blob: 78dc42d003aa1530ebf832df706b46747fe4f0d7 [file] [log] [blame]
// Copyright 2018 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/web_applications/web_app_provider.h"
#include <memory>
#include <utility>
#include "base/barrier_closure.h"
#include "base/check_is_test.h"
#include "base/command_line.h"
#include "base/feature_list.h"
#include "base/functional/bind.h"
#include "base/functional/callback_forward.h"
#include "base/location.h"
#include "base/memory/scoped_refptr.h"
#include "base/run_loop.h"
#include "build/build_config.h"
#include "build/chromeos_buildflags.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/web_applications/extensions_manager.h"
#include "chrome/browser/web_applications/externally_managed_app_manager.h"
#include "chrome/browser/web_applications/file_utils_wrapper.h"
#include "chrome/browser/web_applications/isolated_web_apps/garbage_collect_storage_partitions_command.h"
#include "chrome/browser/web_applications/isolated_web_apps/install_isolated_web_app_from_command_line.h"
#include "chrome/browser/web_applications/manifest_update_manager.h"
#include "chrome/browser/web_applications/os_integration/os_integration_manager.h"
#include "chrome/browser/web_applications/os_integration/url_handler_manager.h"
#include "chrome/browser/web_applications/os_integration/web_app_file_handler_manager.h"
#include "chrome/browser/web_applications/os_integration/web_app_protocol_handler_manager.h"
#include "chrome/browser/web_applications/os_integration/web_app_shortcut_manager.h"
#include "chrome/browser/web_applications/policy/web_app_policy_manager.h"
#include "chrome/browser/web_applications/preinstalled_web_app_manager.h"
#include "chrome/browser/web_applications/web_app_audio_focus_id_map.h"
#include "chrome/browser/web_applications/web_app_command_manager.h"
#include "chrome/browser/web_applications/web_app_command_scheduler.h"
#include "chrome/browser/web_applications/web_app_database_factory.h"
#include "chrome/browser/web_applications/web_app_icon_manager.h"
#include "chrome/browser/web_applications/web_app_install_finalizer.h"
#include "chrome/browser/web_applications/web_app_install_manager.h"
#include "chrome/browser/web_applications/web_app_prefs_utils.h"
#include "chrome/browser/web_applications/web_app_provider_factory.h"
#include "chrome/browser/web_applications/web_app_registrar.h"
#include "chrome/browser/web_applications/web_app_registry_update.h"
#include "chrome/browser/web_applications/web_app_sync_bridge.h"
#include "chrome/browser/web_applications/web_app_translation_manager.h"
#include "chrome/browser/web_applications/web_app_ui_manager.h"
#include "chrome/browser/web_applications/web_app_utils.h"
#include "chrome/browser/web_applications/web_contents/web_contents_manager.h"
#include "chrome/common/chrome_features.h"
#include "chrome/common/pref_names.h"
#include "components/prefs/pref_service.h"
#include "content/public/browser/web_contents.h"
#if (BUILDFLAG(IS_CHROMEOS))
#include "chrome/browser/web_applications/isolated_web_apps/isolated_web_app_update_manager.h"
#endif
#if BUILDFLAG(IS_CHROMEOS_ASH)
#include "ash/constants/ash_features.h"
#include "chromeos/ash/components/browser_context_helper/browser_context_types.h"
#endif
namespace web_app {
namespace {
WebAppProvider::OsIntegrationManagerFactory
g_os_integration_manager_factory_for_testing = nullptr;
} // namespace
// static
WebAppProvider* WebAppProvider::GetDeprecated(Profile* profile) {
return WebAppProviderFactory::GetForProfile(profile);
}
// static
WebAppProvider* WebAppProvider::GetForWebApps(Profile* profile) {
#if BUILDFLAG(IS_CHROMEOS_ASH)
// If features::kWebAppsCrosapi is enabled, Ash browser only manages system
// web apps (return nullptr here). Otherwise, Ash browser manages all web apps
// (return WebAppProvider).
// An exception is that Shimless RMA app always requires loading IWA on Ash.
if (IsWebAppsCrosapiEnabled() &&
(!::ash::features::IsShimlessRMA3pDiagnosticsEnabled() ||
!::ash::IsShimlessRmaAppBrowserContext(profile))) {
return nullptr;
}
#endif
return WebAppProviderFactory::GetForProfile(profile);
}
// static
WebAppProvider* WebAppProvider::GetForLocalAppsUnchecked(Profile* profile) {
return WebAppProviderFactory::GetForProfile(profile);
}
// static
WebAppProvider* WebAppProvider::GetForTest(Profile* profile) {
// Running a nested base::RunLoop outside of tests causes a deadlock. Crash
// immediately instead of deadlocking for easier debugging (especially for
// TAST tests which use prod binaries).
CHECK_IS_TEST();
WebAppProvider* provider = GetForLocalAppsUnchecked(profile);
if (!provider)
return nullptr;
if (provider->on_registry_ready().is_signaled())
return provider;
base::RunLoop run_loop;
provider->on_registry_ready().Post(FROM_HERE, run_loop.QuitClosure());
run_loop.Run();
return provider;
}
// static
WebAppProvider* WebAppProvider::GetForWebContents(
content::WebContents* web_contents) {
Profile* profile =
Profile::FromBrowserContext(web_contents->GetBrowserContext());
DCHECK(profile);
return WebAppProvider::GetForLocalAppsUnchecked(profile);
}
// static
void WebAppProvider::SetOsIntegrationManagerFactoryForTesting(
OsIntegrationManagerFactory factory) {
g_os_integration_manager_factory_for_testing = factory;
}
WebAppProvider::WebAppProvider(Profile* profile) : profile_(profile) {
DCHECK(AreWebAppsEnabled(profile_));
// WebApp System must have only one instance in original profile.
// Exclude secondary off-the-record profiles.
#if BUILDFLAG(IS_CHROMEOS_ASH)
if (!profile_->IsGuestSession())
DCHECK(!profile_->IsOffTheRecord());
#else
DCHECK(!profile_->IsOffTheRecord());
#endif
CreateSubsystems(profile_);
}
WebAppProvider::~WebAppProvider() = default;
void WebAppProvider::Start() {
CHECK(!started_);
ConnectSubsystems();
started_ = true;
StartImpl();
}
WebAppCommandScheduler& WebAppProvider::scheduler() {
return *command_scheduler_;
}
WebAppCommandManager& WebAppProvider::command_manager() {
// Note: It is OK to access the command manager before connection or start.
// Internally it will queue commands to only happen after it has started.
return *command_manager_;
}
WebAppRegistrar& WebAppProvider::registrar_unsafe() {
CheckIsConnected();
return *registrar_;
}
const WebAppRegistrar& WebAppProvider::registrar_unsafe() const {
CheckIsConnected();
return *registrar_;
}
WebAppRegistrarMutable& WebAppProvider::registrar_mutable(
base::PassKey<WebAppSyncBridge>) {
CheckIsConnected();
return *registrar_;
}
WebAppSyncBridge& WebAppProvider::sync_bridge_unsafe() {
CheckIsConnected();
return *sync_bridge_;
}
WebAppInstallManager& WebAppProvider::install_manager() {
CheckIsConnected();
return *install_manager_;
}
WebAppInstallFinalizer& WebAppProvider::install_finalizer() {
CheckIsConnected();
return *install_finalizer_;
}
ManifestUpdateManager& WebAppProvider::manifest_update_manager() {
CheckIsConnected();
return *manifest_update_manager_;
}
ExternallyManagedAppManager& WebAppProvider::externally_managed_app_manager() {
CheckIsConnected();
return *externally_managed_app_manager_;
}
WebAppPolicyManager& WebAppProvider::policy_manager() {
CheckIsConnected();
return *web_app_policy_manager_;
}
IsolatedWebAppCommandLineInstallManager&
WebAppProvider::iwa_command_line_install_manager() {
CheckIsConnected();
return *iwa_command_line_install_manager_;
}
#if (BUILDFLAG(IS_CHROMEOS))
IsolatedWebAppUpdateManager& WebAppProvider::iwa_update_manager() {
CheckIsConnected();
return *iwa_update_manager_;
}
#endif
WebAppUiManager& WebAppProvider::ui_manager() {
CheckIsConnected();
return *ui_manager_;
}
WebAppAudioFocusIdMap& WebAppProvider::audio_focus_id_map() {
CheckIsConnected();
return *audio_focus_id_map_;
}
scoped_refptr<FileUtilsWrapper> WebAppProvider::file_utils() {
CheckIsConnected();
return file_utils_;
}
WebAppIconManager& WebAppProvider::icon_manager() {
CheckIsConnected();
return *icon_manager_;
}
WebAppTranslationManager& WebAppProvider::translation_manager() {
CheckIsConnected();
return *translation_manager_;
}
OsIntegrationManager& WebAppProvider::os_integration_manager() {
CheckIsConnected();
return *os_integration_manager_;
}
const OsIntegrationManager& WebAppProvider::os_integration_manager() const {
CheckIsConnected();
return *os_integration_manager_;
}
WebAppOriginAssociationManager& WebAppProvider::origin_association_manager() {
return *origin_association_manager_;
}
WebContentsManager& WebAppProvider::web_contents_manager() {
return *web_contents_manager_;
}
PreinstalledWebAppManager& WebAppProvider::preinstalled_web_app_manager() {
return *preinstalled_web_app_manager_;
}
ExtensionsManager& WebAppProvider::extensions_manager() {
return *extensions_manager_;
}
AbstractWebAppDatabaseFactory& WebAppProvider::database_factory() {
return *database_factory_;
}
void WebAppProvider::Shutdown() {
command_scheduler_->Shutdown();
command_manager_->Shutdown();
ui_manager_->Shutdown();
externally_managed_app_manager_->Shutdown();
manifest_update_manager_->Shutdown();
#if (BUILDFLAG(IS_CHROMEOS))
iwa_update_manager_->Shutdown();
#endif
install_manager_->Shutdown();
icon_manager_->Shutdown();
install_finalizer_->Shutdown();
registrar_->Shutdown();
is_registry_ready_ = false;
}
base::WeakPtr<WebAppProvider> WebAppProvider::AsWeakPtr() {
return weak_ptr_factory_.GetWeakPtr();
}
void WebAppProvider::StartImpl() {
StartSyncBridge();
MaybeScheduleGarbageCollection();
}
void WebAppProvider::CreateSubsystems(Profile* profile) {
audio_focus_id_map_ = std::make_unique<WebAppAudioFocusIdMap>();
ui_manager_ = WebAppUiManager::Create(profile);
install_manager_ = std::make_unique<WebAppInstallManager>(profile);
manifest_update_manager_ = std::make_unique<ManifestUpdateManager>();
externally_managed_app_manager_ =
std::make_unique<ExternallyManagedAppManager>(profile);
preinstalled_web_app_manager_ =
std::make_unique<PreinstalledWebAppManager>(profile);
web_app_policy_manager_ = std::make_unique<WebAppPolicyManager>(profile);
iwa_command_line_install_manager_ =
std::make_unique<IsolatedWebAppCommandLineInstallManager>(*profile);
#if (BUILDFLAG(IS_CHROMEOS))
iwa_update_manager_ = std::make_unique<IsolatedWebAppUpdateManager>(*profile);
#endif
extensions_manager_ = std::make_unique<ExtensionsManager>(profile);
database_factory_ = std::make_unique<WebAppDatabaseFactory>(profile);
registrar_ = std::make_unique<WebAppRegistrarMutable>(profile);
sync_bridge_ = std::make_unique<WebAppSyncBridge>(registrar_.get());
file_utils_ = base::MakeRefCounted<FileUtilsWrapper>();
icon_manager_ = std::make_unique<WebAppIconManager>(profile);
translation_manager_ = std::make_unique<WebAppTranslationManager>(profile);
install_finalizer_ = std::make_unique<WebAppInstallFinalizer>(profile);
if (g_os_integration_manager_factory_for_testing) {
os_integration_manager_ =
g_os_integration_manager_factory_for_testing(profile);
} else {
auto file_handler_manager =
std::make_unique<WebAppFileHandlerManager>(profile);
auto protocol_handler_manager =
std::make_unique<WebAppProtocolHandlerManager>(profile);
auto shortcut_manager = std::make_unique<WebAppShortcutManager>(
profile, file_handler_manager.get(), protocol_handler_manager.get());
// TODO(crbug.com/1072058): Remove UrlHandlerManager from
// OsIntegrationManager.
os_integration_manager_ = std::make_unique<OsIntegrationManager>(
profile, std::move(shortcut_manager), std::move(file_handler_manager),
std::move(protocol_handler_manager), /*url_handler_manager=*/nullptr);
}
command_manager_ = std::make_unique<WebAppCommandManager>(profile);
command_scheduler_ = std::make_unique<WebAppCommandScheduler>(*profile);
origin_association_manager_ =
std::make_unique<WebAppOriginAssociationManager>();
#if (BUILDFLAG(IS_CHROMEOS))
web_app_run_on_os_login_manager_ =
std::make_unique<WebAppRunOnOsLoginManager>(command_scheduler_.get());
#endif
web_contents_manager_ = std::make_unique<WebContentsManager>();
}
void WebAppProvider::ConnectSubsystems() {
DCHECK(!started_);
// TODO(https://issuetracker.google.com/283816014): Replace SetSubsystems()
// with SetProvider().
sync_bridge_->SetSubsystems(database_factory_.get(), command_manager_.get(),
command_scheduler_.get(), install_manager_.get());
base::PassKey<WebAppProvider> pass_key;
icon_manager_->SetProvider(pass_key, *this);
install_finalizer_->SetProvider(pass_key, *this);
manifest_update_manager_->SetProvider(pass_key, *this);
externally_managed_app_manager_->SetProvider(pass_key, *this);
preinstalled_web_app_manager_->SetProvider(pass_key, *this);
web_app_policy_manager_->SetProvider(pass_key, *this);
registrar_->SetProvider(pass_key, *this);
os_integration_manager_->SetProvider(pass_key, *this);
command_manager_->SetProvider(pass_key, *this);
command_scheduler_->SetProvider(pass_key, *this);
iwa_command_line_install_manager_->SetProvider(pass_key, *this);
#if (BUILDFLAG(IS_CHROMEOS))
iwa_update_manager_->SetProvider(pass_key, *this);
#endif
icon_manager_->SetProvider(pass_key, *this);
translation_manager_->SetProvider(pass_key, *this);
connected_ = true;
}
void WebAppProvider::StartSyncBridge() {
sync_bridge_->Init(
base::BindOnce(&WebAppProvider::OnSyncBridgeReady, AsWeakPtr()));
}
void WebAppProvider::OnSyncBridgeReady() {
DCHECK(!on_registry_ready_.is_signaled());
DoMigrateProfilePrefs(profile_);
// Note: This does not wait for the call from the ChromeOS
// SystemWebAppManager, which is a separate keyed service.
int num_barrier_calls = 2;
base::RepeatingClosure external_manager_barrier = base::BarrierClosure(
num_barrier_calls,
base::BindOnce(
[](base::WeakPtr<WebAppProvider> provider) {
if (!provider)
return;
provider->on_external_managers_synchronized_.Signal();
},
AsWeakPtr()));
registrar_->Start();
install_finalizer_->Start();
icon_manager_->Start();
translation_manager_->Start();
install_manager_->Start();
preinstalled_web_app_manager_->Start(external_manager_barrier);
web_app_policy_manager_->Start(external_manager_barrier);
iwa_command_line_install_manager_->Start();
#if (BUILDFLAG(IS_CHROMEOS))
iwa_update_manager_->Start();
on_external_managers_synchronized_.Post(
FROM_HERE,
base::BindOnce(&WebAppRunOnOsLoginManager::Start,
web_app_run_on_os_login_manager_->GetWeakPtr()));
#endif
manifest_update_manager_->Start();
os_integration_manager_->Start();
ui_manager_->Start();
command_manager_->Start();
on_registry_ready_.Signal();
is_registry_ready_ = true;
}
void WebAppProvider::CheckIsConnected() const {
DCHECK(connected_) << "Attempted to access Web App subsystem while "
"WebAppProvider is not connected. You may need to wait "
"for on_registry_ready().";
}
void WebAppProvider::DoMigrateProfilePrefs(Profile* profile) {
std::map<AppId, int> sources =
TakeAllWebAppInstallSources(profile->GetPrefs());
ScopedRegistryUpdate update = sync_bridge_->BeginUpdate();
for (const auto& iter : sources) {
WebApp* web_app = update->UpdateApp(iter.first);
if (web_app && !web_app->latest_install_source()) {
web_app->SetLatestInstallSource(
static_cast<webapps::WebappInstallSource>(iter.second));
}
}
}
void WebAppProvider::MaybeScheduleGarbageCollection() {
// We are mirating from ExtensionsPref::kStorageGarbageCollect to
// prefs::kShouldGarbageCollectStoragePartitions. During migration, either
// one of the prefs can trigget garbge collection.
// TODO(crbug.com/1463825): Delete ExtensionsPref::kStorageGarbageCollect.
if (profile_->GetPrefs()->GetBoolean(
prefs::kShouldGarbageCollectStoragePartitions) ||
extensions_manager_->ShouldGarbageCollectStoragePartitions()) {
command_manager().ScheduleCommand(
std::make_unique<web_app::GarbageCollectStoragePartititonsCommand>(
profile_, base::DoNothing()));
}
}
} // namespace web_app