* * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ declare(strict_types=1); namespace Symfony\Component\Mercure; use Symfony\Component\HttpClient\HttpClient; use Symfony\Component\Mercure\Jwt\TokenFactoryInterface; use Symfony\Component\Mercure\Jwt\TokenProviderInterface; use Symfony\Contracts\HttpClient\Exception\ExceptionInterface; use Symfony\Contracts\HttpClient\HttpClientInterface; /** * @author Saif Eddin Gmati * * @experimental */ final class Hub implements HubInterface { private $url; private $jwtProvider; private $jwtFactory; private $publicUrl; private $httpClient; public function __construct( string $url, TokenProviderInterface $jwtProvider, TokenFactoryInterface $jwtFactory = null, string $publicUrl = null, HttpClientInterface $httpClient = null ) { $this->url = $url; $this->jwtProvider = $jwtProvider; $this->publicUrl = $publicUrl; $this->jwtFactory = $jwtFactory; $this->httpClient = $httpClient ?? HttpClient::create(); } /** * {@inheritDoc} */ public function getUrl(): string { return $this->url; } /** * {@inheritDoc} */ public function getPublicUrl(): string { return $this->publicUrl ?? $this->getUrl(); } /** * {@inheritDoc} */ public function getProvider(): TokenProviderInterface { return $this->jwtProvider; } /** * {@inheritDoc} */ public function getFactory(): ?TokenFactoryInterface { return $this->jwtFactory; } /** * {@inheritDoc} */ public function publish(Update $update): string { $postData = [ 'topic' => $update->getTopics(), 'data' => $update->getData(), 'private' => $update->isPrivate() ? 'on' : null, 'id' => $update->getId(), 'type' => $update->getType(), 'retry' => $update->getRetry(), ]; $jwt = $this->getProvider()->getJwt(); $this->validateJwt($jwt); try { return $this->httpClient->request('POST', $this->getUrl(), [ 'auth_bearer' => $jwt, 'headers' => [ 'Content-Type' => 'application/x-www-form-urlencoded', ], 'body' => Internal\QueryBuilder::build($postData), ])->getContent(); } catch (ExceptionInterface $exception) { throw new Exception\RuntimeException('Failed to send an update.', 0, $exception); } } /** * Regex ported from Windows Azure Active Directory IdentityModel Extensions for .Net. * * @throws Exception\InvalidArgumentException * * @license MIT * @copyright Copyright (c) Microsoft Corporation * * @see https://github.com/AzureAD/azure-activedirectory-identitymodel-extensions-for-dotnet/blob/6e7a53e241e4566998d3bf365f03acd0da699a31/src/Microsoft.IdentityModel.JsonWebTokens/JwtConstants.cs#L58 */ private function validateJwt(string $jwt): void { if (!preg_match('/^[A-Za-z0-9-_]+\.[A-Za-z0-9-_]+\.[A-Za-z0-9-_]*$/', $jwt)) { throw new Exception\InvalidArgumentException('The provided JWT is not valid.'); } } }