Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

1111 #1

Merged
merged 13 commits into from
Mar 10, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
13 commits
Select commit Hold shift + click to select a range
bcf7534
添加多个组件和自定义 Hook,更新 CI 配置以支持单元测试和覆盖率检查
TD-gaowei Mar 10, 2025
b79a69e
更新 CI 配置,使用 --legacy-peer-deps 安装依赖,并修改测试命令以收集覆盖率
TD-gaowei Mar 10, 2025
4a3cddd
更新 CI 配置,将 Node.js 版本从 16 升级到 20
TD-gaowei Mar 10, 2025
778e646
更新 CI 配置,添加测试报告输出,修改测试命令以支持 JUnit 格式
TD-gaowei Mar 10, 2025
5247c78
更新 CI 配置,添加测试报告输出,修改测试命令以支持 JUnit 格式
TD-gaowei Mar 10, 2025
e5f98e9
更新 CI 配置,修改覆盖率报告路径并增强错误处理,更新测试命令以支持 lcov 格式
TD-gaowei Mar 10, 2025
b2f6787
更新 CI 配置,修改测试命令以收集覆盖率,调整 .gitignore 文件以保留覆盖率目录
TD-gaowei Mar 10, 2025
4f644ec
更新 CI 配置,添加调试覆盖率文件步骤,升级 Codecov Action 版本并增强覆盖率检查功能
TD-gaowei Mar 10, 2025
ce66d07
更新 Vitest 配置,修改覆盖率提供者为 c8,添加 lcov 和文本报告格式,调整报告目录及排除项
TD-gaowei Mar 10, 2025
2a0094c
更新测试命令,使用 c8 作为覆盖率工具并支持 lcov 格式
TD-gaowei Mar 10, 2025
99e48e4
更新 CI 配置,修正测试命令为 npm run,调整 Vitest 覆盖率报告顺序
TD-gaowei Mar 10, 2025
478a01c
更新 CI 配置,移除多余的覆盖率检查步骤,简化工作流
TD-gaowei Mar 10, 2025
c6671c6
更新 CI 配置,改进覆盖率检查逻辑,确保覆盖率数据有效性并输出当前覆盖率
TD-gaowei Mar 10, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
58 changes: 58 additions & 0 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
name: CI - Unit Tests & Coverage

on:
pull_request:
branches:
- main # 或者你用于主开发分支的其他名称
push:
branches:
- main

jobs:
test:
runs-on: ubuntu-latest

steps:
- name: Checkout code
uses: actions/checkout@v3

- name: Set up Node.js
uses: actions/setup-node@v3
with:
node-version: "20"

- name: Install dependencies
run: npm install --legacy-peer-deps

- name: Run unit tests and collect coverage
run: npm run test:coverage
env:
CI: true

- name: Debug coverage files
run: ls -R coverage

- name: Upload coverage to Codecov
uses: codecov/codecov-action@v5
with:
token: ${{ secrets.CODECOV_TOKEN }}
slug: TD-gaowei/unit-test-react
files: ./coverage/lcov.info # 修改为你实际的覆盖率报告路径
fail_ci_if_error: true
root_dir: /home/runner/work/unit-test-react/unit-test-react
verbose: true
flags: unittests
# name: codecov-umbrella

- name: Fail the PR if coverage drops in the new commit
run: |
COMMIT_SHA=${{ github.event.pull_request.head.sha }}
coverage=$(curl -s https://codecov.io/api/gh/${{ github.repository }}/commit/$COMMIT_SHA \
| jq -r '.commit.totals.coverage')

echo "Current coverage: $coverage"

if [[ -z "$coverage" || $(echo "$coverage < 80" | bc) -eq 1 ]]; then
echo "Coverage is below 80% or not found, failing PR"
exit 1
fi
2 changes: 1 addition & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -24,4 +24,4 @@ dist-ssr
*.sw?

node_modules
coverage
# coverage
15 changes: 11 additions & 4 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,13 @@
- 工具方法测试 ✅
- 组件 UI 测试 ✅
- 组件 UI 交互测试 ✅
- React Hooks 测试 ✅. -> 本质上是函数测试

如果可以用当作 hooks 测试,那就用 Hooks 的方式测试,要么可以当下组件中进行测试

