rnd-cosoft / api-tools-content-negotiation
Laminas Module providing content-negotiation features
Requires
- php: ~8.1.0 || ~8.2.0 || ~8.3.0
- laminas/laminas-eventmanager: ^3.4.0
- laminas/laminas-filter: ^2.13.0
- laminas/laminas-http: ^2.15.1
- laminas/laminas-json: ^3.3
- laminas/laminas-mvc: ^3.3.0
- laminas/laminas-servicemanager: ^3.8
- laminas/laminas-stdlib: ^3.6.1
- laminas/laminas-validator: ^2.15.0
- laminas/laminas-view: ^2.14.0
- laminas/laminas-zendframework-bridge: ^1.1
- rnd-cosoft/api-tools-api-problem: ^1.6.0
Requires (Dev)
- laminas/laminas-coding-standard: ~2.3.0
- phpspec/prophecy-phpunit: ^2.0.1
- phpunit/phpunit: ^9.5.27
- psalm/plugin-phpunit: ^0.16.1
- rnd-cosoft/api-tools-hal: ^1.10
- vimeo/psalm: ^4.30
Replaces
This package is not auto-updated.
Last update: 2024-12-20 12:51:16 UTC
README
π·πΊ Π ΡΡΡΠΊΠΈΠΌ Π³ΡΠ°ΠΆΠ΄Π°Π½Π°ΠΌ
ΠΡ, ΡΡΠ°ΡΡΠ½ΠΈΠΊΠΈ Laminas, ΡΠΎΠ΄ΠΈΠ»ΠΈΡΡ ΠΈ ΠΆΠΈΠ²Π΅ΠΌ Π² ΡΠ°Π·Π½ΡΡ ΡΡΡΠ°Π½Π°Ρ . Π£ ΠΌΠ½ΠΎΠ³ΠΈΡ ΠΈΠ· Π½Π°Ρ Π΅ΡΡΡ Π΄ΡΡΠ·ΡΡ, ΡΠΎΠ΄ΡΡΠ²Π΅Π½Π½ΠΈΠΊΠΈ ΠΈ ΠΊΠΎΠ»Π»Π΅Π³ΠΈ ΠΊΠ°ΠΊ Π² Π ΠΎΡΡΠΈΠΈ, ΡΠ°ΠΊ ΠΈ Π² Π£ΠΊΡΠ°ΠΈΠ½Π΅. ΠΠ΅ΠΊΠΎΡΠΎΡΡΠ΅ ΠΈΠ· Π½Π°Ρ ΡΠΎΠ΄ΠΈΠ»ΠΈΡΡ Π² Π ΠΎΡΡΠΈΠΈ. ΠΠ΅ΠΊΠΎΡΠΎΡΡΠ΅ ΠΈΠ· Π½Π°Ρ ΠΆΠΈΠ²ΡΡ Π² Π ΠΎΡΡΠΈΠΈ. Π£ Π½Π΅ΠΊΠΎΡΠΎΡΡΡ Π±Π°Π±ΡΡΠΊΠΈ ΠΈ Π΄Π΅Π΄ΡΡΠΊΠΈ ΡΡΠ°ΠΆΠ°Π»ΠΈΡΡ Ρ ΡΠ°ΡΠΈΡΡΠ°ΠΌΠΈ Π²ΠΎ ΠΡΠΎΡΠΎΠΉ ΠΌΠΈΡΠΎΠ²ΠΎΠΉ Π²ΠΎΠΉΠ½Π΅. ΠΠ΄Π΅ΡΡ Π½ΠΈΠΊΡΠΎ Π½Π΅ ΠΏΠΎΠ΄Π΄Π΅ΡΠΆΠΈΠ²Π°Π΅Ρ ΡΠ°ΡΠΈΠ·ΠΌ.
Π£ ΠΎΠ΄Π½ΠΎΠ³ΠΎ ΠΈΠ· Π½Π°Ρ Π΅ΡΡΡ ΡΠΊΡΠ°ΠΈΠ½ΡΠΊΠ°Ρ ΡΠΎΠ΄ΡΡΠ²Π΅Π½Π½ΠΈΡΠ°, ΠΊΠΎΡΠΎΡΠ°Ρ ΡΠΏΠ°ΡΠ»Π°ΡΡ ΠΈΠ· Π΄ΠΎΠΌΠ° Π²ΠΌΠ΅ΡΡΠ΅ Ρ ΡΡΠ½ΠΎΠΌ. ΠΠΎΠ΅Π·Π΄ Π·Π°Π΄Π΅ΡΠΆΠ°Π»ΡΡ ΠΈΠ·-Π·Π° Π±ΠΎΠΌΠ±Π΅ΠΆΠΊΠΈ Π½Π° Π΄ΠΎΡΠΎΠ³Π΅ Π²ΠΏΠ΅ΡΠ΅Π΄ΠΈ. Π£ Π½Π°Ρ Π΅ΡΡΡ Π΄ΡΡΠ·ΡΡ, ΠΊΠΎΡΠΎΡΡΠ΅ ΠΏΡΡΡΡΡΡΡ Π² Π±ΠΎΠΌΠ±ΠΎΡΠ±Π΅ΠΆΠΈΡΠ°Ρ . ΠΡ Ρ ΡΡΠ΅Π²ΠΎΠ³ΠΎΠΉ ΠΆΠ΄Π΅ΠΌ Π²Π΅ΡΡΠΎΡΠΊΠΈ ΠΎΡ Π½ΠΈΡ ΠΏΠΎΡΠ»Π΅ Π²ΠΎΠ·Π΄ΡΡΠ½ΡΡ Π½Π°Π»Π΅ΡΠΎΠ², ΠΊΠΎΡΠΎΡΡΠ΅ Π±Π΅ΡΠΏΠΎΡΡΠ΄ΠΎΡΠ½ΠΎ Π½Π°Π½ΠΎΡΡΡ ΡΠ΄Π°ΡΡ ΠΈ ΠΏΠΎΠΏΠ°Π΄Π°ΡΡ ΠΏΠΎ Π±ΠΎΠ»ΡΠ½ΠΈΡΠ°ΠΌ, ΡΠΊΠΎΠ»Π°ΠΌ, Π΄Π΅ΡΡΠΊΠΈΠΌ ΡΠ°Π΄Π°ΠΌ ΠΈ Π΄ΠΎΠΌΠ°ΠΌ. ΠΡ Π½Π΅ Π±Π΅ΡΠ΅ΠΌ ΡΡΠΎ ΠΈΠ· ΠΊΠ°ΠΊΠΈΡ -Π»ΠΈΠ±ΠΎ Π‘ΠΠ. ΠΡ Π½Π°Π±Π»ΡΠ΄Π°Π΅ΠΌ ΡΡΠΎ Π½Π°ΠΏΡΡΠΌΡΡ.
ΠΡ Π΄ΠΎΠ²Π΅ΡΡΠ΅ΡΠ΅ Π½Π°ΠΌ Π΄ΠΎΡΡΠ°ΡΠΎΡΠ½ΠΎ, ΡΡΠΎΠ± ΠΈΡΠΏΠΎΠ»ΡΠ·ΠΎΠ²Π°ΡΡ Π½Π°ΡΠΈ ΠΏΡΠΎΠ³ΡΠ°ΠΌΠΌΡ, ΠΈ ΠΌΡ ΠΏΡΠΎΡΠΈΠΌ Π²Π°Ρ Π΄ΠΎΠ²Π΅ΡΠΈΡΡΡΡ Π½Π°ΠΌ Π²Π½ΠΎΠ²Ρ. ΠΡ Π½ΡΠΆΠ΄Π°Π΅ΠΌΡΡ Π² ΠΏΠΎΠΌΠΎΡΠΈ. ΠΡΡ ΠΎΠ΄ΠΈΡΠ΅ ΠΈ ΠΏΡΠΎΡΠ΅ΡΡΡΠΉΡΠ΅ ΠΏΡΠΎΡΠΈΠ² ΡΡΠΎΠΉ Π±Π΅ΡΠΏΠΎΠ»Π΅Π·Π½ΠΎΠΉ Π²ΠΎΠΉΠ½Ρ. ΠΡΡΠ°Π½ΠΎΠ²ΠΈΡΠ΅ ΠΊΡΠΎΠ²ΠΎΠΏΡΠΎΠ»ΠΈΡΠΈΠ΅. Π‘ΠΊΠ°ΠΆΠΈΡΠ΅ "ΠΠ΅Ρ Π²ΠΎΠΉΠ½Π΅!"
πΊπΈ To Citizens of Russia
We at Laminas come from all over the world. Many of us have friends, family and colleagues in both Russia and Ukraine. Some of us were born in Russia. Some of us currently live in Russia. Some have grandparents who fought Nazis in World War II. Nobody here supports fascism.
One team member has a Ukrainian relative who fled her home with her son. The train was delayed due to bombing on the road ahead. We have friends who are hiding in bomb shelters. We anxiously follow up on them after the air raids, which indiscriminately fire at hospitals, schools, kindergartens and houses. We're not taking this from any media. These are our actual experiences.
You trust us enough to use our software. We ask that you trust us to say the truth on this. We need your help. Go out and protest this unnecessary war. Stop the bloodshed. Say "stop the war!"
Introduction
api-tools-content-negotiation
is a module for automating content negotiation tasks within a Laminas
Framework application.
The following features are provided
- Mapping
Accept
header media types to specific view model types, and automatically casting controller results to those view model types. - Defining
Accept
header media type whitelists; requests withAccept
media types that fall outside the whitelist will be immediately rejected with a406 Not Acceptable
response. - Defining
Content-Type
header media type whitelists; requests sending content bodies withContent-Type
media types that fall outside the whitelist will be immediately rejected with a415 Unsupported Media Type
response.
Requirements
Please see the composer.json file.
Installation
Run the following composer
command:
$ composer require laminas-api-tools/api-tools-content-negotiation
Alternately, manually add the following to your composer.json
, in the require
section:
"require": { "laminas-api-tools/api-tools-content-negotiation": "^1.2" }
And then run composer update
to ensure the module is installed.
Finally, add the module name to your project's config/application.config.php
under the modules
key:
return [ /* ... */ 'modules' => [ /* ... */ 'Laminas\ApiTools\ContentNegotiation', ], /* ... */ ];
laminas-component-installer
If you use laminas-component-installer, that plugin will install api-tools-content-negotiation as a module for you.
Configuration
User Configuration
The top-level configuration key for user configuration of this module is api-tools-content-negotiation
.
Key: controllers
The controllers
key is utilized for mapping a content negotiation strategy to a particular
controller service name (from the top-level controllers
section). The value portion
of the controller array can either be a named selector (see selectors
below), or a
selector definition.
A selector definition consists of an array with the key of the array being the name of a view model, and the value of it being an indexed array of media types that, when matched, will select that view model.
Example:
'controllers' => [ // Named selector: 'Application\Controller\HelloWorld1' => 'Json', // Selector definition: 'Application\Controller\HelloWorld2' => [ 'Laminas\ApiTools\ContentNegotiation\JsonModel' => [ 'application/json', 'application/*+json', ], ], ],
Key: selectors
The selectors
key is utilized to create named selector definitions for reuse between many different
controllers. The key part of the selectors array will be a name used to correlate the selector
definition (which uses the format described in the controllers key).
Example:
'selectors' => [ 'Json' => [ 'Laminas\ApiTools\ContentNegotiation\JsonModel' => [ 'application/json', 'application/*+json', ], ], ],
A selector can contain multiple view models, each associated with different media types, allowing you to provide multiple representations. As an example, the following selector would allow a given controller to return either JSON or HTML output:
'selectors' => [ 'HTML-Json' => [ 'Laminas\ApiTools\ContentNegotiation\JsonModel' => [ 'application/json', 'application/*+json', ], 'Laminas\ApiTools\ContentNegotiation\ViewModel' => [ 'text/html', ], ], ],
Key: accept_whitelist
The accept_whitelist
key is utilized to instruct the content negotiation module which media types
are acceptable for a given controller service name. When a controller service name is configured
in this key, along with an indexed array of matching media types, only media types that match
the Accept
header of a given request will be allowed to be dispatched. Unmatched media types
will receive a 406 Cannot honor Accept type specified
response.
The value of each controller service name key can either be a string or an array of strings.
Example:
'accept_whitelist' => [ 'Application\\Controller\\HelloApiController' => [ 'application/vnd.application-hello+json', 'application/hal+json', 'application/json', ], ],
Key: content_type_whitelist
The content_type_whitelist
key is utilized to instruct the content negotiation module which media
types are valid for the Content-Type
of a request. When a controller service name is
configured in this key, along with an indexed array of matching media types, only media types
that match the Content-Type
header of a given request will be allowed to be dispatched. Unmatched
media types will receive a 415 Invalid content-type specified
response.
The value of each controller service name key can either be a string or an array of strings.
Example:
'content_type_whitelist' => [ 'Application\\Controller\\HelloWorldController' => [ 'application/vnd.application-hello-world+json', 'application/json', ], ],
Key: x_http_method_override_enabled
- Since 1.3.0
This boolean flag determines whether or not the HttpMethodOverrideListener
will be enabled by default.
Key: http_override_methods
- Since 1.3.0
The http_override_methods
key is utilized to provide the
HttpMethodOverrideListener
with a map of allowed override methods for a given
HTTP method, as specified via the X-HTTP-Method-Override
header. Essentially,
the values are:
'Incoming HTTP request method' => $arrayOfAllowedOverrideMethods,
As an example, if you want to allow the X-HTTP-Method-Override
header to allow
overriding HTTP GET
requests with an alternate method, you might define this
as follows:
'x_http_method_override_enabled' => true, 'http_override_methods' => [ 'GET' => [ 'HEAD', 'POST', 'PUT', 'DELETE', 'PATCH', ], ];
Then, to make a request, you could do the following:
GET /foo HTTP/1.1 Host: example.com X-HTTP-Method-Override: PATCH some=content&more=content
The above would then be interpreted as a PATCH
request. If the same request
were made via HTTP POST
, or if a GET
request were made with an override
value of OPTIONS
, the listener would raise a problem, as, in the former case,
no maps are defined for POST
, and, in the latter, OPTIONS
is not in the map
for GET
.
System Configuration
The following configuration is provided in config/module.config.php
to enable the module to
function:
'filters' => [ 'aliases' => [ 'Laminas\Filter\File\RenameUpload' => 'filerenameupload', ], 'factories' => [ 'filerenameupload' => Factory\RenameUploadFilterFactory::class, ], ], 'validators' => [ 'aliases' => [ 'Laminas\Validator\File\UploadFile' => 'fileuploadfile', ], 'factories' => [ 'fileuploadfile' => Factory\UploadFileValidatorFactory::class, ], ], 'service_manager' => [ 'factories' => [ ContentTypeListener::class => InvokableFactory::class, AcceptListener::class => Factory\AcceptListenerFactory::class, AcceptFilterListener::class => Factory\AcceptFilterListenerFactory::class, ContentTypeFilterListener::class => Factory\ContentTypeFilterListenerFactory::class, ContentNegotiationOptions::class => Factory\ContentNegotiationOptionsFactory::class, HttpMethodOverrideListener::class => Factory\HttpMethodOverrideListenerFactory::class, ], ], 'controller_plugins' => [ 'aliases' => [ 'routeParam' => ControllerPlugin\RouteParam::class, 'queryParam' => ControllerPlugin\QueryParam::class, 'bodyParam' => ControllerPlugin\BodyParam::class, 'routeParams' => ControllerPlugin\RouteParams::class, 'queryParams' => ControllerPlugin\QueryParams::class, 'bodyParams' => ControllerPlugin\BodyParams::class, ], 'factories' => [ ControllerPlugin\RouteParam::class => InvokableFactory::class, ControllerPlugin\QueryParam::class => InvokableFactory::class, ControllerPlugin\BodyParam::class => InvokableFactory::class, ControllerPlugin\RouteParams::class => InvokableFactory::class, ControllerPlugin\QueryParams::class => InvokableFactory::class, ControllerPlugin\BodyParams::class => InvokableFactory::class, ], ],
Laminas Events
Listeners
Laminas\ApiTools\ContentNegotiation\AcceptListener
This listener is attached to the MvcEvent::EVENT_DISPATCH
event with priority -10
. It is
responsible for performing the actual selection and casting of a controller's view model based on
the content negotiation configuration.
Laminas\ApiTools\ContentNegotiation\ContentTypeListener
This listener is attached to the MvcEvent::EVENT_ROUTE
event with a priority of -625
. It is
responsible for examining the Content-Type
header in order to determine how the content body
should be deserialized. Values are then persisted inside of a ParameterDataContainer
which is
stored in the LaminasContentNegotiationParameterData
key of the MvcEvent
object.
Laminas\ApiTools\ContentNegotiation\AcceptFilterListener
This listener is attached to the MvcEvent::EVENT_ROUTE
event with a priority of -625
. It is
responsible for ensuring the controller selected by routing is configured to respond to the specific
media type in the current request's Accept
header. If it cannot, it will short-circuit the MVC
dispatch process by returning a 406 Cannot honor Accept type specified
response.
Laminas\ApiTools\ContentNegotiation\ContentTypeFilterListener
This listener is attached to the MvcEvent::EVENT_ROUTE
event with a priority of -625
. It is
responsible for ensuring the route matched controller can accept content in the request body
specified by the media type in the current request's Content-Type
header. If it cannot, it will
short-circuit the MVC dispatch process by returning a 415 Invalid content-type specified
response.
Laminas\ApiTools\ContentNegotiation\HttpMethodOverrideListener
- Since 1.3.0
This listener is attached to the MvcEvent::EVENT_ROUTE
event with a priority
of -40
, but only if the x_http_method_override_enabled
configuration flag
was toggle on. It is responsible for checking if an X-HTTP-Method-Override
header is present, and, if so, if it contains a value in the set allowed for the
current HTTP request method invoked. If so, it resets the HTTP request method to
the header value.
Laminas Services
Controller Plugins
Laminas\ApiTools\ContentNegotiation\ControllerPlugin\RouteParam (a.k.a "routeParam")
A controller plugin (Laminas\Mvc\Controller\AbstractActionController
callable) that will return a
single parameter with a particular name from the route match.
use Laminas\Mvc\Controller\AbstractActionController; class IndexController extends AbstractActionController { public function indexAction() { return $this->routeParam('id', 'someDefaultValue'); } }
Laminas\ApiTools\ContentNegotiation\ControllerPlugin\QueryParam (a.k.a "queryParam")
A controller plugin (Laminas\Mvc\Controller\AbstractActionController
callable) that will return a
single parameter from the current request query string.
use Laminas\Mvc\Controller\AbstractActionController; class IndexController extends AbstractActionController { public function indexAction() { return $this->queryParam('foo', 'someDefaultValue'); } }
Laminas\ApiTools\ContentNegotiation\ControllerPlugin\BodyParam (a.k.a "bodyParam")
A controller plugin (Laminas\Mvc\Controller\AbstractActionController
callable) that will return a
single parameter from the content-negotiated content body.
use Laminas\Mvc\Controller\AbstractActionController; class IndexController extends AbstractActionController { public function indexAction() { return $this->bodyParam('foo', 'someDefaultValue'); } }
Laminas\ApiTools\ContentNegotiation\ControllerPlugin\RouteParams (a.k.a "routeParams")
A controller plugin (Laminas\Mvc\Controller\AbstractActionController
callable) that will return a
all the route parameters.
use Laminas\Mvc\Controller\AbstractActionController; class IndexController extends AbstractActionController { public function indexAction() { return $this->routeParams() } }
Laminas\ApiTools\ContentNegotiation\ControllerPlugin\QueryParams (a.k.a "queryParams")
A controller plugin (Laminas\Mvc\Controller\AbstractActionController
callable) that will return a
all the query parameters.
use Laminas\Mvc\Controller\AbstractActionController; class IndexController extends AbstractActionController { public function indexAction() { return $this->queryParams() } }
Laminas\ApiTools\ContentNegotiation\ControllerPlugin\BodyParams (a.k.a "bodyParams")
A controller plugin (Laminas\Mvc\Controller\AbstractActionController
callable) that will return a
all the content-negotiated body parameters.
use Laminas\Mvc\Controller\AbstractActionController; class IndexController extends AbstractActionController { public function indexAction() { return $this->bodyParams() } }