2023-12-30 22:42:11 +00:00
|
|
|
//SPDX-FileCopyrightText: 2023 Yury Gubich <blue@macaw.me>
|
|
|
|
//SPDX-License-Identifier: GPL-3.0-or-later
|
2023-12-10 23:23:15 +00:00
|
|
|
|
2023-12-07 20:32:43 +00:00
|
|
|
#include "mysql.h"
|
|
|
|
|
2023-12-08 22:26:16 +00:00
|
|
|
#include <fstream>
|
2023-12-10 23:23:15 +00:00
|
|
|
#include <iostream>
|
2024-04-15 00:16:36 +00:00
|
|
|
#include <chrono>
|
2023-12-08 22:26:16 +00:00
|
|
|
|
|
|
|
#include "mysqld_error.h"
|
|
|
|
|
|
|
|
#include "statement.h"
|
2023-12-20 22:42:13 +00:00
|
|
|
#include "transaction.h"
|
2023-12-22 23:25:20 +00:00
|
|
|
#include "database/exceptions.h"
|
2023-12-08 22:26:16 +00:00
|
|
|
|
2023-12-20 22:42:13 +00:00
|
|
|
constexpr const char* versionQuery = "SELECT value FROM system WHERE `key` = 'version'";
|
2023-12-08 22:26:16 +00:00
|
|
|
constexpr const char* updateQuery = "UPDATE system SET `value` = ? WHERE `key` = 'version'";
|
2023-12-20 22:42:13 +00:00
|
|
|
constexpr const char* registerQuery = "INSERT INTO accounts (`login`, `type`, `password`) VALUES (?, 1, ?)";
|
|
|
|
constexpr const char* lastIdQuery = "SELECT LAST_INSERT_ID() AS id";
|
|
|
|
constexpr const char* assignRoleQuery = "INSERT INTO roleBindings (`account`, `role`) SELECT ?, roles.id FROM roles WHERE roles.name = ?";
|
2023-12-22 23:25:20 +00:00
|
|
|
constexpr const char* selectHash = "SELECT password FROM accounts where login = ?";
|
2023-12-23 20:23:38 +00:00
|
|
|
constexpr const char* createSessionQuery = "INSERT INTO sessions (`owner`, `access`, `renew`, `persist`, `device`)"
|
2024-01-11 21:33:46 +00:00
|
|
|
" SELECT accounts.id, ?, ?, true, ? FROM accounts WHERE accounts.login = ?"
|
|
|
|
" RETURNING id, owner";
|
2024-01-12 23:39:41 +00:00
|
|
|
constexpr const char* selectSession = "SELECT id, owner, access, renew FROM sessions where access = ?";
|
2024-01-22 18:21:55 +00:00
|
|
|
constexpr const char* selectAssets = "SELECT id, owner, currency, title, icon, color, archived FROM assets where owner = ?";
|
|
|
|
constexpr const char* insertAsset = "INSERT INTO assets (`owner`, `currency`, `title`, `icon`, `color`, `archived`, `type`)"
|
|
|
|
" VALUES (?, ?, ?, ?, ?, ?, 1)";
|
2024-04-07 23:03:10 +00:00
|
|
|
constexpr const char* updateAssetQuery = "UPDATE assets SET `owner` = ?, `currency` = ?, `title` = ?, `icon` = ?, `color` = ?, `archived` = ?"
|
|
|
|
" WHERE `id` = ?";
|
2024-01-21 19:23:48 +00:00
|
|
|
constexpr const char* removeAsset = "DELETE FROM assets where `id` = ? AND `owner` = ?";
|
2024-04-07 23:03:10 +00:00
|
|
|
constexpr const char* selectUsedCurrencies = "SELECT DISTINCT c.id, c.code, c.title, c.manual, c.icon FROM currencies c"
|
2024-03-28 23:20:21 +00:00
|
|
|
" JOIN assets a ON c.id = a.currency"
|
|
|
|
" WHERE a.owner = ?";
|
2024-04-10 23:09:45 +00:00
|
|
|
constexpr const char* addTransactionQuery = "INSERT INTO transactions"
|
|
|
|
" (`initiator`, `type`, `asset`, `parent`, `value`, `performed`)"
|
|
|
|
" VALUES (?, 1, ?, ?, ?, ?)";
|
2024-04-15 00:16:36 +00:00
|
|
|
constexpr const char* updateTransactionQuery = "UPDATE transactions SET"
|
|
|
|
" `initiator` = ?, `type` = 1, `asset` = ?,"
|
|
|
|
" `parent` = ?, `value` = ?, `performed` = ?"
|
|
|
|
" WHERE `id` = ?";
|
2024-04-10 23:09:45 +00:00
|
|
|
|
2023-12-08 22:26:16 +00:00
|
|
|
|
2023-12-10 23:23:15 +00:00
|
|
|
static const std::filesystem::path buildSQLPath = "database";
|
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
DB::MySQL::MySQL ():
|
2023-12-29 17:40:00 +00:00
|
|
|
Interface(Type::mysql),
|
2023-12-07 20:32:43 +00:00
|
|
|
connection(),
|
|
|
|
login(),
|
|
|
|
password(),
|
|
|
|
database()
|
|
|
|
{
|
|
|
|
mysql_init(&connection);
|
|
|
|
}
|
|
|
|
|
2023-12-29 17:40:00 +00:00
|
|
|
DB::MySQL::~MySQL() {
|
2023-12-07 20:32:43 +00:00
|
|
|
mysql_close(&connection);
|
|
|
|
}
|
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
void DB::MySQL::connect (const std::string& path) {
|
2023-12-07 20:32:43 +00:00
|
|
|
if (state != State::disconnected)
|
|
|
|
return;
|
|
|
|
|
|
|
|
MYSQL* con = &connection;
|
|
|
|
MYSQL* res = mysql_real_connect(
|
|
|
|
con,
|
|
|
|
NULL,
|
|
|
|
login.c_str(),
|
|
|
|
password.c_str(),
|
|
|
|
database.empty() ? NULL : database.c_str(),
|
|
|
|
0,
|
|
|
|
path.c_str(),
|
|
|
|
0
|
|
|
|
);
|
|
|
|
|
|
|
|
if (res != con)
|
|
|
|
throw std::runtime_error(std::string("Error changing connecting: ") + mysql_error(con));
|
|
|
|
|
|
|
|
state = State::connected;
|
|
|
|
}
|
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
void DB::MySQL::setCredentials (const std::string& login, const std::string& password) {
|
2023-12-07 20:32:43 +00:00
|
|
|
if (MySQL::login == login && MySQL::password == password)
|
|
|
|
return;
|
|
|
|
|
|
|
|
MySQL::login = login;
|
|
|
|
MySQL::password = password;
|
|
|
|
|
|
|
|
if (state == State::disconnected)
|
|
|
|
return;
|
|
|
|
|
|
|
|
MYSQL* con = &connection;
|
|
|
|
int result = mysql_change_user(
|
|
|
|
con,
|
|
|
|
login.c_str(),
|
|
|
|
password.c_str(),
|
|
|
|
database.empty() ? NULL : database.c_str()
|
|
|
|
);
|
|
|
|
|
|
|
|
if (result != 0)
|
|
|
|
throw std::runtime_error(std::string("Error changing credetials: ") + mysql_error(con));
|
|
|
|
}
|
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
void DB::MySQL::setDatabase (const std::string& database) {
|
2023-12-07 20:32:43 +00:00
|
|
|
if (MySQL::database == database)
|
|
|
|
return;
|
|
|
|
|
|
|
|
MySQL::database = database;
|
|
|
|
|
|
|
|
if (state == State::disconnected)
|
|
|
|
return;
|
|
|
|
|
|
|
|
MYSQL* con = &connection;
|
|
|
|
int result = mysql_select_db(con, database.c_str());
|
|
|
|
|
|
|
|
if (result != 0)
|
|
|
|
throw std::runtime_error(std::string("Error changing db: ") + mysql_error(con));
|
|
|
|
}
|
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
void DB::MySQL::disconnect () {
|
2023-12-07 20:32:43 +00:00
|
|
|
if (state == State::disconnected)
|
|
|
|
return;
|
|
|
|
|
|
|
|
MYSQL* con = &connection;
|
|
|
|
mysql_close(con);
|
|
|
|
mysql_init(con); //this is ridiculous!
|
|
|
|
}
|
2023-12-08 22:26:16 +00:00
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
void DB::MySQL::executeFile (const std::filesystem::path& relativePath) {
|
2023-12-08 22:26:16 +00:00
|
|
|
MYSQL* con = &connection;
|
2023-12-10 23:23:15 +00:00
|
|
|
std::filesystem::path path = sharedPath() / relativePath;
|
|
|
|
if (!std::filesystem::exists(path))
|
|
|
|
throw std::runtime_error("Error executing file "
|
|
|
|
+ std::filesystem::absolute(path).string()
|
|
|
|
+ ": file doesn't exist");
|
|
|
|
|
|
|
|
std::cout << "Executing file " << path << std::endl;
|
2023-12-08 22:26:16 +00:00
|
|
|
std::ifstream inputFile(path);
|
2024-04-15 00:16:36 +00:00
|
|
|
std::string block, comment;
|
|
|
|
while (getBlock(inputFile, block, comment)) {
|
|
|
|
if (!comment.empty())
|
|
|
|
std::cout << '\t' << comment << std::endl;
|
|
|
|
|
|
|
|
if (block.empty())
|
2023-12-11 23:29:55 +00:00
|
|
|
continue;
|
|
|
|
|
2024-04-15 00:16:36 +00:00
|
|
|
int result = mysql_query(con, block.c_str());
|
2023-12-08 22:26:16 +00:00
|
|
|
if (result != 0) {
|
|
|
|
int errcode = mysql_errno(con);
|
|
|
|
if (errcode == ER_EMPTY_QUERY)
|
|
|
|
continue;
|
|
|
|
|
2023-12-10 23:23:15 +00:00
|
|
|
throw std::runtime_error("Error executing file " + path.string() + ": " + mysql_error(con));
|
2023-12-08 22:26:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-15 00:16:36 +00:00
|
|
|
bool DB::MySQL::getBlock(std::ifstream& file, std::string& block, std::string& name) {
|
|
|
|
if (file.eof())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
block.clear();
|
|
|
|
name.clear();
|
|
|
|
|
|
|
|
if (file.peek() == '-') {
|
|
|
|
file.get();
|
|
|
|
if (file.peek() == '-') {
|
|
|
|
file.get();
|
|
|
|
std::getline(file, name);
|
|
|
|
} else {
|
|
|
|
file.unget();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
std::string line;
|
|
|
|
while (!file.eof()) {
|
|
|
|
if (file.peek() == '-')
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (!std::getline(file, line))
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (!block.empty())
|
|
|
|
block.append(1, '\n');
|
|
|
|
|
|
|
|
block += line;
|
|
|
|
}
|
|
|
|
|
|
|
|
return !block.empty() || !name.empty();
|
|
|
|
}
|
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
uint8_t DB::MySQL::getVersion () {
|
2023-12-08 22:26:16 +00:00
|
|
|
MYSQL* con = &connection;
|
2023-12-20 22:42:13 +00:00
|
|
|
int result = mysql_query(con, versionQuery);
|
2023-12-08 22:26:16 +00:00
|
|
|
|
|
|
|
if (result != 0) {
|
|
|
|
unsigned int errcode = mysql_errno(con);
|
|
|
|
if (errcode == ER_NO_SUCH_TABLE)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
throw std::runtime_error(std::string("Error executing retreiving version: ") + mysql_error(con));
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<MYSQL_RES, ResDeleter> res(mysql_store_result(con));
|
|
|
|
if (!res)
|
|
|
|
throw std::runtime_error(std::string("Querying version returned no result: ") + mysql_error(con));
|
|
|
|
|
|
|
|
MYSQL_ROW row = mysql_fetch_row(res.get());
|
|
|
|
if (row)
|
|
|
|
return std::stoi(row[0]);
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
void DB::MySQL::setVersion (uint8_t version) {
|
2023-12-08 22:26:16 +00:00
|
|
|
std::string strVersion = std::to_string(version);
|
|
|
|
Statement statement(&connection, updateQuery);
|
|
|
|
statement.bind(strVersion.data(), MYSQL_TYPE_VAR_STRING);
|
|
|
|
statement.execute();
|
|
|
|
}
|
2023-12-10 23:23:15 +00:00
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
void DB::MySQL::migrate (uint8_t targetVersion) {
|
2023-12-10 23:23:15 +00:00
|
|
|
uint8_t currentVersion = getVersion();
|
|
|
|
|
|
|
|
while (currentVersion < targetVersion) {
|
2023-12-11 23:29:55 +00:00
|
|
|
if (currentVersion == 255)
|
|
|
|
throw std::runtime_error("Maximum possible database version reached");
|
|
|
|
|
|
|
|
uint8_t nextVersion = currentVersion + 1;
|
2023-12-10 23:23:15 +00:00
|
|
|
std::string fileName = "migrations/m" + std::to_string(currentVersion) + ".sql";
|
|
|
|
std::cout << "Performing migration "
|
|
|
|
<< std::to_string(currentVersion)
|
|
|
|
<< " -> "
|
2023-12-11 23:29:55 +00:00
|
|
|
<< std::to_string(nextVersion)
|
2023-12-10 23:23:15 +00:00
|
|
|
<< std::endl;
|
|
|
|
executeFile(fileName);
|
2023-12-11 23:29:55 +00:00
|
|
|
setVersion(nextVersion);
|
|
|
|
currentVersion = nextVersion;
|
2023-12-10 23:23:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::cout << "Database is now on actual version " << std::to_string(targetVersion) << std::endl;
|
|
|
|
}
|
2023-12-11 23:29:55 +00:00
|
|
|
|
2024-01-22 18:21:55 +00:00
|
|
|
uint32_t DB::MySQL::registerAccount (const std::string& login, const std::string& hash) {
|
2023-12-20 22:42:13 +00:00
|
|
|
//TODO validate filed lengths!
|
|
|
|
MYSQL* con = &connection;
|
|
|
|
MySQL::Transaction txn(con);
|
|
|
|
|
|
|
|
Statement addAcc(con, registerQuery);
|
|
|
|
|
|
|
|
std::string l = login; //I hate copying just to please this horible API
|
|
|
|
std::string h = hash;
|
|
|
|
addAcc.bind(l.data(), MYSQL_TYPE_STRING);
|
|
|
|
addAcc.bind(h.data(), MYSQL_TYPE_STRING);
|
2023-12-22 23:25:20 +00:00
|
|
|
try {
|
|
|
|
addAcc.execute();
|
|
|
|
} catch (const Duplicate& dup) {
|
|
|
|
throw DuplicateLogin(dup.what());
|
|
|
|
}
|
2023-12-20 22:42:13 +00:00
|
|
|
|
2024-01-22 18:21:55 +00:00
|
|
|
uint32_t id = lastInsertedId();
|
2023-12-20 22:42:13 +00:00
|
|
|
static std::string defaultRole("default");
|
|
|
|
|
|
|
|
Statement addRole(con, assignRoleQuery);
|
|
|
|
addRole.bind(&id, MYSQL_TYPE_LONG, true);
|
|
|
|
addRole.bind(defaultRole.data(), MYSQL_TYPE_STRING);
|
|
|
|
addRole.execute();
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
std::string DB::MySQL::getAccountHash (const std::string& login) {
|
2023-12-22 23:25:20 +00:00
|
|
|
std::string l = login;
|
|
|
|
MYSQL* con = &connection;
|
|
|
|
|
|
|
|
Statement getHash(con, selectHash);
|
|
|
|
getHash.bind(l.data(), MYSQL_TYPE_STRING);
|
|
|
|
getHash.execute();
|
|
|
|
|
2023-12-23 20:23:38 +00:00
|
|
|
std::vector<std::vector<std::any>> result = getHash.fetchResult();
|
2023-12-22 23:25:20 +00:00
|
|
|
if (result.empty())
|
|
|
|
throw NoLogin("Couldn't find login " + l);
|
|
|
|
|
|
|
|
if (result[0].empty())
|
|
|
|
throw std::runtime_error("Error with the query \"selectHash\"");
|
|
|
|
|
2023-12-23 20:23:38 +00:00
|
|
|
return std::any_cast<const std::string&>(result[0][0]);
|
|
|
|
}
|
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
DB::Session DB::MySQL::createSession (const std::string& login, const std::string& access, const std::string& renew) {
|
2024-01-11 21:33:46 +00:00
|
|
|
std::string l = login;
|
|
|
|
DB::Session res;
|
|
|
|
res.accessToken = access;
|
|
|
|
res.renewToken = renew;
|
2023-12-23 20:23:38 +00:00
|
|
|
static std::string testingDevice("Testing...");
|
|
|
|
|
|
|
|
MYSQL* con = &connection;
|
|
|
|
|
|
|
|
Statement session(con, createSessionQuery);
|
2024-01-11 21:33:46 +00:00
|
|
|
session.bind(res.accessToken.data(), MYSQL_TYPE_STRING);
|
|
|
|
session.bind(res.renewToken.data(), MYSQL_TYPE_STRING);
|
2023-12-23 20:23:38 +00:00
|
|
|
session.bind(testingDevice.data(), MYSQL_TYPE_STRING);
|
|
|
|
session.bind(l.data(), MYSQL_TYPE_STRING);
|
|
|
|
session.execute();
|
|
|
|
|
2024-01-11 21:33:46 +00:00
|
|
|
std::vector<std::vector<std::any>> result = session.fetchResult();
|
|
|
|
if (result.empty())
|
|
|
|
throw std::runtime_error("Error returning ids after insertion in sessions table");
|
|
|
|
|
2024-01-22 18:21:55 +00:00
|
|
|
res.id = std::any_cast<uint32_t>(result[0][0]);
|
|
|
|
res.owner = std::any_cast<uint32_t>(result[0][1]);
|
2024-01-11 21:33:46 +00:00
|
|
|
|
|
|
|
return res;
|
2023-12-22 23:25:20 +00:00
|
|
|
}
|
|
|
|
|
2024-01-22 18:21:55 +00:00
|
|
|
uint32_t DB::MySQL::lastInsertedId () {
|
2023-12-20 22:42:13 +00:00
|
|
|
MYSQL* con = &connection;
|
|
|
|
int result = mysql_query(con, lastIdQuery);
|
|
|
|
|
|
|
|
if (result != 0)
|
|
|
|
throw std::runtime_error(std::string("Error executing last inserted id: ") + mysql_error(con));
|
|
|
|
|
|
|
|
std::unique_ptr<MYSQL_RES, ResDeleter> res(mysql_store_result(con));
|
|
|
|
if (!res)
|
|
|
|
throw std::runtime_error(std::string("Querying last inserted id returned no result: ") + mysql_error(con));
|
|
|
|
|
|
|
|
MYSQL_ROW row = mysql_fetch_row(res.get());
|
|
|
|
if (row)
|
|
|
|
return std::stoi(row[0]);
|
|
|
|
else
|
|
|
|
throw std::runtime_error(std::string("Querying last inserted id returned no rows"));
|
|
|
|
}
|
2024-01-12 23:39:41 +00:00
|
|
|
DB::Session DB::MySQL::findSession (const std::string& accessToken) {
|
2024-01-03 22:20:01 +00:00
|
|
|
std::string a = accessToken;
|
|
|
|
MYSQL* con = &connection;
|
|
|
|
|
|
|
|
Statement session(con, selectSession);
|
|
|
|
session.bind(a.data(), MYSQL_TYPE_STRING);
|
2024-01-09 17:02:56 +00:00
|
|
|
session.execute();
|
2023-12-20 22:42:13 +00:00
|
|
|
|
2024-01-03 22:20:01 +00:00
|
|
|
std::vector<std::vector<std::any>> result = session.fetchResult();
|
|
|
|
if (result.empty())
|
|
|
|
throw NoSession("Couldn't find session with token " + a);
|
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
return DB::Session(result[0]);
|
2024-01-03 22:20:01 +00:00
|
|
|
}
|
2023-12-20 22:42:13 +00:00
|
|
|
|
2024-01-22 18:21:55 +00:00
|
|
|
std::vector<DB::Asset> DB::MySQL::listAssets (uint32_t owner) {
|
2024-01-11 21:33:46 +00:00
|
|
|
MYSQL* con = &connection;
|
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
Statement st(con, selectAssets);
|
2024-01-11 21:33:46 +00:00
|
|
|
st.bind(&owner, MYSQL_TYPE_LONG, true);
|
|
|
|
st.execute();
|
|
|
|
std::vector<std::vector<std::any>> res = st.fetchResult();
|
|
|
|
|
|
|
|
std::size_t size = res.size();
|
|
|
|
std::vector<DB::Asset> result(size);
|
2024-01-12 23:39:41 +00:00
|
|
|
for (std::size_t i = 0; i < size; ++i)
|
|
|
|
result[i].parse(res[i]);
|
2024-01-11 21:33:46 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2023-12-20 22:42:13 +00:00
|
|
|
|
2024-01-12 23:39:41 +00:00
|
|
|
DB::Asset DB::MySQL::addAsset(const Asset& asset) {
|
|
|
|
MYSQL* con = &connection;
|
|
|
|
Asset result = asset;
|
|
|
|
|
2024-01-21 19:23:48 +00:00
|
|
|
Statement add(con, insertAsset);
|
|
|
|
add.bind(&result.owner, MYSQL_TYPE_LONG, true);
|
|
|
|
add.bind(&result.currency, MYSQL_TYPE_LONG, true);
|
|
|
|
add.bind(result.title.data(), MYSQL_TYPE_STRING);
|
|
|
|
add.bind(result.icon.data(), MYSQL_TYPE_STRING);
|
2024-01-22 18:21:55 +00:00
|
|
|
add.bind(&result.color, MYSQL_TYPE_LONG, true);
|
2024-01-21 19:23:48 +00:00
|
|
|
add.bind(&result.archived, MYSQL_TYPE_TINY);
|
|
|
|
add.execute();
|
2024-01-12 23:39:41 +00:00
|
|
|
|
|
|
|
result.id = lastInsertedId();
|
|
|
|
|
2024-01-21 19:23:48 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2024-04-07 23:03:10 +00:00
|
|
|
void DB::MySQL::updateAsset(const Asset& asset) {
|
|
|
|
MYSQL* con = &connection;
|
|
|
|
Asset result = asset;
|
|
|
|
|
|
|
|
Statement update(con, updateAssetQuery);
|
|
|
|
update.bind(&result.owner, MYSQL_TYPE_LONG, true);
|
|
|
|
update.bind(&result.currency, MYSQL_TYPE_LONG, true);
|
|
|
|
update.bind(result.title.data(), MYSQL_TYPE_STRING);
|
|
|
|
update.bind(result.icon.data(), MYSQL_TYPE_STRING);
|
|
|
|
update.bind(&result.color, MYSQL_TYPE_LONG, true);
|
|
|
|
update.bind(&result.archived, MYSQL_TYPE_TINY);
|
|
|
|
update.bind(&result.id, MYSQL_TYPE_LONG, true);
|
|
|
|
update.execute();
|
|
|
|
}
|
|
|
|
|
2024-01-22 18:21:55 +00:00
|
|
|
bool DB::MySQL::deleteAsset(uint32_t assetId, uint32_t actorId) {
|
|
|
|
Statement del(&connection, removeAsset);
|
2024-01-21 19:23:48 +00:00
|
|
|
del.bind(&assetId, MYSQL_TYPE_LONG, true);
|
|
|
|
del.bind(&actorId, MYSQL_TYPE_LONG, true);
|
|
|
|
del.execute();
|
|
|
|
|
|
|
|
if (del.affectedRows() == 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
2024-01-12 23:39:41 +00:00
|
|
|
}
|
2024-01-22 18:21:55 +00:00
|
|
|
|
|
|
|
std::vector<DB::Currency> DB::MySQL::listUsedCurrencies(uint32_t owner) {
|
2024-03-28 23:20:21 +00:00
|
|
|
Statement list(&connection, selectUsedCurrencies);
|
2024-01-22 18:21:55 +00:00
|
|
|
list.bind(&owner, MYSQL_TYPE_LONG, true);
|
|
|
|
list.execute();
|
|
|
|
|
|
|
|
std::vector<std::vector<std::any>> res = list.fetchResult();
|
|
|
|
|
|
|
|
std::size_t size = res.size();
|
|
|
|
std::vector<DB::Currency> result(size);
|
|
|
|
for (std::size_t i = 0; i < size; ++i)
|
|
|
|
result[i].parse(res[i]);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2024-04-10 23:09:45 +00:00
|
|
|
|
|
|
|
DB::Transaction DB::MySQL::addTransaction(const DB::Transaction& transaction) {
|
|
|
|
MYSQL* con = &connection;
|
|
|
|
DB::Transaction result = transaction;
|
|
|
|
|
|
|
|
std::string value = std::to_string(result.value);
|
|
|
|
|
|
|
|
Statement add(con, addTransactionQuery);
|
|
|
|
add.bind(&result.initiator, MYSQL_TYPE_LONG, true);
|
|
|
|
add.bind(&result.asset, MYSQL_TYPE_LONG, true);
|
|
|
|
add.bind(&result.parent, MYSQL_TYPE_LONG, true);
|
|
|
|
add.bind(value.data(), MYSQL_TYPE_STRING);
|
|
|
|
add.bind(&result.performed, MYSQL_TYPE_LONG, true);
|
|
|
|
add.execute();
|
|
|
|
|
|
|
|
result.id = lastInsertedId();
|
2024-04-15 00:16:36 +00:00
|
|
|
std::chrono::time_point currently = std::chrono::time_point_cast<std::chrono::seconds>(
|
|
|
|
std::chrono::system_clock::now()
|
|
|
|
);
|
|
|
|
result.modified = currently.time_since_epoch().count();
|
|
|
|
//todo actual value which could have changed after insertion
|
2024-04-10 23:09:45 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DB::MySQL::updateTransaction(const DB::Transaction& transaction) {
|
2024-04-15 00:16:36 +00:00
|
|
|
MYSQL* con = &connection;
|
|
|
|
DB::Transaction result = transaction;
|
|
|
|
|
|
|
|
std::string value = std::to_string(result.value);
|
|
|
|
|
|
|
|
Statement upd(con, updateTransactionQuery);
|
|
|
|
upd.bind(&result.initiator, MYSQL_TYPE_LONG, true);
|
|
|
|
upd.bind(&result.asset, MYSQL_TYPE_LONG, true);
|
|
|
|
upd.bind(&result.parent, MYSQL_TYPE_LONG, true);
|
|
|
|
upd.bind(value.data(), MYSQL_TYPE_STRING);
|
|
|
|
upd.bind(&result.performed, MYSQL_TYPE_LONG, true);
|
|
|
|
upd.bind(&result.id, MYSQL_TYPE_LONG, true);
|
|
|
|
upd.execute();
|
2024-04-10 23:09:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<DB::Transaction> DB::MySQL::listTransactions(uint32_t owner) {
|
|
|
|
return std::vector<DB::Transaction>();
|
|
|
|
}
|