2023-03-23 17:27:46 +00:00
|
|
|
/*
|
|
|
|
* LMDB Abstraction Layer.
|
|
|
|
* Copyright (C) 2023 Yury Gubich <blue@macaw.me>
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2022-09-05 20:25:39 +00:00
|
|
|
|
2023-03-21 11:05:54 +00:00
|
|
|
#ifndef LMDBAL_BASE_H
|
|
|
|
#define LMDBAL_BASE_H
|
2022-09-05 20:25:39 +00:00
|
|
|
|
|
|
|
#include <map>
|
2023-03-20 15:37:13 +00:00
|
|
|
#include <set>
|
2022-09-05 20:25:39 +00:00
|
|
|
#include <string>
|
2022-10-10 20:51:48 +00:00
|
|
|
#include <optional>
|
2023-03-20 15:37:13 +00:00
|
|
|
#include <limits>
|
2022-09-05 20:25:39 +00:00
|
|
|
|
|
|
|
#include <QString>
|
|
|
|
#include <QStandardPaths>
|
|
|
|
#include <QDir>
|
2023-03-20 15:37:13 +00:00
|
|
|
|
2022-09-05 20:25:39 +00:00
|
|
|
#include <lmdb.h>
|
|
|
|
|
2023-03-20 15:37:13 +00:00
|
|
|
#include "exceptions.h"
|
|
|
|
|
2023-03-21 11:05:54 +00:00
|
|
|
namespace LMDBAL {
|
2023-03-20 15:37:13 +00:00
|
|
|
|
2023-03-21 11:05:54 +00:00
|
|
|
class iStorage;
|
2023-03-20 15:37:13 +00:00
|
|
|
|
|
|
|
template<class T>
|
|
|
|
class Serializer;
|
|
|
|
|
|
|
|
template <class K, class V>
|
|
|
|
class Storage;
|
|
|
|
|
|
|
|
template <class K, class V>
|
|
|
|
class Cache;
|
|
|
|
|
|
|
|
typedef MDB_txn* TransactionID;
|
|
|
|
|
2023-03-21 11:05:54 +00:00
|
|
|
class Base {
|
|
|
|
friend class iStorage;
|
2022-09-05 20:25:39 +00:00
|
|
|
public:
|
|
|
|
|
2023-03-21 11:05:54 +00:00
|
|
|
Base(const QString& name, uint16_t mapSize = 10);
|
|
|
|
~Base();
|
2022-09-05 20:25:39 +00:00
|
|
|
|
|
|
|
void open();
|
|
|
|
void close();
|
2022-09-15 21:34:39 +00:00
|
|
|
bool ready() const;
|
|
|
|
bool removeDirectory();
|
2022-12-17 10:06:37 +00:00
|
|
|
QString createDirectory();
|
2022-09-05 20:25:39 +00:00
|
|
|
QString getName() const;
|
2022-09-17 12:31:58 +00:00
|
|
|
void drop();
|
2022-09-05 20:25:39 +00:00
|
|
|
|
2023-03-20 15:37:13 +00:00
|
|
|
TransactionID beginReadOnlyTransaction() const;
|
|
|
|
TransactionID beginTransaction() const;
|
|
|
|
void commitTransaction(TransactionID id) const;
|
|
|
|
void abortTransaction(TransactionID id) const;
|
2022-12-17 10:06:37 +00:00
|
|
|
|
2022-09-05 20:25:39 +00:00
|
|
|
template <class K, class V>
|
2023-03-23 17:27:46 +00:00
|
|
|
LMDBAL::Storage<K, V>* addStorage(const std::string& name);
|
2022-09-15 21:34:39 +00:00
|
|
|
|
2022-09-20 17:16:48 +00:00
|
|
|
template <class K, class V>
|
2023-03-23 17:27:46 +00:00
|
|
|
LMDBAL::Cache<K, V>* addCache(const std::string& name);
|
2022-09-20 17:16:48 +00:00
|
|
|
|
2022-09-15 21:34:39 +00:00
|
|
|
template <class K, class V>
|
2023-03-23 17:27:46 +00:00
|
|
|
LMDBAL::Storage<K, V>* getStorage(const std::string& name);
|
2022-09-05 20:25:39 +00:00
|
|
|
|
2022-09-20 17:16:48 +00:00
|
|
|
template <class K, class V>
|
2023-03-23 17:27:46 +00:00
|
|
|
LMDBAL::Cache<K, V>* getCache(const std::string& name);
|
2022-09-20 17:16:48 +00:00
|
|
|
|
2023-03-20 15:37:13 +00:00
|
|
|
private:
|
2023-04-02 13:00:21 +00:00
|
|
|
typedef std::map<std::string, LMDBAL::iStorage*> Storages;
|
2023-03-20 15:37:13 +00:00
|
|
|
typedef std::set<TransactionID> Transactions;
|
|
|
|
|
|
|
|
TransactionID beginReadOnlyTransaction(const std::string& storageName) const;
|
|
|
|
TransactionID beginTransaction(const std::string& storageName) const;
|
|
|
|
void commitTransaction(TransactionID id, const std::string& storageName) const;
|
|
|
|
void abortTransaction(TransactionID id, const std::string& storageName) const;
|
2022-09-05 20:25:39 +00:00
|
|
|
|
2023-04-02 13:00:21 +00:00
|
|
|
TransactionID beginPrivateReadOnlyTransaction(const std::string& storageName) const;
|
|
|
|
TransactionID beginPrivateTransaction(const std::string& storageName) const;
|
|
|
|
void commitPrivateTransaction(TransactionID id, const std::string& storageName) const;
|
|
|
|
void abortPrivateTransaction(TransactionID id, const std::string& storageName) const;
|
|
|
|
|
2022-09-05 20:25:39 +00:00
|
|
|
private:
|
|
|
|
std::string name;
|
|
|
|
bool opened;
|
|
|
|
uint16_t size;
|
|
|
|
MDB_env* environment;
|
2023-04-02 13:00:21 +00:00
|
|
|
Storages storages;
|
2023-03-20 15:37:13 +00:00
|
|
|
Transactions* transactions;
|
|
|
|
|
|
|
|
inline static const std::string emptyName = "";
|
2022-09-05 20:25:39 +00:00
|
|
|
};
|
|
|
|
|
2023-03-20 15:37:13 +00:00
|
|
|
}
|
2022-12-18 14:45:12 +00:00
|
|
|
#include "operators.hpp"
|
2022-09-14 22:18:31 +00:00
|
|
|
|
2023-03-23 17:27:46 +00:00
|
|
|
/**
|
|
|
|
* Adds LMDBAL::Storage with the given name to the LMDBAL::Base, also returns it. The LMDBAL::Base must be closed
|
|
|
|
*/
|
2022-09-14 22:18:31 +00:00
|
|
|
template <class K, class V>
|
2023-03-23 17:27:46 +00:00
|
|
|
LMDBAL::Storage<K, V>* LMDBAL::Base::addStorage(const std::string& p_name) {
|
2022-09-14 22:18:31 +00:00
|
|
|
if (opened) {
|
2023-04-02 13:00:21 +00:00
|
|
|
throw Opened(name, "add storage " + p_name);
|
2022-09-14 22:18:31 +00:00
|
|
|
}
|
2023-04-02 13:00:21 +00:00
|
|
|
Storage<K, V>* storage = new Storage<K, V>(p_name, this);
|
|
|
|
storages.insert(std::make_pair(p_name, (iStorage*)storage));
|
|
|
|
return storage;
|
2022-09-05 20:25:39 +00:00
|
|
|
}
|
|
|
|
|
2023-03-23 17:27:46 +00:00
|
|
|
/**
|
|
|
|
* Adds LMDBAL::Cache with given the name to the LMDBAL::Base, also returns it. The LMDBAL::Base must be closed
|
|
|
|
*/
|
2022-09-20 17:16:48 +00:00
|
|
|
template<class K, class V>
|
2023-03-21 11:05:54 +00:00
|
|
|
LMDBAL::Cache<K, V> * LMDBAL::Base::addCache(const std::string& p_name) {
|
2022-09-20 17:16:48 +00:00
|
|
|
if (opened) {
|
|
|
|
throw Opened(name, "add cache " + p_name);
|
|
|
|
}
|
2023-03-20 15:37:13 +00:00
|
|
|
Cache<K, V>* cache = new Cache<K, V>(p_name, this);
|
2023-04-02 13:00:21 +00:00
|
|
|
storages.insert(std::make_pair(p_name, (iStorage*)cache));
|
2022-09-20 17:16:48 +00:00
|
|
|
return cache;
|
|
|
|
}
|
|
|
|
|
2023-03-23 17:27:46 +00:00
|
|
|
/**
|
|
|
|
* Returns LMDBAL::Storage with the given name
|
|
|
|
*/
|
2022-09-15 21:34:39 +00:00
|
|
|
template <class K, class V>
|
2023-03-23 17:27:46 +00:00
|
|
|
LMDBAL::Storage<K, V>* LMDBAL::Base::getStorage(const std::string& p_name) {
|
2023-04-02 13:00:21 +00:00
|
|
|
return static_cast<Storage<K, V>*>(storages.at(p_name));
|
2022-09-15 21:34:39 +00:00
|
|
|
}
|
|
|
|
|
2023-03-23 17:27:46 +00:00
|
|
|
/**
|
|
|
|
* Returns LMDBAL::Cache with the given name
|
|
|
|
*/
|
2022-09-20 17:16:48 +00:00
|
|
|
template <class K, class V>
|
2023-03-21 11:05:54 +00:00
|
|
|
LMDBAL::Cache<K, V>* LMDBAL::Base::getCache(const std::string& p_name) {
|
2023-04-02 13:00:21 +00:00
|
|
|
return static_cast<Cache<K, V>*>(storages.at(p_name));
|
2022-09-20 17:16:48 +00:00
|
|
|
}
|
|
|
|
|
2023-03-21 11:05:54 +00:00
|
|
|
#endif //LMDBAL_BASE_H
|