blob: 0351e00a081bf7c34a7172097c352e8766265567 [file] [log] [blame]
// Copyright 2019 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef CHROME_BROWSER_WEB_APPLICATIONS_WEB_APP_SYNC_BRIDGE_H_
#define CHROME_BROWSER_WEB_APPLICATIONS_WEB_APP_SYNC_BRIDGE_H_
#include <memory>
#include "base/functional/callback_forward.h"
#include "base/functional/callback_helpers.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/weak_ptr.h"
#include "base/one_shot_event.h"
#include "build/build_config.h"
#include "chrome/browser/web_applications/mojom/user_display_mode.mojom.h"
#include "chrome/browser/web_applications/web_app.h"
#include "chrome/browser/web_applications/web_app_command_scheduler.h"
#include "chrome/browser/web_applications/web_app_constants.h"
#include "chrome/browser/web_applications/web_app_registrar.h"
#include "components/sync/model/entity_change.h"
#include "components/sync/model/model_type_sync_bridge.h"
#include "components/webapps/browser/uninstall_result_code.h"
#include "third_party/abseil-cpp/absl/types/optional.h"
namespace base {
class Time;
}
namespace syncer {
class MetadataBatch;
class MetadataChangeList;
class ModelError;
class ModelTypeChangeProcessor;
} // namespace syncer
namespace sync_pb {
class WebAppSpecifics;
} // namespace sync_pb
namespace syncer {
struct EntityData;
}
namespace web_app {
class AbstractWebAppDatabaseFactory;
class AppLock;
class ScopedRegistryUpdate;
class WebAppCommandManager;
class WebAppDatabase;
class WebAppRegistryUpdate;
class WebAppInstallManager;
struct RegistryUpdateData;
// A unified sync and storage controller.
//
// While WebAppRegistrar is a read-only model, WebAppSyncBridge is a
// controller for that model. WebAppSyncBridge is responsible for:
// - Registry initialization (reading model from a persistent storage like
// LevelDb or prefs).
// - Writing all the registry updates to a persistent store and sync.
//
// WebAppSyncBridge is the key class to support integration with Unified Sync
// and Storage (USS) system. The sync bridge exclusively owns
// ModelTypeChangeProcessor and WebAppDatabase (the storage).
class WebAppSyncBridge : public syncer::ModelTypeSyncBridge {
public:
explicit WebAppSyncBridge(WebAppRegistrarMutable* registrar);
// Tests may inject mocks using this ctor.
WebAppSyncBridge(
WebAppRegistrarMutable* registrar,
std::unique_ptr<syncer::ModelTypeChangeProcessor> change_processor);
WebAppSyncBridge(const WebAppSyncBridge&) = delete;
WebAppSyncBridge& operator=(const WebAppSyncBridge&) = delete;
~WebAppSyncBridge() override;
void SetSubsystems(AbstractWebAppDatabaseFactory* database_factory,
WebAppCommandManager* command_manager,
WebAppCommandScheduler* command_scheduler_,
WebAppInstallManager* install_manager_);
using CommitCallback = base::OnceCallback<void(bool success)>;
using RepeatingInstallCallback =
base::RepeatingCallback<void(const AppId& app_id,
webapps::InstallResultCode code)>;
using RepeatingUninstallCallback =
base::RepeatingCallback<void(const AppId& app_id,
webapps::UninstallResultCode code)>;
// This is the writable API for the registry. Any updates will be written to
// LevelDb and sync service. There can be only 1 update at a time. The
// returned update will be committed to the database automatically on
// destruction.
//
// Writes to the RAM database are synchronous. It is normally not necessary to
// wait for the disk write to complete, because:
// - All reads and writes happen from the RAM database.
// - The disk database is only read during startup before the system starts,
// and is only written to during the rest of the browser's lifetime.
//
// The only reason waiting may be necessary here is to handle the edge case
// that a crash happens and the operation wants to ensure that the disk state
// is updated before continuing. This may be necessary for, say, a two-phase
// commit involving another browser system with its own storage.
[[nodiscard]] ScopedRegistryUpdate BeginUpdate(
CommitCallback callback = base::DoNothing());
void Init(base::OnceClosure callback);
void SetAppUserDisplayMode(const AppId& app_id,
mojom::UserDisplayMode user_display_mode,
bool is_user_action);
void SetAppIsDisabled(AppLock& lock, const AppId& app_id, bool is_disabled);
void UpdateAppsDisableMode();
void SetAppLastBadgingTime(const AppId& app_id, const base::Time& time);
void SetAppLastLaunchTime(const AppId& app_id, const base::Time& time);
void SetAppFirstInstallTime(const AppId& app_id, const base::Time& time);
void SetAppManifestUpdateTime(const AppId& app_id, const base::Time& time);
void SetAppWindowControlsOverlayEnabled(const AppId& app_id, bool enabled);
// These methods are used by extensions::AppSorting, which manages the sorting
// of web apps on chrome://apps.
void SetUserPageOrdinal(const AppId& app_id,
syncer::StringOrdinal user_page_ordinal);
void SetUserLaunchOrdinal(const AppId& app_id,
syncer::StringOrdinal user_launch_ordinal);
// Stores the user's preference for the app's use of the File Handling API.
void SetAppFileHandlerApprovalState(const AppId& app_id,
ApiApprovalState state);
#if BUILDFLAG(IS_MAC)
void SetAlwaysShowToolbarInFullscreen(const AppId& app_id, bool show);
#endif
// An access to read-only registry. Does an upcast to read-only type.
const WebAppRegistrar& registrar() const { return *registrar_; }
// syncer::ModelTypeSyncBridge:
std::unique_ptr<syncer::MetadataChangeList> CreateMetadataChangeList()
override;
absl::optional<syncer::ModelError> MergeFullSyncData(
std::unique_ptr<syncer::MetadataChangeList> metadata_change_list,
syncer::EntityChangeList entity_data) override;
absl::optional<syncer::ModelError> ApplyIncrementalSyncChanges(
std::unique_ptr<syncer::MetadataChangeList> metadata_change_list,
syncer::EntityChangeList entity_changes) override;
void GetData(StorageKeyList storage_keys, DataCallback callback) override;
void GetAllDataForDebugging(DataCallback callback) override;
std::string GetClientTag(const syncer::EntityData& entity_data) override;
std::string GetStorageKey(const syncer::EntityData& entity_data) override;
// Signals that the sync system has received data from the server at some
// point, potentially on a previous startup. Apps may still be installing or
// uninstalling.
const base::OneShotEvent& on_sync_connected() const {
return on_sync_connected_;
}
// Used for testing only.
void set_disable_checks_for_testing(bool disable_checks_for_testing) {
disable_checks_for_testing_ = disable_checks_for_testing;
}
using RetryIncompleteUninstallsCallback = base::RepeatingCallback<void(
const base::flat_set<AppId>& apps_to_uninstall)>;
void SetRetryIncompleteUninstallsCallbackForTesting(
RetryIncompleteUninstallsCallback callback);
using InstallWebAppsAfterSyncCallback =
base::RepeatingCallback<void(std::vector<WebApp*> web_apps,
RepeatingInstallCallback callback)>;
void SetInstallWebAppsAfterSyncCallbackForTesting(
InstallWebAppsAfterSyncCallback callback);
using UninstallFromSyncCallback =
base::RepeatingCallback<void(const std::vector<AppId>& web_apps,
RepeatingUninstallCallback callback)>;
void SetUninstallFromSyncCallbackForTesting(
UninstallFromSyncCallback callback);
WebAppDatabase* GetDatabaseForTesting() const { return database_.get(); }
void SetAppIsLocallyInstalledForTesting(const AppId& app_id,
bool is_locally_installed);
private:
void CommitUpdate(CommitCallback callback,
std::unique_ptr<WebAppRegistryUpdate> update);
void CheckRegistryUpdateData(const RegistryUpdateData& update_data) const;
// Update the in-memory model.
void UpdateRegistrar(std::unique_ptr<RegistryUpdateData> update_data);
// Update the remote sync server.
void UpdateSync(const RegistryUpdateData& update_data,
syncer::MetadataChangeList* metadata_change_list);
void OnDatabaseOpened(base::OnceClosure callback,
Registry registry,
std::unique_ptr<syncer::MetadataBatch> metadata_batch);
void OnDataWritten(CommitCallback callback, bool success);
void OnWebAppUninstallComplete(const AppId& app,
webapps::UninstallResultCode code);
void ReportErrorToChangeProcessor(const syncer::ModelError& error);
// Any local entities that don’t exist remotely must be provided to sync.
void MergeLocalAppsToSync(const syncer::EntityChangeList& entity_data,
syncer::MetadataChangeList* metadata_change_list);
void PrepareLocalUpdateFromSyncChange(
const syncer::EntityChange& change,
RegistryUpdateData* update_local_data,
std::vector<AppId>& apps_display_mode_changed);
// Update registrar and Install/Uninstall missing/excessive local apps.
void ApplyIncrementalSyncChangesToRegistrar(
std::unique_ptr<RegistryUpdateData> update_local_data,
const std::vector<AppId>& apps_display_mode_changed);
void MaybeUninstallAppsPendingUninstall();
void MaybeInstallAppsFromSyncAndPendingInstallation();
void InstallWebAppsAfterSync(std::vector<WebApp*> web_apps,
RepeatingInstallCallback callback);
std::unique_ptr<WebAppDatabase> database_;
const raw_ptr<WebAppRegistrarMutable, DanglingUntriaged> registrar_;
raw_ptr<WebAppCommandManager, AcrossTasksDanglingUntriaged> command_manager_ =
nullptr;
raw_ptr<WebAppCommandScheduler, AcrossTasksDanglingUntriaged>
command_scheduler_ = nullptr;
raw_ptr<WebAppInstallManager, AcrossTasksDanglingUntriaged> install_manager_ =
nullptr;
base::OneShotEvent on_sync_connected_;
bool is_in_update_ = false;
bool disable_checks_for_testing_ = false;
RetryIncompleteUninstallsCallback
retry_incomplete_uninstalls_callback_for_testing_;
InstallWebAppsAfterSyncCallback
install_web_apps_after_sync_callback_for_testing_;
UninstallFromSyncCallback
uninstall_from_sync_before_registry_update_callback_for_testing_;
base::WeakPtrFactory<WebAppSyncBridge> weak_ptr_factory_{this};
};
std::unique_ptr<syncer::EntityData> CreateSyncEntityData(const WebApp& app);
void ApplySyncDataToApp(const sync_pb::WebAppSpecifics& sync_data, WebApp* app);
} // namespace web_app
#endif // CHROME_BROWSER_WEB_APPLICATIONS_WEB_APP_SYNC_BRIDGE_H_