| // Copyright (c) 2015 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 "net/third_party/quiche/src/quic/tools/quic_spdy_client_base.h" | 
 |  | 
 | #include "net/third_party/quiche/src/quic/core/crypto/quic_random.h" | 
 | #include "net/third_party/quiche/src/quic/core/http/spdy_utils.h" | 
 | #include "net/third_party/quiche/src/quic/core/quic_server_id.h" | 
 | #include "net/third_party/quiche/src/quic/platform/api/quic_flags.h" | 
 | #include "net/third_party/quiche/src/quic/platform/api/quic_logging.h" | 
 | #include "net/third_party/quiche/src/quic/platform/api/quic_ptr_util.h" | 
 | #include "net/third_party/quiche/src/quic/platform/api/quic_text_utils.h" | 
 |  | 
 | using spdy::SpdyHeaderBlock; | 
 |  | 
 | namespace quic { | 
 |  | 
 | void QuicSpdyClientBase::ClientQuicDataToResend::Resend() { | 
 |   client_->SendRequest(*headers_, body_, fin_); | 
 |   headers_ = nullptr; | 
 | } | 
 |  | 
 | QuicSpdyClientBase::QuicDataToResend::QuicDataToResend( | 
 |     std::unique_ptr<SpdyHeaderBlock> headers, | 
 |     QuicStringPiece body, | 
 |     bool fin) | 
 |     : headers_(std::move(headers)), body_(body), fin_(fin) {} | 
 |  | 
 | QuicSpdyClientBase::QuicDataToResend::~QuicDataToResend() = default; | 
 |  | 
 | QuicSpdyClientBase::QuicSpdyClientBase( | 
 |     const QuicServerId& server_id, | 
 |     const ParsedQuicVersionVector& supported_versions, | 
 |     const QuicConfig& config, | 
 |     QuicConnectionHelperInterface* helper, | 
 |     QuicAlarmFactory* alarm_factory, | 
 |     std::unique_ptr<NetworkHelper> network_helper, | 
 |     std::unique_ptr<ProofVerifier> proof_verifier) | 
 |     : QuicClientBase(server_id, | 
 |                      supported_versions, | 
 |                      config, | 
 |                      helper, | 
 |                      alarm_factory, | 
 |                      std::move(network_helper), | 
 |                      std::move(proof_verifier)), | 
 |       store_response_(false), | 
 |       latest_response_code_(-1) {} | 
 |  | 
 | QuicSpdyClientBase::~QuicSpdyClientBase() { | 
 |   // We own the push promise index. We need to explicitly kill | 
 |   // the session before the push promise index goes out of scope. | 
 |   ResetSession(); | 
 | } | 
 |  | 
 | QuicSpdyClientSession* QuicSpdyClientBase::client_session() { | 
 |   return static_cast<QuicSpdyClientSession*>(QuicClientBase::session()); | 
 | } | 
 |  | 
 | void QuicSpdyClientBase::InitializeSession() { | 
 |   client_session()->Initialize(); | 
 |   client_session()->CryptoConnect(); | 
 | } | 
 |  | 
 | void QuicSpdyClientBase::OnClose(QuicSpdyStream* stream) { | 
 |   DCHECK(stream != nullptr); | 
 |   QuicSpdyClientStream* client_stream = | 
 |       static_cast<QuicSpdyClientStream*>(stream); | 
 |  | 
 |   const SpdyHeaderBlock& response_headers = client_stream->response_headers(); | 
 |   if (response_listener_ != nullptr) { | 
 |     response_listener_->OnCompleteResponse(stream->id(), response_headers, | 
 |                                            client_stream->data()); | 
 |   } | 
 |  | 
 |   // Store response headers and body. | 
 |   if (store_response_) { | 
 |     auto status = response_headers.find(":status"); | 
 |     if (status == response_headers.end() || | 
 |         !QuicTextUtils::StringToInt(status->second, &latest_response_code_)) { | 
 |       QUIC_LOG(ERROR) << "Invalid response headers"; | 
 |     } | 
 |     latest_response_headers_ = response_headers.DebugString(); | 
 |     preliminary_response_headers_ = | 
 |         client_stream->preliminary_headers().DebugString(); | 
 |     latest_response_header_block_ = response_headers.Clone(); | 
 |     latest_response_body_ = client_stream->data(); | 
 |     latest_response_trailers_ = | 
 |         client_stream->received_trailers().DebugString(); | 
 |   } | 
 | } | 
 |  | 
 | std::unique_ptr<QuicSession> QuicSpdyClientBase::CreateQuicClientSession( | 
 |     const quic::ParsedQuicVersionVector& supported_versions, | 
 |     QuicConnection* connection) { | 
 |   return QuicMakeUnique<QuicSpdyClientSession>( | 
 |       *config(), supported_versions, connection, server_id(), crypto_config(), | 
 |       &push_promise_index_); | 
 | } | 
 |  | 
 | void QuicSpdyClientBase::SendRequest(const SpdyHeaderBlock& headers, | 
 |                                      QuicStringPiece body, | 
 |                                      bool fin) { | 
 |   QuicClientPushPromiseIndex::TryHandle* handle; | 
 |   QuicAsyncStatus rv = push_promise_index()->Try(headers, this, &handle); | 
 |   if (rv == QUIC_SUCCESS) | 
 |     return; | 
 |  | 
 |   if (rv == QUIC_PENDING) { | 
 |     // May need to retry request if asynchronous rendezvous fails. | 
 |     AddPromiseDataToResend(headers, body, fin); | 
 |     return; | 
 |   } | 
 |  | 
 |   QuicSpdyClientStream* stream = CreateClientStream(); | 
 |   if (stream == nullptr) { | 
 |     QUIC_BUG << "stream creation failed!"; | 
 |     return; | 
 |   } | 
 |   stream->SendRequest(headers.Clone(), body, fin); | 
 |   // Record this in case we need to resend. | 
 |   MaybeAddDataToResend(headers, body, fin); | 
 | } | 
 |  | 
 | void QuicSpdyClientBase::SendRequestAndWaitForResponse( | 
 |     const SpdyHeaderBlock& headers, | 
 |     QuicStringPiece body, | 
 |     bool fin) { | 
 |   SendRequest(headers, body, fin); | 
 |   while (WaitForEvents()) { | 
 |   } | 
 | } | 
 |  | 
 | void QuicSpdyClientBase::SendRequestsAndWaitForResponse( | 
 |     const std::vector<std::string>& url_list) { | 
 |   for (size_t i = 0; i < url_list.size(); ++i) { | 
 |     SpdyHeaderBlock headers; | 
 |     if (!SpdyUtils::PopulateHeaderBlockFromUrl(url_list[i], &headers)) { | 
 |       QUIC_BUG << "Unable to create request"; | 
 |       continue; | 
 |     } | 
 |     SendRequest(headers, "", true); | 
 |   } | 
 |   while (WaitForEvents()) { | 
 |   } | 
 | } | 
 |  | 
 | QuicSpdyClientStream* QuicSpdyClientBase::CreateClientStream() { | 
 |   if (!connected()) { | 
 |     return nullptr; | 
 |   } | 
 |  | 
 |   auto* stream = static_cast<QuicSpdyClientStream*>( | 
 |       client_session()->CreateOutgoingBidirectionalStream()); | 
 |   if (stream) { | 
 |     stream->SetPriority(QuicStream::kDefaultPriority); | 
 |     stream->set_visitor(this); | 
 |   } | 
 |   return stream; | 
 | } | 
 |  | 
 | int QuicSpdyClientBase::GetNumSentClientHellosFromSession() { | 
 |   return client_session()->GetNumSentClientHellos(); | 
 | } | 
 |  | 
 | int QuicSpdyClientBase::GetNumReceivedServerConfigUpdatesFromSession() { | 
 |   return client_session()->GetNumReceivedServerConfigUpdates(); | 
 | } | 
 |  | 
 | void QuicSpdyClientBase::MaybeAddDataToResend(const SpdyHeaderBlock& headers, | 
 |                                               QuicStringPiece body, | 
 |                                               bool fin) { | 
 |   if (!GetQuicReloadableFlag(enable_quic_stateless_reject_support)) { | 
 |     return; | 
 |   } | 
 |  | 
 |   if (client_session()->IsCryptoHandshakeConfirmed()) { | 
 |     // The handshake is confirmed.  No need to continue saving requests to | 
 |     // resend. | 
 |     data_to_resend_on_connect_.clear(); | 
 |     return; | 
 |   } | 
 |  | 
 |   // The handshake is not confirmed.  Push the data onto the queue of data to | 
 |   // resend if statelessly rejected. | 
 |   std::unique_ptr<SpdyHeaderBlock> new_headers( | 
 |       new SpdyHeaderBlock(headers.Clone())); | 
 |   std::unique_ptr<QuicDataToResend> data_to_resend( | 
 |       new ClientQuicDataToResend(std::move(new_headers), body, fin, this)); | 
 |   MaybeAddQuicDataToResend(std::move(data_to_resend)); | 
 | } | 
 |  | 
 | void QuicSpdyClientBase::MaybeAddQuicDataToResend( | 
 |     std::unique_ptr<QuicDataToResend> data_to_resend) { | 
 |   data_to_resend_on_connect_.push_back(std::move(data_to_resend)); | 
 | } | 
 |  | 
 | void QuicSpdyClientBase::ClearDataToResend() { | 
 |   data_to_resend_on_connect_.clear(); | 
 | } | 
 |  | 
 | void QuicSpdyClientBase::ResendSavedData() { | 
 |   // Calling Resend will re-enqueue the data, so swap out | 
 |   //  data_to_resend_on_connect_ before iterating. | 
 |   std::vector<std::unique_ptr<QuicDataToResend>> old_data; | 
 |   old_data.swap(data_to_resend_on_connect_); | 
 |   for (const auto& data : old_data) { | 
 |     data->Resend(); | 
 |   } | 
 | } | 
 |  | 
 | void QuicSpdyClientBase::AddPromiseDataToResend(const SpdyHeaderBlock& headers, | 
 |                                                 QuicStringPiece body, | 
 |                                                 bool fin) { | 
 |   std::unique_ptr<SpdyHeaderBlock> new_headers( | 
 |       new SpdyHeaderBlock(headers.Clone())); | 
 |   push_promise_data_to_resend_.reset( | 
 |       new ClientQuicDataToResend(std::move(new_headers), body, fin, this)); | 
 | } | 
 |  | 
 | bool QuicSpdyClientBase::CheckVary(const SpdyHeaderBlock& client_request, | 
 |                                    const SpdyHeaderBlock& promise_request, | 
 |                                    const SpdyHeaderBlock& promise_response) { | 
 |   return true; | 
 | } | 
 |  | 
 | void QuicSpdyClientBase::OnRendezvousResult(QuicSpdyStream* stream) { | 
 |   std::unique_ptr<ClientQuicDataToResend> data_to_resend = | 
 |       std::move(push_promise_data_to_resend_); | 
 |   if (stream) { | 
 |     stream->set_visitor(this); | 
 |     stream->OnBodyAvailable(); | 
 |   } else if (data_to_resend) { | 
 |     data_to_resend->Resend(); | 
 |   } | 
 | } | 
 |  | 
 | size_t QuicSpdyClientBase::latest_response_code() const { | 
 |   QUIC_BUG_IF(!store_response_) << "Response not stored!"; | 
 |   return latest_response_code_; | 
 | } | 
 |  | 
 | const std::string& QuicSpdyClientBase::latest_response_headers() const { | 
 |   QUIC_BUG_IF(!store_response_) << "Response not stored!"; | 
 |   return latest_response_headers_; | 
 | } | 
 |  | 
 | const std::string& QuicSpdyClientBase::preliminary_response_headers() const { | 
 |   QUIC_BUG_IF(!store_response_) << "Response not stored!"; | 
 |   return preliminary_response_headers_; | 
 | } | 
 |  | 
 | const SpdyHeaderBlock& QuicSpdyClientBase::latest_response_header_block() | 
 |     const { | 
 |   QUIC_BUG_IF(!store_response_) << "Response not stored!"; | 
 |   return latest_response_header_block_; | 
 | } | 
 |  | 
 | const std::string& QuicSpdyClientBase::latest_response_body() const { | 
 |   QUIC_BUG_IF(!store_response_) << "Response not stored!"; | 
 |   return latest_response_body_; | 
 | } | 
 |  | 
 | const std::string& QuicSpdyClientBase::latest_response_trailers() const { | 
 |   QUIC_BUG_IF(!store_response_) << "Response not stored!"; | 
 |   return latest_response_trailers_; | 
 | } | 
 |  | 
 | bool QuicSpdyClientBase::HasActiveRequests() { | 
 |   return client_session()->HasActiveRequestStreams(); | 
 | } | 
 |  | 
 | }  // namespace quic |