mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-11-24 06:16:30 +03:00
189 lines
5.7 KiB
C++
189 lines
5.7 KiB
C++
// 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/test/spawned_test_server/remote_test_server.h"
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <limits>
|
|
#include <vector>
|
|
|
|
#include "base/base_paths.h"
|
|
#include "base/files/file_path.h"
|
|
#include "base/files/file_util.h"
|
|
#include "base/json/json_writer.h"
|
|
#include "base/logging.h"
|
|
#include "base/message_loop/message_loop.h"
|
|
#include "base/path_service.h"
|
|
#include "base/strings/string_number_conversions.h"
|
|
#include "base/strings/string_split.h"
|
|
#include "base/strings/stringprintf.h"
|
|
#include "base/values.h"
|
|
#include "net/base/host_port_pair.h"
|
|
#include "net/base/ip_endpoint.h"
|
|
#include "net/base/net_errors.h"
|
|
#include "net/test/spawned_test_server/remote_test_server_proxy.h"
|
|
#include "net/test/spawned_test_server/remote_test_server_spawner_request.h"
|
|
#include "url/gurl.h"
|
|
|
|
namespace net {
|
|
|
|
namespace {
|
|
|
|
// Please keep in sync with dictionary SERVER_TYPES in testserver.py
|
|
std::string GetServerTypeString(BaseTestServer::Type type) {
|
|
switch (type) {
|
|
case BaseTestServer::TYPE_FTP:
|
|
return "ftp";
|
|
case BaseTestServer::TYPE_HTTP:
|
|
case BaseTestServer::TYPE_HTTPS:
|
|
return "http";
|
|
case BaseTestServer::TYPE_WS:
|
|
case BaseTestServer::TYPE_WSS:
|
|
return "ws";
|
|
case BaseTestServer::TYPE_TCP_ECHO:
|
|
return "tcpecho";
|
|
case BaseTestServer::TYPE_UDP_ECHO:
|
|
return "udpecho";
|
|
default:
|
|
NOTREACHED();
|
|
}
|
|
return std::string();
|
|
}
|
|
|
|
} // namespace
|
|
|
|
RemoteTestServer::RemoteTestServer(Type type,
|
|
const base::FilePath& document_root)
|
|
: BaseTestServer(type), io_thread_("RemoteTestServer IO Thread") {
|
|
if (!Init(document_root))
|
|
NOTREACHED();
|
|
}
|
|
|
|
RemoteTestServer::RemoteTestServer(Type type,
|
|
const SSLOptions& ssl_options,
|
|
const base::FilePath& document_root)
|
|
: BaseTestServer(type, ssl_options),
|
|
io_thread_("RemoteTestServer IO Thread") {
|
|
if (!Init(document_root))
|
|
NOTREACHED();
|
|
}
|
|
|
|
RemoteTestServer::~RemoteTestServer() {
|
|
Stop();
|
|
}
|
|
|
|
bool RemoteTestServer::StartInBackground() {
|
|
DCHECK(!started());
|
|
DCHECK(!start_request_);
|
|
|
|
base::DictionaryValue arguments_dict;
|
|
if (!GenerateArguments(&arguments_dict))
|
|
return false;
|
|
|
|
arguments_dict.Set("on-remote-server", std::make_unique<base::Value>());
|
|
|
|
// Append the 'server-type' argument which is used by spawner server to
|
|
// pass right server type to Python test server.
|
|
arguments_dict.SetString("server-type", GetServerTypeString(type()));
|
|
|
|
// Generate JSON-formatted argument string.
|
|
std::string arguments_string;
|
|
base::JSONWriter::Write(arguments_dict, &arguments_string);
|
|
if (arguments_string.empty())
|
|
return false;
|
|
|
|
start_request_ = std::make_unique<RemoteTestServerSpawnerRequest>(
|
|
io_thread_.task_runner(), config_.GetSpawnerUrl("start"),
|
|
arguments_string);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool RemoteTestServer::BlockUntilStarted() {
|
|
DCHECK(start_request_);
|
|
|
|
std::string server_data;
|
|
bool request_result = start_request_->WaitForCompletion(&server_data);
|
|
start_request_.reset();
|
|
if (!request_result)
|
|
return false;
|
|
|
|
// Parse server_data.
|
|
if (server_data.empty() ||
|
|
!SetAndParseServerData(server_data, &remote_port_)) {
|
|
LOG(ERROR) << "Could not parse server_data: " << server_data;
|
|
return false;
|
|
}
|
|
|
|
// If the server is not on localhost then start a proxy on localhost to
|
|
// forward connections to the server.
|
|
if (config_.address() != IPAddress::IPv4Localhost()) {
|
|
test_server_proxy_ = std::make_unique<RemoteTestServerProxy>(
|
|
IPEndPoint(config_.address(), remote_port_), io_thread_.task_runner());
|
|
SetPort(test_server_proxy_->local_port());
|
|
} else {
|
|
SetPort(remote_port_);
|
|
}
|
|
|
|
return SetupWhenServerStarted();
|
|
}
|
|
|
|
bool RemoteTestServer::Stop() {
|
|
DCHECK(!start_request_);
|
|
|
|
if (remote_port_) {
|
|
std::unique_ptr<RemoteTestServerSpawnerRequest> kill_request =
|
|
std::make_unique<RemoteTestServerSpawnerRequest>(
|
|
io_thread_.task_runner(),
|
|
config_.GetSpawnerUrl(
|
|
base::StringPrintf("kill?port=%d", remote_port_)),
|
|
std::string());
|
|
|
|
if (!kill_request->WaitForCompletion(nullptr))
|
|
LOG(ERROR) << "Failed stopping RemoteTestServer";
|
|
|
|
remote_port_ = 0;
|
|
}
|
|
|
|
CleanUpWhenStoppingServer();
|
|
|
|
return true;
|
|
}
|
|
|
|
// On Android, the document root in the device is not the same as the document
|
|
// root in the host machine where the test server is launched. So prepend
|
|
// DIR_SOURCE_ROOT here to get the actual path of document root on the Android
|
|
// device.
|
|
base::FilePath RemoteTestServer::GetDocumentRoot() const {
|
|
base::FilePath src_dir;
|
|
PathService::Get(base::DIR_SOURCE_ROOT, &src_dir);
|
|
return src_dir.Append(document_root());
|
|
}
|
|
|
|
bool RemoteTestServer::Init(const base::FilePath& document_root) {
|
|
if (document_root.IsAbsolute())
|
|
return false;
|
|
|
|
config_ = RemoteTestServerConfig::Load();
|
|
|
|
bool thread_started = io_thread_.StartWithOptions(
|
|
base::Thread::Options(base::MessageLoop::TYPE_IO, 0));
|
|
CHECK(thread_started);
|
|
|
|
// Unlike LocalTestServer, RemoteTestServer passes relative paths to the test
|
|
// server. The test server fails on empty strings in some configurations.
|
|
base::FilePath fixed_root = document_root;
|
|
if (fixed_root.empty())
|
|
fixed_root = base::FilePath(base::FilePath::kCurrentDirectory);
|
|
SetResourcePath(fixed_root, base::FilePath()
|
|
.AppendASCII("net")
|
|
.AppendASCII("data")
|
|
.AppendASCII("ssl")
|
|
.AppendASCII("certificates"));
|
|
return true;
|
|
}
|
|
|
|
} // namespace net
|