|  | // Copyright 2025 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. | 
|  |  | 
|  | #ifndef QUICHE_QUIC_MASQUE_MASQUE_H2_CONNECTION_H_ | 
|  | #define QUICHE_QUIC_MASQUE_MASQUE_H2_CONNECTION_H_ | 
|  |  | 
|  | #include <cstddef> | 
|  | #include <cstdint> | 
|  | #include <memory> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #include "absl/container/flat_hash_map.h" | 
|  | #include "absl/strings/string_view.h" | 
|  | #include "quiche/http2/adapter/http2_protocol.h" | 
|  | #include "quiche/http2/adapter/http2_visitor_interface.h" | 
|  | #include "quiche/http2/adapter/oghttp2_adapter.h" | 
|  | #include "openssl/base.h" | 
|  | #include "quiche/common/http/http_header_block.h" | 
|  | #include "quiche/common/platform/api/quiche_export.h" | 
|  |  | 
|  | namespace quic { | 
|  |  | 
|  | class QUICHE_NO_EXPORT MasqueH2Connection | 
|  | : public http2::adapter::Http2VisitorInterface { | 
|  | public: | 
|  | using Http2ErrorCode = http2::adapter::Http2ErrorCode; | 
|  | using Http2PingId = http2::adapter::Http2PingId; | 
|  | using Http2Setting = http2::adapter::Http2Setting; | 
|  | using Http2StreamId = http2::adapter::Http2StreamId; | 
|  | using http2::adapter::Http2VisitorInterface::ConnectionError; | 
|  | using http2::adapter::Http2VisitorInterface::DataFrameHeaderInfo; | 
|  | using http2::adapter::Http2VisitorInterface::InvalidFrameError; | 
|  | using http2::adapter::Http2VisitorInterface::OnHeaderResult; | 
|  |  | 
|  | class QUICHE_NO_EXPORT Visitor { | 
|  | public: | 
|  | virtual ~Visitor() = default; | 
|  | virtual void OnConnectionReady(MasqueH2Connection *connection) = 0; | 
|  | virtual void OnConnectionFinished(MasqueH2Connection *connection) = 0; | 
|  | virtual void OnRequest(MasqueH2Connection *connection, int32_t stream_id, | 
|  | const quiche::HttpHeaderBlock &headers, | 
|  | const std::string &body) = 0; | 
|  | virtual void OnResponse(MasqueH2Connection *connection, int32_t stream_id, | 
|  | const quiche::HttpHeaderBlock &headers, | 
|  | const std::string &body) = 0; | 
|  | }; | 
|  |  | 
|  | // `ssl` and `visitor` must outlive this object. | 
|  | explicit MasqueH2Connection(SSL *ssl, bool is_server, Visitor *visitor); | 
|  |  | 
|  | MasqueH2Connection(const MasqueH2Connection &) = delete; | 
|  | MasqueH2Connection(MasqueH2Connection &&) = delete; | 
|  | MasqueH2Connection &operator=(const MasqueH2Connection &) = delete; | 
|  | MasqueH2Connection &operator=(MasqueH2Connection &&) = delete; | 
|  |  | 
|  | ~MasqueH2Connection(); | 
|  |  | 
|  | bool aborted() const { return aborted_; } | 
|  | // Call when there is more data to be read from SSL. | 
|  | void OnTransportReadable(); | 
|  | // Call when there is more data to be written to SSL. | 
|  | bool AttemptToSend(); | 
|  | int32_t SendRequest(const quiche::HttpHeaderBlock &headers, | 
|  | const std::string &body); | 
|  | void SendResponse(int32_t stream_id, const quiche::HttpHeaderBlock &headers, | 
|  | const std::string &body); | 
|  |  | 
|  | private: | 
|  | struct MasqueH2Stream { | 
|  | quiche::HttpHeaderBlock received_headers; | 
|  | std::string received_body; | 
|  | std::string body_to_send; | 
|  | }; | 
|  | static constexpr size_t kBioBufferSize = 16384; | 
|  | void Abort(); | 
|  | void StartH2(); | 
|  | bool TryRead(); | 
|  | MasqueH2Stream *GetOrCreateH2Stream(Http2StreamId stream_id); | 
|  | std::vector<http2::adapter::Header> ConvertHeaders( | 
|  | const quiche::HttpHeaderBlock &headers); | 
|  |  | 
|  | int WriteDataToTls(absl::string_view data); | 
|  |  | 
|  | // From http2::adapter::Http2VisitorInterface. | 
|  | int64_t OnReadyToSend(absl::string_view serialized) override; | 
|  | DataFrameHeaderInfo OnReadyToSendDataForStream(Http2StreamId stream_id, | 
|  | size_t max_length) override; | 
|  | bool SendDataFrame(Http2StreamId stream_id, absl::string_view frame_header, | 
|  | size_t payload_bytes) override; | 
|  | void OnConnectionError(ConnectionError error) override; | 
|  | void OnSettingsStart() override; | 
|  | void OnSetting(Http2Setting setting) override; | 
|  | void OnSettingsEnd() override; | 
|  | void OnSettingsAck() override; | 
|  | bool OnBeginHeadersForStream(Http2StreamId stream_id) override; | 
|  | OnHeaderResult OnHeaderForStream(Http2StreamId stream_id, | 
|  | absl::string_view key, | 
|  | absl::string_view value) override; | 
|  | bool OnEndHeadersForStream(Http2StreamId stream_id) override; | 
|  | bool OnBeginDataForStream(Http2StreamId stream_id, | 
|  | size_t payload_length) override; | 
|  | bool OnDataPaddingLength(Http2StreamId stream_id, | 
|  | size_t padding_length) override; | 
|  | bool OnDataForStream(Http2StreamId stream_id, | 
|  | absl::string_view data) override; | 
|  | bool OnEndStream(Http2StreamId stream_id) override; | 
|  | void OnRstStream(Http2StreamId stream_id, Http2ErrorCode error_code) override; | 
|  | bool OnCloseStream(Http2StreamId stream_id, | 
|  | Http2ErrorCode error_code) override; | 
|  | void OnPriorityForStream(Http2StreamId stream_id, | 
|  | Http2StreamId parent_stream_id, int weight, | 
|  | bool exclusive) override; | 
|  | void OnPing(Http2PingId ping_id, bool is_ack) override; | 
|  | void OnPushPromiseForStream(Http2StreamId stream_id, | 
|  | Http2StreamId promised_stream_id) override; | 
|  | bool OnGoAway(Http2StreamId last_accepted_stream_id, | 
|  | Http2ErrorCode error_code, | 
|  | absl::string_view opaque_data) override; | 
|  | void OnWindowUpdate(Http2StreamId stream_id, int window_increment) override; | 
|  | int OnBeforeFrameSent(uint8_t frame_type, Http2StreamId stream_id, | 
|  | size_t length, uint8_t flags) override; | 
|  | int OnFrameSent(uint8_t frame_type, Http2StreamId stream_id, size_t length, | 
|  | uint8_t flags, uint32_t error_code) override; | 
|  | bool OnInvalidFrame(Http2StreamId stream_id, | 
|  | InvalidFrameError error) override; | 
|  | void OnBeginMetadataForStream(Http2StreamId stream_id, | 
|  | size_t payload_length) override; | 
|  | bool OnMetadataForStream(Http2StreamId stream_id, | 
|  | absl::string_view metadata) override; | 
|  | bool OnMetadataEndForStream(Http2StreamId stream_id) override; | 
|  | void OnErrorDebug(absl::string_view message) override; | 
|  |  | 
|  | SSL *ssl_; | 
|  | std::unique_ptr<http2::adapter::OgHttp2Adapter> h2_adapter_; | 
|  | const bool is_server_; | 
|  | bool tls_connected_ = false; | 
|  | bool aborted_ = false; | 
|  | absl::flat_hash_map<Http2StreamId, std::unique_ptr<MasqueH2Stream>> | 
|  | h2_streams_; | 
|  | Visitor *visitor_; | 
|  | }; | 
|  |  | 
|  | // Logs an SSL error that was provided by BoringSSL. | 
|  | void PrintSSLError(const char *msg, int ssl_err, int ret); | 
|  |  | 
|  | }  // namespace quic | 
|  |  | 
|  | #endif  // QUICHE_QUIC_MASQUE_MASQUE_H2_CONNECTION_H_ |