From a3dbb2477d1a3a9492510e54ae088d980f4eb184 Mon Sep 17 00:00:00 2001 From: Bob Weinand Date: Sun, 14 Jan 2024 00:42:31 +0100 Subject: [PATCH] Add Priority handling for HTTP/3 --- composer.json | 15 + src/Driver/Http3Driver.php | 26 +- src/Driver/Internal/Http3/Http3Frame.php | 2 +- src/Driver/Internal/Http3/Http3Parser.php | 35 +- src/Driver/Internal/Http3/Rfc8941.php | 491 ++++++++++++++++++ src/Driver/Internal/Http3/Rfc8941/Boolean.php | 18 + src/Driver/Internal/Http3/Rfc8941/Bytes.php | 18 + src/Driver/Internal/Http3/Rfc8941/Date.php | 18 + .../Internal/Http3/Rfc8941/DisplayString.php | 18 + .../Internal/Http3/Rfc8941/InnerList.php | 18 + src/Driver/Internal/Http3/Rfc8941/Item.php | 18 + src/Driver/Internal/Http3/Rfc8941/Number.php | 18 + src/Driver/Internal/Http3/Rfc8941/Str.php | 18 + src/Driver/Internal/Http3/Rfc8941/Token.php | 18 + test/Http/Rfc8941Test.php | 133 +++++ 15 files changed, 858 insertions(+), 6 deletions(-) create mode 100644 src/Driver/Internal/Http3/Rfc8941.php create mode 100644 src/Driver/Internal/Http3/Rfc8941/Boolean.php create mode 100644 src/Driver/Internal/Http3/Rfc8941/Bytes.php create mode 100644 src/Driver/Internal/Http3/Rfc8941/Date.php create mode 100644 src/Driver/Internal/Http3/Rfc8941/DisplayString.php create mode 100644 src/Driver/Internal/Http3/Rfc8941/InnerList.php create mode 100644 src/Driver/Internal/Http3/Rfc8941/Item.php create mode 100644 src/Driver/Internal/Http3/Rfc8941/Number.php create mode 100644 src/Driver/Internal/Http3/Rfc8941/Str.php create mode 100644 src/Driver/Internal/Http3/Rfc8941/Token.php create mode 100644 test/Http/Rfc8941Test.php diff --git a/composer.json b/composer.json index f4b23cd3..85a5f7fa 100644 --- a/composer.json +++ b/composer.json @@ -50,6 +50,7 @@ "amphp/http-client": "^5", "amphp/log": "^2", "amphp/php-cs-fixer-config": "^2", + "httpwg/structured-field-tests": "1.0", "league/uri-components": "^2.4.2 | ^7.1", "monolog/monolog": "^3", "phpunit/phpunit": "^9", @@ -82,6 +83,20 @@ "test": "@php -dzend.assertions=1 -dassert.exception=1 ./vendor/bin/phpunit", "code-style": "@php ./vendor/bin/php-cs-fixer fix" }, + "repositories": [ + { + "type": "package", + "package": { + "name": "httpwg/structured-field-tests", + "version": "1.0", + "source": { + "url": "https://github.com/httpwg/structured-field-tests", + "type": "git", + "reference": "origin/main" + } + } + } + ], "config": { "allow-plugins": { "ocramius/package-versions": false diff --git a/src/Driver/Http3Driver.php b/src/Driver/Http3Driver.php index db9a76b4..b3095de0 100644 --- a/src/Driver/Http3Driver.php +++ b/src/Driver/Http3Driver.php @@ -2,7 +2,6 @@ namespace Amp\Http\Server\Driver; -use Amp\ByteStream\ClosedException; use Amp\ByteStream\ReadableIterableStream; use Amp\CancelledException; use Amp\DeferredCancellation; @@ -255,7 +254,7 @@ public function handleConnection(Client $client, QuicConnection|Socket $connecti } $trailerDeferred = new DeferredFuture; - $bodyQueue = new Queue(); + $bodyQueue = new Queue; try { $trailers = new Trailers( @@ -296,6 +295,10 @@ public function handleConnection(Client $client, QuicConnection|Socket $connecti $expectedLength = null; } + if (isset($headers["priority"])) { + $this->updatePriority($stream, $headers["priority"]); + } + $dataSuspension = null; $body = new RequestBody( new ReadableIterableStream($bodyQueue->pipe()), @@ -439,6 +442,18 @@ function (int $bodySize) use (&$bodySizeLimit, &$dataSuspension) { $parser->abort(new Http3ConnectionException("A push stream must not be initiated by the client", Http3Error::H3_STREAM_CREATION_ERROR)); break; + case Http3Frame::PRIORITY_UPDATE_Request: + [, $streamId, $structuredUpdate] = $frame; + // The RFC says we _should_ temporarily buffer unknown stream ids. We currently don't for simplicity. To eventually improve? + if ($stream = $connection->getStream($streamId)) { + $this->updatePriority($stream, $structuredUpdate); + } + break; + + case Http3Frame::PRIORITY_UPDATE_Push: + $parser->abort(new Http3ConnectionException("No PRIORITY_UPDATE frame may be sent for unpromised push streams", Http3Error::H3_ID_ERROR)); + break; + default: $parser->abort(new Http3ConnectionException("An unexpected stream or frame was received", Http3Error::H3_FRAME_UNEXPECTED)); } @@ -451,6 +466,12 @@ function (int $bodySize) use (&$bodySizeLimit, &$dataSuspension) { } } + public function updatePriority(QuicSocket $socket, array|string $headers) + { + [$urgency, $incremental] = Http3Parser::parsePriority($headers); + $socket->setPriority($urgency + 124 /* 127 is default for QUIC, 3 is default for HTTP */, $incremental); + } + public function getPendingRequestCount(): int { return $this->requestStreams->count(); @@ -473,7 +494,6 @@ public function stop(): void $this->highestStreamId, )) || true); - $outstanding = $this->requestStreams->count(); if ($outstanding === 0) { $this->writer->close(); diff --git a/src/Driver/Internal/Http3/Http3Frame.php b/src/Driver/Internal/Http3/Http3Frame.php index 90e2c2ae..96bf24fb 100644 --- a/src/Driver/Internal/Http3/Http3Frame.php +++ b/src/Driver/Internal/Http3/Http3Frame.php @@ -13,5 +13,5 @@ enum Http3Frame: int case ORIGIN = 0x0c; case MAX_PUSH_ID = 0x0d; case PRIORITY_UPDATE_Request = 0xF0700; - case PRIORITY_UPDATE_Response = 0xF0701; + case PRIORITY_UPDATE_Push = 0xF0701; } diff --git a/src/Driver/Internal/Http3/Http3Parser.php b/src/Driver/Internal/Http3/Http3Parser.php index 51665323..48bbead1 100644 --- a/src/Driver/Internal/Http3/Http3Parser.php +++ b/src/Driver/Internal/Http3/Http3Parser.php @@ -3,6 +3,8 @@ namespace Amp\Http\Server\Driver\Internal\Http3; use Amp\Http\Http2\Http2Parser; +use Amp\Http\Server\Driver\Internal\Http3\Rfc8941\Boolean; +use Amp\Http\Server\Driver\Internal\Http3\Rfc8941\Number; use Amp\Pipeline\ConcurrentIterator; use Amp\Pipeline\Queue; use Amp\Quic\QuicConnection; @@ -310,7 +312,7 @@ public function process(): ConcurrentIterator return; } - if ($frame !== Http3Frame::GOAWAY || $frame !== Http3Frame::MAX_PUSH_ID || $frame !== Http3Frame::CANCEL_PUSH) { + if ($frame !== Http3Frame::GOAWAY && $frame !== Http3Frame::MAX_PUSH_ID && $frame !== Http3Frame::CANCEL_PUSH && $frame !== Http3Frame::PRIORITY_UPDATE_Request && $frame !== Http3Frame::PRIORITY_UPDATE_Push) { throw new Http3ConnectionException("An unexpected frame was received on the control stream", Http3Error::H3_FRAME_UNEXPECTED); } @@ -321,7 +323,11 @@ public function process(): ConcurrentIterator } return; } - $this->queue->push([$frame, $id]); + if ($frame === Http3Frame::PRIORITY_UPDATE_Request || $frame === Http3Frame::PRIORITY_UPDATE_Push) { + $this->queue->push([$frame, $id, \substr($contents, $tmpOff)]); + } else { + $this->queue->push([$frame, $id]); + } } // no break @@ -368,6 +374,31 @@ public function process(): ConcurrentIterator return $this->queue->iterate(); } + // Note: format is shared with HTTP/2 + public static function parsePriority(array|string $headers): ?array + { + $urgency = 3; + $incremental = false; + if ($priority = Rfc8941::parseDictionary($headers)) { + if (isset($priority["u"])) { + $number = $priority["u"]; + if ($number instanceof Number) { + $value = $number->item; + if (\is_int($value) && $value >= 0 && $value <= 7) { + $urgency = $number->item; + } + } + } + if (isset($priority["i"])) { + $bool = $priority["i"]; + if ($bool instanceof Boolean) { + $incremental = $bool->item; + } + } + } + return [$urgency, $incremental]; + } + public function abort(Http3ConnectionException $exception) { if (!$this->queue->isComplete()) { diff --git a/src/Driver/Internal/Http3/Rfc8941.php b/src/Driver/Internal/Http3/Rfc8941.php new file mode 100644 index 00000000..95b070ad --- /dev/null +++ b/src/Driver/Internal/Http3/Rfc8941.php @@ -0,0 +1,491 @@ +> + * @psalm-type Rfc8941SingleItem = Item + * @psalm-type Rfc8941BareItem = int|float|string|bool + * @psalm-type Rfc8941Parameters = array + */ +class Rfc8941 +{ + /** + * @param string[]|string $value + * @psalm-return null|list + */ + public static function parseList(array|string $value): ?array + { + $string = \is_array($value) ? \implode(",", $value) : $value; + + $i = \strspn($string, " "); + $len = \strlen($string); + if ($len === $i) { + return []; + } + + $list = []; + while (true) { + if (null === $list[] = self::parseItemOrInnerList($string, $i)) { + return null; + } + $i += \strspn($string, " \t", $i); + if ($i >= $len) { + return $list; + } + if ($string[$i] !== ",") { + return null; + } + $i += \strspn($string, " \t", ++$i); + if ($i >= $len) { + return null; + } + } + } + + /** + * @param string[]|string $value + * @psalm-return array + */ + public static function parseDictionary(array|string $value): ?array + { + $string = \is_array($value) ? \implode(",", $value) : $value; + + $i = \strspn($string, " "); + $len = \strlen($string); + if ($len === $i) { + return []; + } + + $values = []; + while (true) { + $i += \strspn($string, " ", $i); + if (null === $key = self::parseKey($string, $i)) { + return null; + } + if ($i < $len && $string[$i] === "=") { + ++$i; + if (null === $values[$key] = self::parseItemOrInnerList($string, $i)) { + return null; + } + } else { + if (null === $parameters = self::parseParameters($string, $i)) { + return null; + } + $values[$key] = new Boolean(true, $parameters); + } + $i += \strspn($string, " \t", $i); + if ($i >= $len) { + return $values; + } + if ($string[$i] !== ",") { + return null; + } + $i += \strspn($string, " \t", ++$i); + if ($i >= $len) { + return null; + } + } + } + + /** @psalm-param null|Rfc8941SingleItem */ + public static function parseItem(string $string): ?Item + { + $i = \strspn($string, " "); + if ($i === \strlen($string)) { + return null; + } + $parsed = self::parseItemInternal($string, $i); + return $i + \strspn($string, " ", $i) < \strlen($string) ? null : $parsed; + } + + /** @psalm-return null|Rfc8941ListItem */ + private static function parseItemOrInnerList(string $string, int &$i): ?Item + { + $len = \strlen($string); + if ($string[$i] === "(") { + $innerList = []; + ++$i; + while (true) { + $i += \strspn($string, " ", $i); + if ($i >= $len) { + return null; + } + if ($string[$i] === ")") { + ++$i; + if (null === $params = self::parseParameters($string, $i)) { + return null; + } + return new InnerList($innerList, $params); + } + $chr = $string[$i - 1]; + if ($chr !== " " && $chr !== "(") { + return null; + } + if (null === $innerList[] = self::parseItemInternal($string, $i)) { + return null; + } + } + } + return self::parseItemInternal($string, $i); + } + + /** @psalm-param null|Rfc8941SingleItem */ + private static function parseItemInternal(string $string, int &$i): ?Item + { + if (null === $value = self::parseBareItem($string, $i, $class)) { + return null; + } + if (null === $parameters = self::parseParameters($string, $i)) { + return null; + } + return new $class($value, $parameters); + } + + public static function parseIntegerOrDecimal(string $string): null|int|float + { + if ($string === "") { + return null; + } + + $i = \strspn($string, " "); + $chr = \ord($string[$i]); + if ($chr === \ord("-") || ($chr >= \ord('0') || $chr <= \ord('9'))) { + $parsed = self::parseIntegerOrDecimalInternal($string, $i); + return $i + \strspn($string, " ", $i) < \strlen($string) ? null : $parsed; + } + return null; + } + + private static function parseIntegerOrDecimalInternal(string $string, int &$i): null|int|float + { + $len = \strlen($string); + $sign = 1; + if ($string[$i] === "-") { + ++$i; + $sign = -1; + } + $digits = \strspn($string, "0123456789", $i); + if ($digits < 1) { + return null; + } + $decimaldot = $i + $digits + 1; + if ($decimaldot < $len && $string[$decimaldot - 1] === ".") { + if ($digits > 12) { + return null; + } + $decimals = \strspn($string, "0123456789", $decimaldot); + if ($decimals < 1 || $decimals > 3) { + return null; + } + $length = $decimaldot - $i + $decimals; + $num = $sign * \substr($string, $i, $length); + $i += $length; + } elseif ($digits > 15) { + return null; + } else { + $num = $sign * \substr($string, $i, $digits); + $i += $digits; + } + return $num; + } + + public static function parseString(string $string): ?string + { + if ($string === "") { + return null; + } + + $i = \strspn($string, " "); + if ($string[$i++] === '"') { + $parsed = self::parseStringInternal($string, $i); + return $i + \strspn($string, " ", $i) < \strlen($string) ? null : $parsed; + } + return null; + } + + private static function parseStringInternal(string $string, int &$i): ?string + { + $start = $i; + $len = \strlen($string); + $foundslash = false; + while (true) { + $i += \strspn($string, " !#$%&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[]^_`abcdefghijklmnopqrstuvwxyz{|}~", $i); + if ($i >= $len) { + return null; + } + if ($string[$i] === '"') { + $str = \substr($string, $start, $i++ - $start); + if ($foundslash) { + return \stripslashes($str); + } + return $str; + } + if ($string[$i] === "\\") { + if (++$i >= $len) { + return null; + } + $foundslash = true; + $chr = $string[$i++]; + if ($chr !== '"' && $chr !== "\\") { + return null; + } + } else { + return null; + } + } + } + + public static function parseToken(string $string): ?string + { + if ($string === "") { + return null; + } + + $i = \strspn($string, " "); + $chr = \ord($string[$i]); + if ($chr === \ord("*") || ($chr >= \ord('A') && $chr <= \ord("Z")) || ($chr >= \ord('a') && $chr <= \ord("z"))) { + $parsed = self::parseTokenInternal($string, $i); + return $i + \strspn($string, " ", $i) < \strlen($string) ? null : $parsed; + } + return null; + } + + private static function parseTokenInternal(string $string, int &$i): string + { + $length = \strspn($string, ":/!#$%&'*+-.^_`|~0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ", $i); + $str = \substr($string, $i, $length); + $i += $length; + return $str; + } + + public static function parseByteSequence(string $string): ?string + { + if ($string === "") { + return null; + } + + $i = \strspn($string, " "); + if ($string[$i++] === ':') { + $parsed = self::parseByteSequenceInternal($string, $i); + return $i + \strspn($string, " ", $i) < \strlen($string) ? null : $parsed; + } + return null; + } + + private static function parseByteSequenceInternal(string $string, int &$i): ?string + { + $length = \strspn($string, "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=", $i); + $str = \base64_decode(\substr($string, $i, $length)); + $i += $length; + if (!isset($string[$i]) || $string[$i++] !== ":") { + return null; + } + return $str === false ? null : $str; + } + + public static function parseBoolean(string $string): ?bool + { + if ($string === "") { + return null; + } + + $i = \strspn($string, " "); + if ($string[$i++] === '?') { + $parsed = self::parseBooleanInternal($string, $i); + return $i + \strspn($string, " ", $i) < \strlen($string) ? null : $parsed; + } + return null; + } + + private static function parseBooleanInternal(string $string, int &$i): ?bool + { + if (!isset($string[$i])) { + return null; + } + $chr = $string[$i++]; + if ($chr === "0") { + return false; + } + if ($chr === "1") { + return true; + } + return null; + } + + public static function parseDate(string $string): ?int + { + if ($string === "") { + return null; + } + + $i = \strspn($string, " "); + if ($string[$i++] === '@') { + $parsed = self::parseDateInternal($string, $i); + return $i + \strspn($string, " ", $i) < \strlen($string) ? null : $parsed; + } + return null; + } + + private static function parseDateInternal(string $string, int &$i): ?int + { + if (!isset($string[$i])) { + return null; + } + $start = $i; + if ($string[$i] === "-") { + ++$i; + } + $length = \strspn($string, "0123456789", $i); + if ($length < 1 || $length > 15) { + return null; + } + if ($start !== $i) { + $i += $length++; + } else { + $i += $length; + } + return (int) \substr($string, $start, $length); + } + + public static function parseDisplayString(string $string): ?string + { + $i = \strspn($string, " "); + if (\strlen($string) < $i + 3) { + return null; + } + + if ($string[$i++] === '%' && $string[$i++] === '"') { + $parsed = self::parseDisplayStringInternal($string, $i); + return $i + \strspn($string, " ", $i) < \strlen($string) ? null : $parsed; + } + return null; + } + + private static function parseDisplayStringInternal(string $string, int &$i): ?string + { + $start = $i; + $len = \strlen($string); + $buf = ""; + while (true) { + $i += \strspn($string, " !#$&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[]^_`abcdefghijklmnopqrstuvwxyz{|}~\\", $i); + if ($i >= $len) { + return null; + } + if ($string[$i] === '%') { + $buf .= \substr($string, $start, $i++ - $start); + $hexlen = \strspn($string, "0123456789abcdef", $i, 2); + if ($hexlen !== 2) { + return null; + } + $buf .= \chr(\hexdec(\substr($string, $i, 2))); + $start = $i += 2; + continue; + } + if ($string[$i] === '"') { + $buf .= \substr($string, $start, $i++ - $start); + if (!\preg_match('//u', $buf)) { + return null; + } + return $buf; + } + return null; + } + } + + /** @psalm-return null|Rfc8941BareItem */ + private static function parseBareItem(string $string, int &$i, &$class = ""): null|int|float|string|bool + { + $chr = \ord($string[$i]); + if ($chr === \ord("-") || ($chr >= \ord('0') && $chr <= \ord('9'))) { + $class = Number::class; + return self::parseIntegerOrDecimalInternal($string, $i); + } + if ($chr === \ord('"')) { + $class = Str::class; + ++$i; + return self::parseStringInternal($string, $i); + } + if ($chr === \ord("*") || ($chr >= \ord('A') && $chr <= \ord("Z")) || ($chr >= \ord('a') && $chr <= \ord("z"))) { + $class = Token::class; + return self::parseTokenInternal($string, $i); + } + if ($chr === \ord(":")) { + $class = Bytes::class; + ++$i; + return self::parseByteSequenceInternal($string, $i); + } + if ($chr === \ord("%")) { + $class = DisplayString::class; + if (!isset($string[++$i]) || $string[$i++] !== '"') { + return null; + } + return self::parseDisplayStringInternal($string, $i); + } + if ($chr === \ord("?")) { + $class = Boolean::class; + ++$i; + return self::parseBooleanInternal($string, $i); + } + if ($chr === \ord("@")) { + $class = Date::class; + ++$i; + return self::parseDateInternal($string, $i); + } + return null; + } + + /** @psalm-return null|Rfc8941Parameters */ + private static function parseParameters(string $string, int &$i): ?array + { + $parameters = []; + for ($len = \strlen($string); $i < $len;) { + if ($string[$i] !== ";") { + break; + } + $i += \strspn($string, " ", ++$i); + + if ($i >= $len) { + return null; + } + + if (null === $key = self::parseKey($string, $i)) { + return null; + } + + if ($i < $len && $string[$i] === "=") { + ++$i; + if (null === $item = self::parseBareItem($string, $i)) { + return null; + } + $parameters[$key] = $item; + } else { + $parameters[$key] = true; + } + } + return $parameters; + } + + private static function parseKey(string $string, int &$i): ?string + { + $chr = \ord($string[$i]); + if ($chr !== \ord("*") && ($chr < \ord('a') || $chr > \ord('z'))) { + return null; + } + + $keystart = $i++; + $i += \strspn($string, "*.-_abcdefghijklmnopqrstuvwxyz0123456789", $i); + return \substr($string, $keystart, $i - $keystart); + } +} diff --git a/src/Driver/Internal/Http3/Rfc8941/Boolean.php b/src/Driver/Internal/Http3/Rfc8941/Boolean.php new file mode 100644 index 00000000..c67b1dd5 --- /dev/null +++ b/src/Driver/Internal/Http3/Rfc8941/Boolean.php @@ -0,0 +1,18 @@ +executeTest(...$args); + } + + public function executeTest(array $raw, string $header_type, $expected = null, bool $must_fail = false, bool $can_fail = false, $canonical = null, ...$ignoredArgs) + { + switch ($header_type) { + case "item": + $parsed = Rfc8941::parseItem($raw[0]); + break; + + case "list": + $parsed = Rfc8941::parseList($raw); + break; + + case "dictionary": + $parsed = Rfc8941::parseDictionary($raw); + break; + + default: + $this->fail("Unknown $header_type in dataset"); + } + + self::destructureItems($parsed); + + if ($parsed === null) { + $this->assertTrue($must_fail || $can_fail); + } elseif ($must_fail) { + $this->assertNull($parsed); + } else { + $this->assertSame($expected, $parsed); + } + } + + public function provideAll(): array + { + $cases = []; + foreach (\glob(__DIR__ . "/../../vendor/httpwg/structured-field-tests/*.json") as $file) { + foreach (\json_decode(\file_get_contents($file), true) as $case) { + + // We don't test merging two headers here + self::sanitize($case); + $cases[\basename($file) . ": {$case["name"]}"] = [$case]; + } + } + return $cases; + } + + public static function destructureItems(&$parsed) + { + if ($parsed instanceof Item) { + $parsed = [$parsed->item, $parsed->parameters]; + } + if (\is_array($parsed)) { + foreach ($parsed as &$value) { + self::destructureItems($value); + } + } + } + + public static function sanitize(&$case) + { + $expected = &$case["expected"]; + self::sanitizeType($expected); + if (\is_array($expected)) { + if ($case["header_type"] === "list" || $case["header_type"] === "dictionary") { + if ($case["header_type"] === "dictionary") { + self::recombineDictonary($expected); + } + foreach ($expected as &$item) { + if (\is_array($item[0])) { + foreach ($item[0] as &$innerItem) { + self::recombineDictonary($innerItem[1]); + } + } + self::recombineDictonary($item[1]); + } + } else { + self::recombineDictonary($expected[1]); + } + } + } + + public static function recombineDictonary(&$data) + { + $data = \array_combine(\array_column($data, 0), \array_column($data, 1)); + } + + public static function sanitizeType(&$data) + { + if (\is_array($data)) { + foreach ($data as &$value) { + if (isset($value["__type"])) { + if ($value["__type"] === "binary") { + // base32? wtf. + $str = \rtrim($value["value"], "="); + $buf = ""; + $keys = \array_flip(\array_merge(\range('A', 'Z'), \range(2, 7))); + $byte = 0; + for ($i = 0, $len = \strlen($str); $i < $len; ++$i) { + $shift = (5 * ($i + 1)) % 8; + $byte = $byte << 5 | $keys[$str[$i]]; + if ($shift < 5) { + $buf .= \chr($byte >> $shift); + $byte &= (1 << $shift) - 1; + } + } + if ((5 * $i) % 8 >= 5) { + $buf .= \chr($byte << (8 - (5 * $i) % 8)); + } + $value = $buf; + } else { + $value = $value["value"]; + } + } else { + self::sanitizeType($value); + } + } + } + } +}