持续创作,加速成长!这是我参与「掘金日新计划 · 10 月更文挑战」的第19天,点击查看活动详情
在 单元测试框架 Jest study(一) 搭建好项目,对 Jest 这门框架有了一定的了解后,我们进入项目实战,以此加深学习记忆。
测试逻辑代码
准备工作
mockData.js文件导出userlist
用户列表信息
export const userlist = [{
"id": 1,
"name": "Leanne Graham",
"username": "Bret",
"email": "Sincere@april.biz",
"address": {
"street": "Kulas Light",
"suite": "Apt. 556",
"city": "Gwenborough",
"zipcode": "92998-3874",
"geo": {
"lat": "-37.3159",
"lng": "81.1496"
}
},
"phone": "1-770-736-8031 x56442",
"website": "hildegard.org",
"company": {
"name": "Romaguera-Crona",
"catchPhrase": "Multi-layered client-server neural-net",
"bs": "harness real-time e-markets"
}
},
{
"id": 2,
"name": "Ervin Howell",
"username": "Antonette",
"email": "Shanna@melissa.tv",
"address": {
"street": "Victor Plains",
"suite": "Suite 879",
"city": "Wisokyburgh",
"zipcode": "90566-7771",
"geo": {
"lat": "-43.9509",
"lng": "-34.4618"
}
},
"phone": "010-692-6593 x09125",
"website": "anastasia.net",
"company": {
"name": "Deckow-Crist",
"catchPhrase": "Proactive didactic contingency",
"bs": "synergize scalable supply-chains"
}
},
{
"id": 3,
"name": "Clementine Bauch",
"username": "Samantha",
"email": "Nathan@yesenia.net",
"address": {
"street": "Douglas Extension",
"suite": "Suite 847",
"city": "McKenziehaven",
"zipcode": "59590-4157",
"geo": {
"lat": "-68.6102",
"lng": "-47.0653"
}
},
"phone": "1-463-123-4447",
"website": "ramiro.info",
"company": {
"name": "Romaguera-Jacobson",
"catchPhrase": "Face to face bifurcated interface",
"bs": "e-enable strategic applications"
}
},
{
"id": 4,
"name": "Patricia Lebsack",
"username": "Karianne",
"email": "Julianne.OConner@kory.org",
"address": {
"street": "Hoeger Mall",
"suite": "Apt. 692",
"city": "South Elvis",
"zipcode": "53919-4257",
"geo": {
"lat": "29.4572",
"lng": "-164.2990"
}
},
"phone": "493-170-9623 x156",
"website": "kale.biz",
"company": {
"name": "Robel-Corkery",
"catchPhrase": "Multi-tiered zero tolerance productivity",
"bs": "transition cutting-edge web services"
}
},
{
"id": 5,
"name": "Chelsey Dietrich",
"username": "Kamren",
"email": "Lucio_Hettinger@annie.ca",
"address": {
"street": "Skiles Walks",
"suite": "Suite 351",
"city": "Roscoeview",
"zipcode": "33263",
"geo": {
"lat": "-31.8129",
"lng": "62.5342"
}
},
"phone": "(254)954-1289",
"website": "demarco.info",
"company": {
"name": "Keebler LLC",
"catchPhrase": "User-centric fault-tolerant solution",
"bs": "revolutionize end-to-end systems"
}
},
{
"id": 6,
"name": "Mrs. Dennis Schulist",
"username": "Leopoldo_Corkery",
"email": "Karley_Dach@jasper.info",
"address": {
"street": "Norberto Crossing",
"suite": "Apt. 950",
"city": "South Christy",
"zipcode": "23505-1337",
"geo": {
"lat": "-71.4197",
"lng": "71.7478"
}
},
"phone": "1-477-935-8478 x6430",
"website": "ola.org",
"company": {
"name": "Considine-Lockman",
"catchPhrase": "Synchronised bottom-line interface",
"bs": "e-enable innovative applications"
}
},
{
"id": 7,
"name": "Kurtis Weissnat",
"username": "Elwyn.Skiles",
"email": "Telly.Hoeger@billy.biz",
"address": {
"street": "Rex Trail",
"suite": "Suite 280",
"city": "Howemouth",
"zipcode": "58804-1099",
"geo": {
"lat": "24.8918",
"lng": "21.8984"
}
},
"phone": "210.067.6132",
"website": "elvis.io",
"company": {
"name": "Johns Group",
"catchPhrase": "Configurable multimedia task-force",
"bs": "generate enterprise e-tailers"
}
},
{
"id": 8,
"name": "Nicholas Runolfsdottir V",
"username": "Maxime_Nienow",
"email": "Sherwood@rosamond.me",
"address": {
"street": "Ellsworth Summit",
"suite": "Suite 729",
"city": "Aliyaview",
"zipcode": "45169",
"geo": {
"lat": "-14.3990",
"lng": "-120.7677"
}
},
"phone": "586.493.6943 x140",
"website": "jacynthe.com",
"company": {
"name": "Abernathy Group",
"catchPhrase": "Implemented secondary concept",
"bs": "e-enable extensible e-tailers"
}
},
{
"id": 9,
"name": "Glenna Reichert",
"username": "Delphine",
"email": "Chaim_McDermott@dana.io",
"address": {
"street": "Dayna Park",
"suite": "Suite 449",
"city": "Bartholomebury",
"zipcode": "76495-3109",
"geo": {
"lat": "24.6463",
"lng": "-168.8889"
}
},
"phone": "(775)976-6794 x41206",
"website": "conrad.com",
"company": {
"name": "Yost and Sons",
"catchPhrase": "Switchable contextually-based project",
"bs": "aggregate real-time technologies"
}
},
{
"id": 10,
"name": "Clementina DuBuque",
"username": "Moriah.Stanton",
"email": "Rey.Padberg@karina.biz",
"address": {
"street": "Kattie Turnpike",
"suite": "Suite 198",
"city": "Lebsackbury",
"zipcode": "31428-2261",
"geo": {
"lat": "-38.2386",
"lng": "57.2232"
}
},
"phone": "024-648-3804",
"website": "ambrose.net",
"company": {
"name": "Hoeger LLC",
"catchPhrase": "Centralized empowering task-force",
"bs": "target end-to-end models"
}
}
];
redux准备
store.js:
import { createStore, combineReducers} from 'redux';
import reducers from './reducers/index';
const store = createStore(combineReducers(reducers));
export default store;
reducers/index.js:
import userlist from './userlistReducer';
export default {
userlist
}
userlistReducer.js:
const defaultState = [];
const userlist = (state = defaultState, action) => {
switch (action.type) {
case 'UPDATE_USERLIST':
return action.payload;
default:
return state;
}
};
export default userlist;
- test 文件开始 coding
import { mount } from "enzyme";
import { Provider } from "react-redux";
import { BrowserRouter } from "react-router-dom";
import Landing from "../pages/Landing"; // 登录注册的页面,请根据实际情况导入
import Home from "../pages/Home"; // 退出登录的页面,请根据实际情况导入
import store from "../redux/store"; // 一样的道理,根据实际情况导入
import { userlist } from "./mockData"; // mock 数据
describe("身份验证", () => {
beforeEach(() => {
Object.defineProperty(window, "localStorage", {
value: {
getItem: jest.fn(() => null),
setItem: jest.fn(() => null),
removeItem: jest.fn(() => null),
},
writable: true,
});
});
}
登录
describe("身份验证", () => {
beforeEach(() => {...};
it("用户登录", () => {
store.dispatch({
type: "UPDATE_USERLIST",
payload: userlist,
});
const wrapper = mount(
<Provider store={store}>
<BrowserRouter>
<Landing />
</BrowserRouter>
</Provider>
);
let accountCom = wrapper
.find(TextField)
.at(8)
.find("input");
let passwordCom = wrapper
.find(TextField)
.at(9)
.find("input");
// account name 输入框模拟输入正确账号Bret
accountCom.simulate("change", { target: { value: "Bret" } });
// password 输入框模拟输入密码 Kulas Light
passwordCom.simulate("change", { target: { value: "Kulas Light" } });
// 是否account name模拟赋值成功
expect(
wrapper
.find(TextField)
.at(8)
.find("input")
.prop("value")
).toBe("Bret");
// 是否password 模拟赋值成功
expect(
wrapper
.find(TextField)
.at(9)
.find("input")
.prop("value")
).toBe("Kulas Light");
//模拟点击login按钮
wrapper
.find(Button)
.at(1)
.find("button")
.simulate("click");
// 因为输入的是正确的密码,所以会在本地存储localStroage.userinfo生成值
expect(window.localStorage.setItem).toHaveBeenCalledTimes(1);
});
}
注册
describe("身份验证", () => {
beforeEach(() => {...};
it("用户登录", () => {...};
it("用户注册", () => {
store.dispatch({
type: "UPDATE_USERLIST",
payload: userlist,
});
const wrapper = mount(
<Provider store={store}>
<BrowserRouter>
<Landing />
</BrowserRouter>
</Provider>
);
let accountCom = wrapper
.find(TextField)
.at(8)
.find("input");
let passwordCom = wrapper
.find(TextField)
.at(9)
.find("input");
// account name 输入框模拟输入账号user123
accountCom.simulate("change", { target: { value: "user123" } });
// password 输入框模拟输入密码 123456
passwordCom.simulate("change", { target: { value: "123456" } });
// 是否account name模拟赋值成功
expect(
wrapper
.find(TextField)
.at(8)
.find("input")
.prop("value")
).toBe("user123");
// 是否password 模拟赋值成功
expect(
wrapper
.find(TextField)
.at(9)
.find("input")
.prop("value")
).toBe("123456");
//模拟点击login按钮
wrapper
.find(Button)
.at(1)
.find("button")
.simulate("click");
// 因为输入的是错误的密码,所以会提示User name or password that does not exist!
expect(wrapper.find(Alert).prop("children")).toBe(
"User name or password that does not exist!"
);
});
}
退出登录
describe("身份验证", () => {
beforeEach(() => {...};
it("用户登录", () => {...};
it("用户注册", () => {...};
it("退出登录", () => {
const wrapper = mount(
<Provider store={store}>
<BrowserRouter>
<Home />
</BrowserRouter>
</Provider>
);
// 模拟点击登出
wrapper.find("#logout").simulate("click");
// 当点击登出按钮,会移除本地缓存localStorage.userinfo
expect(window.localStorage.removeItem).toHaveBeenCalledTimes(1);
});
}