blob: b9c2ed1a355bf48491c8d3371cd884fff53456d7 [file] [log] [blame]
// Copyright 2016 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef EXTENSIONS_RENDERER_BINDINGS_API_REQUEST_HANDLER_H_
#define EXTENSIONS_RENDERER_BINDINGS_API_REQUEST_HANDLER_H_
#include <map>
#include <memory>
#include <optional>
#include <set>
#include "base/functional/callback.h"
#include "base/memory/raw_ptr.h"
#include "base/values.h"
#include "extensions/common/mojom/extra_response_data.mojom.h"
#include "extensions/renderer/bindings/api_binding_types.h"
#include "extensions/renderer/bindings/api_last_error.h"
#include "extensions/renderer/bindings/interaction_provider.h"
#include "v8/include/v8.h"
namespace extensions {
class APIResponseValidator;
class ExceptionHandler;
// A wrapper around a map for extension API calls. Contains all pending requests
// and the associated context and callback. Designed to be used on a single
// thread, but amongst multiple contexts.
class APIRequestHandler {
public:
// TODO(devlin): We may want to coalesce this with the
// ExtensionHostMsg_Request_Params IPC struct.
struct Request {
Request();
Request(const Request&) = delete;
Request& operator=(const Request&) = delete;
~Request();
int request_id = -1;
std::string method_name;
bool has_async_response_handler = false;
bool has_user_gesture = false;
base::Value::List arguments_list;
};
// Details about a newly-added request to provide as a return to callers.
// Contains the id of the request and if this is a promise based request, the
// associated promise.
struct RequestDetails {
RequestDetails(int request_id, v8::Local<v8::Promise> promise);
~RequestDetails();
RequestDetails(const RequestDetails& other);
const int request_id;
v8::Local<v8::Promise> promise;
};
using SendRequestMethod =
base::RepeatingCallback<void(std::unique_ptr<Request>,
v8::Local<v8::Context>)>;
APIRequestHandler(SendRequestMethod send_request,
APILastError last_error,
ExceptionHandler* exception_handler,
const InteractionProvider* interaction_provider);
APIRequestHandler(const APIRequestHandler&) = delete;
APIRequestHandler& operator=(const APIRequestHandler&) = delete;
~APIRequestHandler();
// Begins the process of processing the request. If this is a promise based
// request returns the associated promise, otherwise returns an empty promise.
v8::Local<v8::Promise> StartRequest(
v8::Local<v8::Context> context,
const std::string& method,
base::Value::List arguments_list,
binding::AsyncResponseType async_type,
v8::Local<v8::Function> callback,
v8::Local<v8::Function> custom_callback,
binding::ResultModifierFunction result_modifier);
// Adds a pending request for the request handler to manage (and complete via
// CompleteRequest). This is used by renderer-side implementations that
// shouldn't be dispatched to the browser in the normal flow, but means other
// classes don't have to worry about context invalidation. Returns the details
// of the newly-added request.
// Note: Unlike StartRequest(), this will not track user gesture state.
RequestDetails AddPendingRequest(
v8::Local<v8::Context> context,
binding::AsyncResponseType async_type,
v8::Local<v8::Function> callback,
binding::ResultModifierFunction result_modifier);
// Responds to the request with the given `request_id`, calling the callback
// with the given `response` arguments.
// Invalid ids are ignored.
// Warning: This can run arbitrary JS code, so the `context` may be
// invalidated after this!
void CompleteRequest(int request_id,
const base::Value::List& response_list,
const std::string& error,
mojom::ExtraResponseDataPtr extra_data = nullptr);
void CompleteRequest(int request_id,
const v8::LocalVector<v8::Value>& response,
const std::string& error);
// Invalidates any requests that are associated with `context`.
void InvalidateContext(v8::Local<v8::Context> context);
void SetResponseValidator(std::unique_ptr<APIResponseValidator> validator);
APILastError* last_error() { return &last_error_; }
int last_sent_request_id() const { return last_sent_request_id_; }
bool has_response_validator_for_testing() const {
return response_validator_.get() != nullptr;
}
std::set<int> GetPendingRequestIdsForTesting() const;
private:
class ArgumentAdapter;
class AsyncResultHandler;
struct PendingRequest {
PendingRequest(
v8::Isolate* isolate,
v8::Local<v8::Context> context,
const std::string& method_name,
std::unique_ptr<AsyncResultHandler> async_handler,
std::unique_ptr<InteractionProvider::Token> user_gesture_token);
~PendingRequest();
PendingRequest(PendingRequest&&);
PendingRequest& operator=(PendingRequest&&);
raw_ptr<v8::Isolate> isolate;
v8::Global<v8::Context> context;
std::string method_name;
std::unique_ptr<AsyncResultHandler> async_handler;
// Note: We can't use std::optional here for derived Token instances.
std::unique_ptr<InteractionProvider::Token> user_gesture_token;
};
// Returns the next request ID to be used.
int GetNextRequestId();
// Creates and returns an AsyncResultHandler for a request if the request
// requires an asynchronous response, otherwise returns null. Also populates
// `promise_out` with the associated promise if this is a promise based
// request.
std::unique_ptr<AsyncResultHandler> GetAsyncResultHandler(
v8::Local<v8::Context> context,
binding::AsyncResponseType async_type,
v8::Local<v8::Function> callback,
v8::Local<v8::Function> custom_callback,
binding::ResultModifierFunction result_modifier,
v8::Local<v8::Promise>* promise_out);
// Common implementation for completing a request.
void CompleteRequestImpl(int request_id,
ArgumentAdapter arguments,
const std::string& error);
// The next available request identifier.
int next_request_id_ = 0;
// The id of the last request we sent to the browser. This can be used as a
// flag for whether or not a request was sent (if the last_sent_request_id_
// changes).
int last_sent_request_id_ = -1;
// A map of all pending requests.
std::map<int, PendingRequest> pending_requests_;
SendRequestMethod send_request_;
APILastError last_error_;
// The exception handler for the bindings system; guaranteed to be valid
// during this object's lifetime.
const raw_ptr<ExceptionHandler> exception_handler_;
// The response validator used to check the responses for resolved requests.
// Null if response validation is disabled.
std::unique_ptr<APIResponseValidator> response_validator_;
// Outlives `this`.
const raw_ptr<const InteractionProvider, DanglingUntriaged>
interaction_provider_;
};
} // namespace extensions
#endif // EXTENSIONS_RENDERER_BINDINGS_API_REQUEST_HANDLER_H_