实例展示vue单元测试及难题解惑

通过生动详实的例子带你排遍vue单元测试过程中的所有疑惑与难题。

技术栈:jest、vue-test-utils。

共四个部分:运行时、Mock、Stub、Configuring和CLI。

运行时

在跑测试用例时,大家的第一个绊脚石肯定是各种undifned报错。

解决这些报错的血泪史还历历在目,现在总结来看,大都是缺少运行时变量抑或异步造成的。

这里咱们只说运行时,基本就这两类:

1.缺少window等环境变量

一般通过引入global-jsdom解决,这也是官方推荐的。当然我们也可以自己在测试代码中直接声明定义。

比如我们在业务代码中使用了sessionStorage。

// procudtpay.vue
<script>
const sessionParams = window.sessionStorage.getItem('sessionParams')
export default {
  data () { }
}
</script>

然后在测试代码中直接重定义,这样在运行时,实际取到的值就是我们在这里定义的。

// procudtpay.spec.js
window.sessionStorage = {
  getItem: () => {
    return { name:'name', type:'type' }
  }
}
import procudtpay from '../views/procudtpay.vue'

这里关于执行顺序做一点额外说明:

示例中sessionParams的赋值是在import引入.vue模块就执行了的,所以对sessionStorage的定义赋值需要在引入之前。

如果你的sessionStorage取值是在vue实例化后,比如created中,那么则没有该问题。

2.缺少在main.js中定义/注册的全局属性和方法

这些就需要在测试代码中引入同款,以及通过mount的配置项mocks和stubs,分别对其进行mock或者存根了。

// main.js
import Vue from 'vue'
import Mint from 'mint-ui'
import '../filter'
import axios from 'axios'
Vue.use(Mint)
Vue.prototype.$post = (url, params) => {
  return axios.post(url, params).then(res => res.data)
}
Vue.filter('filterxxx', function (value) {
  // bala bala ba…
})

// xxx.spec.js
import Vue from 'vue'
import '../../filter/filter'   // 引入注册同款过滤器
Vue.filter('filterxxx', function (value) {
  // bala bala ba…
})
import { $post } from './http.js' 
it('快照测试', () => {
    const wrapper = shallowMount(ProductPay, {
      mocks: {
        $post  // 用自己定义的mock数据取代真实http请求
      },
      stubs:['mt-header'] // 存根组件
    })
    // ...
})

通常其他测试文件也会依赖这些全局变量,我们可以通过配置jest的setupFiles实现复用。

Mock

我翻开代码一看,这代码没有注释,歪歪斜斜的每一行都写着‘断言正确’四个字。我横竖睡不着,仔细看了半夜,才从字缝里看出字来,满屏都写着两个字:“造假”!

正应了那一句:人(ce)生(shi)如戏,全靠演技(mock)。总之,mock老重要了。

1.mock简单函数

我们从最简单的mock一个函数开始。

比如我们现在想要测试:当用户购买成功,期望页面能跳转到结果页。

// productpay.vue
<script>
export default {
    ...
    methods:{
        commmit () {
          this.$post('xxx', params).then(data => {
            this.$router.push(`/payresult`)
        })
       }
    }
}
</script>

那么,我们可以通过mock掉$router的push方法,然后断言它有被调用且参数正确,达成测试目的。

// productpay.spec.js
it('当用户购买成功后,页面应该跳转至结果页', async () => {
    const mockFunc = jest.fn()
    const wrapper = shallowMount(ProductPay, {
      mocks: {
        $post,
        $router: {
          push: mockFunc
        }
      }
    })
    
    wrapper.vm.commmit() // 提交购买
    
    expect(mockFunc).toHaveBeenCalledWith('/payresult')
})

2.mockHttp请求,指定返回结果

http请求和上面例子中的$router的区别是,它需要返回值。jest有多种方式指定返回值,这里用的是mockImplementation。

