详解使用jest对vue项目进行单元测试
最近领导对前端提出了新的要求,要进行单元测试。之前使用vue做了一个快报名小程序的pc端页面,既然要做单元测试,就准备用这个项目了,之前有些react的经验,vue还是第一遭
vue-cli3.0单元测试方面更加完备,就先升级到了cli3.0,因为项目是用typescript写的,需要ts-jest,得到jest的配置如下
{ "jest": { "modulefileextensions": [ "js", "jsx", "json", "vue", "ts", "tsx" ], "transform": { "^.+\\.vue$": "vue-jest", ".+\\.(css|styl|less|sass|scss|png|jpg|ttf|woff|woff2)$": "jest-transform-stub", "^.+\\.tsx?$": "ts-jest" }, "modulenamemapper": { "^@/(.*)$": "<rootdir>/src/$1" }, "snapshotserializers": [ "jest-serializer-vue" ], "testmatch": [ "**/tests/unit/**/*.spec.(js|jsx|ts|tsx)|**/__tests__/*.(js|jsx|ts|tsx)" ], "testurl": "http://localhost/" } }
先从简单的开始,测试了一个正则字符串常量文件,完美,一点问题没有
然后开始测方案页面的scheme.vue组件,这个地方主要就想测一个computed属性,将三种有代表性的情况写完测试案例,兴冲冲运行yarn test:unit scheme.test.ts,结果还不错,三个it测试用例都通过了,但后面还有一片红是什么鬼
console.error node_modules/vue/dist/vue.runtime.common.js:589
[vue warn]: invalid prop: type check failed for prop "headerpic". expected string, got object.
原来是这个地方调用了一个组件,这个组件需要一个headerpic
属性,用作图片的src,看源码
<sidenav :header-pic="require('../../assets/scheme/schemesidenavpic.jpg')">
感觉没毛病啊,去vue-devtool,"/img/schemesidenavpic.f988623b.jpg"
是字符串啊,一点毛病没有,应该不是require的问题啊,应该是require在jest里面的处理问题,再查看jest配置,已经对jpg
等静态文件做处理了,看了一下jest-transform-stub模块的源码,很简单
module.exports = { process: function() { return '' } }
既对这些静态文件返回空字符串,不做处理,这不就更不应该了呀,幸亏有vscode这款利器,可以方便调试源码,使用vscode调试没有报错,也没能让调试器进入vue文件,没办法,在ts文件里const pic = require('../../../assets/scheme/schemesidenavpic.jpg')
,再次调试,发现
正是jest-transform-stub的内容,确实是个对象,跟在命令行内运行结果一致,也就是说只需要一直处理方式让其返回为
module.exports = ""
查看jest官网,搜了一下css,运气不错????, 处理静态文件,modulenamemapper
选项完全可以满足需求啊,
"modulenamemapper": { "\\.(jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$": "<rootdir>/__mocks__/filemock.js" }
filemock.js内容
// __mocks__/filemock.js module.exports = 'test-file-stub';
就是说只要返回字符串就ok了,加上modulenamemapper
,测试完美的跑通了
接下来对scheme.vue组件发起模拟点击测试
const createscheme = wrapper.findall('.sn-item').at(1) createscheme.trigger('click') expect((wrapper.vm as any).iscreatedialogshow).tobetruthy() expect(wrapper.find('.create-list-dialog').isvisible()).tobetruthy()
使用vue-test-utils的api获取createscheme元素,对其触发点击,测试iscreatedialogshow这个data值被设置成true, 使用的element-ui
<el-dialog :visible.sync="iscreatedialogshow" width="600px" class="create-list-dialog" title="创建方案"> ... </el-dialog>
此dialog可见,顺利通过
接下来再实验一下新功能,快照,使用tomatchsnapshot
方法也顺利通过了
接下来来个大的,测试一下login.vue,登陆页面,主要测其调接口,然后成功设置store值,但不能走真实的网络接口啊,这太慢不说,具体结果还不能预测,得使用mock数据
在项目中创建了axios.plugin.ts
vue插件,这可怎么mock呀,再看官方文档,感觉manual mocks部分最合适,但是举例也不适合vue 插件mock啊,继续浏览网站,不知道是受哪的启发还是突然开窍了,应该是受fs
模块启发,突然知道怎么mock插件了,mock一个模块只需要模仿其型即可,具体实现,就无所谓了,这个http请求插件的mock必须能返回我们期望的值啊,fs
模块的__setmockfiles
又给了我启示,可以直接给接口的返回result设值啊,然后就有来下面的
__mocks__/axios.plugin.ts
文件
const mockaxios = {} as any let result = {} as any mockaxios.install = (vue: any, options: any) => { vue.prototype.$axios = function () { /* eslint-disable prefer-promise-reject-errors */ return new promise((resolve, reject) => { if (result.resultcode === '200') { return result.info } else { reject({ code: result.resultcode, msg: result.message, info: result.info }) } }) } } mockaxios.__setmockdata = (data: any) => { result = data } export default mockaxios
然后一马平川了,localvue.use(vuex)
, localvue.use(axiosplugin)
const mockdata = { resultcode: '200', msg: true, info: { openid: 99, userid: 92003, }, } axiosplugin.__setmockdata(mockdata) (wrapper.vm as any).login({ code: '29992' }).then(() => { expect(wrapper.vm.$store.state.userinfo.openid).tobe(mockdata.info.openid) expect(wrapper.vm.$store.state.userinfo.userid).tobe(mockdata.info.userid) })
完美通过,vue的单元测试框架算是基本搭好了,也能给领导说说了
给领导看还得有个覆盖率报告
yarn test:unit --coverage
覆盖的文件比较少啊,不包含所有的源文件啊,需要加入collectcoveragefrom
配置项,至此整个单元测试就比较完备了
下面是完整jest的配置
{ "jest": { "modulefileextensions": [ "js", "jsx", "json", "vue", "ts", "tsx" ], "transform": { "^.+\\.vue$": "vue-jest", ".+\\.(css|styl|less|sass|scss|png|jpg|ttf|woff|woff2)$": "jest-transform-stub", "^.+\\.tsx?$": "ts-jest" }, "modulenamemapper": { "\\.(jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$": "<rootdir>/__mocks__/filemock.js", "^@/(.*)$": "<rootdir>/src/$1" }, "snapshotserializers": [ "jest-serializer-vue" ], "testmatch": [ "**/tests/unit/**/*.spec.(js|jsx|ts|tsx)|**/__tests__/*.(js|jsx|ts|tsx)" ], "testurl": "http://localhost/", "collectcoveragefrom": [ "**/*.{vue,ts}", "!**/node_modules/**", "!**/*.d.ts" ] } }
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
下一篇: python图像和办公文档处理总结