本文主要介绍前端单元测试的一些技术方案。
单元测试的技术方案很多,不同工具之间有互相协同,也存在功能重合,给我们搭配测试方案带来不小的困难,而且随着 ES6
, TypeScript
的出现,单元测试又增加了很多其他步骤,完整配置起来往往需要很大的时间成本。我希望通过对这些工具的各自作用的掌握,了解完整的前端测试技术方案。前端单元测试的领域也很多,这里主要讲对于前端组件如何进行单元测试,最后会主要介绍下对于 React
组件的一些测试方法总结。
单元测试最核心的部分就是做断言,比如传统语言中的 assert
函数,如果当前程序的某种状态符合 assert
的期望此程序才能正常执行,否则直接退出应用。所以我们可以直接用 Node
中自带的 assert
模块做断言。
用最简单的例子做个验证
function multiple(a, b) {
let result = 0;
for (let i = 0; i < b; ++i)
result += a;
return result;
}
const assert = require('assert');
assert.equal(multiple(1, 2), 3));
这种例子能够满足基础场景的使用,也可以作为一种单元测试的方法。
nodejs
自带的 assert
模块提供了下面一些断言方法,只能满足一些简单场景的需要。
assert.deepEqual(actual, expected[, message])
assert.deepStrictEqual(actual, expected[, message])
assert.doesNotMatch(string, regexp[, message])
assert.doesNotReject(asyncFn[, error][, message])
assert.doesNotThrow(fn[, error][, message])
assert.equal(actual, expected[, message])
assert.fail([message])
assert.ifError(value)
assert.match(string, regexp[, message])
assert.notDeepEqual(actual, expected[, message])
assert.notDeepStrictEqual(actual, expected[, message])
assert.notEqual(actual, expected[, message])
assert.notStrictEqual(actual, expected[, message])
assert.ok(value[, message])
assert.rejects(asyncFn[, error][, message])
assert.strictEqual(actual, expected[, message])
assert.throws(fn[, error][, message])
自带的 assert
不是专门给单元测试使用, 提供的错误信息文档性不好,上面的 demo
最终执行下来会产生下面的报告:
$ node index.js
assert.js:84
throw new AssertionError(obj);
^
AssertionError [ERR_ASSERTION]: 2 == 3
at Object.<anonymous> (/home/quanwei/git/index.js:4:8)
at Module._compile (internal/modules/cjs/loader.js:778:30)
at Object.Module._extensions..js (internal/modules/cjs/loader.js:789:10)
at Module.load (internal/modules/cjs/loader.js:653:32)
at tryModuleLoad (internal/modules/cjs/loader.js:593:12)
at Function.Module._load (internal/modules/cjs/loader.js:585:3)
at Function.Module.runMain (internal/modules/cjs/loader.js:831:12)
at startup (internal/bootstrap/node.js:283:19)
at bootstrapNodeJSCore (internal/bootstrap/node.js:623:3)
由于自带的模块依赖 Node
自身的版本,没办法自由升级,所以使用内置的包灵活性有时候不太够,另外我们很多断言函数也需要在浏览器端执行,所以我们需要同时支持浏览器和 Node
端的断言库。同时观察上面的输出可以发现,这个报告更像是程序的错误报告,而不是一个单元测试报告。而我们在做单元测时往往需要断言库能够提供良好的测试报告,这样才能一目了然地看到有哪些断言通过没通过,所以使用专业的单元测试断言库还是很有必要。
chai
chai
是目前很流行的断言库,相比于同类产品比较突出。chai
提供了 TDD[2] (Test-driven development)和 BDD[3] (Behavior-driven development) 两种风格的断言函数,这里不会过多介绍两种风格的优缺,本文主要以 BDD
风格做演示。
var assert = require('chai').assert
, foo = 'bar'
, beverages = { tea: [ 'chai', 'matcha', 'oolong' ] };
assert.typeOf(foo, 'string'); // without optional message
assert.typeOf(foo, 'number', 'foo is a number'); // with optional message
assert.equal(foo, 'bar', 'foo equal `bar`');
assert.lengthOf(foo, 3, 'foo`s value has a length of 3');
assert.lengthOf(beverages.tea, 3, 'beverages has 3 types of tea');
chai
比 Node
自带的 assert
增加了一个断言说明参数,可以通过这个参数提高测试报告的可读性
$ node chai-assert.js
/home/quanwei/git/learn-tdd-bdd/node_modules/chai/lib/chai/assertion.js:141
throw new AssertionError(msg, {
^
AssertionError: foo is a number: expected 'bar' to be a number
at Object.<anonymous> (/home/quanwei/git/learn-tdd-bdd/chai-assert.js:6:8)
at Module._compile (internal/modules/cjs/loader.js:778:30)
at Object.Module._extensions..js (internal/modules/cjs/loader.js:789:10)
at Module.load (internal/modules/cjs/loader.js:653:32)
at tryModuleLoad (internal/modules/cjs/loader.js:593:12)
at Function.Module._load (internal/modules/cjs/loader.js:585:3)
at Function.Module.runMain (internal/modules/cjs/loader.js:831:12)
at startup (internal/bootstrap/node.js:283:19)
at bootstrapNodeJSCore (internal/bootstrap/node.js:623:3)
chai
的 BDD
风格使用 expect
函数作为语义的起始,也是目前几乎所有 BDD
工具库都遵循的风格。
chai
的 expect
断言风格如下
expect(foo).to.be.a('string');
expect(foo).to.equal('bar');
expect(foo).to.have.lengthOf(3);
BDD
的思想就是写单元测试就像写产品需求,而不关心内部逻辑,每一个用例阅读起来就像一篇文档。例如下面的用例:
expect(foo).to.be.a('string')
expect(foo).to.include('bar')
expect(foo).to.not.include('biz')
可以看到这种风格的测试用例可读性更强。
其他的断言库还有 expect.js``should.js
better-assert[4] , unexpected.js[5] 这些断言库都只提供纯粹的断言函数,可以根据喜好选择不同的库使用。
有了断言库之后我们还需要使用测试框架将我们的断言更好地组织起来。
mocha jasmine
mocha
是一个经典的测试框架(Test Framework),测试框架提供了一个单元测试的骨架,可以将不同子功能分成多个文件,也可以对一个子模块的不同子功能再进行不同的功能测试,从而生成一份结构型的测试报告。例如 mocha
就提供了describe
和 it
描述用例结构,提供了 before
, after
, beforeEach
, afterEach
生命周期函数,提供了 describe.only
,describe.skip
, it.only
, it.skip
用以执行指定部分测试集。
const { expect } = require('chai');
const { multiple } = require('./index');
describe('Multiple', () => {
it ('should be a function', () => {
expect(multiple).to.be.a('function');
})
it ('expect 2 * 3 = 6', () => {
expect(multiple(2, 3)).to.be.equal(6);
})
})
测试框架不依赖底层的断言库,哪怕使用原生的 assert
模块也可以进行。给每一个文件都要手动引入 chai
比较麻烦 ,这时候可以给 mocha
配置全局脚本,在项目根目录 .mocharc.js
文件中加载断言库, 这样每个文件就可以直接使用 expect
函数了。
// .mocharc.js
global.expect = require('chai').expect;
使用 mocha 可以将我们的单元测试输出成一份良好的测试报告 mocha *.test.js
当出现错误时输出如下
因为运行在不同环境中需要的包格式不同,所以需要我们针对不同环境做不同的包格式转换,为了了解在不同端跑单元测试需要做哪些事情,可以先来了解一下常见的包格式。
目前我们主流有三种模块格式,分别是 AMD
, CommonJS
, ES Module
。
AMD[6] 是 RequireJS
推广过程中流行的一个比较老的规范,目前无论浏览器还是 Node
都没有默认支持。AMD
的标准定义了 define
和 require
函数,define
用来定义模块及其依赖关系,require
用以加载模块。例如
<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8"/>
<title>Document</title>
+ <script
+ src="https://requirejs.org/docs/release/2.3.6/minified/require.js"></script>
+ <script src="./index.js" />
</head>
<body></body>
</html>
// index.js
define('moduleA', ['https://some/of/cdn/path'], function() {
return { name: 'moduleA' };
});
define(function(require) {
const fs = require('fs');
return fs;
})
define('moduleB', function() {
return { name: 'module B' }
});
require(['moduleA', 'moduleB'], function(moduleA, moduleB) {
console.log(module);
});
这里使用了RequireJS
作为 AMD
引擎, 可以看到 define
函数会定义当前依赖了哪些模块并将模块加载完成后异步回调给当前模块,这种特性使得 AMD 尤为适合浏览器端异步加载。
我们可以使用 webpack
打包一份 amd
模块看下真实代码
// entry.js
export default function sayHello() {
return 'hello amd';
}
// webpack.config.js
module.exports = {
mode: 'development',
devtool: false,
entry: './entry.js',
output: {
libraryTarget: 'amd'
}
}
最终生成代码(精简了不相关的逻辑)
// dist/main.js
define(() => ({
default: function sayHello() {
return 'hello amd';
}
}));
在浏览器/Node
中想要使用 AMD
需要全局引入 RequireJS
,对单元测试而言比较典型的问题是在初始化 karma
时会询问是否使用 RequireJS
,不过一般现在很少有人使用了。
可以缩写成CJS
, 其 规范 [7]主要是为了定义 Node
的包格式,CJS
定义了三个关键字, 分别为 require
,exports
, module
, 目前几乎所有Node
包以及前端相关的NPM
包都会转换成该格式, CJS
在浏览器端需要使用 webpack
或者 browserify
等工具打包后才能执行。
ES Module
是 ES 2015
中定义的一种模块规范,该规范定义了 代表为 import
和 export
,是我们开发中常用的一种格式。虽然目前很多新版浏览器都支持<script type="module">
了,支持在浏览器中直接运行 ES6
代码,但是浏览器不支持 node_modules
,所以我们的原始 ES6
代码在浏览器上依然无法运行,所以这里我暂且认为浏览器不支持 ES6
代码, 依然需要做一次转换。
下表为每种格式的支持范围,括号内表示需要借助外部工具支持。
Node | 浏览器 | |
---|---|---|
AMD | 不支持(require.js, r.js) | 不支持(require.js) |
CommonJS | 支持 | 不支持(webpack/browserify) |
ESModule | 不支持(babel) | 不支持(webpack) |
单元测试要在不同的环境下执行就要打不同环境对应的包,所以在搭建测试工具链时要确定自己运行在什么环境中,如果在 Node
中只需要加一层 babel
转换,如果是在真实浏览器中,则需要增加 webpack
处理步骤。
所以为了能够在 Node
环境的 Mocha
中使用 ES Module
有两种方式
Node
环境天生支持 ES Module
(node version >= 15)babel
代码进行一次转换第一种方式略过,第二种方式使用下面的配置
npm install @babel/register @babel/core @babel/preset-env --save-dev
// .mocharc.js
+ require('@babel/register');
global.expect = require('chai').expect;
// .babelrc
+ {
+ "presets": ["@babel/preset-env" ,“@babel/preset-typescript”]
+ }
同样地如果在项目中用到了 TypeScript
, 就可以使用ts-node/register
来解决,因为 TypeScript
本身支持 ES Module
转换成 CJS
, 所以支持了 TypeScript
后就不需要使用 babel
来转换了。(这里假设使用了 TypeScript
的默认配置)
npm install ts-node typescript --save-dev
// .mocharc.js
require('ts-node/register');
Mocha
自身支持浏览器和 Node
端测试,为了在浏览器端测试我们需要写一个 html, 里面使用 <script src="mocha.min.js">
的文件,然后再将本地所有文件插入到html中才能完成测试,手动做工程化效率比较低,所以需要借助工具来实现这个任务,这个工具就是 Karma
。
Karma
本质上就是在本地启动一个web服务器,然后再启动一个外部浏览器加载一个引导脚本,这个脚本将我们所有的源文件和测试文件加载到浏览器中,最终就会在浏览器端执行我们的测试用例代码。所以使用 Karma
+ mocha
+chai
即可搭建一个完整的浏览器端的单元测试工具链。
npm install karma mocha chai karma-mocha karma-chai --save-dev
npx karma init
// Which testing framework do you want to use: mocha
// Do you want to use Require.js: no
// Do you want capture any browsers automatically: Chrome
这里 Karma
初始化时选择了 Mocha
的支持,然后第二个 Require.js
一般为否,除非业务代码中使用了amd
类型的包。第三个选用 Chrome
作为测试浏览器。然后再在代码里单独配置下 chai
。
// karma.conf.js
module.exports = function(config) {
config.set({
// base path that will be used to resolve all patterns (eg. files, exclude)
basePath: '',
// frameworks to use
// available frameworks: https://npmjs.org/browse/keyword/karma-adapter
- frameworks: ['mocha'],
+ frameworks: ['mocha', 'chai'],
// list of files / patterns to load in the browser
files: [],
Karma
的 frameworks
作用是在全局注入一些依赖,这里的配置就是将 Mocha
和 chai
提供的测试相关工具暴露在全局上供代码里使用。Karma
只是将我们的文件发送到浏览器去执行,但是根据前文所述我们的代码需要经过 webpack
或 browserify
打包后才能运行在浏览器端。
如果原始代码已经是 CJS
了,可以使用 browserify
来支持浏览器端运行,基本零配置,但是往往现实世界比较复杂,我们有 ES6
,JSX
以及 TypeScript
要处理,所以这里我们使用 webpack
。
下面是 webpack
的配置信息。
npm install karma-webpack@4 webpack@4 @babel/core @babel/preset-env @babel/preset-react babel-loader --save-dev
// karma.conf.js
module.exports = function(config) {
config.set({
// base path that will be used to resolve all patterns (eg. files, exclude)
basePath: '',
// frameworks to use
// available frameworks: https://npmjs.org/browse/keyword/karma-adapter
frameworks: ['mocha', 'chai'],
// list of files / patterns to load in the browser
files: [
+ { pattern: "test/*.test.js", watched: false }
],
preprocessors: {
+ 'test/**/*.js': [ 'webpack']
},
+ webpack: {
+ module: {
+ rules: [{
+ test: /.*\.js/,
+ use: 'babel-loader'
+ }]
+ }
+ },
// .babelrc
{
"presets": ["@babel/preset-env", "@babel/preset-react"]
}
这里我们测试一个React
程序代码如下
// js/index.js
import React from 'react';
import ReactDOM from 'react-dom';
export function renderToPage(str) {
const container = document.createElement('div');
document.body.appendChild(container);
console.log('there is real browser');
return new Promise(resolve => {
ReactDOM.render(<div>{ str } </div>, container, resolve);
});
}
// test/index.test.js
import { renderToPage } from '../js/index';
describe('renderToPage', () => {
it ('should render to page', async function () {
let content = 'magic string';
await renderToPage(content);
expect(document.documentElement.innerText).to.be.contain(content);
})
})
并且打开了本地浏览器
karma browser
可以看到现在已经在真实浏览器中运行测试程序了。
因为图形化的测试对 CI
机器不友好,所以可以选择 puppeteer
代替 Chrome
。
再者这些都是很重的包,如果对真实浏览器依赖性不强,可以使用 JSDOM
在 Node
端模拟一个浏览器环境。
稍微总结下工具链
mocha
+ chai
+ babel
mocha
+ chai
+ babel
+ jsdom
karma
+ mocha
+ chai
+ webpack
+ babel
一个测试流水线往往需要很多个工具搭配使用,配置起来比较繁琐,还有一些额外的工具例如单元覆盖率(istanbul),函数/时间模拟 (sinon.js)等工具。工具之间的配合有时候不一定能够完美契合,选型费时费力。
jasmine
的出现就稍微缓解了一下这个问题,但也不够完整,jasmine
提供一个测试框架,里面包含了 测试流程框架,断言函数,mock工具等测试中会遇到的工具。可以近似地看作 jasmine = mocha + chai + 辅助工具
。
接下来试一试 jasmine
的工作流程。
使用 npx jasmine init
初始化之后会在当前目录中生成spec
目录, 其中包含一份默认的配置文件
// ./spec/support/jasmine.json
{
"spec_dir": "spec",
"spec_files": [
"**/*[sS]pec.js"
],
"helpers": [
"helpers/**/*.js"
],
"stopSpecOnExpectationFailure": false,
"random": true
}
如果希望加载一些全局的配置可以在 spec/helpers
目录中放一些js
文件, 正如配置所言,jasmine 在启动时会去执行 spec/helpers
目录下的所有js
文件。
比如我们常常使用 es6
语法,就需要增加es6
的支持。
新增 spec/helpers/babel.js
写入如下配置即可。
npm install @babel/register @babel/core @babel/preset-env --save-dev
// spec/helpers/babel.js
require('babel-register');
// .babelrc
{
"presets": ["@babel/preset-env"]
}
和 mocha
一样,如果需要 TypeScript
的支持,可以使用如下配置
npm install ts-node typescript --save-dev
// spec/helpers/typescript.js
require('ts-node/register');
配置文件中的 spec_dir
是 jasmine
约定的用例文件目录,spec_files
规定了用例文件格式为 xxx.spec.js
。
有了这份默认配置就可以按照要求写用例,例如
// ./spec/index.spec.js
import { multiple } from '../index.js';
describe('Multiple', () => {
it ('should be a function', () => {
expect(multiple).toBeInstanceOf(Function);
})
it ('should 7 * 2 = 14', () => {
expect(multiple(7, 2)).toEqual(14);
})
it ('should 7 * -2 = -14', () => {
expect(multiple(7, -2)).toEqual(-14);
})
})
jasmine
的断言风格和 chai
很不一样,jasmine
的 API
如下,与 chai
相比少写了很多 .
,而且支持的功能更加清晰,不用考虑如何组合使用的问题,而且下文介绍的 jest
测试框架也是使用这种风格。
nothing()
toBe(expected)
toBeCloseTo(expected, precisionopt)
toBeDefined()
toBeFalse()
toBeFalsy()
toBeGreaterThan(expected)
toBeGreaterThanOrEqual(expected)
toBeInstanceOf(expected)
toBeLessThan(expected)
toBeLessThanOrEqual(expected)
toBeNaN()
toBeNegativeInfinity()
toBeNull()
toBePositiveInfinity()
toBeTrue()
toBeTruthy()
toBeUndefined()
toContain(expected)
toEqual(expected)
toHaveBeenCalled()
toHaveBeenCalledBefore(expected)
toHaveBeenCalledOnceWith()
toHaveBeenCalledTimes(expected)
toHaveBeenCalledWith()
toHaveClass(expected)
toHaveSize(expected)
toMatch(expected)
toThrow(expectedopt)
toThrowError(expectedopt, messageopt)
toThrowMatching(predicate)
withContext(message) → {matchers}
运行 jasmine
即可生成测试报告
默认的测试报告不是很直观, 如果希望提供类似 Mocha
风格的报告可以安装 jasmine-spec-reporter
,在 spec/helpers
目录中添加一个配置文件, 例如spec/helpers/reporter.js
。
const SpecReporter = require('jasmine-spec-reporter').SpecReporter;
jasmine.getEnv().clearReporters(); // remove default reporter logs
jasmine.getEnv().addReporter(new SpecReporter({ // add jasmine-spec-reporter
spec: {
displayPending: true
}
}));
此时输出的用例报告如下
jasmine如果在 Jasmine
中执行 DOM 级别的测试,就依然需要借助 Karma
或 JSDOM
了,具体的配置这里就不再赘述。
总结下 Jasmine
的工具链
Jasmine
+ babel
JSDOM
测试 : Jasmine
+ JSDOM
+ babel
Karma
+ Jasmine
+ webpack
+ babel
jest
Jest
是 facebook
出的一个完整的单元测试技术方案,集 测试框架, 断言库, 启动器, 快照,沙箱,mock工具于一身,也是 React
官方使用的测试工具。Jest
和 Jasmine
具有非常相似的 API
,所以在 Jasmine
中用到的工具在 Jest
中依然可以很自然地使用。可以近似看作 Jest = JSDOM 启动器 + Jasmine
。
虽然 Jest 提供了很丰富的功能,但是并没有内置 ES6
支持,所以依然需要根据不同运行时对代码进行转换,由于 Jest 主要运行在 Node
中,所以需要使用 babel-jest
将 ES Module
转换成 CommonJS
。
Jest 的默认配置
npm install jest --save-dev
npx jest --init
√ Would you like to use Jest when running "test" script in "package.json"? ... yes
√ Would you like to use Typescript for the configuration file? ... no
√ Choose the test environment that will be used for testing » jsdom (browser-like)
√ Do you want Jest to add coverage reports? ... no
√ Which provider should be used to instrument code for coverage? » babel
√ Automatically clear mock calls and instances between every test? ... yes
在 Node
或 JSDOM
下增加 ES6
代码的支持
npm install jest-babel @babel/core @babel/preset-env
// .babelrc
{
"presets": ["@babel/preset-env"]
}
// jest.config.js
// 下面两行为默认配置,不写也可以
{
+ testEnvironment: "jsdom",
+ transform: {"\\.[jt]sx?$": "babel-jest"}
}
使用 Jest
生成测试报告
jest
对于 React
和 TypeScript
支持也可以通过修改 babel
的配置解决
npm install @babel/preset-react @babel/preset-typescript --save-dev
// .babrlrc
{
"presets": ["@babel/preset-env", "@babel/preset-react", "@babel/preset-typescript"]
}
目前 Jest
不支持直接在真实浏览器中进行测试,其默认的启动器只提供了一个 JSDOM
环境,在浏览器中进行单元测试目前只有 Karma
方案能做到,所以也可以使用 Karma
+ Jest
方案实现,但是不建议这么做,因为 Jest
自身太重,使用 Karma
+ Jasmine
能达到基本一样的效果。
另外还有一个比较流行的 E2E
方案 Jest
+ Puppeteer
, 由于 E2E
不属于单元测试范畴,这里不再展开。
Jest
工具链总结
Jest
+ babel
JSDOM
测试 : Jest
+ babel
E2E
测试 : Jest
+ Puppeteer
上面的内容介绍了 chai
, mocha
, karma
, jasmine
和 jest
, 每种工具分别对应一些自己特有的工具链,在选取合适的测试工具时根据实际需要选择, 测试领域还有非常多的工具数都数不过来,下面来看下 React 单元测试的一些方法。
enzyme
Enzyme
基础配置如下:
npm install enzyme enzyme-adapter-react-16 jest-enzyme jest-environment-enzyme jest-canvas-mock react@16 react-dom@16 --save-dev
// jest.config.js
{
- "testEnvironment": "jsdom",
+ setupFilesAfterEnv: ["jest-enzyme", "jest-canvas-mock"],
+ testEnvironment: "enzyme",
+ testEnvironmentOptions: {
+ "enzymeAdapter": "react16"
+ },
}
jest-canvas-mock
这个包是为了解决一些使用 JSDOM
未实现行为触发警告的问题。
上面建立了一个使用 Enzyme
比较友好的环境,可以直接在全局作用域里引用 React
, shallow
, mount
等 API
。此外 Enzyme
还注册了许多友好的断言函数到 Jest
中,如下所示,参考地址[8]
toBeChecked()
toBeDisabled()
toBeEmptyRender()
toExist()
toContainMatchingElement()
toContainMatchingElements()
toContainExactlyOneMatchingElement()
toContainReact()
toHaveClassName()
toHaveDisplayName()
toHaveHTML()
toHaveProp()
toHaveRef()
toHaveState()
toHaveStyle()
toHaveTagName()
toHaveText()
toIncludeText()
toHaveValue()
toMatchElement()
toMatchSelector()
// js/ClassComponent.js
import React from 'react';
export default class ClassComponent extends React.PureComponent {
constructor() {
super();
this.state = { name: 'classcomponent' };
}
render() {
return (
<div>
a simple class component
<CustomComponent />
</div>
);
}
}
// test/hook.test.js
import HookComponent from '../js/HookComponent';
describe('HookComponent', () => {
it ('test with shallow', () => {
const wrapper = shallow(<HookComponent id={1} />);
expect(wrapper).toHaveState('name', 'classcomponent');
expect(wrapper).toIncludeText('a simple class component');
expect(wrapper).toContainReact(<div>a simple class component</div>);
expect(wrapper).toContainMatchingElement('CustomComponent');
})
})
Enzyme
提供了三种渲染组件方法
shallow
使用 react-test-renderer
将组件渲染成内存中的对象, 可以方便进行 props
, state
等数据方面的测试,对应的操作对象为 ShallowWrapper
,在这种模式下仅能感知到第一层自定义子组件,对于自定义子组件内部结构则无法感知。mount
使用 react-dom
渲染组件,会创建真实 DOM
节点,比 shallow
相比增加了可以使用原生 API
操作 DOM
的能力,对应的操作对象为 ReactWrapper
,这种模式下感知到的是一个完整的 DOM
树。render
使用 react-dom-server
渲染成 html
字符串,基于这份静态文档进行操作,对应的操作对象为 CheerioWrapper
。因为 shallow
模式仅能感知到第一层自定义子组件组件,往往只能用于简单组件测试。例如下面的组件
// js/avatar.js
function Image({ src }) {
return <img src={src} />;
}
function Living({ children }) {
return <div className="icon-living"> { children } </div>;
}
function Avatar({ user, onClick }) {
const { living, avatarUrl } = user;
return (
<div className="container" onClick={onClick}>
<div className="wrapper">
<Living >
<div className="text"> 直播中 </div>
</Living>
</div>
<Image src={avatarUrl} />
</div>
)
}
export default Avatar;
shallow
渲染虽然不是真正的渲染,但是其组件生命周期会完整地走一遍。
使用 shallow(<Avatar />)
能感知到的结构如下, 注意看到 div.text
作为 Living
组件的 children
能够被检测到,但是 Living
的内部结构无法感知。
shallowEnzyme
支持的选择器支持我们熟悉的 css selector
语法,这种情况下我们可以对 DOM
结构做如下测试
// test/avatar.test.js
import Avatar from '../js/avatar';
describe('Avatar', () => {
let wrapper = null, avatarUrl = 'abc';
beforeEach(() => {
wrapper = shallow(<Avatar user={{ avatarUrl: avatarUrl }} />);
})
afterEach(() => {
wrapper.unmount();
jest.clearAllMocks();
})
it ('should render success', () => {
// wrapper 渲染不为空
expect(wrapper).not.toBeEmptyRender();
// Image 组件渲染不为空, 这里会执行 Image 组件的渲染函数
expect(wrapper.find('Image')).not.toBeEmptyRender();
// 包含一个节点
expect(wrapper).toContainMatchingElement('div.container');
// 包含一个自定义组件
expect(wrapper).toContainMatchingElement("Image");
expect(wrapper).toContainMatchingElement('Living');
// shallow 渲染不包含子组件的内部结构
expect(wrapper).not.toContainMatchingElement('img');
// shallow 渲染包含 children 节点
expect(wrapper).toContainMatchingElement('div.text');
// shallow 渲染可以对 children 节点内部结构做测试
expect(wrapper.find('div.text')).toIncludeText('直播中');
})
})
如果我们想去测试对应组件的 props
/ state
也可以很方便测试,不过目前存在缺陷,Class Component
能通过 toHaveProp
, toHaveState
直接测试, 但是 Hook
组件无法测试 useState
。
it ('Image component receive props', () => {
const imageWrapper = wrapper.find('Image');、
// 对于 Hook 组件目前我们只能测试 props
expect(imageWrapper).toHaveProp('src', avatarUrl);
})
wrapper.find
虽然会返回同样的一个 ShallowWrapper
对象,但是这个对象的子结构是未展开的,如果想测试imageWrapper
内部结构,需要再 shallow render
一次。
it ('Image momponent receive props', () => {
const imageWrapper = wrapper.find('Image').shallow();
expect(imageWrapper).toHaveProp('src', avatarUrl);
expect(imageWrapper).toContainMatchingElement('img');
expect(imageWrapper.find('img')).toHaveProp('src', avatarUrl);
})
也可以改变组件的 props
, 触发组件重绘
it ('should rerender when user change', () => {
const newAvatarUrl = '' + Math.random();
wrapper.setProps({ user: { avatarUrl: newAvatarUrl }});
wrapper.update();
expect(wrapper.find('Image')).toHaveProp('src', newAvatarUrl);
})
另一个常见的场景是事件模拟,事件比较接近真实测试场景,这种场景下使用 shallow
存在诸多缺陷,因为 shallow
场景事件不会像真实事件一样有捕获和冒泡流程,所以此时只能简单的触发对应的 callback
达到测试目的。
it ('will call onClick prop when click event fired', () => {
const fn = jest.fn();
wrapper.setProps({ onClick: fn });
wrapper.update();
// 这里触发了两次点击事件,但是 onClick 只会被调用一次。
wrapper.find('div.container').simulate('click');
wrapper.find('div.wrapper').simulate('click');
expect(fn).toHaveBeenCalledTimes(1);
})
关于这些网上有人总结了 shallow
模式下的一些不足
shallow
渲染不会进行事件冒泡,而 mount
会。shallow
渲染因为不会创建真实 DOM
,所以组件中使用 refs
的地方都无法正常获取,如果确实需要使用 refs
, 则必须使用 mount
。simulate
在 mount
中会更加有用,因为它会进行事件冒泡。其实上面几点说明了一个现象是 shallow
往往只适合一种理想的场景,一些依赖浏览器行为表现的操作 shallow
无法满足,这些和真实环境相关的就只能使用mount
了。
Mount
渲染的对象结构为 ReactWrapper
其提供了和 ShallowWrapper
几乎一样的 API
, 差异很小。
在 API
层面的一些差异如下
+ getDOMNode() 获取DOM节点
+ detach() 卸载React组件,相当于 unmountComponentAtNode
+ mount() 挂载组件,unmount之后通过这个方法重新挂载
+ ref(refName) 获取 class component 的 instance.refs 上的属性
+ setProps(nextProps, callback)
- setProps(nextProps)
- shallow()
- dive()
- getElement()
- getElements()
另外由于 mount
使用 ReactDOM
进行渲染,所以其更加接近真实场景,在这种模式下我们能观察到整个 DOM
结构和React组件节点结构。
mount
describe('Mount Avatar', () => {
let wrapper = null, avatarUrl = '123';
beforeEach(() => {
wrapper = mount(<Avatar user={{ avatarUrl }} />);
})
afterEach(() => {
jest.clearAllMocks();
})
it ('should set img src with avatarurl', () => {
expect(wrapper.find('Image')).toExist();
expect(wrapper.find('Image')).toHaveProp('src', avatarUrl);
expect(wrapper.find('img')).toHaveProp('src', avatarUrl);
})
})
在 shallow
中无法模拟的事件触发问题在 mount
下就不再是问题。
it ('will call onClick prop when click event fired', () => {
const fn = jest.fn();
wrapper.setProps({ onClick: fn });
wrapper.update();
wrapper.find('div.container').simulate('click');
wrapper.find('div.wrapper').simulate('click');
expect(fn).toHaveBeenCalledTimes(2);
})
总结一下 shallow
中能做的 mount
都能做,mount
中能做的 shallow
不一定能做。
render
内部使用 react-dom-server
渲染成字符串,再经过 Cherrio
转换成内存中的结构,返回 CheerioWrapper
实例,能够完整地渲染整个DOM
树,但是会将内部实例的状态丢失,所以也称为 Static Rendering
。这种渲染能够进行的操作比较少,这里也不作具体介绍,可以参考 官方文档[9] 。
如果让我推荐的话,对于真实浏览器我会推荐 Karma
+ Jasmine
方案测试,对于 React
测试 Jest
+ Enzyme
在 JSDOM
环境下已经能覆盖大部分场景。另外测试 React
组件除了 Enzyme
提供的操作, Jest
中还有很多其他有用的特性,比如可以 mock
一个 npm
组件的实现,调整 setTimeout
时钟等,真正进行单元测试时,这些工具也是必不可少的,整个单元测试技术体系包含了很多东西,本文无法面面俱到,只介绍了一些距离我们最近的相关的技术体系。
参考
[1]江水: https://www.zhihu.com/people/li-quan-wei-41
[2]TDD: https://en.wikipedia.org/wiki/Test-driven_development
[3]BDD: https://en.wikipedia.org/wiki/Behavior-driven_development
[4]better-assert: https://github.com/tj/better-assert
[5]unexpected.js: https://unexpected.js.org/
[6]AMD: https://github.com/amdjs/amdjs-api/blob/master/AMD.md
[7]规范 : http://wiki.commonjs.org/wiki/Modules/1.1
[8]参考地址: https://github.com/enzymejs/enzyme-matchers/blob/master/packages/jest-enzyme/README.md
[9]官方文档: https://enzymejs.github.io/enzyme/docs/api/render.html
本文由哈喽比特于3年以前收录,如有侵权请联系我们。
文章来源:https://mp.weixin.qq.com/s/doyhqK1EunQz-bwFXLHSOg
京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。
日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。
据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。
今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。
日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。
近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。
据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。
9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...
9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。
据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。
特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。
据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。
近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。
据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。
9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。
《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。
近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。
社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”
2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。
罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。