couch_potato-rspec 3.4.0 → 4.0.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
- SHA1:
3
- metadata.gz: ac4b847aec3ea03fe48ea86a0af0db32bc981acf
4
- data.tar.gz: a9022b27f106bfee7accce3260cfdcf0a847967e
2
+ SHA256:
3
+ metadata.gz: 4e3a5c6e20b27a9c81694aef992aaa523989af4d1cb779e9bf18720342760876
4
+ data.tar.gz: bda628e1399a5f4d6f5881d394cb94c7ccb66af3cab3867a2147764e6dcd7883
5
5
  SHA512:
6
- metadata.gz: f4036cf4ab129a4d260859f885d16d6b0edc871a039ab758c43f16a71df711120d72273c77f5f4a8609fff4d0f9cd773a78f219fd5b2cb5dc119134256cd95c3
7
- data.tar.gz: 9dbc3ee9607610d208d9de4382944a1d3d82c2a46da6c17cfca6a44fb02ba770332447dc8bc8c73cb7fe5ac3c368e122f28e8f84ae2e893934c0274bf500cf06
6
+ metadata.gz: a80a9a6123e99730bdb8f3139c6e3f278e1403a7bfaae5152ddc5e55eedf414e31c39d8fcc0aca8b41c4cf25419894c339b65534f60c59a5605cc4adecc5ba58
7
+ data.tar.gz: 0daebcab0f2134a26a87328dcd7d98cde70f0b15f651e319b445fea995c5214faf1aeed9db49f3d239870ffd11f82ba70317a43e700bb1e701415a7683e94ff3
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: couch_potato-rspec
3
3
  version: !ruby/object:Gem::Version
4
- version: 3.4.0
4
+ version: 4.0.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Alexander Lang
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2016-03-29 00:00:00.000000000 Z
11
+ date: 2021-12-19 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: rspec
@@ -38,20 +38,26 @@ dependencies:
38
38
  - - ">="
39
39
  - !ruby/object:Gem::Version
40
40
  version: '0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: execjs
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: 2.7.0
48
+ type: :runtime
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: 2.7.0
41
55
  description: RSpec matchers for Couch Potato
42
56
  email: alex@upstre.am
43
57
  executables: []
44
58
  extensions: []
45
59
  extra_rdoc_files: []
46
60
  files:
47
- - lib/couch_potato/rspec.rb
48
- - lib/couch_potato/rspec/matchers.rb
49
- - lib/couch_potato/rspec/matchers/json2.js
50
- - lib/couch_potato/rspec/matchers/list_as_matcher.rb
51
- - lib/couch_potato/rspec/matchers/map_reduce_to_matcher.rb
52
- - lib/couch_potato/rspec/matchers/map_to_matcher.rb
53
- - lib/couch_potato/rspec/matchers/reduce_to_matcher.rb
54
- - lib/couch_potato/rspec/stub_db.rb
55
61
  - spec/unit/rspec_matchers_spec.rb
56
62
  homepage: http://github.com/langalex/couch_potato
57
63
  licenses: []
@@ -71,11 +77,9 @@ required_rubygems_version: !ruby/object:Gem::Requirement
71
77
  - !ruby/object:Gem::Version
72
78
  version: '0'
73
79
  requirements: []
74
- rubyforge_project:
75
- rubygems_version: 2.4.7
80
+ rubygems_version: 3.1.6
76
81
  signing_key:
77
82
  specification_version: 4
78
83
  summary: RSpec matchers for Couch Potato
79
84
  test_files:
80
85
  - spec/unit/rspec_matchers_spec.rb
