. */ /** * Implements WebFinger for GNU Social, as well as support for the * '.well-known/host-meta' resource. * * Depends on: LRDD plugin * * @package GNUsocial * @author Mikael Nordfeldth */ if (!defined('GNUSOCIAL')) { exit(1); } class WebFingerPlugin extends Plugin { const PLUGIN_VERSION = '2.0.0'; 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'; public function onRouterInitialized(URLMapper $m) { $m->connect('.well-known/host-meta', ['action' => 'hostmeta']); $m->connect('.well-known/host-meta.:format', ['action' => 'hostmeta'], ['format' => '(xml|json)']); // the resource GET parameter can be anywhere, so don't mention it here $m->connect('.well-known/webfinger', ['action' => 'webfinger']); $m->connect('.well-known/webfinger.:format', ['action' => 'webfinger'], ['format' => '(xml|json)']); $m->connect('main/ownerxrd', ['action' => 'ownerxrd']); return true; } public function onLoginAction($action, &$login) { switch ($action) { case 'hostmeta': case 'webfinger': $login = true; return false; } return true; } public function onStartGetProfileAcctUri(Profile $profile, &$acct) { $wfr = new WebFingerResource_Profile($profile); try { $acct = $wfr->reconstructAcct(); } catch (Exception $e) { return true; } return false; } 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')) { throw new Exception(_('Remote profiles not supported via WebFinger yet.')); } $nick = common_canonical_nickname($nick); $user = User::getKV('nickname', $nick); if (!($user instanceof User)) { throw new NoSuchUserException(array('nickname'=>$nick)); } $profile = $user->getProfile(); } } elseif (!common_valid_http_url($resource)) { // If it's not a URL, we can't do our http<->https legacy fix thingie anyway, // so just try the User URI lookup! try { $user = User::getByUri($resource); $profile = $user->getProfile(); } catch (NoResultException $e) { // not a User, maybe a Notice? we'll try that further down... } } else { // this means $resource is a common_valid_http_url (or https) // First build up a set of alternative resource URLs that we can use. $alt_urls = [$resource => true]; if (strtolower(parse_url($resource, PHP_URL_SCHEME)) === 'https' && common_config('fix', 'legacy_http')) { $alt_urls[preg_replace('/^https:/i', 'http:', $resource, 1)] = true; } if (common_config('fix', 'fancyurls')) { foreach (array_keys($alt_urls) as $url) { try { // if it's a /index.php/ url // common_fake_local_fancy_url can throw an exception $alt_url = common_fake_local_fancy_url($url); } 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($url); } $alt_urls[$alt_url] = true; } } common_debug(__METHOD__.': Generated these alternative URLs for various federation fixes: '._ve(array_keys($alt_urls))); try { common_debug(__METHOD__.': Finding User URI for WebFinger lookup on resource=='._ve($resource)); $user = new User(); $user->whereAddIn('uri', array_keys($alt_urls), $user->columnType('uri')); $user->limit(1); if ($user->find(true)) { $profile = $user->getProfile(); } unset($user); } catch (Exception $e) { // Most likely a UserNoProfileException, if it ever happens // and then we need to do some debugging and perhaps fixes. common_log(LOG_ERR, get_class($e).': '._ve($e->getMessage())); throw $e; } try { common_debug(__METHOD__.': Finding User_group URI for WebFinger lookup on resource=='._ve($resource)); $group = new User_group(); $group->whereAddIn('uri', array_keys($alt_urls), $group->columnType('uri')); $group->limit(1); if ($group->find(true)) { $profile = $group->getProfile(); } unset($group); } catch (Exception $e) { common_log(LOG_ERR, get_class($e).': '._ve($e->getMessage())); throw $e; } // User URI did not match, so let's try our alt_urls as Profile URL values if (!$profile instanceof Profile) { common_debug(__METHOD__.': Finding Profile URLs for WebFinger lookup on resource=='._ve($resource)); // if our rewrite hack didn't work, try to get something by profile URL $profile = new Profile(); $profile->whereAddIn('profileurl', array_keys($alt_urls), $profile->columnType('profileurl')); $profile->limit(1); if (!$profile->find(true) || !$profile->isLocal()) { // * Either we didn't find the profile, then we want to make // the $profile variable null for clarity. // * Or we did find it but for a possibly malicious remote // user who might've set their profile URL to a Notice URL // which would've caused a sort of DoS unless we continue // our search here by discarding the remote profile. $profile = null; } } } if ($profile instanceof Profile) { common_debug(__METHOD__.': Found Profile with ID=='._ve($profile->getID()).' for resource=='._ve($resource)); $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; } 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 } // 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')); } /** * Add a link header for LRDD Discovery */ public function onStartShowHTML($action) { if ($action instanceof ShowstreamAction) { $resource = $action->getTarget()->getUri(); $url = common_local_url('webfinger') . '?resource='.urlencode($resource); foreach (array(Discovery::JRD_MIMETYPE, Discovery::XRD_MIMETYPE) as $type) { header('Link: <'.$url.'>; rel="'. Discovery::LRDD_REL.'"; type="'.$type.'"', false); } } } public function onPluginVersion(array &$versions): bool { $versions[] = array('name' => 'WebFinger', 'version' => self::PLUGIN_VERSION, 'author' => 'Mikael Nordfeldth', 'homepage' => GNUSOCIAL_ENGINE_URL, // TRANS: Plugin description. 'rawdescription' => _m('Adds WebFinger lookup to GNU Social')); return true; } }