blob: 77072e65a2d1a16f2ee9024714e40b55c269fab4 [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"
Lily Houghton582d4622018-01-22 22:43:4014#include "net/proxy_resolution/proxy_service.h"
[email protected]a31ecc02013-12-05 08:30:5515#include "net/socket/socket_test_util.h"
Bence Béky46bfbc12018-02-22 19:28:2016#include "net/spdy/core/spdy_protocol.h"
Bence Béky8d1c6052018-02-07 12:48:1517#include "net/websockets/websocket_basic_handshake_stream.h"
mkwst4997ce82015-07-25 12:00:0518#include "url/origin.h"
[email protected]ce9f7ffd2013-10-11 06:04:1119
20namespace net {
21
22namespace {
tfarinaea94afc232015-10-20 04:23:3623
tfarina8a2c66c22015-10-13 19:14:4924const uint64_t kA = (static_cast<uint64_t>(0x5851f42d) << 32) +
25 static_cast<uint64_t>(0x4c957f2d);
26const uint64_t kC = 12345;
27const uint64_t kM = static_cast<uint64_t>(1) << 48;
[email protected]ce9f7ffd2013-10-11 06:04:1128
29} // namespace
30
tfarina8a2c66c22015-10-13 19:14:4931LinearCongruentialGenerator::LinearCongruentialGenerator(uint32_t seed)
[email protected]ce9f7ffd2013-10-11 06:04:1132 : current_(seed) {}
33
tfarina8a2c66c22015-10-13 19:14:4934uint32_t LinearCongruentialGenerator::Generate() {
35 uint64_t result = current_;
[email protected]ce9f7ffd2013-10-11 06:04:1136 current_ = (current_ * kA + kC) % kM;
tfarina8a2c66c22015-10-13 19:14:4937 return static_cast<uint32_t>(result >> 16);
[email protected]ce9f7ffd2013-10-11 06:04:1138}
39
Bence Béky3a0c48532018-03-02 13:38:5140std::string WebSocketExtraHeadersToString(
41 const WebSocketExtraHeaders& headers) {
42 std::string answer;
43 for (const auto& header : headers) {
44 base::StrAppend(&answer, {header.first, ": ", header.second, "\r\n"});
45 }
46 return answer;
47}
48
alladacef397d2016-06-29 17:52:2349std::string WebSocketStandardRequest(
50 const std::string& path,
51 const std::string& host,
52 const url::Origin& origin,
53 const std::string& send_additional_request_headers,
54 const std::string& extra_headers) {
yhirano01a5d662015-02-12 04:33:0655 return WebSocketStandardRequestWithCookies(path, host, origin, std::string(),
alladacef397d2016-06-29 17:52:2356 send_additional_request_headers,
yhirano01a5d662015-02-12 04:33:0657 extra_headers);
58}
59
60std::string WebSocketStandardRequestWithCookies(
61 const std::string& path,
62 const std::string& host,
mkwst4997ce82015-07-25 12:00:0563 const url::Origin& origin,
yhirano01a5d662015-02-12 04:33:0664 const std::string& cookies,
alladacef397d2016-06-29 17:52:2365 const std::string& send_additional_request_headers,
yhirano01a5d662015-02-12 04:33:0666 const std::string& extra_headers) {
[email protected]a31ecc02013-12-05 08:30:5567 // Unrelated changes in net/http may change the order and default-values of
68 // HTTP headers, causing WebSocket tests to fail. It is safe to update this
alladacef397d2016-06-29 17:52:2369 // in that case.
70 HttpRequestHeaders headers;
71 std::stringstream request_headers;
72
73 request_headers << base::StringPrintf("GET %s HTTP/1.1\r\n", path.c_str());
74 headers.SetHeader("Host", host);
75 headers.SetHeader("Connection", "Upgrade");
76 headers.SetHeader("Pragma", "no-cache");
77 headers.SetHeader("Cache-Control", "no-cache");
78 headers.SetHeader("Upgrade", "websocket");
79 headers.SetHeader("Origin", origin.Serialize());
80 headers.SetHeader("Sec-WebSocket-Version", "13");
81 headers.SetHeader("User-Agent", "");
82 headers.AddHeadersFromString(send_additional_request_headers);
83 headers.SetHeader("Accept-Encoding", "gzip, deflate");
84 headers.SetHeader("Accept-Language", "en-us,fr");
85 headers.AddHeadersFromString(cookies);
86 headers.SetHeader("Sec-WebSocket-Key", "dGhlIHNhbXBsZSBub25jZQ==");
87 headers.SetHeader("Sec-WebSocket-Extensions",
88 "permessage-deflate; client_max_window_bits");
89 headers.AddHeadersFromString(extra_headers);
90
91 request_headers << headers.ToString();
92 return request_headers.str();
[email protected]a31ecc02013-12-05 08:30:5593}
94
95std::string WebSocketStandardResponse(const std::string& extra_headers) {
96 return base::StringPrintf(
97 "HTTP/1.1 101 Switching Protocols\r\n"
98 "Upgrade: websocket\r\n"
99 "Connection: Upgrade\r\n"
100 "Sec-WebSocket-Accept: s3pPLMBiTxaQ9kYGzzhZRbK+xOo=\r\n"
101 "%s\r\n",
102 extra_headers.c_str());
103}
104
Bence Béky46bfbc12018-02-22 19:28:20105SpdyHeaderBlock WebSocketHttp2Request(
106 const std::string& path,
107 const std::string& authority,
108 const std::string& origin,
109 const WebSocketExtraHeaders& extra_headers) {
110 SpdyHeaderBlock request_headers;
111 request_headers[kHttp2MethodHeader] = "CONNECT";
112 request_headers[kHttp2AuthorityHeader] = authority;
113 request_headers[kHttp2SchemeHeader] = "https";
114 request_headers[kHttp2PathHeader] = path;
115 request_headers[kHttp2ProtocolHeader] = "websocket";
116 request_headers["pragma"] = "no-cache";
117 request_headers["cache-control"] = "no-cache";
118 request_headers["origin"] = origin;
119 request_headers["sec-websocket-version"] = "13";
120 request_headers["user-agent"] = "";
121 request_headers["accept-encoding"] = "gzip, deflate";
122 request_headers["accept-language"] = "en-us,fr";
123 request_headers["sec-websocket-extensions"] =
124 "permessage-deflate; client_max_window_bits";
125 for (const auto& header : extra_headers) {
126 request_headers[base::ToLowerASCII(header.first)] = header.second;
127 }
128 return request_headers;
129}
130
131SpdyHeaderBlock WebSocketHttp2Response(
132 const WebSocketExtraHeaders& extra_headers) {
133 SpdyHeaderBlock response_headers;
134 response_headers[kHttp2StatusHeader] = "200";
135 for (const auto& header : extra_headers) {
136 response_headers[base::ToLowerASCII(header.first)] = header.second;
137 }
138 return response_headers;
139}
140
mmenked0d201a2015-06-08 12:00:12141struct WebSocketMockClientSocketFactoryMaker::Detail {
[email protected]a31ecc02013-12-05 08:30:55142 std::string expect_written;
143 std::string return_to_read;
[email protected]94831522014-02-06 12:05:18144 std::vector<MockRead> reads;
[email protected]a31ecc02013-12-05 08:30:55145 MockWrite write;
danakj9c5cab52016-04-16 00:54:33146 std::vector<std::unique_ptr<SequencedSocketData>> socket_data_vector;
147 std::vector<std::unique_ptr<SSLSocketDataProvider>> ssl_socket_data_vector;
mmenked0d201a2015-06-08 12:00:12148 MockClientSocketFactory factory;
[email protected]a31ecc02013-12-05 08:30:55149};
150
mmenked0d201a2015-06-08 12:00:12151WebSocketMockClientSocketFactoryMaker::WebSocketMockClientSocketFactoryMaker()
152 : detail_(new Detail) {
153}
[email protected]a31ecc02013-12-05 08:30:55154
mmenked0d201a2015-06-08 12:00:12155WebSocketMockClientSocketFactoryMaker::
Chris Watkins28c2fdd2017-11-30 06:06:52156 ~WebSocketMockClientSocketFactoryMaker() = default;
[email protected]a31ecc02013-12-05 08:30:55157
mmenked0d201a2015-06-08 12:00:12158MockClientSocketFactory* WebSocketMockClientSocketFactoryMaker::factory() {
[email protected]a31ecc02013-12-05 08:30:55159 return &detail_->factory;
160}
161
mmenked0d201a2015-06-08 12:00:12162void WebSocketMockClientSocketFactoryMaker::SetExpectations(
[email protected]a31ecc02013-12-05 08:30:55163 const std::string& expect_written,
164 const std::string& return_to_read) {
[email protected]94831522014-02-06 12:05:18165 const size_t kHttpStreamParserBufferSize = 4096;
[email protected]a31ecc02013-12-05 08:30:55166 // We need to extend the lifetime of these strings.
167 detail_->expect_written = expect_written;
168 detail_->return_to_read = return_to_read;
[email protected]94831522014-02-06 12:05:18169 int sequence = 0;
[email protected]0be93922014-01-29 00:42:45170 detail_->write = MockWrite(SYNCHRONOUS,
171 detail_->expect_written.data(),
172 detail_->expect_written.size(),
[email protected]94831522014-02-06 12:05:18173 sequence++);
174 // HttpStreamParser reads 4KB at a time. We need to take this implementation
175 // detail into account if |return_to_read| is big enough.
176 for (size_t place = 0; place < detail_->return_to_read.size();
177 place += kHttpStreamParserBufferSize) {
178 detail_->reads.push_back(
179 MockRead(SYNCHRONOUS, detail_->return_to_read.data() + place,
180 std::min(detail_->return_to_read.size() - place,
181 kHttpStreamParserBufferSize),
182 sequence++));
183 }
danakj9c5cab52016-04-16 00:54:33184 std::unique_ptr<SequencedSocketData> socket_data(new SequencedSocketData(
davidben5f8b6bc2015-11-25 03:19:54185 detail_->reads.data(), detail_->reads.size(), &detail_->write, 1));
[email protected]a31ecc02013-12-05 08:30:55186 socket_data->set_connect_data(MockConnect(SYNCHRONOUS, OK));
dchengc7eeda422015-12-26 03:56:48187 AddRawExpectations(std::move(socket_data));
[email protected]a31ecc02013-12-05 08:30:55188}
189
mmenked0d201a2015-06-08 12:00:12190void WebSocketMockClientSocketFactoryMaker::AddRawExpectations(
danakj9c5cab52016-04-16 00:54:33191 std::unique_ptr<SequencedSocketData> socket_data) {
[email protected]a62449522014-06-05 11:11:15192 detail_->factory.AddSocketDataProvider(socket_data.get());
dchengc7eeda422015-12-26 03:56:48193 detail_->socket_data_vector.push_back(std::move(socket_data));
[email protected]a62449522014-06-05 11:11:15194}
195
mmenked0d201a2015-06-08 12:00:12196void WebSocketMockClientSocketFactoryMaker::AddSSLSocketDataProvider(
danakj9c5cab52016-04-16 00:54:33197 std::unique_ptr<SSLSocketDataProvider> ssl_socket_data) {
[email protected]a62449522014-06-05 11:11:15198 detail_->factory.AddSSLSocketDataProvider(ssl_socket_data.get());
dchengc7eeda422015-12-26 03:56:48199 detail_->ssl_socket_data_vector.push_back(std::move(ssl_socket_data));
[email protected]a31ecc02013-12-05 08:30:55200}
201
202WebSocketTestURLRequestContextHost::WebSocketTestURLRequestContextHost()
[email protected]654866142014-06-24 22:53:31203 : url_request_context_(true), url_request_context_initialized_(false) {
[email protected]a31ecc02013-12-05 08:30:55204 url_request_context_.set_client_socket_factory(maker_.factory());
205}
206
Chris Watkins28c2fdd2017-11-30 06:06:52207WebSocketTestURLRequestContextHost::~WebSocketTestURLRequestContextHost() =
208 default;
[email protected]a31ecc02013-12-05 08:30:55209
[email protected]a62449522014-06-05 11:11:15210void WebSocketTestURLRequestContextHost::AddRawExpectations(
danakj9c5cab52016-04-16 00:54:33211 std::unique_ptr<SequencedSocketData> socket_data) {
dchengc7eeda422015-12-26 03:56:48212 maker_.AddRawExpectations(std::move(socket_data));
[email protected]a62449522014-06-05 11:11:15213}
214
215void WebSocketTestURLRequestContextHost::AddSSLSocketDataProvider(
danakj9c5cab52016-04-16 00:54:33216 std::unique_ptr<SSLSocketDataProvider> ssl_socket_data) {
dchengc7eeda422015-12-26 03:56:48217 maker_.AddSSLSocketDataProvider(std::move(ssl_socket_data));
[email protected]a31ecc02013-12-05 08:30:55218}
219
ricea38fc268c2015-02-09 02:41:29220void WebSocketTestURLRequestContextHost::SetProxyConfig(
221 const std::string& proxy_rules) {
222 DCHECK(!url_request_context_initialized_);
Lily Houghton8c2f97d2018-01-22 05:06:59223 proxy_resolution_service_ = ProxyResolutionService::CreateFixed(proxy_rules);
224 url_request_context_.set_proxy_resolution_service(
225 proxy_resolution_service_.get());
ricea38fc268c2015-02-09 02:41:29226}
227
[email protected]a31ecc02013-12-05 08:30:55228TestURLRequestContext*
229WebSocketTestURLRequestContextHost::GetURLRequestContext() {
[email protected]654866142014-06-24 22:53:31230 if (!url_request_context_initialized_) {
231 url_request_context_.Init();
232 // A Network Delegate is required to make the URLRequest::Delegate work.
233 url_request_context_.set_network_delegate(&network_delegate_);
234 url_request_context_initialized_ = true;
235 }
[email protected]a31ecc02013-12-05 08:30:55236 return &url_request_context_;
237}
238
Bence Béky8d1c6052018-02-07 12:48:15239void TestWebSocketHandshakeStreamCreateHelper::OnBasicStreamCreated(
240 WebSocketBasicHandshakeStream* stream) {
241 stream->SetWebSocketKeyForTesting("dGhlIHNhbXBsZSBub25jZQ==");
Bence Békydca6bd92018-01-30 13:43:06242}
243
[email protected]ce9f7ffd2013-10-11 06:04:11244} // namespace net