blob: 0182d70b95b27924fe9c182078c31bc0c1ac1fa3 [file] [log] [blame]
[email protected]ce9f7ffd2013-10-11 06:04:111// Copyright 2013 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/websockets/websocket_test_util.h"
6
tfarinaea94afc232015-10-20 04:23:367#include <stddef.h>
[email protected]94831522014-02-06 12:05:188#include <algorithm>
dchengc7eeda422015-12-26 03:56:489#include <utility>
[email protected]94831522014-02-06 12:05:1810
Bence Béky3a0c48532018-03-02 13:38:5111#include "base/strings/strcat.h"
Bence Béky46bfbc12018-02-22 19:28:2012#include "base/strings/string_util.h"
[email protected]a31ecc02013-12-05 08:30:5513#include "base/strings/stringprintf.h"
Bence Béky70dd7662018-03-02 17:13:2414#include "net/http/http_network_session.h"
Lily Houghtonffe89daa02018-03-09 18:30:0315#include "net/proxy_resolution/proxy_resolution_service.h"
[email protected]a31ecc02013-12-05 08:30:5516#include "net/socket/socket_test_util.h"
Ryan Hamilton2e003eea2018-05-02 00:24:2917#include "net/third_party/spdy/core/spdy_protocol.h"
Ramin Halavatica8d5252018-03-12 05:33:4918#include "net/traffic_annotation/network_traffic_annotation_test_helper.h"
Bence Béky8d1c6052018-02-07 12:48:1519#include "net/websockets/websocket_basic_handshake_stream.h"
mkwst4997ce82015-07-25 12:00:0520#include "url/origin.h"
[email protected]ce9f7ffd2013-10-11 06:04:1121
22namespace net {
23
24namespace {
tfarinaea94afc232015-10-20 04:23:3625
tfarina8a2c66c22015-10-13 19:14:4926const uint64_t kA = (static_cast<uint64_t>(0x5851f42d) << 32) +
27 static_cast<uint64_t>(0x4c957f2d);
28const uint64_t kC = 12345;
29const uint64_t kM = static_cast<uint64_t>(1) << 48;
[email protected]ce9f7ffd2013-10-11 06:04:1130
31} // namespace
32
tfarina8a2c66c22015-10-13 19:14:4933LinearCongruentialGenerator::LinearCongruentialGenerator(uint32_t seed)
[email protected]ce9f7ffd2013-10-11 06:04:1134 : current_(seed) {}
35
tfarina8a2c66c22015-10-13 19:14:4936uint32_t LinearCongruentialGenerator::Generate() {
37 uint64_t result = current_;
[email protected]ce9f7ffd2013-10-11 06:04:1138 current_ = (current_ * kA + kC) % kM;
tfarina8a2c66c22015-10-13 19:14:4939 return static_cast<uint32_t>(result >> 16);
[email protected]ce9f7ffd2013-10-11 06:04:1140}
41
Bence Béky3a0c48532018-03-02 13:38:5142std::string WebSocketExtraHeadersToString(
43 const WebSocketExtraHeaders& headers) {
44 std::string answer;
45 for (const auto& header : headers) {
46 base::StrAppend(&answer, {header.first, ": ", header.second, "\r\n"});
47 }
48 return answer;
49}
50
Yutaka Hirano2f65eec2018-05-23 01:58:2251HttpRequestHeaders WebSocketExtraHeadersToHttpRequestHeaders(
52 const WebSocketExtraHeaders& headers) {
53 HttpRequestHeaders headers_to_return;
54 for (const auto& header : headers)
55 headers_to_return.SetHeader(header.first, header.second);
56 return headers_to_return;
57}
58
alladacef397d2016-06-29 17:52:2359std::string WebSocketStandardRequest(
60 const std::string& path,
61 const std::string& host,
62 const url::Origin& origin,
63 const std::string& send_additional_request_headers,
64 const std::string& extra_headers) {
yhirano01a5d662015-02-12 04:33:0665 return WebSocketStandardRequestWithCookies(path, host, origin, std::string(),
alladacef397d2016-06-29 17:52:2366 send_additional_request_headers,
yhirano01a5d662015-02-12 04:33:0667 extra_headers);
68}
69
70std::string WebSocketStandardRequestWithCookies(
71 const std::string& path,
72 const std::string& host,
mkwst4997ce82015-07-25 12:00:0573 const url::Origin& origin,
yhirano01a5d662015-02-12 04:33:0674 const std::string& cookies,
alladacef397d2016-06-29 17:52:2375 const std::string& send_additional_request_headers,
yhirano01a5d662015-02-12 04:33:0676 const std::string& extra_headers) {
[email protected]a31ecc02013-12-05 08:30:5577 // Unrelated changes in net/http may change the order and default-values of
78 // HTTP headers, causing WebSocket tests to fail. It is safe to update this
alladacef397d2016-06-29 17:52:2379 // in that case.
80 HttpRequestHeaders headers;
81 std::stringstream request_headers;
82
83 request_headers << base::StringPrintf("GET %s HTTP/1.1\r\n", path.c_str());
84 headers.SetHeader("Host", host);
85 headers.SetHeader("Connection", "Upgrade");
86 headers.SetHeader("Pragma", "no-cache");
87 headers.SetHeader("Cache-Control", "no-cache");
Yutaka Hirano2f65eec2018-05-23 01:58:2288 headers.AddHeadersFromString(send_additional_request_headers);
alladacef397d2016-06-29 17:52:2389 headers.SetHeader("Upgrade", "websocket");
90 headers.SetHeader("Origin", origin.Serialize());
91 headers.SetHeader("Sec-WebSocket-Version", "13");
Yutaka Hirano2f65eec2018-05-23 01:58:2292 if (!headers.HasHeader("User-Agent"))
93 headers.SetHeader("User-Agent", "");
alladacef397d2016-06-29 17:52:2394 headers.SetHeader("Accept-Encoding", "gzip, deflate");
95 headers.SetHeader("Accept-Language", "en-us,fr");
96 headers.AddHeadersFromString(cookies);
97 headers.SetHeader("Sec-WebSocket-Key", "dGhlIHNhbXBsZSBub25jZQ==");
98 headers.SetHeader("Sec-WebSocket-Extensions",
99 "permessage-deflate; client_max_window_bits");
100 headers.AddHeadersFromString(extra_headers);
101
102 request_headers << headers.ToString();
103 return request_headers.str();
[email protected]a31ecc02013-12-05 08:30:55104}
105
106std::string WebSocketStandardResponse(const std::string& extra_headers) {
107 return base::StringPrintf(
108 "HTTP/1.1 101 Switching Protocols\r\n"
109 "Upgrade: websocket\r\n"
110 "Connection: Upgrade\r\n"
111 "Sec-WebSocket-Accept: s3pPLMBiTxaQ9kYGzzhZRbK+xOo=\r\n"
112 "%s\r\n",
113 extra_headers.c_str());
114}
115
Ryan Hamilton0239aac2018-05-19 00:03:13116spdy::SpdyHeaderBlock WebSocketHttp2Request(
Bence Béky46bfbc12018-02-22 19:28:20117 const std::string& path,
118 const std::string& authority,
119 const std::string& origin,
120 const WebSocketExtraHeaders& extra_headers) {
Ryan Hamilton0239aac2018-05-19 00:03:13121 spdy::SpdyHeaderBlock request_headers;
122 request_headers[spdy::kHttp2MethodHeader] = "CONNECT";
123 request_headers[spdy::kHttp2AuthorityHeader] = authority;
124 request_headers[spdy::kHttp2SchemeHeader] = "https";
125 request_headers[spdy::kHttp2PathHeader] = path;
126 request_headers[spdy::kHttp2ProtocolHeader] = "websocket";
Bence Béky46bfbc12018-02-22 19:28:20127 request_headers["pragma"] = "no-cache";
128 request_headers["cache-control"] = "no-cache";
129 request_headers["origin"] = origin;
130 request_headers["sec-websocket-version"] = "13";
131 request_headers["user-agent"] = "";
132 request_headers["accept-encoding"] = "gzip, deflate";
133 request_headers["accept-language"] = "en-us,fr";
134 request_headers["sec-websocket-extensions"] =
135 "permessage-deflate; client_max_window_bits";
136 for (const auto& header : extra_headers) {
137 request_headers[base::ToLowerASCII(header.first)] = header.second;
138 }
139 return request_headers;
140}
141
Ryan Hamilton0239aac2018-05-19 00:03:13142spdy::SpdyHeaderBlock WebSocketHttp2Response(
Bence Béky46bfbc12018-02-22 19:28:20143 const WebSocketExtraHeaders& extra_headers) {
Ryan Hamilton0239aac2018-05-19 00:03:13144 spdy::SpdyHeaderBlock response_headers;
145 response_headers[spdy::kHttp2StatusHeader] = "200";
Bence Béky46bfbc12018-02-22 19:28:20146 for (const auto& header : extra_headers) {
147 response_headers[base::ToLowerASCII(header.first)] = header.second;
148 }
149 return response_headers;
150}
151
mmenked0d201a2015-06-08 12:00:12152struct WebSocketMockClientSocketFactoryMaker::Detail {
[email protected]a31ecc02013-12-05 08:30:55153 std::string expect_written;
154 std::string return_to_read;
[email protected]94831522014-02-06 12:05:18155 std::vector<MockRead> reads;
[email protected]a31ecc02013-12-05 08:30:55156 MockWrite write;
danakj9c5cab52016-04-16 00:54:33157 std::vector<std::unique_ptr<SequencedSocketData>> socket_data_vector;
158 std::vector<std::unique_ptr<SSLSocketDataProvider>> ssl_socket_data_vector;
mmenked0d201a2015-06-08 12:00:12159 MockClientSocketFactory factory;
[email protected]a31ecc02013-12-05 08:30:55160};
161
mmenked0d201a2015-06-08 12:00:12162WebSocketMockClientSocketFactoryMaker::WebSocketMockClientSocketFactoryMaker()
Bence Béky65623972018-03-05 15:31:56163 : detail_(std::make_unique<Detail>()) {}
[email protected]a31ecc02013-12-05 08:30:55164
mmenked0d201a2015-06-08 12:00:12165WebSocketMockClientSocketFactoryMaker::
Chris Watkins28c2fdd2017-11-30 06:06:52166 ~WebSocketMockClientSocketFactoryMaker() = default;
[email protected]a31ecc02013-12-05 08:30:55167
mmenked0d201a2015-06-08 12:00:12168MockClientSocketFactory* WebSocketMockClientSocketFactoryMaker::factory() {
[email protected]a31ecc02013-12-05 08:30:55169 return &detail_->factory;
170}
171
mmenked0d201a2015-06-08 12:00:12172void WebSocketMockClientSocketFactoryMaker::SetExpectations(
[email protected]a31ecc02013-12-05 08:30:55173 const std::string& expect_written,
174 const std::string& return_to_read) {
[email protected]94831522014-02-06 12:05:18175 const size_t kHttpStreamParserBufferSize = 4096;
[email protected]a31ecc02013-12-05 08:30:55176 // We need to extend the lifetime of these strings.
177 detail_->expect_written = expect_written;
178 detail_->return_to_read = return_to_read;
[email protected]94831522014-02-06 12:05:18179 int sequence = 0;
[email protected]0be93922014-01-29 00:42:45180 detail_->write = MockWrite(SYNCHRONOUS,
181 detail_->expect_written.data(),
182 detail_->expect_written.size(),
[email protected]94831522014-02-06 12:05:18183 sequence++);
184 // HttpStreamParser reads 4KB at a time. We need to take this implementation
185 // detail into account if |return_to_read| is big enough.
186 for (size_t place = 0; place < detail_->return_to_read.size();
187 place += kHttpStreamParserBufferSize) {
188 detail_->reads.push_back(
189 MockRead(SYNCHRONOUS, detail_->return_to_read.data() + place,
190 std::min(detail_->return_to_read.size() - place,
191 kHttpStreamParserBufferSize),
192 sequence++));
193 }
Bence Béky65623972018-03-05 15:31:56194 auto socket_data = std::make_unique<SequencedSocketData>(
Ryan Sleevib8d7ea02018-05-07 20:01:01195 detail_->reads, base::make_span(&detail_->write, 1));
[email protected]a31ecc02013-12-05 08:30:55196 socket_data->set_connect_data(MockConnect(SYNCHRONOUS, OK));
dchengc7eeda422015-12-26 03:56:48197 AddRawExpectations(std::move(socket_data));
[email protected]a31ecc02013-12-05 08:30:55198}
199
mmenked0d201a2015-06-08 12:00:12200void WebSocketMockClientSocketFactoryMaker::AddRawExpectations(
danakj9c5cab52016-04-16 00:54:33201 std::unique_ptr<SequencedSocketData> socket_data) {
[email protected]a62449522014-06-05 11:11:15202 detail_->factory.AddSocketDataProvider(socket_data.get());
dchengc7eeda422015-12-26 03:56:48203 detail_->socket_data_vector.push_back(std::move(socket_data));
[email protected]a62449522014-06-05 11:11:15204}
205
mmenked0d201a2015-06-08 12:00:12206void WebSocketMockClientSocketFactoryMaker::AddSSLSocketDataProvider(
danakj9c5cab52016-04-16 00:54:33207 std::unique_ptr<SSLSocketDataProvider> ssl_socket_data) {
[email protected]a62449522014-06-05 11:11:15208 detail_->factory.AddSSLSocketDataProvider(ssl_socket_data.get());
dchengc7eeda422015-12-26 03:56:48209 detail_->ssl_socket_data_vector.push_back(std::move(ssl_socket_data));
[email protected]a31ecc02013-12-05 08:30:55210}
211
212WebSocketTestURLRequestContextHost::WebSocketTestURLRequestContextHost()
[email protected]654866142014-06-24 22:53:31213 : url_request_context_(true), url_request_context_initialized_(false) {
[email protected]a31ecc02013-12-05 08:30:55214 url_request_context_.set_client_socket_factory(maker_.factory());
Bence Béky70dd7662018-03-02 17:13:24215 auto params = std::make_unique<HttpNetworkSession::Params>();
216 params->enable_spdy_ping_based_connection_checking = false;
217 params->enable_quic = false;
218 params->enable_websocket_over_http2 = true;
219 params->disable_idle_sockets_close_on_memory_pressure = false;
220 url_request_context_.set_http_network_session_params(std::move(params));
[email protected]a31ecc02013-12-05 08:30:55221}
222
Chris Watkins28c2fdd2017-11-30 06:06:52223WebSocketTestURLRequestContextHost::~WebSocketTestURLRequestContextHost() =
224 default;
[email protected]a31ecc02013-12-05 08:30:55225
[email protected]a62449522014-06-05 11:11:15226void WebSocketTestURLRequestContextHost::AddRawExpectations(
danakj9c5cab52016-04-16 00:54:33227 std::unique_ptr<SequencedSocketData> socket_data) {
dchengc7eeda422015-12-26 03:56:48228 maker_.AddRawExpectations(std::move(socket_data));
[email protected]a62449522014-06-05 11:11:15229}
230
231void WebSocketTestURLRequestContextHost::AddSSLSocketDataProvider(
danakj9c5cab52016-04-16 00:54:33232 std::unique_ptr<SSLSocketDataProvider> ssl_socket_data) {
dchengc7eeda422015-12-26 03:56:48233 maker_.AddSSLSocketDataProvider(std::move(ssl_socket_data));
[email protected]a31ecc02013-12-05 08:30:55234}
235
ricea38fc268c2015-02-09 02:41:29236void WebSocketTestURLRequestContextHost::SetProxyConfig(
237 const std::string& proxy_rules) {
238 DCHECK(!url_request_context_initialized_);
Ramin Halavatica8d5252018-03-12 05:33:49239 proxy_resolution_service_ = ProxyResolutionService::CreateFixed(
240 proxy_rules, TRAFFIC_ANNOTATION_FOR_TESTS);
Lily Houghton8c2f97d2018-01-22 05:06:59241 url_request_context_.set_proxy_resolution_service(
242 proxy_resolution_service_.get());
ricea38fc268c2015-02-09 02:41:29243}
244
[email protected]a31ecc02013-12-05 08:30:55245TestURLRequestContext*
246WebSocketTestURLRequestContextHost::GetURLRequestContext() {
[email protected]654866142014-06-24 22:53:31247 if (!url_request_context_initialized_) {
248 url_request_context_.Init();
249 // A Network Delegate is required to make the URLRequest::Delegate work.
250 url_request_context_.set_network_delegate(&network_delegate_);
251 url_request_context_initialized_ = true;
252 }
[email protected]a31ecc02013-12-05 08:30:55253 return &url_request_context_;
254}
255
Adam Rice6f75c0f2018-06-04 08:00:05256void TestWebSocketStreamRequestAPI::OnBasicHandshakeStreamCreated(
257 WebSocketBasicHandshakeStream* handshake_stream) {
258 handshake_stream->SetWebSocketKeyForTesting("dGhlIHNhbXBsZSBub25jZQ==");
Bence Békydca6bd92018-01-30 13:43:06259}
260
Adam Rice6f75c0f2018-06-04 08:00:05261void TestWebSocketStreamRequestAPI::OnHttp2HandshakeStreamCreated(
262 WebSocketHttp2HandshakeStream* handshake_stream) {}
263
[email protected]ce9f7ffd2013-10-11 06:04:11264} // namespace net