Added supplier location endpoint
This commit is contained in:
parent
df90f62b7a
commit
5e078f8f8b
8 changed files with 414 additions and 1 deletions
4
cpanfile
4
cpanfile
|
@ -32,3 +32,7 @@ feature 'postgres', 'PostgreSQL Support' => sub {
|
||||||
requires 'Test::PostgreSQL';
|
requires 'Test::PostgreSQL';
|
||||||
};
|
};
|
||||||
|
|
||||||
|
feature 'codepoint-open', 'Code Point Open manipulation' => sub {
|
||||||
|
requires 'Geo::UK::Postcode::CodePointOpen';
|
||||||
|
};
|
||||||
|
|
||||||
|
|
|
@ -148,6 +148,10 @@ sub startup {
|
||||||
|
|
||||||
my $api_v1 = $api->under('/v1');
|
my $api_v1 = $api->under('/v1');
|
||||||
|
|
||||||
|
my $api_v1_supplier = $api_v1->under('/supplier');
|
||||||
|
|
||||||
|
$api_v1_supplier->post('/location')->to('api-v1-supplier-location#index');
|
||||||
|
|
||||||
my $api_v1_org = $api_v1->under('/organisation')->to('api-v1-organisation#auth');
|
my $api_v1_org = $api_v1->under('/organisation')->to('api-v1-organisation#auth');
|
||||||
|
|
||||||
$api_v1_org->post('/graphs')->to('api-v1-organisation-graphs#index');
|
$api_v1_org->post('/graphs')->to('api-v1-organisation-graphs#index');
|
||||||
|
|
|
@ -36,7 +36,6 @@ sub run {
|
||||||
split_postcode => 1,
|
split_postcode => 1,
|
||||||
);
|
);
|
||||||
|
|
||||||
use Devel::Dwarn;
|
|
||||||
my $pc_rs = $self->app->schema->resultset('GbPostcode');
|
my $pc_rs = $self->app->schema->resultset('GbPostcode');
|
||||||
while ( my $pc = $iter->() ) {
|
while ( my $pc = $iter->() ) {
|
||||||
$pc_rs->find_or_create(
|
$pc_rs->find_or_create(
|
||||||
|
|
109
lib/Pear/LocalLoop/Controller/Api/V1/Supplier/Location.pm
Normal file
109
lib/Pear/LocalLoop/Controller/Api/V1/Supplier/Location.pm
Normal file
|
@ -0,0 +1,109 @@
|
||||||
|
package Pear::LocalLoop::Controller::Api::V1::Supplier::Location;
|
||||||
|
use Mojo::Base 'Mojolicious::Controller';
|
||||||
|
|
||||||
|
has validation_data => sub {
|
||||||
|
my $children_errors = {
|
||||||
|
latitude => {
|
||||||
|
validation => [
|
||||||
|
{ required => {} },
|
||||||
|
{ number => { error_prefix => 'not_number' } },
|
||||||
|
{ in_range => { args => [ -90, 90 ], error_prefix => 'outside_range' } },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
longitude => {
|
||||||
|
validation => [
|
||||||
|
{ required => {} },
|
||||||
|
{ number => { error_prefix => 'not_number' } },
|
||||||
|
{ in_range => { args => [ -180, 180 ], error_prefix => 'outside_range' } },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
index => {
|
||||||
|
north_east => {
|
||||||
|
validation => [
|
||||||
|
{ required => {} },
|
||||||
|
{ is_object => { error_prefix => 'not_object' } },
|
||||||
|
],
|
||||||
|
children => $children_errors,
|
||||||
|
},
|
||||||
|
south_west => {
|
||||||
|
validation => [
|
||||||
|
{ required => {} },
|
||||||
|
{ is_object => { error_prefix => 'not_object' } },
|
||||||
|
],
|
||||||
|
children => $children_errors,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sub index {
|
||||||
|
my $c = shift;
|
||||||
|
|
||||||
|
return if $c->validation_error('index');
|
||||||
|
|
||||||
|
my $json = $c->stash->{api_json};
|
||||||
|
|
||||||
|
# Extra custom error, because its funny
|
||||||
|
if ( $json->{north_east}->{latitude} < $json->{south_west}->{latitude} ) {
|
||||||
|
return $c->render(
|
||||||
|
json => {
|
||||||
|
success => Mojo::JSON->false,
|
||||||
|
errors => [ 'upside_down' ],
|
||||||
|
},
|
||||||
|
status => 400,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
my $entity = $c->stash->{api_user}->entity;
|
||||||
|
my $entity_type_object = $entity->type_object;
|
||||||
|
|
||||||
|
# need: organisations only, with name, latitude, and longitude
|
||||||
|
my $org_rs = $entity->purchases->search_related('seller',
|
||||||
|
{
|
||||||
|
'seller.type' => 'organisation',
|
||||||
|
'organisation.latitude' => { -between => [
|
||||||
|
$json->{south_west}->{latitude},
|
||||||
|
$json->{north_east}->{latitude},
|
||||||
|
] },
|
||||||
|
'organisation.longitude' => { -between => [
|
||||||
|
$json->{south_west}->{longitude},
|
||||||
|
$json->{north_east}->{longitude},
|
||||||
|
] },
|
||||||
|
},
|
||||||
|
{
|
||||||
|
join => [ qw/ organisation / ],
|
||||||
|
columns => [
|
||||||
|
'organisation.name',
|
||||||
|
'organisation.latitude',
|
||||||
|
'organisation.longitude',
|
||||||
|
],
|
||||||
|
group_by => [ qw/ organisation.id / ],
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
$org_rs->result_class('DBIx::Class::ResultClass::HashRefInflator');
|
||||||
|
|
||||||
|
my $suppliers = [ map {
|
||||||
|
{
|
||||||
|
latitude => $_->{organisation}->{latitude} * 1,
|
||||||
|
longitude => $_->{organisation}->{longitude} * 1,
|
||||||
|
name => $_->{organisation}->{name},
|
||||||
|
}
|
||||||
|
} $org_rs->all ];
|
||||||
|
|
||||||
|
$c->render(
|
||||||
|
json => {
|
||||||
|
success => Mojo::JSON->true,
|
||||||
|
suppliers => $suppliers,
|
||||||
|
self => {
|
||||||
|
latitude => $entity_type_object->latitude,
|
||||||
|
longitude => $entity_type_object->longitude,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
1;
|
|
@ -64,6 +64,115 @@ sub register {
|
||||||
$value = $app->parse_iso_datetime( $value );
|
$value = $app->parse_iso_datetime( $value );
|
||||||
return defined $value ? undef : 1;
|
return defined $value ? undef : 1;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
$app->validator->add_check( is_object => sub {
|
||||||
|
my ( $validation, $name, $value ) = @_;
|
||||||
|
return ref ( $value ) eq 'HASH' ? undef : 1;
|
||||||
|
});
|
||||||
|
|
||||||
|
$app->validator->add_check( in_range => sub {
|
||||||
|
my ( $validation, $name, $value, $low, $high ) = @_;
|
||||||
|
return $low < $value && $value < $high ? undef : 1;
|
||||||
|
});
|
||||||
|
|
||||||
|
$app->helper( validation_error => sub { _validation_error(@_) } );
|
||||||
|
}
|
||||||
|
|
||||||
|
=head2 validation_error
|
||||||
|
|
||||||
|
Returns undef if there is no validation error, returns true otherwise - having
|
||||||
|
set the errors up as required. Renders out the errors as an array, with status
|
||||||
|
400
|
||||||
|
|
||||||
|
=cut
|
||||||
|
|
||||||
|
sub _validation_error {
|
||||||
|
my ( $c, $sub_name ) = @_;
|
||||||
|
|
||||||
|
my $val_data = $c->validation_data->{ $sub_name };
|
||||||
|
return unless defined $val_data;
|
||||||
|
my $data = $c->stash->{api_json};
|
||||||
|
|
||||||
|
my @errors = _validate_set( $c, $val_data, $data );
|
||||||
|
|
||||||
|
if ( scalar @errors ) {
|
||||||
|
my @sorted_errors = sort @errors;
|
||||||
|
$c->render(
|
||||||
|
json => {
|
||||||
|
success => Mojo::JSON->false,
|
||||||
|
errors => \@sorted_errors,
|
||||||
|
},
|
||||||
|
status => 400,
|
||||||
|
);
|
||||||
|
return \@errors;
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub _validate_set {
|
||||||
|
my ( $c, $val_data, $data, $parent_name ) = @_;
|
||||||
|
|
||||||
|
my @errors;
|
||||||
|
|
||||||
|
# MUST get a raw validation object
|
||||||
|
my $validation = $c->app->validator->validation;
|
||||||
|
$validation->input( $data );
|
||||||
|
|
||||||
|
for my $val_data_key ( keys %$val_data ) {
|
||||||
|
|
||||||
|
$validation->topic( $val_data_key );
|
||||||
|
|
||||||
|
my $val_set = $val_data->{$val_data_key};
|
||||||
|
|
||||||
|
my $custom_check_prefix = {};
|
||||||
|
|
||||||
|
for my $val_error ( @{$val_set->{validation}} ) {
|
||||||
|
my ( $val_validator ) = keys %$val_error;
|
||||||
|
|
||||||
|
unless (
|
||||||
|
$validation->validator->checks->{$val_validator}
|
||||||
|
|| $val_validator =~ /required|optional/
|
||||||
|
) {
|
||||||
|
$c->app->log->warn( 'Unknown Validator [' . $val_validator . ']' );
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( my $custom_prefix = $val_error->{ $val_validator }->{ error_prefix } ) {
|
||||||
|
$custom_check_prefix->{ $val_validator } = $custom_prefix;
|
||||||
|
}
|
||||||
|
my $val_args = $val_error->{ $val_validator }->{ args };
|
||||||
|
|
||||||
|
$validation->$val_validator(
|
||||||
|
( $val_validator =~ /required|optional/ ? $val_data_key : () ),
|
||||||
|
( defined $val_args ? @$val_args : () )
|
||||||
|
);
|
||||||
|
|
||||||
|
# stop bothering checking if failed, validation stops after first failure
|
||||||
|
last if $validation->has_error( $val_data_key );
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( $validation->has_error( $val_data_key ) ) {
|
||||||
|
my ( $check ) = @{ $validation->error( $val_data_key ) };
|
||||||
|
my $error_prefix = defined $custom_check_prefix->{ $check }
|
||||||
|
? $custom_check_prefix->{ $check }
|
||||||
|
: $check;
|
||||||
|
my $error_string = join ('_',
|
||||||
|
$error_prefix,
|
||||||
|
( defined $parent_name ? $parent_name : () ),
|
||||||
|
$val_data_key,
|
||||||
|
);
|
||||||
|
push @errors, $error_string;
|
||||||
|
} elsif ( defined $val_set->{ children } ) {
|
||||||
|
push @errors, _validate_set(
|
||||||
|
$c,
|
||||||
|
$val_set->{ children },
|
||||||
|
$data->{ $val_data_key },
|
||||||
|
$val_data_key );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return @errors;
|
||||||
}
|
}
|
||||||
|
|
||||||
1;
|
1;
|
||||||
|
|
|
@ -63,4 +63,16 @@ sub name {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sub type_object {
|
||||||
|
my $self = shift;
|
||||||
|
|
||||||
|
if ( $self->type eq 'customer' ) {
|
||||||
|
return $self->customer;
|
||||||
|
} elsif ( $self->type eq 'organisation' ) {
|
||||||
|
return $self->organisation;
|
||||||
|
} else {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
1;
|
1;
|
||||||
|
|
59
t/api/v1/supplier/location.t
Normal file
59
t/api/v1/supplier/location.t
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
use Mojo::Base -strict;
|
||||||
|
|
||||||
|
use FindBin qw/ $Bin /;
|
||||||
|
|
||||||
|
use Test::More;
|
||||||
|
use Mojo::JSON;
|
||||||
|
use Test::Pear::LocalLoop;
|
||||||
|
use DateTime;
|
||||||
|
|
||||||
|
my $framework = Test::Pear::LocalLoop->new(
|
||||||
|
etc_dir => "$Bin/../../../etc",
|
||||||
|
);
|
||||||
|
$framework->install_fixtures('full');
|
||||||
|
|
||||||
|
my $t = $framework->framework;
|
||||||
|
my $schema = $t->app->schema;
|
||||||
|
|
||||||
|
my $session_key = $framework->login({
|
||||||
|
email => 'org1@example.com',
|
||||||
|
password => 'abc123',
|
||||||
|
});
|
||||||
|
|
||||||
|
$t->post_ok('/api/upload' => json => {
|
||||||
|
transaction_value => 10,
|
||||||
|
transaction_type => 1,
|
||||||
|
purchase_time => "2017-08-14T11:29:07.965+01:00",
|
||||||
|
organisation_id => 2,
|
||||||
|
session_key => $session_key,
|
||||||
|
})
|
||||||
|
->status_is(200)
|
||||||
|
->json_is('/success', Mojo::JSON->true);
|
||||||
|
|
||||||
|
# Rough area around Lancaster
|
||||||
|
$t->post_ok('/api/v1/supplier/location' => json => {
|
||||||
|
session_key => $session_key,
|
||||||
|
north_east => {
|
||||||
|
latitude => 54.077665,
|
||||||
|
longitude => -2.761860,
|
||||||
|
},
|
||||||
|
south_west => {
|
||||||
|
latitude => 54.013361,
|
||||||
|
longitude => -2.857647,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
->status_is(200)->or($framework->dump_error)
|
||||||
|
->json_is('/success', Mojo::JSON->true)
|
||||||
|
->json_is('/suppliers', [
|
||||||
|
{
|
||||||
|
name => 'Test Org 2',
|
||||||
|
latitude => 54.04679,
|
||||||
|
longitude => -2.7963,
|
||||||
|
},
|
||||||
|
])
|
||||||
|
->json_is('/self', {
|
||||||
|
latitude => 54.04725,
|
||||||
|
longitude => -2.79611,
|
||||||
|
});
|
||||||
|
|
||||||
|
done_testing;
|
117
t/api/v1/supplier/location_errors.t
Normal file
117
t/api/v1/supplier/location_errors.t
Normal file
|
@ -0,0 +1,117 @@
|
||||||
|
use Mojo::Base -strict;
|
||||||
|
|
||||||
|
use FindBin qw/ $Bin /;
|
||||||
|
|
||||||
|
use Test::More;
|
||||||
|
use Mojo::JSON;
|
||||||
|
use Test::Pear::LocalLoop;
|
||||||
|
use DateTime;
|
||||||
|
|
||||||
|
my $framework = Test::Pear::LocalLoop->new(
|
||||||
|
etc_dir => "$Bin/../../../etc",
|
||||||
|
);
|
||||||
|
$framework->install_fixtures('full');
|
||||||
|
|
||||||
|
my $t = $framework->framework;
|
||||||
|
my $schema = $t->app->schema;
|
||||||
|
|
||||||
|
my $session_key = $framework->login({
|
||||||
|
email => 'org1@example.com',
|
||||||
|
password => 'abc123',
|
||||||
|
});
|
||||||
|
|
||||||
|
$t->post_ok('/api/v1/supplier/location' => json => {
|
||||||
|
session_key => $session_key,
|
||||||
|
})
|
||||||
|
->status_is(400)->or($framework->dump_error)
|
||||||
|
->json_is('/success', Mojo::JSON->false)
|
||||||
|
->json_is('/errors', [
|
||||||
|
'required_north_east',
|
||||||
|
'required_south_west',
|
||||||
|
]);
|
||||||
|
|
||||||
|
$t->post_ok('/api/v1/supplier/location' => json => {
|
||||||
|
session_key => $session_key,
|
||||||
|
north_east => 'banana',
|
||||||
|
south_west => 'apple',
|
||||||
|
})
|
||||||
|
->status_is(400)->or($framework->dump_error)
|
||||||
|
->json_is('/success', Mojo::JSON->false)
|
||||||
|
->json_is('/errors', [
|
||||||
|
'not_object_north_east',
|
||||||
|
'not_object_south_west',
|
||||||
|
]);
|
||||||
|
|
||||||
|
$t->post_ok('/api/v1/supplier/location' => json => {
|
||||||
|
session_key => $session_key,
|
||||||
|
north_east => {},
|
||||||
|
south_west => {},
|
||||||
|
})
|
||||||
|
->status_is(400)->or($framework->dump_error)
|
||||||
|
->json_is('/success', Mojo::JSON->false)
|
||||||
|
->json_is('/errors', [
|
||||||
|
'required_north_east_latitude',
|
||||||
|
'required_north_east_longitude',
|
||||||
|
'required_south_west_latitude',
|
||||||
|
'required_south_west_longitude',
|
||||||
|
]);
|
||||||
|
|
||||||
|
$t->post_ok('/api/v1/supplier/location' => json => {
|
||||||
|
session_key => $session_key,
|
||||||
|
north_east => {
|
||||||
|
latitude => 'banana',
|
||||||
|
longitude => 'apple',
|
||||||
|
},
|
||||||
|
south_west => {
|
||||||
|
latitude => 'grapefruit',
|
||||||
|
longitude => 'orange',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
->status_is(400)->or($framework->dump_error)
|
||||||
|
->json_is('/success', Mojo::JSON->false)
|
||||||
|
->json_is('/errors', [
|
||||||
|
'not_number_north_east_latitude',
|
||||||
|
'not_number_north_east_longitude',
|
||||||
|
'not_number_south_west_latitude',
|
||||||
|
'not_number_south_west_longitude',
|
||||||
|
]);
|
||||||
|
|
||||||
|
$t->post_ok('/api/v1/supplier/location' => json => {
|
||||||
|
session_key => $session_key,
|
||||||
|
north_east => {
|
||||||
|
latitude => 90.00001,
|
||||||
|
longitude => 180.00001,
|
||||||
|
},
|
||||||
|
south_west => {
|
||||||
|
latitude => -90.00001,
|
||||||
|
longitude => -180.00001,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
->status_is(400)->or($framework->dump_error)
|
||||||
|
->json_is('/success', Mojo::JSON->false)
|
||||||
|
->json_is('/errors', [
|
||||||
|
'outside_range_north_east_latitude',
|
||||||
|
'outside_range_north_east_longitude',
|
||||||
|
'outside_range_south_west_latitude',
|
||||||
|
'outside_range_south_west_longitude',
|
||||||
|
]);
|
||||||
|
|
||||||
|
# upside down when NeLat < SwLat
|
||||||
|
$t->post_ok('/api/v1/supplier/location' => json => {
|
||||||
|
session_key => $session_key,
|
||||||
|
north_east => {
|
||||||
|
latitude => -89,
|
||||||
|
longitude => 170,
|
||||||
|
},
|
||||||
|
south_west => {
|
||||||
|
latitude => 89,
|
||||||
|
longitude => -170,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
->status_is(400)->or($framework->dump_error)
|
||||||
|
->json_is('/success', Mojo::JSON->false)
|
||||||
|
->json_is('/errors', [
|
||||||
|
'upside_down',
|
||||||
|
])->or($framework->dump_error);
|
||||||
|
|
||||||
|
done_testing;
|
Reference in a new issue