| // Copyright 2011 Software Freedom Conservancy | |
| // Licensed under the Apache License, Version 2.0 (the "License"); | |
| // you may not use this file except in compliance with the License. | |
| // You may obtain a copy of the License at | |
| // | |
| // http://www.apache.org/licenses/LICENSE-2.0 | |
| // | |
| // Unless required by applicable law or agreed to in writing, software | |
| // distributed under the License is distributed on an "AS IS" BASIS, | |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
| // See the License for the specific language governing permissions and | |
| // limitations under the License. | |
| #ifndef WEBDRIVER_IE_EXECUTEASYNCSCRIPTCOMMANDHANDLER_H_ | |
| #define WEBDRIVER_IE_EXECUTEASYNCSCRIPTCOMMANDHANDLER_H_ | |
| #include "../Browser.h" | |
| #include "../IECommandHandler.h" | |
| #include "../IECommandExecutor.h" | |
| #include "ExecuteScriptCommandHandler.h" | |
| #define GUID_STRING_LEN 40 | |
| namespace webdriver { | |
| class ExecuteAsyncScriptCommandHandler : public ExecuteScriptCommandHandler { | |
| public: | |
| ExecuteAsyncScriptCommandHandler(void) { | |
| } | |
| virtual ~ExecuteAsyncScriptCommandHandler(void) { | |
| } | |
| protected: | |
| void ExecuteInternal(const IECommandExecutor& executor, | |
| const LocatorMap& locator_parameters, | |
| const ParametersMap& command_parameters, | |
| Response* response) { | |
| ParametersMap::const_iterator script_parameter_iterator = command_parameters.find("script"); | |
| ParametersMap::const_iterator args_parameter_iterator = command_parameters.find("args"); | |
| if (script_parameter_iterator == command_parameters.end()) { | |
| response->SetErrorResponse(400, "Missing parameter: script"); | |
| return; | |
| } else if (args_parameter_iterator == command_parameters.end()) { | |
| response->SetErrorResponse(400, "Missing parameter: args"); | |
| return; | |
| } else { | |
| wchar_t page_id_buffer[GUID_STRING_LEN] = {0}; | |
| GUID page_id_guid; | |
| ::CoCreateGuid(&page_id_guid); | |
| ::StringFromGUID2(page_id_guid, page_id_buffer, GUID_STRING_LEN); | |
| std::wstring page_id = &page_id_buffer[0]; | |
| wchar_t pending_id_buffer[GUID_STRING_LEN] = {0}; | |
| GUID pending_id_guid; | |
| ::CoCreateGuid(&pending_id_guid); | |
| ::StringFromGUID2(pending_id_guid, pending_id_buffer, GUID_STRING_LEN); | |
| std::wstring pending_id = &pending_id_buffer[0]; | |
| Json::Value json_args = args_parameter_iterator->second; | |
| int timeout_value = executor.async_script_timeout(); | |
| wchar_t timeout_buffer[12] = {0}; | |
| _itow_s(timeout_value, &timeout_buffer[0], 12, 10); | |
| std::wstring timeout = &timeout_buffer[0]; | |
| std::wstring script_body = CA2W(script_parameter_iterator->second.asCString(), CP_UTF8); | |
| std::wstring async_script = L"(function() { return function(){\n"; | |
| async_script += L"document.__$webdriverAsyncExecutor = {\n"; | |
| async_script += L" pageId: '" + page_id + L"',\n"; | |
| async_script += L" asyncTimeout: 0\n"; | |
| async_script += L"};\n"; | |
| async_script += L"var timeoutId = window.setTimeout(function() {\n"; | |
| async_script += L" window.setTimeout(function() {\n"; | |
| async_script += L" document.__$webdriverAsyncExecutor.asyncTimeout = 1;\n"; | |
| async_script += L" }, 0);\n"; | |
| async_script += L"}," + timeout + L");\n"; | |
| async_script += L"var callback = function(value) {\n"; | |
| async_script += L" document.__$webdriverAsyncExecutor.asyncTimeout = 0;\n"; | |
| async_script += L" document.__$webdriverAsyncExecutor.asyncScriptResult = value;\n"; | |
| async_script += L" window.clearTimeout(timeoutId);\n"; | |
| async_script += L"};\n"; | |
| async_script += L"var argsArray = Array.prototype.slice.call(arguments);\n"; | |
| async_script += L"argsArray.push(callback);\n"; | |
| async_script += L"if (document.__$webdriverAsyncExecutor.asyncScriptResult !== undefined) {\n"; | |
| async_script += L" delete document.__$webdriverAsyncExecutor.asyncScriptResult;\n"; | |
| async_script += L"}\n"; | |
| async_script += L"(function() {" + script_body + L"}).apply(null, argsArray);\n"; | |
| async_script += L"};})();"; | |
| std::wstring polling_script = L"(function() { return function(){\n"; | |
| polling_script += L"var pendingId = '" + pending_id + L"';\n"; | |
| polling_script += L"if ('__$webdriverAsyncExecutor' in document) {\n"; | |
| polling_script += L" if (document.__$webdriverAsyncExecutor.pageId != '" + page_id + L"') {\n"; | |
| polling_script += L" return [pendingId, -1];\n"; | |
| polling_script += L" } else if ('asyncScriptResult' in document.__$webdriverAsyncExecutor) {\n"; | |
| polling_script += L" var value = document.__$webdriverAsyncExecutor.asyncScriptResult;\n"; | |
| polling_script += L" delete document.__$webdriverAsyncExecutor.asyncScriptResult;\n"; | |
| polling_script += L" return value;\n"; | |
| polling_script += L" } else {\n"; | |
| polling_script += L" return [pendingId, document.__$webdriverAsyncExecutor.asyncTimeout];\n"; | |
| polling_script += L" }\n"; | |
| polling_script += L"} else {\n"; | |
| polling_script += L" return [pendingId, -1];\n"; | |
| polling_script += L"}\n"; | |
| polling_script += L"};})();"; | |
| BrowserHandle browser_wrapper; | |
| int status_code = executor.GetCurrentBrowser(&browser_wrapper); | |
| if (status_code != SUCCESS) { | |
| response->SetErrorResponse(status_code, "Unable to get browser"); | |
| return; | |
| } | |
| CComPtr<IHTMLDocument2> doc; | |
| browser_wrapper->GetDocument(&doc); | |
| Script async_script_wrapper(doc, async_script, json_args.size()); | |
| status_code = this->PopulateArgumentArray(executor, | |
| async_script_wrapper, | |
| json_args); | |
| if (status_code != SUCCESS) { | |
| response->SetErrorResponse(status_code, | |
| "Error setting arguments for script"); | |
| return; | |
| } | |
| status_code = async_script_wrapper.Execute(); | |
| if (status_code != SUCCESS) { | |
| response->SetErrorResponse(status_code, | |
| "JavaScript error in async script."); | |
| return; | |
| } else { | |
| Script polling_script_wrapper(doc, polling_script, 0); | |
| while (true) { | |
| Json::Value polling_result; | |
| status_code = polling_script_wrapper.Execute(); | |
| if (status_code != SUCCESS) { | |
| // Assume that if the polling script errors, it's because | |
| // of a page reload. Note that experience shows this to | |
| // happen most frequently when a refresh occurs, since | |
| // the document object is not yet ready for accessing. | |
| // However, this is still a big assumption,and could be faulty. | |
| response->SetErrorResponse(EUNEXPECTEDJSERROR, | |
| "Page reload detected during async script"); | |
| break; | |
| } | |
| polling_script_wrapper.ConvertResultToJsonValue(executor, &polling_result); | |
| Json::UInt index = 0; | |
| std::string narrow_pending_id(CW2A(pending_id.c_str(), CP_UTF8)); | |
| if (polling_result.isArray() && | |
| polling_result.size() == 2 && | |
| polling_result[index].isString() && | |
| polling_result[index].asString() == narrow_pending_id) { | |
| int timeout_flag = polling_result[1].asInt(); | |
| if (timeout_flag < 0) { | |
| response->SetErrorResponse(EUNEXPECTEDJSERROR, | |
| "Page reload detected during async script"); | |
| break; | |
| } | |
| if (timeout_flag > 0) { | |
| response->SetErrorResponse(ESCRIPTTIMEOUT, | |
| "Timeout expired waiting for async script"); | |
| break; | |
| } | |
| } else { | |
| response->SetSuccessResponse(polling_result); | |
| break; | |
| } | |
| } | |
| return; | |
| } | |
| } | |
| } | |
| }; | |
| } // namespace webdriver | |
| #endif // WEBDRIVER_IE_EXECUTEASYNCSCRIPTCOMMANDHANDLER_H_ |