7#include <blockfilter.h>
77 size_t n_cache_size,
bool f_memory,
bool f_wipe)
79 , m_filter_type(filter_type)
82 if (filter_name.empty())
throw std::invalid_argument(
"unknown filter_type");
85 fs::create_directories(path);
87 m_db = std::make_unique<BaseIndex::DB>(path /
"db", n_cache_size, f_memory, f_wipe);
105 LogError(
"Cannot read current %s state; index may be corrupted",
117 if (!op_last_header) {
118 LogError(
"Cannot read last block filter header; index may be corrupted");
137 if (!file.Commit()) {
142 if (file.fclose() != 0) {
154 if (filein.IsNull()) {
160 std::vector<uint8_t> encoded_filter;
162 filein >> block_hash >> encoded_filter;
163 if (
Hash(encoded_filter) != hash) {
164 LogError(
"Checksum mismatch in filter decode.");
169 catch (
const std::exception& e) {
170 LogError(
"Failed to deserialize block filter from disk: %s", e.what());
188 if (last_file.IsNull()) {
192 if (!last_file.Truncate(pos.
nPos)) {
196 if (!last_file.Commit()) {
198 (void)last_file.fclose();
201 if (last_file.fclose() != 0) {
219 if (fileout.IsNull()) {
226 if (fileout.fclose() != 0) {
236 std::pair<uint256, DBVal> read_out;
241 if (read_out.first != expected_block_hash) {
242 LogError(
"previous block header belongs to unexpected block %s; expected %s",
243 read_out.first.ToString(), expected_block_hash.
ToString());
247 return read_out.second.header;
262 if (bytes_written == 0)
return false;
264 std::pair<uint256, DBVal> value;
266 value.second.hash = filter.
GetHash();
267 value.second.header = filter_header;
279 std::unique_ptr<CDBIterator> db_it(
m_db->NewIterator());
284 if (!index_util::CopyHeightIndexToHashIndex<DBVal>(*db_it, batch,
m_name, block.
height)) {
292 m_db->WriteBatch(batch);
300 const CBlockIndex* stop_index, std::vector<DBVal>& results)
302 if (start_height < 0) {
303 LogError(
"start height (%d) is negative", start_height);
306 if (start_height > stop_index->
nHeight) {
307 LogError(
"start height (%d) is greater than stop height (%d)",
308 start_height, stop_index->
nHeight);
312 size_t results_size =
static_cast<size_t>(stop_index->
nHeight - start_height + 1);
313 std::vector<std::pair<uint256, DBVal>>
values(results_size);
316 std::unique_ptr<CDBIterator> db_it(db.
NewIterator());
318 for (
int height = start_height; height <= stop_index->
nHeight; ++height) {
319 if (!db_it->Valid() || !db_it->GetKey(key) || key.
height != height) {
323 size_t i =
static_cast<size_t>(height - start_height);
324 if (!db_it->GetValue(
values[i])) {
325 LogError(
"unable to read value in %s at key (%c, %d)",
333 results.resize(results_size);
338 block_index && block_index->
nHeight >= start_height;
339 block_index = block_index->pprev) {
340 uint256 block_hash = block_index->GetBlockHash();
342 size_t i =
static_cast<size_t>(block_index->nHeight - start_height);
343 if (block_hash ==
values[i].first) {
344 results[i] = std::move(
values[i].second);
349 LogError(
"unable to read value in %s at key (%c, %s)",
376 auto header = m_headers_cache.find(block_index->
GetBlockHash());
377 if (header != m_headers_cache.end()) {
378 header_out = header->second;
391 m_headers_cache.emplace(block_index->
GetBlockHash(), entry.header);
394 header_out = entry.header;
399 std::vector<BlockFilter>& filters_out)
const
401 std::vector<DBVal> entries;
406 filters_out.resize(entries.size());
407 auto filter_pos_it = filters_out.begin();
408 for (
const auto& entry : entries) {
419 std::vector<uint256>& hashes_out)
const
422 std::vector<DBVal> entries;
428 hashes_out.reserve(entries.size());
429 for (
const auto& entry : entries) {
430 hashes_out.push_back(entry.hash);
447 size_t n_cache_size,
bool f_memory,
bool f_wipe)
450 std::forward_as_tuple(filter_type),
451 std::forward_as_tuple(make_chain(), filter_type,
452 n_cache_size, f_memory, f_wipe));
453 return result.second;
const std::string & BlockFilterTypeName(BlockFilterType filter_type)
Get the human-readable name for a filter type.
constexpr unsigned int FLTR_FILE_CHUNK_SIZE
The pre-allocation chunk size for fltr?????.dat files.
bool DestroyBlockFilterIndex(BlockFilterType filter_type)
Destroy the block filter index with the given type.
void DestroyAllBlockFilterIndexes()
Destroy all open block filter indexes.
BlockFilterIndex * GetBlockFilterIndex(BlockFilterType filter_type)
Get a block filter index by type.
constexpr uint8_t DB_FILTER_POS
constexpr unsigned int MAX_FLTR_FILE_SIZE
void ForEachBlockFilterIndex(std::function< void(BlockFilterIndex &)> fn)
Iterate over all running block filter indexes, invoking fn on each.
constexpr size_t CF_HEADERS_CACHE_MAX_SZ
Maximum size of the cfheaders cache We have a limit to prevent a bug in filling this cache potentiall...
bool InitBlockFilterIndex(std::function< std::unique_ptr< interfaces::Chain >()> make_chain, BlockFilterType filter_type, size_t n_cache_size, bool f_memory, bool f_wipe)
Initialize a block filter index for the given type if one does not already exist.
static bool LookupRange(CDBWrapper &db, const std::string &index_name, int start_height, const CBlockIndex *stop_index, std::vector< DBVal > &results)
static std::map< BlockFilterType, BlockFilterIndex > g_filter_indexes
static constexpr int CFCHECKPT_INTERVAL
Interval between compact filter checkpoints.
#define Assert(val)
Identity function.
fs::path GetDataDirNet() const EXCLUSIVE_LOCKS_REQUIRED(!cs_args)
Get data directory path with appended network identifier.
Non-refcounted RAII wrapper for FILE*.
Base class for indices of blockchain data.
const std::string & GetName() const LIFETIMEBOUND
Get the name of the index for display in logs.
Complete block filter struct as defined in BIP 157.
const uint256 & GetBlockHash() const LIFETIMEBOUND
const std::vector< unsigned char > & GetEncodedFilter() const LIFETIMEBOUND
uint256 ComputeHeader(const uint256 &prev_header) const
Compute the filter header given the previous one.
BlockFilterType GetFilterType() const
uint256 GetHash() const
Compute the filter hash.
BlockFilterIndex is used to store and retrieve block filters, hashes, and headers for a range of bloc...
std::unique_ptr< BaseIndex::DB > m_db
bool CustomInit(const std::optional< interfaces::BlockRef > &block) override
Initialize internal state from the database and block index.
bool LookupFilterRange(int start_height, const CBlockIndex *stop_index, std::vector< BlockFilter > &filters_out) const
Get a range of filters between two heights on a chain.
BlockFilterType GetFilterType() const
bool CustomRemove(const interfaces::BlockInfo &block) override
Rewind index by one block during a chain reorg.
bool CustomCommit(CDBBatch &batch) override
Virtual method called internally by Commit that can be overridden to atomically commit more index sta...
BlockFilterType m_filter_type
BlockFilterIndex(std::unique_ptr< interfaces::Chain > chain, BlockFilterType filter_type, size_t n_cache_size, bool f_memory=false, bool f_wipe=false)
Constructs the index, which becomes available to be queried.
interfaces::Chain::NotifyOptions CustomOptions() override
Return custom notification options for index.
std::unique_ptr< FlatFileSeq > m_filter_fileseq
bool LookupFilter(const CBlockIndex *block_index, BlockFilter &filter_out) const
Get a single filter by block.
bool ReadFilterFromDisk(const FlatFilePos &pos, const uint256 &hash, BlockFilter &filter) const
bool LookupFilterHashRange(int start_height, const CBlockIndex *stop_index, std::vector< uint256 > &hashes_out) const
Get a range of filter hashes between two heights on a chain.
bool CustomAppend(const interfaces::BlockInfo &block) override
Write update index entries for a newly connected block.
size_t WriteFilterToDisk(FlatFilePos &pos, const BlockFilter &filter)
bool LookupFilterHeader(const CBlockIndex *block_index, uint256 &header_out) EXCLUSIVE_LOCKS_REQUIRED(!m_cs_headers_cache)
Get a single filter header by block.
std::optional< uint256 > ReadFilterHeader(int height, const uint256 &expected_block_hash)
bool Write(const BlockFilter &filter, uint32_t block_height, const uint256 &filter_header)
FlatFilePos m_next_filter_pos
The block chain is a tree shaped structure starting with the genesis block at the root,...
uint256 GetBlockHash() const
int nHeight
height of the entry in the chain. The genesis block has height 0
Batch of changes queued to be written to a CDBWrapper.
void Write(const K &key, const V &value)
bool Read(const K &key, V &value) const
CDBIterator * NewIterator()
std::string ToString() const
static path u8path(std::string_view utf8_str)
uint256 Hash(const T &in1)
Compute the 256-bit hash of an object.
static bool LookUpOne(const CDBWrapper &db, const interfaces::BlockRef &block, DBVal &result)
static constexpr uint8_t DB_BLOCK_HASH
static constexpr uint8_t DB_BLOCK_HEIGHT
static const int64_t values[]
A selection of numbers that do not trigger int64_t overflow when added/subtracted.
#define SERIALIZE_METHODS(cls, obj)
Implement the Serialize and Unserialize methods by delegating to a single templated static method tha...
uint64_t GetSerializeSize(const T &t)
Block data sent with blockConnected, blockDisconnected notifications.
const uint256 * prev_hash
const CBlockUndo * undo_data
Options specifying which chain notifications are required.
bool connect_undo_data
Include undo data with block connected notifications.
std::string SysErrorString(int err)
Return system error string from errno value.