| OLD | NEW |
| 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "chrome/test/webdriver/commands/create_session.h" | 5 #include "chrome/test/webdriver/commands/create_session.h" |
| 6 | 6 |
| 7 #include <sstream> | |
| 8 #include <string> | 7 #include <string> |
| 9 | 8 |
| 10 #include "base/base64.h" | |
| 11 #include "base/command_line.h" | 9 #include "base/command_line.h" |
| 12 #include "base/file_path.h" | 10 #include "base/file_path.h" |
| 13 #include "base/file_util.h" | |
| 14 #include "base/logging.h" | 11 #include "base/logging.h" |
| 15 #include "base/scoped_temp_dir.h" | 12 #include "base/scoped_temp_dir.h" |
| 16 #include "base/string_number_conversions.h" | |
| 17 #include "base/stringprintf.h" | |
| 18 #include "base/values.h" | 13 #include "base/values.h" |
| 19 #include "chrome/app/chrome_command_ids.h" | |
| 20 #include "chrome/common/chrome_constants.h" | |
| 21 #include "chrome/common/chrome_switches.h" | |
| 22 #include "chrome/common/zip.h" | |
| 23 #include "chrome/test/webdriver/commands/response.h" | 14 #include "chrome/test/webdriver/commands/response.h" |
| 15 #include "chrome/test/webdriver/webdriver_capabilities_parser.h" |
| 24 #include "chrome/test/webdriver/webdriver_error.h" | 16 #include "chrome/test/webdriver/webdriver_error.h" |
| 25 #include "chrome/test/webdriver/webdriver_session.h" | 17 #include "chrome/test/webdriver/webdriver_session.h" |
| 26 #include "chrome/test/webdriver/webdriver_session_manager.h" | 18 #include "chrome/test/webdriver/webdriver_session_manager.h" |
| 27 | 19 |
| 28 namespace webdriver { | 20 namespace webdriver { |
| 29 | 21 |
| 30 namespace { | |
| 31 | |
| 32 bool WriteBase64DataToFile(const FilePath& filename, | |
| 33 const std::string& base64data, | |
| 34 std::string* error_msg) { | |
| 35 std::string data; | |
| 36 if (!base::Base64Decode(base64data, &data)) { | |
| 37 *error_msg = "Invalid base64 encoded data."; | |
| 38 return false; | |
| 39 } | |
| 40 if (!file_util::WriteFile(filename, data.c_str(), data.length())) { | |
| 41 *error_msg = "Could not write data to file."; | |
| 42 return false; | |
| 43 } | |
| 44 return true; | |
| 45 } | |
| 46 | |
| 47 Error* GetBooleanCapability( | |
| 48 const base::DictionaryValue* dict, const std::string& key, bool* option) { | |
| 49 Value* value = NULL; | |
| 50 if (dict->GetWithoutPathExpansion(key, &value)) { | |
| 51 if (!value->GetAsBoolean(option)) { | |
| 52 return new Error(kUnknownError, key + " must be a boolean"); | |
| 53 } | |
| 54 } | |
| 55 return NULL; | |
| 56 } | |
| 57 | |
| 58 } // namespace | |
| 59 | |
| 60 CreateSession::CreateSession(const std::vector<std::string>& path_segments, | 22 CreateSession::CreateSession(const std::vector<std::string>& path_segments, |
| 61 const DictionaryValue* const parameters) | 23 const DictionaryValue* const parameters) |
| 62 : Command(path_segments, parameters) {} | 24 : Command(path_segments, parameters) {} |
| 63 | 25 |
| 64 CreateSession::~CreateSession() {} | 26 CreateSession::~CreateSession() {} |
| 65 | 27 |
| 66 bool CreateSession::DoesPost() { return true; } | 28 bool CreateSession::DoesPost() { return true; } |
| 67 | 29 |
| 68 void CreateSession::ExecutePost(Response* const response) { | 30 void CreateSession::ExecutePost(Response* const response) { |
| 69 DictionaryValue *capabilities = NULL; | 31 DictionaryValue* dict; |
| 70 if (!GetDictionaryParameter("desiredCapabilities", &capabilities)) { | 32 if (!GetDictionaryParameter("desiredCapabilities", &dict)) { |
| 71 response->SetError(new Error( | 33 response->SetError(new Error( |
| 72 kBadRequest, "Missing or invalid 'desiredCapabilities'")); | 34 kBadRequest, "Missing or invalid 'desiredCapabilities'")); |
| 73 return; | 35 return; |
| 74 } | 36 } |
| 75 | 37 ScopedTempDir temp_dir; |
| 76 Automation::BrowserOptions browser_options; | 38 if (!temp_dir.CreateUniqueTempDir()) { |
| 77 FilePath::StringType path; | |
| 78 if (capabilities->GetStringWithoutPathExpansion("chrome.binary", &path)) | |
| 79 browser_options.command = CommandLine(FilePath(path)); | |
| 80 | |
| 81 ListValue* switches = NULL; | |
| 82 const char* kCustomSwitchesKey = "chrome.switches"; | |
| 83 if (capabilities->GetListWithoutPathExpansion(kCustomSwitchesKey, | |
| 84 &switches)) { | |
| 85 for (size_t i = 0; i < switches->GetSize(); ++i) { | |
| 86 std::string switch_string; | |
| 87 if (!switches->GetString(i, &switch_string)) { | |
| 88 response->SetError(new Error( | |
| 89 kBadRequest, "Custom switch is not a string")); | |
| 90 return; | |
| 91 } | |
| 92 size_t separator_index = switch_string.find("="); | |
| 93 if (separator_index != std::string::npos) { | |
| 94 CommandLine::StringType switch_string_native; | |
| 95 if (!switches->GetString(i, &switch_string_native)) { | |
| 96 response->SetError(new Error( | |
| 97 kBadRequest, "Custom switch is not a string")); | |
| 98 return; | |
| 99 } | |
| 100 browser_options.command.AppendSwitchNative( | |
| 101 switch_string.substr(0, separator_index), | |
| 102 switch_string_native.substr(separator_index + 1)); | |
| 103 } else { | |
| 104 browser_options.command.AppendSwitch(switch_string); | |
| 105 } | |
| 106 } | |
| 107 } else if (capabilities->HasKey(kCustomSwitchesKey)) { | |
| 108 response->SetError(new Error( | 39 response->SetError(new Error( |
| 109 kBadRequest, "Custom switches must be a list")); | 40 kUnknownError, "Unable to create temp directory for unpacking")); |
| 110 return; | 41 return; |
| 111 } | 42 } |
| 112 | 43 Capabilities caps; |
| 113 Value* verbose_value; | 44 CapabilitiesParser parser(dict, temp_dir.path(), &caps); |
| 114 if (capabilities->GetWithoutPathExpansion("chrome.verbose", &verbose_value)) { | 45 Error* error = parser.Parse(); |
| 115 bool verbose = false; | |
| 116 if (verbose_value->GetAsBoolean(&verbose)) { | |
| 117 // Since logging is shared among sessions, if any session requests verbose | |
| 118 // logging, verbose logging will be enabled for all sessions. It is not | |
| 119 // possible to turn it off. | |
| 120 if (verbose) | |
| 121 logging::SetMinLogLevel(logging::LOG_INFO); | |
| 122 } else { | |
| 123 response->SetError(new Error( | |
| 124 kBadRequest, "verbose must be a boolean true or false")); | |
| 125 return; | |
| 126 } | |
| 127 } | |
| 128 | |
| 129 capabilities->GetStringWithoutPathExpansion( | |
| 130 "chrome.channel", &browser_options.channel_id); | |
| 131 | |
| 132 ScopedTempDir temp_profile_dir; | |
| 133 std::string base64_profile; | |
| 134 if (capabilities->GetStringWithoutPathExpansion("chrome.profile", | |
| 135 &base64_profile)) { | |
| 136 if (!temp_profile_dir.CreateUniqueTempDir()) { | |
| 137 response->SetError(new Error( | |
| 138 kBadRequest, "Could not create temporary profile directory.")); | |
| 139 return; | |
| 140 } | |
| 141 FilePath temp_profile_zip( | |
| 142 temp_profile_dir.path().AppendASCII("profile.zip")); | |
| 143 std::string message; | |
| 144 if (!WriteBase64DataToFile(temp_profile_zip, base64_profile, &message)) { | |
| 145 response->SetError(new Error(kBadRequest, message)); | |
| 146 return; | |
| 147 } | |
| 148 | |
| 149 browser_options.user_data_dir = | |
| 150 temp_profile_dir.path().AppendASCII("user_data_dir"); | |
| 151 if (!zip::Unzip(temp_profile_zip, browser_options.user_data_dir)) { | |
| 152 response->SetError(new Error( | |
| 153 kBadRequest, "Could not unarchive provided user profile")); | |
| 154 return; | |
| 155 } | |
| 156 } | |
| 157 | |
| 158 const char* kExtensions = "chrome.extensions"; | |
| 159 ScopedTempDir extensions_dir; | |
| 160 ListValue* extensions_list = NULL; | |
| 161 std::vector<FilePath> extensions; | |
| 162 if (capabilities->GetListWithoutPathExpansion(kExtensions, | |
| 163 &extensions_list)) { | |
| 164 if (!extensions_dir.CreateUniqueTempDir()) { | |
| 165 response->SetError(new Error( | |
| 166 kBadRequest, "Could create temporary extensions directory.")); | |
| 167 return; | |
| 168 } | |
| 169 for (size_t i = 0; i < extensions_list->GetSize(); ++i) { | |
| 170 std::string base64_extension; | |
| 171 if (!extensions_list->GetString(i, &base64_extension)) { | |
| 172 response->SetError(new Error( | |
| 173 kBadRequest, "Extension must be a base64 encoded string.")); | |
| 174 return; | |
| 175 } | |
| 176 FilePath extension_file( | |
| 177 extensions_dir.path().AppendASCII("extension" + | |
| 178 base::IntToString(i) + ".crx")); | |
| 179 std::string message; | |
| 180 if (!WriteBase64DataToFile(extension_file, base64_extension, &message)) { | |
| 181 response->SetError(new Error(kBadRequest, message)); | |
| 182 return; | |
| 183 } | |
| 184 extensions.push_back(extension_file); | |
| 185 } | |
| 186 } else if (capabilities->HasKey(kExtensions)) { | |
| 187 response->SetError(new Error( | |
| 188 kBadRequest, "Extensions must be a list of base64 encoded strings")); | |
| 189 return; | |
| 190 } | |
| 191 | |
| 192 Session::Options session_options; | |
| 193 Error* error = NULL; | |
| 194 error = GetBooleanCapability(capabilities, "chrome.nativeEvents", | |
| 195 &session_options.use_native_events); | |
| 196 if (!error) { | |
| 197 error = GetBooleanCapability(capabilities, "chrome.loadAsync", | |
| 198 &session_options.load_async); | |
| 199 } | |
| 200 if (!error) { | |
| 201 error = GetBooleanCapability(capabilities, "chrome.detach", | |
| 202 &browser_options.detach_process); | |
| 203 } | |
| 204 if (error) { | 46 if (error) { |
| 205 response->SetError(error); | 47 response->SetError(error); |
| 206 return; | 48 return; |
| 207 } | 49 } |
| 208 | 50 |
| 51 // Since logging is shared among sessions, if any session requests verbose |
| 52 // logging, verbose logging will be enabled for all sessions. It is not |
| 53 // possible to turn it off. |
| 54 if (caps.verbose) |
| 55 logging::SetMinLogLevel(logging::LOG_INFO); |
| 56 |
| 57 Session::Options session_options; |
| 58 session_options.load_async = caps.load_async; |
| 59 session_options.use_native_events = caps.native_events; |
| 60 |
| 61 Automation::BrowserOptions browser_options; |
| 62 browser_options.command = caps.command; |
| 63 browser_options.channel_id = caps.channel; |
| 64 browser_options.detach_process = caps.detach; |
| 65 browser_options.user_data_dir = caps.profile; |
| 66 |
| 209 // Session manages its own liftime, so do not call delete. | 67 // Session manages its own liftime, so do not call delete. |
| 210 Session* session = new Session(session_options); | 68 Session* session = new Session(session_options); |
| 211 error = session->Init(browser_options); | 69 error = session->Init(browser_options); |
| 212 if (error) { | 70 if (error) { |
| 213 response->SetError(error); | 71 response->SetError(error); |
| 214 return; | 72 return; |
| 215 } | 73 } |
| 216 | 74 |
| 217 // Install extensions. | 75 // Install extensions. |
| 218 for (size_t i = 0; i < extensions.size(); ++i) { | 76 for (size_t i = 0; i < caps.extensions.size(); ++i) { |
| 219 Error* error = session->InstallExtension(extensions[i]); | 77 Error* error = session->InstallExtension(caps.extensions[i]); |
| 220 if (error) { | 78 if (error) { |
| 221 response->SetError(error); | 79 response->SetError(error); |
| 222 return; | 80 return; |
| 223 } | 81 } |
| 224 } | 82 } |
| 225 | 83 |
| 226 LOG(INFO) << "Created session " << session->id(); | 84 LOG(INFO) << "Created session " << session->id(); |
| 227 // Redirect to a relative URI. Although prohibited by the HTTP standard, | 85 // Redirect to a relative URI. Although prohibited by the HTTP standard, |
| 228 // this is what the IEDriver does. Finding the actual IP address is | 86 // this is what the IEDriver does. Finding the actual IP address is |
| 229 // difficult, and returning the hostname causes perf problems with the python | 87 // difficult, and returning the hostname causes perf problems with the python |
| 230 // bindings on Windows. | 88 // bindings on Windows. |
| 231 std::ostringstream stream; | 89 std::ostringstream stream; |
| 232 stream << SessionManager::GetInstance()->url_base() << "/session/" | 90 stream << SessionManager::GetInstance()->url_base() << "/session/" |
| 233 << session->id(); | 91 << session->id(); |
| 234 response->SetStatus(kSeeOther); | 92 response->SetStatus(kSeeOther); |
| 235 response->SetValue(Value::CreateStringValue(stream.str())); | 93 response->SetValue(Value::CreateStringValue(stream.str())); |
| 236 } | 94 } |
| 237 | 95 |
| 238 } // namespace webdriver | 96 } // namespace webdriver |
| OLD | NEW |