blob: 0fe3a531873b968559968693ad23664e104bfbc3 [file] [log] [blame]
// Copyright 2019 The Chromium Authors. All rights reserved.
// 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/callback_forward.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/weak_ptr.h"
#include "chrome/browser/web_applications/web_app.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 "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 SyncInstallDelegate;
class WebAppDatabase;
class WebAppRegistryUpdate;
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:
WebAppSyncBridge(AbstractWebAppDatabaseFactory* database_factory,
WebAppRegistrarMutable* registrar,
SyncInstallDelegate* install_delegate);
// Tests may inject mocks using this ctor.
WebAppSyncBridge(
AbstractWebAppDatabaseFactory* database_factory,
WebAppRegistrarMutable* registrar,
SyncInstallDelegate* install_delegate,
std::unique_ptr<syncer::ModelTypeChangeProcessor> change_processor);
WebAppSyncBridge(const WebAppSyncBridge&) = delete;
WebAppSyncBridge& operator=(const WebAppSyncBridge&) = delete;
~WebAppSyncBridge() override;
using CommitCallback = base::OnceCallback<void(bool success)>;
// 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.
std::unique_ptr<WebAppRegistryUpdate> BeginUpdate();
void CommitUpdate(std::unique_ptr<WebAppRegistryUpdate> update,
CommitCallback callback);
void Init(base::OnceClosure callback);
void SetAppUserDisplayMode(const AppId& app_id,
DisplayMode user_display_mode,
bool is_user_action);
void SetAppIsDisabled(const AppId& app_id, bool is_disabled);
void UpdateAppsDisableMode();
void SetAppIsLocallyInstalled(const AppId& app_id, bool is_locally_installed);
void SetAppLastBadgingTime(const AppId& app_id, const base::Time& time);
void SetAppLastLaunchTime(const AppId& app_id, const base::Time& time);
void SetAppInstallTime(const AppId& app_id, const base::Time& time);
void SetAppManifestUpdateTime(const AppId& app_id, const base::Time& time);
void SetAppRunOnOsLoginMode(const AppId& app_id, RunOnOsLoginMode mode);
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);
// These methods are used by web apps to add or remove allowed
// protocol schemes based on user approval or withdrawal of that approval.
// Allowed protocol schemes will allow web apps to handle launches from
// urls that start with that scheme without asking the user.
void AddAllowedLaunchProtocol(const AppId& app_id,
const std::string& protocol_scheme);
void RemoveAllowedLaunchProtocol(const AppId& app_id,
const std::string& protocol_scheme);
// These methods are used by web apps to add or remove disallowed
// protocol schemes based on user preference or withdrawal of that preference.
// Disallowed protocol schemes will never allow web apps to handle launches
// from urls that start with that scheme.
void AddDisallowedLaunchProtocol(const AppId& app_id,
const std::string& protocol_scheme);
void RemoveDisallowedLaunchProtocol(const AppId& app_id,
const std::string& protocol_scheme);
// 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> MergeSyncData(
std::unique_ptr<syncer::MetadataChangeList> metadata_change_list,
syncer::EntityChangeList entity_data) override;
absl::optional<syncer::ModelError> ApplySyncChanges(
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;
const std::set<AppId>& GetAppsInSyncUninstallForTest();
void set_disable_checks_for_testing(bool disable_checks_for_testing) {
disable_checks_for_testing_ = disable_checks_for_testing;
}
private:
void CheckRegistryUpdateData(const RegistryUpdateData& update_data) const;
// Update the in-memory model. Returns unregistered apps which may be
// disposed.
std::vector<std::unique_ptr<WebApp>> UpdateRegistrar(
std::unique_ptr<RegistryUpdateData> update_data);
// Useful for identifying apps that have not yet been fully uninstalled.
std::set<AppId> apps_in_sync_uninstall_;
// 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 WebAppUninstalled(const AppId& app, bool uninstalled);
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 ApplySyncDataChange(const syncer::EntityChange& change,
RegistryUpdateData* update_local_data);
// Update registrar and Install/Uninstall missing/excessive local apps.
void ApplySyncChangesToRegistrar(
std::unique_ptr<RegistryUpdateData> update_local_data);
void MaybeUninstallAppsPendingUninstall();
void MaybeInstallAppsFromSyncAndPendingInstallation();
std::unique_ptr<WebAppDatabase> database_;
const raw_ptr<WebAppRegistrarMutable> registrar_;
const raw_ptr<SyncInstallDelegate> install_delegate_;
bool is_in_update_ = false;
bool disable_checks_for_testing_ = false;
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_