backbone-jasmine 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,41 @@
1
+ (function(global) {
2
+
3
+ var spyMatchers = "called calledOnce calledTwice calledThrice calledBefore calledAfter calledOn alwaysCalledOn calledWith alwaysCalledWith calledWithExactly alwaysCalledWithExactly".split(" "),
4
+ i = spyMatchers.length,
5
+ spyMatcherHash = {},
6
+ unusualMatchers = {
7
+ "returned": "toHaveReturned",
8
+ "alwaysReturned": "toHaveAlwaysReturned"
9
+ },
10
+
11
+ getMatcherFunction = function(sinonName) {
12
+ return function() {
13
+ var sinonProperty = this.actual[sinonName];
14
+ return (typeof sinonProperty === 'function') ? sinonProperty.apply(this.actual, arguments) : sinonProperty;
15
+ };
16
+ };
17
+
18
+ while(i--) {
19
+ var sinonName = spyMatchers[i],
20
+ matcherName = "toHaveBeen" + sinonName.charAt(0).toUpperCase() + sinonName.slice(1);
21
+
22
+ spyMatcherHash[matcherName] = getMatcherFunction(sinonName);
23
+ };
24
+
25
+ for (var j in unusualMatchers) {
26
+ spyMatcherHash[unusualMatchers[j]] = getMatcherFunction(j);
27
+ }
28
+
29
+ global.sinonJasmine = {
30
+ getMatchers: function() {
31
+ return spyMatcherHash;
32
+ }
33
+ };
34
+
35
+ })(window);
36
+
37
+ beforeEach(function() {
38
+
39
+ this.addMatchers(sinonJasmine.getMatchers());
40
+
41
+ });
@@ -0,0 +1,76 @@
1
+ # src_files
2
+ #
3
+ # Return an array of filepaths relative to src_dir to include before jasmine specs.
4
+ # Default: []
5
+ #
6
+ # EXAMPLE:
7
+ #
8
+ # src_files:
9
+ # - lib/source1.js
10
+ # - lib/source2.js
11
+ # - dist/**/*.js
12
+ #
13
+ src_files:
14
+ - spec/javascripts/support/*.js
15
+ - assets/application.js
16
+ # stylesheets
17
+ #
18
+ # Return an array of stylesheet filepaths relative to src_dir to include before jasmine specs.
19
+ # Default: []
20
+ #
21
+ # EXAMPLE:
22
+ #
23
+ # stylesheets:
24
+ # - css/style.css
25
+ # - stylesheets/*.css
26
+ #
27
+ stylesheets:
28
+ - stylesheets/**/*.css
29
+
30
+ # helpers
31
+ #
32
+ # Return an array of filepaths relative to spec_dir to include before jasmine specs.
33
+ # Default: ["helpers/**/*.js"]
34
+ #
35
+ # EXAMPLE:
36
+ #
37
+ # helpers:
38
+ # - helpers/**/*.js
39
+ #
40
+ helpers:
41
+ - helpers/**/*.js
42
+
43
+ # spec_files
44
+ #
45
+ # Return an array of filepaths relative to spec_dir to include.
46
+ # Default: ["**/*[sS]pec.js"]
47
+ #
48
+ # EXAMPLE:
49
+ #
50
+ # spec_files:
51
+ # - **/*[sS]pec.js
52
+ #
53
+ spec_files:
54
+ - '**/*[sS]pec.js'
55
+
56
+ # src_dir
57
+ #
58
+ # Source directory path. Your src_files must be returned relative to this path. Will use root if left blank.
59
+ # Default: project root
60
+ #
61
+ # EXAMPLE:
62
+ #
63
+ # src_dir: public
64
+ #
65
+ src_dir:
66
+
67
+ # spec_dir
68
+ #
69
+ # Spec directory path. Your spec_files must be returned relative to this path.
70
+ # Default: spec/javascripts
71
+ #
72
+ # EXAMPLE:
73
+ #
74
+ # spec_dir: spec/javascripts
75
+ #
76
+ spec_dir: spec/javascripts
@@ -0,0 +1,23 @@
1
+ module Jasmine
2
+ class Config
3
+
4
+ # Add your overrides or custom config code here
5
+
6
+ end
7
+ end
8
+
9
+
10
+ # Note - this is necessary for rspec2, which has removed the backtrace
11
+ module Jasmine
12
+ class SpecBuilder
13
+ def declare_spec(parent, spec)
14
+ me = self
15
+ example_name = spec["name"]
16
+ @spec_ids << spec["id"]
17
+ backtrace = @example_locations[parent.description + " " + example_name]
18
+ parent.it example_name, {} do
19
+ me.report_spec(spec["id"])
20
+ end
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,32 @@
1
+ $:.unshift(ENV['JASMINE_GEM_PATH']) if ENV['JASMINE_GEM_PATH'] # for gem testing purposes
2
+
3
+ require 'rubygems'
4
+ require 'jasmine'
5
+ jasmine_config_overrides = File.expand_path(File.join(File.dirname(__FILE__), 'jasmine_config.rb'))
6
+ require jasmine_config_overrides if File.exist?(jasmine_config_overrides)
7
+ if Jasmine::rspec2?
8
+ require 'rspec'
9
+ else
10
+ require 'spec'
11
+ end
12
+
13
+ jasmine_config = Jasmine::Config.new
14
+ spec_builder = Jasmine::SpecBuilder.new(jasmine_config)
15
+
16
+ should_stop = false
17
+
18
+ if Jasmine::rspec2?
19
+ RSpec.configuration.after(:suite) do
20
+ spec_builder.stop if should_stop
21
+ end
22
+ else
23
+ Spec::Runner.configure do |config|
24
+ config.after(:suite) do
25
+ spec_builder.stop if should_stop
26
+ end
27
+ end
28
+ end
29
+
30
+ spec_builder.start
31
+ should_stop = true
32
+ spec_builder.declare_suites
@@ -0,0 +1,2821 @@
1
+ /**
2
+ * Sinon.JS 1.1.1, 2011/05/17
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
+ /*jslint eqeqeq: false, onevar: false, forin: true, nomen: false, regexp: false, plusplus: false*/
37
+ /*global module, require, __dirname, document*/
38
+ /**
39
+ * Sinon core utilities. For internal use only.
40
+ *
41
+ * @author Christian Johansen (christian@cjohansen.no)
42
+ * @license BSD
43
+ *
44
+ * Copyright (c) 2010-2011 Christian Johansen
45
+ */
46
+
47
+ var sinon = (function () {
48
+ var div = typeof document != "undefined" && document.createElement("div");
49
+
50
+ function isNode(obj) {
51
+ var success = false;
52
+
53
+ try {
54
+ obj.appendChild(div);
55
+ success = div.parentNode == obj;
56
+ } catch (e) {
57
+ return false;
58
+ } finally {
59
+ try {
60
+ obj.removeChild(div);
61
+ } catch (e) {}
62
+ }
63
+
64
+ return success;
65
+ }
66
+
67
+ function isElement(obj) {
68
+ return div && obj && obj.nodeType === 1 && isNode(obj);
69
+ }
70
+
71
+ return {
72
+ wrapMethod: function wrapMethod(object, property, method) {
73
+ if (!object) {
74
+ throw new TypeError("Should wrap property of object");
75
+ }
76
+
77
+ if (typeof method != "function") {
78
+ throw new TypeError("Method wrapper should be function");
79
+ }
80
+
81
+ var wrappedMethod = object[property];
82
+ var type = typeof wrappedMethod;
83
+
84
+ if (type != "function") {
85
+ throw new TypeError("Attempted to wrap " + type + " property " + property +
86
+ " as function");
87
+ }
88
+
89
+ if (wrappedMethod.restore && wrappedMethod.restore.sinon) {
90
+ throw new TypeError("Attempted to wrap " + property + " which is already wrapped");
91
+ }
92
+
93
+ if (wrappedMethod.calledBefore) {
94
+ var verb = !!wrappedMethod.returns ? "stubbed" : "spied on";
95
+ throw new TypeError("Attempted to wrap " + property + " which is already " + verb);
96
+ }
97
+
98
+ object[property] = method;
99
+ method.displayName = property;
100
+
101
+ method.restore = function () {
102
+ object[property] = wrappedMethod;
103
+ };
104
+
105
+ method.restore.sinon = true;
106
+
107
+ return method;
108
+ },
109
+
110
+ extend: function extend(target) {
111
+ for (var i = 1, l = arguments.length; i < l; i += 1) {
112
+ for (var prop in arguments[i]) {
113
+ if (arguments[i].hasOwnProperty(prop)) {
114
+ target[prop] = arguments[i][prop];
115
+ }
116
+
117
+ // DONT ENUM bug, only care about toString
118
+ if (arguments[i].hasOwnProperty("toString") &&
119
+ arguments[i].toString != target.toString) {
120
+ target.toString = arguments[i].toString;
121
+ }
122
+ }
123
+ }
124
+
125
+ return target;
126
+ },
127
+
128
+ create: function create(proto) {
129
+ var F = function () {};
130
+ F.prototype = proto;
131
+ return new F();
132
+ },
133
+
134
+ deepEqual: function deepEqual(a, b) {
135
+ if (typeof a != "object" || typeof b != "object") {
136
+ return a === b;
137
+ }
138
+
139
+ if (isElement(a) || isElement(b)) {
140
+ return a === b;
141
+ }
142
+
143
+ if (a === b) {
144
+ return true;
145
+ }
146
+
147
+ if (Object.prototype.toString.call(a) == "[object Array]") {
148
+ if (a.length !== b.length) {
149
+ return false;
150
+ }
151
+
152
+ for (var i = 0, l = a.length; i < l; i += 1) {
153
+ if (!deepEqual(a[i], b[i])) {
154
+ return false;
155
+ }
156
+ }
157
+
158
+ return true;
159
+ }
160
+
161
+ var prop, aLength = 0, bLength = 0;
162
+
163
+ for (prop in a) {
164
+ aLength += 1;
165
+
166
+ if (!deepEqual(a[prop], b[prop])) {
167
+ return false;
168
+ }
169
+ }
170
+
171
+ for (prop in b) {
172
+ bLength += 1;
173
+ }
174
+
175
+ if (aLength != bLength) {
176
+ return false;
177
+ }
178
+
179
+ return true;
180
+ },
181
+
182
+ keys: function keys(object) {
183
+ var objectKeys = [];
184
+
185
+ for (var prop in object) {
186
+ if (object.hasOwnProperty(prop)) {
187
+ objectKeys.push(prop);
188
+ }
189
+ }
190
+
191
+ return objectKeys.sort();
192
+ },
193
+
194
+ functionName: function functionName(func) {
195
+ var name = func.displayName || func.name;
196
+
197
+ // Use function decomposition as a last resort to get function
198
+ // name. Does not rely on function decomposition to work - if it
199
+ // doesn't debugging will be slightly less informative
200
+ // (i.e. toString will say 'spy' rather than 'myFunc').
201
+ if (!name) {
202
+ var matches = func.toString().match(/function ([^\s\(]+)/);
203
+ name = matches && matches[1];
204
+ }
205
+
206
+ return name;
207
+ },
208
+
209
+ functionToString: function toString() {
210
+ if (this.getCall && this.callCount) {
211
+ var thisValue, prop, i = this.callCount;
212
+
213
+ while (i--) {
214
+ thisValue = this.getCall(i).thisValue;
215
+
216
+ for (prop in thisValue) {
217
+ if (thisValue[prop] === this) {
218
+ return prop;
219
+ }
220
+ }
221
+ }
222
+ }
223
+
224
+ return this.displayName || "sinon fake";
225
+ },
226
+
227
+ getConfig: function (custom) {
228
+ var config = {};
229
+ custom = custom || {};
230
+ var defaults = sinon.defaultConfig;
231
+
232
+ for (var prop in defaults) {
233
+ if (defaults.hasOwnProperty(prop)) {
234
+ config[prop] = custom.hasOwnProperty(prop) ? custom[prop] : defaults[prop];
235
+ }
236
+ }
237
+
238
+ return config;
239
+ },
240
+
241
+ format: function (val) {
242
+ return "" + val;
243
+ },
244
+
245
+ defaultConfig: {
246
+ injectIntoThis: true,
247
+ injectInto: null,
248
+ properties: ["spy", "stub", "mock", "clock", "server", "requests"],
249
+ useFakeTimers: true,
250
+ useFakeServer: true
251
+ }
252
+ };
253
+ }());
254
+
255
+ if (typeof module == "object" && typeof require == "function") {
256
+ module.exports = sinon;
257
+ module.exports.spy = require("./sinon/spy");
258
+ module.exports.stub = require("./sinon/stub");
259
+ module.exports.mock = require("./sinon/mock");
260
+ module.exports.collection = require("./sinon/collection");
261
+ module.exports.assert = require("./sinon/assert");
262
+ module.exports.sandbox = require("./sinon/sandbox");
263
+ module.exports.test = require("./sinon/test");
264
+ module.exports.testCase = require("./sinon/test_case");
265
+ module.exports.assert = require("./sinon/assert");
266
+ }
267
+
268
+ /* @depend ../sinon.js */
269
+ /*jslint eqeqeq: false, onevar: false, plusplus: false*/
270
+ /*global module, require, sinon*/
271
+ /**
272
+ * Spy functions
273
+ *
274
+ * @author Christian Johansen (christian@cjohansen.no)
275
+ * @license BSD
276
+ *
277
+ * Copyright (c) 2010-2011 Christian Johansen
278
+ */
279
+
280
+ (function (sinon) {
281
+ var commonJSModule = typeof module == "object" && typeof require == "function";
282
+ var spyCall;
283
+ var callId = 0;
284
+
285
+ if (!sinon && commonJSModule) {
286
+ sinon = require("sinon");
287
+ }
288
+
289
+ if (!sinon) {
290
+ return;
291
+ }
292
+
293
+ function spy(object, property) {
294
+ if (!property && typeof object == "function") {
295
+ return spy.create(object);
296
+ }
297
+
298
+ if (!object || !property) {
299
+ return spy.create(function () {});
300
+ }
301
+
302
+ var method = object[property];
303
+ return sinon.wrapMethod(object, property, spy.create(method));
304
+ }
305
+
306
+ sinon.extend(spy, (function () {
307
+ var slice = Array.prototype.slice;
308
+
309
+ function delegateToCalls(api, method, matchAny, actual) {
310
+ api[method] = function () {
311
+ if (!this.called) {
312
+ return false;
313
+ }
314
+
315
+ var currentCall;
316
+ var matches = 0;
317
+
318
+ for (var i = 0, l = this.callCount; i < l; i += 1) {
319
+ currentCall = this.getCall(i);
320
+
321
+ if (currentCall[actual || method].apply(currentCall, arguments)) {
322
+ matches += 1;
323
+
324
+ if (matchAny) {
325
+ return true;
326
+ }
327
+ }
328
+ }
329
+
330
+ return matches === this.callCount;
331
+ };
332
+ }
333
+
334
+ function matchingFake(fakes, args, strict) {
335
+ if (!fakes) {
336
+ return;
337
+ }
338
+
339
+ var alen = args.length;
340
+
341
+ for (var i = 0, l = fakes.length; i < l; i++) {
342
+ if (fakes[i].matches(args, strict)) {
343
+ return fakes[i];
344
+ }
345
+ }
346
+ }
347
+
348
+ var uuid = 0;
349
+
350
+ // Public API
351
+ var spyApi = {
352
+ reset: function () {
353
+ this.called = false;
354
+ this.calledOnce = false;
355
+ this.calledTwice = false;
356
+ this.calledThrice = false;
357
+ this.callCount = 0;
358
+ this.args = [];
359
+ this.returnValues = [];
360
+ this.thisValues = [];
361
+ this.exceptions = [];
362
+ this.callIds = [];
363
+ },
364
+
365
+ create: function create(func) {
366
+ var name;
367
+
368
+ if (typeof func != "function") {
369
+ func = function () {};
370
+ } else {
371
+ name = sinon.functionName(func);
372
+ }
373
+
374
+ function proxy() {
375
+ return proxy.invoke(func, this, slice.call(arguments));
376
+ }
377
+
378
+ sinon.extend(proxy, spy);
379
+ delete proxy.create;
380
+ sinon.extend(proxy, func);
381
+
382
+ proxy.reset();
383
+ proxy.prototype = func.prototype;
384
+ proxy.displayName = name || "spy";
385
+ proxy.toString = sinon.functionToString;
386
+ proxy._create = sinon.spy.create;
387
+ proxy.id = "spy#" + uuid++;
388
+
389
+ return proxy;
390
+ },
391
+
392
+ invoke: function invoke(func, thisValue, args) {
393
+ var matching = matchingFake(this.fakes, args);
394
+ var exception, returnValue;
395
+ this.called = true;
396
+ this.callCount += 1;
397
+ this.calledOnce = this.callCount == 1;
398
+ this.calledTwice = this.callCount == 2;
399
+ this.calledThrice = this.callCount == 3;
400
+ this.thisValues.push(thisValue);
401
+ this.args.push(args);
402
+ this.callIds.push(callId++);
403
+
404
+ try {
405
+ if (matching) {
406
+ returnValue = matching.invoke(func, thisValue, args);
407
+ } else {
408
+ returnValue = (this.func || func).apply(thisValue, args);
409
+ }
410
+ } catch (e) {
411
+ this.returnValues.push(undefined);
412
+ exception = e;
413
+ throw e;
414
+ } finally {
415
+ this.exceptions.push(exception);
416
+ }
417
+
418
+ this.returnValues.push(returnValue);
419
+
420
+ return returnValue;
421
+ },
422
+
423
+ getCall: function getCall(i) {
424
+ if (i < 0 || i >= this.callCount) {
425
+ return null;
426
+ }
427
+
428
+ return spyCall.create(this, this.thisValues[i], this.args[i],
429
+ this.returnValues[i], this.exceptions[i],
430
+ this.callIds[i]);
431
+ },
432
+
433
+ calledBefore: function calledBefore(spyFn) {
434
+ if (!this.called) {
435
+ return false;
436
+ }
437
+
438
+ if (!spyFn.called) {
439
+ return true;
440
+ }
441
+
442
+ return this.callIds[0] < spyFn.callIds[0];
443
+ },
444
+
445
+ calledAfter: function calledAfter(spyFn) {
446
+ if (!this.called || !spyFn.called) {
447
+ return false;
448
+ }
449
+
450
+ return this.callIds[this.callCount - 1] > spyFn.callIds[spyFn.callCount - 1];
451
+ },
452
+
453
+ withArgs: function () {
454
+ var args = slice.call(arguments);
455
+
456
+ if (this.fakes) {
457
+ var match = matchingFake(this.fakes, args, true);
458
+
459
+ if (match) {
460
+ return match;
461
+ }
462
+ } else {
463
+ this.fakes = [];
464
+ }
465
+
466
+ var original = this;
467
+ var fake = this._create();
468
+ fake.matchingAguments = args;
469
+ this.fakes.push(fake);
470
+
471
+ fake.withArgs = function () {
472
+ return original.withArgs.apply(original, arguments);
473
+ };
474
+
475
+ return fake;
476
+ },
477
+
478
+ matches: function (args, strict) {
479
+ var margs = this.matchingAguments;
480
+
481
+ if (margs.length <= args.length &&
482
+ sinon.deepEqual(margs, args.slice(0, margs.length))) {
483
+ return !strict || margs.length == args.length;
484
+ }
485
+ }
486
+ };
487
+
488
+ delegateToCalls(spyApi, "calledOn", true);
489
+ delegateToCalls(spyApi, "alwaysCalledOn", false, "calledOn");
490
+ delegateToCalls(spyApi, "calledWith", true);
491
+ delegateToCalls(spyApi, "alwaysCalledWith", false, "calledWith");
492
+ delegateToCalls(spyApi, "calledWithExactly", true);
493
+ delegateToCalls(spyApi, "alwaysCalledWithExactly", false, "calledWithExactly");
494
+ delegateToCalls(spyApi, "threw", true);
495
+ delegateToCalls(spyApi, "alwaysThrew", false, "threw");
496
+ delegateToCalls(spyApi, "returned", true);
497
+ delegateToCalls(spyApi, "alwaysReturned", false, "returned");
498
+
499
+ return spyApi;
500
+ }()));
501
+
502
+ spyCall = (function () {
503
+ return {
504
+ create: function create(spy, thisValue, args, returnValue, exception, id) {
505
+ var proxyCall = sinon.create(spyCall);
506
+ delete proxyCall.create;
507
+ proxyCall.proxy = spy;
508
+ proxyCall.thisValue = thisValue;
509
+ proxyCall.args = args;
510
+ proxyCall.returnValue = returnValue;
511
+ proxyCall.exception = exception;
512
+ proxyCall.callId = typeof id == "number" && id || callId++;
513
+
514
+ return proxyCall;
515
+ },
516
+
517
+ calledOn: function calledOn(thisValue) {
518
+ return this.thisValue === thisValue;
519
+ },
520
+
521
+ calledWith: function calledWith() {
522
+ for (var i = 0, l = arguments.length; i < l; i += 1) {
523
+ if (!sinon.deepEqual(arguments[i], this.args[i])) {
524
+ return false;
525
+ }
526
+ }
527
+
528
+ return true;
529
+ },
530
+
531
+ calledWithExactly: function calledWithExactly() {
532
+ return arguments.length == this.args.length &&
533
+ this.calledWith.apply(this, arguments);
534
+ },
535
+
536
+ returned: function returned(value) {
537
+ return this.returnValue === value;
538
+ },
539
+
540
+ threw: function threw(error) {
541
+ if (typeof error == "undefined" || !this.exception) {
542
+ return !!this.exception;
543
+ }
544
+
545
+ if (typeof error == "string") {
546
+ return this.exception.name == error;
547
+ }
548
+
549
+ return this.exception === error;
550
+ },
551
+
552
+ calledBefore: function (other) {
553
+ return this.callId < other.callId;
554
+ },
555
+
556
+ calledAfter: function (other) {
557
+ return this.callId > other.callId;
558
+ },
559
+
560
+ toString: function () {
561
+ var callStr = this.proxy.toString() + "(";
562
+ var args = [];
563
+
564
+ for (var i = 0, l = this.args.length; i < l; ++i) {
565
+ args.push(sinon.format(this.args[i]));
566
+ }
567
+
568
+ callStr = callStr + args.join(", ") + ")";
569
+
570
+ if (typeof this.returnValue != "undefined") {
571
+ callStr += " => " + sinon.format(this.returnValue);
572
+ }
573
+
574
+ if (this.exception) {
575
+ callStr += " !" + this.exception.name;
576
+
577
+ if (this.exception.message) {
578
+ callStr += "(" + this.exception.message + ")";
579
+ }
580
+ }
581
+
582
+ return callStr;
583
+ }
584
+ };
585
+ }());
586
+
587
+ if (commonJSModule) {
588
+ module.exports = spy;
589
+ } else {
590
+ sinon.spy = spy;
591
+ }
592
+
593
+ sinon.spyCall = spyCall;
594
+ }(typeof sinon == "object" && sinon || null));
595
+
596
+ /**
597
+ * @depend ../sinon.js
598
+ * @depend spy.js
599
+ */
600
+ /*jslint eqeqeq: false, onevar: false*/
601
+ /*global module, require, sinon*/
602
+ /**
603
+ * Stub functions
604
+ *
605
+ * @author Christian Johansen (christian@cjohansen.no)
606
+ * @license BSD
607
+ *
608
+ * Copyright (c) 2010-2011 Christian Johansen
609
+ */
610
+
611
+ (function (sinon) {
612
+ var commonJSModule = typeof module == "object" && typeof require == "function";
613
+
614
+ if (!sinon && commonJSModule) {
615
+ sinon = require("sinon");
616
+ }
617
+
618
+ if (!sinon) {
619
+ return;
620
+ }
621
+
622
+ function stub(object, property, func) {
623
+ if (!!func && typeof func != "function") {
624
+ throw new TypeError("Custom stub should be function");
625
+ }
626
+
627
+ var wrapper;
628
+
629
+ if (func) {
630
+ wrapper = sinon.spy && sinon.spy.create ? sinon.spy.create(func) : func;
631
+ } else {
632
+ wrapper = stub.create();
633
+ }
634
+
635
+ if (!object && !property) {
636
+ return sinon.stub.create();
637
+ }
638
+
639
+ if (!property && !!object && typeof object == "object") {
640
+ for (var prop in object) {
641
+ if (object.hasOwnProperty(prop) && typeof object[prop] == "function") {
642
+ stub(object, prop);
643
+ }
644
+ }
645
+
646
+ return object;
647
+ }
648
+
649
+ return sinon.wrapMethod(object, property, wrapper);
650
+ }
651
+
652
+ function getCallback(stub, args) {
653
+ if (stub.callArgAt < 0) {
654
+ for (var i = 0, l = args.length; i < l; ++i) {
655
+ if (!stub.callArgProp && typeof args[i] == "function") {
656
+ return args[i];
657
+ }
658
+
659
+ if (stub.callArgProp && args[i] &&
660
+ typeof args[i][stub.callArgProp] == "function") {
661
+ return args[i][stub.callArgProp];
662
+ }
663
+ }
664
+
665
+ return null;
666
+ }
667
+
668
+ return args[stub.callArgAt];
669
+ }
670
+
671
+ var join = Array.prototype.join;
672
+
673
+ function getCallbackError(stub, func, args) {
674
+ if (stub.callArgAt < 0) {
675
+ var msg;
676
+
677
+ if (stub.callArgProp) {
678
+ msg = sinon.functionName(stub) +
679
+ " expected to yield to '" + stub.callArgProp +
680
+ "', but no object with such a property was passed."
681
+ } else {
682
+ msg = sinon.functionName(stub) +
683
+ " expected to yield, but no callback was passed."
684
+ }
685
+
686
+ if (args.length > 0) {
687
+ msg += " Received [" + join.call(args, ", ") + "]";
688
+ }
689
+
690
+ return msg;
691
+ }
692
+
693
+ return "argument at index " + stub.callArgAt + " is not a function: " + func;
694
+ }
695
+
696
+ function callCallback(stub, args) {
697
+ if (typeof stub.callArgAt == "number") {
698
+ var func = getCallback(stub, args);
699
+
700
+ if (typeof func != "function") {
701
+ throw new TypeError(getCallbackError(stub, func, args));
702
+ }
703
+
704
+ func.apply(null, stub.callbackArguments);
705
+ }
706
+ }
707
+
708
+ var uuid = 0;
709
+
710
+ sinon.extend(stub, (function () {
711
+ var slice = Array.prototype.slice;
712
+
713
+ function throwsException(error, message) {
714
+ if (typeof error == "string") {
715
+ this.exception = new Error(message || "");
716
+ this.exception.name = error;
717
+ } else if (!error) {
718
+ this.exception = new Error("Error");
719
+ } else {
720
+ this.exception = error;
721
+ }
722
+
723
+ return this;
724
+ }
725
+
726
+ return {
727
+ create: function create() {
728
+ var functionStub = function () {
729
+ if (functionStub.exception) {
730
+ throw functionStub.exception;
731
+ }
732
+
733
+ callCallback(functionStub, arguments);
734
+
735
+ return functionStub.returnValue;
736
+ };
737
+
738
+ functionStub.id = "stub#" + uuid++;
739
+ var orig = functionStub;
740
+ functionStub = sinon.spy.create(functionStub);
741
+ functionStub.func = orig;
742
+
743
+ sinon.extend(functionStub, stub);
744
+ functionStub._create = sinon.stub.create;
745
+ functionStub.displayName = "stub";
746
+ functionStub.toString = sinon.functionToString;
747
+
748
+ return functionStub;
749
+ },
750
+
751
+ returns: function returns(value) {
752
+ this.returnValue = value;
753
+
754
+ return this;
755
+ },
756
+
757
+ "throws": throwsException,
758
+ throwsException: throwsException,
759
+
760
+ callsArg: function callsArg(pos) {
761
+ if (typeof pos != "number") {
762
+ throw new TypeError("argument index is not number");
763
+ }
764
+
765
+ this.callArgAt = pos;
766
+ this.callbackArguments = [];
767
+
768
+ return this;
769
+ },
770
+
771
+ callsArgWith: function callsArgWith(pos) {
772
+ if (typeof pos != "number") {
773
+ throw new TypeError("argument index is not number");
774
+ }
775
+
776
+ this.callArgAt = pos;
777
+ this.callbackArguments = slice.call(arguments, 1);
778
+
779
+ return this;
780
+ },
781
+
782
+ yields: function () {
783
+ this.callArgAt = -1;
784
+ this.callbackArguments = slice.call(arguments, 0);
785
+
786
+ return this;
787
+ },
788
+
789
+ yieldsTo: function (prop) {
790
+ this.callArgAt = -1;
791
+ this.callArgProp = prop;
792
+ this.callbackArguments = slice.call(arguments, 1);
793
+
794
+ return this;
795
+ }
796
+ };
797
+ }()));
798
+
799
+ if (commonJSModule) {
800
+ module.exports = stub;
801
+ } else {
802
+ sinon.stub = stub;
803
+ }
804
+ }(typeof sinon == "object" && sinon || null));
805
+
806
+ /**
807
+ * @depend ../sinon.js
808
+ * @depend stub.js
809
+ */
810
+ /*jslint eqeqeq: false, onevar: false, nomen: false*/
811
+ /*global module, require, sinon*/
812
+ /**
813
+ * Mock functions.
814
+ *
815
+ * @author Christian Johansen (christian@cjohansen.no)
816
+ * @license BSD
817
+ *
818
+ * Copyright (c) 2010-2011 Christian Johansen
819
+ */
820
+
821
+ (function (sinon) {
822
+ var commonJSModule = typeof module == "object" && typeof require == "function";
823
+
824
+ if (!sinon && commonJSModule) {
825
+ sinon = require("sinon");
826
+ }
827
+
828
+ if (!sinon) {
829
+ return;
830
+ }
831
+
832
+ function mock(object) {
833
+ if (!object) {
834
+ return sinon.expectation.create("Anonymous mock");
835
+ }
836
+
837
+ return mock.create(object);
838
+ }
839
+
840
+ sinon.mock = mock;
841
+
842
+ sinon.extend(mock, (function () {
843
+ function each(collection, callback) {
844
+ if (!collection) {
845
+ return;
846
+ }
847
+
848
+ for (var i = 0, l = collection.length; i < l; i += 1) {
849
+ callback(collection[i]);
850
+ }
851
+ }
852
+
853
+ return {
854
+ create: function create(object) {
855
+ if (!object) {
856
+ throw new TypeError("object is null");
857
+ }
858
+
859
+ var mockObject = sinon.extend({}, mock);
860
+ mockObject.object = object;
861
+ delete mockObject.create;
862
+
863
+ return mockObject;
864
+ },
865
+
866
+ expects: function expects(method) {
867
+ if (!method) {
868
+ throw new TypeError("method is falsy");
869
+ }
870
+
871
+ if (!this.expectations) {
872
+ this.expectations = {};
873
+ this.proxies = [];
874
+ }
875
+
876
+ if (!this.expectations[method]) {
877
+ this.expectations[method] = [];
878
+ var mockObject = this;
879
+
880
+ sinon.wrapMethod(this.object, method, function () {
881
+ return mockObject.invokeMethod(method, this, arguments);
882
+ });
883
+
884
+ this.proxies.push(method);
885
+ }
886
+
887
+ var expectation = sinon.expectation.create(method);
888
+ this.expectations[method].push(expectation);
889
+
890
+ return expectation;
891
+ },
892
+
893
+ restore: function restore() {
894
+ var object = this.object;
895
+
896
+ each(this.proxies, function (proxy) {
897
+ if (typeof object[proxy].restore == "function") {
898
+ object[proxy].restore();
899
+ }
900
+ });
901
+ },
902
+
903
+ verify: function verify() {
904
+ var expectations = this.expectations || {};
905
+ var messages = [], met = [];
906
+
907
+ each(this.proxies, function (proxy) {
908
+ each(expectations[proxy], function (expectation) {
909
+ if (!expectation.met()) {
910
+ messages.push(expectation.toString());
911
+ } else {
912
+ met.push(expectation.toString());
913
+ }
914
+ });
915
+ });
916
+
917
+ this.restore();
918
+
919
+ if (messages.length > 0) {
920
+ err(messages.concat(met).join("\n"));
921
+ }
922
+
923
+ return true;
924
+ },
925
+
926
+ invokeMethod: function invokeMethod(method, thisValue, args) {
927
+ var expectations = this.expectations && this.expectations[method];
928
+ var length = expectations && expectations.length || 0;
929
+
930
+ for (var i = 0; i < length; i += 1) {
931
+ if (!expectations[i].met() &&
932
+ expectations[i].allowsCall(thisValue, args)) {
933
+ return expectations[i].apply(thisValue, args);
934
+ }
935
+ }
936
+
937
+ var messages = [];
938
+
939
+ for (i = 0; i < length; i += 1) {
940
+ messages.push(" " + expectations[i].toString());
941
+ }
942
+
943
+ messages.unshift("Unexpected call: " + sinon.spyCall.toString.call({
944
+ proxy: method,
945
+ args: args
946
+ }));
947
+
948
+ err(messages.join("\n"));
949
+ }
950
+ };
951
+ }()));
952
+
953
+ function err(message) {
954
+ var exception = new Error(message);
955
+ exception.name = "ExpectationError";
956
+
957
+ throw exception;
958
+ }
959
+
960
+ sinon.expectation = (function () {
961
+ var slice = Array.prototype.slice;
962
+ var _invoke = sinon.spy.invoke;
963
+
964
+ function timesInWords(times) {
965
+ if (times == 0) {
966
+ return "never";
967
+ } else if (times == 1) {
968
+ return "once";
969
+ } else if (times == 2) {
970
+ return "twice";
971
+ } else if (times == 3) {
972
+ return "thrice";
973
+ }
974
+
975
+ return times + " times";
976
+ }
977
+
978
+ function callCountInWords(callCount) {
979
+ if (callCount == 0) {
980
+ return "never called";
981
+ } else {
982
+ return "called " + timesInWords(callCount);
983
+ }
984
+ }
985
+
986
+ function expectedCallCountInWords(expectation) {
987
+ var min = expectation.minCalls;
988
+ var max = expectation.maxCalls;
989
+
990
+ if (typeof min == "number" && typeof max == "number") {
991
+ var str = timesInWords(min);
992
+
993
+ if (min != max) {
994
+ str = "at least " + str + " and at most " + timesInWords(max);
995
+ }
996
+
997
+ return str;
998
+ }
999
+
1000
+ if (typeof min == "number") {
1001
+ return "at least " + timesInWords(min);
1002
+ }
1003
+
1004
+ return "at most " + timesInWords(max);
1005
+ }
1006
+
1007
+ function receivedMinCalls(expectation) {
1008
+ var hasMinLimit = typeof expectation.minCalls == "number";
1009
+ return !hasMinLimit || expectation.callCount >= expectation.minCalls;
1010
+ }
1011
+
1012
+ function receivedMaxCalls(expectation) {
1013
+ if (typeof expectation.maxCalls != "number") {
1014
+ return false;
1015
+ }
1016
+
1017
+ return expectation.callCount == expectation.maxCalls;
1018
+ }
1019
+
1020
+ return {
1021
+ minCalls: 1,
1022
+ maxCalls: 1,
1023
+
1024
+ create: function create(methodName) {
1025
+ var expectation = sinon.extend(sinon.stub.create(), sinon.expectation);
1026
+ delete expectation.create;
1027
+ expectation.method = methodName;
1028
+
1029
+ return expectation;
1030
+ },
1031
+
1032
+ invoke: function invoke(func, thisValue, args) {
1033
+ this.verifyCallAllowed(thisValue, args);
1034
+
1035
+ return _invoke.apply(this, arguments);
1036
+ },
1037
+
1038
+ atLeast: function atLeast(num) {
1039
+ if (typeof num != "number") {
1040
+ throw new TypeError("'" + num + "' is not number");
1041
+ }
1042
+
1043
+ if (!this.limitsSet) {
1044
+ this.maxCalls = null;
1045
+ this.limitsSet = true;
1046
+ }
1047
+
1048
+ this.minCalls = num;
1049
+
1050
+ return this;
1051
+ },
1052
+
1053
+ atMost: function atMost(num) {
1054
+ if (typeof num != "number") {
1055
+ throw new TypeError("'" + num + "' is not number");
1056
+ }
1057
+
1058
+ if (!this.limitsSet) {
1059
+ this.minCalls = null;
1060
+ this.limitsSet = true;
1061
+ }
1062
+
1063
+ this.maxCalls = num;
1064
+
1065
+ return this;
1066
+ },
1067
+
1068
+ never: function never() {
1069
+ return this.exactly(0);
1070
+ },
1071
+
1072
+ once: function once() {
1073
+ return this.exactly(1);
1074
+ },
1075
+
1076
+ twice: function twice() {
1077
+ return this.exactly(2);
1078
+ },
1079
+
1080
+ thrice: function thrice() {
1081
+ return this.exactly(3);
1082
+ },
1083
+
1084
+ exactly: function exactly(num) {
1085
+ if (typeof num != "number") {
1086
+ throw new TypeError("'" + num + "' is not a number");
1087
+ }
1088
+
1089
+ this.atLeast(num);
1090
+ return this.atMost(num);
1091
+ },
1092
+
1093
+ met: function met() {
1094
+ return !this.failed && receivedMinCalls(this);
1095
+ },
1096
+
1097
+ verifyCallAllowed: function verifyCallAllowed(thisValue, args) {
1098
+ if (receivedMaxCalls(this)) {
1099
+ this.failed = true;
1100
+ err(this.method + " already called " + timesInWords(this.maxCalls));
1101
+ }
1102
+
1103
+ if ("expectedThis" in this && this.expectedThis !== thisValue) {
1104
+ err(this.method + " called with " + thisValue + " as thisValue, expected " +
1105
+ this.expectedThis);
1106
+ }
1107
+
1108
+ if (!("expectedArguments" in this)) {
1109
+ return;
1110
+ }
1111
+
1112
+ if (!args || args.length === 0) {
1113
+ err(this.method + " received no arguments, expected " +
1114
+ this.expectedArguments.join());
1115
+ }
1116
+
1117
+ if (args.length < this.expectedArguments.length) {
1118
+ err(this.method + " received too few arguments (" + args.join() +
1119
+ "), expected " + this.expectedArguments.join());
1120
+ }
1121
+
1122
+ if (this.expectsExactArgCount &&
1123
+ args.length != this.expectedArguments.length) {
1124
+ err(this.method + " received too many arguments (" + args.join() +
1125
+ "), expected " + this.expectedArguments.join());
1126
+ }
1127
+
1128
+ for (var i = 0, l = this.expectedArguments.length; i < l; i += 1) {
1129
+ if (!sinon.deepEqual(this.expectedArguments[i], args[i])) {
1130
+ err(this.method + " received wrong arguments (" + args.join() +
1131
+ "), expected " + this.expectedArguments.join());
1132
+ }
1133
+ }
1134
+ },
1135
+
1136
+ allowsCall: function allowsCall(thisValue, args) {
1137
+ if (this.met()) {
1138
+ return false;
1139
+ }
1140
+
1141
+ if ("expectedThis" in this && this.expectedThis !== thisValue) {
1142
+ return false;
1143
+ }
1144
+
1145
+ if (!("expectedArguments" in this)) {
1146
+ return true;
1147
+ }
1148
+
1149
+ args = args || [];
1150
+
1151
+ if (args.length < this.expectedArguments.length) {
1152
+ return false;
1153
+ }
1154
+
1155
+ if (this.expectsExactArgCount &&
1156
+ args.length != this.expectedArguments.length) {
1157
+ return false;
1158
+ }
1159
+
1160
+ for (var i = 0, l = this.expectedArguments.length; i < l; i += 1) {
1161
+ if (!sinon.deepEqual(this.expectedArguments[i], args[i])) {
1162
+ return false;
1163
+ }
1164
+ }
1165
+
1166
+ return true;
1167
+ },
1168
+
1169
+ withArgs: function withArgs() {
1170
+ this.expectedArguments = slice.call(arguments);
1171
+ return this;
1172
+ },
1173
+
1174
+ withExactArgs: function withExactArgs() {
1175
+ this.withArgs.apply(this, arguments);
1176
+ this.expectsExactArgCount = true;
1177
+ return this;
1178
+ },
1179
+
1180
+ on: function on(thisValue) {
1181
+ this.expectedThis = thisValue;
1182
+ return this;
1183
+ },
1184
+
1185
+ toString: function () {
1186
+ var args = (this.expectedArguments || []).slice();
1187
+
1188
+ if (!this.expectsExactArgCount) {
1189
+ args.push("[...]");
1190
+ }
1191
+
1192
+ var callStr = sinon.spyCall.toString.call({
1193
+ proxy: this.method, args: args
1194
+ });
1195
+
1196
+ var message = callStr.replace(", [...", "[, ...") + " " +
1197
+ expectedCallCountInWords(this);
1198
+
1199
+ if (this.met()) {
1200
+ return "Expectation met: " + message;
1201
+ }
1202
+
1203
+ return "Expected " + message + " (" +
1204
+ callCountInWords(this.callCount) + ")";
1205
+ },
1206
+
1207
+ verify: function verify() {
1208
+ if (!this.met()) {
1209
+ err(this.toString());
1210
+ }
1211
+
1212
+ return true;
1213
+ }
1214
+ };
1215
+ }());
1216
+
1217
+ if (commonJSModule) {
1218
+ module.exports = mock;
1219
+ } else {
1220
+ sinon.mock = mock;
1221
+ }
1222
+ }(typeof sinon == "object" && sinon || null));
1223
+
1224
+ /**
1225
+ * @depend ../sinon.js
1226
+ * @depend stub.js
1227
+ * @depend mock.js
1228
+ */
1229
+ /*jslint eqeqeq: false, onevar: false, forin: true*/
1230
+ /*global module, require, sinon*/
1231
+ /**
1232
+ * Collections of stubs, spies and mocks.
1233
+ *
1234
+ * @author Christian Johansen (christian@cjohansen.no)
1235
+ * @license BSD
1236
+ *
1237
+ * Copyright (c) 2010-2011 Christian Johansen
1238
+ */
1239
+
1240
+ (function (sinon) {
1241
+ var commonJSModule = typeof module == "object" && typeof require == "function";
1242
+
1243
+ if (!sinon && commonJSModule) {
1244
+ sinon = require("sinon");
1245
+ }
1246
+
1247
+ if (!sinon) {
1248
+ return;
1249
+ }
1250
+
1251
+ function getFakes(fakeCollection) {
1252
+ if (!fakeCollection.fakes) {
1253
+ fakeCollection.fakes = [];
1254
+ }
1255
+
1256
+ return fakeCollection.fakes;
1257
+ }
1258
+
1259
+ function each(fakeCollection, method) {
1260
+ var fakes = getFakes(fakeCollection);
1261
+
1262
+ for (var i = 0, l = fakes.length; i < l; i += 1) {
1263
+ if (typeof fakes[i][method] == "function") {
1264
+ fakes[i][method]();
1265
+ }
1266
+ }
1267
+ }
1268
+
1269
+ var collection = {
1270
+ verify: function resolve() {
1271
+ each(this, "verify");
1272
+ },
1273
+
1274
+ restore: function restore() {
1275
+ each(this, "restore");
1276
+ },
1277
+
1278
+ verifyAndRestore: function verifyAndRestore() {
1279
+ var exception;
1280
+
1281
+ try {
1282
+ this.verify();
1283
+ } catch (e) {
1284
+ exception = e;
1285
+ }
1286
+
1287
+ this.restore();
1288
+
1289
+ if (exception) {
1290
+ throw exception;
1291
+ }
1292
+ },
1293
+
1294
+ add: function add(fake) {
1295
+ getFakes(this).push(fake);
1296
+
1297
+ return fake;
1298
+ },
1299
+
1300
+ spy: function spy() {
1301
+ return this.add(sinon.spy.apply(sinon, arguments));
1302
+ },
1303
+
1304
+ stub: function stub(object, property, value) {
1305
+ if (property) {
1306
+ var original = object[property];
1307
+
1308
+ if (typeof original != "function") {
1309
+ if (!object.hasOwnProperty(property)) {
1310
+ throw new TypeError("Cannot stub non-existent own property " + property);
1311
+ }
1312
+
1313
+ object[property] = value;
1314
+
1315
+ return this.add({
1316
+ restore: function () {
1317
+ object[property] = original;
1318
+ }
1319
+ });
1320
+ }
1321
+ }
1322
+
1323
+ return this.add(sinon.stub.apply(sinon, arguments));
1324
+ },
1325
+
1326
+ mock: function mock() {
1327
+ return this.add(sinon.mock.apply(sinon, arguments));
1328
+ },
1329
+
1330
+ inject: function inject(obj) {
1331
+ var col = this;
1332
+
1333
+ obj.spy = function () {
1334
+ return col.spy.apply(col, arguments);
1335
+ };
1336
+
1337
+ obj.stub = function () {
1338
+ return col.stub.apply(col, arguments);
1339
+ };
1340
+
1341
+ obj.mock = function () {
1342
+ return col.mock.apply(col, arguments);
1343
+ };
1344
+
1345
+ return obj;
1346
+ }
1347
+ };
1348
+
1349
+ if (commonJSModule) {
1350
+ module.exports = collection;
1351
+ } else {
1352
+ sinon.collection = collection;
1353
+ }
1354
+ }(typeof sinon == "object" && sinon || null));
1355
+
1356
+ /*jslint eqeqeq: false, plusplus: false, evil: true, onevar: false, browser: true, forin: false*/
1357
+ /*global module, require, window*/
1358
+ /**
1359
+ * Fake timer API
1360
+ * setTimeout
1361
+ * setInterval
1362
+ * clearTimeout
1363
+ * clearInterval
1364
+ * tick
1365
+ * reset
1366
+ * Date
1367
+ *
1368
+ * Inspired by jsUnitMockTimeOut from JsUnit
1369
+ *
1370
+ * @author Christian Johansen (christian@cjohansen.no)
1371
+ * @license BSD
1372
+ *
1373
+ * Copyright (c) 2010-2011 Christian Johansen
1374
+ */
1375
+
1376
+ if (typeof sinon == "undefined") {
1377
+ var sinon = {};
1378
+ }
1379
+
1380
+ sinon.clock = (function () {
1381
+ var id = 0;
1382
+
1383
+ function addTimer(args, recurring) {
1384
+ if (args.length === 0) {
1385
+ throw new Error("Function requires at least 1 parameter");
1386
+ }
1387
+
1388
+ var toId = id++;
1389
+ var delay = args[1] || 0;
1390
+
1391
+ if (!this.timeouts) {
1392
+ this.timeouts = {};
1393
+ }
1394
+
1395
+ this.timeouts[toId] = {
1396
+ id: toId,
1397
+ func: args[0],
1398
+ callAt: this.now + delay
1399
+ };
1400
+
1401
+ if (recurring === true) {
1402
+ this.timeouts[toId].interval = delay;
1403
+ }
1404
+
1405
+ return toId;
1406
+ }
1407
+
1408
+ function parseTime(str) {
1409
+ if (!str) {
1410
+ return 0;
1411
+ }
1412
+
1413
+ var strings = str.split(":");
1414
+ var l = strings.length, i = l;
1415
+ var ms = 0, parsed;
1416
+
1417
+ if (l > 3 || !/^(\d\d:){0,2}\d\d?$/.test(str)) {
1418
+ throw new Error("tick only understands numbers and 'h:m:s'");
1419
+ }
1420
+
1421
+ while (i--) {
1422
+ parsed = parseInt(strings[i], 10);
1423
+
1424
+ if (parsed >= 60) {
1425
+ throw new Error("Invalid time " + str);
1426
+ }
1427
+
1428
+ ms += parsed * Math.pow(60, (l - i - 1));
1429
+ }
1430
+
1431
+ return ms * 1000;
1432
+ }
1433
+
1434
+ function createObject(object) {
1435
+ var newObject;
1436
+
1437
+ if (Object.create) {
1438
+ newObject = Object.create(object);
1439
+ } else {
1440
+ var F = function () {};
1441
+ F.prototype = object;
1442
+ newObject = new F();
1443
+ }
1444
+
1445
+ newObject.Date.clock = newObject;
1446
+ return newObject;
1447
+ }
1448
+
1449
+ return {
1450
+ now: 0,
1451
+
1452
+ create: function create(now) {
1453
+ var clock = createObject(this);
1454
+
1455
+ if (typeof now == "number") {
1456
+ this.now = now;
1457
+ }
1458
+
1459
+ return clock;
1460
+ },
1461
+
1462
+ setTimeout: function setTimeout(callback, timeout) {
1463
+ return addTimer.call(this, arguments, false);
1464
+ },
1465
+
1466
+ clearTimeout: function clearTimeout(timerId) {
1467
+ if (!this.timeouts) {
1468
+ this.timeouts = [];
1469
+ }
1470
+
1471
+ delete this.timeouts[timerId];
1472
+ },
1473
+
1474
+ setInterval: function setInterval(callback, timeout) {
1475
+ return addTimer.call(this, arguments, true);
1476
+ },
1477
+
1478
+ clearInterval: function clearInterval(timerId) {
1479
+ this.clearTimeout(timerId);
1480
+ },
1481
+
1482
+ tick: function tick(ms) {
1483
+ ms = typeof ms == "number" ? ms : parseTime(ms);
1484
+ var tickFrom = this.now, tickTo = this.now + ms, previous = this.now;
1485
+ var timer = this.firstTimerInRange(tickFrom, tickTo);
1486
+
1487
+ while (timer && tickFrom <= tickTo) {
1488
+ if (this.timeouts[timer.id]) {
1489
+ tickFrom = this.now = timer.callAt;
1490
+ this.callTimer(timer);
1491
+ }
1492
+
1493
+ timer = this.firstTimerInRange(previous, tickTo);
1494
+ previous = tickFrom;
1495
+ }
1496
+
1497
+ this.now = tickTo;
1498
+ },
1499
+
1500
+ firstTimerInRange: function (from, to) {
1501
+ var timer, smallest, originalTimer;
1502
+
1503
+ for (var id in this.timeouts) {
1504
+ if (this.timeouts.hasOwnProperty(id)) {
1505
+ if (this.timeouts[id].callAt < from || this.timeouts[id].callAt > to) {
1506
+ continue;
1507
+ }
1508
+
1509
+ if (!smallest || this.timeouts[id].callAt < smallest) {
1510
+ originalTimer = this.timeouts[id];
1511
+ smallest = this.timeouts[id].callAt;
1512
+
1513
+ timer = {
1514
+ func: this.timeouts[id].func,
1515
+ callAt: this.timeouts[id].callAt,
1516
+ interval: this.timeouts[id].interval,
1517
+ id: this.timeouts[id].id
1518
+ };
1519
+ }
1520
+ }
1521
+ }
1522
+
1523
+ return timer || null;
1524
+ },
1525
+
1526
+ callTimer: function (timer) {
1527
+ try {
1528
+ if (typeof timer.func == "function") {
1529
+ timer.func.call(null);
1530
+ } else {
1531
+ eval(timer.func);
1532
+ }
1533
+ } catch (e) {}
1534
+
1535
+ if (!this.timeouts[timer.id]) {
1536
+ return;
1537
+ }
1538
+
1539
+ if (typeof timer.interval == "number") {
1540
+ this.timeouts[timer.id].callAt += timer.interval;
1541
+ } else {
1542
+ delete this.timeouts[timer.id];
1543
+ }
1544
+ },
1545
+
1546
+ reset: function reset() {
1547
+ this.timeouts = {};
1548
+ },
1549
+
1550
+ Date: (function () {
1551
+ var NativeDate = Date;
1552
+
1553
+ function ClockDate(year, month, date, hour, minute, second, ms) {
1554
+ // Defensive and verbose to avoid potential harm in passing
1555
+ // explicit undefined when user does not pass argument
1556
+ switch (arguments.length) {
1557
+ case 0:
1558
+ return new NativeDate(ClockDate.clock.now);
1559
+ case 1:
1560
+ return new NativeDate(year);
1561
+ case 2:
1562
+ return new NativeDate(year, month);
1563
+ case 3:
1564
+ return new NativeDate(year, month, date);
1565
+ case 4:
1566
+ return new NativeDate(year, month, date, hour);
1567
+ case 5:
1568
+ return new NativeDate(year, month, date, hour, minute);
1569
+ case 6:
1570
+ return new NativeDate(year, month, date, hour, minute, second);
1571
+ default:
1572
+ return new NativeDate(year, month, date, hour, minute, second, ms);
1573
+ }
1574
+ }
1575
+
1576
+ if (NativeDate.now) {
1577
+ ClockDate.now = function now() {
1578
+ return ClockDate.clock.now;
1579
+ };
1580
+ }
1581
+
1582
+ if (NativeDate.toSource) {
1583
+ ClockDate.toSource = function toSource() {
1584
+ return NativeDate.toSource();
1585
+ };
1586
+ }
1587
+
1588
+ ClockDate.toString = function toString() {
1589
+ return NativeDate.toString();
1590
+ };
1591
+
1592
+ ClockDate.prototype = NativeDate.prototype;
1593
+ ClockDate.parse = NativeDate.parse;
1594
+ ClockDate.UTC = NativeDate.UTC;
1595
+
1596
+ return ClockDate;
1597
+ }())
1598
+ };
1599
+ }());
1600
+
1601
+ sinon.timers = {
1602
+ setTimeout: setTimeout,
1603
+ clearTimeout: clearTimeout,
1604
+ setInterval: setInterval,
1605
+ clearInterval: clearInterval,
1606
+ Date: Date
1607
+ };
1608
+
1609
+ sinon.useFakeTimers = (function (global) {
1610
+ var methods = ["Date", "setTimeout", "setInterval", "clearTimeout", "clearInterval"];
1611
+
1612
+ function restore() {
1613
+ var method;
1614
+
1615
+ for (var i = 0, l = this.methods.length; i < l; i++) {
1616
+ method = this.methods[i];
1617
+ global[method] = this["_" + method];
1618
+ }
1619
+ }
1620
+
1621
+ function stubGlobal(method, clock) {
1622
+ clock["_" + method] = global[method];
1623
+
1624
+ global[method] = function () {
1625
+ return clock[method].apply(clock, arguments);
1626
+ };
1627
+
1628
+ for (var prop in clock[method]) {
1629
+ if (clock[method].hasOwnProperty(prop)) {
1630
+ global[method][prop] = clock[method][prop];
1631
+ }
1632
+ }
1633
+
1634
+ global[method].clock = clock;
1635
+ }
1636
+
1637
+ return function useFakeTimers(now) {
1638
+ var clock = sinon.clock.create(now);
1639
+ clock.restore = restore;
1640
+ clock.methods = Array.prototype.slice.call(arguments,
1641
+ typeof now == "number" ? 1 : 0);
1642
+
1643
+ if (clock.methods.length === 0) {
1644
+ clock.methods = methods;
1645
+ }
1646
+
1647
+ for (var i = 0, l = clock.methods.length; i < l; i++) {
1648
+ stubGlobal(clock.methods[i], clock);
1649
+ }
1650
+
1651
+ return clock;
1652
+ };
1653
+ }(typeof global != "undefined" ? global : this));
1654
+
1655
+ if (typeof module == "object" && typeof require == "function") {
1656
+ module.exports = sinon;
1657
+ }
1658
+
1659
+ /*jslint eqeqeq: false, onevar: false*/
1660
+ /*global sinon, module, require, ActiveXObject, XMLHttpRequest, DOMParser*/
1661
+ /**
1662
+ * Fake XMLHttpRequest object
1663
+ *
1664
+ * @author Christian Johansen (christian@cjohansen.no)
1665
+ * @license BSD
1666
+ *
1667
+ * Copyright (c) 2010-2011 Christian Johansen
1668
+ */
1669
+
1670
+ if (typeof sinon == "undefined") {
1671
+ this.sinon = {};
1672
+ }
1673
+
1674
+ sinon.xhr = { XMLHttpRequest: this.XMLHttpRequest };
1675
+
1676
+ sinon.FakeXMLHttpRequest = (function () {
1677
+ /*jsl:ignore*/
1678
+ var unsafeHeaders = {
1679
+ "Accept-Charset": true,
1680
+ "Accept-Encoding": true,
1681
+ "Connection": true,
1682
+ "Content-Length": true,
1683
+ "Cookie": true,
1684
+ "Cookie2": true,
1685
+ "Content-Transfer-Encoding": true,
1686
+ "Date": true,
1687
+ "Expect": true,
1688
+ "Host": true,
1689
+ "Keep-Alive": true,
1690
+ "Referer": true,
1691
+ "TE": true,
1692
+ "Trailer": true,
1693
+ "Transfer-Encoding": true,
1694
+ "Upgrade": true,
1695
+ "User-Agent": true,
1696
+ "Via": true
1697
+ };
1698
+ /*jsl:end*/
1699
+
1700
+ function FakeXMLHttpRequest() {
1701
+ this.readyState = FakeXMLHttpRequest.UNSENT;
1702
+ this.requestHeaders = {};
1703
+ this.requestBody = null;
1704
+ this.status = 0;
1705
+ this.statusText = "";
1706
+
1707
+ if (typeof FakeXMLHttpRequest.onCreate == "function") {
1708
+ FakeXMLHttpRequest.onCreate(this);
1709
+ }
1710
+ }
1711
+
1712
+ function verifyState(xhr) {
1713
+ if (xhr.readyState !== FakeXMLHttpRequest.OPENED) {
1714
+ throw new Error("INVALID_STATE_ERR");
1715
+ }
1716
+
1717
+ if (xhr.sendFlag) {
1718
+ throw new Error("INVALID_STATE_ERR");
1719
+ }
1720
+ }
1721
+
1722
+ sinon.extend(FakeXMLHttpRequest.prototype, {
1723
+ async: true,
1724
+
1725
+ open: function open(method, url, async, username, password) {
1726
+ this.method = method;
1727
+ this.url = url;
1728
+ this.async = typeof async == "boolean" ? async : true;
1729
+ this.username = username;
1730
+ this.password = password;
1731
+ this.responseText = null;
1732
+ this.responseXML = null;
1733
+ this.requestHeaders = {};
1734
+ this.sendFlag = false;
1735
+ this.readyStateChange(FakeXMLHttpRequest.OPENED);
1736
+ },
1737
+
1738
+ readyStateChange: function readyStateChange(state) {
1739
+ this.readyState = state;
1740
+
1741
+ if (typeof this.onreadystatechange == "function") {
1742
+ this.onreadystatechange();
1743
+ }
1744
+ },
1745
+
1746
+ setRequestHeader: function setRequestHeader(header, value) {
1747
+ verifyState(this);
1748
+
1749
+ if (unsafeHeaders[header] || /^(Sec-|Proxy-)/.test(header)) {
1750
+ throw new Error("Refused to set unsafe header \"" + header + "\"");
1751
+ }
1752
+
1753
+ if (this.requestHeaders[header]) {
1754
+ this.requestHeaders[header] += "," + value;
1755
+ } else {
1756
+ this.requestHeaders[header] = value;
1757
+ }
1758
+ },
1759
+
1760
+ // Helps testing
1761
+ setResponseHeaders: function setResponseHeaders(headers) {
1762
+ this.responseHeaders = {};
1763
+
1764
+ for (var header in headers) {
1765
+ if (headers.hasOwnProperty(header)) {
1766
+ this.responseHeaders[header] = headers[header];
1767
+ }
1768
+ }
1769
+
1770
+ if (this.async) {
1771
+ this.readyStateChange(FakeXMLHttpRequest.HEADERS_RECEIVED);
1772
+ }
1773
+ },
1774
+
1775
+ // Currently treats ALL data as a DOMString (i.e. no Document)
1776
+ send: function send(data) {
1777
+ verifyState(this);
1778
+
1779
+ if (!/^(get|head)$/i.test(this.method)) {
1780
+ if (this.requestHeaders["Content-Type"]) {
1781
+ var value = this.requestHeaders["Content-Type"].split(";");
1782
+ this.requestHeaders["Content-Type"] = value[0] + ";charset=utf-8";
1783
+ } else {
1784
+ this.requestHeaders["Content-Type"] = "text/plain;charset=utf-8";
1785
+ }
1786
+
1787
+ this.requestBody = data;
1788
+ }
1789
+
1790
+ this.errorFlag = false;
1791
+ this.sendFlag = this.async;
1792
+ this.readyStateChange(FakeXMLHttpRequest.OPENED);
1793
+
1794
+ if (typeof this.onSend == "function") {
1795
+ this.onSend(this);
1796
+ }
1797
+ },
1798
+
1799
+ abort: function abort() {
1800
+ this.aborted = true;
1801
+ this.responseText = null;
1802
+ this.errorFlag = true;
1803
+ this.requestHeaders = {};
1804
+
1805
+ if (this.readyState > sinon.FakeXMLHttpRequest.OPENED) {
1806
+ this.readyStateChange(sinon.FakeXMLHttpRequest.DONE);
1807
+ this.sendFlag = false;
1808
+ }
1809
+
1810
+ this.readyState = sinon.FakeXMLHttpRequest.UNSENT;
1811
+ },
1812
+
1813
+ getResponseHeader: function getResponseHeader(header) {
1814
+ if (this.readyState < FakeXMLHttpRequest.HEADERS_RECEIVED) {
1815
+ return null;
1816
+ }
1817
+
1818
+ if (/^Set-Cookie2?$/i.test(header)) {
1819
+ return null;
1820
+ }
1821
+
1822
+ header = header.toLowerCase();
1823
+
1824
+ for (var h in this.responseHeaders) {
1825
+ if (h.toLowerCase() == header) {
1826
+ return this.responseHeaders[h];
1827
+ }
1828
+ }
1829
+
1830
+ return null;
1831
+ },
1832
+
1833
+ getAllResponseHeaders: function getAllResponseHeaders() {
1834
+ if (this.readyState < FakeXMLHttpRequest.HEADERS_RECEIVED) {
1835
+ return "";
1836
+ }
1837
+
1838
+ var headers = "";
1839
+
1840
+ for (var header in this.responseHeaders) {
1841
+ if (this.responseHeaders.hasOwnProperty(header) &&
1842
+ !/^Set-Cookie2?$/i.test(header)) {
1843
+ headers += header + ": " + this.responseHeaders[header] + "\r\n";
1844
+ }
1845
+ }
1846
+
1847
+ return headers;
1848
+ },
1849
+
1850
+ setResponseBody: function setResponseBody(body) {
1851
+ if (this.readyState == FakeXMLHttpRequest.DONE) {
1852
+ throw new Error("Request done");
1853
+ }
1854
+
1855
+ if (this.async && this.readyState != FakeXMLHttpRequest.HEADERS_RECEIVED) {
1856
+ throw new Error("No headers received");
1857
+ }
1858
+
1859
+ var chunkSize = this.chunkSize || 10;
1860
+ var index = 0;
1861
+ this.responseText = "";
1862
+
1863
+ do {
1864
+ if (this.async) {
1865
+ this.readyStateChange(FakeXMLHttpRequest.LOADING);
1866
+ }
1867
+
1868
+ this.responseText += body.substring(index, index + chunkSize);
1869
+ index += chunkSize;
1870
+ } while (index < body.length);
1871
+
1872
+ var type = this.getResponseHeader("Content-Type");
1873
+
1874
+ if (this.responseText &&
1875
+ (!type || /(text\/xml)|(application\/xml)|(\+xml)/.test(type))) {
1876
+ try {
1877
+ this.responseXML = FakeXMLHttpRequest.parseXML(this.responseText);
1878
+ } catch (e) {}
1879
+ }
1880
+
1881
+ if (this.async) {
1882
+ this.readyStateChange(FakeXMLHttpRequest.DONE);
1883
+ } else {
1884
+ this.readyState = FakeXMLHttpRequest.DONE;
1885
+ }
1886
+ },
1887
+
1888
+ respond: function respond(status, headers, body) {
1889
+ this.setResponseHeaders(headers || {});
1890
+ this.status = typeof status == "number" ? status : 200;
1891
+ this.statusText = FakeXMLHttpRequest.statusCodes[this.status];
1892
+ this.setResponseBody(body || "");
1893
+ }
1894
+ });
1895
+
1896
+ sinon.extend(FakeXMLHttpRequest, {
1897
+ UNSENT: 0,
1898
+ OPENED: 1,
1899
+ HEADERS_RECEIVED: 2,
1900
+ LOADING: 3,
1901
+ DONE: 4
1902
+ });
1903
+
1904
+ // Borrowed from JSpec
1905
+ FakeXMLHttpRequest.parseXML = function parseXML(text) {
1906
+ var xmlDoc;
1907
+
1908
+ if (typeof DOMParser != "undefined") {
1909
+ var parser = new DOMParser();
1910
+ xmlDoc = parser.parseFromString(text, "text/xml");
1911
+ } else {
1912
+ xmlDoc = new ActiveXObject("Microsoft.XMLDOM");
1913
+ xmlDoc.async = "false";
1914
+ xmlDoc.loadXML(text);
1915
+ }
1916
+
1917
+ return xmlDoc;
1918
+ };
1919
+
1920
+ FakeXMLHttpRequest.statusCodes = {
1921
+ 100: "Continue",
1922
+ 101: "Switching Protocols",
1923
+ 200: "OK",
1924
+ 201: "Created",
1925
+ 202: "Accepted",
1926
+ 203: "Non-Authoritative Information",
1927
+ 204: "No Content",
1928
+ 205: "Reset Content",
1929
+ 206: "Partial Content",
1930
+ 300: "Multiple Choice",
1931
+ 301: "Moved Permanently",
1932
+ 302: "Found",
1933
+ 303: "See Other",
1934
+ 304: "Not Modified",
1935
+ 305: "Use Proxy",
1936
+ 307: "Temporary Redirect",
1937
+ 400: "Bad Request",
1938
+ 401: "Unauthorized",
1939
+ 402: "Payment Required",
1940
+ 403: "Forbidden",
1941
+ 404: "Not Found",
1942
+ 405: "Method Not Allowed",
1943
+ 406: "Not Acceptable",
1944
+ 407: "Proxy Authentication Required",
1945
+ 408: "Request Timeout",
1946
+ 409: "Conflict",
1947
+ 410: "Gone",
1948
+ 411: "Length Required",
1949
+ 412: "Precondition Failed",
1950
+ 413: "Request Entity Too Large",
1951
+ 414: "Request-URI Too Long",
1952
+ 415: "Unsupported Media Type",
1953
+ 416: "Requested Range Not Satisfiable",
1954
+ 417: "Expectation Failed",
1955
+ 422: "Unprocessable Entity",
1956
+ 500: "Internal Server Error",
1957
+ 501: "Not Implemented",
1958
+ 502: "Bad Gateway",
1959
+ 503: "Service Unavailable",
1960
+ 504: "Gateway Timeout",
1961
+ 505: "HTTP Version Not Supported"
1962
+ };
1963
+
1964
+ return FakeXMLHttpRequest;
1965
+ }());
1966
+
1967
+ (function (global) {
1968
+ var GlobalXMLHttpRequest = global.XMLHttpRequest;
1969
+ var GlobalActiveXObject = global.ActiveXObject;
1970
+ var supportsActiveX = typeof ActiveXObject != "undefined";
1971
+ var supportsXHR = typeof XMLHttpRequest != "undefined";
1972
+
1973
+ sinon.useFakeXMLHttpRequest = function () {
1974
+ sinon.FakeXMLHttpRequest.restore = function restore(keepOnCreate) {
1975
+ if (supportsXHR) {
1976
+ global.XMLHttpRequest = GlobalXMLHttpRequest;
1977
+ }
1978
+
1979
+ if (supportsActiveX) {
1980
+ global.ActiveXObject = GlobalActiveXObject;
1981
+ }
1982
+
1983
+ delete sinon.FakeXMLHttpRequest.restore;
1984
+
1985
+ if (keepOnCreate !== true) {
1986
+ delete sinon.FakeXMLHttpRequest.onCreate;
1987
+ }
1988
+ };
1989
+
1990
+ if (supportsXHR) {
1991
+ global.XMLHttpRequest = sinon.FakeXMLHttpRequest;
1992
+ }
1993
+
1994
+ if (supportsActiveX) {
1995
+ global.ActiveXObject = function ActiveXObject(objId) {
1996
+ if (objId == "Microsoft.XMLHTTP" || /^Msxml2\.XMLHTTP/.test(objId)) {
1997
+ return new sinon.FakeXMLHttpRequest();
1998
+ }
1999
+
2000
+ return new GlobalActiveXObject(objId);
2001
+ };
2002
+ }
2003
+
2004
+ return sinon.FakeXMLHttpRequest;
2005
+ };
2006
+ }(this));
2007
+
2008
+ if (typeof module == "object" && typeof require == "function") {
2009
+ module.exports = sinon;
2010
+ }
2011
+
2012
+ /**
2013
+ * @depend fake_xml_http_request.js
2014
+ */
2015
+ /*jslint eqeqeq: false, onevar: false, regexp: false, plusplus: false*/
2016
+ /*global module, require, window*/
2017
+ /**
2018
+ * The Sinon "server" mimics a web server that receives requests from
2019
+ * sinon.FakeXMLHttpRequest and provides an API to respond to those requests,
2020
+ * both synchronously and asynchronously. To respond synchronuously, canned
2021
+ * answers have to be provided upfront.
2022
+ *
2023
+ * @author Christian Johansen (christian@cjohansen.no)
2024
+ * @license BSD
2025
+ *
2026
+ * Copyright (c) 2010-2011 Christian Johansen
2027
+ */
2028
+
2029
+ if (typeof sinon == "undefined") {
2030
+ var sinon = {};
2031
+ }
2032
+
2033
+ sinon.fakeServer = (function () {
2034
+ function F() {}
2035
+
2036
+ function create(proto) {
2037
+ F.prototype = proto;
2038
+ return new F();
2039
+ }
2040
+
2041
+ function responseArray(handler) {
2042
+ var response = handler;
2043
+
2044
+ if (Object.prototype.toString.call(handler) != "[object Array]") {
2045
+ response = [200, {}, handler];
2046
+ }
2047
+
2048
+ if (typeof response[2] != "string") {
2049
+ throw new TypeError("Fake server response body should be string, but was " +
2050
+ typeof response[2]);
2051
+ }
2052
+
2053
+ return response;
2054
+ }
2055
+
2056
+ var wloc = window.location;
2057
+ var rCurrLoc = new RegExp("^" + wloc.protocol + "//" + wloc.host);
2058
+
2059
+ function matchOne(response, reqMethod, reqUrl) {
2060
+ var rmeth = response.method;
2061
+ var matchMethod = !rmeth || rmeth.toLowerCase() == reqMethod.toLowerCase();
2062
+ var url = response.url;
2063
+ var matchUrl = !url || url == reqUrl || (typeof url.test == "function" && url.test(reqUrl));
2064
+
2065
+ return matchMethod && matchUrl;
2066
+ }
2067
+
2068
+ function match(response, request) {
2069
+ var requestMethod = this.getHTTPMethod(request);
2070
+ var requestUrl = request.url;
2071
+
2072
+ if (!/^https?:\/\//.test(requestUrl) || rCurrLoc.test(requestUrl)) {
2073
+ requestUrl = requestUrl.replace(rCurrLoc, "");
2074
+ }
2075
+
2076
+ if (matchOne(response, this.getHTTPMethod(request), requestUrl)) {
2077
+ if (typeof response.response == "function") {
2078
+ var args = [request].concat(requestUrl.match(response.url).slice(1));
2079
+ return response.response.apply(response, args);
2080
+ }
2081
+
2082
+ return true;
2083
+ }
2084
+
2085
+ return false;
2086
+ }
2087
+
2088
+ return {
2089
+ create: function () {
2090
+ var server = create(this);
2091
+ this.xhr = sinon.useFakeXMLHttpRequest();
2092
+ server.requests = [];
2093
+
2094
+ this.xhr.onCreate = function (xhrObj) {
2095
+ server.addRequest(xhrObj);
2096
+ };
2097
+
2098
+ return server;
2099
+ },
2100
+
2101
+ addRequest: function addRequest(xhrObj) {
2102
+ var server = this;
2103
+ this.requests.push(xhrObj);
2104
+
2105
+ xhrObj.onSend = function () {
2106
+ server.handleRequest(this);
2107
+ };
2108
+
2109
+ if (this.autoRespond && !this.responding) {
2110
+ setTimeout(function () {
2111
+ server.responding = false;
2112
+ server.respond();
2113
+ }, this.autoRespondAfter || 10);
2114
+
2115
+ this.responding = true;
2116
+ }
2117
+ },
2118
+
2119
+ getHTTPMethod: function getHTTPMethod(request) {
2120
+ if (this.fakeHTTPMethods && /post/i.test(request.method)) {
2121
+ var matches = request.requestBody.match(/_method=([^\b;]+)/);
2122
+ return !!matches ? matches[1] : request.method;
2123
+ }
2124
+
2125
+ return request.method;
2126
+ },
2127
+
2128
+ handleRequest: function handleRequest(xhr) {
2129
+ if (xhr.async) {
2130
+ if (!this.queue) {
2131
+ this.queue = [];
2132
+ }
2133
+
2134
+ this.queue.push(xhr);
2135
+ } else {
2136
+ this.processRequest(xhr);
2137
+ }
2138
+ },
2139
+
2140
+ respondWith: function respondWith(method, url, body) {
2141
+ if (arguments.length == 1) {
2142
+ this.response = responseArray(method);
2143
+ } else {
2144
+ if (!this.responses) {
2145
+ this.responses = [];
2146
+ }
2147
+
2148
+ if (arguments.length == 2) {
2149
+ body = url;
2150
+ url = method;
2151
+ method = null;
2152
+ }
2153
+
2154
+ this.responses.push({
2155
+ method: method,
2156
+ url: url,
2157
+ response: typeof body == "function" ? body : responseArray(body)
2158
+ });
2159
+ }
2160
+ },
2161
+
2162
+ respond: function respond() {
2163
+ var queue = this.queue || [];
2164
+ var request;
2165
+
2166
+ while(request = queue.shift()) {
2167
+ this.processRequest(request);
2168
+ }
2169
+ },
2170
+
2171
+ processRequest: function processRequest(request) {
2172
+ try {
2173
+ if (request.aborted) {
2174
+ return;
2175
+ }
2176
+
2177
+ var response = this.response || [404, {}, ""];
2178
+
2179
+ if (this.responses) {
2180
+ for (var i = 0, l = this.responses.length; i < l; i++) {
2181
+ if (match.call(this, this.responses[i], request)) {
2182
+ response = this.responses[i].response;
2183
+ break;
2184
+ }
2185
+ }
2186
+ }
2187
+
2188
+ if (request.readyState != 4) {
2189
+ request.respond(response[0], response[1], response[2]);
2190
+ }
2191
+ } catch (e) {}
2192
+ },
2193
+
2194
+ restore: function restore() {
2195
+ return this.xhr.restore && this.xhr.restore.apply(this.xhr, arguments);
2196
+ }
2197
+ };
2198
+ }());
2199
+
2200
+ if (typeof module == "object" && typeof require == "function") {
2201
+ module.exports = sinon;
2202
+ }
2203
+
2204
+ /**
2205
+ * @depend fake_server.js
2206
+ * @depend fake_timers.js
2207
+ */
2208
+ /*jslint browser: true, eqeqeq: false, onevar: false*/
2209
+ /*global sinon*/
2210
+ /**
2211
+ * Add-on for sinon.fakeServer that automatically handles a fake timer along with
2212
+ * the FakeXMLHttpRequest. The direct inspiration for this add-on is jQuery
2213
+ * 1.3.x, which does not use xhr object's onreadystatehandler at all - instead,
2214
+ * it polls the object for completion with setInterval. Dispite the direct
2215
+ * motivation, there is nothing jQuery-specific in this file, so it can be used
2216
+ * in any environment where the ajax implementation depends on setInterval or
2217
+ * setTimeout.
2218
+ *
2219
+ * @author Christian Johansen (christian@cjohansen.no)
2220
+ * @license BSD
2221
+ *
2222
+ * Copyright (c) 2010-2011 Christian Johansen
2223
+ */
2224
+
2225
+ (function () {
2226
+ function Server() {}
2227
+ Server.prototype = sinon.fakeServer;
2228
+
2229
+ sinon.fakeServerWithClock = new Server();
2230
+
2231
+ sinon.fakeServerWithClock.addRequest = function addRequest(xhr) {
2232
+ if (xhr.async) {
2233
+ if (typeof setTimeout.clock == "object") {
2234
+ this.clock = setTimeout.clock;
2235
+ } else {
2236
+ this.clock = sinon.useFakeTimers();
2237
+ this.resetClock = true;
2238
+ }
2239
+
2240
+ if (!this.longestTimeout) {
2241
+ var clockSetTimeout = this.clock.setTimeout;
2242
+ var clockSetInterval = this.clock.setInterval;
2243
+ var server = this;
2244
+
2245
+ this.clock.setTimeout = function (fn, timeout) {
2246
+ server.longestTimeout = Math.max(timeout, server.longestTimeout || 0);
2247
+
2248
+ return clockSetTimeout.apply(this, arguments);
2249
+ };
2250
+
2251
+ this.clock.setInterval = function (fn, timeout) {
2252
+ server.longestTimeout = Math.max(timeout, server.longestTimeout || 0);
2253
+
2254
+ return clockSetInterval.apply(this, arguments);
2255
+ };
2256
+ }
2257
+ }
2258
+
2259
+ return sinon.fakeServer.addRequest.call(this, xhr);
2260
+ };
2261
+
2262
+ sinon.fakeServerWithClock.respond = function respond() {
2263
+ var returnVal = sinon.fakeServer.respond.apply(this, arguments);
2264
+
2265
+ if (this.clock) {
2266
+ this.clock.tick(this.longestTimeout || 0);
2267
+ this.longestTimeout = 0;
2268
+
2269
+ if (this.resetClock) {
2270
+ this.clock.restore();
2271
+ this.resetClock = false;
2272
+ }
2273
+ }
2274
+
2275
+ return returnVal;
2276
+ };
2277
+
2278
+ sinon.fakeServerWithClock.restore = function restore() {
2279
+ if (this.clock) {
2280
+ this.clock.restore();
2281
+ }
2282
+
2283
+ return sinon.fakeServer.restore.apply(this, arguments);
2284
+ };
2285
+ }());
2286
+
2287
+ /**
2288
+ * @depend ../sinon.js
2289
+ * @depend collection.js
2290
+ * @depend util/fake_timers.js
2291
+ * @depend util/fake_server_with_clock.js
2292
+ */
2293
+ /*jslint eqeqeq: false, onevar: false, plusplus: false*/
2294
+ /*global require, module*/
2295
+ /**
2296
+ * Manages fake collections as well as fake utilities such as Sinon's
2297
+ * timers and fake XHR implementation in one convenient object.
2298
+ *
2299
+ * @author Christian Johansen (christian@cjohansen.no)
2300
+ * @license BSD
2301
+ *
2302
+ * Copyright (c) 2010-2011 Christian Johansen
2303
+ */
2304
+
2305
+ if (typeof module == "object" && typeof require == "function") {
2306
+ var sinon = require("sinon");
2307
+ sinon.extend(sinon, require("./util/fake_timers"));
2308
+ }
2309
+
2310
+ (function () {
2311
+ function exposeValue(sandbox, config, key, value) {
2312
+ if (!value) {
2313
+ return;
2314
+ }
2315
+
2316
+ if (config.injectInto) {
2317
+ config.injectInto[key] = value;
2318
+ } else {
2319
+ sandbox.args.push(value);
2320
+ }
2321
+ }
2322
+
2323
+ function prepareSandboxFromConfig(config) {
2324
+ var sandbox = sinon.create(sinon.sandbox);
2325
+
2326
+ if (config.useFakeServer) {
2327
+ if (typeof config.useFakeServer == "object") {
2328
+ sandbox.serverPrototype = config.useFakeServer;
2329
+ }
2330
+
2331
+ sandbox.useFakeServer();
2332
+ }
2333
+
2334
+ if (config.useFakeTimers) {
2335
+ if (typeof config.useFakeTimers == "object") {
2336
+ sandbox.useFakeTimers.apply(sandbox, config.useFakeTimers);
2337
+ } else {
2338
+ sandbox.useFakeTimers();
2339
+ }
2340
+ }
2341
+
2342
+ return sandbox;
2343
+ }
2344
+
2345
+ sinon.sandbox = sinon.extend(sinon.create(sinon.collection), {
2346
+ useFakeTimers: function useFakeTimers() {
2347
+ this.clock = sinon.useFakeTimers.apply(sinon, arguments);
2348
+
2349
+ return this.add(this.clock);
2350
+ },
2351
+
2352
+ serverPrototype: sinon.fakeServer,
2353
+
2354
+ useFakeServer: function useFakeServer() {
2355
+ var proto = this.serverPrototype || sinon.fakeServer;
2356
+
2357
+ if (!proto || !proto.create) {
2358
+ return null;
2359
+ }
2360
+
2361
+ this.server = proto.create();
2362
+ return this.add(this.server);
2363
+ },
2364
+
2365
+ inject: function (obj) {
2366
+ sinon.collection.inject.call(this, obj);
2367
+
2368
+ if (this.clock) {
2369
+ obj.clock = this.clock;
2370
+ }
2371
+
2372
+ if (this.server) {
2373
+ obj.server = this.server;
2374
+ obj.requests = this.server.requests;
2375
+ }
2376
+
2377
+ return obj;
2378
+ },
2379
+
2380
+ create: function (config) {
2381
+ if (!config) {
2382
+ return sinon.create(sinon.sandbox);
2383
+ }
2384
+
2385
+ var sandbox = prepareSandboxFromConfig(config);
2386
+ sandbox.args = sandbox.args || [];
2387
+ var prop, value, exposed = sandbox.inject({});
2388
+
2389
+ if (config.properties) {
2390
+ for (var i = 0, l = config.properties.length; i < l; i++) {
2391
+ prop = config.properties[i];
2392
+ value = exposed[prop] || prop == "sandbox" && sandbox;
2393
+ exposeValue(sandbox, config, prop, value);
2394
+ }
2395
+ } else {
2396
+ exposeValue(sandbox, config, "sandbox", value);
2397
+ }
2398
+
2399
+ return sandbox;
2400
+ }
2401
+ });
2402
+
2403
+ sinon.sandbox.useFakeXMLHttpRequest = sinon.sandbox.useFakeServer;
2404
+
2405
+ if (typeof module != "undefined") {
2406
+ module.exports = sinon.sandbox;
2407
+ }
2408
+ }());
2409
+
2410
+ /**
2411
+ * @depend ../sinon.js
2412
+ * @depend stub.js
2413
+ * @depend mock.js
2414
+ * @depend sandbox.js
2415
+ */
2416
+ /*jslint eqeqeq: false, onevar: false, forin: true, plusplus: false*/
2417
+ /*global module, require, sinon*/
2418
+ /**
2419
+ * Test function, sandboxes fakes
2420
+ *
2421
+ * @author Christian Johansen (christian@cjohansen.no)
2422
+ * @license BSD
2423
+ *
2424
+ * Copyright (c) 2010-2011 Christian Johansen
2425
+ */
2426
+
2427
+ (function (sinon) {
2428
+ var commonJSModule = typeof module == "object" && typeof require == "function";
2429
+
2430
+ if (!sinon && commonJSModule) {
2431
+ sinon = require("sinon");
2432
+ }
2433
+
2434
+ if (!sinon) {
2435
+ return;
2436
+ }
2437
+
2438
+ function test(callback) {
2439
+ var type = typeof callback;
2440
+
2441
+ if (type != "function") {
2442
+ throw new TypeError("sinon.test needs to wrap a test function, got " + type);
2443
+ }
2444
+
2445
+ return function () {
2446
+ var config = sinon.getConfig(sinon.config);
2447
+ config.injectInto = config.injectIntoThis && this || config.injectInto;
2448
+ var sandbox = sinon.sandbox.create(config);
2449
+ var exception, result;
2450
+ var args = Array.prototype.slice.call(arguments).concat(sandbox.args);
2451
+
2452
+ try {
2453
+ result = callback.apply(this, args);
2454
+ } catch (e) {
2455
+ exception = e;
2456
+ }
2457
+
2458
+ sandbox.verifyAndRestore();
2459
+
2460
+ if (exception) {
2461
+ throw exception;
2462
+ }
2463
+
2464
+ return result;
2465
+ };
2466
+ }
2467
+
2468
+ test.config = {
2469
+ injectIntoThis: true,
2470
+ injectInto: null,
2471
+ properties: ["spy", "stub", "mock", "clock", "server", "requests"],
2472
+ useFakeTimers: true,
2473
+ useFakeServer: true
2474
+ };
2475
+
2476
+ if (commonJSModule) {
2477
+ module.exports = test;
2478
+ } else {
2479
+ sinon.test = test;
2480
+ }
2481
+ }(typeof sinon == "object" && sinon || null));
2482
+
2483
+ /**
2484
+ * @depend ../sinon.js
2485
+ * @depend test.js
2486
+ */
2487
+ /*jslint eqeqeq: false, onevar: false, eqeqeq: false*/
2488
+ /*global module, require, sinon*/
2489
+ /**
2490
+ * Test case, sandboxes all test functions
2491
+ *
2492
+ * @author Christian Johansen (christian@cjohansen.no)
2493
+ * @license BSD
2494
+ *
2495
+ * Copyright (c) 2010-2011 Christian Johansen
2496
+ */
2497
+
2498
+ (function (sinon) {
2499
+ var commonJSModule = typeof module == "object" && typeof require == "function";
2500
+
2501
+ if (!sinon && commonJSModule) {
2502
+ sinon = require("sinon");
2503
+ }
2504
+
2505
+ if (!sinon || !Object.prototype.hasOwnProperty) {
2506
+ return;
2507
+ }
2508
+
2509
+ function createTest(property, setUp, tearDown) {
2510
+ return function () {
2511
+ if (setUp) {
2512
+ setUp.apply(this, arguments);
2513
+ }
2514
+
2515
+ var exception, result;
2516
+
2517
+ try {
2518
+ result = property.apply(this, arguments);
2519
+ } catch (e) {
2520
+ exception = e;
2521
+ }
2522
+
2523
+ if (tearDown) {
2524
+ tearDown.apply(this, arguments);
2525
+ }
2526
+
2527
+ if (exception) {
2528
+ throw exception;
2529
+ }
2530
+
2531
+ return result;
2532
+ };
2533
+ }
2534
+
2535
+ function testCase(tests, prefix) {
2536
+ /*jsl:ignore*/
2537
+ if (!tests || typeof tests != "object") {
2538
+ throw new TypeError("sinon.testCase needs an object with test functions");
2539
+ }
2540
+ /*jsl:end*/
2541
+
2542
+ prefix = prefix || "test";
2543
+ var rPrefix = new RegExp("^" + prefix);
2544
+ var methods = {}, testName, property, method;
2545
+ var setUp = tests.setUp;
2546
+ var tearDown = tests.tearDown;
2547
+
2548
+ for (testName in tests) {
2549
+ if (tests.hasOwnProperty(testName)) {
2550
+ property = tests[testName];
2551
+
2552
+ if (/^(setUp|tearDown)$/.test(testName)) {
2553
+ continue;
2554
+ }
2555
+
2556
+ if (typeof property == "function" && rPrefix.test(testName)) {
2557
+ method = property;
2558
+
2559
+ if (setUp || tearDown) {
2560
+ method = createTest(property, setUp, tearDown);
2561
+ }
2562
+
2563
+ methods[testName] = sinon.test(method);
2564
+ } else {
2565
+ methods[testName] = tests[testName];
2566
+ }
2567
+ }
2568
+ }
2569
+
2570
+ return methods;
2571
+ }
2572
+
2573
+ if (commonJSModule) {
2574
+ module.exports = testCase;
2575
+ } else {
2576
+ sinon.testCase = testCase;
2577
+ }
2578
+ }(typeof sinon == "object" && sinon || null));
2579
+
2580
+ /**
2581
+ * @depend ../sinon.js
2582
+ * @depend stub.js
2583
+ */
2584
+ /*jslint eqeqeq: false, onevar: false, nomen: false, plusplus: false*/
2585
+ /*global module, require, sinon*/
2586
+ /**
2587
+ * Assertions matching the test spy retrieval interface.
2588
+ *
2589
+ * @author Christian Johansen (christian@cjohansen.no)
2590
+ * @license BSD
2591
+ *
2592
+ * Copyright (c) 2010-2011 Christian Johansen
2593
+ */
2594
+
2595
+ (function (sinon) {
2596
+ var commonJSModule = typeof module == "object" && typeof require == "function";
2597
+ var slice = Array.prototype.slice;
2598
+ var assert;
2599
+
2600
+ if (!sinon && commonJSModule) {
2601
+ sinon = require("sinon");
2602
+ }
2603
+
2604
+ if (!sinon) {
2605
+ return;
2606
+ }
2607
+
2608
+ function times(count) {
2609
+ return count == 1 && "once" ||
2610
+ count == 2 && "twice" ||
2611
+ count == 3 && "thrice" ||
2612
+ (count || 0) + " times";
2613
+ }
2614
+
2615
+ function verifyIsStub(method) {
2616
+ if (!method) {
2617
+ assert.fail("fake is not a spy");
2618
+ }
2619
+
2620
+ if (typeof method != "function") {
2621
+ assert.fail(method + " is not a function");
2622
+ }
2623
+
2624
+ if (typeof method.getCall != "function") {
2625
+ assert.fail(method + " is not stubbed");
2626
+ }
2627
+ }
2628
+
2629
+ function failAssertion(object, msg) {
2630
+ var failMethod = object.fail || assert.fail;
2631
+ failMethod.call(object, msg);
2632
+ }
2633
+
2634
+ function mirrorAssertion(method, message) {
2635
+ assert[method] = function (fake) {
2636
+ verifyIsStub(fake);
2637
+
2638
+ var failed = typeof fake[method] == "function" ?
2639
+ !fake[method].apply(fake, slice.call(arguments, 1)) : !fake[method];
2640
+
2641
+ if (failed) {
2642
+ var msg = message.replace("%c", times(fake.callCount));
2643
+ msg = msg.replace("%n", fake + "");
2644
+
2645
+ msg = msg.replace("%C", function (m) {
2646
+ return formatSpyCalls(fake);
2647
+ });
2648
+
2649
+ msg = msg.replace("%t", function (m) {
2650
+ return formatThisValues(fake);
2651
+ });
2652
+
2653
+ msg = msg.replace("%*", [].slice.call(arguments, 1).join(", "));
2654
+
2655
+ for (var i = 0, l = arguments.length; i < l; i++) {
2656
+ msg = msg.replace("%" + i, arguments[i]);
2657
+ }
2658
+
2659
+ failAssertion(this, msg);
2660
+ } else {
2661
+ assert.pass(method);
2662
+ }
2663
+ };
2664
+ }
2665
+
2666
+ function formatSpyCalls(spy) {
2667
+ var calls = [];
2668
+
2669
+ for (var i = 0, l = spy.callCount; i < l; ++i) {
2670
+ calls.push(" " + spy.getCall(i).toString());
2671
+ }
2672
+
2673
+ return calls.length > 0 ? "\n" + calls.join("\n") : "";
2674
+ }
2675
+
2676
+ function formatThisValues(spy) {
2677
+ var objects = [];
2678
+
2679
+ for (var i = 0, l = spy.callCount; i < l; ++i) {
2680
+ objects.push(sinon.format(spy.thisValues[i]));
2681
+ }
2682
+
2683
+ return objects.join(", ");
2684
+ }
2685
+
2686
+ assert = {
2687
+ failException: "AssertError",
2688
+
2689
+ fail: function fail(message) {
2690
+ var error = new Error(message);
2691
+ error.name = this.failException || assert.failException;
2692
+
2693
+ throw error;
2694
+ },
2695
+
2696
+ pass: function pass(assertion) {},
2697
+
2698
+ called: function assertCalled(method) {
2699
+ verifyIsStub(method);
2700
+
2701
+ if (!method.called) {
2702
+ failAssertion(this, "expected " + method +
2703
+ " to have been called at least once but was never called");
2704
+ } else {
2705
+ assert.pass("called");
2706
+ }
2707
+ },
2708
+
2709
+ notCalled: function assertNotCalled(method) {
2710
+ verifyIsStub(method);
2711
+
2712
+ if (method.called) {
2713
+ failAssertion(
2714
+ this, "expected " + method + " to not have been called but was " +
2715
+ "called " + times(method.callCount) + formatSpyCalls(method));
2716
+ } else {
2717
+ assert.pass("notCalled");
2718
+ }
2719
+ },
2720
+
2721
+ callOrder: function assertCallOrder() {
2722
+ verifyIsStub(arguments[0]);
2723
+ var expected = [];
2724
+ var actual = [];
2725
+ var failed = false;
2726
+ expected.push(arguments[0]);
2727
+
2728
+ for (var i = 1, l = arguments.length; i < l; i++) {
2729
+ verifyIsStub(arguments[i]);
2730
+ expected.push(arguments[i]);
2731
+
2732
+ if (!arguments[i - 1].calledBefore(arguments[i])) {
2733
+ failed = true;
2734
+ }
2735
+ }
2736
+
2737
+ if (failed) {
2738
+ actual = [].concat(expected).sort(function (a, b) {
2739
+ var aId = a.getCall(0).callId;
2740
+ var bId = b.getCall(0).callId;
2741
+
2742
+ // uuid, won't ever be equal
2743
+ return aId < bId ? -1 : 1;
2744
+ });
2745
+
2746
+ var expectedStr, actualStr;
2747
+
2748
+ try {
2749
+ expectedStr = expected.join(", ");
2750
+ actualStr = actual.join(", ");
2751
+ } catch (e) {}
2752
+
2753
+ failAssertion(this, "expected " + (expectedStr || "") + " to be " +
2754
+ "called in order but were called as " + actualStr);
2755
+ } else {
2756
+ assert.pass("callOrder");
2757
+ }
2758
+ },
2759
+
2760
+ callCount: function assertCallCount(method, count) {
2761
+ verifyIsStub(method);
2762
+
2763
+ if (method.callCount != count) {
2764
+ failAssertion(this, "expected " + method + " to be called " +
2765
+ times(count) + " but was called " +
2766
+ times(method.callCount) + formatSpyCalls(method));
2767
+ } else {
2768
+ assert.pass("callCount");
2769
+ }
2770
+ },
2771
+
2772
+ expose: function expose(target, options) {
2773
+ if (!target) {
2774
+ throw new TypeError("target is null or undefined");
2775
+ }
2776
+
2777
+ options = options || {};
2778
+ var prefix = typeof options.prefix == "undefined" && "assert" || options.prefix;
2779
+
2780
+ var name = function (prop) {
2781
+ if (!prefix) {
2782
+ return prop;
2783
+ }
2784
+
2785
+ return prefix + prop.substring(0, 1).toUpperCase() + prop.substring(1);
2786
+ };
2787
+
2788
+ for (var assertion in this) {
2789
+ if (!/^(fail|expose)/.test(assertion)) {
2790
+ target[name(assertion)] = this[assertion];
2791
+ }
2792
+ }
2793
+
2794
+ if (typeof options.includeFail == "undefined" || !!options.includeFail) {
2795
+ target.fail = this.fail;
2796
+ target.failException = this.failException;
2797
+ }
2798
+
2799
+ return target;
2800
+ }
2801
+ };
2802
+
2803
+ mirrorAssertion("calledOnce", "expected %n to be called once but was called %c%C");
2804
+ mirrorAssertion("calledTwice", "expected %n to be called twice but was called %c%C");
2805
+ mirrorAssertion("calledThrice", "expected %n to be called thrice but was called %c%C");
2806
+ mirrorAssertion("calledOn", "expected %n to be called with %1 as this but was called with %t");
2807
+ mirrorAssertion("alwaysCalledOn", "expected %n to always be called with %1 as this but was called with %t");
2808
+ mirrorAssertion("calledWith", "expected %n to be called with arguments %*%C");
2809
+ mirrorAssertion("alwaysCalledWith", "expected %n to always be called with arguments %*%C");
2810
+ mirrorAssertion("calledWithExactly", "expected %n to be called with exact arguments %*%C");
2811
+ mirrorAssertion("alwaysCalledWithExactly", "expected %n to always be called with exact arguments %*%C");
2812
+ mirrorAssertion("threw", "%n did not throw exception%C");
2813
+ mirrorAssertion("alwaysThrew", "%n did not always throw exception%C");
2814
+
2815
+ if (commonJSModule) {
2816
+ module.exports = assert;
2817
+ } else {
2818
+ sinon.assert = assert;
2819
+ }
2820
+ }(typeof sinon == "object" && sinon || null));
2821
+