123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513 |
- <?php
- /**
- * StatusNet, the distributed open-source microblogging tool
- *
- * Module to convert string locations to Geonames IDs and vice versa
- *
- * PHP version 5
- *
- * LICENCE: This program is free software: you can redistribute it and/or modify
- * it under the terms of the GNU Affero General Public License as published by
- * the Free Software Foundation, either version 3 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU Affero General Public License for more details.
- *
- * You should have received a copy of the GNU Affero General Public License
- * along with this program. If not, see <http://www.gnu.org/licenses/>.
- *
- * @category Action
- * @package StatusNet
- * @author Evan Prodromou <evan@status.net>
- * @copyright 2009 StatusNet Inc.
- * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
- * @link http://status.net/
- */
- if (!defined('STATUSNET')) {
- exit(1);
- }
- /**
- * Module to convert string locations to Geonames IDs and vice versa
- *
- * This handles most of the events that Location class emits. It uses
- * the geonames.org Web service to convert names like 'Montreal, Quebec, Canada'
- * into IDs and lat/lon pairs.
- *
- * @category Module
- * @package StatusNet
- * @author Evan Prodromou <evan@status.net>
- * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
- * @link http://status.net/
- *
- * @seeAlso Location
- */
- class GeonamesModule extends Module
- {
- const PLUGIN_VERSION = '2.0.0';
- const LOCATION_NS = 1;
- public $host = 'ws.geonames.org';
- public $username = null;
- public $token = null;
- public $expiry = 7776000; // 90-day expiry
- public $timeout = 2; // Web service timeout in seconds.
- public $timeoutWindow = 60; // Further lookups in this process will be disabled for N seconds after a timeout.
- public $cachePrefix = null; // Optional shared memcache prefix override
- // to share lookups between local instances.
- protected $lastTimeout = null; // timestamp of last web service timeout
- /**
- * convert a name into a Location object
- *
- * @param string $name Name to convert
- * @param string $language ISO code for anguage the name is in
- * @param Location &$location Location object (may be null)
- *
- * @return boolean whether to continue (results in $location)
- */
- function onLocationFromName($name, $language, &$location)
- {
- $loc = $this->getCache(array('name' => $name,
- 'language' => $language));
- if ($loc !== false) {
- $location = $loc;
- return false;
- }
- try {
- $geonames = $this->getGeonames('search',
- array('maxRows' => 1,
- 'q' => $name,
- 'lang' => $language,
- 'type' => 'xml'));
- } catch (Exception $e) {
- $this->log(LOG_WARNING, "Error for $name: " . $e->getMessage());
- return true;
- }
- if (count($geonames) == 0) {
- // no results
- $this->setCache(array('name' => $name,
- 'language' => $language),
- null);
- return true;
- }
- $n = $geonames[0];
- $location = new Location();
- $location->lat = $this->canonical($n->lat);
- $location->lon = $this->canonical($n->lng);
- $location->names[$language] = (string)$n->name;
- $location->location_id = (string)$n->geonameId;
- $location->location_ns = self::LOCATION_NS;
- $this->setCache(array('name' => $name,
- 'language' => $language),
- $location);
- // handled, don't continue processing!
- return false;
- }
- /**
- * convert an id into a Location object
- *
- * @param string $id Name to convert
- * @param string $ns Name to convert
- * @param string $language ISO code for language for results
- * @param Location &$location Location object (may be null)
- *
- * @return boolean whether to continue (results in $location)
- */
- function onLocationFromId($id, $ns, $language, &$location)
- {
- if ($ns != self::LOCATION_NS) {
- // It's not one of our IDs... keep processing
- return true;
- }
- $loc = $this->getCache(array('id' => $id));
- if ($loc !== false) {
- $location = $loc;
- return false;
- }
- try {
- $geonames = $this->getGeonames('hierarchy',
- array('geonameId' => $id,
- 'lang' => $language));
- } catch (Exception $e) {
- $this->log(LOG_WARNING, "Error for ID $id: " . $e->getMessage());
- return false;
- }
- $parts = array();
- foreach ($geonames as $level) {
- if (in_array($level->fcode, array('PCLI', 'ADM1', 'PPL'))) {
- $parts[] = (string)$level->name;
- }
- }
- $last = $geonames[count($geonames)-1];
- if (!in_array($level->fcode, array('PCLI', 'ADM1', 'PPL'))) {
- $parts[] = (string)$last->name;
- }
- $location = new Location();
- $location->location_id = (string)$last->geonameId;
- $location->location_ns = self::LOCATION_NS;
- $location->lat = $this->canonical($last->lat);
- $location->lon = $this->canonical($last->lng);
- $location->names[$language] = implode(', ', array_reverse($parts));
- $this->setCache(array('id' => (string)$last->geonameId),
- $location);
- // We're responsible for this namespace; nobody else
- // can resolve it
- return false;
- }
- /**
- * convert a lat/lon pair into a Location object
- *
- * Given a lat/lon, we try to find a Location that's around
- * it or nearby. We prefer populated places (cities, towns, villages).
- *
- * @param string $lat Latitude
- * @param string $lon Longitude
- * @param string $language ISO code for language for results
- * @param Location &$location Location object (may be null)
- *
- * @return boolean whether to continue (results in $location)
- */
- function onLocationFromLatLon($lat, $lon, $language, &$location)
- {
- // Make sure they're canonical
- $lat = $this->canonical($lat);
- $lon = $this->canonical($lon);
- $loc = $this->getCache(array('lat' => $lat,
- 'lon' => $lon));
- if ($loc !== false) {
- $location = $loc;
- return false;
- }
- try {
- $geonames = $this->getGeonames('findNearbyPlaceName',
- array('lat' => $lat,
- 'lng' => $lon,
- 'lang' => $language));
- } catch (Exception $e) {
- $this->log(LOG_WARNING, "Error for coords $lat, $lon: " . $e->getMessage());
- return true;
- }
- if (count($geonames) == 0) {
- // no results
- $this->setCache(array('lat' => $lat,
- 'lon' => $lon),
- null);
- return true;
- }
- $n = $geonames[0];
- $parts = array();
- $location = new Location();
- $parts[] = (string)$n->name;
- if (!empty($n->adminName1)) {
- $parts[] = (string)$n->adminName1;
- }
- if (!empty($n->countryName)) {
- $parts[] = (string)$n->countryName;
- }
- $location->location_id = (string)$n->geonameId;
- $location->location_ns = self::LOCATION_NS;
- $location->lat = $this->canonical($n->lat);
- $location->lon = $this->canonical($n->lng);
- $location->names[$language] = implode(', ', $parts);
- $this->setCache(array('lat' => $lat,
- 'lon' => $lon),
- $location);
- // Success! We handled it, so no further processing
- return false;
- }
- /**
- * Human-readable name for a location
- *
- * Given a location, we try to retrieve a human-readable name
- * in the target language.
- *
- * @param Location $location Location to get the name for
- * @param string $language ISO code for language to find name in
- * @param string &$name Place to put the name
- *
- * @return boolean whether to continue
- */
- function onLocationNameLanguage($location, $language, &$name)
- {
- if ($location->location_ns != self::LOCATION_NS) {
- // It's not one of our IDs... keep processing
- return true;
- }
- $id = $location->location_id;
- $n = $this->getCache(array('id' => $id,
- 'language' => $language));
- if ($n !== false) {
- $name = $n;
- return false;
- }
- try {
- $geonames = $this->getGeonames('hierarchy',
- array('geonameId' => $id,
- 'lang' => $language));
- } catch (Exception $e) {
- $this->log(LOG_WARNING, "Error for ID $id: " . $e->getMessage());
- return false;
- }
- if (count($geonames) == 0) {
- $this->setCache(array('id' => $id,
- 'language' => $language),
- null);
- return false;
- }
- $parts = array();
- foreach ($geonames as $level) {
- if (in_array($level->fcode, array('PCLI', 'ADM1', 'PPL'))) {
- $parts[] = (string)$level->name;
- }
- }
- $last = $geonames[count($geonames)-1];
- if (!in_array($level->fcode, array('PCLI', 'ADM1', 'PPL'))) {
- $parts[] = (string)$last->name;
- }
- if (count($parts)) {
- $name = implode(', ', array_reverse($parts));
- $this->setCache(array('id' => $id,
- 'language' => $language),
- $name);
- }
- return false;
- }
- /**
- * Human-readable URL for a location
- *
- * Given a location, we try to retrieve a geonames.org URL.
- *
- * @param Location $location Location to get the url for
- * @param string &$url Place to put the url
- *
- * @return boolean whether to continue
- */
- function onLocationUrl($location, &$url)
- {
- if ($location->location_ns != self::LOCATION_NS) {
- // It's not one of our IDs... keep processing
- return true;
- }
- $url = 'http://www.geonames.org/' . $location->location_id;
- // it's been filled, so don't process further.
- return false;
- }
- /**
- * Machine-readable name for a location
- *
- * Given a location, we try to retrieve a geonames.org URL.
- *
- * @param Location $location Location to get the url for
- * @param string &$url Place to put the url
- *
- * @return boolean whether to continue
- */
- function onLocationRdfUrl($location, &$url)
- {
- if ($location->location_ns != self::LOCATION_NS) {
- // It's not one of our IDs... keep processing
- return true;
- }
- $url = 'http://sws.geonames.org/' . $location->location_id . '/';
- // it's been filled, so don't process further.
- return false;
- }
- function getCache($attrs)
- {
- $c = Cache::instance();
- if (empty($c)) {
- return null;
- }
- $key = $this->cacheKey($attrs);
- $value = $c->get($key);
- return $value;
- }
- function setCache($attrs, $loc)
- {
- $c = Cache::instance();
- if (empty($c)) {
- return null;
- }
- $key = $this->cacheKey($attrs);
- $result = $c->set($key, $loc, 0, time() + $this->expiry);
- return $result;
- }
- function cacheKey($attrs)
- {
- $key = 'geonames:' .
- implode(',', array_keys($attrs)) . ':'.
- Cache::keyize(implode(',', array_values($attrs)));
- if ($this->cachePrefix) {
- return $this->cachePrefix . ':' . $key;
- } else {
- return Cache::key($key);
- }
- }
- function wsUrl($method, $params)
- {
- if (!empty($this->username)) {
- $params['username'] = $this->username;
- }
- if (!empty($this->token)) {
- $params['token'] = $this->token;
- }
- $str = http_build_query($params, null, '&');
- return 'http://'.$this->host.'/'.$method.'?'.$str;
- }
- function getGeonames($method, $params)
- {
- if ($this->lastTimeout && (time() - $this->lastTimeout < $this->timeoutWindow)) {
- // TRANS: Exception thrown when a geo names service is not used because of a recent timeout.
- throw new Exception(_m('Skipping due to recent web service timeout.'));
- }
- $client = HTTPClient::start();
- $client->setConfig('connect_timeout', $this->timeout);
- $client->setConfig('timeout', $this->timeout);
- try {
- $result = $client->get($this->wsUrl($method, $params));
- } catch (Exception $e) {
- common_log(LOG_ERR, __METHOD__ . ": " . $e->getMessage());
- $this->lastTimeout = time();
- throw $e;
- }
- if (!$result->isOk()) {
- // TRANS: Exception thrown when a geo names service does not return an expected response.
- // TRANS: %s is an HTTP error code.
- throw new Exception(sprintf(_m('HTTP error code %s.'),$result->getStatus()));
- }
- $body = $result->getBody();
- if (empty($body)) {
- // TRANS: Exception thrown when a geo names service returns an empty body.
- throw new Exception(_m('Empty HTTP body in response.'));
- }
- // This will throw an exception if the XML is mal-formed
- $document = new SimpleXMLElement($body);
- // No children, usually no results
- $children = $document->children();
- if (count($children) == 0) {
- return array();
- }
- if (isset($document->status)) {
- // TRANS: Exception thrown when a geo names service return a specific error number and error text.
- // TRANS: %1$s is an error code, %2$s is an error message.
- throw new Exception(sprintf(_m('Error #%1$s ("%2$s").'),$document->status['value'],$document->status['message']));
- }
- // Array of elements, >0 elements
- return $document->geoname;
- }
- function onModuleVersion(array &$versions)
- {
- $versions[] = array('name' => 'Geonames',
- 'version' => self::PLUGIN_VERSION,
- 'author' => 'Evan Prodromou',
- 'homepage' => 'https://git.gnu.io/gnu/gnu-social/tree/master/plugins/Geonames',
- 'rawdescription' =>
- // TRANS: Module description.
- _m('Uses <a href="http://geonames.org/">Geonames</a> service to get human-readable '.
- 'names for locations based on user-provided lat/long pairs.'));
- return true;
- }
- function canonical($coord)
- {
- $coord = rtrim($coord, "0");
- $coord = rtrim($coord, ".");
- return $coord;
- }
- }
|