news 2026/4/26 19:31:59

开发中的测试知识—— 集成测试策略

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
开发中的测试知识—— 集成测试策略

背景问题:
需要对复杂功能进行集成测试。

方案思考:
使用端到端测试和组件集成测试相结合的策略。

具体实现:
端到端测试配置(Playwright):

# 安装 Playwrightnpminstall-D @playwright/test npx playwrightinstall

Playwright 配置:

// playwright.config.jsimport{defineConfig,devices}from'@playwright/test'exportdefaultdefineConfig({testDir:'./tests/e2e',fullyParallel:true,forbidOnly:!!process.env.CI,retries:process.env.CI?2:0,workers:process.env.CI?1:undefined,reporter:'html',use:{baseURL:'http://localhost:3000',trace:'on-first-retry',},projects:[{name:'chromium',use:{...devices['Desktop Chrome']},},{name:'firefox',use:{...devices['Desktop Firefox']},},{name:'webkit',use:{...devices['Desktop Safari']},},],webServer:{command:'npm run dev',url:'http://localhost:3000',reuseExistingServer:!process.env.CI,},})

端到端测试示例:

// tests/e2e/login.spec.jsimport{test,expect}from'@playwright/test'test.describe('Login functionality',()=>{test.beforeEach(async({page})=>{awaitpage.goto('/login')})test('should allow valid user to login',async({page})=>{// 输入用户名awaitpage.locator('input[name="username"]').fill('admin')// 输入密码awaitpage.locator('input[name="password"]').fill('admin123')// 点击登录按钮awaitpage.locator('button[type="submit"]').click()// 验证跳转到主页awaitexpect(page).toHaveURL('/dashboard')awaitexpect(page.locator('.user-info')).toContainText('admin')})test('should show error for invalid credentials',async({page})=>{// 输入错误的用户名和密码awaitpage.locator('input[name="username"]').fill('invalid')awaitpage.locator('input[name="password"]').fill('wrongpass')// 点击登录按钮awaitpage.locator('button[type="submit"]').click()// 验证错误消息显示awaitexpect(page.locator('.error-message')).toBeVisible()awaitexpect(page.locator('.error-message')).toContainText('用户名或密码错误')})test('should validate form inputs',async({page})=>{// 点击登录按钮而不填写任何内容awaitpage.locator('button[type="submit"]').click()// 验证验证错误消息awaitexpect(page.locator('.el-form-item__error')).toHaveCount(2)})})

集成测试示例:

<!-- views/UserManagement.vue --> <template> <div class="user-management"> <el-input v-model="searchTerm" placeholder="搜索用户..." @input="performSearch" /> <el-table :data="filteredUsers" v-loading="loading"> <el-table-column prop="name" label="姓名" /> <el-table-column prop="email" label="邮箱" /> <el-table-column label="操作"> <template #default="{ row }"> <el-button @click="editUser(row)">编辑</el-button> <el-button @click="deleteUser(row.id)" type="danger">删除</el-button> </template> </el-table-column> </el-table> </div> </template> <script setup> import { ref, computed, onMounted } from 'vue' import { useUserStore } from '@/stores/user' const userStore = useUserStore() const searchTerm = ref('') const loading = ref(false) // 获取过滤后的用户 const filteredUsers = computed(() => { if (!searchTerm.value) { return userStore.users } return userStore.users.filter(user => user.name.toLowerCase().includes(searchTerm.value.toLowerCase()) || user.email.toLowerCase().includes(searchTerm.value.toLowerCase()) ) }) // 执行搜索 const performSearch = () => { // 实际项目中可能会调用 API } // 编辑用户 const editUser = (user) => { // 编辑逻辑 } // 删除用户 const deleteUser = async (id) => { await userStore.deleteUser(id) } onMounted(() => { loading.value = true userStore.fetchUsers().finally(() => { loading.value = false }) }) </script>
// tests/integration/UserManagement.test.jsimport{describe,it,expect,beforeEach,vi}from'vitest'import{mount}from'@vue/test-utils'import{createTestingPinia}from'@pinia/testing'importUserManagementfrom'@/views/UserManagement.vue'describe('UserManagement Integration',()=>{letwrapperbeforeEach(()=>{// 创建测试用的 Pinia 实例consttestingPinia=createTestingPinia({createSpy:vi.fn,stubActions:false})// Mock storeconstuserStore=testingPinia.useUserStore()userStore.users=[{id:1,name:'John Doe',email:'john@example.com'},{id:2,name:'Jane Smith',email:'jane@example.com'}]wrapper=mount(UserManagement,{global:{plugins:[testingPinia]}})})it('displays users from store',async()=>{awaitwrapper.vm.$nextTick()// 等待组件更新consttableRows=wrapper.findAll('tbody tr')expect(tableRows).toHaveLength(2)constfirstRowCells=tableRows[0].findAll('td')expect(firstRowCells[0].text()).toContain('John Doe')expect(firstRowCells[1].text()).toContain('john@example.com')})it('filters users based on search term',async()=>{constsearchInput=wrapper.find('input')// 搜索 JohnawaitsearchInput.setValue('John')awaitsearchInput.trigger('input')consttableRows=wrapper.findAll('tbody tr')expect(tableRows).toHaveLength(1)expect(tableRows[0].find('td').text()).toContain('John Doe')})it('deletes user when delete button is clicked',async()=>{constuserStore=wrapper.vm.$pinia.state.value.userexpect(userStore.users).toHaveLength(2)// 点击第一个删除按钮constdeleteButtons=wrapper.findAll('button[type="danger"]')awaitdeleteButtons[0].trigger('click')// 验证 store 中的用户数量减少expect(userStore.users).toHaveLength(1)expect(userStore.users[0].id).toBe(2)})})

