mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-02-11 11:16:09 -05:00
![TheCharlatan](/assets/img/avatar_default.png)
Keep the generic serialization in the header, while moving leveldb-specifics to the implementation file. The context of this commit is an effort to decouple the dbwrapper header file from leveldb includes. To this end, the includes are moved to the dbwrapper implementation file. This is done as part of the kernel project to reduce the number of required includes for users of the kernel.
345 lines
9.3 KiB
C++
345 lines
9.3 KiB
C++
// Copyright (c) 2012-2022 The Bitcoin Core developers
|
|
// Distributed under the MIT software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
#ifndef BITCOIN_DBWRAPPER_H
|
|
#define BITCOIN_DBWRAPPER_H
|
|
|
|
#include <clientversion.h>
|
|
#include <logging.h>
|
|
#include <serialize.h>
|
|
#include <span.h>
|
|
#include <streams.h>
|
|
#include <util/fs.h>
|
|
|
|
#include <cstddef>
|
|
#include <cstdint>
|
|
#include <exception>
|
|
#include <leveldb/db.h>
|
|
#include <leveldb/options.h>
|
|
#include <leveldb/slice.h>
|
|
#include <leveldb/status.h>
|
|
#include <memory>
|
|
#include <optional>
|
|
#include <stdexcept>
|
|
#include <string>
|
|
#include <vector>
|
|
namespace leveldb {
|
|
class Env;
|
|
class Iterator;
|
|
}
|
|
|
|
static const size_t DBWRAPPER_PREALLOC_KEY_SIZE = 64;
|
|
static const size_t DBWRAPPER_PREALLOC_VALUE_SIZE = 1024;
|
|
|
|
//! User-controlled performance and debug options.
|
|
struct DBOptions {
|
|
//! Compact database on startup.
|
|
bool force_compact = false;
|
|
};
|
|
|
|
//! Application-specific storage settings.
|
|
struct DBParams {
|
|
//! Location in the filesystem where leveldb data will be stored.
|
|
fs::path path;
|
|
//! Configures various leveldb cache settings.
|
|
size_t cache_bytes;
|
|
//! If true, use leveldb's memory environment.
|
|
bool memory_only = false;
|
|
//! If true, remove all existing data.
|
|
bool wipe_data = false;
|
|
//! If true, store data obfuscated via simple XOR. If false, XOR with a
|
|
//! zero'd byte array.
|
|
bool obfuscate = false;
|
|
//! Passed-through options.
|
|
DBOptions options{};
|
|
};
|
|
|
|
inline auto CharCast(const std::byte* data) { return reinterpret_cast<const char*>(data); }
|
|
|
|
class dbwrapper_error : public std::runtime_error
|
|
{
|
|
public:
|
|
explicit dbwrapper_error(const std::string& msg) : std::runtime_error(msg) {}
|
|
};
|
|
|
|
class CDBWrapper;
|
|
|
|
/** These should be considered an implementation detail of the specific database.
|
|
*/
|
|
namespace dbwrapper_private {
|
|
|
|
/** Handle database error by throwing dbwrapper_error exception.
|
|
*/
|
|
void HandleError(const leveldb::Status& status);
|
|
|
|
/** Work around circular dependency, as well as for testing in dbwrapper_tests.
|
|
* Database obfuscation should be considered an implementation detail of the
|
|
* specific database.
|
|
*/
|
|
const std::vector<unsigned char>& GetObfuscateKey(const CDBWrapper &w);
|
|
|
|
}; // namespace dbwrapper_private
|
|
|
|
bool DestroyDB(const std::string& path_str);
|
|
|
|
/** Batch of changes queued to be written to a CDBWrapper */
|
|
class CDBBatch
|
|
{
|
|
friend class CDBWrapper;
|
|
|
|
private:
|
|
const CDBWrapper &parent;
|
|
|
|
struct WriteBatchImpl;
|
|
const std::unique_ptr<WriteBatchImpl> m_impl_batch;
|
|
|
|
DataStream ssKey{};
|
|
CDataStream ssValue;
|
|
|
|
size_t size_estimate{0};
|
|
|
|
void WriteImpl(Span<const std::byte> ssKey, CDataStream& ssValue);
|
|
void EraseImpl(Span<const std::byte> ssKey);
|
|
|
|
public:
|
|
/**
|
|
* @param[in] _parent CDBWrapper that this batch is to be submitted to
|
|
*/
|
|
explicit CDBBatch(const CDBWrapper& _parent);
|
|
~CDBBatch();
|
|
void Clear();
|
|
|
|
template <typename K, typename V>
|
|
void Write(const K& key, const V& value)
|
|
{
|
|
ssKey.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
|
|
ssValue.reserve(DBWRAPPER_PREALLOC_VALUE_SIZE);
|
|
ssKey << key;
|
|
ssValue << value;
|
|
WriteImpl(ssKey, ssValue);
|
|
ssKey.clear();
|
|
ssValue.clear();
|
|
}
|
|
|
|
template <typename K>
|
|
void Erase(const K& key)
|
|
{
|
|
ssKey.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
|
|
ssKey << key;
|
|
EraseImpl(ssKey);
|
|
ssKey.clear();
|
|
}
|
|
|
|
size_t SizeEstimate() const { return size_estimate; }
|
|
};
|
|
|
|
class CDBIterator
|
|
{
|
|
private:
|
|
const CDBWrapper &parent;
|
|
leveldb::Iterator *piter;
|
|
|
|
void SeekImpl(Span<const std::byte> ssKey);
|
|
Span<const std::byte> GetKeyImpl() const;
|
|
Span<const std::byte> GetValueImpl() const;
|
|
|
|
public:
|
|
|
|
/**
|
|
* @param[in] _parent Parent CDBWrapper instance.
|
|
* @param[in] _piter The original leveldb iterator.
|
|
*/
|
|
CDBIterator(const CDBWrapper &_parent, leveldb::Iterator *_piter) :
|
|
parent(_parent), piter(_piter) { };
|
|
~CDBIterator();
|
|
|
|
bool Valid() const;
|
|
|
|
void SeekToFirst();
|
|
|
|
template<typename K> void Seek(const K& key) {
|
|
DataStream ssKey{};
|
|
ssKey.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
|
|
ssKey << key;
|
|
SeekImpl(ssKey);
|
|
}
|
|
|
|
void Next();
|
|
|
|
template<typename K> bool GetKey(K& key) {
|
|
try {
|
|
DataStream ssKey{GetKeyImpl()};
|
|
ssKey >> key;
|
|
} catch (const std::exception&) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
template<typename V> bool GetValue(V& value) {
|
|
try {
|
|
CDataStream ssValue{GetValueImpl(), SER_DISK, CLIENT_VERSION};
|
|
ssValue.Xor(dbwrapper_private::GetObfuscateKey(parent));
|
|
ssValue >> value;
|
|
} catch (const std::exception&) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
};
|
|
|
|
class CDBWrapper
|
|
{
|
|
friend const std::vector<unsigned char>& dbwrapper_private::GetObfuscateKey(const CDBWrapper &w);
|
|
private:
|
|
//! custom environment this database is using (may be nullptr in case of default environment)
|
|
leveldb::Env* penv;
|
|
|
|
//! database options used
|
|
leveldb::Options options;
|
|
|
|
//! options used when reading from the database
|
|
leveldb::ReadOptions readoptions;
|
|
|
|
//! options used when iterating over values of the database
|
|
leveldb::ReadOptions iteroptions;
|
|
|
|
//! options used when writing to the database
|
|
leveldb::WriteOptions writeoptions;
|
|
|
|
//! options used when sync writing to the database
|
|
leveldb::WriteOptions syncoptions;
|
|
|
|
//! the database itself
|
|
leveldb::DB* pdb;
|
|
|
|
//! the name of this database
|
|
std::string m_name;
|
|
|
|
//! a key used for optional XOR-obfuscation of the database
|
|
std::vector<unsigned char> obfuscate_key;
|
|
|
|
//! the key under which the obfuscation key is stored
|
|
static const std::string OBFUSCATE_KEY_KEY;
|
|
|
|
//! the length of the obfuscate key in number of bytes
|
|
static const unsigned int OBFUSCATE_KEY_NUM_BYTES;
|
|
|
|
std::vector<unsigned char> CreateObfuscateKey() const;
|
|
|
|
//! path to filesystem storage
|
|
const fs::path m_path;
|
|
|
|
//! whether or not the database resides in memory
|
|
bool m_is_memory;
|
|
|
|
public:
|
|
CDBWrapper(const DBParams& params);
|
|
~CDBWrapper();
|
|
|
|
CDBWrapper(const CDBWrapper&) = delete;
|
|
CDBWrapper& operator=(const CDBWrapper&) = delete;
|
|
|
|
template <typename K, typename V>
|
|
bool Read(const K& key, V& value) const
|
|
{
|
|
DataStream ssKey{};
|
|
ssKey.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
|
|
ssKey << key;
|
|
leveldb::Slice slKey(CharCast(ssKey.data()), ssKey.size());
|
|
|
|
std::string strValue;
|
|
leveldb::Status status = pdb->Get(readoptions, slKey, &strValue);
|
|
if (!status.ok()) {
|
|
if (status.IsNotFound())
|
|
return false;
|
|
LogPrintf("LevelDB read failure: %s\n", status.ToString());
|
|
dbwrapper_private::HandleError(status);
|
|
}
|
|
try {
|
|
CDataStream ssValue{MakeByteSpan(strValue), SER_DISK, CLIENT_VERSION};
|
|
ssValue.Xor(obfuscate_key);
|
|
ssValue >> value;
|
|
} catch (const std::exception&) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
template <typename K, typename V>
|
|
bool Write(const K& key, const V& value, bool fSync = false)
|
|
{
|
|
CDBBatch batch(*this);
|
|
batch.Write(key, value);
|
|
return WriteBatch(batch, fSync);
|
|
}
|
|
|
|
//! @returns filesystem path to the on-disk data.
|
|
std::optional<fs::path> StoragePath() {
|
|
if (m_is_memory) {
|
|
return {};
|
|
}
|
|
return m_path;
|
|
}
|
|
|
|
template <typename K>
|
|
bool Exists(const K& key) const
|
|
{
|
|
DataStream ssKey{};
|
|
ssKey.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
|
|
ssKey << key;
|
|
leveldb::Slice slKey(CharCast(ssKey.data()), ssKey.size());
|
|
|
|
std::string strValue;
|
|
leveldb::Status status = pdb->Get(readoptions, slKey, &strValue);
|
|
if (!status.ok()) {
|
|
if (status.IsNotFound())
|
|
return false;
|
|
LogPrintf("LevelDB read failure: %s\n", status.ToString());
|
|
dbwrapper_private::HandleError(status);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
template <typename K>
|
|
bool Erase(const K& key, bool fSync = false)
|
|
{
|
|
CDBBatch batch(*this);
|
|
batch.Erase(key);
|
|
return WriteBatch(batch, fSync);
|
|
}
|
|
|
|
bool WriteBatch(CDBBatch& batch, bool fSync = false);
|
|
|
|
// Get an estimate of LevelDB memory usage (in bytes).
|
|
size_t DynamicMemoryUsage() const;
|
|
|
|
CDBIterator *NewIterator()
|
|
{
|
|
return new CDBIterator(*this, pdb->NewIterator(iteroptions));
|
|
}
|
|
|
|
/**
|
|
* Return true if the database managed by this class contains no entries.
|
|
*/
|
|
bool IsEmpty();
|
|
|
|
template<typename K>
|
|
size_t EstimateSize(const K& key_begin, const K& key_end) const
|
|
{
|
|
DataStream ssKey1{}, ssKey2{};
|
|
ssKey1.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
|
|
ssKey2.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
|
|
ssKey1 << key_begin;
|
|
ssKey2 << key_end;
|
|
leveldb::Slice slKey1(CharCast(ssKey1.data()), ssKey1.size());
|
|
leveldb::Slice slKey2(CharCast(ssKey2.data()), ssKey2.size());
|
|
uint64_t size = 0;
|
|
leveldb::Range range(slKey1, slKey2);
|
|
pdb->GetApproximateSizes(&range, 1, &size);
|
|
return size;
|
|
}
|
|
};
|
|
|
|
#endif // BITCOIN_DBWRAPPER_H
|