| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  | <?php | 
					
						
							|  |  |  | /* | 
					
						
							|  |  |  |  * StatusNet - the distributed open-source microblogging tool | 
					
						
							|  |  |  |  * Copyright (C) 2010, StatusNet, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Some utilities for generating hint data | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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/>. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class DiscoveryHints { | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  |     static function fromXRD(XML_XRD $xrd) | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         $hints = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-30 23:12:35 +02:00
										 |  |  |         foreach ($xrd->links as $link) { | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  |             switch ($link->rel) { | 
					
						
							| 
									
										
										
										
											2013-11-09 00:49:00 +01:00
										 |  |  |             case WebFingerResource_Profile::PROFILEPAGE: | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  |                 $hints['profileurl'] = $link->href; | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2013-11-01 17:00:12 +01:00
										 |  |  |             case Salmon::REL_SALMON: | 
					
						
							|  |  |  |             case Salmon::NS_MENTIONS:   // XXX: deprecated, remove in the future
 | 
					
						
							|  |  |  |             case Salmon::NS_REPLIES:    // XXX: deprecated, remove in the future
 | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  |                 $hints['salmon'] = $link->href; | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |                 break; | 
					
						
							|  |  |  |             case Discovery::UPDATESFROM: | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  |                 if (empty($link->type) || $link->type == 'application/atom+xml') { | 
					
						
							|  |  |  |                     $hints['feedurl'] = $link->href; | 
					
						
							| 
									
										
										
										
											2012-05-04 12:01:13 -04:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |                 break; | 
					
						
							|  |  |  |             case Discovery::HCARD: | 
					
						
							| 
									
										
										
										
											2014-06-23 20:51:00 +02:00
										 |  |  |             case Discovery::MF2_HCARD: | 
					
						
							|  |  |  |                 $hints['hcard'] = $link->href; | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |                 break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $hints; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static function fromHcardUrl($url) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $client = new HTTPClient(); | 
					
						
							|  |  |  |         $client->setHeader('Accept', 'text/html,application/xhtml+xml'); | 
					
						
							|  |  |  |         $response = $client->get($url); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!$response->isOk()) { | 
					
						
							|  |  |  |             return null; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return self::hcardHints($response->getBody(), | 
					
						
							|  |  |  |                                 $response->getUrl()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static function hcardHints($body, $url) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  |         $hcard = self::_hcard($body, $url); | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-19 15:28:06 -05:00
										 |  |  |         if (empty($hcard)) { | 
					
						
							|  |  |  |             return array(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |         $hints = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  |         // XXX: don't copy stuff into an array and then copy it again
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |         if (array_key_exists('nickname', $hcard)) { | 
					
						
							|  |  |  |             $hints['nickname'] = $hcard['nickname']; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (array_key_exists('fn', $hcard)) { | 
					
						
							|  |  |  |             $hints['fullname'] = $hcard['fn']; | 
					
						
							|  |  |  |         } else if (array_key_exists('n', $hcard)) { | 
					
						
							|  |  |  |             $hints['fullname'] = implode(' ', $hcard['n']); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-21 10:18:13 -07:00
										 |  |  |         if (array_key_exists('photo', $hcard) && count($hcard['photo'])) { | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  |             $hints['avatar'] = $hcard['photo'][0]; | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (array_key_exists('note', $hcard)) { | 
					
						
							|  |  |  |             $hints['bio'] = $hcard['note']; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (array_key_exists('adr', $hcard)) { | 
					
						
							|  |  |  |             if (is_string($hcard['adr'])) { | 
					
						
							|  |  |  |                 $hints['location'] = $hcard['adr']; | 
					
						
							|  |  |  |             } else if (is_array($hcard['adr'])) { | 
					
						
							|  |  |  |                 $hints['location'] = implode(' ', $hcard['adr']); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (array_key_exists('url', $hcard)) { | 
					
						
							|  |  |  |             if (is_string($hcard['url'])) { | 
					
						
							|  |  |  |                 $hints['homepage'] = $hcard['url']; | 
					
						
							| 
									
										
										
										
											2010-03-19 15:54:16 -07:00
										 |  |  |             } else if (is_array($hcard['url']) && !empty($hcard['url'])) { | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |                 // HACK get the last one; that's how our hcards look
 | 
					
						
							|  |  |  |                 $hints['homepage'] = $hcard['url'][count($hcard['url'])-1]; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $hints; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  |     static function _hcard($body, $url) | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-08-13 10:51:00 -07:00
										 |  |  |         // DOMDocument::loadHTML may throw warnings on unrecognized elements,
 | 
					
						
							|  |  |  |         // and notices on unrecognized namespaces.
 | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-13 10:51:00 -07:00
										 |  |  |         $old = error_reporting(error_reporting() & ~(E_WARNING | E_NOTICE)); | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $doc = new DOMDocument(); | 
					
						
							|  |  |  |         $doc->loadHTML($body); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         error_reporting($old); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $xp = new DOMXPath($doc); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $hcardNodes = self::_getChildrenByClass($doc->documentElement, 'vcard', $xp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $hcards = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for ($i = 0; $i < $hcardNodes->length; $i++) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $hcardNode = $hcardNodes->item($i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $hcard = self::_hcardFromNode($hcardNode, $xp, $url); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $hcards[] = $hcard; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $repr = null; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         foreach ($hcards as $hcard) { | 
					
						
							|  |  |  |             if (in_array($url, $hcard['url'])) { | 
					
						
							|  |  |  |                 $repr = $hcard; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!is_null($repr)) { | 
					
						
							|  |  |  |             return $repr; | 
					
						
							|  |  |  |         } else if (count($hcards) > 0) { | 
					
						
							|  |  |  |             return $hcards[0]; | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  |             return null; | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  |     function _getChildrenByClass($el, $cls, $xp) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // borrowed from hkit. Thanks dudes!
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $qry = ".//*[contains(concat(' ',normalize-space(@class),' '),' $cls ')]"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $nodes = $xp->query($qry, $el); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $nodes; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function _hcardFromNode($hcardNode, $xp, $base) | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  |         $hcard = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $hcard['url'] = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $urlNodes = self::_getChildrenByClass($hcardNode, 'url', $xp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for ($j = 0; $j < $urlNodes->length; $j++) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $urlNode = $urlNodes->item($j); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if ($urlNode->hasAttribute('href')) { | 
					
						
							|  |  |  |                 $url = $urlNode->getAttribute('href'); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 $url = $urlNode->textContent; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $hcard['url'][] = self::_rel2abs($url, $base); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $hcard['photo'] = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $photoNodes = self::_getChildrenByClass($hcardNode, 'photo', $xp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for ($j = 0; $j < $photoNodes->length; $j++) { | 
					
						
							|  |  |  |             $photoNode = $photoNodes->item($j); | 
					
						
							|  |  |  |             if ($photoNode->hasAttribute('src')) { | 
					
						
							|  |  |  |                 $url = $photoNode->getAttribute('src'); | 
					
						
							|  |  |  |             } else if ($photoNode->hasAttribute('href')) { | 
					
						
							|  |  |  |                 $url = $photoNode->getAttribute('href'); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 $url = $photoNode->textContent; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             $hcard['photo'][] = self::_rel2abs($url, $base); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $singles = array('nickname', 'note', 'fn', 'n', 'adr'); | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  |         foreach ($singles as $single) { | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  |             $nodes = self::_getChildrenByClass($hcardNode, $single, $xp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if ($nodes->length > 0) { | 
					
						
							|  |  |  |                 $node = $nodes->item(0); | 
					
						
							|  |  |  |                 $hcard[$single] = $node->textContent; | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  |         return $hcard; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // XXX: this is a first pass; we probably need
 | 
					
						
							|  |  |  |     // to handle things like ../ and ./ and so on
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static function _rel2abs($rel, $wrt) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $parts = parse_url($rel); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($parts === false) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // If it's got a scheme, use it
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-19 15:54:16 -07:00
										 |  |  |         if (!empty($parts['scheme'])) { | 
					
						
							| 
									
										
										
										
											2010-03-18 20:52:00 -05:00
										 |  |  |             return $rel; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $w = parse_url($wrt); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $base = $w['scheme'].'://'.$w['host']; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($rel[0] == '/') { | 
					
						
							|  |  |  |             return $base.$rel; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $wp = explode('/', $w['path']); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         array_pop($wp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $base.implode('/', $wp).'/'.$rel; | 
					
						
							| 
									
										
										
										
											2010-03-16 11:25:18 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | } |