发起 HTTP 请求
现代测试运行器在测试 HTTP 请求方面已经提供了许多优秀的功能。因此,Vue Test Utils 并没有提供任何独特的工具来处理这一点。
然而,测试 HTTP 请求是一个重要的功能,我们希望强调一些注意事项。
在本节中,我们将探讨一些执行、模拟和断言 HTTP 请求的模式。
博客文章列表
让我们从一个基本的用例开始。以下的 PostList
组件渲染了从外部 API 获取的博客文章列表。为了获取这些文章,组件包含一个触发请求的 button
元素:
<template>
<button @click="getPosts">Get posts</button>
<ul>
<li v-for="post in posts" :key="post.id" data-test="post">
{{ post.title }}
</li>
</ul>
</template>
<script>
import axios from 'axios'
export default {
data() {
return {
posts: null
}
},
methods: {
async getPosts() {
this.posts = await axios.get('/api/posts')
}
}
}
</script>
为了正确测试这个组件,我们需要做几件事。
我们的第一个目标是测试这个组件而不实际访问 API。这样会导致测试变得不稳定且可能执行缓慢。
其次,我们需要断言组件是否以正确的参数进行了正确的调用。虽然我们不会从 API 获取结果,但仍需确保请求了正确的资源。
此外,我们还需要确保 DOM 已相应更新并显示数据。我们可以使用 @vue/test-utils
中的 flushPromises()
函数来实现。
import { mount, flushPromises } from '@vue/test-utils'
import axios from 'axios'
import PostList from './PostList.vue'
const mockPostList = [
{ id: 1, title: 'title1' },
{ id: 2, title: 'title2' }
]
// 以下代码告诉 Jest 模拟任何对 `axios.get` 的调用并返回 `mockPostList`
jest.spyOn(axios, 'get').mockResolvedValue(mockPostList)
test('loads posts on button click', async () => {
const wrapper = mount(PostList)
await wrapper.get('button').trigger('click')
// 断言我们已正确调用 axios.get 的次数和参数。
expect(axios.get).toHaveBeenCalledTimes(1)
expect(axios.get).toHaveBeenCalledWith('/api/posts')
// 等待 DOM 更新。
await flushPromises()
// 最后,确保我们已渲染 API 的内容。
const posts = wrapper.findAll('[data-test="post"]')
expect(posts).toHaveLength(2)
expect(posts[0].text()).toContain('title1')
expect(posts[1].text()).toContain('title2')
})
请注意,我们在变量 mockPostList
前添加了前缀 mock
。如果不这样做,我们将收到错误:“The module factory of jest.mock() is not allowed to reference any out-of-scope variables。” 这是 Jest 特有的,你可以在他们的文档中了解更多关于这种行为的信息。
还要注意,我们在等待 flushPromises
之后再与组件进行交互。这样可以确保在运行断言之前,DOM 已更新。
jest.mock() 的替代方案
在 Jest 中设置模拟有多种方法。上面示例中使用的方式是最简单的。对于更强大的替代方案,你可能想查看 axios-mock-adapter 或 msw 等其他工具。
断言加载状态
现在,这个 PostList
组件非常实用,但缺少一些其他很棒的功能。让我们扩展它,使其在加载文章时显示一个漂亮的消息!
同时,让我们在加载时禁用 <button>
元素。我们不希望用户在获取数据时继续发送请求!
<template>
<button :disabled="loading" @click="getPosts">Get posts</button>
<p v-if="loading" role="alert">Loading your posts…</p>
<ul v-else>
<li v-for="post in posts" :key="post.id" data-test="post">
{{ post.title }}
</li>
</ul>
</template>
<script>
import axios from 'axios'
export default {
data() {
return {
posts: null,
loading: null
}
},
methods: {
async getPosts() {
this.loading = true
this.posts = await axios.get('/api/posts')
this.loading = null
}
}
}
</script>
让我们编写一个测试,以断言所有与加载相关的元素都能及时渲染。
test('displays loading state on button click', async () => {
const wrapper = mount(PostList)
// 注意,我们在点击按钮之前运行以下断言
// 此时组件应该处于“未加载”状态。
expect(wrapper.find('[role="alert"]').exists()).toBe(false)
expect(wrapper.get('button').attributes()).not.toHaveProperty('disabled')
// 现在像往常一样触发它。
await wrapper.get('button').trigger('click')
// 我们在等待所有承诺完成之前,断言“加载状态”。
expect(wrapper.find('[role="alert"]').exists()).toBe(true)
expect(wrapper.get('button').attributes()).toHaveProperty('disabled')
// 和之前一样,等待 DOM 更新。
await flushPromises()
// 之后,我们回到“未加载”状态。
expect(wrapper.find('[role="alert"]').exists()).toBe(false)
expect(wrapper.get('button').attributes()).not.toHaveProperty('disabled')
})
从 Vuex 发起 HTTP 请求
对于更复杂的应用程序,一个典型的场景是触发执行 HTTP 请求的 Vuex 动作。
这与上面概述的示例没有什么不同。我们可能希望像往常一样加载 store 并模拟像 axios
这样的服务。通过这种方式,我们可以模拟系统的边界,从而在测试中获得更高的信心。
你可以查看 Testing Vuex 文档,以获取有关使用 Vue Test Utils 测试 Vuex 的更多信息。
结论
- Vue Test Utils 不需要特殊工具来测试 HTTP 请求。唯一需要考虑的是测试异步行为。
- 测试不应依赖于外部服务。使用模拟工具如
jest.mock
来避免这种情况。 flushPromises()
是一个有用的工具,可以确保在异步操作后 DOM 更新。- 通过与组件进行交互来直接触发 HTTP 请求,可以让你的测试更加稳健。