// test/**.spec.js
it('当用户xxxx,应该xxxx', async () => {
    const respSuccess = { data: [...], code:0 }
    const respError = { data: [...], code:888 }
    // 定义mock函数
    const mockPost = jest.fn() 
    const wrapper = shallowMount(index, { 
       mocks: {
        $post:mockPost // 应用该mock函数
        }
   })
   // 指定异步返回数据
   mockPost.mockImplementation(() => Promise.resolve(respError))
   // 可以对调用情况进行断言
   expect(mockPost).toHaveBeenCalled() 
  
   mockPost.mockImplementation(() => Promise.resolve(respSuccess))
   //也可以等待异步结束,对结果进行断言
   await flushPromises()
   expect(wrapper.vm.list).toEqual(respSuccess.data)
})

实际上我们项目中调用的接口会很多,且不乏返回大量数据的情况。如果这些都定义在测试代码里就会很臃肿。这时候,我们可以对该功能做个简单的模块化。

// 常见的业务代码
// main.js中把axios挂载到了vue实例
Vue.prototype.$post = (url, params) => {
  return axios.post(url, params).then(res => res.data)
}
// Index.vue中的请求
getProductList () {
    this.$post('/ProductListQry', {}).then(data => {
        this.ProductList = data.List
    })
}
// 1. 在单独js中存放模拟数据 data/ProductListQry.js
export default {
    data:[{ id:1,name:'name',...},...],
    code:0
}

// 2. 定义post方法,并做个数据匹配 test/http.js
import ProductListQry from '@/data/ProductListQry.js'
const mockData = {
  ProductListQry,
  ... //可以用同样的方式引入更多mock数据
}
const $post = (url = '') => {
  return new Promise((resolve, reject) => {
    const jsName = String(url).split('/')[1]
    resolve(mockData[jsName])
  })
}
export { $post }

// 3. 引入并使用 test/index.spec.js
import Index from '@/views/Index.vue'
import { $post } from './http.js'
it('...',()=>{
    const wrapper = shallowMount(Index, {
      mocks: {
        $post
      }
    })
    wrapper.vm.getProductList() //触发请求
    await flushPromises() //等待异步请求结束
    //可以看到wrapper中就有了我们指定的模拟数据
    console.log(wrapper.vm.ProductList) 
})

同理,如果要测试请求失败的情形,可以再定义一个返回错误数据的方法,比如就叫$postError。

// test/**.spec.js
import { $postError } from './http.js'
it('...',()=>{
    const wrapper = shallowMount(Index, {
        mocks: {
            $post:$postError
        }
    })
    
    wrapper.vm.getProductList() //触发请求
    await flushPromises() //等待异步请求结束
    
    // 我们就可以就获取到错误数据的场景进行测试了
    console.log(wrapper.vm.ProductList) 
})

3.mock整个模块

当业务代码中直接使用了引入的组件/方法时,我们对其测试可能就需要mock整个模块。下面是一个用弹窗做表单验证的场景:

// productpay.vue
<script>
import { MessageBox } from '../Component'
export default {
    methods:{
        makeSurebuy () {
            let payAmount = delcommafy(this.payAmount)
                if (!payAmount) {
                    MessageBox({
                    message: '请先输入购买金额'
                })
                return
            }
            if (payAmount < this.resData.BaseAmt) {
                MessageBox({
                    message: '购买金额不能小于起存金额'
                })
                return
            }
            if (payAmount > this.Balance) {
                MessageBox({
                    message: '购买金额不能大于可用余额'
                })
                return
            }
            // 校验通过,发起交易...
        }
    }
}
<script>

//productpay.spce.js
import Component from '../Component'
jest.mock('../../../components/ZyComponent')

it('当用户点击购买按钮,如果输入非法金额,应该有相应的错误提示', async () => {
    wrapper.findAll('.btn-commit').at(0).trigger('click')
    expect(Component.MessageBox.mock.calls[0][0])
        .toEqual({ message: '请先输入购买金额' })
    
    wrapper.setData({payAmount: '100'})
    
    wrapper.findAll('.btn-commit').at(0).trigger('click')
    expect(Component.MessageBox.mock.calls[1][0])
        .toEqual({ message: '购买金额不能小于起存金额' })
    
    wrapper.setData({payAmount: '100000000000000000'})
    
    wrapper.findAll('.btn-commit').at(0).trigger('click')
    expect(Component.MessageBox.mock.calls[2][0])
        .toEqual({ message: '购买金额不能大于可用余额' })
})

