| OLD | NEW |
| (Empty) |
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "net/quic/quic_client_session_base.h" | |
| 6 | |
| 7 #include "net/quic/quic_client_promised_info.h" | |
| 8 #include "net/quic/quic_flags.h" | |
| 9 #include "net/quic/spdy_utils.h" | |
| 10 | |
| 11 using base::StringPiece; | |
| 12 using std::string; | |
| 13 | |
| 14 namespace net { | |
| 15 | |
| 16 QuicClientSessionBase::QuicClientSessionBase( | |
| 17 QuicConnection* connection, | |
| 18 QuicClientPushPromiseIndex* push_promise_index, | |
| 19 const QuicConfig& config) | |
| 20 : QuicSpdySession(connection, config), | |
| 21 push_promise_index_(push_promise_index), | |
| 22 largest_promised_stream_id_(kInvalidStreamId) {} | |
| 23 | |
| 24 QuicClientSessionBase::~QuicClientSessionBase() { | |
| 25 // all promised streams for this session | |
| 26 for (auto& it : promised_by_id_) { | |
| 27 DVLOG(1) << "erase stream " << it.first << " url " << it.second->url(); | |
| 28 push_promise_index_->promised_by_url()->erase(it.second->url()); | |
| 29 } | |
| 30 } | |
| 31 | |
| 32 void QuicClientSessionBase::OnConfigNegotiated() { | |
| 33 QuicSpdySession::OnConfigNegotiated(); | |
| 34 } | |
| 35 | |
| 36 void QuicClientSessionBase::OnCryptoHandshakeEvent(CryptoHandshakeEvent event) { | |
| 37 QuicSession::OnCryptoHandshakeEvent(event); | |
| 38 } | |
| 39 | |
| 40 void QuicClientSessionBase::OnPromiseHeaders(QuicStreamId stream_id, | |
| 41 StringPiece headers_data) { | |
| 42 QuicSpdyStream* stream = GetSpdyDataStream(stream_id); | |
| 43 if (!stream) { | |
| 44 // It's quite possible to receive headers after a stream has been reset. | |
| 45 return; | |
| 46 } | |
| 47 stream->OnPromiseHeaders(headers_data); | |
| 48 } | |
| 49 | |
| 50 void QuicClientSessionBase::OnInitialHeadersComplete( | |
| 51 QuicStreamId stream_id, | |
| 52 const SpdyHeaderBlock& response_headers) { | |
| 53 // Note that the strong ordering of the headers stream means that | |
| 54 // QuicSpdyClientStream::OnPromiseHeadersComplete must have already | |
| 55 // been called (on the associated stream) if this is a promised | |
| 56 // stream. However, this stream may not have existed at this time, | |
| 57 // hence the need to query the session. | |
| 58 QuicClientPromisedInfo* promised = GetPromisedById(stream_id); | |
| 59 if (!promised) | |
| 60 return; | |
| 61 | |
| 62 promised->OnResponseHeaders(response_headers); | |
| 63 } | |
| 64 | |
| 65 void QuicClientSessionBase::OnPromiseHeadersComplete( | |
| 66 QuicStreamId stream_id, | |
| 67 QuicStreamId promised_stream_id, | |
| 68 size_t frame_len) { | |
| 69 if (promised_stream_id != kInvalidStreamId && | |
| 70 promised_stream_id <= largest_promised_stream_id_) { | |
| 71 connection()->CloseConnection( | |
| 72 QUIC_INVALID_STREAM_ID, | |
| 73 "Received push stream id lesser or equal to the" | |
| 74 " last accepted before", | |
| 75 ConnectionCloseBehavior::SEND_CONNECTION_CLOSE_PACKET); | |
| 76 return; | |
| 77 } | |
| 78 largest_promised_stream_id_ = promised_stream_id; | |
| 79 | |
| 80 QuicSpdyStream* stream = GetSpdyDataStream(stream_id); | |
| 81 if (!stream) { | |
| 82 // It's quite possible to receive headers after a stream has been reset. | |
| 83 return; | |
| 84 } | |
| 85 stream->OnPromiseHeadersComplete(promised_stream_id, frame_len); | |
| 86 } | |
| 87 | |
| 88 void QuicClientSessionBase::OnPromiseHeaderList( | |
| 89 QuicStreamId stream_id, | |
| 90 QuicStreamId promised_stream_id, | |
| 91 size_t frame_len, | |
| 92 const QuicHeaderList& header_list) { | |
| 93 if (promised_stream_id != kInvalidStreamId && | |
| 94 promised_stream_id <= largest_promised_stream_id_) { | |
| 95 connection()->CloseConnection( | |
| 96 QUIC_INVALID_STREAM_ID, | |
| 97 "Received push stream id lesser or equal to the" | |
| 98 " last accepted before", | |
| 99 ConnectionCloseBehavior::SEND_CONNECTION_CLOSE_PACKET); | |
| 100 return; | |
| 101 } | |
| 102 largest_promised_stream_id_ = promised_stream_id; | |
| 103 | |
| 104 QuicSpdyStream* stream = GetSpdyDataStream(stream_id); | |
| 105 if (!stream) { | |
| 106 // It's quite possible to receive headers after a stream has been reset. | |
| 107 return; | |
| 108 } | |
| 109 stream->OnPromiseHeaderList(promised_stream_id, frame_len, header_list); | |
| 110 } | |
| 111 | |
| 112 void QuicClientSessionBase::HandlePromised(QuicStreamId /* associated_id */, | |
| 113 QuicStreamId id, | |
| 114 const SpdyHeaderBlock& headers) { | |
| 115 // Due to pathalogical packet re-ordering, it is possible that | |
| 116 // frames for the promised stream have already arrived, and the | |
| 117 // promised stream could be active or closed. | |
| 118 if (IsClosedStream(id)) { | |
| 119 // There was a RST on the data stream already, perhaps | |
| 120 // QUIC_REFUSED_STREAM? | |
| 121 DVLOG(1) << "Promise ignored for stream " << id | |
| 122 << " that is already closed"; | |
| 123 return; | |
| 124 } | |
| 125 | |
| 126 if (push_promise_index_->promised_by_url()->size() >= get_max_promises()) { | |
| 127 DVLOG(1) << "Too many promises, rejecting promise for stream " << id; | |
| 128 ResetPromised(id, QUIC_REFUSED_STREAM); | |
| 129 return; | |
| 130 } | |
| 131 | |
| 132 const string url = SpdyUtils::GetUrlFromHeaderBlock(headers); | |
| 133 QuicClientPromisedInfo* old_promised = GetPromisedByUrl(url); | |
| 134 if (old_promised) { | |
| 135 DVLOG(1) << "Promise for stream " << id << " is duplicate URL " << url | |
| 136 << " of previous promise for stream " << old_promised->id(); | |
| 137 ResetPromised(id, QUIC_DUPLICATE_PROMISE_URL); | |
| 138 return; | |
| 139 } | |
| 140 | |
| 141 if (GetPromisedById(id)) { | |
| 142 // OnPromiseHeadersComplete() would have closed the connection if | |
| 143 // promised id is a duplicate. | |
| 144 QUIC_BUG << "Duplicate promise for id " << id; | |
| 145 return; | |
| 146 } | |
| 147 | |
| 148 QuicClientPromisedInfo* promised = new QuicClientPromisedInfo(this, id, url); | |
| 149 std::unique_ptr<QuicClientPromisedInfo> promised_owner(promised); | |
| 150 promised->Init(); | |
| 151 DVLOG(1) << "stream " << id << " emplace url " << url; | |
| 152 (*push_promise_index_->promised_by_url())[url] = promised; | |
| 153 promised_by_id_[id] = std::move(promised_owner); | |
| 154 promised->OnPromiseHeaders(headers); | |
| 155 } | |
| 156 | |
| 157 QuicClientPromisedInfo* QuicClientSessionBase::GetPromisedByUrl( | |
| 158 const string& url) { | |
| 159 QuicPromisedByUrlMap::iterator it = | |
| 160 push_promise_index_->promised_by_url()->find(url); | |
| 161 if (it != push_promise_index_->promised_by_url()->end()) { | |
| 162 return it->second; | |
| 163 } | |
| 164 return nullptr; | |
| 165 } | |
| 166 | |
| 167 QuicClientPromisedInfo* QuicClientSessionBase::GetPromisedById( | |
| 168 const QuicStreamId id) { | |
| 169 QuicPromisedByIdMap::iterator it = promised_by_id_.find(id); | |
| 170 if (it != promised_by_id_.end()) { | |
| 171 return it->second.get(); | |
| 172 } | |
| 173 return nullptr; | |
| 174 } | |
| 175 | |
| 176 QuicSpdyStream* QuicClientSessionBase::GetPromisedStream( | |
| 177 const QuicStreamId id) { | |
| 178 if (IsClosedStream(id)) { | |
| 179 return nullptr; | |
| 180 } | |
| 181 DynamicStreamMap::iterator it = dynamic_streams().find(id); | |
| 182 if (it != dynamic_streams().end()) { | |
| 183 return static_cast<QuicSpdyStream*>(it->second); | |
| 184 } | |
| 185 QUIC_BUG << "Open promised stream " << id << " is missing!"; | |
| 186 return nullptr; | |
| 187 } | |
| 188 | |
| 189 void QuicClientSessionBase::DeletePromised(QuicClientPromisedInfo* promised) { | |
| 190 push_promise_index_->promised_by_url()->erase(promised->url()); | |
| 191 // Since promised_by_id_ contains the unique_ptr, this will destroy | |
| 192 // promised. | |
| 193 promised_by_id_.erase(promised->id()); | |
| 194 } | |
| 195 | |
| 196 void QuicClientSessionBase::ResetPromised(QuicStreamId id, | |
| 197 QuicRstStreamErrorCode error_code) { | |
| 198 SendRstStream(id, error_code, 0); | |
| 199 if (!IsOpenStream(id)) { | |
| 200 MaybeIncreaseLargestPeerStreamId(id); | |
| 201 InsertLocallyClosedStreamsHighestOffset(id, 0); | |
| 202 } | |
| 203 } | |
| 204 | |
| 205 } // namespace net | |
| OLD | NEW |