// script.aculo.us unittest.js v1.7.0, Fri Jan 19 19:16:36 CET 2007// Copyright (c) 2005, 2006 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)//           (c) 2005, 2006 Jon Tirsen (http://www.tirsen.com)//           (c) 2005, 2006 Michael Schuerig (http://www.schuerig.de/michael/)//// script.aculo.us is freely distributable under the terms of an MIT-style license.// For details, see the script.aculo.us web site: http://script.aculo.us/// experimental, Firefox-onlyEvent.simulateMouse = function(element, eventName) {  var options = Object.extend({    pointerX: 0,    pointerY: 0,    buttons:  0,    ctrlKey:  false,    altKey:   false,    shiftKey: false,    metaKey:  false  }, arguments[2] || {});  var oEvent = document.createEvent("MouseEvents");  oEvent.initMouseEvent(eventName, true, true, document.defaultView,     options.buttons, options.pointerX, options.pointerY, options.pointerX, options.pointerY,     options.ctrlKey, options.altKey, options.shiftKey, options.metaKey, 0, $(element));    if(this.mark) Element.remove(this.mark);  this.mark = document.createElement('div');  this.mark.appendChild(document.createTextNode(" "));  document.body.appendChild(this.mark);  this.mark.style.position = 'absolute';  this.mark.style.top = options.pointerY + "px";  this.mark.style.left = options.pointerX + "px";  this.mark.style.width = "5px";  this.mark.style.height = "5px;";  this.mark.style.borderTop = "1px solid red;"  this.mark.style.borderLeft = "1px solid red;"    if(this.step)    alert('['+new Date().getTime().toString()+'] '+eventName+'/'+Test.Unit.inspect(options));    $(element).dispatchEvent(oEvent);};// Note: Due to a fix in Firefox 1.0.5/6 that probably fixed "too much", this doesn't work in 1.0.6 or DP2.// You need to downgrade to 1.0.4 for now to get this working// See https://bugzilla.mozilla.org/show_bug.cgi?id=289940 for the fix that fixed too muchEvent.simulateKey = function(element, eventName) {  var options = Object.extend({    ctrlKey: false,    altKey: false,    shiftKey: false,    metaKey: false,    keyCode: 0,    charCode: 0  }, arguments[2] || {});  var oEvent = document.createEvent("KeyEvents");  oEvent.initKeyEvent(eventName, true, true, window,     options.ctrlKey, options.altKey, options.shiftKey, options.metaKey,    options.keyCode, options.charCode );  $(element).dispatchEvent(oEvent);};Event.simulateKeys = function(element, command) {  for(var i=0; i<command.length; i++) {    Event.simulateKey(element,'keypress',{charCode:command.charCodeAt(i)});  }};var Test = {}Test.Unit = {};// security exception workaroundTest.Unit.inspect = Object.inspect;Test.Unit.Logger = Class.create();Test.Unit.Logger.prototype = {  initialize: function(log) {    this.log = $(log);
    if (this.log) {      this._createLogTable();    }  },  start: function(testName) {    if (!this.log) return;    this.testName = testName;    this.lastLogLine = document.createElement('tr');    this.statusCell = document.createElement('td');    this.nameCell = document.createElement('td');    this.nameCell.className = "nameCell";    this.nameCell.appendChild(document.createTextNode(testName));    this.messageCell = document.createElement('td');    this.lastLogLine.appendChild(this.statusCell);    this.lastLogLine.appendChild(this.nameCell);    this.lastLogLine.appendChild(this.messageCell);    this.loglines.appendChild(this.lastLogLine);  },  finish: function(status, summary) {    if (!this.log) return;    this.lastLogLine.className = status;    this.statusCell.innerHTML = status;    this.messageCell.innerHTML = this._toHTML(summary);    this.addLinksToResults();  },  message: function(message) {    if (!this.log) return;    this.messageCell.innerHTML = this._toHTML(message);  },  summary: function(summary) {    if (!this.log) return;    this.logsummary.innerHTML = this._toHTML(summary);  },  _createLogTable: function() {    this.log.innerHTML =    '<div id="logsummary"></div>' +    '<table id="logtable">' +    '<thead><tr><th>Status</th><th>Test</th><th>Message</th></tr></thead>' +    '<tbody id="loglines"></tbody>' +    '</table>';    this.logsummary = $('logsummary')    this.loglines = $('loglines');  },  _toHTML: function(txt) {    return txt.escapeHTML().replace(/\n/g,"<br/>");  },  addLinksToResults: function(){     $$("tr.failed .nameCell").each( function(td){ // todo: limit to children of this.log      td.title = "Run only this test"      Event.observe(td, 'click', function(){ window.location.search = "?tests=" + td.innerHTML;});    });    $$("tr.passed .nameCell").each( function(td){ // todo: limit to children of this.log      td.title = "Run all tests"      Event.observe(td, 'click', function(){ window.location.search = "";});    });  }}Test.Unit.Runner = Class.create();Test.Unit.Runner.prototype = {  initialize: function(testcases) {    this.options = Object.extend({      testLog: 'testlog'    }, arguments[1] || {});    this.options.resultsURL = this.parseResultsURLQueryParameter();    this.options.tests      = this.parseTestsQueryParameter();    if (this.options.testLog) {      this.options.testLog = $(this.options.testLog) || null;    }    if(this.options.tests) {      this.tests = [];      for(var i = 0; i < this.options.tests.length; i++) {        if(/^test/.test(this.options.tests[i])) {          this.tests.push(new Test.Unit.Testcase(this.options.tests[i], testcases[this.options.tests[i]], testcases["setup"], testcases["teardown"]));        }      }    } else {      if (this.options.test) {        this.tests = [new Test.Unit.Testcase(this.options.test, testcases[this.options.test], testcases["setup"], testcases["teardown"])];      } else {        this.tests = [];        for(var testcase in testcases) {          if(/^test/.test(testcase)) {            this.tests.push(               new Test.Unit.Testcase(                 this.options.context ? ' -> ' + this.options.titles[testcase] : testcase,                  testcases[testcase], testcases["setup"], testcases["teardown"]               ));          }        }      }    }    this.currentTest = 0;    this.logger = new Test.Unit.Logger(this.options.testLog);    setTimeout(this.runTests.bind(this), 1000);  },  parseResultsURLQueryParameter: function() {    return window.location.search.parseQuery()["resultsURL"];  },  parseTestsQueryParameter: function(){    if (window.location.search.parseQuery()["tests"]){        return window.location.search.parseQuery()["tests"].split(',');    };  },  // Returns:  //  "ERROR" if there was an error,  //  "FAILURE" if there was a failure, or  //  "SUCCESS" if there was neither  getResult: function() {    var hasFailure = false;    for(var i=0;i<this.tests.length;i++) {      if (this.tests[i].errors > 0) {        return "ERROR";      }      if (this.tests[i].failures > 0) {        hasFailure = true;      }    }    if (hasFailure) {
      return "FAILURE";    } else {      return "SUCCESS";    }  },  postResults: function() {    if (this.options.resultsURL) {      new Ajax.Request(this.options.resultsURL,         { method: 'get', parameters: 'result=' + this.getResult(), asynchronous: false });    }  },  runTests: function() {    var test = this.tests[this.currentTest];    if (!test) {      // finished!      this.postResults();      this.logger.summary(this.summary());      return;    }    if(!test.isWaiting) {      this.logger.start(test.name);    }    test.run();    if(test.isWaiting) {      this.logger.message("Waiting for " + test.timeToWait + "ms");      setTimeout(this.runTests.bind(this), test.timeToWait || 1000);    } else {      this.logger.finish(test.status(), test.summary());      this.currentTest++;      // tail recursive, hopefully the browser will skip the stackframe      this.runTests();    }  },  summary: function() {    var assertions = 0;    var failures = 0;    var errors = 0;    var messages = [];    for(var i=0;i<this.tests.length;i++) {      assertions +=   this.tests[i].assertions;      failures   +=   this.tests[i].failures;      errors     +=   this.tests[i].errors;    }    return (      (this.options.context ? this.options.context + ': ': '') +       this.tests.length + " tests, " +       assertions + " assertions, " +       failures   + " failures, " +      errors     + " errors");  }}Test.Unit.Assertions = Class.create();Test.Unit.Assertions.prototype = {  initialize: function() {    this.assertions = 0;    this.failures   = 0;    this.errors     = 0;    this.messages   = [];  },  summary: function() {    return (      this.assertions + " assertions, " +       this.failures   + " failures, " +      this.errors     + " errors" + "\n" +      this.messages.join("\n"));  },  pass: function() {    this.assertions++;  },  fail: function(message) {    this.failures++;    this.messages.push("Failure: " + message);  },  info: function(message) {    this.messages.push("Info: " + message);  },  error: function(error) {    this.errors++;    this.messages.push(error.name + ": "+ error.message + "(" + Test.Unit.inspect(error) +")");  },  status: function() {    if (this.failures > 0) return 'failed';    if (this.errors > 0) return 'error';    return 'passed';  },  assert: function(expression) {    var message = arguments[1] || 'assert: got "' + Test.Unit.inspect(expression) + '"';    try { expression ? this.pass() :       this.fail(message); }    catch(e) { this.error(e); }  },  assertEqual: function(expected, actual) {    var message = arguments[2] || "assertEqual";    try { (expected == actual) ? this.pass() :      this.fail(message + ': expected "' + Test.Unit.inspect(expected) +         '", actual "' + Test.Unit.inspect(actual) + '"'); }    catch(e) { this.error(e); }  },  assertInspect: function(expected, actual) {    var message = arguments[2] || "assertInspect";    try { (expected == actual.inspect()) ? this.pass() :      this.fail(message + ': expected "' + Test.Unit.inspect(expected) +         '", actual "' + Test.Unit.inspect(actual) + '"'); }    catch(e) { this.error(e); }  },  assertEnumEqual: function(expected, actual) {    var message = arguments[2] || "assertEnumEqual";    try { $A(expected).length == $A(actual).length &&       expected.zip(actual).all(function(pair) { return pair[0] == pair[1] }) ?        this.pass() : this.fail(message + ': expected ' + Test.Unit.inspect(expected) +           ', actual ' + Test.Unit.inspect(actual)); }    catch(e) { this.error(e); }  },  assertNotEqual: function(expected, actual) {    var message = arguments[2] || "assertNotEqual";    try { (expected != actual) ? this.pass() :       this.fail(message + ': got "' + Test.Unit.inspect(actual) + '"'); }    catch(e) { this.error(e); }  },  assertIdentical: function(expected, actual) {     var message = arguments[2] || "assertIdentical";     try { (expected === actual) ? this.pass() :       this.fail(message + ': expected "' + Test.Unit.inspect(expected) +          '", actual "' + Test.Unit.inspect(actual) + '"'); }     catch(e) { this.error(e); }   },  assertNotIdentical: function(expected, actual) {     var message = arguments[2] || "assertNotIdentical";     try { !(expected === actual) ? this.pass() :       this.fail(message + ': expected "' + Test.Unit.inspect(expected) +          '", actual "' + Test.Unit.inspect(actual) + '"'); }     catch(e) { this.error(e); }   },  assertNull: function(obj) {    var message = arguments[1] || 'assertNull'    try { (obj==null) ? this.pass() :       this.fail(message + ': got "' + Test.Unit.inspect(obj) + '"'); }    catch(e) { this.error(e); }  },  assertMatch: function(expected, actual) {    var message = arguments[2] || 'assertMatch';    var regex = new RegExp(expected);    try { (regex.exec(actual)) ? this.pass() :      this.fail(message + ' : regex: "' +  Test.Unit.inspect(expected) + ' did not match: ' + Test.Unit.inspect(actual) + '"'); }    catch(e) { this.error(e); }  },  assertHidden: function(element) {    var message = arguments[1] || 'assertHidden';    this.assertEqual("none", element.style.display, message);  },  assertNotNull: function(object) {    var message = arguments[1] || 'assertNotNull';    this.assert(object != null, message);  },  assertType: function(expected, actual) {    var message = arguments[2] || 'assertType';    try {       (actual.constructor == expected) ? this.pass() :       this.fail(message + ': expected "' + Test.Unit.inspect(expected) +          '", actual "' + (actual.constructor) + '"'); }    catch(e) { this.error(e); }  },  assertNotOfType: function(expected, actual) {    var message = arguments[2] || 'assertNotOfType';    try {       (actual.constructor != expected) ? this.pass() :       this.fail(message + ': expected "' + Test.Unit.inspect(expected) +          '", actual "' + (actual.constructor) + '"'); }    catch(e) { this.error(e); }  },  assertInstanceOf: function(expected, actual) {    var message = arguments[2] || 'assertInstanceOf';    try {       (actual instanceof expected) ? this.pass() :       this.fail(message + ": object was not an instance of the expected type"); }    catch(e) { this.error(e); }   },  assertNotInstanceOf: function(expected, actual) {    var message = arguments[2] || 'assertNotInstanceOf';    try {       !(actual instanceof expected) ? this.pass() :       this.fail(message + ": object was an instance of the not expected type"); }    catch(e) { this.error(e); }   },  assertRespondsTo: function(method, obj) {    var message = arguments[2] || 'assertRespondsTo';    try {      (obj[method] && typeof obj[method] == 'function') ? this.pass() :       this.fail(message + ": object doesn't respond to [" + method + "]"); }    catch(e) { this.error(e); }  },  assertReturnsTrue: function(method, obj) {    var message = arguments[2] || 'assertReturnsTrue';    try {      var m = obj[method];      if(!m) m = obj['is'+method.charAt(0).toUpperCase()+method.slice(1)];      m() ? this.pass() :       this.fail(message + ": method returned false"); }    catch(e) { this.error(e); }  },  assertReturnsFalse: function(method, obj) {    var message = arguments[2] || 'assertReturnsFalse';    try {      var m = obj[method];      if(!m) m = obj['is'+method.charAt(0).toUpperCase()+method.slice(1)];      !m() ? this.pass() :       this.fail(message + ": method returned true"); }    catch(e) { this.error(e); }  },  assertRaise: function(exceptionName, method) {    var message = arguments[2] || 'assertRaise';    try {       method();      this.fail(message + ": exception expected but none was raised"); }    catch(e) {      ((exceptionName == null) || (e.name==exceptionName)) ? this.pass() : this.error(e);     }  },  assertElementsMatch: function() {    var expressions = $A(arguments), elements = $A(expressions.shift());    if (elements.length != expressions.length) {      this.fail('assertElementsMatch: size mismatch: ' + elements.length + ' elements, ' + expressions.length + ' expressions');      return false;    }    elements.zip(expressions).all(function(pair, index) {      var element = $(pair.first()), expression = pair.last();      if (element.match(expression)) return true;      this.fail('assertElementsMatch: (in index ' + index + ') expected ' + expression.inspect() + ' but got ' + element.inspect());    }.bind(this)) && this.pass();  },  assertElementMatches: function(element, expression) {    this.assertElementsMatch([element], expression);  },  benchmark: function(operation, iterations) {    var startAt = new Date();    (iterations || 1).times(operation);    var timeTaken = ((new Date())-startAt);    this.info((arguments[2] || 'Operation') + ' finished ' +        iterations + ' iterations in ' + (timeTaken/1000)+'s' );    return timeTaken;  },  _isVisible: function(element) {    element = $(element);    if(!element.parentNode) return true;    this.assertNotNull(element);    if(element.style && Element.getStyle(element, 'display') == 'none')      return false;        return this._isVisible(element.parentNode);  },  assertNotVisible: function(element) {    this.assert(!this._isVisible(element), Test.Unit.inspect(element) + " was not hidden and didn't have a hidden parent either. " + ("" || arguments[1]));  },  assertVisible: function(element) {    this.assert(this._isVisible(element), Test.Unit.inspect(element) + " was not visible. " + ("" || arguments[1]));  },  benchmark: function(operation, iterations) {    var startAt = new Date();    (iterations || 1).times(operation);    var timeTaken = ((new Date())-startAt);    this.info((arguments[2] || 'Operation') + ' finished ' +        iterations + ' iterations in ' + (timeTaken/1000)+'s' );    return timeTaken;  }}Test.Unit.Testcase = Class.create();Object.extend(Object.extend(Test.Unit.Testcase.prototype, Test.Unit.Assertions.prototype), {  initialize: function(name, test, setup, teardown) {    Test.Unit.Assertions.prototype.initialize.bind(this)();    this.name           = name;        if(typeof test == 'string') {      test = test.gsub(/(\.should[^\(]+\()/,'#{0}this,');      test = test.gsub(/(\.should[^\(]+)\(this,\)/,'#{1}(this)');      this.test = function() {        eval('with(this){'+test+'}');      }    } else {      this.test = test || function() {};    }        this.setup          = setup || function() {};    this.teardown       = teardown || function() {};    this.isWaiting      = false;    this.timeToWait     = 1000;  },  wait: function(time, nextPart) {    this.isWaiting = true;    this.test = nextPart;    this.timeToWait = time;  },  run: function() {    try {      try {        if (!this.isWaiting) this.setup.bind(this)();        this.isWaiting = false;        this.test.bind(this)();      } finally {        if(!this.isWaiting) {          this.teardown.bind(this)();        }      }    }    catch(e) { this.error(e); }  }});// *EXPERIMENTAL* BDD-style testing to please non-technical folk// This draws many ideas from RSpec http://rspec.rubyforge.org/Test.setupBDDExtensionMethods = function(){  var METHODMAP = {    shouldEqual:     'assertEqual',    shouldNotEqual:  'assertNotEqual',    shouldEqualEnum: 'assertEnumEqual',    shouldBeA:       'assertType',    shouldNotBeA:    'assertNotOfType',    shouldBeAn:      'assertType',    shouldNotBeAn:   'assertNotOfType',    shouldBeNull:    'assertNull',    shouldNotBeNull: 'assertNotNull',        shouldBe:        'assertReturnsTrue',    shouldNotBe:     'assertReturnsFalse',    shouldRespondTo: 'assertRespondsTo'  };  Test.BDDMethods = {};  for(m in METHODMAP) {    Test.BDDMethods[m] = eval(      'function(){'+      'var args = $A(arguments);'+      'var scope = args.shift();'+      'scope.'+METHODMAP[m]+'.apply(scope,(args || []).concat([this])); }');  }  [Array.prototype, String.prototype, Number.prototype].each(    function(p){ Object.extend(p, Test.BDDMethods) }  );}Test.context = function(name, spec, log){  Test.setupBDDExtensionMethods();    var compiledSpec = {};  var titles = {};  for(specName in spec) {    switch(specName){      case "setup":      case "teardown":        compiledSpec[specName] = spec[specName];        break;      default:        var testName = 'test'+specName.gsub(/\s+/,'-').camelize();        var body = spec[specName].toString().split('\n').slice(1);        if(/^\{/.test(body[0])) body = body.slice(1);        body.pop();        body = body.map(function(statement){           return statement.strip()        });        compiledSpec[testName] = body.join('\n');        titles[testName] = specName;    }  }  new Test.Unit.Runner(compiledSpec, { titles: titles, testLog: log || 'testlog', context: name });};