diff --git a/src/Laravel/ApiPlatformProvider.php b/src/Laravel/ApiPlatformProvider.php index 2338ea4b766..d909169a500 100644 --- a/src/Laravel/ApiPlatformProvider.php +++ b/src/Laravel/ApiPlatformProvider.php @@ -183,6 +183,7 @@ use ApiPlatform\State\Pagination\PaginationOptions; use ApiPlatform\State\ParameterProviderInterface; use ApiPlatform\State\Processor\AddLinkHeaderProcessor; +use ApiPlatform\State\Processor\LinkedDataPlatformProcessor; use ApiPlatform\State\Processor\RespondProcessor; use ApiPlatform\State\Processor\SerializeProcessor; use ApiPlatform\State\Processor\WriteProcessor; @@ -555,11 +556,23 @@ public function register(): void }); $this->app->singleton(RespondProcessor::class, function () { - return new AddLinkHeaderProcessor(new RespondProcessor(), new HttpHeaderSerializer()); + return new RespondProcessor(); + }); + + $this->app->singleton(AddLinkHeaderProcessor::class, function (Application $app) { + return new AddLinkHeaderProcessor($app->make(RespondProcessor::class), new HttpHeaderSerializer()); + }); + + $this->app->singleton(LinkedDataPlatformProcessor::class, function (Application $app) { + return new LinkedDataPlatformProcessor( + $app->make(AddLinkHeaderProcessor::class), // Original service + $app->make(ResourceClassResolverInterface::class), + $app->make(ResourceMetadataCollectionFactoryInterface::class) + ); }); $this->app->singleton(SerializeProcessor::class, function (Application $app) { - return new SerializeProcessor($app->make(RespondProcessor::class), $app->make(Serializer::class), $app->make(SerializerContextBuilderInterface::class)); + return new SerializeProcessor($app->make(LinkedDataPlatformProcessor::class), $app->make(Serializer::class), $app->make(SerializerContextBuilderInterface::class)); }); $this->app->singleton(WriteProcessor::class, function (Application $app) { diff --git a/src/Laravel/Tests/LinkedDataPlatformTest.php b/src/Laravel/Tests/LinkedDataPlatformTest.php new file mode 100644 index 00000000000..f8123f1d9a4 --- /dev/null +++ b/src/Laravel/Tests/LinkedDataPlatformTest.php @@ -0,0 +1,68 @@ +<?php + +/* + * This file is part of the API Platform project. + * + * (c) Kévin Dunglas <dunglas@gmail.com> + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +declare(strict_types=1); + +use ApiPlatform\Laravel\Test\ApiTestAssertionsTrait; +use Illuminate\Contracts\Config\Repository; +use Illuminate\Foundation\Application; +use Illuminate\Foundation\Testing\RefreshDatabase; +use Orchestra\Testbench\Concerns\WithWorkbench; +use Orchestra\Testbench\TestCase; +use Workbench\App\Models\Book; +use Workbench\Database\Factories\BookFactory; + +class LinkedDataPlatformTest extends TestCase +{ + use ApiTestAssertionsTrait; + use RefreshDatabase; + use WithWorkbench; + + /** + * @param Application $app + */ + protected function defineEnvironment($app): void + { + tap($app['config'], function (Repository $config): void { + $config->set('api-platform.formats', ['jsonld' => ['application/ld+json'], 'turtle' => ['text/turtle']]); + $config->set('api-platform.resources', [app_path('Models'), app_path('ApiResource')]); + $config->set('app.debug', true); + }); + } + + public function testHeadersAcceptPostIsReturnWhenPostAllowed(): void + { + $response = $this->get('/api/books', ['accept' => ['application/ld+json']]); + $response->assertStatus(200); + $response->assertHeader('accept-post', 'application/ld+json, text/turtle, text/html'); + } + + public function testHeadersAcceptPostIsNotSetWhenPostIsNotAllowed(): void + { + BookFactory::new()->createOne(); + $book = Book::first(); + $response = $this->get($this->getIriFromResource($book), ['accept' => ['application/ld+json']]); + $response->assertStatus(200); + $response->assertHeaderMissing('accept-post'); + } + + public function testHeaderAllowReflectsResourceAllowedMethods(): void + { + $response = $this->get('/api/books', ['accept' => ['application/ld+json']]); + $response->assertHeader('allow', 'OPTIONS, HEAD, POST, GET'); + + BookFactory::new()->createOne(); + $book = Book::first(); + $response = $this->get($this->getIriFromResource($book), ['accept' => ['application/ld+json']]); + $response->assertStatus(200); + $response->assertHeader('allow', 'OPTIONS, HEAD, PUT, PATCH, GET, DELETE'); + } +} diff --git a/src/State/Processor/LinkedDataPlatformProcessor.php b/src/State/Processor/LinkedDataPlatformProcessor.php new file mode 100644 index 00000000000..a9fddb85a38 --- /dev/null +++ b/src/State/Processor/LinkedDataPlatformProcessor.php @@ -0,0 +1,76 @@ +<?php + +/* + * This file is part of the API Platform project. + * + * (c) Kévin Dunglas <dunglas@gmail.com> + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +declare(strict_types=1); + +namespace ApiPlatform\State\Processor; + +use ApiPlatform\Metadata\Error; +use ApiPlatform\Metadata\HttpOperation; +use ApiPlatform\Metadata\Operation; +use ApiPlatform\Metadata\Resource\Factory\ResourceMetadataCollectionFactoryInterface; +use ApiPlatform\Metadata\ResourceClassResolverInterface; +use ApiPlatform\State\ProcessorInterface; +use Symfony\Component\HttpFoundation\Response; + +/** + * @template T1 + * @template T2 + * + * @implements ProcessorInterface<T1, T2> + */ +final class LinkedDataPlatformProcessor implements ProcessorInterface +{ + private const DEFAULT_ALLOWED_METHODS = ['OPTIONS', 'HEAD']; + + /** + * @param ProcessorInterface<T1, T2> $decorated + */ + public function __construct( + private readonly ProcessorInterface $decorated, + private readonly ?ResourceClassResolverInterface $resourceClassResolver = null, + private readonly ?ResourceMetadataCollectionFactoryInterface $resourceMetadataCollectionFactory = null, + ) { + } + + public function process(mixed $data, Operation $operation, array $uriVariables = [], array $context = []): mixed + { + $response = $this->decorated->process($data, $operation, $uriVariables, $context); + if ( + !$response instanceof Response + || !$operation instanceof HttpOperation + || $operation instanceof Error + || !$this->resourceMetadataCollectionFactory + || !($context['resource_class'] ?? null) + || !$operation->getUriTemplate() + || !$this->resourceClassResolver?->isResourceClass($context['resource_class']) + ) { + return $response; + } + + $allowedMethods = self::DEFAULT_ALLOWED_METHODS; + $resourceCollection = $this->resourceMetadataCollectionFactory->create($context['resource_class']); + foreach ($resourceCollection as $resource) { + foreach ($resource->getOperations() as $op) { + if ($op->getUriTemplate() === $operation->getUriTemplate()) { + $allowedMethods[] = $method = $op->getMethod(); + if ('POST' === $method && \is_array($outputFormats = $op->getOutputFormats())) { + $response->headers->set('Accept-Post', implode(', ', array_merge(...array_values($outputFormats)))); + } + } + } + } + + $response->headers->set('Allow', implode(', ', $allowedMethods)); + + return $response; + } +} diff --git a/src/State/Tests/Processor/LinkedDataPlatformProcessorTest.php b/src/State/Tests/Processor/LinkedDataPlatformProcessorTest.php new file mode 100644 index 00000000000..8018b8ba23d --- /dev/null +++ b/src/State/Tests/Processor/LinkedDataPlatformProcessorTest.php @@ -0,0 +1,183 @@ +<?php + +/* + * This file is part of the API Platform project. + * + * (c) Kévin Dunglas <dunglas@gmail.com> + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +declare(strict_types=1); + +namespace ApiPlatform\State\Tests\Processor; + +use ApiPlatform\Metadata\ApiResource; +use ApiPlatform\Metadata\Delete; +use ApiPlatform\Metadata\Error; +use ApiPlatform\Metadata\Get; +use ApiPlatform\Metadata\GetCollection; +use ApiPlatform\Metadata\HttpOperation; +use ApiPlatform\Metadata\Post; +use ApiPlatform\Metadata\Put; +use ApiPlatform\Metadata\Resource\Factory\ResourceMetadataCollectionFactoryInterface; +use ApiPlatform\Metadata\Resource\ResourceMetadataCollection; +use ApiPlatform\Metadata\ResourceClassResolverInterface; +use ApiPlatform\State\Processor\LinkedDataPlatformProcessor; +use ApiPlatform\State\ProcessorInterface; +use PHPUnit\Framework\MockObject\MockObject; +use PHPUnit\Framework\TestCase; +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\HttpFoundation\Response; + +class LinkedDataPlatformProcessorTest extends TestCase +{ + private ResourceMetadataCollectionFactoryInterface&MockObject $resourceMetadataCollectionFactory; + private ResourceClassResolverInterface&MockObject $resourceClassResolver; + + private ProcessorInterface&MockObject $decorated; + + protected function setUp(): void + { + $this->resourceClassResolver = $this->createMock(ResourceClassResolverInterface::class); + $this->resourceClassResolver + ->method('isResourceClass') + ->willReturn(true); + + $this->resourceMetadataCollectionFactory = $this->createMock(ResourceMetadataCollectionFactoryInterface::class); + $this->resourceMetadataCollectionFactory + ->method('create') + ->willReturn( + new ResourceMetadataCollection('DummyResource', [ // todo mock $dummy_resource + new ApiResource(operations: [ + new Get(uriTemplate: '/dummy_resources/{dummyResourceId}{._format}', name: 'get'), + new GetCollection(uriTemplate: '/dummy_resources{._format}', name: 'get_collections'), + new Post(uriTemplate: '/dummy_resources{._format}', outputFormats: ['jsonld' => ['application/ld+json'], 'text/turtle' => ['text/turtle']], name: 'post'), + new Delete(uriTemplate: '/dummy_resources/{dummyResourceId}{._format}', name: 'delete'), + new Put(uriTemplate: '/dummy_resources/{dummyResourceId}{._format}', name: 'put'), + ]), + ]) + ); + + $this->decorated = $this->createMock(ProcessorInterface::class); + $this->decorated->method('process')->willReturn(new Response()); + } + + public function testHeadersAcceptPostIsReturnWhenPostAllowed(): void + { + $operation = (new HttpOperation('GET', '/dummy_resources{._format}')); + + $context = $this->getContext(); + + $processor = new LinkedDataPlatformProcessor( + $this->decorated, + $this->resourceClassResolver, + $this->resourceMetadataCollectionFactory + ); + /** @var Response $response */ + $response = $processor->process(null, $operation, [], $context); + + $this->assertSame('application/ld+json, text/turtle', $response->headers->get('Accept-Post')); + } + + public function testHeadersAcceptPostIsNotSetWhenPostIsNotAllowed(): void + { + $operation = (new HttpOperation('GET', '/dummy_resources/{dummyResourceId}{._format}')); + $context = $this->getContext(); + + $processor = new LinkedDataPlatformProcessor( + $this->decorated, + $this->resourceClassResolver, + $this->resourceMetadataCollectionFactory + ); + /** @var Response $response */ + $response = $processor->process(null, $operation, [], $context); + + $this->assertNull($response->headers->get('Accept-Post')); + } + + public function testHeaderAllowReflectsResourceAllowedMethods(): void + { + $operation = (new HttpOperation('GET', '/dummy_resources{._format}')); + $context = $this->getContext(); + + $processor = new LinkedDataPlatformProcessor( + $this->decorated, + $this->resourceClassResolver, + $this->resourceMetadataCollectionFactory + ); + /** @var Response $response */ + $response = $processor->process(null, $operation, [], $context); + $allowHeader = $response->headers->get('Allow'); + $this->assertStringContainsString('OPTIONS', $allowHeader); + $this->assertStringContainsString('HEAD', $allowHeader); + $this->assertStringContainsString('GET', $allowHeader); + $this->assertStringContainsString('POST', $allowHeader); + + $operation = (new HttpOperation('GET', '/dummy_resources/{dummyResourceId}{._format}')); + + /** @var Response $response */ + $processor = new LinkedDataPlatformProcessor( + $this->decorated, + $this->resourceClassResolver, + $this->resourceMetadataCollectionFactory + ); + /** @var Response $response */ + $response = $processor->process('data', $operation, [], $this->getContext()); + $allowHeader = $response->headers->get('Allow'); + $this->assertStringContainsString('OPTIONS', $allowHeader); + $this->assertStringContainsString('HEAD', $allowHeader); + $this->assertStringContainsString('GET', $allowHeader); + $this->assertStringContainsString('PUT', $allowHeader); + $this->assertStringContainsString('DELETE', $allowHeader); + } + + public function testProcessorWithoutRequiredConditionReturnOriginalResponse(): void + { + $operation = (new HttpOperation('GET', '/dummy_resources/{dummyResourceId}{._format}')); + + // No collection factory + $processor = new LinkedDataPlatformProcessor($this->decorated, $this->resourceClassResolver, null); + /** @var Response $response */ + $response = $processor->process(null, $operation, [], $this->getContext()); + + $this->assertNull($response->headers->get('Allow')); + + // No uri variable in context + $processor = new LinkedDataPlatformProcessor($this->decorated, null, $this->resourceMetadataCollectionFactory); + $response = $processor->process(null, $operation, [], $this->getContext()); + $this->assertNull($response->headers->get('Allow')); + + // Operation is an Error + $processor = new LinkedDataPlatformProcessor($this->decorated, $this->resourceClassResolver, $this->resourceMetadataCollectionFactory); + $response = $processor->process(null, new Error(), $this->getContext()); + $this->assertNull($response->headers->get('Allow')); + + // Not a resource class + $this->resourceClassResolver + ->method('isResourceClass') + ->willReturn(false); + $processor = new LinkedDataPlatformProcessor($this->decorated, $this->resourceClassResolver, $this->resourceMetadataCollectionFactory); + $response = $processor->process(null, $operation, []); + $this->assertNull($response->headers->get('Allow')); + } + + private function createGetRequest(): Request + { + $request = new Request(); + $request->setMethod('GET'); + $request->setRequestFormat('json'); + $request->headers->set('Accept', 'application/ld+json'); + + return $request; + } + + private function getContext(): array + { + return [ + 'resource_class' => 'DummyResource', + 'request' => $this->createGetRequest(), + ]; + } +} diff --git a/src/Symfony/Bundle/Resources/config/state/processor.xml b/src/Symfony/Bundle/Resources/config/state/processor.xml index 627d3742957..b448f4c3cf6 100644 --- a/src/Symfony/Bundle/Resources/config/state/processor.xml +++ b/src/Symfony/Bundle/Resources/config/state/processor.xml @@ -26,5 +26,11 @@ <service id="api_platform.state_processor.add_link_header" class="ApiPlatform\State\Processor\AddLinkHeaderProcessor" decorates="api_platform.state_processor.respond"> <argument type="service" id="api_platform.state_processor.add_link_header.inner" /> </service> + + <service id="api_platform.state_processor.linked_data_platform" class="ApiPlatform\State\Processor\LinkedDataPlatformProcessor" decorates="api_platform.state_processor.respond"> + <argument type="service" id="api_platform.state_processor.linked_data_platform.inner" /> + <argument type="service" id="api_platform.resource_class_resolver" /> + <argument type="service" id="api_platform.metadata.resource.metadata_collection_factory" /> + </service> </services> </container> diff --git a/src/Symfony/Bundle/Resources/config/symfony/events.xml b/src/Symfony/Bundle/Resources/config/symfony/events.xml index 4ecdb892921..2ee684128f7 100644 --- a/src/Symfony/Bundle/Resources/config/symfony/events.xml +++ b/src/Symfony/Bundle/Resources/config/symfony/events.xml @@ -67,6 +67,12 @@ <argument type="service" id="api_platform.state_processor.add_link_header.inner" /> </service> + <service id="api_platform.state_processor.linked_data_platform" class="ApiPlatform\State\Processor\LinkedDataPlatformProcessor" decorates="api_platform.state_processor.respond"> + <argument type="service" id="api_platform.state_processor.linked_data_platform.inner" /> + <argument type="service" id="api_platform.resource_class_resolver" /> + <argument type="service" id="api_platform.metadata.resource.metadata_collection_factory" /> + </service> + <service id="api_platform.listener.view.write" class="ApiPlatform\Symfony\EventListener\WriteListener"> <argument type="service" id="api_platform.state_processor.write" /> <argument type="service" id="api_platform.metadata.resource.metadata_collection_factory" /> diff --git a/tests/Fixtures/TestBundle/ApiResource/DummyGetPostDeleteOperation.php b/tests/Fixtures/TestBundle/ApiResource/DummyGetPostDeleteOperation.php new file mode 100644 index 00000000000..d46018fb298 --- /dev/null +++ b/tests/Fixtures/TestBundle/ApiResource/DummyGetPostDeleteOperation.php @@ -0,0 +1,63 @@ +<?php + +/* + * This file is part of the API Platform project. + * + * (c) Kévin Dunglas <dunglas@gmail.com> + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +declare(strict_types=1); + +namespace ApiPlatform\Tests\Fixtures\TestBundle\ApiResource; + +use ApiPlatform\Metadata\ApiResource; +use ApiPlatform\Metadata\Delete; +use ApiPlatform\Metadata\Get; +use ApiPlatform\Metadata\GetCollection; +use ApiPlatform\Metadata\Operation; +use ApiPlatform\Metadata\Post; + +#[ApiResource(operations: [ + new Get( + uriTemplate: '/dummy_get_post_delete_operations/{id}', + provider: [self::class, 'provideItem'], + ), + new GetCollection( + uriTemplate: '/dummy_get_post_delete_operations', + provider: [self::class, 'provide'], ), + new Post( + uriTemplate: '/dummy_get_post_delete_operations', + provider: [self::class, 'provide'], ), + new Delete( + uriTemplate: '/dummy_get_post_delete_operations/{id}', + provider: [self::class, 'provideItem'], ), +])] +class DummyGetPostDeleteOperation +{ + public ?int $id; + + public ?string $name = null; + + public static function provide(Operation $operation, array $uriVariables = [], array $context = []): array + { + $dummyResource = new self(); + $dummyResource->id = 1; + $dummyResource->name = 'Dummy name'; + + return [ + $dummyResource, + ]; + } + + public static function provideItem(Operation $operation, array $uriVariables = [], array $context = []): self + { + $dummyResource = new self(); + $dummyResource->id = $uriVariables['id']; + $dummyResource->name = 'Dummy name'; + + return $dummyResource; + } +} diff --git a/tests/Functional/LinkDataPlatformTest.php b/tests/Functional/LinkDataPlatformTest.php new file mode 100644 index 00000000000..b6ef9c608ba --- /dev/null +++ b/tests/Functional/LinkDataPlatformTest.php @@ -0,0 +1,76 @@ +<?php + +/* + * This file is part of the API Platform project. + * + * (c) Kévin Dunglas <dunglas@gmail.com> + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +declare(strict_types=1); + +namespace ApiPlatform\Tests\Functional; + +use ApiPlatform\Symfony\Bundle\Test\ApiTestCase; +use ApiPlatform\Tests\Fixtures\TestBundle\ApiResource\DummyGetPostDeleteOperation; +use ApiPlatform\Tests\SetupClassResourcesTrait; + +class LinkDataPlatformTest extends ApiTestCase +{ + use SetupClassResourcesTrait; + + /** + * @return class-string[] + */ + public static function getResources(): array + { + return [DummyGetPostDeleteOperation::class]; + } + + public function testAllowHeadersForResourceCollectionReflectsAllowedMethods(): void + { + $client = static::createClient(); + $client->request('GET', '/dummy_get_post_delete_operations', [ + 'headers' => [ + 'Content-Type' => 'application/ld+json', + ], + ]); + + $this->assertResponseHeaderSame('allow', 'OPTIONS, HEAD, GET, POST'); + + $client = static::createClient(); + $client->request('GET', '/dummy_get_post_delete_operations/1', [ + 'headers' => [ + 'Content-Type' => 'application/ld+json', + ], + ]); + + $this->assertResponseHeaderSame('allow', 'OPTIONS, HEAD, GET, DELETE'); + } + + public function testAcceptPostHeaderForResourceWithPostReflectsAllowedTypes(): void + { + $client = static::createClient(); + $client->request('GET', '/dummy_get_post_delete_operations', [ + 'headers' => [ + 'Content-Type' => 'application/ld+json', + ], + ]); + + $this->assertResponseHeaderSame('accept-post', 'application/ld+json, application/hal+json, application/vnd.api+json, application/xml, text/xml, application/json, text/html, application/graphql, multipart/form-data'); + } + + public function testAcceptPostHeaderDoesNotExistResourceWithoutPost(): void + { + $client = static::createClient(); + $client->request('GET', '/dummy_get_post_delete_operations/1', [ + 'headers' => [ + 'Content-Type' => 'application/ld+json', + ], + ]); + + $this->assertResponseNotHasHeader('accept-post'); + } +}