electron/shell/browser/api/atom_api_debugger.cc

210 lines
6.1 KiB
C++
Raw Normal View History

2016-01-22 04:57:25 +00:00
// Copyright (c) 2016 GitHub, Inc.
2016-01-21 18:22:23 +00:00
// Use of this source code is governed by the MIT license that can be
// found in the LICENSE file.
#include "shell/browser/api/atom_api_debugger.h"
2016-01-21 18:22:23 +00:00
#include <memory>
2016-01-21 18:22:23 +00:00
#include <string>
#include <utility>
2016-01-21 18:22:23 +00:00
#include "base/json/json_reader.h"
2016-01-21 18:22:23 +00:00
#include "base/json/json_writer.h"
#include "content/public/browser/devtools_agent_host.h"
#include "content/public/browser/web_contents.h"
2016-01-23 04:02:21 +00:00
#include "native_mate/dictionary.h"
#include "shell/common/native_mate_converters/callback_converter_deprecated.h"
#include "shell/common/native_mate_converters/value_converter.h"
#include "shell/common/node_includes.h"
2016-01-21 18:22:23 +00:00
using content::DevToolsAgentHost;
namespace electron {
2016-01-21 18:22:23 +00:00
namespace api {
2016-04-25 01:17:54 +00:00
Debugger::Debugger(v8::Isolate* isolate, content::WebContents* web_contents)
: content::WebContentsObserver(web_contents), web_contents_(web_contents) {
2016-04-25 01:17:54 +00:00
Init(isolate);
2016-01-21 18:22:23 +00:00
}
2018-04-18 01:55:30 +00:00
Debugger::~Debugger() {}
2016-01-21 18:22:23 +00:00
void Debugger::AgentHostClosed(DevToolsAgentHost* agent_host) {
DCHECK(agent_host == agent_host_);
agent_host_ = nullptr;
ClearPendingRequests();
Emit("detach", "target closed");
2016-01-21 18:22:23 +00:00
}
void Debugger::DispatchProtocolMessage(DevToolsAgentHost* agent_host,
const std::string& message) {
DCHECK(agent_host == agent_host_);
2016-01-21 18:22:23 +00:00
2017-04-28 17:42:01 +00:00
v8::Locker locker(isolate());
v8::HandleScope handle_scope(isolate());
std::unique_ptr<base::Value> parsed_message =
base::JSONReader::ReadDeprecated(message);
if (!parsed_message || !parsed_message->is_dict())
2017-04-28 17:42:01 +00:00
return;
base::DictionaryValue* dict =
static_cast<base::DictionaryValue*>(parsed_message.get());
2016-01-21 18:22:23 +00:00
int id;
if (!dict->GetInteger("id", &id)) {
std::string method;
if (!dict->GetString("method", &method))
return;
2016-01-22 08:47:23 +00:00
base::DictionaryValue* params_value = nullptr;
base::DictionaryValue params;
if (dict->GetDictionary("params", &params_value))
params.Swap(params_value);
Emit("message", method, params);
2016-01-21 18:22:23 +00:00
} else {
auto it = pending_requests_.find(id);
if (it == pending_requests_.end())
2016-01-21 18:22:23 +00:00
return;
2016-01-22 08:47:23 +00:00
electron::util::Promise<base::DictionaryValue> promise =
std::move(it->second);
pending_requests_.erase(it);
base::DictionaryValue* error = nullptr;
if (dict->GetDictionary("error", &error)) {
std::string message;
error->GetString("message", &message);
promise.RejectWithErrorMessage(message);
} else {
base::DictionaryValue* result_body = nullptr;
base::DictionaryValue result;
if (dict->GetDictionary("result", &result_body)) {
result.Swap(result_body);
}
promise.Resolve(result);
}
2016-01-21 18:22:23 +00:00
}
}
void Debugger::RenderFrameHostChanged(content::RenderFrameHost* old_rfh,
content::RenderFrameHost* new_rfh) {
if (agent_host_) {
agent_host_->DisconnectWebContents();
auto* web_contents = content::WebContents::FromRenderFrameHost(new_rfh);
agent_host_->ConnectWebContents(web_contents);
}
}
2016-01-21 18:22:23 +00:00
void Debugger::Attach(mate::Arguments* args) {
std::string protocol_version;
args->GetNext(&protocol_version);
if (agent_host_) {
args->ThrowError("Debugger is already attached to the target");
return;
}
2016-01-21 18:22:23 +00:00
if (!protocol_version.empty() &&
!DevToolsAgentHost::IsSupportedProtocolVersion(protocol_version)) {
args->ThrowError("Requested protocol version is not supported");
return;
}
2016-01-21 18:22:23 +00:00
agent_host_ = DevToolsAgentHost::GetOrCreateFor(web_contents_);
if (!agent_host_) {
2016-01-21 18:22:23 +00:00
args->ThrowError("No target available");
return;
}
agent_host_->AttachClient(this);
}
2016-01-22 08:47:23 +00:00
bool Debugger::IsAttached() {
return agent_host_ && agent_host_->IsAttached();
2016-01-22 08:47:23 +00:00
}
2016-01-21 18:22:23 +00:00
void Debugger::Detach() {
if (!agent_host_)
2016-01-22 04:57:25 +00:00
return;
2016-09-06 08:24:37 +00:00
agent_host_->DetachClient(this);
AgentHostClosed(agent_host_.get());
2016-01-21 18:22:23 +00:00
}
v8::Local<v8::Promise> Debugger::SendCommand(mate::Arguments* args) {
electron::util::Promise<base::DictionaryValue> promise(isolate());
v8::Local<v8::Promise> handle = promise.GetHandle();
if (!agent_host_) {
promise.RejectWithErrorMessage("No target available");
return handle;
}
2016-01-21 18:22:23 +00:00
std::string method;
if (!args->GetNext(&method)) {
promise.RejectWithErrorMessage("Invalid method");
return handle;
2016-01-21 18:22:23 +00:00
}
2016-01-21 18:22:23 +00:00
base::DictionaryValue command_params;
args->GetNext(&command_params);
base::DictionaryValue request;
int request_id = ++previous_request_id_;
pending_requests_.emplace(request_id, std::move(promise));
2016-01-21 18:22:23 +00:00
request.SetInteger("id", request_id);
request.SetString("method", method);
if (!command_params.empty())
request.Set("params",
base::Value::ToUniquePtrValue(command_params.Clone()));
2016-01-21 18:22:23 +00:00
std::string json_args;
base::JSONWriter::Write(request, &json_args);
2016-09-06 08:24:37 +00:00
agent_host_->DispatchProtocolMessage(this, json_args);
return handle;
2016-01-21 18:22:23 +00:00
}
void Debugger::ClearPendingRequests() {
for (auto& it : pending_requests_)
it.second.RejectWithErrorMessage("target closed while handling command");
}
2016-01-21 18:22:23 +00:00
// static
2018-04-18 01:55:30 +00:00
mate::Handle<Debugger> Debugger::Create(v8::Isolate* isolate,
content::WebContents* web_contents) {
2016-08-02 11:38:35 +00:00
return mate::CreateHandle(isolate, new Debugger(isolate, web_contents));
2016-01-21 18:22:23 +00:00
}
// static
void Debugger::BuildPrototype(v8::Isolate* isolate,
v8::Local<v8::FunctionTemplate> prototype) {
2016-08-02 10:28:12 +00:00
prototype->SetClassName(mate::StringToV8(isolate, "Debugger"));
mate::ObjectTemplateBuilder(isolate, prototype->PrototypeTemplate())
2016-01-21 18:22:23 +00:00
.SetMethod("attach", &Debugger::Attach)
2016-01-22 08:47:23 +00:00
.SetMethod("isAttached", &Debugger::IsAttached)
2016-01-21 18:22:23 +00:00
.SetMethod("detach", &Debugger::Detach)
2016-01-22 04:57:25 +00:00
.SetMethod("sendCommand", &Debugger::SendCommand);
2016-01-21 18:22:23 +00:00
}
} // namespace api
} // namespace electron
2016-01-23 04:02:21 +00:00
namespace {
using electron::api::Debugger;
2016-08-02 11:38:35 +00:00
2018-04-18 01:55:30 +00:00
void Initialize(v8::Local<v8::Object> exports,
v8::Local<v8::Value> unused,
v8::Local<v8::Context> context,
void* priv) {
2016-01-23 04:02:21 +00:00
v8::Isolate* isolate = context->GetIsolate();
2016-08-02 11:38:35 +00:00
mate::Dictionary(isolate, exports)
.Set("Debugger", Debugger::GetConstructor(isolate)
->GetFunction(context)
.ToLocalChecked());
2016-01-23 04:02:21 +00:00
}
} // namespace
2019-02-27 19:14:23 +00:00
NODE_LINKED_MODULE_CONTEXT_AWARE(atom_browser_debugger, Initialize)