2018-06-12 17:29:22 +08:00
|
|
|
#include <rocksdb/db.h>
|
|
|
|
#include <rocksdb/options.h>
|
2018-04-27 17:23:40 +08:00
|
|
|
|
2020-01-24 17:45:05 +08:00
|
|
|
#include "kvstore/kvstore.hpp"
|
2018-04-27 17:23:40 +08:00
|
|
|
#include "utils/file.hpp"
|
|
|
|
|
2020-01-24 17:45:05 +08:00
|
|
|
namespace kvstore {
|
2018-04-27 17:23:40 +08:00
|
|
|
|
2018-06-12 17:29:22 +08:00
|
|
|
struct KVStore::impl {
|
2019-04-23 17:00:49 +08:00
|
|
|
std::filesystem::path storage;
|
2018-06-12 17:29:22 +08:00
|
|
|
std::unique_ptr<rocksdb::DB> db;
|
|
|
|
rocksdb::Options options;
|
|
|
|
};
|
|
|
|
|
2019-04-23 17:00:49 +08:00
|
|
|
KVStore::KVStore(std::filesystem::path storage)
|
2019-02-14 17:34:09 +08:00
|
|
|
: pimpl_(std::make_unique<impl>()) {
|
2018-06-12 17:29:22 +08:00
|
|
|
pimpl_->storage = storage;
|
|
|
|
if (!utils::EnsureDir(pimpl_->storage))
|
|
|
|
throw KVStoreError("Folder for the key-value store " +
|
|
|
|
pimpl_->storage.string() + " couldn't be initialized!");
|
|
|
|
pimpl_->options.create_if_missing = true;
|
2018-04-27 17:23:40 +08:00
|
|
|
rocksdb::DB *db = nullptr;
|
2018-06-12 17:29:22 +08:00
|
|
|
auto s = rocksdb::DB::Open(pimpl_->options, storage.c_str(), &db);
|
2018-04-27 17:23:40 +08:00
|
|
|
if (!s.ok())
|
|
|
|
throw KVStoreError("RocksDB couldn't be initialized inside " +
|
2018-06-12 17:29:22 +08:00
|
|
|
storage.string() + " -- " + std::string(s.ToString()));
|
|
|
|
pimpl_->db.reset(db);
|
|
|
|
}
|
|
|
|
|
|
|
|
KVStore::~KVStore() {}
|
|
|
|
|
|
|
|
KVStore::KVStore(KVStore &&other) { pimpl_ = std::move(other.pimpl_); }
|
|
|
|
|
|
|
|
KVStore &KVStore::operator=(KVStore &&other) {
|
|
|
|
pimpl_ = std::move(other.pimpl_);
|
|
|
|
return *this;
|
2018-04-27 17:23:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool KVStore::Put(const std::string &key, const std::string &value) {
|
2018-06-12 17:29:22 +08:00
|
|
|
auto s = pimpl_->db->Put(rocksdb::WriteOptions(), key, value);
|
|
|
|
return s.ok();
|
2018-04-27 17:23:40 +08:00
|
|
|
}
|
|
|
|
|
2018-08-14 17:34:00 +08:00
|
|
|
bool KVStore::PutMultiple(const std::map<std::string, std::string> &items) {
|
|
|
|
rocksdb::WriteBatch batch;
|
|
|
|
for (const auto &item : items) {
|
|
|
|
batch.Put(item.first, item.second);
|
|
|
|
}
|
|
|
|
auto s = pimpl_->db->Write(rocksdb::WriteOptions(), &batch);
|
|
|
|
return s.ok();
|
|
|
|
}
|
|
|
|
|
2019-04-23 17:00:49 +08:00
|
|
|
std::optional<std::string> KVStore::Get(const std::string &key) const noexcept {
|
2018-04-27 17:23:40 +08:00
|
|
|
std::string value;
|
2018-06-12 17:29:22 +08:00
|
|
|
auto s = pimpl_->db->Get(rocksdb::ReadOptions(), key, &value);
|
2019-04-23 17:00:49 +08:00
|
|
|
if (!s.ok()) return std::nullopt;
|
2018-04-27 17:23:40 +08:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool KVStore::Delete(const std::string &key) {
|
2018-06-12 17:29:22 +08:00
|
|
|
auto s = pimpl_->db->Delete(rocksdb::WriteOptions(), key);
|
|
|
|
return s.ok();
|
|
|
|
}
|
|
|
|
|
2018-08-14 17:34:00 +08:00
|
|
|
bool KVStore::DeleteMultiple(const std::vector<std::string> &keys) {
|
|
|
|
rocksdb::WriteBatch batch;
|
|
|
|
for (const auto &key : keys) {
|
|
|
|
batch.Delete(key);
|
|
|
|
}
|
|
|
|
auto s = pimpl_->db->Write(rocksdb::WriteOptions(), &batch);
|
|
|
|
return s.ok();
|
|
|
|
}
|
|
|
|
|
2018-06-12 17:29:22 +08:00
|
|
|
bool KVStore::DeletePrefix(const std::string &prefix) {
|
|
|
|
std::unique_ptr<rocksdb::Iterator> iter = std::unique_ptr<rocksdb::Iterator>(
|
|
|
|
pimpl_->db->NewIterator(rocksdb::ReadOptions()));
|
|
|
|
for (iter->Seek(prefix); iter->Valid() && iter->key().starts_with(prefix);
|
|
|
|
iter->Next()) {
|
|
|
|
if (!pimpl_->db->Delete(rocksdb::WriteOptions(), iter->key()).ok())
|
|
|
|
return false;
|
|
|
|
}
|
2018-04-27 17:23:40 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-08-14 17:34:00 +08:00
|
|
|
bool KVStore::PutAndDeleteMultiple(
|
|
|
|
const std::map<std::string, std::string> &items,
|
|
|
|
const std::vector<std::string> &keys) {
|
|
|
|
rocksdb::WriteBatch batch;
|
|
|
|
for (const auto &item : items) {
|
|
|
|
batch.Put(item.first, item.second);
|
|
|
|
}
|
|
|
|
for (const auto &key : keys) {
|
|
|
|
batch.Delete(key);
|
|
|
|
}
|
|
|
|
auto s = pimpl_->db->Write(rocksdb::WriteOptions(), &batch);
|
|
|
|
return s.ok();
|
|
|
|
}
|
|
|
|
|
2018-06-12 17:29:22 +08:00
|
|
|
// iterator
|
|
|
|
|
|
|
|
struct KVStore::iterator::impl {
|
|
|
|
const KVStore *kvstore;
|
|
|
|
std::string prefix;
|
|
|
|
std::unique_ptr<rocksdb::Iterator> it;
|
|
|
|
std::pair<std::string, std::string> disk_prop;
|
|
|
|
};
|
|
|
|
|
|
|
|
KVStore::iterator::iterator(const KVStore *kvstore, const std::string &prefix,
|
|
|
|
bool at_end)
|
|
|
|
: pimpl_(std::make_unique<impl>()) {
|
|
|
|
pimpl_->kvstore = kvstore;
|
|
|
|
pimpl_->prefix = prefix;
|
|
|
|
pimpl_->it = std::unique_ptr<rocksdb::Iterator>(
|
|
|
|
pimpl_->kvstore->pimpl_->db->NewIterator(rocksdb::ReadOptions()));
|
|
|
|
pimpl_->it->Seek(pimpl_->prefix);
|
|
|
|
if (!pimpl_->it->Valid() || !pimpl_->it->key().starts_with(pimpl_->prefix) ||
|
|
|
|
at_end)
|
|
|
|
pimpl_->it = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
KVStore::iterator::iterator(KVStore::iterator &&other) {
|
|
|
|
pimpl_ = std::move(other.pimpl_);
|
|
|
|
}
|
|
|
|
|
|
|
|
KVStore::iterator::~iterator() {}
|
|
|
|
|
|
|
|
KVStore::iterator &KVStore::iterator::operator=(KVStore::iterator &&other) {
|
|
|
|
pimpl_ = std::move(other.pimpl_);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
KVStore::iterator &KVStore::iterator::operator++() {
|
|
|
|
pimpl_->it->Next();
|
|
|
|
if (!pimpl_->it->Valid() || !pimpl_->it->key().starts_with(pimpl_->prefix))
|
|
|
|
pimpl_->it = nullptr;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool KVStore::iterator::operator==(const iterator &other) const {
|
|
|
|
return pimpl_->kvstore == other.pimpl_->kvstore &&
|
|
|
|
pimpl_->prefix == other.pimpl_->prefix &&
|
|
|
|
pimpl_->it == other.pimpl_->it;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool KVStore::iterator::operator!=(const iterator &other) const {
|
|
|
|
return !(*this == other);
|
|
|
|
}
|
|
|
|
|
|
|
|
KVStore::iterator::reference KVStore::iterator::operator*() {
|
|
|
|
pimpl_->disk_prop = {pimpl_->it->key().ToString(),
|
|
|
|
pimpl_->it->value().ToString()};
|
|
|
|
return pimpl_->disk_prop;
|
|
|
|
}
|
|
|
|
|
|
|
|
KVStore::iterator::pointer KVStore::iterator::operator->() { return &**this; }
|
|
|
|
|
|
|
|
void KVStore::iterator::SetInvalid() { pimpl_->it = nullptr; }
|
|
|
|
|
|
|
|
bool KVStore::iterator::IsValid() { return pimpl_->it != nullptr; }
|
|
|
|
|
|
|
|
// TODO(ipaljak) The complexity of the size function should be at most
|
|
|
|
// logarithmic.
|
|
|
|
size_t KVStore::Size(const std::string &prefix) {
|
|
|
|
size_t size = 0;
|
|
|
|
for (auto it = this->begin(prefix); it != this->end(prefix); ++it) ++size;
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2019-03-29 21:14:08 +08:00
|
|
|
bool KVStore::CompactRange(const std::string &begin_prefix,
|
|
|
|
const std::string &end_prefix) {
|
|
|
|
rocksdb::CompactRangeOptions options;
|
|
|
|
rocksdb::Slice begin(begin_prefix);
|
|
|
|
rocksdb::Slice end(end_prefix);
|
|
|
|
auto s = pimpl_->db->CompactRange(options, &begin, &end);
|
|
|
|
return s.ok();
|
|
|
|
}
|
|
|
|
|
2020-01-24 17:45:05 +08:00
|
|
|
} // namespace kvstore
|