newjs 1.5.0 → 1.5.1

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,36 @@
1
+ require 'erb'
2
+
3
+ class String
4
+ def lines
5
+ split $/
6
+ end
7
+
8
+ def strip_whitespace_at_line_ends
9
+ lines.map {|line| line.gsub(/\s+$/, '')} * $/
10
+ end
11
+ end
12
+
13
+ module Protodoc
14
+ module Environment
15
+ def include(*filenames)
16
+ filenames.map {|filename| Preprocessor.new(filename).to_s}.join("\n")
17
+ end
18
+ end
19
+
20
+ class Preprocessor
21
+ include Environment
22
+
23
+ def initialize(filename)
24
+ @filename = File.expand_path(filename)
25
+ @template = ERB.new(IO.read(@filename), nil, '%')
26
+ end
27
+
28
+ def to_s
29
+ @template.result(binding).strip_whitespace_at_line_ends
30
+ end
31
+ end
32
+ end
33
+
34
+ if __FILE__ == $0
35
+ print Protodoc::Preprocessor.new(ARGV.first)
36
+ end
@@ -0,0 +1 @@
1
+ script/rstakeout "rake test:recent:javascript" test/unit/*_test.html src/*.js
@@ -0,0 +1,97 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ config = File.dirname(__FILE__) + "/../config/javascript_test_autotest.yml"
4
+ unless File.exists?(config)
5
+ puts <<-EOS
6
+ Edit config/javascript_test_autotest.yml for the browser(s) to use for autotesting.
7
+ See config/javascript_test_autotest.yml.sample for examples.
8
+ EOS
9
+ exit
10
+ end
11
+
12
+ ##
13
+ # Originally by Mike Clark.
14
+ #
15
+ # From http://www.pragmaticautomation.com/cgi-bin/pragauto.cgi/Monitor/StakingOutFileChanges.rdoc
16
+ #
17
+ # Runs a user-defined command when files are modified.
18
+ #
19
+ # Like autotest, but more customizable. This is useful when you want to do
20
+ # something other than run tests. For example, generate a PDF book, run
21
+ # a single test, or run a legacy Test::Unit suite in an app that also
22
+ # has an rSpec suite.
23
+ #
24
+ # Can use Ruby's Dir[] to get file glob. Quote your args to take advantage of this.
25
+ #
26
+ # rstakeout 'rake test:recent' **/*.rb
27
+ # => Only watches Ruby files one directory down (no quotes)
28
+ #
29
+ # rstakeout 'rake test:recent' '**/*.rb'
30
+ # => Watches all Ruby files in all directories and subdirectories
31
+ #
32
+ # Modified (with permission) by Geoffrey Grosenbach to call growlnotify for
33
+ # rspec and Test::Unit output.
34
+ #
35
+ # See the PeepCode screencast on rSpec or other blog articles for instructions on
36
+ # setting up growlnotify.
37
+
38
+ def growl(title, msg, img, pri=0, sticky="")
39
+ system "growlnotify -n autotest --image ~/.autotest_images/#{img} -p #{pri} -m #{msg.inspect} #{title} #{sticky}"
40
+ end
41
+
42
+ def self.growl_fail(output)
43
+ growl "FAIL", "#{output}", "fail.png", 2
44
+ end
45
+
46
+ def self.growl_pass(output)
47
+ growl "Pass", "#{output}", "pass.png"
48
+ end
49
+
50
+ command = ARGV.shift
51
+ files = {}
52
+
53
+ ARGV.each do |arg|
54
+ Dir[arg].each { |file|
55
+ files[file] = File.mtime(file)
56
+ }
57
+ end
58
+
59
+ puts "Watching #{files.keys.join(', ')}\n\nFiles: #{files.keys.length}"
60
+
61
+ trap('INT') do
62
+ puts "\nQuitting..."
63
+ exit
64
+ end
65
+
66
+
67
+ loop do
68
+
69
+ sleep 1
70
+
71
+ changed_file, last_changed = files.find { |file, last_changed|
72
+ File.mtime(file) > last_changed
73
+ }
74
+
75
+ if changed_file
76
+ files[changed_file] = File.mtime(changed_file)
77
+ puts "=> #{changed_file} changed, running #{command}"
78
+ results = `#{command}`
79
+ puts results
80
+
81
+ if results.include? 'tests'
82
+ output = results.slice(/(\d+)\s+tests?,\s*(\d+)\s+assertions?,\s*(\d+)\s+failures?(,\s*(\d+)\s+errors)?/)
83
+ if output
84
+ $~[3].to_i + $~[5].to_i > 0 ? growl_fail(output) : growl_pass(output)
85
+ end
86
+ else
87
+ output = results.slice(/(\d+)\s+examples?,\s*(\d+)\s+failures?(,\s*(\d+)\s+not implemented)?/)
88
+ if output
89
+ $~[2].to_i > 0 ? growl_fail(output) : growl_pass(output)
90
+ end
91
+ end
92
+ # TODO Generic growl notification for other actions
93
+
94
+ puts "=> done"
95
+ end
96
+
97
+ end
@@ -0,0 +1 @@
1
+ @ruby <%= File.join("script", filename) %> %*
@@ -0,0 +1 @@
1
+ <%%= include 'some_library_from_src_folder.js' %>
@@ -0,0 +1,46 @@
1
+ TEST_CHANGES_SINCE = Time.now - 600
2
+
3
+ namespace :test do
4
+ namespace :recent do
5
+ desc "Open recently modified files into browser"
6
+ task :javascript do
7
+ require 'rubygems'
8
+ gem 'activesupport'
9
+ require 'active_support'
10
+
11
+ APP_ROOT = File.dirname(__FILE__) + "/../"
12
+ since = TEST_CHANGES_SINCE
13
+ touched = FileList[
14
+ 'test/unit/*_test.html',
15
+ 'src/*.js'].select { |path| File.mtime(path) > since }
16
+ next if touched.blank?
17
+
18
+ gem 'newjs'
19
+ require 'newjs'
20
+ require 'newjs/autotest'
21
+
22
+ touched.each do |file|
23
+ if file =~ /\/([^\/]+)\.js$/
24
+ file = "test/unit/#{$1}_test.html"
25
+ end
26
+ file = "#{APP_ROOT}/#{file}"
27
+ unless File.exists?(file)
28
+ # puts "Notice: Test file does not exist: #{file}"
29
+ next
30
+ end
31
+ puts "Launching test: #{file}"
32
+ browsers = JavascriptTestAutotest::Config.get :browsers
33
+ if browsers.blank?
34
+ puts "WARNING: No browsers setup in config/javascript_test_autotest.yml"
35
+ next
36
+ end
37
+ browsers.each_pair do |name, path|
38
+ browser = JavascriptTestAutotest::Browser.browser(name, path)
39
+ browser.setup
40
+ browser.visit(file)
41
+ browser.teardown
42
+ end
43
+ end
44
+ end
45
+ end
46
+ end
@@ -0,0 +1,976 @@
1
+ /* Jsunittest, version 0.6.3
2
+ * (c) 2008 Dr Nic Williams
3
+ *
4
+ * Jsunittest is freely distributable under
5
+ * the terms of an MIT-style license.
6
+ * For details, see the web site: http://jsunittest.rubyforge.org
7
+ *
8
+ *--------------------------------------------------------------------------*/
9
+
10
+ var JsUnitTest = {
11
+ Unit: {},
12
+ inspect: function(object) {
13
+ try {
14
+ if (typeof object == "undefined") return 'undefined';
15
+ if (object === null) return 'null';
16
+ if (typeof object == "string") {
17
+ var useDoubleQuotes = arguments[1];
18
+ var escapedString = this.gsub(object, /[\x00-\x1f\\]/, function(match) {
19
+ var character = String.specialChar[match[0]];
20
+ return character ? character : '\\u00' + match[0].charCodeAt().toPaddedString(2, 16);
21
+ });
22
+ if (useDoubleQuotes) return '"' + escapedString.replace(/"/g, '\\"') + '"';
23
+ return "'" + escapedString.replace(/'/g, '\\\'') + "'";
24
+ };
25
+ return String(object);
26
+ } catch (e) {
27
+ if (e instanceof RangeError) return '...';
28
+ throw e;
29
+ }
30
+ },
31
+ $: function(element) {
32
+ if (arguments.length > 1) {
33
+ for (var i = 0, elements = [], length = arguments.length; i < length; i++)
34
+ elements.push(this.$(arguments[i]));
35
+ return elements;
36
+ }
37
+ if (typeof element == "string")
38
+ element = document.getElementById(element);
39
+ return element;
40
+ },
41
+ gsub: function(source, pattern, replacement) {
42
+ var result = '', match;
43
+ replacement = arguments.callee.prepareReplacement(replacement);
44
+
45
+ while (source.length > 0) {
46
+ if (match = source.match(pattern)) {
47
+ result += source.slice(0, match.index);
48
+ result += JsUnitTest.String.interpret(replacement(match));
49
+ source = source.slice(match.index + match[0].length);
50
+ } else {
51
+ result += source, source = '';
52
+ }
53
+ }
54
+ return result;
55
+ },
56
+ scan: function(source, pattern, iterator) {
57
+ this.gsub(source, pattern, iterator);
58
+ return String(source);
59
+ },
60
+ escapeHTML: function(data) {
61
+ return data.replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;');
62
+ },
63
+ arrayfromargs: function(args) {
64
+ var myarray = new Array();
65
+ var i;
66
+
67
+ for (i=0;i<args.length;i++)
68
+ myarray[i] = args[i];
69
+
70
+ return myarray;
71
+ },
72
+ hashToSortedArray: function(hash) {
73
+ var results = [];
74
+ for (key in hash) {
75
+ results.push([key, hash[key]]);
76
+ }
77
+ return results.sort();
78
+ },
79
+ flattenArray: function(array) {
80
+ var results = arguments[1] || [];
81
+ for (var i=0; i < array.length; i++) {
82
+ var object = array[i];
83
+ if (object != null && typeof object == "object" &&
84
+ 'splice' in object && 'join' in object) {
85
+ this.flattenArray(object, results);
86
+ } else {
87
+ results.push(object);
88
+ }
89
+ };
90
+ return results;
91
+ },
92
+ selectorMatch: function(expression, element) {
93
+ var tokens = [];
94
+ var patterns = {
95
+ // combinators must be listed first
96
+ // (and descendant needs to be last combinator)
97
+ laterSibling: /^\s*~\s*/,
98
+ child: /^\s*>\s*/,
99
+ adjacent: /^\s*\+\s*/,
100
+ descendant: /^\s/,
101
+
102
+ // selectors follow
103
+ tagName: /^\s*(\*|[\w\-]+)(\b|$)?/,
104
+ id: /^#([\w\-\*]+)(\b|$)/,
105
+ className: /^\.([\w\-\*]+)(\b|$)/,
106
+ pseudo:
107
+ /^:((first|last|nth|nth-last|only)(-child|-of-type)|empty|checked|(en|dis)abled|not)(\((.*?)\))?(\b|$|(?=\s|[:+~>]))/,
108
+ attrPresence: /^\[((?:[\w]+:)?[\w]+)\]/,
109
+ attr: /\[((?:[\w-]*:)?[\w-]+)\s*(?:([!^$*~|]?=)\s*((['"])([^\4]*?)\4|([^'"][^\]]*?)))?\]/
110
+ };
111
+
112
+ var assertions = {
113
+ tagName: function(element, matches) {
114
+ return matches[1].toUpperCase() == element.tagName.toUpperCase();
115
+ },
116
+
117
+ className: function(element, matches) {
118
+ return Element.hasClassName(element, matches[1]);
119
+ },
120
+
121
+ id: function(element, matches) {
122
+ return element.id === matches[1];
123
+ },
124
+
125
+ attrPresence: function(element, matches) {
126
+ return Element.hasAttribute(element, matches[1]);
127
+ },
128
+
129
+ attr: function(element, matches) {
130
+ var nodeValue = Element.readAttribute(element, matches[1]);
131
+ return nodeValue && operators[matches[2]](nodeValue, matches[5] || matches[6]);
132
+ }
133
+ };
134
+ var e = this.expression, ps = patterns, as = assertions;
135
+ var le, p, m;
136
+
137
+ while (e && le !== e && (/\S/).test(e)) {
138
+ le = e;
139
+ for (var i in ps) {
140
+ p = ps[i];
141
+ if (m = e.match(p)) {
142
+ // use the Selector.assertions methods unless the selector
143
+ // is too complex.
144
+ if (as[i]) {
145
+ tokens.push([i, Object.clone(m)]);
146
+ e = e.replace(m[0], '');
147
+ }
148
+ }
149
+ }
150
+ }
151
+
152
+ var match = true, name, matches;
153
+ for (var i = 0, token; token = tokens[i]; i++) {
154
+ name = token[0], matches = token[1];
155
+ if (!assertions[name](element, matches)) {
156
+ match = false; break;
157
+ }
158
+ }
159
+
160
+ return match;
161
+ },
162
+ toQueryParams: function(query, separator) {
163
+ var query = query || window.location.search;
164
+ var match = query.replace(/^\s+/, '').replace(/\s+$/, '').match(/([^?#]*)(#.*)?$/);
165
+ if (!match) return { };
166
+
167
+ var hash = {};
168
+ var parts = match[1].split(separator || '&');
169
+ for (var i=0; i < parts.length; i++) {
170
+ var pair = parts[i].split('=');
171
+ if (pair[0]) {
172
+ var key = decodeURIComponent(pair.shift());
173
+ var value = pair.length > 1 ? pair.join('=') : pair[0];
174
+ if (value != undefined) value = decodeURIComponent(value);
175
+
176
+ if (key in hash) {
177
+ var object = hash[key];
178
+ var isArray = object != null && typeof object == "object" &&
179
+ 'splice' in object && 'join' in object
180
+ if (!isArray) hash[key] = [hash[key]];
181
+ hash[key].push(value);
182
+ }
183
+ else hash[key] = value;
184
+ }
185
+ };
186
+ return hash;
187
+ },
188
+
189
+ String: {
190
+ interpret: function(value) {
191
+ return value == null ? '' : String(value);
192
+ }
193
+ }
194
+ };
195
+
196
+ JsUnitTest.gsub.prepareReplacement = function(replacement) {
197
+ if (typeof replacement == "function") return replacement;
198
+ var template = new Template(replacement);
199
+ return function(match) { return template.evaluate(match) };
200
+ };
201
+
202
+ JsUnitTest.Version = '0.6.3';
203
+
204
+ JsUnitTest.Template = function(template, pattern) {
205
+ this.template = template; //template.toString();
206
+ this.pattern = pattern || JsUnitTest.Template.Pattern;
207
+ };
208
+
209
+ JsUnitTest.Template.prototype.evaluate = function(object) {
210
+ if (typeof object.toTemplateReplacements == "function")
211
+ object = object.toTemplateReplacements();
212
+
213
+ return JsUnitTest.gsub(this.template, this.pattern, function(match) {
214
+ if (object == null) return '';
215
+
216
+ var before = match[1] || '';
217
+ if (before == '\\') return match[2];
218
+
219
+ var ctx = object, expr = match[3];
220
+ var pattern = /^([^.[]+|\[((?:.*?[^\\])?)\])(\.|\[|$)/;
221
+ match = pattern.exec(expr);
222
+ if (match == null) return before;
223
+
224
+ while (match != null) {
225
+ var comp = (match[1].indexOf('[]') === 0) ? match[2].gsub('\\\\]', ']') : match[1];
226
+ ctx = ctx[comp];
227
+ if (null == ctx || '' == match[3]) break;
228
+ expr = expr.substring('[' == match[3] ? match[1].length : match[0].length);
229
+ match = pattern.exec(expr);
230
+ }
231
+
232
+ return before + JsUnitTest.String.interpret(ctx);
233
+ });
234
+ }
235
+
236
+ JsUnitTest.Template.Pattern = /(^|.|\r|\n)(#\{(.*?)\})/;
237
+ JsUnitTest.Event = {};
238
+ // written by Dean Edwards, 2005
239
+ // with input from Tino Zijdel, Matthias Miller, Diego Perini
240
+ // namespaced by Dr Nic Williams 2008
241
+
242
+ // http://dean.edwards.name/weblog/2005/10/add-event/
243
+ // http://dean.edwards.name/weblog/2005/10/add-event2/
244
+ JsUnitTest.Event.addEvent = function(element, type, handler) {
245
+ if (element.addEventListener) {
246
+ element.addEventListener(type, handler, false);
247
+ } else {
248
+ // assign each event handler a unique ID
249
+ if (!handler.$$guid) handler.$$guid = JsUnitTest.Event.addEvent.guid++;
250
+ // create a hash table of event types for the element
251
+ if (!element.events) element.events = {};
252
+ // create a hash table of event handlers for each element/event pair
253
+ var handlers = element.events[type];
254
+ if (!handlers) {
255
+ handlers = element.events[type] = {};
256
+ // store the existing event handler (if there is one)
257
+ if (element["on" + type]) {
258
+ handlers[0] = element["on" + type];
259
+ }
260
+ }
261
+ // store the event handler in the hash table
262
+ handlers[handler.$$guid] = handler;
263
+ // assign a global event handler to do all the work
264
+ element["on" + type] = this.handleEvent;
265
+ }
266
+ };
267
+ // a counter used to create unique IDs
268
+ JsUnitTest.Event.addEvent.guid = 1;
269
+
270
+ JsUnitTest.Event.removeEvent = function(element, type, handler) {
271
+ if (element.removeEventListener) {
272
+ element.removeEventListener(type, handler, false);
273
+ } else {
274
+ // delete the event handler from the hash table
275
+ if (element.events && element.events[type]) {
276
+ delete element.events[type][handler.$$guid];
277
+ }
278
+ }
279
+ };
280
+
281
+ JsUnitTest.Event.handleEvent = function(event) {
282
+ var returnValue = true;
283
+ // grab the event object (IE uses a global event object)
284
+ event = event || JsUnitTest.Event.fixEvent(((this.ownerDocument || this.document || this).parentWindow || window).event);
285
+ // get a reference to the hash table of event handlers
286
+ var handlers = this.events[event.type];
287
+ // execute each event handler
288
+ for (var i in handlers) {
289
+ this.$$handleEvent = handlers[i];
290
+ if (this.$$handleEvent(event) === false) {
291
+ returnValue = false;
292
+ }
293
+ }
294
+ return returnValue;
295
+ };
296
+
297
+ JsUnitTest.Event.fixEvent = function(event) {
298
+ // add W3C standard event methods
299
+ event.preventDefault = this.fixEvent.preventDefault;
300
+ event.stopPropagation = this.fixEvent.stopPropagation;
301
+ return event;
302
+ };
303
+ JsUnitTest.Event.fixEvent.preventDefault = function() {
304
+ this.returnValue = false;
305
+ };
306
+ JsUnitTest.Event.fixEvent.stopPropagation = function() {
307
+ this.cancelBubble = true;
308
+ };
309
+
310
+ JsUnitTest.Unit.Logger = function(element) {
311
+ this.element = JsUnitTest.$(element);
312
+ if (this.element) this._createLogTable();
313
+ };
314
+
315
+ JsUnitTest.Unit.Logger.prototype.start = function(testName) {
316
+ if (!this.element) return;
317
+ var tbody = this.element.getElementsByTagName('tbody')[0];
318
+
319
+ var tr = document.createElement('tr');
320
+ var td;
321
+
322
+ //testname
323
+ td = document.createElement('td');
324
+ td.appendChild(document.createTextNode(testName));
325
+ tr.appendChild(td)
326
+
327
+ tr.appendChild(document.createElement('td'));//status
328
+ tr.appendChild(document.createElement('td'));//message
329
+
330
+ tbody.appendChild(tr);
331
+ };
332
+
333
+ JsUnitTest.Unit.Logger.prototype.setStatus = function(status) {
334
+ var logline = this.getLastLogLine();
335
+ logline.className = status;
336
+ var statusCell = logline.getElementsByTagName('td')[1];
337
+ statusCell.appendChild(document.createTextNode(status));
338
+ };
339
+
340
+ JsUnitTest.Unit.Logger.prototype.finish = function(status, summary) {
341
+ if (!this.element) return;
342
+ this.setStatus(status);
343
+ this.message(summary);
344
+ };
345
+
346
+ JsUnitTest.Unit.Logger.prototype.message = function(message) {
347
+ if (!this.element) return;
348
+ var cell = this.getMessageCell();
349
+
350
+ // cell.appendChild(document.createTextNode(this._toHTML(message)));
351
+ cell.innerHTML = this._toHTML(message);
352
+ };
353
+
354
+ JsUnitTest.Unit.Logger.prototype.summary = function(summary) {
355
+ if (!this.element) return;
356
+ var div = this.element.getElementsByTagName('div')[0];
357
+ div.innerHTML = this._toHTML(summary);
358
+ };
359
+
360
+ JsUnitTest.Unit.Logger.prototype.getLastLogLine = function() {
361
+ var tbody = this.element.getElementsByTagName('tbody')[0];
362
+ var loglines = tbody.getElementsByTagName('tr');
363
+ return loglines[loglines.length - 1];
364
+ };
365
+
366
+ JsUnitTest.Unit.Logger.prototype.getMessageCell = function() {
367
+ var logline = this.getLastLogLine();
368
+ return logline.getElementsByTagName('td')[2];
369
+ };
370
+
371
+ JsUnitTest.Unit.Logger.prototype._createLogTable = function() {
372
+ var html = '<div class="logsummary">running...</div>' +
373
+ '<table class="logtable">' +
374
+ '<thead><tr><th>Status</th><th>Test</th><th>Message</th></tr></thead>' +
375
+ '<tbody class="loglines"></tbody>' +
376
+ '</table>';
377
+ this.element.innerHTML = html;
378
+ };
379
+
380
+ JsUnitTest.Unit.Logger.prototype.appendActionButtons = function(actions) {
381
+ // actions = $H(actions);
382
+ // if (!actions.any()) return;
383
+ // var div = new Element("div", {className: 'action_buttons'});
384
+ // actions.inject(div, function(container, action) {
385
+ // var button = new Element("input").setValue(action.key).observe("click", action.value);
386
+ // button.type = "button";
387
+ // return container.insert(button);
388
+ // });
389
+ // this.getMessageCell().insert(div);
390
+ };
391
+
392
+ JsUnitTest.Unit.Logger.prototype._toHTML = function(txt) {
393
+ return JsUnitTest.escapeHTML(txt).replace(/\n/g,"<br/>");
394
+ };
395
+ JsUnitTest.Unit.MessageTemplate = function(string) {
396
+ var parts = [];
397
+ var str = JsUnitTest.scan((string || ''), /(?=[^\\])\?|(?:\\\?|[^\?])+/, function(part) {
398
+ parts.push(part[0]);
399
+ });
400
+ this.parts = parts;
401
+ };
402
+
403
+ JsUnitTest.Unit.MessageTemplate.prototype.evaluate = function(params) {
404
+ var results = [];
405
+ for (var i=0; i < this.parts.length; i++) {
406
+ var part = this.parts[i];
407
+ var result = (part == '?') ? JsUnitTest.inspect(params.shift()) : part.replace(/\\\?/, '?');
408
+ results.push(result);
409
+ };
410
+ return results.join('');
411
+ };
412
+ // A generic function for performming AJAX requests
413
+ // It takes one argument, which is an object that contains a set of options
414
+ // All of which are outline in the comments, below
415
+ // From John Resig's book Pro JavaScript Techniques
416
+ // published by Apress, 2006-8
417
+ JsUnitTest.ajax = function( options ) {
418
+
419
+ // Load the options object with defaults, if no
420
+ // values were provided by the user
421
+ options = {
422
+ // The type of HTTP Request
423
+ type: options.type || "POST",
424
+
425
+ // The URL the request will be made to
426
+ url: options.url || "",
427
+
428
+ // How long to wait before considering the request to be a timeout
429
+ timeout: options.timeout || 5000,
430
+
431
+ // Functions to call when the request fails, succeeds,
432
+ // or completes (either fail or succeed)
433
+ onComplete: options.onComplete || function(){},
434
+ onError: options.onError || function(){},
435
+ onSuccess: options.onSuccess || function(){},
436
+
437
+ // The data type that'll be returned from the server
438
+ // the default is simply to determine what data was returned from the
439
+ // and act accordingly.
440
+ data: options.data || ""
441
+ };
442
+
443
+ // Create the request object
444
+ var xml = new XMLHttpRequest();
445
+
446
+ // Open the asynchronous POST request
447
+ xml.open(options.type, options.url, true);
448
+
449
+ // We're going to wait for a request for 5 seconds, before giving up
450
+ var timeoutLength = 5000;
451
+
452
+ // Keep track of when the request has been succesfully completed
453
+ var requestDone = false;
454
+
455
+ // Initalize a callback which will fire 5 seconds from now, cancelling
456
+ // the request (if it has not already occurred).
457
+ setTimeout(function(){
458
+ requestDone = true;
459
+ }, timeoutLength);
460
+
461
+ // Watch for when the state of the document gets updated
462
+ xml.onreadystatechange = function(){
463
+ // Wait until the data is fully loaded,
464
+ // and make sure that the request hasn't already timed out
465
+ if ( xml.readyState == 4 && !requestDone ) {
466
+
467
+ // Check to see if the request was successful
468
+ if ( httpSuccess( xml ) ) {
469
+
470
+ // Execute the success callback with the data returned from the server
471
+ options.onSuccess( httpData( xml, options.type ) );
472
+
473
+ // Otherwise, an error occurred, so execute the error callback
474
+ } else {
475
+ options.onError();
476
+ }
477
+
478
+ // Call the completion callback
479
+ options.onComplete();
480
+
481
+ // Clean up after ourselves, to avoid memory leaks
482
+ xml = null;
483
+ }
484
+ };
485
+
486
+ // Establish the connection to the server
487
+ xml.send(null);
488
+
489
+ // Determine the success of the HTTP response
490
+ function httpSuccess(r) {
491
+ try {
492
+ // If no server status is provided, and we're actually
493
+ // requesting a local file, then it was successful
494
+ return !r.status && location.protocol == "file:" ||
495
+
496
+ // Any status in the 200 range is good
497
+ ( r.status >= 200 && r.status < 300 ) ||
498
+
499
+ // Successful if the document has not been modified
500
+ r.status == 304 ||
501
+
502
+ // Safari returns an empty status if the file has not been modified
503
+ navigator.userAgent.indexOf("Safari") >= 0 && typeof r.status == "undefined";
504
+ } catch(e){}
505
+
506
+ // If checking the status failed, then assume that the request failed too
507
+ return false;
508
+ }
509
+
510
+ // Extract the correct data from the HTTP response
511
+ function httpData(r,type) {
512
+ // Get the content-type header
513
+ var ct = r.getResponseHeader("content-type");
514
+
515
+ // If no default type was provided, determine if some
516
+ // form of XML was returned from the server
517
+ var data = !type && ct && ct.indexOf("xml") >= 0;
518
+
519
+ // Get the XML Document object if XML was returned from
520
+ // the server, otherwise return the text contents returned by the server
521
+ data = type == "xml" || data ? r.responseXML : r.responseText;
522
+
523
+ // If the specified type is "script", execute the returned text
524
+ // response as if it was JavaScript
525
+ if ( type == "script" )
526
+ eval.call( window, data );
527
+
528
+ // Return the response data (either an XML Document or a text string)
529
+ return data;
530
+ }
531
+
532
+ }
533
+ JsUnitTest.Unit.Assertions = {
534
+ buildMessage: function(message, template) {
535
+ var args = JsUnitTest.arrayfromargs(arguments).slice(2);
536
+ return (message ? message + '\n' : '') +
537
+ new JsUnitTest.Unit.MessageTemplate(template).evaluate(args);
538
+ },
539
+
540
+ flunk: function(message) {
541
+ this.assertBlock(message || 'Flunked', function() { return false });
542
+ },
543
+
544
+ assertBlock: function(message, block) {
545
+ try {
546
+ block.call(this) ? this.pass() : this.fail(message);
547
+ } catch(e) { this.error(e) }
548
+ },
549
+
550
+ assert: function(expression, message) {
551
+ message = this.buildMessage(message || 'assert', 'got <?>', expression);
552
+ this.assertBlock(message, function() { return expression });
553
+ },
554
+
555
+ assertEqual: function(expected, actual, message) {
556
+ message = this.buildMessage(message || 'assertEqual', 'expected <?>, actual: <?>', expected, actual);
557
+ this.assertBlock(message, function() { return expected == actual });
558
+ },
559
+
560
+ assertNotEqual: function(expected, actual, message) {
561
+ message = this.buildMessage(message || 'assertNotEqual', 'expected <?>, actual: <?>', expected, actual);
562
+ this.assertBlock(message, function() { return expected != actual });
563
+ },
564
+
565
+ assertEnumEqual: function(expected, actual, message) {
566
+ message = this.buildMessage(message || 'assertEnumEqual', 'expected <?>, actual: <?>', expected, actual);
567
+ var expected_array = JsUnitTest.flattenArray(expected);
568
+ var actual_array = JsUnitTest.flattenArray(actual);
569
+ this.assertBlock(message, function() {
570
+ if (expected_array.length == actual_array.length) {
571
+ for (var i=0; i < expected_array.length; i++) {
572
+ if (expected_array[i] != actual_array[i]) return false;
573
+ };
574
+ return true;
575
+ }
576
+ return false;
577
+ });
578
+ },
579
+
580
+ assertEnumNotEqual: function(expected, actual, message) {
581
+ message = this.buildMessage(message || 'assertEnumNotEqual', '<?> was the same as <?>', expected, actual);
582
+ var expected_array = JsUnitTest.flattenArray(expected);
583
+ var actual_array = JsUnitTest.flattenArray(actual);
584
+ this.assertBlock(message, function() {
585
+ if (expected_array.length == actual_array.length) {
586
+ for (var i=0; i < expected_array.length; i++) {
587
+ if (expected_array[i] != actual_array[i]) return true;
588
+ };
589
+ return false;
590
+ }
591
+ return true;
592
+ });
593
+ },
594
+
595
+ assertHashEqual: function(expected, actual, message) {
596
+ message = this.buildMessage(message || 'assertHashEqual', 'expected <?>, actual: <?>', expected, actual);
597
+ var expected_array = JsUnitTest.flattenArray(JsUnitTest.hashToSortedArray(expected));
598
+ var actual_array = JsUnitTest.flattenArray(JsUnitTest.hashToSortedArray(actual));
599
+ var block = function() {
600
+ if (expected_array.length == actual_array.length) {
601
+ for (var i=0; i < expected_array.length; i++) {
602
+ if (expected_array[i] != actual_array[i]) return false;
603
+ };
604
+ return true;
605
+ }
606
+ return false;
607
+ };
608
+ this.assertBlock(message, block);
609
+ },
610
+
611
+ assertHashNotEqual: function(expected, actual, message) {
612
+ message = this.buildMessage(message || 'assertHashNotEqual', '<?> was the same as <?>', expected, actual);
613
+ var expected_array = JsUnitTest.flattenArray(JsUnitTest.hashToSortedArray(expected));
614
+ var actual_array = JsUnitTest.flattenArray(JsUnitTest.hashToSortedArray(actual));
615
+ // from now we recursively zip & compare nested arrays
616
+ var block = function() {
617
+ if (expected_array.length == actual_array.length) {
618
+ for (var i=0; i < expected_array.length; i++) {
619
+ if (expected_array[i] != actual_array[i]) return true;
620
+ };
621
+ return false;
622
+ }
623
+ return true;
624
+ };
625
+ this.assertBlock(message, block);
626
+ },
627
+
628
+ assertIdentical: function(expected, actual, message) {
629
+ message = this.buildMessage(message || 'assertIdentical', 'expected <?>, actual: <?>', expected, actual);
630
+ this.assertBlock(message, function() { return expected === actual });
631
+ },
632
+
633
+ assertNotIdentical: function(expected, actual, message) {
634
+ message = this.buildMessage(message || 'assertNotIdentical', 'expected <?>, actual: <?>', expected, actual);
635
+ this.assertBlock(message, function() { return expected !== actual });
636
+ },
637
+
638
+ assertNull: function(obj, message) {
639
+ message = this.buildMessage(message || 'assertNull', 'got <?>', obj);
640
+ this.assertBlock(message, function() { return obj === null });
641
+ },
642
+
643
+ assertNotNull: function(obj, message) {
644
+ message = this.buildMessage(message || 'assertNotNull', 'got <?>', obj);
645
+ this.assertBlock(message, function() { return obj !== null });
646
+ },
647
+
648
+ assertUndefined: function(obj, message) {
649
+ message = this.buildMessage(message || 'assertUndefined', 'got <?>', obj);
650
+ this.assertBlock(message, function() { return typeof obj == "undefined" });
651
+ },
652
+
653
+ assertNotUndefined: function(obj, message) {
654
+ message = this.buildMessage(message || 'assertNotUndefined', 'got <?>', obj);
655
+ this.assertBlock(message, function() { return typeof obj != "undefined" });
656
+ },
657
+
658
+ assertNullOrUndefined: function(obj, message) {
659
+ message = this.buildMessage(message || 'assertNullOrUndefined', 'got <?>', obj);
660
+ this.assertBlock(message, function() { return obj == null });
661
+ },
662
+
663
+ assertNotNullOrUndefined: function(obj, message) {
664
+ message = this.buildMessage(message || 'assertNotNullOrUndefined', 'got <?>', obj);
665
+ this.assertBlock(message, function() { return obj != null });
666
+ },
667
+
668
+ assertMatch: function(expected, actual, message) {
669
+ message = this.buildMessage(message || 'assertMatch', 'regex <?> did not match <?>', expected, actual);
670
+ this.assertBlock(message, function() { return new RegExp(expected).exec(actual) });
671
+ },
672
+
673
+ assertNoMatch: function(expected, actual, message) {
674
+ message = this.buildMessage(message || 'assertNoMatch', 'regex <?> matched <?>', expected, actual);
675
+ this.assertBlock(message, function() { return !(new RegExp(expected).exec(actual)) });
676
+ },
677
+
678
+ assertHidden: function(element, message) {
679
+ message = this.buildMessage(message || 'assertHidden', '? isn\'t hidden.', element);
680
+ this.assertBlock(message, function() { return !element.style.display || element.style.display == 'none' });
681
+ },
682
+
683
+ assertInstanceOf: function(expected, actual, message) {
684
+ message = this.buildMessage(message || 'assertInstanceOf', '<?> was not an instance of the expected type', actual);
685
+ this.assertBlock(message, function() { return actual instanceof expected });
686
+ },
687
+
688
+ assertNotInstanceOf: function(expected, actual, message) {
689
+ message = this.buildMessage(message || 'assertNotInstanceOf', '<?> was an instance of the expected type', actual);
690
+ this.assertBlock(message, function() { return !(actual instanceof expected) });
691
+ },
692
+
693
+ assertRespondsTo: function(method, obj, message) {
694
+ message = this.buildMessage(message || 'assertRespondsTo', 'object doesn\'t respond to <?>', method);
695
+ this.assertBlock(message, function() { return (method in obj && typeof obj[method] == 'function') });
696
+ },
697
+
698
+ assertRaise: function(exceptionName, method, message) {
699
+ message = this.buildMessage(message || 'assertRaise', '<?> exception expected but none was raised', exceptionName);
700
+ var block = function() {
701
+ try {
702
+ method();
703
+ return false;
704
+ } catch(e) {
705
+ if (e.name == exceptionName) return true;
706
+ else throw e;
707
+ }
708
+ };
709
+ this.assertBlock(message, block);
710
+ },
711
+
712
+ assertNothingRaised: function(method, message) {
713
+ try {
714
+ method();
715
+ this.assert(true, "Expected nothing to be thrown");
716
+ } catch(e) {
717
+ message = this.buildMessage(message || 'assertNothingRaised', '<?> was thrown when nothing was expected.', e);
718
+ this.flunk(message);
719
+ }
720
+ },
721
+
722
+ _isVisible: function(element) {
723
+ element = JsUnitTest.$(element);
724
+ if(!element.parentNode) return true;
725
+ this.assertNotNull(element);
726
+ if(element.style && (element.style.display == 'none'))
727
+ return false;
728
+
729
+ return arguments.callee.call(this, element.parentNode);
730
+ },
731
+
732
+ assertVisible: function(element, message) {
733
+ message = this.buildMessage(message, '? was not visible.', element);
734
+ this.assertBlock(message, function() { return this._isVisible(element) });
735
+ },
736
+
737
+ assertNotVisible: function(element, message) {
738
+ message = this.buildMessage(message, '? was not hidden and didn\'t have a hidden parent either.', element);
739
+ this.assertBlock(message, function() { return !this._isVisible(element) });
740
+ },
741
+
742
+ assertElementsMatch: function() {
743
+ var pass = true, expressions = JsUnitTest.arrayfromargs(arguments);
744
+ var elements = expressions.shift();
745
+ if (elements.length != expressions.length) {
746
+ message = this.buildMessage('assertElementsMatch', 'size mismatch: ? elements, ? expressions (?).', elements.length, expressions.length, expressions);
747
+ this.flunk(message);
748
+ pass = false;
749
+ }
750
+ for (var i=0; i < expressions.length; i++) {
751
+ var expression = expressions[i];
752
+ var element = JsUnitTest.$(elements[i]);
753
+ if (JsUnitTest.selectorMatch(expression, element)) {
754
+ pass = true;
755
+ break;
756
+ }
757
+ message = this.buildMessage('assertElementsMatch', 'In index <?>: expected <?> but got ?', index, expression, element);
758
+ this.flunk(message);
759
+ pass = false;
760
+ };
761
+ this.assert(pass, "Expected all elements to match.");
762
+ },
763
+
764
+ assertElementMatches: function(element, expression, message) {
765
+ this.assertElementsMatch([element], expression);
766
+ }
767
+ };
768
+ JsUnitTest.Unit.Runner = function(testcases) {
769
+ var argumentOptions = arguments[1] || {};
770
+ var options = this.options = {};
771
+ options.testLog = ('testLog' in argumentOptions) ? argumentOptions.testLog : 'testlog';
772
+ options.resultsURL = this.queryParams.resultsURL;
773
+ options.testLog = JsUnitTest.$(options.testLog);
774
+
775
+ this.tests = this.getTests(testcases);
776
+ this.currentTest = 0;
777
+ this.logger = new JsUnitTest.Unit.Logger(options.testLog);
778
+
779
+ var self = this;
780
+ JsUnitTest.Event.addEvent(window, "load", function() {
781
+ setTimeout(function() {
782
+ self.runTests();
783
+ }, 0.1);
784
+ });
785
+ };
786
+
787
+ JsUnitTest.Unit.Runner.prototype.queryParams = JsUnitTest.toQueryParams();
788
+
789
+ JsUnitTest.Unit.Runner.prototype.portNumber = function() {
790
+ if (window.location.search.length > 0) {
791
+ var matches = window.location.search.match(/\:(\d{3,5})\//);
792
+ if (matches) {
793
+ return parseInt(matches[1]);
794
+ }
795
+ }
796
+ return null;
797
+ };
798
+
799
+ JsUnitTest.Unit.Runner.prototype.getTests = function(testcases) {
800
+ var tests = [], options = this.options;
801
+ if (this.queryParams.tests) tests = this.queryParams.tests.split(',');
802
+ else if (options.tests) tests = options.tests;
803
+ else if (options.test) tests = [option.test];
804
+ else {
805
+ for (testname in testcases) {
806
+ if (testname.match(/^test/)) tests.push(testname);
807
+ }
808
+ }
809
+ var results = [];
810
+ for (var i=0; i < tests.length; i++) {
811
+ var test = tests[i];
812
+ if (testcases[test])
813
+ results.push(
814
+ new JsUnitTest.Unit.Testcase(test, testcases[test], testcases.setup, testcases.teardown)
815
+ );
816
+ };
817
+ return results;
818
+ };
819
+
820
+ JsUnitTest.Unit.Runner.prototype.getResult = function() {
821
+ var results = {
822
+ tests: this.tests.length,
823
+ assertions: 0,
824
+ failures: 0,
825
+ errors: 0
826
+ };
827
+
828
+ for (var i=0; i < this.tests.length; i++) {
829
+ var test = this.tests[i];
830
+ results.assertions += test.assertions;
831
+ results.failures += test.failures;
832
+ results.errors += test.errors;
833
+ };
834
+ return results;
835
+ };
836
+
837
+ JsUnitTest.Unit.Runner.prototype.postResults = function() {
838
+ if (this.options.resultsURL) {
839
+ // new Ajax.Request(this.options.resultsURL,
840
+ // { method: 'get', parameters: this.getResult(), asynchronous: false });
841
+ var results = this.getResult();
842
+ var url = this.options.resultsURL + "?";
843
+ url += "assertions="+ results.assertions + "&";
844
+ url += "failures=" + results.failures + "&";
845
+ url += "errors=" + results.errors;
846
+ JsUnitTest.ajax({
847
+ url: url,
848
+ type: 'GET'
849
+ })
850
+ }
851
+ };
852
+
853
+ JsUnitTest.Unit.Runner.prototype.runTests = function() {
854
+ var test = this.tests[this.currentTest], actions;
855
+
856
+ if (!test) return this.finish();
857
+ if (!test.isWaiting) this.logger.start(test.name);
858
+ test.run();
859
+ var self = this;
860
+ if(test.isWaiting) {
861
+ this.logger.message("Waiting for " + test.timeToWait + "ms");
862
+ // setTimeout(this.runTests.bind(this), test.timeToWait || 1000);
863
+ setTimeout(function() {
864
+ self.runTests();
865
+ }, test.timeToWait || 1000);
866
+ return;
867
+ }
868
+
869
+ this.logger.finish(test.status(), test.summary());
870
+ if (actions = test.actions) this.logger.appendActionButtons(actions);
871
+ this.currentTest++;
872
+ // tail recursive, hopefully the browser will skip the stackframe
873
+ this.runTests();
874
+ };
875
+
876
+ JsUnitTest.Unit.Runner.prototype.finish = function() {
877
+ this.postResults();
878
+ this.logger.summary(this.summary());
879
+ };
880
+
881
+ JsUnitTest.Unit.Runner.prototype.summary = function() {
882
+ return new JsUnitTest.Template('#{tests} tests, #{assertions} assertions, #{failures} failures, #{errors} errors').evaluate(this.getResult());
883
+ };
884
+ JsUnitTest.Unit.Testcase = function(name, test, setup, teardown) {
885
+ this.name = name;
886
+ this.test = test || function() {};
887
+ this.setup = setup || function() {};
888
+ this.teardown = teardown || function() {};
889
+ this.messages = [];
890
+ this.actions = {};
891
+ };
892
+ // import JsUnitTest.Unit.Assertions
893
+
894
+ for (method in JsUnitTest.Unit.Assertions) {
895
+ JsUnitTest.Unit.Testcase.prototype[method] = JsUnitTest.Unit.Assertions[method];
896
+ }
897
+
898
+ JsUnitTest.Unit.Testcase.prototype.isWaiting = false;
899
+ JsUnitTest.Unit.Testcase.prototype.timeToWait = 1000;
900
+ JsUnitTest.Unit.Testcase.prototype.assertions = 0;
901
+ JsUnitTest.Unit.Testcase.prototype.failures = 0;
902
+ JsUnitTest.Unit.Testcase.prototype.errors = 0;
903
+ // JsUnitTest.Unit.Testcase.prototype.isRunningFromRake = window.location.port == 4711;
904
+ JsUnitTest.Unit.Testcase.prototype.isRunningFromRake = window.location.port;
905
+
906
+ JsUnitTest.Unit.Testcase.prototype.wait = function(time, nextPart) {
907
+ this.isWaiting = true;
908
+ this.test = nextPart;
909
+ this.timeToWait = time;
910
+ };
911
+
912
+ JsUnitTest.Unit.Testcase.prototype.run = function(rethrow) {
913
+ try {
914
+ try {
915
+ if (!this.isWaiting) this.setup();
916
+ this.isWaiting = false;
917
+ this.test();
918
+ } finally {
919
+ if(!this.isWaiting) {
920
+ this.teardown();
921
+ }
922
+ }
923
+ }
924
+ catch(e) {
925
+ if (rethrow) throw e;
926
+ this.error(e, this);
927
+ }
928
+ };
929
+
930
+ JsUnitTest.Unit.Testcase.prototype.summary = function() {
931
+ var msg = '#{assertions} assertions, #{failures} failures, #{errors} errors\n';
932
+ return new JsUnitTest.Template(msg).evaluate(this) +
933
+ this.messages.join("\n");
934
+ };
935
+
936
+ JsUnitTest.Unit.Testcase.prototype.pass = function() {
937
+ this.assertions++;
938
+ };
939
+
940
+ JsUnitTest.Unit.Testcase.prototype.fail = function(message) {
941
+ this.failures++;
942
+ var line = "";
943
+ try {
944
+ throw new Error("stack");
945
+ } catch(e){
946
+ line = (/\.html:(\d+)/.exec(e.stack || '') || ['',''])[1];
947
+ }
948
+ this.messages.push("Failure: " + message + (line ? " Line #" + line : ""));
949
+ };
950
+
951
+ JsUnitTest.Unit.Testcase.prototype.info = function(message) {
952
+ this.messages.push("Info: " + message);
953
+ };
954
+
955
+ JsUnitTest.Unit.Testcase.prototype.error = function(error, test) {
956
+ this.errors++;
957
+ this.actions['retry with throw'] = function() { test.run(true) };
958
+ this.messages.push(error.name + ": "+ error.message + "(" + JsUnitTest.inspect(error) + ")");
959
+ };
960
+
961
+ JsUnitTest.Unit.Testcase.prototype.status = function() {
962
+ if (this.failures > 0) return 'failed';
963
+ if (this.errors > 0) return 'error';
964
+ return 'passed';
965
+ };
966
+
967
+ JsUnitTest.Unit.Testcase.prototype.benchmark = function(operation, iterations) {
968
+ var startAt = new Date();
969
+ (iterations || 1).times(operation);
970
+ var timeTaken = ((new Date())-startAt);
971
+ this.info((arguments[2] || 'Operation') + ' finished ' +
972
+ iterations + ' iterations in ' + (timeTaken/1000)+'s' );
973
+ return timeTaken;
974
+ };
975
+
976
+ Test = JsUnitTest