run-qunit.js 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. var system = require('system');
  2. /**
  3. * Wait until the test condition is true or a timeout occurs. Useful for waiting
  4. * on a server response or for a ui change (fadeIn, etc.) to occur.
  5. *
  6. * @param testFx javascript condition that evaluates to a boolean,
  7. * it can be passed in as a string (e.g.: "1 == 1" or "$('#bar').is(':visible')" or
  8. * as a callback function.
  9. * @param onReady what to do when testFx condition is fulfilled,
  10. * it can be passed in as a string (e.g.: "1 == 1" or "$('#bar').is(':visible')" or
  11. * as a callback function.
  12. * @param timeOutMillis the max amount of time to wait. If not specified, 3 sec is used.
  13. */
  14. function waitFor(testFx, onReady, timeOutMillis) {
  15. var maxtimeOutMillis = timeOutMillis ? timeOutMillis : 10001, //< Default Max Timout is 3s
  16. start = new Date().getTime(),
  17. condition = false,
  18. interval = setInterval(function() {
  19. if ( (new Date().getTime() - start < maxtimeOutMillis) && !condition ) {
  20. // If not time-out yet and condition not yet fulfilled
  21. condition = (typeof(testFx) === "string" ? eval(testFx) : testFx()); //< defensive code
  22. } else {
  23. if(!condition) {
  24. // If condition still not fulfilled (timeout but condition is 'false')
  25. console.log("'waitFor()' timeout");
  26. phantom.exit(1);
  27. } else {
  28. // Condition fulfilled (timeout and/or condition is 'true')
  29. //console.log("'waitFor()' finished in " + (new Date().getTime() - start) + "ms.");
  30. typeof(onReady) === "string" ? eval(onReady) : onReady(); //< Do what it's supposed to do once the condition is fulfilled
  31. clearInterval(interval); //< Stop this interval
  32. }
  33. }
  34. }, 100); //< repeat check every 100ms
  35. };
  36. if (system.args.length !== 2) {
  37. console.log('Usage: run-qunit.js URL');
  38. phantom.exit(1);
  39. }
  40. var fs = require('fs');
  41. var page = require('webpage').create();
  42. // Route "console.log()" calls from within the Page context to the main Phantom context (i.e. current "this")
  43. page.onConsoleMessage = function(msg) {
  44. console.log(msg);
  45. };
  46. page.onError = function (msg, trace) {
  47. console.log(msg);
  48. trace.forEach(function(item) {
  49. console.log(' ', item.file, ':', item.line);
  50. })
  51. }
  52. var _openPath = phantom.args[0].replace(/^.*(\\|\/)/, '');
  53. var openPath = _openPath;
  54. var origdir = '../js/';
  55. var basedir = '../instrumented/';
  56. var coverageBase = fs.read('_coverage.html');
  57. if (fs.exists(basedir)){
  58. var script = /<script.*><\/script>/g,
  59. src = /src=(["'])(.*?)\1/,
  60. contents = fs.read(openPath),
  61. _contents = contents,
  62. srcs = [],
  63. s;
  64. while (script.exec(contents)){
  65. s = src.exec(RegExp.lastMatch)[2];
  66. if (s && s.indexOf(origdir) != -1)
  67. _contents = _contents.replace(s, s.replace(origdir, basedir))
  68. }
  69. if (_contents != contents){
  70. openPath += '.cov.html';
  71. fs.write(openPath, _contents);
  72. }
  73. }
  74. page.open(openPath, function(status){
  75. if (status !== "success") {
  76. console.log("Unable to access network");
  77. phantom.exit(1);
  78. } else {
  79. // Inject instrumented sources if they exist
  80. if (fs.exists(basedir))
  81. for (var i=0; i<srcs.length; i++)
  82. page.includeJs(srcs[i]);
  83. waitFor(function(){
  84. return page.evaluate(function(){
  85. var el = document.getElementById('qunit-testresult');
  86. if (el && el.innerText.match('completed')) {
  87. return true;
  88. }
  89. return false;
  90. });
  91. }, function(){
  92. // output colorized code coverage
  93. // reach into page context and pull out coverage info. stringify to pass context boundaries.
  94. var coverageInfo = JSON.parse(page.evaluate(function() { return JSON.stringify(getCoverageByLine()); }));
  95. if (coverageInfo.key){
  96. var lineCoverage = coverageInfo.lines;
  97. var originalFile = origdir + fs.separator + coverageInfo.key;
  98. var source = coverageInfo.source;
  99. var fileLines = readFileLines(originalFile);
  100. var colorized = '';
  101. for (var idx=0; idx < lineCoverage.length; idx++) {
  102. //+1: coverage lines count from 1.
  103. var cvg = lineCoverage[idx + 1];
  104. var hitmiss = '';
  105. if (typeof cvg === 'number') {
  106. hitmiss = ' ' + (cvg>0 ? 'hit' : 'miss');
  107. } else {
  108. hitmiss = ' ' + 'undef';
  109. }
  110. var htmlLine = fileLines[idx]
  111. if (!source)
  112. htmlLine = htmlLine.replace('<', '&lt;').replace('>', '&gt;');
  113. colorized += '<div class="code' + hitmiss + '">' + htmlLine + '</div>\n';
  114. };
  115. colorized = coverageBase.replace('COLORIZED_LINE_HTML', colorized);
  116. fs.write('coverage.html', colorized, 'w');
  117. console.log('Coverage for ' + coverageInfo.key + ' in coverage.html');
  118. }
  119. if (_openPath != openPath)
  120. fs.remove(openPath);
  121. var failedNum = page.evaluate(function(){
  122. var el = document.getElementById('qunit-testresult');
  123. console.log(el.innerText);
  124. try {
  125. return el.getElementsByClassName('failed')[0].innerHTML;
  126. } catch (e) { }
  127. return 10000;
  128. });
  129. phantom.exit((parseInt(failedNum, 10) > 0) ? 1 : 0);
  130. });
  131. }
  132. });
  133. function readFileLines(filename) {
  134. var stream = fs.open(filename, 'r');
  135. var lines = [];
  136. var line;
  137. while (!stream.atEnd()) {
  138. lines.push(stream.readLine());
  139. }
  140. stream.close();
  141. return lines;
  142. }