234 lines
6.4 KiB
TypeScript

import { Readable, Writable } from 'node:stream'
import fs from 'node:fs'
import { fileURLToPath } from 'node:url'
import type { UserConfig as ViteUserConfig } from 'vite'
import { type WorkerGlobalState, afterEach, onTestFinished } from 'vitest'
import type { UserConfig, Vitest, VitestRunMode } from 'vitest/node'
import { startVitest } from 'vitest/node'
import type { Options } from 'tinyexec'
import { x } from 'tinyexec'
import { dirname, resolve } from 'pathe'
import { getCurrentTest } from 'vitest/suite'
import { Cli } from './cli'
interface VitestRunnerCLIOptions {
std?: 'inherit'
fails?: boolean
}
export async function runVitest(
config: UserConfig,
cliFilters: string[] = [],
mode: VitestRunMode = 'test',
viteOverrides: ViteUserConfig = {},
runnerOptions: VitestRunnerCLIOptions = {},
) {
// Reset possible previous runs
process.exitCode = 0
let exitCode = process.exitCode
// Prevent possible process.exit() calls, e.g. from --browser
const exit = process.exit
process.exit = (() => { }) as never
const stdout = new Writable({
write(chunk, __, callback) {
if (runnerOptions.std === 'inherit') {
process.stdout.write(chunk.toString())
}
callback()
},
})
const stderr = new Writable({
write(chunk, __, callback) {
if (runnerOptions.std === 'inherit') {
process.stderr.write(chunk.toString())
}
callback()
},
})
// "node:tty".ReadStream doesn't work on Github Windows CI, let's simulate it
const stdin = new Readable({ read: () => '' }) as NodeJS.ReadStream
stdin.isTTY = true
stdin.setRawMode = () => stdin
const cli = new Cli({ stdin, stdout, stderr })
let ctx: Vitest | undefined
try {
const { reporters, ...rest } = config
ctx = await startVitest(mode, cliFilters, {
watch: false,
// "none" can be used to disable passing "reporter" option so that default value is used (it's not same as reporters: ["default"])
...(reporters === 'none' ? {} : reporters ? { reporters } : { reporters: ['verbose'] }),
...rest,
}, {
...viteOverrides,
server: {
// we never need a websocket connection for the root config because it doesn't connect to the browser
// browser mode uses a separate config that doesn't inherit CLI overrides
ws: false,
watch: {
// During tests we edit the files too fast and sometimes chokidar
// misses change events, so enforce polling for consistency
// https://github.com/vitejs/vite/blob/b723a753ced0667470e72b4853ecda27b17f546a/playground/vitestSetup.ts#L211
usePolling: true,
interval: 100,
},
...viteOverrides?.server,
},
}, {
stdin,
stdout,
stderr,
})
}
catch (e: any) {
if (runnerOptions.fails !== true) {
console.error(e)
}
cli.stderr += e.stack
}
finally {
exitCode = process.exitCode
process.exitCode = 0
if (getCurrentTest()) {
onTestFinished(async () => {
await ctx?.close()
await ctx?.closingPromise
process.exit = exit
})
}
else {
afterEach(async () => {
await ctx?.close()
await ctx?.closingPromise
process.exit = exit
})
}
}
return {
ctx,
exitCode,
vitest: cli,
stdout: cli.stdout,
stderr: cli.stderr,
waitForClose: async () => {
await new Promise<void>(resolve => ctx!.onClose(resolve))
return ctx?.closingPromise
},
}
}
export async function runCli(command: string, _options?: Partial<Options> | string, ...args: string[]) {
let options = _options
if (typeof _options === 'string') {
args.unshift(_options)
options = undefined
}
const subprocess = x(command, args, options as Options).process!
const cli = new Cli({
stdin: subprocess.stdin!,
stdout: subprocess.stdout!,
stderr: subprocess.stderr!,
})
let setDone: (value?: unknown) => void
const isDone = new Promise(resolve => (setDone = resolve))
subprocess.on('exit', () => setDone())
function output() {
return {
vitest: cli,
exitCode: subprocess.exitCode,
stdout: cli.stdout || '',
stderr: cli.stderr || '',
waitForClose: () => isDone,
}
}
// Manually stop the processes so that each test don't have to do this themselves
afterEach(async () => {
if (subprocess.exitCode === null) {
subprocess.kill()
}
await isDone
})
if (args.includes('--inspect') || args.includes('--inspect-brk')) {
return output()
}
if (args[0] !== 'list' && args.includes('--watch')) {
if (command === 'vitest') {
// Wait for initial test run to complete
await cli.waitForStdout('Waiting for file changes')
}
// make sure watcher is ready
await cli.waitForStdout('[debug] watcher is ready')
cli.stdout = cli.stdout.replace('[debug] watcher is ready\n', '')
}
else {
await isDone
}
return output()
}
export async function runVitestCli(_options?: Partial<Options> | string, ...args: string[]) {
process.env.VITE_TEST_WATCHER_DEBUG = 'true'
return runCli('vitest', _options, ...args)
}
export async function runViteNodeCli(_options?: Partial<Options> | string, ...args: string[]) {
process.env.VITE_TEST_WATCHER_DEBUG = 'true'
const { vitest, ...rest } = await runCli('vite-node', _options, ...args)
return { viteNode: vitest, ...rest }
}
export function getInternalState(): WorkerGlobalState {
// @ts-expect-error untyped global
return globalThis.__vitest_worker__
}
const originalFiles = new Map<string, string>()
const createdFiles = new Set<string>()
afterEach(() => {
originalFiles.forEach((content, file) => {
fs.writeFileSync(file, content, 'utf-8')
})
createdFiles.forEach((file) => {
if (fs.existsSync(file)) {
fs.unlinkSync(file)
}
})
originalFiles.clear()
createdFiles.clear()
})
export function createFile(file: string, content: string) {
createdFiles.add(file)
fs.mkdirSync(dirname(file), { recursive: true })
fs.writeFileSync(file, content, 'utf-8')
}
export function editFile(file: string, callback: (content: string) => string) {
const content = fs.readFileSync(file, 'utf-8')
if (!originalFiles.has(file)) {
originalFiles.set(file, content)
}
fs.writeFileSync(file, callback(content), 'utf-8')
}
export function resolvePath(baseUrl: string, path: string) {
const filename = fileURLToPath(baseUrl)
return resolve(dirname(filename), path)
}