getMyUrl(); $my_url = Network::isUrlValid($my_url); if (empty($my_url) || DI::userSession()->getLocalUserId()) { return; } $addr = $_GET['addr'] ?? $my_url; $arr = ['zrl' => $my_url, 'url' => DI::args()->getCommand()]; Hook::callAll('zrl_init', $arr); // Try to find the public contact entry of the visitor. $contact = Contact::getByURL($my_url, null, ['id', 'url', 'gsid']); if (empty($contact)) { Logger::info('No contact record found', ['url' => $my_url]); return; } if (DI::userSession()->getRemoteUserId() && DI::userSession()->getRemoteUserId() == $contact['id']) { Logger::info('The visitor is already authenticated', ['url' => $my_url]); return; } $gserver = DBA::selectFirst('gserver', ['url', 'authredirect'], ['id' => $contact['gsid']]); if (empty($gserver) || empty($gserver['authredirect'])) { Logger::info('No server record found or magic path not defined for server', ['id' => $contact['gsid'], 'gserver' => $gserver]); return; } // Avoid endless loops $cachekey = 'zrlInit:' . $my_url; if (DI::cache()->get($cachekey)) { Logger::info('URL ' . $my_url . ' already tried to authenticate.'); return; } else { DI::cache()->set($cachekey, true, Duration::MINUTE); } Logger::info('Not authenticated. Invoking reverse magic-auth', ['url' => $my_url]); // Remove the "addr" parameter from the destination. It is later added as separate parameter again. $addr_request = 'addr=' . urlencode($addr); $query = rtrim(str_replace($addr_request, '', DI::args()->getQueryString()), '?&'); // The other instance needs to know where to redirect. $dest = urlencode(DI::baseUrl() . '/' . $query); if ($gserver['url'] != DI::baseUrl() && !strstr($dest, '/magic')) { $magic_path = $gserver['authredirect'] . '?f=&rev=1&owa=1&dest=' . $dest . '&' . $addr_request; Logger::info('Doing magic auth for visitor ' . $my_url . ' to ' . $magic_path); System::externalRedirect($magic_path); } } /** * OpenWebAuth authentication. * * Ported from Hubzilla: https://framagit.org/hubzilla/core/blob/master/include/zid.php * * @param string $token * * @return void * @throws \Friendica\Network\HTTPException\InternalServerErrorException * @throws \ImagickException */ public static function init(string $token) { $appHelper = DI::appHelper(); // Clean old OpenWebAuthToken entries. OpenWebAuthToken::purge('owt', '3 MINUTE'); // Check if the token we got is the same one // we have stored in the database. $visitor_handle = OpenWebAuthToken::getMeta('owt', 0, $token); if ($visitor_handle === false) { return; } $visitor = self::addVisitorCookieForHandle($visitor_handle); if (empty($visitor)) { return; } $arr = [ 'visitor' => $visitor, 'url' => DI::args()->getQueryString() ]; /** * @hooks magic_auth_success * Called when a magic-auth was successful. * * \e array \b visitor * * \e string \b url */ Hook::callAll('magic_auth_success', $arr); $appHelper->setContactId($arr['visitor']['id']); DI::sysmsg()->addInfo(DI::l10n()->t('OpenWebAuth: %1$s welcomes %2$s', DI::baseUrl()->getHost(), $visitor['name'])); Logger::info('OpenWebAuth: auth success from ' . $visitor['addr']); } /** * Set the visitor cookies (see remote_user()) for the given handle * * @param string $handle Visitor handle * * @return array Visitor contact array */ public static function addVisitorCookieForHandle(string $handle): array { $appHelper = DI::appHelper(); // Try to find the public contact entry of the visitor. $cid = Contact::getIdForURL($handle); if (!$cid) { Logger::info('Handle not found', ['handle' => $handle]); return []; } $visitor = Contact::getById($cid); // Authenticate the visitor. DI::userSession()->setMultiple([ 'authenticated' => 1, 'visitor_id' => $visitor['id'], 'visitor_handle' => $visitor['addr'], 'visitor_home' => $visitor['url'], 'my_url' => $visitor['url'], 'remote_comment' => $visitor['subscribe'], ]); DI::userSession()->setVisitorsContacts($visitor['url']); $appHelper->setContactId($visitor['id']); Logger::info('Authenticated visitor', ['url' => $visitor['url']]); return $visitor; } /** * Set the visitor cookies (see remote_user()) for signed HTTP requests * * @param array $server The content of the $_SERVER superglobal * @return array Visitor contact array * @throws InternalServerErrorException */ public static function addVisitorCookieForHTTPSigner(array $server): array { $requester = HTTPSignature::getSigner('', $server); if (empty($requester)) { return []; } return self::addVisitorCookieForHandle($requester); } /** * Returns URL with URL-encoded zrl parameter * * @param string $url URL to enhance * @param bool $force Either to force adding zrl parameter * * @return string URL with 'zrl' parameter or original URL in case of no Friendica profile URL */ public static function getZrlUrl(string $url, bool $force = false): string { if (!strlen($url)) { return $url; } if (!strpos($url, '/profile/') && !$force) { return $url; } if ($force && substr($url, -1, 1) !== '/') { $url = $url . '/'; } $achar = strpos($url, '?') ? '&' : '?'; $mine = DI::userSession()->getMyUrl(); if ($mine && !Strings::compareLink($mine, $url)) { return $url . $achar . 'zrl=' . urlencode($mine); } return $url; } }