Angular /data/database REST API using CouchDB view
[angular-mojolicious.git] / 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
10 sub new_uuid { Time::HiRes::time * 100000 }
11
12 # based on
13 # http://docs.getangular.com/REST.Basic
14 # http://angular.getangular.com/data
15
16 my $couchdb = 'http://localhost:5984';
17 my $client = Mojo::Client->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         warn "# _couchdb_put $url = $json";
27         $client->put( "$couchdb/$url" => $json => sub {
28                 my ($client,$tx) = @_;
29                 if ($tx->error) {
30                         die "ERROR CouchDB ",$tx->error;
31                 }
32                 my $response = $tx->res->json;
33                 warn "## CouchDB response ",dump($response);
34         })->process;
35 }
36
37 sub _couchdb_get {
38         my ( $url ) = @_;
39         my $return = $client->get( "$couchdb/$url" )->res->json;
40         warn "# _couchdb_get $url = ",dump($return);
41         return $return;
42 }
43
44
45 our $id2nr;
46
47
48 sub _render_jsonp {
49         my ( $self, $json ) = @_;
50 #warn "## _render_json ",dump($json);
51         my $data = $self->render( json => $json, partial => 1 );
52 warn "## _render_json $data";
53         if ( my $callback = $self->param('callback') ) {
54                 $data = "$callback($data)";
55         }
56         $self->render( data => $data, format => 'js' );
57 }
58
59 #get '/' => 'index';
60
61 get '/_replicate' => sub {
62         my $self = shift;
63
64         if ( my $from = $self->param('from') ) {
65                 my $got = $self->client->get( $from )->res->json;
66                 warn "# from $from ",dump($got);
67                 _render_jsonp( $self,  $got );
68
69                 my $database = $got->{name};
70                 my $entities = $got->{entities};
71
72                 if ( $database && $entities ) {
73                         foreach my $entity ( keys %$entities ) {
74                                 my $url = $from;
75                                 $url =~ s{/?$}{/}; # add slash at end
76                                 $url .= $entity;
77                                 my $e = $self->client->get( $url )->res->json;
78                                 warn "# replicated $url ", dump($e);
79                                 _chouchdb_put( $self, $database, $entity, $e->{'$id'}, $e );
80                         }
81                 }
82         }
83 };
84
85 get '/data/' => sub {
86         my $self = shift;
87         _render_jsonp( $self, _couchdb_get('/_all_dbs') );
88 };
89
90 get '/data/:database' => sub {
91         my $self = shift;
92         my $database = $self->param('database');
93
94         my $list_databases = { name => $database };
95
96         my $counts = _couchdb_get("/$database/_design/entity/_view/counts?group=true");
97         if ( exists $counts->{error} ) {
98                 warn "creating CouchDB view because of ", dump($counts);
99                 _couchdb_put "/$database/_design/entity", {
100                         _id => '_design/entity',
101                         language => 'javascript',
102                         views => {
103                                 counts => {
104                                         map    => q| function(doc) { emit(doc.$entity,1); } |,
105                                         reduce => q| function(keys,values,rereduce) { return sum(values); } |,
106                                 }
107                         }
108                 };
109                 $counts = _couchdb_get("/$database/_design/entity/_view/counts?group=true")
110                 || die "give up!";
111         }
112
113         warn "# counts ",dump($counts);
114
115         foreach my $row ( @{ $counts->{rows} } ) {
116                 my $n = $row->{value};
117                 $list_databases->{entities}->{ $row->{key} } = $n;
118                 $list_databases->{document_counts} += $n;
119         }
120         warn dump($list_databases);
121         _render_jsonp( $self,  $list_databases );
122 };
123
124 get '/data/:database/:entity' => sub {
125         my $self = shift;
126         _render_jsonp( $self, _couchdb_get( '/' . $self->param('database') . '/_all_docs' ) ); # FIXME
127 };
128
129 get '/data/:database/:entity/:id' => sub {
130     my $self = shift;
131
132         my $database = $self->param('database');
133         my $entity   = $self->param('entity');
134         my $id       = $self->param('id');
135
136         _render_jsonp( $self, _couchdb_get( "/$database/$entity.$id" ) );
137 };
138
139 any [ 'post' ] => '/data/:database/:entity' => sub {
140         my $self = shift;
141         my $database = $self->param('database');
142         my $entity   = $self->param('entity');
143         my $json = $self->req->json;
144         my $id = $json->{'$id'} # XXX we don't get it back from angular.js
145                 || new_uuid;
146         warn "## $database $entity $id body ",dump($self->req->body, $json);
147
148         $json->{'$id'} ||= $id; # make sure $id is in there
149
150         _couchdb_put "/$database/$entity.$id" => $json;
151
152         _render_jsonp( $self,  $json );
153 };
154
155
156 get '/' => sub { shift->redirect_to('/Cookbook') };
157
158 get '/Cookbook' => 'Cookbook';
159 get '/Cookbook/:example' => sub {
160         my $self = shift;
161         $self->render( "Cookbook/" . $self->param('example'), layout => 'angular' );
162 };
163
164 get '/conference/:page' => sub {
165         my $self = shift;
166         $self->render( "conference/" . $self->param('page'), layout => 'angular' );
167 };
168
169 app->start;
170 __DATA__
171
172 @@ index.html.ep
173 % layout 'funky';
174 Yea baby!
175
176 @@ layouts/funky.html.ep
177 <!doctype html><html>
178     <head><title>Funky!</title></head>
179     <body><%== content %></body>
180 </html>
181
182 @@ layouts/angular.html.ep
183 <!DOCTYPE HTML>
184 <html xmlns:ng="http://angularjs.org">
185   <head>
186    <meta charset="utf-8">
187 % my $ANGULAR_JS = $ENV{ANGULAR_JS} || ( -e 'public/angular/build/angular.js' ? '/angular/build/angular.js' : '/angular/src/angular-bootstrap.js' );
188     <script type="text/javascript"
189          src="<%== $ANGULAR_JS %>" ng:autobind></script>
190   </head>
191   <body><%== content %></body>
192 </html>