correctly extract regitration id from _id
[angular-drzb] / 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 push @{app->static->paths}, 'app'; # default angular-seed app directory
14
15
16 my $couchdb = $ENV{COUCHDB} || 'http://localhost:5984';
17 my $client = Mojo::UserAgent->new;
18
19 sub _couchdb_put {
20         my ( $url, $data ) = @_;
21
22         $data->{'$entity'} = $1 if $url =~ m{/(\w+)\.\d+/$/};
23
24         my $json = Mojo::JSON->new->encode( $data );
25
26         my $rev;
27
28         warn "# _couchdb_put $url = $json";
29         return $client->put( "$couchdb/$url" => $json)->res->json;
30 }
31
32 sub _couchdb_get {
33         my ( $url ) = @_;
34         my $return = $client->get( "$couchdb/$url" )->res->json;
35         warn "# _couchdb_get $url = ",dump($return);
36         return $return;
37 }
38
39
40 our $id2nr;
41
42
43 sub _render_jsonp {
44         my ( $self, $json ) = @_;
45 #warn "## _render_json ",dump($json);
46         my $data = $self->render( json => $json, partial => 1 );
47 warn "## _render_json $data";
48         if ( my $callback = $self->param('callback') ) {
49                 $data = "$callback($data)";
50         }
51         $self->render( data => $data, format => 'js' );
52 }
53
54 #get '/' => 'index';
55
56
57 get '/data/' => sub {
58         my $self = shift;
59         _render_jsonp( $self, _couchdb_get('/_all_dbs') );
60 };
61
62 get '/data/:database' => sub {
63         my $self = shift;
64         my $database = $self->param('database');
65
66         my $list_databases = { name => $database };
67
68         my $counts = _couchdb_get("/$database/_design/entity/_view/counts?group=true");
69         if ( exists $counts->{error} ) {
70                 warn "creating CouchDB view because of ", dump($counts);
71                 _couchdb_put "/$database/_design/entity", {
72                         _id => '_design/entity',
73                         language => 'javascript',
74                         views => {
75                                 counts => {
76                                         map    => q| function(doc) { emit(doc._id.split('.')[0],1); } |,
77                                         reduce => q| function(keys,values,rereduce) { return sum(values); } |,
78                                 }
79                         }
80                 };
81                 $counts = _couchdb_get("/$database/_design/entity/_view/counts?group=true")
82                 || die "give up!";
83         }
84
85         warn "# counts ",dump($counts);
86
87         foreach my $row ( @{ $counts->{rows} } ) {
88                 my $n = $row->{value};
89                 $list_databases->{entities}->{ $row->{key} } = $n;
90                 $list_databases->{document_counts} += $n;
91         }
92         warn dump($list_databases);
93         _render_jsonp( $self,  $list_databases );
94 };
95
96 get '/data/:database/:entity' => sub {
97         my $self = shift;
98
99         my $database = $self->param('database');
100         my $entity   = $self->param('entity');
101
102         my $endkey = $entity;
103         $endkey++;
104
105         my $counts = _couchdb_get qq|/$database/_all_docs?startkey="$entity";endkey="$endkey";include_docs=true|;
106         warn "# counts ",dump($counts);
107
108         _render_jsonp( $self, [ map { $_->{doc} } @{ $counts->{rows} } ] )
109 };
110
111 get '/data/:database/:entity/:id' => sub {
112     my $self = shift;
113
114         my $database = $self->param('database');
115         my $entity   = $self->param('entity');
116         my $id       = $self->param('id');
117
118         _render_jsonp( $self, _couchdb_get( "/$database/$entity.$id" ) );
119 };
120
121 any [ 'post' ] => '/data/:database/:entity' => sub {
122         my $self = shift;
123         my $database = $self->param('database');
124         my $entity   = $self->param('entity');
125         my $json = $self->req->json;
126         my $id = $1 if $json->{'_id'} =~ m/^\Q$entity\E\.(.+)$/;
127         $id ||= new_uuid;
128         warn "## $database $entity $id body ",dump($self->req->body, $json);
129
130         my $new = _couchdb_put "/$database/$entity.$id" => $json;
131         warn "new: ",dump($new);
132         if ( $new->{ok} ) {
133                 $json->{'_'.$_} = $new->{$_} foreach ( 'rev','id' );
134         } else {
135                 warn "ERROR: ",dump($new);
136                 $json->{error} = $new;
137         }
138
139         _render_jsonp( $self,  $json );
140 };
141
142
143 #get '/' => sub { shift->redirect_to('/app/') };
144
145 # CouchDB proxy for _design _view
146
147 get '/:database/_design/:design/_view/:view' => sub {
148         my $self = shift;
149         my $url = join('/', $self->param('database'),'_design',$self->param('design'),'_view',$self->param('view') );
150         my $param = $self->req->url->query->clone->remove('callback')->to_string;
151         $url .= '?' . $param if $param;
152         warn "CouchDB proxy $url";
153         _render_jsonp( $self, _couchdb_get($url));
154 };
155
156 # static JSON files from public/json/database/entity/json
157
158 get '/json' => sub {
159         _render_jsonp( shift, [ map { s{public/json/}{}; $_ } glob 'public/json/*' ] );
160 };
161
162 get '/json/:database' => sub {
163         my $self = shift;
164         my $database = $self->param('database');
165
166         my $status = {
167                 document_counts => 0,
168                 name => $database,
169         };
170
171         foreach my $path ( glob "public/json/$database/*" ) {
172                 my @entities = glob "$path/*";
173                 $path =~ s{public/json/$database/}{};
174                 $status->{entities}->{$path} = scalar @entities;
175                 $status->{document_counts}++;
176         }
177
178         _render_jsonp( $self, $status );
179 };
180
181 get '/json/:database/:entity' => sub {
182         my $self = shift;
183
184         my $database = $self->param('database');
185         my $entity   = $self->param('entity');
186
187         my $path = "public/json/$database/$entity";
188         die "$path: $!" unless -d $path;
189
190         my $docs;
191         foreach my $path ( sort glob "$path/*" ) {
192                 open(my $fh, '<', $path) || die $!;
193                 local $/ = undef;
194                 my $str = <$fh>;
195                 warn "# $path $str";
196                 my $data = Mojo::JSON->new->decode( $str );
197                 $data->{_key} = $1 if $path =~ m{/([^/]+$)};
198                 push @$docs, $data;
199         }
200
201         _render_jsonp( $self, $docs )
202 };
203
204 get '/_utils/script/(*url)' => sub { $_[0]->proxy_to( "$couchdb/_utils/script/" . $_[0]->param('url') , with_query_params => 1 ) };
205
206 app->start;
207 __DATA__
208
209 @@ index.html.ep
210 % layout 'funky';
211 Yea baby!
212
213 @@ layouts/funky.html.ep
214 <!doctype html><html>
215     <head><title>Funky!</title></head>
216     <body><%== content %></body>
217 </html>
218
219 @@ layouts/angular.html.ep
220 <!DOCTYPE HTML>
221 <html xmlns:ng="http://angularjs.org">
222   <head>
223    <meta charset="utf-8">
224 % my $ANGULAR_JS = $ENV{ANGULAR_JS} || ( -e 'public/angular/build/angular.js' ? '/angular/build/angular.js' : '/angular/src/angular-bootstrap.js' );
225     <script type="text/javascript"
226          src="<%== $ANGULAR_JS %>" ng:autobind></script>
227   </head>
228   <body><%== content %></body>
229 </html>