keep id and entity in json object
[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;
127         if ( exists $json->{'id'} ) { # @id in resource
128                 $id = $json->{'id'};
129                 warn "EXISTING $id\n";
130         } else {
131                 $id = $json->{'id'} = new_uuid;
132                 $json->{entity} = $entity;
133                 warn "NEW $id\n";
134         }
135         warn "## $database $entity $id body ",dump($self->req->body, $json);
136
137         my $new = _couchdb_put "/$database/$entity.$id" => $json;
138         warn "new: ",dump($new);
139         if ( $new->{ok} ) {
140                 $json->{'_'.$_} = $new->{$_} foreach ( 'rev','id' );
141         } else {
142                 warn "ERROR: ",dump($new);
143                 $json->{error} = $new;
144         }
145
146         _render_jsonp( $self,  $json );
147 };
148
149
150 #get '/' => sub { shift->redirect_to('/app/') };
151
152 # CouchDB proxy for _design _view
153
154 get '/:database/_design/:design/_view/:view' => sub {
155         my $self = shift;
156         my $url = join('/', $self->param('database'),'_design',$self->param('design'),'_view',$self->param('view') );
157         my $param = $self->req->url->query->clone->remove('callback')->to_string;
158         $url .= '?' . $param if $param;
159         warn "CouchDB proxy $url";
160         _render_jsonp( $self, _couchdb_get($url));
161 };
162
163 # static JSON files from public/json/database/entity/json
164
165 get '/json' => sub {
166         _render_jsonp( shift, [ map { s{public/json/}{}; $_ } glob 'public/json/*' ] );
167 };
168
169 get '/json/:database' => sub {
170         my $self = shift;
171         my $database = $self->param('database');
172
173         my $status = {
174                 document_counts => 0,
175                 name => $database,
176         };
177
178         foreach my $path ( glob "public/json/$database/*" ) {
179                 my @entities = glob "$path/*";
180                 $path =~ s{public/json/$database/}{};
181                 $status->{entities}->{$path} = scalar @entities;
182                 $status->{document_counts}++;
183         }
184
185         _render_jsonp( $self, $status );
186 };
187
188 get '/json/:database/:entity' => sub {
189         my $self = shift;
190
191         my $database = $self->param('database');
192         my $entity   = $self->param('entity');
193
194         my $path = "public/json/$database/$entity";
195         die "$path: $!" unless -d $path;
196
197         my $docs;
198         foreach my $path ( sort glob "$path/*" ) {
199                 open(my $fh, '<', $path) || die $!;
200                 local $/ = undef;
201                 my $str = <$fh>;
202                 warn "# $path $str";
203                 my $data = Mojo::JSON->new->decode( $str );
204                 $data->{_key} = $1 if $path =~ m{/([^/]+$)};
205                 push @$docs, $data;
206         }
207
208         _render_jsonp( $self, $docs )
209 };
210
211 get '/_utils/script/(*url)' => sub { $_[0]->proxy_to( "$couchdb/_utils/script/" . $_[0]->param('url') , with_query_params => 1 ) };
212
213 app->start;
214 __DATA__
215
216 @@ index.html.ep
217 % layout 'funky';
218 Yea baby!
219
220 @@ layouts/funky.html.ep
221 <!doctype html><html>
222     <head><title>Funky!</title></head>
223     <body><%== content %></body>
224 </html>
225
226 @@ layouts/angular.html.ep
227 <!DOCTYPE HTML>
228 <html xmlns:ng="http://angularjs.org">
229   <head>
230    <meta charset="utf-8">
231 % my $ANGULAR_JS = $ENV{ANGULAR_JS} || ( -e 'public/angular/build/angular.js' ? '/angular/build/angular.js' : '/angular/src/angular-bootstrap.js' );
232     <script type="text/javascript"
233          src="<%== $ANGULAR_JS %>" ng:autobind></script>
234   </head>
235   <body><%== content %></body>
236 </html>