mirror of
https://github.com/minetest/minetestmapper.git
synced 2025-02-23 15:30:24 +01:00
Refactor postgres into a base class too
This commit is contained in:
parent
cd36f16775
commit
e982efe94e
@ -9,6 +9,66 @@
|
|||||||
|
|
||||||
#define ARRLEN(x) (sizeof(x) / sizeof((x)[0]))
|
#define ARRLEN(x) (sizeof(x) / sizeof((x)[0]))
|
||||||
|
|
||||||
|
/* PostgreSQLBase */
|
||||||
|
|
||||||
|
PostgreSQLBase::~PostgreSQLBase()
|
||||||
|
{
|
||||||
|
if (db)
|
||||||
|
PQfinish(db);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PostgreSQLBase::openDatabase(const char *connect_string)
|
||||||
|
{
|
||||||
|
if (db)
|
||||||
|
throw std::logic_error("Database already open");
|
||||||
|
|
||||||
|
db = PQconnectdb(connect_string);
|
||||||
|
if (PQstatus(db) != CONNECTION_OK) {
|
||||||
|
throw std::runtime_error(std::string("PostgreSQL database error: ") +
|
||||||
|
PQerrorMessage(db)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
PGresult *PostgreSQLBase::checkResults(PGresult *res, bool clear)
|
||||||
|
{
|
||||||
|
ExecStatusType statusType = PQresultStatus(res);
|
||||||
|
|
||||||
|
switch (statusType) {
|
||||||
|
case PGRES_COMMAND_OK:
|
||||||
|
case PGRES_TUPLES_OK:
|
||||||
|
break;
|
||||||
|
case PGRES_FATAL_ERROR:
|
||||||
|
throw std::runtime_error(
|
||||||
|
std::string("PostgreSQL database error: ") +
|
||||||
|
PQresultErrorMessage(res)
|
||||||
|
);
|
||||||
|
default:
|
||||||
|
throw std::runtime_error(
|
||||||
|
std::string("Unhandled PostgreSQL result code ") +
|
||||||
|
std::to_string(statusType)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (clear)
|
||||||
|
PQclear(res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
PGresult *PostgreSQLBase::execPrepared(
|
||||||
|
const char *stmtName, const int paramsNumber,
|
||||||
|
const void **params,
|
||||||
|
const int *paramsLengths, const int *paramsFormats,
|
||||||
|
bool clear)
|
||||||
|
{
|
||||||
|
return checkResults(PQexecPrepared(db, stmtName, paramsNumber,
|
||||||
|
(const char* const*) params, paramsLengths, paramsFormats,
|
||||||
|
1 /* binary output */), clear
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* DBPostgreSQL */
|
||||||
|
|
||||||
DBPostgreSQL::DBPostgreSQL(const std::string &mapdir)
|
DBPostgreSQL::DBPostgreSQL(const std::string &mapdir)
|
||||||
{
|
{
|
||||||
std::ifstream ifs(mapdir + "world.mt");
|
std::ifstream ifs(mapdir + "world.mt");
|
||||||
@ -16,14 +76,8 @@ DBPostgreSQL::DBPostgreSQL(const std::string &mapdir)
|
|||||||
throw std::runtime_error("Failed to read world.mt");
|
throw std::runtime_error("Failed to read world.mt");
|
||||||
std::string connect_string = read_setting("pgsql_connection", ifs);
|
std::string connect_string = read_setting("pgsql_connection", ifs);
|
||||||
ifs.close();
|
ifs.close();
|
||||||
db = PQconnectdb(connect_string.c_str());
|
|
||||||
|
|
||||||
if (PQstatus(db) != CONNECTION_OK) {
|
openDatabase(connect_string.c_str());
|
||||||
throw std::runtime_error(std::string(
|
|
||||||
"PostgreSQL database error: ") +
|
|
||||||
PQerrorMessage(db)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
prepareStatement(
|
prepareStatement(
|
||||||
"get_block_pos",
|
"get_block_pos",
|
||||||
@ -56,7 +110,6 @@ DBPostgreSQL::~DBPostgreSQL()
|
|||||||
} catch (const std::exception& caught) {
|
} catch (const std::exception& caught) {
|
||||||
std::cerr << "could not finalize: " << caught.what() << std::endl;
|
std::cerr << "could not finalize: " << caught.what() << std::endl;
|
||||||
}
|
}
|
||||||
PQfinish(db);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -169,50 +222,6 @@ void DBPostgreSQL::getBlocksByPos(BlockList &blocks,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
PGresult *DBPostgreSQL::checkResults(PGresult *res, bool clear)
|
|
||||||
{
|
|
||||||
ExecStatusType statusType = PQresultStatus(res);
|
|
||||||
|
|
||||||
switch (statusType) {
|
|
||||||
case PGRES_COMMAND_OK:
|
|
||||||
case PGRES_TUPLES_OK:
|
|
||||||
break;
|
|
||||||
case PGRES_FATAL_ERROR:
|
|
||||||
throw std::runtime_error(
|
|
||||||
std::string("PostgreSQL database error: ") +
|
|
||||||
PQresultErrorMessage(res)
|
|
||||||
);
|
|
||||||
default:
|
|
||||||
throw std::runtime_error(
|
|
||||||
"Unhandled PostgreSQL result code"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (clear)
|
|
||||||
PQclear(res);
|
|
||||||
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
void DBPostgreSQL::prepareStatement(const std::string &name, const std::string &sql)
|
|
||||||
{
|
|
||||||
checkResults(PQprepare(db, name.c_str(), sql.c_str(), 0, NULL));
|
|
||||||
}
|
|
||||||
|
|
||||||
PGresult *DBPostgreSQL::execPrepared(
|
|
||||||
const char *stmtName, const int paramsNumber,
|
|
||||||
const void **params,
|
|
||||||
const int *paramsLengths, const int *paramsFormats,
|
|
||||||
bool clear
|
|
||||||
)
|
|
||||||
{
|
|
||||||
return checkResults(PQexecPrepared(db, stmtName, paramsNumber,
|
|
||||||
(const char* const*) params, paramsLengths, paramsFormats,
|
|
||||||
1 /* binary output */), clear
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
int DBPostgreSQL::pg_binary_to_int(PGresult *res, int row, int col)
|
int DBPostgreSQL::pg_binary_to_int(PGresult *res, int row, int col)
|
||||||
{
|
{
|
||||||
int32_t* raw = reinterpret_cast<int32_t*>(PQgetvalue(res, row, col));
|
int32_t* raw = reinterpret_cast<int32_t*>(PQgetvalue(res, row, col));
|
||||||
|
@ -3,7 +3,28 @@
|
|||||||
#include "db.h"
|
#include "db.h"
|
||||||
#include <libpq-fe.h>
|
#include <libpq-fe.h>
|
||||||
|
|
||||||
class DBPostgreSQL : public DB {
|
class PostgreSQLBase {
|
||||||
|
public:
|
||||||
|
~PostgreSQLBase();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void openDatabase(const char *connect_string);
|
||||||
|
|
||||||
|
PGresult *checkResults(PGresult *res, bool clear = true);
|
||||||
|
void prepareStatement(const std::string &name, const std::string &sql) {
|
||||||
|
checkResults(PQprepare(db, name.c_str(), sql.c_str(), 0, NULL));
|
||||||
|
}
|
||||||
|
PGresult *execPrepared(
|
||||||
|
const char *stmtName, const int paramsNumber,
|
||||||
|
const void **params,
|
||||||
|
const int *paramsLengths = nullptr, const int *paramsFormats = nullptr,
|
||||||
|
bool clear = true
|
||||||
|
);
|
||||||
|
|
||||||
|
PGconn *db = NULL;
|
||||||
|
};
|
||||||
|
|
||||||
|
class DBPostgreSQL : public DB, PostgreSQLBase {
|
||||||
public:
|
public:
|
||||||
DBPostgreSQL(const std::string &mapdir);
|
DBPostgreSQL(const std::string &mapdir);
|
||||||
std::vector<BlockPos> getBlockPosXZ(BlockPos min, BlockPos max) override;
|
std::vector<BlockPos> getBlockPosXZ(BlockPos min, BlockPos max) override;
|
||||||
@ -15,17 +36,6 @@ public:
|
|||||||
|
|
||||||
bool preferRangeQueries() const override { return true; }
|
bool preferRangeQueries() const override { return true; }
|
||||||
|
|
||||||
protected:
|
|
||||||
PGresult *checkResults(PGresult *res, bool clear = true);
|
|
||||||
void prepareStatement(const std::string &name, const std::string &sql);
|
|
||||||
PGresult *execPrepared(
|
|
||||||
const char *stmtName, const int paramsNumber,
|
|
||||||
const void **params,
|
|
||||||
const int *paramsLengths = nullptr, const int *paramsFormats = nullptr,
|
|
||||||
bool clear = true
|
|
||||||
);
|
|
||||||
int pg_binary_to_int(PGresult *res, int row, int col);
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
PGconn *db;
|
int pg_binary_to_int(PGresult *res, int row, int col);
|
||||||
};
|
};
|
||||||
|
Loading…
x
Reference in New Issue
Block a user