blob: bbf1c1a602c3f6767f201ff1b1a1cebbd8a404ff [file] [log] [blame]
// Copyright 2014 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_BROWSER_API_RUNTIME_RUNTIME_API_H_
#define EXTENSIONS_BROWSER_API_RUNTIME_RUNTIME_API_H_
#include <memory>
#include <optional>
#include <string>
#include "base/memory/raw_ptr.h"
#include "base/scoped_observation.h"
#include "base/time/time.h"
#include "base/timer/timer.h"
#include "base/values.h"
#include "extensions/browser/api/runtime/runtime_api_delegate.h"
#include "extensions/browser/browser_context_keyed_api_factory.h"
#include "extensions/browser/events/lazy_event_dispatch_util.h"
#include "extensions/browser/extension_function.h"
#include "extensions/browser/extension_registry.h"
#include "extensions/browser/extension_registry_observer.h"
#include "extensions/browser/lazy_context_task_queue.h"
#include "extensions/browser/process_manager.h"
#include "extensions/browser/process_manager_observer.h"
#include "extensions/browser/update_observer.h"
#include "extensions/buildflags/buildflags.h"
#include "extensions/common/api/runtime.h"
#include "extensions/common/extension_id.h"
static_assert(BUILDFLAG(ENABLE_EXTENSIONS_CORE));
namespace base {
class Version;
}
namespace content {
class BrowserContext;
}
class PrefRegistrySimple;
namespace extensions {
namespace api::runtime {
struct PlatformInfo;
}
class Extension;
class ExtensionRegistry;
// Runtime API dispatches onStartup, onInstalled, and similar events to
// extensions. There is one instance shared between a browser context and
// its related incognito instance.
class RuntimeAPI : public BrowserContextKeyedAPI,
public ExtensionRegistryObserver,
public UpdateObserver,
public ProcessManagerObserver,
public LazyEventDispatchUtil::Observer {
public:
// The status of the restartAfterDelay request.
enum class RestartAfterDelayStatus {
// The request was made by a different extension other than the first one to
// invoke the restartAfterDelay runtime API.
FAILED_NOT_FIRST_EXTENSION,
// The request came too soon after a previous restart induced by the
// restartAfterDelay API. It failed to be scheduled as requested, and was
// instead throttled.
FAILED_THROTTLED,
// Any previously scheduled restart was successfully canceled.
SUCCESS_RESTART_CANCELED,
// A restart was successfully scheduled.
SUCCESS_RESTART_SCHEDULED,
};
// After this many suspiciously fast consecutive reloads, an extension will
// get disabled.
static constexpr int kFastReloadCount = 5;
// Same as above, but we increase the fast reload count for unpacked
// extensions.
static constexpr int kUnpackedFastReloadCount = 30;
static BrowserContextKeyedAPIFactory<RuntimeAPI>* GetFactoryInstance();
static void RegisterPrefs(PrefRegistrySimple* registry);
explicit RuntimeAPI(content::BrowserContext* context);
RuntimeAPI(const RuntimeAPI&) = delete;
RuntimeAPI& operator=(const RuntimeAPI&) = delete;
~RuntimeAPI() override;
void ReloadExtension(const ExtensionId& extension_id);
bool CheckForUpdates(const ExtensionId& extension_id,
RuntimeAPIDelegate::UpdateCheckCallback callback);
void OpenURL(const GURL& uninstall_url);
bool GetPlatformInfo(api::runtime::PlatformInfo* info);
bool RestartDevice(std::string* error_message);
RestartAfterDelayStatus RestartDeviceAfterDelay(
const ExtensionId& extension_id,
int seconds_from_now);
bool OpenOptionsPage(const Extension* extension,
content::BrowserContext* browser_context);
private:
friend class BrowserContextKeyedAPIFactory<RuntimeAPI>;
friend class RestartAfterDelayApiTest;
// ExtensionRegistryObserver implementation.
void OnExtensionLoaded(content::BrowserContext* browser_context,
const Extension* extension) override;
void OnExtensionUninstalled(content::BrowserContext* browser_context,
const Extension* extension,
UninstallReason reason) override;
// LazyEventDispatchUtil::Observer:
void OnExtensionInstalledAndLoaded(
content::BrowserContext* browser_context,
const Extension* extension,
const base::Version& previous_version) override;
// Cancels any previously scheduled restart request.
void MaybeCancelRunningDelayedRestartTimer();
// Handler for the signal from ExtensionSystem::ready().
void OnExtensionsReady();
RestartAfterDelayStatus ScheduleDelayedRestart(const base::Time& now,
int seconds_from_now);
// Called when the delayed restart timer times out so that it attempts to
// execute the restart request scheduled earlier.
void OnDelayedRestartTimerTimeout();
// BrowserContextKeyedAPI implementation:
static const char* service_name() { return "RuntimeAPI"; }
static const bool kServiceRedirectedInIncognito = true;
static const bool kServiceIsNULLWhileTesting = true;
void Shutdown() override;
// extensions::UpdateObserver overrides:
void OnAppUpdateAvailable(const Extension& extension) override;
void OnChromeUpdateAvailable() override;
// ProcessManagerObserver implementation:
void OnBackgroundHostStartup(const Extension* extension) override;
void AllowNonKioskAppsInRestartAfterDelayForTesting();
void set_min_duration_between_restarts_for_testing(base::TimeDelta delta) {
minimum_duration_between_restarts_ = delta;
}
std::unique_ptr<RuntimeAPIDelegate> delegate_;
raw_ptr<content::BrowserContext> browser_context_;
// Listen to extension notifications.
base::ScopedObservation<ExtensionRegistry, ExtensionRegistryObserver>
extension_registry_observation_{this};
base::ScopedObservation<ProcessManager, ProcessManagerObserver>
process_manager_observation_{this};
// The ID of the first extension to call the restartAfterDelay API. Any other
// extensions to call this API after that will fail.
std::string schedule_restart_first_extension_id_;
// The timer that will trigger a device restart when it times out.
base::OneShotTimer restart_after_delay_timer_;
// The minimum allowed duration between two successive restarts caused by
// restartAfterDelay calls.
base::TimeDelta minimum_duration_between_restarts_;
// The last restart time which was a result of a successful call to
// chrome.runtime.restartAfterDelay().
base::Time last_delayed_restart_time_;
// True if we should dispatch the chrome.runtime.onInstalled event with
// reason "chrome_update" upon loading each extension.
bool dispatch_chrome_updated_event_;
bool did_read_delayed_restart_preferences_;
bool was_last_restart_due_to_delayed_restart_api_;
base::WeakPtrFactory<RuntimeAPI> weak_ptr_factory_{this};
};
template <>
void BrowserContextKeyedAPIFactory<RuntimeAPI>::DeclareFactoryDependencies();
class RuntimeEventRouter {
public:
// Dispatches the onStartup event to all currently-loaded extensions.
static void DispatchOnStartupEvent(content::BrowserContext* context,
const ExtensionId& extension_id);
// Dispatches the onInstalled event to the given extension.
static void DispatchOnInstalledEvent(void* context_id,
const ExtensionId& extension_id,
const base::Version& old_version,
bool chrome_updated);
// Dispatches the onUpdateAvailable event to the given extension.
static void DispatchOnUpdateAvailableEvent(content::BrowserContext* context,
const ExtensionId& extension_id,
const base::Value::Dict* manifest);
// Dispatches the onBrowserUpdateAvailable event to all extensions.
static void DispatchOnBrowserUpdateAvailableEvent(
content::BrowserContext* context);
// Dispatches the onRestartRequired event to the given app.
static void DispatchOnRestartRequiredEvent(
content::BrowserContext* context,
const std::string& app_id,
api::runtime::OnRestartRequiredReason reason);
// Does any work needed at extension uninstall (e.g. load uninstall url).
static void OnExtensionUninstalled(content::BrowserContext* context,
const ExtensionId& extension_id,
UninstallReason reason);
};
class RuntimeGetBackgroundPageFunction : public ExtensionFunction {
public:
DECLARE_EXTENSION_FUNCTION("runtime.getBackgroundPage",
RUNTIME_GETBACKGROUNDPAGE)
protected:
~RuntimeGetBackgroundPageFunction() override = default;
ResponseAction Run() override;
private:
void OnPageLoaded(
std::unique_ptr<LazyContextTaskQueue::ContextInfo> context_info);
};
class RuntimeOpenOptionsPageFunction : public ExtensionFunction {
public:
DECLARE_EXTENSION_FUNCTION("runtime.openOptionsPage", RUNTIME_OPENOPTIONSPAGE)
protected:
~RuntimeOpenOptionsPageFunction() override = default;
ResponseAction Run() override;
};
class RuntimeSetUninstallURLFunction : public ExtensionFunction {
public:
DECLARE_EXTENSION_FUNCTION("runtime.setUninstallURL", RUNTIME_SETUNINSTALLURL)
protected:
~RuntimeSetUninstallURLFunction() override = default;
ResponseAction Run() override;
};
class RuntimeReloadFunction : public ExtensionFunction {
public:
DECLARE_EXTENSION_FUNCTION("runtime.reload", RUNTIME_RELOAD)
protected:
~RuntimeReloadFunction() override = default;
ResponseAction Run() override;
};
class RuntimeRequestUpdateCheckFunction : public ExtensionFunction {
public:
DECLARE_EXTENSION_FUNCTION("runtime.requestUpdateCheck",
RUNTIME_REQUESTUPDATECHECK)
protected:
~RuntimeRequestUpdateCheckFunction() override = default;
ResponseAction Run() override;
private:
void CheckComplete(const RuntimeAPIDelegate::UpdateCheckResult& result);
};
class RuntimeRestartFunction : public ExtensionFunction {
public:
DECLARE_EXTENSION_FUNCTION("runtime.restart", RUNTIME_RESTART)
protected:
~RuntimeRestartFunction() override = default;
ResponseAction Run() override;
};
class RuntimeRestartAfterDelayFunction : public ExtensionFunction {
public:
DECLARE_EXTENSION_FUNCTION("runtime.restartAfterDelay",
RUNTIME_RESTARTAFTERDELAY)
protected:
~RuntimeRestartAfterDelayFunction() override = default;
ResponseAction Run() override;
};
class RuntimeGetPlatformInfoFunction : public ExtensionFunction {
public:
DECLARE_EXTENSION_FUNCTION("runtime.getPlatformInfo", RUNTIME_GETPLATFORMINFO)
protected:
~RuntimeGetPlatformInfoFunction() override = default;
ResponseAction Run() override;
};
class RuntimeGetPackageDirectoryEntryFunction : public ExtensionFunction {
public:
DECLARE_EXTENSION_FUNCTION("runtime.getPackageDirectoryEntry",
RUNTIME_GETPACKAGEDIRECTORYENTRY)
protected:
~RuntimeGetPackageDirectoryEntryFunction() override = default;
ResponseAction Run() override;
};
class RuntimeGetContextsFunction : public ExtensionFunction {
public:
DECLARE_EXTENSION_FUNCTION("runtime.getContexts", RUNTIME_GETCONTEXTS)
RuntimeGetContextsFunction();
RuntimeGetContextsFunction(const RuntimeGetContextsFunction&) = delete;
RuntimeGetContextsFunction& operator=(const RuntimeGetContextsFunction&) =
delete;
private:
// ExtensionFunction:
~RuntimeGetContextsFunction() override;
ResponseAction Run() override;
// Returns the context for the extension background service worker, if the
// worker is active. Otherwise, returns nullopt.
std::optional<api::runtime::ExtensionContext> GetWorkerContext();
// Returns a collection of all frame-based extension contexts for the
// extension.
std::vector<api::runtime::ExtensionContext> GetFrameContexts();
// Helper methods to return tab id, frame id and window id for a given
// context.
int GetTabId(content::WebContents& web_contents);
int GetFrameId(content::RenderFrameHost& host);
int GetWindowId(content::WebContents& web_contents);
};
} // namespace extensions
#endif // EXTENSIONS_BROWSER_API_RUNTIME_RUNTIME_API_H_