|  | // Copyright (c) 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. | 
|  |  | 
|  | #include "quiche/quic/qbone/qbone_server_session.h" | 
|  |  | 
|  | #include <string> | 
|  | #include <utility> | 
|  |  | 
|  | #include "absl/strings/string_view.h" | 
|  | #include "quiche/quic/core/quic_connection_id.h" | 
|  | #include "quiche/quic/core/quic_types.h" | 
|  | #include "quiche/quic/core/quic_utils.h" | 
|  | #include "quiche/quic/qbone/qbone_constants.h" | 
|  | #include "quiche/common/platform/api/quiche_command_line_flags.h" | 
|  |  | 
|  | DEFINE_QUICHE_COMMAND_LINE_FLAG( | 
|  | bool, qbone_server_defer_control_stream_creation, true, | 
|  | "If true, control stream in QBONE server session is created after " | 
|  | "encryption established."); | 
|  |  | 
|  | namespace quic { | 
|  |  | 
|  | bool QboneCryptoServerStreamHelper::CanAcceptClientHello( | 
|  | const CryptoHandshakeMessage& chlo, const QuicSocketAddress& client_address, | 
|  | const QuicSocketAddress& peer_address, | 
|  | const QuicSocketAddress& self_address, std::string* error_details) const { | 
|  | absl::string_view alpn; | 
|  | chlo.GetStringPiece(quic::kALPN, &alpn); | 
|  | if (alpn != QboneConstants::kQboneAlpn) { | 
|  | *error_details = "ALPN-indicated protocol is not qbone"; | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | QboneServerSession::QboneServerSession( | 
|  | const quic::ParsedQuicVersionVector& supported_versions, | 
|  | QuicConnection* connection, Visitor* owner, const QuicConfig& config, | 
|  | const QuicCryptoServerConfig* quic_crypto_server_config, | 
|  | QuicCompressedCertsCache* compressed_certs_cache, QbonePacketWriter* writer, | 
|  | QuicIpAddress self_ip, QuicIpAddress client_ip, | 
|  | size_t client_ip_subnet_length, QboneServerControlStream::Handler* handler) | 
|  | : QboneSessionBase(connection, owner, config, supported_versions, writer), | 
|  | processor_(self_ip, client_ip, client_ip_subnet_length, this, this), | 
|  | quic_crypto_server_config_(quic_crypto_server_config), | 
|  | compressed_certs_cache_(compressed_certs_cache), | 
|  | handler_(handler) {} | 
|  |  | 
|  | QboneServerSession::~QboneServerSession() {} | 
|  |  | 
|  | std::unique_ptr<QuicCryptoStream> QboneServerSession::CreateCryptoStream() { | 
|  | return CreateCryptoServerStream(quic_crypto_server_config_, | 
|  | compressed_certs_cache_, this, | 
|  | &stream_helper_); | 
|  | } | 
|  |  | 
|  | void QboneServerSession::CreateControlStream() { | 
|  | if (control_stream_ != nullptr) { | 
|  | return; | 
|  | } | 
|  | // Register the reserved control stream. | 
|  | auto control_stream = | 
|  | std::make_unique<QboneServerControlStream>(this, handler_); | 
|  | control_stream_ = control_stream.get(); | 
|  | ActivateStream(std::move(control_stream)); | 
|  | } | 
|  |  | 
|  | QuicStream* QboneServerSession::CreateControlStreamFromPendingStream( | 
|  | PendingStream* pending) { | 
|  | QUICHE_DCHECK(control_stream_ == nullptr); | 
|  | // Register the reserved control stream. | 
|  | auto control_stream = | 
|  | std::make_unique<QboneServerControlStream>(pending, this, handler_); | 
|  | control_stream_ = control_stream.get(); | 
|  | ActivateStream(std::move(control_stream)); | 
|  | return control_stream_; | 
|  | } | 
|  |  | 
|  | void QboneServerSession::Initialize() { | 
|  | QboneSessionBase::Initialize(); | 
|  | if (!quiche::GetQuicheCommandLineFlag( | 
|  | FLAGS_qbone_server_defer_control_stream_creation)) { | 
|  | CreateControlStream(); | 
|  | } | 
|  | } | 
|  |  | 
|  | void QboneServerSession::SetDefaultEncryptionLevel( | 
|  | quic::EncryptionLevel level) { | 
|  | QboneSessionBase::SetDefaultEncryptionLevel(level); | 
|  | if (quiche::GetQuicheCommandLineFlag( | 
|  | FLAGS_qbone_server_defer_control_stream_creation) && | 
|  | level == quic::ENCRYPTION_FORWARD_SECURE) { | 
|  | CreateControlStream(); | 
|  | } | 
|  | } | 
|  |  | 
|  | bool QboneServerSession::SendClientRequest(const QboneClientRequest& request) { | 
|  | if (!control_stream_) { | 
|  | QUIC_BUG(quic_bug_11026_1) | 
|  | << "Cannot send client request before control stream is created."; | 
|  | return false; | 
|  | } | 
|  | return control_stream_->SendRequest(request); | 
|  | } | 
|  |  | 
|  | void QboneServerSession::ProcessPacketFromNetwork(absl::string_view packet) { | 
|  | std::string buffer = std::string(packet); | 
|  | processor_.ProcessPacket(&buffer, | 
|  | QbonePacketProcessor::Direction::FROM_NETWORK); | 
|  | } | 
|  |  | 
|  | void QboneServerSession::ProcessPacketFromPeer(absl::string_view packet) { | 
|  | std::string buffer = std::string(packet); | 
|  | processor_.ProcessPacket(&buffer, | 
|  | QbonePacketProcessor::Direction::FROM_OFF_NETWORK); | 
|  | } | 
|  |  | 
|  | void QboneServerSession::SendPacketToClient(absl::string_view packet) { | 
|  | SendPacketToPeer(packet); | 
|  | } | 
|  |  | 
|  | void QboneServerSession::SendPacketToNetwork(absl::string_view packet) { | 
|  | QUICHE_DCHECK(writer_ != nullptr); | 
|  | writer_->WritePacketToNetwork(packet.data(), packet.size()); | 
|  | } | 
|  |  | 
|  | }  // namespace quic |