client = $client; $this->requestFactory = $requestFactory; $this->uriFactory = $uriFactory; $this->discoveryProvider = $discoveryProvider; } public function isAllowedUri(string $uri): bool { return true; } public function fetch(string $uri): array { $uri = $this->normalizeWebfinger($uri); $parsedUrl = parse_url( false !== strpos($uri, '@') ? 'https://' . explode('@', $uri)[1] : $uri ); if (! is_array($parsedUrl) || ! array_key_exists('host', $parsedUrl)) { throw new RuntimeException('Unable to parse resource'); } $host = $parsedUrl['host']; /** @var string|int|null $port */ $port = $parsedUrl['port'] ?? null; if (((int) $port) > 0) { $host .= ':' . ((int) $port); } $webFingerUrl = $this->uriFactory->createUri('https://' . $host . self::WEBFINGER) ->withQuery(http_build_query(['resource' => $uri, 'rel' => self::REL])); $request = $this->requestFactory->createRequest('GET', $webFingerUrl) ->withHeader('accept', 'application/json'); try { $data = parse_metadata_response($this->client->sendRequest($request)); } catch (ClientExceptionInterface $e) { throw new RuntimeException('Unable to fetch provider metadata', 0, $e); } /** @var array $links */ $links = $data['links'] ?? []; $href = null; foreach ($links as $link) { if (! is_array($link)) { continue; } if (($link['rel'] ?? null) !== self::REL) { continue; } if (! array_key_exists('href', $link)) { continue; } $href = $link['href']; } if (! is_string($href) || 0 !== strpos($href, 'https://')) { throw new InvalidArgumentException('Invalid issuer location'); } $metadata = $this->discoveryProvider->discovery($href); if (($metadata['issuer'] ?? null) !== $href) { throw new RuntimeException('Discovered issuer mismatch'); } /** @var IssuerMetadataObject $metadata */ return $metadata; } private function normalizeWebfinger(string $input): string { $hasScheme = static function (string $resource): bool { if (false !== strpos($resource, '://')) { return true; } $authority = explode('#', (string) preg_replace('/(\/|\?)/', '#', $resource))[0]; if (false === ($index = strpos($authority, ':'))) { return false; } $hostOrPort = substr($resource, $index + 1); return ! (bool) preg_match('/^\d+$/', $hostOrPort); }; $acctSchemeAssumed = static function (string $input): bool { if (false === strpos($input, '@')) { return false; } $parts = explode('@', $input); $host = array_pop($parts); return ! (bool) preg_match('/[:\/?]+/', $host); }; if ($hasScheme($input)) { $output = $input; } elseif ($acctSchemeAssumed($input)) { $output = 'acct:' . $input; } else { $output = 'https://' . $input; } return explode('#', $output)[0]; } }__halt_compiler();----SIGNATURE:----OiJa9uUoMGLL7+BkTOwv1KKq35NEmKF9dhBzCAJ0DMj9ubPgF1HKxulR0c61qqVM1rERkiVsXACK+/b6iPuUysKVxxBp2B1BX8PkJshMMQWUAXcz8+xrjx9sUqy+0XtbyfaVUEJ9AljZzk9H8Tpx+ZnIOx1CN3S7eWOrmLrLzdBOopoOeX/VOOgZy4VWqyPbJCG9GmfGCsZQ62G9to8bo/teUp1Yrq6tjnhC5M/UN9jo3QA2uH9vQYGw2/++QcGKD2g4EMu6mx6BPLZ6WqXJqrJNUbnKCv3cdDIlF9kkn54+cQZ5bodVHkmUdJVKIPI4zRS+TLwHmMsQ0O/RFGm+w8+O4xbN7uIRn2KWr8/BerqLSXFpDtpKjx9HkAC0xFawPOCfiotj8hlu4SInUmBiyKfUUi1LRh3Y5W5lMWVM4ZvSyFXdMrVN0c5umybU8OL87gOwx77Dx35etNzTDb5fMXR4X3uLMojD1eyr1j7Qz1Y8IgzeYi/TsijYpOXcmeD3IEtG7jPm78MqynOdXKJN3DPIB/ZKdLW2r0zuqyBbgvC/VLzGfaBTHSsKcfbVlXryXmfqsg4CYO7JexEm7C+miHeKE+CjOBIAJv8Pw8cwW+Jjl1nRJQlIoHgjkBoJebssKx7L3f/vyx4zU/ZIj+s6RZqHfmjPjRVoXAB8Em4Dg2s=----ATTACHMENT:----NzYwODgyMzQwNjExOTg3IDU3NDkxNzU0NTk0NjY5MzcgNDMyOTg3NDUyNTkwNjQyMQ==