| /** @file |
| * |
| * Fundamental HTTP/2 protocol definitions and parsers. |
| * |
| * @section license License |
| * |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance |
| * with the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include "proxy/hdrs/VersionConverter.h" |
| #include "proxy/hdrs/HeaderValidator.h" |
| #include "proxy/http2/HTTP2.h" |
| #include "proxy/http2/HPACK.h" |
| |
| #include "tscore/ink_assert.h" |
| #include "tsutil/LocalBuffer.h" |
| |
| #include "../../records/P_RecCore.h" |
| #include "../../records/P_RecProcess.h" |
| |
| const char *const HTTP2_CONNECTION_PREFACE = "PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n"; |
| |
| static const uint32_t HTTP2_MAX_TABLE_SIZE_LIMIT = 64 * 1024; |
| |
| namespace |
| { |
| struct Http2HeaderName { |
| const char *name = nullptr; |
| int name_len = 0; |
| }; |
| |
| static VersionConverter hvc; |
| |
| } // namespace |
| |
| // Statistics |
| Http2StatsBlock http2_rsb; |
| |
| Metrics::Counter::AtomicType *http2_frame_metrics_in[HTTP2_FRAME_TYPE_MAX + 1]; |
| |
| union byte_pointer { |
| byte_pointer(void *p) : ptr(p) {} |
| void *ptr; |
| uint8_t *u8; |
| uint16_t *u16; |
| uint32_t *u32; |
| }; |
| |
| template <typename T> union byte_addressable_value { |
| uint8_t bytes[sizeof(T)]; |
| T value; |
| }; |
| |
| static void |
| write_and_advance(byte_pointer &dst, const uint8_t *src, size_t length) |
| { |
| memcpy(dst.u8, src, length); |
| dst.u8 += length; |
| } |
| |
| static void |
| write_and_advance(byte_pointer &dst, uint32_t src) |
| { |
| byte_addressable_value<uint32_t> pval; |
| |
| // cppcheck-suppress unreadVariable ; it's an union and be read as pval.bytes |
| pval.value = htonl(src); |
| memcpy(dst.u8, pval.bytes, sizeof(pval.bytes)); |
| dst.u8 += sizeof(pval.bytes); |
| } |
| |
| static void |
| write_and_advance(byte_pointer &dst, uint16_t src) |
| { |
| byte_addressable_value<uint16_t> pval; |
| |
| // cppcheck-suppress unreadVariable ; it's an union and be read as pval.bytes |
| pval.value = htons(src); |
| memcpy(dst.u8, pval.bytes, sizeof(pval.bytes)); |
| dst.u8 += sizeof(pval.bytes); |
| } |
| |
| static void |
| write_and_advance(byte_pointer &dst, uint8_t src) |
| { |
| *dst.u8 = src; |
| dst.u8++; |
| } |
| |
| template <unsigned N> |
| static void |
| memcpy_and_advance(uint8_t (&dst)[N], byte_pointer &src) |
| { |
| memcpy(dst, src.u8, N); |
| src.u8 += N; |
| } |
| |
| static void |
| memcpy_and_advance(uint8_t(&dst), byte_pointer &src) |
| { |
| dst = *src.u8; |
| ++src.u8; |
| } |
| |
| bool |
| http2_frame_header_is_valid(const Http2FrameHeader &hdr, unsigned max_frame_size) |
| { |
| // 6.1 If a DATA frame is received whose stream identifier field is 0x0, the recipient MUST |
| // respond with a connection error (Section 5.4.1) of type PROTOCOL_ERROR. |
| if (hdr.type == HTTP2_FRAME_TYPE_DATA && hdr.streamid == 0) { |
| return false; |
| } |
| |
| return true; |
| } |
| |
| bool |
| http2_settings_parameter_is_valid(const Http2SettingsParameter ¶m) |
| { |
| // Static maximum values for Settings parameters. |
| static const uint32_t settings_max[HTTP2_SETTINGS_MAX] = { |
| 0, |
| UINT_MAX, // HTTP2_SETTINGS_HEADER_TABLE_SIZE |
| 1, // HTTP2_SETTINGS_ENABLE_PUSH |
| UINT_MAX, // HTTP2_SETTINGS_MAX_CONCURRENT_STREAMS |
| HTTP2_MAX_WINDOW_SIZE, // HTTP2_SETTINGS_INITIAL_WINDOW_SIZE |
| 16777215, // HTTP2_SETTINGS_MAX_FRAME_SIZE |
| UINT_MAX, // HTTP2_SETTINGS_MAX_HEADER_LIST_SIZE |
| }; |
| |
| if (param.id == 0 || param.id >= HTTP2_SETTINGS_MAX) { |
| // Do nothing - 6.5.2 Unsupported parameters MUST be ignored |
| return true; |
| } |
| |
| if (param.value > settings_max[param.id]) { |
| return false; |
| } |
| |
| if (param.id == HTTP2_SETTINGS_ENABLE_PUSH && param.value != 0 && param.value != 1) { |
| return false; |
| } |
| |
| if (param.id == HTTP2_SETTINGS_MAX_FRAME_SIZE && (param.value < (1 << 14) || param.value > (1 << 24) - 1)) { |
| return false; |
| } |
| |
| return true; |
| } |
| |
| // 4.1. Frame Format |
| // |
| // 0 1 2 3 |
| // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 |
| // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ |
| // | Length (24) | |
| // +---------------+---------------+---------------+ |
| // | Type (8) | Flags (8) | |
| // +-+-+-----------+---------------+-------------------------------+ |
| // |R| Stream Identifier (31) | |
| // +=+=============================================================+ |
| // | Frame Payload (0...) ... |
| // +---------------------------------------------------------------+ |
| |
| bool |
| http2_parse_frame_header(IOVec iov, Http2FrameHeader &hdr) |
| { |
| byte_pointer ptr(iov.iov_base); |
| byte_addressable_value<uint32_t> length_and_type; |
| byte_addressable_value<uint32_t> streamid; |
| |
| if (unlikely(iov.iov_len < HTTP2_FRAME_HEADER_LEN)) { |
| return false; |
| } |
| |
| memcpy_and_advance(length_and_type.bytes, ptr); |
| memcpy_and_advance(hdr.flags, ptr); |
| memcpy_and_advance(streamid.bytes, ptr); |
| |
| hdr.length = ntohl(length_and_type.value) >> 8; |
| hdr.type = ntohl(length_and_type.value) & 0xff; |
| streamid.bytes[0] &= 0x7f; // Clear the high reserved bit |
| hdr.streamid = ntohl(streamid.value); |
| |
| return true; |
| } |
| |
| bool |
| http2_write_frame_header(const Http2FrameHeader &hdr, IOVec iov) |
| { |
| byte_pointer ptr(iov.iov_base); |
| |
| if (unlikely(iov.iov_len < HTTP2_FRAME_HEADER_LEN)) { |
| return false; |
| } |
| |
| byte_addressable_value<uint32_t> length; |
| // cppcheck-suppress unreadVariable ; it's an union and be read as pval.bytes |
| length.value = htonl(hdr.length); |
| // MSB length.bytes[0] is unused. |
| write_and_advance(ptr, length.bytes[1]); |
| write_and_advance(ptr, length.bytes[2]); |
| write_and_advance(ptr, length.bytes[3]); |
| |
| write_and_advance(ptr, hdr.type); |
| write_and_advance(ptr, hdr.flags); |
| write_and_advance(ptr, hdr.streamid); |
| |
| return true; |
| } |
| |
| bool |
| http2_write_rst_stream(uint32_t error_code, IOVec iov) |
| { |
| byte_pointer ptr(iov.iov_base); |
| |
| write_and_advance(ptr, error_code); |
| |
| return true; |
| } |
| |
| bool |
| http2_write_settings(const Http2SettingsParameter ¶m, const IOVec &iov) |
| { |
| byte_pointer ptr(iov.iov_base); |
| |
| if (unlikely(iov.iov_len < HTTP2_SETTINGS_PARAMETER_LEN)) { |
| return false; |
| } |
| |
| write_and_advance(ptr, param.id); |
| write_and_advance(ptr, param.value); |
| |
| return true; |
| } |
| |
| bool |
| http2_write_ping(const uint8_t *opaque_data, IOVec iov) |
| { |
| byte_pointer ptr(iov.iov_base); |
| |
| if (unlikely(iov.iov_len < HTTP2_PING_LEN)) { |
| return false; |
| } |
| |
| write_and_advance(ptr, opaque_data, HTTP2_PING_LEN); |
| |
| return true; |
| } |
| |
| bool |
| http2_write_goaway(const Http2Goaway &goaway, IOVec iov) |
| { |
| byte_pointer ptr(iov.iov_base); |
| |
| if (unlikely(iov.iov_len < HTTP2_GOAWAY_LEN)) { |
| return false; |
| } |
| |
| write_and_advance(ptr, goaway.last_streamid); |
| write_and_advance(ptr, static_cast<uint32_t>(goaway.error_code)); |
| |
| return true; |
| } |
| |
| bool |
| http2_write_window_update(const uint32_t new_size, const IOVec &iov) |
| { |
| byte_pointer ptr(iov.iov_base); |
| write_and_advance(ptr, new_size); |
| |
| return true; |
| } |
| |
| bool |
| http2_write_push_promise(const Http2PushPromise &push_promise, const uint8_t *src, size_t length, const IOVec &iov) |
| { |
| byte_pointer ptr(iov.iov_base); |
| write_and_advance(ptr, push_promise.promised_streamid); |
| write_and_advance(ptr, src, length); |
| return true; |
| } |
| |
| bool |
| http2_parse_headers_parameter(IOVec iov, Http2HeadersParameter ¶ms) |
| { |
| byte_pointer ptr(iov.iov_base); |
| memcpy_and_advance(params.pad_length, ptr); |
| |
| return true; |
| } |
| |
| bool |
| http2_parse_priority_parameter(IOVec iov, Http2Priority &priority) |
| { |
| byte_pointer ptr(iov.iov_base); |
| byte_addressable_value<uint32_t> dependency; |
| |
| memcpy_and_advance(dependency.bytes, ptr); |
| |
| priority.exclusive_flag = dependency.bytes[0] & 0x80; |
| |
| dependency.bytes[0] &= 0x7f; // Clear the highest bit for exclusive flag |
| priority.stream_dependency = ntohl(dependency.value); |
| |
| memcpy_and_advance(priority.weight, ptr); |
| |
| return true; |
| } |
| |
| bool |
| http2_parse_rst_stream(IOVec iov, Http2RstStream &rst_stream) |
| { |
| byte_pointer ptr(iov.iov_base); |
| byte_addressable_value<uint32_t> ec; |
| |
| memcpy_and_advance(ec.bytes, ptr); |
| |
| rst_stream.error_code = ntohl(ec.value); |
| |
| return true; |
| } |
| |
| bool |
| http2_parse_settings_parameter(IOVec iov, Http2SettingsParameter ¶m) |
| { |
| byte_pointer ptr(iov.iov_base); |
| byte_addressable_value<uint16_t> pid; |
| byte_addressable_value<uint32_t> pval; |
| |
| if (unlikely(iov.iov_len < HTTP2_SETTINGS_PARAMETER_LEN)) { |
| return false; |
| } |
| |
| memcpy_and_advance(pid.bytes, ptr); |
| memcpy_and_advance(pval.bytes, ptr); |
| |
| param.id = ntohs(pid.value); |
| param.value = ntohl(pval.value); |
| |
| return true; |
| } |
| |
| bool |
| http2_parse_goaway(IOVec iov, Http2Goaway &goaway) |
| { |
| byte_pointer ptr(iov.iov_base); |
| byte_addressable_value<uint32_t> sid; |
| byte_addressable_value<uint32_t> ec; |
| |
| memcpy_and_advance(sid.bytes, ptr); |
| memcpy_and_advance(ec.bytes, ptr); |
| |
| goaway.last_streamid = ntohl(sid.value); |
| goaway.error_code = static_cast<Http2ErrorCode>(ntohl(ec.value)); |
| return true; |
| } |
| |
| bool |
| http2_parse_window_update(IOVec iov, uint32_t &size) |
| { |
| byte_pointer ptr(iov.iov_base); |
| byte_addressable_value<uint32_t> s; |
| |
| memcpy_and_advance(s.bytes, ptr); |
| |
| size = ntohl(s.value); |
| |
| return true; |
| } |
| |
| ParseResult |
| http2_convert_header_from_2_to_1_1(HTTPHdr *headers) |
| { |
| if (hvc.convert(*headers, 2, 1) == 0) { |
| return PARSE_RESULT_DONE; |
| } else { |
| return PARSE_RESULT_ERROR; |
| } |
| } |
| |
| /** |
| Convert HTTP/1.1 HTTPHdr to HTTP/2 |
| |
| Assuming HTTP/2 Pseudo-Header Fields are reserved by passing a version to `HTTPHdr::create()`. |
| */ |
| ParseResult |
| http2_convert_header_from_1_1_to_2(HTTPHdr *headers) |
| { |
| if (hvc.convert(*headers, 1, 2) == 0) { |
| return PARSE_RESULT_DONE; |
| } else { |
| return PARSE_RESULT_ERROR; |
| } |
| |
| return PARSE_RESULT_DONE; |
| } |
| |
| Http2ErrorCode |
| http2_encode_header_blocks(HTTPHdr *in, uint8_t *out, uint32_t out_len, uint32_t *len_written, HpackHandle &handle, |
| int32_t maximum_table_size) |
| { |
| // Limit the maximum table size to the configured value or 64kB at maximum, which is the size advertised by major clients |
| maximum_table_size = |
| std::min(maximum_table_size, static_cast<int32_t>(std::min(Http2::header_table_size_limit, HTTP2_MAX_TABLE_SIZE_LIMIT))); |
| // Set maximum table size only if it is different from current maximum size |
| if (maximum_table_size == hpack_get_maximum_table_size(handle)) { |
| maximum_table_size = -1; |
| } |
| |
| // TODO: It would be better to split Cookie header value |
| int64_t result = hpack_encode_header_block(handle, out, out_len, in, maximum_table_size); |
| if (result < 0) { |
| return Http2ErrorCode::HTTP2_ERROR_COMPRESSION_ERROR; |
| } |
| if (len_written) { |
| *len_written = result; |
| } |
| return Http2ErrorCode::HTTP2_ERROR_NO_ERROR; |
| } |
| |
| /* |
| * Decode Header Blocks to Header List. |
| */ |
| Http2ErrorCode |
| http2_decode_header_blocks(HTTPHdr *hdr, const uint8_t *buf_start, const uint32_t buf_len, uint32_t *len_read, HpackHandle &handle, |
| bool is_trailing_header, uint32_t maximum_table_size, bool is_outbound) |
| { |
| int64_t result = hpack_decode_header_block(handle, hdr, buf_start, buf_len, Http2::max_header_list_size, maximum_table_size); |
| |
| if (result < 0) { |
| if (result == HPACK_ERROR_COMPRESSION_ERROR) { |
| return Http2ErrorCode::HTTP2_ERROR_COMPRESSION_ERROR; |
| } else if (result == HPACK_ERROR_SIZE_EXCEEDED_ERROR) { |
| return Http2ErrorCode::HTTP2_ERROR_ENHANCE_YOUR_CALM; |
| } |
| |
| return Http2ErrorCode::HTTP2_ERROR_PROTOCOL_ERROR; |
| } |
| if (len_read) { |
| *len_read = result; |
| } |
| return HeaderValidator::is_h2_h3_header_valid(*hdr, is_outbound, is_trailing_header) ? Http2ErrorCode::HTTP2_ERROR_NO_ERROR : |
| Http2ErrorCode::HTTP2_ERROR_PROTOCOL_ERROR; |
| } |
| |
| // Initialize this subsystem with librecords configs (for now) |
| uint32_t Http2::max_concurrent_streams_in = 100; |
| uint32_t Http2::min_concurrent_streams_in = 10; |
| uint32_t Http2::max_active_streams_in = 0; |
| bool Http2::throttling = false; |
| uint32_t Http2::stream_priority_enabled = 0; |
| uint32_t Http2::initial_window_size_in = 65535; |
| Http2FlowControlPolicy Http2::flow_control_policy_in = Http2FlowControlPolicy::STATIC_SESSION_AND_STATIC_STREAM; |
| uint32_t Http2::max_frame_size = 16384; |
| uint32_t Http2::header_table_size = 4096; |
| uint32_t Http2::max_header_list_size = 4294967295; |
| uint32_t Http2::accept_no_activity_timeout = 120; |
| uint32_t Http2::no_activity_timeout_in = 120; |
| uint32_t Http2::active_timeout_in = 0; |
| uint32_t Http2::push_diary_size = 256; |
| uint32_t Http2::zombie_timeout_in = 0; |
| |
| uint32_t Http2::max_concurrent_streams_out = 100; |
| uint32_t Http2::min_concurrent_streams_out = 10; |
| uint32_t Http2::max_active_streams_out = 0; |
| uint32_t Http2::initial_window_size_out = 65535; |
| Http2FlowControlPolicy Http2::flow_control_policy_out = Http2FlowControlPolicy::STATIC_SESSION_AND_STATIC_STREAM; |
| uint32_t Http2::no_activity_timeout_out = 120; |
| |
| float Http2::stream_error_rate_threshold = 0.1; |
| uint32_t Http2::stream_error_sampling_threshold = 10; |
| uint32_t Http2::max_settings_per_frame = 7; |
| uint32_t Http2::max_settings_per_minute = 14; |
| uint32_t Http2::max_settings_frames_per_minute = 14; |
| uint32_t Http2::max_ping_frames_per_minute = 60; |
| uint32_t Http2::max_priority_frames_per_minute = 120; |
| uint32_t Http2::max_rst_stream_frames_per_minute = 200; |
| float Http2::min_avg_window_update = 2560.0; |
| uint32_t Http2::con_slow_log_threshold = 0; |
| uint32_t Http2::stream_slow_log_threshold = 0; |
| uint32_t Http2::header_table_size_limit = 65536; |
| uint32_t Http2::write_buffer_block_size = 262144; |
| float Http2::write_size_threshold = 0.5; |
| uint32_t Http2::write_time_threshold = 100; |
| uint32_t Http2::buffer_water_mark = 0; |
| |
| void |
| Http2::init() |
| { |
| REC_EstablishStaticConfigInt32U(max_concurrent_streams_in, "proxy.config.http2.max_concurrent_streams_in"); |
| REC_EstablishStaticConfigInt32U(min_concurrent_streams_in, "proxy.config.http2.min_concurrent_streams_in"); |
| REC_EstablishStaticConfigInt32U(max_concurrent_streams_out, "proxy.config.http2.max_concurrent_streams_out"); |
| REC_EstablishStaticConfigInt32U(min_concurrent_streams_out, "proxy.config.http2.min_concurrent_streams_out"); |
| |
| REC_EstablishStaticConfigInt32U(max_active_streams_in, "proxy.config.http2.max_active_streams_in"); |
| REC_EstablishStaticConfigInt32U(stream_priority_enabled, "proxy.config.http2.stream_priority_enabled"); |
| |
| REC_EstablishStaticConfigInt32U(initial_window_size_in, "proxy.config.http2.initial_window_size_in"); |
| uint32_t flow_control_policy_in_int = 0; |
| REC_EstablishStaticConfigInt32U(flow_control_policy_in_int, "proxy.config.http2.flow_control.policy_in"); |
| if (flow_control_policy_in_int > 2) { |
| Error("Invalid value for proxy.config.http2.flow_control.policy_in: %d", flow_control_policy_in_int); |
| flow_control_policy_in_int = 0; |
| } |
| flow_control_policy_in = static_cast<Http2FlowControlPolicy>(flow_control_policy_in_int); |
| |
| REC_EstablishStaticConfigInt32U(initial_window_size_out, "proxy.config.http2.initial_window_size_out"); |
| uint32_t flow_control_policy_out_int = 0; |
| REC_EstablishStaticConfigInt32U(flow_control_policy_out_int, "proxy.config.http2.flow_control.policy_out"); |
| if (flow_control_policy_out_int > 2) { |
| Error("Invalid value for proxy.config.http2.flow_control.policy_out: %d", flow_control_policy_out_int); |
| flow_control_policy_out_int = 0; |
| } |
| flow_control_policy_out = static_cast<Http2FlowControlPolicy>(flow_control_policy_out_int); |
| |
| REC_EstablishStaticConfigInt32U(max_frame_size, "proxy.config.http2.max_frame_size"); |
| REC_EstablishStaticConfigInt32U(header_table_size, "proxy.config.http2.header_table_size"); |
| REC_EstablishStaticConfigInt32U(max_header_list_size, "proxy.config.http2.max_header_list_size"); |
| REC_EstablishStaticConfigInt32U(accept_no_activity_timeout, "proxy.config.http2.accept_no_activity_timeout"); |
| REC_EstablishStaticConfigInt32U(no_activity_timeout_in, "proxy.config.http2.no_activity_timeout_in"); |
| REC_EstablishStaticConfigInt32U(no_activity_timeout_out, "proxy.config.http2.no_activity_timeout_out"); |
| REC_EstablishStaticConfigInt32U(active_timeout_in, "proxy.config.http2.active_timeout_in"); |
| REC_EstablishStaticConfigInt32U(push_diary_size, "proxy.config.http2.push_diary_size"); |
| REC_EstablishStaticConfigInt32U(zombie_timeout_in, "proxy.config.http2.zombie_debug_timeout_in"); |
| REC_EstablishStaticConfigFloat(stream_error_rate_threshold, "proxy.config.http2.stream_error_rate_threshold"); |
| REC_EstablishStaticConfigInt32U(stream_error_sampling_threshold, "proxy.config.http2.stream_error_sampling_threshold"); |
| REC_EstablishStaticConfigInt32U(max_settings_per_frame, "proxy.config.http2.max_settings_per_frame"); |
| REC_EstablishStaticConfigInt32U(max_settings_per_minute, "proxy.config.http2.max_settings_per_minute"); |
| REC_EstablishStaticConfigInt32U(max_settings_frames_per_minute, "proxy.config.http2.max_settings_frames_per_minute"); |
| REC_EstablishStaticConfigInt32U(max_ping_frames_per_minute, "proxy.config.http2.max_ping_frames_per_minute"); |
| REC_EstablishStaticConfigInt32U(max_priority_frames_per_minute, "proxy.config.http2.max_priority_frames_per_minute"); |
| REC_EstablishStaticConfigInt32U(max_rst_stream_frames_per_minute, "proxy.config.http2.max_rst_stream_frames_per_minute"); |
| REC_EstablishStaticConfigFloat(min_avg_window_update, "proxy.config.http2.min_avg_window_update"); |
| REC_EstablishStaticConfigInt32U(con_slow_log_threshold, "proxy.config.http2.connection.slow.log.threshold"); |
| REC_EstablishStaticConfigInt32U(stream_slow_log_threshold, "proxy.config.http2.stream.slow.log.threshold"); |
| REC_EstablishStaticConfigInt32U(header_table_size_limit, "proxy.config.http2.header_table_size_limit"); |
| REC_EstablishStaticConfigInt32U(write_buffer_block_size, "proxy.config.http2.write_buffer_block_size"); |
| REC_EstablishStaticConfigFloat(write_size_threshold, "proxy.config.http2.write_size_threshold"); |
| REC_EstablishStaticConfigInt32U(write_time_threshold, "proxy.config.http2.write_time_threshold"); |
| REC_EstablishStaticConfigInt32U(buffer_water_mark, "proxy.config.http2.default_buffer_water_mark"); |
| |
| // If any settings is broken, ATS should not start |
| ink_release_assert(http2_settings_parameter_is_valid({HTTP2_SETTINGS_MAX_CONCURRENT_STREAMS, max_concurrent_streams_in})); |
| ink_release_assert(http2_settings_parameter_is_valid({HTTP2_SETTINGS_MAX_CONCURRENT_STREAMS, min_concurrent_streams_in})); |
| ink_release_assert(http2_settings_parameter_is_valid({HTTP2_SETTINGS_MAX_CONCURRENT_STREAMS, max_concurrent_streams_out})); |
| ink_release_assert(http2_settings_parameter_is_valid({HTTP2_SETTINGS_MAX_CONCURRENT_STREAMS, min_concurrent_streams_out})); |
| ink_release_assert(http2_settings_parameter_is_valid({HTTP2_SETTINGS_INITIAL_WINDOW_SIZE, initial_window_size_in})); |
| ink_release_assert(http2_settings_parameter_is_valid({HTTP2_SETTINGS_MAX_FRAME_SIZE, max_frame_size})); |
| ink_release_assert(http2_settings_parameter_is_valid({HTTP2_SETTINGS_HEADER_TABLE_SIZE, header_table_size})); |
| ink_release_assert(http2_settings_parameter_is_valid({HTTP2_SETTINGS_MAX_HEADER_LIST_SIZE, max_header_list_size})); |
| |
| // Setup statistics |
| http2_rsb.current_client_session_count = Metrics::Gauge::createPtr("proxy.process.http2.current_client_connections"); |
| http2_rsb.current_server_session_count = Metrics::Gauge::createPtr("proxy.process.http2.current_server_connections"); |
| http2_rsb.current_active_client_connection_count = |
| Metrics::Gauge::createPtr("proxy.process.http2.current_active_client_connections"); |
| http2_rsb.current_active_server_connection_count = |
| Metrics::Gauge::createPtr("proxy.process.http2.current_active_server_connections"); |
| http2_rsb.current_client_stream_count = Metrics::Gauge::createPtr("proxy.process.http2.current_client_streams"); |
| http2_rsb.current_server_stream_count = Metrics::Gauge::createPtr("proxy.process.http2.current_server_streams"); |
| http2_rsb.total_client_stream_count = Metrics::Counter::createPtr("proxy.process.http2.total_client_streams"); |
| http2_rsb.total_server_stream_count = Metrics::Counter::createPtr("proxy.process.http2.total_server_streams"); |
| http2_rsb.total_transactions_time = Metrics::Counter::createPtr("proxy.process.http2.total_transactions_time"); |
| http2_rsb.total_client_connection_count = Metrics::Counter::createPtr("proxy.process.http2.total_client_connections"); |
| http2_rsb.total_server_connection_count = Metrics::Counter::createPtr("proxy.process.http2.total_server_connections"); |
| http2_rsb.stream_errors_count = Metrics::Counter::createPtr("proxy.process.http2.stream_errors"); |
| http2_rsb.connection_errors_count = Metrics::Counter::createPtr("proxy.process.http2.connection_errors"); |
| http2_rsb.session_die_default = Metrics::Counter::createPtr("proxy.process.http2.session_die_default"); |
| http2_rsb.session_die_other = Metrics::Counter::createPtr("proxy.process.http2.session_die_other"); |
| http2_rsb.session_die_active = Metrics::Counter::createPtr("proxy.process.http2.session_die_active"); |
| http2_rsb.session_die_inactive = Metrics::Counter::createPtr("proxy.process.http2.session_die_inactive"); |
| http2_rsb.session_die_eos = Metrics::Counter::createPtr("proxy.process.http2.session_die_eos"); |
| http2_rsb.session_die_error = Metrics::Counter::createPtr("proxy.process.http2.session_die_error"); |
| http2_rsb.session_die_high_error_rate = Metrics::Counter::createPtr("proxy.process.http2.session_die_high_error_rate"); |
| http2_rsb.max_settings_per_frame_exceeded = Metrics::Counter::createPtr("proxy.process.http2.max_settings_per_frame_exceeded"); |
| http2_rsb.max_settings_per_minute_exceeded = Metrics::Counter::createPtr("proxy.process.http2.max_settings_per_minute_exceeded"); |
| http2_rsb.max_settings_frames_per_minute_exceeded = |
| Metrics::Counter::createPtr("proxy.process.http2.max_settings_frames_per_minute_exceeded"); |
| http2_rsb.max_ping_frames_per_minute_exceeded = |
| Metrics::Counter::createPtr("proxy.process.http2.max_ping_frames_per_minute_exceeded"); |
| http2_rsb.max_priority_frames_per_minute_exceeded = |
| Metrics::Counter::createPtr("proxy.process.http2.max_priority_frames_per_minute_exceeded"); |
| http2_rsb.max_rst_stream_frames_per_minute_exceeded = |
| Metrics::Counter::createPtr("proxy.process.http2.max_rst_stream_frames_per_minute_exceeded"); |
| http2_rsb.insufficient_avg_window_update = Metrics::Counter::createPtr("proxy.process.http2.insufficient_avg_window_update"); |
| http2_rsb.max_concurrent_streams_exceeded_in = |
| Metrics::Counter::createPtr("proxy.process.http2.max_concurrent_streams_exceeded_in"); |
| http2_rsb.max_concurrent_streams_exceeded_out = |
| Metrics::Counter::createPtr("proxy.process.http2.max_concurrent_streams_exceeded_out"); |
| http2_rsb.data_frames_in = Metrics::Counter::createPtr("proxy.process.http2.data_frames_in"), |
| http2_rsb.headers_frames_in = Metrics::Counter::createPtr("proxy.process.http2.headers_frames_in"), |
| http2_rsb.priority_frames_in = Metrics::Counter::createPtr("proxy.process.http2.priority_frames_in"), |
| http2_rsb.rst_stream_frames_in = Metrics::Counter::createPtr("proxy.process.http2.rst_stream_frames_in"), |
| http2_rsb.settings_frames_in = Metrics::Counter::createPtr("proxy.process.http2.settings_frames_in"), |
| http2_rsb.push_promise_frames_in = Metrics::Counter::createPtr("proxy.process.http2.push_promise_frames_in"), |
| http2_rsb.ping_frames_in = Metrics::Counter::createPtr("proxy.process.http2.ping_frames_in"), |
| http2_rsb.goaway_frames_in = Metrics::Counter::createPtr("proxy.process.http2.goaway_frames_in"), |
| http2_rsb.window_update_frames_in = Metrics::Counter::createPtr("proxy.process.http2.window_update_frames_in"), |
| http2_rsb.continuation_frames_in = Metrics::Counter::createPtr("proxy.process.http2.continuation_frames_in"), |
| http2_rsb.unknown_frames_in = Metrics::Counter::createPtr("proxy.process.http2.unknown_frames_in"), |
| |
| http2_frame_metrics_in[0] = http2_rsb.data_frames_in; |
| http2_frame_metrics_in[1] = http2_rsb.headers_frames_in; |
| http2_frame_metrics_in[2] = http2_rsb.priority_frames_in; |
| http2_frame_metrics_in[3] = http2_rsb.rst_stream_frames_in; |
| http2_frame_metrics_in[4] = http2_rsb.settings_frames_in; |
| http2_frame_metrics_in[5] = http2_rsb.push_promise_frames_in; |
| http2_frame_metrics_in[6] = http2_rsb.ping_frames_in; |
| http2_frame_metrics_in[7] = http2_rsb.goaway_frames_in; |
| http2_frame_metrics_in[8] = http2_rsb.window_update_frames_in; |
| http2_frame_metrics_in[9] = http2_rsb.continuation_frames_in; |
| http2_frame_metrics_in[10] = http2_rsb.unknown_frames_in; |
| |
| http2_init(); |
| } |
| |
| void |
| http2_init() |
| { |
| } |