Vitest
Vitest 实例需要当前的测试模式。它可以是以下之一:
test:运行运行时测试时benchmark:运行基准测试时 实验性
New in Vitest 4
Vitest 4 新增了多个 API(它们都标记有 "4.0.0+" 徽章),并移除了已弃用的 API:
invalidateschangedTests(使用onFilterWatchedSpecification代替)server(使用vite代替)getProjectsByTestFile(使用getModuleSpecifications代替)getFileWorkspaceSpecs(使用getModuleSpecifications代替)getModuleProjects(自行通过this.projects过滤)updateLastChanged(重命名为invalidateFile)globTestSpecs(使用globTestSpecifications代替)globTestFiles(使用globTestSpecifications代替)listFile(使用getRelevantTestSpecifications代替)
mode
test
测试模式只会调用 test 或 it 中的函数,并在遇到 bench 时抛出错误。此模式使用配置中的 include 和 exclude 选项来查找测试文件。
benchmark 实验性
基准测试模式调用 bench 函数,并在遇到 test 或 it 时抛出错误。此模式使用配置中的 benchmark.include 和 benchmark.exclude 选项来查找基准测试文件。
config
这是根配置(也叫全局配置)。如果你在配置中定义了多个项目,这些项目都会将这个配置视作它们的 globalConfig 并进行继承或引用。
WARNING
这是 Vitest 配置,它不扩展 Vite 配置。它仅包含从 test 属性解析的值。
vite
这是全局的 ViteDevServer。
state 实验性
WARNING
公共 state 是一个实验性 API(除了 vitest.state.getReportedEntity)。破坏性更改可能不遵循 SemVer,请在使用时固定 Vitest 的版本。
全局状态存储有关当前测试的信息。默认情况下,它使用与 @vitest/runner 相同的 API,但我们建议通过调用 @vitest/runner API 上的 state.getReportedEntity() 来使用 报告任务 API:
const task = vitest.state.idMap.get(taskId) // 旧 API
const testCase = vitest.state.getReportedEntity(task) // 新 API未来,旧 API 将不再公开。
snapshot
全局快照管理器。Vitest 使用 snapshot.add 方法跟踪所有快照。
我们可以通过 vitest.snapshot.summary 属性获取快照的最新摘要。
cache
缓存管理器,存储有关最新测试结果和测试文件状态的信息。在 Vitest 中,这仅由默认的排序器用于排序测试。
watcher 4.0.0+
这是 Vitest 的 watcher 实例,提供追踪文件变更并重新执行测试的便利方法。若关闭内置 watcher ,你仍可在自定义 watcher 中调用 onFileChange、onFileDelete 或 onFileCreate 完成相同任务。
projects
这是一个数组,里面包含了所有 测试项目 ,这些项目是用户自己定义的。如果用户没有显式指定任何项目,那么这个数组中只会包含一个 根项目 。
Vitest 会保证这个数组里至少有一个项目可用。如果用户在命令行里通过 --project 参数指定了不存在的项目名称,Vitest 会在创建这个数组前就报错。
getRootProject
function getRootProject(): TestProject该方法会返回根测试项目。一般情况下,根项目并不会实际执行测试,也不会被加入到 vitest.projects 列表中,除非用户在配置中主动包含了根配置,或者没有定义任何独立的测试项目。
根项目的主要目标是设置全局配置。实际上,rootProject.config 直接引用 rootProject.globalConfig 和 vitest.config:
rootProject.config === rootProject.globalConfig === rootProject.vitest.configprovide
function provide<T extends keyof ProvidedContext & string>(
key: T,
value: ProvidedContext[T],
): voidVitest 公开了 provide 方法,它是 vitest.getRootProject().provide 的简写。通过此方法,我们可以从主线程传递值到测试中。所有值在存储之前都通过 structuredClone 进行检查,但值本身不会被克隆。
为了接收测试中的值,我们需要从 vitest 入口点导入 inject 方法:
import { inject } from 'vitest'
const port = inject('wsPort') // 3000为了更好的类型安全性,我们鼓励我们扩展 ProvidedContext 的类型:
import { createVitest } from 'vitest/node'
const vitest = await createVitest('test', {
watch: false,
})
vitest.provide('wsPort', 3000)
declare module 'vitest' {
export interface ProvidedContext {
wsPort: number
}
}WARNING
从技术上讲,provide 是 TestProject 的一种方法,因此它仅限于特定项目。但是,所有项目都会从根项目继承值,这使得 vitest.provide 成为将值传递给测试的通用方法。
getProvidedContext
function getProvidedContext(): ProvidedContext返回根上下文对象。这是 vitest.getRootProject().getProvidedContext 的简写。
getProjectByName
function getProjectByName(name: string): TestProject此方法通过名称返回项目。类似于调用 vitest.projects.find 。
WARNING
如果项目不存在,此方法将返回根项目 - 请确保再次检查返回的项目是否是我们要找的项目。
如果用户没有自定义名称,Vitest 将分配一个空字符串作为名称。
globTestSpecifications
function globTestSpecifications(
filters?: string[],
): Promise<TestSpecification[]>此方法通过收集所有项目中的每个测试来构造新的 测试规范,使用 project.globTestFiles。它接受字符串过滤器以匹配测试文件 - 这些过滤器与 CLI 支持的过滤器 相同。
此方法自动缓存所有测试规范。当我们下次调用 getModuleSpecifications 时,它将返回相同的规范,除非在此之前调用了 clearSpecificationsCache。
WARNING
从 Vitest 3 开始,如果 poolMatchGlob 有多个池或启用了 typecheck,则可能有多个具有相同模块 ID(文件路径)的测试规范。这种可能性将在 Vitest 4 中移除。
const specifications = await vitest.globTestSpecifications(['my-filter'])
// [TestSpecification{ moduleId: '/tests/my-filter.test.ts' }]
console.log(specifications)getRelevantTestSpecifications
function getRelevantTestSpecifications(
filters?: string[]
): Promise<TestSpecification[]>此方法通过调用 project.globTestFiles 解析每个测试规范。它接受字符串过滤器以匹配测试文件 - 这些过滤器与 CLI 支持的过滤器 相同。如果指定了 --changed 标志,则列表将被过滤为仅包含已更改的文件。getRelevantTestSpecifications 不会运行任何测试文件。
WARNING
此方法可能很慢,因为它需要过滤 --changed 标志。如果我们只需要测试文件列表,请不要使用它。
- 如果我们需要获取已知测试文件的规范列表,请使用
getModuleSpecifications代替。 - 如果我们需要获取所有可能的测试文件列表,请使用
globTestSpecifications。
mergeReports
function mergeReports(directory?: string): Promise<TestRunResult>合并指定目录中的多个运行的报告(如果未指定,则使用 --merge-reports 的值)。此值也可以在 config.mergeReports 上设置(默认情况下,它将读取 .vitest-reports 文件夹)。
请注意,directory 将始终相对于工作目录解析。
如果设置了 config.mergeReports,则此方法由 startVitest 自动调用。
collect
function collect(filters?: string[]): Promise<TestRunResult>执行测试文件而不运行测试回调。collect 返回未处理的错误和 测试模块 数组。它接受字符串过滤器以匹配测试文件 - 这些过滤器与 CLI 支持的过滤器 相同。
此方法根据配置的 include、exclude 和 includeSource 值解析测试规范。有关更多信息,请参阅 project.globTestFiles。如果指定了 --changed 标志,则列表将被过滤为仅包含已更改的文件。
WARNING
请注意,Vitest 不使用静态分析来收集测试。Vitest 将像运行常规测试一样在隔离环境中运行每个测试文件。
这使得此方法非常慢,除非我们在收集测试之前禁用隔离。
start
function start(filters?: string[]): Promise<TestRunResult>初始化报告器、覆盖率提供者并运行测试。此方法接受字符串过滤器以匹配测试文件 - 这些过滤器与 CLI 支持的过滤器 相同。
WARNING
如果还调用了 vitest.init(),则不应调用此方法。如果我们需要在 Vitest 初始化后运行测试,请使用 runTestSpecifications 或 rerunTestSpecifications。
如果未设置 config.mergeReports 和 config.standalone,则此方法由 startVitest 自动调用。
init
function init(): Promise<void>初始化报告器和覆盖率提供者。此方法不运行任何测试。如果提供了 --watch 标志,Vitest 仍将运行更改的测试,即使未调用此方法。
在内部,仅当启用了 --standalone 标志时才会调用此方法。
WARNING
如果还调用了 vitest.start(),则不应调用此方法。
如果设置了 config.standalone,则此方法由 startVitest 自动调用。
getModuleSpecifications
function getModuleSpecifications(moduleId: string): TestSpecification[]返回与模块 ID 相关的测试规范列表。ID 应已解析为绝对文件路径。如果 ID 不匹配 include 或 includeSource 模式,则返回的数组将为空。
此方法可以根据 moduleId 和 pool 返回已缓存的规范。但请注意,project.createSpecification 总是返回一个新实例,并且不会自动缓存。但是,当调用 runTestSpecifications 时,规范会自动缓存。
WARNING
从 Vitest 3 开始,此方法使用缓存来检查文件是否为测试文件。为确保缓存不为空,请至少调用一次 globTestSpecifications。
clearSpecificationsCache
function clearSpecificationsCache(moduleId?: string): void当调用 globTestSpecifications 或 runTestSpecifications 时,Vitest 会自动缓存每个文件的测试规范。此方法会根据第一个参数清除给定文件的缓存或整个缓存。
runTestSpecifications
function runTestSpecifications(
specifications: TestSpecification[],
allTestsRun = false
): Promise<TestRunResult>该方法会遍历并执行所有根据 测试规格 定义的测试用例。第二个参数 allTestsRun 则供覆盖率工具判断是否应在覆盖率报告中加入那些没有被任何测试覆盖到的文件。
WARNING
此方法不会触发 onWatcherRerun、onWatcherStart 和 onTestsRerun 回调。如果我们基于文件更改重新运行测试,请考虑使用 rerunTestSpecifications 代替。
rerunTestSpecifications
function rerunTestSpecifications(
specifications: TestSpecification[],
allTestsRun = false
): Promise<TestRunResult>此方法发出 reporter.onWatcherRerun 和 onTestsRerun 事件,然后使用 runTestSpecifications 运行测试。如果主进程中没有错误,它将发出 reporter.onWatcherStart 事件。
updateSnapshot
function updateSnapshot(files?: string[]): Promise<TestRunResult>更新指定文件中的快照。如果未提供文件,它将更新具有失败测试和过时快照的文件。
collectTests
function collectTests(
specifications: TestSpecification[]
): Promise<TestRunResult>执行测试文件而不运行测试回调。collectTests 返回未处理的错误和 测试模块 数组。
此方法与 collect 完全相同,但我们需要自己提供测试规范。
WARNING
请注意,Vitest 不使用静态分析来收集测试。Vitest 将像运行常规测试一样在隔离环境中运行每个测试文件。
这使得此方法非常慢,除非我们在收集测试之前禁用隔离。
cancelCurrentRun
function cancelCurrentRun(reason: CancelReason): Promise<void>此方法将优雅地取消所有正在进行的测试。它将等待已启动的测试完成运行,并且不会运行已计划运行但尚未启动的测试。
setGlobalTestNamePattern
function setGlobalTestNamePattern(pattern: string | RegExp): void此方法覆盖全局的 测试名称模式。
WARNING
此方法不会开始运行任何测试。要使用更新后的模式运行测试,请调用 runTestSpecifications。
getGlobalTestNamePattern
function getGlobalTestNamePattern(): RegExp | undefinedReturns the regexp used for the global test name pattern.
resetGlobalTestNamePattern
function resetGlobalTestNamePattern(): void此方法重置 测试名称模式。这意味着 Vitest 现在不会跳过任何测试。
WARNING
此方法不会开始运行任何测试。要运行没有模式的测试,请调用 runTestSpecifications。
enableSnapshotUpdate
function enableSnapshotUpdate(): void启用允许在运行测试时更新快照的模式。在此方法调用后运行的每个测试都将更新快照。要禁用此模式,请调用 resetSnapshotUpdate。
WARNING
此方法不会开始运行任何测试。要更新快照,请使用 runTestSpecifications 运行测试。
resetSnapshotUpdate
function resetSnapshotUpdate(): void禁用允许在运行测试时更新快照的模式。此方法不会开始运行任何测试。
invalidateFile
function invalidateFile(filepath: string): void此方法使每个项目缓存中的文件失效。如果我们依赖自己的观察器,则此方法非常有用,因为 Vite 的缓存会持久保存在内存中。
DANGER
如果我们禁用 Vitest 的观察器但保持 Vitest 运行,则必须使用此方法手动清除缓存,因为无法禁用缓存。此方法还将使文件的导入者失效。
import
function import<T>(moduleId: string): Promise<T>使用 Vite 模块运行器导入文件。文件将通过全局配置由 Vite 转换,并在单独的上下文中执行。请注意,moduleId 将相对于 config.root。
DANGER
project.import 重用 Vite 的模块图,因此使用常规导入导入同一模块将返回不同的模块:
import * as staticExample from './example.js'
const dynamicExample = await vitest.import('./example.js')
dynamicExample !== staticExample // ✅INFO
Vitest 在内部会通过这个方法加载全局设置、自定义的覆盖率工具和报告器。只要这些组件都挂载在同一个 Vite 服务器下,它们就会共享相同的模块依赖图。
close
function close(): Promise<void>关闭所有项目及其相关资源。此方法只能调用一次;关闭的 Promise 会被缓存,直到服务器重新启动。
exit
function exit(force = false): Promise<void>关闭所有项目并退出进程。如果 force 设置为 true,则进程将在关闭项目后立即退出。
如果进程在 config.teardownTimeout 毫秒后仍然处于活动状态,此方法还将强制调用 process.exit()。
shouldKeepServer
function shouldKeepServer(): boolean如果测试完成后服务器应继续运行,则此方法将返回 true。这通常意味着启用了 watch 模式。
onServerRestart
function onServerRestart(fn: OnServerRestartHandler): void注册一个处理程序,当服务器由于配置更改而重新启动时调用。
onCancel
function onCancel(fn: (reason: CancelReason) => Awaitable<void>): void注册一个处理程序,当测试运行被 vitest.cancelCurrentRun 取消时调用。
onClose
function onClose(fn: () => Awaitable<void>): void注册一个处理程序,当服务器关闭时调用。
onTestsRerun
function onTestsRerun(fn: OnTestsRerunHandler): void注册一个处理程序,当测试重新运行时调用。当手动调用 rerunTestSpecifications 或文件更改且内置观察器安排重新运行时,测试会重新运行。
onFilterWatchedSpecification
function onFilterWatchedSpecification(
fn: (specification: TestSpecification) => boolean
): void注册一个处理程序,当文件更改时调用。此回调应返回 true 或 false,指示是否需要重新运行测试文件。
通过此方法,我们可以挂钩到默认的观察器逻辑,以延迟或丢弃用户当前不想跟踪的测试:
const continuesTests: string[] = []
myCustomWrapper.onContinuesRunEnabled(testItem =>
continuesTests.push(item.fsPath)
)
vitest.onFilterWatchedSpecification(specification =>
continuesTests.includes(specification.moduleId)
)Vitest 可以根据 pool 或 locations 选项为同一文件创建不同的规范,因此不要依赖引用。Vitest 还可以从 vitest.getModuleSpecifications 返回缓存的规范 - 缓存基于 moduleId 和 pool。请注意,project.createSpecification 总是返回一个新实例。
matchesProjectFilter 3.1.0+
function matchesProjectFilter(name: string): boolean检查名称是否与当前 项目过滤器 匹配。如果没有项目过滤器,则始终返回 true 。
无法通过编程方式更改 --project CLI 选项。
waitForTestRunEnd 4.0.0+
function waitForTestRunEnd(): Promise<void>若测试正在运行,则返回一个 Promise ,它会在测试运行完毕后兑现。
createCoverageProvider 4.0.0+
function createCoverageProvider(): Promise<CoverageProvider | null>当配置中启用了 coverage 时,创建覆盖率提供器。若使用 start 或 init 方法启动测试,这一步会自动完成。
WARNING
若未将 coverage.clean 显式设为 false ,此方法还会清空之前的所有报告。
enableCoverage 4.0.0+
function enableCoverage(): Promise<void>此方法为在此调用之后运行的测试启用覆盖率收集。enableCoverage 不会运行任何测试;它只是设置 Vitest 来收集覆盖率。
如果尚不存在覆盖率提供者,它将创建一个新的覆盖率提供者。
disableCoverage 4.0.0+
function disableCoverage(): void此方法会禁用后续运行的测试的覆盖率收集功能。
getSeed 4.0.0+
function getSeed(): number | null如果测试以随机顺序运行,则返回种子值。
experimental_parseSpecification 4.0.0+ experimental
function experimental_parseSpecification(
specification: TestSpecification
): Promise<TestModule>该函数会收集文件内的所有测试,但不会执行它们。它借助 Vite 的 ssrTransform ,并在其之上使用 rollup 的 parseAst 进行静态分析,从而提取所有可识别的测试用例。
WARNING
如果 Vitest 无法解析测试的名称,它将在测试或套件中注入一个 dynamic: true 属性。id 也会带有 -dynamic 后缀,以避免破坏已正确收集的测试。
Vitest 总是在带有 for 或 each 修饰符的测试,或者名称是动态生成的测试(如 hello ${property} 或 'hello' + ${property})中注入此属性。Vitest 仍会为测试分配一个名称,但该名称不能用于过滤测试。
Vitest 无法做到让动态测试可以被过滤,但你可以使用 escapeTestName 函数将带有 for 或 each 修饰符的测试转换为名称模式:
若 Vitest 无法解析测试名称,它会在测试或套件中注入一个隐藏的 dynamic: true 属性,并在 id 后追加 -dynamic ,以免破坏已正确收集的测试。
含 for 或 each 修饰符的测试,以及名称动态生成的测试(如 hello ${property} 或 'hello' + ${property} ) , Vitest 一律会注入此属性。 Vitest 仍会为其分配名称,但该名称无法用于过滤测试。
Vitest 无法让动态测试支持过滤,但你可以使用 escapeTestName 函数,将带 for 或 each 的测试转换成名称模式:
import { escapeTestName } from 'vitest/node'
// 转换为 /hello, .+?/
const escapedPattern = new RegExp(escapeTestName('hello, %s', true))WARNING
Vitest 只会收集当前文件内定义的测试,绝不会跟随导入去其他文件搜寻。
无论是否从 vitest 入口点导入, Vitest 都会收集所有 it 、test 、suite 和 describe 的定义。
experimental_parseSpecifications 4.0.0+ 实验
function experimental_parseSpecifications(
specifications: TestSpecification[],
options?: {
concurrency?: number
}
): Promise<TestModule[]>该方法会依据规格数组 collect tests。
默认情况下, Vitest 仅同时运行 os.availableParallelism() 个规格,以避免性能骤降。我们可以在第二个参数中指定其他并发数。