glasner-couchrest 0.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (90) hide show
  1. data/LICENSE +176 -0
  2. data/README.md +93 -0
  3. data/Rakefile +66 -0
  4. data/THANKS.md +18 -0
  5. data/examples/model/example.rb +144 -0
  6. data/examples/word_count/markov +38 -0
  7. data/examples/word_count/views/books/chunked-map.js +3 -0
  8. data/examples/word_count/views/books/united-map.js +1 -0
  9. data/examples/word_count/views/markov/chain-map.js +6 -0
  10. data/examples/word_count/views/markov/chain-reduce.js +7 -0
  11. data/examples/word_count/views/word_count/count-map.js +6 -0
  12. data/examples/word_count/views/word_count/count-reduce.js +3 -0
  13. data/examples/word_count/word_count.rb +46 -0
  14. data/examples/word_count/word_count_query.rb +40 -0
  15. data/examples/word_count/word_count_views.rb +26 -0
  16. data/lib/couchrest.rb +189 -0
  17. data/lib/couchrest/commands/generate.rb +71 -0
  18. data/lib/couchrest/commands/push.rb +103 -0
  19. data/lib/couchrest/core/database.rb +313 -0
  20. data/lib/couchrest/core/design.rb +89 -0
  21. data/lib/couchrest/core/document.rb +96 -0
  22. data/lib/couchrest/core/response.rb +16 -0
  23. data/lib/couchrest/core/server.rb +88 -0
  24. data/lib/couchrest/core/view.rb +4 -0
  25. data/lib/couchrest/helper/pager.rb +103 -0
  26. data/lib/couchrest/helper/streamer.rb +44 -0
  27. data/lib/couchrest/mixins.rb +4 -0
  28. data/lib/couchrest/mixins/attachments.rb +31 -0
  29. data/lib/couchrest/mixins/callbacks.rb +483 -0
  30. data/lib/couchrest/mixins/design_doc.rb +64 -0
  31. data/lib/couchrest/mixins/document_queries.rb +48 -0
  32. data/lib/couchrest/mixins/extended_attachments.rb +68 -0
  33. data/lib/couchrest/mixins/extended_document_mixins.rb +6 -0
  34. data/lib/couchrest/mixins/properties.rb +125 -0
  35. data/lib/couchrest/mixins/validation.rb +234 -0
  36. data/lib/couchrest/mixins/views.rb +168 -0
  37. data/lib/couchrest/monkeypatches.rb +119 -0
  38. data/lib/couchrest/more/casted_model.rb +28 -0
  39. data/lib/couchrest/more/extended_document.rb +217 -0
  40. data/lib/couchrest/more/property.rb +40 -0
  41. data/lib/couchrest/support/blank.rb +42 -0
  42. data/lib/couchrest/support/class.rb +191 -0
  43. data/lib/couchrest/validation/auto_validate.rb +163 -0
  44. data/lib/couchrest/validation/contextual_validators.rb +78 -0
  45. data/lib/couchrest/validation/validation_errors.rb +118 -0
  46. data/lib/couchrest/validation/validators/absent_field_validator.rb +74 -0
  47. data/lib/couchrest/validation/validators/confirmation_validator.rb +99 -0
  48. data/lib/couchrest/validation/validators/format_validator.rb +117 -0
  49. data/lib/couchrest/validation/validators/formats/email.rb +66 -0
  50. data/lib/couchrest/validation/validators/formats/url.rb +43 -0
  51. data/lib/couchrest/validation/validators/generic_validator.rb +120 -0
  52. data/lib/couchrest/validation/validators/length_validator.rb +134 -0
  53. data/lib/couchrest/validation/validators/method_validator.rb +89 -0
  54. data/lib/couchrest/validation/validators/numeric_validator.rb +104 -0
  55. data/lib/couchrest/validation/validators/required_field_validator.rb +109 -0
  56. data/spec/couchrest/core/couchrest_spec.rb +201 -0
  57. data/spec/couchrest/core/database_spec.rb +745 -0
  58. data/spec/couchrest/core/design_spec.rb +131 -0
  59. data/spec/couchrest/core/document_spec.rb +311 -0
  60. data/spec/couchrest/core/server_spec.rb +35 -0
  61. data/spec/couchrest/helpers/pager_spec.rb +122 -0
  62. data/spec/couchrest/helpers/streamer_spec.rb +23 -0
  63. data/spec/couchrest/more/casted_extended_doc_spec.rb +40 -0
  64. data/spec/couchrest/more/casted_model_spec.rb +98 -0
  65. data/spec/couchrest/more/extended_doc_attachment_spec.rb +130 -0
  66. data/spec/couchrest/more/extended_doc_spec.rb +509 -0
  67. data/spec/couchrest/more/extended_doc_view_spec.rb +207 -0
  68. data/spec/couchrest/more/property_spec.rb +130 -0
  69. data/spec/couchrest/support/class_spec.rb +59 -0
  70. data/spec/fixtures/attachments/README +3 -0
  71. data/spec/fixtures/attachments/couchdb.png +0 -0
  72. data/spec/fixtures/attachments/test.html +11 -0
  73. data/spec/fixtures/more/article.rb +34 -0
  74. data/spec/fixtures/more/card.rb +20 -0
  75. data/spec/fixtures/more/course.rb +14 -0
  76. data/spec/fixtures/more/event.rb +6 -0
  77. data/spec/fixtures/more/invoice.rb +17 -0
  78. data/spec/fixtures/more/person.rb +8 -0
  79. data/spec/fixtures/more/question.rb +6 -0
  80. data/spec/fixtures/more/service.rb +12 -0
  81. data/spec/fixtures/views/lib.js +3 -0
  82. data/spec/fixtures/views/test_view/lib.js +3 -0
  83. data/spec/fixtures/views/test_view/only-map.js +4 -0
  84. data/spec/fixtures/views/test_view/test-map.js +3 -0
  85. data/spec/fixtures/views/test_view/test-reduce.js +3 -0
  86. data/spec/spec.opts +6 -0
  87. data/spec/spec_helper.rb +26 -0
  88. data/utils/remap.rb +27 -0
  89. data/utils/subset.rb +30 -0
  90. metadata +219 -0
