// Copyright (c) 2013 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 "content/browser/devtools/devtools_protocol.h"
#include "base/json/json_reader.h"
#include "base/json/json_writer.h"
#include "base/strings/stringprintf.h"
namespace content {
namespace {
const char kIdParam[] = "id";
const char kMethodParam[] = "method";
const char kParamsParam[] = "params";
const char kResultParam[] = "result";
const char kErrorParam[] = "error";
const char kErrorCodeParam[] = "code";
const char kErrorMessageParam[] = "message";
const int kNoId = -1;
// JSON RPC 2.0 spec: http://www.jsonrpc.org/specification#error_object
enum Error {
kErrorParseError = -32700,
kErrorInvalidRequest = -32600,
kErrorNoSuchMethod = -32601,
kErrorInvalidParams = -32602,
kErrorInternalError = -32603,
kErrorServerError = -32000
};
} // namespace
using base::Value;
DevToolsProtocol::Message::~Message() {
}
DevToolsProtocol::Message::Message(const std::string& method,
base::DictionaryValue* params)
: method_(method),
params_(params) {
size_t pos = method.find(".");
if (pos != std::string::npos && pos > 0)
domain_ = method.substr(0, pos);
}
DevToolsProtocol::Command::~Command() {
}
std::string DevToolsProtocol::Command::Serialize() {
base::DictionaryValue command;
command.SetInteger(kIdParam, id_);
command.SetString(kMethodParam, method_);
if (params_)
command.Set(kParamsParam, params_->DeepCopy());
std::string json_command;
base::JSONWriter::Write(&command, &json_command);
return json_command;
}
scoped_refptr<DevToolsProtocol::Response>
DevToolsProtocol::Command::SuccessResponse(base::DictionaryValue* result) {
return new DevToolsProtocol::Response(id_, result);
}
scoped_refptr<DevToolsProtocol::Response>
DevToolsProtocol::Command::InternalErrorResponse(const std::string& message) {
return new DevToolsProtocol::Response(id_, kErrorInternalError, message);
}
scoped_refptr<DevToolsProtocol::Response>
DevToolsProtocol::Command::InvalidParamResponse(const std::string& param) {
std::string message =
base::StringPrintf("Missing or invalid '%s' parameter", param.c_str());
return new DevToolsProtocol::Response(id_, kErrorInvalidParams, message);
}
scoped_refptr<DevToolsProtocol::Response>
DevToolsProtocol::Command::NoSuchMethodErrorResponse() {
return new Response(id_, kErrorNoSuchMethod, "No such method");
}
scoped_refptr<DevToolsProtocol::Response>
DevToolsProtocol::Command::ServerErrorResponse(const std::string& message) {
return new Response(id_, kErrorServerError, message);
}
scoped_refptr<DevToolsProtocol::Response>
DevToolsProtocol::Command::AsyncResponsePromise() {
scoped_refptr<DevToolsProtocol::Response> promise =
new DevToolsProtocol::Response(0, NULL);
promise->is_async_promise_ = true;
return promise;
}
DevToolsProtocol::Command::Command(int id,
const std::string& method,
base::DictionaryValue* params)
: Message(method, params),
id_(id) {
}
DevToolsProtocol::Response::~Response() {
}
std::string DevToolsProtocol::Response::Serialize() {
base::DictionaryValue response;
if (id_ != kNoId)
response.SetInteger(kIdParam, id_);
if (error_code_) {
base::DictionaryValue* error_object = new base::DictionaryValue();
response.Set(kErrorParam, error_object);
error_object->SetInteger(kErrorCodeParam, error_code_);
if (!error_message_.empty())
error_object->SetString(kErrorMessageParam, error_message_);
} else {
if (result_)
response.Set(kResultParam, result_->DeepCopy());
else
response.Set(kResultParam, new base::DictionaryValue());
}
std::string json_response;
base::JSONWriter::Write(&response, &json_response);
return json_response;
}
DevToolsProtocol::Response::Response(int id, base::DictionaryValue* result)
: id_(id),
result_(result),
error_code_(0),
is_async_promise_(false) {
}
DevToolsProtocol::Response::Response(int id,
int error_code,
const std::string& error_message)
: id_(id),
error_code_(error_code),
error_message_(error_message),
is_async_promise_(false) {
}
DevToolsProtocol::Notification::Notification(const std::string& method,
base::DictionaryValue* params)
: Message(method, params) {
}
DevToolsProtocol::Notification::~Notification() {
}
std::string DevToolsProtocol::Notification::Serialize() {
base::DictionaryValue notification;
notification.SetString(kMethodParam, method_);
if (params_)
notification.Set(kParamsParam, params_->DeepCopy());
std::string json_notification;
base::JSONWriter::Write(¬ification, &json_notification);
return json_notification;
}
DevToolsProtocol::Handler::~Handler() {
}
scoped_refptr<DevToolsProtocol::Response>
DevToolsProtocol::Handler::HandleCommand(
scoped_refptr<DevToolsProtocol::Command> command) {
CommandHandlers::iterator it = command_handlers_.find(command->method());
if (it == command_handlers_.end())
return NULL;
return (it->second).Run(command);
}
void DevToolsProtocol::Handler::SetNotifier(const Notifier& notifier) {
notifier_ = notifier;
}
DevToolsProtocol::Handler::Handler() {
}
void DevToolsProtocol::Handler::RegisterCommandHandler(
const std::string& command,
const CommandHandler& handler) {
command_handlers_[command] = handler;
}
void DevToolsProtocol::Handler::SendNotification(
const std::string& method,
base::DictionaryValue* params) {
scoped_refptr<DevToolsProtocol::Notification> notification =
new DevToolsProtocol::Notification(method, params);
SendRawMessage(notification->Serialize());
}
void DevToolsProtocol::Handler::SendAsyncResponse(
scoped_refptr<DevToolsProtocol::Response> response) {
SendRawMessage(response->Serialize());
}
void DevToolsProtocol::Handler::SendRawMessage(const std::string& message) {
if (!notifier_.is_null())
notifier_.Run(message);
}
static bool ParseMethod(base::DictionaryValue* command,
std::string* method) {
if (!command->GetString(kMethodParam, method))
return false;
size_t pos = method->find(".");
if (pos == std::string::npos || pos == 0)
return false;
return true;
}
// static
scoped_refptr<DevToolsProtocol::Command> DevToolsProtocol::ParseCommand(
const std::string& json,
std::string* error_response) {
scoped_ptr<base::DictionaryValue> command_dict(
ParseMessage(json, error_response));
return ParseCommand(command_dict.get(), error_response);
}
// static
scoped_refptr<DevToolsProtocol::Command> DevToolsProtocol::ParseCommand(
base::DictionaryValue* command_dict,
std::string* error_response) {
if (!command_dict)
return NULL;
int id;
std::string method;
bool ok = command_dict->GetInteger(kIdParam, &id) && id >= 0;
ok = ok && ParseMethod(command_dict, &method);
if (!ok) {
scoped_refptr<Response> response =
new Response(kNoId, kErrorInvalidRequest, "No such method");
*error_response = response->Serialize();
return NULL;
}
base::DictionaryValue* params = NULL;
command_dict->GetDictionary(kParamsParam, ¶ms);
return new Command(id, method, params ? params->DeepCopy() : NULL);
}
// static
scoped_refptr<DevToolsProtocol::Command>
DevToolsProtocol::CreateCommand(
int id,
const std::string& method,
base::DictionaryValue* params) {
return new Command(id, method, params);
}
//static
scoped_refptr<DevToolsProtocol::Response>
DevToolsProtocol::ParseResponse(
base::DictionaryValue* response_dict) {
int id;
if (!response_dict->GetInteger(kIdParam, &id))
id = kNoId;
const base::DictionaryValue* error_dict;
if (response_dict->GetDictionary(kErrorParam, &error_dict)) {
int error_code = kErrorInternalError;
response_dict->GetInteger(kErrorCodeParam, &error_code);
std::string error_message;
response_dict->GetString(kErrorMessageParam, &error_message);
return new Response(id, error_code, error_message);
}
const base::DictionaryValue* result = NULL;
response_dict->GetDictionary(kResultParam, &result);
return new Response(id, result ? result->DeepCopy() : NULL);
}
// static
scoped_refptr<DevToolsProtocol::Notification>
DevToolsProtocol::ParseNotification(const std::string& json) {
scoped_ptr<base::DictionaryValue> dict(ParseMessage(json, NULL));
if (!dict)
return NULL;
std::string method;
bool ok = ParseMethod(dict.get(), &method);
if (!ok)
return NULL;
base::DictionaryValue* params = NULL;
dict->GetDictionary(kParamsParam, ¶ms);
return new Notification(method, params ? params->DeepCopy() : NULL);
}
//static
scoped_refptr<DevToolsProtocol::Notification>
DevToolsProtocol::CreateNotification(
const std::string& method,
base::DictionaryValue* params) {
return new Notification(method, params);
}
// static
base::DictionaryValue* DevToolsProtocol::ParseMessage(
const std::string& json,
std::string* error_response) {
int parse_error_code;
std::string error_message;
scoped_ptr<base::Value> message(
base::JSONReader::ReadAndReturnError(
json, 0, &parse_error_code, &error_message));
if (!message || !message->IsType(base::Value::TYPE_DICTIONARY)) {
scoped_refptr<Response> response =
new Response(0, kErrorParseError, error_message);
if (error_response)
*error_response = response->Serialize();
return NULL;
}
return static_cast<base::DictionaryValue*>(message.release());
}
} // namespace content