当你希望确保函数的输出不会意外更改时,快照测试是一个非常有用的工具。
使用快照时,Vitest 将获取给定值的快照,将其比较时将参考存储在测试旁边的快照文件。如果两个快照不匹配,则测试将失败:要么更改是意外的,要么参考快照需要更新到测试结果的新版本。
使用快照
要将一个值快照,你可以使用 expect()
的 toMatchSnapshot()
API:
// ---cut---
import { expect, it } from 'vitest'
function toUpperCase(str: string) {
return str
}
it('toUpperCase', () => {
const result = toUpperCase('foobar')
expect(result).toMatchSnapshot()
})
此测试在第一次运行时,Vitest 会创建一个快照文件,如下所示:
// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html
exports['toUpperCase 1'] = '"FOOBAR"'
快照文件应该与代码更改一起提交,并作为代码审查过程的一部分进行审查。在随后的测试运行中,Vitest 会将执行的输出与之前的快照进行比较。如果他们匹配,测试就会通过。如果它们不匹配,要么测试运行时在你的代码中发现了应该修复的错误,要么实现已经更改,需要更新快照。
内联快照
在异步并发测试中使用快照时,由于 JavaScript 的限制,你需要使用 测试环境 中的 expect
来确保检测到正确的测试。
如同前文,你可以使用 toMatchInlineSnapshot()
将内联快照存储在测试文件中。
// ---cut---
import { expect, it } from 'vitest'
function toUpperCase(str: string) {
return str
}
it('toUpperCase', () => {
const result = toUpperCase('foobar')
expect(result).toMatchInlineSnapshot()
})
Vitest 不会创建快照文件,而是直接修改测试文件,将快照作为字符串更新到文件中:
// ---cut---
import { expect, it } from 'vitest'
function toUpperCase(str: string) {
return str
}
it('toUpperCase', () => {
const result = toUpperCase('foobar')
expect(result).toMatchInlineSnapshot('"FOOBAR"')
})
这允许你直接查看期望输出,而无需跨不同的文件跳转。
更新快照
在异步并发测试中使用快照时,由于 JavaScript 的限制,你需要使用 测试环境 中的 expect
来确保检测到正确的测试。
当接收到的值与快照不匹配时,测试将失败,并显示它们之间的差异。当需要更改快照时,你可能希望从当前状态更新快照。
在监听(watch)模式下, 你可以在终端中键入 u
键直接更新失败的快照。
或者,你可以在 CLI 中使用 --update
或 -u
标记使 Vitest 进入快照更新模式。
vitest -u
文件快照
调用 toMatchSnapshot()
时,我们将所有快照存储在格式化的快照文件中。这意味着我们需要转义快照字符串中的一些字符(即双引号 "
和反引号 ```)。同时,你可能会丢失快照内容的语法突出显示(如果它们是某种语言)。
为了改善这种情况,我们引入 toMatchFileSnapshot()
以在文件中显式快照。这允许你为快照文件分配任何文件扩展名,并使它们更具可读性。
import { expect, it } from 'vitest'
it('render basic', async () => {
const result = renderHTML(h('div', { class: 'foo' }))
await expect(result).toMatchFileSnapshot('./test/basic.output.html')
})
它将与 ./test/basic.output.html
的内容进行比较。并且可以用 --update
标志写回。
图像快照
快照图像也可以使用 jest-image-snapshot
。
npm i -D jest-image-snapshot
test('image snapshot', () => {
expect(readFileSync('./test/stubs/input-image.png')).toMatchImageSnapshot()
})
自定义序列化程序
你可以添加自己的逻辑来修改快照的序列化方式。像 Jest 一样,Vitest 为内置的 JavaScript 类型、HTML 元素、ImmutableJS 和 React 元素提供了默认的序列化程序。
可以使用 expect.addSnapshotSerializer
添加自定义序列器。
expect.addSnapshotSerializer({
serialize(val, config, indentation, depth, refs, printer) {
// `printer` is a function that serializes a value using existing plugins.
return `Pretty foo: ${printer(val.foo, config, indentation, depth, refs)}`
},
test(val) {
return val && Object.prototype.hasOwnProperty.call(val, 'foo')
},
})
我们还支持 snapshotSerializers 选项来隐式添加自定义序列化器。
import { SnapshotSerializer } from 'vitest'
export default {
serialize(val, config, indentation, depth, refs, printer) {
// `printer` 是一个使用现有插件序列化数值的函数。
return `Pretty foo: ${printer(val.foo, config, indentation, depth, refs)}`
},
test(val) {
return val && Object.prototype.hasOwnProperty.call(val, 'foo')
},
} satisfies SnapshotSerializer
import { defineConfig } from 'vite'
export default defineConfig({
test: {
snapshotSerializers: ['path/to/custom-serializer.ts'],
},
})
如下所示的测试添加后:
test('foo snapshot test', () => {
const bar = {
foo: {
x: 1,
y: 2,
},
}
expect(bar).toMatchSnapshot()
})
你将获得以下快照:
Pretty foo: Object {
"x": 1,
"y": 2,
}
我们使用的是 Jest 的 pretty-format
来序列化快照。你可以在这里阅读更多相关内容:pretty-format.
与 Jest 的区别
Vitest 提供了与 Jest 几乎兼容的快照功能,除少数例外:
1. 快照文件中的注释标头不同
- // Jest Snapshot v1, https://goo.gl/fbAQLP
+ // Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html
这实际上不会影响功能,但在从 Jest 迁移时可能会影响提交差异。
2. printBasicPrototype
默认为 false
Jest 和 Vitest 的快照都是由 pretty-format
支持的。在 Vitest 中,我们将 printBasicPrototype
的默认值设置为 false
以提供更清晰的快照输出,在 Jest 版本 < 29.0.0 中默认为 true
。
import { expect, test } from 'vitest'
test('snapshot', () => {
const bar = [
{
foo: 'bar',
},
]
// in Jest
expect(bar).toMatchInlineSnapshot(`
Array [
Object {
"foo": "bar",
},
]
`)
// in Vitest
expect(bar).toMatchInlineSnapshot(`
[
{
"foo": "bar",
},
]
`)
})
我们相信这种预设有更好的可读性和开发体验。如果你仍然喜欢 Jest 的行为,可以通过以下方式更改配置:
// vitest.config.js
export default defineConfig({
test: {
snapshotFormat: {
printBasicPrototype: true,
},
},
})
3. 使用 V 形 >
而非冒号 :
作为自定义消息的分隔符
当创建快照文件期间传递自定义消息时,Vitest 使用 V 形 >
作为分隔符而不是冒号 :
以提高自定义消息可读性。
对于以下示例测试代码:
test('toThrowErrorMatchingSnapshot', () => {
expect(() => {
throw new Error('error')
}).toThrowErrorMatchingSnapshot('hint')
})
在 Jest 中,快照将是:
exports[`toThrowErrorMatchingSnapshot: hint 1`] = `"error"`;
在 Vitest 中,等效的快照将是:
exports[`toThrowErrorMatchingSnapshot > hint 1`] = `[Error: error]`;
4. toThrowErrorMatchingSnapshot
和 toThrowErrorMatchingInlineSnapshot
的默认 Error
快照不同
import { expect, test } from 'vitest'
// ---cut---
test('snapshot', () => {
//
// in Jest
//
expect(new Error('error')).toMatchInlineSnapshot('[Error: error]')
// Jest 在 `Error` 实例上记录 `Error.message` 快照
expect(() => {
throw new Error('error')
}).toThrowErrorMatchingInlineSnapshot('"error"')
//
// in Vitest
//
expect(new Error('error')).toMatchInlineSnapshot('[Error: error]')
expect(() => {
throw new Error('error')
}).toThrowErrorMatchingInlineSnapshot('[Error: error]')
})