websocket-rails 0.1.5 → 0.1.6

Sign up to get free protection for your applications and to get access to all the features.
Files changed (50) hide show
  1. data/CHANGELOG.md +22 -0
  2. data/Gemfile +4 -0
  3. data/README.md +66 -159
  4. data/Rakefile +31 -4
  5. data/bin/thin-socketrails +16 -1
  6. data/lib/assets/javascripts/websocket_rails/channel.js.coffee +23 -8
  7. data/lib/assets/javascripts/websocket_rails/event.js.coffee +40 -0
  8. data/lib/assets/javascripts/websocket_rails/http_connection.js.coffee +18 -10
  9. data/lib/assets/javascripts/websocket_rails/main.js +1 -0
  10. data/lib/assets/javascripts/websocket_rails/websocket_connection.js.coffee +15 -10
  11. data/lib/assets/javascripts/websocket_rails/websocket_rails.js.coffee +41 -23
  12. data/lib/websocket-rails.rb +4 -4
  13. data/lib/websocket_rails/base_controller.rb +61 -29
  14. data/lib/websocket_rails/channel.rb +14 -5
  15. data/lib/websocket_rails/channel_manager.rb +3 -1
  16. data/lib/websocket_rails/connection_adapters.rb +34 -12
  17. data/lib/websocket_rails/connection_manager.rb +4 -0
  18. data/lib/websocket_rails/dispatcher.rb +27 -3
  19. data/lib/websocket_rails/engine.rb +2 -5
  20. data/lib/websocket_rails/event.rb +87 -42
  21. data/lib/websocket_rails/event_map.rb +70 -20
  22. data/lib/websocket_rails/event_queue.rb +4 -0
  23. data/lib/websocket_rails/internal_events.rb +21 -3
  24. data/lib/websocket_rails/logging.rb +18 -0
  25. data/lib/websocket_rails/version.rb +1 -1
  26. data/spec/dummy/log/test.log +0 -429
  27. data/spec/integration/connection_manager_spec.rb +3 -5
  28. data/spec/javascripts/generated/assets/channel.js +98 -0
  29. data/spec/javascripts/generated/assets/event.js +78 -0
  30. data/spec/javascripts/generated/assets/http_connection.js +108 -0
  31. data/spec/javascripts/generated/assets/websocket_connection.js +66 -0
  32. data/spec/javascripts/generated/assets/websocket_rails.js +180 -0
  33. data/spec/javascripts/generated/specs/channel_spec.js +66 -0
  34. data/spec/javascripts/generated/specs/event_spec.js +107 -0
  35. data/spec/javascripts/generated/specs/websocket_connection_spec.js +117 -0
  36. data/spec/javascripts/generated/specs/websocket_rails_spec.js +232 -0
  37. data/spec/javascripts/support/jasmine.yml +44 -0
  38. data/spec/javascripts/support/jasmine_config.rb +63 -0
  39. data/spec/javascripts/support/vendor/sinon-1.3.4.js +3555 -0
  40. data/spec/javascripts/websocket_rails/channel_spec.coffee +51 -0
  41. data/spec/javascripts/websocket_rails/event_spec.coffee +69 -0
  42. data/spec/javascripts/websocket_rails/websocket_connection_spec.coffee +86 -0
  43. data/spec/javascripts/websocket_rails/websocket_rails_spec.coffee +166 -0
  44. data/spec/support/helper_methods.rb +10 -1
  45. data/spec/unit/channel_spec.rb +28 -4
  46. data/spec/unit/connection_adapters_spec.rb +17 -0
  47. data/spec/unit/connection_manager_spec.rb +1 -1
  48. data/spec/unit/dispatcher_spec.rb +1 -1
  49. data/spec/unit/event_spec.rb +15 -11
  50. metadata +22 -4
