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