memgraph/include/transactions/engine.hpp

136 lines
2.6 KiB
C++
Raw Normal View History

#pragma once
#include <atomic>
#include <vector>
#include "threading/sync/lockable.hpp"
#include "threading/sync/spinlock.hpp"
#include "transactions/commit_log.hpp"
#include "transactions/transaction.hpp"
#include "transactions/transaction_store.hpp"
2015-07-31 18:36:41 +08:00
#include "utils/counters/simple_counter.hpp"
2015-10-04 15:47:15 +08:00
namespace tx
{
class TransactionError : std::runtime_error
{
public:
using std::runtime_error::runtime_error;
};
class Engine : Lockable<SpinLock>
2015-07-31 18:36:41 +08:00
{
public:
using sptr = std::shared_ptr<Engine>;
Engine() : counter(1) {}
// Begins transaction and runs given functions in same atomic step.
// Functions will be given Transaction&
template <class... F>
Transaction &begin(F... fun)
{
auto guard = this->acquire_unique();
auto id = Id(counter.next());
auto t = new Transaction(id, active, *this);
active.insert(id);
store.put(id, t);
2015-10-04 15:47:15 +08:00
call(*t, fun...);
2015-10-04 15:47:15 +08:00
return *t;
}
Transaction &advance(const Id &id)
2015-10-04 15:47:15 +08:00
{
auto guard = this->acquire_unique();
2015-10-04 15:47:15 +08:00
auto *t = store.get(id);
2015-10-04 15:47:15 +08:00
if (t == nullptr) throw TransactionError("transaction does not exist");
2015-10-04 15:47:15 +08:00
// this is a new command
t->cid++;
return *t;
}
// Returns copy of current snapshot
Snapshot<Id> snapshot()
{
auto guard = this->acquire_unique();
return active;
}
void commit(const Transaction &t)
{
auto guard = this->acquire_unique();
2016-03-12 19:26:56 +08:00
clog.set_committed(t.id);
2015-07-31 18:36:41 +08:00
finalize(t);
}
void abort(const Transaction &t)
{
auto guard = this->acquire_unique();
2016-03-12 19:26:56 +08:00
clog.set_aborted(t.id);
2015-07-31 18:36:41 +08:00
finalize(t);
}
Id last_known_active()
{
auto guard = this->acquire_unique();
2015-07-31 18:36:41 +08:00
return active.front();
}
// total number of transactions started from the beginning of time
2015-07-31 18:36:41 +08:00
uint64_t count()
{
auto guard = this->acquire_unique();
2015-07-31 18:36:41 +08:00
return counter.count();
}
// the number of currently active transactions
size_t size()
{
auto guard = this->acquire_unique();
return active.size();
}
2016-03-12 19:26:56 +08:00
CommitLog clog;
private:
template <class T, class... F>
void call(Transaction &t, T fun, F... funs)
{
call(t, fun);
call(t, funs...);
}
template <class T>
void call(Transaction &t, T fun)
{
fun(t);
}
void call(Transaction &t) {}
void finalize(const Transaction &t)
{
active.remove(t.id);
2015-10-04 15:47:15 +08:00
// remove transaction from store
store.del(t.id);
}
2015-07-31 18:36:41 +08:00
SimpleCounter<uint64_t> counter;
Snapshot<Id> active;
TransactionStore<uint64_t> store;
};
2015-10-04 15:47:15 +08:00
}