#pragma once #include "database/db_transaction.hpp" #include "mvcc/version_list.hpp" #include "storage/indexes/index_record.hpp" #include "storage/indexes/index_update.hpp" #include "storage/model/properties/properties.hpp" #include "storage/model/properties/property.hpp" #include "storage/model/properties/property_family.hpp" #include "storage/model/properties/stored_property.hpp" #include "transactions/transaction.hpp" template class RecordAccessor { friend DbAccessor; using vlist_t = typename TG::vlist_t; using T = typename TG::record_t; public: RecordAccessor(vlist_t *vlist, DbTransaction &db) : vlist(vlist), db(db) { assert(vlist != nullptr); } RecordAccessor(T *t, vlist_t *vlist, DbTransaction &db) : record(t), vlist(vlist), db(db) { assert(record != nullptr); assert(vlist != nullptr); } 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) { } bool empty() const { return record == nullptr; } // Fills accessor and returns true if there is valid data for current // transaction false otherwise. bool fill() const { const_cast(this)->record = vlist->find(db.trans); return record != nullptr; } const Id &id() const { return vlist->id; } // True if record visible for current transaction is visible to given // transaction id. bool is_visble_to(tx::TransactionRead const &id) { return record->visible(id); } // Returns new IndexRecord with given key. template IndexRecord create_index_record(K &&key) { return IndexRecord(std::move(key), record, vlist); } // TODO: Test this Derived update() const { assert(!empty()); if (record->is_visible_write(db.trans)) { // TODO: VALIDATE THIS BRANCH. THEN ONLY THIS TRANSACTION CAN SEE // THIS DATA WHICH MEANS THAT IT CAN CHANGE IT. 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(this)->record = new_record; // Add record to update index. db.to_update_index(vlist, new_record); return Derived(new_record, vlist, db); } } bool contains(property_key &key) const { return properties().contains(key); } const StoredProperty &at(PropertyFamily &key) const { return properties().at(key); } const StoredProperty &at(property_key &key) const { return properties().at(key); } template OptionPtr at(type_key_t &key) const { return properties().template at(key); } void set(property_key &key, Property value) { properties().set(StoredProperty(std::move(value), key)); } void set(StoredProperty value) { properties().set(std::move(value)); } void clear(property_key &key) { properties().clear(key); } void clear(PropertyFamily &key) { properties().clear(key); } template void accept(Handler &handler) const { properties().template accept(handler); } template void handle(Handler &handler) const { properties().template handle(handler); } Properties &properties() const { return record->data.props; } explicit operator bool() const { return record != nullptr; } T const *operator->() const { return record; } T *operator->() { return record; } RecordAccessor &operator=(const RecordAccessor &other) { record = other.record; vlist_t *&vl = const_cast(vlist); vl = other.vlist; return *this; } RecordAccessor &operator=(RecordAccessor &&other) { record = other.record; vlist_t *&vl = const_cast(vlist); vl = other.vlist; return *this; } // 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); } protected: void remove() const; T *record{nullptr}; vlist_t *const vlist; DbTransaction &db; };