7#include <chainparams.h>
20#include <boost/test/unit_test.hpp>
27BOOST_AUTO_TEST_SUITE(blockfilter_index_tests)
30 CBlock CreateBlock(
const CBlockIndex* prev,
const std::vector<CMutableTransaction>& txns,
const CScript& scriptPubKey);
31 bool BuildChain(
const CBlockIndex* pindex,
const CScript& coinbase_script_pub_key,
size_t length, std::vector<std::shared_ptr<CBlock>>& chain);
39 BOOST_ERROR(
"ComputeFilter failed on block " << block_index->
nHeight);
45 std::vector<BlockFilter> filters;
46 std::vector<uint256> filter_hashes;
63 filter_hashes.clear();
64 last_header = filter_header;
69 const std::vector<CMutableTransaction>& txns,
72 BlockAssembler::Options options;
73 options.coinbase_output_script = scriptPubKey;
74 options.include_dummy_extranonce =
true;
76 CBlock& block = pblocktemplate->block;
87 tx_coinbase.nLockTime =
static_cast<uint32_t
>(prev->
nHeight);
99 const CScript& coinbase_script_pub_key,
101 std::vector<std::shared_ptr<CBlock>>& chain)
103 std::vector<CMutableTransaction> no_txns;
105 chain.resize(length);
106 for (
auto& block : chain) {
107 block = std::make_shared<CBlock>(
CreateBlock(pindex, no_txns, coinbase_script_pub_key));
121 BOOST_REQUIRE(filter_index.
Init());
131 std::vector<BlockFilter> filters;
132 std::vector<uint256> filter_hashes;
135 block_index !=
nullptr;
136 block_index =
m_node.
chainman->ActiveChain().Next(*block_index)) {
146 BOOST_CHECK(!filter_index.BlockUntilSyncedToCurrentChain());
155 block_index !=
nullptr;
156 block_index =
m_node.
chainman->ActiveChain().Next(*block_index)) {
171 std::vector<std::shared_ptr<CBlock>> chainA, chainB;
172 BOOST_REQUIRE(BuildChain(tip, coinbase_script_pub_key_A, 10, chainA));
173 BOOST_REQUIRE(BuildChain(tip, coinbase_script_pub_key_B, 10, chainB));
176 uint256 chainA_last_header = last_header;
177 for (
size_t i = 0; i < 2; i++) {
178 const auto& block = chainA[i];
181 for (
size_t i = 0; i < 2; i++) {
182 const auto& block = chainA[i];
186 block_index =
m_node.
chainman->m_blockman.LookupBlockIndex(block->GetHash());
189 BOOST_CHECK(filter_index.BlockUntilSyncedToCurrentChain());
194 uint256 chainB_last_header = last_header;
195 for (
size_t i = 0; i < 3; i++) {
196 const auto& block = chainB[i];
199 for (
size_t i = 0; i < 3; i++) {
200 const auto& block = chainB[i];
204 block_index =
m_node.
chainman->m_blockman.LookupBlockIndex(block->GetHash());
207 BOOST_CHECK(filter_index.BlockUntilSyncedToCurrentChain());
212 chainA_last_header = last_header;
213 for (
size_t i = 0; i < 2; i++) {
214 const auto& block = chainA[i];
218 block_index =
m_node.
chainman->m_blockman.LookupBlockIndex(block->GetHash());
221 BOOST_CHECK(filter_index.BlockUntilSyncedToCurrentChain());
226 for (
size_t i = 2; i < 4; i++) {
227 const auto& block = chainA[i];
232 chainA_last_header = last_header;
233 chainB_last_header = last_header;
234 for (
size_t i = 0; i < 3; i++) {
239 block_index =
m_node.
chainman->m_blockman.LookupBlockIndex(chainA[i]->GetHash());
241 BOOST_CHECK(filter_index.BlockUntilSyncedToCurrentChain());
246 block_index =
m_node.
chainman->m_blockman.LookupBlockIndex(chainB[i]->GetHash());
248 BOOST_CHECK(filter_index.BlockUntilSyncedToCurrentChain());
253 std::vector<BlockFilter> filters;
254 std::vector<uint256> filter_hashes;
268 filter_hashes.clear();
314 std::unique_ptr<BaseIndex::DB>
m_db;
319 explicit IndexReorgCrash(std::unique_ptr<interfaces::Chain> chain, std::shared_future<void> blocker,
324 fs::create_directories(path);
325 m_db = std::make_unique<BaseIndex::DB>(path /
"db", 0,
true,
false);
350 std::promise<void> promise;
351 std::shared_future<void> blocker(promise.get_future());
355 BOOST_REQUIRE(index.
Init());
358 auto func_wait_until = [&](
int height, std::chrono::milliseconds timeout) {
359 auto deadline = std::chrono::steady_clock::now() + timeout;
361 if (std::chrono::steady_clock::now() > deadline) {
365 std::this_thread::sleep_for(100ms);
370 func_wait_until(blocking_height - 1, 5
s);
373 std::vector<std::shared_ptr<CBlock>> fork;
377 for (
const auto& block : fork) {
378 BOOST_REQUIRE(
m_node.
chainman->ProcessNewBlock(block,
true,
true,
nullptr));
384 func_wait_until(blocking_height + 2, 5
s);
CScript GetScriptForDestination(const CTxDestination &dest)
Generate a Bitcoin scriptPubKey for the given CTxDestination.
static bool CheckFilterLookups(BlockFilterIndex &filter_index, const CBlockIndex *block_index, uint256 &last_header, const BlockManager &blockman)
BOOST_FIXTURE_TEST_CASE(blockfilter_index_initial_sync, BuildChainTestingSetup)
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.
void ForEachBlockFilterIndex(std::function< void(BlockFilterIndex &)> fn)
Iterate over all running block filter indexes, invoking fn on each.
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.
#define Assert(val)
Identity function.
fs::path GetDataDirNet() const EXCLUSIVE_LOCKS_REQUIRED(!cs_args)
Get data directory path with appended network identifier.
The database stores a block locator of the chain the database is synced to so that the index can effi...
Base class for indices of blockchain data.
void Stop()
Stops the instance from staying in sync with blockchain updates.
bool Init()
Initializes the sync state and registers the instance to the validation interface so that it stays in...
bool BlockUntilSyncedToCurrentChain() const LOCKS_EXCLUDED(void Interrupt()
Blocks the current thread until the index is caught up to the current state of the block chain.
IndexSummary GetSummary() const
Get a summary of the index and its state.
void Sync()
Sync the index with the block index starting from the current best block.
bool StartBackgroundSync()
Starts the initial sync process on a background thread.
Complete block filter struct as defined in BIP 157.
uint256 ComputeHeader(const uint256 &prev_header) const
Compute the filter header given the previous one.
uint256 GetHash() const
Compute the filter hash.
BlockFilterIndex is used to store and retrieve block filters, hashes, and headers for a range of bloc...
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 LookupFilter(const CBlockIndex *block_index, BlockFilter &filter_out) const
Get a single filter by block.
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 LookupFilterHeader(const CBlockIndex *block_index, uint256 &header_out) EXCLUSIVE_LOCKS_REQUIRED(!m_cs_headers_cache)
Get a single filter header by block.
std::vector< CTransactionRef > vtx
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
An encapsulated private key.
CPubKey GetPubKey() const
Compute the public key from a private key.
Serialized script, used inside transaction inputs and outputs.
bool CustomAppend(const interfaces::BlockInfo &block) override
Write update index entries for a newly connected block.
std::unique_ptr< BaseIndex::DB > m_db
BaseIndex::DB & GetDB() const override
IndexReorgCrash(std::unique_ptr< interfaces::Chain > chain, std::shared_future< void > blocker, int blocking_height)
bool AllowPrune() const override
std::shared_future< void > m_blocker
Generate a new block, without valid proof-of-work.
Maintains a tree of blocks (stored in m_block_index) which is consulted to determine where the most-w...
uint256 BlockMerkleRoot(const CBlock &block, bool *mutated)
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
BOOST_AUTO_TEST_SUITE_END()
CKey GenerateRandomKey(bool compressed) noexcept
BOOST_FAIL("Test unconditionally fails.")
std::unique_ptr< Chain > MakeChain(node::NodeContext &node)
Return implementation of Chain interface.
#define BOOST_CHECK_EQUAL(v1, v2)
#define BOOST_CHECK(expr)
bool CheckProofOfWork(uint256 hash, unsigned int nBits, const Consensus::Params ¶ms)
Check whether a block hash satisfies the proof-of-work requirement specified by nBits.
static CTransactionRef MakeTransactionRef(Tx &&txIn)
static bool GetPubKey(const SigningProvider &provider, const SignatureData &sigdata, const CKeyID &address, CPubKey &pubkey)
bool BuildChain(const CBlockIndex *pindex, const CScript &coinbase_script_pub_key, size_t length, std::vector< std::shared_ptr< CBlock > > &chain)
CBlock CreateBlock(const CBlockIndex *prev, const std::vector< CMutableTransaction > &txns, const CScript &scriptPubKey)
A mutable version of CTransaction.
Testing fixture that pre-creates a 100-block REGTEST-mode block chain.
Block data sent with blockConnected, blockDisconnected notifications.
std::unique_ptr< CTxMemPool > mempool
std::unique_ptr< ChainstateManager > chainman
#define WITH_LOCK(cs, code)
Run code while locking a mutex.
bool ComputeFilter(BlockFilterType filter_type, const CBlockIndex &block_index, BlockFilter &filter, const BlockManager &blockman)
void SetMockTime(int64_t nMockTimeIn)
DEPRECATED Use SetMockTime with chrono type.