This source file includes following definitions.
- Traversal
- ExpandToInclude
- top
- begin_children
- end_children
- PushAddedItem
- PushDeletedItem
- PushUpdatedItem
- SetExtraDataForId
- SetSpecificsForId
- Clear
- IsEmpty
- GetAllChangesInTreeOrder
#include "sync/internal_api/change_reorder_buffer.h"
#include <limits>
#include <queue>
#include <set>
#include <utility>
#include "sync/internal_api/public/base_node.h"
#include "sync/internal_api/public/base_transaction.h"
#include "sync/syncable/entry.h"
#include "sync/syncable/syncable_base_transaction.h"
using std::numeric_limits;
using std::pair;
using std::queue;
using std::set;
namespace syncer {
class ChangeReorderBuffer::Traversal {
public:
typedef pair<int64, int64> ParentChildLink;
typedef set<ParentChildLink> LinkSet;
Traversal() : top_(kInvalidId) { }
void ExpandToInclude(syncable::BaseTransaction* trans,
int64 child_handle) {
if (top_ == kInvalidId) {
top_ = child_handle;
return;
}
int64 node_to_include = child_handle;
while (node_to_include != kInvalidId && node_to_include != top_) {
int64 node_parent = 0;
syncable::Entry node(trans, syncable::GET_BY_HANDLE, node_to_include);
CHECK(node.good());
if (node.GetId().IsRoot()) {
node_to_include = top_;
top_ = node.GetMetahandle();
} else {
syncable::Entry parent(trans, syncable::GET_BY_ID,
node.GetParentId());
CHECK(parent.good());
node_parent = parent.GetMetahandle();
ParentChildLink link(node_parent, node_to_include);
if (links_.find(link) != links_.end())
return;
links_.insert(link);
node_to_include = node_parent;
}
}
}
int64 top() const { return top_; }
LinkSet::const_iterator begin_children(int64 parent_id) const {
return links_.upper_bound(
ParentChildLink(parent_id, numeric_limits<int64>::min()));
}
LinkSet::const_iterator end_children(int64 parent_id) const {
return begin_children(parent_id + 1);
}
private:
int64 top_;
LinkSet links_;
DISALLOW_COPY_AND_ASSIGN(Traversal);
};
ChangeReorderBuffer::ChangeReorderBuffer() {
}
ChangeReorderBuffer::~ChangeReorderBuffer() {
}
void ChangeReorderBuffer::PushAddedItem(int64 id) {
operations_[id] = ChangeRecord::ACTION_ADD;
}
void ChangeReorderBuffer::PushDeletedItem(int64 id) {
operations_[id] = ChangeRecord::ACTION_DELETE;
}
void ChangeReorderBuffer::PushUpdatedItem(int64 id) {
operations_[id] = ChangeRecord::ACTION_UPDATE;
}
void ChangeReorderBuffer::SetExtraDataForId(
int64 id,
ExtraPasswordChangeRecordData* extra) {
extra_data_[id] = make_linked_ptr<ExtraPasswordChangeRecordData>(extra);
}
void ChangeReorderBuffer::SetSpecificsForId(
int64 id,
const sync_pb::EntitySpecifics& specifics) {
specifics_[id] = specifics;
}
void ChangeReorderBuffer::Clear() {
operations_.clear();
}
bool ChangeReorderBuffer::IsEmpty() const {
return operations_.empty();
}
bool ChangeReorderBuffer::GetAllChangesInTreeOrder(
const BaseTransaction* sync_trans,
ImmutableChangeRecordList* changes) {
syncable::BaseTransaction* trans = sync_trans->GetWrappedTrans();
Traversal traversal;
ChangeRecordList changelist;
OperationMap::const_iterator i;
for (i = operations_.begin(); i != operations_.end(); ++i) {
if (i->second == ChangeRecord::ACTION_DELETE) {
ChangeRecord record;
record.id = i->first;
record.action = i->second;
if (specifics_.find(record.id) != specifics_.end())
record.specifics = specifics_[record.id];
if (extra_data_.find(record.id) != extra_data_.end())
record.extra = extra_data_[record.id];
changelist.push_back(record);
} else {
traversal.ExpandToInclude(trans, i->first);
}
}
queue<int64> to_visit;
to_visit.push(traversal.top());
while (!to_visit.empty()) {
int64 next = to_visit.front();
to_visit.pop();
i = operations_.find(next);
if (i != operations_.end()) {
ChangeRecord record;
record.id = next;
record.action = i->second;
if (specifics_.find(record.id) != specifics_.end())
record.specifics = specifics_[record.id];
if (extra_data_.find(record.id) != extra_data_.end())
record.extra = extra_data_[record.id];
changelist.push_back(record);
}
Traversal::LinkSet::const_iterator j = traversal.begin_children(next);
Traversal::LinkSet::const_iterator end = traversal.end_children(next);
for (; j != end; ++j) {
CHECK(j->first == next);
to_visit.push(j->second);
}
}
*changes = ImmutableChangeRecordList(&changelist);
return true;
}
}