306 lines
6.6 KiB
JavaScript
Executable File
306 lines
6.6 KiB
JavaScript
Executable File
// Crockford's helpers
|
|
|
|
// Object.create instead of new Object
|
|
if (typeof Object.create !== 'function') {
|
|
Object.create = function (o) {
|
|
var F = function () {
|
|
};
|
|
F.prototype = o;
|
|
return new F();
|
|
};
|
|
}
|
|
|
|
// Klass.method instead of Klass.prototype.name = function
|
|
if (typeof Function.method !== 'function') {
|
|
Function.prototype.method = function (name, func) {
|
|
this.prototype[name] = func;
|
|
return this;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* object for holding results; allows for the results array to hold another nestedResults()
|
|
*
|
|
*/
|
|
|
|
var nestedResults = function() {
|
|
var that = {
|
|
totalCount: 0,
|
|
passedCount: 0,
|
|
failedCount: 0,
|
|
results: [],
|
|
|
|
rollupCounts: function (result) {
|
|
that.totalCount += result.totalCount;
|
|
that.passedCount += result.passedCount;
|
|
that.failedCount += result.failedCount;
|
|
},
|
|
|
|
push: function (result) {
|
|
if (result.results) {
|
|
that.rollupCounts(result);
|
|
} else {
|
|
that.totalCount++;
|
|
result.passed ? that.passedCount++ : that.failedCount++;
|
|
}
|
|
that.results.push(result);
|
|
}
|
|
}
|
|
|
|
return that;
|
|
}
|
|
|
|
|
|
/*
|
|
* base for Runner & Suite: allows for a queue of functions to get executed, allowing for
|
|
* any one action to complete, including asynchronous calls, before going to the next
|
|
* action.
|
|
*
|
|
**/
|
|
var actionCollection = function () {
|
|
var that = {
|
|
actions: [],
|
|
index: 0,
|
|
finished: false,
|
|
results: nestedResults(),
|
|
|
|
report: function (result) {
|
|
that.results.push(result);
|
|
},
|
|
|
|
execute: function () {
|
|
if (that.actions.length > 0) {
|
|
that.next();
|
|
}
|
|
},
|
|
|
|
getCurrentAction: function () {
|
|
return that.actions[that.index];
|
|
},
|
|
|
|
next: function() {
|
|
if (that.index >= that.actions.length) {
|
|
that.finished = true;
|
|
return;
|
|
}
|
|
|
|
var currentAction = that.getCurrentAction();
|
|
|
|
if (that.beforeEach) {
|
|
that.beforeEach.apply(currentAction);
|
|
}
|
|
|
|
currentAction.execute();
|
|
that.waitForDone(currentAction);
|
|
},
|
|
|
|
waitForDone: function(action) {
|
|
var id = setInterval(function () {
|
|
if (action.finished) {
|
|
clearInterval(id);
|
|
that.report(action.results);
|
|
|
|
if (that.afterEach) {
|
|
that.afterEach.apply(action);
|
|
}
|
|
|
|
that.index++;
|
|
that.next();
|
|
}
|
|
}, 150);
|
|
}
|
|
}
|
|
|
|
return that;
|
|
}
|
|
|
|
/******************************************************************************
|
|
* Jasmine
|
|
******************************************************************************/
|
|
|
|
/*
|
|
* Matchers methods; add your own with Matchers.method()
|
|
*/
|
|
|
|
Matchers = function (actual, results) {
|
|
this.actual = actual;
|
|
this.passing_message = 'Passed.'
|
|
this.results = results || nestedResults();
|
|
}
|
|
|
|
Matchers.method('report', function (result, failing_message) {
|
|
|
|
this.results.push({
|
|
passed: result,
|
|
message: result ? this.passing_message : failing_message
|
|
});
|
|
|
|
return result;
|
|
});
|
|
|
|
Matchers.method('should_equal', function (expected) {
|
|
return this.report((this.actual === expected),
|
|
'Expected ' + expected + ' but got ' + this.actual + '.');
|
|
});
|
|
|
|
Matchers.method('should_not_equal', function (expected) {
|
|
return this.report((this.actual !== expected),
|
|
'Expected ' + expected + ' to not equal ' + this.actual + ', but it does.');
|
|
});
|
|
|
|
/*
|
|
* Jasmine spec constructor
|
|
*/
|
|
|
|
var queuedFunction = function(func, timeout, spec) {
|
|
var that = {
|
|
func: func,
|
|
next: function () {
|
|
spec.finish(); // default value is to be done after one function
|
|
},
|
|
execute: function () {
|
|
if (timeout > 0) {
|
|
setTimeout(function () {
|
|
that.func.apply(spec);
|
|
that.next();
|
|
}, timeout);
|
|
} else {
|
|
that.func.apply(spec);
|
|
that.next();
|
|
}
|
|
}
|
|
}
|
|
return that;
|
|
}
|
|
|
|
var it = function (description, func) {
|
|
var that = {
|
|
description: description,
|
|
queue: [],
|
|
currentTimeout: 0,
|
|
finished: false,
|
|
|
|
results: nestedResults(),
|
|
|
|
expects_that: function (actual) {
|
|
return new Matchers(actual, that.results);
|
|
},
|
|
|
|
waits: function (timeout) {
|
|
that.currentTimeout = timeout;
|
|
return that;
|
|
},
|
|
|
|
resetTimeout: function() {
|
|
that.currentTimeout = 0;
|
|
},
|
|
|
|
finish: function() {
|
|
that.finished = true;
|
|
},
|
|
|
|
execute: function () {
|
|
if (that.queue[0]) {
|
|
that.queue[0].execute();
|
|
}
|
|
}
|
|
};
|
|
|
|
var addToQueue = function(func) {
|
|
var currentFunction = queuedFunction(func, that.currentTimeout, that);
|
|
that.queue.push(currentFunction);
|
|
|
|
if (that.queue.length > 1) {
|
|
var previousFunction = that.queue[that.queue.length - 2];
|
|
previousFunction.next = function () {
|
|
currentFunction.execute();
|
|
}
|
|
}
|
|
|
|
that.resetTimeout();
|
|
return that;
|
|
}
|
|
|
|
that.runs = addToQueue;
|
|
|
|
currentSuite.specs.push(that);
|
|
currentSpec = that;
|
|
|
|
if (func) {
|
|
func();
|
|
}
|
|
|
|
that.results.description = description;
|
|
return that;
|
|
}
|
|
|
|
var runs = function (func) {
|
|
currentSpec.runs(func);
|
|
}
|
|
|
|
var waits = function (timeout) {
|
|
currentSpec.waits(timeout);
|
|
}
|
|
|
|
var beforeEach = function (beforeEach) {
|
|
currentSuite.beforeEach = beforeEach;
|
|
}
|
|
|
|
var afterEach = function (afterEach) {
|
|
currentSuite.afterEach = afterEach;
|
|
}
|
|
|
|
var describe = function (description, spec_definitions) {
|
|
var that = actionCollection();
|
|
|
|
that.description = description;
|
|
that.specs = that.actions;
|
|
|
|
currentSuite = that;
|
|
Jasmine.suites.push(that);
|
|
|
|
spec_definitions();
|
|
|
|
that.results.description = description;
|
|
|
|
return that;
|
|
}
|
|
|
|
var Runner = function () {
|
|
var that = actionCollection();
|
|
|
|
that.suites = that.actions;
|
|
that.results.description = 'All Jasmine Suites';
|
|
|
|
Jasmine = that;
|
|
return that;
|
|
}
|
|
|
|
var Jasmine = Runner();
|
|
var currentSuite;
|
|
var currentSpec;
|
|
|
|
/*
|
|
* TODO:
|
|
//* - add spec or description to results
|
|
//* - spec.execute needs to wait until the spec is done
|
|
//* - an async test will be killed after X ms if not done and then listed as failed with an "async fail" message of some sort
|
|
//* - Suite to run tests in order, constructed with a function called describe
|
|
* - Suite supports before
|
|
* - Suite supports after
|
|
//* - Suite supports beforeEach
|
|
//* - Suite supports afterEach
|
|
//* - Suite rolls up spec results
|
|
//* - Suite supports asynch
|
|
//* - Runner that runs suites in order
|
|
//* - Runner supports async
|
|
* - HTML reporter
|
|
* - Shows pass/fail progress (just like bootstrap reporter)
|
|
* - Lists a Summary: total # specs, # of passed, # of failed
|
|
* - Failed reports lists all specs that failed and what the failure was
|
|
* - Failed output is styled with red
|
|
* - JSON reporter
|
|
* - Lists full results as a JSON object/string
|
|
*/
|
|
|