removed debug code
[editimage_extension] / lib / unittest.js
1 // Copyright (c) 2005 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
2 //           (c) 2005 Jon Tirsen (http://www.tirsen.com)
3 //           (c) 2005 Michael Schuerig (http://www.schuerig.de/michael/)
4 //
5 // Permission is hereby granted, free of charge, to any person obtaining
6 // a copy of this software and associated documentation files (the
7 // "Software"), to deal in the Software without restriction, including
8 // without limitation the rights to use, copy, modify, merge, publish,
9 // distribute, sublicense, and/or sell copies of the Software, and to
10 // permit persons to whom the Software is furnished to do so, subject to
11 // the following conditions:
12 // 
13 // The above copyright notice and this permission notice shall be
14 // included in all copies or substantial portions of the Software.
15 // 
16 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23
24
25 // experimental, Firefox-only
26 Event.simulateMouse = function(element, eventName) {
27   var options = Object.extend({
28     pointerX: 0,
29     pointerY: 0,
30     buttons: 0
31   }, arguments[2] || {});
32   var oEvent = document.createEvent("MouseEvents");
33   oEvent.initMouseEvent(eventName, true, true, document.defaultView, 
34     options.buttons, options.pointerX, options.pointerY, options.pointerX, options.pointerY, 
35     false, false, false, false, 0, $(element));
36   
37   if(this.mark) Element.remove(this.mark);
38   this.mark = document.createElement('div');
39   this.mark.appendChild(document.createTextNode(" "));
40   document.body.appendChild(this.mark);
41   this.mark.style.position = 'absolute';
42   this.mark.style.top = options.pointerY + "px";
43   this.mark.style.left = options.pointerX + "px";
44   this.mark.style.width = "5px";
45   this.mark.style.height = "5px;";
46   this.mark.style.borderTop = "1px solid red;"
47   this.mark.style.borderLeft = "1px solid red;"
48   
49   if(this.step)
50     alert('['+new Date().getTime().toString()+'] '+eventName+'/'+Test.Unit.inspect(options));
51   
52   $(element).dispatchEvent(oEvent);
53 };
54
55 // 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.
56 // You need to downgrade to 1.0.4 for now to get this working
57 // See https://bugzilla.mozilla.org/show_bug.cgi?id=289940 for the fix that fixed too much
58 Event.simulateKey = function(element, eventName) {
59   var options = Object.extend({
60     ctrlKey: false,
61     altKey: false,
62     shiftKey: false,
63     metaKey: false,
64     keyCode: 0,
65     charCode: 0
66   }, arguments[2] || {});
67
68   var oEvent = document.createEvent("KeyEvents");
69   oEvent.initKeyEvent(eventName, true, true, window, 
70     options.ctrlKey, options.altKey, options.shiftKey, options.metaKey,
71     options.keyCode, options.charCode );
72   $(element).dispatchEvent(oEvent);
73 };
74
75 Event.simulateKeys = function(element, command) {
76   for(var i=0; i<command.length; i++) {
77     Event.simulateKey(element,'keypress',{charCode:command.charCodeAt(i)});
78   }
79 };
80
81 var Test = {}
82 Test.Unit = {};
83
84 // security exception workaround
85 Test.Unit.inspect = Object.inspect;
86
87 Test.Unit.Logger = Class.create();
88 Test.Unit.Logger.prototype = {
89   initialize: function(log) {
90     this.log = $(log);
91     if (this.log) {
92       this._createLogTable();
93     }
94   },
95   start: function(testName) {
96     if (!this.log) return;
97     this.testName = testName;
98     this.lastLogLine = document.createElement('tr');
99     this.statusCell = document.createElement('td');
100     this.nameCell = document.createElement('td');
101     this.nameCell.appendChild(document.createTextNode(testName));
102     this.messageCell = document.createElement('td');
103     this.lastLogLine.appendChild(this.statusCell);
104     this.lastLogLine.appendChild(this.nameCell);
105     this.lastLogLine.appendChild(this.messageCell);
106     this.loglines.appendChild(this.lastLogLine);
107   },
108   finish: function(status, summary) {
109     if (!this.log) return;
110     this.lastLogLine.className = status;
111     this.statusCell.innerHTML = status;
112     this.messageCell.innerHTML = this._toHTML(summary);
113   },
114   message: function(message) {
115     if (!this.log) return;
116     this.messageCell.innerHTML = this._toHTML(message);
117   },
118   summary: function(summary) {
119     if (!this.log) return;
120     this.logsummary.innerHTML = this._toHTML(summary);
121   },
122   _createLogTable: function() {
123     this.log.innerHTML =
124     '<div id="logsummary"></div>' +
125     '<table id="logtable">' +
126     '<thead><tr><th>Status</th><th>Test</th><th>Message</th></tr></thead>' +
127     '<tbody id="loglines"></tbody>' +
128     '</table>';
129     this.logsummary = $('logsummary')
130     this.loglines = $('loglines');
131   },
132   _toHTML: function(txt) {
133     return txt.escapeHTML().replace(/\n/g,"<br/>");
134   }
135 }
136
137 Test.Unit.Runner = Class.create();
138 Test.Unit.Runner.prototype = {
139   initialize: function(testcases) {
140     this.options = Object.extend({
141       testLog: 'testlog'
142     }, arguments[1] || {});
143     this.options.resultsURL = this.parseResultsURLQueryParameter();
144     if (this.options.testLog) {
145       this.options.testLog = $(this.options.testLog) || null;
146     }
147     if(this.options.tests) {
148       this.tests = [];
149       for(var i = 0; i < this.options.tests.length; i++) {
150         if(/^test/.test(this.options.tests[i])) {
151           this.tests.push(new Test.Unit.Testcase(this.options.tests[i], testcases[this.options.tests[i]], testcases["setup"], testcases["teardown"]));
152         }
153       }
154     } else {
155       if (this.options.test) {
156         this.tests = [new Test.Unit.Testcase(this.options.test, testcases[this.options.test], testcases["setup"], testcases["teardown"])];
157       } else {
158         this.tests = [];
159         for(var testcase in testcases) {
160           if(/^test/.test(testcase)) {
161             this.tests.push(new Test.Unit.Testcase(testcase, testcases[testcase], testcases["setup"], testcases["teardown"]));
162           }
163         }
164       }
165     }
166     this.currentTest = 0;
167     this.logger = new Test.Unit.Logger(this.options.testLog);
168     setTimeout(this.runTests.bind(this), 1000);
169   },
170   parseResultsURLQueryParameter: function() {
171     return window.location.search.parseQuery()["resultsURL"];
172   },
173   // Returns:
174   //  "ERROR" if there was an error,
175   //  "FAILURE" if there was a failure, or
176   //  "SUCCESS" if there was neither
177   getResult: function() {
178     var hasFailure = false;
179     for(var i=0;i<this.tests.length;i++) {
180       if (this.tests[i].errors > 0) {
181         return "ERROR";
182       }
183       if (this.tests[i].failures > 0) {
184         hasFailure = true;
185       }
186     }
187     if (hasFailure) {
188       return "FAILURE";
189     } else {
190       return "SUCCESS";
191     }
192   },
193   postResults: function() {
194     if (this.options.resultsURL) {
195       new Ajax.Request(this.options.resultsURL, 
196         { method: 'get', parameters: 'result=' + this.getResult(), asynchronous: false });
197     }
198   },
199   runTests: function() {
200     var test = this.tests[this.currentTest];
201     if (!test) {
202       // finished!
203       this.postResults();
204       this.logger.summary(this.summary());
205       return;
206     }
207     if(!test.isWaiting) {
208       this.logger.start(test.name);
209     }
210     test.run();
211     if(test.isWaiting) {
212       this.logger.message("Waiting for " + test.timeToWait + "ms");
213       setTimeout(this.runTests.bind(this), test.timeToWait || 1000);
214     } else {
215       this.logger.finish(test.status(), test.summary());
216       this.currentTest++;
217       // tail recursive, hopefully the browser will skip the stackframe
218       this.runTests();
219     }
220   },
221   summary: function() {
222     var assertions = 0;
223     var failures = 0;
224     var errors = 0;
225     var messages = [];
226     for(var i=0;i<this.tests.length;i++) {
227       assertions +=   this.tests[i].assertions;
228       failures   +=   this.tests[i].failures;
229       errors     +=   this.tests[i].errors;
230     }
231     return (
232       this.tests.length + " tests, " + 
233       assertions + " assertions, " + 
234       failures   + " failures, " +
235       errors     + " errors");
236   }
237 }
238
239 Test.Unit.Assertions = Class.create();
240 Test.Unit.Assertions.prototype = {
241   initialize: function() {
242     this.assertions = 0;
243     this.failures   = 0;
244     this.errors     = 0;
245     this.messages   = [];
246   },
247   summary: function() {
248     return (
249       this.assertions + " assertions, " + 
250       this.failures   + " failures, " +
251       this.errors     + " errors" + "\n" +
252       this.messages.join("\n"));
253   },
254   pass: function() {
255     this.assertions++;
256   },
257   fail: function(message) {
258     this.failures++;
259     this.messages.push("Failure: " + message);
260   },
261   info: function(message) {
262     this.messages.push("Info: " + message);
263   },
264   error: function(error) {
265     this.errors++;
266     this.messages.push(error.name + ": "+ error.message + "(" + Test.Unit.inspect(error) +")");
267   },
268   status: function() {
269     if (this.failures > 0) return 'failed';
270     if (this.errors > 0) return 'error';
271     return 'passed';
272   },
273   assert: function(expression) {
274     var message = arguments[1] || 'assert: got "' + Test.Unit.inspect(expression) + '"';
275     try { expression ? this.pass() : 
276       this.fail(message); }
277     catch(e) { this.error(e); }
278   },
279   assertEqual: function(expected, actual) {
280     var message = arguments[2] || "assertEqual";
281     try { (expected == actual) ? this.pass() :
282       this.fail(message + ': expected "' + Test.Unit.inspect(expected) + 
283         '", actual "' + Test.Unit.inspect(actual) + '"'); }
284     catch(e) { this.error(e); }
285   },
286   assertEnumEqual: function(expected, actual) {
287     var message = arguments[2] || "assertEnumEqual";
288     try { $A(expected).length == $A(actual).length && 
289       expected.zip(actual).all(function(pair) { return pair[0] == pair[1] }) ?
290         this.pass() : this.fail(message + ': expected ' + Test.Unit.inspect(expected) + 
291           ', actual ' + Test.Unit.inspect(actual)); }
292     catch(e) { this.error(e); }
293   },
294   assertNotEqual: function(expected, actual) {
295     var message = arguments[2] || "assertNotEqual";
296     try { (expected != actual) ? this.pass() : 
297       this.fail(message + ': got "' + Test.Unit.inspect(actual) + '"'); }
298     catch(e) { this.error(e); }
299   },
300   assertNull: function(obj) {
301     var message = arguments[1] || 'assertNull'
302     try { (obj==null) ? this.pass() : 
303       this.fail(message + ': got "' + Test.Unit.inspect(obj) + '"'); }
304     catch(e) { this.error(e); }
305   },
306   assertHidden: function(element) {
307     var message = arguments[1] || 'assertHidden';
308     this.assertEqual("none", element.style.display, message);
309   },
310   assertNotNull: function(object) {
311     var message = arguments[1] || 'assertNotNull';
312     this.assert(object != null, message);
313   },
314   assertInstanceOf: function(expected, actual) {
315     var message = arguments[2] || 'assertInstanceOf';
316     try { 
317       (actual instanceof expected) ? this.pass() : 
318       this.fail(message + ": object was not an instance of the expected type"); }
319     catch(e) { this.error(e); } 
320   },
321   assertNotInstanceOf: function(expected, actual) {
322     var message = arguments[2] || 'assertNotInstanceOf';
323     try { 
324       !(actual instanceof expected) ? this.pass() : 
325       this.fail(message + ": object was an instance of the not expected type"); }
326     catch(e) { this.error(e); } 
327   },
328   _isVisible: function(element) {
329     element = $(element);
330     if(!element.parentNode) return true;
331     this.assertNotNull(element);
332     if(element.style && Element.getStyle(element, 'display') == 'none')
333       return false;
334     
335     return this._isVisible(element.parentNode);
336   },
337   assertNotVisible: function(element) {
338     this.assert(!this._isVisible(element), Test.Unit.inspect(element) + " was not hidden and didn't have a hidden parent either. " + ("" || arguments[1]));
339   },
340   assertVisible: function(element) {
341     this.assert(this._isVisible(element), Test.Unit.inspect(element) + " was not visible. " + ("" || arguments[1]));
342   },
343   benchmark: function(operation, iterations) {
344     var startAt = new Date();
345     (iterations || 1).times(operation);
346     var timeTaken = ((new Date())-startAt);
347     this.info((arguments[2] || 'Operation') + ' finished ' + 
348        iterations + ' iterations in ' + (timeTaken/1000)+'s' );
349     return timeTaken;
350   }
351 }
352
353 Test.Unit.Testcase = Class.create();
354 Object.extend(Object.extend(Test.Unit.Testcase.prototype, Test.Unit.Assertions.prototype), {
355   initialize: function(name, test, setup, teardown) {
356     Test.Unit.Assertions.prototype.initialize.bind(this)();
357     this.name           = name;
358     this.test           = test || function() {};
359     this.setup          = setup || function() {};
360     this.teardown       = teardown || function() {};
361     this.isWaiting      = false;
362     this.timeToWait     = 1000;
363   },
364   wait: function(time, nextPart) {
365     this.isWaiting = true;
366     this.test = nextPart;
367     this.timeToWait = time;
368   },
369   run: function() {
370     try {
371       try {
372         if (!this.isWaiting) this.setup.bind(this)();
373         this.isWaiting = false;
374         this.test.bind(this)();
375       } finally {
376         if(!this.isWaiting) {
377           this.teardown.bind(this)();
378         }
379       }
380     }
381     catch(e) { this.error(e); }
382   }
383 });

Benjamin Mako Hill || Want to submit a patch?