middlewares / negotiation
Middleware to implement content negotiation
Installs: 378 397
Dependents: 15
Suggesters: 2
Security: 0
Stars: 45
Watchers: 3
Forks: 2
Open Issues: 0
Requires
- php: ^7.2 || ^8.0
- middlewares/utils: ^3.0 || ^4.0
- psr/http-server-middleware: ^1.0
- willdurand/negotiation: ^3.0
Requires (Dev)
- friendsofphp/php-cs-fixer: ^2.0
- laminas/laminas-diactoros: ^2.2
- oscarotero/php-cs-fixer-config: ^1.0
- phpstan/phpstan: ^0.12
- phpunit/phpunit: ^8|^9|^10|^11
- squizlabs/php_codesniffer: ^3.0
README
Middleware using wildurand/Negotiation to implement content negotiation. Contains the following components:
Requirements
- PHP >= 7.2
- A PSR-7 http library
- A PSR-15 middleware dispatcher
Installation
This package is installable and autoloadable via Composer as middlewares/negotiation.
composer require middlewares/negotiation
Example
Dispatcher::run([ new Middlewares\ContentType(), new Middlewares\ContentLanguage(['en', 'gl', 'es']), new Middlewares\ContentEncoding(['gzip', 'deflate']), ]);
ContentType
To detect the preferred mime type using the Accept
header and the file extension and edit the header with this value. A Content-Type
header is also added to the response if it's missing.
Define the formats to negotiate sorted by priority in the first argument. By default uses these
//Use the default types $negotiator = new Middlewares\ContentType(); //Use only few types $negotiator = new Middlewares\ContentType(['html', 'json']); //Use only few types and configure some of them $negotiator = new Middlewares\ContentType([ 'html', 'json', 'txt' => [ 'extension' => ['txt'], 'mime-type' => ['text/plain'], 'charset' => true, ] ]);
errorResponse
If no format matches the negotiation, by default the middleware use the first value in the list of available formats (by default text/html
). Use this option to return a 406
error. Optionally, you can provide a Psr\Http\Message\ResponseFactoryInterface
that will be used to create the response. If it's not defined, Middleware\Utils\Factory will be used to detect it automatically.
$responseFactory = new MyOwnResponseFactory(); //Use default html format (the first provided) if no valid format was detected (By default) $negotiator = new Middlewares\ContentType(['html', 'json']); //Return a 406 response if no valid format was detected $negotiator = (new Middlewares\ContentType(['html', 'json']))->errorResponse(); //Return a 406 response using a specific responseFactory if no valid format was detected $negotiator = (new Middlewares\ContentType(['html', 'json']))->errorResponse($responseFactory);
charsets
The available charsets to negotiate with the Accept-Charset
header. By default is UTF-8
.
$negotiator = (new Middlewares\ContentType())->charsets(['UTF-8', 'ISO-8859-1']);
noSniff
Adds the X-Content-Type-Options: nosniff
header, to mitigating MIME confusiĆ³n attacks.. Enabled by default.
//Disable noSniff header $negotiator = (new Middlewares\ContentType())->noSniff(false);
attribute
To store the format name (json
, html
, css
etc) in an attribute of the ServerRequest
.
ContentLanguage
To detect the preferred language using the Accept-Language
header or the path prefix and edit the header with this value. A Content-Language
header is also added to the response if it's missing.
The first argument is an array with the available languages to negotiate sorted by priority. The first value will be used as default if no other languages is choosen in the negotiation.
$request = Factory::createServerRequest('GET', '/') ->withHeader('Accept-Language', 'gl-es, es;q=0.8, en;q=0.7'); Dispatcher::run([ new Middlewares\ContentLanguage(['es', 'en']), function ($request) { $language = $request->getHeaderLine('Accept-Language'); switch ($language) { case 'es': return 'Hola mundo'; case 'en': return 'Hello world'; } } ], $request);
usePath
By enabling this option, the base path will be used to detect the language. This is useful if you have different paths for each language, for example /gl/foo
and /en/foo
.
Note: the language in the path has preference over the Accept-Language
header.
$request = Factory::createServerRequest('GET', '/en/hello-world'); Dispatcher::run([ (new Middlewares\ContentLanguage(['es', 'en']))->usePath(), function ($request) { $language = $request->getHeaderLine('Accept-Language'); switch ($language) { case 'es': return 'Hola mundo'; case 'en': return 'Hello world'; } } ], $request);
redirect
Used to return a 302
responses redirecting to the path containing the language. This only works if usePath
is enabled, so for example, if the request uri is /welcome
, returns a redirection to /en/welcome
.
$responseFactory = new MyOwnResponseFactory(); //Use not only the Accept-Language header but also the path prefix to detect the language $negotiator = (new Middlewares\ContentLanguage(['es', 'en']))->usePath(); //Returns a redirection with the language in the path if it's missing $negotiator = (new Middlewares\ContentLanguage(['es', 'en']))->usePath()->redirect(); //Returns a redirection using a specific response factory $negotiator = (new Middlewares\ContentLanguage(['es', 'en']))->usePath()->redirect($responseFactory);
ContentEncoding
To detect the preferred encoding type using the Accept-Encoding
header and edit the header with this value.
$request = Factory::createServerRequest('GET', '/') ->withHeader('Accept-Encoding', 'gzip,deflate'); Dispatcher::run([ new Middlewares\ContentEncoding(['gzip']), function ($request) { echo $request->getHeaderLine('Accept-Encoding'); //gzip } ], $request);
Please see CHANGELOG for more information about recent changes and CONTRIBUTING for contributing details.
The MIT License (MIT). Please see LICENSE for more information.