首页 > 其他分享 >[TDD] 如何测试 React 异步组件?

[TDD] 如何测试 React 异步组件?

时间:2022-10-07 18:31:46浏览次数:67  
标签:username 异步 screen posts TDD React expect import password

前言

本文承接上文 如何测试驱动开发 React 组件?​,这次我将继续使用 @testing-library/react 来测试我们的 React 应用,并简要简要说明如何测试异步组件。

异步组件的测试内容

我们知道异步请求主要用于从服务器上获取数据,这个异步请求可能是主动触发的,也可能是(鼠标)事件响应,本文主要包含 2 方面内容:

  1. 如何测试在​​componentDidMount​​ 生命周期中发出的异步请求 ?
  2. 如何测试(鼠标)事件发出的异步请求 ?

对于异步组件,有两件步骤需要进行测试:

第一:测试异步方法本身有没有被调用,并且传了正确的参数。

第二:在调用之后,应用程序应该做出响应。

一起来看看代码中该如何实现?

假设你有一个用 React 编写的小型博客应用程序。有一个登录页面,还有有一个文章列表页面,内容就跟我的博客一样。

登录测试

先来实现登录页,先脑补一个效果图吧

[TDD] 如何测试 React 异步组件?_数据

我们先来写下测试用例

  1. 界面包含账号和密码输入框
  2. 接口请求包含 username 和 password
  3. 防止登录重复点击
  4. 登录成功跳转页面
  5. 登录失败显示错误信息

测试渲染

代码未动,测试先行,先确保我们的组件可以渲染。

import React from "react";
import { render } from "@testing-library/react";
import Login from "./index";

