proxy script requests to CouchDB
[angular-mojolicious.git] / angular-server.pl
1 #!/usr/bin/env perl
2
3 use lib 'common/mojo/lib';
4
5 use Mojolicious::Lite;
6 use Data::Dump qw(dump);
7 use Time::HiRes;
8 use Clone qw(clone);
9 use Mojo::UserAgent;
10
11 sub new_uuid { Time::HiRes::time * 100000 }
12
13 # based on
14 # http://docs.getangular.com/REST.Basic
15 # http://angular.getangular.com/data
16
17 my $couchdb = $ENV{COUCHDB} || 'http://localhost:5984';
18 my $client = Mojo::UserAgent->new;
19
20 sub _couchdb_put {
21         my ( $url, $data ) = @_;
22
23         $data->{'$entity'} = $1 if $url =~ m{/(\w+)\.\d+/$/};
24
25         my $json = Mojo::JSON->new->encode( $data );
26
27         my $rev;
28
29         warn "# _couchdb_put $url = $json";
30         return $client->put( "$couchdb/$url" => $json)->res->json;
31 }
32
33 sub _couchdb_get {
34         my ( $url ) = @_;
35         my $return = $client->get( "$couchdb/$url" )->res->json;
36         warn "# _couchdb_get $url = ",dump($return);
37         return $return;
38 }
39
40
41 our $id2nr;
42
43
44 sub _render_jsonp {
45         my ( $self, $json ) = @_;
46 #warn "## _render_json ",dump($json);
47         my $data = $self->render( json => $json, partial => 1 );
48 warn "## _render_json $data";
49         if ( my $callback = $self->param('callback') ) {
50                 $data = "$callback($data)";
51         }
52         $self->render( data => $data, format => 'js' );
53 }
54
55 #get '/' => 'index';
56
57
58 get '/data/' => sub {
59         my $self = shift;
60         _render_jsonp( $self, _couchdb_get('/_all_dbs') );
61 };
62
63 get '/data/:database' => sub {
64         my $self = shift;
65         my $database = $self->param('database');
66
67         my $list_databases = { name => $database };
68
69         my $counts = _couchdb_get("/$database/_design/entity/_view/counts?group=true");
70         if ( exists $counts->{error} ) {
71                 warn "creating CouchDB view because of ", dump($counts);
72                 _couchdb_put "/$database/_design/entity", {
73                         _id => '_design/entity',
74                         language => 'javascript',
75                         views => {
76                                 counts => {
77                                         map    => q| function(doc) { emit(doc._id.split('.')[0],1); } |,
78                                         reduce => q| function(keys,values,rereduce) { return sum(values); } |,
79                                 }
80                         }
81                 };
82                 $counts = _couchdb_get("/$database/_design/entity/_view/counts?group=true")
83                 || die "give up!";
84         }
85
86         warn "# counts ",dump($counts);
87
88         foreach my $row ( @{ $counts->{rows} } ) {
89                 my $n = $row->{value};
90                 $list_databases->{entities}->{ $row->{key} } = $n;
91                 $list_databases->{document_counts} += $n;
92         }
93         warn dump($list_databases);
94         _render_jsonp( $self,  $list_databases );
95 };
96
97 get '/data/:database/:entity' => sub {
98         my $self = shift;
99
100         my $database = $self->param('database');
101         my $entity   = $self->param('entity');
102
103         my $endkey = $entity;
104         $endkey++;
105
106         my $counts = _couchdb_get qq|/$database/_all_docs?startkey="$entity";endkey="$endkey";include_docs=true|;
107         warn "# counts ",dump($counts);
108
109         _render_jsonp( $self, [ map { $_->{doc} } @{ $counts->{rows} } ] )
110 };
111
112 get '/data/:database/:entity/:id' => sub {
113     my $self = shift;
114
115         my $database = $self->param('database');
116         my $entity   = $self->param('entity');
117         my $id       = $self->param('id');
118
119         _render_jsonp( $self, _couchdb_get( "/$database/$entity.$id" ) );
120 };
121
122 any [ 'post' ] => '/data/:database/:entity' => sub {
123         my $self = shift;
124         my $database = $self->param('database');
125         my $entity   = $self->param('entity');
126         my $json = $self->req->json;
127         my $id = $json->{'$id'} # XXX we don't get it back from angular.js
128                 || new_uuid;
129         warn "## $database $entity $id body ",dump($self->req->body, $json);
130
131         $json->{'$id'} ||= $id; # make sure $id is in there
132
133         my $new = _couchdb_put "/$database/$entity.$id" => $json;
134         warn "new: ",dump($new);
135         if ( $new->{ok} ) {
136                 $json->{'_'.$_} = $new->{$_} foreach ( 'rev','id' );
137         } else {
138                 warn "ERROR: ",dump($new);
139                 $json->{error} = $new;
140         }
141
142         _render_jsonp( $self,  $json );
143 };
144
145
146 get '/' => sub { shift->redirect_to('/Cookbook') };
147
148 get '/Cookbook' => 'Cookbook';
149 get '/Cookbook/:example' => sub {
150         my $self = shift;
151         $self->render( "Cookbook/" . $self->param('example'), layout => 'angular' );
152 };
153
154 get '/conference/:page' => sub {
155         my $self = shift;
156         $self->render( "conference/" . $self->param('page'), layout => 'angular' );
157 };
158
159 # /app/
160
161 get '/app/:database/angular.js' => sub {
162         my $self = shift;
163         my $ANGULAR_JS = $ENV{ANGULAR_JS} || ( -e 'public/angular/build/angular.js' ? '/angular/build/angular.js' : '/angular/src/angular-bootstrap.js' );
164         warn "# $ANGULAR_JS";
165         $self->render_static( $ANGULAR_JS );
166 };
167
168 # CouchDB proxy for _design _view
169
170 get '/:database/_design/:design/_view/:view' => sub {
171         my $self = shift;
172         my $url = join('/', $self->param('database'),'_design',$self->param('design'),'_view',$self->param('view') );
173         my $param = $self->req->url->query->clone->remove('callback')->to_string;
174         $url .= '?' . $param if $param;
175         warn "CouchDB proxy $url";
176         _render_jsonp( $self, _couchdb_get($url));
177 };
178
179 # static JSON files from public/json/database/entity/json
180
181 get '/json' => sub {
182         _render_jsonp( shift, [ map { s{public/json/}{}; $_ } glob 'public/json/*' ] );
183 };
184
185 get '/json/:database' => sub {
186         my $self = shift;
187         my $database = $self->param('database');
188
189         my $status = {
190                 document_counts => 0,
191                 name => $database,
192         };
193
194         foreach my $path ( glob "public/json/$database/*" ) {
195                 my @entities = glob "$path/*";
196                 $path =~ s{public/json/$database/}{};
197                 $status->{entities}->{$path} = scalar @entities;
198                 $status->{document_counts}++;
199         }
200
201         _render_jsonp( $self, $status );
202 };
203
204 get '/json/:database/:entity' => sub {
205         my $self = shift;
206
207         my $database = $self->param('database');
208         my $entity   = $self->param('entity');
209
210         my $path = "public/json/$database/$entity";
211         die "$path: $!" unless -d $path;
212
213         my $docs;
214         foreach my $path ( sort glob "$path/*" ) {
215                 open(my $fh, '<', $path) || die $!;
216                 local $/ = undef;
217                 my $str = <$fh>;
218                 warn "# $path $str";
219                 my $data = Mojo::JSON->new->decode( $str );
220                 $data->{_key} = $1 if $path =~ m{/([^/]+$)};
221                 push @$docs, $data;
222         }
223
224         _render_jsonp( $self, $docs )
225 };
226
227 # app/resevations
228 use Encode;
229 use iCal::Parser;
230
231 plugin 'proxy';
232
233 get '/reservations/get/(*url)' => sub {
234         my $self = shift;
235
236         my $text = $client->get( 'http://' . $self->param('url') )->res->body;
237         warn "# get ", $self->param('url'), dump($text);
238
239         $text = decode( 'utf-8', $text );
240         $text =~ s{\\,}{,}gs;
241         $text =~ s{\\n}{ }gs;
242
243         my $c = iCal::Parser->new->parse_strings( $text );
244
245 #       warn "# iCal::Parser = ",dump($c);
246
247         my $ical = {
248                 cal => $c->{cals}->[0], # FIXME assume single calendar
249         };
250
251         my $e = $c->{events};
252         my @events;
253
254         foreach my $yyyy ( sort keys %$e ) {
255                 foreach my $mm ( sort keys %{ $e->{$yyyy} } ) {
256                         foreach my $dd ( sort keys %{ $e->{$yyyy}->{$mm} } ) {
257                                 push @events, values %{ $e->{$yyyy}->{$mm}->{$dd} };
258                         }
259                 }
260         }
261
262         @events = map {
263                 foreach my $check_slot ( qw(
264                         LOCATION
265                         STATUS
266                         SUMMARY
267                 )) {
268                         next unless exists $_->{$check_slot};
269                         $_->{slots} = $1 if $_->{$check_slot} =~ m/(\d+)\s*mjesta/s;
270                 }
271                 $_;
272         } @events;
273
274         $ical->{events} = [ sort {
275                                         $a->{DTSTART} cmp $b->{DTSTART}
276         } @events ];
277
278         _render_jsonp( $self, $ical );
279 };
280
281 get '/reservations/events/:view_name' => sub {
282         my $self = shift;
283
284         my $view = _couchdb_get('/reservations/_design/events/_view/' . $self->param('view_name') . '?group=true');
285         my $hash;
286
287         if ( exists $view->{error} ) {
288                 _couchdb_put "/reservations/_design/events", {
289                         _id => '_design/events',
290                         language => 'javascript',
291                         views => {
292                                 submited => {
293                                         map    => q|
294                                                 function(doc) {
295                                                         if ( doc.event && doc.event.UID ) emit(doc.event.UID, 1)
296                                                 }
297                                         |,
298                                         reduce => q|_sum|,
299                                 }
300                         }
301                 };
302         }
303
304         _render_jsonp( $self, {} ) unless ref $view->{rows} eq 'ARRAY';
305
306         foreach my $row ( @{ $view->{rows} } ) {
307                 $hash->{ $row->{key} } = $row->{value};
308         }
309
310         _render_jsonp( $self, $hash );
311 };
312
313 get '/_utils/script/(*url)' => sub { $_[0]->proxy_to( "$couchdb/_utils/script/" . $_[0]->param('url') , with_query_params => 1 ) };
314
315 app->start;
316 __DATA__
317
318 @@ index.html.ep
319 % layout 'funky';
320 Yea baby!
321
322 @@ layouts/funky.html.ep
323 <!doctype html><html>
324     <head><title>Funky!</title></head>
325     <body><%== content %></body>
326 </html>
327
328 @@ layouts/angular.html.ep
329 <!DOCTYPE HTML>
330 <html xmlns:ng="http://angularjs.org">
331   <head>
332    <meta charset="utf-8">
333 % my $ANGULAR_JS = $ENV{ANGULAR_JS} || ( -e 'public/angular/build/angular.js' ? '/angular/build/angular.js' : '/angular/src/angular-bootstrap.js' );
334     <script type="text/javascript"
335          src="<%== $ANGULAR_JS %>" ng:autobind></script>
336   </head>
337   <body><%== content %></body>
338 </html>