#ifndef NET_QUIC_QUIC_CHROMIUM_CLIENT_SESSION_H_
#define NET_QUIC_QUIC_CHROMIUM_CLIENT_SESSION_H_
#include <stddef.h>
#include <list>
#include <memory>
#include <set>
#include <string>
#include <string_view>
#include <vector>
#include "base/containers/flat_map.h"
#include "base/feature_list.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/raw_ptr_exclusion.h"
#include "base/observer_list.h"
#include "base/observer_list_types.h"
#include "base/task/sequenced_task_runner.h"
#include "base/time/time.h"
#include "base/timer/timer.h"
#include "base/values.h"
#include "net/base/completion_once_callback.h"
#include "net/base/connection_migration_information.h"
#include "net/base/load_timing_info.h"
#include "net/base/net_error_details.h"
#include "net/base/net_export.h"
#include "net/base/network_handle.h"
#include "net/log/net_log_with_source.h"
#include "net/net_buildflags.h"
#include "net/quic/quic_chromium_client_stream.h"
#include "net/quic/quic_chromium_packet_reader.h"
#include "net/quic/quic_chromium_packet_writer.h"
#include "net/quic/quic_connection_logger.h"
#include "net/quic/quic_crypto_client_config_handle.h"
#include "net/quic/quic_http3_logger.h"
#include "net/quic/quic_session_alias_key.h"
#include "net/quic/quic_session_key.h"
#include "net/socket/socket_performance_watcher.h"
#include "net/spdy/http2_priority_dependencies.h"
#include "net/spdy/multiplexed_session.h"
#include "net/spdy/multiplexed_session_creation_initiator.h"
#include "net/third_party/quiche/src/quiche/common/http/http_header_block.h"
#include "net/third_party/quiche/src/quiche/quic/core/http/quic_spdy_client_session_base.h"
#include "net/third_party/quiche/src/quiche/quic/core/quic_crypto_client_stream.h"
#include "net/third_party/quiche/src/quiche/quic/core/quic_packet_writer.h"
#include "net/third_party/quiche/src/quiche/quic/core/quic_packets.h"
#include "net/third_party/quiche/src/quiche/quic/core/quic_path_validator.h"
#include "net/third_party/quiche/src/quiche/quic/core/quic_server_id.h"
#include "net/third_party/quiche/src/quiche/quic/core/quic_time.h"
#include "net/third_party/quiche/src/quiche/quic/core/quic_types.h"
#include "net/traffic_annotation/network_traffic_annotation.h"
#include "url/origin.h"
#include "url/scheme_host_port.h"
#if BUILDFLAG(ENABLE_WEBSOCKETS)
#include "net/websockets/websocket_basic_stream_adapters.h"
#endif
namespace net {
class CertVerifyResult;
class DatagramClientSocket;
struct ConnectionEndpointMetadata;
class QuicCryptoClientStreamFactory;
class QuicServerInfo;
class QuicSessionPool;
class SSLConfigService;
class SSLInfo;
class TransportSecurityState;
namespace test {
class QuicChromiumClientSessionPeer;
}
const size_t kQuicMaxHeaderListSize = 256 * 1024;
enum class MigrationResult {
SUCCESS,
NO_NEW_NETWORK,
FAILURE,
};
enum class ConnectionMigrationMode {
NO_MIGRATION,
NO_MIGRATION_ON_PATH_DEGRADING_V1,
FULL_MIGRATION_V1,
NO_MIGRATION_ON_PATH_DEGRADING_V2,
FULL_MIGRATION_V2
};
enum MigrationCause {
UNKNOWN_CAUSE,
ON_NETWORK_CONNECTED,
ON_NETWORK_DISCONNECTED,
ON_WRITE_ERROR,
ON_NETWORK_MADE_DEFAULT,
ON_MIGRATE_BACK_TO_DEFAULT_NETWORK,
CHANGE_NETWORK_ON_PATH_DEGRADING,
CHANGE_PORT_ON_PATH_DEGRADING,
NEW_NETWORK_CONNECTED_POST_PATH_DEGRADING,
ON_SERVER_PREFERRED_ADDRESS_AVAILABLE,
MIGRATION_CAUSE_MAX
};
enum QuicConnectionMigrationStatus {
MIGRATION_STATUS_NO_MIGRATABLE_STREAMS,
MIGRATION_STATUS_ALREADY_MIGRATED,
MIGRATION_STATUS_INTERNAL_ERROR,
MIGRATION_STATUS_TOO_MANY_CHANGES,
MIGRATION_STATUS_SUCCESS,
MIGRATION_STATUS_NON_MIGRATABLE_STREAM,
MIGRATION_STATUS_NOT_ENABLED,
MIGRATION_STATUS_NO_ALTERNATE_NETWORK,
MIGRATION_STATUS_ON_PATH_DEGRADING_DISABLED,
MIGRATION_STATUS_DISABLED_BY_CONFIG,
MIGRATION_STATUS_PATH_DEGRADING_NOT_ENABLED,
MIGRATION_STATUS_TIMEOUT,
MIGRATION_STATUS_ON_WRITE_ERROR_DISABLED,
MIGRATION_STATUS_PATH_DEGRADING_BEFORE_HANDSHAKE_CONFIRMED,
MIGRATION_STATUS_IDLE_MIGRATION_TIMEOUT,
MIGRATION_STATUS_NO_UNUSED_CONNECTION_ID,
MIGRATION_STATUS_MAX
};
enum class ProbingResult {
PENDING,
DISABLED_WITH_IDLE_SESSION,
DISABLED_BY_CONFIG,
DISABLED_BY_NON_MIGRABLE_STREAM,
INTERNAL_ERROR,
FAILURE,
};
enum class EcnPermutations {
kUnknown = 0,
kNotEct = 1,
kEct1 = 2,
kNotEctEct1 = 3,
kEct0 = 4,
kNotEctEct0 = 5,
kEct1Ect0 = 6,
kNotEctEct1Ect0 = 7,
kCe = 8,
kNotEctCe = 9,
kEct1Ce = 10,
kNotEctEct1Ce = 11,
kEct0Ce = 12,
kNotEctEct0Ce = 13,
kEct1Ect0Ce = 14,
kNotEctEct1Ect0Ce = 15,
kMaxValue = kNotEctEct1Ect0Ce,
};
class NET_EXPORT_PRIVATE QuicChromiumClientSession
: public quic::QuicSpdyClientSessionBase,
public MultiplexedSession,
public QuicChromiumPacketReader::Visitor,
public QuicChromiumPacketWriter::Delegate {
public:
static void SetMidMigrationCallbackForTesting(base::OnceClosure callback);
class StreamRequest;
class NET_EXPORT_PRIVATE ConnectivityObserver : public base::CheckedObserver {
public:
virtual void OnSessionPathDegrading(QuicChromiumClientSession* session,
handles::NetworkHandle network) = 0;
virtual void OnSessionResumedPostPathDegrading(
QuicChromiumClientSession* session,
handles::NetworkHandle network) = 0;
virtual void OnSessionEncounteringWriteError(
QuicChromiumClientSession* session,
handles::NetworkHandle network,
int error_code) = 0;
virtual void OnSessionClosedAfterHandshake(
QuicChromiumClientSession* session,
handles::NetworkHandle network,
quic::ConnectionCloseSource source,
quic::QuicErrorCode error_code) = 0;
virtual void OnSessionRegistered(QuicChromiumClientSession* session,
handles::NetworkHandle network) = 0;
virtual void OnSessionRemoved(QuicChromiumClientSession* session) = 0;
};
class NET_EXPORT_PRIVATE Handle : public MultiplexedSessionHandle {
public:
Handle(const base::WeakPtr<QuicChromiumClientSession>& session,
url::SchemeHostPort destination);
Handle(const Handle& other) = delete;
~Handle() override;
bool IsConnected() const;
bool OneRttKeysAvailable() const;
int RequestStream(bool requires_confirmation,
CompletionOnceCallback callback,
const NetworkTrafficAnnotationTag& traffic_annotation);
std::unique_ptr<QuicChromiumClientStream::Handle> ReleaseStream();
std::unique_ptr<quic::QuicConnection::ScopedPacketFlusher>
CreatePacketBundler();
void PopulateNetErrorDetails(NetErrorDetails* details) const;
const LoadTimingInfo::ConnectTiming& GetConnectTiming();
bool SharesSameSession(const Handle& other) const;
quic::ParsedQuicVersion GetQuicVersion() const;
int GetPeerAddress(IPEndPoint* address) const;
int GetSelfAddress(IPEndPoint* address) const;
quic::QuicServerId server_id() const { return server_id_; }
const url::SchemeHostPort& destination() const { return destination_; }
const NetLogWithSource& net_log() const { return net_log_; }
ConnectionMigrationMode connection_migration_mode() const {
return session_->connection_migration_mode();
}
bool WasEverUsed() const;
const std::set<std::string>& GetDnsAliasesForSessionKey(
const QuicSessionKey& key) const;
quic::QuicPacketLength GetGuaranteedLargestMessagePayload() const;
const ConnectionMigrationInformation GetConnectionMigrationInfoSinceInit()
const;
#if BUILDFLAG(ENABLE_WEBSOCKETS)
std::unique_ptr<WebSocketQuicStreamAdapter>
CreateWebSocketQuicStreamAdapter(
WebSocketQuicStreamAdapter::Delegate* delegate,
base::OnceCallback<void(std::unique_ptr<WebSocketQuicStreamAdapter>)>
callback,
const NetworkTrafficAnnotationTag& traffic_annotation);
#endif
private:
friend class QuicChromiumClientSession;
friend class QuicChromiumClientSession::StreamRequest;
#if BUILDFLAG(ENABLE_WEBSOCKETS)
friend class WebSocketHttp3HandshakeStream;
#endif
int WaitForHandshakeConfirmation(CompletionOnceCallback callback);
void OnCryptoHandshakeConfirmed();
void OnSessionClosed(quic::ParsedQuicVersion quic_version,
int net_error,
quic::QuicErrorCode quic_error,
quic::ConnectionCloseSource source,
bool port_migration_detected,
bool quic_connection_migration_attempted,
bool quic_connection_migration_successful,
LoadTimingInfo::ConnectTiming connect_timing,
bool was_ever_used,
const ConnectionMigrationInformation& migration_info);
int TryCreateStream(StreamRequest* request);
void CancelRequest(StreamRequest* request);
base::WeakPtr<QuicChromiumClientSession> session_;
url::SchemeHostPort destination_;
std::unique_ptr<StreamRequest> stream_request_;
NetLogWithSource net_log_;
bool was_handshake_confirmed_;
int net_error_ = OK;
quic::QuicErrorCode quic_error_ = quic::QUIC_NO_ERROR;
quic::ConnectionCloseSource source_ =
quic::ConnectionCloseSource::FROM_SELF;
bool port_migration_detected_ = false;
bool quic_connection_migration_attempted_ = false;
bool quic_connection_migration_successful_ = false;
quic::QuicServerId server_id_;
quic::ParsedQuicVersion quic_version_;
LoadTimingInfo::ConnectTiming connect_timing_;
const ConnectionMigrationInformation initial_migration_information_;
ConnectionMigrationInformation last_migration_information_;
bool was_ever_used_ = false;
};
class NET_EXPORT_PRIVATE StreamRequest {
public:
StreamRequest(const StreamRequest&) = delete;
StreamRequest& operator=(const StreamRequest&) = delete;
~StreamRequest();
int StartRequest(CompletionOnceCallback callback);
std::unique_ptr<QuicChromiumClientStream::Handle> ReleaseStream();
const NetworkTrafficAnnotationTag traffic_annotation() {
return traffic_annotation_;
}
private:
friend class QuicChromiumClientSession;
enum State {
STATE_NONE,
STATE_WAIT_FOR_CONFIRMATION,
STATE_WAIT_FOR_CONFIRMATION_COMPLETE,
STATE_REQUEST_STREAM,
STATE_REQUEST_STREAM_COMPLETE,
};
StreamRequest(QuicChromiumClientSession::Handle* session,
bool requires_confirmation,
const NetworkTrafficAnnotationTag& traffic_annotation);
void OnIOComplete(int rv);
void DoCallback(int rv);
int DoLoop(int rv);
int DoWaitForConfirmation();
int DoWaitForConfirmationComplete(int rv);
int DoRequestStream();
int DoRequestStreamComplete(int rv);
void OnRequestCompleteSuccess(
std::unique_ptr<QuicChromiumClientStream::Handle> stream);
void OnRequestCompleteFailure(int rv);
const raw_ptr<QuicChromiumClientSession::Handle> session_;
const bool requires_confirmation_;
CompletionOnceCallback callback_;
std::unique_ptr<QuicChromiumClientStream::Handle> stream_;
base::TimeTicks pending_start_time_;
State next_state_;
const NetworkTrafficAnnotationTag traffic_annotation_;
#if BUILDFLAG(ENABLE_WEBSOCKETS)
bool for_websockets_ = false;
raw_ptr<WebSocketQuicStreamAdapter::Delegate> websocket_adapter_delegate_;
base::OnceCallback<void(std::unique_ptr<WebSocketQuicStreamAdapter>)>
start_websocket_callback_;
#endif
base::WeakPtrFactory<StreamRequest> weak_factory_{this};
};
class NET_EXPORT_PRIVATE QuicChromiumPathValidationContext
: public quic::QuicPathValidationContext {
public:
QuicChromiumPathValidationContext(
const quic::QuicSocketAddress& self_address,
const quic::QuicSocketAddress& peer_address,
handles::NetworkHandle network,
std::unique_ptr<QuicChromiumPacketWriter> writer,
std::unique_ptr<QuicChromiumPacketReader> reader);
~QuicChromiumPathValidationContext() override;
handles::NetworkHandle network();
quic::QuicPacketWriter* WriterToUse() override;
std::unique_ptr<QuicChromiumPacketWriter> ReleaseWriter();
std::unique_ptr<QuicChromiumPacketReader> ReleaseReader();
private:
handles::NetworkHandle network_handle_;
std::unique_ptr<QuicChromiumPacketReader> reader_;
std::unique_ptr<QuicChromiumPacketWriter> writer_;
};
class NET_EXPORT_PRIVATE ConnectionMigrationValidationResultDelegate
: public quic::QuicPathValidator::ResultDelegate {
public:
explicit ConnectionMigrationValidationResultDelegate(
QuicChromiumClientSession* session);
void OnPathValidationSuccess(
std::unique_ptr<quic::QuicPathValidationContext> context,
quic::QuicTime start_time) override;
void OnPathValidationFailure(
std::unique_ptr<quic::QuicPathValidationContext> context) override;
private:
raw_ptr<QuicChromiumClientSession> session_;
};
class NET_EXPORT_PRIVATE PortMigrationValidationResultDelegate
: public quic::QuicPathValidator::ResultDelegate {
public:
explicit PortMigrationValidationResultDelegate(
QuicChromiumClientSession* session);
void OnPathValidationSuccess(
std::unique_ptr<quic::QuicPathValidationContext> context,
quic::QuicTime start_time) override;
void OnPathValidationFailure(
std::unique_ptr<quic::QuicPathValidationContext> context) override;
private:
raw_ptr<QuicChromiumClientSession> session_;
};
class NET_EXPORT_PRIVATE ServerPreferredAddressValidationResultDelegate
: public quic::QuicPathValidator::ResultDelegate {
public:
explicit ServerPreferredAddressValidationResultDelegate(
QuicChromiumClientSession* session);
void OnPathValidationSuccess(
std::unique_ptr<quic::QuicPathValidationContext> context,
quic::QuicTime start_time) override;
void OnPathValidationFailure(
std::unique_ptr<quic::QuicPathValidationContext> context) override;
private:
raw_ptr<QuicChromiumClientSession> session_;
};
class NET_EXPORT_PRIVATE QuicChromiumPathValidationWriterDelegate
: public QuicChromiumPacketWriter::Delegate {
public:
QuicChromiumPathValidationWriterDelegate(
QuicChromiumClientSession* session,
base::SequencedTaskRunner* task_runner);
QuicChromiumPathValidationWriterDelegate(
const QuicChromiumPathValidationWriterDelegate&) = delete;
QuicChromiumPathValidationWriterDelegate& operator=(
const QuicChromiumPathValidationWriterDelegate&) = delete;
~QuicChromiumPathValidationWriterDelegate();
int HandleWriteError(
int error_code,
scoped_refptr<QuicChromiumPacketWriter::ReusableIOBuffer> last_packet)
override;
void OnWriteError(int error_code) override;
void OnWriteUnblocked() override;
void set_peer_address(const quic::QuicSocketAddress& peer_address);
void set_network(handles::NetworkHandle network);
private:
void NotifySessionProbeFailed(handles::NetworkHandle network);
raw_ptr<QuicChromiumClientSession> session_;
raw_ptr<base::SequencedTaskRunner> task_runner_;
handles::NetworkHandle network_;
quic::QuicSocketAddress peer_address_;
base::WeakPtrFactory<QuicChromiumPathValidationWriterDelegate>
weak_factory_{this};
};
QuicChromiumClientSession(
quic::QuicConnection* connection,
std::unique_ptr<DatagramClientSocket> socket,
QuicSessionPool* session_pool,
QuicCryptoClientStreamFactory* crypto_client_stream_factory,
const quic::QuicClock* clock,
TransportSecurityState* transport_security_state,
SSLConfigService* ssl_config_service,
std::unique_ptr<QuicServerInfo> server_info,
QuicSessionAliasKey session_alias_key,
bool require_confirmation,
bool migrate_sesion_early_v2,
bool migrate_session_on_network_change_v2,
handles::NetworkHandle default_network,
quic::QuicTime::Delta retransmittable_on_wire_timeout,
bool migrate_idle_session,
bool allow_port_migration,
base::TimeDelta idle_migration_period,
int multi_port_probing_interval,
base::TimeDelta max_time_on_non_default_network,
int max_migrations_to_non_default_network_on_write_error,
int max_migrations_to_non_default_network_on_path_degrading,
int yield_after_packets,
quic::QuicTime::Delta yield_after_duration,
int cert_verify_flags,
const quic::QuicConfig& config,
std::unique_ptr<QuicCryptoClientConfigHandle> crypto_config,
const char* const connection_description,
base::TimeTicks dns_resolution_start_time,
base::TimeTicks dns_resolution_end_time,
const base::TickClock* tick_clock,
base::SequencedTaskRunner* task_runner,
std::unique_ptr<SocketPerformanceWatcher> socket_performance_watcher,
const ConnectionEndpointMetadata& metadata,
bool enable_origin_frame,
bool allow_server_preferred_address,
MultiplexedSessionCreationInitiator session_creation_initiator,
const NetLogWithSource& net_log);
QuicChromiumClientSession(const QuicChromiumClientSession&) = delete;
QuicChromiumClientSession& operator=(const QuicChromiumClientSession&) =
delete;
~QuicChromiumClientSession() override;
void Initialize() override;
void AddHandle(Handle* handle);
void RemoveHandle(Handle* handle);
void AddConnectivityObserver(ConnectivityObserver* observer);
void RemoveConnectivityObserver(ConnectivityObserver* observer);
ConnectionMigrationMode connection_migration_mode() const;
int WaitForHandshakeConfirmation(CompletionOnceCallback callback);
int TryCreateStream(StreamRequest* request);
void CancelRequest(StreamRequest* request);
int HandleWriteError(int error_code,
scoped_refptr<QuicChromiumPacketWriter::ReusableIOBuffer>
last_packet) override;
void OnWriteError(int error_code) override;
void OnWriteUnblocked() override;
void OnConnectionMigrationProbeSucceeded(
handles::NetworkHandle network,
const quic::QuicSocketAddress& peer_address,
const quic::QuicSocketAddress& self_address,
std::unique_ptr<QuicChromiumPacketWriter> writer,
std::unique_ptr<QuicChromiumPacketReader> reader);
void OnPortMigrationProbeSucceeded(
handles::NetworkHandle network,
const quic::QuicSocketAddress& peer_address,
const quic::QuicSocketAddress& self_address,
std::unique_ptr<QuicChromiumPacketWriter> writer,
std::unique_ptr<QuicChromiumPacketReader> reader);
void OnServerPreferredAddressProbeSucceeded(
handles::NetworkHandle network,
const quic::QuicSocketAddress& peer_address,
const quic::QuicSocketAddress& self_address,
std::unique_ptr<QuicChromiumPacketWriter> writer,
std::unique_ptr<QuicChromiumPacketReader> reader);
void OnProbeFailed(handles::NetworkHandle network,
const quic::QuicSocketAddress& peer_address);
size_t WriteHeadersOnHeadersStream(
quic::QuicStreamId id,
quiche::HttpHeaderBlock headers,
bool fin,
const spdy::SpdyStreamPrecedence& precedence,
quiche::QuicheReferenceCountedPointer<quic::QuicAckListenerInterface>
ack_listener) override;
void OnHttp3GoAway(uint64_t id) override;
void OnAcceptChFrameReceivedViaAlps(
const quic::AcceptChFrame& frame) override;
void OnOriginFrame(const quic::OriginFrame& frame) override;
QuicChromiumClientStream* CreateOutgoingBidirectionalStream() override;
const quic::QuicCryptoClientStream* GetCryptoStream() const override;
quic::QuicCryptoClientStream* GetMutableCryptoStream() override;
void SetDefaultEncryptionLevel(quic::EncryptionLevel level) override;
void OnTlsHandshakeComplete() override;
void OnTlsHandshakeConfirmed() override;
void OnNewEncryptionKeyAvailable(
quic::EncryptionLevel level,
std::unique_ptr<quic::QuicEncrypter> encrypter) override;
void OnCryptoHandshakeMessageSent(
const quic::CryptoHandshakeMessage& message) override;
void OnCryptoHandshakeMessageReceived(
const quic::CryptoHandshakeMessage& message) override;
void OnGoAway(const quic::QuicGoAwayFrame& frame) override;
void OnCanCreateNewOutgoingStream(bool unidirectional) override;
quic::QuicSSLConfig GetSSLConfig() const override;
void OnProofValid(
const quic::QuicCryptoClientConfig::CachedState& cached) override;
void OnProofVerifyDetailsAvailable(
const quic::ProofVerifyDetails& verify_details) override;
void OnConnectionClosed(const quic::QuicConnectionCloseFrame& frame,
quic::ConnectionCloseSource source) override;
void OnSuccessfulVersionNegotiation(
const quic::ParsedQuicVersion& version) override;
void OnPathDegrading() override;
void OnForwardProgressMadeAfterPathDegrading() override;
void SendRetireConnectionId(uint64_t sequence_number) override;
void OnKeyUpdate(quic::KeyUpdateReason reason) override;
void CreateContextForMultiPortPath(
std::unique_ptr<quic::MultiPortPathContextObserver> context_observer)
override;
void MigrateToMultiPortPath(
std::unique_ptr<quic::QuicPathValidationContext> context) override;
#if BUILDFLAG(ARKWEB_NETWORK_LOAD)
std::string GetServerHostForQuicBroken() const override {
return server_id().host();
}
#endif
bool OnReadError(int result, const DatagramClientSocket* socket) override;
bool OnPacket(const quic::QuicReceivedPacket& packet,
const quic::QuicSocketAddress& local_address,
const quic::QuicSocketAddress& peer_address) override;
void OnStreamClosed(quic::QuicStreamId stream_id) override;
bool ShouldKeepConnectionAlive() const override;
int GetRemoteEndpoint(IPEndPoint* endpoint) override;
bool GetSSLInfo(SSLInfo* ssl_info) const override;
std::string_view GetAcceptChViaAlps(
const url::SchemeHostPort& scheme_host_port) const override;
void FinishCreateContextForMultiPortPath(
std::unique_ptr<quic::MultiPortPathContextObserver> context_observer,
std::unique_ptr<DatagramClientSocket> probing_socket,
int rv);
int CryptoConnect(CompletionOnceCallback callback);
void StartReading();
void CloseSessionOnError(int net_error,
quic::QuicErrorCode quic_error,
quic::ConnectionCloseBehavior behavior);
void CloseSessionOnErrorLater(int net_error,
quic::QuicErrorCode quic_error,
quic::ConnectionCloseBehavior behavior);
base::Value::Dict GetInfoAsValue(const std::set<HostPortPair>& aliases);
const NetLogWithSource& net_log() const { return net_log_; }
bool gquic_zero_rtt_disabled() const;
virtual std::unique_ptr<QuicChromiumClientSession::Handle> CreateHandle(
url::SchemeHostPort destination);
int GetNumSentClientHellos() const;
bool CanPool(std::string_view hostname,
const QuicSessionKey& other_session_key) const;
const quic::QuicServerId& server_id() const {
return session_key_.server_id();
}
const ConnectionMigrationInformation& migration_info() const {
return migration_info_;
}
const QuicSessionKey& quic_session_key() const { return session_key_; }
const QuicSessionAliasKey& session_alias_key() const {
return session_alias_key_;
}
void MigrateSessionOnWriteError(int error_code,
quic::QuicPacketWriter* writer);
void FinishMigrateSessionOnWriteError(handles::NetworkHandle new_network,
MigrationResult result);
void WriteToNewSocket();
using MigrationCallback = base::OnceCallback<void(MigrationResult)>;
void Migrate(handles::NetworkHandle network,
IPEndPoint peer_address,
bool close_session_on_error,
MigrationCallback migration_callback);
void FinishMigrate(std::unique_ptr<DatagramClientSocket> socket,
IPEndPoint peer_address,
bool close_session_on_error,
MigrationCallback callback,
int rv);
void DoMigrationCallback(MigrationCallback callback, MigrationResult rv);
bool MigrateToSocket(const quic::QuicSocketAddress& self_address,
const quic::QuicSocketAddress& peer_address,
std::unique_ptr<QuicChromiumPacketReader> reader,
std::unique_ptr<QuicChromiumPacketWriter> writer);
void OnNetworkConnected(handles::NetworkHandle network);
void OnNetworkDisconnectedV2(handles::NetworkHandle disconnected_network);
void OnNetworkMadeDefault(handles::NetworkHandle new_network);
void OnNoNewNetwork();
void OnMigrationTimeout(size_t num_sockets);
void PopulateNetErrorDetails(NetErrorDetails* details) const;
const DatagramClientSocket* GetDefaultSocket() const;
handles::NetworkHandle GetCurrentNetwork() const;
void OnServerPreferredAddressAvailable(
const quic::QuicSocketAddress& server_preferred_address) override;
const LoadTimingInfo::ConnectTiming& GetConnectTiming();
quic::ParsedQuicVersion GetQuicVersion() const;
void SendPing();
bool require_confirmation() const { return require_confirmation_; }
const std::set<std::string>& GetDnsAliasesForSessionKey(
const QuicSessionKey& key) const;
const std::set<url::SchemeHostPort>& received_origins() const {
return received_origins_;
}
void SetPeriodicConnectionKeepAlive(bool enable_periodic_ping) {
enable_periodic_ping_ = enable_periodic_ping;
}
void SetGoingAwayForTesting(bool going_away) { going_away_ = going_away; }
void SetConnectionMigrationInformationForTesting(
net::ConnectionMigrationInformation migration_info) {
migration_info_ = migration_info;
}
protected:
bool ShouldCreateIncomingStream(quic::QuicStreamId id) override;
bool ShouldCreateOutgoingBidirectionalStream() override;
QuicChromiumClientStream* CreateIncomingStream(
quic::QuicStreamId id) override;
QuicChromiumClientStream* CreateIncomingStream(
quic::PendingStream* pending) override;
private:
friend class test::QuicChromiumClientSessionPeer;
typedef std::set<raw_ptr<Handle>> HandleSet;
typedef std::list<raw_ptr<StreamRequest>> StreamRequestQueue;
bool WasConnectionEverUsed();
QuicChromiumClientStream* CreateOutgoingReliableStreamImpl(
const NetworkTrafficAnnotationTag& traffic_annotation);
QuicChromiumClientStream* CreateIncomingReliableStreamImpl(
quic::QuicStreamId id,
const NetworkTrafficAnnotationTag& traffic_annotation);
QuicChromiumClientStream* CreateIncomingReliableStreamImpl(
quic::PendingStream* pending,
const NetworkTrafficAnnotationTag& traffic_annotation);
void OnReadComplete(int result);
void NotifyAllStreamsOfError(int net_error);
void CloseAllHandles(int net_error);
void CancelAllRequests(int net_error);
void NotifyRequestsOfConfirmation(int net_error);
using ProbingCallback = base::OnceCallback<void(ProbingResult)>;
void StartProbing(ProbingCallback probing_callback,
handles::NetworkHandle network,
const quic::QuicSocketAddress& peer_address);
void FinishStartProbing(ProbingCallback probing_callback,
std::unique_ptr<DatagramClientSocket> probing_socket,
handles::NetworkHandle network,
const quic::QuicSocketAddress& peer_address,
int rv);
void MaybeStartProbing(ProbingCallback probing_callback,
handles::NetworkHandle network,
const quic::QuicSocketAddress& peer_address);
void MaybeMigrateToAlternateNetworkOnPathDegrading();
void MaybeMigrateToDifferentPortOnPathDegrading();
void MigrateNetworkImmediately(handles::NetworkHandle network);
void FinishMigrateNetworkImmediately(handles::NetworkHandle network,
MigrationResult result);
void StartMigrateBackToDefaultNetworkTimer(base::TimeDelta delay);
void CancelMigrateBackToDefaultNetworkTimer();
void TryMigrateBackToDefaultNetwork(base::TimeDelta timeout);
void FinishTryMigrateBackToDefaultNetwork(base::TimeDelta timeout,
ProbingResult result);
void MaybeRetryMigrateBackToDefaultNetwork();
bool CheckIdleTimeExceedsIdleMigrationPeriod();
void ResetNonMigratableStreams();
void LogMetricsOnNetworkDisconnected();
void LogMetricsOnNetworkMadeDefault();
void LogMigrationResultToHistogram(QuicConnectionMigrationStatus status);
void LogHandshakeStatusOnMigrationSignal() const;
void HistogramAndLogMigrationFailure(QuicConnectionMigrationStatus status,
quic::QuicConnectionId connection_id,
const char* reason);
void HistogramAndLogMigrationSuccess(quic::QuicConnectionId connection_id);
void NotifyFactoryOfSessionGoingAway();
void NotifyFactoryOfSessionClosedLater();
void NotifyFactoryOfSessionClosed();
void OnCryptoHandshakeComplete();
void LogZeroRttStats();
#if BUILDFLAG(ENABLE_WEBSOCKETS)
std::unique_ptr<WebSocketQuicStreamAdapter>
CreateWebSocketQuicStreamAdapterImpl(
WebSocketQuicStreamAdapter::Delegate* delegate);
std::unique_ptr<WebSocketQuicStreamAdapter> CreateWebSocketQuicStreamAdapter(
WebSocketQuicStreamAdapter::Delegate* delegate,
base::OnceCallback<void(std::unique_ptr<WebSocketQuicStreamAdapter>)>
callback,
StreamRequest* stream_request);
#endif
virtual void RegisterQuicConnectionClosePayload();
virtual void UnregisterQuicConnectionClosePayload();
const QuicSessionAliasKey session_alias_key_;
QuicSessionKey session_key_;
bool require_confirmation_;
bool migrate_session_early_v2_;
bool migrate_session_on_network_change_v2_;
bool pending_migrate_session_on_write_error_ = false;
bool pending_migrate_network_immediately_ = false;
bool migrate_idle_session_;
bool allow_port_migration_;
base::TimeDelta idle_migration_period_;
base::TimeDelta max_time_on_non_default_network_;
int max_migrations_to_non_default_network_on_write_error_;
int current_migrations_to_non_default_network_on_write_error_ = 0;
int max_migrations_to_non_default_network_on_path_degrading_;
int current_migrations_to_non_default_network_on_path_degrading_ = 0;
raw_ptr<const quic::QuicClock> clock_;
int yield_after_packets_;
quic::QuicTime::Delta yield_after_duration_;
base::TimeTicks most_recent_path_degrading_timestamp_;
base::TimeTicks most_recent_network_disconnected_timestamp_;
raw_ptr<const base::TickClock> tick_clock_;
base::TimeTicks most_recent_stream_close_time_;
int most_recent_write_error_ = 0;
base::TimeTicks most_recent_write_error_timestamp_;
std::unique_ptr<QuicCryptoClientConfigHandle> crypto_config_;
std::unique_ptr<quic::QuicCryptoClientStream> crypto_stream_;
raw_ptr<QuicSessionPool> session_pool_;
base::ObserverList<ConnectivityObserver> connectivity_observer_list_;
std::vector<std::unique_ptr<QuicChromiumPacketReader>> packet_readers_;
raw_ptr<TransportSecurityState> transport_security_state_;
raw_ptr<SSLConfigService> ssl_config_service_;
std::unique_ptr<QuicServerInfo> server_info_;
std::unique_ptr<CertVerifyResult> cert_verify_result_;
bool pkp_bypassed_ = false;
bool is_fatal_cert_error_ = false;
const bool enable_origin_frame_;
HandleSet handles_;
StreamRequestQueue stream_requests_;
std::vector<CompletionOnceCallback> waiting_for_confirmation_callbacks_;
CompletionOnceCallback callback_;
size_t num_total_streams_ = 0;
raw_ptr<base::SequencedTaskRunner> task_runner_;
NetLogWithSource net_log_;
LoadTimingInfo::ConnectTiming connect_timing_;
std::unique_ptr<QuicConnectionLogger> logger_;
std::unique_ptr<QuicHttp3Logger> http3_logger_;
bool going_away_ = false;
quic::ConnectionCloseSource source_ = quic::ConnectionCloseSource::FROM_SELF;
bool port_migration_detected_ = false;
bool quic_connection_migration_attempted_ = false;
bool quic_connection_migration_successful_ = false;
scoped_refptr<QuicChromiumPacketWriter::ReusableIOBuffer> packet_;
handles::NetworkHandle default_network_;
int retry_migrate_back_count_ = 0;
base::OneShotTimer migrate_back_to_default_timer_;
MigrationCause current_migration_cause_ = UNKNOWN_CAUSE;
bool send_packet_after_migration_ = false;
bool wait_for_new_network_ = false;
bool ignore_read_error_ = false;
bool attempted_zero_rtt_ = false;
size_t num_migrations_ = 0;
ConnectionMigrationInformation migration_info_;
quic::KeyUpdateReason last_key_update_reason_ =
quic::KeyUpdateReason::kInvalid;
QuicChromiumPathValidationWriterDelegate path_validation_writer_delegate_;
base::flat_map<url::SchemeHostPort, std::string>
accept_ch_entries_received_via_alps_;
std::set<url::SchemeHostPort> received_origins_;
std::vector<uint8_t> ech_config_list_;
std::vector<std::vector<uint8_t>> trust_anchor_ids_;
const bool allow_server_preferred_address_;
const MultiplexedSessionCreationInitiator session_creation_initiator_;
bool enable_periodic_ping_ = false;
bool crypto_handshake_complete_ = false;
base::WeakPtrFactory<QuicChromiumClientSession> weak_factory_{this};
};
namespace features {
NET_EXPORT BASE_DECLARE_FEATURE(
kQuicMigrationIgnoreDisconnectSignalDuringProbing);
NET_EXPORT BASE_DECLARE_FEATURE(kQuicRegisterConnectionClosePayload);
}
}
#endif