| // Copyright 2019 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_CLIENT_SESSION_H_ | 
 | #define QUICHE_QUIC_MASQUE_MASQUE_CLIENT_SESSION_H_ | 
 |  | 
 | #include "absl/container/flat_hash_map.h" | 
 | #include "absl/strings/string_view.h" | 
 | #include "quic/core/http/quic_spdy_client_session.h" | 
 | #include "quic/masque/masque_compression_engine.h" | 
 | #include "quic/masque/masque_utils.h" | 
 | #include "quic/platform/api/quic_export.h" | 
 | #include "quic/platform/api/quic_socket_address.h" | 
 |  | 
 | namespace quic { | 
 |  | 
 | // QUIC client session for connection to MASQUE proxy. This session establishes | 
 | // a connection to a MASQUE proxy and handles sending and receiving DATAGRAM | 
 | // frames for operation of the MASQUE protocol. Multiple end-to-end encapsulated | 
 | // sessions can then coexist inside this session. Once these are created, they | 
 | // need to be registered with this session. | 
 | class QUIC_NO_EXPORT MasqueClientSession : public QuicSpdyClientSession { | 
 |  public: | 
 |   // Interface meant to be implemented by the owner of the | 
 |   // MasqueClientSession instance. | 
 |   class QUIC_NO_EXPORT Owner { | 
 |    public: | 
 |     virtual ~Owner() {} | 
 |  | 
 |     // Notifies the owner that the client connection ID is no longer in use. | 
 |     virtual void UnregisterClientConnectionId( | 
 |         QuicConnectionId client_connection_id) = 0; | 
 |   }; | 
 |   // Interface meant to be implemented by encapsulated client sessions, i.e. | 
 |   // the end-to-end QUIC client sessions that run inside MASQUE encapsulation. | 
 |   class QUIC_NO_EXPORT EncapsulatedClientSession { | 
 |    public: | 
 |     virtual ~EncapsulatedClientSession() {} | 
 |  | 
 |     // Process packet that was just decapsulated. | 
 |     virtual void ProcessPacket(absl::string_view packet, | 
 |                                QuicSocketAddress target_server_address) = 0; | 
 |  | 
 |     // Close the encapsulated connection. | 
 |     virtual void CloseConnection( | 
 |         QuicErrorCode error, | 
 |         const std::string& details, | 
 |         ConnectionCloseBehavior connection_close_behavior) = 0; | 
 |   }; | 
 |  | 
 |   // Takes ownership of |connection|, but not of |crypto_config| or | 
 |   // |push_promise_index| or |owner|. All pointers must be non-null. Caller | 
 |   // must ensure that |push_promise_index| and |owner| stay valid for the | 
 |   // lifetime of the newly created MasqueClientSession. | 
 |   MasqueClientSession(MasqueMode masque_mode, | 
 |                       const QuicConfig& config, | 
 |                       const ParsedQuicVersionVector& supported_versions, | 
 |                       QuicConnection* connection, | 
 |                       const QuicServerId& server_id, | 
 |                       QuicCryptoClientConfig* crypto_config, | 
 |                       QuicClientPushPromiseIndex* push_promise_index, | 
 |                       Owner* owner); | 
 |  | 
 |   // Disallow copy and assign. | 
 |   MasqueClientSession(const MasqueClientSession&) = delete; | 
 |   MasqueClientSession& operator=(const MasqueClientSession&) = delete; | 
 |  | 
 |   // From QuicSession. | 
 |   void OnMessageReceived(absl::string_view message) override; | 
 |   void OnMessageAcked(QuicMessageId message_id, | 
 |                       QuicTime receive_timestamp) override; | 
 |   void OnMessageLost(QuicMessageId message_id) override; | 
 |   void OnConnectionClosed(const QuicConnectionCloseFrame& frame, | 
 |                           ConnectionCloseSource source) override; | 
 |   void OnStreamClosed(QuicStreamId stream_id) override; | 
 |  | 
 |   // Send encapsulated packet. | 
 |   void SendPacket(QuicConnectionId client_connection_id, | 
 |                   QuicConnectionId server_connection_id, | 
 |                   absl::string_view packet, | 
 |                   const QuicSocketAddress& target_server_address, | 
 |                   EncapsulatedClientSession* encapsulated_client_session); | 
 |  | 
 |   // Register encapsulated client. This allows clients that are encapsulated | 
 |   // within this MASQUE session to indicate they own a given client connection | 
 |   // ID so incoming packets with that connection ID are routed back to them. | 
 |   // Callers must not register a second different |encapsulated_client_session| | 
 |   // with the same |client_connection_id|. Every call must be matched with a | 
 |   // call to UnregisterConnectionId. | 
 |   void RegisterConnectionId( | 
 |       QuicConnectionId client_connection_id, | 
 |       EncapsulatedClientSession* encapsulated_client_session); | 
 |  | 
 |   // Unregister encapsulated client. |client_connection_id| must match a | 
 |   // value previously passed to RegisterConnectionId. | 
 |   void UnregisterConnectionId( | 
 |       QuicConnectionId client_connection_id, | 
 |       EncapsulatedClientSession* encapsulated_client_session); | 
 |  | 
 |  private: | 
 |   // State that the MasqueClientSession keeps for each CONNECT-UDP request. | 
 |   class QUIC_NO_EXPORT ConnectUdpClientState | 
 |       : public QuicSpdySession::Http3DatagramVisitor { | 
 |    public: | 
 |     // |stream| and |encapsulated_client_session| must be valid for the lifetime | 
 |     // of the ConnectUdpClientState. | 
 |     explicit ConnectUdpClientState( | 
 |         QuicSpdyClientStream* stream, | 
 |         EncapsulatedClientSession* encapsulated_client_session, | 
 |         MasqueClientSession* masque_session, | 
 |         QuicDatagramFlowId flow_id, | 
 |         const QuicSocketAddress& target_server_address); | 
 |  | 
 |     ~ConnectUdpClientState(); | 
 |  | 
 |     // Disallow copy but allow move. | 
 |     ConnectUdpClientState(const ConnectUdpClientState&) = delete; | 
 |     ConnectUdpClientState(ConnectUdpClientState&&); | 
 |     ConnectUdpClientState& operator=(const ConnectUdpClientState&) = delete; | 
 |     ConnectUdpClientState& operator=(ConnectUdpClientState&&); | 
 |  | 
 |     QuicSpdyClientStream* stream() const { return stream_; } | 
 |     EncapsulatedClientSession* encapsulated_client_session() const { | 
 |       return encapsulated_client_session_; | 
 |     } | 
 |     QuicDatagramFlowId flow_id() const { | 
 |       QUICHE_DCHECK(flow_id_.has_value()); | 
 |       return *flow_id_; | 
 |     } | 
 |     const QuicSocketAddress& target_server_address() const { | 
 |       return target_server_address_; | 
 |     } | 
 |  | 
 |     // From QuicSpdySession::Http3DatagramVisitor. | 
 |     void OnHttp3Datagram(QuicDatagramFlowId flow_id, | 
 |                          absl::string_view payload) override; | 
 |  | 
 |    private: | 
 |     QuicSpdyClientStream* stream_;                            // Unowned. | 
 |     EncapsulatedClientSession* encapsulated_client_session_;  // Unowned. | 
 |     MasqueClientSession* masque_session_;                     // Unowned. | 
 |     absl::optional<QuicDatagramFlowId> flow_id_; | 
 |     QuicSocketAddress target_server_address_; | 
 |   }; | 
 |  | 
 |   const ConnectUdpClientState* GetOrCreateConnectUdpClientState( | 
 |       const QuicSocketAddress& target_server_address, | 
 |       EncapsulatedClientSession* encapsulated_client_session); | 
 |  | 
 |   MasqueMode masque_mode_; | 
 |   std::list<ConnectUdpClientState> connect_udp_client_states_; | 
 |   absl::flat_hash_map<QuicConnectionId, | 
 |                       EncapsulatedClientSession*, | 
 |                       QuicConnectionIdHash> | 
 |       client_connection_id_registrations_; | 
 |   Owner* owner_;  // Unowned; | 
 |   MasqueCompressionEngine compression_engine_; | 
 | }; | 
 |  | 
 | }  // namespace quic | 
 |  | 
 | #endif  // QUICHE_QUIC_MASQUE_MASQUE_CLIENT_SESSION_H_ |