Storage::RecordAccessor - added multiple pointers to RecordAcessors and switching capabilities. Rewired GraphDbAccessor and VersionList accordingly. Added tests for new RecordAccessor functions.
Reviewers: teon.banek, dtomicevic, mislav.bradac, dgleich, buda Reviewed By: buda Subscribers: pullbot Differential Revision: https://phabricator.memgraph.io/D207
This commit is contained in:
parent
863f55dc2f
commit
9ce2081103
@ -38,46 +38,48 @@ void GraphDbAccessor::abort() {
|
|||||||
|
|
||||||
VertexAccessor GraphDbAccessor::insert_vertex() {
|
VertexAccessor GraphDbAccessor::insert_vertex() {
|
||||||
// create a vertex
|
// create a vertex
|
||||||
Vertex *vertex = nullptr;
|
auto vertex_vlist = new mvcc::VersionList<Vertex>(*transaction_);
|
||||||
auto vertex_vlist = new mvcc::VersionList<Vertex>(*transaction_, vertex);
|
|
||||||
|
|
||||||
bool success = db_.vertices_.access().insert(vertex_vlist).second;
|
bool success = db_.vertices_.access().insert(vertex_vlist).second;
|
||||||
if (success) return VertexAccessor(*vertex_vlist, *vertex, *this);
|
if (success) return VertexAccessor(*vertex_vlist, *this);
|
||||||
throw CreationException("Unable to create a Vertex after 5 attempts");
|
throw CreationException("Unable to create a Vertex after 5 attempts");
|
||||||
}
|
}
|
||||||
|
|
||||||
bool GraphDbAccessor::remove_vertex(VertexAccessor &vertex_accessor) {
|
bool GraphDbAccessor::remove_vertex(VertexAccessor &vertex_accessor) {
|
||||||
// TODO consider if this works well with MVCC
|
vertex_accessor.SwitchNew();
|
||||||
if (vertex_accessor.out_degree() > 0 || vertex_accessor.in_degree() > 0)
|
if (vertex_accessor.out_degree() > 0 || vertex_accessor.in_degree() > 0)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
vertex_accessor.vlist_->remove(&vertex_accessor.update(), *transaction_);
|
vertex_accessor.vlist_->remove(vertex_accessor.current_, *transaction_);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void GraphDbAccessor::detach_remove_vertex(VertexAccessor &vertex_accessor) {
|
void GraphDbAccessor::detach_remove_vertex(VertexAccessor &vertex_accessor) {
|
||||||
// removing edges via accessors is both safe
|
vertex_accessor.SwitchNew();
|
||||||
// and it should remove all the pointers in the relevant
|
|
||||||
// vertices (including this one)
|
|
||||||
vertex_accessor.vlist_->remove(&vertex_accessor.update(), *transaction_);
|
|
||||||
for (auto edge_accessor : vertex_accessor.in()) remove_edge(edge_accessor);
|
for (auto edge_accessor : vertex_accessor.in()) remove_edge(edge_accessor);
|
||||||
|
vertex_accessor.SwitchNew();
|
||||||
for (auto edge_accessor : vertex_accessor.out()) remove_edge(edge_accessor);
|
for (auto edge_accessor : vertex_accessor.out()) remove_edge(edge_accessor);
|
||||||
|
vertex_accessor.vlist_->remove(vertex_accessor.SwitchNew().current_, *transaction_);
|
||||||
}
|
}
|
||||||
|
|
||||||
EdgeAccessor GraphDbAccessor::insert_edge(VertexAccessor &from,
|
EdgeAccessor GraphDbAccessor::insert_edge(VertexAccessor &from,
|
||||||
VertexAccessor &to,
|
VertexAccessor &to,
|
||||||
GraphDbTypes::EdgeType edge_type) {
|
GraphDbTypes::EdgeType edge_type) {
|
||||||
// create an edge
|
// create an edge
|
||||||
Edge *edge = nullptr;
|
|
||||||
auto edge_vlist = new mvcc::VersionList<Edge>(
|
auto edge_vlist = new mvcc::VersionList<Edge>(
|
||||||
*transaction_, edge, *from.vlist_, *to.vlist_, edge_type);
|
*transaction_, *from.vlist_, *to.vlist_, edge_type);
|
||||||
|
|
||||||
// set the vertex connections to this edge
|
// ensure that the "from" accessor has the latest version
|
||||||
|
from.SwitchNew();
|
||||||
from.update().out_.emplace_back(edge_vlist);
|
from.update().out_.emplace_back(edge_vlist);
|
||||||
|
// ensure that the "to" accessor has the latest version
|
||||||
|
// WARNING: must do that after the above "from.update()" for cases when
|
||||||
|
// we are creating a cycle and "from" and "to" are the same vlist
|
||||||
|
to.SwitchNew();
|
||||||
to.update().in_.emplace_back(edge_vlist);
|
to.update().in_.emplace_back(edge_vlist);
|
||||||
|
|
||||||
bool success = db_.edges_.access().insert(edge_vlist).second;
|
bool success = db_.edges_.access().insert(edge_vlist).second;
|
||||||
if (success) return EdgeAccessor(*edge_vlist, *edge, *this);
|
if (success) return EdgeAccessor(*edge_vlist, *this);
|
||||||
|
|
||||||
throw CreationException("Unable to create an Edge after 5 attempts");
|
throw CreationException("Unable to create an Edge after 5 attempts");
|
||||||
}
|
}
|
||||||
@ -97,7 +99,7 @@ void swap_out_edge(std::vector<mvcc::VersionList<Edge> *> &edges,
|
|||||||
void GraphDbAccessor::remove_edge(EdgeAccessor &edge_accessor) {
|
void GraphDbAccessor::remove_edge(EdgeAccessor &edge_accessor) {
|
||||||
swap_out_edge(edge_accessor.from().update().out_, edge_accessor.vlist_);
|
swap_out_edge(edge_accessor.from().update().out_, edge_accessor.vlist_);
|
||||||
swap_out_edge(edge_accessor.to().update().in_, edge_accessor.vlist_);
|
swap_out_edge(edge_accessor.to().update().in_, edge_accessor.vlist_);
|
||||||
edge_accessor.vlist_->remove(&edge_accessor.update(), *transaction_);
|
edge_accessor.vlist_->remove(edge_accessor.SwitchNew().current_, *transaction_);
|
||||||
}
|
}
|
||||||
|
|
||||||
GraphDbTypes::Label GraphDbAccessor::label(const std::string &label_name) {
|
GraphDbTypes::Label GraphDbAccessor::label(const std::string &label_name) {
|
||||||
|
@ -77,15 +77,17 @@ class GraphDbAccessor {
|
|||||||
* visible to the current transaction.
|
* visible to the current transaction.
|
||||||
*/
|
*/
|
||||||
auto vertices() {
|
auto vertices() {
|
||||||
// filter out the accessors not visible to the current transaction
|
// wrap version lists into accessors, which will look for visible versions
|
||||||
auto filtered = iter::filter(
|
auto accessors =
|
||||||
[this](auto vlist) { return vlist->find(*transaction_) != nullptr; },
|
iter::imap([this](auto vlist) { return VertexAccessor(*vlist, *this); },
|
||||||
db_.vertices_.access());
|
db_.vertices_.access());
|
||||||
|
|
||||||
// return accessors of the filtered out vlists
|
// filter out the accessors not visible to the current transaction
|
||||||
return iter::imap(
|
return iter::filter(
|
||||||
[this](auto vlist) { return VertexAccessor(*vlist, *this); },
|
[this](const VertexAccessor &accessor) {
|
||||||
std::move(filtered));
|
return accessor.old_ != nullptr;
|
||||||
|
},
|
||||||
|
std::move(accessors));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -111,15 +113,17 @@ class GraphDbAccessor {
|
|||||||
* visible to the current transaction.
|
* visible to the current transaction.
|
||||||
*/
|
*/
|
||||||
auto edges() {
|
auto edges() {
|
||||||
// filter out the accessors not visible to the current transaction
|
// wrap version lists into accessors, which will look for visible versions
|
||||||
auto filtered = iter::filter(
|
auto accessors =
|
||||||
[this](auto vlist) { return vlist->find(*transaction_) != nullptr; },
|
iter::imap([this](auto vlist) { return EdgeAccessor(*vlist, *this); },
|
||||||
db_.edges_.access());
|
db_.edges_.access());
|
||||||
|
|
||||||
// return accessors of the filtered out vlists
|
// filter out the accessors not visible to the current transaction
|
||||||
return iter::imap(
|
return iter::filter(
|
||||||
[this](auto vlist) { return EdgeAccessor(*vlist, *this); },
|
[this](const EdgeAccessor &accessor) {
|
||||||
std::move(filtered));
|
return accessor.old_ != nullptr;
|
||||||
|
},
|
||||||
|
std::move(accessors));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -211,22 +215,48 @@ class GraphDbAccessor {
|
|||||||
void abort();
|
void abort();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Init accessor record with vlist.
|
* Initializes the record pointers in the given accessor.
|
||||||
* @args accessor whose record to initialize.
|
* The old_ and new_ pointers need to be initialized
|
||||||
|
* with appropriate values, and current_ set to old_
|
||||||
|
* if it exists and to new_ otherwise.
|
||||||
*/
|
*/
|
||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
void init_record(RecordAccessor<TRecord> &accessor) {
|
void Reconstruct(RecordAccessor<TRecord> &accessor) {
|
||||||
accessor.record_ = accessor.vlist_->find(*transaction_);
|
accessor.vlist_->find_set_new_old(*transaction_, accessor.old_,
|
||||||
|
accessor.new_);
|
||||||
|
accessor.current_ = accessor.old_ ? accessor.old_ : accessor.new_;
|
||||||
|
// TODO review: we should never use a record accessor that
|
||||||
|
// does not have either old_ or new_ (both are null), but
|
||||||
|
// we can't assert that here because we construct such an accessor
|
||||||
|
// and filter it out in GraphDbAccessor::[Vertices|Edges]
|
||||||
|
// any ideas?
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Update accessor record with vlist.
|
* Update accessor record with vlist.
|
||||||
|
*
|
||||||
|
* It is not legal
|
||||||
|
* to call this function on a Vertex/Edge that has
|
||||||
|
* been deleted in the current transaction+command.
|
||||||
|
*
|
||||||
* @args accessor whose record to update if possible.
|
* @args accessor whose record to update if possible.
|
||||||
*/
|
*/
|
||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
void update(RecordAccessor<TRecord> &accessor) {
|
void update(RecordAccessor<TRecord> &accessor) {
|
||||||
if (!accessor.record_->is_visible_write(*transaction_))
|
// can't update a deleted record if:
|
||||||
accessor.record_ = accessor.vlist_->update(*transaction_);
|
// - we only have old_ and it hasn't been deleted
|
||||||
|
// - we have new_ and it hasn't been deleted
|
||||||
|
if (!accessor.new_) {
|
||||||
|
debug_assert(
|
||||||
|
!accessor.old_->is_deleted_by(*transaction_),
|
||||||
|
"Can't update a record deleted in the current transaction+command");
|
||||||
|
} else {
|
||||||
|
debug_assert(
|
||||||
|
!accessor.new_->is_deleted_by(*transaction_),
|
||||||
|
"Can't update a record deleted in the current transaction+command");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!accessor.new_) accessor.new_ = accessor.vlist_->update(*transaction_);
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -125,6 +125,22 @@ class Record : public Version<T> {
|
|||||||
cmd.exp() >= t.cid))); // but not before this command,
|
cmd.exp() >= t.cid))); // but not before this command,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* True if this record is created in the current command
|
||||||
|
* of the given transaction.
|
||||||
|
*/
|
||||||
|
bool is_created_by(const tx::Transaction &t) {
|
||||||
|
return tx.cre() == t.id && cmd.cre() == t.cid;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* True if this record is deleted in the current command
|
||||||
|
* of the given transaction.
|
||||||
|
*/
|
||||||
|
bool is_deleted_by(const tx::Transaction &t) {
|
||||||
|
return tx.exp() == t.id && cmd.exp() == t.cid;
|
||||||
|
}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
template <class U>
|
template <class U>
|
||||||
bool committed(U &hints, const Id &id, const tx::Transaction &t) {
|
bool committed(U &hints, const Id &id, const tx::Transaction &t) {
|
||||||
|
@ -18,13 +18,18 @@ class VersionList {
|
|||||||
|
|
||||||
/* @brief Constructor that is used to insert one item into VersionList.
|
/* @brief Constructor that is used to insert one item into VersionList.
|
||||||
@param t - transaction
|
@param t - transaction
|
||||||
@param T item - item which will point to new and only entry in
|
@param args - args forwarded to constructor of item T (for
|
||||||
version_list.
|
creating the first Record (Version) in this VersionList.
|
||||||
@param args - args forwarded to constructor of item T.
|
|
||||||
*/
|
*/
|
||||||
template <typename... Args>
|
template <typename... Args>
|
||||||
VersionList(tx::Transaction &t, T *&item, Args &&... args) {
|
VersionList(tx::Transaction &t, Args &&... args) {
|
||||||
item = insert(t, std::forward<Args>(args)...);
|
// TODO replace 'new' with something better
|
||||||
|
auto v1 = new T(std::forward<Args>(args)...);
|
||||||
|
|
||||||
|
// mark the record as created by the transaction t
|
||||||
|
v1->mark_created(t);
|
||||||
|
|
||||||
|
head.store(v1, std::memory_order_seq_cst);
|
||||||
}
|
}
|
||||||
|
|
||||||
VersionList() = delete;
|
VersionList() = delete;
|
||||||
@ -127,6 +132,34 @@ class VersionList {
|
|||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Looks for and sets two versions. The 'old' version is the
|
||||||
|
* newest version that is visible by the current transaction+command,
|
||||||
|
* but has not been created by it. The 'new' version is the version
|
||||||
|
* that has been created by current transaction+command.
|
||||||
|
*
|
||||||
|
* It is possible that both, either or neither are found:
|
||||||
|
* - both are found when an existing record has been modified
|
||||||
|
* - only old is found when an existing record has not been modified
|
||||||
|
* - only new is found when the whole vlist was created
|
||||||
|
* - neither is found when for example the record has been deleted but not
|
||||||
|
* garbage collected yet
|
||||||
|
*
|
||||||
|
* @param t The transaction
|
||||||
|
*/
|
||||||
|
void find_set_new_old(const tx::Transaction &t, T *&old_ref,
|
||||||
|
T *&new_ref) const {
|
||||||
|
// assume that the sought old record is further down the list
|
||||||
|
// from new record, so that if we found old we can stop looking
|
||||||
|
new_ref = nullptr;
|
||||||
|
old_ref = head.load(std::memory_order_seq_cst);
|
||||||
|
while (old_ref != nullptr && !old_ref->visible(t)) {
|
||||||
|
if (!new_ref && old_ref->is_created_by(t))
|
||||||
|
new_ref = old_ref;
|
||||||
|
old_ref = old_ref->next(std::memory_order_seq_cst);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
T *update(tx::Transaction &t) {
|
T *update(tx::Transaction &t) {
|
||||||
debug_assert(head != nullptr, "Head is nullptr on update.");
|
debug_assert(head != nullptr, "Head is nullptr on update.");
|
||||||
auto record = find(t);
|
auto record = find(t);
|
||||||
@ -190,27 +223,6 @@ class VersionList {
|
|||||||
throw SerializationError();
|
throw SerializationError();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* This is private because this should only be called from the constructor.
|
|
||||||
* Otherwise head might be nullptr while version_list exists and it could
|
|
||||||
* interfere with GC.
|
|
||||||
* @tparam Args forwarded to the constructor of T
|
|
||||||
*/
|
|
||||||
template <typename... Args>
|
|
||||||
T *insert(tx::Transaction &t, Args &&... args) {
|
|
||||||
debug_assert(head == nullptr, "Head is not nullptr on creation.");
|
|
||||||
|
|
||||||
// create a first version of the record
|
|
||||||
// TODO replace 'new' with something better
|
|
||||||
auto v1 = new T(std::forward<Args>(args)...);
|
|
||||||
|
|
||||||
// mark the record as created by the transaction t
|
|
||||||
v1->mark_created(t);
|
|
||||||
|
|
||||||
head.store(v1, std::memory_order_seq_cst);
|
|
||||||
return v1;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::atomic<T *> head{nullptr};
|
std::atomic<T *> head{nullptr};
|
||||||
RecordLock lock;
|
RecordLock lock;
|
||||||
};
|
};
|
||||||
|
@ -5,12 +5,12 @@ void EdgeAccessor::set_edge_type(GraphDbTypes::EdgeType edge_type) {
|
|||||||
update().edge_type_ = edge_type;
|
update().edge_type_ = edge_type;
|
||||||
}
|
}
|
||||||
|
|
||||||
GraphDbTypes::EdgeType EdgeAccessor::edge_type() const { return view().edge_type_; }
|
GraphDbTypes::EdgeType EdgeAccessor::edge_type() const { return current().edge_type_; }
|
||||||
|
|
||||||
VertexAccessor EdgeAccessor::from() const {
|
VertexAccessor EdgeAccessor::from() const {
|
||||||
return VertexAccessor(view().from_, db_accessor());
|
return VertexAccessor(current().from_, db_accessor());
|
||||||
}
|
}
|
||||||
|
|
||||||
VertexAccessor EdgeAccessor::to() const {
|
VertexAccessor EdgeAccessor::to() const {
|
||||||
return VertexAccessor(view().to_, db_accessor());
|
return VertexAccessor(current().to_, db_accessor());
|
||||||
}
|
}
|
||||||
|
@ -7,23 +7,14 @@
|
|||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
RecordAccessor<TRecord>::RecordAccessor(mvcc::VersionList<TRecord> &vlist,
|
RecordAccessor<TRecord>::RecordAccessor(mvcc::VersionList<TRecord> &vlist,
|
||||||
GraphDbAccessor &db_accessor)
|
GraphDbAccessor &db_accessor)
|
||||||
: db_accessor_(&db_accessor), vlist_(&vlist), record_(nullptr) {
|
: db_accessor_(&db_accessor), vlist_(&vlist) {
|
||||||
db_accessor.init_record(*this);
|
Reconstruct();
|
||||||
debug_assert(record_ != nullptr, "Record is nullptr.");
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename TRecord>
|
|
||||||
RecordAccessor<TRecord>::RecordAccessor(mvcc::VersionList<TRecord> &vlist,
|
|
||||||
TRecord &record,
|
|
||||||
GraphDbAccessor &db_accessor)
|
|
||||||
: db_accessor_(&db_accessor), vlist_(&vlist), record_(&record) {
|
|
||||||
debug_assert(record_ != nullptr, "Record is nullptr.");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
const PropertyValue &RecordAccessor<TRecord>::PropsAt(
|
const PropertyValue &RecordAccessor<TRecord>::PropsAt(
|
||||||
GraphDbTypes::Property key) const {
|
GraphDbTypes::Property key) const {
|
||||||
return view().properties_.at(key);
|
return current().properties_.at(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
@ -39,7 +30,7 @@ void RecordAccessor<TRecord>::PropsClear() {
|
|||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
const PropertyValueStore<GraphDbTypes::Property>
|
const PropertyValueStore<GraphDbTypes::Property>
|
||||||
&RecordAccessor<TRecord>::Properties() const {
|
&RecordAccessor<TRecord>::Properties() const {
|
||||||
return view().properties_;
|
return current().properties_;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
@ -48,7 +39,7 @@ void RecordAccessor<TRecord>::PropertiesAccept(
|
|||||||
const PropertyValue &prop)>
|
const PropertyValue &prop)>
|
||||||
handler,
|
handler,
|
||||||
std::function<void()> finish) const {
|
std::function<void()> finish) const {
|
||||||
view().properties_.Accept(handler, finish);
|
current().properties_.Accept(handler, finish);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
@ -58,35 +49,54 @@ GraphDbAccessor &RecordAccessor<TRecord>::db_accessor() const {
|
|||||||
|
|
||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
const uint64_t RecordAccessor<TRecord>::temporary_id() const {
|
const uint64_t RecordAccessor<TRecord>::temporary_id() const {
|
||||||
return (uint64_t) vlist_;
|
return (uint64_t)vlist_;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
RecordAccessor<TRecord> &RecordAccessor<TRecord>::SwitchNew() {
|
RecordAccessor<TRecord> &RecordAccessor<TRecord>::SwitchNew() {
|
||||||
// TODO implement
|
if (!new_) {
|
||||||
|
// if new_ is not set yet, look for it
|
||||||
|
// we can just Reconstruct the pointers, old_ will get initialized
|
||||||
|
// to the same value as it has now, and the amount of work is the
|
||||||
|
// same as just looking for a new_ record
|
||||||
|
Reconstruct();
|
||||||
|
}
|
||||||
|
// set new if we have it
|
||||||
|
// if we don't then current_ is old_ and remains there
|
||||||
|
if (new_) current_ = new_;
|
||||||
|
debug_assert(current_ != nullptr,
|
||||||
|
"RecordAccessor::SwitchNew - current_ is nullptr");
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
RecordAccessor<TRecord> &RecordAccessor<TRecord>::SwitchOld() {
|
RecordAccessor<TRecord> &RecordAccessor<TRecord>::SwitchOld() {
|
||||||
// TODO implement
|
// if this whole record is new (new version-list) then we don't
|
||||||
|
// have a valid old_ version. in such a situation SwitchOld
|
||||||
|
// is not a legal function call
|
||||||
|
debug_assert(old_ != nullptr,
|
||||||
|
"RecordAccessor.old_ is nullptr and SwitchOld called");
|
||||||
|
current_ = old_;
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
void RecordAccessor<TRecord>::Reconstruct() {
|
void RecordAccessor<TRecord>::Reconstruct() {
|
||||||
// TODO implement
|
db_accessor().Reconstruct(*this);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
TRecord &RecordAccessor<TRecord>::update() {
|
TRecord &RecordAccessor<TRecord>::update() {
|
||||||
db_accessor().update(*this);
|
db_accessor().update(*this);
|
||||||
return *record_;
|
debug_assert(new_ != nullptr, "RecordAccessor.new_ is null after update");
|
||||||
|
return *new_;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename TRecord>
|
template <typename TRecord>
|
||||||
const TRecord &RecordAccessor<TRecord>::view() const {
|
const TRecord &RecordAccessor<TRecord>::current() const {
|
||||||
return *record_;
|
debug_assert(current_ != nullptr,
|
||||||
|
"RecordAccessor.current_ pointer is nullptr");
|
||||||
|
return *current_;
|
||||||
}
|
}
|
||||||
|
|
||||||
template class RecordAccessor<Vertex>;
|
template class RecordAccessor<Vertex>;
|
||||||
|
@ -39,18 +39,6 @@ class RecordAccessor {
|
|||||||
RecordAccessor(mvcc::VersionList<TRecord> &vlist,
|
RecordAccessor(mvcc::VersionList<TRecord> &vlist,
|
||||||
GraphDbAccessor &db_accessor);
|
GraphDbAccessor &db_accessor);
|
||||||
|
|
||||||
/**
|
|
||||||
* @param vlist MVCC record that this accessor wraps.
|
|
||||||
* @param record MVCC version (that is viewable from this
|
|
||||||
* db_accessor.transaction)
|
|
||||||
* of the given record. Slightly more optimal then the constructor that does
|
|
||||||
* not
|
|
||||||
* accept an already found record.
|
|
||||||
* @param db_accessor The DB accessor that "owns" this record accessor.
|
|
||||||
*/
|
|
||||||
RecordAccessor(mvcc::VersionList<TRecord> &vlist, TRecord &record,
|
|
||||||
GraphDbAccessor &db_accessor);
|
|
||||||
|
|
||||||
// this class is default copyable, movable and assignable
|
// this class is default copyable, movable and assignable
|
||||||
RecordAccessor(const RecordAccessor &other) = default;
|
RecordAccessor(const RecordAccessor &other) = default;
|
||||||
RecordAccessor(RecordAccessor &&other) = default;
|
RecordAccessor(RecordAccessor &&other) = default;
|
||||||
@ -160,6 +148,9 @@ class RecordAccessor {
|
|||||||
* Switches this record accessor to use the old
|
* Switches this record accessor to use the old
|
||||||
* (not updated) version visible to the current transaction+command.
|
* (not updated) version visible to the current transaction+command.
|
||||||
*
|
*
|
||||||
|
* It is not legal to call this function on a Vertex/Edge that
|
||||||
|
* was created by the current transaction+command.
|
||||||
|
*
|
||||||
* @return A reference to this.
|
* @return A reference to this.
|
||||||
*/
|
*/
|
||||||
RecordAccessor<TRecord> &SwitchOld();
|
RecordAccessor<TRecord> &SwitchOld();
|
||||||
@ -175,18 +166,21 @@ class RecordAccessor {
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
/**
|
/**
|
||||||
* Returns the update-ready version of the record.
|
* Ensures there is an updateable version of the record
|
||||||
|
* in the version_list, and that the `new_` pointer
|
||||||
|
* points to it. Returns a reference to that version.
|
||||||
*
|
*
|
||||||
* @return See above.
|
* @return See above.
|
||||||
*/
|
*/
|
||||||
TRecord &update();
|
TRecord &update();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a version of the record that is only for viewing.
|
* Returns the current version (either new_ or old_)
|
||||||
|
* set on this RecordAccessor.
|
||||||
*
|
*
|
||||||
* @return See above.
|
* @return See above.
|
||||||
*/
|
*/
|
||||||
const TRecord &view() const;
|
const TRecord ¤t() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// The database accessor for which this record accessor is created
|
// The database accessor for which this record accessor is created
|
||||||
@ -198,16 +192,34 @@ class RecordAccessor {
|
|||||||
// Immutable, set in the constructor and never changed.
|
// Immutable, set in the constructor and never changed.
|
||||||
mvcc::VersionList<TRecord> *vlist_;
|
mvcc::VersionList<TRecord> *vlist_;
|
||||||
|
|
||||||
/* The version of the record currently used in this transaction. Defaults to
|
/**
|
||||||
* the
|
* Latest version which is visible to the current transaction+command
|
||||||
* latest viewable version (set in the constructor). After the first update
|
* but has not been created nor modified by the current transaction+command.
|
||||||
* done
|
|
||||||
* through this accessor a new, editable version, is created for this
|
|
||||||
* transaction,
|
|
||||||
* and set as the value of this variable.
|
|
||||||
*
|
*
|
||||||
* Stored as a pointer due to it's mutability (the update() function changes
|
* Can be null only when the record itself (the version-list) has
|
||||||
* it).
|
* been created by the current transaction+command.
|
||||||
*/
|
*/
|
||||||
TRecord *record_;
|
TRecord *old_{nullptr};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Version that has been modified (created or updated) by the current
|
||||||
|
* transaction+command.
|
||||||
|
*
|
||||||
|
* Can be null when the record has not been modified in the current
|
||||||
|
* transaction+command. It is also possible that the modification
|
||||||
|
* has happened, but this RecordAccessor does not know this. To
|
||||||
|
* ensure correctness, the `SwitchNew` function must check if this
|
||||||
|
* is null, and if it is it must check with the vlist_ if there is
|
||||||
|
* an update.
|
||||||
|
*/
|
||||||
|
TRecord *new_{nullptr};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pointer to the version (either old_ or new_) that READ operations
|
||||||
|
* in the accessor should take data from. Note that WRITE operations
|
||||||
|
* should always use new_.
|
||||||
|
*
|
||||||
|
* This pointer can never ever be null.
|
||||||
|
*/
|
||||||
|
TRecord *current_{nullptr};
|
||||||
};
|
};
|
||||||
|
@ -5,12 +5,12 @@
|
|||||||
#include "storage/util.hpp"
|
#include "storage/util.hpp"
|
||||||
#include "storage/vertex_accessor.hpp"
|
#include "storage/vertex_accessor.hpp"
|
||||||
|
|
||||||
size_t VertexAccessor::out_degree() const { return view().out_.size(); }
|
size_t VertexAccessor::out_degree() const { return current().out_.size(); }
|
||||||
|
|
||||||
size_t VertexAccessor::in_degree() const { return view().in_.size(); }
|
size_t VertexAccessor::in_degree() const { return current().in_.size(); }
|
||||||
|
|
||||||
bool VertexAccessor::add_label(GraphDbTypes::Label label) {
|
bool VertexAccessor::add_label(GraphDbTypes::Label label) {
|
||||||
auto &labels_view = view().labels_;
|
auto &labels_view = current().labels_;
|
||||||
auto found = std::find(labels_view.begin(), labels_view.end(), label);
|
auto found = std::find(labels_view.begin(), labels_view.end(), label);
|
||||||
if (found != labels_view.end()) return false;
|
if (found != labels_view.end()) return false;
|
||||||
|
|
||||||
@ -31,10 +31,10 @@ size_t VertexAccessor::remove_label(GraphDbTypes::Label label) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool VertexAccessor::has_label(GraphDbTypes::Label label) const {
|
bool VertexAccessor::has_label(GraphDbTypes::Label label) const {
|
||||||
auto &labels = this->view().labels_;
|
auto &labels = this->current().labels_;
|
||||||
return std::find(labels.begin(), labels.end(), label) != labels.end();
|
return std::find(labels.begin(), labels.end(), label) != labels.end();
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::vector<GraphDbTypes::Label> &VertexAccessor::labels() const {
|
const std::vector<GraphDbTypes::Label> &VertexAccessor::labels() const {
|
||||||
return this->view().labels_;
|
return this->current().labels_;
|
||||||
}
|
}
|
||||||
|
@ -66,10 +66,10 @@ class VertexAccessor : public RecordAccessor<Vertex> {
|
|||||||
/**
|
/**
|
||||||
* Returns EdgeAccessors for all incoming edges.
|
* Returns EdgeAccessors for all incoming edges.
|
||||||
*/
|
*/
|
||||||
auto in() { return make_accessor_iterator<EdgeAccessor>(view().in_, db_accessor()); }
|
auto in() { return make_accessor_iterator<EdgeAccessor>(current().in_, db_accessor()); }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns EdgeAccessors for all outgoing edges.
|
* Returns EdgeAccessors for all outgoing edges.
|
||||||
*/
|
*/
|
||||||
auto out() { return make_accessor_iterator<EdgeAccessor>(view().out_, db_accessor()); }
|
auto out() { return make_accessor_iterator<EdgeAccessor>(current().out_, db_accessor()); }
|
||||||
};
|
};
|
||||||
|
@ -162,38 +162,42 @@ TEST(GraphDbAccessorTest, DetachRemoveVertex) {
|
|||||||
Dbms dbms;
|
Dbms dbms;
|
||||||
auto dba = dbms.active();
|
auto dba = dbms.active();
|
||||||
|
|
||||||
// setup (v1)- []->(v2)<-[]-(v3)<-[]-(v4)
|
// setup (v0)- []->(v1)<-[]-(v2)<-[]-(v3)
|
||||||
auto va1 = dba->insert_vertex();
|
std::vector<VertexAccessor> vertices;
|
||||||
auto va2 = dba->insert_vertex();
|
for (int i = 0; i < 4; ++i) vertices.emplace_back(dba->insert_vertex());
|
||||||
auto va3 = dba->insert_vertex();
|
|
||||||
auto va4 = dba->insert_vertex();
|
|
||||||
auto edge_type = dba->edge_type("type");
|
auto edge_type = dba->edge_type("type");
|
||||||
dba->insert_edge(va1, va2, edge_type);
|
dba->insert_edge(vertices[0], vertices[1], edge_type);
|
||||||
dba->insert_edge(va1, va3, edge_type);
|
dba->insert_edge(vertices[2], vertices[1], edge_type);
|
||||||
dba->insert_edge(va4, va3, edge_type);
|
dba->insert_edge(vertices[3], vertices[2], edge_type);
|
||||||
|
|
||||||
dba->advance_command();
|
dba->advance_command();
|
||||||
|
for (auto &vertex : vertices) vertex.Reconstruct();
|
||||||
|
|
||||||
// ensure that plain remove does NOT work
|
// ensure that plain remove does NOT work
|
||||||
EXPECT_EQ(CountVertices(*dba), 4);
|
EXPECT_EQ(CountVertices(*dba), 4);
|
||||||
EXPECT_EQ(CountEdges(*dba), 3);
|
EXPECT_EQ(CountEdges(*dba), 3);
|
||||||
EXPECT_FALSE(dba->remove_vertex(va1));
|
EXPECT_FALSE(dba->remove_vertex(vertices[0]));
|
||||||
EXPECT_FALSE(dba->remove_vertex(va2));
|
EXPECT_FALSE(dba->remove_vertex(vertices[1]));
|
||||||
EXPECT_FALSE(dba->remove_vertex(va3));
|
EXPECT_FALSE(dba->remove_vertex(vertices[2]));
|
||||||
EXPECT_EQ(CountVertices(*dba), 4);
|
EXPECT_EQ(CountVertices(*dba), 4);
|
||||||
EXPECT_EQ(CountEdges(*dba), 3);
|
EXPECT_EQ(CountEdges(*dba), 3);
|
||||||
|
|
||||||
dba->detach_remove_vertex(va3);
|
dba->detach_remove_vertex(vertices[2]);
|
||||||
dba->advance_command();
|
dba->advance_command();
|
||||||
|
for (auto &vertex : vertices) vertex.Reconstruct();
|
||||||
|
|
||||||
EXPECT_EQ(CountVertices(*dba), 3);
|
EXPECT_EQ(CountVertices(*dba), 3);
|
||||||
EXPECT_EQ(CountEdges(*dba), 1);
|
EXPECT_EQ(CountEdges(*dba), 1);
|
||||||
EXPECT_TRUE(dba->remove_vertex(va4));
|
EXPECT_TRUE(dba->remove_vertex(vertices[3]));
|
||||||
dba->advance_command();
|
dba->advance_command();
|
||||||
|
for (auto &vertex : vertices) vertex.Reconstruct();
|
||||||
|
|
||||||
EXPECT_EQ(CountVertices(*dba), 2);
|
EXPECT_EQ(CountVertices(*dba), 2);
|
||||||
EXPECT_EQ(CountEdges(*dba), 1);
|
EXPECT_EQ(CountEdges(*dba), 1);
|
||||||
for (auto va : dba->vertices()) EXPECT_FALSE(dba->remove_vertex(va));
|
for (auto va : dba->vertices()) EXPECT_FALSE(dba->remove_vertex(va));
|
||||||
dba->advance_command();
|
dba->advance_command();
|
||||||
|
for (auto &vertex : vertices) vertex.Reconstruct();
|
||||||
|
|
||||||
EXPECT_EQ(CountVertices(*dba), 2);
|
EXPECT_EQ(CountVertices(*dba), 2);
|
||||||
EXPECT_EQ(CountEdges(*dba), 1);
|
EXPECT_EQ(CountEdges(*dba), 1);
|
||||||
@ -203,6 +207,7 @@ TEST(GraphDbAccessorTest, DetachRemoveVertex) {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
dba->advance_command();
|
dba->advance_command();
|
||||||
|
for (auto &vertex : vertices) vertex.Reconstruct();
|
||||||
|
|
||||||
EXPECT_EQ(CountVertices(*dba), 1);
|
EXPECT_EQ(CountVertices(*dba), 1);
|
||||||
EXPECT_EQ(CountEdges(*dba), 0);
|
EXPECT_EQ(CountEdges(*dba), 0);
|
||||||
@ -228,12 +233,13 @@ TEST(GraphDbAccessorTest, DetachRemoveVertexMultiple) {
|
|||||||
int N = 7;
|
int N = 7;
|
||||||
std::vector<VertexAccessor> vertices;
|
std::vector<VertexAccessor> vertices;
|
||||||
auto edge_type = dba->edge_type("edge");
|
auto edge_type = dba->edge_type("edge");
|
||||||
for (int i = 0; i < N; ++i)
|
for (int i = 0; i < N; ++i) vertices.emplace_back(dba->insert_vertex());
|
||||||
vertices.emplace_back(dba->insert_vertex());
|
|
||||||
for (int j = 0; j < N; ++j)
|
for (int j = 0; j < N; ++j)
|
||||||
for (int k = 0; k < N; ++k)
|
for (int k = 0; k < N; ++k)
|
||||||
dba->insert_edge(vertices[j], vertices[k], edge_type);
|
dba->insert_edge(vertices[j], vertices[k], edge_type);
|
||||||
|
|
||||||
dba->advance_command();
|
dba->advance_command();
|
||||||
|
for (auto &vertex : vertices) vertex.Reconstruct();
|
||||||
|
|
||||||
EXPECT_EQ(CountVertices(*dba), N);
|
EXPECT_EQ(CountVertices(*dba), N);
|
||||||
EXPECT_EQ(CountEdges(*dba), N * N);
|
EXPECT_EQ(CountEdges(*dba), N * N);
|
||||||
@ -241,6 +247,7 @@ TEST(GraphDbAccessorTest, DetachRemoveVertexMultiple) {
|
|||||||
// detach delete one edge
|
// detach delete one edge
|
||||||
dba->detach_remove_vertex(vertices[0]);
|
dba->detach_remove_vertex(vertices[0]);
|
||||||
dba->advance_command();
|
dba->advance_command();
|
||||||
|
for (auto &vertex : vertices) vertex.Reconstruct();
|
||||||
EXPECT_EQ(CountVertices(*dba), N - 1);
|
EXPECT_EQ(CountVertices(*dba), N - 1);
|
||||||
EXPECT_EQ(CountEdges(*dba), (N - 1) * (N - 1));
|
EXPECT_EQ(CountEdges(*dba), (N - 1) * (N - 1));
|
||||||
|
|
||||||
@ -248,13 +255,14 @@ TEST(GraphDbAccessorTest, DetachRemoveVertexMultiple) {
|
|||||||
dba->detach_remove_vertex(vertices[1]);
|
dba->detach_remove_vertex(vertices[1]);
|
||||||
dba->detach_remove_vertex(vertices[2]);
|
dba->detach_remove_vertex(vertices[2]);
|
||||||
dba->advance_command();
|
dba->advance_command();
|
||||||
|
for (auto &vertex : vertices) vertex.Reconstruct();
|
||||||
EXPECT_EQ(CountVertices(*dba), N - 3);
|
EXPECT_EQ(CountVertices(*dba), N - 3);
|
||||||
EXPECT_EQ(CountEdges(*dba), (N - 3) * (N - 3));
|
EXPECT_EQ(CountEdges(*dba), (N - 3) * (N - 3));
|
||||||
|
|
||||||
// detach delete everything, buwahahahaha
|
// detach delete everything, buwahahahaha
|
||||||
for (int l = 3; l < N ; ++l)
|
for (int l = 3; l < N; ++l) dba->detach_remove_vertex(vertices[l]);
|
||||||
dba->detach_remove_vertex(vertices[l]);
|
|
||||||
dba->advance_command();
|
dba->advance_command();
|
||||||
|
for (auto &vertex : vertices) vertex.Reconstruct();
|
||||||
EXPECT_EQ(CountVertices(*dba), 0);
|
EXPECT_EQ(CountVertices(*dba), 0);
|
||||||
EXPECT_EQ(CountEdges(*dba), 0);
|
EXPECT_EQ(CountEdges(*dba), 0);
|
||||||
}
|
}
|
||||||
@ -303,6 +311,6 @@ TEST(GraphDbAccessorTest, Properties) {
|
|||||||
|
|
||||||
int main(int argc, char **argv) {
|
int main(int argc, char **argv) {
|
||||||
::testing::InitGoogleTest(&argc, argv);
|
::testing::InitGoogleTest(&argc, argv);
|
||||||
// ::testing::GTEST_FLAG(filter) = "*.DetachRemoveVertex";
|
// ::testing::GTEST_FLAG(filter) = "*.DetachRemoveVertex";
|
||||||
return RUN_ALL_TESTS();
|
return RUN_ALL_TESTS();
|
||||||
}
|
}
|
||||||
|
@ -1000,6 +1000,7 @@ TEST(Interpreter, SetLabels) {
|
|||||||
EXPECT_EQ(2, PullAll(label_set, *dba, symbol_table));
|
EXPECT_EQ(2, PullAll(label_set, *dba, symbol_table));
|
||||||
|
|
||||||
for (VertexAccessor vertex : dba->vertices()) {
|
for (VertexAccessor vertex : dba->vertices()) {
|
||||||
|
vertex.SwitchNew();
|
||||||
EXPECT_EQ(3, vertex.labels().size());
|
EXPECT_EQ(3, vertex.labels().size());
|
||||||
EXPECT_TRUE(vertex.has_label(label2));
|
EXPECT_TRUE(vertex.has_label(label2));
|
||||||
EXPECT_TRUE(vertex.has_label(label3));
|
EXPECT_TRUE(vertex.has_label(label3));
|
||||||
@ -1083,6 +1084,7 @@ TEST(Interpreter, RemoveLabels) {
|
|||||||
EXPECT_EQ(2, PullAll(label_remove, *dba, symbol_table));
|
EXPECT_EQ(2, PullAll(label_remove, *dba, symbol_table));
|
||||||
|
|
||||||
for (VertexAccessor vertex : dba->vertices()) {
|
for (VertexAccessor vertex : dba->vertices()) {
|
||||||
|
vertex.SwitchNew();
|
||||||
EXPECT_EQ(1, vertex.labels().size());
|
EXPECT_EQ(1, vertex.labels().size());
|
||||||
EXPECT_FALSE(vertex.has_label(label1));
|
EXPECT_FALSE(vertex.has_label(label1));
|
||||||
EXPECT_FALSE(vertex.has_label(label2));
|
EXPECT_FALSE(vertex.has_label(label2));
|
||||||
|
@ -11,8 +11,7 @@ class Prop : public mvcc::Record<Prop> {};
|
|||||||
TEST(MVCC, Case1Test3) {
|
TEST(MVCC, Case1Test3) {
|
||||||
tx::Engine engine;
|
tx::Engine engine;
|
||||||
auto t1 = engine.begin();
|
auto t1 = engine.begin();
|
||||||
Prop *prop;
|
mvcc::VersionList<Prop> version_list(*t1);
|
||||||
mvcc::VersionList<Prop> version_list(*t1, prop);
|
|
||||||
t1->commit();
|
t1->commit();
|
||||||
|
|
||||||
auto t2 = engine.begin();
|
auto t2 = engine.begin();
|
||||||
@ -28,9 +27,8 @@ TEST(MVCC, Case1Test3) {
|
|||||||
|
|
||||||
TEST(MVCC, InSnapshot) {
|
TEST(MVCC, InSnapshot) {
|
||||||
tx::Engine engine;
|
tx::Engine engine;
|
||||||
Prop *prop;
|
|
||||||
auto t1 = engine.begin();
|
auto t1 = engine.begin();
|
||||||
mvcc::VersionList<Prop> version_list(*t1, prop);
|
mvcc::VersionList<Prop> version_list(*t1);
|
||||||
t1->commit();
|
t1->commit();
|
||||||
|
|
||||||
auto t2 = engine.begin();
|
auto t2 = engine.begin();
|
||||||
|
@ -37,15 +37,14 @@ TEST(VersionList, GcDeleted) {
|
|||||||
std::vector<uint64_t> ids;
|
std::vector<uint64_t> ids;
|
||||||
auto t1 = engine.begin();
|
auto t1 = engine.begin();
|
||||||
std::atomic<int> count{0};
|
std::atomic<int> count{0};
|
||||||
Prop *prop;
|
mvcc::VersionList<Prop> version_list(*t1, count);
|
||||||
mvcc::VersionList<Prop> version_list(*t1, prop, count);
|
|
||||||
ids.push_back(t1->id);
|
ids.push_back(t1->id);
|
||||||
t1->commit();
|
t1->commit();
|
||||||
|
|
||||||
for (int i = 0; i < UPDATES; ++i) {
|
for (int i = 0; i < UPDATES; ++i) {
|
||||||
auto t2 = engine.begin();
|
auto t2 = engine.begin();
|
||||||
ids.push_back(t2->id);
|
ids.push_back(t2->id);
|
||||||
prop = version_list.update(prop, *t2);
|
version_list.update(*t2);
|
||||||
t2->commit();
|
t2->commit();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -72,9 +71,8 @@ TEST(GarbageCollector, WaitAndClean) {
|
|||||||
gc.Run(std::chrono::seconds(1));
|
gc.Run(std::chrono::seconds(1));
|
||||||
|
|
||||||
auto t1 = engine.begin();
|
auto t1 = engine.begin();
|
||||||
Prop *prop;
|
|
||||||
std::atomic<int> count;
|
std::atomic<int> count;
|
||||||
auto vl = new mvcc::VersionList<Prop>(*t1, prop, count);
|
auto vl = new mvcc::VersionList<Prop>(*t1, count);
|
||||||
|
|
||||||
auto access = skiplist.access();
|
auto access = skiplist.access();
|
||||||
access.insert(vl);
|
access.insert(vl);
|
||||||
@ -99,9 +97,8 @@ TEST(GarbageCollector, WaitAndDontClean) {
|
|||||||
// the top one except GC is never run.
|
// the top one except GC is never run.
|
||||||
|
|
||||||
auto t1 = engine.begin();
|
auto t1 = engine.begin();
|
||||||
Prop *prop;
|
|
||||||
std::atomic<int> count;
|
std::atomic<int> count;
|
||||||
auto vl = new mvcc::VersionList<Prop>(*t1, prop, count);
|
auto vl = new mvcc::VersionList<Prop>(*t1, count);
|
||||||
|
|
||||||
auto access = skiplist.access();
|
auto access = skiplist.access();
|
||||||
access.insert(vl);
|
access.insert(vl);
|
||||||
|
@ -77,6 +77,76 @@ TEST(RecordAccessor, RecordLessThan) {
|
|||||||
EXPECT_FALSE(e2 < e2);
|
EXPECT_FALSE(e2 < e2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST(RecordAccessor, SwitchOldAndSwitchNewMemberFunctionTest) {
|
||||||
|
Dbms dbms;
|
||||||
|
|
||||||
|
// test SwitchOld failure on new record, SwitchNew OK
|
||||||
|
{
|
||||||
|
auto dba = dbms.active();
|
||||||
|
auto v1 = dba->insert_vertex();
|
||||||
|
EXPECT_DEATH(v1.SwitchOld(), "");
|
||||||
|
v1.SwitchNew();
|
||||||
|
dba->commit();
|
||||||
|
}
|
||||||
|
|
||||||
|
// test both Switches work on existing record
|
||||||
|
{
|
||||||
|
auto dba = dbms.active();
|
||||||
|
auto v1 = *dba->vertices().begin();
|
||||||
|
v1.SwitchOld();
|
||||||
|
v1.SwitchNew();
|
||||||
|
}
|
||||||
|
|
||||||
|
// ensure switch exposes the right data
|
||||||
|
{
|
||||||
|
auto dba = dbms.active();
|
||||||
|
auto label = dba->label("label");
|
||||||
|
auto v1 = *dba->vertices().begin();
|
||||||
|
|
||||||
|
EXPECT_FALSE(v1.has_label(label)); // old record
|
||||||
|
v1.add_label(label); // modifying data does not switch to new
|
||||||
|
EXPECT_FALSE(v1.has_label(label)); // old record
|
||||||
|
v1.SwitchNew();
|
||||||
|
EXPECT_TRUE(v1.has_label(label));
|
||||||
|
v1.SwitchOld();
|
||||||
|
EXPECT_FALSE(v1.has_label(label));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(RecordAccessor, Reconstruct) {
|
||||||
|
Dbms dbms;
|
||||||
|
auto label = dbms.active()->label("label");
|
||||||
|
|
||||||
|
{
|
||||||
|
// we must operate on an old vertex
|
||||||
|
// because otherwise we only have new
|
||||||
|
// so create a vertex and commit it
|
||||||
|
auto dba = dbms.active();
|
||||||
|
dba->insert_vertex();
|
||||||
|
dba->commit();
|
||||||
|
}
|
||||||
|
|
||||||
|
// ensure we don't have label set
|
||||||
|
auto dba = dbms.active();
|
||||||
|
auto v1 = *dba->vertices().begin();
|
||||||
|
v1.SwitchNew();
|
||||||
|
EXPECT_FALSE(v1.has_label(label));
|
||||||
|
|
||||||
|
{
|
||||||
|
// update the record through a different accessor
|
||||||
|
auto v1_other_accessor = *dba->vertices().begin();
|
||||||
|
v1_other_accessor.add_label(label);
|
||||||
|
EXPECT_FALSE(v1.has_label(label));
|
||||||
|
v1_other_accessor.SwitchNew();
|
||||||
|
EXPECT_TRUE(v1_other_accessor.has_label(label));
|
||||||
|
}
|
||||||
|
|
||||||
|
EXPECT_FALSE(v1.has_label(label));
|
||||||
|
v1.Reconstruct();
|
||||||
|
v1.SwitchNew();
|
||||||
|
EXPECT_TRUE(v1.has_label(label));
|
||||||
|
}
|
||||||
|
|
||||||
TEST(RecordAccessor, VertexLabels) {
|
TEST(RecordAccessor, VertexLabels) {
|
||||||
Dbms dbms;
|
Dbms dbms;
|
||||||
auto dba = dbms.active();
|
auto dba = dbms.active();
|
||||||
|
Loading…
Reference in New Issue
Block a user