blob: d07da11672b913540ac1c01fa848541780104624 [file] [log] [blame]
vasilvvd88f1622019-11-04 13:50:53 -08001// Copyright (c) 2019 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/third_party/quiche/src/quic/tools/quic_transport_simple_server_session.h"
6
7#include <memory>
8
9#include "url/gurl.h"
10#include "url/origin.h"
vasilvv2b0ab242020-01-07 07:32:09 -080011#include "net/third_party/quiche/src/quic/core/quic_buffer_allocator.h"
vasilvvd88f1622019-11-04 13:50:53 -080012#include "net/third_party/quiche/src/quic/core/quic_types.h"
13#include "net/third_party/quiche/src/quic/core/quic_versions.h"
14#include "net/third_party/quiche/src/quic/platform/api/quic_flags.h"
15#include "net/third_party/quiche/src/quic/platform/api/quic_logging.h"
vasilvvd88f1622019-11-04 13:50:53 -080016#include "net/third_party/quiche/src/quic/quic_transport/quic_transport_protocol.h"
17#include "net/third_party/quiche/src/quic/quic_transport/quic_transport_stream.h"
18
19namespace quic {
20
21namespace {
22
23// Discards any incoming data.
24class DiscardVisitor : public QuicTransportStream::Visitor {
25 public:
26 DiscardVisitor(QuicTransportStream* stream) : stream_(stream) {}
27
28 void OnCanRead() override {
29 std::string buffer;
30 size_t bytes_read = stream_->Read(&buffer);
31 QUIC_DVLOG(2) << "Read " << bytes_read << " bytes from stream "
32 << stream_->id();
33 }
34
35 void OnFinRead() override {}
36 void OnCanWrite() override {}
37
38 private:
39 QuicTransportStream* stream_;
40};
41
42// Echoes any incoming data back on the same stream.
43class BidirectionalEchoVisitor : public QuicTransportStream::Visitor {
44 public:
45 BidirectionalEchoVisitor(QuicTransportStream* stream) : stream_(stream) {}
46
47 void OnCanRead() override {
48 stream_->Read(&buffer_);
49 OnCanWrite();
50 }
51
52 void OnFinRead() override {
53 bool success = stream_->SendFin();
54 DCHECK(success);
55 }
56
57 void OnCanWrite() override {
58 if (buffer_.empty()) {
59 return;
60 }
61
62 bool success = stream_->Write(buffer_);
63 if (success) {
64 buffer_ = "";
65 }
66 }
67
68 private:
69 QuicTransportStream* stream_;
70 std::string buffer_;
71};
72
73// Buffers all of the data and calls EchoStreamBack() on the parent session.
74class UnidirectionalEchoReadVisitor : public QuicTransportStream::Visitor {
75 public:
76 UnidirectionalEchoReadVisitor(QuicTransportSimpleServerSession* session,
77 QuicTransportStream* stream)
78 : session_(session), stream_(stream) {}
79
80 void OnCanRead() override {
81 bool success = stream_->Read(&buffer_);
82 DCHECK(success);
83 }
84
85 void OnFinRead() override {
86 QUIC_DVLOG(1) << "Finished receiving data on stream " << stream_->id()
87 << ", queueing up the echo";
88 session_->EchoStreamBack(buffer_);
89 }
90
91 void OnCanWrite() override { QUIC_NOTREACHED(); }
92
93 private:
94 QuicTransportSimpleServerSession* session_;
95 QuicTransportStream* stream_;
96 std::string buffer_;
97};
98
99// Sends supplied data.
100class UnidirectionalEchoWriteVisitor : public QuicTransportStream::Visitor {
101 public:
102 UnidirectionalEchoWriteVisitor(QuicTransportStream* stream,
103 const std::string& data)
104 : stream_(stream), data_(data) {}
105
106 void OnCanRead() override { QUIC_NOTREACHED(); }
107 void OnFinRead() override { QUIC_NOTREACHED(); }
108 void OnCanWrite() override {
109 if (data_.empty()) {
110 return;
111 }
112 if (!stream_->Write(data_)) {
113 return;
114 }
115 data_ = "";
116 bool fin_sent = stream_->SendFin();
117 DCHECK(fin_sent);
118 }
119
120 private:
121 QuicTransportStream* stream_;
122 std::string data_;
123};
124
125} // namespace
126
127QuicTransportSimpleServerSession::QuicTransportSimpleServerSession(
128 QuicConnection* connection,
129 bool owns_connection,
130 Visitor* owner,
131 const QuicConfig& config,
132 const ParsedQuicVersionVector& supported_versions,
133 const QuicCryptoServerConfig* crypto_config,
134 QuicCompressedCertsCache* compressed_certs_cache,
vasilvvd88f1622019-11-04 13:50:53 -0800135 std::vector<url::Origin> accepted_origins)
136 : QuicTransportServerSession(connection,
137 owner,
138 config,
139 supported_versions,
140 crypto_config,
141 compressed_certs_cache,
142 this),
vasilvvd88f1622019-11-04 13:50:53 -0800143 owns_connection_(owns_connection),
vasilvvd7692cd2019-12-06 08:06:31 -0800144 mode_(DISCARD),
wub50c6a372019-11-25 05:34:56 -0800145 accepted_origins_(accepted_origins) {}
vasilvvd88f1622019-11-04 13:50:53 -0800146
147QuicTransportSimpleServerSession::~QuicTransportSimpleServerSession() {
148 if (owns_connection_) {
ianswett6aefa0b2019-12-10 07:26:15 -0800149 DeleteConnection();
vasilvvd88f1622019-11-04 13:50:53 -0800150 }
151}
152
153void QuicTransportSimpleServerSession::OnIncomingDataStream(
154 QuicTransportStream* stream) {
155 switch (mode_) {
156 case DISCARD:
157 stream->set_visitor(std::make_unique<DiscardVisitor>(stream));
158 break;
159
160 case ECHO:
161 switch (stream->type()) {
162 case BIDIRECTIONAL:
163 QUIC_DVLOG(1) << "Opening bidirectional echo stream " << stream->id();
164 stream->set_visitor(
165 std::make_unique<BidirectionalEchoVisitor>(stream));
166 break;
167 case READ_UNIDIRECTIONAL:
168 QUIC_DVLOG(1)
169 << "Started receiving data on unidirectional echo stream "
170 << stream->id();
171 stream->set_visitor(
172 std::make_unique<UnidirectionalEchoReadVisitor>(this, stream));
173 break;
174 default:
175 QUIC_NOTREACHED();
176 break;
177 }
178 break;
QUICHE team76c186d2020-09-17 18:54:18 -0700179
180 case OUTGOING_BIDIRECTIONAL:
181 stream->set_visitor(std::make_unique<DiscardVisitor>(stream));
182 ++pending_outgoing_bidirectional_streams_;
183 MaybeCreateOutgoingBidirectionalStream();
184 break;
vasilvvd88f1622019-11-04 13:50:53 -0800185 }
186}
187
188void QuicTransportSimpleServerSession::OnCanCreateNewOutgoingStream(
189 bool unidirectional) {
190 if (mode_ == ECHO && unidirectional) {
191 MaybeEchoStreamsBack();
QUICHE team76c186d2020-09-17 18:54:18 -0700192 } else if (mode_ == OUTGOING_BIDIRECTIONAL && !unidirectional) {
193 MaybeCreateOutgoingBidirectionalStream();
vasilvvd88f1622019-11-04 13:50:53 -0800194 }
195}
196
197bool QuicTransportSimpleServerSession::CheckOrigin(url::Origin origin) {
198 if (accepted_origins_.empty()) {
199 return true;
200 }
201
202 for (const url::Origin& accepted_origin : accepted_origins_) {
203 if (origin.IsSameOriginWith(accepted_origin)) {
204 return true;
205 }
206 }
207 return false;
208}
209
vasilvv78571892019-12-06 07:14:57 -0800210bool QuicTransportSimpleServerSession::ProcessPath(const GURL& url) {
vasilvvd7692cd2019-12-06 08:06:31 -0800211 if (url.path() == "/discard") {
212 mode_ = DISCARD;
213 return true;
214 }
215 if (url.path() == "/echo") {
216 mode_ = ECHO;
217 return true;
218 }
QUICHE team76c186d2020-09-17 18:54:18 -0700219 if (url.path() == "/receive-bidirectional") {
220 mode_ = OUTGOING_BIDIRECTIONAL;
221 return true;
222 }
vasilvvd7692cd2019-12-06 08:06:31 -0800223
224 QUIC_DLOG(WARNING) << "Unknown path requested: " << url.path();
225 return false;
vasilvv78571892019-12-06 07:14:57 -0800226}
227
vasilvv2b0ab242020-01-07 07:32:09 -0800228void QuicTransportSimpleServerSession::OnMessageReceived(
229 quiche::QuicheStringPiece message) {
230 if (mode_ != ECHO) {
231 return;
232 }
233 QuicUniqueBufferPtr buffer = MakeUniqueBuffer(
234 connection()->helper()->GetStreamSendBufferAllocator(), message.size());
235 memcpy(buffer.get(), message.data(), message.size());
236 datagram_queue()->SendOrQueueDatagram(
237 QuicMemSlice(std::move(buffer), message.size()));
238}
239
vasilvvd88f1622019-11-04 13:50:53 -0800240void QuicTransportSimpleServerSession::MaybeEchoStreamsBack() {
241 while (!streams_to_echo_back_.empty() &&
242 CanOpenNextOutgoingUnidirectionalStream()) {
243 // Remove the stream from the queue first, in order to avoid accidentally
244 // entering an infinite loop in case any of the following code calls
245 // OnCanCreateNewOutgoingStream().
246 std::string data = std::move(streams_to_echo_back_.front());
247 streams_to_echo_back_.pop_front();
248
249 auto stream_owned = std::make_unique<QuicTransportStream>(
250 GetNextOutgoingUnidirectionalStreamId(), this, this);
251 QuicTransportStream* stream = stream_owned.get();
252 ActivateStream(std::move(stream_owned));
253 QUIC_DVLOG(1) << "Opened echo response stream " << stream->id();
254
255 stream->set_visitor(
256 std::make_unique<UnidirectionalEchoWriteVisitor>(stream, data));
257 stream->visitor()->OnCanWrite();
258 }
259}
260
QUICHE team76c186d2020-09-17 18:54:18 -0700261void QuicTransportSimpleServerSession::
262 MaybeCreateOutgoingBidirectionalStream() {
263 while (pending_outgoing_bidirectional_streams_ > 0 &&
264 CanOpenNextOutgoingBidirectionalStream()) {
265 auto stream_owned = std::make_unique<QuicTransportStream>(
266 GetNextOutgoingBidirectionalStreamId(), this, this);
267 QuicTransportStream* stream = stream_owned.get();
268 ActivateStream(std::move(stream_owned));
269 QUIC_DVLOG(1) << "Opened outgoing bidirectional stream " << stream->id();
270 stream->set_visitor(std::make_unique<BidirectionalEchoVisitor>(stream));
271 if (!stream->Write("hello")) {
272 QUIC_DVLOG(1) << "Write failed.";
273 }
274 --pending_outgoing_bidirectional_streams_;
275 }
276}
277
vasilvvd88f1622019-11-04 13:50:53 -0800278} // namespace quic