#!/usr/bin/env perl
+use lib 'common/mojo/lib';
+
use Mojolicious::Lite;
use Data::Dump qw(dump);
+use Time::HiRes;
+use Clone qw(clone);
+
+sub new_uuid { Time::HiRes::time * 100000 }
# based on
# http://docs.getangular.com/REST.Basic
{ '$id' => 3, bar => 2 },
{ '$id' => 4, baz => 3 },
],
+ },
+ 'AddressBook' => {
+ people => [
+ {name=>'Misko'},
+ {name=>'Igor'},
+ {name=>'Adam'},
+ {name=>'Elliott'}
+ ]
}
};
our $id2nr;
-get '/' => 'index';
+sub _render_jsonp {
+ my ( $self, $json ) = @_;
+#warn "## _render_json ",dump($json);
+ my $data = $self->render( json => $json, partial => 1 );
+warn "## _render_json $data";
+ if ( my $callback = $self->param('callback') ) {
+ $data = "$callback($data)";
+ }
+ $self->render( data => $data, format => 'js' );
+}
+
+#get '/' => 'index';
get '/_replicate' => sub {
my $self = shift;
if ( my $from = $self->param('from') ) {
my $got = $self->client->get( $from )->res->json;
warn "# from $from ",dump($got);
- $self->render_json( $got );
+ _render_jsonp( $self, $got );
my $database = $got->{name};
my $entities = $got->{entities};
};
get '/_data' => sub {
- shift->render_json( $data )
+ my $self = shift;
+ _render_jsonp( $self, $data )
};
get '/data/' => sub {
my $self = shift;
- $self->render_json( [ keys %$data ] );
+ _render_jsonp( $self, [ keys %$data ] );
};
get '/data/:database' => sub {
$list_databases->{document_count} += $count;
}
warn dump($list_databases);
- $self->render_json( $list_databases );
+ _render_jsonp( $self, $list_databases );
};
get '/data/:database/:entity' => sub {
my $self = shift;
- $self->render_json( $data->{ $self->param('database') }->{ $self->param('entity' ) } );
+ _render_jsonp( $self, $data->{ $self->param('database') }->{ $self->param('entity' ) } );
};
get '/data/:database/:entity/:id' => sub {
if ( exists $id2nr->{$database}->{$entity}->{$id} ) {
my $nr = $id2nr->{$database}->{$entity}->{$id};
warn "# entity $id -> $nr\n";
- $self->render_json( $data->{$database}->{$entity}->[$nr] );
+ _render_jsonp( $self, $data->{$database}->{$entity}->[$nr] );
} else {
die "no entity $entity $id in ", dump( $id2nr->{$database}->{$entity} );
}
};
-any [ 'put', 'post' ] => '/data/:database/:entity/:id' => sub {
+any [ 'post' ] => '/data/:database/:entity' => sub {
my $self = shift;
- my $data = $self->req->json;
- warn "# body ",dump($self->req->body, $data);
+ my $json = $self->req->json;
+ my $id = $json->{'$id'} # XXX we don't get it back from angular.js
+ || $json->{'_id'} # so we use our version
+ || new_uuid;
+ warn "## $id body ",dump($self->req->body, $json);
die "no data" unless $data;
- $data->{ $self->param('database') }->{ $self->param('entity') }->{ $self->param('id') } = $data;
- $self->render_json( $data );
+
+ $json->{'$id'} ||= $id; # angular.js doesn't resend this one
+ $json->{'_id'} = $id; # but does this one :-)
+
+ my $database = $self->param('database');
+ my $entity = $self->param('entity');
+
+ my $nr = $id2nr->{$database}->{$entity}->{$id};
+ if ( defined $nr ) {
+ $data->{$database}->{$entity}->[$nr] = $json;
+ warn "# update $nr $id ",dump($json);
+ } else {
+ push @{ $data->{$database}->{$entity} }, $json;
+ my $nr = $#{ $data->{$database}->{$entity} };
+ $id2nr->{$database}->{$entity}->{$id} = $nr;
+ warn "# added $nr $id ",dump($json);
+ }
+ _render_jsonp( $self, $json );
};
get '/demo/:groovy' => sub {
$self->render(text => $self->param('groovy'), layout => 'funky');
};
+get '/' => sub { shift->redirect_to('/Cookbook') };
get '/Cookbook' => 'Cookbook';
get '/Cookbook/:example' => sub {
my $self = shift;
$self->render( "Cookbook/" . $self->param('example'), layout => 'angular' );
};
+get '/conference/:page' => sub {
+ my $self = shift;
+ $self->render( "conference/" . $self->param('page'), layout => 'angular' );
+};
app->start;
__DATA__
<!DOCTYPE HTML>
<html xmlns:ng="http://angularjs.org">
<head>
+% my $ANGULAR_JS = $ENV{ANGULAR_JS} || ( -e 'public/angular/build/angular.js' ? '/angular/build/angular.js' : '/angular/src/angular-bootstrap.js' );
<script type="text/javascript"
- src="<%== $ENV{ANGULAR_JS} || '/build/angular.js' %>" ng:autobind></script>
+ src="<%== $ANGULAR_JS %>" ng:autobind></script>
</head>
<body><%== content %></body>
</html>