81
- has_rdoc:
@@ -1,482 +0,0 @@
1
- /*
2
- http://www.JSON.org/json2.js
3
- 2010-03-20
4
-
5
- Public Domain.
6
-
7
- NO WARRANTY EXPRESSED OR IMPLIED. USE AT YOUR OWN RISK.
8
-
9
- See http://www.JSON.org/js.html
10
-
11
-
12
- This code should be minified before deployment.
13
- See http://javascript.crockford.com/jsmin.html
14
-
15
- USE YOUR OWN COPY. IT IS EXTREMELY UNWISE TO LOAD CODE FROM SERVERS YOU DO
16
- NOT CONTROL.
17
-
18
-
19
- This file creates a global JSON object containing two methods: stringify
20
- and parse.
21
-
22
- JSON.stringify(value, replacer, space)
23
- value any JavaScript value, usually an object or array.
24
-
25
- replacer an optional parameter that determines how object
26
- values are stringified for objects. It can be a
27
- function or an array of strings.
28
-
29
- space an optional parameter that specifies the indentation
30
- of nested structures. If it is omitted, the text will
31
- be packed without extra whitespace. If it is a number,
32
- it will specify the number of spaces to indent at each
33
- level. If it is a string (such as '\t' or ' '),
34
- it contains the characters used to indent at each level.
35
-
36
- This method produces a JSON text from a JavaScript value.
37
-
38
- When an object value is found, if the object contains a toJSON
39
- method, its toJSON method will be called and the result will be
40
- stringified. A toJSON method does not serialize: it returns the
41
- value represented by the name/value pair that should be serialized,
42
- or undefined if nothing should be serialized. The toJSON method
43
- will be passed the key associated with the value, and this will be
44
- bound to the value
45
-
46
- For example, this would serialize Dates as ISO strings.
47
-
48
- Date.prototype.toJSON = function (key) {
49
- function f(n) {
50
- // Format integers to have at least two digits.
51
- return n < 10 ? '0' + n : n;
52
- }
53
-
54
- return this.getUTCFullYear() + '-' +
55
- f(this.getUTCMonth() + 1) + '-' +
56
- f(this.getUTCDate()) + 'T' +
57
- f(this.getUTCHours()) + ':' +
58
- f(this.getUTCMinutes()) + ':' +
59
- f(this.getUTCSeconds()) + 'Z';
60
- };
61
-
62
- You can provide an optional replacer method. It will be passed the
63
- key and value of each member, with this bound to the containing
64
- object. The value that is returned from your method will be
65
- serialized. If your method returns undefined, then the member will
66
- be excluded from the serialization.
67
-
68
- If the replacer parameter is an array of strings, then it will be
69
- used to select the members to be serialized. It filters the results
70
- such that only members with keys listed in the replacer array are
71
- stringified.
72
-
73
- Values that do not have JSON representations, such as undefined or
74
- functions, will not be serialized. Such values in objects will be
75
- dropped; in arrays they will be replaced with null. You can use
76
- a replacer function to replace those with JSON values.
77
- JSON.stringify(undefined) returns undefined.
78
-
79
- The optional space parameter produces a stringification of the
80
- value that is filled with line breaks and indentation to make it
81
- easier to read.
82
-
83
- If the space parameter is a non-empty string, then that string will
84
- be used for indentation. If the space parameter is a number, then
85
- the indentation will be that many spaces.
86
-
87
- Example:
88
-
89
- text = JSON.stringify(['e', {pluribus: 'unum'}]);
90
- // text is '["e",{"pluribus":"unum"}]'
91
-
92
-
93
- text = JSON.stringify(['e', {pluribus: 'unum'}], null, '\t');
94
- // text is '[\n\t"e",\n\t{\n\t\t"pluribus": "unum"\n\t}\n]'
95
-
96
- text = JSON.stringify([new Date()], function (key, value) {
97
- return this[key] instanceof Date ?
98
- 'Date(' + this[key] + ')' : value;
99
- });
100
- // text is '["Date(---current time---)"]'
101
-
102
-
103
- JSON.parse(text, reviver)
104
- This method parses a JSON text to produce an object or array.
105
- It can throw a SyntaxError exception.
106
-
107
- The optional reviver parameter is a function that can filter and
108
- transform the results. It receives each of the keys and values,
109
- and its return value is used instead of the original value.
110
- If it returns what it received, then the structure is not modified.
111
- If it returns undefined then the member is deleted.
112
-
113
- Example:
114
-
115
- // Parse the text. Values that look like ISO date strings will
116
- // be converted to Date objects.
117
-
118
- myData = JSON.parse(text, function (key, value) {
119
- var a;
120
- if (typeof value === 'string') {
121
- a =
122
- /^(\d{4})-(\d{2})-(\d{2})T(\d{2}):(\d{2}):(\d{2}(?:\.\d*)?)Z$/.exec(value);
123
- if (a) {
124
- return new Date(Date.UTC(+a[1], +a[2] - 1, +a[3], +a[4],
125
- +a[5], +a[6]));
126
- }
127
- }
128
- return value;
129
- });
130
-
131
- myData = JSON.parse('["Date(09/09/2001)"]', function (key, value) {
132
- var d;
133
- if (typeof value === 'string' &&
134
- value.slice(0, 5) === 'Date(' &&
135
- value.slice(-1) === ')') {
136
- d = new Date(value.slice(5, -1));
137
- if (d) {
138
- return d;
139
- }
140
- }
141
- return value;
142
- });
143
-
144
-
145
- This is a reference implementation. You are free to copy, modify, or
146
- redistribute.
147
- */
148
-
149
- /*jslint evil: true, strict: false */
150
-
151
- /*members "", "\b", "\t", "\n", "\f", "\r", "\"", JSON, "\\", apply,
152
- call, charCodeAt, getUTCDate, getUTCFullYear, getUTCHours,
153
- getUTCMinutes, getUTCMonth, getUTCSeconds, hasOwnProperty, join,
154
- lastIndex, length, parse, prototype, push, replace, slice, stringify,
155
- test, toJSON, toString, valueOf
156
- */
157
-
158
-
159
- // Create a JSON object only if one does not already exist. We create the
160
- // methods in a closure to avoid creating global variables.
161
-
162
- if (!this.JSON) {
163
- this.JSON = {};
164
- }
165
-
166
- (function () {
167
-
168
- function f(n) {
169
- // Format integers to have at least two digits.
170
- return n < 10 ? '0' + n : n;
171
- }
172
-
173
- if (typeof Date.prototype.toJSON !== 'function') {
174
-
175
- Date.prototype.toJSON = function (key) {
176
-
177
- return isFinite(this.valueOf()) ?
178
- this.getUTCFullYear() + '-' +
179
- f(this.getUTCMonth() + 1) + '-' +
180
- f(this.getUTCDate()) + 'T' +
181
- f(this.getUTCHours()) + ':' +
182
- f(this.getUTCMinutes()) + ':' +
183
- f(this.getUTCSeconds()) + 'Z' : null;
184
- };
185
-
186
- String.prototype.toJSON =
187
- Number.prototype.toJSON =
188
- Boolean.prototype.toJSON = function (key) {
189
- return this.valueOf();
190
- };
191
- }
192
-
193
- var cx = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g,
194
- escapable = /[\\\"\x00-\x1f\x7f-\x9f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g,
195
- gap,
196
- indent,
197
- meta = { // table of character substitutions
198
- '\b': '\\b',
199
- '\t': '\\t',
200
- '\n': '\\n',
201
- '\f': '\\f',
202
- '\r': '\\r',
203
- '"' : '\\"',
204
- '\\': '\\\\'
205
- },
206
- rep;
207
-
208
-
209
- function quote(string) {
210
-
211
- // If the string contains no control characters, no quote characters, and no
212
- // backslash characters, then we can safely slap some quotes around it.
213
- // Otherwise we must also replace the offending characters with safe escape
214
- // sequences.
215
-
216
- escapable.lastIndex = 0;
217
- return escapable.test(string) ?
218
- '"' + string.replace(escapable, function (a) {
219
- var c = meta[a];
220
- return typeof c === 'string' ? c :
221
- '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4);
222
- }) + '"' :
223
- '"' + string + '"';
224
- }
225
-
226
-
227
- function str(key, holder) {
228
-
229
- // Produce a string from holder[key].
230
-
231
- var i, // The loop counter.
232
- k, // The member key.
233
- v, // The member value.
234
- length,
235
- mind = gap,
236
- partial,
237
- value = holder[key];
238
-
239
- // If the value has a toJSON method, call it to obtain a replacement value.
240
-
241
- if (value && typeof value === 'object' &&
242
- typeof value.toJSON === 'function') {
243
- value = value.toJSON(key);
244
- }
245
-
246
- // If we were called with a replacer function, then call the replacer to
247
- // obtain a replacement value.
248
-
249
- if (typeof rep === 'function') {
250
- value = rep.call(holder, key, value);
251
- }
252
-
253
- // What happens next depends on the value's type.
254
-
255
- switch (typeof value) {
256
- case 'string':
257
- return quote(value);
258
-
259
- case 'number':
260
-
261
- // JSON numbers must be finite. Encode non-finite numbers as null.
262
-
263
- return isFinite(value) ? String(value) : 'null';
264
-
265
- case 'boolean':
266
- case 'null':
267
-
268
- // If the value is a boolean or null, convert it to a string. Note:
269
- // typeof null does not produce 'null'. The case is included here in
270
- // the remote chance that this gets fixed someday.
271
-
272
- return String(value);
273
-
274
- // If the type is 'object', we might be dealing with an object or an array or
275
- // null.
276
-
277
- case 'object':
278
-
279
- // Due to a specification blunder in ECMAScript, typeof null is 'object',
280
- // so watch out for that case.
281
-
282
- if (!value) {
283
- return 'null';
284
- }
285
-
286
- // Make an array to hold the partial results of stringifying this object value.
287
-
288
- gap += indent;
289
- partial = [];
290
-
291
- // Is the value an array?
292
-
293
- if (Object.prototype.toString.apply(value) === '[object Array]') {
294
-
295
- // The value is an array. Stringify every element. Use null as a placeholder
296
- // for non-JSON values.
297
-
298
- length = value.length;
299
- for (i = 0; i < length; i += 1) {
300
- partial[i] = str(i, value) || 'null';
301
- }
302
-
303
- // Join all of the elements together, separated with commas, and wrap them in
304
- // brackets.
305
-
306
- v = partial.length === 0 ? '[]' :
307
- gap ? '[\n' + gap +
308
- partial.join(',\n' + gap) + '\n' +
309
- mind + ']' :
310
- '[' + partial.join(',') + ']';
311
- gap = mind;
312
- return v;
313
- }
314
-
315
- // If the replacer is an array, use it to select the members to be stringified.
316
-
317
- if (rep && typeof rep === 'object') {
318
- length = rep.length;
319
- for (i = 0; i < length; i += 1) {
320
- k = rep[i];
321
- if (typeof k === 'string') {
322
- v = str(k, value);
323
- if (v) {
324
- partial.push(quote(k) + (gap ? ': ' : ':') + v);
325
- }
326
- }
327
- }
328
- } else {
329
-
330
- // Otherwise, iterate through all of the keys in the object.
331
-
332
- for (k in value) {
333
- if (Object.hasOwnProperty.call(value, k)) {
334
- v = str(k, value);
335
- if (v) {
336
- partial.push(quote(k) + (gap ? ': ' : ':') + v);
337
- }
338
- }
339
- }
340
- }
341
-
342
- // Join all of the member texts together, separated with commas,
343
- // and wrap them in braces.
344
-
345
- v = partial.length === 0 ? '{}' :
346
- gap ? '{\n' + gap + partial.join(',\n' + gap) + '\n' +
347
- mind + '}' : '{' + partial.join(',') + '}';
348
- gap = mind;
349
- return v;
350
- }
351
- }
352
-
353
- // If the JSON object does not yet have a stringify method, give it one.
354
-
355
- if (typeof JSON.stringify !== 'function') {
356
- JSON.stringify = function (value, replacer, space) {
357
-
358
- // The stringify method takes a value and an optional replacer, and an optional
359
- // space parameter, and returns a JSON text. The replacer can be a function
360
- // that can replace values, or an array of strings that will select the keys.
361
- // A default replacer method can be provided. Use of the space parameter can
362
- // produce text that is more easily readable.
363
-
364
- var i;
365
- gap = '';
366
- indent = '';
367
-
368
- // If the space parameter is a number, make an indent string containing that
369
- // many spaces.
370
-
371
- if (typeof space === 'number') {
372
- for (i = 0; i < space; i += 1) {
373
- indent += ' ';
374
- }
375
-
376
- // If the space parameter is a string, it will be used as the indent string.
377
-
378
- } else if (typeof space === 'string') {
379
- indent = space;
380
- }
381
-
382
- // If there is a replacer, it must be a function or an array.
383
- // Otherwise, throw an error.
384
-
385
- rep = replacer;
386
- if (replacer && typeof replacer !== 'function' &&
387
- (typeof replacer !== 'object' ||
388
- typeof replacer.length !== 'number')) {
389
- throw new Error('JSON.stringify');
390
- }
391
-
392
- // Make a fake root object containing our value under the key of ''.
393
- // Return the result of stringifying the value.
394
-
395
- return str('', {'': value});
396
- };
397
- }
398
-
399
-
400
- // If the JSON object does not yet have a parse method, give it one.
401
-
402
- if (typeof JSON.parse !== 'function') {
403
- JSON.parse = function (text, reviver) {
404
-
405
- // The parse method takes a text and an optional reviver function, and returns
406
- // a JavaScript value if the text is a valid JSON text.
407
-
408
- var j;
409
-
410
- function walk(holder, key) {
411
-
412
- // The walk method is used to recursively walk the resulting structure so
413
- // that modifications can be made.
414
-
415
- var k, v, value = holder[key];
416
- if (value && typeof value === 'object') {
417
- for (k in value) {
418
- if (Object.hasOwnProperty.call(value, k)) {
419
- v = walk(value, k);
420
- if (v !== undefined) {
421
- value[k] = v;
422
- } else {
423
- delete value[k];
424
- }
425
- }
426
- }
427
- }
428
- return reviver.call(holder, key, value);
429
- }
430
-
431
-
432
- // Parsing happens in four stages. In the first stage, we replace certain
433
- // Unicode characters with escape sequences. JavaScript handles many characters
434
- // incorrectly, either silently deleting them, or treating them as line endings.
435
-
436
- text = String(text);
437
- cx.lastIndex = 0;
438
- if (cx.test(text)) {
439
- text = text.replace(cx, function (a) {
440
- return '\\u' +
441
- ('0000' + a.charCodeAt(0).toString(16)).slice(-4);
442
- });
443
- }
444
-
445
- // In the second stage, we run the text against regular expressions that look
446
- // for non-JSON patterns. We are especially concerned with '()' and 'new'
447
- // because they can cause invocation, and '=' because it can cause mutation.
448
- // But just to be safe, we want to reject all unexpected forms.
449
-
450
- // We split the second stage into 4 regexp operations in order to work around
451
- // crippling inefficiencies in IE's and Safari's regexp engines. First we
452
- // replace the JSON backslash pairs with '@' (a non-JSON character). Second, we
453
- // replace all simple value tokens with ']' characters. Third, we delete all
454
- // open brackets that follow a colon or comma or that begin the text. Finally,
455
- // we look to see that the remaining characters are only whitespace or ']' or
456
- // ',' or ':' or '{' or '}'. If that is so, then the text is safe for eval.
457
-
458
- if (/^[\],:{}\s]*$/.
459
- test(text.replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, '@').
460
- replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']').
461
- replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) {
462
-
463
- // In the third stage we use the eval function to compile the text into a
464
- // JavaScript structure. The '{' operator is subject to a syntactic ambiguity
465
- // in JavaScript: it can begin a block or an object literal. We wrap the text
466
- // in parens to eliminate the ambiguity.
467
-
468
- j = eval('(' + text + ')');
469
-
470
- // In the optional fourth stage, we recursively walk the new structure, passing
471
- // each name/value pair to a reviver function for possible transformation.
472
-
473
- return typeof reviver === 'function' ?
474
- walk({'': j}, '') : j;
475
- }
476
-
477
- // If the text is not JSON parseable, then a SyntaxError is thrown.
478
-
479
- throw new SyntaxError('JSON.parse');
480
- };
481
- }
482
- }());
@@ -1,53 +0,0 @@
1
- module CouchPotato
2
- module RSpec
3
- class ListAsProxy
4
- def initialize(results_ruby)
5
- @results_ruby = results_ruby
6
- end
7
-
8
- def as(expected_ruby)
9
- ListAsMatcher.new(expected_ruby, @results_ruby)
10
- end
11
- end
12
-
13
- class ListAsMatcher
14
- include RunJS
15
-
16
- def initialize(expected_ruby, results_ruby)
17
- @expected_ruby = expected_ruby
18
- @results_ruby = results_ruby
19
- end
20
-
21
- def matches?(view_spec)
22
- js = <<-JS
23
- #{File.read(File.dirname(__FILE__) + '/json2.js')}
24
- var results = #{@results_ruby.to_json};
25
- var listed = '';
26
- var list = #{view_spec.list_function};
27
-
28
- var getRow = function() {
29
- return results.rows.shift();
30
- };
31
- var send = function(text) {
32
- listed = listed + text;
33
- };
34
- list();
35
- JSON.stringify(JSON.parse(listed));
36
- JS
37
-
38
- @actual_ruby = JSON.parse(run_js(js))
39
-
40
- @expected_ruby == @actual_ruby
41
- end
42
-
43
- def failure_message
44
- "Expected to list as #{@expected_ruby.inspect} but got #{@actual_ruby.inspect}."
45
- end
46
-
47
- def failure_message_when_negated
48
- "Expected to not list as #{@expected_ruby.inspect} but did."
49
- end
50
-
51
- end
52
- end
53
- end
@@ -1,166 +0,0 @@
1
- require 'json'
2
-
3
- module CouchPotato
4
- module RSpec
5
- class MapReduceToProxy
6
- def initialize(*input_ruby)
7
- @input_ruby, @options = input_ruby.flatten, {}
8
- end
9
-
10
- def with_options(options)
11
- @options = options
12
- self
13
- end
14
-
15
- def to(*expected_ruby)
16
- MapReduceToMatcher.new(expected_ruby, @input_ruby, @options)
17
- end
18
- end
19
-
20
- class MapReduceToMatcher
21
- include RunJS
22
-
23
- def initialize(expected_ruby, input_ruby, options)
24
- @expected_ruby = expected_ruby
25
- @input_ruby = input_ruby
26
- @options = options
27
- end
28
-
29
- def matches?(view_spec)
30
- js = <<-JS
31
- var sum = function(values) {
32
- return values.reduce(function(memo, value) { return memo + value; });
33
- };
34
- // Equivalents of couchdb built-in reduce functions whose names can be
35
- // given as the reduce function in the view_spec:
36
- var _sum = function(keys, values, rereduce) {
37
- return sum(values);
38
- }
39
- var _count = function(keys, values, rereduce) {
40
- if (rereduce) {
41
- return sum(values);
42
- } else {
43
- return values.length;
44
- }
45
- }
46
- var _stats = function(keys, values, rereduce) {
47
- var result = {sum: 0, count: 0, min: Number.MAX_VALUE, max: Number.MIN_VALUE, sumsqr: 0};
48
- if (rereduce) {
49
- for (var i in values) {
50
- var value = values[i];
51
- result.sum += value.sum;
52
- result.count += value.count;
53
- result.min = Math.min(result.min, value.min);
54
- result.max = Math.max(result.max, value.max);
55
- result.sumsqr += value.sumsqr;
56
- }
57
- } else {
58
- for (var i in values) {
59
- var value = values[i];
60
- result.sum += value;
61
- result.count += 1;
62
- result.min = Math.min(result.min, value);
63
- result.max = Math.max(result.max, value);
64
- result.sumsqr += Math.pow(value, 2);
65
- }
66
- }
67
- return result;
68
- }
69
-
70
- var docs = #{@input_ruby.to_json};
71
- var options = #{@options.to_json};
72
- var map = #{view_spec.map_function};
73
- var reduce = #{view_spec.reduce_function};
74
- var lib = #{view_spec.respond_to?(:lib) && view_spec.lib.to_json};
75
-
76
- // Map the input docs
77
- var require = function(modulePath) {
78
- var module = {exports: {}};
79
- var exports = module.exports;
80
- var pathArray = modulePath.split("/").slice(2);
81
- var result = lib;
82
- for (var i in pathArray) {
83
- result = result[pathArray[i]];
84
- }
85
- eval(result);
86
- return module.exports;
87
- }
88
-
89
- var mapResults = [];
90
- var emit = function(key, value) {
91
- mapResults.push({key: key, value: value});
92
- };
93
- for (var i in docs) {
94
- map(docs[i]);
95
- }
96
-
97
- // Group the map results, honoring the group and group_level options
98
- var grouped = [];
99
- if (options.group || options.group_level) {
100
- var groupLevel = options.group_level;
101
- if (groupLevel == "exact" || options.group == true)
102
- groupLevel = 9999;
103
- var keysEqual = function(a, b) { return !(a < b || b < a); }
104
-
105
- for (var mr in mapResults) {
106
- var mapResult = mapResults[mr];
107
- var groupedKey = mapResult.key.slice(0, groupLevel);
108
- var groupFound = false;
109
- for (var g in grouped) {
110
- var group = grouped[g];
111
- if (keysEqual(groupedKey, group.groupedKey)) {
112
- group.keys.push(mapResult.key);
113
- group.values.push(mapResult.value);
114
- groupFound = true;
115
- break;
116
- }
117
- }
118
-
119
- if (!groupFound)
120
- grouped.push({keys: [mapResult.key], groupedKey: groupedKey, values: [mapResult.value]});
121
- }
122
- } else {
123
- var group = {keys: null, groupedKey: null, values: []};
124
- for (var mr in mapResults)
125
- group.values.push(mapResults[mr].value);
126
- grouped.push(group);
127
- }
128
-
129
- // Reduce the grouped map results
130
- var results = [];
131
- for (var g in grouped) {
132
- var group = grouped[g], reduced = null;
133
- if (group.values.length >= 2) {
134
- // Split the values into two parts, reduce each part, then rereduce those results
135
- var mid = parseInt(group.values.length / 2);
136
- var keys1 = (group.keys || []).slice(0, mid),
137
- values1 = group.values.slice(0, mid);
138
- var reduced1 = reduce(keys1, values1, false);
139
- var keys2 = (group.keys || []).slice(mid, group.values.length),
140
- values2 = group.values.slice(mid, group.values.length);
141
- var reduced2 = reduce(keys2, values2, false);
142
- reduced = reduce(null, [reduced1, reduced2], true);
143
- } else {
144
- // Not enough values to split, so just reduce, and then rereduce the single result
145
- reduced = reduce(group.keys, group.values, false);
146
- reduced = reduce(null, [reduced], true);
147
- }
148
- results.push({key: group.groupedKey, value: reduced});
149
- }
150
-
151
- JSON.stringify(results);
152
- JS
153
- @actual_ruby = JSON.parse(run_js(js))
154
- @expected_ruby == @actual_ruby
155
- end
156
-
157
- def failure_message
158
- "Expected to map/reduce to #{@expected_ruby.inspect} but got #{@actual_ruby.inspect}."
159
- end
160
-
161
- def failure_message_when_negated
162
- "Expected not to map/reduce to #{@actual_ruby.inspect} but did."
163
- end
164
- end
165
- end
166
- end
@@ -1,61 +0,0 @@
1
- require 'json'
2
-
3
-
4
- module CouchPotato
5
- module RSpec
6
- class MapToProxy
7
- def initialize(input_ruby)
8
- @input_ruby = input_ruby
9
- end
10
-
11
- def to(*expected_ruby)
12
- MapToMatcher.new(expected_ruby, @input_ruby)
13
- end
14
- end
15
-
16
- class MapToMatcher
17
- include RunJS
18
-
19
- def initialize(expected_ruby, input_ruby)
20
- @expected_ruby = expected_ruby
21
- @input_ruby = input_ruby
22
- end
23
-
24
- def matches?(view_spec)
25
- js = <<-JS
26
- var doc = #{@input_ruby.to_json};
27
- var map = #{view_spec.map_function};
28
- var lib = #{view_spec.respond_to?(:lib) && view_spec.lib.to_json};
29
- var result = [];
30
- var require = function(modulePath) {
31
- var module = {exports: {}};
32
- var exports = module.exports;
33
- var pathArray = modulePath.split("/").slice(2);
34
- var result = lib;
35
- for (var i in pathArray) {
36
- result = result[pathArray[i]];
37
- }
38
- eval(result);
39
- return module.exports;
40
- }
41
-
42
- var emit = function(key, value) {
43
- result.push([key, value]);
44
- };
45
- map(doc);
46
- JSON.stringify(result);
47
- JS
48
- @actual_ruby = JSON.parse(run_js(js))
49
- @expected_ruby == @actual_ruby
50
- end
51
-
52
- def failure_message
53
- "Expected to map to #{@expected_ruby.inspect} but got #{@actual_ruby.inspect}."
54
- end
55
-
56
- def failure_message_when_negated
57
- "Expected not to map to #{@actual_ruby.inspect} but did."
58
- end
59
- end
60
- end
61
- end
@@ -1,48 +0,0 @@
1
- module CouchPotato
2
- module RSpec
3
- class ReduceToProxy
4
- def initialize(keys, values, rereduce = false)
5
- @keys, @values, @rereduce = keys, values, rereduce
6
- end
7
-
8
- def to(expected_ruby)
9
- ReduceToMatcher.new(expected_ruby, @keys, @values, @rereduce)
10
- end
11
- end
12
-
13
- class ReduceToMatcher
14
- include RunJS
15
-
16
- def initialize(expected_ruby, keys, values, rereduce = false)
17
- @expected_ruby, @keys, @values, @rereduce = expected_ruby, keys, values, rereduce
18
- end
19
-
20
- def matches?(view_spec)
21
- js = <<-JS
22
- sum = function(values) {
23
- var rv = 0;
24
- for (var i in values) {
25
- rv += values[i];
26
- }
27
- return rv;
28
- };
29
-
30
- var keys = #{@keys.to_json};
31
- var values = #{@values.to_json};
32
- var reduce = #{view_spec.reduce_function};
33
- JSON.stringify({result: reduce(keys, values, #{@rereduce})});
34
- JS
35
- @actual_ruby = JSON.parse(run_js(js))['result']
36
- @expected_ruby == @actual_ruby
37
- end
38
-
39
- def failure_message
40
- "Expected to reduce to #{@expected_ruby.inspect} but got #{@actual_ruby.inspect}."
41
- end
42
-
43
- def failure_message_when_negated
44
- "Expected not to reduce to #{@actual_ruby.inspect} but did."
45
- end
46
- end
47
- end
48
- end
@@ -1,56 +0,0 @@
1
- begin
2
- require 'v8'
3
- rescue LoadError
4
- begin
5
- require 'rhino'
6
- rescue LoadError
7
- STDERR.puts "You need to install therubyracer (or therhinoracer on jruby) to use matchers"
8
- end
9
- end
10
-
11
- module CouchPotato
12
- module RSpec
13
- module RunJS
14
- private
15
-
16
- def run_js(js)
17
- if defined?(V8)
18
- V8::Context.new.eval(js)
19
- else
20
- Rhino::Context.open{|cxt| cxt.eval(js)}
21
- end
22
- end
23
- end
24
- end
25
- end
26
-
27
-
28
- require 'couch_potato/rspec/matchers/map_to_matcher'
29
- require 'couch_potato/rspec/matchers/reduce_to_matcher'
30
- require 'couch_potato/rspec/matchers/map_reduce_to_matcher'
31
- require 'couch_potato/rspec/matchers/list_as_matcher'
32
-
33
- module RSpec
34
- module Matchers
35
- def map(document)
36
- CouchPotato::RSpec::MapToProxy.new(document)
37
- end
38
-
39
- def reduce(keys, values)
40
- CouchPotato::RSpec::ReduceToProxy.new(keys, values)
41
- end
42
-
43
- def rereduce(keys, values)
44
- CouchPotato::RSpec::ReduceToProxy.new(keys, values, true)
45
- end
46
-
47
- def list(results)
48
- CouchPotato::RSpec::ListAsProxy.new(results)
49
- end
50
-
51
- def map_reduce(*docs)
52
- CouchPotato::RSpec::MapReduceToProxy.new(docs)
53
- end
54
- end
55
- end
56
-
@@ -1,55 +0,0 @@
1
- require 'rspec/mocks'
2
-
3
- module CouchPotato::RSpec
4
- module StubView
5
- class ViewStub
6
- include RSpec::Mocks::ExampleMethods
7
-
8
- def initialize(clazz, view, db)
9
- @clazz = clazz
10
- @view = view
11
- @db = db
12
- end
13
-
14
- def with(*args, &block)
15
- @args = args
16
- and_return(block.call) if block
17
- self
18
- end
19
-
20
- def and_return(return_value)
21
- view_stub = double("#{@clazz}.#{@view}(#{@args.try(:join, ', ')}) view")
22
- stub = allow(@clazz).to receive(@view)
23
- stub.with(*@args) if @args
24
- stub.and_return(view_stub)
25
- allow(@db).to receive(:view).with(view_stub).and_return(return_value)
26
- return unless return_value.respond_to?(:first)
27
- allow(@db).to receive(:first).with(view_stub).and_return(return_value.first)
28
- if return_value.first
29
- allow(@db).to receive(:first!).with(view_stub).and_return(return_value.first)
30
- else
31
- allow(@db).to receive(:first!).with(view_stub).and_raise(CouchPotato::NotFound)
32
- end
33
- end
34
- end
35
-
36
- def stub_view(clazz, view, &block)
37
- stub = ViewStub.new clazz, view, self
38
- stub.and_return(block.call) if block
39
- stub
40
- end
41
- end
42
-
43
- module StubDb
44
- include ::RSpec::Mocks::ExampleMethods
45
-
46
- def stub_db(options = {})
47
- db = double(:db, options)
48
- db.extend CouchPotato::RSpec::StubView
49
- allow(self).to receive(:database) { db }
50
- db
51
- end
52
- end
53
-
54
- ::CouchPotato.extend StubDb
55
- end
@@ -1,2 +0,0 @@
1
- require 'couch_potato/rspec/stub_db'
2
- require 'couch_potato/rspec/matchers'