Summary: What's done: - `RecordAccessor` can represent remote data - `GraphDbAccessor` manages remote data - Cleanup: different `EdgeAccessor lazyness (@dgleich: take a look), unused methods, documentation... - `TODO` placeholders for remote implementation What's not done: - RPC and data transfer - how exactly remote errors are handled - not sure if any MVCC Record info for remote data should be tracked - WAL and RPC Deltas properly handled (Gleich working on extracting `Wal::Op`) This implementation should not break single-node execution, and should provide good abstractions and placeholders for distributed. Once that's satisfied, it should land. Reviewers: dgleich, buda, mislav.bradac Reviewed By: dgleich Subscribers: dgleich, pullbot Differential Revision: https://phabricator.memgraph.io/D1030
205 lines
6.8 KiB
C++
205 lines
6.8 KiB
C++
#pragma once
|
|
|
|
#include "glog/logging.h"
|
|
|
|
#include "database/graph_db_datatypes.hpp"
|
|
#include "mvcc/version_list.hpp"
|
|
#include "storage/address.hpp"
|
|
#include "storage/gid.hpp"
|
|
#include "storage/property_value.hpp"
|
|
#include "storage/property_value_store.hpp"
|
|
#include "utils/total_ordering.hpp"
|
|
|
|
class GraphDbAccessor;
|
|
|
|
/// Mock class for a DB delta.
|
|
// TODO replace with the real thing.
|
|
class GraphStateDelta {
|
|
public:
|
|
/// Indicates what the result of applying the delta to the remote worker
|
|
/// (owner of the Vertex/Edge the delta affects).
|
|
enum class RemoteResult {
|
|
SUCCES,
|
|
SERIALIZATION_ERROR,
|
|
LOCK_TIMEOUT_ERROR
|
|
// TODO: network error?
|
|
};
|
|
};
|
|
|
|
/**
|
|
* An accessor to a database record (an Edge or a Vertex).
|
|
*
|
|
* Exposes view and update functions to the client programmer.
|
|
* Assumes responsibility of doing all the relevant book-keeping
|
|
* (such as index updates etc).
|
|
*
|
|
* @tparam TRecord Type of record (MVCC Version) of the accessor.
|
|
*/
|
|
template <typename TRecord>
|
|
class RecordAccessor : public TotalOrdering<RecordAccessor<TRecord>> {
|
|
using AddressT = storage::Address<mvcc::VersionList<TRecord>>;
|
|
/**
|
|
* The GraphDbAccessor is friend to this accessor so it can
|
|
* operate on it's data (mvcc version-list and the record itself).
|
|
* This is legitemate because GraphDbAccessor creates RecordAccessors
|
|
* and is semantically their parent/owner. It is necessary because
|
|
* the GraphDbAccessor handles insertions and deletions, and these
|
|
* operations modify data intensively.
|
|
*/
|
|
friend GraphDbAccessor;
|
|
|
|
public:
|
|
/**
|
|
* @param address Address (local or global) of the Vertex/Edge of this
|
|
* accessor.
|
|
* @param db_accessor The DB accessor that "owns" this record accessor.
|
|
*/
|
|
RecordAccessor(AddressT address, GraphDbAccessor &db_accessor);
|
|
|
|
// this class is default copyable, movable and assignable
|
|
RecordAccessor(const RecordAccessor &other) = default;
|
|
RecordAccessor(RecordAccessor &&other) = default;
|
|
RecordAccessor &operator=(const RecordAccessor &other) = default;
|
|
RecordAccessor &operator=(RecordAccessor &&other) = default;
|
|
|
|
/** Gets the property for the given key. */
|
|
const PropertyValue &PropsAt(GraphDbTypes::Property key) const;
|
|
|
|
/** Sets a value on the record for the given property. */
|
|
void PropsSet(GraphDbTypes::Property key, PropertyValue value);
|
|
|
|
/** Erases the property for the given key. */
|
|
size_t PropsErase(GraphDbTypes::Property key);
|
|
|
|
/** Removes all the properties from this record. */
|
|
void PropsClear();
|
|
|
|
/** Returns the properties of this record. */
|
|
const PropertyValueStore<GraphDbTypes::Property> &Properties() const;
|
|
|
|
bool operator==(const RecordAccessor &other) const;
|
|
|
|
/** Returns a GraphDB accessor of this record accessor. */
|
|
GraphDbAccessor &db_accessor() const;
|
|
|
|
/**
|
|
* Returns a globally-unique ID of this vertex or edge. Note that vertices
|
|
* and edges have separate ID domains, there can be a vertex with ID X and an
|
|
* edge with the same id.
|
|
*/
|
|
gid::Gid gid() const;
|
|
|
|
AddressT address() const;
|
|
|
|
/*
|
|
* Switches this record accessor to use the latest version visible to the
|
|
* current transaction+command. Possibly the one that was created by this
|
|
* transaction+command.
|
|
*
|
|
* @return A reference to this.
|
|
*/
|
|
RecordAccessor<TRecord> &SwitchNew();
|
|
|
|
/**
|
|
* Attempts to switch this accessor to use the latest version not updated by
|
|
* the current transaction+command. If that is not possible (vertex/edge was
|
|
* created by the current transaction/command), it does nothing (current
|
|
* remains pointing to the new version).
|
|
*
|
|
* @return A reference to this.
|
|
*/
|
|
RecordAccessor<TRecord> &SwitchOld();
|
|
|
|
/**
|
|
* Reconstructs the internal state of the record accessor so it uses the
|
|
* versions appropriate to this transaction+command.
|
|
*
|
|
* @return True if this accessor is valid after reconstruction. This means
|
|
* that at least one record pointer was found (either new_ or old_), possibly
|
|
* both.
|
|
*/
|
|
bool Reconstruct() const;
|
|
|
|
/**
|
|
* Returns true if the given accessor is visible to the given transaction.
|
|
*
|
|
* @param current_state If true then the graph state for the
|
|
* current transaction+command is returned (insertions, updates and
|
|
* deletions performed in the current transaction+command are not
|
|
* ignored).
|
|
*/
|
|
bool Visible(const tx::Transaction &t, bool current_state) const {
|
|
return (old_ && !(current_state && old_->is_expired_by(t))) ||
|
|
(current_state && new_ && !new_->is_expired_by(t));
|
|
}
|
|
|
|
protected:
|
|
/**
|
|
* 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 be null if created by an accessor which lazily reads from
|
|
* mvcc.
|
|
*/
|
|
mutable TRecord *current_{nullptr};
|
|
|
|
/**
|
|
* 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.
|
|
*
|
|
* It is not legal to call this function on a Vertex/Edge that has been
|
|
* deleted in the current transaction+command.
|
|
*/
|
|
TRecord &update() const;
|
|
|
|
/** Returns the current version (either new_ or old_) set on this
|
|
* RecordAccessor. */
|
|
const TRecord ¤t() const;
|
|
|
|
/** Indicates if this accessor represents a local Vertex/Edge, or one whose
|
|
* owner is some other worker in a distributed system. */
|
|
bool is_local() const { return address_.is_local(); }
|
|
|
|
/**
|
|
* Processes the delta that's a consequence of changes in this accessor. If
|
|
* the accessor is local that means writing the delta to the write-ahead log.
|
|
* If it's remote, then the delta needs to be sent to it's owner for
|
|
* processing.
|
|
*
|
|
* @param delta The delta to process.
|
|
*/
|
|
void ProcessDelta(const GraphStateDelta &delta) const;
|
|
|
|
private:
|
|
// The database accessor for which this record accessor is created
|
|
// Provides means of getting to the transaction and database functions.
|
|
// Immutable, set in the constructor and never changed.
|
|
GraphDbAccessor *db_accessor_;
|
|
|
|
AddressT address_;
|
|
|
|
/**
|
|
* Latest version which is visible to the current transaction+command
|
|
* but has not been created nor modified by the current transaction+command.
|
|
*
|
|
* Can be null only when the record itself (the version-list) has
|
|
* been created by the current transaction+command.
|
|
*/
|
|
mutable 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.
|
|
*/
|
|
mutable TRecord *new_{nullptr};
|
|
};
|