ozone: evdev: Sync caps lock LED state to evdev
[chromium-blink-merge.git] / net / test / spawned_test_server / local_test_server.cc
blob6ecb7d9cc7769363c900f26a3fefa957c5f50060
1 // 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.
5 #include "net/test/spawned_test_server/local_test_server.h"
7 #include "base/command_line.h"
8 #include "base/json/json_reader.h"
9 #include "base/logging.h"
10 #include "base/path_service.h"
11 #include "base/process/kill.h"
12 #include "base/strings/string_number_conversions.h"
13 #include "base/values.h"
14 #include "net/base/host_port_pair.h"
15 #include "net/base/net_errors.h"
16 #include "net/test/python_utils.h"
17 #include "url/gurl.h"
19 namespace net {
21 namespace {
23 bool AppendArgumentFromJSONValue(const std::string& key,
24 const base::Value& value_node,
25 base::CommandLine* command_line) {
26 std::string argument_name = "--" + key;
27 switch (value_node.GetType()) {
28 case base::Value::TYPE_NULL:
29 command_line->AppendArg(argument_name);
30 break;
31 case base::Value::TYPE_INTEGER: {
32 int value;
33 bool result = value_node.GetAsInteger(&value);
34 DCHECK(result);
35 command_line->AppendArg(argument_name + "=" + base::IntToString(value));
36 break;
38 case base::Value::TYPE_STRING: {
39 std::string value;
40 bool result = value_node.GetAsString(&value);
41 if (!result || value.empty())
42 return false;
43 command_line->AppendArg(argument_name + "=" + value);
44 break;
46 case base::Value::TYPE_BOOLEAN:
47 case base::Value::TYPE_DOUBLE:
48 case base::Value::TYPE_LIST:
49 case base::Value::TYPE_DICTIONARY:
50 case base::Value::TYPE_BINARY:
51 default:
52 NOTREACHED() << "improper json type";
53 return false;
55 return true;
58 } // namespace
60 LocalTestServer::LocalTestServer(Type type,
61 const std::string& host,
62 const base::FilePath& document_root)
63 : BaseTestServer(type, host) {
64 if (!Init(document_root))
65 NOTREACHED();
68 LocalTestServer::LocalTestServer(Type type,
69 const SSLOptions& ssl_options,
70 const base::FilePath& document_root)
71 : BaseTestServer(type, ssl_options) {
72 if (!Init(document_root))
73 NOTREACHED();
76 LocalTestServer::~LocalTestServer() {
77 Stop();
80 bool LocalTestServer::GetTestServerPath(base::FilePath* testserver_path) const {
81 base::FilePath testserver_dir;
82 if (!PathService::Get(base::DIR_SOURCE_ROOT, &testserver_dir)) {
83 LOG(ERROR) << "Failed to get DIR_SOURCE_ROOT";
84 return false;
86 testserver_dir = testserver_dir.Append(FILE_PATH_LITERAL("net"))
87 .Append(FILE_PATH_LITERAL("tools"))
88 .Append(FILE_PATH_LITERAL("testserver"));
89 *testserver_path = testserver_dir.Append(FILE_PATH_LITERAL("testserver.py"));
90 return true;
93 bool LocalTestServer::Start() {
94 return StartInBackground() && BlockUntilStarted();
97 bool LocalTestServer::StartInBackground() {
98 // Get path to Python server script.
99 base::FilePath testserver_path;
100 if (!GetTestServerPath(&testserver_path))
101 return false;
103 if (!SetPythonPath())
104 return false;
106 if (!LaunchPython(testserver_path))
107 return false;
109 return true;
112 bool LocalTestServer::BlockUntilStarted() {
113 if (!WaitToStart()) {
114 Stop();
115 return false;
118 return SetupWhenServerStarted();
121 bool LocalTestServer::Stop() {
122 CleanUpWhenStoppingServer();
124 if (!process_.IsValid())
125 return true;
127 // First check if the process has already terminated.
128 int exit_code;
129 bool ret = process_.WaitForExitWithTimeout(base::TimeDelta(), &exit_code);
130 if (!ret)
131 ret = base::KillProcess(process_.Handle(), 1, true);
133 if (ret)
134 process_.Close();
135 else
136 VLOG(1) << "Kill failed?";
138 return ret;
141 bool LocalTestServer::Init(const base::FilePath& document_root) {
142 if (document_root.IsAbsolute())
143 return false;
145 // At this point, the port that the test server will listen on is unknown.
146 // The test server will listen on an ephemeral port, and write the port
147 // number out over a pipe that this TestServer object will read from. Once
148 // that is complete, the host port pair will contain the actual port.
149 DCHECK(!GetPort());
151 base::FilePath src_dir;
152 if (!PathService::Get(base::DIR_SOURCE_ROOT, &src_dir))
153 return false;
154 SetResourcePath(src_dir.Append(document_root),
155 src_dir.AppendASCII("net")
156 .AppendASCII("data")
157 .AppendASCII("ssl")
158 .AppendASCII("certificates"));
159 return true;
162 bool LocalTestServer::SetPythonPath() const {
163 base::FilePath third_party_dir;
164 if (!PathService::Get(base::DIR_SOURCE_ROOT, &third_party_dir)) {
165 LOG(ERROR) << "Failed to get DIR_SOURCE_ROOT";
166 return false;
168 third_party_dir = third_party_dir.AppendASCII("third_party");
170 // For simplejson. (simplejson, unlike all the other Python modules
171 // we include, doesn't have an extra 'simplejson' directory, so we
172 // need to include its parent directory, i.e. third_party_dir).
173 AppendToPythonPath(third_party_dir);
175 AppendToPythonPath(third_party_dir.AppendASCII("tlslite"));
176 AppendToPythonPath(
177 third_party_dir.AppendASCII("pyftpdlib").AppendASCII("src"));
178 AppendToPythonPath(
179 third_party_dir.AppendASCII("pywebsocket").AppendASCII("src"));
181 // Locate the Python code generated by the protocol buffers compiler.
182 base::FilePath pyproto_dir;
183 if (!GetPyProtoPath(&pyproto_dir)) {
184 LOG(WARNING) << "Cannot find pyproto dir for generated code. "
185 << "Testserver features that rely on it will not work";
186 return true;
188 AppendToPythonPath(pyproto_dir);
190 return true;
193 bool LocalTestServer::AddCommandLineArguments(
194 base::CommandLine* command_line) const {
195 base::DictionaryValue arguments_dict;
196 if (!GenerateArguments(&arguments_dict))
197 return false;
199 // Serialize the argument dictionary into CommandLine.
200 for (base::DictionaryValue::Iterator it(arguments_dict); !it.IsAtEnd();
201 it.Advance()) {
202 const base::Value& value = it.value();
203 const std::string& key = it.key();
205 // Add arguments from a list.
206 if (value.IsType(base::Value::TYPE_LIST)) {
207 const base::ListValue* list = NULL;
208 if (!value.GetAsList(&list) || !list || list->empty())
209 return false;
210 for (base::ListValue::const_iterator list_it = list->begin();
211 list_it != list->end(); ++list_it) {
212 if (!AppendArgumentFromJSONValue(key, *(*list_it), command_line))
213 return false;
215 } else if (!AppendArgumentFromJSONValue(key, value, command_line)) {
216 return false;
220 // Append the appropriate server type argument.
221 switch (type()) {
222 case TYPE_HTTP: // The default type is HTTP, no argument required.
223 break;
224 case TYPE_HTTPS:
225 command_line->AppendArg("--https");
226 break;
227 case TYPE_WS:
228 case TYPE_WSS:
229 command_line->AppendArg("--websocket");
230 break;
231 case TYPE_FTP:
232 command_line->AppendArg("--ftp");
233 break;
234 case TYPE_TCP_ECHO:
235 command_line->AppendArg("--tcp-echo");
236 break;
237 case TYPE_UDP_ECHO:
238 command_line->AppendArg("--udp-echo");
239 break;
240 case TYPE_BASIC_AUTH_PROXY:
241 command_line->AppendArg("--basic-auth-proxy");
242 break;
243 default:
244 NOTREACHED();
245 return false;
248 return true;
251 } // namespace net