/[Frey]/trunk/lib/Frey/CouchAPI.pm
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Contents of /trunk/lib/Frey/CouchAPI.pm

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1062 - (show annotations)
Mon Apr 27 16:23:52 2009 UTC (15 years ago) by dpavlin
File size: 7848 byte(s)
added startkey_docid which is really simple for us since we have same key and id and report correct number of total_rows
1 package Frey::CouchAPI;
2
3 =head1 DESCRIPTION
4
5 This is REST wrapper using following L<Mojo> implement Apache's CouchDB API
6
7
8 L<Mojo::URL>
9
10 L<Mojo::Transaction>
11
12
13 =head1 Supported HTTP API
14
15 =cut
16
17 use warnings;
18 use strict;
19
20 use JSON;
21 use Data::Dump qw/dump/;
22 use URI::Escape;
23 use File::Path qw(make_path remove_tree);
24 use Storable;
25
26 our $VERSION = '0.3';
27 $VERSION .= " (Frey $Frey::VERSION)" if $Frey::VERSION;
28
29 our $debug = $Frey::debug || 0;
30
31 sub rewrite_urls {
32 my ( $self, $tx ) = @_;
33 if ( $tx->req->url->path =~ m{/_utils/} ) {
34 my $path = $tx->req->url->path;
35 $path =~ s{(/_utils)/?$}{$1/index.html}; # poor man's DirectoryIndex
36 $path =~ s{/_utils}{/static/futon};
37 $tx->req->url->path( $path );
38 my $url = $tx->req->url->to_string;
39 my $old = $url;
40 $url = $tx->req->url->to_string;
41 warn "# rewrite $old -> $url\n";
42 }
43 }
44
45 our $config = {
46 database => {
47 path => '/data/webpac2/var/row',
48 name_glob => '/*/*',
49 }
50 };
51
52 my $regex_dbs = '[a-z][a-z0-9_\$\(\)\+\-/]+';
53
54 our $json = {};
55 our $status;
56
57 sub ok {
58 $json = { ok => JSON::true };
59 $status = 200;
60 warn "ok from ",join(' ',caller),$/;
61 }
62
63 sub file_rev { (stat($_[0]))[9] } # mtime
64
65 sub data_from_tx {
66 my $tx = shift;
67 my $data = $tx->req->content->file->slurp;
68 $data = JSON->new->allow_nonref->decode( $data );
69 warn "## data ",dump( $data );
70 return $data;
71 }
72
73 sub dispatch {
74 my ($self,$tx) = @_;
75
76 $status = 500; # Internal Error
77
78 my $url = $tx->req->url->to_string;
79 $url = uri_unescape( $url );
80 my $method = $tx->req->method;
81 my $path = $config->{database}->{path};
82
83 if ( $url eq '/' ) {
84 $json = {
85 couchdb => "Welcome",
86 version => "CouchAPI $VERSION",
87 };
88 $status = 200;
89 } elsif ( $url eq '/_all_dbs' ) {
90 $json = [
91 map {
92 s{^\Q$path\E/*}{};
93 $_;
94 } glob $path . $config->{database}->{name_glob}
95 ];
96 $status = 200;
97 } elsif ( $url =~ m{^/_config/?(.+)} ) {
98
99 $json = $config;
100
101 if ( $method eq 'PUT' ) {
102
103 my $part = $1;
104 my $data = data_from_tx( $tx );
105 warn "## part $part = $data\n";
106
107 $part =~ s!/!'}->{'!;
108
109 # poor man's transaction :-)
110 my $code = "\$config->{'$part'} = \$data;";
111 eval $code;
112 if ( $@ ) {
113 warn "ERROR: $code -> $@";
114 $status = 500;
115 } else {
116 $status = 200;
117 }
118
119 warn "# config after $code is ",dump( $config ),$/;
120
121 } elsif ( $method eq 'GET' ) {
122 $status = 200;
123 } else {
124 $status = 501;
125 }
126
127 =head2 Database
128
129 L<http://wiki.apache.org/couchdb/HTTP_database_API> except compaction
130
131 =cut
132
133 } elsif (
134 $url =~ m{($regex_dbs)/$}
135 # DELETE doesn't have trailing slash
136 || $method eq 'DELETE' && $url =~ m{($regex_dbs)$}
137 ) {
138
139 my $database = $1;
140
141 my $dir = "$path/$database";
142
143 if ( $method eq 'GET' ) {
144 $json = database_get( $database );
145 } elsif ( $method eq 'DELETE' ) {
146 if ( ! -e $dir ) {
147 $status = 404;
148 } else {
149 remove_tree($dir);
150 if ( ! -d $dir ) {
151 ok;
152 } else {
153 $status = 500;
154 }
155 }
156 } elsif ( $method eq 'PUT' ) {
157 if ( -e $dir ) {
158 $status = 412;
159 } else {
160 make_path($dir);
161 if ( -e $path ) {
162 ok;
163 $status = 201;
164 } else {
165 $status = 500;
166 }
167 }
168 }
169
170 warn "## database $database $status ",dump( $json );
171
172 } elsif ( $url =~ m{($regex_dbs)/([^?]+)\??(.+)?$} ) {
173 my ($database,$id,$args) = ($1,$2,$3);
174
175 =head2 Document
176
177 L<http://wiki.apache.org/couchdb/HTTP_Document_API>
178
179 =cut
180
181 my $arg;
182 if ( $args ) {
183 foreach my $a ( split(/[&;]/,$args) ) {
184 my ($n,$v) = split(/=/,$a);
185 $v =~ s{(["'])(.+)\1}{$2};
186 $arg->{$n} = $v;
187 }
188 }
189
190 warn "ERROR: path $path doesn't exist\n" unless -e $path;
191
192 my $p = "$path/$database/$id";
193 warn "## database: $database id: $id -> $p ", dump( $arg ),"\n";
194
195
196 if ( $id =~ m{_all_docs(\w*)$} ) {
197
198 my $by = $1;
199 my $offset = 0;
200 my $startkey = delete $arg->{startkey};
201 $startkey ||= delete $arg->{startkey_docid}; # XXX key == id
202 my $endkey = delete $arg->{endkey};
203 my $limit = delete $arg->{limit};
204 my $skip = delete $arg->{skip};
205 my $total_rows = 0;
206 my $collected_rows = 0;
207
208 my @docs = grep { length $_ } map {
209
210 $total_rows++;
211
212 if ( $limit > 0 && $collected_rows == $limit ) {
213 '';
214 } else {
215
216 s{^$path/$database/}{};
217
218 if ( defined $endkey && $_ gt $endkey ) {
219 '';
220 } elsif ( $startkey ) {
221 if ( $_ ge $startkey ) {
222 $collected_rows++;
223 $_;
224 } else {
225 $offset++;
226 '';
227 }
228 } else {
229 $collected_rows++;
230 $_;
231 }
232 }
233
234 } glob( "$path/$database/*" );
235
236 $offset += $skip if $skip;
237
238 warn "## docs $startkey -> $endkey limit $limit ", dump( @docs ); # if $debug;
239
240 $json = {
241 total_rows => $total_rows,
242 offset => $offset,
243 rows => [],
244 };
245
246 my $rows;
247 my @ids;
248
249 foreach my $id ( @docs ) {
250 warn "++ $id\n" if $debug;
251 my $p = "$path/$database/$id";
252 my $data = eval { Storable::retrieve( $p ) };
253 if ( $@ ) {
254 warn "ERROR: $p | $@\n";
255 next;
256 }
257 push @ids, $id;
258 $rows->{$id} = {
259 id => $id,
260 key => $id,
261 value => {
262 rev => file_rev $p,
263 }
264 };
265 }
266
267 my $descending = delete $arg->{descending};
268 my @sorted = sort @ids;
269
270 warn "creating rows in ", $descending ? "descending" : "", " order\n";
271
272 foreach my $id ( $descending ? reverse @sorted : @sorted ) {
273 warn ">> $id ", $descending ? 'desc' : 'asc', "\n" if $debug;
274 push @{ $json->{rows} }, $rows->{$id};
275 }
276
277 $status = 200;
278
279 } elsif ( $method eq 'PUT' ) {
280
281 warn "## ",dump( $tx->req ) if $debug;
282
283 my $data = $tx->req->content->file->slurp;
284
285 my $db_path = "$path/$database";
286 make_path $db_path unless -e $db_path;
287
288 Storable::store( from_json($data), $p );
289 my $rev = file_rev $p;
290 warn "store $p $rev size ", -s $p, " bytes | $data\n";
291
292 $status = 201; # Created
293 $json = {
294 id => $id,
295 ok => JSON::true,
296 rev => $rev,
297 };
298
299 } elsif ( $method eq 'GET' ) {
300 if ( ! -e $p ) {
301 $status = 404;
302 } else {
303 warn "retrive $p ", -s $p, " bytes\n";
304 $json = Storable::retrieve( $p );
305 if ( delete $arg->{revs_info} ) {
306 my $rev = file_rev $p;
307 $json->{_rev} = $rev;
308 $json->{_revs_info} = [
309 { rev => $rev, status => 'available' }
310 ];
311 }
312 $status = 200;
313
314 }
315 } elsif ( $method eq 'DELETE' ) {
316 if ( -e $p ) {
317 unlink $p && ok || { $status = 500 };
318 } else {
319 $status = 404;
320 }
321 } elsif ( $method eq 'POST' ) {
322 my $data = data_from_tx( $tx );
323
324 # FIXME implement real view server and return 200
325 $json = { total_rows => 0, offset => 0 };
326 $status = 202;
327
328 } else {
329 $status = 501;
330 }
331
332 if ( keys %$arg ) {
333 warn "WARNING: arg left from $url = ",dump( $arg ),$/;
334 $status = 501;
335 }
336
337 }
338
339 $json = { error => 'not_found', reason => 'Missing' } if $status == 404;
340
341 if ( $method =~ m{(DELETE|PUT)} ) {
342 $tx->res->headers->add_line( 'Location' => $tx->req->url->to_abs );
343 }
344
345 $tx->res->code( $status );
346 $tx->res->headers->content_type( 'text/plain;charset=utf-8' );
347 my $body = to_json $json;
348 $tx->res->body( $body );
349 $tx->res->headers->add_line( 'Cache-Control' => 'must-revalidate' );
350 $tx->res->headers->add_line( 'Server' => "Frey::CouchAPI/$VERSION" );
351
352 print "$method $url $status\n$body\n";
353
354 warn "## headers ", $tx->res->headers->to_string;
355
356 return $tx;
357
358 }
359
360 sub database_get {
361 my ($db_name) = @_;
362 my $path = $config->{database}->{path} || die;
363 warn "# collecting docs from $path/$db_name/*\n";
364 my @docs = glob "$path/$db_name/*";
365 my $json = {
366 db_name => $db_name,
367 doc_count => $#docs + 1,
368 doc_del_count => 0,
369 update_seq => 0,
370 purge_seq => 0,
371 capacity_running => JSON::false,
372 disk_size => 0,
373 instance_start_time => time(),
374 };
375
376 warn "## calculating disk_size\n";
377 $json->{disk_size} += -s $_ foreach @docs;
378 $status = 200;
379 return $json;
380 }
381
382 1;
383 __END__
384
385 =head1 SEE ALSO
386
387 L<http://wiki.apache.org/couchdb/Reference>
388

  ViewVC Help
Powered by ViewVC 1.1.26