#include "db.hh" #include "util.hh" #include <memory> /* Wrapper class to ensure proper destruction. */ class DestroyDb { Db * db; public: DestroyDb(Db * _db) : db(_db) { } ~DestroyDb() { db->close(0); delete db; } }; class DestroyDbc { Dbc * dbc; public: DestroyDbc(Dbc * _dbc) : dbc(_dbc) { } ~DestroyDbc() { dbc->close(); /* close() frees dbc */ } }; static void rethrow(DbException & e) { throw Error(e.what()); } Transaction::Transaction() : txn(0) { } Transaction::Transaction(Database & db) { db.requireEnv(); db.env->txn_begin(0, &txn, 0); } Transaction::~Transaction() { if (txn) { txn->abort(); txn = 0; } } void Transaction::commit() { if (!txn) throw Error("commit called on null transaction"); txn->commit(0); txn = 0; } void Database::requireEnv() { if (!env) throw Error("database environment not open"); } Db * Database::openDB(const Transaction & txn, const string & table, bool create) { requireEnv(); Db * db = new Db(env, 0); try { // !!! fixme when switching to BDB 4.1: use txn. db->open(table.c_str(), 0, DB_HASH, create ? DB_CREATE : 0, 0666); } catch (...) { delete db; throw; } return db; } Database::Database() : env(0) { } Database::~Database() { if (env) { env->close(0); delete env; } } void Database::open(const string & path) { try { if (env) throw Error(format("environment already open")); env = new DbEnv(0); debug("foo" + path); env->open(path.c_str(), DB_INIT_LOCK | DB_INIT_LOG | DB_INIT_MPOOL | DB_INIT_TXN | DB_CREATE, 0666); } catch (DbException e) { rethrow(e); } } void Database::createTable(const string & table) { try { Db * db = openDB(noTxn, table, true); DestroyDb destroyDb(db); } catch (DbException e) { rethrow(e); } } bool Database::queryString(const Transaction & txn, const string & table, const string & key, string & data) { try { Db * db = openDB(txn, table, false); DestroyDb destroyDb(db); Dbt kt((void *) key.c_str(), key.length()); Dbt dt; int err = db->get(txn.txn, &kt, &dt, 0); if (err) return false; if (!dt.get_data()) data = ""; else data = string((char *) dt.get_data(), dt.get_size()); } catch (DbException e) { rethrow(e); } return true; } bool Database::queryStrings(const Transaction & txn, const string & table, const string & key, Strings & data) { string d; if (!queryString(txn, table, key, d)) return false; string::iterator it = d.begin(); while (it != d.end()) { if (it + 4 > d.end()) throw Error(format("short db entry: `%1%'") % d); unsigned int len; len = (unsigned char) *it++; len |= ((unsigned char) *it++) << 8; len |= ((unsigned char) *it++) << 16; len |= ((unsigned char) *it++) << 24; if (it + len > d.end()) throw Error(format("short db entry: `%1%'") % d); string s; while (len--) s += *it++; data.push_back(s); } return true; } void Database::setString(const Transaction & txn, const string & table, const string & key, const string & data) { try { Db * db = openDB(txn, table, false); DestroyDb destroyDb(db); Dbt kt((void *) key.c_str(), key.length()); Dbt dt((void *) data.c_str(), data.length()); db->put(txn.txn, &kt, &dt, 0); } catch (DbException e) { rethrow(e); } } void Database::setStrings(const Transaction & txn, const string & table, const string & key, const Strings & data) { string d; for (Strings::const_iterator it = data.begin(); it != data.end(); it++) { string s = *it; unsigned int len = s.size(); d += len & 0xff; d += (len >> 8) & 0xff; d += (len >> 16) & 0xff; d += (len >> 24) & 0xff; d += s; } setString(txn, table, key, d); } void Database::delPair(const Transaction & txn, const string & table, const string & key) { try { Db * db = openDB(txn, table, false); DestroyDb destroyDb(db); Dbt kt((void *) key.c_str(), key.length()); db->del(txn.txn, &kt, 0); } catch (DbException e) { rethrow(e); } } void Database::enumTable(const Transaction & txn, const string & table, Strings & keys) { try { Db * db = openDB(txn, table, false); DestroyDb destroyDb(db); Dbc * dbc; db->cursor(0, &dbc, 0); DestroyDbc destroyDbc(dbc); Dbt kt, dt; while (dbc->get(&kt, &dt, DB_NEXT) != DB_NOTFOUND) keys.push_back( string((char *) kt.get_data(), kt.get_size())); } catch (DbException e) { rethrow(e); } }