. */ declare(strict_types=1); namespace Fisharebest\Webtrees\Module; use Fig\Http\Message\StatusCodeInterface; use Fisharebest\Webtrees\Html; use Fisharebest\Webtrees\I18N; use Fisharebest\Webtrees\Registry; use GuzzleHttp\Client; use GuzzleHttp\Psr7\Request; use JsonException; use Psr\Http\Message\RequestInterface; use Psr\Http\Message\ResponseInterface; use function json_decode; use const JSON_THROW_ON_ERROR; /** * Trait ModuleMapGeoLocationTrait - default implementation of ModuleMapGeoLocationInterface */ trait ModuleMapGeoLocationTrait { /** * A sentence describing what this module does. * * @return string */ public function description(): string { return I18N::translate('Use an external service to find locations.'); } /** * @param string $place * * @return array */ public function searchPlaceNames(string $place): array { if (strlen($place) <= 2) { return []; } $key = $this->name() . $place; $cache = Registry::cache()->file(); $ttl = 86400; return $cache->remember($key, function () use ($place) { $request = $this->searchLocationsRequest($place); $client = new Client([ 'timeout' => 3, ]); $response = $client->send($request); if ($response->getStatusCode() === StatusCodeInterface::STATUS_OK) { return $this->extractLocationsFromResponse($response); } return []; }, $ttl); } /** * @param string $place * * @return RequestInterface */ protected function searchLocationsRequest(string $place): RequestInterface { $uri = Html::url('https://nominatim.openstreetmap.org/search', [ 'accept-language' => I18N::languageTag(), 'format' => 'jsonv2', 'limit' => 50, 'q' => $place, ]); return new Request('GET', $uri); } /** * @param ResponseInterface $response * * @return array */ protected function extractLocationsFromResponse(ResponseInterface $response): array { $body = $response->getBody()->getContents(); try { return json_decode($body, false, 512, JSON_THROW_ON_ERROR); } catch (JsonException $ex) { return []; } } }