我们通过jest.mock()mock整个模块,当该模块的方法被调用后它就会有一个mock属性,可以通过ZyComponent.ZyMessageBox.mock进行访问,其中ZyComponent.ZyMessageBox.mock.calls会返回被调用情况的数组,我们可以根据这个数据对函数被调用次数、入参情况进行断言测试。

Stub存根组件

进行单元测试,理论上我们不用、也不应该在它的测试用例中测试子组件,不然就叫集成测试了。vue-test-utils是通过配置stubs实现对组件mock的。

const wrapper = shallowMount(index, {
    stubs: ['mt-header', 'mt-loadmore']
}

但是业务中难免会有调用子组件方法的时候,比如说mint-ui的loadmore。

// procuctlist.vue
<script>
export default {
    ...
    methods:{
        getProductList () {
          this.$post('xxx', params).then(data => {
          ...
            this.ProductList = this.ProductList.concat(data.List)
            this.$refs.loadmore.onBottomLoaded()
        })
       }
    }
}
</script>

这时候我们是可以改用mount方法使页面渲染子组件,这样通过$refs就能正常的获取到子组件实例。但更合适的做法应该是自定义存根组件的内部实现,以满足测试需求。

// procuctlist.spec.js
it('当用户上拉产品列表,应该能看到的更多的产品', () => {
    const mockOnBottomLoaded = jest.fn()
    const mtLoadMore = {
      render: () => { },
      methods: {
        onBottomLoaded: mockOnBottomLoaded
      }
    }
    const mtHeader = {
      render: () => { }
    }
    const wrapper = shallowMount(Index, {
      stubs: { 'mt-loadmore': mtLoadMore, 'mt-header': mtHeader },
      mocks: {
        $post
      }
    })
    const currentPage = wrapper.vm.currentPage

    wrapper.vm.loadMoreProduction()

    expect(wrapper.vm.currentPage).toEqual(currentPage + 1)
    expect(mockOnBottomLoaded).toHaveBeenCalled()
})

最后提一嘴,存根组件后,业务代码中子组件还是会被引入的,只是没有被实例化和渲染。

Configuring和CLI

1.统计代码覆盖率忽略某些文件

使用coveragePathIgnorePatterns配置即可,把这个列出来是应为我遇到两个项目相同配置,有一个死活不生效的问题。最后才从官方文档中得知是babel插件istanbul问题。目前还未解决,只是粗暴的在.balelrc中把istanbul去掉了。有真正解决方案的大佬,留言教下……跪谢。

// jest.config.js
{
    coveragePathIgnorePatterns: ['<rootDir>/src/assets/']
}

2.通过t模式,可以仅执行指定的测试用例

当测试用例写的多了,每次执行跑一堆用例,效率很低,如果代码里有很多console,那就更难受了,找个报错都能找半天。当时就想如果能仅测试当前用例就好了。

然后就找到了t模式,jest命令带–watch参数进入监听模式,然后输入t,再输入匹配规则即可。世界一下子就清净了,舒服……

// package.json
{
    "scripts":{
        "tets":"jest --watch"
    }
}

3.vue-awesome-swiper测试运行时报错

如果组件中引入了swiper,那么在执行测试用例时,vue-awesome-swiper中的js会报错,引用即报错,且是第三方代码。

最后通过把swiper组件由局部注册改为全局注册得以解决。

行动吧,在路上总比一直观望的要好,未来的你肯定会感 谢现在拼搏的自己!如果想学习提升找不到资料,没人答疑解惑时,请及时加入扣群: 320231853,里面有各种软件测试+开发资料和技术可以一起交流学习哦。

最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/637471.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

网络协议测试仪设计方案:474-便携式手提万兆网络协议测试仪

便携式手提万兆网络协议测试仪 一、平台简介 便携式手提万兆网络协议测试仪&#xff0c;以FPGA万兆卡和X86主板为基础&#xff0c;构建便携式的手提设备。 FPGA万兆卡是以Kintex-7XC7K325T PCIeX4的双路万兆光纤网络卡&#xff0c;支持万兆网络数据的收发和网络协议…

同旺科技 FLUKE ADPT 隔离版发布 ---- 3

所需设备&#xff1a; 1、FLUKE ADPT 隔离版 内附链接&#xff1b; 应用于&#xff1a;福禄克Fluke 12E / 15BMax / 17B Max / 101 / 106 / 107 应用于&#xff1a;福禄克Fluke 15B / 17B / 18B 总体连接&#xff1a; 连接线&#xff0c;根据自己实际需求而定&#xff1b; …

java操作Redis缓存设置过期时间

如何用java操作Redis缓存设置过期时间&#xff1f;很多新手对此不是很清楚&#xff0c;为了帮助大家解决这个难题&#xff0c;下面小编将为大家详细讲解&#xff0c;有这方面需求的人可以来学习下&#xff0c;希望你能有所收获。 在应用中我们会需要使用redis设置过期时间&…

WPS PPT学习笔记 1 排版4原则等基本技巧整理

排版原则 PPT的排版需要满足4原则&#xff1a;密性、对齐、重复和对比4个基本原则。 亲密性 彼此相关的元素应该靠近&#xff0c;成为一个视觉单位&#xff0c;减少混乱&#xff0c;形成清晰的结构。 两端对齐&#xff0c;1.5倍行距 在本例中&#xff0c;19年放左边&#x…

融资融券利率4.0%!融资融券保证金比例和余额

融资融券利率最低是4.0%~5.0%&#xff0c;这是目前市场最低的利率水平&#xff0c;股票佣金万一。 各家券商的利率差异是较大的&#xff0c;现在无门槛利率是5%&#xff0c;根据投资者的资金量大小在4.0%~5%之间浮动&#xff0c;具体需要通过对应证券经理协商办理&#xff01;…

C++_vector操作使用

文章目录 &#x1f680;1.1 vector介绍&#x1f680;1.2 vector的初始化&#x1f680;1.3 vector的常用内置函数&#x1f680;1.4 vector的遍历 &#x1f680;1.1 vector介绍 vector是表示可变大小数组的序列容器。就像数组一样&#xff0c;vector也采用的连续存储空间来存储元…

什么是线程安全?如何保证线程安全?

目录 一、引入线程安全 &#x1f447; 二、 线程安全&#x1f447; 1、线程安全概念 &#x1f50d; 2、线程不安全的原因 &#x1f50d; 抢占式执行&#xff08;罪魁祸首&#xff0c;万恶之源&#xff09;导致了线程之间的调度是“随机的” 多个线程修改同一个变量 修改…

Java代码审计-XSS审计

一、漏洞简介 XSS是Cross Site Scripting的缩写&#xff0c;意为"跨站脚本攻击"&#xff0c;为了避免与层叠样式表(Cascading Style Sheet&#xff0c;CSS)的缩写混淆&#xff0c;故将跨站脚本攻击缩写为XSS。XSS是一种针对网站应用程序的安全漏洞攻击技术&#xff…

线上申请流量卡一些必知的小知识,愿每个人都能刷到!

很多朋友都想办理一张大流量卡&#xff0c;但是又怕被套路&#xff0c;一时不知道该怎么选择&#xff0c;那个纠结啊。 今天&#xff0c;小编用自己多年的行业经验给大家整理了一些办卡攻略&#xff0c;希望能帮助大家选到适合自己的流量卡。 ​1、有的流量卡都是免费申请&…

使用JavaScript日历小部件和DHTMLX Gantt的应用场景(三)

DHTMLX Suite UI 组件库允许您更快地构建跨平台、跨浏览器 Web 和移动应用程序。它包括一组丰富的即用式 HTML5 组件&#xff0c;这些组件可以轻松组合到单个应用程序界面中。 DHTMLX Gantt是用于跨浏览器和跨平台应用程序的功能齐全的Gantt图表&#xff0c;可满足项目管理应用…

了解 Linux 网络卡绑定:提高网络性能与冗余性

在现代 IT 基础设施中&#xff0c;网络性能和可靠性至关重要。对于许多企业和个人用户来说&#xff0c;确保网络的高可用性和冗余性是首要任务之一。Linux 提供了一个强大的解决方案——网络卡绑定&#xff08;Network Interface Card Bonding&#xff0c;简称 NIC Bonding&…

DevExpress Office File API中文教程 - 如何用OpenAI模型增强Office文档可访问性?

DevExpress Office File API是一个专为C#, VB.NET 和 ASP.NET等开发人员提供的非可视化.NET库。有了这个库&#xff0c;不用安装Microsoft Office&#xff0c;就可以完全自动处理Excel、Word等文档。开发人员使用一个非常易于操作的API就可以生成XLS, XLSx, DOC, DOCx, RTF, CS…

58同城如何降低 80%的机器成本 | OceanBase案例

本文作者&#xff1a;58同城架构师刘春雷 一、背景介绍 58同城作为中国互联网生活服务领域的领军者&#xff0c;其平台规模居国内之首&#xff0c;涵盖了包括车辆交易、房产服务、人才招聘、本地生活服务以及金融等多元化的业务场景。 因其业务的广泛性和多样性&#xff0c;我…

Keil MDK map文件学习笔记

Keil MDK map文件学习笔记 map文件组成1.Section Cross References段交叉引用2.Removing Unused input sections from the image移除无用的段3.Image Symbol Table镜像符号表局部符号表全局符号表 4.Memory Map of the image镜像存储器映射ROM区执行域RAM区执行域 5. Image com…

DLRover:蚂蚁集团开源的AI训练革命

在当前的深度学习领域&#xff0c;大规模训练作业面临着一系列挑战。首先&#xff0c;硬件故障或软件错误导致的停机时间会严重影响训练效率和进度。其次&#xff0c;传统的检查点机制在大规模训练中效率低下&#xff0c;耗时长且容易降低训练的有效时间。资源管理的复杂性也给…

关于新配置的adb,设备管理器找不到此设备问题

上面页面中一开始没有找到此android设备&#xff0c; 可能是因为我重新配置的adb和设备驱动&#xff0c; 只把adb配置了环境变量&#xff0c;驱动没有更新到电脑中&#xff0c; 点击添加驱动&#xff0c; 选择路径&#xff0c;我安装时都放在了SDK下面&#xff0c;可以尝试…

卷爆短剧出海:五大关键,由AIGC重构

短剧高温下&#xff0c;谈谈AIGC的助攻路线。 短剧&#xff0c;一个席卷全球的高温赛道。 以往只是踏着霸总题材&#xff0c;如今&#xff0c;内容循着精品化、IP化的自然发展风向&#xff0c;给内容、制作、平台等产业全链都带来新机&#xff0c;也让短剧消费走向文化深处&am…

【C语言回顾】动态内存管理

前言1. 动态内存管理初步概述2. malloc3. calloc4. realloc5. free6. 常见的动态内存错误7. 柔性数组8. 程序内存区域划分结语 #include<GUIQU.h> int main { 上期回顾: 【C语言回顾】联合和枚举 个人主页&#xff1a;C_GUIQU 专栏&#xff1a;【C语言学习】 return 一键…

win32-鼠标消息、键盘消息、计时器消息、菜单资源

承接前文&#xff1a; win32窗口编程windows 开发基础win32-注册窗口类、创建窗口win32-显示窗口、消息循环、消息队列 本文目录 键盘消息键盘消息的分类WM_CHAR 字符消息 鼠标消息鼠标消息附带信息 定时器消息 WM_TIMER创建销毁定时器 菜单资源资源相关菜单资源使用命令消息的…

远动通讯屏具体干啥作用

远动通讯屏具体干啥作用 远动通讯屏主要用于电力系统中的各类发电厂、变电站、光伏电站、开闭所、配电房等&#xff0c;具有实时传输数据和远程控制功能。它的主要作用包括&#xff1a; 数据采集&#xff1a;远动通讯屏能够采集各种模拟量、开关量和数字量等信息&#xff0c…