diff options
Diffstat (limited to 'vendor/league/uri-interfaces/KeyValuePair/Converter.php')
-rw-r--r-- | vendor/league/uri-interfaces/KeyValuePair/Converter.php | 209 |
1 files changed, 209 insertions, 0 deletions
diff --git a/vendor/league/uri-interfaces/KeyValuePair/Converter.php b/vendor/league/uri-interfaces/KeyValuePair/Converter.php new file mode 100644 index 000000000..f208060c6 --- /dev/null +++ b/vendor/league/uri-interfaces/KeyValuePair/Converter.php @@ -0,0 +1,209 @@ +<?php + +/** + * League.Uri (https://uri.thephpleague.com) + * + * (c) Ignace Nyamagana Butera <nyamsprod@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 League\Uri\KeyValuePair; + +use League\Uri\Contracts\UriComponentInterface; +use League\Uri\Exceptions\SyntaxError; +use Stringable; + +use function array_combine; +use function explode; +use function implode; +use function is_float; +use function is_int; +use function is_string; +use function json_encode; +use function preg_match; +use function str_replace; + +use const JSON_PRESERVE_ZERO_FRACTION; +use const PHP_QUERY_RFC1738; +use const PHP_QUERY_RFC3986; + +final class Converter +{ + private const REGEXP_INVALID_CHARS = '/[\x00-\x1f\x7f]/'; + + /** + * @param non-empty-string $separator the query string separator + * @param array<string> $fromRfc3986 contains all the RFC3986 encoded characters to be converted + * @param array<string> $toEncoding contains all the expected encoded characters + */ + private function __construct( + private readonly string $separator, + private readonly array $fromRfc3986 = [], + private readonly array $toEncoding = [], + ) { + if ('' === $this->separator) { + throw new SyntaxError('The separator character must be a non empty string.'); + } + } + + /** + * @param non-empty-string $separator + */ + public static function new(string $separator): self + { + return new self($separator); + } + + /** + * @param non-empty-string $separator + */ + public static function fromRFC3986(string $separator = '&'): self + { + return self::new($separator); + } + + /** + * @param non-empty-string $separator + */ + public static function fromRFC1738(string $separator = '&'): self + { + return self::new($separator) + ->withEncodingMap(['%20' => '+']); + } + + /** + * @param non-empty-string $separator + * + * @see https://url.spec.whatwg.org/#application/x-www-form-urlencoded + */ + public static function fromFormData(string $separator = '&'): self + { + return self::new($separator) + ->withEncodingMap(['%20' => '+', '%2A' => '*']); + } + + public static function fromEncodingType(int $encType): self + { + return match ($encType) { + PHP_QUERY_RFC3986 => self::fromRFC3986(), + PHP_QUERY_RFC1738 => self::fromRFC1738(), + default => throw new SyntaxError('Unknown or Unsupported encoding.'), + }; + } + + /** + * @return non-empty-string + */ + public function separator(): string + { + return $this->separator; + } + + /** + * @return array<string, string> + */ + public function encodingMap(): array + { + return array_combine($this->fromRfc3986, $this->toEncoding); + } + + /** + * @return array<non-empty-list<string|null>> + */ + public function toPairs(Stringable|string|int|float|bool|null $value): array + { + $value = match (true) { + $value instanceof UriComponentInterface => $value->value(), + $value instanceof Stringable, is_int($value) => (string) $value, + false === $value => '0', + true === $value => '1', + default => $value, + }; + + if (null === $value) { + return []; + } + + $value = match (1) { + preg_match(self::REGEXP_INVALID_CHARS, (string) $value) => throw new SyntaxError('Invalid query string: `'.$value.'`.'), + default => str_replace($this->toEncoding, $this->fromRfc3986, (string) $value), + }; + + return array_map( + fn (string $pair): array => explode('=', $pair, 2) + [1 => null], + explode($this->separator, $value) + ); + } + + private static function vString(Stringable|string|bool|int|float|null $value): ?string + { + return match (true) { + $value => '1', + false === $value => '0', + null === $value => null, + is_float($value) => (string) json_encode($value, JSON_PRESERVE_ZERO_FRACTION), + default => (string) $value, + }; + } + + /** + * @param iterable<array{0:string|null, 1:Stringable|string|bool|int|float|null}> $pairs + */ + public function toValue(iterable $pairs): ?string + { + $filteredPairs = []; + foreach ($pairs as $pair) { + $filteredPairs[] = match (true) { + !is_string($pair[0]) => throw new SyntaxError('the pair key MUST be a string;, `'.gettype($pair[0]).'` given.'), + null === $pair[1] => self::vString($pair[0]), + default => self::vString($pair[0]).'='.self::vString($pair[1]), + }; + } + + return match ([]) { + $filteredPairs => null, + default => str_replace($this->fromRfc3986, $this->toEncoding, implode($this->separator, $filteredPairs)), + }; + } + + /** + * @param non-empty-string $separator + */ + public function withSeparator(string $separator): self + { + return match ($this->separator) { + $separator => $this, + default => new self($separator, $this->fromRfc3986, $this->toEncoding), + }; + } + + /** + * Sets the conversion map. + * + * Each key from the iterable structure represents the RFC3986 encoded characters as string, + * while each value represents the expected output encoded characters + */ + public function withEncodingMap(iterable $encodingMap): self + { + $fromRfc3986 = []; + $toEncoding = []; + foreach ($encodingMap as $from => $to) { + [$fromRfc3986[], $toEncoding[]] = match (true) { + !is_string($from) => throw new SyntaxError('The encoding output must be a string; `'.gettype($from).'` given.'), + $to instanceof Stringable, + is_string($to) => [$from, (string) $to], + default => throw new SyntaxError('The encoding output must be a string; `'.gettype($to).'` given.'), + }; + } + + return match (true) { + $fromRfc3986 !== $this->fromRfc3986, + $toEncoding !== $this->toEncoding => new self($this->separator, $fromRfc3986, $toEncoding), + default => $this, + }; + } +} |