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