| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  | <?php | 
					
						
							|  |  |  | /* | 
					
						
							|  |  |  |  * GNU Social - a federating social network | 
					
						
							|  |  |  |  * Copyright (C) 2013, Free Software Foundation, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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/>. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Implements WebFinger for GNU Social, as well as support for the | 
					
						
							|  |  |  |  * '.well-known/host-meta' resource. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Depends on: LRDD plugin | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2013-10-15 00:20:36 +02:00
										 |  |  |  * @package GNUsocial | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  |  * @author  Mikael Nordfeldth <mmn@hethane.se> | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if (!defined('GNUSOCIAL')) { exit(1); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class WebFingerPlugin extends Plugin | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-06-04 18:54:09 +02:00
										 |  |  |     const OAUTH_ACCESS_TOKEN_REL    = 'http://apinamespace.org/oauth/access_token'; | 
					
						
							|  |  |  |     const OAUTH_REQUEST_TOKEN_REL   = 'http://apinamespace.org/oauth/request_token'; | 
					
						
							|  |  |  |     const OAUTH_AUTHORIZE_REL       = 'http://apinamespace.org/oauth/authorize'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  |     public function onRouterInitialized($m) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $m->connect('.well-known/host-meta', array('action' => 'hostmeta')); | 
					
						
							|  |  |  |         $m->connect('.well-known/host-meta.:format', | 
					
						
							|  |  |  |                         array('action' => 'hostmeta', | 
					
						
							|  |  |  |                               'format' => '(xml|json)')); | 
					
						
							|  |  |  |         // the resource GET parameter can be anywhere, so don't mention it here
 | 
					
						
							|  |  |  |         $m->connect('.well-known/webfinger', array('action' => 'webfinger')); | 
					
						
							|  |  |  |         $m->connect('.well-known/webfinger.:format', | 
					
						
							|  |  |  |                         array('action' => 'webfinger', | 
					
						
							|  |  |  |                               'format' => '(xml|json)')); | 
					
						
							|  |  |  |         $m->connect('main/ownerxrd', array('action' => 'ownerxrd')); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public function onLoginAction($action, &$login) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         switch ($action) { | 
					
						
							|  |  |  |         case 'hostmeta': | 
					
						
							|  |  |  |         case 'webfinger': | 
					
						
							|  |  |  |             $login = true; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-28 18:26:00 +01:00
										 |  |  |     public function onStartGetProfileAcctUri(Profile $profile, &$acct) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $wfr = new WebFingerResource_Profile($profile); | 
					
						
							|  |  |  |         try { | 
					
						
							|  |  |  |             $acct = $wfr->reconstructAcct(); | 
					
						
							|  |  |  |         } catch (Exception $e) { | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-20 15:32:56 +02:00
										 |  |  |     public function onEndGetWebFingerResource($resource, WebFingerResource &$target=null, array $args=array()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $profile = null; | 
					
						
							|  |  |  |         if (Discovery::isAcct($resource)) { | 
					
						
							|  |  |  |             $parts = explode('@', substr(urldecode($resource), 5)); // 5 is strlen of 'acct:'
 | 
					
						
							|  |  |  |             if (count($parts) == 2) { | 
					
						
							|  |  |  |                 list($nick, $domain) = $parts; | 
					
						
							|  |  |  |                 if ($domain === common_config('site', 'server')) { | 
					
						
							|  |  |  |                     $nick = common_canonical_nickname($nick); | 
					
						
							|  |  |  |                     $user = User::getKV('nickname', $nick); | 
					
						
							|  |  |  |                     if (!($user instanceof User)) { | 
					
						
							|  |  |  |                         throw new NoSuchUserException(array('nickname'=>$nick)); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     $profile = $user->getProfile(); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     throw new Exception(_('Remote profiles not supported via WebFinger yet.')); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2016-02-21 18:48:48 +01:00
										 |  |  |             try { | 
					
						
							|  |  |  |                 $user = User::getByUri($resource); | 
					
						
							| 
									
										
										
										
											2013-10-20 15:32:56 +02:00
										 |  |  |                 $profile = $user->getProfile(); | 
					
						
							| 
									
										
										
										
											2016-02-21 18:48:48 +01:00
										 |  |  |             } catch (NoResultException $e) { | 
					
						
							| 
									
										
										
										
											2016-02-23 14:00:59 +01:00
										 |  |  |                 if (common_config('fix', 'fancyurls')) { | 
					
						
							| 
									
										
										
										
											2016-02-21 20:05:32 +01:00
										 |  |  |                     try { | 
					
						
							|  |  |  |                         try {   // if it's a /index.php/ url
 | 
					
						
							|  |  |  |                             // common_fake_local_fancy_url can throw an exception
 | 
					
						
							|  |  |  |                             $alt_url = common_fake_local_fancy_url($resource); | 
					
						
							|  |  |  |                         } catch (Exception $e) {    // let's try to create a fake local /index.php/ url
 | 
					
						
							|  |  |  |                             // this too if it can't do anything about the URL
 | 
					
						
							|  |  |  |                             $alt_url = common_fake_local_nonfancy_url($resource); | 
					
						
							|  |  |  |                         } | 
					
						
							| 
									
										
										
										
											2016-02-21 18:48:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-21 20:05:32 +01:00
										 |  |  |                         // and this will throw a NoResultException if not found
 | 
					
						
							|  |  |  |                         $user = User::getByUri($alt_url); | 
					
						
							|  |  |  |                         $profile = $user->getProfile(); | 
					
						
							|  |  |  |                     } catch (Exception $e) { | 
					
						
							|  |  |  |                         // apparently we didn't get any matches with that, so continue...
 | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2016-02-21 18:48:48 +01:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2013-10-20 15:32:56 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-21 20:05:32 +01:00
										 |  |  |         // if we still haven't found a match...
 | 
					
						
							|  |  |  |         if (!$profile instanceof Profile) { | 
					
						
							|  |  |  |             // if our rewrite hack didn't work, try to get something by profile URL
 | 
					
						
							|  |  |  |             $profile = Profile::getKV('profileurl', $resource); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-20 15:32:56 +02:00
										 |  |  |         if ($profile instanceof Profile) { | 
					
						
							|  |  |  |             $target = new WebFingerResource_Profile($profile); | 
					
						
							|  |  |  |             return false;   // We got our target, stop handler execution
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $notice = Notice::getKV('uri', $resource); | 
					
						
							|  |  |  |         if ($notice instanceof Notice) { | 
					
						
							|  |  |  |             $target = new WebFingerResource_Notice($notice); | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  |     public function onStartHostMetaLinks(array &$links) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         foreach (Discovery::supportedMimeTypes() as $type) { | 
					
						
							|  |  |  |             $links[] = new XML_XRD_Element_Link(Discovery::LRDD_REL, | 
					
						
							|  |  |  |                             common_local_url('webfinger') . '?resource={uri}', | 
					
						
							|  |  |  |                             $type, | 
					
						
							|  |  |  |                             true);    // isTemplate
 | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-06-04 18:54:09 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // OAuth connections
 | 
					
						
							|  |  |  |         $links[] = new XML_XRD_Element_link(self::OAUTH_ACCESS_TOKEN_REL,  common_local_url('ApiOAuthAccessToken')); | 
					
						
							|  |  |  |         $links[] = new XML_XRD_Element_link(self::OAUTH_REQUEST_TOKEN_REL, common_local_url('ApiOAuthRequestToken')); | 
					
						
							|  |  |  |         $links[] = new XML_XRD_Element_link(self::OAUTH_AUTHORIZE_REL,     common_local_url('ApiOAuthAuthorize')); | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Add a link header for LRDD Discovery | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function onStartShowHTML($action) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if ($action instanceof ShowstreamAction) { | 
					
						
							| 
									
										
										
										
											2016-02-17 22:36:33 +01:00
										 |  |  |             $resource = $action->getTarget()->getUri(); | 
					
						
							|  |  |  |             $url = common_local_url('webfinger') . '?resource='.urlencode($resource); | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             foreach (array(Discovery::JRD_MIMETYPE, Discovery::XRD_MIMETYPE) as $type) { | 
					
						
							| 
									
										
										
										
											2015-10-25 18:42:37 +00:00
										 |  |  |                 header('Link: <'.$url.'>; rel="'. Discovery::LRDD_REL.'"; type="'.$type.'"', false); | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-06 22:04:01 +02:00
										 |  |  |     public function onPluginVersion(array &$versions) | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         $versions[] = array('name' => 'WebFinger', | 
					
						
							| 
									
										
										
										
											2013-11-01 13:51:41 +01:00
										 |  |  |                             'version' => GNUSOCIAL_VERSION, | 
					
						
							| 
									
										
										
										
											2013-09-30 17:13:03 +02:00
										 |  |  |                             'author' => 'Mikael Nordfeldth', | 
					
						
							|  |  |  |                             'homepage' => 'http://www.gnu.org/software/social/', | 
					
						
							|  |  |  |                             // TRANS: Plugin description.
 | 
					
						
							|  |  |  |                             'rawdescription' => _m('Adds WebFinger lookup to GNU Social')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |