8fc713fb13
The logic which was in place to convert array-like objects to arrays prevented the circular reference detection to work properly for arrays and array-like objects.
268 lines
7.8 KiB
JavaScript
268 lines
7.8 KiB
JavaScript
/*global window:false, alert:false, jasmine:false, Node:false, */
|
|
/*jshint curly:false*/
|
|
|
|
'use strict';
|
|
|
|
var phantom = {};
|
|
|
|
if (window._phantom) {
|
|
console.log = function(){
|
|
phantom.sendMessage('verbose',Array.prototype.slice.apply(arguments).join(', '));
|
|
};
|
|
}
|
|
|
|
phantom.sendMessage = function() {
|
|
var args = [].slice.call( arguments );
|
|
var payload = JSON.stringify( args );
|
|
if (window._phantom) {
|
|
// alerts are the communication bridge to grunt
|
|
alert( payload );
|
|
}
|
|
};
|
|
|
|
(function(){
|
|
|
|
function PhantomReporter() {
|
|
this.started = false;
|
|
this.finished = false;
|
|
this.suites_ = [];
|
|
this.results_ = {};
|
|
this.buffer = '';
|
|
}
|
|
|
|
PhantomReporter.prototype.reportRunnerStarting = function(runner) {
|
|
this.started = true;
|
|
|
|
var suites = runner.topLevelSuites();
|
|
for (var i = 0; i < suites.length; i++) {
|
|
var suite = suites[i];
|
|
this.suites_.push(this.summarize_(suite));
|
|
}
|
|
phantom.sendMessage('jasmine.reportRunnerStarting', this.suites_);
|
|
};
|
|
|
|
PhantomReporter.prototype.reportSpecStarting = function(spec) {
|
|
spec.startTime = (new Date()).getTime();
|
|
var message = {
|
|
suite : {
|
|
description : spec.suite.description
|
|
},
|
|
description : spec.description
|
|
};
|
|
phantom.sendMessage('jasmine.reportSpecStarting', message);
|
|
};
|
|
|
|
PhantomReporter.prototype.suites = function() {
|
|
return this.suites_;
|
|
};
|
|
|
|
PhantomReporter.prototype.summarize_ = function(suiteOrSpec) {
|
|
var isSuite = suiteOrSpec instanceof jasmine.Suite;
|
|
var summary = {
|
|
id: suiteOrSpec.id,
|
|
name: suiteOrSpec.description,
|
|
type: isSuite ? 'suite' : 'spec',
|
|
children: []
|
|
};
|
|
|
|
if (isSuite) {
|
|
var children = suiteOrSpec.children();
|
|
for (var i = 0; i < children.length; i++) {
|
|
summary.children.push(this.summarize_(children[i]));
|
|
}
|
|
}
|
|
return summary;
|
|
};
|
|
|
|
PhantomReporter.prototype.results = function() {
|
|
return this.results_;
|
|
};
|
|
|
|
PhantomReporter.prototype.resultsForSpec = function(specId) {
|
|
return this.results_[specId];
|
|
};
|
|
|
|
PhantomReporter.prototype.reportRunnerResults = function(runner) {
|
|
this.finished = true;
|
|
var specIds = runner.specs().map(function(a){return a.id;});
|
|
var summary = this.resultsForSpecs(specIds);
|
|
phantom.sendMessage('jasmine.reportRunnerResults',summary);
|
|
phantom.sendMessage('jasmine.reportJUnitResults', this.generateJUnitSummary(runner));
|
|
phantom.sendMessage('jasmine.done.PhantomReporter');
|
|
};
|
|
|
|
PhantomReporter.prototype.reportSuiteResults = function(suite) {
|
|
if (suite.specs().length) {
|
|
suite.timestamp = new Date();
|
|
suite.duration = suite.timestamp.getTime() - suite.specs()[0].startTime;
|
|
phantom.sendMessage('jasmine.reportSuiteResults',{
|
|
description : suite.description,
|
|
results : suite.results()
|
|
});
|
|
}
|
|
};
|
|
|
|
function stringify(obj) {
|
|
if (typeof obj !== 'object') return obj;
|
|
|
|
var cache = [], keyMap = [], index;
|
|
|
|
var string = JSON.stringify(obj, function(key, value) {
|
|
// Let json stringify falsy values
|
|
if (!value) return value;
|
|
|
|
// If we're a node
|
|
if (value instanceof Node) return '[ Node ]';
|
|
|
|
// If we're a window (logic stolen from jQuery)
|
|
if (value.window && value.window === value.window.window) return '[ Window ]';
|
|
|
|
// Simple function reporting
|
|
if (typeof value === 'function') return '[ Function ]';
|
|
|
|
if (typeof value === 'object' && value !== null) {
|
|
|
|
if (index = cache.indexOf(value) !== -1) {
|
|
// If we have it in cache, report the circle with the key we first found it in
|
|
return '[ Circular {' + (keyMap[index] || 'root') + '} ]';
|
|
}
|
|
cache.push(value);
|
|
keyMap.push(key);
|
|
}
|
|
return value;
|
|
});
|
|
return string;
|
|
}
|
|
|
|
PhantomReporter.prototype.reportSpecResults = function(spec) {
|
|
spec.duration = (new Date()).getTime() - spec.startTime;
|
|
var _results = spec.results();
|
|
var results = {
|
|
description : _results.description,
|
|
messages : _results.getItems(),
|
|
failedCount : _results.failedCount,
|
|
totalCount : _results.totalCount,
|
|
passedCount : _results.passedCount,
|
|
skipped : _results.skipped,
|
|
passed : _results.passed(),
|
|
msg : _results.failedCount > 0 ? "failed" : "passed"
|
|
};
|
|
this.results_[spec.id] = results;
|
|
|
|
// Quick hack to alleviate cyclical object breaking JSONification.
|
|
results.messages.forEach(function(item){
|
|
if (item.expected) {
|
|
item.expected = stringify(item.expected);
|
|
}
|
|
if (item.actual) {
|
|
item.actual = stringify(item.actual);
|
|
}
|
|
});
|
|
|
|
phantom.sendMessage( 'jasmine.reportSpecResults', spec.id, results, this.getFullName(spec));
|
|
};
|
|
|
|
PhantomReporter.prototype.getFullName = function(spec) {
|
|
return getNestedSuiteName(spec.suite, ':: ') + ':: ' + spec.description;
|
|
};
|
|
|
|
PhantomReporter.prototype.resultsForSpecs = function(specIds){
|
|
var results = {};
|
|
for (var i = 0; i < specIds.length; i++) {
|
|
var specId = specIds[i];
|
|
results[specId] = this.summarizeResult_(this.results_[specId]);
|
|
}
|
|
return results;
|
|
};
|
|
|
|
PhantomReporter.prototype.summarizeResult_ = function(result){
|
|
var summaryMessages = [];
|
|
var messagesLength = result.messages.length;
|
|
for (var messageIndex = 0; messageIndex < messagesLength; messageIndex++) {
|
|
var resultMessage = result.messages[messageIndex];
|
|
summaryMessages.push({
|
|
text: resultMessage.type === 'log' ? resultMessage.toString() : jasmine.undefined,
|
|
passed: resultMessage.passed ? resultMessage.passed() : true,
|
|
type: resultMessage.type,
|
|
message: resultMessage.message,
|
|
trace: {
|
|
stack: resultMessage.passed && !resultMessage.passed() ? resultMessage.trace.stack : jasmine.undefined
|
|
}
|
|
});
|
|
}
|
|
|
|
return {
|
|
result : result.result,
|
|
messages : summaryMessages
|
|
};
|
|
};
|
|
|
|
function getNestedSuiteName(suite, sep) {
|
|
var names = [];
|
|
while (suite) {
|
|
names.unshift(suite.description);
|
|
suite = suite.parentSuite;
|
|
}
|
|
return names.join(sep ? sep : ' ');
|
|
}
|
|
|
|
function getTopLevelSuiteId(suite) {
|
|
var id;
|
|
while (suite) {
|
|
id = suite.id;
|
|
suite = suite.parentSuite;
|
|
}
|
|
return id;
|
|
}
|
|
|
|
PhantomReporter.prototype.generateJUnitSummary = function(runner) {
|
|
var consolidatedSuites = {},
|
|
suites = runner.suites().map(function(suite) {
|
|
var failures = 0;
|
|
|
|
var testcases = suite.specs().map(function(spec) {
|
|
var failureMessages = [];
|
|
if (spec.results().failedCount) {
|
|
failures++;
|
|
spec.results().items_.forEach(function(expectation) {
|
|
if (!expectation.passed()) {
|
|
failureMessages.push(expectation.message);
|
|
}
|
|
});
|
|
}
|
|
return {
|
|
assertions: spec.results().items_.length,
|
|
className: getNestedSuiteName(spec.suite),
|
|
name: spec.description,
|
|
time: spec.duration / 1000,
|
|
failureMessages: failureMessages
|
|
};
|
|
});
|
|
|
|
var data = {
|
|
name: getNestedSuiteName(suite),
|
|
time: suite.duration / 1000,
|
|
timestamp: suite.timestamp,
|
|
tests: suite.specs().length,
|
|
errors: 0, // TODO: These exist in the JUnit XML but not sure how they map to jasmine things
|
|
testcases: testcases,
|
|
failures: failures
|
|
};
|
|
|
|
if (suite.parentSuite) {
|
|
consolidatedSuites[getTopLevelSuiteId(suite)].push(data);
|
|
} else {
|
|
consolidatedSuites[suite.id] = [data];
|
|
}
|
|
return data;
|
|
});
|
|
|
|
return {
|
|
suites: suites,
|
|
consolidatedSuites: consolidatedSuites
|
|
};
|
|
};
|
|
|
|
jasmine.getEnv().addReporter( new PhantomReporter() );
|
|
}());
|