// 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 "chrome/test/automation/tab_proxy.h"
#include <algorithm>
#include "base/json/json_string_value_serializer.h"
#include "base/logging.h"
#include "base/strings/string16.h"
#include "base/strings/utf_string_conversions.h"
#include "base/threading/platform_thread.h"
#include "chrome/common/automation_constants.h"
#include "chrome/common/automation_messages.h"
#include "chrome/test/automation/automation_proxy.h"
#include "chrome/test/automation/browser_proxy.h"
#include "url/gurl.h"
TabProxy::TabProxy(AutomationMessageSender* sender,
AutomationHandleTracker* tracker,
int handle)
: AutomationResourceProxy(tracker, sender, handle) {
}
bool TabProxy::GetTabTitle(std::wstring* title) const {
if (!is_valid())
return false;
if (!title) {
NOTREACHED();
return false;
}
int tab_title_size_response = 0;
bool succeeded = sender_->Send(
new AutomationMsg_TabTitle(handle_, &tab_title_size_response, title));
return succeeded;
}
bool TabProxy::GetTabIndex(int* index) const {
if (!is_valid())
return false;
if (!index) {
NOTREACHED();
return false;
}
return sender_->Send(new AutomationMsg_TabIndex(handle_, index));
}
int TabProxy::FindInPage(const std::wstring& search_string,
FindInPageDirection forward,
FindInPageCase match_case,
bool find_next,
int* ordinal) {
if (!is_valid())
return -1;
AutomationMsg_Find_Params params;
params.search_string = WideToUTF16Hack(search_string);
params.find_next = find_next;
params.match_case = (match_case == CASE_SENSITIVE);
params.forward = (forward == FWD);
int matches = 0;
int ordinal2 = 0;
bool succeeded = sender_->Send(new AutomationMsg_Find(handle_,
params,
&ordinal2,
&matches));
if (!succeeded)
return -1;
if (ordinal)
*ordinal = ordinal2;
return matches;
}
AutomationMsg_NavigationResponseValues TabProxy::NavigateToURL(
const GURL& url) {
return NavigateToURLBlockUntilNavigationsComplete(url, 1);
}
AutomationMsg_NavigationResponseValues
TabProxy::NavigateToURLBlockUntilNavigationsComplete(
const GURL& url, int number_of_navigations) {
if (!is_valid())
return AUTOMATION_MSG_NAVIGATION_ERROR;
AutomationMsg_NavigationResponseValues navigate_response =
AUTOMATION_MSG_NAVIGATION_ERROR;
sender_->Send(new AutomationMsg_NavigateToURLBlockUntilNavigationsComplete(
handle_, url, number_of_navigations, &navigate_response));
return navigate_response;
}
AutomationMsg_NavigationResponseValues TabProxy::GoBack() {
return GoBackBlockUntilNavigationsComplete(1);
}
AutomationMsg_NavigationResponseValues
TabProxy::GoBackBlockUntilNavigationsComplete(int number_of_navigations) {
if (!is_valid())
return AUTOMATION_MSG_NAVIGATION_ERROR;
AutomationMsg_NavigationResponseValues navigate_response =
AUTOMATION_MSG_NAVIGATION_ERROR;
sender_->Send(new AutomationMsg_GoBackBlockUntilNavigationsComplete(
handle_, number_of_navigations, &navigate_response));
return navigate_response;
}
AutomationMsg_NavigationResponseValues TabProxy::GoForward() {
return GoForwardBlockUntilNavigationsComplete(1);
}
AutomationMsg_NavigationResponseValues
TabProxy::GoForwardBlockUntilNavigationsComplete(
int number_of_navigations) {
if (!is_valid())
return AUTOMATION_MSG_NAVIGATION_ERROR;
AutomationMsg_NavigationResponseValues navigate_response =
AUTOMATION_MSG_NAVIGATION_ERROR;
sender_->Send(new AutomationMsg_GoForwardBlockUntilNavigationsComplete(
handle_, number_of_navigations, &navigate_response));
return navigate_response;
}
AutomationMsg_NavigationResponseValues TabProxy::Reload() {
if (!is_valid())
return AUTOMATION_MSG_NAVIGATION_ERROR;
AutomationMsg_NavigationResponseValues navigate_response =
AUTOMATION_MSG_NAVIGATION_ERROR;
sender_->Send(new AutomationMsg_Reload(handle_, &navigate_response));
return navigate_response;
}
bool TabProxy::GetCurrentURL(GURL* url) const {
if (!is_valid())
return false;
if (!url) {
NOTREACHED();
return false;
}
bool succeeded = false;
sender_->Send(new AutomationMsg_TabURL(handle_, &succeeded, url));
return succeeded;
}
bool TabProxy::NavigateToURLAsync(const GURL& url) {
if (!is_valid())
return false;
bool status = false;
sender_->Send(new AutomationMsg_NavigationAsync(handle_,
url,
&status));
return status;
}
bool TabProxy::ExecuteAndExtractString(const std::wstring& frame_xpath,
const std::wstring& jscript,
std::wstring* string_value) {
scoped_ptr<Value> root(ExecuteAndExtractValue(frame_xpath, jscript));
if (root == NULL)
return false;
DCHECK(root->IsType(Value::TYPE_LIST));
Value* value = NULL;
bool succeeded = static_cast<ListValue*>(root.get())->Get(0, &value);
if (succeeded) {
string16 read_value;
succeeded = value->GetAsString(&read_value);
if (succeeded) {
// TODO(viettrungluu): remove conversion. (But should |jscript| be UTF-8?)
*string_value = UTF16ToWideHack(read_value);
}
}
return succeeded;
}
bool TabProxy::ExecuteAndExtractBool(const std::wstring& frame_xpath,
const std::wstring& jscript,
bool* bool_value) {
scoped_ptr<Value> root(ExecuteAndExtractValue(frame_xpath, jscript));
if (root == NULL)
return false;
bool read_value = false;
DCHECK(root->IsType(Value::TYPE_LIST));
Value* value = NULL;
bool succeeded = static_cast<ListValue*>(root.get())->Get(0, &value);
if (succeeded) {
succeeded = value->GetAsBoolean(&read_value);
if (succeeded) {
*bool_value = read_value;
}
}
return succeeded;
}
bool TabProxy::ExecuteAndExtractInt(const std::wstring& frame_xpath,
const std::wstring& jscript,
int* int_value) {
scoped_ptr<Value> root(ExecuteAndExtractValue(frame_xpath, jscript));
if (root == NULL)
return false;
int read_value = 0;
DCHECK(root->IsType(Value::TYPE_LIST));
Value* value = NULL;
bool succeeded = static_cast<ListValue*>(root.get())->Get(0, &value);
if (succeeded) {
succeeded = value->GetAsInteger(&read_value);
if (succeeded) {
*int_value = read_value;
}
}
return succeeded;
}
Value* TabProxy::ExecuteAndExtractValue(const std::wstring& frame_xpath,
const std::wstring& jscript) {
if (!is_valid())
return NULL;
std::string json;
if (!sender_->Send(new AutomationMsg_DomOperation(handle_, frame_xpath,
jscript, &json))) {
return NULL;
}
// Wrap |json| in an array before deserializing because valid JSON has an
// array or an object as the root.
json.insert(0, "[");
json.append("]");
JSONStringValueSerializer deserializer(json);
return deserializer.Deserialize(NULL, NULL);
}
bool TabProxy::GetCookies(const GURL& url, std::string* cookies) {
if (!is_valid())
return false;
int size = 0;
return sender_->Send(new AutomationMsg_GetCookies(url, handle_, &size,
cookies));
}
bool TabProxy::GetCookieByName(const GURL& url,
const std::string& name,
std::string* cookie) {
std::string cookies;
if (!GetCookies(url, &cookies))
return false;
std::string namestr = name + "=";
std::string::size_type idx = cookies.find(namestr);
if (idx != std::string::npos) {
cookies.erase(0, idx + namestr.length());
*cookie = cookies.substr(0, cookies.find(";"));
} else {
cookie->clear();
}
return true;
}
bool TabProxy::Close() {
return Close(false);
}
bool TabProxy::Close(bool wait_until_closed) {
if (!is_valid())
return false;
bool succeeded = false;
sender_->Send(new AutomationMsg_CloseTab(handle_, wait_until_closed,
&succeeded));
return succeeded;
}
#if defined(OS_WIN)
bool TabProxy::ProcessUnhandledAccelerator(const MSG& msg) {
if (!is_valid())
return false;
return sender_->Send(
new AutomationMsg_ProcessUnhandledAccelerator(handle_, msg));
// This message expects no response
}
bool TabProxy::SetInitialFocus(bool reverse, bool restore_focus_to_view) {
if (!is_valid())
return false;
return sender_->Send(
new AutomationMsg_SetInitialFocus(handle_, reverse,
restore_focus_to_view));
// This message expects no response
}
AutomationMsg_NavigationResponseValues TabProxy::NavigateInExternalTab(
const GURL& url, const GURL& referrer) {
if (!is_valid())
return AUTOMATION_MSG_NAVIGATION_ERROR;
AutomationMsg_NavigationResponseValues rv = AUTOMATION_MSG_NAVIGATION_ERROR;
sender_->Send(new AutomationMsg_NavigateInExternalTab(handle_, url,
referrer, &rv));
return rv;
}
AutomationMsg_NavigationResponseValues TabProxy::NavigateExternalTabAtIndex(
int index) {
if (!is_valid())
return AUTOMATION_MSG_NAVIGATION_ERROR;
AutomationMsg_NavigationResponseValues rv = AUTOMATION_MSG_NAVIGATION_ERROR;
sender_->Send(new AutomationMsg_NavigateExternalTabAtIndex(handle_, index,
&rv));
return rv;
}
void TabProxy::HandleMessageFromExternalHost(const std::string& message,
const std::string& origin,
const std::string& target) {
if (!is_valid())
return;
sender_->Send(
new AutomationMsg_HandleMessageFromExternalHost(
handle_, message, origin, target));
}
#endif // defined(OS_WIN)
bool TabProxy::PrintAsync() {
if (!is_valid())
return false;
return sender_->Send(new AutomationMsg_PrintAsync(handle_));
}
bool TabProxy::WaitForInfoBarCount(size_t target_count) {
if (!is_valid())
return false;
bool success = false;
return sender_->Send(new AutomationMsg_WaitForInfoBarCount(
handle_, target_count, &success)) && success;
}
bool TabProxy::OverrideEncoding(const std::string& encoding) {
if (!is_valid())
return false;
bool succeeded = false;
sender_->Send(new AutomationMsg_OverrideEncoding(handle_, encoding,
&succeeded));
return succeeded;
}
#if defined(OS_WIN)
void TabProxy::Reposition(HWND window, HWND window_insert_after, int left,
int top, int width, int height, int flags,
HWND parent_window) {
Reposition_Params params;
params.window = window;
params.window_insert_after = window_insert_after;
params.left = left;
params.top = top;
params.width = width;
params.height = height;
params.flags = flags;
params.set_parent = (::IsWindow(parent_window) ? true : false);
params.parent_window = parent_window;
sender_->Send(new AutomationMsg_TabReposition(handle_, params));
}
void TabProxy::SendContextMenuCommand(int selected_command) {
sender_->Send(new AutomationMsg_ForwardContextMenuCommandToChrome(
handle_, selected_command));
}
void TabProxy::OnHostMoved() {
sender_->Send(new AutomationMsg_BrowserMove(handle_));
}
#endif // defined(OS_WIN)
void TabProxy::SelectAll() {
sender_->Send(new AutomationMsg_SelectAll(handle_));
}
void TabProxy::Cut() {
sender_->Send(new AutomationMsg_Cut(handle_));
}
void TabProxy::Copy() {
sender_->Send(new AutomationMsg_Copy(handle_));
}
void TabProxy::Paste() {
sender_->Send(new AutomationMsg_Paste(handle_));
}
void TabProxy::ReloadAsync() {
sender_->Send(new AutomationMsg_ReloadAsync(handle_));
}
void TabProxy::StopAsync() {
sender_->Send(new AutomationMsg_StopAsync(handle_));
}
void TabProxy::SaveAsAsync() {
sender_->Send(new AutomationMsg_SaveAsAsync(handle_));
}
void TabProxy::JavaScriptStressTestControl(int cmd, int param) {
if (!is_valid())
return;
sender_->Send(new AutomationMsg_JavaScriptStressTestControl(
handle_, cmd, param));
}
void TabProxy::AddObserver(TabProxyDelegate* observer) {
base::AutoLock lock(list_lock_);
observers_list_.AddObserver(observer);
}
void TabProxy::RemoveObserver(TabProxyDelegate* observer) {
base::AutoLock lock(list_lock_);
observers_list_.RemoveObserver(observer);
}
// Called on Channel background thread, if TabMessages filter is installed.
bool TabProxy::OnMessageReceived(const IPC::Message& message) {
base::AutoLock lock(list_lock_);
FOR_EACH_OBSERVER(TabProxyDelegate, observers_list_,
OnMessageReceived(this, message));
return true;
}
void TabProxy::OnChannelError() {
base::AutoLock lock(list_lock_);
FOR_EACH_OBSERVER(TabProxyDelegate, observers_list_, OnChannelError(this));
}
TabProxy::~TabProxy() {}
void TabProxy::FirstObjectAdded() {
AddRef();
}
void TabProxy::LastObjectRemoved() {
Release();
}