| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | <?php | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * StatusNet, the distributed open-source microblogging tool | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Plugin 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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Plugin 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 Plugin | 
					
						
							|  |  |  |  * @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 GeonamesPlugin extends Plugin | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-06-03 01:56:52 +01:00
										 |  |  |     const PLUGIN_VERSION = '2.0.0'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-17 16:56:43 -08:00
										 |  |  |     const LOCATION_NS = 1; | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-04 15:30:43 -05:00
										 |  |  |     public $host     = 'ws.geonames.org'; | 
					
						
							|  |  |  |     public $username = null; | 
					
						
							|  |  |  |     public $token    = null; | 
					
						
							| 
									
										
										
										
											2009-12-04 16:25:05 -05:00
										 |  |  |     public $expiry   = 7776000; // 90-day expiry
 | 
					
						
							| 
									
										
										
										
											2010-04-06 15:14:28 -07:00
										 |  |  |     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.
 | 
					
						
							| 
									
										
										
										
											2010-03-30 12:19:25 -07:00
										 |  |  |     public $cachePrefix = null; // Optional shared memcache prefix override
 | 
					
						
							|  |  |  |                                 // to share lookups between local instances.
 | 
					
						
							| 
									
										
										
										
											2009-12-04 15:30:43 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-06 15:14:28 -07:00
										 |  |  |     protected $lastTimeout = null; // timestamp of last web service timeout
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * 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) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |         $loc = $this->getCache(array('name' => $name, | 
					
						
							|  |  |  |                                      'language' => $language)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         if ($loc !== false) { | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |             $location = $loc; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         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()); | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         if (count($geonames) == 0) { | 
					
						
							|  |  |  |             // no results
 | 
					
						
							|  |  |  |             $this->setCache(array('name' => $name, | 
					
						
							|  |  |  |                                   'language' => $language), | 
					
						
							|  |  |  |                             null); | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         $n = $geonames[0]; | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         $location = new Location(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         $location->lat              = $this->canonical($n->lat); | 
					
						
							|  |  |  |         $location->lon              = $this->canonical($n->lng); | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         $location->names[$language] = (string)$n->name; | 
					
						
							|  |  |  |         $location->location_id      = (string)$n->geonameId; | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         $location->location_ns      = self::LOCATION_NS; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->setCache(array('name' => $name, | 
					
						
							|  |  |  |                               'language' => $language), | 
					
						
							|  |  |  |                         $location); | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         // handled, don't continue processing!
 | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 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) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-11-17 16:56:43 -08:00
										 |  |  |         if ($ns != self::LOCATION_NS) { | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  |             // It's not one of our IDs... keep processing
 | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |         $loc = $this->getCache(array('id' => $id)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         if ($loc !== false) { | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |             $location = $loc; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         try { | 
					
						
							|  |  |  |             $geonames = $this->getGeonames('hierarchy', | 
					
						
							|  |  |  |                                            array('geonameId' => $id, | 
					
						
							|  |  |  |                                                  'lang' => $language)); | 
					
						
							|  |  |  |         } catch (Exception $e) { | 
					
						
							|  |  |  |             $this->log(LOG_WARNING, "Error for ID $id: " . $e->getMessage()); | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         $parts = array(); | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         foreach ($geonames as $level) { | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |             if (in_array($level->fcode, array('PCLI', 'ADM1', 'PPL'))) { | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |                 $parts[] = (string)$level->name; | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         $last = $geonames[count($geonames)-1]; | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         if (!in_array($level->fcode, array('PCLI', 'ADM1', 'PPL'))) { | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |             $parts[] = (string)$last->name; | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         $location = new Location(); | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         $location->location_id      = (string)$last->geonameId; | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         $location->location_ns      = self::LOCATION_NS; | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         $location->lat              = $this->canonical($last->lat); | 
					
						
							|  |  |  |         $location->lon              = $this->canonical($last->lng); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         $location->names[$language] = implode(', ', array_reverse($parts)); | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         $this->setCache(array('id' => (string)$last->geonameId), | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |                         $location); | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         // We're responsible for this namespace; nobody else
 | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  |         // 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) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         // Make sure they're canonical
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $lat = $this->canonical($lat); | 
					
						
							|  |  |  |         $lon = $this->canonical($lon); | 
					
						
							| 
									
										
										
										
											2009-12-04 16:25:05 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |         $loc = $this->getCache(array('lat' => $lat, | 
					
						
							|  |  |  |                                      'lon' => $lon)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         if ($loc !== false) { | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |             $location = $loc; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         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()); | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         if (count($geonames) == 0) { | 
					
						
							|  |  |  |             // no results
 | 
					
						
							|  |  |  |             $this->setCache(array('lat' => $lat, | 
					
						
							|  |  |  |                                   'lon' => $lon), | 
					
						
							|  |  |  |                             null); | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         $n = $geonames[0]; | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         $parts = array(); | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         $location = new Location(); | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         $parts[] = (string)$n->name; | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         if (!empty($n->adminName1)) { | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |             $parts[] = (string)$n->adminName1; | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         if (!empty($n->countryName)) { | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |             $parts[] = (string)$n->countryName; | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         $location->location_id = (string)$n->geonameId; | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         $location->location_ns = self::LOCATION_NS; | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         $location->lat         = $this->canonical($n->lat); | 
					
						
							|  |  |  |         $location->lon         = $this->canonical($n->lng); | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         $location->names[$language] = implode(', ', $parts); | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         $this->setCache(array('lat' => $lat, | 
					
						
							|  |  |  |                               'lon' => $lon), | 
					
						
							|  |  |  |                         $location); | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         // Success! We handled it, so no further processing
 | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 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) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-11-17 16:56:43 -08:00
										 |  |  |         if ($location->location_ns != self::LOCATION_NS) { | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  |             // It's not one of our IDs... keep processing
 | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         $id = $location->location_id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $n = $this->getCache(array('id' => $id, | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |                                    'language' => $language)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         if ($n !== false) { | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |             $name = $n; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         try { | 
					
						
							|  |  |  |             $geonames = $this->getGeonames('hierarchy', | 
					
						
							|  |  |  |                                            array('geonameId' => $id, | 
					
						
							|  |  |  |                                                  'lang' => $language)); | 
					
						
							|  |  |  |         } catch (Exception $e) { | 
					
						
							|  |  |  |             $this->log(LOG_WARNING, "Error for ID $id: " . $e->getMessage()); | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         if (count($geonames) == 0) { | 
					
						
							|  |  |  |             $this->setCache(array('id' => $id, | 
					
						
							|  |  |  |                                   'language' => $language), | 
					
						
							|  |  |  |                             null); | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         $parts = array(); | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         foreach ($geonames as $level) { | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |             if (in_array($level->fcode, array('PCLI', 'ADM1', 'PPL'))) { | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |                 $parts[] = (string)$level->name; | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         $last = $geonames[count($geonames)-1]; | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         if (!in_array($level->fcode, array('PCLI', 'ADM1', 'PPL'))) { | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |             $parts[] = (string)$last->name; | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         if (count($parts)) { | 
					
						
							|  |  |  |             $name = implode(', ', array_reverse($parts)); | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |             $this->setCache(array('id' => $id, | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |                                   'language' => $language), | 
					
						
							|  |  |  |                             $name); | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 09:26:43 -08:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-10-29 15:07:26 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |      * Human-readable URL for a location | 
					
						
							| 
									
										
										
										
											2009-10-29 15:07:26 -04:00
										 |  |  |      * | 
					
						
							|  |  |  |      * 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) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-11-17 16:56:43 -08:00
										 |  |  |         if ($location->location_ns != self::LOCATION_NS) { | 
					
						
							| 
									
										
										
										
											2009-10-29 15:07:26 -04:00
										 |  |  |             // It's not one of our IDs... keep processing
 | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $url = 'http://www.geonames.org/' . $location->location_id; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-09 20:01:46 +01:00
										 |  |  |         // it's been filled, so don't process further.
 | 
					
						
							| 
									
										
										
										
											2009-10-29 15:07:26 -04:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-11-19 12:00:25 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 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; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-10 10:30:07 -07:00
										 |  |  |         $url = 'http://sws.geonames.org/' . $location->location_id . '/'; | 
					
						
							| 
									
										
										
										
											2009-11-19 12:00:25 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // it's been filled, so don't process further.
 | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     function getCache($attrs) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-09-06 09:56:45 -04:00
										 |  |  |         $c = Cache::instance(); | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-04 15:30:43 -05:00
										 |  |  |         if (empty($c)) { | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |             return null; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-04 16:25:05 -05:00
										 |  |  |         $key = $this->cacheKey($attrs); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $value = $c->get($key); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $value; | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function setCache($attrs, $loc) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-09-06 09:56:45 -04:00
										 |  |  |         $c = Cache::instance(); | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-04 15:30:43 -05:00
										 |  |  |         if (empty($c)) { | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |             return null; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-04 16:25:05 -05:00
										 |  |  |         $key = $this->cacheKey($attrs); | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-04 16:25:05 -05:00
										 |  |  |         $result = $c->set($key, $loc, 0, time() + $this->expiry); | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-04 16:25:05 -05:00
										 |  |  |         return $result; | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function cacheKey($attrs) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-03-30 12:19:25 -07:00
										 |  |  |         $key = 'geonames:' . | 
					
						
							|  |  |  |                implode(',', array_keys($attrs)) . ':'. | 
					
						
							| 
									
										
										
										
											2010-09-06 10:03:51 -04:00
										 |  |  |                Cache::keyize(implode(',', array_values($attrs))); | 
					
						
							| 
									
										
										
										
											2010-03-30 12:19:25 -07:00
										 |  |  |         if ($this->cachePrefix) { | 
					
						
							|  |  |  |             return $this->cachePrefix . ':' . $key; | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-09-06 10:07:43 -04:00
										 |  |  |             return Cache::key($key); | 
					
						
							| 
									
										
										
										
											2010-03-30 12:19:25 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-11-20 07:58:28 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-12-04 15:30:43 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     function wsUrl($method, $params) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!empty($this->username)) { | 
					
						
							|  |  |  |             $params['username'] = $this->username; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($this->token)) { | 
					
						
							|  |  |  |             $params['token'] = $this->token; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:16:22 -08:00
										 |  |  |         $str = http_build_query($params, null, '&'); | 
					
						
							| 
									
										
										
										
											2009-12-04 15:30:43 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return 'http://'.$this->host.'/'.$method.'?'.$str; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     function getGeonames($method, $params) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-04-06 15:14:28 -07:00
										 |  |  |         if ($this->lastTimeout && (time() - $this->lastTimeout < $this->timeoutWindow)) { | 
					
						
							| 
									
										
										
										
											2011-04-08 18:46:41 +02:00
										 |  |  |             // 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.')); | 
					
						
							| 
									
										
										
										
											2010-04-06 15:14:28 -07:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         $client = HTTPClient::start(); | 
					
						
							| 
									
										
										
										
											2010-04-06 15:14:28 -07:00
										 |  |  |         $client->setConfig('connect_timeout', $this->timeout); | 
					
						
							|  |  |  |         $client->setConfig('timeout', $this->timeout); | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-06 15:14:28 -07:00
										 |  |  |         try { | 
					
						
							|  |  |  |             $result = $client->get($this->wsUrl($method, $params)); | 
					
						
							|  |  |  |         } catch (Exception $e) { | 
					
						
							|  |  |  |             common_log(LOG_ERR, __METHOD__ . ": " . $e->getMessage()); | 
					
						
							|  |  |  |             $this->lastTimeout = time(); | 
					
						
							|  |  |  |             throw $e; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (!$result->isOk()) { | 
					
						
							| 
									
										
										
										
											2011-04-08 18:46:41 +02:00
										 |  |  |             // 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())); | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         $body = $result->getBody(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (empty($body)) { | 
					
						
							| 
									
										
										
										
											2011-04-08 18:46:41 +02:00
										 |  |  |             // TRANS: Exception thrown when a geo names service returns an empty body.
 | 
					
						
							|  |  |  |             throw new Exception(_m('Empty HTTP body in response.')); | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // This will throw an exception if the XML is mal-formed
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $document = new SimpleXMLElement($body); | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         // No children, usually no results
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $children = $document->children(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (count($children) == 0) { | 
					
						
							|  |  |  |             return array(); | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (isset($document->status)) { | 
					
						
							| 
									
										
										
										
											2011-04-08 18:46:41 +02:00
										 |  |  |             // 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'])); | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  |         // Array of elements, >0 elements
 | 
					
						
							| 
									
										
										
										
											2009-12-23 12:09:11 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return $document->geoname; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-01-07 17:27:01 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-12 15:03:30 +01:00
										 |  |  |     public function onPluginVersion(array &$versions): bool | 
					
						
							| 
									
										
										
										
											2010-01-07 17:27:01 -08:00
										 |  |  |     { | 
					
						
							|  |  |  |         $versions[] = array('name' => 'Geonames', | 
					
						
							| 
									
										
										
										
											2019-06-03 01:56:52 +01:00
										 |  |  |                             'version' => self::PLUGIN_VERSION, | 
					
						
							| 
									
										
										
										
											2010-01-07 17:27:01 -08:00
										 |  |  |                             'author' => 'Evan Prodromou', | 
					
						
							| 
									
										
										
										
											2016-01-22 16:38:42 +00:00
										 |  |  |                             'homepage' => 'https://git.gnu.io/gnu/gnu-social/tree/master/plugins/Geonames', | 
					
						
							| 
									
										
										
										
											2010-01-07 17:27:01 -08:00
										 |  |  |                             'rawdescription' => | 
					
						
							| 
									
										
										
										
											2011-04-08 18:46:41 +02:00
										 |  |  |                             // TRANS: Plugin description.
 | 
					
						
							| 
									
										
										
										
											2010-01-07 17:27:01 -08:00
										 |  |  |                             _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; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-01-30 12:40:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     function canonical($coord) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $coord = rtrim($coord, "0"); | 
					
						
							|  |  |  |         $coord = rtrim($coord, "."); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $coord; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-10-22 15:44:36 -04:00
										 |  |  | } |