/Users/eugenesiegel/btc/bitcoin/src/wallet/transaction.h
Line | Count | Source (jump to first uncovered line) |
1 | | // Copyright (c) 2021-2022 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 | | |
5 | | #ifndef BITCOIN_WALLET_TRANSACTION_H |
6 | | #define BITCOIN_WALLET_TRANSACTION_H |
7 | | |
8 | | #include <attributes.h> |
9 | | #include <consensus/amount.h> |
10 | | #include <primitives/transaction.h> |
11 | | #include <tinyformat.h> |
12 | | #include <uint256.h> |
13 | | #include <util/check.h> |
14 | | #include <util/overloaded.h> |
15 | | #include <util/strencodings.h> |
16 | | #include <util/string.h> |
17 | | #include <wallet/types.h> |
18 | | |
19 | | #include <bitset> |
20 | | #include <cstdint> |
21 | | #include <map> |
22 | | #include <utility> |
23 | | #include <variant> |
24 | | #include <vector> |
25 | | |
26 | | namespace interfaces { |
27 | | class Chain; |
28 | | } // namespace interfaces |
29 | | |
30 | | namespace wallet { |
31 | | //! State of transaction confirmed in a block. |
32 | | struct TxStateConfirmed { |
33 | | uint256 confirmed_block_hash; |
34 | | int confirmed_block_height; |
35 | | int position_in_block; |
36 | | |
37 | 0 | explicit TxStateConfirmed(const uint256& block_hash, int height, int index) : confirmed_block_hash(block_hash), confirmed_block_height(height), position_in_block(index) {} |
38 | 0 | std::string toString() const { return strprintf("Confirmed (block=%s, height=%i, index=%i)", confirmed_block_hash.ToString(), confirmed_block_height, position_in_block); } Line | Count | Source | 1172 | 0 | #define strprintf tfm::format |
|
39 | | }; |
40 | | |
41 | | //! State of transaction added to mempool. |
42 | | struct TxStateInMempool { |
43 | 0 | std::string toString() const { return strprintf("InMempool"); } Line | Count | Source | 1172 | 0 | #define strprintf tfm::format |
|
44 | | }; |
45 | | |
46 | | //! State of rejected transaction that conflicts with a confirmed block. |
47 | | struct TxStateBlockConflicted { |
48 | | uint256 conflicting_block_hash; |
49 | | int conflicting_block_height; |
50 | | |
51 | 0 | explicit TxStateBlockConflicted(const uint256& block_hash, int height) : conflicting_block_hash(block_hash), conflicting_block_height(height) {} |
52 | 0 | std::string toString() const { return strprintf("BlockConflicted (block=%s, height=%i)", conflicting_block_hash.ToString(), conflicting_block_height); } Line | Count | Source | 1172 | 0 | #define strprintf tfm::format |
|
53 | | }; |
54 | | |
55 | | //! State of transaction not confirmed or conflicting with a known block and |
56 | | //! not in the mempool. May conflict with the mempool, or with an unknown block, |
57 | | //! or be abandoned, never broadcast, or rejected from the mempool for another |
58 | | //! reason. |
59 | | struct TxStateInactive { |
60 | | bool abandoned; |
61 | | |
62 | 0 | explicit TxStateInactive(bool abandoned = false) : abandoned(abandoned) {} |
63 | 0 | std::string toString() const { return strprintf("Inactive (abandoned=%i)", abandoned); } Line | Count | Source | 1172 | 0 | #define strprintf tfm::format |
|
64 | | }; |
65 | | |
66 | | //! State of transaction loaded in an unrecognized state with unexpected hash or |
67 | | //! index values. Treated as inactive (with serialized hash and index values |
68 | | //! preserved) by default, but may enter another state if transaction is added |
69 | | //! to the mempool, or confirmed, or abandoned, or found conflicting. |
70 | | struct TxStateUnrecognized { |
71 | | uint256 block_hash; |
72 | | int index; |
73 | | |
74 | 0 | TxStateUnrecognized(const uint256& block_hash, int index) : block_hash(block_hash), index(index) {} |
75 | 0 | std::string toString() const { return strprintf("Unrecognized (block=%s, index=%i)", block_hash.ToString(), index); } Line | Count | Source | 1172 | 0 | #define strprintf tfm::format |
|
76 | | }; |
77 | | |
78 | | //! All possible CWalletTx states |
79 | | using TxState = std::variant<TxStateConfirmed, TxStateInMempool, TxStateBlockConflicted, TxStateInactive, TxStateUnrecognized>; |
80 | | |
81 | | //! Subset of states transaction sync logic is implemented to handle. |
82 | | using SyncTxState = std::variant<TxStateConfirmed, TxStateInMempool, TxStateInactive>; |
83 | | |
84 | | //! Try to interpret deserialized TxStateUnrecognized data as a recognized state. |
85 | | static inline TxState TxStateInterpretSerialized(TxStateUnrecognized data) |
86 | 0 | { |
87 | 0 | if (data.block_hash == uint256::ZERO) { |
88 | 0 | if (data.index == 0) return TxStateInactive{}; |
89 | 0 | } else if (data.block_hash == uint256::ONE) { |
90 | 0 | if (data.index == -1) return TxStateInactive{/*abandoned=*/true}; |
91 | 0 | } else if (data.index >= 0) { |
92 | 0 | return TxStateConfirmed{data.block_hash, /*height=*/-1, data.index}; |
93 | 0 | } else if (data.index == -1) { |
94 | 0 | return TxStateBlockConflicted{data.block_hash, /*height=*/-1}; |
95 | 0 | } |
96 | 0 | return data; |
97 | 0 | } Unexecuted instantiation: fees.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: scriptpubkeyman.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: spend.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: dump.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: feebumper.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: interfaces.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: load.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: receive.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: addresses.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: backup.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: coins.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: encrypt.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: signmessage.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: transactions.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: util.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: wallet.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: transaction.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: walletdb.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE Unexecuted instantiation: init.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE |
98 | | |
99 | | //! Get TxState serialized block hash. Inverse of TxStateInterpretSerialized. |
100 | | static inline uint256 TxStateSerializedBlockHash(const TxState& state) |
101 | 0 | { |
102 | 0 | return std::visit(util::Overloaded{ |
103 | 0 | [](const TxStateInactive& inactive) { return inactive.abandoned ? uint256::ONE : uint256::ZERO; }, Unexecuted instantiation: wallet.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS5_E_clESB_ Unexecuted instantiation: walletdb.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS5_E_clESB_ |
104 | 0 | [](const TxStateInMempool& in_mempool) { return uint256::ZERO; }, Unexecuted instantiation: wallet.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS3_E_clESB_ Unexecuted instantiation: walletdb.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS3_E_clESB_ |
105 | 0 | [](const TxStateConfirmed& confirmed) { return confirmed.confirmed_block_hash; }, Unexecuted instantiation: wallet.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS2_E_clESB_ Unexecuted instantiation: walletdb.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS2_E_clESB_ |
106 | 0 | [](const TxStateBlockConflicted& conflicted) { return conflicted.conflicting_block_hash; }, Unexecuted instantiation: wallet.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS4_E_clESB_ Unexecuted instantiation: walletdb.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS4_E_clESB_ |
107 | 0 | [](const TxStateUnrecognized& unrecognized) { return unrecognized.block_hash; } Unexecuted instantiation: wallet.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS6_E_clESB_ Unexecuted instantiation: walletdb.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS6_E_clESB_ |
108 | 0 | }, state); |
109 | 0 | } Unexecuted instantiation: fees.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: scriptpubkeyman.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: spend.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: dump.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: feebumper.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: interfaces.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: load.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: receive.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: addresses.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: backup.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: coins.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: encrypt.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: signmessage.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: transactions.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: util.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: wallet.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: transaction.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: walletdb.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: init.cpp:_ZN6walletL26TxStateSerializedBlockHashERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE |
110 | | |
111 | | //! Get TxState serialized block index. Inverse of TxStateInterpretSerialized. |
112 | | static inline int TxStateSerializedIndex(const TxState& state) |
113 | 0 | { |
114 | 0 | return std::visit(util::Overloaded{ |
115 | 0 | [](const TxStateInactive& inactive) { return inactive.abandoned ? -1 : 0; }, Unexecuted instantiation: wallet.cpp:_ZZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS5_E_clESB_ Unexecuted instantiation: walletdb.cpp:_ZZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS5_E_clESB_ |
116 | 0 | [](const TxStateInMempool& in_mempool) { return 0; }, Unexecuted instantiation: wallet.cpp:_ZZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS3_E_clESB_ Unexecuted instantiation: walletdb.cpp:_ZZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS3_E_clESB_ |
117 | 0 | [](const TxStateConfirmed& confirmed) { return confirmed.position_in_block; }, Unexecuted instantiation: wallet.cpp:_ZZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS2_E_clESB_ Unexecuted instantiation: walletdb.cpp:_ZZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS2_E_clESB_ |
118 | 0 | [](const TxStateBlockConflicted& conflicted) { return -1; }, Unexecuted instantiation: wallet.cpp:_ZZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS4_E_clESB_ Unexecuted instantiation: walletdb.cpp:_ZZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS4_E_clESB_ |
119 | 0 | [](const TxStateUnrecognized& unrecognized) { return unrecognized.index; } Unexecuted instantiation: wallet.cpp:_ZZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS6_E_clESB_ Unexecuted instantiation: walletdb.cpp:_ZZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENKUlRKS6_E_clESB_ |
120 | 0 | }, state); |
121 | 0 | } Unexecuted instantiation: fees.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: scriptpubkeyman.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: spend.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: dump.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: feebumper.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: interfaces.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: load.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: receive.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: addresses.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: backup.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: coins.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: encrypt.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: signmessage.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: transactions.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: util.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: wallet.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: transaction.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: walletdb.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE Unexecuted instantiation: init.cpp:_ZN6walletL22TxStateSerializedIndexERKNSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEE |
122 | | |
123 | | //! Return TxState or SyncTxState as a string for logging or debugging. |
124 | | template<typename T> |
125 | | std::string TxStateString(const T& state) |
126 | 0 | { |
127 | 0 | return std::visit([](const auto& s) { return s.toString(); }, state); Unexecuted instantiation: _ZZN6wallet13TxStateStringINSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEEENS1_12basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERKT_ENKUlSH_E_clIS3_EEDaSH_ Unexecuted instantiation: _ZZN6wallet13TxStateStringINSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEEENS1_12basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERKT_ENKUlSH_E_clIS4_EEDaSH_ Unexecuted instantiation: _ZZN6wallet13TxStateStringINSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEEENS1_12basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERKT_ENKUlSH_E_clIS5_EEDaSH_ Unexecuted instantiation: _ZZN6wallet13TxStateStringINSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEEENS1_12basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERKT_ENKUlSH_E_clIS6_EEDaSH_ Unexecuted instantiation: _ZZN6wallet13TxStateStringINSt3__17variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEEENS1_12basic_stringIcNS1_11char_traitsIcEENS1_9allocatorIcEEEERKT_ENKUlSH_E_clIS7_EEDaSH_ |
128 | 0 | } |
129 | | |
130 | | /** |
131 | | * Cachable amount subdivided into avoid reuse and all balances |
132 | | */ |
133 | | struct CachableAmount |
134 | | { |
135 | | std::optional<CAmount> m_avoid_reuse_value; |
136 | | std::optional<CAmount> m_all_value; |
137 | | inline void Reset() |
138 | 0 | { |
139 | 0 | m_avoid_reuse_value.reset(); |
140 | 0 | m_all_value.reset(); |
141 | 0 | } |
142 | | void Set(bool avoid_reuse, CAmount value) |
143 | 0 | { |
144 | 0 | if (avoid_reuse) { |
145 | 0 | m_avoid_reuse_value = value; |
146 | 0 | } else { |
147 | 0 | m_all_value = value; |
148 | 0 | } |
149 | 0 | } |
150 | | CAmount Get(bool avoid_reuse) |
151 | 0 | { |
152 | 0 | if (avoid_reuse) { |
153 | 0 | Assert(m_avoid_reuse_value.has_value()); Line | Count | Source | 106 | 0 | #define Assert(val) inline_assertion_check<true>(val, __FILE__, __LINE__, __func__, #val) |
|
154 | 0 | return m_avoid_reuse_value.value(); |
155 | 0 | } |
156 | 0 | Assert(m_all_value.has_value()); Line | Count | Source | 106 | 0 | #define Assert(val) inline_assertion_check<true>(val, __FILE__, __LINE__, __func__, #val) |
|
157 | 0 | return m_all_value.value(); |
158 | 0 | } |
159 | | bool IsCached(bool avoid_reuse) |
160 | 0 | { |
161 | 0 | if (avoid_reuse) return m_avoid_reuse_value.has_value(); |
162 | 0 | return m_all_value.has_value(); |
163 | 0 | } |
164 | | }; |
165 | | |
166 | | |
167 | | typedef std::map<std::string, std::string> mapValue_t; |
168 | | |
169 | | |
170 | | /** Legacy class used for deserializing vtxPrev for backwards compatibility. |
171 | | * vtxPrev was removed in commit 93a18a3650292afbb441a47d1fa1b94aeb0164e3, |
172 | | * but old wallet.dat files may still contain vtxPrev vectors of CMerkleTxs. |
173 | | * These need to get deserialized for field alignment when deserializing |
174 | | * a CWalletTx, but the deserialized values are discarded.**/ |
175 | | class CMerkleTx |
176 | | { |
177 | | public: |
178 | | template<typename Stream> |
179 | | void Unserialize(Stream& s) |
180 | 0 | { |
181 | 0 | CTransactionRef tx; |
182 | 0 | uint256 hashBlock; |
183 | 0 | std::vector<uint256> vMerkleBranch; |
184 | 0 | int nIndex; |
185 | |
|
186 | 0 | s >> TX_WITH_WITNESS(tx) >> hashBlock >> vMerkleBranch >> nIndex; |
187 | 0 | } |
188 | | }; |
189 | | |
190 | | /** |
191 | | * A transaction with a bunch of additional info that only the owner cares about. |
192 | | * It includes any unrecorded transactions needed to link it back to the block chain. |
193 | | */ |
194 | | class CWalletTx |
195 | | { |
196 | | public: |
197 | | /** |
198 | | * Key/value map with information about the transaction. |
199 | | * |
200 | | * The following keys can be read and written through the map and are |
201 | | * serialized in the wallet database: |
202 | | * |
203 | | * "comment", "to" - comment strings provided to sendtoaddress, |
204 | | * and sendmany wallet RPCs |
205 | | * "replaces_txid" - txid (as HexStr) of transaction replaced by |
206 | | * bumpfee on transaction created by bumpfee |
207 | | * "replaced_by_txid" - txid (as HexStr) of transaction created by |
208 | | * bumpfee on transaction replaced by bumpfee |
209 | | * "from", "message" - obsolete fields that could be set in UI prior to |
210 | | * 2011 (removed in commit 4d9b223) |
211 | | * |
212 | | * The following keys are serialized in the wallet database, but shouldn't |
213 | | * be read or written through the map (they will be temporarily added and |
214 | | * removed from the map during serialization): |
215 | | * |
216 | | * "fromaccount" - serialized strFromAccount value |
217 | | * "n" - serialized nOrderPos value |
218 | | * "timesmart" - serialized nTimeSmart value |
219 | | * "spent" - serialized vfSpent value that existed prior to |
220 | | * 2014 (removed in commit 93a18a3) |
221 | | */ |
222 | | mapValue_t mapValue; |
223 | | std::vector<std::pair<std::string, std::string> > vOrderForm; |
224 | | unsigned int nTimeReceived; //!< time received by this node |
225 | | /** |
226 | | * Stable timestamp that never changes, and reflects the order a transaction |
227 | | * was added to the wallet. Timestamp is based on the block time for a |
228 | | * transaction added as part of a block, or else the time when the |
229 | | * transaction was received if it wasn't part of a block, with the timestamp |
230 | | * adjusted in both cases so timestamp order matches the order transactions |
231 | | * were added to the wallet. More details can be found in |
232 | | * CWallet::ComputeTimeSmart(). |
233 | | */ |
234 | | unsigned int nTimeSmart; |
235 | | int64_t nOrderPos; //!< position in ordered transaction list |
236 | | std::multimap<int64_t, CWalletTx*>::const_iterator m_it_wtxOrdered; |
237 | | |
238 | | // memory only |
239 | | enum AmountType { DEBIT, CREDIT, AMOUNTTYPE_ENUM_ELEMENTS }; |
240 | | mutable CachableAmount m_amounts[AMOUNTTYPE_ENUM_ELEMENTS]; |
241 | | /** |
242 | | * This flag is true if all m_amounts caches are empty. This is particularly |
243 | | * useful in places where MarkDirty is conditionally called and the |
244 | | * condition can be expensive and thus can be skipped if the flag is true. |
245 | | * See MarkDestinationsDirty. |
246 | | */ |
247 | | mutable bool m_is_cache_empty{true}; |
248 | | mutable bool fChangeCached; |
249 | | mutable CAmount nChangeCached; |
250 | | |
251 | 0 | CWalletTx(CTransactionRef tx, const TxState& state) : tx(std::move(tx)), m_state(state) |
252 | 0 | { |
253 | 0 | Init(); |
254 | 0 | } |
255 | | |
256 | | void Init() |
257 | 0 | { |
258 | 0 | mapValue.clear(); |
259 | 0 | vOrderForm.clear(); |
260 | 0 | nTimeReceived = 0; |
261 | 0 | nTimeSmart = 0; |
262 | 0 | fChangeCached = false; |
263 | 0 | nChangeCached = 0; |
264 | 0 | nOrderPos = -1; |
265 | 0 | } |
266 | | |
267 | | CTransactionRef tx; |
268 | | TxState m_state; |
269 | | |
270 | | // Set of mempool transactions that conflict |
271 | | // directly with the transaction, or that conflict |
272 | | // with an ancestor transaction. This set will be |
273 | | // empty if state is InMempool or Confirmed, but |
274 | | // can be nonempty if state is Inactive or |
275 | | // BlockConflicted. |
276 | | std::set<Txid> mempool_conflicts; |
277 | | |
278 | | // Track v3 mempool tx that spends from this tx |
279 | | // so that we don't try to create another unconfirmed child |
280 | | std::optional<Txid> truc_child_in_mempool; |
281 | | |
282 | | template<typename Stream> |
283 | | void Serialize(Stream& s) const |
284 | 0 | { |
285 | 0 | mapValue_t mapValueCopy = mapValue; |
286 | |
|
287 | 0 | mapValueCopy["fromaccount"] = ""; |
288 | 0 | if (nOrderPos != -1) { |
289 | 0 | mapValueCopy["n"] = util::ToString(nOrderPos); |
290 | 0 | } |
291 | 0 | if (nTimeSmart) { |
292 | 0 | mapValueCopy["timesmart"] = strprintf("%u", nTimeSmart); Line | Count | Source | 1172 | 0 | #define strprintf tfm::format |
|
293 | 0 | } |
294 | |
|
295 | 0 | std::vector<uint8_t> dummy_vector1; //!< Used to be vMerkleBranch |
296 | 0 | std::vector<uint8_t> dummy_vector2; //!< Used to be vtxPrev |
297 | 0 | bool dummy_bool = false; //!< Used to be fFromMe, and fSpent |
298 | 0 | uint32_t dummy_int = 0; // Used to be fTimeReceivedIsTxTime |
299 | 0 | uint256 serializedHash = TxStateSerializedBlockHash(m_state); |
300 | 0 | int serializedIndex = TxStateSerializedIndex(m_state); |
301 | 0 | s << TX_WITH_WITNESS(tx) << serializedHash << dummy_vector1 << serializedIndex << dummy_vector2 << mapValueCopy << vOrderForm << dummy_int << nTimeReceived << dummy_bool << dummy_bool; |
302 | 0 | } |
303 | | |
304 | | template<typename Stream> |
305 | | void Unserialize(Stream& s) |
306 | 0 | { |
307 | 0 | Init(); |
308 | |
|
309 | 0 | std::vector<uint256> dummy_vector1; //!< Used to be vMerkleBranch |
310 | 0 | std::vector<CMerkleTx> dummy_vector2; //!< Used to be vtxPrev |
311 | 0 | bool dummy_bool; //! Used to be fFromMe, and fSpent |
312 | 0 | uint32_t dummy_int; // Used to be fTimeReceivedIsTxTime |
313 | 0 | uint256 serialized_block_hash; |
314 | 0 | int serializedIndex; |
315 | 0 | s >> TX_WITH_WITNESS(tx) >> serialized_block_hash >> dummy_vector1 >> serializedIndex >> dummy_vector2 >> mapValue >> vOrderForm >> dummy_int >> nTimeReceived >> dummy_bool >> dummy_bool; |
316 | |
|
317 | 0 | m_state = TxStateInterpretSerialized({serialized_block_hash, serializedIndex}); |
318 | |
|
319 | 0 | const auto it_op = mapValue.find("n"); |
320 | 0 | nOrderPos = (it_op != mapValue.end()) ? LocaleIndependentAtoi<int64_t>(it_op->second) : -1; |
321 | 0 | const auto it_ts = mapValue.find("timesmart"); |
322 | 0 | nTimeSmart = (it_ts != mapValue.end()) ? static_cast<unsigned int>(LocaleIndependentAtoi<int64_t>(it_ts->second)) : 0; |
323 | |
|
324 | 0 | mapValue.erase("fromaccount"); |
325 | 0 | mapValue.erase("spent"); |
326 | 0 | mapValue.erase("n"); |
327 | 0 | mapValue.erase("timesmart"); |
328 | 0 | } |
329 | | |
330 | | void SetTx(CTransactionRef arg) |
331 | 0 | { |
332 | 0 | tx = std::move(arg); |
333 | 0 | } |
334 | | |
335 | | //! make sure balances are recalculated |
336 | | void MarkDirty() |
337 | 0 | { |
338 | 0 | m_amounts[DEBIT].Reset(); |
339 | 0 | m_amounts[CREDIT].Reset(); |
340 | 0 | fChangeCached = false; |
341 | 0 | m_is_cache_empty = true; |
342 | 0 | } |
343 | | |
344 | | /** True if only scriptSigs are different */ |
345 | | bool IsEquivalentTo(const CWalletTx& tx) const; |
346 | | |
347 | | bool InMempool() const; |
348 | | |
349 | | int64_t GetTxTime() const; |
350 | | |
351 | 0 | template<typename T> const T* state() const { return std::get_if<T>(&m_state); } Unexecuted instantiation: _ZNK6wallet9CWalletTx5stateINS_15TxStateInactiveEEEPKT_v Unexecuted instantiation: _ZNK6wallet9CWalletTx5stateINS_22TxStateBlockConflictedEEEPKT_v Unexecuted instantiation: _ZNK6wallet9CWalletTx5stateINS_16TxStateConfirmedEEEPKT_v Unexecuted instantiation: _ZNK6wallet9CWalletTx5stateINS_16TxStateInMempoolEEEPKT_v |
352 | 0 | template<typename T> T* state() { return std::get_if<T>(&m_state); } Unexecuted instantiation: _ZN6wallet9CWalletTx5stateINS_16TxStateConfirmedEEEPT_v Unexecuted instantiation: _ZN6wallet9CWalletTx5stateINS_22TxStateBlockConflictedEEEPT_v Unexecuted instantiation: _ZN6wallet9CWalletTx5stateINS_16TxStateInMempoolEEEPT_v |
353 | | |
354 | | //! Update transaction state when attaching to a chain, filling in heights |
355 | | //! of conflicted and confirmed blocks |
356 | | void updateState(interfaces::Chain& chain); |
357 | | |
358 | 0 | bool isAbandoned() const { return state<TxStateInactive>() && state<TxStateInactive>()->abandoned; } |
359 | 0 | bool isMempoolConflicted() const { return !mempool_conflicts.empty(); } |
360 | 0 | bool isBlockConflicted() const { return state<TxStateBlockConflicted>(); } |
361 | 0 | bool isInactive() const { return state<TxStateInactive>(); } |
362 | 0 | bool isUnconfirmed() const { return !isAbandoned() && !isBlockConflicted() && !isMempoolConflicted() && !isConfirmed(); } |
363 | 0 | bool isConfirmed() const { return state<TxStateConfirmed>(); } |
364 | 0 | const Txid& GetHash() const LIFETIMEBOUND { return tx->GetHash(); } |
365 | 0 | const Wtxid& GetWitnessHash() const LIFETIMEBOUND { return tx->GetWitnessHash(); } |
366 | 0 | bool IsCoinBase() const { return tx->IsCoinBase(); } |
367 | | |
368 | | private: |
369 | | // Disable copying of CWalletTx objects to prevent bugs where instances get |
370 | | // copied in and out of the mapWallet map, and fields are updated in the |
371 | | // wrong copy. |
372 | | CWalletTx(const CWalletTx&) = default; |
373 | 0 | CWalletTx& operator=(const CWalletTx&) = default; |
374 | | public: |
375 | | // Instead have an explicit copy function |
376 | | void CopyFrom(const CWalletTx&); |
377 | | }; |
378 | | |
379 | | struct WalletTxOrderComparator { |
380 | | bool operator()(const CWalletTx* a, const CWalletTx* b) const |
381 | 0 | { |
382 | 0 | return a->nOrderPos < b->nOrderPos; |
383 | 0 | } |
384 | | }; |
385 | | |
386 | | class WalletTXO |
387 | | { |
388 | | private: |
389 | | const CWalletTx& m_wtx; |
390 | | const CTxOut& m_output; |
391 | | |
392 | | public: |
393 | | WalletTXO(const CWalletTx& wtx, const CTxOut& output) |
394 | 0 | : m_wtx(wtx), |
395 | 0 | m_output(output) |
396 | 0 | { |
397 | 0 | Assume(std::ranges::find(wtx.tx->vout, output) != wtx.tx->vout.end()); Line | Count | Source | 118 | 0 | #define Assume(val) inline_assertion_check<false>(val, __FILE__, __LINE__, __func__, #val) |
|
398 | 0 | } |
399 | | |
400 | 0 | const CWalletTx& GetWalletTx() const { return m_wtx; } |
401 | | |
402 | 0 | const CTxOut& GetTxOut() const { return m_output; } |
403 | | }; |
404 | | } // namespace wallet |
405 | | |
406 | | #endif // BITCOIN_WALLET_TRANSACTION_H |