Bitcoin Core 31.99.0
P2P Digital Currency
txospenderindex.cpp
Go to the documentation of this file.
1// Copyright (c) The Bitcoin Core developers
2// Distributed under the MIT software license, see the accompanying
3// file COPYING or http://www.opensource.org/licenses/mit-license.php.
4
6
7#include <common/args.h>
8#include <crypto/siphash.h>
9#include <dbwrapper.h>
10#include <flatfile.h>
11#include <index/base.h>
12#include <index/disktxpos.h>
13#include <interfaces/chain.h>
14#include <logging.h>
15#include <node/blockstorage.h>
16#include <primitives/block.h>
18#include <random.h>
19#include <serialize.h>
20#include <streams.h>
21#include <tinyformat.h>
22#include <uint256.h>
23#include <util/fs.h>
24#include <validation.h>
25
26#include <cstdio>
27#include <exception>
28#include <ios>
29#include <string>
30#include <utility>
31#include <vector>
32
33/* The database is used to find the spending transaction of a given utxo.
34 * For every input of every transaction it stores a key that is a pair(siphash(input outpoint), transaction location on disk) and an empty value.
35 * To find the spending transaction of an outpoint, we perform a range query on siphash(outpoint), and for each returned key load the transaction
36 * and return it if it does spend the provided outpoint.
37 */
38
39// LevelDB key prefix. We only have one key for now but it will make it easier to add others if needed.
40constexpr uint8_t DB_TXOSPENDERINDEX{'s'};
41
42std::unique_ptr<TxoSpenderIndex> g_txospenderindex;
43
44struct DBKey {
45 uint64_t hash;
47
48 explicit DBKey(const uint64_t& hash_in, const CDiskTxPos& pos_in) : hash(hash_in), pos(pos_in) {}
49
51 {
55 throw std::ios_base::failure("Invalid format for spender index DB key");
56 }
57 READWRITE(obj.hash);
58 READWRITE(obj.pos);
59 }
60};
61
62TxoSpenderIndex::TxoSpenderIndex(std::unique_ptr<interfaces::Chain> chain, size_t n_cache_size, bool f_memory, bool f_wipe)
63 : BaseIndex(std::move(chain), "txospenderindex"), m_db{std::make_unique<DB>(gArgs.GetDataDirNet() / "indexes" / "txospenderindex" / "db", n_cache_size, f_memory, f_wipe)}
64{
65 if (!m_db->Read("siphash_key", m_siphash_key)) {
66 FastRandomContext rng(false);
67 m_siphash_key = {rng.rand64(), rng.rand64()};
68 m_db->Write("siphash_key", m_siphash_key, /*fSync=*/ true);
69 }
70}
71
73{
75 options.disconnect_data = true;
76 return options;
77}
78
79static uint64_t CreateKeyPrefix(std::pair<uint64_t, uint64_t> siphash_key, const COutPoint& vout)
80{
81 return PresaltedSipHasher(siphash_key.first, siphash_key.second)(vout.hash.ToUint256(), vout.n);
82}
83
84static DBKey CreateKey(std::pair<uint64_t, uint64_t> siphash_key, const COutPoint& vout, const CDiskTxPos& pos)
85{
86 return DBKey(CreateKeyPrefix(siphash_key, vout), pos);
87}
88
89void TxoSpenderIndex::WriteSpenderInfos(const std::vector<std::pair<COutPoint, CDiskTxPos>>& items)
90{
91 CDBBatch batch(*m_db);
92 for (const auto& [outpoint, pos] : items) {
93 DBKey key(CreateKey(m_siphash_key, outpoint, pos));
94 // key is hash(spent outpoint) | disk pos, value is empty
95 batch.Write(key, "");
96 }
97 m_db->WriteBatch(batch);
98}
99
100
101void TxoSpenderIndex::EraseSpenderInfos(const std::vector<std::pair<COutPoint, CDiskTxPos>>& items)
102{
103 CDBBatch batch(*m_db);
104 for (const auto& [outpoint, pos] : items) {
105 batch.Erase(CreateKey(m_siphash_key, outpoint, pos));
106 }
107 m_db->WriteBatch(batch);
108}
109
110static std::vector<std::pair<COutPoint, CDiskTxPos>> BuildSpenderPositions(const interfaces::BlockInfo& block)
111{
112 std::vector<std::pair<COutPoint, CDiskTxPos>> items;
113 items.reserve(block.data->vtx.size());
114
115 CDiskTxPos pos({block.file_number, block.data_pos}, GetSizeOfCompactSize(block.data->vtx.size()));
116 for (const auto& tx : block.data->vtx) {
117 if (!tx->IsCoinBase()) {
118 for (const auto& input : tx->vin) {
119 items.emplace_back(input.prevout, pos);
120 }
121 }
123 }
124
125 return items;
126}
127
128
130{
132 return true;
133}
134
136{
138 return true;
139}
140
142{
143 AutoFile file{m_chainstate->m_blockman.OpenBlockFile(tx_pos, /*fReadOnly=*/true)};
144 if (file.IsNull()) {
145 return util::Unexpected("cannot open block");
146 }
147 CBlockHeader header;
148 TxoSpender spender;
149 try {
150 file >> header;
151 file.seek(tx_pos.nTxOffset, SEEK_CUR);
152 file >> TX_WITH_WITNESS(spender.tx);
153 spender.block_hash = header.GetHash();
154 return spender;
155 } catch (const std::exception& e) {
156 return util::Unexpected(e.what());
157 }
158}
159
161{
162 const uint64_t prefix{CreateKeyPrefix(m_siphash_key, txo)};
163 std::unique_ptr<CDBIterator> it(m_db->NewIterator());
164 DBKey key(prefix, CDiskTxPos());
165
166 // find all keys that start with the outpoint hash, load the transaction at the location specified in the key
167 // and return it if it does spend the provided outpoint
168 for (it->Seek(std::pair{DB_TXOSPENDERINDEX, prefix}); it->Valid() && it->GetKey(key) && key.hash == prefix; it->Next()) {
169 if (const auto spender{ReadTransaction(key.pos)}) {
170 for (const auto& input : spender->tx->vin) {
171 if (input.prevout == txo) {
172 return std::optional{*spender};
173 }
174 }
175 } else {
176 LogError("Deserialize or I/O error - %s", spender.error());
177 return util::Unexpected{strprintf("IO error finding spending tx for outpoint %s:%d.", txo.hash.GetHex(), txo.n)};
178 }
179 }
180 return util::Expected<std::optional<TxoSpender>, std::string>(std::nullopt);
181}
182
ArgsManager gArgs
Definition: args.cpp:40
Non-refcounted RAII wrapper for FILE*.
Definition: streams.h:373
The database stores a block locator of the chain the database is synced to so that the index can effi...
Definition: base.h:65
Base class for indices of blockchain data.
Definition: base.h:55
Chainstate * m_chainstate
Definition: base.h:118
Nodes collect new transactions into a block, hash them into a hash tree, and scan through nonce value...
Definition: block.h:27
std::vector< CTransactionRef > vtx
Definition: block.h:77
Batch of changes queued to be written to a CDBWrapper.
Definition: dbwrapper.h:72
void Erase(const K &key)
Definition: dbwrapper.h:108
void Write(const K &key, const V &value)
Definition: dbwrapper.h:96
An outpoint - a combination of a transaction hash and an index n into its vout.
Definition: transaction.h:29
uint32_t n
Definition: transaction.h:32
Txid hash
Definition: transaction.h:31
node::BlockManager & m_blockman
Reference to a BlockManager instance which itself is shared across all Chainstate instances.
Definition: validation.h:578
Fast randomness source.
Definition: random.h:386
uint64_t rand64() noexcept
Generate a random 64-bit integer.
Definition: random.h:404
Optimized SipHash-2-4 implementation for uint256.
Definition: siphash.h:56
void EraseSpenderInfos(const std::vector< std::pair< COutPoint, CDiskTxPos > > &items)
std::pair< uint64_t, uint64_t > m_siphash_key
TxoSpenderIndex(std::unique_ptr< interfaces::Chain > chain, size_t n_cache_size, bool f_memory=false, bool f_wipe=false)
std::unique_ptr< BaseIndex::DB > m_db
bool CustomRemove(const interfaces::BlockInfo &block) override
Rewind index by one block during a chain reorg.
util::Expected< TxoSpender, std::string > ReadTransaction(const CDiskTxPos &pos) const
util::Expected< std::optional< TxoSpender >, std::string > FindSpender(const COutPoint &txo) const
Search the index for a transaction that spends the given outpoint.
interfaces::Chain::NotifyOptions CustomOptions() override
Return custom notification options for index.
void WriteSpenderInfos(const std::vector< std::pair< COutPoint, CDiskTxPos > > &items)
BaseIndex::DB & GetDB() const override
bool CustomAppend(const interfaces::BlockInfo &block) override
Write update index entries for a newly connected block.
bool IsBlockPruned(const CBlockIndex &block) const EXCLUSIVE_LOCKS_REQUIRED(void UpdatePruneLock(const std::string &name, const PruneLockInfo &lock_info) EXCLUSIVE_LOCKS_REQUIRED(AutoFile OpenBlockFile(const FlatFilePos &pos, bool fReadOnly) const
Check whether the block associated with this index entry is pruned or not.
Definition: blockstorage.h:459
const uint256 & ToUint256() const LIFETIMEBOUND
std::string GetHex() const
The util::Expected class provides a standard way for low-level functions to return either error value...
Definition: expected.h:45
The util::Unexpected class represents an unexpected value stored in util::Expected.
Definition: expected.h:22
#define LogError(...)
Definition: log.h:99
Definition: common.h:29
static constexpr TransactionSerParams TX_WITH_WITNESS
Definition: transaction.h:180
const char * prefix
Definition: rest.cpp:1142
constexpr unsigned int GetSizeOfCompactSize(uint64_t nSize)
Compact Size size < 253 – 1 byte size <= USHRT_MAX – 3 bytes (253 + 2 bytes) size <= UINT_MAX – 5 byt...
Definition: serialize.h:290
uint64_t GetSerializeSize(const T &t)
Definition: serialize.h:1097
#define READWRITE(...)
Definition: serialize.h:147
uint32_t nTxOffset
Definition: disktxpos.h:13
CDiskTxPos pos
uint64_t hash
SERIALIZE_METHODS(DBKey, obj)
DBKey(const uint64_t &hash_in, const CDiskTxPos &pos_in)
uint256 block_hash
CTransactionRef tx
Block data sent with blockConnected, blockDisconnected notifications.
Definition: chain.h:19
unsigned data_pos
Definition: chain.h:24
const CBlock * data
Definition: chain.h:25
Options specifying which chain notifications are required.
Definition: chain.h:319
bool disconnect_data
Include block data with block disconnected notifications.
Definition: chain.h:323
#define strprintf
Format arguments and return the string or write to given std::ostream (see tinyformat::format doc for...
Definition: tinyformat.h:1172
constexpr uint8_t DB_TXOSPENDERINDEX
static DBKey CreateKey(std::pair< uint64_t, uint64_t > siphash_key, const COutPoint &vout, const CDiskTxPos &pos)
std::unique_ptr< TxoSpenderIndex > g_txospenderindex
The global txo spender index. May be null.
static std::vector< std::pair< COutPoint, CDiskTxPos > > BuildSpenderPositions(const interfaces::BlockInfo &block)
static uint64_t CreateKeyPrefix(std::pair< uint64_t, uint64_t > siphash_key, const COutPoint &vout)