added backwards compatibile urls
[angular-drzb] / angular-server.pl
1 #!/usr/bin/env perl
2
3 use Mojolicious::Lite;
4 use Data::Dump qw(dump);
5 use Time::HiRes;
6 use Clone qw(clone);
7 use Mojo::UserAgent;
8
9 sub new_uuid { Time::HiRes::time * 100000 }
10
11 #push @{app->static->paths}, 'app'; # default angular-seed app directory
12
13
14 my $couchdb = $ENV{COUCHDB} || 'http://localhost:5984';
15 my $client = Mojo::UserAgent->new;
16
17 sub _couchdb_put {
18         my ( $url, $data ) = @_;
19
20         $data->{'$entity'} = $1 if $url =~ m{/(\w+)\.\d+/$/};
21
22         my $json = Mojo::JSON->new->encode( $data );
23
24         my $rev;
25
26         warn "# _couchdb_put $url = $json";
27         return $client->put( "$couchdb/$url" => $json)->res->json;
28 }
29
30 sub _couchdb_get {
31         my ( $url ) = @_;
32         my $return = $client->get( "$couchdb/$url" )->res->json;
33 #       warn "# _couchdb_get $url = ",dump($return);
34         return $return;
35 }
36
37
38 our $id2nr;
39
40
41 sub _render_jsonp {
42         my ( $self, $json ) = @_;
43 #warn "## _render_json ",dump($json);
44         my $data = $self->render( json => $json, partial => 1 );
45 #warn "## _render_json $data";
46         if ( my $callback = $self->param('callback') ) {
47                 $data = "$callback($data)";
48         }
49         $self->render( data => $data, format => 'js' );
50 }
51
52 get '/' => sub {
53         my $self = shift;
54         $self->render_text("...");
55 };
56
57 # define languages
58
59 helper locale => sub {
60         my $self = shift;
61         my %locale = @_;
62         my $lang = $self->stash('lang');
63         $lang =~ s/-dev$//;
64         return $locale{ $lang } || "MISSING $lang $_[1]";
65 };
66
67 # short public URLs
68 get '/hr' => sub { shift->redirect_to('/lang/hr/drzb2013') };
69 get '/en' => sub { shift->redirect_to('/lang/en/drzb2013') };
70
71 get '/lang/:lang/:template' => sub {
72         my $self = shift;
73         $self->render( $self->stash('template') , lang => $self->stash('lang') );
74 };
75
76 get '/lang/:lang/partials/:template' => sub {
77         my $self = shift;
78         $self->render( 'partials/' . $self->stash('template') , lang => $self->stash('lang') );
79 };
80
81 get '/lang/:lang/.template' => sub {
82         my $self = shift;
83         $self->render( $self->stash('template') , lang => $self->stash('lang') );
84 };
85
86
87 get '/data/' => sub {
88         my $self = shift;
89         _render_jsonp( $self, _couchdb_get('/_all_dbs') );
90 };
91
92 get '/data/:database' => sub {
93         my $self = shift;
94         my $database = $self->param('database');
95
96         my $list_databases = { name => $database };
97
98         my $counts = _couchdb_get("/$database/_design/entity/_view/counts?group=true");
99         if ( exists $counts->{error} ) {
100                 warn "creating CouchDB view because of ", dump($counts);
101                 _couchdb_put "/$database/_design/entity", {
102                         _id => '_design/entity',
103                         language => 'javascript',
104                         views => {
105                                 counts => {
106                                         map    => q| function(doc) { emit(doc._id.split('.')[0],1); } |,
107                                         reduce => q| function(keys,values,rereduce) { return sum(values); } |,
108                                 }
109                         }
110                 };
111                 $counts = _couchdb_get("/$database/_design/entity/_view/counts?group=true")
112                 || die "give up!";
113         }
114
115         warn "# counts ",dump($counts);
116
117         foreach my $row ( @{ $counts->{rows} } ) {
118                 my $n = $row->{value};
119                 $list_databases->{entities}->{ $row->{key} } = $n;
120                 $list_databases->{document_counts} += $n;
121         }
122         warn dump($list_databases);
123         _render_jsonp( $self,  $list_databases );
124 };
125
126 get '/data/:database/:entity' => sub {
127         my $self = shift;
128
129         my $database = $self->param('database');
130         my $entity   = $self->param('entity');
131
132         my $endkey = $entity;
133         $endkey++;
134
135         my $counts = _couchdb_get qq|/$database/_all_docs?startkey="$entity";endkey="$endkey";include_docs=true|;
136 #       warn "# counts ",dump($counts);
137
138         _render_jsonp( $self, [ map { $_->{doc} } @{ $counts->{rows} } ] )
139 };
140
141 get '/data/:database/:entity/:id' => sub {
142     my $self = shift;
143
144         my $database = $self->param('database');
145         my $entity   = $self->param('entity');
146         my $id       = $self->param('id');
147
148         _render_jsonp( $self, _couchdb_get( "/$database/$entity.$id" ) );
149 };
150
151 any [ 'post' ] => '/data/:database/:entity' => sub {
152         my $self = shift;
153         my $database = $self->param('database');
154         my $entity   = $self->param('entity');
155         my $json = $self->req->json;
156         my $id;
157         if ( exists $json->{'id'} ) { # @id in resource
158                 $id = $json->{'id'};
159                 warn "EXISTING $id\n";
160         } else {
161                 $id = $json->{'id'} = new_uuid;
162                 $json->{entity} = $entity;
163                 warn "NEW $id\n";
164         }
165         warn "## $database $entity $id body ",dump($self->req->body, $json);
166
167         my $new = _couchdb_put "/$database/$entity.$id" => $json;
168         warn "new: ",dump($new);
169         if ( $new->{ok} ) {
170                 $json->{'_'.$_} = $new->{$_} foreach ( 'rev','id' );
171         } else {
172                 warn "ERROR: ",dump($new);
173                 $json->{error} = $new;
174         }
175
176         _render_jsonp( $self,  $json );
177 };
178
179
180 #get '/' => sub { shift->redirect_to('/app/') };
181
182 # CouchDB proxy for _design _view
183
184 get '/:database/_design/:design/_view/:view' => sub {
185         my $self = shift;
186         my $format = $self->param('format');
187         my $url = join('/', $self->param('database'),'_design',$self->param('design'),'_view',$self->param('view') );
188         if ( my $param = $self->req->url->query->clone->remove('callback')->remove('format')->to_string ) {
189                 $url .= '?' . $param
190         }
191         warn "CouchDB proxy $url";
192         my $json = _couchdb_get($url);
193
194         if ( exists $json->{error} ) {
195                 warn "creating CouchDB view because of ", dump($json);
196                 my $url = "/" . $self->param('database') . "/_design/registration";
197                 _couchdb_put $url, {
198                         _id => '_design/registration',
199                         language => 'javascript',
200                         views => {
201                                 organizations => {
202                                         map    => q| function(doc) {
203 if ( doc.user.organization != '' ) {
204   emit(doc.user.organization, 1);
205   if ( doc.user.persons ) {
206     doc.user.persons.forEach( function(person) {
207       emit(person.organization, 1);
208     });
209   }
210 }
211                                         }q|,
212                                         reduce => q| function(keys,values,rereduce) { return sum(values); } |,
213                                 }
214                         }
215                 };
216                 $json = _couchdb_get($url)
217                 || die "give up!";
218         }
219
220         if ( $format eq 'key_array' ) { # array of keys sorted by value
221                 $json->{rows} = [ map { $_->{key} } sort { $b->{value} <=> $a->{value} } @{ $json->{rows} } ];
222         }
223         _render_jsonp( $self, $json );
224 };
225
226 # http://showmetheco.de/articles/2010/10/adding-etag-caching-to-your-mojolicious-app.html
227
228 hook after_dispatch => sub {
229         my $self = shift;
230
231         return unless $self->req->method eq 'GET';
232
233         my $body = $self->res->body;
234         return unless defined $body;
235
236         return if $self->res->headers->header('ETag');
237
238         my $our_etag = Mojo::ByteStream->new($body)->md5_sum;
239         $self->res->headers->header('ETag' => $our_etag);
240
241         my $browser_etag = $self->req->headers->header('If-None-Match');
242         return unless $browser_etag && $browser_etag eq $our_etag;
243
244         $self->app->log->info("HTTP cache hit ", dump( $self->req->url->to_string ), $our_etag );
245
246         $self->res->code(304);
247         $self->res->body('');
248 };
249
250
251 app->start;
252