diff --git a/perllib/FixMyStreet/Cobrand/Brent.pm b/perllib/FixMyStreet/Cobrand/Brent.pm index c300fb56e37..c0860146528 100644 --- a/perllib/FixMyStreet/Cobrand/Brent.pm +++ b/perllib/FixMyStreet/Cobrand/Brent.pm @@ -707,9 +707,16 @@ sub open311_extra_data_exclude { return []; } +=head2 open311_pre_send + +We check in Echo to see if something has already been sent there first. + +=cut + sub open311_pre_send { my ($self, $row, $open311) = @_; return 'SKIP' if $row->category eq 'Request new container' && $row->get_extra_field_value('request_referral'); + return 'SENT' if $self->open311_pre_send_check($row, 'FMS'); } =head2 open311_post_send @@ -745,6 +752,8 @@ sub open311_post_send { if ($error =~ /Selected reservations expired|Invalid reservation reference/) { $self->bulky_refetch_slots($row2); $row->discard_changes; + } elsif ($error =~ /Internal error/) { + $self->open311_post_send_error_check("FMS", $row, $row2, $sender); } }); } diff --git a/perllib/FixMyStreet/Cobrand/Bromley.pm b/perllib/FixMyStreet/Cobrand/Bromley.pm index 879ef8d7df8..167ee100d66 100644 --- a/perllib/FixMyStreet/Cobrand/Bromley.pm +++ b/perllib/FixMyStreet/Cobrand/Bromley.pm @@ -288,6 +288,8 @@ sub open311_pre_send { my $text = $row->detail . "\n\nPrivate comments: $private_comments"; $row->detail($text); } + + return 'SENT' if $self->open311_pre_send_check($row, 'FMS'); } sub _include_user_title_in_extra { @@ -348,6 +350,8 @@ sub open311_post_send { } elsif ($error =~ /Selected reservations expired|Invalid reservation reference/) { $self->bulky_refetch_slots($row2); $row->discard_changes; + } elsif ($error =~ /Internal error/) { + $self->open311_post_send_error_check("FMS", $row, $row2, $sender); } }); } diff --git a/perllib/FixMyStreet/Roles/Cobrand/Echo.pm b/perllib/FixMyStreet/Roles/Cobrand/Echo.pm index 0b9f1657c2b..9ee8bb397e6 100644 --- a/perllib/FixMyStreet/Roles/Cobrand/Echo.pm +++ b/perllib/FixMyStreet/Roles/Cobrand/Echo.pm @@ -673,6 +673,43 @@ sub admin_templates_external_status_code_hook { return $code; } +sub open311_pre_send_check { + my ($self, $row, $prefix) = @_; + + my $guid = $self->open311_check_existing_event("$prefix-" . $row->id, "ClientReference"); + if ($guid) { + # Event already exists + $row->discard_changes; + $row->update({ external_id => $guid }); + return 1; + } + return 0; +} + +sub open311_post_send_error_check { + my ($self, $prefix, $row, $row2, $sender) = @_; + $self->open311_post_send_check("$prefix-" . $row->id, 'ClientReference', $row, $row2, $sender); +} + +sub open311_post_send_check { + my ($self, $id, $type, $row, $row2, $sender) = @_; + if (my $guid = $self->open311_check_existing_event($id, $type)) { + $row2->external_id($guid); + $sender->success(1); + $row2->update; + $row->discard_changes; + } +} + +sub open311_check_existing_event { + my ($self, $id, $type) = @_; + + my $cfg = $self->feature('echo'); + my $echo = Integrations::Echo->new(%$cfg); + my $event = $echo->GetEvent($id, $type) || {}; + return $event->{Guid}; +} + =item waste_fetch_events Loop through all open waste events to see if there have been any updates diff --git a/perllib/FixMyStreet/Roles/Cobrand/SLWP.pm b/perllib/FixMyStreet/Roles/Cobrand/SLWP.pm index 3520dae816a..73235fe27d9 100644 --- a/perllib/FixMyStreet/Roles/Cobrand/SLWP.pm +++ b/perllib/FixMyStreet/Roles/Cobrand/SLWP.pm @@ -478,6 +478,23 @@ sub open311_extra_data_include { return $open311_only; } +sub echo_reference_prefix { + my $self = shift; + return 'LBS' if $self->moniker eq 'sutton'; + return 'RBK' if $self->moniker eq 'kingston'; + return 'MRT' if $self->moniker eq 'merton'; +} + +=item * We check in Echo to see if something has already been sent there first + +=cut + +sub open311_pre_send { + my ($self, $row, $open311) = @_; + + return 'SENT' if $self->open311_pre_send_check($row, $self->echo_reference_prefix); +} + =item * If Echo errors, we try and deal with standard issues - a renewal on an expired subscription, or a duplicate event =cut @@ -503,13 +520,9 @@ sub open311_post_send { $row->discard_changes; } elsif ($error =~ /Duplicate Event! Original eventID: (\d+)/) { my $id = $1; - my $cfg = $self->feature('echo'); - my $echo = Integrations::Echo->new(%$cfg); - my $event = $echo->GetEvent($id, 'Id'); - $row2->external_id($event->{Guid}); - $sender->success(1); - $row2->update; - $row->discard_changes; + $self->open311_post_send_check($id, "Id", $row, $row2, $sender); + } elsif ($error =~ /Internal error/) { + $self->open311_post_send_error_check($self->echo_reference_prefix, $row, $row2, $sender); } }); } diff --git a/perllib/FixMyStreet/SendReport/Open311.pm b/perllib/FixMyStreet/SendReport/Open311.pm index f4bdfcea355..83187efe6af 100644 --- a/perllib/FixMyStreet/SendReport/Open311.pm +++ b/perllib/FixMyStreet/SendReport/Open311.pm @@ -77,7 +77,7 @@ sub send { my $open311 = Open311->new( %open311_params ); my $skip = $cobrand->call_hook(open311_pre_send => $row, $open311); - $skip = $skip && $skip eq 'SKIP'; + $skip = $skip && ($skip eq 'SKIP' || $skip eq 'SENT'); my $resp; if (!$skip) { @@ -88,7 +88,7 @@ sub send { $row->discard_changes; if ( $skip || $resp ) { - $row->update({ external_id => $resp }); + $row->update({ external_id => $resp }) if $resp; $self->success( 1 ); } else { $self->error( "Failed to send over Open311\n" ) unless $self->error; diff --git a/t/app/controller/waste_bromley_bulky.t b/t/app/controller/waste_bromley_bulky.t index 0549955a875..b48ec1109f6 100644 --- a/t/app/controller/waste_bromley_bulky.t +++ b/t/app/controller/waste_bromley_bulky.t @@ -162,6 +162,7 @@ FixMyStreet::override_config { $echo->mock( 'CancelReservedSlotsForEvent', sub { [] } ); $echo->mock( 'GetTasks', sub { [] } ); $echo->mock( 'GetEventsForObject', sub { [] } ); + $echo->mock( 'GetEvent', sub { {} } ); $echo->mock( 'FindPoints',sub { [ { Description => '2 Example Street, Bromley, BR1 1AF', diff --git a/t/app/controller/waste_kingston_r.t b/t/app/controller/waste_kingston_r.t index a25f10e6230..c8204611267 100644 --- a/t/app/controller/waste_kingston_r.t +++ b/t/app/controller/waste_kingston_r.t @@ -485,6 +485,7 @@ sub shared_echo_mocks { }); $e->mock('GetServiceUnitsForObject', sub { $bin_data }); $e->mock('GetEventsForObject', sub { [] }); + $e->mock('GetEvent', sub { {} }); $e->mock('GetTasks', sub { [] }); $e->mock( 'CancelReservedSlotsForEvent', sub { my (undef, $guid) = @_; diff --git a/t/app/controller/waste_sutton_r.t b/t/app/controller/waste_sutton_r.t index c433a7e3622..4645a22bd75 100644 --- a/t/app/controller/waste_sutton_r.t +++ b/t/app/controller/waste_sutton_r.t @@ -454,6 +454,7 @@ sub shared_echo_mocks { }); $e->mock('GetServiceUnitsForObject', sub { $bin_data }); $e->mock('GetEventsForObject', sub { [] }); + $e->mock('GetEvent', sub { {} }); $e->mock('GetTasks', sub { [] }); $e->mock( 'CancelReservedSlotsForEvent', sub { my (undef, $guid) = @_; diff --git a/t/cobrand/brent.t b/t/cobrand/brent.t index f3b93eb03d3..a18a648c611 100644 --- a/t/cobrand/brent.t +++ b/t/cobrand/brent.t @@ -223,6 +223,8 @@ create_contact({ category => 'Additional collection', email => 'general@brent.go { code => 'service_id', required => 1, automated => 'hidden_field' }, ); +my $echo = shared_echo_mocks(); + subtest "title is labelled 'location of problem' in open311 extended description" => sub { my ($problem) = $mech->create_problems_for_body(1, $brent->id, 'title', { category => 'Graffiti' , @@ -698,6 +700,8 @@ FixMyStreet::override_config { $mech->host("brent.fixmystreet.com"); }; +undef $echo; # Otherwise tests below fail + package SOAP::Result; sub result { return $_[0]->{result}; } sub new { my $c = shift; bless { @_ }, $c; } @@ -906,6 +910,8 @@ FixMyStreet::override_config { } }; +$echo = shared_echo_mocks(); + FixMyStreet::override_config { ALLOWED_COBRANDS => [ 'brent', 'tfl' ], MAPIT_URL => 'http://mapit.uk/', @@ -1587,6 +1593,7 @@ sub shared_echo_mocks { }; }); $e->mock('GetEventsForObject', sub { [] }); + $e->mock('GetEvent', sub { {} }); $e->mock('GetTasks', sub { [] }); return $e; } diff --git a/t/cobrand/bromley.t b/t/cobrand/bromley.t index b4405060470..fe2b6e69a35 100644 --- a/t/cobrand/bromley.t +++ b/t/cobrand/bromley.t @@ -160,6 +160,9 @@ subtest 'Updates from staff with no text but with private comments are sent' => }; }; +my $echo = Test::MockModule->new('Integrations::Echo'); +$echo->mock(GetEvent => sub { {} }); + for my $test ( { desc => 'testing special Open311 behaviour', diff --git a/t/cobrand/bromley_waste.t b/t/cobrand/bromley_waste.t index b3aa0adfa00..50698e63378 100644 --- a/t/cobrand/bromley_waste.t +++ b/t/cobrand/bromley_waste.t @@ -94,6 +94,9 @@ subtest 'check footer is powered by SocietyWorks' => sub { }; subtest 'test waste duplicate' => sub { + my $echo = Test::MockModule->new('Integrations::Echo'); + $echo->mock(GetEvent => sub { {} }); + my $sender = FixMyStreet::SendReport::Open311->new( bodies => [ $body ], body_config => { $body->id => $body }, ); @@ -111,6 +114,9 @@ subtest 'test waste duplicate' => sub { }; subtest 'test DD taking so long it expires' => sub { + my $echo = Test::MockModule->new('Integrations::Echo'); + $echo->mock(GetEvent => sub { {} }); + my $title = $report->title; $report->update({ title => "Garden Subscription - Renew" }); my $sender = FixMyStreet::SendReport::Open311->new( diff --git a/t/cobrand/sutton.t b/t/cobrand/sutton.t index c305abf3d03..6efeb5161f6 100644 --- a/t/cobrand/sutton.t +++ b/t/cobrand/sutton.t @@ -72,6 +72,9 @@ FixMyStreet::override_config { MAPIT_URL => 'http://mapit.uk/', STAGING_FLAGS => { send_reports => 1 }, }, sub { + my $echo = Test::MockModule->new('Integrations::Echo'); + $echo->mock('GetEvent', sub { {} }); + subtest 'test waste duplicate' => sub { my $sender = FixMyStreet::SendReport::Open311->new( bodies => [ $body ], body_config => { $body->id => $body }, @@ -106,11 +109,10 @@ FixMyStreet::override_config { my $sender = FixMyStreet::SendReport::Open311->new( bodies => [ $body ], body_config => { $body->id => $body }, ); - my $echo = Test::MockModule->new('Integrations::Echo'); - $echo->mock('GetEvent', sub { { - Guid => 'a-guid', - Id => 123, - } } ); + $echo->mock('GetEvent', sub { + return {} if $_[2] eq 'ClientReference'; + return { Guid => 'a-guid', Id => 123 } + }); Open311->_inject_response('/requests.xml', 'Duplicate Event! Original eventID: 123', 500); $sender->send($report, { easting => 1, @@ -121,6 +123,61 @@ FixMyStreet::override_config { is $report->external_id, 'a-guid'; }; + subtest 'test internal error, no event, at the Echo side' => sub { + $report->update({ external_id => undef }); + my $sender = FixMyStreet::SendReport::Open311->new( + bodies => [ $body ], body_config => { $body->id => $body }, + ); + $echo->mock('GetEvent', sub { {} }); + Open311->_inject_response('/requests.xml', 'Internal error', 500); + $sender->send($report, { + easting => 1, + northing => 2, + url => 'http://example.org/', + }); + is $sender->success, 0; + is $report->external_id, undef; + }; + + subtest 'test internal error, event accepted, at the Echo side' => sub { + my $sender = FixMyStreet::SendReport::Open311->new( + bodies => [ $body ], body_config => { $body->id => $body }, + ); + $echo->mock('GetEvent', sub { + my $fn = (caller(2))[3]; + if ($fn =~ /open311_post_send_check/) { + return { Guid => 'c-guid', Id => 123 } + } else { + return {}; + } + }); + Open311->_inject_response('/requests.xml', 'Internal error', 500); + $sender->send($report, { + easting => 1, + northing => 2, + url => 'http://example.org/', + }); + is $sender->success, 1; + is $report->external_id, 'c-guid'; + }; + + subtest 'test event already existing at the Echo side' => sub { + my $sender = FixMyStreet::SendReport::Open311->new( + bodies => [ $body ], body_config => { $body->id => $body }, + ); + $echo->mock('GetEvent', sub { + return { Guid => 'b-guid', Id => 123 }; + }); + $sender->send($report, { + easting => 1, + northing => 2, + url => 'http://example.org/', + }); + is $sender->success, 1; + is $report->external_id, 'b-guid'; + $echo->mock('GetEvent', sub { {} }); + }; + subtest 'correct payment data sent across' => sub { $report->category('Garden Subscription'); $report->update_extra_field({ name => 'PaymentCode', value => 'Code4321' }); @@ -159,7 +216,7 @@ FixMyStreet::override_config { like $body, qr/Dear Sutton Council,\s+A user of FixMyStreet has submitted/; like $body, qr{http://www.example.org/report/$id}; }; - + undef $echo; }; package SOAP::Result; @@ -402,6 +459,7 @@ FixMyStreet::override_config { bodies => [ $body ], body_config => { $body->id => $body }, ); my $echo = Test::MockModule->new('Integrations::Echo'); + $echo->mock('GetEvent', sub { {} }); $echo->mock('CancelReservedSlotsForEvent', sub { is $_[1], $report->get_extra_field_value('GUID'); }); diff --git a/t/sendreport/open311.t b/t/sendreport/open311.t index d50da9835bf..60878b598fe 100644 --- a/t/sendreport/open311.t +++ b/t/sendreport/open311.t @@ -16,6 +16,9 @@ $ukc->mock('lookup_site_code', sub { return "Road ID"; }); +my $echo = Test::MockModule->new('Integrations::Echo'); +$echo->mock(GetEvent => sub { {} }); + package main; sub test_overrides; # defined below