// Copyright 2014 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 "sync/internal_api/sync_rollback_manager.h" #include "sync/internal_api/public/base/model_type.h" #include "sync/internal_api/public/read_node.h" #include "sync/internal_api/public/read_transaction.h" #include "sync/internal_api/public/util/syncer_error.h" #include "sync/internal_api/public/write_transaction.h" #include "sync/syncable/directory.h" #include "sync/syncable/mutable_entry.h" #include "url/gurl.h" namespace syncer { SyncRollbackManager::SyncRollbackManager() : change_delegate_(NULL) { } SyncRollbackManager::~SyncRollbackManager() { } void SyncRollbackManager::Init(InitArgs* args) { if (SyncRollbackManagerBase::InitInternal( args->database_location, args->internal_components_factory.get(), InternalComponentsFactory::STORAGE_ON_DISK, args->unrecoverable_error_handler.Pass(), args->report_unrecoverable_error_function)) { change_delegate_ = args->change_delegate; for (size_t i = 0; i < args->workers.size(); ++i) { ModelSafeGroup group = args->workers[i]->GetModelSafeGroup(); CHECK(workers_.find(group) == workers_.end()); workers_[group] = args->workers[i]; } rollback_ready_types_ = GetUserShare()->directory->InitialSyncEndedTypes(); rollback_ready_types_.RetainAll(BackupTypes()); } } void SyncRollbackManager::StartSyncingNormally( const ModelSafeRoutingInfo& routing_info, base::Time last_poll_time){ if (rollback_ready_types_.Empty()) { NotifyRollbackDone(); return; } std::map to_delete; { WriteTransaction trans(FROM_HERE, GetUserShare()); syncable::Directory::Metahandles unsynced; GetUserShare()->directory->GetUnsyncedMetaHandles(trans.GetWrappedTrans(), &unsynced); for (size_t i = 0; i < unsynced.size(); ++i) { syncable::MutableEntry e(trans.GetWrappedWriteTrans(), syncable::GET_BY_HANDLE, unsynced[i]); if (!e.good() || e.GetIsDel() || e.GetId().ServerKnows()) continue; // TODO(haitaol): roll back entries that are backed up but whose content // is merged with local model during association. ModelType type = GetModelTypeFromSpecifics(e.GetSpecifics()); if (!rollback_ready_types_.Has(type)) continue; to_delete[type].push_back(unsynced[i]); } } for (std::map::iterator it = to_delete.begin(); it != to_delete.end(); ++it) { ModelSafeGroup group = routing_info.find(it->first)->second; CHECK(workers_.find(group) != workers_.end()); workers_[group]->DoWorkAndWaitUntilDone( base::Bind(&SyncRollbackManager::DeleteOnWorkerThread, base::Unretained(this), it->first, it->second)); } NotifyRollbackDone(); } SyncerError SyncRollbackManager::DeleteOnWorkerThread( ModelType type, std::vector handles) { CHECK(change_delegate_); { ChangeRecordList deletes; WriteTransaction trans(FROM_HERE, GetUserShare()); for (size_t i = 0; i < handles.size(); ++i) { syncable::MutableEntry e(trans.GetWrappedWriteTrans(), syncable::GET_BY_HANDLE, handles[i]); if (!e.good() || e.GetIsDel()) continue; ChangeRecord del; del.action = ChangeRecord::ACTION_DELETE; del.id = handles[i]; del.specifics = e.GetSpecifics(); deletes.push_back(del); } change_delegate_->OnChangesApplied(type, 1, &trans, MakeImmutable(&deletes)); } change_delegate_->OnChangesComplete(type); return SYNCER_OK; } void SyncRollbackManager::NotifyRollbackDone() { SyncProtocolError error; error.action = ROLLBACK_DONE; FOR_EACH_OBSERVER(SyncManager::Observer, *GetObservers(), OnActionableError(error)); } } // namespace syncer