nix-super/src/libstore/sqlite.cc

190 lines
4.6 KiB
C++
Raw Normal View History

2016-03-30 14:27:25 +03:00
#include "sqlite.hh"
#include "util.hh"
#include <sqlite3.h>
namespace nix {
[[noreturn]] void throwSQLiteError(sqlite3 * db, const format & f)
{
int err = sqlite3_errcode(db);
auto path = sqlite3_db_filename(db, nullptr);
if (!path) path = "(in-memory)";
2016-03-30 14:27:25 +03:00
if (err == SQLITE_BUSY || err == SQLITE_PROTOCOL) {
if (err == SQLITE_PROTOCOL)
printError("warning: SQLite database %s is busy (SQLITE_PROTOCOL)", path);
2016-03-30 14:27:25 +03:00
else {
static bool warned = false;
if (!warned) {
printError("warning: SQLite database %s is busy", path);
2016-03-30 14:27:25 +03:00
warned = true;
}
}
/* Sleep for a while since retrying the transaction right away
is likely to fail again. */
checkInterrupt();
2016-03-30 14:27:25 +03:00
#if HAVE_NANOSLEEP
struct timespec t;
t.tv_sec = 0;
t.tv_nsec = (random() % 100) * 1000 * 1000; /* <= 0.1s */
nanosleep(&t, 0);
#else
sleep(1);
#endif
throw SQLiteBusy("%s: %s (in %s)", f.str(), sqlite3_errstr(err), path);
2016-03-30 14:27:25 +03:00
}
else
throw SQLiteError("%s: %s (in %s)", f.str(), sqlite3_errstr(err), path);
2016-03-30 14:27:25 +03:00
}
2016-08-09 15:27:30 +03:00
SQLite::SQLite(const Path & path)
{
if (sqlite3_open_v2(path.c_str(), &db,
SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE, 0) != SQLITE_OK)
throw Error(format("cannot open SQLite database %s") % path);
2016-08-09 15:27:30 +03:00
}
2016-03-30 14:27:25 +03:00
SQLite::~SQLite()
{
try {
if (db && sqlite3_close(db) != SQLITE_OK)
throwSQLiteError(db, "closing database");
} catch (...) {
ignoreException();
}
}
2016-08-09 15:27:30 +03:00
void SQLite::exec(const std::string & stmt)
{
if (sqlite3_exec(db, stmt.c_str(), 0, 0, 0) != SQLITE_OK)
throwSQLiteError(db, format("executing SQLite statement %s") % stmt);
2016-08-09 15:27:30 +03:00
}
2016-03-30 14:27:25 +03:00
void SQLiteStmt::create(sqlite3 * db, const string & s)
{
checkInterrupt();
assert(!stmt);
if (sqlite3_prepare_v2(db, s.c_str(), -1, &stmt, 0) != SQLITE_OK)
throwSQLiteError(db, "creating statement");
this->db = db;
}
SQLiteStmt::~SQLiteStmt()
{
try {
if (stmt && sqlite3_finalize(stmt) != SQLITE_OK)
throwSQLiteError(db, "finalizing statement");
} catch (...) {
ignoreException();
}
}
SQLiteStmt::Use::Use(SQLiteStmt & stmt)
: stmt(stmt)
2016-03-30 14:27:25 +03:00
{
assert(stmt.stmt);
/* Note: sqlite3_reset() returns the error code for the most
recent call to sqlite3_step(). So ignore it. */
sqlite3_reset(stmt);
2016-04-05 16:29:56 +03:00
}
SQLiteStmt::Use::~Use()
{
sqlite3_reset(stmt);
2016-03-30 14:27:25 +03:00
}
SQLiteStmt::Use & SQLiteStmt::Use::operator () (const std::string & value, bool notNull)
2016-03-30 14:27:25 +03:00
{
if (notNull) {
if (sqlite3_bind_text(stmt, curArg++, value.c_str(), -1, SQLITE_TRANSIENT) != SQLITE_OK)
throwSQLiteError(stmt.db, "binding argument");
} else
bind();
return *this;
2016-03-30 14:27:25 +03:00
}
SQLiteStmt::Use & SQLiteStmt::Use::operator () (int64_t value, bool notNull)
2016-03-30 14:27:25 +03:00
{
if (notNull) {
if (sqlite3_bind_int64(stmt, curArg++, value) != SQLITE_OK)
throwSQLiteError(stmt.db, "binding argument");
} else
bind();
return *this;
2016-03-30 14:27:25 +03:00
}
SQLiteStmt::Use & SQLiteStmt::Use::bind()
2016-03-30 14:27:25 +03:00
{
if (sqlite3_bind_null(stmt, curArg++) != SQLITE_OK)
throwSQLiteError(stmt.db, "binding argument");
return *this;
2016-03-30 14:27:25 +03:00
}
int SQLiteStmt::Use::step()
2016-03-30 14:27:25 +03:00
{
return sqlite3_step(stmt);
2016-03-30 14:27:25 +03:00
}
void SQLiteStmt::Use::exec()
2016-03-30 14:27:25 +03:00
{
int r = step();
assert(r != SQLITE_ROW);
if (r != SQLITE_DONE)
throwSQLiteError(stmt.db, "executing SQLite statement");
}
bool SQLiteStmt::Use::next()
{
int r = step();
if (r != SQLITE_DONE && r != SQLITE_ROW)
throwSQLiteError(stmt.db, "executing SQLite query");
return r == SQLITE_ROW;
}
std::string SQLiteStmt::Use::getStr(int col)
{
auto s = (const char *) sqlite3_column_text(stmt, col);
assert(s);
return s;
}
int64_t SQLiteStmt::Use::getInt(int col)
{
// FIXME: detect nulls?
return sqlite3_column_int64(stmt, col);
2016-03-30 14:27:25 +03:00
}
bool SQLiteStmt::Use::isNull(int col)
{
return sqlite3_column_type(stmt, col) == SQLITE_NULL;
}
2016-03-30 14:27:25 +03:00
SQLiteTxn::SQLiteTxn(sqlite3 * db)
{
this->db = db;
if (sqlite3_exec(db, "begin;", 0, 0, 0) != SQLITE_OK)
throwSQLiteError(db, "starting transaction");
active = true;
}
void SQLiteTxn::commit()
{
if (sqlite3_exec(db, "commit;", 0, 0, 0) != SQLITE_OK)
throwSQLiteError(db, "committing transaction");
active = false;
}
SQLiteTxn::~SQLiteTxn()
{
try {
if (active && sqlite3_exec(db, "rollback;", 0, 0, 0) != SQLITE_OK)
throwSQLiteError(db, "aborting transaction");
} catch (...) {
ignoreException();
}
}
}