- 工具方法测试
- 组件UI测试
- 组件UI交互测试
- React Hooks测试
- 使用 Zustand 的 hooks 测试 ✅
- 快照测试 ✅
- 异步测试 ✅
- 异常测试 ✅
- 代码覆盖率测试 ✅
- 边界测试 - 本质上是多种输入不同参数测试的组合 ✅
6 changes: 4 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -8,12 +8,13 @@
"lint": "eslint .",
"preview": "vite preview",
"test": "vitest",
"test:coverage": "vitest --coverage"
"test:coverage": "npx c8 --reporter=lcov vitest"
},
"dependencies": {
"@testing-library/jest-dom": "^6.6.3",
"react": "18.3.1",
"react-dom": "18.3.1"
"react-dom": "18.3.1",
"zustand": "^5.0.3"
},
"devDependencies": {
"@eslint/js": "^9.21.0",
Expand All @@ -24,6 +25,7 @@
"@types/react-dom": "^19.0.4",
"@vitejs/plugin-react": "^4.3.4",
"@vitest/coverage-v8": "3.0.8",
"c8": "^10.1.3",
"eslint": "^9.21.0",
"eslint-plugin-react-hooks": "^5.1.0",
"eslint-plugin-react-refresh": "^0.4.19",
Expand Down
5 changes: 5 additions & 0 deletions src/App.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@ import { useState } from "react";
import reactLogo from "./assets/react.svg";

import "./App.css";
import Login from "./components/Login/Login";
import UserList from "./components/UserList/UserList";

function App() {
const [count, setCount] = useState(0);
Expand All @@ -25,6 +27,9 @@ function App() {
<p className="read-the-docs">
Click on the Vite and React logos to learn more
</p>

<Login />
<UserList />
</>
);
}
Expand Down
36 changes: 36 additions & 0 deletions src/components/AccountList/AccountList.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
import React, { useState, useEffect } from "react";

export default function AccountList() {
const [users, setUsers] = useState([]);
const [error, setError] = useState(null);

useEffect(() => {
// 模拟API请求
async function fetchUsers() {
try {
const response = await fetch("/api/users");
if (!response.ok) {
throw new Error("Failed to fetch users");
}
const data = await response.json();
setUsers(data);
} catch (err) {
setError(err.message);
}
}

fetchUsers();
}, []);

if (error) {
return <div>Error: {error}</div>;
}

return (
<ul>
{users.map((user) => (
<li key={user.id}>{user.name}</li>
))}
</ul>
);
}
3 changes: 3 additions & 0 deletions src/components/Button/Button.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
export default function Button({ label }) {
return <button>{label}</button>;
}
41 changes: 41 additions & 0 deletions src/components/Login/Login.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
import React, { useState } from "react";

const Login = () => {
const [username, setUsername] = useState("");
const [password, setPassword] = useState("");

const handleSubmit = (e) => {
e.preventDefault();
// Handle login logic here
console.log("Username:", username);
console.log("Password:", password);
};

return (
<div className="login-container">
<form onSubmit={handleSubmit}>
<div className="form-group">
<label htmlFor="username">Username:</label>
<input
type="text"
id="username"
value={username}
onChange={(e) => setUsername(e.target.value)}
/>
</div>
<div className="form-group">
<label htmlFor="password">Password:</label>
<input
type="password"
id="password"
value={password}
onChange={(e) => setPassword(e.target.value)}
/>
</div>
<button type="submit">登录</button>
</form>
</div>
);
};

export default Login;
44 changes: 44 additions & 0 deletions src/components/Login/__tests__/Login.test.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
import { render, screen, fireEvent } from "@testing-library/react";

import Login from "../Login";
import { expect } from "vitest";

describe("Login Component", () => {
it("renders the login form", () => {
render(<Login />);
expect(screen.getByLabelText(/username/i)).toBeInTheDocument();
expect(screen.getByLabelText(/password/i)).toBeInTheDocument();
expect(screen.getByRole("button", { name: /登录/i })).toBeInTheDocument();
});

it("updates the username and password fields", () => {
render(<Login />);
const usernameInput = screen.getByLabelText(/username/i);
const passwordInput = screen.getByLabelText(/password/i);

fireEvent.change(usernameInput, { target: { value: "testuser" } });
fireEvent.change(passwordInput, { target: { value: "password123" } });

expect(usernameInput.value).toBe("testuser");
expect(passwordInput.value).toBe("password123");
});

it("handles form submission", () => {
render(<Login />);
const usernameInput = screen.getByLabelText(/username/i);
const passwordInput = screen.getByLabelText(/password/i);
const button = screen.getByRole("button", { name: /登录/i });

const spyConsole = vi.spyOn(console, "log").mockImplementation(() => {});

fireEvent.change(usernameInput, { target: { value: "testuser" } });
fireEvent.change(passwordInput, { target: { value: "password123" } });

fireEvent.click(button);

expect(usernameInput.value).toBe("testuser");
expect(passwordInput.value).toBe("password123");

expect(spyConsole).toHaveBeenCalledTimes(2);
});
});
26 changes: 26 additions & 0 deletions src/components/UserList/UserList.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
import React, { useEffect, useState } from "react";

const mockUsers = [{ name: "Alice" }, { name: "Bob" }, { name: "Charlie" }];

const UserList = () => {
const [users, setUsers] = useState([]);

useEffect(() => {
setTimeout(() => {
setUsers(mockUsers);
}, 2000);
});

return (
<div>
<h2>User List</h2>
<ul>
{users.map((user, index) => (
<li key={index}>{user.name}</li>
))}
</ul>
</div>
);
};

export default UserList;
63 changes: 63 additions & 0 deletions src/components/__tests__/AccountList.test.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
// AccountList.test.js
import { render, screen, waitFor } from "@testing-library/react";
import AccountList from "../AccountList/AccountList";

// 模拟 API 请求
global.fetch = vi.fn();

describe("AccountList", () => {
it("should display users after successful fetch", async () => {
// 模拟成功的 API 响应
fetch.mockResolvedValueOnce({
ok: true,
json: async () => [
{ id: 1, name: "Alice" },
{ id: 2, name: "Bob" },
],
});

render(<AccountList />);

// 等待用户数据加载
await waitFor(() => screen.getByText("Alice"));
await waitFor(() => screen.getByText("Bob"));

// 检查用户是否渲染
expect(screen.getByText("Alice")).toBeInTheDocument();
expect(screen.getByText("Bob")).toBeInTheDocument();
});

it("should display an error message when fetch fails", async () => {
// 模拟失败的 API 响应
fetch.mockRejectedValueOnce(new Error("Failed to fetch users"));

render(<AccountList />);

// 等待错误消息渲染
await waitFor(() => screen.getByText(/Error:/));

// 检查错误消息
expect(
screen.getByText("Error: Failed to fetch users")
).toBeInTheDocument();
});

it("should display an error message when API response is not ok", async () => {
// 模拟 API 返回失败的响应(非 2xx 状态)
fetch.mockResolvedValueOnce({
ok: false,
status: 500,
json: async () => ({ message: "Internal Server Error" }),
});

render(<AccountList />);

// 等待错误消息渲染
await waitFor(() => screen.getByText(/Error:/));

// 检查错误消息
expect(
screen.getByText("Error: Failed to fetch users")
).toBeInTheDocument();
});
});
13 changes: 13 additions & 0 deletions src/components/__tests__/Button.test.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
// Button.test.js
import { render } from "@testing-library/react";
import Button from "../Button/Button";

describe("Button Component Snapshot", () => {
it("should match the snapshot", () => {
// 渲染组件
const { asFragment } = render(<Button label="Click Me" />);

// 使用快照 API 将渲染结果与之前保存的快照进行比较
expect(asFragment()).toMatchSnapshot();
});
});
22 changes: 22 additions & 0 deletions src/components/__tests__/UserList.test.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
import { render, screen, waitFor } from "@testing-library/react";
import UserList from "../UserList/UserList";
import { expect } from "vitest";

describe("UserList component", () => {
it("renders User List heading", () => {
render(<UserList />);
expect(screen.getByText("User List")).toBeInTheDocument();
});

it("renders users after timeout", async () => {
render(<UserList />);

const alice = await screen.findByText("Alice", {}, { timeout: 3000 });
const bob = await screen.findByText("Bob", {}, { timeout: 3000 });
const charlie = await screen.findByText("Charlie", {}, { timeout: 3000 });

expect(alice).toBeInTheDocument();
expect(bob).toBeInTheDocument();
expect(charlie).toBeInTheDocument();
});
});
9 changes: 9 additions & 0 deletions src/components/__tests__/__snapshots__/Button.test.jsx.snap
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html

exports[`Button Component Snapshot > should match the snapshot 1`] = `
<DocumentFragment>
<button>
Click Me
</button>
</DocumentFragment>
`;
Loading