couch_potato-rspec 3.4.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: ac4b847aec3ea03fe48ea86a0af0db32bc981acf
4
+ data.tar.gz: a9022b27f106bfee7accce3260cfdcf0a847967e
5
+ SHA512:
6
+ metadata.gz: f4036cf4ab129a4d260859f885d16d6b0edc871a039ab758c43f16a71df711120d72273c77f5f4a8609fff4d0f9cd773a78f219fd5b2cb5dc119134256cd95c3
7
+ data.tar.gz: 9dbc3ee9607610d208d9de4382944a1d3d82c2a46da6c17cfca6a44fb02ba770332447dc8bc8c73cb7fe5ac3c368e122f28e8f84ae2e893934c0274bf500cf06
@@ -0,0 +1,2 @@
1
+ require 'couch_potato/rspec/stub_db'
2
+ require 'couch_potato/rspec/matchers'
@@ -0,0 +1,56 @@
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
+
@@ -0,0 +1,482 @@
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
+ }());
@@ -0,0 +1,53 @@
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
@@ -0,0 +1,166 @@
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
@@ -0,0 +1,61 @@
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
@@ -0,0 +1,48 @@
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
@@ -0,0 +1,55 @@
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
@@ -0,0 +1,317 @@
1
+ require 'spec_helper'
2
+ require 'couch_potato-rspec'
3
+
4
+ describe CouchPotato::RSpec::MapToMatcher do
5
+
6
+ describe "basic map function" do
7
+ before(:each) do
8
+ @view_spec = double(:map_function => "function(doc) {emit(doc.name, doc.tags.length);}")
9
+ end
10
+
11
+ it "should pass if the given function emits the expected javascript" do
12
+ expect(@view_spec).to map({:name => 'horst', :tags => ['person', 'male']}).to(['horst', 2])
13
+ end
14
+
15
+ it "should not pass if the given function emits different javascript" do
16
+ expect(@view_spec).not_to map({:name => 'horst', :tags => ['person', 'male']}).to(['horst', 3])
17
+ end
18
+ end
19
+
20
+ describe "functions emitting multiple times" do
21
+ before(:each) do
22
+ @view_spec = double(:map_function => "function(doc) {emit(doc.name, doc.tags.length); emit(doc.tags[0], doc.tags[1])};")
23
+ end
24
+
25
+ it "should pass if the given function emits the expected javascript" do
26
+ expect(@view_spec).to map({:name => 'horst', :tags => ['person', 'male']}).to(['horst', 2], ['person', 'male'])
27
+ end
28
+
29
+ it "should return false if the given function emits different javascript" do
30
+ expect(@view_spec).not_to map({:name => 'horst', :tags => ['person', 'male']}).to(['horst', 2], ['male', 'person'])
31
+ end
32
+ end
33
+
34
+ it "should work with date emit values" do
35
+ spec = double(:map_function => "function(doc) { emit(null, new Date(1368802800000)); }")
36
+ expect(spec).to map({}).to([nil, "2013-05-17T15:00:00.000Z"])
37
+ end
38
+
39
+ it "should work with commonJS modules that use 'exports'" do
40
+ spec = double(
41
+ :map_function => "function(doc) { var test = require('views/lib/test'); emit(null, test.test); }",
42
+ :lib => {:test => "exports.test = 'test';"}
43
+ )
44
+ expect(spec).to map({}).to([nil, "test"])
45
+ end
46
+
47
+ it "should work with commonJS modules that use 'module.exports'" do
48
+ spec = double(
49
+ :map_function => "function(doc) { var test = require('views/lib/test'); emit(null, test.test); }",
50
+ :lib => {:test => "module.exports.test = 'test';"}
51
+ )
52
+ expect(spec).to map({}).to([nil, "test"])
53
+ end
54
+
55
+ describe "failing specs" do
56
+ before(:each) do
57
+ @view_spec = double(:map_function => "function(doc) {emit(doc.name, null)}")
58
+ end
59
+
60
+ it "should have a nice error message for failing should" do
61
+ expect {
62
+ expect(@view_spec).to map({:name => 'bill'}).to(['linus', nil])
63
+ }.to raise_error('Expected to map to [["linus", nil]] but got [["bill", nil]].')
64
+ end
65
+
66
+ it "should have a nice error message for failing should not" do
67
+ expect {
68
+ expect(@view_spec).not_to map({:name => 'bill'}).to(['bill', nil])
69
+ }.to raise_error('Expected not to map to [["bill", nil]] but did.')
70
+ end
71
+ end
72
+ end
73
+
74
+ describe CouchPotato::RSpec::ReduceToMatcher do
75
+ before(:each) do
76
+ @view_spec = double(:reduce_function => "function(keys, values, rereduce) {
77
+ if(rereduce) {
78
+ return(sum(values) * 2);
79
+ } else {
80
+ return(sum(values));
81
+ };
82
+ }")
83
+ end
84
+
85
+ it "should pass if the given function return the expected javascript" do
86
+ expect(@view_spec).to reduce([], [1, 2, 3]).to(6)
87
+ end
88
+
89
+ it "should not pass if the given function returns different javascript" do
90
+ expect(@view_spec).not_to reduce([], [1, 2, 3]).to(7)
91
+ end
92
+
93
+ it "should work with date return values" do
94
+ spec = double(:reduce_function => "function() { return new Date(1368802800000); }")
95
+ expect(spec).to reduce([], []).to("2013-05-17T15:00:00.000Z")
96
+ end
97
+
98
+ describe "rereduce" do
99
+ it "should pass if the given function return the expected javascript" do
100
+ expect(@view_spec).to rereduce([], [1, 2, 3]).to(12)
101
+ end
102
+
103
+ it "should not pass if the given function returns different javascript" do
104
+ expect(@view_spec).not_to rereduce([], [1, 2, 3]).to(13)
105
+ end
106
+ end
107
+
108
+ describe 'failing specs' do
109
+
110
+ it "should have a nice error message for failing should" do
111
+ expect {
112
+ expect(@view_spec).to reduce([], [1, 2, 3]).to(7)
113
+ }.to raise_error('Expected to reduce to 7 but got 6.')
114
+ end
115
+
116
+ it "should have a nice error message for failing should not" do
117
+ expect {
118
+ expect(@view_spec).not_to reduce([], [1, 2, 3]).to(6)
119
+ }.to raise_error('Expected not to reduce to 6 but did.')
120
+ end
121
+ end
122
+ end
123
+
124
+ describe CouchPotato::RSpec::MapReduceToMatcher do
125
+ before(:each) do
126
+ @view_spec = double(
127
+ :map_function => "function(doc) {
128
+ for (var i in doc.numbers)
129
+ emit([doc.age, doc.name], doc.numbers[i]);
130
+ }",
131
+ :reduce_function => "function (keys, values, rereduce) {
132
+ return Math.max.apply(this, values);
133
+ }"
134
+ )
135
+ @docs = [
136
+ {:name => "a", :age => 25, :numbers => [1, 2]},
137
+ {:name => "b", :age => 25, :numbers => [3, 4]},
138
+ {:name => "c", :age => 26, :numbers => [5, 6]},
139
+ {:name => "d", :age => 27, :numbers => [7, 8]}]
140
+ end
141
+
142
+ it "should handle date return values" do
143
+ spec = double(:map_function => "function() { emit(null, null); }",
144
+ :reduce_function => "function() { return new Date(1368802800000); }")
145
+ expect(spec).to map_reduce({}).to({"key" => nil, "value" => "2013-05-17T15:00:00.000Z"})
146
+ end
147
+
148
+ it "should handle CommonJS requires for modules that use 'exports'" do
149
+ spec = double(
150
+ :map_function => "function() { var test = require('views/lib/test'); emit(null, test.test); }",
151
+ :reduce_function => "function(keys, values) { return 'test' }",
152
+ :lib => {:test => "exports.test = 'test'"})
153
+ expect(spec).to map_reduce({}).to({"key" => nil, "value" => "test"})
154
+ end
155
+
156
+ it "should handle CommonJS requires for modules that use 'module.exports'" do
157
+ spec = double(
158
+ :map_function => "function() { var test = require('views/lib/test'); emit(null, test.test); }",
159
+ :reduce_function => "function(keys, values) { return 'test' }",
160
+ :lib => {:test => "module.exports.test = 'test'"})
161
+ expect(spec).to map_reduce({}).to({"key" => nil, "value" => "test"})
162
+ end
163
+
164
+ it "should handle sum function" do
165
+ spec = double(
166
+ :map_function => "function(doc) { emit(null, doc.age); }",
167
+ :reduce_function => "function(keys, values) { return sum(values); }")
168
+ expect(spec).to map_reduce(@docs).to({"key" => nil, "value" => 103})
169
+ end
170
+
171
+ context "without grouping" do
172
+ it "should not group by key by default" do
173
+ expect(@view_spec).to map_reduce(@docs).to({"key" => nil, "value" => 8})
174
+ end
175
+
176
+ it "should group by key with :group => false" do
177
+ expect(@view_spec).to map_reduce(@docs).with_options(:group => false).to({"key" => nil, "value" => 8})
178
+ end
179
+ end
180
+
181
+ context "with grouping" do
182
+ [true, "exact"].each do |group_value|
183
+ it "should group by the full key with option :group => #{group_value}" do
184
+ expect(@view_spec).to map_reduce(@docs).with_options(:group => group_value).to(
185
+ {"key" => [25, "a"], "value" => 2},
186
+ {"key" => [25, "b"], "value" => 4},
187
+ {"key" => [26, "c"], "value" => 6},
188
+ {"key" => [27, "d"], "value" => 8})
189
+ end
190
+ end
191
+
192
+ it "should group by parts of the keys based on the :group_level option" do
193
+ expect(@view_spec).to map_reduce(@docs).with_options(:group_level => 1).to(
194
+ {"key" => [25], "value" => 4},
195
+ {"key" => [26], "value" => 6},
196
+ {"key" => [27], "value" => 8})
197
+ end
198
+ end
199
+
200
+ describe "rereducing" do
201
+ before :each do
202
+ @view_spec = double(:map_function => "function(doc) {
203
+ emit(doc.name, doc.number);
204
+ }",
205
+ :reduce_function => "function (keys, values, rereduce) {
206
+ if (rereduce) {
207
+ var result = {rereduce_values: []};
208
+ for (var v in values) {
209
+ result.rereduce_values.push(values[v].reduce_values);
210
+ }
211
+ return result;
212
+ }
213
+ return {reduce_values: values};
214
+ }")
215
+ end
216
+
217
+ it "should reduce and rereduce for a single emit" do
218
+ expect(@view_spec).to map_reduce({:name => "a", :number => 1}).to({"key" => nil, "value" => {"rereduce_values" => [[1]]}})
219
+ end
220
+
221
+ it "should split and reduce each half of emitted values separately and rereduce the results" do
222
+ docs = [
223
+ {:name => "a", :number => 1},
224
+ {:name => "a", :number => 2},
225
+ {:name => "a", :number => 3},
226
+ {:name => "a", :number => 4}]
227
+ expect(@view_spec).to map_reduce(docs).to({"key" => nil, "value" => {"rereduce_values" => [[1, 2], [3, 4]]}})
228
+ end
229
+
230
+ it "should correctly split and rereduce with an odd number of emits" do
231
+ docs = [
232
+ {:name => "a", :number => 1},
233
+ {:name => "a", :number => 2},
234
+ {:name => "a", :number => 3}]
235
+ expect(@view_spec).to map_reduce(docs).to({"key" => nil, "value" => {"rereduce_values" => [[1], [2, 3]]}})
236
+ end
237
+ end
238
+
239
+ describe "failing specs" do
240
+ it "should have a nice error message for failing should" do
241
+ expect {
242
+ expect(@view_spec).to map_reduce(@docs).with_options(:group => false).to({"key" => nil, "value" => 9})
243
+ }.to raise_error('Expected to map/reduce to [{"key"=>nil, "value"=>9}] but got [{"key"=>nil, "value"=>8}].')
244
+ end
245
+
246
+ it "should have a nice error message for failing should not" do
247
+ expect {
248
+ expect(@view_spec).not_to map_reduce(@docs).with_options(:group => false).to({"key" => nil, "value" => 8})
249
+ }.to raise_error('Expected not to map/reduce to [{"key"=>nil, "value"=>8}] but did.')
250
+ end
251
+ end
252
+
253
+ describe "couchdb built-in reduce functions" do
254
+ describe "_sum" do
255
+ it "should return the sum of emitted values" do
256
+ spec = double(:map_function => @view_spec.map_function, :reduce_function => "_sum")
257
+ expect(spec).to map_reduce(@docs).to({"key" => nil, "value" => 36})
258
+ end
259
+ end
260
+
261
+ describe "_count" do
262
+ it "should return the count of emitted values" do
263
+ spec = double(:map_function => @view_spec.map_function, :reduce_function => "_count")
264
+ expect(spec).to map_reduce(@docs).to({"key" => nil, "value" => 8})
265
+ end
266
+ end
267
+
268
+ describe "_stats" do
269
+ it "should return the numerical statistics of emitted values" do
270
+ spec = double(:map_function => @view_spec.map_function, :reduce_function => "_stats")
271
+ expect(spec).to map_reduce(@docs).to({
272
+ "key" => nil,
273
+ "value" => {
274
+ "sum" => 36,
275
+ "count" => 8,
276
+ "min" => 1,
277
+ "max" => 8,
278
+ "sumsqr" => 204
279
+ }
280
+ })
281
+ end
282
+ end
283
+ end
284
+ end
285
+
286
+ describe CouchPotato::RSpec::ListAsMatcher do
287
+ before(:each) do
288
+ @view_spec = double(:list_function => "function() {var row = getRow(); send(JSON.stringify([{text: row.text + ' world'}]));}")
289
+ end
290
+
291
+ it "should pass if the function return the expected json" do
292
+ expect(@view_spec).to list({'rows' => [{:text => 'hello'}]}).as([{'text' => 'hello world'}])
293
+ end
294
+
295
+ it "should not pass if the function does not return the expected json" do
296
+ expect(@view_spec).not_to list({'rows' => [{:text => 'hello'}]}).as([{'text' => 'hello there'}])
297
+ end
298
+
299
+ it "should work with date values" do
300
+ spec = double(:list_function => "function() { send(JSON.stringify([{date: new Date(1368802800000)}])); }")
301
+ expect(spec).to list({"rows" => [{}]}).as([{"date" => "2013-05-17T15:00:00.000Z"}])
302
+ end
303
+
304
+ describe "failing specs" do
305
+ it "should have a nice error message for failing should" do
306
+ expect {
307
+ expect(@view_spec).to list({'rows' => [{:text => 'hello'}]}).as([{'text' => 'hello there'}])
308
+ }.to raise_error('Expected to list as [{"text"=>"hello there"}] but got [{"text"=>"hello world"}].')
309
+ end
310
+
311
+ it "should have a nice error message for failing should not" do
312
+ expect {
313
+ expect(@view_spec).not_to list({'rows' => [{:text => 'hello'}]}).as([{'text' => 'hello world'}])
314
+ }.to raise_error('Expected to not list as [{"text"=>"hello world"}] but did.')
315
+ end
316
+ end
317
+ end
metadata ADDED
@@ -0,0 +1,81 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: couch_potato-rspec
3
+ version: !ruby/object:Gem::Version
4
+ version: 3.4.0
5
+ platform: ruby
6
+ authors:
7
+ - Alexander Lang
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2016-03-29 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: rspec
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: '3.4'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: '3.4'
27
+ - !ruby/object:Gem::Dependency
28
+ name: rake
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: '0'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
39
+ - !ruby/object:Gem::Version
40
+ version: '0'
41
+ description: RSpec matchers for Couch Potato
42
+ email: alex@upstre.am
43
+ executables: []
44
+ extensions: []
45
+ extra_rdoc_files: []
46
+ 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
+ - spec/unit/rspec_matchers_spec.rb
56
+ homepage: http://github.com/langalex/couch_potato
57
+ licenses: []
58
+ metadata: {}
59
+ post_install_message:
60
+ rdoc_options: []
61
+ require_paths:
62
+ - lib
63
+ required_ruby_version: !ruby/object:Gem::Requirement
64
+ requirements:
65
+ - - ">="
66
+ - !ruby/object:Gem::Version
67
+ version: '0'
68
+ required_rubygems_version: !ruby/object:Gem::Requirement
69
+ requirements:
70
+ - - ">="
71
+ - !ruby/object:Gem::Version
72
+ version: '0'
73
+ requirements: []
74
+ rubyforge_project:
75
+ rubygems_version: 2.4.7
76
+ signing_key:
77
+ specification_version: 4
78
+ summary: RSpec matchers for Couch Potato
79
+ test_files:
80
+ - spec/unit/rspec_matchers_spec.rb
81
+ has_rdoc: