| // 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_ |