blob: a7eceee8a8005fec4740f98cfdc05fdb0d5f53e8 [file] [log] [blame]
// Copyright 2013 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.
#include "chrome/browser/sync_file_system/drive_backend/sync_engine.h"
#include <utility>
#include <vector>
#include "base/bind.h"
#include "base/callback_helpers.h"
#include "base/macros.h"
#include "base/memory/ptr_util.h"
#include "base/metrics/histogram_macros.h"
#include "base/task/post_task.h"
#include "base/task/thread_pool.h"
#include "base/threading/thread_task_runner_handle.h"
#include "base/time/time.h"
#include "base/values.h"
#include "chrome/browser/drive/drive_notification_manager_factory.h"
#include "chrome/browser/extensions/extension_service.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/signin/identity_manager_factory.h"
#include "chrome/browser/sync_file_system/drive_backend/callback_helper.h"
#include "chrome/browser/sync_file_system/drive_backend/conflict_resolver.h"
#include "chrome/browser/sync_file_system/drive_backend/drive_backend_constants.h"
#include "chrome/browser/sync_file_system/drive_backend/drive_service_on_worker.h"
#include "chrome/browser/sync_file_system/drive_backend/drive_service_wrapper.h"
#include "chrome/browser/sync_file_system/drive_backend/drive_uploader_on_worker.h"
#include "chrome/browser/sync_file_system/drive_backend/drive_uploader_wrapper.h"
#include "chrome/browser/sync_file_system/drive_backend/list_changes_task.h"
#include "chrome/browser/sync_file_system/drive_backend/local_to_remote_syncer.h"
#include "chrome/browser/sync_file_system/drive_backend/metadata_database.h"
#include "chrome/browser/sync_file_system/drive_backend/register_app_task.h"
#include "chrome/browser/sync_file_system/drive_backend/remote_change_processor_on_worker.h"
#include "chrome/browser/sync_file_system/drive_backend/remote_change_processor_wrapper.h"
#include "chrome/browser/sync_file_system/drive_backend/remote_to_local_syncer.h"
#include "chrome/browser/sync_file_system/drive_backend/sync_engine_context.h"
#include "chrome/browser/sync_file_system/drive_backend/sync_engine_initializer.h"
#include "chrome/browser/sync_file_system/drive_backend/sync_task.h"
#include "chrome/browser/sync_file_system/drive_backend/sync_worker.h"
#include "chrome/browser/sync_file_system/drive_backend/sync_worker_interface.h"
#include "chrome/browser/sync_file_system/drive_backend/uninstall_app_task.h"
#include "chrome/browser/sync_file_system/file_status_observer.h"
#include "chrome/browser/sync_file_system/logger.h"
#include "chrome/browser/sync_file_system/syncable_file_system_util.h"
#include "components/drive/drive_notification_manager.h"
#include "components/drive/drive_uploader.h"
#include "components/drive/service/drive_api_service.h"
#include "components/drive/service/drive_service_interface.h"
#include "components/signin/public/identity_manager/identity_manager.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/device_service.h"
#include "content/public/browser/network_service_instance.h"
#include "content/public/browser/storage_partition.h"
#include "extensions/browser/extension_registry.h"
#include "extensions/browser/extension_system.h"
#include "extensions/browser/extension_system_provider.h"
#include "extensions/browser/extensions_browser_client.h"
#include "extensions/common/extension.h"
#include "google_apis/drive/drive_api_url_generator.h"
#include "mojo/public/cpp/bindings/pending_remote.h"
#include "net/traffic_annotation/network_traffic_annotation.h"
#include "services/device/public/mojom/wake_lock_provider.mojom.h"
#include "storage/browser/blob/scoped_file.h"
#include "storage/common/file_system/file_system_util.h"
namespace sync_file_system {
class RemoteChangeProcessor;
namespace drive_backend {
constexpr net::NetworkTrafficAnnotationTag kSyncFileSystemTrafficAnnotation =
net::DefineNetworkTrafficAnnotation("sync_file_system", R"(
semantics {
sender: "Sync FileSystem Chrome API"
description:
"Sync FileSystem API provides an isolated FileSystem to Chrome "
"Apps. The contents of the FileSystem are automatically synced "
"among application instances through a hidden folder on Google "
"Drive. This service uploades or downloads these files for "
"synchronization."
trigger:
"When a Chrome App uses Sync FileSystem API, or when a file on "
"Google Drive is modified."
data:
"Files created by Chrome Apps via Sync FileSystem API."
destination: GOOGLE_OWNED_SERVICE
}
policy {
cookies_allowed: NO
setting: "This feature cannot be disabled in settings."
policy_exception_justification: "Not implemented."
})");
std::unique_ptr<drive::DriveServiceInterface>
SyncEngine::DriveServiceFactory::CreateDriveService(
signin::IdentityManager* identity_manager,
scoped_refptr<network::SharedURLLoaderFactory> url_loader_factory,
base::SequencedTaskRunner* blocking_task_runner) {
return std::make_unique<drive::DriveAPIService>(
identity_manager, url_loader_factory, blocking_task_runner,
GURL(google_apis::DriveApiUrlGenerator::kBaseUrlForProduction),
GURL(google_apis::DriveApiUrlGenerator::kBaseThumbnailUrlForProduction),
std::string(), /* custom_user_agent */
kSyncFileSystemTrafficAnnotation);
}
class SyncEngine::WorkerObserver : public SyncWorkerInterface::Observer {
public:
WorkerObserver(base::SequencedTaskRunner* ui_task_runner,
base::WeakPtr<SyncEngine> sync_engine)
: ui_task_runner_(ui_task_runner),
sync_engine_(sync_engine) {
sequence_checker_.DetachFromSequence();
}
~WorkerObserver() override {
DCHECK(sequence_checker_.CalledOnValidSequence());
}
void OnPendingFileListUpdated(int item_count) override {
if (ui_task_runner_->RunsTasksInCurrentSequence()) {
if (sync_engine_)
sync_engine_->OnPendingFileListUpdated(item_count);
return;
}
DCHECK(sequence_checker_.CalledOnValidSequence());
ui_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncEngine::OnPendingFileListUpdated,
sync_engine_, item_count));
}
void OnFileStatusChanged(const storage::FileSystemURL& url,
SyncFileType file_type,
SyncFileStatus file_status,
SyncAction sync_action,
SyncDirection direction) override {
if (ui_task_runner_->RunsTasksInCurrentSequence()) {
if (sync_engine_)
sync_engine_->OnFileStatusChanged(
url, file_type, file_status, sync_action, direction);
return;
}
DCHECK(sequence_checker_.CalledOnValidSequence());
ui_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&SyncEngine::OnFileStatusChanged, sync_engine_, url,
file_type, file_status, sync_action, direction));
}
void UpdateServiceState(RemoteServiceState state,
const std::string& description) override {
if (ui_task_runner_->RunsTasksInCurrentSequence()) {
if (sync_engine_)
sync_engine_->UpdateServiceState(state, description);
return;
}
DCHECK(sequence_checker_.CalledOnValidSequence());
ui_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncEngine::UpdateServiceState, sync_engine_,
state, description));
}
void DetachFromSequence() {
sequence_checker_.DetachFromSequence();
}
private:
scoped_refptr<base::SequencedTaskRunner> ui_task_runner_;
base::WeakPtr<SyncEngine> sync_engine_;
base::SequenceChecker sequence_checker_;
DISALLOW_COPY_AND_ASSIGN(WorkerObserver);
};
namespace {
void DidRegisterOrigin(const base::TimeTicks& start_time,
SyncStatusCallback callback,
SyncStatusCode status) {
base::TimeDelta delta(base::TimeTicks::Now() - start_time);
LOCAL_HISTOGRAM_TIMES("SyncFileSystem.RegisterOriginTime", delta);
std::move(callback).Run(status);
}
} // namespace
std::unique_ptr<SyncEngine> SyncEngine::CreateForBrowserContext(
content::BrowserContext* context,
TaskLogger* task_logger) {
scoped_refptr<base::SingleThreadTaskRunner> ui_task_runner =
base::ThreadTaskRunnerHandle::Get();
scoped_refptr<base::SequencedTaskRunner> worker_task_runner =
base::ThreadPool::CreateSequencedTaskRunner(
{base::MayBlock(), base::TaskPriority::BEST_EFFORT,
base::TaskShutdownBehavior::SKIP_ON_SHUTDOWN});
scoped_refptr<base::SequencedTaskRunner> drive_task_runner =
base::ThreadPool::CreateSequencedTaskRunner(
{base::MayBlock(), base::TaskPriority::BEST_EFFORT,
base::TaskShutdownBehavior::SKIP_ON_SHUTDOWN});
Profile* profile = Profile::FromBrowserContext(context);
drive::DriveNotificationManager* notification_manager =
drive::DriveNotificationManagerFactory::GetForBrowserContext(context);
extensions::ExtensionService* extension_service =
extensions::ExtensionSystem::Get(context)->extension_service();
signin::IdentityManager* identity_manager =
IdentityManagerFactory::GetForProfile(profile);
scoped_refptr<network::SharedURLLoaderFactory> url_loader_factory =
content::BrowserContext::GetDefaultStoragePartition(context)
->GetURLLoaderFactoryForBrowserProcess();
extensions::ExtensionRegistry* extension_registry =
extensions::ExtensionRegistry::Get(context);
// Use WrapUnique instead of std::make_unique because of the private ctor.
auto sync_engine = base::WrapUnique(new SyncEngine(
ui_task_runner.get(), worker_task_runner.get(), drive_task_runner.get(),
GetSyncFileSystemDir(context->GetPath()), task_logger,
notification_manager, extension_service, extension_registry,
identity_manager, url_loader_factory,
std::make_unique<DriveServiceFactory>(), nullptr /* env_override */));
sync_engine->Initialize();
return sync_engine;
}
void SyncEngine::AppendDependsOnFactories(
std::set<BrowserContextKeyedServiceFactory*>* factories) {
DCHECK(factories);
factories->insert(drive::DriveNotificationManagerFactory::GetInstance());
factories->insert(
extensions::ExtensionsBrowserClient::Get()->GetExtensionSystemFactory());
factories->insert(IdentityManagerFactory::GetInstance());
}
SyncEngine::~SyncEngine() {
Reset();
content::GetNetworkConnectionTracker()->RemoveNetworkConnectionObserver(this);
if (identity_manager_)
identity_manager_->RemoveObserver(this);
if (notification_manager_)
notification_manager_->RemoveObserver(this);
}
void SyncEngine::Reset() {
if (drive_service_)
drive_service_->RemoveObserver(this);
worker_task_runner_->DeleteSoon(FROM_HERE, sync_worker_.release());
worker_task_runner_->DeleteSoon(FROM_HERE, worker_observer_.release());
worker_task_runner_->DeleteSoon(FROM_HERE,
remote_change_processor_on_worker_.release());
drive_service_wrapper_.reset();
drive_service_.reset();
drive_uploader_wrapper_.reset();
drive_uploader_.reset();
remote_change_processor_wrapper_.reset();
callback_tracker_.AbortAll();
}
void SyncEngine::Initialize() {
DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
Reset();
if (!identity_manager_ || !identity_manager_->HasPrimaryAccount())
return;
DCHECK(drive_service_factory_);
std::unique_ptr<drive::DriveServiceInterface> drive_service =
drive_service_factory_->CreateDriveService(
identity_manager_, url_loader_factory_, drive_task_runner_.get());
mojo::PendingRemote<device::mojom::WakeLockProvider> wake_lock_provider;
content::GetDeviceService().BindWakeLockProvider(
wake_lock_provider.InitWithNewPipeAndPassReceiver());
auto drive_uploader = std::make_unique<drive::DriveUploader>(
drive_service.get(), drive_task_runner_.get(),
std::move(wake_lock_provider));
InitializeInternal(std::move(drive_service), std::move(drive_uploader),
nullptr);
}
void SyncEngine::InitializeForTesting(
std::unique_ptr<drive::DriveServiceInterface> drive_service,
std::unique_ptr<drive::DriveUploaderInterface> drive_uploader,
std::unique_ptr<SyncWorkerInterface> sync_worker) {
Reset();
InitializeInternal(std::move(drive_service), std::move(drive_uploader),
std::move(sync_worker));
}
void SyncEngine::InitializeInternal(
std::unique_ptr<drive::DriveServiceInterface> drive_service,
std::unique_ptr<drive::DriveUploaderInterface> drive_uploader,
std::unique_ptr<SyncWorkerInterface> sync_worker) {
drive_service_ = std::move(drive_service);
drive_service_wrapper_ =
std::make_unique<DriveServiceWrapper>(drive_service_.get());
CoreAccountId account_id;
if (identity_manager_)
account_id = identity_manager_->GetPrimaryAccountId();
drive_service_->Initialize(account_id);
drive_uploader_ = std::move(drive_uploader);
drive_uploader_wrapper_ =
std::make_unique<DriveUploaderWrapper>(drive_uploader_.get());
// DriveServiceWrapper and DriveServiceOnWorker relay communications
// between DriveService and syncers in SyncWorker.
auto drive_service_on_worker = std::make_unique<DriveServiceOnWorker>(
drive_service_wrapper_->AsWeakPtr(), ui_task_runner_.get(),
worker_task_runner_.get());
auto drive_uploader_on_worker = std::make_unique<DriveUploaderOnWorker>(
drive_uploader_wrapper_->AsWeakPtr(), ui_task_runner_.get(),
worker_task_runner_.get());
auto sync_engine_context = std::make_unique<SyncEngineContext>(
std::move(drive_service_on_worker), std::move(drive_uploader_on_worker),
task_logger_, ui_task_runner_.get(), worker_task_runner_.get());
worker_observer_ = std::make_unique<WorkerObserver>(
ui_task_runner_.get(), weak_ptr_factory_.GetWeakPtr());
base::WeakPtr<extensions::ExtensionServiceInterface>
extension_service_weak_ptr;
if (extension_service_)
extension_service_weak_ptr = extension_service_->AsWeakPtr();
if (!sync_worker) {
sync_worker = std::make_unique<SyncWorker>(
sync_file_system_dir_, extension_service_weak_ptr, extension_registry_,
env_override_);
}
sync_worker_ = std::move(sync_worker);
sync_worker_->AddObserver(worker_observer_.get());
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::Initialize,
base::Unretained(sync_worker_.get()),
std::move(sync_engine_context)));
if (remote_change_processor_)
SetRemoteChangeProcessor(remote_change_processor_);
drive_service_->AddObserver(this);
service_state_ = REMOTE_SERVICE_TEMPORARY_UNAVAILABLE;
auto connection_type = network::mojom::ConnectionType::CONNECTION_NONE;
if (content::GetNetworkConnectionTracker()->GetConnectionType(
&connection_type, base::BindOnce(&SyncEngine::OnConnectionChanged,
weak_ptr_factory_.GetWeakPtr()))) {
OnConnectionChanged(connection_type);
}
if (drive_service_->HasRefreshToken())
OnReadyToSendRequests();
else
OnRefreshTokenInvalid();
}
void SyncEngine::AddServiceObserver(SyncServiceObserver* observer) {
service_observers_.AddObserver(observer);
}
void SyncEngine::AddFileStatusObserver(FileStatusObserver* observer) {
file_status_observers_.AddObserver(observer);
}
void SyncEngine::RegisterOrigin(const GURL& origin,
SyncStatusCallback callback) {
if (!sync_worker_) {
// TODO(tzik): Record |origin| and retry the registration after late
// sign-in. Then, return SYNC_STATUS_OK.
if (!identity_manager_ || !identity_manager_->HasPrimaryAccount())
std::move(callback).Run(SYNC_STATUS_AUTHENTICATION_FAILED);
else
std::move(callback).Run(SYNC_STATUS_ABORT);
return;
}
SyncStatusCallback relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, base::BindOnce(&DidRegisterOrigin, base::TimeTicks::Now(),
TrackCallback(std::move(callback))));
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::RegisterOrigin,
base::Unretained(sync_worker_.get()), origin,
std::move(relayed_callback)));
}
void SyncEngine::EnableOrigin(const GURL& origin, SyncStatusCallback callback) {
if (!sync_worker_) {
// It's safe to return OK immediately since this is also checked in
// SyncWorker initialization.
std::move(callback).Run(SYNC_STATUS_OK);
return;
}
SyncStatusCallback relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, TrackCallback(std::move(callback)));
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::EnableOrigin,
base::Unretained(sync_worker_.get()), origin,
std::move(relayed_callback)));
}
void SyncEngine::DisableOrigin(const GURL& origin,
SyncStatusCallback callback) {
if (!sync_worker_) {
// It's safe to return OK immediately since this is also checked in
// SyncWorker initialization.
std::move(callback).Run(SYNC_STATUS_OK);
return;
}
SyncStatusCallback relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, TrackCallback(std::move(callback)));
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::DisableOrigin,
base::Unretained(sync_worker_.get()), origin,
std::move(relayed_callback)));
}
void SyncEngine::UninstallOrigin(const GURL& origin,
UninstallFlag flag,
SyncStatusCallback callback) {
if (!sync_worker_) {
// It's safe to return OK immediately since this is also checked in
// SyncWorker initialization.
std::move(callback).Run(SYNC_STATUS_OK);
return;
}
SyncStatusCallback relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, TrackCallback(std::move(callback)));
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::UninstallOrigin,
base::Unretained(sync_worker_.get()), origin,
flag, std::move(relayed_callback)));
}
void SyncEngine::ProcessRemoteChange(SyncFileCallback callback) {
if (GetCurrentState() == REMOTE_SERVICE_DISABLED) {
std::move(callback).Run(SYNC_STATUS_SYNC_DISABLED,
storage::FileSystemURL());
return;
}
if (!sync_worker_) {
std::move(callback).Run(SYNC_STATUS_ABORT, storage::FileSystemURL());
return;
}
auto split_callback = base::SplitOnceCallback(std::move(callback));
base::OnceClosure abort_closure =
base::BindOnce(std::move(split_callback.first), SYNC_STATUS_ABORT,
storage::FileSystemURL());
SyncFileCallback tracked_callback = callback_tracker_.Register(
std::move(abort_closure), std::move(split_callback.second));
SyncFileCallback relayed_callback =
RelayCallbackToCurrentThread(FROM_HERE, std::move(tracked_callback));
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::ProcessRemoteChange,
base::Unretained(sync_worker_.get()),
std::move(relayed_callback)));
}
void SyncEngine::SetRemoteChangeProcessor(RemoteChangeProcessor* processor) {
remote_change_processor_ = processor;
if (!sync_worker_)
return;
remote_change_processor_wrapper_ =
std::make_unique<RemoteChangeProcessorWrapper>(processor);
remote_change_processor_on_worker_ =
std::make_unique<RemoteChangeProcessorOnWorker>(
remote_change_processor_wrapper_->AsWeakPtr(), ui_task_runner_.get(),
worker_task_runner_.get());
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::SetRemoteChangeProcessor,
base::Unretained(sync_worker_.get()),
remote_change_processor_on_worker_.get()));
}
LocalChangeProcessor* SyncEngine::GetLocalChangeProcessor() {
return this;
}
RemoteServiceState SyncEngine::GetCurrentState() const {
if (!sync_enabled_)
return REMOTE_SERVICE_DISABLED;
if (!has_refresh_token_)
return REMOTE_SERVICE_AUTHENTICATION_REQUIRED;
return service_state_;
}
void SyncEngine::GetOriginStatusMap(StatusMapCallback callback) {
if (!sync_worker_) {
std::move(callback).Run(nullptr);
return;
}
auto split_callback = base::SplitOnceCallback(std::move(callback));
base::OnceClosure abort_closure =
base::BindOnce(std::move(split_callback.first), nullptr);
StatusMapCallback tracked_callback = callback_tracker_.Register(
std::move(abort_closure), std::move(split_callback.second));
StatusMapCallback relayed_callback =
RelayCallbackToCurrentThread(FROM_HERE, std::move(tracked_callback));
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::GetOriginStatusMap,
base::Unretained(sync_worker_.get()),
std::move(relayed_callback)));
}
void SyncEngine::DumpFiles(const GURL& origin, ListCallback callback) {
if (!sync_worker_) {
std::move(callback).Run(nullptr);
return;
}
auto split_callback = base::SplitOnceCallback(std::move(callback));
base::OnceClosure abort_closure =
base::BindOnce(std::move(split_callback.first), nullptr);
ListCallback tracked_callback = callback_tracker_.Register(
std::move(abort_closure), std::move(split_callback.second));
PostTaskAndReplyWithResult(
worker_task_runner_.get(), FROM_HERE,
base::BindOnce(&SyncWorkerInterface::DumpFiles,
base::Unretained(sync_worker_.get()), origin),
std::move(tracked_callback));
}
void SyncEngine::DumpDatabase(ListCallback callback) {
if (!sync_worker_) {
std::move(callback).Run(nullptr);
return;
}
auto split_callback = base::SplitOnceCallback(std::move(callback));
base::OnceClosure abort_closure =
base::BindOnce(std::move(split_callback.first), nullptr);
ListCallback tracked_callback = callback_tracker_.Register(
std::move(abort_closure), std::move(split_callback.second));
PostTaskAndReplyWithResult(
worker_task_runner_.get(), FROM_HERE,
base::BindOnce(&SyncWorkerInterface::DumpDatabase,
base::Unretained(sync_worker_.get())),
std::move(tracked_callback));
}
void SyncEngine::SetSyncEnabled(bool sync_enabled) {
if (sync_enabled_ == sync_enabled)
return;
sync_enabled_ = sync_enabled;
if (sync_enabled_) {
if (!sync_worker_)
Initialize();
// Have no login credential.
if (!sync_worker_)
return;
worker_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&SyncWorkerInterface::SetSyncEnabled,
base::Unretained(sync_worker_.get()), sync_enabled_));
return;
}
if (!sync_worker_)
return;
// TODO(tzik): Consider removing SyncWorkerInterface::SetSyncEnabled and
// let SyncEngine handle the flag.
worker_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&SyncWorkerInterface::SetSyncEnabled,
base::Unretained(sync_worker_.get()), sync_enabled_));
Reset();
}
void SyncEngine::PromoteDemotedChanges(base::OnceClosure callback) {
if (!sync_worker_) {
std::move(callback).Run();
return;
}
auto split_callback = base::SplitOnceCallback(std::move(callback));
base::OnceClosure relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, callback_tracker_.Register(std::move(split_callback.first),
std::move(split_callback.second)));
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::PromoteDemotedChanges,
base::Unretained(sync_worker_.get()),
std::move(relayed_callback)));
}
void SyncEngine::ApplyLocalChange(const FileChange& local_change,
const base::FilePath& local_path,
const SyncFileMetadata& local_metadata,
const storage::FileSystemURL& url,
SyncStatusCallback callback) {
if (GetCurrentState() == REMOTE_SERVICE_DISABLED) {
std::move(callback).Run(SYNC_STATUS_SYNC_DISABLED);
return;
}
if (!sync_worker_) {
std::move(callback).Run(SYNC_STATUS_ABORT);
return;
}
SyncStatusCallback relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, TrackCallback(std::move(callback)));
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::ApplyLocalChange,
base::Unretained(sync_worker_.get()),
local_change, local_path, local_metadata, url,
std::move(relayed_callback)));
}
void SyncEngine::OnNotificationReceived(
const std::map<std::string, int64_t>& invalidations) {
OnNotificationTimerFired();
}
void SyncEngine::OnNotificationTimerFired() {
if (!sync_worker_)
return;
worker_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&SyncWorkerInterface::ActivateService,
base::Unretained(sync_worker_.get()), REMOTE_SERVICE_OK,
"Got push notification for Drive"));
}
void SyncEngine::OnPushNotificationEnabled(bool /* enabled */) {}
void SyncEngine::OnReadyToSendRequests() {
has_refresh_token_ = true;
if (!sync_worker_)
return;
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::ActivateService,
base::Unretained(sync_worker_.get()),
REMOTE_SERVICE_OK, "Authenticated"));
}
void SyncEngine::OnRefreshTokenInvalid() {
has_refresh_token_ = false;
if (!sync_worker_)
return;
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::DeactivateService,
base::Unretained(sync_worker_.get()),
"Found invalid refresh token."));
}
void SyncEngine::OnConnectionChanged(network::mojom::ConnectionType type) {
if (!sync_worker_)
return;
bool network_available_old = network_available_;
network_available_ =
(type != network::mojom::ConnectionType::CONNECTION_NONE);
if (!network_available_old && network_available_) {
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::ActivateService,
base::Unretained(sync_worker_.get()),
REMOTE_SERVICE_OK, "Connected"));
} else if (network_available_old && !network_available_) {
worker_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&SyncWorkerInterface::DeactivateService,
base::Unretained(sync_worker_.get()), "Disconnected"));
}
}
void SyncEngine::OnPrimaryAccountSet(
const CoreAccountInfo& primary_account_info) {
Initialize();
}
void SyncEngine::OnPrimaryAccountCleared(
const CoreAccountInfo& previous_primary_account_info) {
Reset();
UpdateServiceState(REMOTE_SERVICE_AUTHENTICATION_REQUIRED,
"User signed out.");
}
SyncEngine::SyncEngine(
const scoped_refptr<base::SingleThreadTaskRunner>& ui_task_runner,
const scoped_refptr<base::SequencedTaskRunner>& worker_task_runner,
const scoped_refptr<base::SequencedTaskRunner>& drive_task_runner,
const base::FilePath& sync_file_system_dir,
TaskLogger* task_logger,
drive::DriveNotificationManager* notification_manager,
extensions::ExtensionServiceInterface* extension_service,
extensions::ExtensionRegistry* extension_registry,
signin::IdentityManager* identity_manager,
scoped_refptr<network::SharedURLLoaderFactory> url_loader_factory,
std::unique_ptr<DriveServiceFactory> drive_service_factory,
leveldb::Env* env_override)
: ui_task_runner_(ui_task_runner),
worker_task_runner_(worker_task_runner),
drive_task_runner_(drive_task_runner),
sync_file_system_dir_(sync_file_system_dir),
task_logger_(task_logger),
notification_manager_(notification_manager),
extension_service_(extension_service),
extension_registry_(extension_registry),
identity_manager_(identity_manager),
url_loader_factory_(url_loader_factory),
drive_service_factory_(std::move(drive_service_factory)),
remote_change_processor_(nullptr),
service_state_(REMOTE_SERVICE_TEMPORARY_UNAVAILABLE),
has_refresh_token_(false),
network_available_(false),
sync_enabled_(false),
env_override_(env_override) {
DCHECK(sync_file_system_dir_.IsAbsolute());
if (notification_manager_)
notification_manager_->AddObserver(this);
if (identity_manager_)
identity_manager_->AddObserver(this);
content::GetNetworkConnectionTracker()->AddNetworkConnectionObserver(this);
}
void SyncEngine::OnPendingFileListUpdated(int item_count) {
for (auto& observer : service_observers_)
observer.OnRemoteChangeQueueUpdated(item_count);
}
void SyncEngine::OnFileStatusChanged(const storage::FileSystemURL& url,
SyncFileType file_type,
SyncFileStatus file_status,
SyncAction sync_action,
SyncDirection direction) {
for (auto& observer : file_status_observers_) {
observer.OnFileStatusChanged(url, file_type, file_status, sync_action,
direction);
}
}
void SyncEngine::UpdateServiceState(RemoteServiceState state,
const std::string& description) {
service_state_ = state;
for (auto& observer : service_observers_)
observer.OnRemoteServiceStateUpdated(GetCurrentState(), description);
}
SyncStatusCallback SyncEngine::TrackCallback(SyncStatusCallback callback) {
auto split_callback = base::SplitOnceCallback(std::move(callback));
return callback_tracker_.Register(
base::BindOnce(std::move(split_callback.first), SYNC_STATUS_ABORT),
std::move(split_callback.second));
}
} // namespace drive_backend
} // namespace sync_file_system