爲了研究與學習某些測試框架的工做原理,同時也爲了完成培訓中實現一個簡單的測試框架的緣由,我對should.js的代碼進行了學習與分析,如今與你們來進行交流下。javascript
以前是放在segmentfault上面的,如今往掘金上面也同步一份。java
其中ext
爲文件夾,其他爲js文件。segmentfault
其中should.js
爲整個項目入口,asssertion.js
爲should.js中的類,負責對測試信息進行記錄。assertion-error.js
爲should.js定義了一個錯誤類,負責存儲錯誤信息。config.js
中存儲了一些should.js中的一些配置信息。util.js
中則定義了一些項目中經常使用的工具函數。bash
should.js
var should = function should(obj) {
return (new should.Assertion(obj));
};
should.AssertionError = require('./assertion-error');
should.Assertion = require('./assertion');
should.format = util.format;
should.type = require('should-type');
should.util = util;
should.config = require('./config');
exports = module.exports = should;
複製代碼
should.js
入口文件初始化了一個類,並將全部文件中其餘的模塊進行引入。同時將本身export出去,讓本身可以被require到。app
should.extend = function (propertyName, proto) {
propertyName = propertyName || 'should';
proto = proto || Object.prototype;
var prevDescriptor = Object.getOwnPropertyDescriptor(proto, propertyName);
Object.defineProperty(proto, propertyName, {
set: function () {
},
get: function () {
return should(util.isWrapperType(this) ? this.valueOf() : this);
},
configurable: true
});
return {
name: propertyName, descriptor: prevDescriptor, proto: proto};
};
複製代碼
should.js
自身定義了一個extend方法,用於兼容should.js的另外一種調用方式,即should(obj)
的方式等於should.js
的常規調用方式obj.should
,從而兼容另外一種寫法。 框架
should
.use(require('./ext/assert'))
.use(require('./ext/chain'))
.use(require('./ext/bool'))
.use(require('./ext/number'))
.use(require('./ext/eql'))
.use(require('./ext/type'))
.use(require('./ext/string'))
.use(require('./ext/property'))
.use(require('./ext/error'))
.use(require('./ext/match'))
.use(require('./ext/contain'));
複製代碼
should.js
中還定義了use方法,從而讓咱們可以本身編寫一些類型判斷例如isNumber等函數導入到項目中,從而方便進行測試。項目目錄中的ext
文件夾就是編寫的一些簡單的should.js的擴展。後面將在介紹擴展時對二者的工做原理以及使用方法進行介紹。函數
assertion.js
function Assertion(obj) {
this.obj = obj;
/** * any標誌位 * @type {boolean} */
this.anyOne = false;
/** * not標誌位 * @type {boolean} */
this.negate = false;
this.params = {actual: obj};
}
複製代碼
assertion.js
中定義了一個Assertion類,其中any爲should.js中的any
方法的標誌位,而not則爲其not
方法的標誌位。工具
Assertion.add = function(name, func) {
var prop = {enumerable: true, configurable: true};
prop.value = function() {
var context = new Assertion(this.obj, this, name);
context.anyOne = this.anyOne;
try {
func.apply(context, arguments);
} catch(e) {
//check for fail
if(e instanceof AssertionError) {
//negative fail
if(this.negate) {
this.obj = context.obj;
this.negate = false;
return this;
}
if(context !== e.assertion) {
context.params.previous = e;
}
//positive fail
context.negate = false;
context.fail();
}
// throw if it is another exception
throw e;
}
//negative pass
if(this.negate) {
context.negate = true;//because .fail will set negate
context.params.details = 'false negative fail';
context.fail();
}
//positive pass
if(!this.params.operator) this.params = context.params;//shortcut
this.obj = context.obj;
this.negate = false;
return this;
};
Object.defineProperty(Assertion.prototype, name, prop);
};
複製代碼
assertion.js
中的add方法在Assertion的原型鏈中添加自定義命名的方法,從而讓咱們可以打包一些判斷的方法來進行調用,不須要重複進行代碼的編寫。該方法具體的使用方式咱們在後面對擴展進行講解時將會提到。 學習
Assertion.addChain = function(name, onCall) {
onCall = onCall || function() {
};
Object.defineProperty(Assertion.prototype, name, {
get: function() {
onCall();
return this;
},
enumerable: true
});
};
複製代碼
addChain
方法添加屬性到原型鏈中,該屬性在調用方法後返回調用者自己。該方法在should.js
的鏈式調用中起着重要的做用。測試
同時,Assertion類還支持別名功能,alias
方法使用Object對象的getOwnPropertyDescriptor
方法來對屬性是否存在進行判斷,並調用defineProperty
進行賦值。
Assertion
類在原型鏈中定義了assert
方法,用來對各級限制條件進行判斷。assert
方法與普通方法不一樣,它並未採用參數來進行一些參數的傳遞,而是經過assert
方法所在的Assertion
對象的params
屬性來進行參數的傳遞。由於在Assertion
對象中存儲了相關的信息,使用這個方法來進行參數傳遞方便在各級中assert
函數的調用方便。具體使用方法咱們將在擴展的分析時提到。
assert: function(expr) {
if(expr) return this;
var params = this.params;
if('obj' in params && !('actual' in params)) {
params.actual = params.obj;
} else if(!('obj' in params) && !('actual' in params)) {
params.actual = this.obj;
}
params.stackStartFunction = params.stackStartFunction || this.assert;
params.negate = this.negate;
params.assertion = this;
throw new AssertionError(params);
}
複製代碼
Assertion
類也定義了一個fail
方法可以讓用戶直接調用從而拋出一個Assertion的Error。
fail: function() {
return this.assert(false);
}
複製代碼
assertion-error.js
在此文件中,定義了assertion中拋出來的錯誤,同時在其中定義了一些信息存儲的函數例如message
和detail
等,可以讓錯誤在被捕獲的時候帶上一些特定的信息從而方便進行判斷與處理。因爲實現較爲簡單,所以在此就不貼出代碼,須要瞭解的人能夠本身去查閱should.js的源碼。
ext/bool.js
下面簡單介紹一個Assertion
的擴展的工做方式。讓咱們可以對should.js的工做原理有一個更加深入的理解。
module.exports = function(should, Assertion) {
/** * 判斷是否爲true */
Assertion.add('true', function() {
this.is.exactly(true);
});
/** * 別名爲True */
Assertion.alias('true', 'True');
/** * 判斷是否爲false */
Assertion.add('false', function() {
this.is.exactly(false);
});
/** * 別名False */
Assertion.alias('false', 'False');
/** * 經過對象檢查來判斷對象是否爲空 */
Assertion.add('ok', function() {
this.params = {operator: 'to be truthy'};
this.assert(this.obj);
});
};
//should.js
should.use = function (f) {
f(should, should.Assertion);
return this;
};
//use
'1'.should.be.true();
複製代碼
經過上面的擴展模塊代碼以及should.js
文件中的use
函數,咱們能夠發現,use
函數向擴展模塊傳入了should
方法和Assertion
構造函數。在bool.js
這個擴展模塊中,它經過調用Assertion
對象上的add函數來添加新的判斷方式,而且經過params
參數來告訴Assertion
對象若是判斷失敗應該如何提示用戶。
should.js
如何實現鏈式調用?在Assertion
類中,有一個addChain
方法,該方法爲某些屬性定義了一些在getter函數中調用的操做方法,而且返回對象自己。經過這個方法,在ext/chain.js
中,它爲should.js
中常見的語義詞添加了屬性,並經過返回對象自己來達到鏈式調用的Assertion
對象傳遞。
['an', 'of', 'a', 'and', 'be', 'has', 'have', 'with', 'is', 'which', 'the', 'it'].forEach(function(name) {
Assertion.addChain(name);
});
複製代碼
如下兩段代碼在結果上是如出一轍的效果:
'1'.shoud.be.a.Number();
'1'.should.be.be.be.be.a.a.a.a.Number();
複製代碼
should.js
的實現方式有哪些值得借鑑的地方?should.js
中,經過將一些語義詞添加爲屬性值並返回Assertion
對象自己,所以有效解決了鏈式調用的問題。Asseriton
對象的屬性來進行參數的傳遞,而不是經過函數參數,從而有效避免了函數調用時參數的傳遞問題以及多層調用時結構的複雜。should.js
經過擴展的方式來添加其判斷的函數,保證了良好的擴展性,避免了代碼耦合在一塊兒,經過也爲其餘人編寫更多的擴展代碼提供了接口。should.js
經過extend方法,讓should(obj)
與obj.should
兩種方式達到了相同的效果。經過在defineProperty
中定義should屬性而且在回調函數中用should(obj)
的方式來獲取obj
對象。總的來講,should.js
是一個比較小而精的測試框架,他可以知足在開發過程當中所須要的大部分測試場景,同時也支持本身編寫擴展來強化它的功能。在設計上,這個框架使用了很多巧妙的方法,避免了一些複雜的鏈式調用與參數傳遞等問題,並且結構清晰,比較適合進行閱讀與學習。