1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
|
# FixMyStreet::Geocode::Zurich
# Geocoding with Zurich web service.
#
# Thanks to http://msdn.microsoft.com/en-us/library/ms995764.aspx
# and http://noisemore.wordpress.com/2009/03/19/perl-soaplite-wsse-web-services-security-soapheader/
# for SOAP::Lite pointers
#
# Copyright (c) 2012 UK Citizens Online Democracy. All rights reserved.
# Email: matthew@mysociety.org; WWW: http://www.mysociety.org/
package FixMyStreet::Geocode::Zurich;
use strict;
use Digest::MD5 qw(md5_hex);
use File::Path ();
use Geo::Coordinates::CH1903;
use Storable;
use Utils;
my ($soap, $method, $security);
sub setup_soap {
return if $soap;
# Variables for the SOAP web service
my $geocoder = FixMyStreet->config('GEOCODER');
my $url = $geocoder->{url};
my $username = $geocoder->{username};
my $password = $geocoder->{password};
my $attr = 'http://ch/geoz/fixmyzuerich/service';
my $action = "$attr/IFixMyZuerich/";
require SOAP::Lite;
# Set up the SOAP handler
$security = SOAP::Header->name("Security")->attr({
'mustUnderstand' => 'true',
'xmlns' => 'http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd'
})->value(
\SOAP::Header->name(
"UsernameToken" => \SOAP::Header->value(
SOAP::Header->name('Username', $username),
SOAP::Header->name('Password', $password)
)
)
);
$soap = SOAP::Lite->on_action( sub { $action . $_[1]; } )->proxy($url);
$method = SOAP::Data->name('getLocation')->attr({ xmlns => $attr });
}
# string STRING CONTEXT
# Looks up on Zurich web service a user-inputted location.
# Returns array of (LAT, LON, ERROR), where ERROR is either undef, a string, or
# an array of matches if there are more than one.
# If there is no ambiguity, returns only a {lat,long} hash, unless allow_single_match_string is true
# (because the auto-complete use of this (in /around) should send the matched name even though it's not ambiguous).
#
# The information in the query may be used to disambiguate the location in cobranded
# versions of the site.
sub string {
my ( $s, $c ) = @_;
setup_soap();
my $cache_dir = FixMyStreet->config('GEO_CACHE') . 'zurich/';
my $cache_file = $cache_dir . md5_hex($s);
my $result;
if (-s $cache_file && -M $cache_file <= 7) {
$result = retrieve($cache_file);
} else {
my $search = SOAP::Data->name('search' => $s)->type('');
my $count = SOAP::Data->name('count' => 10)->type('');
eval {
$result = $soap->call($method, $security, $search, $count);
};
if ($@) {
warn $@ if FixMyStreet->config('STAGING_SITE');
return { error => 'The geocoder appears to be down.' };
}
$result = $result->result;
File::Path::mkpath($cache_dir);
store $result, $cache_file if $result;
}
if (!$result || !$result->{Location}) {
return { error => _('Sorry, we could not parse that location. Please try again.') };
}
my $results = $result->{Location};
$results = [ $results ] unless ref $results eq 'ARRAY';
my ( $error, @valid_locations, $latitude, $longitude );
foreach (@$results) {
($latitude, $longitude) =
map { Utils::truncate_coordinate($_) }
Geo::Coordinates::CH1903::to_latlon($_->{easting}, $_->{northing});
push (@$error, {
address => $_->{text},
latitude => $latitude,
longitude => $longitude
});
push (@valid_locations, $_);
last if lc($_->{text}) eq lc($s);
}
if (scalar @valid_locations == 1 && ! $c->stash->{allow_single_geocode_match_strings} ) {
return { latitude => $latitude, longitude => $longitude };
}
return { error => $error };
}
1;
|