|  | // Copyright 2016 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "quiche/http2/http2_constants.h" | 
|  |  | 
|  | #include "absl/strings/str_cat.h" | 
|  | #include "absl/strings/str_format.h" | 
|  | #include "absl/strings/string_view.h" | 
|  | #include "quiche/common/platform/api/quiche_flag_utils.h" | 
|  | #include "quiche/common/platform/api/quiche_flags.h" | 
|  | #include "quiche/common/platform/api/quiche_logging.h" | 
|  |  | 
|  | namespace http2 { | 
|  |  | 
|  | std::string Http2FrameTypeToString(Http2FrameType v) { | 
|  | switch (v) { | 
|  | case Http2FrameType::DATA: | 
|  | return "DATA"; | 
|  | case Http2FrameType::HEADERS: | 
|  | return "HEADERS"; | 
|  | case Http2FrameType::PRIORITY: | 
|  | return "PRIORITY"; | 
|  | case Http2FrameType::RST_STREAM: | 
|  | return "RST_STREAM"; | 
|  | case Http2FrameType::SETTINGS: | 
|  | return "SETTINGS"; | 
|  | case Http2FrameType::PUSH_PROMISE: | 
|  | return "PUSH_PROMISE"; | 
|  | case Http2FrameType::PING: | 
|  | return "PING"; | 
|  | case Http2FrameType::GOAWAY: | 
|  | return "GOAWAY"; | 
|  | case Http2FrameType::WINDOW_UPDATE: | 
|  | return "WINDOW_UPDATE"; | 
|  | case Http2FrameType::CONTINUATION: | 
|  | return "CONTINUATION"; | 
|  | case Http2FrameType::ALTSVC: | 
|  | return "ALTSVC"; | 
|  | case Http2FrameType::PRIORITY_UPDATE: | 
|  | return "PRIORITY_UPDATE"; | 
|  | } | 
|  | return absl::StrCat("UnknownFrameType(", static_cast<int>(v), ")"); | 
|  | } | 
|  |  | 
|  | std::string Http2FrameTypeToString(uint8_t v) { | 
|  | return Http2FrameTypeToString(static_cast<Http2FrameType>(v)); | 
|  | } | 
|  |  | 
|  | std::string Http2FrameFlagsToString(Http2FrameType type, uint8_t flags) { | 
|  | std::string s; | 
|  | // Closure to append flag name |v| to the std::string |s|, | 
|  | // and to clear |bit| from |flags|. | 
|  | auto append_and_clear = [&s, &flags](absl::string_view v, uint8_t bit) { | 
|  | if (!s.empty()) { | 
|  | s.push_back('|'); | 
|  | } | 
|  | absl::StrAppend(&s, v); | 
|  | flags ^= bit; | 
|  | }; | 
|  | if (flags & 0x01) { | 
|  | if (type == Http2FrameType::DATA || type == Http2FrameType::HEADERS) { | 
|  | append_and_clear("END_STREAM", Http2FrameFlag::END_STREAM); | 
|  | } else if (type == Http2FrameType::SETTINGS || | 
|  | type == Http2FrameType::PING) { | 
|  | append_and_clear("ACK", Http2FrameFlag::ACK); | 
|  | } | 
|  | } | 
|  | if (flags & 0x04) { | 
|  | if (type == Http2FrameType::HEADERS || | 
|  | type == Http2FrameType::PUSH_PROMISE || | 
|  | type == Http2FrameType::CONTINUATION) { | 
|  | append_and_clear("END_HEADERS", Http2FrameFlag::END_HEADERS); | 
|  | } | 
|  | } | 
|  | if (flags & 0x08) { | 
|  | if (type == Http2FrameType::DATA || type == Http2FrameType::HEADERS || | 
|  | type == Http2FrameType::PUSH_PROMISE) { | 
|  | append_and_clear("PADDED", Http2FrameFlag::PADDED); | 
|  | } | 
|  | } | 
|  | if (flags & 0x20) { | 
|  | if (type == Http2FrameType::HEADERS) { | 
|  | append_and_clear("PRIORITY", Http2FrameFlag::PRIORITY); | 
|  | } | 
|  | } | 
|  | if (flags != 0) { | 
|  | append_and_clear(absl::StrFormat("0x%02x", flags), flags); | 
|  | } | 
|  | QUICHE_DCHECK_EQ(0, flags); | 
|  | return s; | 
|  | } | 
|  | std::string Http2FrameFlagsToString(uint8_t type, uint8_t flags) { | 
|  | return Http2FrameFlagsToString(static_cast<Http2FrameType>(type), flags); | 
|  | } | 
|  |  | 
|  | std::string Http2ErrorCodeToString(uint32_t v) { | 
|  | switch (v) { | 
|  | case 0x0: | 
|  | return "NO_ERROR"; | 
|  | case 0x1: | 
|  | return "PROTOCOL_ERROR"; | 
|  | case 0x2: | 
|  | return "INTERNAL_ERROR"; | 
|  | case 0x3: | 
|  | return "FLOW_CONTROL_ERROR"; | 
|  | case 0x4: | 
|  | return "SETTINGS_TIMEOUT"; | 
|  | case 0x5: | 
|  | return "STREAM_CLOSED"; | 
|  | case 0x6: | 
|  | return "FRAME_SIZE_ERROR"; | 
|  | case 0x7: | 
|  | return "REFUSED_STREAM"; | 
|  | case 0x8: | 
|  | return "CANCEL"; | 
|  | case 0x9: | 
|  | return "COMPRESSION_ERROR"; | 
|  | case 0xa: | 
|  | return "CONNECT_ERROR"; | 
|  | case 0xb: | 
|  | return "ENHANCE_YOUR_CALM"; | 
|  | case 0xc: | 
|  | return "INADEQUATE_SECURITY"; | 
|  | case 0xd: | 
|  | return "HTTP_1_1_REQUIRED"; | 
|  | } | 
|  | return absl::StrCat("UnknownErrorCode(0x", absl::Hex(v), ")"); | 
|  | } | 
|  | std::string Http2ErrorCodeToString(Http2ErrorCode v) { | 
|  | return Http2ErrorCodeToString(static_cast<uint32_t>(v)); | 
|  | } | 
|  |  | 
|  | std::string Http2SettingsParameterToString(uint32_t v) { | 
|  | switch (v) { | 
|  | case 0x1: | 
|  | return "HEADER_TABLE_SIZE"; | 
|  | case 0x2: | 
|  | return "ENABLE_PUSH"; | 
|  | case 0x3: | 
|  | return "MAX_CONCURRENT_STREAMS"; | 
|  | case 0x4: | 
|  | return "INITIAL_WINDOW_SIZE"; | 
|  | case 0x5: | 
|  | return "MAX_FRAME_SIZE"; | 
|  | case 0x6: | 
|  | return "MAX_HEADER_LIST_SIZE"; | 
|  | } | 
|  | return absl::StrCat("UnknownSettingsParameter(0x", absl::Hex(v), ")"); | 
|  | } | 
|  | std::string Http2SettingsParameterToString(Http2SettingsParameter v) { | 
|  | return Http2SettingsParameterToString(static_cast<uint32_t>(v)); | 
|  | } | 
|  |  | 
|  | // Invalid HTTP/2 header names according to | 
|  | // https://datatracker.ietf.org/doc/html/rfc7540#section-8.1.2.2. | 
|  | // TODO(b/78024822): Consider adding "upgrade" to this set. | 
|  | constexpr char const* kHttp2InvalidHeaderNames[] = { | 
|  | "connection",        "host", "keep-alive", "proxy-connection", | 
|  | "transfer-encoding", "", | 
|  | }; | 
|  |  | 
|  | constexpr char const* kHttp2InvalidHeaderNamesOld[] = { | 
|  | "connection", "host", "keep-alive", "proxy-connection", "transfer-encoding", | 
|  | }; | 
|  |  | 
|  | const InvalidHeaderSet& GetInvalidHttp2HeaderSet() { | 
|  | if (!GetQuicheReloadableFlag(quic, quic_verify_request_headers_2)) { | 
|  | static const auto* invalid_header_set_old = | 
|  | new InvalidHeaderSet(std::begin(http2::kHttp2InvalidHeaderNamesOld), | 
|  | std::end(http2::kHttp2InvalidHeaderNamesOld)); | 
|  | return *invalid_header_set_old; | 
|  | } | 
|  | QUICHE_RELOADABLE_FLAG_COUNT_N(quic_verify_request_headers_2, 3, 3); | 
|  | static const auto* invalid_header_set = | 
|  | new InvalidHeaderSet(std::begin(http2::kHttp2InvalidHeaderNames), | 
|  | std::end(http2::kHttp2InvalidHeaderNames)); | 
|  | return *invalid_header_set; | 
|  | } | 
|  |  | 
|  | }  // namespace http2 |