// Squawk messenger. // Copyright (C) 2019 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/>. #include "database.h" #include "exceptions.h" #include "table.h" DataBase::DataBase(const QString& p_name, uint16_t mapSize): name(p_name.toStdString()), opened(false), size(mapSize), environment(), tables() { } DataBase::~DataBase() { close(); for (const std::pair<const std::string, _Table*>& pair : tables) { delete pair.second; } } void DataBase::close() { if (opened) { for (const std::pair<const std::string, _Table*>& pair : tables) { _Table* table = pair.second; mdb_dbi_close(environment, table->dbi); } mdb_env_close(environment); opened = false; } } void DataBase::open() { if (!opened) { mdb_env_create(&environment); QString path = createDirectory(); mdb_env_set_maxdbs(environment, tables.size()); mdb_env_set_mapsize(environment, size * 1024UL * 1024UL); mdb_env_open(environment, path.toStdString().c_str(), 0, 0664); MDB_txn *txn; mdb_txn_begin(environment, NULL, 0, &txn); for (const std::pair<const std::string, _Table*>& pair : tables) { _Table* table = pair.second; int rc = table->createTable(txn); if (rc) { throw Unknown(name, mdb_strerror(rc)); } } mdb_txn_commit(txn); opened = true; } } bool DataBase::removeDirectory() { if (opened) { throw Opened(name, "remove database directory"); } QString path(QStandardPaths::writableLocation(QStandardPaths::CacheLocation)); path += "/" + getName(); QDir cache(path); if (cache.exists()) { return cache.removeRecursively(); } else { return true; } } QString DataBase::createDirectory() { if (opened) { throw Opened(name, "create database directory"); } QString path(QStandardPaths::writableLocation(QStandardPaths::CacheLocation)); path += "/" + getName(); QDir cache(path); if (!cache.exists()) { bool res = cache.mkpath(path); if (!res) { throw Directory(path.toStdString()); } } return path; } QString DataBase::getName() const { return QString::fromStdString(name); } bool DataBase::ready() const { return opened; } void DataBase::drop() { if (!opened) { throw Closed("drop", name); } MDB_txn *txn; int rc = mdb_txn_begin(environment, NULL, 0, &txn); if (rc) { throw Unknown(name, mdb_strerror(rc)); } for (const std::pair<const std::string, _Table*>& pair : tables) { rc = pair.second->drop(txn); if (rc) { throw Unknown(name, mdb_strerror(rc), pair.first); } } mdb_txn_commit(txn); }