@@ -0,0 +1,103 @@
1
+ module CouchRest
2
+
3
+ module Commands
4
+
5
+ module Push
6
+
7
+ def self.run(options)
8
+ directory = options[:directory]
9
+ database = options[:trailing_args].first
10
+
11
+ fm = CouchRest::FileManager.new(database)
12
+ fm.loud = options[:loud]
13
+
14
+ if options[:loud]
15
+ puts "Pushing views from directory #{directory} to database #{fm.db}"
16
+ end
17
+
18
+ fm.push_views(directory)
19
+ end
20
+
21
+ def self.help
22
+ helpstring = <<-GEN
23
+
24
+ == Pushing views with Couchview ==
25
+
26
+ Usage: couchview push directory dbname
27
+
28
+ Couchview expects a specific filesystem layout for your CouchDB views (see
29
+ example below). It also supports advanced features like inlining of library
30
+ code (so you can keep DRY) as well as avoiding unnecessary document
31
+ modification.
32
+
33
+ Couchview also solves a problem with CouchDB's view API, which only provides
34
+ access to the final reduce side of any views which have both a map and a
35
+ reduce function defined. The intermediate map results are often useful for
36
+ development and production. CouchDB is smart enough to reuse map indexes for
37
+ functions duplicated across views within the same design document.
38
+
39
+ For views with a reduce function defined, Couchview creates both a reduce view
40
+ and a map-only view, so that you can browse and query the map side as well as
41
+ the reduction, with no performance penalty.
42
+
43
+ == Example ==
44
+
45
+ couchview push foo-project/bar-views baz-database
46
+
47
+ This will push the views defined in foo-project/bar-views into a database
48
+ called baz-database. Couchview expects the views to be defined in files with
49
+ names like:
50
+
51
+ foo-project/bar-views/my-design/viewname-map.js
52
+ foo-project/bar-views/my-design/viewname-reduce.js
53
+ foo-project/bar-views/my-design/noreduce-map.js
54
+
55
+ Pushed to => http://127.0.0.1:5984/baz-database/_design/my-design
56
+
57
+ And the design document:
58
+ {
59
+ "views" : {
60
+ "viewname-map" : {
61
+ "map" : "### contents of view-name-map.js ###"
62
+ },
63
+ "viewname-reduce" : {
64
+ "map" : "### contents of view-name-map.js ###",
65
+ "reduce" : "### contents of view-name-reduce.js ###"
66
+ },
67
+ "noreduce-map" : {
68
+ "map" : "### contents of noreduce-map.js ###"
69
+ }
70
+ }
71
+ }
72
+
73
+ Couchview will create a design document for each subdirectory of the views
74
+ directory specified on the command line.
75
+
76
+ == Library Inlining ==
77
+
78
+ Couchview can optionally inline library code into your views so you only have
79
+ to maintain it in one place. It looks for any files named lib.* in your
80
+ design-doc directory (for doc specific libs) and in the parent views directory
81
+ (for project global libs). These libraries are only inserted into views which
82
+ include the text
83
+
84
+ // !include lib
85
+
86
+ or
87
+
88
+ # !include lib
89
+
90
+ Couchview is a result of scratching my own itch. I'd be happy to make it more
91
+ general, so please contact me at jchris@grabb.it if you'd like to see anything
92
+ added or changed.
93
+
94
+ GEN
95
+ helpstring.gsub(/^ /, '')
96
+ end
97
+
98
+ end
99
+
100
+
101
+ end
102
+
103
+ end
@@ -0,0 +1,313 @@
1
+ require 'cgi'
2
+ require "base64"
3
+
4
+ module CouchRest
5
+ class Database
6
+ attr_reader :server, :host, :name, :root, :uri
7
+ attr_accessor :bulk_save_cache_limit
8
+
9
+ # Create a CouchRest::Database adapter for the supplied CouchRest::Server
10
+ # and database name.
11
+ #
12
+ # ==== Parameters
13
+ # server<CouchRest::Server>:: database host
14
+ # name<String>:: database name
15
+ #
16
+ def initialize(server, name)
17
+ @name = name
18
+ @server = server
19
+ @host = server.uri
20
+ @uri = @root = "#{host}/#{name}"
21
+ @streamer = Streamer.new(self)
22
+ @bulk_save_cache = []
23
+ @bulk_save_cache_limit = 500 # must be smaller than the uuid count
24
+ end
25
+
26
+ # returns the database's uri
27
+ def to_s
28
+ @uri
29
+ end
30
+
31
+ # GET the database info from CouchDB
32
+ def info
33
+ CouchRest.get @uri
34
+ end
35
+
36
+ # Query the <tt>_all_docs</tt> view. Accepts all the same arguments as view.
37
+ def documents(params = {})
38
+ keys = params.delete(:keys)
39
+ url = CouchRest.paramify_url "#{@uri}/_all_docs", params
40
+ if keys
41
+ CouchRest.post(url, {:keys => keys})
42
+ else
43
+ CouchRest.get url
44
+ end
45
+ end
46
+
47
+ # POST a temporary view function to CouchDB for querying. This is not
48
+ # recommended, as you don't get any performance benefit from CouchDB's
49
+ # materialized views. Can be quite slow on large databases.
50
+ def slow_view(funcs, params = {})
51
+ keys = params.delete(:keys)
52
+ funcs = funcs.merge({:keys => keys}) if keys
53
+ url = CouchRest.paramify_url "#{@uri}/_temp_view", params
54
+ JSON.parse(RestClient.post(url, funcs.to_json, {"Content-Type" => 'application/json'}))
55
+ end
56
+
57
+ # backwards compatibility is a plus
58
+ alias :temp_view :slow_view
59
+
60
+ # Query a CouchDB view as defined by a <tt>_design</tt> document. Accepts
61
+ # paramaters as described in http://wiki.apache.org/couchdb/HttpViewApi
62
+ def view(name, params = {}, &block)
63
+ keys = params.delete(:keys)
64
+ url = CouchRest.paramify_url "#{@uri}/_view/#{name}", params
65
+ if keys
66
+ CouchRest.post(url, {:keys => keys})
67
+ else
68
+ if block_given?
69
+ @streamer.view(name, params, &block)
70
+ else
71
+ CouchRest.get url
72
+ end
73
+ end
74
+ end
75
+
76
+ # GET a document from CouchDB, by id. Returns a Ruby Hash.
77
+ def get(id)
78
+ slug = escape_docid(id)
79
+ hash = CouchRest.get("#{@uri}/#{slug}")
80
+ doc = if /^_design/ =~ hash["_id"]
81
+ Design.new(hash)
82
+ else
83
+ Document.new(hash)
84
+ end
85
+ doc.database = self
86
+ doc
87
+ end
88
+
89
+ # GET an attachment directly from CouchDB
90
+ def fetch_attachment(doc, name)
91
+ # slug = escape_docid(docid)
92
+ # name = CGI.escape(name)
93
+ uri = uri_for_attachment(doc, name)
94
+ RestClient.get uri
95
+ # "#{@uri}/#{slug}/#{name}"
96
+ end
97
+
98
+ # PUT an attachment directly to CouchDB
99
+ def put_attachment(doc, name, file, options = {})
100
+ docid = escape_docid(doc['_id'])
101
+ name = CGI.escape(name)
102
+ uri = uri_for_attachment(doc, name)
103
+ JSON.parse(RestClient.put(uri, file, options))
104
+ end
105
+
106
+ # DELETE an attachment directly from CouchDB
107
+ def delete_attachment doc, name
108
+ uri = uri_for_attachment(doc, name)
109
+ # this needs a rev
110
+ JSON.parse(RestClient.delete(uri))
111
+ end
112
+
113
+ # Save a document to CouchDB. This will use the <tt>_id</tt> field from
114
+ # the document as the id for PUT, or request a new UUID from CouchDB, if
115
+ # no <tt>_id</tt> is present on the document. IDs are attached to
116
+ # documents on the client side because POST has the curious property of
117
+ # being automatically retried by proxies in the event of network
118
+ # segmentation and lost responses.
119
+ #
120
+ # If <tt>bulk</tt> is true (false by default) the document is cached for bulk-saving later.
121
+ # Bulk saving happens automatically when #bulk_save_cache limit is exceded, or on the next non bulk save.
122
+ def save_doc(doc, bulk = false)
123
+ if doc['_attachments']
124
+ doc['_attachments'] = encode_attachments(doc['_attachments'])
125
+ end
126
+ if bulk
127
+ @bulk_save_cache << doc
128
+ return bulk_save if @bulk_save_cache.length >= @bulk_save_cache_limit
129
+ return {"ok" => true} # Compatibility with Document#save
130
+ elsif !bulk && @bulk_save_cache.length > 0
131
+ bulk_save
132
+ end
133
+ result = if doc['_id']
134
+ slug = escape_docid(doc['_id'])
135
+ CouchRest.put "#{@uri}/#{slug}", doc
136
+ else
137
+ begin
138
+ slug = doc['_id'] = @server.next_uuid
139
+ CouchRest.put "#{@uri}/#{slug}", doc
140
+ rescue #old version of couchdb
141
+ CouchRest.post @uri, doc
142
+ end
143
+ end
144
+ if result['ok']
145
+ doc['_id'] = result['id']
146
+ doc['_rev'] = result['rev']
147
+ doc.database = self if doc.respond_to?(:database=)
148
+ end
149
+ result
150
+ end
151
+
152
+ ### DEPRECATION NOTICE
153
+ def save(doc, bulk=false)
154
+ puts "CouchRest::Database's save method is being deprecated, please use save_doc instead"
155
+ save_doc(doc, bulk)
156
+ end
157
+
158
+
159
+ # POST an array of documents to CouchDB. If any of the documents are
160
+ # missing ids, supply one from the uuid cache.
161
+ #
162
+ # If called with no arguments, bulk saves the cache of documents to be bulk saved.
163
+ def bulk_save(docs = nil, use_uuids = true)
164
+ if docs.nil?
165
+ docs = @bulk_save_cache
166
+ @bulk_save_cache = []
167
+ end
168
+ if (use_uuids)
169
+ ids, noids = docs.partition{|d|d['_id']}
170
+ uuid_count = [noids.length, @server.uuid_batch_count].max
171
+ noids.each do |doc|
172
+ nextid = @server.next_uuid(uuid_count) rescue nil
173
+ doc['_id'] = nextid if nextid
174
+ end
175
+ end
176
+ CouchRest.post "#{@uri}/_bulk_docs", {:docs => docs}
177
+ end
178
+ alias :bulk_delete :bulk_save
179
+
180
+ # DELETE the document from CouchDB that has the given <tt>_id</tt> and
181
+ # <tt>_rev</tt>.
182
+ #
183
+ # If <tt>bulk</tt> is true (false by default) the deletion is recorded for bulk-saving (bulk-deletion :) later.
184
+ # Bulk saving happens automatically when #bulk_save_cache limit is exceded, or on the next non bulk save.
185
+ def delete_doc(doc, bulk = false)
186
+ raise ArgumentError, "_id and _rev required for deleting" unless doc['_id'] && doc['_rev']
187
+ if bulk
188
+ @bulk_save_cache << { '_id' => doc['_id'], '_rev' => doc['_rev'], '_deleted' => true }
189
+ return bulk_save if @bulk_save_cache.length >= @bulk_save_cache_limit
190
+ return { "ok" => true } # Mimic the non-deferred version
191
+ end
192
+ slug = escape_docid(doc['_id'])
193
+ CouchRest.delete "#{@uri}/#{slug}?rev=#{doc['_rev']}"
194
+ end
195
+
196
+ ### DEPRECATION NOTICE
197
+ def delete(doc, bulk=false)
198
+ puts "CouchRest::Database's delete method is being deprecated, please use delete_doc instead"
199
+ delete_doc(doc, bulk)
200
+ end
201
+
202
+ # COPY an existing document to a new id. If the destination id currently exists, a rev must be provided.
203
+ # <tt>dest</tt> can take one of two forms if overwriting: "id_to_overwrite?rev=revision" or the actual doc
204
+ # hash with a '_rev' key
205
+ def copy_doc(doc, dest)
206
+ raise ArgumentError, "_id is required for copying" unless doc['_id']
207
+ slug = escape_docid(doc['_id'])
208
+ destination = if dest.respond_to?(:has_key?) && dest['_id'] && dest['_rev']
209
+ "#{dest['_id']}?rev=#{dest['_rev']}"
210
+ else
211
+ dest
212
+ end
213
+ CouchRest.copy "#{@uri}/#{slug}", destination
214
+ end
215
+
216
+ ### DEPRECATION NOTICE
217
+ def copy(doc, dest)
218
+ puts "CouchRest::Database's copy method is being deprecated, please use copy_doc instead"
219
+ copy_doc(doc, dest)
220
+ end
221
+
222
+ # MOVE an existing document to a new id. If the destination id currently exists, a rev must be provided.
223
+ # <tt>dest</tt> can take one of two forms if overwriting: "id_to_overwrite?rev=revision" or the actual doc
224
+ # hash with a '_rev' key
225
+ def move_doc(doc, dest)
226
+ raise ArgumentError, "_id and _rev are required for moving" unless doc['_id'] && doc['_rev']
227
+ slug = escape_docid(doc['_id'])
228
+ destination = if dest.respond_to?(:has_key?) && dest['_id'] && dest['_rev']
229
+ "#{dest['_id']}?rev=#{dest['_rev']}"
230
+ else
231
+ dest
232
+ end
233
+ CouchRest.move "#{@uri}/#{slug}?rev=#{doc['_rev']}", destination
234
+ end
235
+
236
+ ### DEPRECATION NOTICE
237
+ def move(doc, dest)
238
+ puts "CouchRest::Database's move method is being deprecated, please use move_doc instead"
239
+ move_doc(doc, dest)
240
+ end
241
+
242
+ # Compact the database, removing old document revisions and optimizing space use.
243
+ def compact!
244
+ CouchRest.post "#{@uri}/_compact"
245
+ end
246
+
247
+ # Create the database
248
+ def create!
249
+ bool = server.create_db(@name) rescue false
250
+ bool && true
251
+ end
252
+
253
+ # Delete and re create the database
254
+ def recreate!
255
+ delete!
256
+ create!
257
+ rescue RestClient::ResourceNotFound
258
+ ensure
259
+ create!
260
+ end
261
+
262
+ # Replicates via "pulling" from another database to this database. Makes no attempt to deal with conflicts.
263
+ def replicate_from other_db
264
+ raise ArgumentError, "must provide a CouchReset::Database" unless other_db.kind_of?(CouchRest::Database)
265
+ CouchRest.post "#{@host}/_replicate", :source => other_db.root, :target => name
266
+ end
267
+
268
+ # Replicates via "pushing" to another database. Makes no attempt to deal with conflicts.
269
+ def replicate_to other_db
270
+ raise ArgumentError, "must provide a CouchReset::Database" unless other_db.kind_of?(CouchRest::Database)
271
+ CouchRest.post "#{@host}/_replicate", :target => other_db.root, :source => name
272
+ end
273
+
274
+ # DELETE the database itself. This is not undoable and could be rather
275
+ # catastrophic. Use with care!
276
+ def delete!
277
+ CouchRest.delete @uri
278
+ end
279
+
280
+ private
281
+
282
+ def uri_for_attachment(doc, name)
283
+ if doc.is_a?(String)
284
+ puts "CouchRest::Database#fetch_attachment will eventually require a doc as the first argument, not a doc.id"
285
+ docid = doc
286
+ rev = nil
287
+ else
288
+ docid = doc['_id']
289
+ rev = doc['_rev']
290
+ end
291
+ docid = escape_docid(docid)
292
+ name = CGI.escape(name)
293
+ rev = "?rev=#{doc['_rev']}" if rev
294
+ "#{@root}/#{docid}/#{name}#{rev}"
295
+ end
296
+
297
+ def escape_docid id
298
+ /^_design\/(.*)/ =~ id ? "_design/#{CGI.escape($1)}" : CGI.escape(id)
299
+ end
300
+
301
+ def encode_attachments(attachments)
302
+ attachments.each do |k,v|
303
+ next if v['stub']
304
+ v['data'] = base64(v['data'])
305
+ end
306
+ attachments
307
+ end
308
+
309
+ def base64(data)
310
+ Base64.encode64(data).gsub(/\s/,'')
311
+ end
312
+ end
313
+ end
@@ -0,0 +1,89 @@
1
+ module CouchRest
2
+ class Design < Document
3
+ def view_by *keys
4
+ opts = keys.pop if keys.last.is_a?(Hash)
5
+ opts ||= {}
6
+ self['views'] ||= {}
7
+ method_name = "by_#{keys.join('_and_')}"
8
+
9
+ if opts[:map]
10
+ view = {}
11
+ view['map'] = opts.delete(:map)
12
+ if opts[:reduce]
13
+ view['reduce'] = opts.delete(:reduce)
14
+ opts[:reduce] = false
15
+ end
16
+ self['views'][method_name] = view
17
+ else
18
+ doc_keys = keys.collect{|k|"doc['#{k}']"} # this is where :require => 'doc.x == true' would show up
19
+ key_emit = doc_keys.length == 1 ? "#{doc_keys.first}" : "[#{doc_keys.join(', ')}]"
20
+ guards = opts.delete(:guards) || []
21
+ guards.concat doc_keys
22
+ map_function = <<-JAVASCRIPT
23
+ function(doc) {
24
+ if (#{guards.join(' && ')}) {
25
+ emit(#{key_emit}, null);
26
+ }
27
+ }
28
+ JAVASCRIPT
29
+ self['views'][method_name] = {
30
+ 'map' => map_function
31
+ }
32
+ end
33
+ self['views'][method_name]['couchrest-defaults'] = opts unless opts.empty?
34
+ method_name
35
+ end
36
+
37
+ # Dispatches to any named view.
38
+ def view view_name, query={}, &block
39
+ view_name = view_name.to_s
40
+ view_slug = "#{name}/#{view_name}"
41
+ defaults = (self['views'][view_name] && self['views'][view_name]["couchrest-defaults"]) || {}
42
+ fetch_view(view_slug, defaults.merge(query), &block)
43
+ end
44
+
45
+ def name
46
+ id.sub('_design/','') if id
47
+ end
48
+
49
+ def name= newname
50
+ self['_id'] = "_design/#{newname}"
51
+ end
52
+
53
+ def save
54
+ raise ArgumentError, "_design docs require a name" unless name && name.length > 0
55
+ super
56
+ end
57
+
58
+ private
59
+
60
+ # returns stored defaults if the there is a view named this in the design doc
61
+ def has_view?(view)
62
+ view = view.to_s
63
+ self['views'][view] &&
64
+ (self['views'][view]["couchrest-defaults"]||{})
65
+ end
66
+
67
+ # def fetch_view_with_docs name, opts, raw=false, &block
68
+ # if raw
69
+ # fetch_view name, opts, &block
70
+ # else
71
+ # begin
72
+ # view = fetch_view name, opts.merge({:include_docs => true}), &block
73
+ # view['rows'].collect{|r|new(r['doc'])} if view['rows']
74
+ # rescue
75
+ # # fallback for old versions of couchdb that don't
76
+ # # have include_docs support
77
+ # view = fetch_view name, opts, &block
78
+ # view['rows'].collect{|r|new(database.get(r['id']))} if view['rows']
79
+ # end
80
+ # end
81
+ # end
82
+
83
+ def fetch_view view_name, opts, &block
84
+ database.view(view_name, opts, &block)
85
+ end
86
+
87
+ end
88
+
89
+ end