| // Copyright (c) 2012 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/local_test_server.h" |
| |
| #include "base/command_line.h" |
| #include "base/json/json_reader.h" |
| #include "base/logging.h" |
| #include "base/path_service.h" |
| #include "base/process_util.h" |
| #include "base/string_number_conversions.h" |
| #include "base/values.h" |
| #include "googleurl/src/gurl.h" |
| #include "net/base/host_port_pair.h" |
| #include "net/base/net_errors.h" |
| #include "net/test/python_utils.h" |
| |
| namespace net { |
| |
| namespace { |
| |
| bool AppendArgumentFromJSONValue(const std::string& key, |
| const base::Value& value_node, |
| CommandLine* command_line) { |
| std::string argument_name = "--" + key; |
| switch (value_node.GetType()) { |
| case base::Value::TYPE_NULL: |
| command_line->AppendArg(argument_name); |
| break; |
| case base::Value::TYPE_INTEGER: { |
| int value; |
| bool result = value_node.GetAsInteger(&value); |
| DCHECK(result); |
| command_line->AppendArg(argument_name + "=" + base::IntToString(value)); |
| break; |
| } |
| case Value::TYPE_STRING: { |
| std::string value; |
| bool result = value_node.GetAsString(&value); |
| if (!result || value.empty()) |
| return false; |
| command_line->AppendArg(argument_name + "=" + value); |
| break; |
| } |
| case base::Value::TYPE_BOOLEAN: |
| case base::Value::TYPE_DOUBLE: |
| case base::Value::TYPE_LIST: |
| case base::Value::TYPE_DICTIONARY: |
| case base::Value::TYPE_BINARY: |
| default: |
| NOTREACHED() << "improper json type"; |
| return false; |
| } |
| return true; |
| } |
| |
| } // namespace |
| |
| LocalTestServer::LocalTestServer(Type type, |
| const std::string& host, |
| const FilePath& document_root) |
| : BaseTestServer(type, host) { |
| if (!Init(document_root)) |
| NOTREACHED(); |
| } |
| |
| LocalTestServer::LocalTestServer(Type type, |
| const SSLOptions& ssl_options, |
| const FilePath& document_root) |
| : BaseTestServer(type, ssl_options) { |
| if (!Init(document_root)) |
| NOTREACHED(); |
| } |
| |
| LocalTestServer::~LocalTestServer() { |
| Stop(); |
| } |
| |
| // static |
| bool LocalTestServer::GetTestServerDirectory(FilePath* directory) { |
| // Get path to python server script. |
| FilePath testserver_dir; |
| if (!PathService::Get(base::DIR_SOURCE_ROOT, &testserver_dir)) { |
| LOG(ERROR) << "Failed to get DIR_SOURCE_ROOT"; |
| return false; |
| } |
| |
| testserver_dir = testserver_dir |
| .Append(FILE_PATH_LITERAL("net")) |
| .Append(FILE_PATH_LITERAL("tools")) |
| .Append(FILE_PATH_LITERAL("testserver")); |
| *directory = testserver_dir; |
| return true; |
| } |
| |
| bool LocalTestServer::GetTestServerPath(FilePath* testserver_path) const { |
| if (!GetTestServerDirectory(testserver_path)) |
| return false; |
| *testserver_path = testserver_path->Append(FILE_PATH_LITERAL( |
| "testserver.py")); |
| return true; |
| } |
| |
| bool LocalTestServer::Start() { |
| // Get path to Python server script. |
| FilePath testserver_path; |
| if (!GetTestServerPath(&testserver_path)) |
| return false; |
| |
| if (!SetPythonPath()) |
| return false; |
| |
| if (!LaunchPython(testserver_path)) |
| return false; |
| |
| if (!WaitToStart()) { |
| Stop(); |
| return false; |
| } |
| |
| return SetupWhenServerStarted(); |
| } |
| |
| bool LocalTestServer::Stop() { |
| CleanUpWhenStoppingServer(); |
| |
| if (!process_handle_) |
| return true; |
| |
| // First check if the process has already terminated. |
| bool ret = base::WaitForSingleProcess(process_handle_, base::TimeDelta()); |
| if (!ret) |
| ret = base::KillProcess(process_handle_, 1, true); |
| |
| if (ret) { |
| base::CloseProcessHandle(process_handle_); |
| process_handle_ = base::kNullProcessHandle; |
| } else { |
| VLOG(1) << "Kill failed?"; |
| } |
| |
| return ret; |
| } |
| |
| bool LocalTestServer::Init(const FilePath& document_root) { |
| if (document_root.IsAbsolute()) |
| return false; |
| |
| // At this point, the port that the test server will listen on is unknown. |
| // The test server will listen on an ephemeral port, and write the port |
| // number out over a pipe that this TestServer object will read from. Once |
| // that is complete, the host port pair will contain the actual port. |
| DCHECK(!GetPort()); |
| process_handle_ = base::kNullProcessHandle; |
| |
| FilePath src_dir; |
| if (!PathService::Get(base::DIR_SOURCE_ROOT, &src_dir)) |
| return false; |
| SetResourcePath(src_dir.Append(document_root), |
| src_dir.AppendASCII("net") |
| .AppendASCII("data") |
| .AppendASCII("ssl") |
| .AppendASCII("certificates")); |
| return true; |
| } |
| |
| bool LocalTestServer::SetPythonPath() const { |
| return SetPythonPathStatic(); |
| } |
| |
| // static |
| bool LocalTestServer::SetPythonPathStatic() { |
| FilePath third_party_dir; |
| if (!PathService::Get(base::DIR_SOURCE_ROOT, &third_party_dir)) { |
| LOG(ERROR) << "Failed to get DIR_SOURCE_ROOT"; |
| return false; |
| } |
| third_party_dir = third_party_dir.AppendASCII("third_party"); |
| |
| // For simplejson. (simplejson, unlike all the other Python modules |
| // we include, doesn't have an extra 'simplejson' directory, so we |
| // need to include its parent directory, i.e. third_party_dir). |
| AppendToPythonPath(third_party_dir); |
| |
| AppendToPythonPath(third_party_dir.AppendASCII("tlslite")); |
| AppendToPythonPath( |
| third_party_dir.AppendASCII("pyftpdlib").AppendASCII("src")); |
| AppendToPythonPath( |
| third_party_dir.AppendASCII("pywebsocket").AppendASCII("src")); |
| |
| // Locate the Python code generated by the protocol buffers compiler. |
| FilePath pyproto_dir; |
| if (!GetPyProtoPath(&pyproto_dir)) { |
| LOG(WARNING) << "Cannot find pyproto dir for generated code. " |
| << "Testserver features that rely on it will not work"; |
| return true; |
| } |
| |
| AppendToPythonPath(pyproto_dir); |
| AppendToPythonPath(pyproto_dir.AppendASCII("sync").AppendASCII("protocol")); |
| AppendToPythonPath(pyproto_dir.AppendASCII("chrome") |
| .AppendASCII("browser") |
| .AppendASCII("policy") |
| .AppendASCII("proto")); |
| |
| return true; |
| } |
| |
| bool LocalTestServer::AddCommandLineArguments(CommandLine* command_line) const { |
| base::DictionaryValue arguments_dict; |
| if (!GenerateArguments(&arguments_dict)) |
| return false; |
| |
| // Serialize the argument dictionary into CommandLine. |
| for (DictionaryValue::Iterator it(arguments_dict); it.HasNext(); |
| it.Advance()) { |
| const base::Value& value = it.value(); |
| const std::string& key = it.key(); |
| |
| // Add arguments from a list. |
| if (value.IsType(Value::TYPE_LIST)) { |
| const base::ListValue* list = NULL; |
| if (!value.GetAsList(&list) || !list || list->empty()) |
| return false; |
| for (base::ListValue::const_iterator list_it = list->begin(); |
| list_it != list->end(); ++list_it) { |
| if (!AppendArgumentFromJSONValue(key, *(*list_it), command_line)) |
| return false; |
| } |
| } else if (!AppendArgumentFromJSONValue(key, value, command_line)) { |
| return false; |
| } |
| } |
| |
| // Append the appropriate server type argument. |
| switch (type()) { |
| case TYPE_HTTP: |
| case TYPE_HTTPS: |
| // The default type is HTTP, no argument required. |
| break; |
| case TYPE_WS: |
| case TYPE_WSS: |
| command_line->AppendArg("--websocket"); |
| break; |
| case TYPE_FTP: |
| command_line->AppendArg("-f"); |
| break; |
| case TYPE_SYNC: |
| command_line->AppendArg("--sync"); |
| break; |
| case TYPE_TCP_ECHO: |
| command_line->AppendArg("--tcp-echo"); |
| break; |
| case TYPE_UDP_ECHO: |
| command_line->AppendArg("--udp-echo"); |
| break; |
| case TYPE_BASIC_AUTH_PROXY: |
| command_line->AppendArg("--basic-auth-proxy"); |
| break; |
| default: |
| NOTREACHED(); |
| return false; |
| } |
| |
| return true; |
| } |
| |
| } // namespace net |