| // Copyright (c) 2012 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_simple_server_session.h" | 
 |  | 
 | #include <utility> | 
 |  | 
 | #include "net/third_party/quiche/src/quic/core/http/quic_spdy_session.h" | 
 | #include "net/third_party/quiche/src/quic/core/quic_connection.h" | 
 | #include "net/third_party/quiche/src/quic/core/quic_utils.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/tools/quic_simple_server_stream.h" | 
 |  | 
 | namespace quic { | 
 |  | 
 | QuicSimpleServerSession::QuicSimpleServerSession( | 
 |     const QuicConfig& config, | 
 |     const ParsedQuicVersionVector& supported_versions, | 
 |     QuicConnection* connection, | 
 |     QuicSession::Visitor* visitor, | 
 |     QuicCryptoServerStream::Helper* helper, | 
 |     const QuicCryptoServerConfig* crypto_config, | 
 |     QuicCompressedCertsCache* compressed_certs_cache, | 
 |     QuicSimpleServerBackend* quic_simple_server_backend) | 
 |     : QuicServerSessionBase(config, | 
 |                             supported_versions, | 
 |                             connection, | 
 |                             visitor, | 
 |                             helper, | 
 |                             crypto_config, | 
 |                             compressed_certs_cache), | 
 |       highest_promised_stream_id_( | 
 |           QuicUtils::GetInvalidStreamId(connection->transport_version())), | 
 |       quic_simple_server_backend_(quic_simple_server_backend) { | 
 |   DCHECK(quic_simple_server_backend_); | 
 | } | 
 |  | 
 | QuicSimpleServerSession::~QuicSimpleServerSession() { | 
 |   delete connection(); | 
 | } | 
 |  | 
 | QuicCryptoServerStreamBase* | 
 | QuicSimpleServerSession::CreateQuicCryptoServerStream( | 
 |     const QuicCryptoServerConfig* crypto_config, | 
 |     QuicCompressedCertsCache* compressed_certs_cache) { | 
 |   return new QuicCryptoServerStream(crypto_config, compressed_certs_cache, this, | 
 |                                     stream_helper()); | 
 | } | 
 |  | 
 | void QuicSimpleServerSession::OnStreamFrame(const QuicStreamFrame& frame) { | 
 |   if (!IsIncomingStream(frame.stream_id)) { | 
 |     QUIC_LOG(WARNING) << "Client shouldn't send data on server push stream"; | 
 |     connection()->CloseConnection( | 
 |         QUIC_INVALID_STREAM_ID, "Client sent data on server push stream", | 
 |         ConnectionCloseBehavior::SEND_CONNECTION_CLOSE_PACKET); | 
 |     return; | 
 |   } | 
 |   QuicSpdySession::OnStreamFrame(frame); | 
 | } | 
 |  | 
 | void QuicSimpleServerSession::PromisePushResources( | 
 |     const std::string& request_url, | 
 |     const std::list<QuicBackendResponse::ServerPushInfo>& resources, | 
 |     QuicStreamId original_stream_id, | 
 |     const spdy::SpdyStreamPrecedence& original_precedence, | 
 |     const spdy::SpdyHeaderBlock& original_request_headers) { | 
 |   if (!server_push_enabled()) { | 
 |     return; | 
 |   } | 
 |  | 
 |   for (QuicBackendResponse::ServerPushInfo resource : resources) { | 
 |     spdy::SpdyHeaderBlock headers = SynthesizePushRequestHeaders( | 
 |         request_url, resource, original_request_headers); | 
 |     highest_promised_stream_id_ += | 
 |         QuicUtils::StreamIdDelta(transport_version()); | 
 |     if (VersionUsesHttp3(transport_version()) && | 
 |         highest_promised_stream_id_ > max_allowed_push_id()) { | 
 |       return; | 
 |     } | 
 |     SendPushPromise(original_stream_id, highest_promised_stream_id_, | 
 |                     headers.Clone()); | 
 |     promised_streams_.push_back(PromisedStreamInfo( | 
 |         std::move(headers), highest_promised_stream_id_, | 
 |         use_http2_priority_write_scheduler() | 
 |             ? original_precedence | 
 |             : spdy::SpdyStreamPrecedence(resource.priority))); | 
 |   } | 
 |  | 
 |   // Procese promised push request as many as possible. | 
 |   HandlePromisedPushRequests(); | 
 | } | 
 |  | 
 | QuicSpdyStream* QuicSimpleServerSession::CreateIncomingStream(QuicStreamId id) { | 
 |   if (!ShouldCreateIncomingStream(id)) { | 
 |     return nullptr; | 
 |   } | 
 |  | 
 |   QuicSpdyStream* stream = new QuicSimpleServerStream( | 
 |       id, this, BIDIRECTIONAL, quic_simple_server_backend_); | 
 |   ActivateStream(QuicWrapUnique(stream)); | 
 |   return stream; | 
 | } | 
 |  | 
 | QuicSpdyStream* QuicSimpleServerSession::CreateIncomingStream( | 
 |     PendingStream* pending) { | 
 |   QuicSpdyStream* stream = new QuicSimpleServerStream( | 
 |       pending, this, BIDIRECTIONAL, quic_simple_server_backend_); | 
 |   ActivateStream(QuicWrapUnique(stream)); | 
 |   return stream; | 
 | } | 
 |  | 
 | QuicSimpleServerStream* | 
 | QuicSimpleServerSession::CreateOutgoingBidirectionalStream() { | 
 |   DCHECK(false); | 
 |   return nullptr; | 
 | } | 
 |  | 
 | QuicSimpleServerStream* | 
 | QuicSimpleServerSession::CreateOutgoingUnidirectionalStream() { | 
 |   if (!ShouldCreateOutgoingUnidirectionalStream()) { | 
 |     return nullptr; | 
 |   } | 
 |  | 
 |   QuicSimpleServerStream* stream = new QuicSimpleServerStream( | 
 |       GetNextOutgoingUnidirectionalStreamId(), this, WRITE_UNIDIRECTIONAL, | 
 |       quic_simple_server_backend_); | 
 |   ActivateStream(QuicWrapUnique(stream)); | 
 |   return stream; | 
 | } | 
 |  | 
 | void QuicSimpleServerSession::HandleFrameOnNonexistentOutgoingStream( | 
 |     QuicStreamId stream_id) { | 
 |   // If this stream is a promised but not created stream (stream_id within the | 
 |   // range of next_outgoing_stream_id_ and highes_promised_stream_id_), | 
 |   // connection shouldn't be closed. | 
 |   // Otherwise behave in the same way as base class. | 
 |   if (highest_promised_stream_id_ == | 
 |           QuicUtils::GetInvalidStreamId(transport_version()) || | 
 |       stream_id > highest_promised_stream_id_) { | 
 |     QuicSession::HandleFrameOnNonexistentOutgoingStream(stream_id); | 
 |   } | 
 | } | 
 |  | 
 | void QuicSimpleServerSession::HandleRstOnValidNonexistentStream( | 
 |     const QuicRstStreamFrame& frame) { | 
 |   QuicSession::HandleRstOnValidNonexistentStream(frame); | 
 |   if (!IsClosedStream(frame.stream_id)) { | 
 |     // If a nonexistent stream is not a closed stream and still valid, it must | 
 |     // be a locally preserved stream. Resetting this kind of stream means | 
 |     // cancelling the promised server push. | 
 |     // Since PromisedStreamInfo are queued in sequence, the corresponding | 
 |     // index for it in promised_streams_ can be calculated. | 
 |     QuicStreamId next_stream_id = next_outgoing_unidirectional_stream_id(); | 
 |     if (VersionHasIetfQuicFrames(transport_version())) { | 
 |       DCHECK(!QuicUtils::IsBidirectionalStreamId(frame.stream_id)); | 
 |     } | 
 |     DCHECK_GE(frame.stream_id, next_stream_id); | 
 |     size_t index = (frame.stream_id - next_stream_id) / | 
 |                    QuicUtils::StreamIdDelta(transport_version()); | 
 |     DCHECK_LE(index, promised_streams_.size()); | 
 |     promised_streams_[index].is_cancelled = true; | 
 |     control_frame_manager().WriteOrBufferRstStream(frame.stream_id, | 
 |                                                    QUIC_RST_ACKNOWLEDGEMENT, 0); | 
 |     connection()->OnStreamReset(frame.stream_id, QUIC_RST_ACKNOWLEDGEMENT); | 
 |   } | 
 | } | 
 |  | 
 | spdy::SpdyHeaderBlock QuicSimpleServerSession::SynthesizePushRequestHeaders( | 
 |     std::string request_url, | 
 |     QuicBackendResponse::ServerPushInfo resource, | 
 |     const spdy::SpdyHeaderBlock& original_request_headers) { | 
 |   QuicUrl push_request_url = resource.request_url; | 
 |  | 
 |   spdy::SpdyHeaderBlock spdy_headers = original_request_headers.Clone(); | 
 |   // :authority could be different from original request. | 
 |   spdy_headers[":authority"] = push_request_url.host(); | 
 |   spdy_headers[":path"] = push_request_url.path(); | 
 |   // Push request always use GET. | 
 |   spdy_headers[":method"] = "GET"; | 
 |   spdy_headers["referer"] = request_url; | 
 |   spdy_headers[":scheme"] = push_request_url.scheme(); | 
 |   // It is not possible to push a response to a request that includes a request | 
 |   // body. | 
 |   spdy_headers["content-length"] = "0"; | 
 |   // Remove "host" field as push request is a directly generated HTTP2 request | 
 |   // which should use ":authority" instead of "host". | 
 |   spdy_headers.erase("host"); | 
 |   return spdy_headers; | 
 | } | 
 |  | 
 | void QuicSimpleServerSession::SendPushPromise(QuicStreamId original_stream_id, | 
 |                                               QuicStreamId promised_stream_id, | 
 |                                               spdy::SpdyHeaderBlock headers) { | 
 |   QUIC_DLOG(INFO) << "stream " << original_stream_id | 
 |                   << " send PUSH_PROMISE for promised stream " | 
 |                   << promised_stream_id; | 
 |   WritePushPromise(original_stream_id, promised_stream_id, std::move(headers)); | 
 | } | 
 |  | 
 | void QuicSimpleServerSession::HandlePromisedPushRequests() { | 
 |   while (!promised_streams_.empty() && | 
 |          ShouldCreateOutgoingUnidirectionalStream()) { | 
 |     PromisedStreamInfo& promised_info = promised_streams_.front(); | 
 |     DCHECK_EQ(next_outgoing_unidirectional_stream_id(), | 
 |               promised_info.stream_id); | 
 |  | 
 |     if (promised_info.is_cancelled) { | 
 |       // This stream has been reset by client. Skip this stream id. | 
 |       promised_streams_.pop_front(); | 
 |       GetNextOutgoingUnidirectionalStreamId(); | 
 |       return; | 
 |     } | 
 |  | 
 |     QuicSimpleServerStream* promised_stream = | 
 |         static_cast<QuicSimpleServerStream*>( | 
 |             CreateOutgoingUnidirectionalStream()); | 
 |     DCHECK_NE(promised_stream, nullptr); | 
 |     DCHECK_EQ(promised_info.stream_id, promised_stream->id()); | 
 |     QUIC_DLOG(INFO) << "created server push stream " << promised_stream->id(); | 
 |  | 
 |     promised_stream->SetPriority(promised_info.precedence); | 
 |  | 
 |     spdy::SpdyHeaderBlock request_headers( | 
 |         std::move(promised_info.request_headers)); | 
 |  | 
 |     promised_streams_.pop_front(); | 
 |     promised_stream->PushResponse(std::move(request_headers)); | 
 |   } | 
 | } | 
 |  | 
 | void QuicSimpleServerSession::OnCanCreateNewOutgoingStream( | 
 |     bool unidirectional) { | 
 |   QuicSpdySession::OnCanCreateNewOutgoingStream(unidirectional); | 
 |   if (unidirectional) { | 
 |     HandlePromisedPushRequests(); | 
 |   } | 
 | } | 
 |  | 
 | void QuicSimpleServerSession::MaybeInitializeHttp3UnidirectionalStreams() { | 
 |   size_t previous_static_stream_count = num_outgoing_static_streams(); | 
 |   QuicSpdySession::MaybeInitializeHttp3UnidirectionalStreams(); | 
 |   size_t current_static_stream_count = num_outgoing_static_streams(); | 
 |   DCHECK_GE(current_static_stream_count, previous_static_stream_count); | 
 |   highest_promised_stream_id_ += | 
 |       QuicUtils::StreamIdDelta(transport_version()) * | 
 |       (current_static_stream_count - previous_static_stream_count); | 
 | } | 
 | }  // namespace quic |