blob: 6b27e464890c05e11f849640d07b5168566db35f [file] [log] [blame]
// 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/browser/automation/automation_provider_json.h"
#include "base/json/json_writer.h"
#include "base/json/string_escape.h"
#include "base/values.h"
#include "chrome/browser/autocomplete/autocomplete_match.h"
#include "chrome/browser/automation/automation_provider.h"
#include "chrome/browser/automation/automation_util.h"
#include "chrome/browser/extensions/extension_service.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/common/automation_id.h"
#include "chrome/common/automation_messages.h"
#include "chrome/common/extensions/extension.h"
#include "content/public/browser/web_contents.h"
using automation::Error;
using automation::ErrorCode;
using content::WebContents;
AutomationJSONReply::AutomationJSONReply(AutomationProvider* provider,
IPC::Message* reply_message)
: provider_(provider),
message_(reply_message) {
}
AutomationJSONReply::~AutomationJSONReply() {
DCHECK(!message_) << "JSON automation request not replied!";
}
void AutomationJSONReply::SendSuccess(const Value* value) {
DCHECK(message_) << "Resending reply for JSON automation request";
std::string json_string = "{}";
if (value)
base::JSONWriter::Write(value, &json_string);
AutomationMsg_SendJSONRequest::WriteReplyParams(
message_, json_string, true);
provider_->Send(message_);
message_ = NULL;
}
void AutomationJSONReply::SendError(const std::string& error_message) {
SendError(Error(error_message));
}
void AutomationJSONReply::SendErrorCode(ErrorCode code) {
SendError(Error(code));
}
void AutomationJSONReply::SendError(const Error& error) {
DCHECK(message_) << "Resending reply for JSON automation request";
base::DictionaryValue dict;
dict.SetString("error", error.message());
dict.SetInteger("code", error.code());
std::string json;
base::JSONWriter::Write(&dict, &json);
AutomationMsg_SendJSONRequest::WriteReplyParams(message_, json, false);
provider_->Send(message_);
message_ = NULL;
}
bool GetBrowserFromJSONArgs(
DictionaryValue* args,
Browser** browser,
std::string* error) {
if (args->HasKey("auto_id")) {
AutomationId id;
if (!GetAutomationIdFromJSONArgs(args, "auto_id", &id, error))
return false;
WebContents* tab;
if (!automation_util::GetTabForId(id, &tab)) {
*error = "'auto_id' does not refer to an open tab";
return false;
}
Browser* container = automation_util::GetBrowserForTab(tab);
if (!container) {
*error = "tab does not belong to an open browser";
return false;
}
*browser = container;
} else {
int browser_index;
if (!args->GetInteger("windex", &browser_index)) {
*error = "'windex' missing or invalid";
return false;
}
*browser = automation_util::GetBrowserAt(browser_index);
if (!*browser) {
*error = "Cannot locate browser from given index";
return false;
}
}
return true;
}
bool GetTabFromJSONArgs(
DictionaryValue* args,
WebContents** tab,
std::string* error) {
if (args->HasKey("auto_id")) {
AutomationId id;
if (!GetAutomationIdFromJSONArgs(args, "auto_id", &id, error))
return false;
if (!automation_util::GetTabForId(id, tab)) {
*error = "'auto_id' does not refer to an open tab";
return false;
}
} else {
int browser_index, tab_index;
if (!args->GetInteger("windex", &browser_index)) {
*error = "'windex' missing or invalid";
return false;
}
if (!args->GetInteger("tab_index", &tab_index)) {
*error = "'tab_index' missing or invalid";
return false;
}
*tab = automation_util::GetWebContentsAt(browser_index, tab_index);
if (!*tab) {
*error = "Cannot locate tab from given indices";
return false;
}
}
return true;
}
bool GetBrowserAndTabFromJSONArgs(
DictionaryValue* args,
Browser** browser,
WebContents** tab,
std::string* error) {
return GetBrowserFromJSONArgs(args, browser, error) &&
GetTabFromJSONArgs(args, tab, error);
}
bool GetAutomationIdFromJSONArgs(
DictionaryValue* args,
const std::string& key,
AutomationId* id,
std::string* error) {
Value* id_value;
if (!args->Get(key, &id_value)) {
*error = base::StringPrintf("Missing parameter '%s'", key.c_str());
return false;
}
return AutomationId::FromValue(id_value, id, error);
}
bool GetRenderViewFromJSONArgs(
DictionaryValue* args,
Profile* profile,
content::RenderViewHost** rvh,
std::string* error) {
Value* id_value;
if (args->Get("auto_id", &id_value)) {
AutomationId id;
if (!AutomationId::FromValue(id_value, &id, error))
return false;
if (!automation_util::GetRenderViewForId(id, profile, rvh)) {
*error = "ID does not correspond to an open view";
return false;
}
} else {
// If the render view id is not specified, check for browser/tab indices.
WebContents* tab = NULL;
if (!GetTabFromJSONArgs(args, &tab, error))
return false;
*rvh = tab->GetRenderViewHost();
}
return true;
}
namespace {
bool GetExtensionFromJSONArgsHelper(
base::DictionaryValue* args,
const std::string& key,
Profile* profile,
bool include_disabled,
const extensions::Extension** extension,
std::string* error) {
std::string id;
if (!args->GetString(key, &id)) {
*error = base::StringPrintf("Missing or invalid key: %s", key.c_str());
return false;
}
ExtensionService* service = profile->GetExtensionService();
if (!service) {
*error = "No extensions service.";
return false;
}
if (!service->GetInstalledExtension(id)) {
// The extension ID does not correspond to any extension, whether crashed
// or not.
*error = base::StringPrintf("Extension %s is not installed.",
id.c_str());
return false;
}
const extensions::Extension* installed_extension =
service->GetExtensionById(id, include_disabled);
if (!installed_extension) {
*error = "Extension is disabled or has crashed.";
return false;
}
*extension = installed_extension;
return true;
}
} // namespace
bool GetExtensionFromJSONArgs(
base::DictionaryValue* args,
const std::string& key,
Profile* profile,
const extensions::Extension** extension,
std::string* error) {
return GetExtensionFromJSONArgsHelper(
args, key, profile, true /* include_disabled */, extension, error);
}
bool GetEnabledExtensionFromJSONArgs(
base::DictionaryValue* args,
const std::string& key,
Profile* profile,
const extensions::Extension** extension,
std::string* error) {
return GetExtensionFromJSONArgsHelper(
args, key, profile, false /* include_disabled */, extension, error);
}