i18n-js 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,168 @@
1
+ require 'test_helper'
2
+
3
+ class I18nJSTest < ActiveSupport::TestCase
4
+ setup do
5
+ # Remove temporary directory if already present
6
+ FileUtils.rm_r(Rails.root) if File.exist?(Rails.root)
7
+
8
+ # Create temporary directory to test the files generation
9
+ %w( config public/javascripts ).each do |path|
10
+ FileUtils.mkdir_p Rails.root.join(path)
11
+ end
12
+
13
+ # Overwrite defaut locales path to use fixtures
14
+ I18n.load_path = [File.dirname(__FILE__) + "/resources/locales.yml"]
15
+ end
16
+
17
+ teardown do
18
+ # Remove temporary directory
19
+ FileUtils.rm_r(Rails.root)
20
+ end
21
+
22
+ test "copy configuration file" do
23
+ assert_equal false, File.file?(SimplesIdeias::I18n.config_file)
24
+ SimplesIdeias::I18n.setup!
25
+ assert File.file?(SimplesIdeias::I18n.config_file)
26
+ end
27
+
28
+ test "don't overwrite existing configuration file" do
29
+ File.open(SimplesIdeias::I18n.config_file, "w+") {|f| f << "ORIGINAL"}
30
+ SimplesIdeias::I18n.setup!
31
+
32
+ assert_equal "ORIGINAL", File.read(SimplesIdeias::I18n.config_file)
33
+ end
34
+
35
+ test "copy JavaScript library" do
36
+ path = Rails.root.join("public/javascripts/i18n.js")
37
+
38
+ assert_equal false, File.file?(path)
39
+ SimplesIdeias::I18n.setup!
40
+ assert File.file?(path)
41
+ end
42
+
43
+ test "load configuration file" do
44
+ SimplesIdeias::I18n.setup!
45
+
46
+ assert SimplesIdeias::I18n.config?
47
+ assert_kind_of HashWithIndifferentAccess, SimplesIdeias::I18n.config
48
+ assert SimplesIdeias::I18n.config.any?
49
+ end
50
+
51
+ test "export messages to default path when configuration file doesn't exist" do
52
+ SimplesIdeias::I18n.export!
53
+ assert File.file?(Rails.root.join("public/javascripts/translations.js"))
54
+ end
55
+
56
+ test "export messages using the default configuration file" do
57
+ set_config "default.yml"
58
+ SimplesIdeias::I18n.expects(:save).with(translations, "public/javascripts/translations.js")
59
+ SimplesIdeias::I18n.export!
60
+ end
61
+
62
+ test "export messages using custom output path" do
63
+ set_config "custom_path.yml"
64
+ SimplesIdeias::I18n.expects(:save).with(translations, "public/javascripts/translations/all.js")
65
+ SimplesIdeias::I18n.export!
66
+ end
67
+
68
+ test "set default scope to * when not specified" do
69
+ set_config "no_scope.yml"
70
+ SimplesIdeias::I18n.expects(:save).with(translations, "public/javascripts/no_scope.js")
71
+ SimplesIdeias::I18n.export!
72
+ end
73
+
74
+ test "export to multiple files" do
75
+ set_config "multiple_files.yml"
76
+ SimplesIdeias::I18n.export!
77
+
78
+ assert File.file?(Rails.root.join("public/javascripts/all.js"))
79
+ assert File.file?(Rails.root.join("public/javascripts/tudo.js"))
80
+ end
81
+
82
+ test "filtered translations using scope *.date.formats" do
83
+ result = SimplesIdeias::I18n.filter(translations, "*.date.formats")
84
+ assert_equal [:formats], result[:en][:date].keys
85
+ assert_equal [:formats], result[:fr][:date].keys
86
+ end
87
+
88
+ test "filtered translations using scope [*.date.formats, *.number.currency.format]" do
89
+ result = SimplesIdeias::I18n.scoped_translations(["*.date.formats", "*.number.currency.format"])
90
+ assert_equal %w[ date number ], result[:en].keys.collect(&:to_s).sort
91
+ assert_equal %w[ date number ], result[:fr].keys.collect(&:to_s).sort
92
+ end
93
+
94
+ test "filtered translations using multi-star scope" do
95
+ result = SimplesIdeias::I18n.scoped_translations("*.*.formats")
96
+
97
+ assert_equal %w[ date time ], result[:en].keys.collect(&:to_s).sort
98
+ assert_equal %w[ date time ], result[:fr].keys.collect(&:to_s).sort
99
+
100
+ assert_equal [:formats], result[:en][:date].keys
101
+ assert_equal [:formats], result[:en][:time].keys
102
+
103
+ assert_equal [:formats], result[:fr][:date].keys
104
+ assert_equal [:formats], result[:fr][:time].keys
105
+ end
106
+
107
+ test "filtered translations using alternated stars" do
108
+ result = SimplesIdeias::I18n.scoped_translations("*.admin.*.title")
109
+
110
+ assert_equal %w[ edit show ], result[:en][:admin].keys.collect(&:to_s).sort
111
+ assert_equal %w[ edit show ], result[:fr][:admin].keys.collect(&:to_s).sort
112
+
113
+ assert_equal "Show", result[:en][:admin][:show][:title]
114
+ assert_equal "Visualiser", result[:fr][:admin][:show][:title]
115
+
116
+ assert_equal "Edit", result[:en][:admin][:edit][:title]
117
+ assert_equal "Editer", result[:fr][:admin][:edit][:title]
118
+ end
119
+
120
+ test "deep merge" do
121
+ target = {:a => {:b => 1}}
122
+ result = SimplesIdeias::I18n.deep_merge(target, {:a => {:c => 2}})
123
+
124
+ assert_equal result[:a], {:b => 1, :c => 2}
125
+ end
126
+
127
+ test "deep banged merge" do
128
+ target = {:a => {:b => 1}}
129
+ SimplesIdeias::I18n.deep_merge!(target, {:a => {:c => 2}})
130
+
131
+ assert_equal target[:a], {:b => 1, :c => 2}
132
+ end
133
+
134
+ test "sorted hash" do
135
+ assert_equal [:c, :a, :b], {:b => 1, :a => 2, :c => 3}.keys
136
+ assert_equal [:a, :b, :c], SimplesIdeias::I18n.sorted_hash(:b => 1, :a => 2, :c => 3).keys
137
+ end
138
+
139
+ test "sorted multi-levels hash" do
140
+ hash = {
141
+ :foo => {:b => 1, :a => 2, :c => 3}
142
+ }
143
+
144
+ assert_equal [:c, :a, :b], hash[:foo].keys
145
+ assert_equal [:a, :b, :c], SimplesIdeias::I18n.sorted_hash(hash[:foo]).keys
146
+ end
147
+
148
+ test "update javascript library" do
149
+ FakeWeb.register_uri(:get, "http://github.com/fnando/i18n-js/raw/master/lib/i18n.js", :body => "UPDATED")
150
+
151
+ SimplesIdeias::I18n.setup!
152
+ SimplesIdeias::I18n.update!
153
+ assert_equal "UPDATED", File.read(SimplesIdeias::I18n.javascript_file)
154
+ end
155
+
156
+ private
157
+ # Set the configuration as the current one
158
+ def set_config(path)
159
+ config = HashWithIndifferentAccess.new(YAML.load_file(File.dirname(__FILE__) + "/resources/#{path}"))
160
+ SimplesIdeias::I18n.expects(:config?).returns(true)
161
+ SimplesIdeias::I18n.expects(:config).returns(config)
162
+ end
163
+
164
+ # Shortcut to SimplesIdeias::I18n.translations
165
+ def translations
166
+ SimplesIdeias::I18n.translations
167
+ end
168
+ end
@@ -0,0 +1,1017 @@
1
+ /* Jsunittest, version 0.7.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.7.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 = window.ActiveXObject ? new ActiveXObject('Microsoft.XMLHTTP') : 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
+ assertHasClass: function(element, klass, message) {
679
+ element = JsUnitTest.$(element);
680
+ message = this.buildMessage(message || 'assertHasClass', '? doesn\'t have class <?>.', element, klass);
681
+ this.assertBlock(message, function() {
682
+ var elementClassName = element.className;
683
+ return (elementClassName.length > 0 && (elementClassName == klass ||
684
+ new RegExp("(^|\\s)" + klass + "(\\s|$)").test(elementClassName)));
685
+ // return !!element.className.match(new RegExp(klass))
686
+ });
687
+ },
688
+
689
+ assertNotHasClass: function(element, klass, message) {
690
+ element = JsUnitTest.$(element);
691
+ message = this.buildMessage(message || 'assertNotHasClass', '? does have class <?>.', element, klass);
692
+ this.assertBlock(message, function() {
693
+ var elementClassName = element.className;
694
+ return !(elementClassName.length > 0 && (elementClassName == klass ||
695
+ new RegExp("(^|\\s)" + klass + "(\\s|$)").test(elementClassName)));
696
+ });
697
+ },
698
+
699
+ assertHidden: function(element, message) {
700
+ element = JsUnitTest.$(element);
701
+ message = this.buildMessage(message || 'assertHidden', '? isn\'t hidden.', element);
702
+ this.assertBlock(message, function() { return !element.style.display || element.style.display == 'none' });
703
+ },
704
+
705
+ assertInstanceOf: function(expected, actual, message) {
706
+ message = this.buildMessage(message || 'assertInstanceOf', '<?> was not an instance of the expected type', actual);
707
+ this.assertBlock(message, function() { return actual instanceof expected });
708
+ },
709
+
710
+ assertNotInstanceOf: function(expected, actual, message) {
711
+ message = this.buildMessage(message || 'assertNotInstanceOf', '<?> was an instance of the expected type', actual);
712
+ this.assertBlock(message, function() { return !(actual instanceof expected) });
713
+ },
714
+
715
+ assertRespondsTo: function(method, obj, message) {
716
+ message = this.buildMessage(message || 'assertRespondsTo', 'object doesn\'t respond to <?>', method);
717
+ this.assertBlock(message, function() { return (method in obj && typeof obj[method] == 'function') });
718
+ },
719
+
720
+ assertRaise: function(exceptionName, method, message) {
721
+ message = this.buildMessage(message || 'assertRaise', '<?> exception expected but none was raised', exceptionName);
722
+ var block = function() {
723
+ try {
724
+ method();
725
+ return false;
726
+ } catch(e) {
727
+ if (e.name == exceptionName) return true;
728
+ else throw e;
729
+ }
730
+ };
731
+ this.assertBlock(message, block);
732
+ },
733
+
734
+ assertNothingRaised: function(method, message) {
735
+ try {
736
+ method();
737
+ this.assert(true, "Expected nothing to be thrown");
738
+ } catch(e) {
739
+ message = this.buildMessage(message || 'assertNothingRaised', '<?> was thrown when nothing was expected.', e);
740
+ this.flunk(message);
741
+ }
742
+ },
743
+
744
+ _isVisible: function(element) {
745
+ element = JsUnitTest.$(element);
746
+ if(!element.parentNode) return true;
747
+ this.assertNotNull(element);
748
+ if(element.style && (element.style.display == 'none'))
749
+ return false;
750
+
751
+ return arguments.callee.call(this, element.parentNode);
752
+ },
753
+
754
+ assertVisible: function(element, message) {
755
+ message = this.buildMessage(message, '? was not visible.', element);
756
+ this.assertBlock(message, function() { return this._isVisible(element) });
757
+ },
758
+
759
+ assertNotVisible: function(element, message) {
760
+ message = this.buildMessage(message, '? was not hidden and didn\'t have a hidden parent either.', element);
761
+ this.assertBlock(message, function() { return !this._isVisible(element) });
762
+ },
763
+
764
+ assertElementsMatch: function() {
765
+ var pass = true, expressions = JsUnitTest.arrayfromargs(arguments);
766
+ var elements = expressions.shift();
767
+ if (elements.length != expressions.length) {
768
+ message = this.buildMessage('assertElementsMatch', 'size mismatch: ? elements, ? expressions (?).', elements.length, expressions.length, expressions);
769
+ this.flunk(message);
770
+ pass = false;
771
+ }
772
+ for (var i=0; i < expressions.length; i++) {
773
+ var expression = expressions[i];
774
+ var element = JsUnitTest.$(elements[i]);
775
+ if (JsUnitTest.selectorMatch(expression, element)) {
776
+ pass = true;
777
+ break;
778
+ }
779
+ message = this.buildMessage('assertElementsMatch', 'In index <?>: expected <?> but got ?', index, expression, element);
780
+ this.flunk(message);
781
+ pass = false;
782
+ };
783
+ this.assert(pass, "Expected all elements to match.");
784
+ },
785
+
786
+ assertElementMatches: function(element, expression, message) {
787
+ this.assertElementsMatch([element], expression);
788
+ }
789
+ };
790
+ JsUnitTest.Unit.Runner = function(testcases) {
791
+ var argumentOptions = arguments[1] || {};
792
+ var options = this.options = {};
793
+ options.testLog = ('testLog' in argumentOptions) ? argumentOptions.testLog : 'testlog';
794
+ options.resultsURL = this.queryParams.resultsURL;
795
+ options.testLog = JsUnitTest.$(options.testLog);
796
+
797
+ this.tests = this.getTests(testcases);
798
+ this.currentTest = 0;
799
+ this.logger = new JsUnitTest.Unit.Logger(options.testLog);
800
+
801
+ var self = this;
802
+ JsUnitTest.Event.addEvent(window, "load", function() {
803
+ setTimeout(function() {
804
+ self.runTests();
805
+ }, 0.1);
806
+ });
807
+ };
808
+
809
+ JsUnitTest.Unit.Runner.prototype.queryParams = JsUnitTest.toQueryParams();
810
+
811
+ JsUnitTest.Unit.Runner.prototype.portNumber = function() {
812
+ if (window.location.search.length > 0) {
813
+ var matches = window.location.search.match(/\:(\d{3,5})\//);
814
+ if (matches) {
815
+ return parseInt(matches[1]);
816
+ }
817
+ }
818
+ return null;
819
+ };
820
+
821
+ JsUnitTest.Unit.Runner.prototype.getTests = function(testcases) {
822
+ var tests = [], options = this.options;
823
+ if (this.queryParams.tests) tests = this.queryParams.tests.split(',');
824
+ else if (options.tests) tests = options.tests;
825
+ else if (options.test) tests = [option.test];
826
+ else {
827
+ for (testname in testcases) {
828
+ if (testname.match(/^test/)) tests.push(testname);
829
+ }
830
+ }
831
+ var results = [];
832
+ for (var i=0; i < tests.length; i++) {
833
+ var test = tests[i];
834
+ if (testcases[test])
835
+ results.push(
836
+ new JsUnitTest.Unit.Testcase(test, testcases[test], testcases.setup, testcases.teardown)
837
+ );
838
+ };
839
+ return results;
840
+ };
841
+
842
+ JsUnitTest.Unit.Runner.prototype.getResult = function() {
843
+ var results = {
844
+ tests: this.tests.length,
845
+ assertions: 0,
846
+ failures: 0,
847
+ errors: 0,
848
+ warnings: 0
849
+ };
850
+
851
+ for (var i=0; i < this.tests.length; i++) {
852
+ var test = this.tests[i];
853
+ results.assertions += test.assertions;
854
+ results.failures += test.failures;
855
+ results.errors += test.errors;
856
+ results.warnings += test.warnings;
857
+ };
858
+ return results;
859
+ };
860
+
861
+ JsUnitTest.Unit.Runner.prototype.postResults = function() {
862
+ if (this.options.resultsURL) {
863
+ // new Ajax.Request(this.options.resultsURL,
864
+ // { method: 'get', parameters: this.getResult(), asynchronous: false });
865
+ var results = this.getResult();
866
+ var url = this.options.resultsURL + "?";
867
+ url += "tests="+ this.tests.length + "&";
868
+ url += "assertions="+ results.assertions + "&";
869
+ url += "warnings=" + results.warnings + "&";
870
+ url += "failures=" + results.failures + "&";
871
+ url += "errors=" + results.errors;
872
+ JsUnitTest.ajax({
873
+ url: url,
874
+ type: 'GET'
875
+ })
876
+ }
877
+ };
878
+
879
+ JsUnitTest.Unit.Runner.prototype.runTests = function() {
880
+ var test = this.tests[this.currentTest], actions;
881
+
882
+ if (!test) return this.finish();
883
+ if (!test.isWaiting) this.logger.start(test.name);
884
+ test.run();
885
+ var self = this;
886
+ if(test.isWaiting) {
887
+ this.logger.message("Waiting for " + test.timeToWait + "ms");
888
+ // setTimeout(this.runTests.bind(this), test.timeToWait || 1000);
889
+ setTimeout(function() {
890
+ self.runTests();
891
+ }, test.timeToWait || 1000);
892
+ return;
893
+ }
894
+
895
+ this.logger.finish(test.status(), test.summary());
896
+ if (actions = test.actions) this.logger.appendActionButtons(actions);
897
+ this.currentTest++;
898
+ // tail recursive, hopefully the browser will skip the stackframe
899
+ this.runTests();
900
+ };
901
+
902
+ JsUnitTest.Unit.Runner.prototype.finish = function() {
903
+ this.postResults();
904
+ this.logger.summary(this.summary());
905
+ };
906
+
907
+ JsUnitTest.Unit.Runner.prototype.summary = function() {
908
+ return new JsUnitTest.Template('#{tests} tests, #{assertions} assertions, #{failures} failures, #{errors} errors, #{warnings} warnings').evaluate(this.getResult());
909
+ };
910
+ JsUnitTest.Unit.Testcase = function(name, test, setup, teardown) {
911
+ this.name = name;
912
+ this.test = test || function() {};
913
+ this.setup = setup || function() {};
914
+ this.teardown = teardown || function() {};
915
+ this.messages = [];
916
+ this.actions = {};
917
+ };
918
+ // import JsUnitTest.Unit.Assertions
919
+
920
+ for (method in JsUnitTest.Unit.Assertions) {
921
+ JsUnitTest.Unit.Testcase.prototype[method] = JsUnitTest.Unit.Assertions[method];
922
+ }
923
+
924
+ JsUnitTest.Unit.Testcase.prototype.isWaiting = false;
925
+ JsUnitTest.Unit.Testcase.prototype.timeToWait = 1000;
926
+ JsUnitTest.Unit.Testcase.prototype.assertions = 0;
927
+ JsUnitTest.Unit.Testcase.prototype.failures = 0;
928
+ JsUnitTest.Unit.Testcase.prototype.errors = 0;
929
+ JsUnitTest.Unit.Testcase.prototype.warnings = 0;
930
+ JsUnitTest.Unit.Testcase.prototype.isRunningFromRake = window.location.port;
931
+
932
+ // JsUnitTest.Unit.Testcase.prototype.isRunningFromRake = window.location.port == 4711;
933
+
934
+ JsUnitTest.Unit.Testcase.prototype.wait = function(time, nextPart) {
935
+ this.isWaiting = true;
936
+ this.test = nextPart;
937
+ this.timeToWait = time;
938
+ };
939
+
940
+ JsUnitTest.Unit.Testcase.prototype.run = function(rethrow) {
941
+ try {
942
+ try {
943
+ if (!this.isWaiting) this.setup();
944
+ this.isWaiting = false;
945
+ this.test();
946
+ } finally {
947
+ if(!this.isWaiting) {
948
+ this.teardown();
949
+ }
950
+ }
951
+ }
952
+ catch(e) {
953
+ if (rethrow) throw e;
954
+ this.error(e, this);
955
+ }
956
+ };
957
+
958
+ JsUnitTest.Unit.Testcase.prototype.summary = function() {
959
+ var msg = '#{assertions} assertions, #{failures} failures, #{errors} errors, #{warnings} warnings\n';
960
+ return new JsUnitTest.Template(msg).evaluate(this) +
961
+ this.messages.join("\n");
962
+ };
963
+
964
+ JsUnitTest.Unit.Testcase.prototype.pass = function() {
965
+ this.assertions++;
966
+ };
967
+
968
+ JsUnitTest.Unit.Testcase.prototype.fail = function(message) {
969
+ this.failures++;
970
+ var line = "";
971
+ try {
972
+ throw new Error("stack");
973
+ } catch(e){
974
+ line = (/\.html:(\d+)/.exec(e.stack || '') || ['',''])[1];
975
+ }
976
+ this.messages.push("Failure: " + message + (line ? " Line #" + line : ""));
977
+ };
978
+
979
+ JsUnitTest.Unit.Testcase.prototype.warning = function(message) {
980
+ this.warnings++;
981
+ var line = "";
982
+ try {
983
+ throw new Error("stack");
984
+ } catch(e){
985
+ line = (/\.html:(\d+)/.exec(e.stack || '') || ['',''])[1];
986
+ }
987
+ this.messages.push("Warning: " + message + (line ? " Line #" + line : ""));
988
+ };
989
+ JsUnitTest.Unit.Testcase.prototype.warn = JsUnitTest.Unit.Testcase.prototype.warning;
990
+
991
+ JsUnitTest.Unit.Testcase.prototype.info = function(message) {
992
+ this.messages.push("Info: " + message);
993
+ };
994
+
995
+ JsUnitTest.Unit.Testcase.prototype.error = function(error, test) {
996
+ this.errors++;
997
+ this.actions['retry with throw'] = function() { test.run(true) };
998
+ this.messages.push(error.name + ": "+ error.message + "(" + JsUnitTest.inspect(error) + ")");
999
+ };
1000
+
1001
+ JsUnitTest.Unit.Testcase.prototype.status = function() {
1002
+ if (this.failures > 0) return 'failed';
1003
+ if (this.errors > 0) return 'error';
1004
+ if (this.warnings > 0) return 'warning';
1005
+ return 'passed';
1006
+ };
1007
+
1008
+ JsUnitTest.Unit.Testcase.prototype.benchmark = function(operation, iterations) {
1009
+ var startAt = new Date();
1010
+ (iterations || 1).times(operation);
1011
+ var timeTaken = ((new Date())-startAt);
1012
+ this.info((arguments[2] || 'Operation') + ' finished ' +
1013
+ iterations + ' iterations in ' + (timeTaken/1000)+'s' );
1014
+ return timeTaken;
1015
+ };
1016
+
1017
+ Test = JsUnitTest