blob: 8b8121d6563599331c339cea83f4d26ee54b2f76 [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/macros.h"
#include "base/metrics/histogram_macros.h"
#include "base/task_scheduler/post_task.h"
#include "base/threading/thread_task_runner_handle.h"
#include "base/time/time.h"
#include "base/values.h"
#include "chrome/browser/browser_process.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/profile_oauth2_token_service_factory.h"
#include "chrome/browser/signin/signin_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/core/browser/profile_oauth2_token_service.h"
#include "components/signin/core/browser/signin_manager.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/storage_partition.h"
#include "content/public/common/service_manager_connection.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/interface_request.h"
#include "net/traffic_annotation/network_traffic_annotation.h"
#include "net/url_request/url_request_context_getter.h"
#include "services/device/public/mojom/constants.mojom.h"
#include "services/device/public/mojom/wake_lock_provider.mojom.h"
#include "services/service_manager/public/cpp/connector.h"
#include "storage/browser/blob/scoped_file.h"
#include "storage/common/fileapi/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(
OAuth2TokenService* oauth2_token_service,
net::URLRequestContextGetter* url_request_context_getter,
scoped_refptr<network::SharedURLLoaderFactory> url_loader_factory,
base::SequencedTaskRunner* blocking_task_runner) {
return std::unique_ptr<
drive::DriveServiceInterface>(new drive::DriveAPIService(
oauth2_token_service, url_request_context_getter, 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,
const SyncStatusCallback& callback,
SyncStatusCode status) {
base::TimeDelta delta(base::TimeTicks::Now() - start_time);
LOCAL_HISTOGRAM_TIMES("SyncFileSystem.RegisterOriginTime", delta);
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::CreateSequencedTaskRunnerWithTraits(
{base::MayBlock(), base::TaskPriority::BEST_EFFORT,
base::TaskShutdownBehavior::SKIP_ON_SHUTDOWN});
scoped_refptr<base::SequencedTaskRunner> drive_task_runner =
base::CreateSequencedTaskRunnerWithTraits(
{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();
SigninManagerBase* signin_manager =
SigninManagerFactory::GetForProfile(profile);
OAuth2TokenService* token_service =
ProfileOAuth2TokenServiceFactory::GetForProfile(profile);
scoped_refptr<net::URLRequestContextGetter> request_context =
content::BrowserContext::GetDefaultStoragePartition(context)->
GetURLRequestContext();
scoped_refptr<network::SharedURLLoaderFactory> url_loader_factory =
content::BrowserContext::GetDefaultStoragePartition(context)
->GetURLLoaderFactoryForBrowserProcess();
std::unique_ptr<drive_backend::SyncEngine> sync_engine(new SyncEngine(
ui_task_runner.get(), worker_task_runner.get(), drive_task_runner.get(),
GetSyncFileSystemDir(context->GetPath()), task_logger,
notification_manager, extension_service, signin_manager, token_service,
request_context.get(), 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(SigninManagerFactory::GetInstance());
factories->insert(
extensions::ExtensionsBrowserClient::Get()->GetExtensionSystemFactory());
factories->insert(ProfileOAuth2TokenServiceFactory::GetInstance());
}
SyncEngine::~SyncEngine() {
Reset();
g_browser_process->network_connection_tracker()
->RemoveNetworkConnectionObserver(this);
if (signin_manager_)
signin_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 (!signin_manager_ || !signin_manager_->IsAuthenticated())
return;
DCHECK(drive_service_factory_);
std::unique_ptr<drive::DriveServiceInterface> drive_service =
drive_service_factory_->CreateDriveService(
token_service_, request_context_.get(), url_loader_factory_,
drive_task_runner_.get());
device::mojom::WakeLockProviderPtr wake_lock_provider(nullptr);
DCHECK(content::ServiceManagerConnection::GetForProcess());
auto* connector =
content::ServiceManagerConnection::GetForProcess()->GetConnector();
connector->BindInterface(device::mojom::kServiceName,
mojo::MakeRequest(&wake_lock_provider));
std::unique_ptr<drive::DriveUploaderInterface> drive_uploader(
new 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_.reset(new DriveServiceWrapper(drive_service_.get()));
std::string account_id;
if (signin_manager_)
account_id = signin_manager_->GetAuthenticatedAccountId();
drive_service_->Initialize(account_id);
drive_uploader_ = std::move(drive_uploader);
drive_uploader_wrapper_.reset(
new DriveUploaderWrapper(drive_uploader_.get()));
// DriveServiceWrapper and DriveServiceOnWorker relay communications
// between DriveService and syncers in SyncWorker.
std::unique_ptr<drive::DriveServiceInterface> drive_service_on_worker(
new DriveServiceOnWorker(drive_service_wrapper_->AsWeakPtr(),
ui_task_runner_.get(),
worker_task_runner_.get()));
std::unique_ptr<drive::DriveUploaderInterface> drive_uploader_on_worker(
new DriveUploaderOnWorker(drive_uploader_wrapper_->AsWeakPtr(),
ui_task_runner_.get(),
worker_task_runner_.get()));
std::unique_ptr<SyncEngineContext> sync_engine_context(new 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_.reset(new 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.reset(new SyncWorker(
sync_file_system_dir_,
extension_service_weak_ptr,
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;
g_browser_process->network_connection_tracker()->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,
const SyncStatusCallback& callback) {
if (!sync_worker_) {
// TODO(tzik): Record |origin| and retry the registration after late
// sign-in. Then, return SYNC_STATUS_OK.
if (!signin_manager_ || !signin_manager_->IsAuthenticated())
callback.Run(SYNC_STATUS_AUTHENTICATION_FAILED);
else
callback.Run(SYNC_STATUS_ABORT);
return;
}
SyncStatusCallback relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, base::Bind(&DidRegisterOrigin, base::TimeTicks::Now(),
TrackCallback(callback)));
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::RegisterOrigin,
base::Unretained(sync_worker_.get()), origin,
relayed_callback));
}
void SyncEngine::EnableOrigin(
const GURL& origin, const SyncStatusCallback& callback) {
if (!sync_worker_) {
// It's safe to return OK immediately since this is also checked in
// SyncWorker initialization.
callback.Run(SYNC_STATUS_OK);
return;
}
SyncStatusCallback relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, TrackCallback(callback));
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::EnableOrigin,
base::Unretained(sync_worker_.get()), origin,
relayed_callback));
}
void SyncEngine::DisableOrigin(
const GURL& origin, const SyncStatusCallback& callback) {
if (!sync_worker_) {
// It's safe to return OK immediately since this is also checked in
// SyncWorker initialization.
callback.Run(SYNC_STATUS_OK);
return;
}
SyncStatusCallback relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, TrackCallback(callback));
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::DisableOrigin,
base::Unretained(sync_worker_.get()), origin,
relayed_callback));
}
void SyncEngine::UninstallOrigin(
const GURL& origin,
UninstallFlag flag,
const SyncStatusCallback& callback) {
if (!sync_worker_) {
// It's safe to return OK immediately since this is also checked in
// SyncWorker initialization.
callback.Run(SYNC_STATUS_OK);
return;
}
SyncStatusCallback relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, TrackCallback(callback));
worker_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&SyncWorkerInterface::UninstallOrigin,
base::Unretained(sync_worker_.get()), origin,
flag, relayed_callback));
}
void SyncEngine::ProcessRemoteChange(const SyncFileCallback& callback) {
if (GetCurrentState() == REMOTE_SERVICE_DISABLED) {
callback.Run(SYNC_STATUS_SYNC_DISABLED, storage::FileSystemURL());
return;
}
base::Closure abort_closure =
base::Bind(callback, SYNC_STATUS_ABORT, storage::FileSystemURL());
if (!sync_worker_) {
abort_closure.Run();
return;
}
SyncFileCallback tracked_callback = callback_tracker_.Register(
abort_closure, callback);
SyncFileCallback relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, tracked_callback);
worker_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&SyncWorkerInterface::ProcessRemoteChange,
base::Unretained(sync_worker_.get()), relayed_callback));
}
void SyncEngine::SetRemoteChangeProcessor(RemoteChangeProcessor* processor) {
remote_change_processor_ = processor;
if (!sync_worker_)
return;
remote_change_processor_wrapper_.reset(
new RemoteChangeProcessorWrapper(processor));
remote_change_processor_on_worker_.reset(new 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(const StatusMapCallback& callback) {
base::Closure abort_closure =
base::Bind(callback, base::Passed(std::unique_ptr<OriginStatusMap>()));
if (!sync_worker_) {
abort_closure.Run();
return;
}
StatusMapCallback tracked_callback =
callback_tracker_.Register(abort_closure, callback);
StatusMapCallback relayed_callback =
RelayCallbackToCurrentThread(FROM_HERE, tracked_callback);
worker_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&SyncWorkerInterface::GetOriginStatusMap,
base::Unretained(sync_worker_.get()), relayed_callback));
}
void SyncEngine::DumpFiles(const GURL& origin,
const ListCallback& callback) {
base::Closure abort_closure =
base::Bind(callback, base::Passed(std::unique_ptr<base::ListValue>()));
if (!sync_worker_) {
abort_closure.Run();
return;
}
ListCallback tracked_callback =
callback_tracker_.Register(abort_closure, callback);
PostTaskAndReplyWithResult(worker_task_runner_.get(),
FROM_HERE,
base::Bind(&SyncWorkerInterface::DumpFiles,
base::Unretained(sync_worker_.get()),
origin),
tracked_callback);
}
void SyncEngine::DumpDatabase(const ListCallback& callback) {
base::Closure abort_closure =
base::Bind(callback, base::Passed(std::unique_ptr<base::ListValue>()));
if (!sync_worker_) {
abort_closure.Run();
return;
}
ListCallback tracked_callback =
callback_tracker_.Register(abort_closure, callback);
PostTaskAndReplyWithResult(worker_task_runner_.get(),
FROM_HERE,
base::Bind(&SyncWorkerInterface::DumpDatabase,
base::Unretained(sync_worker_.get())),
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(const base::Closure& callback) {
if (!sync_worker_) {
callback.Run();
return;
}
base::Closure relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, callback_tracker_.Register(callback, callback));
worker_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&SyncWorkerInterface::PromoteDemotedChanges,
base::Unretained(sync_worker_.get()), relayed_callback));
}
void SyncEngine::ApplyLocalChange(const FileChange& local_change,
const base::FilePath& local_path,
const SyncFileMetadata& local_metadata,
const storage::FileSystemURL& url,
const SyncStatusCallback& callback) {
if (GetCurrentState() == REMOTE_SERVICE_DISABLED) {
callback.Run(SYNC_STATUS_SYNC_DISABLED);
return;
}
if (!sync_worker_) {
callback.Run(SYNC_STATUS_ABORT);
return;
}
SyncStatusCallback relayed_callback = RelayCallbackToCurrentThread(
FROM_HERE, TrackCallback(callback));
worker_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&SyncWorkerInterface::ApplyLocalChange,
base::Unretained(sync_worker_.get()), local_change,
local_path, local_metadata, url, relayed_callback));
}
void SyncEngine::OnNotificationReceived() {
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::GoogleSigninFailed(const GoogleServiceAuthError& error) {
Reset();
UpdateServiceState(REMOTE_SERVICE_AUTHENTICATION_REQUIRED,
"Failed to sign in.");
}
void SyncEngine::GoogleSigninSucceeded(const std::string& account_id,
const std::string& username) {
Initialize();
}
void SyncEngine::GoogleSignedOut(const std::string& account_id,
const std::string& username) {
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,
SigninManagerBase* signin_manager,
OAuth2TokenService* token_service,
net::URLRequestContextGetter* request_context,
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),
signin_manager_(signin_manager),
token_service_(token_service),
request_context_(request_context),
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),
weak_ptr_factory_(this) {
DCHECK(sync_file_system_dir_.IsAbsolute());
if (notification_manager_)
notification_manager_->AddObserver(this);
if (signin_manager_)
signin_manager_->AddObserver(this);
g_browser_process->network_connection_tracker()->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(
const SyncStatusCallback& callback) {
return callback_tracker_.Register(
base::Bind(callback, SYNC_STATUS_ABORT),
callback);
}
} // namespace drive_backend
} // namespace sync_file_system