diff --git a/module/CLI/src/Command/Api/GenerateKeyCommand.php b/module/CLI/src/Command/Api/GenerateKeyCommand.php index 0a35bef70..a7656189f 100644 --- a/module/CLI/src/Command/Api/GenerateKeyCommand.php +++ b/module/CLI/src/Command/Api/GenerateKeyCommand.php @@ -109,7 +109,7 @@ protected function execute(InputInterface $input, OutputInterface $output): int )); $io = new SymfonyStyle($input, $output); - $io->success(sprintf('Generated API key: "%s"', $apiKey->toString())); + $io->success(sprintf('Generated API key: "%s"', $apiKey->key)); if (! ApiKey::isAdmin($apiKey)) { ShlinkTable::default($io)->render( diff --git a/module/CLI/src/Command/Api/ListKeysCommand.php b/module/CLI/src/Command/Api/ListKeysCommand.php index 40ae8eef8..ab10ebc6a 100644 --- a/module/CLI/src/Command/Api/ListKeysCommand.php +++ b/module/CLI/src/Command/Api/ListKeysCommand.php @@ -54,7 +54,7 @@ protected function execute(InputInterface $input, OutputInterface $output): int $messagePattern = $this->determineMessagePattern($apiKey); // Set columns for this row - $rowData = [sprintf($messagePattern, $apiKey), sprintf($messagePattern, $apiKey->name ?? '-')]; + $rowData = [sprintf($messagePattern, $apiKey->key), sprintf($messagePattern, $apiKey->name ?? '-')]; if (! $enabledOnly) { $rowData[] = sprintf($messagePattern, $this->getEnabledSymbol($apiKey)); } diff --git a/module/CLI/src/Command/ShortUrl/ListShortUrlsCommand.php b/module/CLI/src/Command/ShortUrl/ListShortUrlsCommand.php index 72a92fe85..900402e1e 100644 --- a/module/CLI/src/Command/ShortUrl/ListShortUrlsCommand.php +++ b/module/CLI/src/Command/ShortUrl/ListShortUrlsCommand.php @@ -244,7 +244,7 @@ private function resolveColumnsMap(InputInterface $input): array } if ($input->getOption('show-api-key')) { $columnsMap['API Key'] = static fn (array $_, ShortUrl $shortUrl): string => - $shortUrl->authorApiKey?->__toString() ?? ''; + $shortUrl->authorApiKey?->key ?? ''; } if ($input->getOption('show-api-key-name')) { $columnsMap['API Key Name'] = static fn (array $_, ShortUrl $shortUrl): string|null => diff --git a/module/CLI/test/Command/Api/ListKeysCommandTest.php b/module/CLI/test/Command/Api/ListKeysCommandTest.php index 478dbaa52..6f4b816bb 100644 --- a/module/CLI/test/Command/Api/ListKeysCommandTest.php +++ b/module/CLI/test/Command/Api/ListKeysCommandTest.php @@ -55,11 +55,11 @@ public static function provideKeysAndOutputs(): iterable +--------------------------------------+------+------------+---------------------------+-------+ | Key | Name | Is enabled | Expiration date | Roles | +--------------------------------------+------+------------+---------------------------+-------+ - | {$apiKey1} | - | --- | - | Admin | + | {$apiKey1->key} | - | --- | - | Admin | +--------------------------------------+------+------------+---------------------------+-------+ - | {$apiKey2} | - | --- | 2020-01-01T00:00:00+00:00 | Admin | + | {$apiKey2->key} | - | --- | 2020-01-01T00:00:00+00:00 | Admin | +--------------------------------------+------+------------+---------------------------+-------+ - | {$apiKey3} | - | +++ | - | Admin | + | {$apiKey3->key} | - | +++ | - | Admin | +--------------------------------------+------+------------+---------------------------+-------+ OUTPUT, @@ -71,9 +71,9 @@ public static function provideKeysAndOutputs(): iterable +--------------------------------------+------+-----------------+-------+ | Key | Name | Expiration date | Roles | +--------------------------------------+------+-----------------+-------+ - | {$apiKey1} | - | - | Admin | + | {$apiKey1->key} | - | - | Admin | +--------------------------------------+------+-----------------+-------+ - | {$apiKey2} | - | - | Admin | + | {$apiKey2->key} | - | - | Admin | +--------------------------------------+------+-----------------+-------+ OUTPUT, @@ -97,18 +97,18 @@ public static function provideKeysAndOutputs(): iterable +--------------------------------------+------+-----------------+--------------------------+ | Key | Name | Expiration date | Roles | +--------------------------------------+------+-----------------+--------------------------+ - | {$apiKey1} | - | - | Admin | + | {$apiKey1->key} | - | - | Admin | +--------------------------------------+------+-----------------+--------------------------+ - | {$apiKey2} | - | - | Author only | + | {$apiKey2->key} | - | - | Author only | +--------------------------------------+------+-----------------+--------------------------+ - | {$apiKey3} | - | - | Domain only: example.com | + | {$apiKey3->key} | - | - | Domain only: example.com | +--------------------------------------+------+-----------------+--------------------------+ - | {$apiKey4} | - | - | Admin | + | {$apiKey4->key} | - | - | Admin | +--------------------------------------+------+-----------------+--------------------------+ - | {$apiKey5} | - | - | Author only | + | {$apiKey5->key} | - | - | Author only | | | | | Domain only: example.com | +--------------------------------------+------+-----------------+--------------------------+ - | {$apiKey6} | - | - | Admin | + | {$apiKey6->key} | - | - | Admin | +--------------------------------------+------+-----------------+--------------------------+ OUTPUT, @@ -125,13 +125,13 @@ public static function provideKeysAndOutputs(): iterable +--------------------------------------+---------------+-----------------+-------+ | Key | Name | Expiration date | Roles | +--------------------------------------+---------------+-----------------+-------+ - | {$apiKey1} | Alice | - | Admin | + | {$apiKey1->key} | Alice | - | Admin | +--------------------------------------+---------------+-----------------+-------+ - | {$apiKey2} | Alice and Bob | - | Admin | + | {$apiKey2->key} | Alice and Bob | - | Admin | +--------------------------------------+---------------+-----------------+-------+ - | {$apiKey3} | | - | Admin | + | {$apiKey3->key} | | - | Admin | +--------------------------------------+---------------+-----------------+-------+ - | {$apiKey4} | - | - | Admin | + | {$apiKey4->key} | - | - | Admin | +--------------------------------------+---------------+-----------------+-------+ OUTPUT, diff --git a/module/CLI/test/Command/ShortUrl/ListShortUrlsCommandTest.php b/module/CLI/test/Command/ShortUrl/ListShortUrlsCommandTest.php index 41b3fe88d..3b84d175c 100644 --- a/module/CLI/test/Command/ShortUrl/ListShortUrlsCommandTest.php +++ b/module/CLI/test/Command/ShortUrl/ListShortUrlsCommandTest.php @@ -140,7 +140,7 @@ public function provideOptionalFlagsMakesNewColumnsToBeIncluded( public static function provideOptionalFlags(): iterable { $apiKey = ApiKey::fromMeta(ApiKeyMeta::fromParams(name: 'my api key')); - $key = $apiKey->toString(); + $key = $apiKey->key; yield 'tags only' => [ ['--show-tags' => true], diff --git a/module/Rest/src/ApiKey/Model/ApiKeyMeta.php b/module/Rest/src/ApiKey/Model/ApiKeyMeta.php index 020c1f27d..04b372147 100644 --- a/module/Rest/src/ApiKey/Model/ApiKeyMeta.php +++ b/module/Rest/src/ApiKey/Model/ApiKeyMeta.php @@ -7,16 +7,16 @@ use Cake\Chronos\Chronos; use Ramsey\Uuid\Uuid; -final class ApiKeyMeta +final readonly class ApiKeyMeta { /** * @param iterable $roleDefinitions */ private function __construct( - public readonly string $key, - public readonly string|null $name, - public readonly Chronos|null $expirationDate, - public readonly iterable $roleDefinitions, + public string $key, + public string|null $name, + public Chronos|null $expirationDate, + public iterable $roleDefinitions, ) { } diff --git a/module/Rest/src/Entity/ApiKey.php b/module/Rest/src/Entity/ApiKey.php index 1cca4f3aa..4f7575c51 100644 --- a/module/Rest/src/Entity/ApiKey.php +++ b/module/Rest/src/Entity/ApiKey.php @@ -19,10 +19,9 @@ class ApiKey extends AbstractEntity { /** * @param Collection $roles - * @throws Exception */ private function __construct( - private string $key, + public readonly string $key, public readonly string|null $name = null, public readonly Chronos|null $expirationDate = null, private bool $enabled = true, @@ -75,16 +74,6 @@ public function isValid(): bool return $this->isEnabled() && ! $this->isExpired(); } - public function __toString(): string - { - return $this->key; - } - - public function toString(): string - { - return $this->key; - } - public function spec(string|null $context = null): Specification { $specs = $this->roles->map(fn (ApiKeyRole $role) => Role::toSpec($role, $context))->getValues(); diff --git a/module/Rest/test-api/Fixtures/ApiKeyFixture.php b/module/Rest/test-api/Fixtures/ApiKeyFixture.php index 1b4f64eaa..c734e342a 100644 --- a/module/Rest/test-api/Fixtures/ApiKeyFixture.php +++ b/module/Rest/test-api/Fixtures/ApiKeyFixture.php @@ -8,7 +8,6 @@ use Doctrine\Common\DataFixtures\AbstractFixture; use Doctrine\Common\DataFixtures\DependentFixtureInterface; use Doctrine\Persistence\ObjectManager; -use ReflectionObject; use Shlinkio\Shlink\Core\Domain\Entity\Domain; use Shlinkio\Shlink\Rest\ApiKey\Model\ApiKeyMeta; use Shlinkio\Shlink\Rest\ApiKey\Model\RoleDefinition; @@ -51,12 +50,7 @@ public function load(ObjectManager $manager): void private function buildApiKey(string $key, bool $enabled, Chronos|null $expiresAt = null): ApiKey { - $apiKey = ApiKey::fromMeta(ApiKeyMeta::fromParams(expirationDate: $expiresAt)); - $ref = new ReflectionObject($apiKey); - $keyProp = $ref->getProperty('key'); - $keyProp->setAccessible(true); - $keyProp->setValue($apiKey, $key); - + $apiKey = ApiKey::fromMeta(ApiKeyMeta::fromParams(key: $key, expirationDate: $expiresAt)); if (! $enabled) { $apiKey->disable(); } diff --git a/module/Rest/test/Middleware/AuthenticationMiddlewareTest.php b/module/Rest/test/Middleware/AuthenticationMiddlewareTest.php index 99a8b3e64..5c5304802 100644 --- a/module/Rest/test/Middleware/AuthenticationMiddlewareTest.php +++ b/module/Rest/test/Middleware/AuthenticationMiddlewareTest.php @@ -130,18 +130,17 @@ public function throwsExceptionWhenProvidedApiKeyIsInvalid(): void public function validApiKeyFallsBackToNextMiddleware(): void { $apiKey = ApiKey::create(); - $key = $apiKey->toString(); $request = ServerRequestFactory::fromGlobals() ->withAttribute( RouteResult::class, RouteResult::fromRoute(new Route('bar', self::getDummyMiddleware()), []), ) - ->withHeader('X-Api-Key', $key); + ->withHeader('X-Api-Key', $apiKey->key); $this->handler->expects($this->once())->method('handle')->with( $request->withAttribute(ApiKey::class, $apiKey), )->willReturn(new Response()); - $this->apiKeyService->expects($this->once())->method('check')->with($key)->willReturn( + $this->apiKeyService->expects($this->once())->method('check')->with($apiKey->key)->willReturn( new ApiKeyCheckResult($apiKey), );