安裝必要的包
nodejs的單元測試最經常使用的是使用mocha
包。首先確保你本地安裝nodejs
,以後安裝mocha
包。node
npm install mocha -g
而後還須要安裝相關的斷言工具,Node.js中經常使用的斷言庫有:git
- assert: TDD風格
- should: BDD風格
- expect: BDD風格
- chai: BDD/TDD風格
使用npm install
安裝這些斷言庫其中之一便可。github
PHPStorm配置nodejs單元測試環境
在PHPStorm中選擇菜單:Run -> Edit Configurations,點擊右上角添加mocha
。 分別填寫下面幾項,關於mocha
單元測試能夠參考官網:https://mochajs.org/正則表達式
- Name: 隨便一個運行配置的名稱,如MochaTest
- Working directory: 當前項目目錄
- Mocha package: Mocha安裝包的目錄,node_modules\mocha
- User interface: 測試類型,這裏選擇TDD(對應
assert
庫) - Test directory: 這一項能夠選擇測試目錄或文件
- All in directory: 整個目錄都進行測試
- File patterns: 某種模式的文件,能夠填正則表達式
- Test file: 某個特定的測試文件
填寫完成而且沒有報錯後點擊OK。shell
Nodejs進行單元測試
這裏咱們選擇assert
庫,TDD模式進行單元測試。在上面選定的Test directory
目錄下新建一個測試文件test.js
.npm
const assert = require('assert'); // 測試Array類型的方法 suite('Array', function() { // 測試 indexOf方法 suite('#indexOf()', function() { // 測試用例 test('should return -1 when not present', function() { assert.equal(-1, [1, 2, 3].indexOf(4)); }); }); });
點擊選擇Mocha運行,在PHPStorm下面的輸出框中有測試的結果,綠色表示經過,紅色表示失敗。 數組
斷言庫的使用
mocha進行單元測試的時候,除了可以使用assert
斷言庫,只要斷言代碼中拋出Error
,mocha就能夠正常工做。promise
assert
庫:TDD風格
下面列舉assert
庫中經常使用的斷言函數,詳情可參考官網:https://www.npmjs.com/package/assertapp
- assert.fail(actual, expected, message, operator)
- assert(value, message), assert.ok(value, [message])
- assert.equal(actual, expected, [message])
- assert.notEqual(actual, expected, [message])
- assert.deepEqual(actual, expected, [message])
- assert.notDeepEqual(actual, expected, [message])
- assert.strictEqual(actual, expected, [message])
- assert.notStrictEqual(actual, expected, [message])
- assert.throws(block, [error], [message])
- assert.doesNotThrow(block, [message])
- assert.ifError(value)
其中的參數說明以下:async
- value: 實際值
- actual: 實際值
- expected: 指望值
- block: 語句塊
- message: 附加信息
BDD風格should.js
斷言庫
安裝方法:npm install should --save-dev
,官網地址:https://github.com/shouldjs/should.js
const should = require('should'); const user = { name: 'tj' , pets: ['tobi', 'loki', 'jane', 'bandit'] }; user.should.have.property('name', 'tj'); user.should.have.property('pets').with.lengthOf(4); // If the object was created with Object.create(null) // then it doesn't inherit `Object.prototype`, so it will not have `.should` getter // so you can do: should(user).have.property('name', 'tj'); // also you can test in that way for null's should(null).not.be.ok(); someAsyncTask(foo, function(err, result){ should.not.exist(err); should.exist(result); result.bar.should.equal(foo); });
should
庫可使用鏈式調用,功能很是強大。相關文檔參考:http://shouldjs.github.io/
user.should.be.an.instanceOf(Object).and.have.property('name', 'tj'); user.pets.should.be.instanceof(Array).and.have.lengthOf(4);
經常使用的should
斷言方法:
-
無心義謂詞,沒做用增長可讀性:
.an
,.of
,.a
,.and
,.be
,.have
,.with
,.is
,.which
-
should.equal(actual, expected, [message]): 判斷是否相等
-
should.notEqual(actual, expected, [message]): 判斷是否不相等
-
should.strictEqual(actual, expected, [message]): 判斷是否嚴格相等
-
should.notStrictEqual(actual, expected, [message]): 判斷是否嚴格不相等
-
should.deepEqual(actual, expected, [message]): 判斷是否遞歸相等
-
should.notDeepEqual(actual, expected, [message]): 判斷是否遞歸不想等
-
should.throws(block, [error], [message]): 判斷是否拋出異常
-
should.doesNotThrow(block, [message]): 判斷是否不拋出異常
-
should.fail(actual, expected, message, operator): 判斷是否不等
-
should.ifError(err): 判斷是否爲錯誤
-
should.exist(actual, [message]): 判斷對象是否存在
-
should.not.exist(actual, [message]): 判斷對象是否不存在
另外should
還提供了一系列類型判斷斷言方法:
// bool類型判斷 (true).should.be.true(); false.should.not.be.true(); // 數組是否包含 [ 1, 2, 3].should.containDeep([2, 1]); [ 1, 2, [ 1, 2, 3 ]].should.containDeep([ 1, [ 3, 1 ]]); // 數字比較 (10).should.not.be.NaN(); NaN.should.be.NaN(); (0).should.be.belowOrEqual(10); (0).should.be.belowOrEqual(0); (10).should.be.aboveOrEqual(0); (10).should.be.aboveOrEqual(10); // Promise狀態判斷 // don't forget to handle async nature (new Promise(function(resolve, reject) { resolve(10); })).should.be.fulfilled(); // test example with mocha it is possible to return promise it('is async', () => { return new Promise(resolve => resolve(10)) .should.be.fulfilled(); }); // 對象的屬性判斷 ({ a: 10 }).should.have.property('a'); ({ a: 10, b: 20 }).should.have.properties({ b: 20 }); [1, 2].should.have.length(2); ({}).should.be.empty(); // 類型檢查 [1, 2, 3].should.is.Array(); ({}).should.is.Object();
幾種常見的測試風格代碼舉例
BDD
BDD提供的接口有:describe(), context(), it(), specify(), before(), after(), beforeEach(), and afterEach().
describe('Array', function() { before(function() { // ... }); describe('#indexOf()', function() { context('when not present', function() { it('should not throw an error', function() { (function() { [1, 2, 3].indexOf(4); }.should.not.throw()); }); it('should return -1', function() { [1, 2, 3].indexOf(4).should.equal(-1); }); }); context('when present', function() { it('should return the index where the element first appears in the array', function() { [1, 2, 3].indexOf(3).should.equal(2); }); }); }); });
TDD
提供的接口有: suite(), test(), suiteSetup(), suiteTeardown(), setup(), and teardown():
suite('Array', function() { setup(function() { // ... }); suite('#indexOf()', function() { test('should return -1 when not present', function() { assert.equal(-1, [1, 2, 3].indexOf(4)); }); }); });
QUNIT
和TDD相似,使用suite()和test()標記測試永烈,包含的接口有:before(), after(), beforeEach(), and afterEach()。
function ok(expr, msg) { if (!expr) throw new Error(msg); } suite('Array'); test('#length', function() { var arr = [1, 2, 3]; ok(arr.length == 3); }); test('#indexOf()', function() { var arr = [1, 2, 3]; ok(arr.indexOf(1) == 0); ok(arr.indexOf(2) == 1); ok(arr.indexOf(3) == 2); }); suite('String'); test('#length', function() { ok('foo'.length == 3); });