mirror of
https://github.com/vitest-dev/vitest.git
synced 2025-12-08 18:26:03 +00:00
Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> Co-authored-by: Vladimir Sheremet <sleuths.slews0s@icloud.com>
327 lines
9.2 KiB
TypeScript
327 lines
9.2 KiB
TypeScript
import type { Options } from 'tinyexec'
|
|
import type { UserConfig as ViteUserConfig } from 'vite'
|
|
import type { WorkerGlobalState } from 'vitest'
|
|
import type { WorkspaceProjectConfiguration } from 'vitest/config'
|
|
import type { TestModule, UserConfig, Vitest, VitestRunMode } from 'vitest/node'
|
|
import { webcrypto as crypto } from 'node:crypto'
|
|
import fs from 'node:fs'
|
|
import { Readable, Writable } from 'node:stream'
|
|
import { fileURLToPath } from 'node:url'
|
|
import { inspect } from 'node:util'
|
|
import { dirname, resolve } from 'pathe'
|
|
import { x } from 'tinyexec'
|
|
import * as tinyrainbow from 'tinyrainbow'
|
|
import { afterEach, onTestFinished } from 'vitest'
|
|
import { startVitest } from 'vitest/node'
|
|
import { getCurrentTest } from 'vitest/suite'
|
|
import { Cli } from './cli'
|
|
|
|
// override default colors to disable them in tests
|
|
Object.assign(tinyrainbow.default, tinyrainbow.getDefaultColors())
|
|
|
|
export interface VitestRunnerCLIOptions {
|
|
std?: 'inherit'
|
|
fails?: boolean
|
|
preserveAnsi?: boolean
|
|
tty?: 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()
|
|
},
|
|
})
|
|
|
|
if (runnerOptions?.tty) {
|
|
(stdout as typeof process.stdout).isTTY = true
|
|
}
|
|
|
|
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, preserveAnsi: runnerOptions.preserveAnsi })
|
|
|
|
let ctx: Vitest | undefined
|
|
let thrown = false
|
|
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,
|
|
env: {
|
|
NO_COLOR: 'true',
|
|
...rest.env,
|
|
},
|
|
}, {
|
|
...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)
|
|
}
|
|
thrown = true
|
|
cli.stderr += inspect(e)
|
|
}
|
|
finally {
|
|
exitCode = process.exitCode
|
|
process.exitCode = 0
|
|
|
|
if (getCurrentTest()) {
|
|
onTestFinished(async () => {
|
|
await ctx?.close()
|
|
process.exit = exit
|
|
})
|
|
}
|
|
else {
|
|
afterEach(async () => {
|
|
await ctx?.close()
|
|
process.exit = exit
|
|
})
|
|
}
|
|
}
|
|
|
|
return {
|
|
thrown,
|
|
ctx,
|
|
exitCode,
|
|
vitest: cli,
|
|
stdout: cli.stdout,
|
|
stderr: cli.stderr,
|
|
waitForClose: async () => {
|
|
await new Promise<void>(resolve => ctx!.onClose(resolve))
|
|
return ctx?.closingPromise
|
|
},
|
|
}
|
|
}
|
|
|
|
interface CliOptions extends Partial<Options> {
|
|
earlyReturn?: boolean
|
|
}
|
|
|
|
export async function runCli(command: string, _options?: CliOptions | 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 ((options as CliOptions)?.earlyReturn || args.includes('--inspect') || args.includes('--inspect-brk')) {
|
|
return output()
|
|
}
|
|
|
|
if (args[0] === 'init') {
|
|
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?: CliOptions | string, ...args: string[]) {
|
|
process.env.VITE_TEST_WATCHER_DEBUG = 'true'
|
|
return runCli('vitest', _options, ...args)
|
|
}
|
|
|
|
export async function runViteNodeCli(_options?: CliOptions | 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)
|
|
}
|
|
|
|
export function useFS(root: string, structure: Record<string, string | ViteUserConfig | WorkspaceProjectConfiguration[]>) {
|
|
const files = new Set<string>()
|
|
const hasConfig = Object.keys(structure).some(file => file.includes('.config.'))
|
|
if (!hasConfig) {
|
|
structure['./vitest.config.js'] = {}
|
|
}
|
|
for (const file in structure) {
|
|
const filepath = resolve(root, file)
|
|
files.add(filepath)
|
|
const content = typeof structure[file] === 'string'
|
|
? structure[file]
|
|
: `export default ${JSON.stringify(structure[file])}`
|
|
fs.mkdirSync(dirname(filepath), { recursive: true })
|
|
fs.writeFileSync(filepath, String(content), 'utf-8')
|
|
}
|
|
onTestFinished(() => {
|
|
if (process.env.VITEST_FS_CLEANUP !== 'false') {
|
|
fs.rmSync(root, { recursive: true, force: true })
|
|
}
|
|
})
|
|
return {
|
|
editFile: (file: string, callback: (content: string) => string) => {
|
|
const filepath = resolve(root, file)
|
|
if (!files.has(filepath)) {
|
|
throw new Error(`file ${file} is outside of the test file system`)
|
|
}
|
|
const content = fs.readFileSync(filepath, 'utf-8')
|
|
fs.writeFileSync(filepath, callback(content))
|
|
},
|
|
createFile: (file: string, content: string) => {
|
|
if (file.startsWith('..')) {
|
|
throw new Error(`file ${file} is outside of the test file system`)
|
|
}
|
|
const filepath = resolve(root, file)
|
|
if (!files.has(filepath)) {
|
|
throw new Error(`file ${file} already exists in the test file system`)
|
|
}
|
|
createFile(filepath, content)
|
|
},
|
|
}
|
|
}
|
|
|
|
export async function runInlineTests(
|
|
structure: Record<string, string | ViteUserConfig | WorkspaceProjectConfiguration[]>,
|
|
config?: UserConfig,
|
|
options?: VitestRunnerCLIOptions,
|
|
) {
|
|
const root = resolve(process.cwd(), `vitest-test-${crypto.randomUUID()}`)
|
|
const fs = useFS(root, structure)
|
|
const vitest = await runVitest({
|
|
root,
|
|
...config,
|
|
}, [], 'test', {}, options)
|
|
return {
|
|
fs,
|
|
root,
|
|
...vitest,
|
|
get results() {
|
|
return (vitest.ctx?.state.getFiles() || []).map(file => vitest.ctx?.state.getReportedEntity(file) as TestModule)
|
|
},
|
|
}
|
|
}
|
|
|
|
export const ts = String.raw
|