123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486 |
- <?php
- declare(strict_types=1);
- /*
- * This file is part of the Geocoder package.
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- *
- * @license MIT License
- */
- namespace Geocoder\Provider\GoogleMaps;
- use Geocoder\Collection;
- use Geocoder\Exception\InvalidCredentials;
- use Geocoder\Exception\InvalidServerResponse;
- use Geocoder\Exception\QuotaExceeded;
- use Geocoder\Exception\UnsupportedOperation;
- use Geocoder\Http\Provider\AbstractHttpProvider;
- use Geocoder\Model\AddressBuilder;
- use Geocoder\Model\AddressCollection;
- use Geocoder\Provider\GoogleMaps\Model\GoogleAddress;
- use Geocoder\Provider\Provider;
- use Geocoder\Query\GeocodeQuery;
- use Geocoder\Query\ReverseQuery;
- use Http\Client\HttpClient;
- /**
- * @author William Durand <william.durand1@gmail.com>
- */
- final class GoogleMaps extends AbstractHttpProvider implements Provider
- {
- /**
- * @var string
- */
- const GEOCODE_ENDPOINT_URL_SSL = 'https://maps.googleapis.com/maps/api/geocode/json?address=%s';
- /**
- * @var string
- */
- const REVERSE_ENDPOINT_URL_SSL = 'https://maps.googleapis.com/maps/api/geocode/json?latlng=%F,%F';
- /**
- * @var string|null
- */
- private $region;
- /**
- * @var string|null
- */
- private $apiKey;
- /**
- * @var string|null
- */
- private $clientId;
- /**
- * @var string|null
- */
- private $privateKey;
- /**
- * @var string|null
- */
- private $channel;
- /**
- * Google Maps for Business
- * https://developers.google.com/maps/documentation/business/
- * Maps for Business is no longer accepting new signups.
- *
- * @param HttpClient $client An HTTP adapter
- * @param string $clientId Your Client ID
- * @param string $privateKey Your Private Key (optional)
- * @param string $region Region biasing (optional)
- * @param string $apiKey Google Geocoding API key (optional)
- * @param string $channel Google Channel parameter (optional)
- *
- * @return GoogleMaps
- */
- public static function business(
- HttpClient $client,
- string $clientId,
- string $privateKey = null,
- string $region = null,
- string $apiKey = null,
- string $channel = null
- ) {
- $provider = new self($client, $region, $apiKey);
- $provider->clientId = $clientId;
- $provider->privateKey = $privateKey;
- $provider->channel = $channel;
- return $provider;
- }
- /**
- * @param HttpClient $client An HTTP adapter
- * @param string $region Region biasing (optional)
- * @param string $apiKey Google Geocoding API key (optional)
- */
- public function __construct(HttpClient $client, string $region = null, string $apiKey = null)
- {
- parent::__construct($client);
- $this->region = $region;
- $this->apiKey = $apiKey;
- }
- public function geocodeQuery(GeocodeQuery $query): Collection
- {
- // Google API returns invalid data if IP address given
- // This API doesn't handle IPs
- if (filter_var($query->getText(), FILTER_VALIDATE_IP)) {
- throw new UnsupportedOperation('The GoogleMaps provider does not support IP addresses, only street addresses.');
- }
- $url = sprintf(self::GEOCODE_ENDPOINT_URL_SSL, rawurlencode($query->getText()));
- if (null !== $bounds = $query->getBounds()) {
- $url .= sprintf(
- '&bounds=%s,%s|%s,%s',
- $bounds->getSouth(),
- $bounds->getWest(),
- $bounds->getNorth(),
- $bounds->getEast()
- );
- }
- if (null !== $components = $query->getData('components')) {
- $serializedComponents = is_string($components) ? $components : $this->serializeComponents($components);
- $url .= sprintf('&components=%s', urlencode($serializedComponents));
- }
- return $this->fetchUrl($url, $query->getLocale(), $query->getLimit(), $query->getData('region', $this->region));
- }
- public function reverseQuery(ReverseQuery $query): Collection
- {
- $coordinate = $query->getCoordinates();
- $url = sprintf(self::REVERSE_ENDPOINT_URL_SSL, $coordinate->getLatitude(), $coordinate->getLongitude());
- if (null !== $locationType = $query->getData('location_type')) {
- $url .= '&location_type='.urlencode($locationType);
- }
- if (null !== $resultType = $query->getData('result_type')) {
- $url .= '&result_type='.urlencode($resultType);
- }
- return $this->fetchUrl($url, $query->getLocale(), $query->getLimit(), $query->getData('region', $this->region));
- }
- /**
- * {@inheritdoc}
- */
- public function getName(): string
- {
- return 'google_maps';
- }
- /**
- * @param string $url
- * @param string $locale
- *
- * @return string query with extra params
- */
- private function buildQuery(string $url, string $locale = null, string $region = null): string
- {
- if (null === $this->apiKey && null === $this->clientId) {
- throw new InvalidCredentials('You must provide an API key. Keyless access was removed in June, 2016');
- }
- if (null !== $locale) {
- $url = sprintf('%s&language=%s', $url, $locale);
- }
- if (null !== $region) {
- $url = sprintf('%s®ion=%s', $url, $region);
- }
- if (null !== $this->apiKey) {
- $url = sprintf('%s&key=%s', $url, $this->apiKey);
- }
- if (null !== $this->clientId) {
- $url = sprintf('%s&client=%s', $url, $this->clientId);
- if (null !== $this->channel) {
- $url = sprintf('%s&channel=%s', $url, $this->channel);
- }
- if (null !== $this->privateKey) {
- $url = $this->signQuery($url);
- }
- }
- return $url;
- }
- /**
- * @param string $url
- * @param string $locale
- * @param int $limit
- * @param string $region
- *
- * @return AddressCollection
- *
- * @throws InvalidServerResponse
- * @throws InvalidCredentials
- */
- private function fetchUrl(string $url, string $locale = null, int $limit, string $region = null): AddressCollection
- {
- $url = $this->buildQuery($url, $locale, $region);
- $content = $this->getUrlContents($url);
- $json = $this->validateResponse($url, $content);
- // no result
- if (!isset($json->results) || !count($json->results) || 'OK' !== $json->status) {
- return new AddressCollection([]);
- }
- $results = [];
- foreach ($json->results as $result) {
- $builder = new AddressBuilder($this->getName());
- $this->parseCoordinates($builder, $result);
- // set official Google place id
- if (isset($result->place_id)) {
- $builder->setValue('id', $result->place_id);
- }
- // update address components
- foreach ($result->address_components as $component) {
- foreach ($component->types as $type) {
- $this->updateAddressComponent($builder, $type, $component);
- }
- }
- /** @var GoogleAddress $address */
- $address = $builder->build(GoogleAddress::class);
- $address = $address->withId($builder->getValue('id'));
- if (isset($result->geometry->location_type)) {
- $address = $address->withLocationType($result->geometry->location_type);
- }
- if (isset($result->types)) {
- $address = $address->withResultType($result->types);
- }
- if (isset($result->formatted_address)) {
- $address = $address->withFormattedAddress($result->formatted_address);
- }
- $results[] = $address
- ->withStreetAddress($builder->getValue('street_address'))
- ->withIntersection($builder->getValue('intersection'))
- ->withPolitical($builder->getValue('political'))
- ->withColloquialArea($builder->getValue('colloquial_area'))
- ->withWard($builder->getValue('ward'))
- ->withNeighborhood($builder->getValue('neighborhood'))
- ->withPremise($builder->getValue('premise'))
- ->withSubpremise($builder->getValue('subpremise'))
- ->withNaturalFeature($builder->getValue('natural_feature'))
- ->withAirport($builder->getValue('airport'))
- ->withPark($builder->getValue('park'))
- ->withPointOfInterest($builder->getValue('point_of_interest'))
- ->withEstablishment($builder->getValue('establishment'))
- ->withSubLocalityLevels($builder->getValue('subLocalityLevel', []))
- ->withPostalCodeSuffix($builder->getValue('postal_code_suffix'))
- ->withPartialMatch($result->partial_match ?? false);
- if (count($results) >= $limit) {
- break;
- }
- }
- return new AddressCollection($results);
- }
- /**
- * Update current resultSet with given key/value.
- *
- * @param AddressBuilder $builder
- * @param string $type Component type
- * @param object $values The component values
- */
- private function updateAddressComponent(AddressBuilder $builder, string $type, $values)
- {
- switch ($type) {
- case 'postal_code':
- $builder->setPostalCode($values->long_name);
- break;
- case 'locality':
- case 'postal_town':
- $builder->setLocality($values->long_name);
- break;
- case 'administrative_area_level_1':
- case 'administrative_area_level_2':
- case 'administrative_area_level_3':
- case 'administrative_area_level_4':
- case 'administrative_area_level_5':
- $builder->addAdminLevel(intval(substr($type, -1)), $values->long_name, $values->short_name);
- break;
- case 'sublocality_level_1':
- case 'sublocality_level_2':
- case 'sublocality_level_3':
- case 'sublocality_level_4':
- case 'sublocality_level_5':
- $subLocalityLevel = $builder->getValue('subLocalityLevel', []);
- $subLocalityLevel[] = [
- 'level' => intval(substr($type, -1)),
- 'name' => $values->long_name,
- 'code' => $values->short_name,
- ];
- $builder->setValue('subLocalityLevel', $subLocalityLevel);
- break;
- case 'country':
- $builder->setCountry($values->long_name);
- $builder->setCountryCode($values->short_name);
- break;
- case 'street_number':
- $builder->setStreetNumber($values->long_name);
- break;
- case 'route':
- $builder->setStreetName($values->long_name);
- break;
- case 'sublocality':
- $builder->setSubLocality($values->long_name);
- break;
- case 'street_address':
- case 'intersection':
- case 'political':
- case 'colloquial_area':
- case 'ward':
- case 'neighborhood':
- case 'premise':
- case 'subpremise':
- case 'natural_feature':
- case 'airport':
- case 'park':
- case 'point_of_interest':
- case 'establishment':
- case 'postal_code_suffix':
- $builder->setValue($type, $values->long_name);
- break;
- default:
- }
- }
- /**
- * Sign a URL with a given crypto key
- * Note that this URL must be properly URL-encoded
- * src: http://gmaps-samples.googlecode.com/svn/trunk/urlsigning/UrlSigner.php-source.
- *
- * @param string $query Query to be signed
- *
- * @return string $query query with signature appended
- */
- private function signQuery(string $query): string
- {
- $url = parse_url($query);
- $urlPartToSign = $url['path'].'?'.$url['query'];
- // Decode the private key into its binary format
- $decodedKey = base64_decode(str_replace(['-', '_'], ['+', '/'], $this->privateKey));
- // Create a signature using the private key and the URL-encoded
- // string using HMAC SHA1. This signature will be binary.
- $signature = hash_hmac('sha1', $urlPartToSign, $decodedKey, true);
- $encodedSignature = str_replace(['+', '/'], ['-', '_'], base64_encode($signature));
- return sprintf('%s&signature=%s', $query, $encodedSignature);
- }
- /**
- * Serialize the component query parameter.
- *
- * @param array $components
- *
- * @return string
- */
- private function serializeComponents(array $components): string
- {
- return implode('|', array_map(function ($name, $value) {
- return sprintf('%s:%s', $name, $value);
- }, array_keys($components), $components));
- }
- /**
- * Decode the response content and validate it to make sure it does not have any errors.
- *
- * @param string $url
- * @param string $content
- *
- * @return \Stdclass result form json_decode()
- *
- * @throws InvalidCredentials
- * @throws InvalidServerResponse
- * @throws QuotaExceeded
- */
- private function validateResponse(string $url, $content)
- {
- // Throw exception if invalid clientID and/or privateKey used with GoogleMapsBusinessProvider
- if (false !== strpos($content, "Provided 'signature' is not valid for the provided client ID")) {
- throw new InvalidCredentials(sprintf('Invalid client ID / API Key %s', $url));
- }
- $json = json_decode($content);
- // API error
- if (!isset($json)) {
- throw InvalidServerResponse::create($url);
- }
- if ('REQUEST_DENIED' === $json->status && 'The provided API key is invalid.' === $json->error_message) {
- throw new InvalidCredentials(sprintf('API key is invalid %s', $url));
- }
- if ('REQUEST_DENIED' === $json->status) {
- throw new InvalidServerResponse(sprintf('API access denied. Request: %s - Message: %s', $url, $json->error_message));
- }
- // you are over your quota
- if ('OVER_QUERY_LIMIT' === $json->status) {
- throw new QuotaExceeded(sprintf('Daily quota exceeded %s', $url));
- }
- return $json;
- }
- /**
- * Parse coordinates and bounds.
- *
- * @param AddressBuilder $builder
- * @param \Stdclass $result
- */
- private function parseCoordinates(AddressBuilder $builder, $result)
- {
- $coordinates = $result->geometry->location;
- $builder->setCoordinates($coordinates->lat, $coordinates->lng);
- if (isset($result->geometry->bounds)) {
- $builder->setBounds(
- $result->geometry->bounds->southwest->lat,
- $result->geometry->bounds->southwest->lng,
- $result->geometry->bounds->northeast->lat,
- $result->geometry->bounds->northeast->lng
- );
- } elseif (isset($result->geometry->viewport)) {
- $builder->setBounds(
- $result->geometry->viewport->southwest->lat,
- $result->geometry->viewport->southwest->lng,
- $result->geometry->viewport->northeast->lat,
- $result->geometry->viewport->northeast->lng
- );
- } elseif ('ROOFTOP' === $result->geometry->location_type) {
- // Fake bounds
- $builder->setBounds(
- $coordinates->lat,
- $coordinates->lng,
- $coordinates->lat,
- $coordinates->lng
- );
- }
- }
- }
|