angular['scenario'] = angular['scenario'] || (angular['scenario'] = {});
angular.scenario['dsl'] = angular.scenario['dsl'] || (angular.scenario['dsl'] = {});
angular.scenario.Runner = function(scope, jQuery){
var self = scope.$scenario = this;
this.scope = scope;
this.jQuery = jQuery;
this.scope.$testrun = {done: false, results: []};
var specs = this.specs = {};
this.currentSpec = {name: '', futures: []};
var path = [];
this.scope.describe = function(name, body){
path.push(name);
body();
path.pop();
};
var beforeEach = noop;
var afterEach = noop;
this.scope.beforeEach = function(body) {
beforeEach = body;
};
this.scope.afterEach = function(body) {
afterEach = body;
};
this.scope.expect = function(future) {
return new Matcher(self, future, self.logger);
};
this.scope.it = function(name, body) {
var specName = path.join(' ') + ': it ' + name;
self.currentSpec = specs[specName] = {
name: specName,
futures: []
};
try {
beforeEach();
body();
} catch(err) {
self.addFuture(err.message || 'ERROR', function(){
throw err;
});
} finally {
afterEach();
}
self.currentSpec = _null;
};
this.logger = function returnNoop(){
return extend(returnNoop, {close:noop, fail:noop});
};
};
angular.scenario.Runner.prototype = {
run: function(body){
var jQuery = this.jQuery;
body.append(
'
' +
'' +
'' +
'
');
var console = body.find('#runner .console');
console.find('li').live('click', function(){
jQuery(this).toggleClass('collapsed');
});
this.testFrame = body.find('#testView iframe');
function logger(parent) {
var container;
return function(type, text) {
if (!container) {
container = jQuery('');
parent.append(container);
}
var element = jQuery('');
element.find('span').text(text);
container.append(element);
return extend(logger(element), {
close: function(){
element.removeClass('running');
if(!element.hasClass('fail'))
element.addClass('collapsed');
console.scrollTop(console[0].scrollHeight);
},
fail: function(){
element.removeClass('running');
var current = element;
while (current[0] != console[0]) {
if (current.is('li'))
current.addClass('fail');
current = current.parent();
}
}
});
};
}
this.logger = logger(console);
var specNames = [];
foreach(this.specs, function(spec, name){
specNames.push(name);
}, this);
specNames.sort();
var self = this;
function callback(){
var next = specNames.shift();
if(next) {
self.execute(next, callback);
} else {
self.scope.$testrun.done = true;
}
}
callback();
},
addFuture: function(name, behavior) {
var future = new Future(name, behavior);
this.currentSpec.futures.push(future);
return future;
},
execute: function(name, callback) {
var spec = this.specs[name],
self = this,
futuresFulfilled = [],
result = {
passed: false,
failed: false,
finished: false,
fail: function(error) {
result.passed = false;
result.failed = true;
result.error = error;
result.log('fail', isString(error) ? error : toJson(error)).fail();
}
},
specThis = createScope({
result: result,
jQuery: this.jQuery,
testFrame: this.testFrame,
testWindow: this.testWindow
}, angularService, {});
this.self = specThis;
var futureLogger = this.logger('spec', name);
spec.nextFutureIndex = 0;
function done() {
result.finished = true;
futureLogger.close();
self.self = _null;
(callback||noop).call(specThis);
}
function next(value){
if (spec.nextFutureIndex > 0) {
spec.futures[spec.nextFutureIndex - 1].fulfill(value);
}
var future = spec.futures[spec.nextFutureIndex];
(result.log || {close:noop}).close();
result.log = _null;
if (future) {
spec.nextFutureIndex ++;
result.log = futureLogger('future', future.name);
futuresFulfilled.push(future.name);
try {
future.behavior.call(specThis, next);
} catch (e) {
console.error(e);
result.fail(e);
self.scope.$testrun.results.push(
{name: name, passed: false, error: e, steps: futuresFulfilled});
done();
}
} else {
result.passed = !result.failed;
self.scope.$testrun.results.push({
name: name,
passed: !result.failed,
error: result.error,
steps: futuresFulfilled});
done();
}
}
next();
return specThis;
}
};