os: KeyValueDB: allow finer-grained control of transaction operations

This patch introduces the possibility of using single key/value
modification operations into the transaction interface.

Until now, any 'set' or 'rmkeys' operations required a map of keys to be
provided to the function, which made the task of removing or setting a
bunch of keys easier. Doing these same operations for a single key,
however, would entail creating a map with a single key.

Instead, this patch adds two new virtual abstract functions, to be
implemented by derivative classes, which set or remove one single
key/value, and we then implement the map-based, existing functions in
terms of these new functions.

We also update the derivative classes of KeyValueDB in order to reflect
these changes (i.e., LevelDBStore and KeyValueDBMemory).

Signed-off-by: Joao Eduardo Luis <joao.luis@inktank.com>
This commit is contained in:
Joao Eduardo Luis 2012-07-24 02:23:01 +01:00
parent 4e1d973e46
commit a16d9c64da
5 changed files with 77 additions and 61 deletions

View File

@ -1,4 +1,5 @@
// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*- // -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
// vim: ts=8 sw=2 smarttab
#ifndef KEY_VALUE_DB_H #ifndef KEY_VALUE_DB_H
#define KEY_VALUE_DB_H #define KEY_VALUE_DB_H
@ -20,15 +21,37 @@ public:
class TransactionImpl { class TransactionImpl {
public: public:
/// Set Keys /// Set Keys
virtual void set( void set(
const string &prefix, ///< [in] Prefix for keys const string &prefix, ///< [in] Prefix for keys
const std::map<string, bufferlist> &to_set ///< [in] keys/values to set const std::map<string, bufferlist> &to_set ///< [in] keys/values to set
) {
std::map<string, bufferlist>::const_iterator it;
for (it = to_set.begin(); it != to_set.end(); ++it)
set(prefix, it->first, it->second);
}
/// Set Key
virtual void set(
const string &prefix, ///< [in] Prefix for the key
const string &k, ///< [in] Key to set
const bufferlist &bl ///< [in] Value to set
) = 0; ) = 0;
/// Removes Keys /// Removes Keys
virtual void rmkeys( void rmkeys(
const string &prefix, ///< [in] Prefix to search for const string &prefix, ///< [in] Prefix to search for
const std::set<string> &keys ///< [in] Keys to remove const std::set<string> &keys ///< [in] Keys to remove
) {
std::set<string>::const_iterator it;
for (it = keys.begin(); it != keys.end(); ++it)
rmkey(prefix, *it);
}
/// Remove Key
virtual void rmkey(
const string &prefix, ///< [in] Prefix to search for
const string &k ///< [in] Key to remove
) = 0; ) = 0;
/// Removes keys beginning with prefix /// Removes keys beginning with prefix

View File

@ -1,5 +1,5 @@
// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*- // -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
// vim: ts=8 sw=2 smarttab
#include "LevelDBStore.h" #include "LevelDBStore.h"
#include <set> #include <set>
@ -28,31 +28,25 @@ int LevelDBStore::init(ostream &out)
void LevelDBStore::LevelDBTransactionImpl::set( void LevelDBStore::LevelDBTransactionImpl::set(
const string &prefix, const string &prefix,
const std::map<string, bufferlist> &to_set) const string &k,
const bufferlist &to_set_bl)
{ {
for (std::map<string, bufferlist>::const_iterator i = to_set.begin(); buffers.push_back(to_set_bl);
i != to_set.end(); buffers.rbegin()->rebuild();
++i) { bufferlist &bl = *(buffers.rbegin());
buffers.push_back(i->second); string key = combine_strings(prefix, k);
buffers.rbegin()->rebuild(); keys.push_back(key);
bufferlist &bl = *(buffers.rbegin()); bat.Delete(leveldb::Slice(*(keys.rbegin())));
string key = combine_strings(prefix, i->first); bat.Put(leveldb::Slice(*(keys.rbegin())),
keys.push_back(key); leveldb::Slice(bl.c_str(), bl.length()));
bat.Delete(leveldb::Slice(*(keys.rbegin())));
bat.Put(leveldb::Slice(*(keys.rbegin())),
leveldb::Slice(bl.c_str(), bl.length()));
}
} }
void LevelDBStore::LevelDBTransactionImpl::rmkeys(const string &prefix,
const std::set<string> &to_rm) void LevelDBStore::LevelDBTransactionImpl::rmkey(const string &prefix,
const string &k)
{ {
for (std::set<string>::const_iterator i = to_rm.begin(); string key = combine_strings(prefix, k);
i != to_rm.end(); keys.push_back(key);
++i) { bat.Delete(leveldb::Slice(*(keys.rbegin())));
string key = combine_strings(prefix, *i);
keys.push_back(key);
bat.Delete(leveldb::Slice(*(keys.rbegin())));
}
} }
void LevelDBStore::LevelDBTransactionImpl::rmkeys_by_prefix(const string &prefix) void LevelDBStore::LevelDBTransactionImpl::rmkeys_by_prefix(const string &prefix)

View File

@ -36,12 +36,11 @@ public:
LevelDBTransactionImpl(LevelDBStore *db) : db(db) {} LevelDBTransactionImpl(LevelDBStore *db) : db(db) {}
void set( void set(
const string &prefix, const string &prefix,
const std::map<string, bufferlist> &to_set const string &k,
); const bufferlist &bl);
void rmkeys( void rmkey(
const string &prefix, const string &prefix,
const std::set<string> &keys const string &k);
);
void rmkeys_by_prefix( void rmkeys_by_prefix(
const string &prefix const string &prefix
); );

View File

@ -1,4 +1,5 @@
// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*- // -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
// vim: ts=8 sw=2 smarttab
#include "include/encoding.h" #include "include/encoding.h"
#include "KeyValueDBMemory.h" #include "KeyValueDBMemory.h"
#include <map> #include <map>
@ -133,25 +134,18 @@ int KeyValueDBMemory::get_keys(const string &prefix,
} }
int KeyValueDBMemory::set(const string &prefix, int KeyValueDBMemory::set(const string &prefix,
const map<string, bufferlist> &to_set) { const string &key,
for (map<string, bufferlist>::const_iterator i = to_set.begin(); const bufferlist &bl) {
i != to_set.end(); db[prefix][key] = bl;
++i) {
bufferlist bl = i->second;
db[prefix][i->first] = i->second;
}
return 0; return 0;
} }
int KeyValueDBMemory::rmkeys(const string &prefix, int KeyValueDBMemory::rmkey(const string &prefix,
const std::set<string> &keys) { const string &key) {
if (!db.count(prefix)) db[prefix].erase(key);
return 0; if (db[prefix].size() == 0)
for (std::set<string>::const_iterator i = keys.begin(); db.erase(prefix);
i != keys.end();
++i) {
db[prefix].erase(*i);
}
return 0; return 0;
} }

View File

@ -1,4 +1,5 @@
// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*- // -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
// vim: ts=8 sw=2 smarttab
#include <map> #include <map>
#include <set> #include <set>
#include <string> #include <string>
@ -28,12 +29,13 @@ public:
int set( int set(
const string &prefix, const string &prefix,
const std::map<string, bufferlist> &to_set const string &key,
const bufferlist &bl
); );
int rmkeys( int rmkey(
const string &prefix, const string &prefix,
const std::set<string> &keys const string &key
); );
int rmkeys_by_prefix( int rmkeys_by_prefix(
@ -51,33 +53,37 @@ public:
struct SetOp : public Context { struct SetOp : public Context {
KeyValueDBMemory *db; KeyValueDBMemory *db;
string prefix; string prefix;
std::map<string, bufferlist> to_set; string key;
bufferlist value;
SetOp(KeyValueDBMemory *db, SetOp(KeyValueDBMemory *db,
const string &prefix, const string &prefix,
const std::map<string, bufferlist> &to_set) const string &key,
: db(db), prefix(prefix), to_set(to_set) {} const bufferlist &value)
: db(db), prefix(prefix), key(key), value(value) {}
void finish(int r) { void finish(int r) {
db->set(prefix, to_set); db->set(prefix, key, value);
} }
}; };
void set(const string &prefix, const std::map<string, bufferlist> &to_set) {
on_commit.push_back(new SetOp(db, prefix, to_set)); void set(const string &prefix, const string &k, const bufferlist& bl) {
on_commit.push_back(new SetOp(db, prefix, k, bl));
} }
struct RmKeysOp : public Context { struct RmKeysOp : public Context {
KeyValueDBMemory *db; KeyValueDBMemory *db;
string prefix; string prefix;
std::set<string> keys; string key;
RmKeysOp(KeyValueDBMemory *db, RmKeysOp(KeyValueDBMemory *db,
const string &prefix, const string &prefix,
const std::set<string> &keys) const string &key)
: db(db), prefix(prefix), keys(keys) {} : db(db), prefix(prefix), key(key) {}
void finish(int r) { void finish(int r) {
db->rmkeys(prefix, keys); db->rmkey(prefix, key);
} }
}; };
void rmkeys(const string &prefix, const std::set<string> &to_remove) {
on_commit.push_back(new RmKeysOp(db, prefix, to_remove)); void rmkey(const string &prefix, const string &key) {
on_commit.push_back(new RmKeysOp(db, prefix, key));
} }
struct RmKeysByPrefixOp : public Context { struct RmKeysByPrefixOp : public Context {