基于Jest+Enzyme的React单元测试

本文为实践的产物,总结从搭建到使用过程中遇到的问题,初学者可以参考。

  • Jest 和 Enzyme 的基本介绍
  • 测试环境搭建
  • 测试脚本编写
  • 运行并调试
  • 匹配器使用
  • 测试异步代码
  • 遇到的问题
  • 参考资料

Jest、Enzyme介绍
Jest是Facebook发布的一个开源的、基于Jasmine框架的JavaScript单元测试工具,支持断言、仿真、快照测试、测试覆盖率报告等。

Enzyme获得React 官方的推荐,Airbnb开源的React测试类库Enzyme提供了一套简介强大的API,并通过jQuery风格的方式进行DOM处理,开发体检十分友好。

强烈推荐阅读使用jest+enzyme进行react项目测试 - 介绍篇

测试环境搭建
在开发React应用的基础上,需要安装Jest、Enzyme以及对应的babel-jest。
使用 yarn 安装 Jest︰
yarn add --dev jest enzyme babel-jest

npm
npm install --save-dev jest enzyme babel-jest

将下面的配置部分添加到package.json 里面:

{
  "scripts": {
    "test": "jest"
  } 
"jest": {
    "moduleFileExtensions": [
      "js",
      "jsx"
    ],
    "moduleNameMapper": {
      "\\.(jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$": "<rootDir>/__mocks__/fileMock.js",
      ".*\\.(css|less|scss)$": "<rootDir>/__mocks__/styleMock.js"
    },
    "transform": {
      "^.+\\.js$": "babel-jest"
    }
  },
}
  • moduleFileExtensions:代表支持加载的文件名,与Webpack中的resolve.extensions类似
  • moduleNameMapper:代表需要被Mock的资源名称。如果需要Mock静态资源(如less、scss等),需要配置Mock的路径<rootDir>/mocks/xxMock.js
  • transform:用于编译 ES6/ES7 语法,需配合 babel-jest使用
    这三个是常用的配置,更多 Jest 配置见官方文档:Jest Configuration

然后,创建xx.test.js的文件,写实际的测试逻辑
最后,运行yarn test

测试脚本编写
强烈推荐阅读使用jest+enzyme进行react项目测试 - 测试手法篇
测试用例写法总结

  • 组件UI测试用snapshot
    snapshot可以测试到组件的渲染结果是否符合预期,预期就是指你上一次录入保存的结果,toMatchSnapshot方法会去帮你对比这次将要生成的结构与上次的区别
    需要注意的是一个足够健壮的测试应该覆盖到所有的渲染请况,也就是如果你的组件根据props传入的参数渲染结果可能不同,这样的情况必须写多个测试案例都覆盖到。

  • DOM交互测试用Jest+Enzyme
    enzyme有3种渲染方式:render、mount、shallow
    render、mount、shallow的区别
    render采用的是第三方库Cheerio的渲染,渲染结果是普通的html结构,对于snapshot使用render比较合适。

    shallow和mount对组件的渲染结果不是html的dom树,而是react树,如果你chrome装了react devtool插件,他的渲染结果就是react devtool tab下查看的组件结构,而render函数的结果是element tab下查看的结果。

    这些只是渲染结果上的差别,更大的差别是shallow和mount的结果是个被封装的ReactWrapper,可以进行多种操作,譬如find()、parents()、children()等选择器进行元素查找;state()、props()进行数据查找,setState()、setprops()操作数据;simulate()模拟事件触发。

    shallow只渲染当前组件,只能能对当前组件做断言;mount会渲染当前组件以及所有子组件,对所有子组件也可以做上述操作。一般交互测试都会关心到子组件,我使用的都是mount。但是mount耗时更长,内存啥的也都占用的更多,如果没必要操作和断言子组件,可以使用shallow。

  • 交互测试
    主要利用simulate()接口模拟事件,实际上simulate是通过触发事件绑定函数,来模拟事件的触发。触发事件后,去判断props上特定函数是否被调用,传参是否正确;组件状态是否发生预料之中的修改;某个dom节点是否存在是否符合期望。

  • mock请求
    需要mock掉真正的http请求,模拟返回值。不用担心不准确,你只用保证请求时的参数符合期望就好,mock的返回值按预期编写就好,置于这些请求是否真的能返回这些结果,是接口测试改干的活。

运行并调试
在运行测试脚本过程,Jest 的错误提示信息友好,通过错误信息一般都能找到问题的所在。 同时 Jest 还提供了生成测试覆盖率报告的命令,只需要在package.json 里配置{ "scripts": { "test": "jest --coverage " } 即可生成。会在终端中显示如下报告:

image.png

而且还会在项目中生成 coverage 文件夹,非常方便。

匹配器使用
推荐阅读:
前端测试框架Jest系列教程 -- Matchers(匹配器)
Javascript单元测试工具-Jest 学习笔记(一)

测试异步代码
推荐阅读:
Javascript单元测试工具-Jest 学习笔记(一)
学习Jest——语法篇

遇到的问题
Image.onload不能被监听到
解决办法参考:https://zhuanlan.zhihu.com/p/37329102
我采用了自己 mock Image 构造函数方案

最后,在写测试用例过程中,不知道用什么匹配器,可以来Jest官网搜罗一圈https://jestjs.io/docs/zh-Hans/expect
不知道使用Enzyme什么API时,也来Enzyme搜罗一圈http://airbnb.io/enzyme/docs/api/
总能找到你想要的~

参考资料
https://zhuanlan.zhihu.com/p/26625269

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容