mirror of
https://github.com/minetest/minetestmapper.git
synced 2025-04-04 12:00:42 +02:00
Refactor sqlite3 code into base class
This commit is contained in:
parent
5016bca232
commit
d9c89bd6a2
@ -6,14 +6,37 @@
|
|||||||
#include "db-sqlite3.h"
|
#include "db-sqlite3.h"
|
||||||
#include "types.h"
|
#include "types.h"
|
||||||
|
|
||||||
#define SQLRES(r, good) do { \
|
/* SQLite3Base */
|
||||||
auto _result = (r); \
|
|
||||||
if (_result != good) \
|
|
||||||
throw std::runtime_error(sqlite3_errmsg(db)); \
|
|
||||||
} while (0)
|
|
||||||
|
|
||||||
|
#define SQLRES(r, good) check_result(r, good)
|
||||||
#define SQLOK(r) SQLRES(r, SQLITE_OK)
|
#define SQLOK(r) SQLRES(r, SQLITE_OK)
|
||||||
|
|
||||||
|
SQLite3Base::~SQLite3Base()
|
||||||
|
{
|
||||||
|
if (db && sqlite3_close(db) != SQLITE_OK) {
|
||||||
|
std::cerr << "Error closing SQLite database: "
|
||||||
|
<< sqlite3_errmsg(db) << std::endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SQLite3Base::openDatabase(const char *path, bool readonly)
|
||||||
|
{
|
||||||
|
if (db)
|
||||||
|
throw std::logic_error("Database already open");
|
||||||
|
|
||||||
|
int flags = 0;
|
||||||
|
if (readonly)
|
||||||
|
flags |= SQLITE_OPEN_READONLY | SQLITE_OPEN_PRIVATECACHE;
|
||||||
|
else
|
||||||
|
flags |= SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE;
|
||||||
|
#ifdef SQLITE_OPEN_EXRESCODE
|
||||||
|
flags |= SQLITE_OPEN_EXRESCODE;
|
||||||
|
#endif
|
||||||
|
SQLOK(sqlite3_open_v2(path, &db, flags, 0));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* DBSQLite3 */
|
||||||
|
|
||||||
// make sure a row is available. intended to be used outside a loop.
|
// make sure a row is available. intended to be used outside a loop.
|
||||||
// compare result to SQLITE_ROW afterwards.
|
// compare result to SQLITE_ROW afterwards.
|
||||||
#define SQLROW1(stmt) \
|
#define SQLROW1(stmt) \
|
||||||
@ -36,54 +59,41 @@ DBSQLite3::DBSQLite3(const std::string &mapdir)
|
|||||||
{
|
{
|
||||||
std::string db_name = mapdir + "map.sqlite";
|
std::string db_name = mapdir + "map.sqlite";
|
||||||
|
|
||||||
auto flags = SQLITE_OPEN_READONLY |
|
openDatabase(db_name.c_str());
|
||||||
SQLITE_OPEN_PRIVATECACHE;
|
|
||||||
#ifdef SQLITE_OPEN_EXRESCODE
|
|
||||||
flags |= SQLITE_OPEN_EXRESCODE;
|
|
||||||
#endif
|
|
||||||
SQLOK(sqlite3_open_v2(db_name.c_str(), &db, flags, 0));
|
|
||||||
|
|
||||||
// There's a simple, dumb way to check if we have a new or old database schema.
|
// There's a simple, dumb way to check if we have a new or old database schema.
|
||||||
// If we prepare a statement that references columns that don't exist, it will
|
// If we prepare a statement that references columns that don't exist, it will
|
||||||
// error right there.
|
// error right there.
|
||||||
int result = sqlite3_prepare_v2(db, "SELECT x, y, z FROM blocks", -1,
|
int result = prepare(stmt_get_block_pos, "SELECT x, y, z FROM blocks");
|
||||||
&stmt_get_block_pos, NULL);
|
|
||||||
newFormat = result == SQLITE_OK;
|
newFormat = result == SQLITE_OK;
|
||||||
#ifndef NDEBUG
|
#ifndef NDEBUG
|
||||||
std::cerr << "Detected " << (newFormat ? "new" : "old") << " SQLite schema" << std::endl;
|
std::cerr << "Detected " << (newFormat ? "new" : "old") << " SQLite schema" << std::endl;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (newFormat) {
|
if (newFormat) {
|
||||||
SQLOK(sqlite3_prepare_v2(db,
|
SQLOK(prepare(stmt_get_blocks_xz_range,
|
||||||
"SELECT y, data FROM blocks WHERE "
|
"SELECT y, data FROM blocks WHERE "
|
||||||
"x = ? AND z = ? AND y BETWEEN ? AND ?",
|
"x = ? AND z = ? AND y BETWEEN ? AND ?"));
|
||||||
-1, &stmt_get_blocks_xz_range, NULL));
|
|
||||||
|
|
||||||
SQLOK(sqlite3_prepare_v2(db,
|
SQLOK(prepare(stmt_get_block_exact,
|
||||||
"SELECT data FROM blocks WHERE x = ? AND y = ? AND z = ?",
|
"SELECT data FROM blocks WHERE x = ? AND y = ? AND z = ?"));
|
||||||
-1, &stmt_get_block_exact, NULL));
|
|
||||||
|
|
||||||
SQLOK(sqlite3_prepare_v2(db,
|
SQLOK(prepare(stmt_get_block_pos_range,
|
||||||
"SELECT x, y, z FROM blocks WHERE "
|
"SELECT x, y, z FROM blocks WHERE "
|
||||||
"x >= ? AND y >= ? AND z >= ? AND "
|
"x >= ? AND y >= ? AND z >= ? AND "
|
||||||
"x < ? AND y < ? AND z < ?",
|
"x < ? AND y < ? AND z < ?"));
|
||||||
-1, &stmt_get_block_pos_range, NULL));
|
|
||||||
} else {
|
} else {
|
||||||
SQLOK(sqlite3_prepare_v2(db,
|
SQLOK(prepare(stmt_get_blocks_z,
|
||||||
"SELECT pos, data FROM blocks WHERE pos BETWEEN ? AND ?",
|
"SELECT pos, data FROM blocks WHERE pos BETWEEN ? AND ?"));
|
||||||
-1, &stmt_get_blocks_z, NULL));
|
|
||||||
|
|
||||||
SQLOK(sqlite3_prepare_v2(db,
|
SQLOK(prepare(stmt_get_block_exact,
|
||||||
"SELECT data FROM blocks WHERE pos = ?",
|
"SELECT data FROM blocks WHERE pos = ?"));
|
||||||
-1, &stmt_get_block_exact, NULL));
|
|
||||||
|
|
||||||
SQLOK(sqlite3_prepare_v2(db,
|
SQLOK(prepare(stmt_get_block_pos,
|
||||||
"SELECT pos FROM blocks",
|
"SELECT pos FROM blocks"));
|
||||||
-1, &stmt_get_block_pos, NULL));
|
|
||||||
|
|
||||||
SQLOK(sqlite3_prepare_v2(db,
|
SQLOK(prepare(stmt_get_block_pos_range,
|
||||||
"SELECT pos FROM blocks WHERE pos BETWEEN ? AND ?",
|
"SELECT pos FROM blocks WHERE pos BETWEEN ? AND ?"));
|
||||||
-1, &stmt_get_block_pos_range, NULL));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#undef RANGE
|
#undef RANGE
|
||||||
@ -97,11 +107,6 @@ DBSQLite3::~DBSQLite3()
|
|||||||
sqlite3_finalize(stmt_get_block_pos);
|
sqlite3_finalize(stmt_get_block_pos);
|
||||||
sqlite3_finalize(stmt_get_block_pos_range);
|
sqlite3_finalize(stmt_get_block_pos_range);
|
||||||
sqlite3_finalize(stmt_get_block_exact);
|
sqlite3_finalize(stmt_get_block_exact);
|
||||||
|
|
||||||
if (sqlite3_close(db) != SQLITE_OK) {
|
|
||||||
std::cerr << "Error closing SQLite database: "
|
|
||||||
<< sqlite3_errmsg(db) << std::endl;
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -4,7 +4,39 @@
|
|||||||
#include <unordered_map>
|
#include <unordered_map>
|
||||||
#include <sqlite3.h>
|
#include <sqlite3.h>
|
||||||
|
|
||||||
class DBSQLite3 : public DB {
|
class SQLite3Base {
|
||||||
|
public:
|
||||||
|
~SQLite3Base();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void openDatabase(const char *path, bool readonly = true);
|
||||||
|
|
||||||
|
// check function result or throw error
|
||||||
|
inline void check_result(int result, int good = SQLITE_OK)
|
||||||
|
{
|
||||||
|
if (result != good)
|
||||||
|
throw std::runtime_error(sqlite3_errmsg(db));
|
||||||
|
}
|
||||||
|
|
||||||
|
// prepare a statement
|
||||||
|
inline int prepare(sqlite3_stmt *&stmt, const char *sql)
|
||||||
|
{
|
||||||
|
return sqlite3_prepare_v2(db, sql, -1, &stmt, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
// read blob from statement
|
||||||
|
static inline ustring read_blob(sqlite3_stmt *stmt, int iCol)
|
||||||
|
{
|
||||||
|
auto *data = reinterpret_cast<const unsigned char *>(
|
||||||
|
sqlite3_column_blob(stmt, iCol));
|
||||||
|
size_t size = sqlite3_column_bytes(stmt, iCol);
|
||||||
|
return ustring(data, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
sqlite3 *db = NULL;
|
||||||
|
};
|
||||||
|
|
||||||
|
class DBSQLite3 : public DB, SQLite3Base {
|
||||||
public:
|
public:
|
||||||
DBSQLite3(const std::string &mapdir);
|
DBSQLite3(const std::string &mapdir);
|
||||||
std::vector<BlockPos> getBlockPos(BlockPos min, BlockPos max) override;
|
std::vector<BlockPos> getBlockPos(BlockPos min, BlockPos max) override;
|
||||||
@ -35,17 +67,6 @@ private:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// read blob from statement
|
|
||||||
static inline ustring read_blob(sqlite3_stmt *stmt, int iCol)
|
|
||||||
{
|
|
||||||
auto *data = reinterpret_cast<const unsigned char *>(
|
|
||||||
sqlite3_column_blob(stmt, iCol));
|
|
||||||
size_t size = sqlite3_column_bytes(stmt, iCol);
|
|
||||||
return ustring(data, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
sqlite3 *db = NULL;
|
|
||||||
|
|
||||||
sqlite3_stmt *stmt_get_block_pos = NULL;
|
sqlite3_stmt *stmt_get_block_pos = NULL;
|
||||||
sqlite3_stmt *stmt_get_block_pos_range = NULL;
|
sqlite3_stmt *stmt_get_block_pos_range = NULL;
|
||||||
sqlite3_stmt *stmt_get_blocks_z = NULL;
|
sqlite3_stmt *stmt_get_blocks_z = NULL;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user