2016-01-02 19:20:51 +08:00
|
|
|
#pragma once
|
|
|
|
|
2016-08-15 07:09:58 +08:00
|
|
|
#include "database/db_transaction.hpp"
|
2016-01-02 19:20:51 +08:00
|
|
|
#include "mvcc/version_list.hpp"
|
2016-08-18 22:34:36 +08:00
|
|
|
#include "storage/indexes/index_record.hpp"
|
2016-08-25 22:29:45 +08:00
|
|
|
#include "storage/indexes/index_update.hpp"
|
2016-01-03 05:20:09 +08:00
|
|
|
#include "storage/model/properties/properties.hpp"
|
2016-08-10 03:29:03 +08:00
|
|
|
#include "storage/model/properties/property.hpp"
|
2016-08-18 22:34:36 +08:00
|
|
|
#include "storage/model/properties/property_family.hpp"
|
2016-09-05 17:02:48 +08:00
|
|
|
#include "storage/model/properties/stored_property.hpp"
|
2016-08-10 03:29:03 +08:00
|
|
|
#include "transactions/transaction.hpp"
|
2016-01-02 19:20:51 +08:00
|
|
|
|
2016-08-25 22:29:45 +08:00
|
|
|
template <class TG, class Derived>
|
2016-01-02 19:20:51 +08:00
|
|
|
class RecordAccessor
|
|
|
|
{
|
2016-08-15 07:09:58 +08:00
|
|
|
friend DbAccessor;
|
2016-08-25 22:29:45 +08:00
|
|
|
using vlist_t = typename TG::vlist_t;
|
|
|
|
using T = typename TG::record_t;
|
2016-08-15 07:09:58 +08:00
|
|
|
|
2016-01-02 19:20:51 +08:00
|
|
|
public:
|
2016-08-15 07:09:58 +08:00
|
|
|
RecordAccessor(vlist_t *vlist, DbTransaction &db) : vlist(vlist), db(db)
|
|
|
|
{
|
|
|
|
assert(vlist != nullptr);
|
|
|
|
}
|
2016-01-02 19:20:51 +08:00
|
|
|
|
2016-08-15 07:09:58 +08:00
|
|
|
RecordAccessor(T *t, vlist_t *vlist, DbTransaction &db)
|
|
|
|
: record(t), vlist(vlist), db(db)
|
2016-01-02 19:20:51 +08:00
|
|
|
{
|
2016-01-09 21:43:55 +08:00
|
|
|
assert(record != nullptr);
|
|
|
|
assert(vlist != nullptr);
|
2016-01-02 19:20:51 +08:00
|
|
|
}
|
|
|
|
|
2016-08-23 02:03:45 +08:00
|
|
|
RecordAccessor(RecordAccessor const &other)
|
|
|
|
: record(other.record), vlist(other.vlist), db(other.db)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
RecordAccessor(RecordAccessor &&other)
|
|
|
|
: record(other.record), vlist(other.vlist), db(other.db)
|
|
|
|
{
|
|
|
|
}
|
2016-08-15 07:09:58 +08:00
|
|
|
|
2016-08-10 03:29:03 +08:00
|
|
|
bool empty() const { return record == nullptr; }
|
2016-01-02 19:20:51 +08:00
|
|
|
|
2016-08-15 07:09:58 +08:00
|
|
|
// Fills accessor and returns true if there is valid data for current
|
|
|
|
// transaction false otherwise.
|
|
|
|
bool fill() const
|
|
|
|
{
|
|
|
|
const_cast<RecordAccessor *>(this)->record = vlist->find(db.trans);
|
|
|
|
return record != nullptr;
|
|
|
|
}
|
|
|
|
|
2016-08-20 01:40:04 +08:00
|
|
|
const Id &id() const { return vlist->id; }
|
2016-01-09 21:43:55 +08:00
|
|
|
|
2016-09-08 20:25:52 +08:00
|
|
|
// True if record visible for current transaction is visible to given
|
|
|
|
// transaction id.
|
2016-09-10 03:56:53 +08:00
|
|
|
bool is_visble_to(tx::TransactionRead const &id)
|
2016-09-08 20:25:52 +08:00
|
|
|
{
|
|
|
|
return record->visible(id);
|
|
|
|
}
|
|
|
|
|
2016-09-13 03:13:04 +08:00
|
|
|
// Returns new IndexRecord with given key.
|
|
|
|
template <class K>
|
|
|
|
IndexRecord<TG, K> create_index_record(K &&key)
|
|
|
|
{
|
|
|
|
return IndexRecord<TG, K>(std::move(key), record, vlist);
|
|
|
|
}
|
|
|
|
|
2016-08-25 22:29:45 +08:00
|
|
|
// TODO: Test this
|
2016-08-15 07:09:58 +08:00
|
|
|
Derived update() const
|
2016-01-02 19:20:51 +08:00
|
|
|
{
|
|
|
|
assert(!empty());
|
|
|
|
|
2016-08-25 22:29:45 +08:00
|
|
|
if (record->is_visible_write(db.trans)) {
|
|
|
|
// TODO: VALIDATE THIS BRANCH. THEN ONLY THIS TRANSACTION CAN SEE
|
2016-09-17 22:31:35 +08:00
|
|
|
// THIS DATA WHICH MEANS THAT IT CAN CHANGE IT.
|
2016-08-25 22:29:45 +08:00
|
|
|
return Derived(record, vlist, db);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
auto new_record = vlist->update(db.trans);
|
|
|
|
|
|
|
|
// TODO: Validate that update of record in this accessor is correct.
|
|
|
|
const_cast<RecordAccessor *>(this)->record = new_record;
|
|
|
|
|
|
|
|
// Add record to update index.
|
|
|
|
db.to_update_index<TG>(vlist, new_record);
|
|
|
|
|
|
|
|
return Derived(new_record, vlist, db);
|
|
|
|
}
|
2016-01-02 19:20:51 +08:00
|
|
|
}
|
2016-10-19 08:41:06 +08:00
|
|
|
|
|
|
|
bool contains(property_key<TG> &key) const
|
|
|
|
{
|
|
|
|
return properties().contains(key);
|
|
|
|
}
|
2016-01-02 19:20:51 +08:00
|
|
|
|
2016-09-05 17:02:48 +08:00
|
|
|
const StoredProperty<TG> &at(PropertyFamily<TG> &key) const
|
2016-08-20 01:40:04 +08:00
|
|
|
{
|
|
|
|
return properties().at(key);
|
|
|
|
}
|
|
|
|
|
2016-09-05 17:02:48 +08:00
|
|
|
const StoredProperty<TG> &at(property_key<TG> &key) const
|
2016-08-25 22:29:45 +08:00
|
|
|
{
|
|
|
|
return properties().at(key);
|
|
|
|
}
|
2016-08-18 22:34:36 +08:00
|
|
|
|
|
|
|
template <class V>
|
2016-09-05 17:02:48 +08:00
|
|
|
OptionPtr<const V> at(type_key_t<TG, V> &key) const
|
2016-08-25 22:29:45 +08:00
|
|
|
{
|
|
|
|
return properties().template at<V>(key);
|
|
|
|
}
|
2016-01-02 19:20:51 +08:00
|
|
|
|
2016-09-05 17:02:48 +08:00
|
|
|
void set(property_key<TG> &key, Property value)
|
2016-01-02 19:20:51 +08:00
|
|
|
{
|
2016-09-05 17:02:48 +08:00
|
|
|
properties().set(StoredProperty<TG>(std::move(value), key));
|
2016-01-03 05:20:09 +08:00
|
|
|
}
|
|
|
|
|
2016-09-05 17:02:48 +08:00
|
|
|
void set(StoredProperty<TG> value) { properties().set(std::move(value)); }
|
2016-02-22 05:21:15 +08:00
|
|
|
|
2016-09-05 17:02:48 +08:00
|
|
|
void clear(property_key<TG> &key) { properties().clear(key); }
|
2016-01-02 19:20:51 +08:00
|
|
|
|
2016-08-25 22:29:45 +08:00
|
|
|
void clear(PropertyFamily<TG> &key) { properties().clear(key); }
|
2016-08-20 01:40:04 +08:00
|
|
|
|
2016-08-18 22:34:36 +08:00
|
|
|
template <class Handler>
|
|
|
|
void accept(Handler &handler) const
|
2016-08-15 20:21:38 +08:00
|
|
|
{
|
2016-08-18 22:34:36 +08:00
|
|
|
properties().template accept<Handler>(handler);
|
2016-08-15 20:21:38 +08:00
|
|
|
}
|
|
|
|
|
2016-09-13 19:14:16 +08:00
|
|
|
template <class Handler>
|
|
|
|
void handle(Handler &handler) const
|
|
|
|
{
|
|
|
|
properties().template handle<Handler>(handler);
|
|
|
|
}
|
|
|
|
|
2016-08-25 22:29:45 +08:00
|
|
|
Properties<TG> &properties() const { return record->data.props; }
|
2016-08-18 22:34:36 +08:00
|
|
|
|
2016-08-10 03:29:03 +08:00
|
|
|
explicit operator bool() const { return record != nullptr; }
|
2016-03-13 03:16:19 +08:00
|
|
|
|
2016-08-15 07:09:58 +08:00
|
|
|
T const *operator->() const { return record; }
|
|
|
|
T *operator->() { return record; }
|
|
|
|
|
2016-08-23 02:03:45 +08:00
|
|
|
RecordAccessor &operator=(const RecordAccessor &other)
|
|
|
|
{
|
|
|
|
record = other.record;
|
|
|
|
vlist_t *&vl = const_cast<vlist_t *&>(vlist);
|
|
|
|
vl = other.vlist;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
RecordAccessor &operator=(RecordAccessor &&other)
|
|
|
|
{
|
|
|
|
record = other.record;
|
|
|
|
vlist_t *&vl = const_cast<vlist_t *&>(vlist);
|
|
|
|
vl = other.vlist;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2016-08-15 07:09:58 +08:00
|
|
|
// Assumes same transaction
|
|
|
|
friend bool operator==(const RecordAccessor &a, const RecordAccessor &b)
|
|
|
|
{
|
|
|
|
return a.vlist == b.vlist;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assumes same transaction
|
|
|
|
friend bool operator!=(const RecordAccessor &a, const RecordAccessor &b)
|
|
|
|
{
|
|
|
|
return !(a == b);
|
|
|
|
}
|
|
|
|
|
2016-08-28 22:47:13 +08:00
|
|
|
protected:
|
2016-08-30 15:53:02 +08:00
|
|
|
void remove() const;
|
2016-08-30 13:16:04 +08:00
|
|
|
|
2016-08-15 07:09:58 +08:00
|
|
|
T *record{nullptr};
|
|
|
|
vlist_t *const vlist;
|
|
|
|
DbTransaction &db;
|
2016-01-02 19:20:51 +08:00
|
|
|
};
|