mocha+Typescript的单元测试搭建

Posted 米糕随笔

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了mocha+Typescript的单元测试搭建相关的知识,希望对你有一定的参考价值。


关注我
更多精彩等你发现!



TDD(Test-Driven Development)测试驱动开发是敏捷开发中的一项核心的实践技术,要求在开发编写代码之前,先编写测试用例。既保证了开发过程的严谨,也对后期维护和重构有很大的帮助。今天在这里就和大家一起从0搭建一个单元测试框架。


  • 技术准备
    开发使用typescript作为主要开发语言,为了更好的理解,这里不采用任何集成框架。
    单元测试工具使用mocha,使用chai作为断言工具
    测试报告使用mochawesome作为测试报告生成工具


在这里我们使用之前已有的node + typescript项目作为示例,目录结构如下

mocha+Typescript的单元测试搭建


开始安装mocha和chai。由于我们计划结合测试typescript,并且用其作为测试用例的编写语言,因此我们还要安装@types/mocha 和 @types/chai

yarn add mocha chai @types/mocha @types/chai --save-dev

在使用之前,我们按照惯例去官网阅读一下“说明书”

按照官网的介绍,有两种方式使用mocha,一种通过mocha的cli命令通过控制台运行测试,另一种方式是通过mocha的命令生成一个框架,然后通过页面浏览器环境进行测试。这里,我们采用第一种方式,更方便进行集成测试。

首先我们创建一个待测试的代码

demo.ts:

/** * demo of origin for unit test */
export class Demo { /** * @name getSum * @description 求和demo * @param arg */ public static getSum (arg: Array<number>): number { let sum: number = 0; for (let i in arg) { sum += arg[i]; } return sum; }}

这里是一个简单的求和方法,下面搭建demo.test.ts对其进行测试

demo.test.ts

/** * demo of unit test */
import { expect } from 'chai';import { Demo } from './demo';
// Demo类的测试声明describe('Demo', function () { describe('Demo.getSum', function() { it('1+1 应该等于 2', function() { expect(Demo.getSum([1, 1])).to.be.equal(2); }); });});

describe方法用来创建一个测试描述,第一个参数是描述文本,第二个参数是实体内容,接收一个回调函数。describe是可以嵌套的,像上文中使用了两个describe分别描述了Demo和Demo的静态方法getSum,使用中可以根据自己的项目以及代码结构进行灵活分配,让测试结构更加合理。

it方法用来约定一个测试用例,回调函数中使用chai的expect函数获取结果,并通过其断言语法进行判断,这里使用.to.be.equal()方法判断1+1=2


测试文件搭建好了,接下来要使用测试用例了。但是注意,这里我们忽略了一个步骤。mocha命令运行测试用例是使用nodejs环境的,我们编写的是ts文件,可以直接运行么?答案是不可以!要想通过node环境运行ts文件,我们会想到之前给大家介绍过的神器:ts-node

那mocha如何使用ts-node呢?查询了一下,有许多现成的npm工具如ts-mocha mocha-typescript 等,已经集成了ts-node和mocha,直接拿来就可以用了。

但是,轮毂和轮胎都拿到手了,还要再换个总成么?这明显不是我性格!我们继续查找官网:

mocha+Typescript的单元测试搭建

mocha提供了 --require 参数,通过此参数可以在运行时集成某些模块。Good,我们来试试:

我们的mocha是安装在项目中,因此调用前使用yarn命令作为前缀

yarn mocha ./src/tasks/demo.test.ts --require ts-node/register

运行结果:


Demo Demo.getSum √ 1+1 应该等于 2

1 passing (19ms)


完美!官网还提供了许多命令参数,方便我们集成更多的功能和灵活的设置。但是每次通过命令运行这么多的参数,尤其是在修改的时候,太不方便了。没关系,继续阅读官网:

mocha+Typescript的单元测试搭建

mocha很贴心地提供了配置文件的方式,让我们对mocha运行的参数进行集中配置。(记住,不会了去查官网,总会有收获)

老版本的mocha使用的是opts文件,现在已经不推荐了,但提供了这四种方式。为了配置统一,我们选择package.json中集成的方式进行配置:

