You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
361 lines
11 KiB
361 lines
11 KiB
<?php |
|
|
|
/** |
|
* This file is part of CodeIgniter 4 framework. |
|
* |
|
* (c) CodeIgniter Foundation <admin@codeigniter.com> |
|
* |
|
* For the full copyright and license information, please view |
|
* the LICENSE file that was distributed with this source code. |
|
*/ |
|
|
|
namespace CodeIgniter\HTTP; |
|
|
|
use CodeIgniter\HTTP\Exceptions\HTTPException; |
|
|
|
/** |
|
* Class Negotiate |
|
* |
|
* Provides methods to negotiate with the HTTP headers to determine the best |
|
* type match between what the application supports and what the requesting |
|
* server wants. |
|
* |
|
* @see http://tools.ietf.org/html/rfc7231#section-5.3 |
|
* @see \CodeIgniter\HTTP\NegotiateTest |
|
*/ |
|
class Negotiate |
|
{ |
|
/** |
|
* Request |
|
* |
|
* @var IncomingRequest |
|
*/ |
|
protected $request; |
|
|
|
/** |
|
* Constructor |
|
*/ |
|
public function __construct(?RequestInterface $request = null) |
|
{ |
|
if ($request !== null) { |
|
assert($request instanceof IncomingRequest); |
|
|
|
$this->request = $request; |
|
} |
|
} |
|
|
|
/** |
|
* Stores the request instance to grab the headers from. |
|
* |
|
* @return $this |
|
*/ |
|
public function setRequest(RequestInterface $request) |
|
{ |
|
assert($request instanceof IncomingRequest); |
|
|
|
$this->request = $request; |
|
|
|
return $this; |
|
} |
|
|
|
/** |
|
* Determines the best content-type to use based on the $supported |
|
* types the application says it supports, and the types requested |
|
* by the client. |
|
* |
|
* If no match is found, the first, highest-ranking client requested |
|
* type is returned. |
|
* |
|
* @param bool $strictMatch If TRUE, will return an empty string when no match found. |
|
* If FALSE, will return the first supported element. |
|
*/ |
|
public function media(array $supported, bool $strictMatch = false): string |
|
{ |
|
return $this->getBestMatch($supported, $this->request->getHeaderLine('accept'), true, $strictMatch); |
|
} |
|
|
|
/** |
|
* Determines the best charset to use based on the $supported |
|
* types the application says it supports, and the types requested |
|
* by the client. |
|
* |
|
* If no match is found, the first, highest-ranking client requested |
|
* type is returned. |
|
*/ |
|
public function charset(array $supported): string |
|
{ |
|
$match = $this->getBestMatch( |
|
$supported, |
|
$this->request->getHeaderLine('accept-charset'), |
|
false, |
|
true |
|
); |
|
|
|
// If no charset is shown as a match, ignore the directive |
|
// as allowed by the RFC, and tell it a default value. |
|
if ($match === '') { |
|
return 'utf-8'; |
|
} |
|
|
|
return $match; |
|
} |
|
|
|
/** |
|
* Determines the best encoding type to use based on the $supported |
|
* types the application says it supports, and the types requested |
|
* by the client. |
|
* |
|
* If no match is found, the first, highest-ranking client requested |
|
* type is returned. |
|
*/ |
|
public function encoding(array $supported = []): string |
|
{ |
|
$supported[] = 'identity'; |
|
|
|
return $this->getBestMatch($supported, $this->request->getHeaderLine('accept-encoding')); |
|
} |
|
|
|
/** |
|
* Determines the best language to use based on the $supported |
|
* types the application says it supports, and the types requested |
|
* by the client. |
|
* |
|
* If no match is found, the first, highest-ranking client requested |
|
* type is returned. |
|
*/ |
|
public function language(array $supported): string |
|
{ |
|
return $this->getBestMatch($supported, $this->request->getHeaderLine('accept-language'), false, false, true); |
|
} |
|
|
|
// -------------------------------------------------------------------- |
|
// Utility Methods |
|
// -------------------------------------------------------------------- |
|
|
|
/** |
|
* Does the grunt work of comparing any of the app-supported values |
|
* against a given Accept* header string. |
|
* |
|
* Portions of this code base on Aura.Accept library. |
|
* |
|
* @param array $supported App-supported values |
|
* @param string $header header string |
|
* @param bool $enforceTypes If TRUE, will compare media types and sub-types. |
|
* @param bool $strictMatch If TRUE, will return empty string on no match. |
|
* If FALSE, will return the first supported element. |
|
* @param bool $matchLocales If TRUE, will match locale sub-types to a broad type (fr-FR = fr) |
|
* |
|
* @return string Best match |
|
*/ |
|
protected function getBestMatch( |
|
array $supported, |
|
?string $header = null, |
|
bool $enforceTypes = false, |
|
bool $strictMatch = false, |
|
bool $matchLocales = false |
|
): string { |
|
if ($supported === []) { |
|
throw HTTPException::forEmptySupportedNegotiations(); |
|
} |
|
|
|
if ($header === null || $header === '') { |
|
return $strictMatch ? '' : $supported[0]; |
|
} |
|
|
|
$acceptable = $this->parseHeader($header); |
|
|
|
foreach ($acceptable as $accept) { |
|
// if acceptable quality is zero, skip it. |
|
if ($accept['q'] === 0.0) { |
|
continue; |
|
} |
|
|
|
// if acceptable value is "anything", return the first available |
|
if ($accept['value'] === '*' || $accept['value'] === '*/*') { |
|
return $supported[0]; |
|
} |
|
|
|
// If an acceptable value is supported, return it |
|
foreach ($supported as $available) { |
|
if ($this->match($accept, $available, $enforceTypes, $matchLocales)) { |
|
return $available; |
|
} |
|
} |
|
} |
|
|
|
// No matches? Return the first supported element. |
|
return $strictMatch ? '' : $supported[0]; |
|
} |
|
|
|
/** |
|
* Parses an Accept* header into it's multiple values. |
|
* |
|
* This is based on code from Aura.Accept library. |
|
*/ |
|
public function parseHeader(string $header): array |
|
{ |
|
$results = []; |
|
$acceptable = explode(',', $header); |
|
|
|
foreach ($acceptable as $value) { |
|
$pairs = explode(';', $value); |
|
|
|
$value = $pairs[0]; |
|
|
|
unset($pairs[0]); |
|
|
|
$parameters = []; |
|
|
|
foreach ($pairs as $pair) { |
|
if (preg_match( |
|
'/^(?P<name>.+?)=(?P<quoted>"|\')?(?P<value>.*?)(?:\k<quoted>)?$/', |
|
$pair, |
|
$param |
|
)) { |
|
$parameters[trim($param['name'])] = trim($param['value']); |
|
} |
|
} |
|
|
|
$quality = 1.0; |
|
|
|
if (array_key_exists('q', $parameters)) { |
|
$quality = $parameters['q']; |
|
unset($parameters['q']); |
|
} |
|
|
|
$results[] = [ |
|
'value' => trim($value), |
|
'q' => (float) $quality, |
|
'params' => $parameters, |
|
]; |
|
} |
|
|
|
// Sort to get the highest results first |
|
usort($results, static function ($a, $b) { |
|
if ($a['q'] === $b['q']) { |
|
$aAst = substr_count($a['value'], '*'); |
|
$bAst = substr_count($b['value'], '*'); |
|
|
|
// '*/*' has lower precedence than 'text/*', |
|
// and 'text/*' has lower priority than 'text/plain' |
|
// |
|
// This seems backwards, but needs to be that way |
|
// due to the way PHP7 handles ordering or array |
|
// elements created by reference. |
|
if ($aAst > $bAst) { |
|
return 1; |
|
} |
|
|
|
// If the counts are the same, but one element |
|
// has more params than another, it has higher precedence. |
|
// |
|
// This seems backwards, but needs to be that way |
|
// due to the way PHP7 handles ordering or array |
|
// elements created by reference. |
|
if ($aAst === $bAst) { |
|
return count($b['params']) - count($a['params']); |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
// Still here? Higher q values have precedence. |
|
return ($a['q'] > $b['q']) ? -1 : 1; |
|
}); |
|
|
|
return $results; |
|
} |
|
|
|
/** |
|
* Match-maker |
|
* |
|
* @param bool $matchLocales |
|
*/ |
|
protected function match(array $acceptable, string $supported, bool $enforceTypes = false, $matchLocales = false): bool |
|
{ |
|
$supported = $this->parseHeader($supported); |
|
if (count($supported) === 1) { |
|
$supported = $supported[0]; |
|
} |
|
|
|
// Is it an exact match? |
|
if ($acceptable['value'] === $supported['value']) { |
|
return $this->matchParameters($acceptable, $supported); |
|
} |
|
|
|
// Do we need to compare types/sub-types? Only used |
|
// by negotiateMedia(). |
|
if ($enforceTypes) { |
|
return $this->matchTypes($acceptable, $supported); |
|
} |
|
|
|
// Do we need to match locales against broader locales? |
|
if ($matchLocales) { |
|
return $this->matchLocales($acceptable, $supported); |
|
} |
|
|
|
return false; |
|
} |
|
|
|
/** |
|
* Checks two Accept values with matching 'values' to see if their |
|
* 'params' are the same. |
|
*/ |
|
protected function matchParameters(array $acceptable, array $supported): bool |
|
{ |
|
if (count($acceptable['params']) !== count($supported['params'])) { |
|
return false; |
|
} |
|
|
|
foreach ($supported['params'] as $label => $value) { |
|
if (! isset($acceptable['params'][$label]) |
|
|| $acceptable['params'][$label] !== $value |
|
) { |
|
return false; |
|
} |
|
} |
|
|
|
return true; |
|
} |
|
|
|
/** |
|
* Compares the types/subtypes of an acceptable Media type and |
|
* the supported string. |
|
*/ |
|
public function matchTypes(array $acceptable, array $supported): bool |
|
{ |
|
// PHPDocumentor v2 cannot parse yet the shorter list syntax, |
|
// causing no API generation for the file. |
|
[$aType, $aSubType] = explode('/', $acceptable['value']); |
|
[$sType, $sSubType] = explode('/', $supported['value']); |
|
|
|
// If the types don't match, we're done. |
|
if ($aType !== $sType) { |
|
return false; |
|
} |
|
|
|
// If there's an asterisk, we're cool |
|
if ($aSubType === '*') { |
|
return true; |
|
} |
|
|
|
// Otherwise, subtypes must match also. |
|
return $aSubType === $sSubType; |
|
} |
|
|
|
/** |
|
* Will match locales against their broader pairs, so that fr-FR would |
|
* match a supported localed of fr |
|
*/ |
|
public function matchLocales(array $acceptable, array $supported): bool |
|
{ |
|
$aBroad = mb_strpos($acceptable['value'], '-') > 0 |
|
? mb_substr($acceptable['value'], 0, mb_strpos($acceptable['value'], '-')) |
|
: $acceptable['value']; |
|
$sBroad = mb_strpos($supported['value'], '-') > 0 |
|
? mb_substr($supported['value'], 0, mb_strpos($supported['value'], '-')) |
|
: $supported['value']; |
|
|
|
return strtolower($aBroad) === strtolower($sBroad); |
|
} |
|
}
|
|
|