| // Copyright 2013 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/websockets/websocket_test_util.h" |
| |
| #include <algorithm> |
| #include <vector> |
| |
| #include "base/basictypes.h" |
| #include "base/memory/scoped_vector.h" |
| #include "base/stl_util.h" |
| #include "base/strings/stringprintf.h" |
| #include "net/socket/socket_test_util.h" |
| |
| namespace net { |
| |
| namespace { |
| const uint64 kA = |
| (static_cast<uint64>(0x5851f42d) << 32) + static_cast<uint64>(0x4c957f2d); |
| const uint64 kC = 12345; |
| const uint64 kM = static_cast<uint64>(1) << 48; |
| |
| } // namespace |
| |
| LinearCongruentialGenerator::LinearCongruentialGenerator(uint32 seed) |
| : current_(seed) {} |
| |
| uint32 LinearCongruentialGenerator::Generate() { |
| uint64 result = current_; |
| current_ = (current_ * kA + kC) % kM; |
| return static_cast<uint32>(result >> 16); |
| } |
| |
| std::string WebSocketStandardRequest(const std::string& path, |
| const std::string& origin, |
| const std::string& extra_headers) { |
| // Unrelated changes in net/http may change the order and default-values of |
| // HTTP headers, causing WebSocket tests to fail. It is safe to update this |
| // string in that case. |
| return base::StringPrintf( |
| "GET %s HTTP/1.1\r\n" |
| "Host: localhost\r\n" |
| "Connection: Upgrade\r\n" |
| "Pragma: no-cache\r\n" |
| "Cache-Control: no-cache\r\n" |
| "Upgrade: websocket\r\n" |
| "Origin: %s\r\n" |
| "Sec-WebSocket-Version: 13\r\n" |
| "User-Agent:\r\n" |
| "Accept-Encoding: gzip,deflate\r\n" |
| "Accept-Language: en-us,fr\r\n" |
| "Sec-WebSocket-Key: dGhlIHNhbXBsZSBub25jZQ==\r\n" |
| "Sec-WebSocket-Extensions: permessage-deflate; client_max_window_bits\r\n" |
| "%s\r\n", |
| path.c_str(), |
| origin.c_str(), |
| extra_headers.c_str()); |
| } |
| |
| std::string WebSocketStandardResponse(const std::string& extra_headers) { |
| return base::StringPrintf( |
| "HTTP/1.1 101 Switching Protocols\r\n" |
| "Upgrade: websocket\r\n" |
| "Connection: Upgrade\r\n" |
| "Sec-WebSocket-Accept: s3pPLMBiTxaQ9kYGzzhZRbK+xOo=\r\n" |
| "%s\r\n", |
| extra_headers.c_str()); |
| } |
| |
| struct WebSocketDeterministicMockClientSocketFactoryMaker::Detail { |
| std::string expect_written; |
| std::string return_to_read; |
| std::vector<MockRead> reads; |
| MockWrite write; |
| ScopedVector<DeterministicSocketData> socket_data_vector; |
| ScopedVector<SSLSocketDataProvider> ssl_socket_data_vector; |
| DeterministicMockClientSocketFactory factory; |
| }; |
| |
| WebSocketDeterministicMockClientSocketFactoryMaker:: |
| WebSocketDeterministicMockClientSocketFactoryMaker() |
| : detail_(new Detail) {} |
| |
| WebSocketDeterministicMockClientSocketFactoryMaker:: |
| ~WebSocketDeterministicMockClientSocketFactoryMaker() {} |
| |
| DeterministicMockClientSocketFactory* |
| WebSocketDeterministicMockClientSocketFactoryMaker::factory() { |
| return &detail_->factory; |
| } |
| |
| void WebSocketDeterministicMockClientSocketFactoryMaker::SetExpectations( |
| const std::string& expect_written, |
| const std::string& return_to_read) { |
| const size_t kHttpStreamParserBufferSize = 4096; |
| // We need to extend the lifetime of these strings. |
| detail_->expect_written = expect_written; |
| detail_->return_to_read = return_to_read; |
| int sequence = 0; |
| detail_->write = MockWrite(SYNCHRONOUS, |
| detail_->expect_written.data(), |
| detail_->expect_written.size(), |
| sequence++); |
| // HttpStreamParser reads 4KB at a time. We need to take this implementation |
| // detail into account if |return_to_read| is big enough. |
| for (size_t place = 0; place < detail_->return_to_read.size(); |
| place += kHttpStreamParserBufferSize) { |
| detail_->reads.push_back( |
| MockRead(SYNCHRONOUS, detail_->return_to_read.data() + place, |
| std::min(detail_->return_to_read.size() - place, |
| kHttpStreamParserBufferSize), |
| sequence++)); |
| } |
| scoped_ptr<DeterministicSocketData> socket_data( |
| new DeterministicSocketData(vector_as_array(&detail_->reads), |
| detail_->reads.size(), |
| &detail_->write, |
| 1)); |
| socket_data->set_connect_data(MockConnect(SYNCHRONOUS, OK)); |
| socket_data->SetStop(sequence); |
| AddRawExpectations(socket_data.Pass()); |
| } |
| |
| void WebSocketDeterministicMockClientSocketFactoryMaker::AddRawExpectations( |
| scoped_ptr<DeterministicSocketData> socket_data) { |
| detail_->factory.AddSocketDataProvider(socket_data.get()); |
| detail_->socket_data_vector.push_back(socket_data.release()); |
| } |
| |
| void |
| WebSocketDeterministicMockClientSocketFactoryMaker::AddSSLSocketDataProvider( |
| scoped_ptr<SSLSocketDataProvider> ssl_socket_data) { |
| detail_->factory.AddSSLSocketDataProvider(ssl_socket_data.get()); |
| detail_->ssl_socket_data_vector.push_back(ssl_socket_data.release()); |
| } |
| |
| WebSocketTestURLRequestContextHost::WebSocketTestURLRequestContextHost() |
| : url_request_context_(true), url_request_context_initialized_(false) { |
| url_request_context_.set_client_socket_factory(maker_.factory()); |
| } |
| |
| WebSocketTestURLRequestContextHost::~WebSocketTestURLRequestContextHost() {} |
| |
| void WebSocketTestURLRequestContextHost::AddRawExpectations( |
| scoped_ptr<DeterministicSocketData> socket_data) { |
| maker_.AddRawExpectations(socket_data.Pass()); |
| } |
| |
| void WebSocketTestURLRequestContextHost::AddSSLSocketDataProvider( |
| scoped_ptr<SSLSocketDataProvider> ssl_socket_data) { |
| maker_.AddSSLSocketDataProvider(ssl_socket_data.Pass()); |
| } |
| |
| TestURLRequestContext* |
| WebSocketTestURLRequestContextHost::GetURLRequestContext() { |
| if (!url_request_context_initialized_) { |
| url_request_context_.Init(); |
| // A Network Delegate is required to make the URLRequest::Delegate work. |
| url_request_context_.set_network_delegate(&network_delegate_); |
| url_request_context_initialized_ = true; |
| } |
| return &url_request_context_; |
| } |
| |
| } // namespace net |