describe("Login component", () => {
it("should render", () => {
const { getByText } = render(<Login onSubmit={());
expect(getByText(/账号/)).toBeInTheDocument();
expect(getByText(/密码/)).toBeInTheDocument();
});
});

登录组件实现

为了保证是一个纯组件,将提交方法​​onSubmit​​​作为一个 ​​props​​ 传入,接下来我们实现下组件代码

import React from "react";

function Login({ onSubmit }) {
function handleSubmit(event) {
event.preventDefault();
const { username, password } = event.target.elements;

onSubmit({
username: username.value,
password: password.value
});
}
return (
<form onSubmit={handleSubmit}>
<div>
<label htmlFor="username-field">账号:</label>
<input id="username-field" name="username" type="text"
</div>
<div>
<label htmlFor="password-field">密码:</label>
<input id="password-field" name="password" type="password"
</div>
<div>
<button type="submit">登录</button>
</div>
</form>
);
}

export default Login;

为了方便理解我们这边没有使用其他三方库,若在生产环境中,我推荐使用 react-hook-form

测试提交

接下来测试下 ​​onSubmit​​​ 方法必须包含 ​​username​​​ 和 ​​password​​,

我们需要模拟用户输入,这个时候我们需要安装 ​​@test-library/user-event​

import React from "react";
import { render, screen, waitFor } from "@testing-library/react";
import userEvent from "@testing-library/user-event";
import Login from "./index";

test("onSubmit calls with username and password", () => {
let submittedData;
const handleSubmit = data => (submittedData = data);
render(
<Login onSubmit={handleSubmit}
);
const username = "user";
const password = "password";

userEvent.type(screen.getByPlaceholderText(/username/i), username);
userEvent.type(screen.getByPlaceholderText(/password/i), password);
userEvent.click(screen.getByRole("button", { name: /登录/ }));

expect(submittedData).toEqual({
username,
password
});
});

我们可以选用 ​​@testing-library​​​ 的 ​​get*By*​​​ 函数获取dom 中的元素, 这里使用 ​​getByPlaceholderText​

[TDD] 如何测试 React 异步组件?_测试_02

以上测试用例只测试了登录函数,但是我们并未写登录成功或者失败的逻辑,接下来来我们通过 jest 的 mock 函数功能来模拟登录。

测试登录成功

由于测试登录成功的例子已经包含了"测试提交"和"测试渲染"的功能,所以,可以将前面2个单元测试删除。登录后,按钮改成 loading 状态 disabled。

const fakeData = {
username: "username",
password: "username",
};

test("onSubmit success", async () => {
// mock 登录函数
const login = jest.fn().mockResolvedValueOnce({ data: { success: true } });
render(<Login onSubmit={login});

userEvent.type(screen.getByPlaceholderText(/username/i), fakeData.username);
userEvent.type(screen.getByPlaceholderText(/password/i), fakeData.password);
userEvent.click(screen.getByRole("button", { name: /登录/ }));
//登录后,按钮改成 loading 状态 disabled
const button = screen.getByRole("button", { name: /loading/ });

expect(button).toBeInTheDocument();
expect(button).toBeDisabled();

await waitFor(() => expect(login).toHaveBeenCalledWith(fakeData));

expect(login).toHaveBeenCalledTimes(1);

// 文档中没有 loading 按钮
expect(
screen.queryByRole("button", { name: "loading" })
).not.toBeInTheDocument();
});

接下来我们修改组件, 顺便把登录失败的逻辑也写了,登录失败在登录框下显示服务端返回信息。

import React, { useState } from "react";

function Login({ onSubmit }) {
const [loading, setLoading] = useState(false);
const [message, setMessage] = useState("");
function handleSubmit(event) {
event.preventDefault();
const { username, password } = event.target.elements;
setLoading(true);

onSubmit({
username: username.value,
password: password.value,
})
.then((res) => {
setLoading(false);
})
.catch((res) => {
setLoading(false);
setMessage(res.data.message);
});
}
return (
<form onSubmit={handleSubmit}>
<div>
<label htmlFor="username-field">账号:</label>
<input
id="username-field"
name="username"
placeholder="username"
type="text"
</div>
<div>
<label htmlFor="password-field">密码:</label>
<input
id="password-field"
name="password"
placeholder="password"
type="password"
</div>
<div>
<button disabled={loading} type="submit">
{loading ? "loading" : "登录"}
</button>
</div>
<div>{message}</div>
</form>
);
}

export default Login;

测试登录失败

我们直接复制成功的测试用例,修改失败的逻辑。测试用例:

  • 失败后文档中显示服务端的消息
  • 失败后按钮又显示登录并且可以点击
test("onSubmit failures", async () => {
const message = "账号或密码错误!";
// mock 登录函数失败
const login = jest.fn().mockRejectedValueOnce({
data: { message },
});
render(<Login onSubmit={login});

userEvent.type(screen.getByPlaceholderText(/username/i), fakeData.username);
userEvent.type(screen.getByPlaceholderText(/password/i), fakeData.password);
userEvent.click(screen.getByRole("button", { name: /登录/ }));

const button = screen.getByRole("button", { name: /loading/ });

expect(button).toBeInTheDocument();
expect(button).toBeDisabled();

await waitFor(() => expect(login).toHaveBeenCalledWith(fakeData));

expect(login).toHaveBeenCalledTimes(1);

// 确保文档中有服务端返回的消息
expect(screen.getByText(message)).toBeInTheDocument();

// 文档中没有 loading 按钮
expect(
screen.queryByRole("button", { name: "loading" })
).not.toBeInTheDocument();

expect(screen.getByRole("button", { name: /登录/ })).not.toBeDisabled();
});

博客列表测试

相信经过登录的测试,我们在来写博客列表的测试已经不难了,我们先来写下测试用例:

  • 接口请求中页面显示 loading
  • 请求成功显示博客列表
  • 列表为空显示暂无数据
  • 请求失败显示服务端错误

博客列表代码

下面的代码中, 使用了 ​​react-use​​,首先我们先要安装这个包

import React from "react";
import { fetchPosts } from "./api/posts";
import { useAsync } from "react-use";

export default function Posts() {
const posts = useAsync(fetchPosts, []);

if (posts.loading) return "Loading...";
if (posts.error) return "服务端错误";
if (posts.value && posts.value.length===0) return "暂无数据";


return (
<>
<h1>My Posts</h1>
<ul>
{posts.value.map((post) => (
<li key={post.id}>
<a href={post.url}>{post.title}</a>
</li>
))}
</ul>
</>

当然我们也可以写 class 的实现方式,代码如下:

import React from "react";
import { fetchPosts } from "./api/posts";

export default class Posts extends React.Component {
state = { loading: true, error: null, posts: null };

async componentDidMount() {
try {
this.setState({ loading: true });
const posts = await fetchPosts();
this.setState({ loading: false, posts });
} catch (error) {
this.setState({ loading: false, error });
}
}

render() {
if (this.state.loading) return "Loading...";
if (this.state.error) return "服务端错误";
if (this.state.posts && this.state.posts.length===0) return "暂无数据";

return (
<>
<h1>My Posts</h1>
<ul>
{this.state.posts.map((post) => (
<li key={post.id}>
<a href={post.url}>{post.title}</a>
</li>
))}
</ul>
</>

Mock 接口

jest.mock("./api/posts");

我们可以在官方文档中阅读关于 jest.mock 的更多信息。

它所做的就是告诉 Jest 替换​​/api/posts​​ 模块。

现在我们已经有了 mock,让我们来渲染组件,并且界面显示 loading:

import React from "react";
import { render, screen } from "@testing-library/react";
import Post from "./index";

jest.mock("./api/posts");

test("should show loading", () => {
render(<Posts);
expect(screen.getByText("Loading...")).toBeInTheDocument();
});

这是第一步,现在我们需要确保我们的 ​​fetchPosts​​ 方法被正确调用:

import React from "react";
import { render, screen } from "@testing-library/react";
import Posts from "./index";
import { fetchPosts } from "./api/posts";

jest.mock("./api/posts");

test("should show a list of posts", () => {
render(<Posts);
expect(screen.getByText("Loading...")).toBeInTheDocument();

expect(fetchPosts).toHaveBeenCalledTimes(1);
expect(fetchPosts).toHaveBeenCalledWith();

});

通过 ​​toHaveBeenCalledTimes​​​ 测试调用次数,通过 ​​toHaveBeenCalledWith​​ 测试调用方法的参数,虽然这边是空数据,但是我们也可以写,确保调用参数是空。

此时我们的测试代码会报错,因为我们没有 Mock ​​fetchPosts​​ 方法

import React from "react";
import { render, screen, wait } from "@testing-library/react";
import Posts from "./index";
import { fetchPosts } from "./api/posts";

jest.mock("./api/posts");

test("should show a list of posts", async () => {
// mock 接口
const posts = [{ id: 1, title: "My post", url: "/1" }];
fetchPosts.mockResolvedValueOnce(posts);

render(<Posts);
expect(screen.getByText("Loading...")).toBeInTheDocument();
expect(fetchPosts).toHaveBeenCalledWith();
expect(fetchPosts).toHaveBeenCalledTimes(1);

//等待标题渲染
await waitFor(() => screen.findByText("My Posts"));
posts.forEach((post) =>
expect(screen.getByText(post.title)).toBeInTheDocument()
);
});

我们使用 ​​mockResolvedValueOnce​​​ 返回一些假数据。然后,我们等待异步方法解析并等待 Posts 组件重新渲染。为此,我们使用 ​​waitFor​​ 方法,同时检查标题是否呈现,之后遍历检查,确保每一个标题在页面上。

测试接口错误

接下来我们要测试错误是否被正确呈现,那么只需要修改 mock:

test("should show an error message on failures", async () => {
fetchPosts.mockRejectedValueOnce("Error!");

render(<Posts);
expect(screen.getByText("Loading...")).toBeInTheDocument();

await waitFor(() => {
expect(screen.getByText("服务端错误")).toBeInTheDocument();
});
expect(fetchPosts).toHaveBeenCalledWith();
expect(fetchPosts).toHaveBeenCalledTimes(1);
});

小结

以下是测试异步组件的步骤:

  1. 通过 mock 使组件可以获取静态假数据;
  2. 测试加载状态;
  3. 测试异步方法是否被正确调用,并且带上了正确的参数;
  4. 测试组件是否正确地渲染了数据
  5. 测试异步方法错误时,组件是是否渲染了正确的状态

文中关于登录成功后页面跳转并未测试,那么如何测试 react 路由 ?请关注我,我会尽快出 React test 系列的下文。

希望这篇文章对大家有所帮助,也可以参考我往期的文章或者在评论区交流你的想法和心得,欢迎一起探索前端。

标签:username,异步,screen,posts,TDD,React,expect,import,password
From: https://blog.51cto.com/u_15757429/5734861

相关文章

  • react路由
    -react路由其实是借助BOM中的History对象实现的,专门有一个history.js的库,可以让操作history对象简单起来。用history.js可以通过两种方式创建路由对象:1、History.create......
  • [Typescript + React] Tip: Use generics in React to make dynamic and flexible com
    YoucanusegenericsinReacttomakeincrediblydynamic,flexiblecomponents.Here,ImakeaTablecomponentwithageneric'items'type.interfaceTableProp......
  • Spring Boot 2.x基础教程:配置@Async异步任务的线程池
    上一篇我们介绍了如何使用@Async注解来创建异步任务,我可以用这种方法来实现一些并发操作,以加速任务的执行效率。但是,如果只是如前文那样直接简单的创建来使用,可能还是会碰......
  • Spring Boot 2.x基础教程:使用@Async实现异步调用
    什么是“异步调用”?“异步调用”对应的是“同步调用”,同步调用指程序按照定义顺序依次执行,每一行程序都必须等待上一行程序执行完成之后才能执行;异步调用指程序在顺序执行......
  • 前端学习随笔 react中的方法bind(this)
    1classClickCounterextendsComponent{2constructor(props){3super(props);4this.state={count:0};5}67onClic......
  • React+hook+ts+ant design封装一个具有编辑和新增功能的页面
    前言我是歌谣我有个兄弟巅峰的时候排名c站总榜19叫前端小歌谣曾经我花了三年的时间创作了他现在我要用五年的时间超越他今天又是接近兄弟的一天人生难免坎坷大不了从......
  • vue3的Async Components异步组件
    前言:当我们的项目达到一定的规模时,对于某些组件来说,我们并不希望一开始全部加载,而是需要的时候进行加载;这样的做得目的可以很好的提高用户体验。传统方式引入组件如下,这......
  • 每日一题之Vue的异步更新实现原理是怎样的?
    最近面试总是会被问到这么一个问题:在使用vue的时候,将for循环中声明的变量i从1增加到100,然后将i展示到页面上,页面上的i是从1跳到100,还是会怎样?答案当然是只会显示100,并不会......
  • React组件之间的通信方式总结(下)
    一、写一个时钟用react写一个每秒都可以更新一次的时钟importReactfrom'react'importReactDOMfrom'react-dom'functiontick(){letele=<h1>{new......
  • React-hooks面试考察知识点汇总
    Hook简介Hook出世之前React存在的问题在组件之间复用状态逻辑很难React没有提供将可复用性行为“附加”到组件的途径(例如,把组件连接到store)。有一些解决此类问题的......