901 lines
23 KiB
JavaScript
901 lines
23 KiB
JavaScript
|
/*
|
||
|
* Jasmine internal classes & objects
|
||
|
*/
|
||
|
|
||
|
var Jasmine = {};
|
||
|
|
||
|
Jasmine.util = {};
|
||
|
|
||
|
/** @deprecated Use Jasmine.util instead */
|
||
|
Jasmine.Util = Jasmine.util;
|
||
|
|
||
|
Jasmine.util.inherit = function(childClass, parentClass) {
|
||
|
var subclass = function() { };
|
||
|
subclass.prototype = parentClass.prototype;
|
||
|
childClass.prototype = new subclass;
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* Holds results; allows for the results array to hold another Jasmine.NestedResults
|
||
|
*/
|
||
|
Jasmine.NestedResults = function() {
|
||
|
this.totalCount = 0;
|
||
|
this.passedCount = 0;
|
||
|
this.failedCount = 0;
|
||
|
this.results = [];
|
||
|
};
|
||
|
|
||
|
Jasmine.NestedResults.prototype.rollupCounts = function(result) {
|
||
|
this.totalCount += result.totalCount;
|
||
|
this.passedCount += result.passedCount;
|
||
|
this.failedCount += result.failedCount;
|
||
|
};
|
||
|
|
||
|
Jasmine.NestedResults.prototype.push = function(result) {
|
||
|
if (result.results) {
|
||
|
this.rollupCounts(result);
|
||
|
} else {
|
||
|
this.totalCount++;
|
||
|
if (result.passed) {
|
||
|
this.passedCount++;
|
||
|
} else {
|
||
|
this.failedCount++;
|
||
|
}
|
||
|
}
|
||
|
this.results.push(result);
|
||
|
};
|
||
|
|
||
|
Jasmine.NestedResults.prototype.passed = function() {
|
||
|
return this.passedCount === this.totalCount;
|
||
|
};
|
||
|
|
||
|
|
||
|
/*
|
||
|
* 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.
|
||
|
*
|
||
|
**/
|
||
|
Jasmine.ActionCollection = function() {
|
||
|
this.actions = [];
|
||
|
this.index = 0;
|
||
|
this.finished = false;
|
||
|
this.results = new Jasmine.NestedResults();
|
||
|
};
|
||
|
|
||
|
Jasmine.ActionCollection.prototype.finish = function() {
|
||
|
if (this.finishCallback) {
|
||
|
this.finishCallback();
|
||
|
}
|
||
|
this.finished = true;
|
||
|
};
|
||
|
|
||
|
Jasmine.ActionCollection.prototype.report = function(result) {
|
||
|
this.results.push(result);
|
||
|
};
|
||
|
|
||
|
Jasmine.ActionCollection.prototype.execute = function() {
|
||
|
if (this.actions.length > 0) {
|
||
|
this.next();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Jasmine.ActionCollection.prototype.getCurrentAction = function() {
|
||
|
return this.actions[this.index];
|
||
|
};
|
||
|
|
||
|
Jasmine.ActionCollection.prototype.next = function() {
|
||
|
if (this.index >= this.actions.length) {
|
||
|
this.finish();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
var currentAction = this.getCurrentAction();
|
||
|
|
||
|
currentAction.execute(this);
|
||
|
|
||
|
if (currentAction.afterCallbacks) {
|
||
|
for (var i = 0; i < currentAction.afterCallbacks.length; i++) {
|
||
|
try {
|
||
|
currentAction.afterCallbacks[i]();
|
||
|
} catch (e) {
|
||
|
alert(e);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
this.waitForDone(currentAction);
|
||
|
};
|
||
|
|
||
|
Jasmine.ActionCollection.prototype.waitForDone = function(action) {
|
||
|
var self = this;
|
||
|
var afterExecute = function() {
|
||
|
self.report(action.results);
|
||
|
self.index++;
|
||
|
self.next();
|
||
|
};
|
||
|
|
||
|
if (action.finished) {
|
||
|
afterExecute();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
var id = setInterval(function() {
|
||
|
if (action.finished) {
|
||
|
clearInterval(id);
|
||
|
afterExecute();
|
||
|
}
|
||
|
}, 150);
|
||
|
};
|
||
|
|
||
|
Jasmine.safeExecuteBeforeOrAfter = function(spec, func) {
|
||
|
try {
|
||
|
func.apply(spec);
|
||
|
} catch (e) {
|
||
|
var fail = {passed: false, message: func.typeName + '() fail: ' + Jasmine.util.formatException(e)};
|
||
|
spec.results.push(fail);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* QueuedFunction is how ActionCollections' actions are implemented
|
||
|
*/
|
||
|
Jasmine.QueuedFunction = function(func, timeout, latchFunction, spec) {
|
||
|
this.func = func;
|
||
|
this.timeout = timeout;
|
||
|
this.latchFunction = latchFunction;
|
||
|
this.spec = spec;
|
||
|
|
||
|
this.totalTimeSpentWaitingForLatch = 0;
|
||
|
this.latchTimeoutIncrement = 100;
|
||
|
};
|
||
|
|
||
|
Jasmine.QueuedFunction.prototype.next = function() {
|
||
|
this.spec.finish(); // default value is to be done after one function
|
||
|
};
|
||
|
|
||
|
Jasmine.QueuedFunction.prototype.safeExecute = function() {
|
||
|
if (console) {
|
||
|
console.log('>> Jasmine Running ' + this.spec.suite.description + ' ' + this.spec.description + '...');
|
||
|
}
|
||
|
|
||
|
try {
|
||
|
this.func.apply(this.spec);
|
||
|
} catch (e) {
|
||
|
this.fail(e);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Jasmine.QueuedFunction.prototype.execute = function() {
|
||
|
var self = this;
|
||
|
var executeNow = function() {
|
||
|
self.safeExecute();
|
||
|
self.next();
|
||
|
};
|
||
|
|
||
|
var executeLater = function() {
|
||
|
setTimeout(executeNow, self.timeout);
|
||
|
};
|
||
|
|
||
|
var executeNowOrLater = function() {
|
||
|
var latchFunctionResult;
|
||
|
|
||
|
try {
|
||
|
latchFunctionResult = self.latchFunction.apply(self.spec);
|
||
|
} catch (e) {
|
||
|
self.fail(e);
|
||
|
self.next();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (latchFunctionResult) {
|
||
|
executeNow();
|
||
|
} else if (self.totalTimeSpentWaitingForLatch >= self.timeout) {
|
||
|
var message = 'timed out after ' + self.timeout + ' msec waiting for ' + (self.latchFunction.description || 'something to happen');
|
||
|
self.fail({ name: 'timeout', message: message });
|
||
|
self.next();
|
||
|
} else {
|
||
|
self.totalTimeSpentWaitingForLatch += self.latchTimeoutIncrement;
|
||
|
setTimeout(executeNowOrLater, self.latchTimeoutIncrement);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
if (this.latchFunction !== undefined) {
|
||
|
executeNowOrLater();
|
||
|
} else if (this.timeout > 0) {
|
||
|
executeLater();
|
||
|
} else {
|
||
|
executeNow();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Jasmine.QueuedFunction.prototype.fail = function(e) {
|
||
|
this.spec.results.push({passed:false, message: Jasmine.util.formatException(e)});
|
||
|
};
|
||
|
|
||
|
/******************************************************************************
|
||
|
* Jasmine
|
||
|
******************************************************************************/
|
||
|
|
||
|
Jasmine.Env = function() {
|
||
|
this.currentSpec = null;
|
||
|
this.currentSuite = null;
|
||
|
this.currentRunner = null;
|
||
|
};
|
||
|
|
||
|
Jasmine.Env.prototype.execute = function() {
|
||
|
this.currentRunner.execute();
|
||
|
};
|
||
|
|
||
|
Jasmine.currentEnv_ = new Jasmine.Env();
|
||
|
|
||
|
/** @deprecated use Jasmine.getEnv() instead */
|
||
|
var jasmine = Jasmine.currentEnv_;
|
||
|
|
||
|
Jasmine.getEnv = function() {
|
||
|
return Jasmine.currentEnv_;
|
||
|
};
|
||
|
|
||
|
Jasmine.isArray_ = function(value) {
|
||
|
return value &&
|
||
|
typeof value === 'object' &&
|
||
|
typeof value.length === 'number' &&
|
||
|
typeof value.splice === 'function' &&
|
||
|
!(value.propertyIsEnumerable('length'));
|
||
|
};
|
||
|
|
||
|
Jasmine.arrayToString_ = function(array) {
|
||
|
var formatted_value = '';
|
||
|
for (var i = 0; i < array.length; i++) {
|
||
|
if (i > 0) {
|
||
|
formatted_value += ', ';
|
||
|
}
|
||
|
;
|
||
|
formatted_value += Jasmine.pp(array[i]);
|
||
|
}
|
||
|
return '[ ' + formatted_value + ' ]';
|
||
|
};
|
||
|
|
||
|
Jasmine.objectToString_ = function(obj) {
|
||
|
var formatted_value = '';
|
||
|
var first = true;
|
||
|
for (var property in obj) {
|
||
|
if (property == '__Jasmine_pp_has_traversed__') continue;
|
||
|
|
||
|
if (first) {
|
||
|
first = false;
|
||
|
} else {
|
||
|
formatted_value += ', ';
|
||
|
}
|
||
|
formatted_value += property;
|
||
|
formatted_value += ' : ';
|
||
|
formatted_value += Jasmine.pp(obj[property]);
|
||
|
}
|
||
|
|
||
|
return '{ ' + formatted_value + ' }';
|
||
|
};
|
||
|
|
||
|
Jasmine.ppNestLevel_ = 0;
|
||
|
|
||
|
Jasmine.pp = function(value) {
|
||
|
if (Jasmine.ppNestLevel_ > 40) {
|
||
|
// return '(Jasmine.pp nested too deeply!)';
|
||
|
throw new Error('Jasmine.pp nested too deeply!');
|
||
|
}
|
||
|
|
||
|
Jasmine.ppNestLevel_++;
|
||
|
try {
|
||
|
return Jasmine.pp_(value);
|
||
|
} finally {
|
||
|
Jasmine.ppNestLevel_--;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Jasmine.pp_ = function(value) {
|
||
|
if (value === undefined) {
|
||
|
return 'undefined';
|
||
|
}
|
||
|
if (value === null) {
|
||
|
return 'null';
|
||
|
}
|
||
|
|
||
|
if (value.navigator && value.frames && value.setTimeout) {
|
||
|
return '<window>';
|
||
|
}
|
||
|
|
||
|
if (value instanceof Jasmine.Any) return value.toString();
|
||
|
|
||
|
if (typeof value === 'string') {
|
||
|
return "'" + Jasmine.util.htmlEscape(value) + "'";
|
||
|
}
|
||
|
if (typeof value === 'function') {
|
||
|
return 'Function';
|
||
|
}
|
||
|
if (typeof value.nodeType === 'number') {
|
||
|
return 'HTMLNode';
|
||
|
}
|
||
|
|
||
|
if (value.__Jasmine_pp_has_traversed__) {
|
||
|
return '<circular reference: ' + (Jasmine.isArray_(value) ? 'Array' : 'Object') + '>';
|
||
|
}
|
||
|
|
||
|
if (Jasmine.isArray_(value) || typeof value == 'object') {
|
||
|
value.__Jasmine_pp_has_traversed__ = true;
|
||
|
var stringified = Jasmine.isArray_(value) ? Jasmine.arrayToString_(value) : Jasmine.objectToString_(value);
|
||
|
delete value.__Jasmine_pp_has_traversed__;
|
||
|
return stringified;
|
||
|
}
|
||
|
|
||
|
return Jasmine.util.htmlEscape(value.toString());
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* Jasmine.Matchers methods; add your own by extending Jasmine.Matchers.prototype - don't forget to write a test
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
Jasmine.Matchers = function(actual, results) {
|
||
|
this.actual = actual;
|
||
|
this.passing_message = 'Passed.';
|
||
|
this.results = results || new Jasmine.NestedResults();
|
||
|
};
|
||
|
|
||
|
Jasmine.Matchers.prototype.report = function(result, failing_message) {
|
||
|
|
||
|
this.results.push({
|
||
|
passed: result,
|
||
|
message: result ? this.passing_message : failing_message
|
||
|
});
|
||
|
|
||
|
return result;
|
||
|
};
|
||
|
|
||
|
Jasmine.isDomNode = function(obj) {
|
||
|
return obj['nodeType'] > 0;
|
||
|
};
|
||
|
|
||
|
Jasmine.Any = function(expectedClass) {
|
||
|
this.expectedClass = expectedClass;
|
||
|
};
|
||
|
|
||
|
Jasmine.Any.prototype.matches = function(other) {
|
||
|
if (this.expectedClass == String) {
|
||
|
return typeof other == 'string' || other instanceof String;
|
||
|
}
|
||
|
|
||
|
if (this.expectedClass == Number) {
|
||
|
return typeof other == 'number' || other instanceof Number;
|
||
|
}
|
||
|
|
||
|
return other instanceof this.expectedClass;
|
||
|
};
|
||
|
|
||
|
Jasmine.Any.prototype.toString = function() {
|
||
|
return '<Jasmine.any(' + this.expectedClass + ')>';
|
||
|
};
|
||
|
|
||
|
Jasmine.any = function(clazz) {
|
||
|
return new Jasmine.Any(clazz);
|
||
|
};
|
||
|
|
||
|
Jasmine.Matchers.prototype.toEqual = function(expected) {
|
||
|
var mismatchKeys = [];
|
||
|
var mismatchValues = [];
|
||
|
|
||
|
var hasKey = function(obj, keyName) {
|
||
|
return obj!=null && obj[keyName] !== undefined;
|
||
|
};
|
||
|
|
||
|
var equal = function(a, b) {
|
||
|
if (a == undefined || a == null) {
|
||
|
return (a == undefined && b == undefined);
|
||
|
}
|
||
|
|
||
|
if (Jasmine.isDomNode(a) && Jasmine.isDomNode(b)) {
|
||
|
return a === b;
|
||
|
}
|
||
|
|
||
|
if (typeof a === "object" && typeof b === "object") {
|
||
|
for (var property in b) {
|
||
|
if (!hasKey(a, property) && hasKey(b, property)) {
|
||
|
mismatchKeys.push("expected has key '" + property + "', but missing from <b>actual</b>.");
|
||
|
}
|
||
|
}
|
||
|
for (property in a) {
|
||
|
if (!hasKey(b, property) && hasKey(a, property)) {
|
||
|
mismatchKeys.push("<b>expected</b> missing key '" + property + "', but present in actual.");
|
||
|
}
|
||
|
}
|
||
|
for (property in b) {
|
||
|
if (!equal(a[property], b[property])) {
|
||
|
mismatchValues.push("'" + property + "' was<br /><br />'" + (b[property] ? Jasmine.util.htmlEscape(b[property].toString()) : b[property]) + "'<br /><br />in expected, but was<br /><br />'" + (a[property] ? Jasmine.util.htmlEscape(a[property].toString()) : a[property]) + "'<br /><br />in actual.<br />");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
return (mismatchKeys.length == 0 && mismatchValues.length == 0);
|
||
|
}
|
||
|
|
||
|
if (b instanceof Jasmine.Any) {
|
||
|
return b.matches(a);
|
||
|
}
|
||
|
|
||
|
// functions are considered equivalent if their bodies are equal // todo: remove this
|
||
|
if (typeof a === "function" && typeof b === "function") {
|
||
|
return a.toString() == b.toString();
|
||
|
}
|
||
|
|
||
|
//Straight check
|
||
|
return (a === b);
|
||
|
};
|
||
|
|
||
|
var formatMismatches = function(name, array) {
|
||
|
if (array.length == 0) return '';
|
||
|
var errorOutput = '<br /><br />Different ' + name + ':<br />';
|
||
|
for (var i = 0; i < array.length; i++) {
|
||
|
errorOutput += array[i] + '<br />';
|
||
|
}
|
||
|
return errorOutput;
|
||
|
|
||
|
};
|
||
|
|
||
|
return this.report(equal(this.actual, expected),
|
||
|
'Expected<br /><br />' + Jasmine.pp(expected)
|
||
|
+ '<br /><br />but got<br /><br />' + Jasmine.pp(this.actual)
|
||
|
+ '<br />'
|
||
|
+ formatMismatches('Keys', mismatchKeys)
|
||
|
+ formatMismatches('Values', mismatchValues));
|
||
|
};
|
||
|
/** @deprecated */
|
||
|
Jasmine.Matchers.prototype.should_equal = Jasmine.Matchers.prototype.toEqual;
|
||
|
|
||
|
Jasmine.Matchers.prototype.toNotEqual = function(expected) {
|
||
|
return this.report((this.actual !== expected),
|
||
|
'Expected ' + Jasmine.pp(expected) + ' to not equal ' + Jasmine.pp(this.actual) + ', but it does.');
|
||
|
};
|
||
|
/** @deprecated */
|
||
|
Jasmine.Matchers.prototype.should_not_equal = Jasmine.Matchers.prototype.toNotEqual;
|
||
|
|
||
|
Jasmine.Matchers.prototype.toMatch = function(reg_exp) {
|
||
|
return this.report((new RegExp(reg_exp).test(this.actual)),
|
||
|
'Expected ' + this.actual + ' to match ' + reg_exp + '.');
|
||
|
};
|
||
|
/** @deprecated */
|
||
|
Jasmine.Matchers.prototype.should_match = Jasmine.Matchers.prototype.toMatch;
|
||
|
|
||
|
Jasmine.Matchers.prototype.toNotMatch = function(reg_exp) {
|
||
|
return this.report((!new RegExp(reg_exp).test(this.actual)),
|
||
|
'Expected ' + this.actual + ' to not match ' + reg_exp + '.');
|
||
|
};
|
||
|
/** @deprecated */
|
||
|
Jasmine.Matchers.prototype.should_not_match = Jasmine.Matchers.prototype.toNotMatch;
|
||
|
|
||
|
Jasmine.Matchers.prototype.toBeDefined = function() {
|
||
|
return this.report((this.actual !== undefined),
|
||
|
'Expected a value to be defined but it was undefined.');
|
||
|
};
|
||
|
/** @deprecated */
|
||
|
Jasmine.Matchers.prototype.should_be_defined = Jasmine.Matchers.prototype.toBeDefined;
|
||
|
|
||
|
Jasmine.Matchers.prototype.toBeNull = function() {
|
||
|
return this.report((this.actual === null),
|
||
|
'Expected a value to be null but it was not.');
|
||
|
};
|
||
|
/** @deprecated */
|
||
|
Jasmine.Matchers.prototype.should_be_null = Jasmine.Matchers.prototype.toBeNull;
|
||
|
|
||
|
Jasmine.Matchers.prototype.toBeTruthy = function() {
|
||
|
return this.report((this.actual),
|
||
|
'Expected a value to be truthy but it was not.');
|
||
|
};
|
||
|
/** @deprecated */
|
||
|
Jasmine.Matchers.prototype.should_be_truthy = Jasmine.Matchers.prototype.toBeTruthy;
|
||
|
|
||
|
Jasmine.Matchers.prototype.toBeFalsy = function() {
|
||
|
return this.report((!this.actual),
|
||
|
'Expected a value to be falsy but it was not.');
|
||
|
};
|
||
|
/** @deprecated */
|
||
|
Jasmine.Matchers.prototype.should_be_falsy = Jasmine.Matchers.prototype.toBeFalsy;
|
||
|
|
||
|
Jasmine.Matchers.prototype.wasCalled = function() {
|
||
|
if (!this.actual.isSpy) {
|
||
|
return this.report(false, 'Expected value to be a spy, but it was not.');
|
||
|
}
|
||
|
return this.report((this.actual.wasCalled),
|
||
|
'Expected spy to have been called, but it was not.');
|
||
|
};
|
||
|
/** @deprecated */
|
||
|
Jasmine.Matchers.prototype.was_called = Jasmine.Matchers.prototype.wasCalled;
|
||
|
|
||
|
Jasmine.Matchers.prototype.wasNotCalled = function() {
|
||
|
if (!this.actual.isSpy) {
|
||
|
return this.report(false, 'Expected value to be a spy, but it was not.');
|
||
|
}
|
||
|
return this.report((!this.actual.wasCalled),
|
||
|
'Expected spy to not have been called, but it was.');
|
||
|
};
|
||
|
/** @deprecated */
|
||
|
Jasmine.Matchers.prototype.was_not_called = Jasmine.Matchers.prototype.wasNotCalled;
|
||
|
|
||
|
Jasmine.Matchers.prototype.wasCalledWith = function() {
|
||
|
if (!this.wasCalled()) return false;
|
||
|
var argMatcher = new Jasmine.Matchers(this.actual.mostRecentCall.args, this.results);
|
||
|
return argMatcher.toEqual(Jasmine.util.argsToArray(arguments));
|
||
|
};
|
||
|
/** @deprecated */
|
||
|
Jasmine.Matchers.prototype.was_called = Jasmine.Matchers.prototype.wasCalled;
|
||
|
|
||
|
Jasmine.Matchers.prototype.toContain = function(item) {
|
||
|
return this.report((this.actual.indexOf(item) >= 0),
|
||
|
'Expected ' + Jasmine.pp(this.actual) + ' to contain ' + Jasmine.pp(item) + ', but it does not.');
|
||
|
};
|
||
|
|
||
|
Jasmine.Matchers.prototype.toNotContain = function(item) {
|
||
|
return this.report((this.actual.indexOf(item) < 0),
|
||
|
'Expected ' + Jasmine.pp(this.actual) + ' not to contain ' + Jasmine.pp(item) + ', but it does.');
|
||
|
};
|
||
|
|
||
|
Jasmine.createSpy = function() {
|
||
|
var spyObj = function() {
|
||
|
spyObj.wasCalled = true;
|
||
|
spyObj.callCount++;
|
||
|
var args = Jasmine.util.argsToArray(arguments);
|
||
|
spyObj.mostRecentCall = {
|
||
|
object: this,
|
||
|
args: args
|
||
|
};
|
||
|
spyObj.argsForCall.push(args);
|
||
|
return spyObj.plan.apply(this, arguments);
|
||
|
};
|
||
|
|
||
|
spyObj.isSpy = true;
|
||
|
|
||
|
spyObj.plan = function() {
|
||
|
};
|
||
|
|
||
|
spyObj.andCallThrough = function() {
|
||
|
spyObj.plan = spyObj.originalValue;
|
||
|
return spyObj;
|
||
|
};
|
||
|
spyObj.andReturn = function(value) {
|
||
|
spyObj.plan = function() {
|
||
|
return value;
|
||
|
};
|
||
|
return spyObj;
|
||
|
};
|
||
|
spyObj.andThrow = function(exceptionMsg) {
|
||
|
spyObj.plan = function() {
|
||
|
throw exceptionMsg;
|
||
|
};
|
||
|
return spyObj;
|
||
|
};
|
||
|
spyObj.andCallFake = function(fakeFunc) {
|
||
|
spyObj.plan = fakeFunc;
|
||
|
return spyObj;
|
||
|
};
|
||
|
spyObj.reset = function() {
|
||
|
spyObj.wasCalled = false;
|
||
|
spyObj.callCount = 0;
|
||
|
spyObj.argsForCall = [];
|
||
|
spyObj.mostRecentCall = {};
|
||
|
};
|
||
|
spyObj.reset();
|
||
|
|
||
|
return spyObj;
|
||
|
};
|
||
|
|
||
|
Jasmine.spyOn = function(obj, methodName) {
|
||
|
var spec = Jasmine.getEnv().currentSpec;
|
||
|
spec.after(function() {
|
||
|
spec.removeAllSpies();
|
||
|
});
|
||
|
|
||
|
if (obj == undefined) {
|
||
|
throw "spyOn could not find an object to spy upon";
|
||
|
}
|
||
|
|
||
|
if (obj[methodName] === undefined) {
|
||
|
throw methodName + '() method does not exist';
|
||
|
}
|
||
|
|
||
|
if (obj[methodName].isSpy) {
|
||
|
throw new Error(methodName + ' has already been spied upon');
|
||
|
}
|
||
|
|
||
|
var spyObj = Jasmine.createSpy();
|
||
|
|
||
|
spec.spies_.push(spyObj);
|
||
|
spyObj.baseObj = obj;
|
||
|
spyObj.methodName = methodName;
|
||
|
spyObj.originalValue = obj[methodName];
|
||
|
|
||
|
obj[methodName] = spyObj;
|
||
|
|
||
|
return spyObj;
|
||
|
};
|
||
|
|
||
|
var spyOn = Jasmine.spyOn;
|
||
|
|
||
|
/*
|
||
|
* Jasmine spec constructor
|
||
|
*/
|
||
|
|
||
|
Jasmine.Spec = function(description) {
|
||
|
this.suite = null;
|
||
|
this.description = description;
|
||
|
this.queue = [];
|
||
|
this.currentTimeout = 0;
|
||
|
this.currentLatchFunction = undefined;
|
||
|
this.finished = false;
|
||
|
this.afterCallbacks = [];
|
||
|
this.spies_ = [];
|
||
|
|
||
|
this.results = new Jasmine.NestedResults();
|
||
|
};
|
||
|
|
||
|
Jasmine.Spec.prototype.freezeSuite = function(suite) {
|
||
|
this.suite = suite;
|
||
|
};
|
||
|
|
||
|
/** @deprecated */
|
||
|
Jasmine.Spec.prototype.expects_that = function(actual) {
|
||
|
return new Jasmine.Matchers(actual, this.results);
|
||
|
};
|
||
|
|
||
|
Jasmine.Spec.prototype.waits = function(timeout) {
|
||
|
this.currentTimeout = timeout;
|
||
|
this.currentLatchFunction = undefined;
|
||
|
return this;
|
||
|
};
|
||
|
|
||
|
Jasmine.Spec.prototype.waitsFor = function(timeout, latchFunction, message) {
|
||
|
this.currentTimeout = timeout;
|
||
|
this.currentLatchFunction = latchFunction;
|
||
|
this.currentLatchFunction.description = message;
|
||
|
return this;
|
||
|
};
|
||
|
|
||
|
Jasmine.Spec.prototype.resetTimeout = function() {
|
||
|
this.currentTimeout = 0;
|
||
|
this.currentLatchFunction = undefined;
|
||
|
};
|
||
|
|
||
|
Jasmine.Spec.prototype.finishCallback = function() {
|
||
|
if (Jasmine.getEnv().reporter) {
|
||
|
Jasmine.getEnv().reporter.reportSpecResults(this.results);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Jasmine.Spec.prototype.finish = function() {
|
||
|
if (this.suite.afterEach) {
|
||
|
Jasmine.safeExecuteBeforeOrAfter(this, this.suite.afterEach);
|
||
|
}
|
||
|
this.finishCallback();
|
||
|
this.finished = true;
|
||
|
};
|
||
|
|
||
|
Jasmine.Spec.prototype.after = function(doAfter) {
|
||
|
this.afterCallbacks.push(doAfter);
|
||
|
};
|
||
|
|
||
|
Jasmine.Spec.prototype.execute = function() {
|
||
|
Jasmine.getEnv().currentSpec = this;
|
||
|
if (this.suite.beforeEach) {
|
||
|
Jasmine.safeExecuteBeforeOrAfter(this, this.suite.beforeEach);
|
||
|
}
|
||
|
if (this.queue[0]) {
|
||
|
this.queue[0].execute();
|
||
|
} else {
|
||
|
this.finish();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Jasmine.Spec.prototype.explodes = function() {
|
||
|
throw 'explodes function should not have been called';
|
||
|
};
|
||
|
|
||
|
Jasmine.Spec.prototype.spyOn = Jasmine.spyOn;
|
||
|
|
||
|
Jasmine.Spec.prototype.removeAllSpies = function() {
|
||
|
for (var i = 0; i < this.spies_.length; i++) {
|
||
|
var spy = this.spies_[i];
|
||
|
spy.baseObj[spy.methodName] = spy.originalValue;
|
||
|
}
|
||
|
this.spies_ = [];
|
||
|
};
|
||
|
|
||
|
var it = function(description, func) {
|
||
|
var that = new Jasmine.Spec(description);
|
||
|
|
||
|
var addToQueue = function(func) {
|
||
|
var currentFunction = new Jasmine.QueuedFunction(func, that.currentTimeout, that.currentLatchFunction, 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.expectationResults = that.results.results;
|
||
|
that.runs = addToQueue;
|
||
|
that.freezeSuite(Jasmine.getEnv().currentSuite);
|
||
|
|
||
|
Jasmine.getEnv().currentSuite.specs.push(that);
|
||
|
|
||
|
Jasmine.getEnv().currentSpec = that;
|
||
|
|
||
|
if (func) {
|
||
|
addToQueue(func);
|
||
|
}
|
||
|
|
||
|
that.results.description = description;
|
||
|
return that;
|
||
|
};
|
||
|
|
||
|
//this mirrors the spec syntax so you can define a spec description that will not run.
|
||
|
var xit = function() {
|
||
|
return {runs: function() {
|
||
|
} };
|
||
|
};
|
||
|
|
||
|
var expect = function() {
|
||
|
return Jasmine.getEnv().currentSpec.expects_that.apply(Jasmine.getEnv().currentSpec, arguments);
|
||
|
};
|
||
|
|
||
|
var runs = function(func) {
|
||
|
Jasmine.getEnv().currentSpec.runs(func);
|
||
|
};
|
||
|
|
||
|
var waits = function(timeout) {
|
||
|
Jasmine.getEnv().currentSpec.waits(timeout);
|
||
|
};
|
||
|
|
||
|
var waitsFor = function(timeout, latchFunction, message) {
|
||
|
Jasmine.getEnv().currentSpec.waitsFor(timeout, latchFunction, message);
|
||
|
};
|
||
|
|
||
|
var beforeEach = function(beforeEach) {
|
||
|
beforeEach.typeName = 'beforeEach';
|
||
|
Jasmine.getEnv().currentSuite.beforeEach = beforeEach;
|
||
|
};
|
||
|
|
||
|
var afterEach = function(afterEach) {
|
||
|
afterEach.typeName = 'afterEach';
|
||
|
Jasmine.getEnv().currentSuite.afterEach = afterEach;
|
||
|
};
|
||
|
|
||
|
Jasmine.Description = function(description, specDefinitions) {
|
||
|
Jasmine.ActionCollection.call(this);
|
||
|
|
||
|
this.description = description;
|
||
|
this.specs = this.actions;
|
||
|
};
|
||
|
Jasmine.util.inherit(Jasmine.Description, Jasmine.ActionCollection);
|
||
|
|
||
|
var describe = function(description, spec_definitions) {
|
||
|
var that = new Jasmine.Description(description, spec_definitions);
|
||
|
|
||
|
Jasmine.getEnv().currentSuite = that;
|
||
|
Jasmine.getEnv().currentRunner.suites.push(that);
|
||
|
|
||
|
spec_definitions();
|
||
|
|
||
|
that.results.description = description;
|
||
|
that.specResults = that.results.results;
|
||
|
|
||
|
that.finishCallback = function() {
|
||
|
if (Jasmine.getEnv().reporter) {
|
||
|
Jasmine.getEnv().reporter.reportSuiteResults(that.results);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
return that;
|
||
|
};
|
||
|
|
||
|
var xdescribe = function() {
|
||
|
return {execute: function() {
|
||
|
}};
|
||
|
};
|
||
|
|
||
|
Jasmine.Runner = function() {
|
||
|
Jasmine.ActionCollection.call(this);
|
||
|
|
||
|
this.suites = this.actions;
|
||
|
this.results.description = 'All Jasmine Suites';
|
||
|
};
|
||
|
Jasmine.util.inherit(Jasmine.Runner, Jasmine.ActionCollection);
|
||
|
|
||
|
var Runner = function() {
|
||
|
var that = new Jasmine.Runner();
|
||
|
|
||
|
that.finishCallback = function() {
|
||
|
if (Jasmine.getEnv().reporter) {
|
||
|
Jasmine.getEnv().reporter.reportRunnerResults(that.results);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
that.suiteResults = that.results.results;
|
||
|
|
||
|
Jasmine.getEnv().currentRunner = that;
|
||
|
return that;
|
||
|
};
|
||
|
|
||
|
Jasmine.getEnv().currentRunner = Runner();
|
||
|
|
||
|
/* JasmineReporters.reporter
|
||
|
* Base object that will get called whenever a Spec, Suite, or Runner is done. It is up to
|
||
|
* descendants of this object to do something with the results (see json_reporter.js)
|
||
|
*/
|
||
|
Jasmine.Reporters = {};
|
||
|
|
||
|
Jasmine.Reporters.reporter = function(callbacks) {
|
||
|
var that = {
|
||
|
callbacks: callbacks || {},
|
||
|
|
||
|
doCallback: function(callback, results) {
|
||
|
if (callback) {
|
||
|
callback(results);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
reportRunnerResults: function(results) {
|
||
|
that.doCallback(that.callbacks.runnerCallback, results);
|
||
|
},
|
||
|
reportSuiteResults: function(results) {
|
||
|
that.doCallback(that.callbacks.suiteCallback, results);
|
||
|
},
|
||
|
reportSpecResults: function(results) {
|
||
|
that.doCallback(that.callbacks.specCallback, results);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
return that;
|
||
|
};
|
||
|
|
||
|
Jasmine.util.formatException = function(e) {
|
||
|
var lineNumber;
|
||
|
if (e.line) {
|
||
|
lineNumber = e.line;
|
||
|
}
|
||
|
else if (e.lineNumber) {
|
||
|
lineNumber = e.lineNumber;
|
||
|
}
|
||
|
|
||
|
var file;
|
||
|
|
||
|
if (e.sourceURL) {
|
||
|
file = e.sourceURL;
|
||
|
}
|
||
|
else if (e.fileName) {
|
||
|
file = e.fileName;
|
||
|
}
|
||
|
|
||
|
var message = (e.name && e.message) ? (e.name + ': ' + e.message) : e.toString();
|
||
|
|
||
|
if (file && lineNumber) {
|
||
|
message += ' in ' + file + ' (line ' + lineNumber + ')';
|
||
|
}
|
||
|
|
||
|
return message;
|
||
|
};
|
||
|
|
||
|
Jasmine.util.htmlEscape = function(str) {
|
||
|
if (!str) return str;
|
||
|
return str.replace(/&/g, '&')
|
||
|
.replace(/</g, '<')
|
||
|
.replace(/>/g, '>');
|
||
|
};
|
||
|
|
||
|
Jasmine.util.argsToArray = function(args) {
|
||
|
var arrayOfArgs = [];
|
||
|
for (var i = 0; i < args.length; i++) arrayOfArgs.push(args[i]);
|
||
|
return arrayOfArgs;
|
||
|
};
|