Merge branch 'master' of github.com:dpavlin/angular-mojolicious
[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 my $slot_regex = '(\d+)\s*mjesta';
234
235 get '/reservations/get/(*url)' => sub {
236         my $self = shift;
237
238         my $text = $client->get( 'http://' . $self->param('url') )->res->body;
239         warn "# get ", $self->param('url'), dump($text);
240
241         $text = decode( 'utf-8', $text );
242         $text =~ s{\\,}{,}gs;
243         $text =~ s{\\n}{ }gs;
244
245         my $c = iCal::Parser->new->parse_strings( $text );
246
247 #       warn "# iCal::Parser = ",dump($c);
248
249         my $ical = {
250                 cal => $c->{cals}->[0], # FIXME assume single calendar
251         };
252
253         my $e = $c->{events};
254         my @events;
255
256         foreach my $yyyy ( sort keys %$e ) {
257                 foreach my $mm ( sort keys %{ $e->{$yyyy} } ) {
258                         foreach my $dd ( sort keys %{ $e->{$yyyy}->{$mm} } ) {
259                                 push @events, values %{ $e->{$yyyy}->{$mm}->{$dd} };
260                         }
261                 }
262         }
263
264         @events = map {
265                 foreach my $check_slot ( qw(
266                         DESCRIPTION
267                         LOCATION
268                         STATUS
269                         SUMMARY
270                 )) {
271                         next unless exists $_->{$check_slot};
272                         $_->{slots} = $1 if $_->{$check_slot} =~ m/$slot_regex/is;
273                 }
274                 $_->{slots} ||= $1 if $ical->{cal}->{'X-WR-CALDESC'} =~ m/$slot_regex/s;
275                 $_;
276         } @events;
277
278         $ical->{events} = [ sort {
279                                         $a->{DTSTART} cmp $b->{DTSTART}
280         } @events ];
281
282         _render_jsonp( $self, $ical );
283 };
284
285 get '/reservations/events/:view_name' => sub {
286         my $self = shift;
287
288         my $view = _couchdb_get('/reservations/_design/events/_view/' . $self->param('view_name') . '?group=true');
289         my $hash;
290
291         if ( exists $view->{error} ) {
292                 _couchdb_put "/reservations/_design/events", {
293                         _id => '_design/events',
294                         language => 'javascript',
295                         views => {
296                                 submited => {
297                                         map    => q|
298                                                 function(doc) {
299                                                         if ( doc.event && doc.event.UID ) emit(doc.event.UID, 1)
300                                                 }
301                                         |,
302                                         reduce => q|_sum|,
303                                 }
304                         }
305                 };
306         }
307
308         _render_jsonp( $self, {} ) unless ref $view->{rows} eq 'ARRAY';
309
310         foreach my $row ( @{ $view->{rows} } ) {
311                 $hash->{ $row->{key} } = $row->{value};
312         }
313
314         $hash ||= {};
315
316         _render_jsonp( $self, $hash );
317 };
318
319 get '/_utils/script/(*url)' => sub { $_[0]->proxy_to( "$couchdb/_utils/script/" . $_[0]->param('url') , with_query_params => 1 ) };
320
321 app->start;
322 __DATA__
323
324 @@ index.html.ep
325 % layout 'funky';
326 Yea baby!
327
328 @@ layouts/funky.html.ep
329 <!doctype html><html>
330     <head><title>Funky!</title></head>
331     <body><%== content %></body>
332 </html>
333
334 @@ layouts/angular.html.ep
335 <!DOCTYPE HTML>
336 <html xmlns:ng="http://angularjs.org">
337   <head>
338    <meta charset="utf-8">
339 % my $ANGULAR_JS = $ENV{ANGULAR_JS} || ( -e 'public/angular/build/angular.js' ? '/angular/build/angular.js' : '/angular/src/angular-bootstrap.js' );
340     <script type="text/javascript"
341          src="<%== $ANGULAR_JS %>" ng:autobind></script>
342   </head>
343   <body><%== content %></body>
344 </html>