duplicate templates for translation
[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 $client = Mojo::UserAgent->new;
16
17 sub _couchdb_put {
18         my ( $url, $data ) = @_;
19
20         $data->{'$entity'} = $1 if $url =~ m{/(\w+)\.\d+/$/};
21
22         my $json = Mojo::JSON->new->encode( $data );
23
24         my $rev;
25
26         warn "# _couchdb_put $url = $json";
27         return $client->put( "$couchdb/$url" => $json)->res->json;
28 }
29
30 sub _couchdb_get {
31         my ( $url ) = @_;
32         my $return = $client->get( "$couchdb/$url" )->res->json;
33 #       warn "# _couchdb_get $url = ",dump($return);
34         return $return;
35 }
36
37
38 our $id2nr;
39
40
41 sub _render_jsonp {
42         my ( $self, $json ) = @_;
43 #warn "## _render_json ",dump($json);
44         my $data = $self->render( json => $json, partial => 1 );
45 #warn "## _render_json $data";
46         if ( my $callback = $self->param('callback') ) {
47                 $data = "$callback($data)";
48         }
49         $self->render( data => $data, format => 'js' );
50 }
51
52 get '/' => sub {
53         my $self = shift;
54         $self->render_text("...");
55 };
56
57 get '/en' => sub {
58         my $self = shift;
59         $self->render_static('/en/index.html'); # main AngularJS application page
60 };
61
62 get '/hr' => sub {
63         my $self = shift;
64         $self->render_static('/hr/index.html'); # main AngularJS application page
65 };
66
67
68 get '/data/' => sub {
69         my $self = shift;
70         _render_jsonp( $self, _couchdb_get('/_all_dbs') );
71 };
72
73 get '/data/:database' => sub {
74         my $self = shift;
75         my $database = $self->param('database');
76
77         my $list_databases = { name => $database };
78
79         my $counts = _couchdb_get("/$database/_design/entity/_view/counts?group=true");
80         if ( exists $counts->{error} ) {
81                 warn "creating CouchDB view because of ", dump($counts);
82                 _couchdb_put "/$database/_design/entity", {
83                         _id => '_design/entity',
84                         language => 'javascript',
85                         views => {
86                                 counts => {
87                                         map    => q| function(doc) { emit(doc._id.split('.')[0],1); } |,
88                                         reduce => q| function(keys,values,rereduce) { return sum(values); } |,
89                                 }
90                         }
91                 };
92                 $counts = _couchdb_get("/$database/_design/entity/_view/counts?group=true")
93                 || die "give up!";
94         }
95
96         warn "# counts ",dump($counts);
97
98         foreach my $row ( @{ $counts->{rows} } ) {
99                 my $n = $row->{value};
100                 $list_databases->{entities}->{ $row->{key} } = $n;
101                 $list_databases->{document_counts} += $n;
102         }
103         warn dump($list_databases);
104         _render_jsonp( $self,  $list_databases );
105 };
106
107 get '/data/:database/:entity' => sub {
108         my $self = shift;
109
110         my $database = $self->param('database');
111         my $entity   = $self->param('entity');
112
113         my $endkey = $entity;
114         $endkey++;
115
116         my $counts = _couchdb_get qq|/$database/_all_docs?startkey="$entity";endkey="$endkey";include_docs=true|;
117 #       warn "# counts ",dump($counts);
118
119         _render_jsonp( $self, [ map { $_->{doc} } @{ $counts->{rows} } ] )
120 };
121
122 get '/data/:database/:entity/:id' => sub {
123     my $self = shift;
124
125         my $database = $self->param('database');
126         my $entity   = $self->param('entity');
127         my $id       = $self->param('id');
128
129         _render_jsonp( $self, _couchdb_get( "/$database/$entity.$id" ) );
130 };
131
132 any [ 'post' ] => '/data/:database/:entity' => sub {
133         my $self = shift;
134         my $database = $self->param('database');
135         my $entity   = $self->param('entity');
136         my $json = $self->req->json;
137         my $id;
138         if ( exists $json->{'id'} ) { # @id in resource
139                 $id = $json->{'id'};
140                 warn "EXISTING $id\n";
141         } else {
142                 $id = $json->{'id'} = new_uuid;
143                 $json->{entity} = $entity;
144                 warn "NEW $id\n";
145         }
146         warn "## $database $entity $id body ",dump($self->req->body, $json);
147
148         my $new = _couchdb_put "/$database/$entity.$id" => $json;
149         warn "new: ",dump($new);
150         if ( $new->{ok} ) {
151                 $json->{'_'.$_} = $new->{$_} foreach ( 'rev','id' );
152         } else {
153                 warn "ERROR: ",dump($new);
154                 $json->{error} = $new;
155         }
156
157         _render_jsonp( $self,  $json );
158 };
159
160
161 #get '/' => sub { shift->redirect_to('/app/') };
162 get '/en' => sub { shift->redirect_to('/en/index.html') };
163
164 # CouchDB proxy for _design _view
165
166 get '/:database/_design/:design/_view/:view' => sub {
167         my $self = shift;
168         my $url = join('/', $self->param('database'),'_design',$self->param('design'),'_view',$self->param('view') );
169         my $param = $self->req->url->query->clone->remove('callback')->to_string;
170         $url .= '?' . $param if $param;
171         warn "CouchDB proxy $url";
172         _render_jsonp( $self, _couchdb_get($url));
173 };
174
175 app->start;
176