#include "extensions/renderer/api/messaging/one_time_message_handler.h"
#include <map>
#include "base/containers/contains.h"
#include "base/containers/cxx20_erase.h"
#include "base/functional/bind.h"
#include "base/functional/callback.h"
#include "base/ranges/algorithm.h"
#include "base/supports_user_data.h"
#include "content/public/renderer/render_frame.h"
#include "extensions/common/api/messaging/channel_type.h"
#include "extensions/common/api/messaging/message.h"
#include "extensions/common/api/messaging/port_id.h"
#include "extensions/common/mojom/event_dispatcher.mojom.h"
#include "extensions/renderer/api/messaging/message_target.h"
#include "extensions/renderer/api/messaging/messaging_util.h"
#include "extensions/renderer/bindings/api_binding_types.h"
#include "extensions/renderer/bindings/api_binding_util.h"
#include "extensions/renderer/bindings/api_bindings_system.h"
#include "extensions/renderer/bindings/api_event_handler.h"
#include "extensions/renderer/bindings/api_request_handler.h"
#include "extensions/renderer/bindings/get_per_context_data.h"
#include "extensions/renderer/gc_callback.h"
#include "extensions/renderer/ipc_message_sender.h"
#include "extensions/renderer/native_extension_bindings_system.h"
#include "extensions/renderer/script_context.h"
#include "gin/arguments.h"
#include "gin/dictionary.h"
#include "gin/handle.h"
#include "gin/per_context_data.h"
#include "ipc/ipc_message.h"
#include "v8/include/v8-container.h"
#include "v8/include/v8-exception.h"
#include "v8/include/v8-external.h"
#include "v8/include/v8-function-callback.h"
#include "v8/include/v8-function.h"
#include "v8/include/v8-isolate.h"
#include "v8/include/v8-object.h"
#include "v8/include/v8-persistent-handle.h"
#include "v8/include/v8-primitive.h"
namespace extensions {
namespace {
struct OneTimeOpener {
int request_id = -1;
int routing_id = MSG_ROUTING_NONE;
binding::AsyncResponseType async_type = binding::AsyncResponseType::kNone;
};
struct OneTimeReceiver {
int routing_id = MSG_ROUTING_NONE;
std::string event_name;
v8::Global<v8::Object> sender;
};
using OneTimeMessageCallback =
base::OnceCallback<void(gin::Arguments* arguments)>;
struct OneTimeMessageContextData : public base::SupportsUserData::Data {
static constexpr char kPerContextDataKey[] =
"extension_one_time_message_context_data";
std::map<PortId, OneTimeOpener> openers;
std::map<PortId, OneTimeReceiver> receivers;
std::vector<std::unique_ptr<OneTimeMessageCallback>> pending_callbacks;
};
constexpr char OneTimeMessageContextData::kPerContextDataKey[];
int RoutingIdForScriptContext(ScriptContext* script_context) {
content::RenderFrame* render_frame = script_context->GetRenderFrame();
return render_frame ? render_frame->GetRoutingID() : MSG_ROUTING_NONE;
}
void OneTimeMessageResponseHelper(
const v8::FunctionCallbackInfo<v8::Value>& info) {
CHECK(info.Data()->IsExternal());
gin::Arguments arguments(info);
v8::Isolate* isolate = arguments.isolate();
v8::HandleScope handle_scope(isolate);
v8::Local<v8::Context> context = isolate->GetCurrentContext();
OneTimeMessageContextData* data =
GetPerContextData<OneTimeMessageContextData>(context,
kDontCreateIfMissing);
if (!data)
return;
v8::Local<v8::External> external = info.Data().As<v8::External>();
auto* raw_callback = static_cast<OneTimeMessageCallback*>(external->Value());
auto iter = base::ranges::find(data->pending_callbacks, raw_callback,
&std::unique_ptr<OneTimeMessageCallback>::get);
if (iter == data->pending_callbacks.end())
return;
std::unique_ptr<OneTimeMessageCallback> callback = std::move(*iter);
data->pending_callbacks.erase(iter);
std::move(*callback).Run(&arguments);
}
bool WillListenerReplyAsync(absl::optional<base::Value> result) {
if (!result)
return false;
if (const base::Value::Dict* dict = result->GetIfDict()) {
if (const base::Value::List* list = dict->FindList("results")) {
for (const base::Value& value : *list) {
if (value.is_bool() && value.GetBool())
return true;
}
}
}
return false;
}
}
OneTimeMessageHandler::OneTimeMessageHandler(
NativeExtensionBindingsSystem* bindings_system)
: bindings_system_(bindings_system) {}
OneTimeMessageHandler::~OneTimeMessageHandler() = default;
bool OneTimeMessageHandler::HasPort(ScriptContext* script_context,
const PortId& port_id) {
v8::Isolate* isolate = script_context->isolate();
v8::HandleScope handle_scope(isolate);
OneTimeMessageContextData* data =
GetPerContextData<OneTimeMessageContextData>(script_context->v8_context(),
kDontCreateIfMissing);
if (!data)
return false;
return port_id.is_opener ? base::Contains(data->openers, port_id)
: base::Contains(data->receivers, port_id);
}
v8::Local<v8::Promise> OneTimeMessageHandler::SendMessage(
ScriptContext* script_context,
const PortId& new_port_id,
const MessageTarget& target,
ChannelType channel_type,
const Message& message,
binding::AsyncResponseType async_type,
v8::Local<v8::Function> response_callback) {
v8::Isolate* isolate = script_context->isolate();
v8::EscapableHandleScope handle_scope(isolate);
DCHECK(new_port_id.is_opener);
DCHECK_EQ(script_context->context_id(), new_port_id.context_id);
OneTimeMessageContextData* data =
GetPerContextData<OneTimeMessageContextData>(script_context->v8_context(),
kCreateIfMissing);
DCHECK(data);
v8::Local<v8::Promise> promise;
bool wants_response = async_type != binding::AsyncResponseType::kNone;
int routing_id = RoutingIdForScriptContext(script_context);
if (wants_response) {
if (async_type == binding::AsyncResponseType::kPromise)
DCHECK(response_callback.IsEmpty());
APIRequestHandler::RequestDetails details =
bindings_system_->api_system()->request_handler()->AddPendingRequest(
script_context->v8_context(), async_type, response_callback,
binding::ResultModifierFunction());
OneTimeOpener& port = data->openers[new_port_id];
port.request_id = details.request_id;
port.routing_id = routing_id;
port.async_type = async_type;
promise = details.promise;
DCHECK_EQ(async_type == binding::AsyncResponseType::kPromise,
!promise.IsEmpty());
}
IPCMessageSender* ipc_sender = bindings_system_->GetIPCMessageSender();
std::string channel_name;
switch (channel_type) {
case ChannelType::kSendRequest:
channel_name = messaging_util::kSendRequestChannel;
break;
case ChannelType::kSendMessage:
channel_name = messaging_util::kSendMessageChannel;
break;
case ChannelType::kNative:
break;
case ChannelType::kConnect:
NOTREACHED_NORETURN();
}
ipc_sender->SendOpenMessageChannel(script_context, new_port_id, target,
channel_type, channel_name);
ipc_sender->SendPostMessageToPort(new_port_id, message);
if (!wants_response && target.type != MessageTarget::NATIVE_APP) {
bool close_channel = true;
ipc_sender->SendCloseMessagePort(routing_id, new_port_id, close_channel);
}
return handle_scope.Escape(promise);
}
void OneTimeMessageHandler::AddReceiver(ScriptContext* script_context,
const PortId& target_port_id,
v8::Local<v8::Object> sender,
const std::string& event_name) {
DCHECK(!target_port_id.is_opener);
DCHECK_NE(script_context->context_id(), target_port_id.context_id);
v8::Isolate* isolate = script_context->isolate();
v8::HandleScope handle_scope(isolate);
v8::Local<v8::Context> context = script_context->v8_context();
OneTimeMessageContextData* data =
GetPerContextData<OneTimeMessageContextData>(context, kCreateIfMissing);
DCHECK(data);
DCHECK(!base::Contains(data->receivers, target_port_id));
OneTimeReceiver& receiver = data->receivers[target_port_id];
receiver.sender.Reset(isolate, sender);
receiver.routing_id = RoutingIdForScriptContext(script_context);
receiver.event_name = event_name;
}
bool OneTimeMessageHandler::DeliverMessage(ScriptContext* script_context,
const Message& message,
const PortId& target_port_id) {
v8::Isolate* isolate = script_context->isolate();
v8::HandleScope handle_scope(isolate);
return target_port_id.is_opener
? DeliverReplyToOpener(script_context, message, target_port_id)
: DeliverMessageToReceiver(script_context, message,
target_port_id);
}
bool OneTimeMessageHandler::Disconnect(ScriptContext* script_context,
const PortId& port_id,
const std::string& error_message) {
v8::Isolate* isolate = script_context->isolate();
v8::HandleScope handle_scope(isolate);
return port_id.is_opener
? DisconnectOpener(script_context, port_id, error_message)
: DisconnectReceiver(script_context, port_id);
}
int OneTimeMessageHandler::GetPendingCallbackCountForTest(
ScriptContext* script_context) {
v8::Isolate* isolate = script_context->isolate();
v8::HandleScope handle_scope(isolate);
OneTimeMessageContextData* data =
GetPerContextData<OneTimeMessageContextData>(script_context->v8_context(),
kDontCreateIfMissing);
return data ? data->pending_callbacks.size() : 0;
}
bool OneTimeMessageHandler::DeliverMessageToReceiver(
ScriptContext* script_context,
const Message& message,
const PortId& target_port_id) {
DCHECK(!target_port_id.is_opener);
v8::Isolate* isolate = script_context->isolate();
v8::Local<v8::Context> context = script_context->v8_context();
bool handled = false;
OneTimeMessageContextData* data =
GetPerContextData<OneTimeMessageContextData>(context,
kDontCreateIfMissing);
if (!data)
return handled;
auto iter = data->receivers.find(target_port_id);
if (iter == data->receivers.end())
return handled;
handled = true;
OneTimeReceiver& port = iter->second;
auto callback = std::make_unique<OneTimeMessageCallback>(
base::BindOnce(&OneTimeMessageHandler::OnOneTimeMessageResponse,
weak_factory_.GetWeakPtr(), target_port_id));
v8::Local<v8::External> external = v8::External::New(isolate, callback.get());
v8::Local<v8::Function> response_function;
if (!v8::Function::New(context, &OneTimeMessageResponseHelper, external)
.ToLocal(&response_function)) {
NOTREACHED();
return handled;
}
new GCCallback(
script_context, response_function,
base::BindOnce(&OneTimeMessageHandler::OnResponseCallbackCollected,
weak_factory_.GetWeakPtr(), script_context, target_port_id,
reinterpret_cast<CallbackID>(callback.get())),
base::OnceClosure());
v8::HandleScope handle_scope(isolate);
v8::Local<v8::Value> v8_message =
messaging_util::MessageToV8(context, message);
v8::Local<v8::Object> v8_sender = port.sender.Get(isolate);
std::vector<v8::Local<v8::Value>> args = {v8_message, v8_sender,
response_function};
JSRunner::ResultCallback dispatch_callback;
if (port.event_name == messaging_util::kOnMessageEvent) {
dispatch_callback =
base::BindOnce(&OneTimeMessageHandler::OnEventFired,
weak_factory_.GetWeakPtr(), target_port_id);
}
data->pending_callbacks.push_back(std::move(callback));
bindings_system_->api_system()->event_handler()->FireEventInContext(
port.event_name, context, &args, nullptr, std::move(dispatch_callback));
return handled;
}
bool OneTimeMessageHandler::DeliverReplyToOpener(ScriptContext* script_context,
const Message& message,
const PortId& target_port_id) {
DCHECK(target_port_id.is_opener);
v8::Local<v8::Context> v8_context = script_context->v8_context();
bool handled = false;
OneTimeMessageContextData* data =
GetPerContextData<OneTimeMessageContextData>(v8_context,
kDontCreateIfMissing);
if (!data)
return handled;
auto iter = data->openers.find(target_port_id);
if (iter == data->openers.end())
return handled;
handled = true;
const OneTimeOpener port = iter->second;
DCHECK_NE(-1, port.request_id);
data->openers.erase(iter);
v8::Local<v8::Value> v8_message =
messaging_util::MessageToV8(v8_context, message);
std::vector<v8::Local<v8::Value>> args = {v8_message};
bindings_system_->api_system()->request_handler()->CompleteRequest(
port.request_id, args, std::string());
bool close_channel = true;
bindings_system_->GetIPCMessageSender()->SendCloseMessagePort(
port.routing_id, target_port_id, close_channel);
return handled;
}
bool OneTimeMessageHandler::DisconnectReceiver(ScriptContext* script_context,
const PortId& port_id) {
v8::Local<v8::Context> context = script_context->v8_context();
bool handled = false;
OneTimeMessageContextData* data =
GetPerContextData<OneTimeMessageContextData>(context,
kDontCreateIfMissing);
if (!data)
return handled;
auto iter = data->receivers.find(port_id);
if (iter == data->receivers.end())
return handled;
handled = true;
data->receivers.erase(iter);
return handled;
}
bool OneTimeMessageHandler::DisconnectOpener(ScriptContext* script_context,
const PortId& port_id,
const std::string& error_message) {
bool handled = false;
v8::Local<v8::Context> v8_context = script_context->v8_context();
OneTimeMessageContextData* data =
GetPerContextData<OneTimeMessageContextData>(v8_context,
kDontCreateIfMissing);
if (!data)
return handled;
auto iter = data->openers.find(port_id);
if (iter == data->openers.end())
return handled;
handled = true;
const OneTimeOpener opener = iter->second;
DCHECK_NE(-1, opener.request_id);
data->openers.erase(iter);
std::string error;
if (!error_message.empty()) {
error = error_message;
} else if (opener.async_type == binding::AsyncResponseType::kCallback) {
error = "The message port closed before a response was received.";
}
bindings_system_->api_system()->request_handler()->CompleteRequest(
opener.request_id, std::vector<v8::Local<v8::Value>>(), error);
return handled;
}
void OneTimeMessageHandler::OnOneTimeMessageResponse(
const PortId& port_id,
gin::Arguments* arguments) {
v8::Isolate* isolate = arguments->isolate();
v8::Local<v8::Context> context = isolate->GetCurrentContext();
OneTimeMessageContextData* data =
GetPerContextData<OneTimeMessageContextData>(context,
kDontCreateIfMissing);
if (!data)
return;
auto iter = data->receivers.find(port_id);
if (iter == data->receivers.end())
return;
int routing_id = iter->second.routing_id;
data->receivers.erase(iter);
v8::Local<v8::Value> value;
if (arguments->Length() > 0)
CHECK(arguments->GetNext(&value));
else
value = v8::Undefined(isolate);
std::string error;
std::unique_ptr<Message> message = messaging_util::MessageFromV8(
context, value, port_id.serialization_format, &error);
if (!message) {
arguments->ThrowTypeError(error);
return;
}
IPCMessageSender* ipc_sender = bindings_system_->GetIPCMessageSender();
ipc_sender->SendPostMessageToPort(port_id, *message);
bool close_channel = true;
ipc_sender->SendCloseMessagePort(routing_id, port_id, close_channel);
}
void OneTimeMessageHandler::OnResponseCallbackCollected(
ScriptContext* script_context,
const PortId& port_id,
CallbackID raw_callback) {
v8::HandleScope handle_scope(script_context->isolate());
OneTimeMessageContextData* data =
GetPerContextData<OneTimeMessageContextData>(script_context->v8_context(),
kDontCreateIfMissing);
if (!data)
return;
auto iter = data->receivers.find(port_id);
if (iter == data->receivers.end())
return;
int routing_id = iter->second.routing_id;
data->receivers.erase(iter);
base::EraseIf(
data->pending_callbacks,
[raw_callback](const std::unique_ptr<OneTimeMessageCallback>& callback) {
return reinterpret_cast<CallbackID>(callback.get()) == raw_callback;
});
IPCMessageSender* ipc_sender = bindings_system_->GetIPCMessageSender();
bool close_channel = false;
ipc_sender->SendCloseMessagePort(routing_id, port_id, close_channel);
}
void OneTimeMessageHandler::OnEventFired(const PortId& port_id,
v8::Local<v8::Context> context,
absl::optional<base::Value> result) {
OneTimeMessageContextData* data =
GetPerContextData<OneTimeMessageContextData>(context,
kDontCreateIfMissing);
if (!data)
return;
auto iter = data->receivers.find(port_id);
if (iter == data->receivers.end())
return;
int routing_id = iter->second.routing_id;
IPCMessageSender* ipc_sender = bindings_system_->GetIPCMessageSender();
if (WillListenerReplyAsync(std::move(result))) {
ipc_sender->SendMessageResponsePending(routing_id, port_id);
return;
}
data->receivers.erase(iter);
bool close_channel = false;
ipc_sender->SendCloseMessagePort(routing_id, port_id, close_channel);
}
}