@@ -0,0 +1,3555 @@
1
+ /**
2
+ * Sinon.JS 1.3.4, 2012/04/16
3
+ *
4
+ * @author Christian Johansen (christian@cjohansen.no)
5
+ *
6
+ * (The BSD License)
7
+ *
8
+ * Copyright (c) 2010-2011, Christian Johansen, christian@cjohansen.no
9
+ * All rights reserved.
10
+ *
11
+ * Redistribution and use in source and binary forms, with or without modification,
12
+ * are permitted provided that the following conditions are met:
13
+ *
14
+ * * Redistributions of source code must retain the above copyright notice,
15
+ * this list of conditions and the following disclaimer.
16
+ * * Redistributions in binary form must reproduce the above copyright notice,
17
+ * this list of conditions and the following disclaimer in the documentation
18
+ * and/or other materials provided with the distribution.
19
+ * * Neither the name of Christian Johansen nor the names of his contributors
20
+ * may be used to endorse or promote products derived from this software
21
+ * without specific prior written permission.
22
+ *
23
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
24
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
25
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
26
+ * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
27
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
28
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
29
+ * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
30
+ * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
31
+ * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
32
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
33
+ */
34
+
35
+ "use strict";
36
+ var sinon = (function () {
37
+ var buster = (function (buster, setTimeout) {
38
+ function extend(target) {
39
+ if (!target) {
40
+ return;
41
+ }
42
+
43
+ for (var i = 1, l = arguments.length, prop; i < l; ++i) {
44
+ for (prop in arguments[i]) {
45
+ target[prop] = arguments[i][prop];
46
+ }
47
+ }
48
+
49
+ return target;
50
+ }
51
+
52
+ var div = typeof document != "undefined" && document.createElement("div");
53
+
54
+ return extend(buster, {
55
+ bind: function (obj, methOrProp) {
56
+ var method = typeof methOrProp == "string" ? obj[methOrProp] : methOrProp;
57
+ var args = Array.prototype.slice.call(arguments, 2);
58
+
59
+ return function () {
60
+ var allArgs = args.concat(Array.prototype.slice.call(arguments));
61
+ return method.apply(obj, allArgs);
62
+ };
63
+ },
64
+
65
+ create: (function () {
66
+ function F() {}
67
+
68
+ return function create(object) {
69
+ F.prototype = object;
70
+ return new F();
71
+ }
72
+ }()),
73
+
74
+ extend: extend,
75
+
76
+ nextTick: function (callback) {
77
+ if (typeof process != "undefined" && process.nextTick) {
78
+ return process.nextTick(callback);
79
+ }
80
+
81
+ setTimeout(callback, 0);
82
+ },
83
+
84
+ functionName: function (func) {
85
+ if (!func) return "";
86
+ if (func.displayName) return func.displayName;
87
+ if (func.name) return func.name;
88
+
89
+ var matches = func.toString().match(/function\s+([^\(]+)/m);
90
+ return matches && matches[1] || "";
91
+ },
92
+
93
+ isNode: function (obj) {
94
+ if (!div) return false;
95
+
96
+ try {
97
+ obj.appendChild(div);
98
+ obj.removeChild(div);
99
+ } catch (e) {
100
+ return false;
101
+ }
102
+
103
+ return true;
104
+ },
105
+
106
+ isElement: function (obj) {
107
+ return obj && buster.isNode(obj) && obj.nodeType === 1;
108
+ }
109
+ });
110
+ }(buster || {}, setTimeout));
111
+
112
+ if (typeof module == "object" && typeof require == "function") {
113
+ module.exports = buster;
114
+ buster.eventEmitter = require("./buster-event-emitter");
115
+
116
+ Object.defineProperty(buster, "defineVersionGetter", {
117
+ get: function () {
118
+ return require("./define-version-getter");
119
+ }
120
+ });
121
+ }
122
+ if (typeof buster === "undefined") {
123
+ var buster = {};
124
+ }
125
+
126
+ if (typeof module === "object" && typeof require === "function") {
127
+ buster = require("buster-core");
128
+ }
129
+
130
+ buster.format = buster.format || {};
131
+ buster.format.excludeConstructors = ["Object", /^.$/];
132
+ buster.format.quoteStrings = true;
133
+
134
+ buster.format.ascii = (function () {
135
+
136
+ function keys(object) {
137
+ var k = Object.keys && Object.keys(object) || [];
138
+
139
+ if (k.length == 0) {
140
+ for (var prop in object) {
141
+ if (object.hasOwnProperty(prop)) {
142
+ k.push(prop);
143
+ }
144
+ }
145
+ }
146
+
147
+ return k.sort();
148
+ }
149
+
150
+ function isCircular(object, objects) {
151
+ if (typeof object != "object") {
152
+ return false;
153
+ }
154
+
155
+ for (var i = 0, l = objects.length; i < l; ++i) {
156
+ if (objects[i] === object) {
157
+ return true;
158
+ }
159
+ }
160
+
161
+ return false;
162
+ }
163
+
164
+ function ascii(object, processed, indent) {
165
+ if (typeof object == "string") {
166
+ var quote = typeof this.quoteStrings != "boolean" || this.quoteStrings;
167
+ return processed || quote ? '"' + object + '"' : object;
168
+ }
169
+
170
+ if (typeof object == "function" && !(object instanceof RegExp)) {
171
+ return ascii.func(object);
172
+ }
173
+
174
+ processed = processed || [];
175
+
176
+ if (isCircular(object, processed)) {
177
+ return "[Circular]";
178
+ }
179
+
180
+ if (Object.prototype.toString.call(object) == "[object Array]") {
181
+ return ascii.array.call(this, object);
182
+ }
183
+
184
+ if (!object) {
185
+ return "" + object;
186
+ }
187
+
188
+ if (buster.isElement(object)) {
189
+ return ascii.element(object);
190
+ }
191
+
192
+ if (typeof object.toString == "function" &&
193
+ object.toString !== Object.prototype.toString) {
194
+ return object.toString();
195
+ }
196
+
197
+ return ascii.object.call(this, object, processed, indent);
198
+ }
199
+
200
+ ascii.func = function (func) {
201
+ return "function " + buster.functionName(func) + "() {}";
202
+ };
203
+
204
+ ascii.array = function (array, processed) {
205
+ processed = processed || [];
206
+ processed.push(array);
207
+ var pieces = [];
208
+
209
+ for (var i = 0, l = array.length; i < l; ++i) {
210
+ pieces.push(ascii.call(this, array[i], processed));
211
+ }
212
+
213
+ return "[" + pieces.join(", ") + "]";
214
+ };
215
+
216
+ ascii.object = function (object, processed, indent) {
217
+ processed = processed || [];
218
+ processed.push(object);
219
+ indent = indent || 0;
220
+ var pieces = [], properties = keys(object), prop, str, obj;
221
+ var is = "";
222
+ var length = 3;
223
+
224
+ for (var i = 0, l = indent; i < l; ++i) {
225
+ is += " ";
226
+ }
227
+
228
+ for (i = 0, l = properties.length; i < l; ++i) {
229
+ prop = properties[i];
230
+ obj = object[prop];
231
+
232
+ if (isCircular(obj, processed)) {
233
+ str = "[Circular]";
234
+ } else {
235
+ str = ascii.call(this, obj, processed, indent + 2);
236
+ }
237
+
238
+ str = (/\s/.test(prop) ? '"' + prop + '"' : prop) + ": " + str;
239
+ length += str.length;
240
+ pieces.push(str);
241
+ }
242
+
243
+ var cons = ascii.constructorName.call(this, object);
244
+ var prefix = cons ? "[" + cons + "] " : ""
245
+
246
+ return (length + indent) > 80 ?
247
+ prefix + "{\n " + is + pieces.join(",\n " + is) + "\n" + is + "}" :
248
+ prefix + "{ " + pieces.join(", ") + " }";
249
+ };
250
+
251
+ ascii.element = function (element) {
252
+ var tagName = element.tagName.toLowerCase();
253
+ var attrs = element.attributes, attribute, pairs = [], attrName;
254
+
255
+ for (var i = 0, l = attrs.length; i < l; ++i) {
256
+ attribute = attrs.item(i);
257
+ attrName = attribute.nodeName.toLowerCase().replace("html:", "");
258
+
259
+ if (attrName == "contenteditable" && attribute.nodeValue == "inherit") {
260
+ continue;
261
+ }
262
+
263
+ if (!!attribute.nodeValue) {
264
+ pairs.push(attrName + "=\"" + attribute.nodeValue + "\"");
265
+ }
266
+ }
267
+
268
+ var formatted = "<" + tagName + (pairs.length > 0 ? " " : "");
269
+ var content = element.innerHTML;
270
+
271
+ if (content.length > 20) {
272
+ content = content.substr(0, 20) + "[...]";
273
+ }
274
+
275
+ var res = formatted + pairs.join(" ") + ">" + content + "</" + tagName + ">";
276
+
277
+ return res.replace(/ contentEditable="inherit"/, "");
278
+ };
279
+
280
+ ascii.constructorName = function (object) {
281
+ var name = buster.functionName(object && object.constructor);
282
+ var excludes = this.excludeConstructors || buster.format.excludeConstructors || [];
283
+
284
+ for (var i = 0, l = excludes.length; i < l; ++i) {
285
+ if (typeof excludes[i] == "string" && excludes[i] == name) {
286
+ return "";
287
+ } else if (excludes[i].test && excludes[i].test(name)) {
288
+ return "";
289
+ }
290
+ }
291
+
292
+ return name;
293
+ };
294
+
295
+ return ascii;
296
+ }());
297
+
298
+ if (typeof module != "undefined") {
299
+ module.exports = buster.format;
300
+ }
301
+ /*jslint eqeqeq: false, onevar: false, forin: true, nomen: false, regexp: false, plusplus: false*/
302
+ /*global module, require, __dirname, document*/
303
+ /**
304
+ * Sinon core utilities. For internal use only.
305
+ *
306
+ * @author Christian Johansen (christian@cjohansen.no)
307
+ * @license BSD
308
+ *
309
+ * Copyright (c) 2010-2011 Christian Johansen
310
+ */
311
+
312
+ var sinon = (function (buster) {
313
+ var div = typeof document != "undefined" && document.createElement("div");
314
+ var hasOwn = Object.prototype.hasOwnProperty;
315
+
316
+ function isDOMNode(obj) {
317
+ var success = false;
318
+
319
+ try {
320
+ obj.appendChild(div);
321
+ success = div.parentNode == obj;
322
+ } catch (e) {
323
+ return false;
324
+ } finally {
325
+ try {
326
+ obj.removeChild(div);
327
+ } catch (e) {
328
+ // Remove failed, not much we can do about that
329
+ }
330
+ }
331
+
332
+ return success;
333
+ }
334
+
335
+ function isElement(obj) {
336
+ return div && obj && obj.nodeType === 1 && isDOMNode(obj);
337
+ }
338
+
339
+ function isFunction(obj) {
340
+ return !!(obj && obj.constructor && obj.call && obj.apply);
341
+ }
342
+
343
+ function mirrorProperties(target, source) {
344
+ for (var prop in source) {
345
+ if (!hasOwn.call(target, prop)) {
346
+ target[prop] = source[prop];
347
+ }
348
+ }
349
+ }
350
+
351
+ var sinon = {
352
+ wrapMethod: function wrapMethod(object, property, method) {
353
+ if (!object) {
354
+ throw new TypeError("Should wrap property of object");
355
+ }
356
+
357
+ if (typeof method != "function") {
358
+ throw new TypeError("Method wrapper should be function");
359
+ }
360
+
361
+ var wrappedMethod = object[property];
362
+
363
+ if (!isFunction(wrappedMethod)) {
364
+ throw new TypeError("Attempted to wrap " + (typeof wrappedMethod) + " property " +
365
+ property + " as function");
366
+ }
367
+
368
+ if (wrappedMethod.restore && wrappedMethod.restore.sinon) {
369
+ throw new TypeError("Attempted to wrap " + property + " which is already wrapped");
370
+ }
371
+
372
+ if (wrappedMethod.calledBefore) {
373
+ var verb = !!wrappedMethod.returns ? "stubbed" : "spied on";
374
+ throw new TypeError("Attempted to wrap " + property + " which is already " + verb);
375
+ }
376
+
377
+ // IE 8 does not support hasOwnProperty on the window object.
378
+ var owned = hasOwn.call(object, property);
379
+ object[property] = method;
380
+ method.displayName = property;
381
+
382
+ method.restore = function () {
383
+ if(owned) {
384
+ object[property] = wrappedMethod;
385
+ } else {
386
+ delete object[property];
387
+ }
388
+ };
389
+
390
+ method.restore.sinon = true;
391
+ mirrorProperties(method, wrappedMethod);
392
+
393
+ return method;
394
+ },
395
+
396
+ extend: function extend(target) {
397
+ for (var i = 1, l = arguments.length; i < l; i += 1) {
398
+ for (var prop in arguments[i]) {
399
+ if (arguments[i].hasOwnProperty(prop)) {
400
+ target[prop] = arguments[i][prop];
401
+ }
402
+
403
+ // DONT ENUM bug, only care about toString
404
+ if (arguments[i].hasOwnProperty("toString") &&
405
+ arguments[i].toString != target.toString) {
406
+ target.toString = arguments[i].toString;
407
+ }
408
+ }
409
+ }
410
+
411
+ return target;
412
+ },
413
+
414
+ create: function create(proto) {
415
+ var F = function () {};
416
+ F.prototype = proto;
417
+ return new F();
418
+ },
419
+
420
+ deepEqual: function deepEqual(a, b) {
421
+ if (typeof a != "object" || typeof b != "object") {
422
+ return a === b;
423
+ }
424
+
425
+ if (isElement(a) || isElement(b)) {
426
+ return a === b;
427
+ }
428
+
429
+ if (a === b) {
430
+ return true;
431
+ }
432
+
433
+ var aString = Object.prototype.toString.call(a);
434
+ if (aString != Object.prototype.toString.call(b)) {
435
+ return false;
436
+ }
437
+
438
+ if (aString == "[object Array]") {
439
+ if (a.length !== b.length) {
440
+ return false;
441
+ }
442
+
443
+ for (var i = 0, l = a.length; i < l; i += 1) {
444
+ if (!deepEqual(a[i], b[i])) {
445
+ return false;
446
+ }
447
+ }
448
+
449
+ return true;
450
+ }
451
+
452
+ var prop, aLength = 0, bLength = 0;
453
+
454
+ for (prop in a) {
455
+ aLength += 1;
456
+
457
+ if (!deepEqual(a[prop], b[prop])) {
458
+ return false;
459
+ }
460
+ }
461
+
462
+ for (prop in b) {
463
+ bLength += 1;
464
+ }
465
+
466
+ if (aLength != bLength) {
467
+ return false;
468
+ }
469
+
470
+ return true;
471
+ },
472
+
473
+ functionName: function functionName(func) {
474
+ var name = func.displayName || func.name;
475
+
476
+ // Use function decomposition as a last resort to get function
477
+ // name. Does not rely on function decomposition to work - if it
478
+ // doesn't debugging will be slightly less informative
479
+ // (i.e. toString will say 'spy' rather than 'myFunc').
480
+ if (!name) {
481
+ var matches = func.toString().match(/function ([^\s\(]+)/);
482
+ name = matches && matches[1];
483
+ }
484
+
485
+ return name;
486
+ },
487
+
488
+ functionToString: function toString() {
489
+ if (this.getCall && this.callCount) {
490
+ var thisValue, prop, i = this.callCount;
491
+
492
+ while (i--) {
493
+ thisValue = this.getCall(i).thisValue;
494
+
495
+ for (prop in thisValue) {
496
+ if (thisValue[prop] === this) {
497
+ return prop;
498
+ }
499
+ }
500
+ }
501
+ }
502
+
503
+ return this.displayName || "sinon fake";
504
+ },
505
+
506
+ getConfig: function (custom) {
507
+ var config = {};
508
+ custom = custom || {};
509
+ var defaults = sinon.defaultConfig;
510
+
511
+ for (var prop in defaults) {
512
+ if (defaults.hasOwnProperty(prop)) {
513
+ config[prop] = custom.hasOwnProperty(prop) ? custom[prop] : defaults[prop];
514
+ }
515
+ }
516
+
517
+ return config;
518
+ },
519
+
520
+ format: function (val) {
521
+ return "" + val;
522
+ },
523
+
524
+ defaultConfig: {
525
+ injectIntoThis: true,
526
+ injectInto: null,
527
+ properties: ["spy", "stub", "mock", "clock", "server", "requests"],
528
+ useFakeTimers: true,
529
+ useFakeServer: true
530
+ },
531
+
532
+ timesInWords: function timesInWords(count) {
533
+ return count == 1 && "once" ||
534
+ count == 2 && "twice" ||
535
+ count == 3 && "thrice" ||
536
+ (count || 0) + " times";
537
+ },
538
+
539
+ calledInOrder: function (spies) {
540
+ for (var i = 1, l = spies.length; i < l; i++) {
541
+ if (!spies[i - 1].calledBefore(spies[i])) {
542
+ return false;
543
+ }
544
+ }
545
+
546
+ return true;
547
+ },
548
+
549
+ orderByFirstCall: function (spies) {
550
+ return spies.sort(function (a, b) {
551
+ // uuid, won't ever be equal
552
+ var aCall = a.getCall(0);
553
+ var bCall = b.getCall(0);
554
+ var aId = aCall && aCall.callId || -1;
555
+ var bId = bCall && bCall.callId || -1;
556
+
557
+ return aId < bId ? -1 : 1;
558
+ });
559
+ },
560
+
561
+ log: function () {},
562
+
563
+ logError: function (label, err) {
564
+ var msg = label + " threw exception: "
565
+ sinon.log(msg + "[" + err.name + "] " + err.message);
566
+ if (err.stack) { sinon.log(err.stack); }
567
+
568
+ setTimeout(function () {
569
+ err.message = msg + err.message;
570
+ throw err;
571
+ }, 0);
572
+ }
573
+ };
574
+
575
+ var isNode = typeof module == "object" && typeof require == "function";
576
+
577
+ if (isNode) {
578
+ try {
579
+ buster = { format: require("buster-format") };
580
+ } catch (e) {}
581
+ module.exports = sinon;
582
+ module.exports.spy = require("./sinon/spy");
583
+ module.exports.stub = require("./sinon/stub");
584
+ module.exports.mock = require("./sinon/mock");
585
+ module.exports.collection = require("./sinon/collection");
586
+ module.exports.assert = require("./sinon/assert");
587
+ module.exports.sandbox = require("./sinon/sandbox");
588
+ module.exports.test = require("./sinon/test");
589
+ module.exports.testCase = require("./sinon/test_case");
590
+ module.exports.assert = require("./sinon/assert");
591
+ }
592
+
593
+ if (buster) {
594
+ var formatter = sinon.create(buster.format);
595
+ formatter.quoteStrings = false;
596
+ sinon.format = function () {
597
+ return formatter.ascii.apply(formatter, arguments);
598
+ };
599
+ } else if (isNode) {
600
+ try {
601
+ var util = require("util");
602
+ sinon.format = function (value) {
603
+ return typeof value == "object" ? util.inspect(value) : value;
604
+ };
605
+ } catch (e) {
606
+ /* Node, but no util module - would be very old, but better safe than
607
+ sorry */
608
+ }
609
+ }
610
+
611
+ return sinon;
612
+ }(typeof buster == "object" && buster));
613
+
614
+ /* @depend ../sinon.js */
615
+ /*jslint eqeqeq: false, onevar: false, plusplus: false*/
616
+ /*global module, require, sinon*/
617
+ /**
618
+ * Spy functions
619
+ *
620
+ * @author Christian Johansen (christian@cjohansen.no)
621
+ * @license BSD
622
+ *
623
+ * Copyright (c) 2010-2011 Christian Johansen
624
+ */
625
+
626
+ (function (sinon) {
627
+ var commonJSModule = typeof module == "object" && typeof require == "function";
628
+ var spyCall;
629
+ var callId = 0;
630
+ var push = [].push;
631
+ var slice = Array.prototype.slice;
632
+
633
+ if (!sinon && commonJSModule) {
634
+ sinon = require("../sinon");
635
+ }
636
+
637
+ if (!sinon) {
638
+ return;
639
+ }
640
+
641
+ function spy(object, property) {
642
+ if (!property && typeof object == "function") {
643
+ return spy.create(object);
644
+ }
645
+
646
+ if (!object || !property) {
647
+ return spy.create(function () {});
648
+ }
649
+
650
+ var method = object[property];
651
+ return sinon.wrapMethod(object, property, spy.create(method));
652
+ }
653
+
654
+ sinon.extend(spy, (function () {
655
+
656
+ function delegateToCalls(api, method, matchAny, actual, notCalled) {
657
+ api[method] = function () {
658
+ if (!this.called) {
659
+ if (notCalled) {
660
+ return notCalled.apply(this, arguments);
661
+ }
662
+ return false;
663
+ }
664
+
665
+ var currentCall;
666
+ var matches = 0;
667
+
668
+ for (var i = 0, l = this.callCount; i < l; i += 1) {
669
+ currentCall = this.getCall(i);
670
+
671
+ if (currentCall[actual || method].apply(currentCall, arguments)) {
672
+ matches += 1;
673
+
674
+ if (matchAny) {
675
+ return true;
676
+ }
677
+ }
678
+ }
679
+
680
+ return matches === this.callCount;
681
+ };
682
+ }
683
+
684
+ function matchingFake(fakes, args, strict) {
685
+ if (!fakes) {
686
+ return;
687
+ }
688
+
689
+ var alen = args.length;
690
+
691
+ for (var i = 0, l = fakes.length; i < l; i++) {
692
+ if (fakes[i].matches(args, strict)) {
693
+ return fakes[i];
694
+ }
695
+ }
696
+ }
697
+
698
+ function incrementCallCount() {
699
+ this.called = true;
700
+ this.callCount += 1;
701
+ this.calledOnce = this.callCount == 1;
702
+ this.calledTwice = this.callCount == 2;
703
+ this.calledThrice = this.callCount == 3;
704
+ }
705
+
706
+ function createCallProperties() {
707
+ this.firstCall = this.getCall(0);
708
+ this.secondCall = this.getCall(1);
709
+ this.thirdCall = this.getCall(2);
710
+ this.lastCall = this.getCall(this.callCount - 1);
711
+ }
712
+
713
+ var uuid = 0;
714
+
715
+ // Public API
716
+ var spyApi = {
717
+ reset: function () {
718
+ this.called = false;
719
+ this.calledOnce = false;
720
+ this.calledTwice = false;
721
+ this.calledThrice = false;
722
+ this.callCount = 0;
723
+ this.firstCall = null;
724
+ this.secondCall = null;
725
+ this.thirdCall = null;
726
+ this.lastCall = null;
727
+ this.args = [];
728
+ this.returnValues = [];
729
+ this.thisValues = [];
730
+ this.exceptions = [];
731
+ this.callIds = [];
732
+ },
733
+
734
+ create: function create(func) {
735
+ var name;
736
+
737
+ if (typeof func != "function") {
738
+ func = function () {};
739
+ } else {
740
+ name = sinon.functionName(func);
741
+ }
742
+
743
+ function proxy() {
744
+ return proxy.invoke(func, this, slice.call(arguments));
745
+ }
746
+
747
+ sinon.extend(proxy, spy);
748
+ delete proxy.create;
749
+ sinon.extend(proxy, func);
750
+
751
+ proxy.reset();
752
+ proxy.prototype = func.prototype;
753
+ proxy.displayName = name || "spy";
754
+ proxy.toString = sinon.functionToString;
755
+ proxy._create = sinon.spy.create;
756
+ proxy.id = "spy#" + uuid++;
757
+
758
+ return proxy;
759
+ },
760
+
761
+ invoke: function invoke(func, thisValue, args) {
762
+ var matching = matchingFake(this.fakes, args);
763
+ var exception, returnValue;
764
+
765
+ incrementCallCount.call(this);
766
+ push.call(this.thisValues, thisValue);
767
+ push.call(this.args, args);
768
+ push.call(this.callIds, callId++);
769
+
770
+ try {
771
+ if (matching) {
772
+ returnValue = matching.invoke(func, thisValue, args);
773
+ } else {
774
+ returnValue = (this.func || func).apply(thisValue, args);
775
+ }
776
+ } catch (e) {
777
+ push.call(this.returnValues, undefined);
778
+ exception = e;
779
+ throw e;
780
+ } finally {
781
+ push.call(this.exceptions, exception);
782
+ }
783
+
784
+ push.call(this.returnValues, returnValue);
785
+
786
+ createCallProperties.call(this);
787
+
788
+ return returnValue;
789
+ },
790
+
791
+ getCall: function getCall(i) {
792
+ if (i < 0 || i >= this.callCount) {
793
+ return null;
794
+ }
795
+
796
+ return spyCall.create(this, this.thisValues[i], this.args[i],
797
+ this.returnValues[i], this.exceptions[i],
798
+ this.callIds[i]);
799
+ },
800
+
801
+ calledBefore: function calledBefore(spyFn) {
802
+ if (!this.called) {
803
+ return false;
804
+ }
805
+
806
+ if (!spyFn.called) {
807
+ return true;
808
+ }
809
+
810
+ return this.callIds[0] < spyFn.callIds[spyFn.callIds.length - 1];
811
+ },
812
+
813
+ calledAfter: function calledAfter(spyFn) {
814
+ if (!this.called || !spyFn.called) {
815
+ return false;
816
+ }
817
+
818
+ return this.callIds[this.callCount - 1] > spyFn.callIds[spyFn.callCount - 1];
819
+ },
820
+
821
+ withArgs: function () {
822
+ var args = slice.call(arguments);
823
+
824
+ if (this.fakes) {
825
+ var match = matchingFake(this.fakes, args, true);
826
+
827
+ if (match) {
828
+ return match;
829
+ }
830
+ } else {
831
+ this.fakes = [];
832
+ }
833
+
834
+ var original = this;
835
+ var fake = this._create();
836
+ fake.matchingAguments = args;
837
+ push.call(this.fakes, fake);
838
+
839
+ fake.withArgs = function () {
840
+ return original.withArgs.apply(original, arguments);
841
+ };
842
+
843
+ for (var i = 0; i < this.args.length; i++) {
844
+ if (fake.matches(this.args[i])) {
845
+ incrementCallCount.call(fake);
846
+ push.call(fake.thisValues, this.thisValues[i]);
847
+ push.call(fake.args, this.args[i]);
848
+ push.call(fake.returnValues, this.returnValues[i]);
849
+ push.call(fake.exceptions, this.exceptions[i]);
850
+ push.call(fake.callIds, this.callIds[i]);
851
+ }
852
+ }
853
+ createCallProperties.call(fake);
854
+
855
+ return fake;
856
+ },
857
+
858
+ matches: function (args, strict) {
859
+ var margs = this.matchingAguments;
860
+
861
+ if (margs.length <= args.length &&
862
+ sinon.deepEqual(margs, args.slice(0, margs.length))) {
863
+ return !strict || margs.length == args.length;
864
+ }
865
+ },
866
+
867
+ printf: function (format) {
868
+ var spy = this;
869
+ var args = slice.call(arguments, 1);
870
+ var formatter;
871
+
872
+ return (format || "").replace(/%(.)/g, function (match, specifyer) {
873
+ formatter = spyApi.formatters[specifyer];
874
+
875
+ if (typeof formatter == "function") {
876
+ return formatter.call(null, spy, args);
877
+ } else if (!isNaN(parseInt(specifyer), 10)) {
878
+ return sinon.format(args[specifyer - 1]);
879
+ }
880
+
881
+ return "%" + specifyer;
882
+ });
883
+ }
884
+ };
885
+
886
+ delegateToCalls(spyApi, "calledOn", true);
887
+ delegateToCalls(spyApi, "alwaysCalledOn", false, "calledOn");
888
+ delegateToCalls(spyApi, "calledWith", true);
889
+ delegateToCalls(spyApi, "alwaysCalledWith", false, "calledWith");
890
+ delegateToCalls(spyApi, "calledWithExactly", true);
891
+ delegateToCalls(spyApi, "alwaysCalledWithExactly", false, "calledWithExactly");
892
+ delegateToCalls(spyApi, "neverCalledWith", false, "notCalledWith",
893
+ function () { return true; });
894
+ delegateToCalls(spyApi, "threw", true);
895
+ delegateToCalls(spyApi, "alwaysThrew", false, "threw");
896
+ delegateToCalls(spyApi, "returned", true);
897
+ delegateToCalls(spyApi, "alwaysReturned", false, "returned");
898
+ delegateToCalls(spyApi, "calledWithNew", true);
899
+ delegateToCalls(spyApi, "alwaysCalledWithNew", false, "calledWithNew");
900
+ delegateToCalls(spyApi, "callArg", false, "callArgWith", function () {
901
+ throw new Error(this.toString() + " cannot call arg since it was not yet invoked.");
902
+ });
903
+ spyApi.callArgWith = spyApi.callArg;
904
+ delegateToCalls(spyApi, "yield", false, "yield", function () {
905
+ throw new Error(this.toString() + " cannot yield since it was not yet invoked.");
906
+ });
907
+ // "invokeCallback" is an alias for "yield" since "yield" is invalid in strict mode.
908
+ spyApi.invokeCallback = spyApi.yield;
909
+ delegateToCalls(spyApi, "yieldTo", false, "yieldTo", function (property) {
910
+ throw new Error(this.toString() + " cannot yield to '" + property +
911
+ "' since it was not yet invoked.");
912
+ });
913
+
914
+ spyApi.formatters = {
915
+ "c": function (spy) {
916
+ return sinon.timesInWords(spy.callCount);
917
+ },
918
+
919
+ "n": function (spy) {
920
+ return spy.toString();
921
+ },
922
+
923
+ "C": function (spy) {
924
+ var calls = [];
925
+
926
+ for (var i = 0, l = spy.callCount; i < l; ++i) {
927
+ push.call(calls, " " + spy.getCall(i).toString());
928
+ }
929
+
930
+ return calls.length > 0 ? "\n" + calls.join("\n") : "";
931
+ },
932
+
933
+ "t": function (spy) {
934
+ var objects = [];
935
+
936
+ for (var i = 0, l = spy.callCount; i < l; ++i) {
937
+ push.call(objects, sinon.format(spy.thisValues[i]));
938
+ }
939
+
940
+ return objects.join(", ");
941
+ },
942
+
943
+ "*": function (spy, args) {
944
+ return args.join(", ");
945
+ }
946
+ };
947
+
948
+ return spyApi;
949
+ }()));
950
+
951
+ spyCall = (function () {
952
+
953
+ function throwYieldError(proxy, text, args) {
954
+ var msg = sinon.functionName(proxy) + text;
955
+ if (args.length) {
956
+ msg += " Received [" + slice.call(args).join(", ") + "]";
957
+ }
958
+ throw new Error(msg);
959
+ }
960
+
961
+ return {
962
+ create: function create(spy, thisValue, args, returnValue, exception, id) {
963
+ var proxyCall = sinon.create(spyCall);
964
+ delete proxyCall.create;
965
+ proxyCall.proxy = spy;
966
+ proxyCall.thisValue = thisValue;
967
+ proxyCall.args = args;
968
+ proxyCall.returnValue = returnValue;
969
+ proxyCall.exception = exception;
970
+ proxyCall.callId = typeof id == "number" && id || callId++;
971
+
972
+ return proxyCall;
973
+ },
974
+
975
+ calledOn: function calledOn(thisValue) {
976
+ return this.thisValue === thisValue;
977
+ },
978
+
979
+ calledWith: function calledWith() {
980
+ for (var i = 0, l = arguments.length; i < l; i += 1) {
981
+ if (!sinon.deepEqual(arguments[i], this.args[i])) {
982
+ return false;
983
+ }
984
+ }
985
+
986
+ return true;
987
+ },
988
+
989
+ calledWithExactly: function calledWithExactly() {
990
+ return arguments.length == this.args.length &&
991
+ this.calledWith.apply(this, arguments);
992
+ },
993
+
994
+ notCalledWith: function notCalledWith() {
995
+ for (var i = 0, l = arguments.length; i < l; i += 1) {
996
+ if (!sinon.deepEqual(arguments[i], this.args[i])) {
997
+ return true;
998
+ }
999
+ }
1000
+ return false;
1001
+ },
1002
+
1003
+ returned: function returned(value) {
1004
+ return this.returnValue === value;
1005
+ },
1006
+
1007
+ threw: function threw(error) {
1008
+ if (typeof error == "undefined" || !this.exception) {
1009
+ return !!this.exception;
1010
+ }
1011
+
1012
+ if (typeof error == "string") {
1013
+ return this.exception.name == error;
1014
+ }
1015
+
1016
+ return this.exception === error;
1017
+ },
1018
+
1019
+ calledWithNew: function calledWithNew(thisValue) {
1020
+ return this.thisValue instanceof this.proxy;
1021
+ },
1022
+
1023
+ calledBefore: function (other) {
1024
+ return this.callId < other.callId;
1025
+ },
1026
+
1027
+ calledAfter: function (other) {
1028
+ return this.callId > other.callId;
1029
+ },
1030
+
1031
+ callArg: function (pos) {
1032
+ this.args[pos]();
1033
+ },
1034
+
1035
+ callArgWith: function (pos) {
1036
+ var args = slice.call(arguments, 1);
1037
+ this.args[pos].apply(null, args);
1038
+ },
1039
+
1040
+ "yield": function () {
1041
+ var args = this.args;
1042
+ for (var i = 0, l = args.length; i < l; ++i) {
1043
+ if (typeof args[i] === "function") {
1044
+ args[i].apply(null, slice.call(arguments));
1045
+ return;
1046
+ }
1047
+ }
1048
+ throwYieldError(this.proxy, " cannot yield since no callback was passed.", args);
1049
+ },
1050
+
1051
+ yieldTo: function (prop) {
1052
+ var args = this.args;
1053
+ for (var i = 0, l = args.length; i < l; ++i) {
1054
+ if (args[i] && typeof args[i][prop] === "function") {
1055
+ args[i][prop].apply(null, slice.call(arguments, 1));
1056
+ return;
1057
+ }
1058
+ }
1059
+ throwYieldError(this.proxy, " cannot yield to '" + prop +
1060
+ "' since no callback was passed.", args);
1061
+ },
1062
+
1063
+ toString: function () {
1064
+ var callStr = this.proxy.toString() + "(";
1065
+ var args = [];
1066
+
1067
+ for (var i = 0, l = this.args.length; i < l; ++i) {
1068
+ push.call(args, sinon.format(this.args[i]));
1069
+ }
1070
+
1071
+ callStr = callStr + args.join(", ") + ")";
1072
+
1073
+ if (typeof this.returnValue != "undefined") {
1074
+ callStr += " => " + sinon.format(this.returnValue);
1075
+ }
1076
+
1077
+ if (this.exception) {
1078
+ callStr += " !" + this.exception.name;
1079
+
1080
+ if (this.exception.message) {
1081
+ callStr += "(" + this.exception.message + ")";
1082
+ }
1083
+ }
1084
+
1085
+ return callStr;
1086
+ }
1087
+ };
1088
+ }());
1089
+
1090
+ spy.spyCall = spyCall;
1091
+
1092
+ // This steps outside the module sandbox and will be removed
1093
+ sinon.spyCall = spyCall;
1094
+
1095
+ if (commonJSModule) {
1096
+ module.exports = spy;
1097
+ } else {
1098
+ sinon.spy = spy;
1099
+ }
1100
+ }(typeof sinon == "object" && sinon || null));
1101
+
1102
+ /**
1103
+ * @depend ../sinon.js
1104
+ * @depend spy.js
1105
+ */
1106
+ /*jslint eqeqeq: false, onevar: false*/
1107
+ /*global module, require, sinon*/
1108
+ /**
1109
+ * Stub functions
1110
+ *
1111
+ * @author Christian Johansen (christian@cjohansen.no)
1112
+ * @license BSD
1113
+ *
1114
+ * Copyright (c) 2010-2011 Christian Johansen
1115
+ */
1116
+
1117
+ (function (sinon) {
1118
+ var commonJSModule = typeof module == "object" && typeof require == "function";
1119
+
1120
+ if (!sinon && commonJSModule) {
1121
+ sinon = require("../sinon");
1122
+ }
1123
+
1124
+ if (!sinon) {
1125
+ return;
1126
+ }
1127
+
1128
+ function stub(object, property, func) {
1129
+ if (!!func && typeof func != "function") {
1130
+ throw new TypeError("Custom stub should be function");
1131
+ }
1132
+
1133
+ var wrapper;
1134
+
1135
+ if (func) {
1136
+ wrapper = sinon.spy && sinon.spy.create ? sinon.spy.create(func) : func;
1137
+ } else {
1138
+ wrapper = stub.create();
1139
+ }
1140
+
1141
+ if (!object && !property) {
1142
+ return sinon.stub.create();
1143
+ }
1144
+
1145
+ if (!property && !!object && typeof object == "object") {
1146
+ for (var prop in object) {
1147
+ if (typeof object[prop] === "function") {
1148
+ stub(object, prop);
1149
+ }
1150
+ }
1151
+
1152
+ return object;
1153
+ }
1154
+
1155
+ return sinon.wrapMethod(object, property, wrapper);
1156
+ }
1157
+
1158
+ function getCallback(stub, args) {
1159
+ if (stub.callArgAt < 0) {
1160
+ for (var i = 0, l = args.length; i < l; ++i) {
1161
+ if (!stub.callArgProp && typeof args[i] == "function") {
1162
+ return args[i];
1163
+ }
1164
+
1165
+ if (stub.callArgProp && args[i] &&
1166
+ typeof args[i][stub.callArgProp] == "function") {
1167
+ return args[i][stub.callArgProp];
1168
+ }
1169
+ }
1170
+
1171
+ return null;
1172
+ }
1173
+
1174
+ return args[stub.callArgAt];
1175
+ }
1176
+
1177
+ var join = Array.prototype.join;
1178
+
1179
+ function getCallbackError(stub, func, args) {
1180
+ if (stub.callArgAt < 0) {
1181
+ var msg;
1182
+
1183
+ if (stub.callArgProp) {
1184
+ msg = sinon.functionName(stub) +
1185
+ " expected to yield to '" + stub.callArgProp +
1186
+ "', but no object with such a property was passed."
1187
+ } else {
1188
+ msg = sinon.functionName(stub) +
1189
+ " expected to yield, but no callback was passed."
1190
+ }
1191
+
1192
+ if (args.length > 0) {
1193
+ msg += " Received [" + join.call(args, ", ") + "]";
1194
+ }
1195
+
1196
+ return msg;
1197
+ }
1198
+
1199
+ return "argument at index " + stub.callArgAt + " is not a function: " + func;
1200
+ }
1201
+
1202
+ function callCallback(stub, args) {
1203
+ if (typeof stub.callArgAt == "number") {
1204
+ var func = getCallback(stub, args);
1205
+
1206
+ if (typeof func != "function") {
1207
+ throw new TypeError(getCallbackError(stub, func, args));
1208
+ }
1209
+
1210
+ func.apply(stub.callbackContext, stub.callbackArguments);
1211
+ }
1212
+ }
1213
+
1214
+ var uuid = 0;
1215
+
1216
+ sinon.extend(stub, (function () {
1217
+ var slice = Array.prototype.slice;
1218
+
1219
+ function throwsException(error, message) {
1220
+ if (typeof error == "string") {
1221
+ this.exception = new Error(message || "");
1222
+ this.exception.name = error;
1223
+ } else if (!error) {
1224
+ this.exception = new Error("Error");
1225
+ } else {
1226
+ this.exception = error;
1227
+ }
1228
+
1229
+ return this;
1230
+ }
1231
+
1232
+ return {
1233
+ create: function create() {
1234
+ var functionStub = function () {
1235
+ if (functionStub.exception) {
1236
+ throw functionStub.exception;
1237
+ } else if (typeof functionStub.returnArgAt == 'number') {
1238
+ return arguments[functionStub.returnArgAt];
1239
+ }
1240
+
1241
+ callCallback(functionStub, arguments);
1242
+
1243
+ return functionStub.returnValue;
1244
+ };
1245
+
1246
+ functionStub.id = "stub#" + uuid++;
1247
+ var orig = functionStub;
1248
+ functionStub = sinon.spy.create(functionStub);
1249
+ functionStub.func = orig;
1250
+
1251
+ sinon.extend(functionStub, stub);
1252
+ functionStub._create = sinon.stub.create;
1253
+ functionStub.displayName = "stub";
1254
+ functionStub.toString = sinon.functionToString;
1255
+
1256
+ return functionStub;
1257
+ },
1258
+
1259
+ returns: function returns(value) {
1260
+ this.returnValue = value;
1261
+
1262
+ return this;
1263
+ },
1264
+
1265
+ returnsArg: function returnsArg(pos) {
1266
+ if (typeof pos != "number") {
1267
+ throw new TypeError("argument index is not number");
1268
+ }
1269
+
1270
+ this.returnArgAt = pos;
1271
+
1272
+ return this;
1273
+ },
1274
+
1275
+ "throws": throwsException,
1276
+ throwsException: throwsException,
1277
+
1278
+ callsArg: function callsArg(pos) {
1279
+ if (typeof pos != "number") {
1280
+ throw new TypeError("argument index is not number");
1281
+ }
1282
+
1283
+ this.callArgAt = pos;
1284
+ this.callbackArguments = [];
1285
+
1286
+ return this;
1287
+ },
1288
+
1289
+ callsArgOn: function callsArgOn(pos, context) {
1290
+ if (typeof pos != "number") {
1291
+ throw new TypeError("argument index is not number");
1292
+ }
1293
+ if (typeof context != "object") {
1294
+ throw new TypeError("argument context is not an object");
1295
+ }
1296
+
1297
+ this.callArgAt = pos;
1298
+ this.callbackArguments = [];
1299
+ this.callbackContext = context;
1300
+
1301
+ return this;
1302
+ },
1303
+
1304
+ callsArgWith: function callsArgWith(pos) {
1305
+ if (typeof pos != "number") {
1306
+ throw new TypeError("argument index is not number");
1307
+ }
1308
+
1309
+ this.callArgAt = pos;
1310
+ this.callbackArguments = slice.call(arguments, 1);
1311
+
1312
+ return this;
1313
+ },
1314
+
1315
+ callsArgOnWith: function callsArgWith(pos, context) {
1316
+ if (typeof pos != "number") {
1317
+ throw new TypeError("argument index is not number");
1318
+ }
1319
+ if (typeof context != "object") {
1320
+ throw new TypeError("argument context is not an object");
1321
+ }
1322
+
1323
+ this.callArgAt = pos;
1324
+ this.callbackArguments = slice.call(arguments, 2);
1325
+ this.callbackContext = context;
1326
+
1327
+ return this;
1328
+ },
1329
+
1330
+ yields: function () {
1331
+ this.callArgAt = -1;
1332
+ this.callbackArguments = slice.call(arguments, 0);
1333
+
1334
+ return this;
1335
+ },
1336
+
1337
+ yieldsOn: function (context) {
1338
+ if (typeof context != "object") {
1339
+ throw new TypeError("argument context is not an object");
1340
+ }
1341
+
1342
+ this.callArgAt = -1;
1343
+ this.callbackArguments = slice.call(arguments, 1);
1344
+ this.callbackContext = context;
1345
+
1346
+ return this;
1347
+ },
1348
+
1349
+ yieldsTo: function (prop) {
1350
+ this.callArgAt = -1;
1351
+ this.callArgProp = prop;
1352
+ this.callbackArguments = slice.call(arguments, 1);
1353
+
1354
+ return this;
1355
+ },
1356
+
1357
+ yieldsToOn: function (prop, context) {
1358
+ if (typeof context != "object") {
1359
+ throw new TypeError("argument context is not an object");
1360
+ }
1361
+
1362
+ this.callArgAt = -1;
1363
+ this.callArgProp = prop;
1364
+ this.callbackArguments = slice.call(arguments, 2);
1365
+ this.callbackContext = context;
1366
+
1367
+ return this;
1368
+ }
1369
+ };
1370
+ }()));
1371
+
1372
+ if (commonJSModule) {
1373
+ module.exports = stub;
1374
+ } else {
1375
+ sinon.stub = stub;
1376
+ }
1377
+ }(typeof sinon == "object" && sinon || null));
1378
+
1379
+ /**
1380
+ * @depend ../sinon.js
1381
+ * @depend stub.js
1382
+ */
1383
+ /*jslint eqeqeq: false, onevar: false, nomen: false*/
1384
+ /*global module, require, sinon*/
1385
+ /**
1386
+ * Mock functions.
1387
+ *
1388
+ * @author Christian Johansen (christian@cjohansen.no)
1389
+ * @license BSD
1390
+ *
1391
+ * Copyright (c) 2010-2011 Christian Johansen
1392
+ */
1393
+
1394
+ (function (sinon) {
1395
+ var commonJSModule = typeof module == "object" && typeof require == "function";
1396
+ var push = [].push;
1397
+
1398
+ if (!sinon && commonJSModule) {
1399
+ sinon = require("../sinon");
1400
+ }
1401
+
1402
+ if (!sinon) {
1403
+ return;
1404
+ }
1405
+
1406
+ function mock(object) {
1407
+ if (!object) {
1408
+ return sinon.expectation.create("Anonymous mock");
1409
+ }
1410
+
1411
+ return mock.create(object);
1412
+ }
1413
+
1414
+ sinon.mock = mock;
1415
+
1416
+ sinon.extend(mock, (function () {
1417
+ function each(collection, callback) {
1418
+ if (!collection) {
1419
+ return;
1420
+ }
1421
+
1422
+ for (var i = 0, l = collection.length; i < l; i += 1) {
1423
+ callback(collection[i]);
1424
+ }
1425
+ }
1426
+
1427
+ return {
1428
+ create: function create(object) {
1429
+ if (!object) {
1430
+ throw new TypeError("object is null");
1431
+ }
1432
+
1433
+ var mockObject = sinon.extend({}, mock);
1434
+ mockObject.object = object;
1435
+ delete mockObject.create;
1436
+
1437
+ return mockObject;
1438
+ },
1439
+
1440
+ expects: function expects(method) {
1441
+ if (!method) {
1442
+ throw new TypeError("method is falsy");
1443
+ }
1444
+
1445
+ if (!this.expectations) {
1446
+ this.expectations = {};
1447
+ this.proxies = [];
1448
+ }
1449
+
1450
+ if (!this.expectations[method]) {
1451
+ this.expectations[method] = [];
1452
+ var mockObject = this;
1453
+
1454
+ sinon.wrapMethod(this.object, method, function () {
1455
+ return mockObject.invokeMethod(method, this, arguments);
1456
+ });
1457
+
1458
+ push.call(this.proxies, method);
1459
+ }
1460
+
1461
+ var expectation = sinon.expectation.create(method);
1462
+ push.call(this.expectations[method], expectation);
1463
+
1464
+ return expectation;
1465
+ },
1466
+
1467
+ restore: function restore() {
1468
+ var object = this.object;
1469
+
1470
+ each(this.proxies, function (proxy) {
1471
+ if (typeof object[proxy].restore == "function") {
1472
+ object[proxy].restore();
1473
+ }
1474
+ });
1475
+ },
1476
+
1477
+ verify: function verify() {
1478
+ var expectations = this.expectations || {};
1479
+ var messages = [], met = [];
1480
+
1481
+ each(this.proxies, function (proxy) {
1482
+ each(expectations[proxy], function (expectation) {
1483
+ if (!expectation.met()) {
1484
+ push.call(messages, expectation.toString());
1485
+ } else {
1486
+ push.call(met, expectation.toString());
1487
+ }
1488
+ });
1489
+ });
1490
+
1491
+ this.restore();
1492
+
1493
+ if (messages.length > 0) {
1494
+ sinon.expectation.fail(messages.concat(met).join("\n"));
1495
+ }
1496
+
1497
+ return true;
1498
+ },
1499
+
1500
+ invokeMethod: function invokeMethod(method, thisValue, args) {
1501
+ var expectations = this.expectations && this.expectations[method];
1502
+ var length = expectations && expectations.length || 0;
1503
+
1504
+ for (var i = 0; i < length; i += 1) {
1505
+ if (!expectations[i].met() &&
1506
+ expectations[i].allowsCall(thisValue, args)) {
1507
+ return expectations[i].apply(thisValue, args);
1508
+ }
1509
+ }
1510
+
1511
+ var messages = [];
1512
+
1513
+ for (i = 0; i < length; i += 1) {
1514
+ push.call(messages, " " + expectations[i].toString());
1515
+ }
1516
+
1517
+ messages.unshift("Unexpected call: " + sinon.spyCall.toString.call({
1518
+ proxy: method,
1519
+ args: args
1520
+ }));
1521
+
1522
+ sinon.expectation.fail(messages.join("\n"));
1523
+ }
1524
+ };
1525
+ }()));
1526
+
1527
+ var times = sinon.timesInWords;
1528
+
1529
+ sinon.expectation = (function () {
1530
+ var slice = Array.prototype.slice;
1531
+ var _invoke = sinon.spy.invoke;
1532
+
1533
+ function callCountInWords(callCount) {
1534
+ if (callCount == 0) {
1535
+ return "never called";
1536
+ } else {
1537
+ return "called " + times(callCount);
1538
+ }
1539
+ }
1540
+
1541
+ function expectedCallCountInWords(expectation) {
1542
+ var min = expectation.minCalls;
1543
+ var max = expectation.maxCalls;
1544
+
1545
+ if (typeof min == "number" && typeof max == "number") {
1546
+ var str = times(min);
1547
+
1548
+ if (min != max) {
1549
+ str = "at least " + str + " and at most " + times(max);
1550
+ }
1551
+
1552
+ return str;
1553
+ }
1554
+
1555
+ if (typeof min == "number") {
1556
+ return "at least " + times(min);
1557
+ }
1558
+
1559
+ return "at most " + times(max);
1560
+ }
1561
+
1562
+ function receivedMinCalls(expectation) {
1563
+ var hasMinLimit = typeof expectation.minCalls == "number";
1564
+ return !hasMinLimit || expectation.callCount >= expectation.minCalls;
1565
+ }
1566
+
1567
+ function receivedMaxCalls(expectation) {
1568
+ if (typeof expectation.maxCalls != "number") {
1569
+ return false;
1570
+ }
1571
+
1572
+ return expectation.callCount == expectation.maxCalls;
1573
+ }
1574
+
1575
+ return {
1576
+ minCalls: 1,
1577
+ maxCalls: 1,
1578
+
1579
+ create: function create(methodName) {
1580
+ var expectation = sinon.extend(sinon.stub.create(), sinon.expectation);
1581
+ delete expectation.create;
1582
+ expectation.method = methodName;
1583
+
1584
+ return expectation;
1585
+ },
1586
+
1587
+ invoke: function invoke(func, thisValue, args) {
1588
+ this.verifyCallAllowed(thisValue, args);
1589
+
1590
+ return _invoke.apply(this, arguments);
1591
+ },
1592
+
1593
+ atLeast: function atLeast(num) {
1594
+ if (typeof num != "number") {
1595
+ throw new TypeError("'" + num + "' is not number");
1596
+ }
1597
+
1598
+ if (!this.limitsSet) {
1599
+ this.maxCalls = null;
1600
+ this.limitsSet = true;
1601
+ }
1602
+
1603
+ this.minCalls = num;
1604
+
1605
+ return this;
1606
+ },
1607
+
1608
+ atMost: function atMost(num) {
1609
+ if (typeof num != "number") {
1610
+ throw new TypeError("'" + num + "' is not number");
1611
+ }
1612
+
1613
+ if (!this.limitsSet) {
1614
+ this.minCalls = null;
1615
+ this.limitsSet = true;
1616
+ }
1617
+
1618
+ this.maxCalls = num;
1619
+
1620
+ return this;
1621
+ },
1622
+
1623
+ never: function never() {
1624
+ return this.exactly(0);
1625
+ },
1626
+
1627
+ once: function once() {
1628
+ return this.exactly(1);
1629
+ },
1630
+
1631
+ twice: function twice() {
1632
+ return this.exactly(2);
1633
+ },
1634
+
1635
+ thrice: function thrice() {
1636
+ return this.exactly(3);
1637
+ },
1638
+
1639
+ exactly: function exactly(num) {
1640
+ if (typeof num != "number") {
1641
+ throw new TypeError("'" + num + "' is not a number");
1642
+ }
1643
+
1644
+ this.atLeast(num);
1645
+ return this.atMost(num);
1646
+ },
1647
+
1648
+ met: function met() {
1649
+ return !this.failed && receivedMinCalls(this);
1650
+ },
1651
+
1652
+ verifyCallAllowed: function verifyCallAllowed(thisValue, args) {
1653
+ if (receivedMaxCalls(this)) {
1654
+ this.failed = true;
1655
+ sinon.expectation.fail(this.method + " already called " + times(this.maxCalls));
1656
+ }
1657
+
1658
+ if ("expectedThis" in this && this.expectedThis !== thisValue) {
1659
+ sinon.expectation.fail(this.method + " called with " + thisValue + " as thisValue, expected " +
1660
+ this.expectedThis);
1661
+ }
1662
+
1663
+ if (!("expectedArguments" in this)) {
1664
+ return;
1665
+ }
1666
+
1667
+ if (!args || args.length === 0) {
1668
+ sinon.expectation.fail(this.method + " received no arguments, expected " +
1669
+ this.expectedArguments.join());
1670
+ }
1671
+
1672
+ if (args.length < this.expectedArguments.length) {
1673
+ sinon.expectation.fail(this.method + " received too few arguments (" + args.join() +
1674
+ "), expected " + this.expectedArguments.join());
1675
+ }
1676
+
1677
+ if (this.expectsExactArgCount &&
1678
+ args.length != this.expectedArguments.length) {
1679
+ sinon.expectation.fail(this.method + " received too many arguments (" + args.join() +
1680
+ "), expected " + this.expectedArguments.join());
1681
+ }
1682
+
1683
+ for (var i = 0, l = this.expectedArguments.length; i < l; i += 1) {
1684
+ if (!sinon.deepEqual(this.expectedArguments[i], args[i])) {
1685
+ sinon.expectation.fail(this.method + " received wrong arguments (" + args.join() +
1686
+ "), expected " + this.expectedArguments.join());
1687
+ }
1688
+ }
1689
+ },
1690
+
1691
+ allowsCall: function allowsCall(thisValue, args) {
1692
+ if (this.met()) {
1693
+ return false;
1694
+ }
1695
+
1696
+ if ("expectedThis" in this && this.expectedThis !== thisValue) {
1697
+ return false;
1698
+ }
1699
+
1700
+ if (!("expectedArguments" in this)) {
1701
+ return true;
1702
+ }
1703
+
1704
+ args = args || [];
1705
+
1706
+ if (args.length < this.expectedArguments.length) {
1707
+ return false;
1708
+ }
1709
+
1710
+ if (this.expectsExactArgCount &&
1711
+ args.length != this.expectedArguments.length) {
1712
+ return false;
1713
+ }
1714
+
1715
+ for (var i = 0, l = this.expectedArguments.length; i < l; i += 1) {
1716
+ if (!sinon.deepEqual(this.expectedArguments[i], args[i])) {
1717
+ return false;
1718
+ }
1719
+ }
1720
+
1721
+ return true;
1722
+ },
1723
+
1724
+ withArgs: function withArgs() {
1725
+ this.expectedArguments = slice.call(arguments);
1726
+ return this;
1727
+ },
1728
+
1729
+ withExactArgs: function withExactArgs() {
1730
+ this.withArgs.apply(this, arguments);
1731
+ this.expectsExactArgCount = true;
1732
+ return this;
1733
+ },
1734
+
1735
+ on: function on(thisValue) {
1736
+ this.expectedThis = thisValue;
1737
+ return this;
1738
+ },
1739
+
1740
+ toString: function () {
1741
+ var args = (this.expectedArguments || []).slice();
1742
+
1743
+ if (!this.expectsExactArgCount) {
1744
+ push.call(args, "[...]");
1745
+ }
1746
+
1747
+ var callStr = sinon.spyCall.toString.call({
1748
+ proxy: this.method, args: args
1749
+ });
1750
+
1751
+ var message = callStr.replace(", [...", "[, ...") + " " +
1752
+ expectedCallCountInWords(this);
1753
+
1754
+ if (this.met()) {
1755
+ return "Expectation met: " + message;
1756
+ }
1757
+
1758
+ return "Expected " + message + " (" +
1759
+ callCountInWords(this.callCount) + ")";
1760
+ },
1761
+
1762
+ verify: function verify() {
1763
+ if (!this.met()) {
1764
+ sinon.expectation.fail(this.toString());
1765
+ }
1766
+
1767
+ return true;
1768
+ },
1769
+
1770
+ fail: function (message) {
1771
+ var exception = new Error(message);
1772
+ exception.name = "ExpectationError";
1773
+
1774
+ throw exception;
1775
+ }
1776
+ };
1777
+ }());
1778
+
1779
+ if (commonJSModule) {
1780
+ module.exports = mock;
1781
+ } else {
1782
+ sinon.mock = mock;
1783
+ }
1784
+ }(typeof sinon == "object" && sinon || null));
1785
+
1786
+ /**
1787
+ * @depend ../sinon.js
1788
+ * @depend stub.js
1789
+ * @depend mock.js
1790
+ */
1791
+ /*jslint eqeqeq: false, onevar: false, forin: true*/
1792
+ /*global module, require, sinon*/
1793
+ /**
1794
+ * Collections of stubs, spies and mocks.
1795
+ *
1796
+ * @author Christian Johansen (christian@cjohansen.no)
1797
+ * @license BSD
1798
+ *
1799
+ * Copyright (c) 2010-2011 Christian Johansen
1800
+ */
1801
+
1802
+ (function (sinon) {
1803
+ var commonJSModule = typeof module == "object" && typeof require == "function";
1804
+ var push = [].push;
1805
+
1806
+ if (!sinon && commonJSModule) {
1807
+ sinon = require("../sinon");
1808
+ }
1809
+
1810
+ if (!sinon) {
1811
+ return;
1812
+ }
1813
+
1814
+ function getFakes(fakeCollection) {
1815
+ if (!fakeCollection.fakes) {
1816
+ fakeCollection.fakes = [];
1817
+ }
1818
+
1819
+ return fakeCollection.fakes;
1820
+ }
1821
+
1822
+ function each(fakeCollection, method) {
1823
+ var fakes = getFakes(fakeCollection);
1824
+
1825
+ for (var i = 0, l = fakes.length; i < l; i += 1) {
1826
+ if (typeof fakes[i][method] == "function") {
1827
+ fakes[i][method]();
1828
+ }
1829
+ }
1830
+ }
1831
+
1832
+ function compact(fakeCollection) {
1833
+ var fakes = getFakes(fakeCollection);
1834
+ var i = 0;
1835
+ while (i < fakes.length) {
1836
+ fakes.splice(i, 1);
1837
+ }
1838
+ }
1839
+
1840
+ var collection = {
1841
+ verify: function resolve() {
1842
+ each(this, "verify");
1843
+ },
1844
+
1845
+ restore: function restore() {
1846
+ each(this, "restore");
1847
+ compact(this);
1848
+ },
1849
+
1850
+ verifyAndRestore: function verifyAndRestore() {
1851
+ var exception;
1852
+
1853
+ try {
1854
+ this.verify();
1855
+ } catch (e) {
1856
+ exception = e;
1857
+ }
1858
+
1859
+ this.restore();
1860
+
1861
+ if (exception) {
1862
+ throw exception;
1863
+ }
1864
+ },
1865
+
1866
+ add: function add(fake) {
1867
+ push.call(getFakes(this), fake);
1868
+ return fake;
1869
+ },
1870
+
1871
+ spy: function spy() {
1872
+ return this.add(sinon.spy.apply(sinon, arguments));
1873
+ },
1874
+
1875
+ stub: function stub(object, property, value) {
1876
+ if (property) {
1877
+ var original = object[property];
1878
+
1879
+ if (typeof original != "function") {
1880
+ if (!object.hasOwnProperty(property)) {
1881
+ throw new TypeError("Cannot stub non-existent own property " + property);
1882
+ }
1883
+
1884
+ object[property] = value;
1885
+
1886
+ return this.add({
1887
+ restore: function () {
1888
+ object[property] = original;
1889
+ }
1890
+ });
1891
+ }
1892
+ }
1893
+
1894
+ return this.add(sinon.stub.apply(sinon, arguments));
1895
+ },
1896
+
1897
+ mock: function mock() {
1898
+ return this.add(sinon.mock.apply(sinon, arguments));
1899
+ },
1900
+
1901
+ inject: function inject(obj) {
1902
+ var col = this;
1903
+
1904
+ obj.spy = function () {
1905
+ return col.spy.apply(col, arguments);
1906
+ };
1907
+
1908
+ obj.stub = function () {
1909
+ return col.stub.apply(col, arguments);
1910
+ };
1911
+
1912
+ obj.mock = function () {
1913
+ return col.mock.apply(col, arguments);
1914
+ };
1915
+
1916
+ return obj;
1917
+ }
1918
+ };
1919
+
1920
+ if (commonJSModule) {
1921
+ module.exports = collection;
1922
+ } else {
1923
+ sinon.collection = collection;
1924
+ }
1925
+ }(typeof sinon == "object" && sinon || null));
1926
+
1927
+ /*jslint eqeqeq: false, plusplus: false, evil: true, onevar: false, browser: true, forin: false*/
1928
+ /*global module, require, window*/
1929
+ /**
1930
+ * Fake timer API
1931
+ * setTimeout
1932
+ * setInterval
1933
+ * clearTimeout
1934
+ * clearInterval
1935
+ * tick
1936
+ * reset
1937
+ * Date
1938
+ *
1939
+ * Inspired by jsUnitMockTimeOut from JsUnit
1940
+ *
1941
+ * @author Christian Johansen (christian@cjohansen.no)
1942
+ * @license BSD
1943
+ *
1944
+ * Copyright (c) 2010-2011 Christian Johansen
1945
+ */
1946
+
1947
+ if (typeof sinon == "undefined") {
1948
+ var sinon = {};
1949
+ }
1950
+
1951
+ (function (global) {
1952
+ var id = 1;
1953
+
1954
+ function addTimer(args, recurring) {
1955
+ if (args.length === 0) {
1956
+ throw new Error("Function requires at least 1 parameter");
1957
+ }
1958
+
1959
+ var toId = id++;
1960
+ var delay = args[1] || 0;
1961
+
1962
+ if (!this.timeouts) {
1963
+ this.timeouts = {};
1964
+ }
1965
+
1966
+ this.timeouts[toId] = {
1967
+ id: toId,
1968
+ func: args[0],
1969
+ callAt: this.now + delay
1970
+ };
1971
+
1972
+ if (recurring === true) {
1973
+ this.timeouts[toId].interval = delay;
1974
+ }
1975
+
1976
+ return toId;
1977
+ }
1978
+
1979
+ function parseTime(str) {
1980
+ if (!str) {
1981
+ return 0;
1982
+ }
1983
+
1984
+ var strings = str.split(":");
1985
+ var l = strings.length, i = l;
1986
+ var ms = 0, parsed;
1987
+
1988
+ if (l > 3 || !/^(\d\d:){0,2}\d\d?$/.test(str)) {
1989
+ throw new Error("tick only understands numbers and 'h:m:s'");
1990
+ }
1991
+
1992
+ while (i--) {
1993
+ parsed = parseInt(strings[i], 10);
1994
+
1995
+ if (parsed >= 60) {
1996
+ throw new Error("Invalid time " + str);
1997
+ }
1998
+
1999
+ ms += parsed * Math.pow(60, (l - i - 1));
2000
+ }
2001
+
2002
+ return ms * 1000;
2003
+ }
2004
+
2005
+ function createObject(object) {
2006
+ var newObject;
2007
+
2008
+ if (Object.create) {
2009
+ newObject = Object.create(object);
2010
+ } else {
2011
+ var F = function () {};
2012
+ F.prototype = object;
2013
+ newObject = new F();
2014
+ }
2015
+
2016
+ newObject.Date.clock = newObject;
2017
+ return newObject;
2018
+ }
2019
+
2020
+ sinon.clock = {
2021
+ now: 0,
2022
+
2023
+ create: function create(now) {
2024
+ var clock = createObject(this);
2025
+
2026
+ if (typeof now == "number") {
2027
+ clock.now = now;
2028
+ }
2029
+
2030
+ if (!!now && typeof now == "object") {
2031
+ throw new TypeError("now should be milliseconds since UNIX epoch");
2032
+ }
2033
+
2034
+ return clock;
2035
+ },
2036
+
2037
+ setTimeout: function setTimeout(callback, timeout) {
2038
+ return addTimer.call(this, arguments, false);
2039
+ },
2040
+
2041
+ clearTimeout: function clearTimeout(timerId) {
2042
+ if (!this.timeouts) {
2043
+ this.timeouts = [];
2044
+ }
2045
+
2046
+ if (timerId in this.timeouts) {
2047
+ delete this.timeouts[timerId];
2048
+ }
2049
+ },
2050
+
2051
+ setInterval: function setInterval(callback, timeout) {
2052
+ return addTimer.call(this, arguments, true);
2053
+ },
2054
+
2055
+ clearInterval: function clearInterval(timerId) {
2056
+ this.clearTimeout(timerId);
2057
+ },
2058
+
2059
+ tick: function tick(ms) {
2060
+ ms = typeof ms == "number" ? ms : parseTime(ms);
2061
+ var tickFrom = this.now, tickTo = this.now + ms, previous = this.now;
2062
+ var timer = this.firstTimerInRange(tickFrom, tickTo);
2063
+
2064
+ var firstException;
2065
+ while (timer && tickFrom <= tickTo) {
2066
+ if (this.timeouts[timer.id]) {
2067
+ tickFrom = this.now = timer.callAt;
2068
+ try {
2069
+ this.callTimer(timer);
2070
+ } catch (e) {
2071
+ firstException = firstException || e;
2072
+ }
2073
+ }
2074
+
2075
+ timer = this.firstTimerInRange(previous, tickTo);
2076
+ previous = tickFrom;
2077
+ }
2078
+
2079
+ this.now = tickTo;
2080
+
2081
+ if (firstException) {
2082
+ throw firstException;
2083
+ }
2084
+ },
2085
+
2086
+ firstTimerInRange: function (from, to) {
2087
+ var timer, smallest, originalTimer;
2088
+
2089
+ for (var id in this.timeouts) {
2090
+ if (this.timeouts.hasOwnProperty(id)) {
2091
+ if (this.timeouts[id].callAt < from || this.timeouts[id].callAt > to) {
2092
+ continue;
2093
+ }
2094
+
2095
+ if (!smallest || this.timeouts[id].callAt < smallest) {
2096
+ originalTimer = this.timeouts[id];
2097
+ smallest = this.timeouts[id].callAt;
2098
+
2099
+ timer = {
2100
+ func: this.timeouts[id].func,
2101
+ callAt: this.timeouts[id].callAt,
2102
+ interval: this.timeouts[id].interval,
2103
+ id: this.timeouts[id].id
2104
+ };
2105
+ }
2106
+ }
2107
+ }
2108
+
2109
+ return timer || null;
2110
+ },
2111
+
2112
+ callTimer: function (timer) {
2113
+ try {
2114
+ if (typeof timer.func == "function") {
2115
+ timer.func.call(null);
2116
+ } else {
2117
+ eval(timer.func);
2118
+ }
2119
+ } catch (e) {
2120
+ var exception = e;
2121
+ }
2122
+
2123
+ if (!this.timeouts[timer.id]) {
2124
+ if (exception) {
2125
+ throw exception;
2126
+ }
2127
+ return;
2128
+ }
2129
+
2130
+ if (typeof timer.interval == "number") {
2131
+ this.timeouts[timer.id].callAt += timer.interval;
2132
+ } else {
2133
+ delete this.timeouts[timer.id];
2134
+ }
2135
+
2136
+ if (exception) {
2137
+ throw exception;
2138
+ }
2139
+ },
2140
+
2141
+ reset: function reset() {
2142
+ this.timeouts = {};
2143
+ },
2144
+
2145
+ Date: (function () {
2146
+ var NativeDate = Date;
2147
+
2148
+ function ClockDate(year, month, date, hour, minute, second, ms) {
2149
+ // Defensive and verbose to avoid potential harm in passing
2150
+ // explicit undefined when user does not pass argument
2151
+ switch (arguments.length) {
2152
+ case 0:
2153
+ return new NativeDate(ClockDate.clock.now);
2154
+ case 1:
2155
+ return new NativeDate(year);
2156
+ case 2:
2157
+ return new NativeDate(year, month);
2158
+ case 3:
2159
+ return new NativeDate(year, month, date);
2160
+ case 4:
2161
+ return new NativeDate(year, month, date, hour);
2162
+ case 5:
2163
+ return new NativeDate(year, month, date, hour, minute);
2164
+ case 6:
2165
+ return new NativeDate(year, month, date, hour, minute, second);
2166
+ default:
2167
+ return new NativeDate(year, month, date, hour, minute, second, ms);
2168
+ }
2169
+ }
2170
+
2171
+ return mirrorDateProperties(ClockDate, NativeDate);
2172
+ }())
2173
+ };
2174
+
2175
+ function mirrorDateProperties(target, source) {
2176
+ if (source.now) {
2177
+ target.now = function now() {
2178
+ return target.clock.now;
2179
+ };
2180
+ } else {
2181
+ delete target.now;
2182
+ }
2183
+
2184
+ if (source.toSource) {
2185
+ target.toSource = function toSource() {
2186
+ return source.toSource();
2187
+ };
2188
+ } else {
2189
+ delete target.toSource;
2190
+ }
2191
+
2192
+ target.toString = function toString() {
2193
+ return source.toString();
2194
+ };
2195
+
2196
+ target.prototype = source.prototype;
2197
+ target.parse = source.parse;
2198
+ target.UTC = source.UTC;
2199
+ target.prototype.toUTCString = source.prototype.toUTCString;
2200
+ return target;
2201
+ }
2202
+
2203
+ var methods = ["Date", "setTimeout", "setInterval",
2204
+ "clearTimeout", "clearInterval"];
2205
+
2206
+ function restore() {
2207
+ var method;
2208
+
2209
+ for (var i = 0, l = this.methods.length; i < l; i++) {
2210
+ method = this.methods[i];
2211
+ global[method] = this["_" + method];
2212
+ }
2213
+ }
2214
+
2215
+ function stubGlobal(method, clock) {
2216
+ clock["_" + method] = global[method];
2217
+
2218
+ if (method == "Date") {
2219
+ var date = mirrorDateProperties(clock[method], global[method]);
2220
+ global[method] = date;
2221
+ } else {
2222
+ global[method] = function () {
2223
+ return clock[method].apply(clock, arguments);
2224
+ };
2225
+
2226
+ for (var prop in clock[method]) {
2227
+ if (clock[method].hasOwnProperty(prop)) {
2228
+ global[method][prop] = clock[method][prop];
2229
+ }
2230
+ }
2231
+ }
2232
+
2233
+ global[method].clock = clock;
2234
+ }
2235
+
2236
+ sinon.useFakeTimers = function useFakeTimers(now) {
2237
+ var clock = sinon.clock.create(now);
2238
+ clock.restore = restore;
2239
+ clock.methods = Array.prototype.slice.call(arguments,
2240
+ typeof now == "number" ? 1 : 0);
2241
+
2242
+ if (clock.methods.length === 0) {
2243
+ clock.methods = methods;
2244
+ }
2245
+
2246
+ for (var i = 0, l = clock.methods.length; i < l; i++) {
2247
+ stubGlobal(clock.methods[i], clock);
2248
+ }
2249
+
2250
+ return clock;
2251
+ };
2252
+ }(typeof global != "undefined" && typeof global !== "function" ? global : this));
2253
+
2254
+ sinon.timers = {
2255
+ setTimeout: setTimeout,
2256
+ clearTimeout: clearTimeout,
2257
+ setInterval: setInterval,
2258
+ clearInterval: clearInterval,
2259
+ Date: Date
2260
+ };
2261
+
2262
+ if (typeof module == "object" && typeof require == "function") {
2263
+ module.exports = sinon;
2264
+ }
2265
+
2266
+ /*jslint eqeqeq: false, onevar: false*/
2267
+ /*global sinon, module, require, ActiveXObject, XMLHttpRequest, DOMParser*/
2268
+ /**
2269
+ * Minimal Event interface implementation
2270
+ *
2271
+ * Original implementation by Sven Fuchs: https://gist.github.com/995028
2272
+ * Modifications and tests by Christian Johansen.
2273
+ *
2274
+ * @author Sven Fuchs (svenfuchs@artweb-design.de)
2275
+ * @author Christian Johansen (christian@cjohansen.no)
2276
+ * @license BSD
2277
+ *
2278
+ * Copyright (c) 2011 Sven Fuchs, Christian Johansen
2279
+ */
2280
+
2281
+ if (typeof sinon == "undefined") {
2282
+ this.sinon = {};
2283
+ }
2284
+
2285
+ (function () {
2286
+ var push = [].push;
2287
+
2288
+ sinon.Event = function Event(type, bubbles, cancelable) {
2289
+ this.initEvent(type, bubbles, cancelable);
2290
+ };
2291
+
2292
+ sinon.Event.prototype = {
2293
+ initEvent: function(type, bubbles, cancelable) {
2294
+ this.type = type;
2295
+ this.bubbles = bubbles;
2296
+ this.cancelable = cancelable;
2297
+ },
2298
+
2299
+ stopPropagation: function () {},
2300
+
2301
+ preventDefault: function () {
2302
+ this.defaultPrevented = true;
2303
+ }
2304
+ };
2305
+
2306
+ sinon.EventTarget = {
2307
+ addEventListener: function addEventListener(event, listener, useCapture) {
2308
+ this.eventListeners = this.eventListeners || {};
2309
+ this.eventListeners[event] = this.eventListeners[event] || [];
2310
+ push.call(this.eventListeners[event], listener);
2311
+ },
2312
+
2313
+ removeEventListener: function removeEventListener(event, listener, useCapture) {
2314
+ var listeners = this.eventListeners && this.eventListeners[event] || [];
2315
+
2316
+ for (var i = 0, l = listeners.length; i < l; ++i) {
2317
+ if (listeners[i] == listener) {
2318
+ return listeners.splice(i, 1);
2319
+ }
2320
+ }
2321
+ },
2322
+
2323
+ dispatchEvent: function dispatchEvent(event) {
2324
+ var type = event.type;
2325
+ var listeners = this.eventListeners && this.eventListeners[type] || [];
2326
+
2327
+ for (var i = 0; i < listeners.length; i++) {
2328
+ if (typeof listeners[i] == "function") {
2329
+ listeners[i].call(this, event);
2330
+ } else {
2331
+ listeners[i].handleEvent(event);
2332
+ }
2333
+ }
2334
+
2335
+ return !!event.defaultPrevented;
2336
+ }
2337
+ };
2338
+ }());
2339
+
2340
+ /**
2341
+ * @depend event.js
2342
+ */
2343
+ /*jslint eqeqeq: false, onevar: false*/
2344
+ /*global sinon, module, require, ActiveXObject, XMLHttpRequest, DOMParser*/
2345
+ /**
2346
+ * Fake XMLHttpRequest object
2347
+ *
2348
+ * @author Christian Johansen (christian@cjohansen.no)
2349
+ * @license BSD
2350
+ *
2351
+ * Copyright (c) 2010-2011 Christian Johansen
2352
+ */
2353
+
2354
+ if (typeof sinon == "undefined") {
2355
+ this.sinon = {};
2356
+ }
2357
+ sinon.xhr = { XMLHttpRequest: this.XMLHttpRequest };
2358
+
2359
+ // wrapper for global
2360
+ (function(global) {
2361
+ var xhr = sinon.xhr;
2362
+ xhr.GlobalXMLHttpRequest = global.XMLHttpRequest;
2363
+ xhr.GlobalActiveXObject = global.ActiveXObject;
2364
+ xhr.supportsActiveX = typeof xhr.GlobalActiveXObject != "undefined";
2365
+ xhr.supportsXHR = typeof xhr.GlobalXMLHttpRequest != "undefined";
2366
+ xhr.workingXHR = xhr.supportsXHR ? xhr.GlobalXMLHttpRequest : xhr.supportsActiveX
2367
+ ? function() { return new xhr.GlobalActiveXObject("MSXML2.XMLHTTP.3.0") } : false;
2368
+
2369
+ /*jsl:ignore*/
2370
+ var unsafeHeaders = {
2371
+ "Accept-Charset": true,
2372
+ "Accept-Encoding": true,
2373
+ "Connection": true,
2374
+ "Content-Length": true,
2375
+ "Cookie": true,
2376
+ "Cookie2": true,
2377
+ "Content-Transfer-Encoding": true,
2378
+ "Date": true,
2379
+ "Expect": true,
2380
+ "Host": true,
2381
+ "Keep-Alive": true,
2382
+ "Referer": true,
2383
+ "TE": true,
2384
+ "Trailer": true,
2385
+ "Transfer-Encoding": true,
2386
+ "Upgrade": true,
2387
+ "User-Agent": true,
2388
+ "Via": true
2389
+ };
2390
+ /*jsl:end*/
2391
+
2392
+ function FakeXMLHttpRequest() {
2393
+ this.readyState = FakeXMLHttpRequest.UNSENT;
2394
+ this.requestHeaders = {};
2395
+ this.requestBody = null;
2396
+ this.status = 0;
2397
+ this.statusText = "";
2398
+
2399
+ if (typeof FakeXMLHttpRequest.onCreate == "function") {
2400
+ FakeXMLHttpRequest.onCreate(this);
2401
+ }
2402
+ }
2403
+
2404
+ function verifyState(xhr) {
2405
+ if (xhr.readyState !== FakeXMLHttpRequest.OPENED) {
2406
+ throw new Error("INVALID_STATE_ERR");
2407
+ }
2408
+
2409
+ if (xhr.sendFlag) {
2410
+ throw new Error("INVALID_STATE_ERR");
2411
+ }
2412
+ }
2413
+
2414
+ // filtering to enable a white-list version of Sinon FakeXhr,
2415
+ // where whitelisted requests are passed through to real XHR
2416
+ function each(collection, callback) {
2417
+ if (!collection) return;
2418
+ for (var i = 0, l = collection.length; i < l; i += 1) {
2419
+ callback(collection[i]);
2420
+ }
2421
+ }
2422
+ function some(collection, callback) {
2423
+ for (var index = 0; index < collection.length; index++) {
2424
+ if(callback(collection[index]) === true) return true;
2425
+ };
2426
+ return false;
2427
+ }
2428
+ // largest arity in XHR is 5 - XHR#open
2429
+ var apply = function(obj,method,args) {
2430
+ switch(args.length) {
2431
+ case 0: return obj[method]();
2432
+ case 1: return obj[method](args[0]);
2433
+ case 2: return obj[method](args[0],args[1]);
2434
+ case 3: return obj[method](args[0],args[1],args[2]);
2435
+ case 4: return obj[method](args[0],args[1],args[2],args[3]);
2436
+ case 5: return obj[method](args[0],args[1],args[2],args[3],args[4]);
2437
+ };
2438
+ };
2439
+
2440
+ FakeXMLHttpRequest.filters = [];
2441
+ FakeXMLHttpRequest.addFilter = function(fn) {
2442
+ this.filters.push(fn)
2443
+ };
2444
+ var IE6Re = /MSIE 6/;
2445
+ FakeXMLHttpRequest.defake = function(fakeXhr,xhrArgs) {
2446
+ var xhr = new sinon.xhr.workingXHR();
2447
+ each(["open","setRequestHeader","send","abort","getResponseHeader",
2448
+ "getAllResponseHeaders","addEventListener","overrideMimeType","removeEventListener"],
2449
+ function(method) {
2450
+ fakeXhr[method] = function() {
2451
+ return apply(xhr,method,arguments);
2452
+ };
2453
+ });
2454
+
2455
+ var copyAttrs = function(args) {
2456
+ each(args, function(attr) {
2457
+ try {
2458
+ fakeXhr[attr] = xhr[attr]
2459
+ } catch(e) {
2460
+ if(!IE6Re.test(navigator.userAgent)) throw e;
2461
+ }
2462
+ });
2463
+ };
2464
+
2465
+ var stateChange = function() {
2466
+ fakeXhr.readyState = xhr.readyState;
2467
+ if(xhr.readyState >= FakeXMLHttpRequest.HEADERS_RECEIVED) {
2468
+ copyAttrs(["status","statusText"]);
2469
+ }
2470
+ if(xhr.readyState >= FakeXMLHttpRequest.LOADING) {
2471
+ copyAttrs(["responseText"]);
2472
+ }
2473
+ if(xhr.readyState === FakeXMLHttpRequest.DONE) {
2474
+ copyAttrs(["responseXML"]);
2475
+ }
2476
+ if(fakeXhr.onreadystatechange) fakeXhr.onreadystatechange.call(fakeXhr);
2477
+ };
2478
+ if(xhr.addEventListener) {
2479
+ for(var event in fakeXhr.eventListeners) {
2480
+ if(fakeXhr.eventListeners.hasOwnProperty(event)) {
2481
+ each(fakeXhr.eventListeners[event],function(handler) {
2482
+ xhr.addEventListener(event, handler);
2483
+ });
2484
+ }
2485
+ }
2486
+ xhr.addEventListener("readystatechange",stateChange);
2487
+ } else {
2488
+ xhr.onreadystatechange = stateChange;
2489
+ }
2490
+ apply(xhr,"open",xhrArgs);
2491
+ };
2492
+ FakeXMLHttpRequest.useFilters = false;
2493
+
2494
+ function verifyRequestSent(xhr) {
2495
+ if (xhr.readyState == FakeXMLHttpRequest.DONE) {
2496
+ throw new Error("Request done");
2497
+ }
2498
+ }
2499
+
2500
+ function verifyHeadersReceived(xhr) {
2501
+ if (xhr.async && xhr.readyState != FakeXMLHttpRequest.HEADERS_RECEIVED) {
2502
+ throw new Error("No headers received");
2503
+ }
2504
+ }
2505
+
2506
+ function verifyResponseBodyType(body) {
2507
+ if (typeof body != "string") {
2508
+ var error = new Error("Attempted to respond to fake XMLHttpRequest with " +
2509
+ body + ", which is not a string.");
2510
+ error.name = "InvalidBodyException";
2511
+ throw error;
2512
+ }
2513
+ }
2514
+
2515
+ sinon.extend(FakeXMLHttpRequest.prototype, sinon.EventTarget, {
2516
+ async: true,
2517
+
2518
+ open: function open(method, url, async, username, password) {
2519
+ this.method = method;
2520
+ this.url = url;
2521
+ this.async = typeof async == "boolean" ? async : true;
2522
+ this.username = username;
2523
+ this.password = password;
2524
+ this.responseText = null;
2525
+ this.responseXML = null;
2526
+ this.requestHeaders = {};
2527
+ this.sendFlag = false;
2528
+ if(sinon.FakeXMLHttpRequest.useFilters === true) {
2529
+ var xhrArgs = arguments;
2530
+ var defake = some(FakeXMLHttpRequest.filters,function(filter) {
2531
+ return filter.apply(this,xhrArgs)
2532
+ });
2533
+ if (defake) {
2534
+ return sinon.FakeXMLHttpRequest.defake(this,arguments);
2535
+ }
2536
+ }
2537
+ this.readyStateChange(FakeXMLHttpRequest.OPENED);
2538
+ },
2539
+
2540
+ readyStateChange: function readyStateChange(state) {
2541
+ this.readyState = state;
2542
+
2543
+ if (typeof this.onreadystatechange == "function") {
2544
+ try {
2545
+ this.onreadystatechange();
2546
+ } catch (e) {
2547
+ sinon.logError("Fake XHR onreadystatechange handler", e);
2548
+ }
2549
+ }
2550
+
2551
+ this.dispatchEvent(new sinon.Event("readystatechange"));
2552
+ },
2553
+
2554
+ setRequestHeader: function setRequestHeader(header, value) {
2555
+ verifyState(this);
2556
+
2557
+ if (unsafeHeaders[header] || /^(Sec-|Proxy-)/.test(header)) {
2558
+ throw new Error("Refused to set unsafe header \"" + header + "\"");
2559
+ }
2560
+
2561
+ if (this.requestHeaders[header]) {
2562
+ this.requestHeaders[header] += "," + value;
2563
+ } else {
2564
+ this.requestHeaders[header] = value;
2565
+ }
2566
+ },
2567
+
2568
+ // Helps testing
2569
+ setResponseHeaders: function setResponseHeaders(headers) {
2570
+ this.responseHeaders = {};
2571
+
2572
+ for (var header in headers) {
2573
+ if (headers.hasOwnProperty(header)) {
2574
+ this.responseHeaders[header] = headers[header];
2575
+ }
2576
+ }
2577
+
2578
+ if (this.async) {
2579
+ this.readyStateChange(FakeXMLHttpRequest.HEADERS_RECEIVED);
2580
+ }
2581
+ },
2582
+
2583
+ // Currently treats ALL data as a DOMString (i.e. no Document)
2584
+ send: function send(data) {
2585
+ verifyState(this);
2586
+
2587
+ if (!/^(get|head)$/i.test(this.method)) {
2588
+ if (this.requestHeaders["Content-Type"]) {
2589
+ var value = this.requestHeaders["Content-Type"].split(";");
2590
+ this.requestHeaders["Content-Type"] = value[0] + ";charset=utf-8";
2591
+ } else {
2592
+ this.requestHeaders["Content-Type"] = "text/plain;charset=utf-8";
2593
+ }
2594
+
2595
+ this.requestBody = data;
2596
+ }
2597
+
2598
+ this.errorFlag = false;
2599
+ this.sendFlag = this.async;
2600
+ this.readyStateChange(FakeXMLHttpRequest.OPENED);
2601
+
2602
+ if (typeof this.onSend == "function") {
2603
+ this.onSend(this);
2604
+ }
2605
+ },
2606
+
2607
+ abort: function abort() {
2608
+ this.aborted = true;
2609
+ this.responseText = null;
2610
+ this.errorFlag = true;
2611
+ this.requestHeaders = {};
2612
+
2613
+ if (this.readyState > sinon.FakeXMLHttpRequest.UNSENT && this.sendFlag) {
2614
+ this.readyStateChange(sinon.FakeXMLHttpRequest.DONE);
2615
+ this.sendFlag = false;
2616
+ }
2617
+
2618
+ this.readyState = sinon.FakeXMLHttpRequest.UNSENT;
2619
+ },
2620
+
2621
+ getResponseHeader: function getResponseHeader(header) {
2622
+ if (this.readyState < FakeXMLHttpRequest.HEADERS_RECEIVED) {
2623
+ return null;
2624
+ }
2625
+
2626
+ if (/^Set-Cookie2?$/i.test(header)) {
2627
+ return null;
2628
+ }
2629
+
2630
+ header = header.toLowerCase();
2631
+
2632
+ for (var h in this.responseHeaders) {
2633
+ if (h.toLowerCase() == header) {
2634
+ return this.responseHeaders[h];
2635
+ }
2636
+ }
2637
+
2638
+ return null;
2639
+ },
2640
+
2641
+ getAllResponseHeaders: function getAllResponseHeaders() {
2642
+ if (this.readyState < FakeXMLHttpRequest.HEADERS_RECEIVED) {
2643
+ return "";
2644
+ }
2645
+
2646
+ var headers = "";
2647
+
2648
+ for (var header in this.responseHeaders) {
2649
+ if (this.responseHeaders.hasOwnProperty(header) &&
2650
+ !/^Set-Cookie2?$/i.test(header)) {
2651
+ headers += header + ": " + this.responseHeaders[header] + "\r\n";
2652
+ }
2653
+ }
2654
+
2655
+ return headers;
2656
+ },
2657
+
2658
+ setResponseBody: function setResponseBody(body) {
2659
+ verifyRequestSent(this);
2660
+ verifyHeadersReceived(this);
2661
+ verifyResponseBodyType(body);
2662
+
2663
+ var chunkSize = this.chunkSize || 10;
2664
+ var index = 0;
2665
+ this.responseText = "";
2666
+
2667
+ do {
2668
+ if (this.async) {
2669
+ this.readyStateChange(FakeXMLHttpRequest.LOADING);
2670
+ }
2671
+
2672
+ this.responseText += body.substring(index, index + chunkSize);
2673
+ index += chunkSize;
2674
+ } while (index < body.length);
2675
+
2676
+ var type = this.getResponseHeader("Content-Type");
2677
+
2678
+ if (this.responseText &&
2679
+ (!type || /(text\/xml)|(application\/xml)|(\+xml)/.test(type))) {
2680
+ try {
2681
+ this.responseXML = FakeXMLHttpRequest.parseXML(this.responseText);
2682
+ } catch (e) {
2683
+ // Unable to parse XML - no biggie
2684
+ }
2685
+ }
2686
+
2687
+ if (this.async) {
2688
+ this.readyStateChange(FakeXMLHttpRequest.DONE);
2689
+ } else {
2690
+ this.readyState = FakeXMLHttpRequest.DONE;
2691
+ }
2692
+ },
2693
+
2694
+ respond: function respond(status, headers, body) {
2695
+ this.setResponseHeaders(headers || {});
2696
+ this.status = typeof status == "number" ? status : 200;
2697
+ this.statusText = FakeXMLHttpRequest.statusCodes[this.status];
2698
+ this.setResponseBody(body || "");
2699
+ }
2700
+ });
2701
+
2702
+ sinon.extend(FakeXMLHttpRequest, {
2703
+ UNSENT: 0,
2704
+ OPENED: 1,
2705
+ HEADERS_RECEIVED: 2,
2706
+ LOADING: 3,
2707
+ DONE: 4
2708
+ });
2709
+
2710
+ // Borrowed from JSpec
2711
+ FakeXMLHttpRequest.parseXML = function parseXML(text) {
2712
+ var xmlDoc;
2713
+
2714
+ if (typeof DOMParser != "undefined") {
2715
+ var parser = new DOMParser();
2716
+ xmlDoc = parser.parseFromString(text, "text/xml");
2717
+ } else {
2718
+ xmlDoc = new ActiveXObject("Microsoft.XMLDOM");
2719
+ xmlDoc.async = "false";
2720
+ xmlDoc.loadXML(text);
2721
+ }
2722
+
2723
+ return xmlDoc;
2724
+ };
2725
+
2726
+ FakeXMLHttpRequest.statusCodes = {
2727
+ 100: "Continue",
2728
+ 101: "Switching Protocols",
2729
+ 200: "OK",
2730
+ 201: "Created",
2731
+ 202: "Accepted",
2732
+ 203: "Non-Authoritative Information",
2733
+ 204: "No Content",
2734
+ 205: "Reset Content",
2735
+ 206: "Partial Content",
2736
+ 300: "Multiple Choice",
2737
+ 301: "Moved Permanently",
2738
+ 302: "Found",
2739
+ 303: "See Other",
2740
+ 304: "Not Modified",
2741
+ 305: "Use Proxy",
2742
+ 307: "Temporary Redirect",
2743
+ 400: "Bad Request",
2744
+ 401: "Unauthorized",
2745
+ 402: "Payment Required",
2746
+ 403: "Forbidden",
2747
+ 404: "Not Found",
2748
+ 405: "Method Not Allowed",
2749
+ 406: "Not Acceptable",
2750
+ 407: "Proxy Authentication Required",
2751
+ 408: "Request Timeout",
2752
+ 409: "Conflict",
2753
+ 410: "Gone",
2754
+ 411: "Length Required",
2755
+ 412: "Precondition Failed",
2756
+ 413: "Request Entity Too Large",
2757
+ 414: "Request-URI Too Long",
2758
+ 415: "Unsupported Media Type",
2759
+ 416: "Requested Range Not Satisfiable",
2760
+ 417: "Expectation Failed",
2761
+ 422: "Unprocessable Entity",
2762
+ 500: "Internal Server Error",
2763
+ 501: "Not Implemented",
2764
+ 502: "Bad Gateway",
2765
+ 503: "Service Unavailable",
2766
+ 504: "Gateway Timeout",
2767
+ 505: "HTTP Version Not Supported"
2768
+ };
2769
+
2770
+ sinon.useFakeXMLHttpRequest = function () {
2771
+ sinon.FakeXMLHttpRequest.restore = function restore(keepOnCreate) {
2772
+ if (xhr.supportsXHR) {
2773
+ global.XMLHttpRequest = xhr.GlobalXMLHttpRequest;
2774
+ }
2775
+
2776
+ if (xhr.supportsActiveX) {
2777
+ global.ActiveXObject = xhr.GlobalActiveXObject;
2778
+ }
2779
+
2780
+ delete sinon.FakeXMLHttpRequest.restore;
2781
+
2782
+ if (keepOnCreate !== true) {
2783
+ delete sinon.FakeXMLHttpRequest.onCreate;
2784
+ }
2785
+ };
2786
+ if (xhr.supportsXHR) {
2787
+ global.XMLHttpRequest = sinon.FakeXMLHttpRequest;
2788
+ }
2789
+
2790
+ if (xhr.supportsActiveX) {
2791
+ global.ActiveXObject = function ActiveXObject(objId) {
2792
+ if (objId == "Microsoft.XMLHTTP" || /^Msxml2\.XMLHTTP/i.test(objId)) {
2793
+
2794
+ return new sinon.FakeXMLHttpRequest();
2795
+ }
2796
+
2797
+ return new xhr.GlobalActiveXObject(objId);
2798
+ };
2799
+ }
2800
+
2801
+ return sinon.FakeXMLHttpRequest;
2802
+ };
2803
+
2804
+ sinon.FakeXMLHttpRequest = FakeXMLHttpRequest;
2805
+ })(this);
2806
+
2807
+ if (typeof module == "object" && typeof require == "function") {
2808
+ module.exports = sinon;
2809
+ }
2810
+
2811
+ /**
2812
+ * @depend fake_xml_http_request.js
2813
+ */
2814
+ /*jslint eqeqeq: false, onevar: false, regexp: false, plusplus: false*/
2815
+ /*global module, require, window*/
2816
+ /**
2817
+ * The Sinon "server" mimics a web server that receives requests from
2818
+ * sinon.FakeXMLHttpRequest and provides an API to respond to those requests,
2819
+ * both synchronously and asynchronously. To respond synchronuously, canned
2820
+ * answers have to be provided upfront.
2821
+ *
2822
+ * @author Christian Johansen (christian@cjohansen.no)
2823
+ * @license BSD
2824
+ *
2825
+ * Copyright (c) 2010-2011 Christian Johansen
2826
+ */
2827
+
2828
+ if (typeof sinon == "undefined") {
2829
+ var sinon = {};
2830
+ }
2831
+
2832
+ sinon.fakeServer = (function () {
2833
+ var push = [].push;
2834
+ function F() {}
2835
+
2836
+ function create(proto) {
2837
+ F.prototype = proto;
2838
+ return new F();
2839
+ }
2840
+
2841
+ function responseArray(handler) {
2842
+ var response = handler;
2843
+
2844
+ if (Object.prototype.toString.call(handler) != "[object Array]") {
2845
+ response = [200, {}, handler];
2846
+ }
2847
+
2848
+ if (typeof response[2] != "string") {
2849
+ throw new TypeError("Fake server response body should be string, but was " +
2850
+ typeof response[2]);
2851
+ }
2852
+
2853
+ return response;
2854
+ }
2855
+
2856
+ var wloc = window.location;
2857
+ var rCurrLoc = new RegExp("^" + wloc.protocol + "//" + wloc.host);
2858
+
2859
+ function matchOne(response, reqMethod, reqUrl) {
2860
+ var rmeth = response.method;
2861
+ var matchMethod = !rmeth || rmeth.toLowerCase() == reqMethod.toLowerCase();
2862
+ var url = response.url;
2863
+ var matchUrl = !url || url == reqUrl || (typeof url.test == "function" && url.test(reqUrl));
2864
+
2865
+ return matchMethod && matchUrl;
2866
+ }
2867
+
2868
+ function match(response, request) {
2869
+ var requestMethod = this.getHTTPMethod(request);
2870
+ var requestUrl = request.url;
2871
+
2872
+ if (!/^https?:\/\//.test(requestUrl) || rCurrLoc.test(requestUrl)) {
2873
+ requestUrl = requestUrl.replace(rCurrLoc, "");
2874
+ }
2875
+
2876
+ if (matchOne(response, this.getHTTPMethod(request), requestUrl)) {
2877
+ if (typeof response.response == "function") {
2878
+ var ru = response.url;
2879
+ var args = [request].concat(!ru ? [] : requestUrl.match(ru).slice(1));
2880
+ return response.response.apply(response, args);
2881
+ }
2882
+
2883
+ return true;
2884
+ }
2885
+
2886
+ return false;
2887
+ }
2888
+
2889
+ return {
2890
+ create: function () {
2891
+ var server = create(this);
2892
+ this.xhr = sinon.useFakeXMLHttpRequest();
2893
+ server.requests = [];
2894
+
2895
+ this.xhr.onCreate = function (xhrObj) {
2896
+ server.addRequest(xhrObj);
2897
+ };
2898
+
2899
+ return server;
2900
+ },
2901
+
2902
+ addRequest: function addRequest(xhrObj) {
2903
+ var server = this;
2904
+ push.call(this.requests, xhrObj);
2905
+
2906
+ xhrObj.onSend = function () {
2907
+ server.handleRequest(this);
2908
+ };
2909
+
2910
+ if (this.autoRespond && !this.responding) {
2911
+ setTimeout(function () {
2912
+ server.responding = false;
2913
+ server.respond();
2914
+ }, this.autoRespondAfter || 10);
2915
+
2916
+ this.responding = true;
2917
+ }
2918
+ },
2919
+
2920
+ getHTTPMethod: function getHTTPMethod(request) {
2921
+ if (this.fakeHTTPMethods && /post/i.test(request.method)) {
2922
+ var matches = (request.requestBody || "").match(/_method=([^\b;]+)/);
2923
+ return !!matches ? matches[1] : request.method;
2924
+ }
2925
+
2926
+ return request.method;
2927
+ },
2928
+
2929
+ handleRequest: function handleRequest(xhr) {
2930
+ if (xhr.async) {
2931
+ if (!this.queue) {
2932
+ this.queue = [];
2933
+ }
2934
+
2935
+ push.call(this.queue, xhr);
2936
+ } else {
2937
+ this.processRequest(xhr);
2938
+ }
2939
+ },
2940
+
2941
+ respondWith: function respondWith(method, url, body) {
2942
+ if (arguments.length == 1 && typeof method != "function") {
2943
+ this.response = responseArray(method);
2944
+ return;
2945
+ }
2946
+
2947
+ if (!this.responses) { this.responses = []; }
2948
+
2949
+ if (arguments.length == 1) {
2950
+ body = method;
2951
+ url = method = null;
2952
+ }
2953
+
2954
+ if (arguments.length == 2) {
2955
+ body = url;
2956
+ url = method;
2957
+ method = null;
2958
+ }
2959
+
2960
+ push.call(this.responses, {
2961
+ method: method,
2962
+ url: url,
2963
+ response: typeof body == "function" ? body : responseArray(body)
2964
+ });
2965
+ },
2966
+
2967
+ respond: function respond() {
2968
+ if (arguments.length > 0) this.respondWith.apply(this, arguments);
2969
+ var queue = this.queue || [];
2970
+ var request;
2971
+
2972
+ while(request = queue.shift()) {
2973
+ this.processRequest(request);
2974
+ }
2975
+ },
2976
+
2977
+ processRequest: function processRequest(request) {
2978
+ try {
2979
+ if (request.aborted) {
2980
+ return;
2981
+ }
2982
+
2983
+ var response = this.response || [404, {}, ""];
2984
+
2985
+ if (this.responses) {
2986
+ for (var i = 0, l = this.responses.length; i < l; i++) {
2987
+ if (match.call(this, this.responses[i], request)) {
2988
+ response = this.responses[i].response;
2989
+ break;
2990
+ }
2991
+ }
2992
+ }
2993
+
2994
+ if (request.readyState != 4) {
2995
+ request.respond(response[0], response[1], response[2]);
2996
+ }
2997
+ } catch (e) {
2998
+ sinon.logError("Fake server request processing", e);
2999
+ }
3000
+ },
3001
+
3002
+ restore: function restore() {
3003
+ return this.xhr.restore && this.xhr.restore.apply(this.xhr, arguments);
3004
+ }
3005
+ };
3006
+ }());
3007
+
3008
+ if (typeof module == "object" && typeof require == "function") {
3009
+ module.exports = sinon;
3010
+ }
3011
+
3012
+ /**
3013
+ * @depend fake_server.js
3014
+ * @depend fake_timers.js
3015
+ */
3016
+ /*jslint browser: true, eqeqeq: false, onevar: false*/
3017
+ /*global sinon*/
3018
+ /**
3019
+ * Add-on for sinon.fakeServer that automatically handles a fake timer along with
3020
+ * the FakeXMLHttpRequest. The direct inspiration for this add-on is jQuery
3021
+ * 1.3.x, which does not use xhr object's onreadystatehandler at all - instead,
3022
+ * it polls the object for completion with setInterval. Dispite the direct
3023
+ * motivation, there is nothing jQuery-specific in this file, so it can be used
3024
+ * in any environment where the ajax implementation depends on setInterval or
3025
+ * setTimeout.
3026
+ *
3027
+ * @author Christian Johansen (christian@cjohansen.no)
3028
+ * @license BSD
3029
+ *
3030
+ * Copyright (c) 2010-2011 Christian Johansen
3031
+ */
3032
+
3033
+ (function () {
3034
+ function Server() {}
3035
+ Server.prototype = sinon.fakeServer;
3036
+
3037
+ sinon.fakeServerWithClock = new Server();
3038
+
3039
+ sinon.fakeServerWithClock.addRequest = function addRequest(xhr) {
3040
+ if (xhr.async) {
3041
+ if (typeof setTimeout.clock == "object") {
3042
+ this.clock = setTimeout.clock;
3043
+ } else {
3044
+ this.clock = sinon.useFakeTimers();
3045
+ this.resetClock = true;
3046
+ }
3047
+
3048
+ if (!this.longestTimeout) {
3049
+ var clockSetTimeout = this.clock.setTimeout;
3050
+ var clockSetInterval = this.clock.setInterval;
3051
+ var server = this;
3052
+
3053
+ this.clock.setTimeout = function (fn, timeout) {
3054
+ server.longestTimeout = Math.max(timeout, server.longestTimeout || 0);
3055
+
3056
+ return clockSetTimeout.apply(this, arguments);
3057
+ };
3058
+
3059
+ this.clock.setInterval = function (fn, timeout) {
3060
+ server.longestTimeout = Math.max(timeout, server.longestTimeout || 0);
3061
+
3062
+ return clockSetInterval.apply(this, arguments);
3063
+ };
3064
+ }
3065
+ }
3066
+
3067
+ return sinon.fakeServer.addRequest.call(this, xhr);
3068
+ };
3069
+
3070
+ sinon.fakeServerWithClock.respond = function respond() {
3071
+ var returnVal = sinon.fakeServer.respond.apply(this, arguments);
3072
+
3073
+ if (this.clock) {
3074
+ this.clock.tick(this.longestTimeout || 0);
3075
+ this.longestTimeout = 0;
3076
+
3077
+ if (this.resetClock) {
3078
+ this.clock.restore();
3079
+ this.resetClock = false;
3080
+ }
3081
+ }
3082
+
3083
+ return returnVal;
3084
+ };
3085
+
3086
+ sinon.fakeServerWithClock.restore = function restore() {
3087
+ if (this.clock) {
3088
+ this.clock.restore();
3089
+ }
3090
+
3091
+ return sinon.fakeServer.restore.apply(this, arguments);
3092
+ };
3093
+ }());
3094
+
3095
+ /**
3096
+ * @depend ../sinon.js
3097
+ * @depend collection.js
3098
+ * @depend util/fake_timers.js
3099
+ * @depend util/fake_server_with_clock.js
3100
+ */
3101
+ /*jslint eqeqeq: false, onevar: false, plusplus: false*/
3102
+ /*global require, module*/
3103
+ /**
3104
+ * Manages fake collections as well as fake utilities such as Sinon's
3105
+ * timers and fake XHR implementation in one convenient object.
3106
+ *
3107
+ * @author Christian Johansen (christian@cjohansen.no)
3108
+ * @license BSD
3109
+ *
3110
+ * Copyright (c) 2010-2011 Christian Johansen
3111
+ */
3112
+
3113
+ if (typeof module == "object" && typeof require == "function") {
3114
+ var sinon = require("../sinon");
3115
+ sinon.extend(sinon, require("./util/fake_timers"));
3116
+ }
3117
+
3118
+ (function () {
3119
+ var push = [].push;
3120
+
3121
+ function exposeValue(sandbox, config, key, value) {
3122
+ if (!value) {
3123
+ return;
3124
+ }
3125
+
3126
+ if (config.injectInto) {
3127
+ config.injectInto[key] = value;
3128
+ } else {
3129
+ push.call(sandbox.args, value);
3130
+ }
3131
+ }
3132
+
3133
+ function prepareSandboxFromConfig(config) {
3134
+ var sandbox = sinon.create(sinon.sandbox);
3135
+
3136
+ if (config.useFakeServer) {
3137
+ if (typeof config.useFakeServer == "object") {
3138
+ sandbox.serverPrototype = config.useFakeServer;
3139
+ }
3140
+
3141
+ sandbox.useFakeServer();
3142
+ }
3143
+
3144
+ if (config.useFakeTimers) {
3145
+ if (typeof config.useFakeTimers == "object") {
3146
+ sandbox.useFakeTimers.apply(sandbox, config.useFakeTimers);
3147
+ } else {
3148
+ sandbox.useFakeTimers();
3149
+ }
3150
+ }
3151
+
3152
+ return sandbox;
3153
+ }
3154
+
3155
+ sinon.sandbox = sinon.extend(sinon.create(sinon.collection), {
3156
+ useFakeTimers: function useFakeTimers() {
3157
+ this.clock = sinon.useFakeTimers.apply(sinon, arguments);
3158
+
3159
+ return this.add(this.clock);
3160
+ },
3161
+
3162
+ serverPrototype: sinon.fakeServer,
3163
+
3164
+ useFakeServer: function useFakeServer() {
3165
+ var proto = this.serverPrototype || sinon.fakeServer;
3166
+
3167
+ if (!proto || !proto.create) {
3168
+ return null;
3169
+ }
3170
+
3171
+ this.server = proto.create();
3172
+ return this.add(this.server);
3173
+ },
3174
+
3175
+ inject: function (obj) {
3176
+ sinon.collection.inject.call(this, obj);
3177
+
3178
+ if (this.clock) {
3179
+ obj.clock = this.clock;
3180
+ }
3181
+
3182
+ if (this.server) {
3183
+ obj.server = this.server;
3184
+ obj.requests = this.server.requests;
3185
+ }
3186
+
3187
+ return obj;
3188
+ },
3189
+
3190
+ create: function (config) {
3191
+ if (!config) {
3192
+ return sinon.create(sinon.sandbox);
3193
+ }
3194
+
3195
+ var sandbox = prepareSandboxFromConfig(config);
3196
+ sandbox.args = sandbox.args || [];
3197
+ var prop, value, exposed = sandbox.inject({});
3198
+
3199
+ if (config.properties) {
3200
+ for (var i = 0, l = config.properties.length; i < l; i++) {
3201
+ prop = config.properties[i];
3202
+ value = exposed[prop] || prop == "sandbox" && sandbox;
3203
+ exposeValue(sandbox, config, prop, value);
3204
+ }
3205
+ } else {
3206
+ exposeValue(sandbox, config, "sandbox", value);
3207
+ }
3208
+
3209
+ return sandbox;
3210
+ }
3211
+ });
3212
+
3213
+ sinon.sandbox.useFakeXMLHttpRequest = sinon.sandbox.useFakeServer;
3214
+
3215
+ if (typeof module == "object" && typeof require == "function") {
3216
+ module.exports = sinon.sandbox;
3217
+ }
3218
+ }());
3219
+
3220
+ /**
3221
+ * @depend ../sinon.js
3222
+ * @depend stub.js
3223
+ * @depend mock.js
3224
+ * @depend sandbox.js
3225
+ */
3226
+ /*jslint eqeqeq: false, onevar: false, forin: true, plusplus: false*/
3227
+ /*global module, require, sinon*/
3228
+ /**
3229
+ * Test function, sandboxes fakes
3230
+ *
3231
+ * @author Christian Johansen (christian@cjohansen.no)
3232
+ * @license BSD
3233
+ *
3234
+ * Copyright (c) 2010-2011 Christian Johansen
3235
+ */
3236
+
3237
+ (function (sinon) {
3238
+ var commonJSModule = typeof module == "object" && typeof require == "function";
3239
+
3240
+ if (!sinon && commonJSModule) {
3241
+ sinon = require("../sinon");
3242
+ }
3243
+
3244
+ if (!sinon) {
3245
+ return;
3246
+ }
3247
+
3248
+ function test(callback) {
3249
+ var type = typeof callback;
3250
+
3251
+ if (type != "function") {
3252
+ throw new TypeError("sinon.test needs to wrap a test function, got " + type);
3253
+ }
3254
+
3255
+ return function () {
3256
+ var config = sinon.getConfig(sinon.config);
3257
+ config.injectInto = config.injectIntoThis && this || config.injectInto;
3258
+ var sandbox = sinon.sandbox.create(config);
3259
+ var exception, result;
3260
+ var args = Array.prototype.slice.call(arguments).concat(sandbox.args);
3261
+
3262
+ try {
3263
+ result = callback.apply(this, args);
3264
+ } finally {
3265
+ sandbox.verifyAndRestore();
3266
+ }
3267
+
3268
+ return result;
3269
+ };
3270
+ }
3271
+
3272
+ test.config = {
3273
+ injectIntoThis: true,
3274
+ injectInto: null,
3275
+ properties: ["spy", "stub", "mock", "clock", "server", "requests"],
3276
+ useFakeTimers: true,
3277
+ useFakeServer: true
3278
+ };
3279
+
3280
+ if (commonJSModule) {
3281
+ module.exports = test;
3282
+ } else {
3283
+ sinon.test = test;
3284
+ }
3285
+ }(typeof sinon == "object" && sinon || null));
3286
+
3287
+ /**
3288
+ * @depend ../sinon.js
3289
+ * @depend test.js
3290
+ */
3291
+ /*jslint eqeqeq: false, onevar: false, eqeqeq: false*/
3292
+ /*global module, require, sinon*/
3293
+ /**
3294
+ * Test case, sandboxes all test functions
3295
+ *
3296
+ * @author Christian Johansen (christian@cjohansen.no)
3297
+ * @license BSD
3298
+ *
3299
+ * Copyright (c) 2010-2011 Christian Johansen
3300
+ */
3301
+
3302
+ (function (sinon) {
3303
+ var commonJSModule = typeof module == "object" && typeof require == "function";
3304
+
3305
+ if (!sinon && commonJSModule) {
3306
+ sinon = require("../sinon");
3307
+ }
3308
+
3309
+ if (!sinon || !Object.prototype.hasOwnProperty) {
3310
+ return;
3311
+ }
3312
+
3313
+ function createTest(property, setUp, tearDown) {
3314
+ return function () {
3315
+ if (setUp) {
3316
+ setUp.apply(this, arguments);
3317
+ }
3318
+
3319
+ var exception, result;
3320
+
3321
+ try {
3322
+ result = property.apply(this, arguments);
3323
+ } catch (e) {
3324
+ exception = e;
3325
+ }
3326
+
3327
+ if (tearDown) {
3328
+ tearDown.apply(this, arguments);
3329
+ }
3330
+
3331
+ if (exception) {
3332
+ throw exception;
3333
+ }
3334
+
3335
+ return result;
3336
+ };
3337
+ }
3338
+
3339
+ function testCase(tests, prefix) {
3340
+ /*jsl:ignore*/
3341
+ if (!tests || typeof tests != "object") {
3342
+ throw new TypeError("sinon.testCase needs an object with test functions");
3343
+ }
3344
+ /*jsl:end*/
3345
+
3346
+ prefix = prefix || "test";
3347
+ var rPrefix = new RegExp("^" + prefix);
3348
+ var methods = {}, testName, property, method;
3349
+ var setUp = tests.setUp;
3350
+ var tearDown = tests.tearDown;
3351
+
3352
+ for (testName in tests) {
3353
+ if (tests.hasOwnProperty(testName)) {
3354
+ property = tests[testName];
3355
+
3356
+ if (/^(setUp|tearDown)$/.test(testName)) {
3357
+ continue;
3358
+ }
3359
+
3360
+ if (typeof property == "function" && rPrefix.test(testName)) {
3361
+ method = property;
3362
+
3363
+ if (setUp || tearDown) {
3364
+ method = createTest(property, setUp, tearDown);
3365
+ }
3366
+
3367
+ methods[testName] = sinon.test(method);
3368
+ } else {
3369
+ methods[testName] = tests[testName];
3370
+ }
3371
+ }
3372
+ }
3373
+
3374
+ return methods;
3375
+ }
3376
+
3377
+ if (commonJSModule) {
3378
+ module.exports = testCase;
3379
+ } else {
3380
+ sinon.testCase = testCase;
3381
+ }
3382
+ }(typeof sinon == "object" && sinon || null));
3383
+
3384
+ /**
3385
+ * @depend ../sinon.js
3386
+ * @depend stub.js
3387
+ */
3388
+ /*jslint eqeqeq: false, onevar: false, nomen: false, plusplus: false*/
3389
+ /*global module, require, sinon*/
3390
+ /**
3391
+ * Assertions matching the test spy retrieval interface.
3392
+ *
3393
+ * @author Christian Johansen (christian@cjohansen.no)
3394
+ * @license BSD
3395
+ *
3396
+ * Copyright (c) 2010-2011 Christian Johansen
3397
+ */
3398
+
3399
+ (function (sinon, global) {
3400
+ var commonJSModule = typeof module == "object" && typeof require == "function";
3401
+ var slice = Array.prototype.slice;
3402
+ var assert;
3403
+
3404
+ if (!sinon && commonJSModule) {
3405
+ sinon = require("../sinon");
3406
+ }
3407
+
3408
+ if (!sinon) {
3409
+ return;
3410
+ }
3411
+
3412
+ function verifyIsStub() {
3413
+ var method;
3414
+
3415
+ for (var i = 0, l = arguments.length; i < l; ++i) {
3416
+ method = arguments[i];
3417
+
3418
+ if (!method) {
3419
+ assert.fail("fake is not a spy");
3420
+ }
3421
+
3422
+ if (typeof method != "function") {
3423
+ assert.fail(method + " is not a function");
3424
+ }
3425
+
3426
+ if (typeof method.getCall != "function") {
3427
+ assert.fail(method + " is not stubbed");
3428
+ }
3429
+ }
3430
+ }
3431
+
3432
+ function failAssertion(object, msg) {
3433
+ object = object || global;
3434
+ var failMethod = object.fail || assert.fail;
3435
+ failMethod.call(object, msg);
3436
+ }
3437
+
3438
+ function mirrorPropAsAssertion(name, method, message) {
3439
+ if (arguments.length == 2) {
3440
+ message = method;
3441
+ method = name;
3442
+ }
3443
+
3444
+ assert[name] = function (fake) {
3445
+ verifyIsStub(fake);
3446
+
3447
+ var args = slice.call(arguments, 1);
3448
+ var failed = false;
3449
+
3450
+ if (typeof method == "function") {
3451
+ failed = !method(fake);
3452
+ } else {
3453
+ failed = typeof fake[method] == "function" ?
3454
+ !fake[method].apply(fake, args) : !fake[method];
3455
+ }
3456
+
3457
+ if (failed) {
3458
+ failAssertion(this, fake.printf.apply(fake, [message].concat(args)));
3459
+ } else {
3460
+ assert.pass(name);
3461
+ }
3462
+ };
3463
+ }
3464
+
3465
+ function exposedName(prefix, prop) {
3466
+ return !prefix || /^fail/.test(prop) ? prop :
3467
+ prefix + prop.slice(0, 1).toUpperCase() + prop.slice(1);
3468
+ };
3469
+
3470
+ assert = {
3471
+ failException: "AssertError",
3472
+
3473
+ fail: function fail(message) {
3474
+ var error = new Error(message);
3475
+ error.name = this.failException || assert.failException;
3476
+
3477
+ throw error;
3478
+ },
3479
+
3480
+ pass: function pass(assertion) {},
3481
+
3482
+ callOrder: function assertCallOrder() {
3483
+ verifyIsStub.apply(null, arguments);
3484
+ var expected = "", actual = "";
3485
+
3486
+ if (!sinon.calledInOrder(arguments)) {
3487
+ try {
3488
+ expected = [].join.call(arguments, ", ");
3489
+ actual = sinon.orderByFirstCall(slice.call(arguments)).join(", ");
3490
+ } catch (e) {
3491
+ // If this fails, we'll just fall back to the blank string
3492
+ }
3493
+
3494
+ failAssertion(this, "expected " + expected + " to be " +
3495
+ "called in order but were called as " + actual);
3496
+ } else {
3497
+ assert.pass("callOrder");
3498
+ }
3499
+ },
3500
+
3501
+ callCount: function assertCallCount(method, count) {
3502
+ verifyIsStub(method);
3503
+
3504
+ if (method.callCount != count) {
3505
+ var msg = "expected %n to be called " + sinon.timesInWords(count) +
3506
+ " but was called %c%C";
3507
+ failAssertion(this, method.printf(msg));
3508
+ } else {
3509
+ assert.pass("callCount");
3510
+ }
3511
+ },
3512
+
3513
+ expose: function expose(target, options) {
3514
+ if (!target) {
3515
+ throw new TypeError("target is null or undefined");
3516
+ }
3517
+
3518
+ var o = options || {};
3519
+ var prefix = typeof o.prefix == "undefined" && "assert" || o.prefix;
3520
+ var includeFail = typeof o.includeFail == "undefined" || !!o.includeFail;
3521
+
3522
+ for (var method in this) {
3523
+ if (method != "export" && (includeFail || !/^(fail)/.test(method))) {
3524
+ target[exposedName(prefix, method)] = this[method];
3525
+ }
3526
+ }
3527
+
3528
+ return target;
3529
+ }
3530
+ };
3531
+
3532
+ mirrorPropAsAssertion("called", "expected %n to have been called at least once but was never called");
3533
+ mirrorPropAsAssertion("notCalled", function (spy) { return !spy.called; },
3534
+ "expected %n to not have been called but was called %c%C");
3535
+ mirrorPropAsAssertion("calledOnce", "expected %n to be called once but was called %c%C");
3536
+ mirrorPropAsAssertion("calledTwice", "expected %n to be called twice but was called %c%C");
3537
+ mirrorPropAsAssertion("calledThrice", "expected %n to be called thrice but was called %c%C");
3538
+ mirrorPropAsAssertion("calledOn", "expected %n to be called with %1 as this but was called with %t");
3539
+ mirrorPropAsAssertion("alwaysCalledOn", "expected %n to always be called with %1 as this but was called with %t");
3540
+ mirrorPropAsAssertion("calledWith", "expected %n to be called with arguments %*%C");
3541
+ mirrorPropAsAssertion("alwaysCalledWith", "expected %n to always be called with arguments %*%C");
3542
+ mirrorPropAsAssertion("calledWithExactly", "expected %n to be called with exact arguments %*%C");
3543
+ mirrorPropAsAssertion("alwaysCalledWithExactly", "expected %n to always be called with exact arguments %*%C");
3544
+ mirrorPropAsAssertion("neverCalledWith", "expected %n to never be called with arguments %*%C");
3545
+ mirrorPropAsAssertion("threw", "%n did not throw exception%C");
3546
+ mirrorPropAsAssertion("alwaysThrew", "%n did not always throw exception%C");
3547
+
3548
+ if (commonJSModule) {
3549
+ module.exports = assert;
3550
+ } else {
3551
+ sinon.assert = assert;
3552
+ }
3553
+ }(typeof sinon == "object" && sinon || null, typeof window != "undefined" ? window : global));
3554
+
3555
+ return sinon;}.call(typeof window != 'undefined' && window || {}));