blob: a11faad77c1c5f996dc8d90a8e9a5dda7a414d8a [file] [log] [blame]
// Copyright 2023 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_ISOLATED_WORLD_MANAGER_H_
#define EXTENSIONS_RENDERER_ISOLATED_WORLD_MANAGER_H_
#include <map>
#include <optional>
#include <string>
#include "base/sequence_checker.h"
#include "extensions/common/mojom/execution_world.mojom.h"
#include "url/gurl.h"
class InjectionHost;
namespace extensions {
// A class responsible for managing the creation and maintenance of isolated
// worlds related to extensions APIs.
//
// There is a single instance of this class (per renderer), retrieved via
// `IsolatedWorldManager::GetInstance()`.
//
// This class is *not* thread-safe and is only designed to be accessed from the
// main renderer thread; this is not an issue because all
// script-injection-related functionality happens on the main thread (e.g.,
// there is no situation in which an extension service worker directly accesses
// isolated worlds).
//
// A note on Host IDs: Host IDs are unique identifiers for the entity causing
// the injection. This is *usually* an extension, but can also be WebUI in the
// case of WebUI injecting into an embedded <webview>. The Host ID for an
// extension is the extension's ID.
class IsolatedWorldManager {
public:
IsolatedWorldManager();
IsolatedWorldManager(const IsolatedWorldManager&) = delete;
IsolatedWorldManager& operator=(const IsolatedWorldManager&) = delete;
~IsolatedWorldManager();
// Returns the shared instance of the isolated world manager.
static IsolatedWorldManager& GetInstance();
// Returns the id of the injection host associated with the given `world_id`
// or an empty string if none exists.
std::string GetHostIdForIsolatedWorld(int world_id);
// Returns the execution world for the given `world_id`, if any exists.
std::optional<mojom::ExecutionWorld> GetExecutionWorldForIsolatedWorld(
int world_id);
// Removes all isolated worlds associated with the given `host_id`, if any
// exist.
void RemoveIsolatedWorlds(const std::string& host_id);
// Sets properties for newly-created user script worlds for the associated
// `host_id`. Does not affect any already-created worlds, but does update the
// world data stored in blink so that any newly-created worlds will be
// properly initialized.
void SetUserScriptWorldProperties(const std::string& host_id,
const std::optional<std::string>& world_id,
std::optional<std::string> csp,
bool enable_messaging);
// Clears any properties associated with the given `host_id` and `world_id`.
// Note this does *not* update any existing worlds.
void ClearUserScriptWorldProperties(
const std::string& host_id,
const std::optional<std::string>& world_id);
// Returns whether messaging APIs should be enabled in worlds for the given
// `host_id`.
bool IsMessagingEnabledInUserScriptWorld(int blink_world_id);
// Returns the id of the isolated world associated with the given
// `injection_host`. If none exists, creates a new world for it associated
// with the host's name and CSP.
int GetOrCreateIsolatedWorldForHost(
const InjectionHost& injection_host,
mojom::ExecutionWorld execution_world,
const std::optional<std::string>& world_id);
// Returns true if `blink_world_id` corresponds to the ID for an extension
// isolated world (either a content script world or user script world).
static bool IsExtensionIsolatedWorld(int blink_world_id);
private:
// A structure to track existing isolated world information, where an
// isolated world is any world other than the main world (including both
// "user script" worlds and default "isolated" worlds from extension content
// scripts). Presence in this map indicates there has been a world (i.e., v8
// context) created at the blink layer at least once in this process.
struct IsolatedWorldInfo {
// Defaults, to appease the Chromium clang plugin.
IsolatedWorldInfo();
~IsolatedWorldInfo();
IsolatedWorldInfo(IsolatedWorldInfo&&);
// The integer ID of the world, used to reference the world with the
// blink layer.
int blink_world_id;
// The id of the injection host the world is associated with. For
// extensions, this is the extension ID.
std::string host_id;
// The execution world for the isolated world. Currently, this is restricted
// to mojom::ExecutionWorld::kIsolated.
mojom::ExecutionWorld execution_world;
// An optional extension-provided ID for the world. If omitted, refers to
// the "default" world of the given `exection_world` type.
std::optional<std::string> world_id;
// A human-friendly name for the isolated world host.
std::string name;
// The URL associated with the isolated world host.
GURL url;
// CSP to use for the isolated world, if any.
std::optional<std::string> csp;
// Whether messaging is enabled within this isolated world.
bool enable_messaging;
};
// A set of data to store properties for user script worlds. There may or may
// not be a created world at the blink layer (i.e., a v8 context) for these
// entries.
struct PendingWorldInfo {
PendingWorldInfo();
~PendingWorldInfo();
PendingWorldInfo(PendingWorldInfo&&);
// The CSP to use for newly-created isolated worlds, if any.
std::optional<std::string> csp;
// Whether to enable messaging APIs in newly-created isolated worlds.
bool enable_messaging = false;
};
// Finds the stored `IsolatedWorldInfo` for the given `host_id`,
// `execution_world`, and `world_id`, if any.
IsolatedWorldInfo* FindIsolatedWorldInfo(
const std::string& host_id,
mojom::ExecutionWorld execution_world,
const std::optional<std::string>& world_id);
// Finds the stored `PendingWorldInfo` for the given `host_id`,
// `execution_world`, and `world_id`, if any.
PendingWorldInfo* FindPendingWorldInfo(
const std::string& host_id,
const std::optional<std::string>& world_id);
void UpdateBlinkIsolatedWorldInfo(int world_id,
const IsolatedWorldInfo& world_info);
// A map between the isolated world ID and injection host ID.
using IsolatedWorldMap = std::map<int, IsolatedWorldInfo>;
IsolatedWorldMap isolated_worlds_;
// A map of configuration properties for user script worlds. A user script
// world is unique based on the pairing of <host_id, world_id>, where the
// world ID may be optional (in which case, it indicates the default world of
// that type).
using PendingWorldKey = std::pair<std::string, std::optional<std::string>>;
std::map<PendingWorldKey, PendingWorldInfo> pending_worlds_info_;
SEQUENCE_CHECKER(sequence_checker_);
};
} // namespace extensions
#endif // EXTENSIONS_RENDERER_ISOLATED_WORLD_MANAGER_H_