Negotiation Middleware
, (*1)
The Negotiation Middleware is a PHP library that negotiates accept headers of HTTP requests. The middleware chooses the most fitting options by looking at the accepted values of client and server. It supports the headers accept, accept-language, accept-encoding and accept-charset., (*2)
This library is a middleware for the Slim framework 3 but may be used for any PHP code that uses HTTP messages conform to PSR-7. The Negotiation Middleware is based on the library from William Durand., (*3)
Installation
The recommended way to install the Negotiation Middleware is using Composer:, (*4)
``` bash
$ composer require gofabian/negotiation-middleware, (*5)
Composer fetches all dependencies automatically. The Negotiation Middleware uses common standards and tries to reduce the number of required packages follwing the [KISS principle][kiss]:
[kiss]: https://en.wikipedia.org/wiki/KISS_principle "Keep it simple, stupid"
- PHP 7.3+
- PSR-7
- willdurand/negotiation 3.x
- recommended: slim/slim 3.x
## Usage
The first example describes how to use the Slim framework in combination with the Negotiation Middleware. The following examples are less detailed and bring specific aspects into focus.
### Slim and the Negotiation Middleware
The most common way is to negotiate the media type. In this example the server accepts the media types `text/html` and `application/json`:
``` php
<?php
use Gofabian\Negotiation\NegotiationMiddleware;
// create Slim app
$app = new \Slim\App;
// configure middleware
$app->add(new NegotiationMiddleware([
'accept' => ['text/html', 'application/json']
]));
// use negotiated media type
$app->get('/mediatype', function ($request, $response, $args) {
$negotiation = $request->getAttribute('negotiation');
$mediaType = $negotiation->getMediaType();
return $response->write("media type = " . $mediaType);
});
// run app
$app->run();
Let's have a look at incoming HTTP requests and the priority of accepted media types:, (*6)
-
If the accept header includes text/html
the HTTP response will be media type = text/html
. The first entry of the configured media types has the highest priority., (*7)
-
If the accept header includes application/json
but not text/html
the HTTP response will be media type = application/json
., (*8)
-
If the accept header is empty the HTTP response will be media type = text/html
. By default the entry with highest priority will be chosen if the accept header does not exist., (*9)
-
If the accept header is available but includes neither application/json
nor text/html
the HTTP response will have status 406 "Not Acceptable"., (*10)
In addition to media type the middleware negotiates language, charset and encoding:, (*11)
``` php
$app->add(new NegotiationMiddleware([
'accept' => ['text/html', 'application/json'],
'accept-language' => ['en', 'de-DE'],
'accept-encoding' => ['gzip'],
'accept-charset' => ['utf-8', 'ascii']
]));, (*12)
### HTTP Status 406 - Not Acceptable
If the HTTP request contains an accept header but none of its values is accepted by the Negotiation Middleware the HTTP response will have the HTTP status 406 "Not Acceptable".
If the HTTP request does not contain an accept header the Negotiation Middleware will take the value with highest priority. Alternatively you can answer such requests with HTTP status 406, too:
```php
$app->add(new NegotiationMiddleware(
[ 'accept-language' => ['en', 'de-DE'] ],
false // 406 status for empty accept headers
));
In this example the Negotiation Middleware will return HTTP status 406 if the HTTP request does not contain the accept header accept-language
., (*13)
Negotiation result
The Negotiation Middleware puts the negotiation result into an instance of AcceptProvider
and attaches it as an attribute to the PSR-7 HTTP request. The attribute name is negotiation
by default and may be set like this:, (*14)
$app->add(new NegotiationMiddleware(
[ 'accept' => ['text/xml'] ],
true,
'foo' // custom name
));
$app->get('/example', function($request, $response, $args) {
$result = $request->getAttribute('foo');
return $response->write($result->getMediaType());
});
The easy way to access the negotiation results is to use the attached AcceptProvider
like this:, (*15)
$result = $request->getAttribute('negotiation');
$result->getMediaType(); // these
$result->getLanguage(); // getters
$result->getCharset(); // return
$result->getEncoding(); // strings
Alternatively you can get the original result objects of the negotiation library like this:, (*16)
$result = $request->getAttribute('negotiation');
$result->getAccept(); // results
$result->getAcceptLanguage(); // from
$result->getAcceptCharset(); // negotiation
$result->getAcceptEncoding(); // library
Have a look at the original documentation for more information., (*17)
Contributing
Please see CONTRIBUTING for details., (*18)
Credits
License
The MIT License (MIT). Please see License File for more information., (*19)