模拟 API 服务:

// tests/mocks/server.jsimport{setupServer}from'msw/node'import{http,HttpResponse}from'msw'consthandlers=[http.get('/api/users',()=>{returnHttpResponse.json({code:200,data:[{id:1,name:'John Doe',email:'john@example.com'},{id:2,name:'Jane Smith',email:'jane@example.com'}],total:2})}),http.post('/api/users',async({request})=>{constuserData=awaitrequest.json()returnHttpResponse.json({code:200,data:{...userData,id:Date.now()}})}),http.delete('/api/users/:id',({params})=>{const{id}=paramsreturnHttpResponse.json({code:200,message:`User${id}deleted`})})]exportconstserver=setupServer(...handlers)
// tests/setup.js (更新)import{afterAll,afterEach,beforeAll}from'vitest'import{server}from'./mocks/server'// 开始 mock 服务器beforeAll(()=>server.listen({onUnhandledRequest:'error'}))// 清理请求处理程序afterEach(()=>server.resetHandlers())// 关闭服务器afterAll(()=>server.close())
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/22 1:46:03

MATLAB分形维数计算:1D/2D/3D图形的盒维数实现

一、盒维数计算原理 盒维数&#xff08;Box-counting dimension&#xff09;通过统计覆盖分形对象所需不同尺寸的盒子数量&#xff0c;建立盒子尺寸与数量的幂律关系&#xff0c;其分形维数 D满足&#xff1a;其中 N(ϵ)N(ϵ)N(ϵ)为覆盖对象所需边长为 ϵϵϵ的盒子数量。二、…

作者头像 李华
网站建设 2026/4/17 1:31:52

基于STM32的智能家居安防系统

目录 STM32智能家居安防系统概述核心功能模块软件设计要点低功耗优化扩展功能典型硬件选型 源码文档获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&#xff01; STM32智能家居安防系统概述 STM32系列微控制器因其高性能、低功耗和丰富的外设资源&#xff…

作者头像 李华
网站建设 2026/4/26 13:26:10

Python篇---提升Python代码性能

提升Python代码性能&#xff1a;从蜗牛到火箭的秘诀 一、一句话理解 提升代码性能 减少计算机的"工作量" 选择更快的"工具" 就像送外卖&#xff1a; ❌ 一次只送一单&#xff0c;来回跑10次&#xff08;慢&#xff09; ✅ 一次规划好路线&#xff0c…

作者头像 李华
网站建设 2026/4/26 2:39:09

Composition 英文单词学习

1️、基本信息 单词&#xff1a;composition 词性&#xff1a;名词 发音&#xff1a; &#x1f1fa;&#x1f1f8; /ˌkɑːm.pəˈzɪʃ.ən/&#x1f1ec;&#x1f1e7; /ˌkɒm.pəˈzɪʃ.ən/ 词源&#xff1a; 来自拉丁语 componere&#xff08;一起放置 → 组合、构成…

作者头像 李华
网站建设 2026/4/17 1:33:00

导师严选9个AI论文平台,自考学生轻松搞定毕业论文!

导师严选9个AI论文平台&#xff0c;自考学生轻松搞定毕业论文&#xff01; AI 工具助力论文写作&#xff0c;高效又省心 在当前的学术环境中&#xff0c;越来越多的学生开始借助 AI 工具来辅助论文写作。尤其对于自考学生而言&#xff0c;时间紧张、资料繁杂、写作经验不足等问…

作者头像 李华
网站建设 2026/4/22 13:58:17

戒了么4.0 戒色签到打卡源码

内容目录 一、详细介绍二、效果展示1.部分代码2.效果图展示 三、学习资料下载 一、详细介绍 最近死了么很火&#xff0c;于是我就想写一个戒色的&#xff0c;这款程序也就应运而生 原作者&#xff1a;阿宅 戒了么是一个轻量、纯 PHP 的每日戒色签到程序&#xff0c;面向正在自…

作者头像 李华