内容简介:mocha诞生于2011年,是一个特征丰富的javascript测试框架,可以运行在node.js和浏览器上,使异步测试更简单和有趣。mocha测试连续运行,允许灵活和准确的报告,同时将未捕获的异常映射到正确的测试用例。公司项目中,没有自动化的单元测试,而是通过写if/else判断,多少有点懵逼所在在种种考虑之下,我们就选择mocha测试框架做单元测试
mocha诞生于2011年,是一个特征丰富的javascript测试框架,可以运行在node.js和浏览器上,使异步测试更简单和有趣。mocha测试连续运行,允许灵活和准确的报告,同时将未捕获的异常映射到正确的测试用例。
背景
公司项目中,没有自动化的单元测试,而是通过写if/else判断,多少有点懵逼
所在在种种考虑之下,我们就选择mocha测试框架做单元测试
测试报告
在terminal里运行
npm run mochawesome 复制代码
完成项目中的单元测试
单击 file:///
+项目所在地址+ /mochawesome-report/mochawesome.html
最终得到的就是这一份测试报告
待测试的接口
需要测试的代码如下
'use strict'; const router = require('express').Router(); const passport = require('passport'); const User = require('../collections/user'); const log = require('../services/logger').createLogger('userAuthentication'); const AUTH_ERR = require('../constant/errMessage').AUTH; const COMM_ERR = require('../constant/errMessage').COMMON; /** * @api {get} /v1/auth/ User auth information * @apiName UserAuthInfo * @apiGroup userAuthentication * * @apiParam {null} null. * * @apiSuccess {String} username The username of the current user. * @apiSuccess {date} last User last logon time. * * @apiSuccessExample Success-Response: * HTTP/1.1 200 OK * { * "username": "test", * "last": "2019-06-03T06:22:53.567Z" * } * * @apiError NOT_LOGIN The current User was not logon. * * @apiErrorExample Error-Response: * HTTP/1.1 401 Unauthorized * { * "err": "NOT_LOGIN", * "message": "User has not logon in!" * } */ router.get('/', function(req, res) { if (req.user) { res.json({ username: req.user.username, last: req.user.last }); } else { res.status(401).json({ err: 'NOT_LOGIN', message: AUTH_ERR.NOT_LOGIN }); } }); /** * @api {post} /v1/auth/register User Register * @apiName UserRegister * @apiGroup userAuthentication * * @apiParam {String} username New user's name. * @apiParam {String} password New user's password. * * @apiSuccess {String} username The username of the register user. * @apiSuccess {string} message The registering success info. * * @apiSuccessExample Success-Response: * HTTP/1.1 200 OK * { * "username": "gushen", * "message": "User registered successful" * } * * @apiError REGISTER_FAILURE The register failure. * * @apiErrorExample Error-Response: * HTTP/1.1 500 Internal Server Error * { * "err": "REGISTER_FAILURE", * "message": "User register failure!" * } */ router.post('/register', function(req, res, next) { User.register(new User({ username: req.body.username }), req.body.password, function(err) { if (err) { log.error(err); res.status(500).json({ err: 'REGISTER_FAILURE', message: AUTH_ERR.REGISTER_FAILURE }); return; } log.info('user ' + req.body.username + ' registered successful!'); res.json({ username: req.body.username, message: 'User registered successful' }); }); }); /** * @api {post} /v1/auth/login User login * @apiName UserLogin * @apiGroup userAuthentication * * @apiParam {String} username User's name. * @apiParam {String} password User's password. * * @apiSuccess {String} username The username of the register user. * @apiSuccess {string} message The messgaer if the user login in successful. * * @apiSuccessExample Success-Response: * HTTP/1.1 200 OK * { * "username": "test", * "message": "Authentication Success" * } * * @apiError REGISTER_FAILURE The register failure. * * @apiErrorExample Error-Response: * HTTP/1.1 401 Unauthorized * { * "err": "AUTHENTICATE_FAILURE", * "message": "Authenticate failure" * } */ router.post('/login', isAhenticated, passport.authenticate('local'), function(req, res) { if (req.user) { log.info(`${req.user.username} login in successful`); res.json({ username: req.user.username, message: 'Authentication Success' }); return; } log.info(`${req.user.username} login failure`); res.status(401).json({ err: 'AUTHENTICATE_FAILURE', message: `${req.user.username} login failure` }); }); /** * @api {post} /v1/auth/user/:username User delete * @apiName UserDelete * @apiGroup userAuthentication * * @apiParam {String} username User's name. * * @apiSuccess {String} username The username of the deleted user. * @apiSuccess {string} message The message if deleting successful. * * @apiSuccessExample Success-Response: * HTTP/1.1 200 OK * { * "username": "gushen", * "message": "Delete User Successful" * } * * @apiError NOT_LOGIN The register failure. * * @apiErrorExample Error-Response: * HTTP/1.1 401 Unauthorized * { * "err": "NOT_LOGIN", * "message": "User has not logon in!" * } */ router.delete('/user/:username', function(req, res) { if (!req.user) { res.status(401).json({ err: 'NOT_LOGIN', message: AUTH_ERR.NOT_LOGIN }); return; } // if (!req.params.username) { // res.json({ // err: 'PARAMS_NOT_CORRECT', // message: 'No deleted user name' // }); // return; // } User.deleteOne({ username: req.params.username }, (err) => { if (err) { log.error(err); res.status(500).json({ err: 'SERVER_ERROR', message: COMM_ERR.SERVER_ERROR }); return; } res.json({ username: req.params.username, message: 'Delete User Successful' }); log.info(`${req.params.username} has been deleted`); }); }); /** * @api {post} /v1/auth/changepassword User change password * @apiName UserChangePassword * @apiGroup userAuthentication * * @apiParam {String} username User's name. * @apiParam {String} oldpassword User's old password. * @apiParam {String} newpassword User's old password. * * @apiSuccess {String} username The username of the user. * @apiSuccess {string} message The message if changing password successful. * * @apiSuccessExample Success-Response: * HTTP/1.1 200 OK * { * "username": "test", * "message": "change password successful" * } * * @apiError AUTHENTICATE_FAILURE The register failure. * * @apiErrorExample Error-Response: * HTTP/1.1 401 Unauthorized * { * "err": "AUTHENTICATE_FAILURE", * "message": "Password or username is incorrect" * } */ router.post('/changepassword', function(req, res) { User.findOne({ 'username': req.body.username }, (err, user) => { if (err) { log.error(err); res.status(500).json({ err: 'SERVER_ERROR', message: COMM_ERR.SERVER_ERROR }); return; } if (!user) { res.status(500).json({ err: 'USER_NOT_EXIST', message: AUTH_ERR.USER_NOT_EXIST }); return; } user.changePassword(req.body.oldpassword, req.body.newpassword, (err, value) => { if (err) { log.error(err); res.status(401).json({ err: 'AUTHENTICATE_FAILURE', message: err.message }); return; } log.info(`${req.body.username} change password successful`); res.json({ username: req.body.username, message: 'change password successful' }); }); }); }); /** * @api {get} /v1/auth/logout User login out * @apiName UserLogout * @apiGroup userAuthentication * * @apiSuccess {String} username The username of the user. * @apiSuccess {string} message The message if user login out successful. * * @apiSuccessExample Success-Response: * HTTP/1.1 200 OK * { * "username": "test", * "message": "logout successful" * } * * @apiError NOT_LOGIN There is no user logon in. * * @apiErrorExample Error-Response: * HTTP/1.1 401 Unauthorized * { * "err": "NOT_LOGIN", * "message": "No user has been logon" * } */ router.get('/logout', function(req, res) { const user = req.user; if (!user) { res.status(401).json({ err: 'NOT_LOGIN', message: 'No user has been logon' }); return; } // user login out req.logout(); if (!req.user) { res.json({ username: user.username, message: 'logout successful' }); log.info(`${user.username} has been logon out`); return; } res.status(500).json({ err: 'SERVER_ERROR', message: 'logout failure!' }); }); function isAhenticated(req, res, next) { User.findOne({ 'username': req.body.username }, (err, user) => { if (err) { log.error(err); res.json({ err: 'SERVER_ERROR', message: COMM_ERR.SERVER_ERROR }); return; } // If user is not existed if (!user) { res.json({ err: 'USER_NOT_EXIST', message: AUTH_ERR.USER_NOT_EXIST }); return; } user.authenticate(req.body.password, (err, value) => { if (err) { log.error(err); res.json({ err: 'SERVER_ERROR', message: COMM_ERR.SERVER_ERROR }); } else if (value) { return next(); } else { res.json({ err: 'AUTHENTICATE_FAILURE', message: AUTH_ERR.AUTHENTICATE_FAILURE }); } }); }); } module.exports = router; 复制代码
这是一套常见的有关用户登录注册验证的接口
因为文本只涉及到这个模块,所以将这个模块的接口都写在 userAuthentication
测试套件下
describe('userAuthentication', function() {} 复制代码
测试代码
'use strict'; const request = require('supertest'); const url = 'http://localhost:5001'; // eslint-disable-next-line no-unused-vars const should = require('should'); var userCookie; // 用户名密码 const user = { username: 'name', password: 'password' }; // 测试更改密码(每次测试完调换) const user2 = { username: 'uu2', password: 'oldpassword' }; const newUser2 = { username: 'uu2', oldpassword: 'oldpassword', newpassword: 'newpassword' }; // const user22={ // username: 'uu2', // password: 'newpassword' // }; // const oldUser2 = { // username: 'uu2', // oldpassword: 'newpassword', // newpassword: 'oldpassword' // }; describe('userAuthentication', function() { // 测试注册接口 describe('UserRegister', function() { describe('POST /register', function() { // eslint-disable-next-line max-len it('register success', function(done) { request(url) .post('/api/v1/auth/register') .send(user) .expect(200) .end(function(err, res) { res.body.should.containEql({ message: 'User registered successful' }); if (err) throw err; done(); }); }); it('repeated registration failure.', function(done) { request(url) .post('/api/v1/auth/register') .send(user) .expect(500) .end(function(err, res) { res.body.should.containEql({ err: 'REGISTER_FAILURE' }); if (err) throw err; done(); }); }); }); }); // 测试登录接口 describe('UserLogin', function() { describe('POST /login', function() { it('login success', function(done) { request(url) .post('/api/v1/auth/login') .send(user) .expect(200) .end(function(err, res) { res.body.should.containEql({ message: 'Authentication Success' }); if (err) throw err; done(); }); }); it('USER_NOT_EXIST.', function(done) { request(url) .post('/api/v1/auth/login') .send({ username: 'a', password: 'admin' }) .expect(200) .end(function(err, res) { res.body.should.containEql({ err: 'USER_NOT_EXIST' }); if (err) throw err; done(); }); }); }); }); // 权限验证 describe('UserAuthInfo', function() { describe('GET /api/v1/auth/', function() { // 没有登录,权限验证 it('The current User was not login.', function(done) { request(url) .get('/api/v1/auth/') .set('Accept', 'application/json') .expect('Content-Type', /json/) .expect(401) .end(function(err, res) { res.body.should.containEql({ err: 'NOT_LOGIN' }); if (err) throw err; done(); }); }); // 权限验证前先登录 beforeEach(function(done) { request(url) .post('/api/v1/auth/login') .send(user) .set('Accept', 'application/json') .end(function(err, res) { if (!err) { userCookie = res.header['set-cookie']; done(); } }); }); it('The username of the current user.', function(done) { request(url) .get('/api/v1/auth/') .set('Cookie', userCookie) .expect(200) .end(function(err, res) { res.body.should.have.keys('username'); if (err) throw err; done(); }); }); }); }); // 测试用户注销接口 describe('UserLogout', function() { describe('GET /logout', function() { // 没有登录,测试注销 it('NOT_LOGIN.', function(done) { request(url) .get('/api/v1/auth/logout') .expect(401) .end(function(err, res) { res.body.should.containEql({ err: 'NOT_LOGIN' }); if (err) throw err; done(); }); }); // 注销成功前先登录 beforeEach(function(done) { request(url) .post('/api/v1/auth/login') .send(user) .set('Accept', 'application/json') .end(function(err, res) { if (!err) { userCookie = res.header['set-cookie']; done(); } }); }); it('logout successful.', function(done) { request(url) .get('/api/v1/auth/logout') .set('Cookie', userCookie) .expect(200) .end(function(err, res) { res.body.should.containEql({ message: 'logout successful' }); if (err) throw err; done(); }); }); }); }); // 测试更改用户密码接口 describe('UserChangePassword', function() { describe('POST /changepassword', function() { // 更改用户密码前先注册-登录 // eslint-disable-next-line no-undef before(function(done) { request(url) .post('/api/v1/auth/register') .send(user2) .end(function(err, res) { if (err) throw err; done(); }); }); // eslint-disable-next-line no-undef before(function(done) { request(url) .post('/api/v1/auth/login') .send(user2) .set('Accept', 'application/json') .end(function(err, res) { if (!err) { userCookie = res.header['set-cookie']; done(); } }); }); it('change password successful', function(done) { request(url) .post('/api/v1/auth/changepassword') .set('Cookie', userCookie) .send(newUser2) .expect(200) .end(function(err, res) { res.body.should.containEql({ message: 'change password successful' }); if (err) throw err; done(); }); }); it('AUTHENTICATE_FAILURE', function(done) { request(url) .post('/api/v1/auth/changepassword') .set('Cookie', userCookie) .send(newUser2) .expect(401) .end(function(err, res) { res.body.should.containEql({ err: 'AUTHENTICATE_FAILURE' }); if (err) throw err; done(); }); }); // after(function(done) { // request(url) // .post('/api/v1/auth/login') // .send(user22) // .set('Accept', 'application/json') // .end(function(err, res) { // if (!err) { // userCookie = res.header['set-cookie']; // done(); // } // }); // }); // after(function(done) { // request(url) // .post('/api/v1/auth/changepassword') // .set('Cookie', userCookie) // .send(oldUser2) // .expect(200) // .end(function(err, res) { // res.body.should.containEql({ // message: 'rechange password successful' // }); // if (err) throw err; // done(); // }); // }); }); }); // 测试删除用户接口 describe('UserDelete', function() { describe('DELETE /user/:username', function() { it('NOT_LOGIN.', function(done) { request(url) .delete(`/api/v1/auth/user/${user.username}`) .expect(401) .end(function(err, res) { res.body.should.containEql({ err: 'NOT_LOGIN' }); if (err) throw err; done(); }); }); // 删除用户前先登录 beforeEach(function(done) { request(url) .post('/api/v1/auth/login') .send(user) .set('Accept', 'application/json') .end(function(err, res) { if (!err) { userCookie = res.header['set-cookie']; done(); } }); }); it('delete user success', function(done) { request(url) .delete(`/api/v1/auth/user/${user.username}`) .set('Cookie', userCookie) .expect(200) .end(function(err, res) { res.body.should.containEql({ message: 'Delete User Successful' }); if (err) throw err; done(); }); }); }); }); }); 复制代码
测试前的准备
测试框架
所谓"测试框架",就是运行测试的工具。通过它,可以为JavaScript应用添加测试,从而保证代码的质量。
通常应用会有单元测试(Unit tests)和功能测试(Functional tests),复杂大型应用可能会有整合测试(Integration tests)。
- 单元测试:关注应用中每个零部件的正常运转,防止后续修改影响之前的组件。
- 功能测试:确保其整体表现符合预期,关注能否让用户正常使用。
- 整合测试:确保单独运行正常的零部件整合到一起之后依然能正常运行。
开发人员主要是集中单元测试,作为开发中的反馈。
单元测试的好处:
- 如果能通过单元测试,那么通过后续测试且软件整体正常运行的概率大大提高。
- 单元测试发现的问题定位到细节,容易修改,节省时间。
- 追踪问题变得更加方便。
选择单元测试框架
单元测试应该:简单,快速执行,清晰的错误报告。 测试框架基本上都做了同一件事儿:
- 描述你要测试的东西
- 对其进行测试
- 判断是否符合预期
选择框架会考虑下面的点:
- 断言(Assertions):用于判断结果是否符合预期。有些框架需要单独的断言库。
- 适合 TDD / BDD:是否适合 测试驱动型 / 行为驱动型 的测试风格。
- 异步测试:有些框架对异步测试支持良好。
- 使用的语言:大部分 js 测试框架使用 js。
- 用于特定目的:每个框架可能会擅长处理不同的问题。
- 社区是否活跃。 注:
- TDD:测试驱动型的开发方式,先写测试代码,之后编写能通过测试的业务代码,可以不断的在能通过测试的情况下重构。
- BDD:与 TDD 很相似,测试代码的风格是预期结果,更关注功能,看起来像需求文档。
其实都是先写测试代码,感觉BDD 风格更人性。
测试 工具 的类型
组合使用工具很常见,即使已选框架也能实现类似的功能
- 提供测试框架(Mocha, Jasmine, Jest, Cucumber)
- 提供断言(Chai, Jasmine, Jest, Unexpected)
- 生成,展示测试结果(Mocha, Jasmine, Jest, Karma)
- 快照测试(Jest, Ava)
- 提供仿真(Sinon, Jasmine, enzyme, Jest, testdouble)
- 生成测试覆盖率报告(Istanbul, Jest, Blanket)
- 提供类浏览器环境(Protractor, Nightwatch, Phantom, Casper)
解释上面提到的点:
- 测试框架,即组织你的测试,当前流行 BDD 的测试结构。
- 快照测试(snapshot testing),测试 UI 或数据结构是否和之前完全一致,通常 UI 测试不在单元测试中
- 仿真(mocks, spies, and stubs):获取方法的调用信息,模拟方法,模块,甚至服务器
Jest/Jasmine/Mocha框架特点
-
Jest
- facebook 坐庄
- 基于 Jasmine 至今已经做了大量修改添加了很多特性
- 开箱即用配置少,API简单
- 支持断言和仿真
- 支持快照测试
- 在隔离环境下测试
- 互动模式选择要测试的模块
- 优雅的测试覆盖率报告,基于Istanbul
- 智能并行测试(参考)
- 较新,社区不十分成熟
- 全局环境,比如 describe 不需要引入直接用
- 较多用于 React 项目(但广泛支持各种项目)
-
Jasmine
- 开箱即用(支持断言和仿真)
- 全局环境
- 比较'老',坑基本都有人踩过了
- AVA
- 异步,性能好
- 简约,清晰
- 快照测试和断言需要三方支持
- Tape
- 体积最小,只提供最关键的东西
- 对比其他框架,只提供最底层的 API
-
Mocha
- 灵活(不包括断言和仿真,自己选对应工具)
- 流行的选择:chai,sinon
- 社区成熟用的人多,测试各种东西社区都有示例
- 需要较多配置
- 可以使用快照测试,但依然需要额外配置
综上所述,Mocha 用的人最多,社区最成熟,灵活,可配置性强易拓展,Jest 开箱即用,里边啥都有提供全面的方案,Tape 最精简,提供最基础的东西最底层的API。所以本文就选择用mocha。
mocha特征
- 浏览器支持
- 全局变量泄漏检测
- 简单异步支持,包括promise
- 可以选择运行与regexp匹配的测试
- 测试覆盖率报告
- 自动退出以防止活动循环“挂起”
- 字符串差异支持
- 易于元生成套件和测试用例
- 用于运行测试的javascript API
- 配置文件支持
- CI支持等的正确退出状态
- mocha.opts文件支持
- 自动检测和禁用非tty的着色
- 可单击套件标题以筛选测试执行
- 将未捕获的异常映射到正确的测试用例
- 节点调试器支持
- 异步测试超时支持
- 检测多个要完成的调用
- 测试重试支持
- 测试特定超时
- 咆哮支持
- 报告测试持续时间
- 突出显示慢速测试
- 文件监视程序支持
- 使用所需的任何断言库
- extensible reporting, bundled with 9+ reporters
- 可扩展测试DSL或“接口”
- 每个钩子之前、之后、所有钩子之前、之后
- 任意蒸腾器支持(coffee-script 等)
- TextMate bundle
断言库should
Mocha本身是不包含断言库的,所以我们需要自己选择断言库。should是一个很简单的、贴近自然语言的断言库。当然,Mocha是适配所有的断言库的,如果你喜欢其他的断言库比如expect之类的,你也可以把它包含进来使用。
http测试模块SuperTest
单单使用Mocha和should就几乎可以满足所有JavaScript函数的单元测试。但是对于Node应用而言,不仅仅是函数的集合,比如一个web应用的测试。这时候就需要配合一个http代理,完成Http请求和路由的测试。 Supertest是一个HTTP代理服务引擎,可以模拟一切HTTP请求行为。Supertest可以搭配任意的应用框架,从而进行应用的单元测试。
测试套件describe
describe块称为"测试套件"(test suite),表示一组相关的测试。它是一个函数,第一个参数是测试套件的名称("加法函数的测试"),第二个参数是一个实际执行的函数。
测试用例it
it块称为"测试用例"(test case),表示一个单独的测试,是测试的最小单位。它也是一个函数,第一个参数是测试用例的名称("1 加 1 应该等于 2"),第二个参数是一个实际执行的函数。
钩子hooks
Mocha在describe块之中,提供测试用例的四个钩子:before()、after()、beforeEach()和afterEach()。它们会在指定时间执行。
describe('hooks', function() { before(function() { // 在本区块的所有测试用例之前执行 }); after(function() { // 在本区块的所有测试用例之后执行 }); beforeEach(function() { // 在本区块的每个测试用例之前执行 }); afterEach(function() { // 在本区块的每个测试用例之后执行 }); // test cases }); 复制代码
安装mocha> = v3.0.0,npm的版本应该> = v2.14.2。除此,确保使用Node.js的版本> = v4来运行mocha
mocha小例子
安装
作为项目的依赖进行安装
npm install --save-dev mocha 复制代码
开始
mkdir test cd test touch test.js 复制代码
加入测试代码
'use strict' var assert = require('assert'); describe('Array', function() { describe('#indexOf()', function() { it('should return 0 when the value is not present', function() { assert.equal([1, 2, 3].indexOf(1), 1); }); }); }); 复制代码
安装依赖
npm install --save-dev assert 复制代码
执行测试
./node_modules/mocha/bin/mocha 复制代码
报错结果
Array #indexOf() 1) should return 0 when the value is not present 0 passing (5ms) 1 failing 1) Array #indexOf() should return 0 when the value is not present: AssertionError [ERR_ASSERTION]: 0 == 1 + expected - actual -0 +1 at Context.<anonymous> (test/test.js:6:14) 复制代码
package.json中写入命令
"mocha": "mocha" 复制代码
执行命令
npm run mocha 复制代码
正确测试
'use strict' var assert = require('assert'); describe('Array', function() { describe('#indexOf()', function() { it('should return 0 when the value is not present', function() { assert.equal([1, 2, 3].indexOf(1), 0); }); }); }); 复制代码
正确结果
Array #indexOf() ✓ should return 0 when the value is not present 1 passing (4ms) 复制代码
到这里,对mocha就有了初步的认识
开始测试
了解了背景和框架后,正式开启测试
添加依赖
npm install --save-dev mocha mochawesome should supertest 复制代码
在scripts中添加命令
"mochawesome": "./node_modules/.bin/mocha --reporter mochawesome", "dev": "node index.js" 复制代码
mochawesome生成报告
dev启动项目
注册接口的测试
- 从注册接口中,我得知该接口返回两个状态码,分别是200和500,对应的注册成功和注册失败
- 那么测试中就有两个注册成功和失败的测试用例
- 每个测试用例针对每个状态返回的值判断
- 通过即可
- 不通过,要么是接口有问题,要么是写的测试有问题
/** * @api {post} /v1/auth/register User Register * @apiName UserRegister * @apiGroup userAuthentication * * @apiParam {String} username New user's name. * @apiParam {String} password New user's password. * * @apiSuccess {String} username The username of the register user. * @apiSuccess {string} message The registering success info. * * @apiSuccessExample Success-Response: * HTTP/1.1 200 OK * { * "username": "gushen", * "message": "User registered successful" * } * * @apiError REGISTER_FAILURE The register failure. * * @apiErrorExample Error-Response: * HTTP/1.1 500 Internal Server Error * { * "err": "REGISTER_FAILURE", * "message": "User register failure!" * } */ router.post('/register', function(req, res, next) { User.register(new User({ username: req.body.username }), req.body.password, function(err) { if (err) { log.error(err); res.status(500).json({ err: 'REGISTER_FAILURE', message: AUTH_ERR.REGISTER_FAILURE }); return; } log.info('user ' + req.body.username + ' registered successful!'); res.json({ username: req.body.username, message: 'User registered successful' }); }); }); 复制代码
UserRegister POST /register register success
注意: 每个测试用例结束后必须带上 done
,否则没有结束标识,会超时报错
// 测试注册接口 describe('UserRegister', function() { describe('POST /register', function() { // eslint-disable-next-line max-len it('register success', function(done) { request(url) .post('/api/v1/auth/register') .send(user) .expect(200) .end(function(err, res) { res.body.should.containEql({ message: 'User registered successful' }); if (err) throw err; done(); }); }); it('repeated registration failure.', function(done) { request(url) .post('/api/v1/auth/register') .send(user) .expect(500) .end(function(err, res) { res.body.should.containEql({ err: 'REGISTER_FAILURE' }); if (err) throw err; done(); }); }); }); }); 复制代码
登录接口的测试
没什么好讲的,同测试注册接口步骤一致
describe('UserLogin', function() { describe('POST /login', function() { it('login success', function(done) { request(url) .post('/api/v1/auth/login') .send(user) .expect(200) .end(function(err, res) { res.body.should.containEql({ message: 'Authentication Success' }); if (err) throw err; done(); }); }); it('USER_NOT_EXIST.', function(done) { request(url) .post('/api/v1/auth/login') .send({ username: 'a', password: 'admin' }) .expect(200) .end(function(err, res) { res.body.should.containEql({ err: 'USER_NOT_EXIST' }); if (err) throw err; done(); }); }); }); }); 复制代码
权限验证的测试
before
userCookie = res.header['set-cookie']; 复制代码
- 在断言的请求中带上
Cookie
.set('Cookie', userCookie) 复制代码
// 权限验证 describe('UserAuthInfo', function() { describe('GET /api/v1/auth/', function() { // 没有登录,权限验证 it('The current User was not login.', function(done) { request(url) .get('/api/v1/auth/') .set('Accept', 'application/json') .expect('Content-Type', /json/) .expect(401) .end(function(err, res) { res.body.should.containEql({ err: 'NOT_LOGIN' }); if (err) throw err; done(); }); }); // 权限验证前先登录 beforeEach(function(done) { request(url) .post('/api/v1/auth/login') .send(user) .set('Accept', 'application/json') .end(function(err, res) { if (!err) { userCookie = res.header['set-cookie']; done(); } }); }); it('The username of the current user.', function(done) { request(url) .get('/api/v1/auth/') .set('Cookie', userCookie) .expect(200) .end(function(err, res) { res.body.should.have.keys('username'); if (err) throw err; done(); }); }); }); }); 复制代码
用户注销接口的测试
没什么好讲的,同测试权限验证步骤一致
describe('UserLogout', function() { describe('GET /logout', function() { // 没有登录,测试注销 it('NOT_LOGIN.', function(done) { request(url) .get('/api/v1/auth/logout') .expect(401) .end(function(err, res) { res.body.should.containEql({ err: 'NOT_LOGIN' }); if (err) throw err; done(); }); }); // 注销成功前先登录 beforeEach(function(done) { request(url) .post('/api/v1/auth/login') .send(user) .set('Accept', 'application/json') .end(function(err, res) { if (!err) { userCookie = res.header['set-cookie']; done(); } }); }); it('logout successful.', function(done) { request(url) .get('/api/v1/auth/logout') .set('Cookie', userCookie) .expect(200) .end(function(err, res) { res.body.should.containEql({ message: 'logout successful' }); if (err) throw err; done(); }); }); }); }); 复制代码
更改用户密码的测试
更改用户密码前先注册-登录
// 测试更改用户密码接口 describe('UserChangePassword', function() { describe('POST /changepassword', function() { // 更改用户密码前先注册-登录 // eslint-disable-next-line no-undef before(function(done) { request(url) .post('/api/v1/auth/register') .send(user2) .end(function(err, res) { if (err) throw err; done(); }); }); // eslint-disable-next-line no-undef before(function(done) { request(url) .post('/api/v1/auth/login') .send(user2) .set('Accept', 'application/json') .end(function(err, res) { if (!err) { userCookie = res.header['set-cookie']; done(); } }); }); it('change password successful', function(done) { request(url) .post('/api/v1/auth/changepassword') .set('Cookie', userCookie) .send(newUser2) .expect(200) .end(function(err, res) { res.body.should.containEql({ message: 'change password successful' }); if (err) throw err; done(); }); }); it('AUTHENTICATE_FAILURE', function(done) { request(url) .post('/api/v1/auth/changepassword') .set('Cookie', userCookie) .send(newUser2) .expect(401) .end(function(err, res) { res.body.should.containEql({ err: 'AUTHENTICATE_FAILURE' }); if (err) throw err; done(); }); }); // after(function(done) { // request(url) // .post('/api/v1/auth/login') // .send(user22) // .set('Accept', 'application/json') // .end(function(err, res) { // if (!err) { // userCookie = res.header['set-cookie']; // done(); // } // }); // }); // after(function(done) { // request(url) // .post('/api/v1/auth/changepassword') // .set('Cookie', userCookie) // .send(oldUser2) // .expect(200) // .end(function(err, res) { // res.body.should.containEql({ // message: 'rechange password successful' // }); // if (err) throw err; // done(); // }); // }); }); }); 复制代码
问题是我改完后得将密码改回来,这一步我没有成功,很奇怪为什么?
目前得每次测试完后将新旧密码调换,真的很麻烦
删除用户的测试
没什么好讲的,同测试权限验证步骤一致
describe('UserDelete', function() { describe('DELETE /user/:username', function() { it('NOT_LOGIN.', function(done) { request(url) .delete(`/api/v1/auth/user/${user.username}`) .expect(401) .end(function(err, res) { res.body.should.containEql({ err: 'NOT_LOGIN' }); if (err) throw err; done(); }); }); // 删除用户前先登录 beforeEach(function(done) { request(url) .post('/api/v1/auth/login') .send(user) .set('Accept', 'application/json') .end(function(err, res) { if (!err) { userCookie = res.header['set-cookie']; done(); } }); }); it('delete user success', function(done) { request(url) .delete(`/api/v1/auth/user/${user.username}`) .set('Cookie', userCookie) .expect(200) .end(function(err, res) { res.body.should.containEql({ message: 'Delete User Successful' }); if (err) throw err; done(); }); }); }); }); 复制代码
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网
猜你喜欢:- Springboot Application 集成 OSGI 框架开发
- 禅道 12.3.stable 版本发布,全面集成八种单元测试框架,打通持续集成闭环
- Chef 15.0.118 发布,系统集成框架
- Chef 15.0.118 发布,系统集成框架
- 微服务框架 Micronaut 集成 Kafka、GraphQL 实战
- 学习 Spring Boot(七):集成 Apache Shiro 安全框架
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
网络是怎样连接的
[日]户根勤 / 周自恒 / 人民邮电出版社 / 2017-1-1 / CNY 49.00
本书以探索之旅的形式,从在浏览器中输入网址开始,一路追踪了到显示出网页内容为止的整个过程,以图配文,讲解了网络的全貌,并重点介绍了实际的网络设备和软件是如何工作的。目的是帮助读者理解网络的本质意义,理解实际的设备和软件,进而熟练运用网络技术。同时,专设了“网络术语其实很简单”专栏,以对话的形式介绍了一些网络术语的词源,颇为生动有趣。 本书图文并茂,通俗易懂,非常适合计算机、网络爱好者及相关从......一起来看看 《网络是怎样连接的》 这本书的介绍吧!