package.json

{ "name": "migao-node-service", "version": "0.0.0", "private": true, "scripts": { "debug": "yarn ts-node ./src/bin/www.ts", "start-f": "forever start forever.json", "start": "node ./build/bin/www.js", "status": "forever list", "stop": "forever stopall", "build": "gulp build", "test": "yarn mocha" }, "mocha": { "colors": true, "recursive": [ "src/**/*.test.ts" ], "reporter": [ "mochawesome" ], "require": [ "ts-node/register" ] }, "dependencies": { "@types/chai": "^4.2.11", "@types/mocha": "^8.0.0", "axios": "^0.19.2", "chai": "^4.2.0", "cookie-parser": "~1.4.4", "debug": "~2.6.9", "express": "~4.16.1", "http-errors": "~1.6.3", "jade": "^1.11.0", "mocha": "^8.0.1", "mochawesome": "^6.1.1", "morgan": "~1.9.1", "pug": "^2.0.4" }, "devDependencies": { "@babel/core": "^7.9.0", "@babel/plugin-transform-typescript": "^7.9.4", "@babel/preset-env": "^7.9.0", "@babel/preset-typescript": "^7.9.0", "@types/express": "^4.17.4", "@types/node": "^13.11.0", "babel-preset-minify": "^0.5.1", "del": "^5.1.0", "gulp": "^4.0.2", "gulp-babel": "^8.0.0", "gulp-plumber": "^1.2.1", "gulp-scss": "^1.4.0", "ts-node": "^8.8.2", "typescript": "^3.8.3" }}

由于默认运行mocha命令的时候,mocha会从上下文中找到最近的匹配配置运行,因此在test命令中,直接运行yarn mocha即可

上边的mocha配置中,除了添加了之前描述过的内容以外,还添加了reporter属性,这个属性可以指定报告格式,mocha内置了许多种报告格式,但都是在控制台输出的。我们选择了mochawesome插件来作为测试报告的输出,mochawesome可以通过配置生成测试报告页面和其他格式,非常美观。我们安装一下mochawesome

yarn add mochawesome --save-dev

运行之后,我们可以在根目录获得mocha-awesome的目录,结构如下

mocha+Typescript的单元测试搭建

我们打开mochawesome.html

得到这样一个精美的报告页面。

我们抛出一个问题:如果有测试未通过的用例怎么办?我们来试一下,增加一条用例,让第一条用例测试不通过:

demo.test.ts

/** * demo of unit test */
import { expect } from 'chai';import { Demo } from './demo';
// Demo类的测试声明describe('Demo', function () { describe('Demo.getSum', function() { it('1+1 应该等于 1', function() { expect(Demo.getSum([1, 1])).to.be.equal(1); });
it('1+2 应该等于 3', function() { expect(Demo.getSum([1, 2])).to.be.equal(3); }); });});

报告结果:

在这过程中,我遇到了一个问题,mocha在运行识别到错误之后,直接退出了进程,并没有运行后面的测试用例,并且没有生成报告。经过查找,有两处需要注意:

  1. mocha的 bail和allowUncaught 需要设置为false,这里默认为false,如果添加了上述两个属性为true,则会在报错的时候退出主进程,并执行mocha提供的hooks(mocha提供了before等hooks用于在遇到错误或者用例执行失败的时候进行监控并控制流程)

  2. mocha的reporter属性中,可以设置reporterOption子属性,用于对reporter的插件进行补充设置。其中mochawesome可以设置overwrite为true来覆盖之前生成的报告,防止已存在报告目录但不会被覆盖



以上是关于mocha+Typescript的单元测试搭建的主要内容,如果未能解决你的问题,请参考以下文章

Mocha 测试 TypeScript 奇怪的 TypeError

使用 typescript/mocha 进行单元测试时找不到模块

基于 Vue CLI / Typescript 的项目不再构建或运行 Mocha 测试

给 TypeScript 项目编写单元测试的一些小经验

mocha 单元测试中的 Highchart 导入错误

React-基于Mocha搭建测试框架