Coverage Report

Created: 2025-09-19 18:31

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/root/bitcoin/src/wallet/transaction.h
Line
Count
Source
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); }
39
};
40
41
//! State of transaction added to mempool.
42
struct TxStateInMempool {
43
0
    std::string toString() const { return strprintf("InMempool"); }
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); }
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); }
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); }
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: transaction.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE
Unexecuted instantiation: wallet.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE
Unexecuted instantiation: walletdb.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE
Unexecuted instantiation: interfaces.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE
Unexecuted instantiation: load.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE
Unexecuted instantiation: receive.cpp:_ZN6walletL26TxStateInterpretSerializedENS_19TxStateUnrecognizedE
Unexecuted instantiation: feebumper.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: 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:_ZZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS4_E_clESA_
Unexecuted instantiation: walletdb.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS4_E_clESA_
104
0
        [](const TxStateInMempool& in_mempool) { return uint256::ZERO; },
Unexecuted instantiation: wallet.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS2_E_clESA_
Unexecuted instantiation: walletdb.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS2_E_clESA_
105
0
        [](const TxStateConfirmed& confirmed) { return confirmed.confirmed_block_hash; },
Unexecuted instantiation: wallet.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS1_E_clESA_
Unexecuted instantiation: walletdb.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS1_E_clESA_
106
0
        [](const TxStateBlockConflicted& conflicted) { return conflicted.conflicting_block_hash; },
Unexecuted instantiation: wallet.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS3_E_clESA_
Unexecuted instantiation: walletdb.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS3_E_clESA_
107
0
        [](const TxStateUnrecognized& unrecognized) { return unrecognized.block_hash; }
Unexecuted instantiation: wallet.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS5_E_clESA_
Unexecuted instantiation: walletdb.cpp:_ZZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS5_E_clESA_
108
0
    }, state);
109
0
}
Unexecuted instantiation: fees.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: scriptpubkeyman.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: spend.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: dump.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: transaction.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: wallet.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: walletdb.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: interfaces.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: load.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: receive.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: feebumper.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: addresses.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: backup.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: coins.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: encrypt.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: signmessage.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: transactions.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: util.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: init.cpp:_ZN6walletL26TxStateSerializedBlockHashERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
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:_ZZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS4_E_clESA_
Unexecuted instantiation: walletdb.cpp:_ZZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS4_E_clESA_
116
0
        [](const TxStateInMempool& in_mempool) { return 0; },
Unexecuted instantiation: wallet.cpp:_ZZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS2_E_clESA_
Unexecuted instantiation: walletdb.cpp:_ZZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS2_E_clESA_
117
0
        [](const TxStateConfirmed& confirmed) { return confirmed.position_in_block; },
Unexecuted instantiation: wallet.cpp:_ZZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS1_E_clESA_
Unexecuted instantiation: walletdb.cpp:_ZZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS1_E_clESA_
118
0
        [](const TxStateBlockConflicted& conflicted) { return -1; },
Unexecuted instantiation: wallet.cpp:_ZZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS3_E_clESA_
Unexecuted instantiation: walletdb.cpp:_ZZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS3_E_clESA_
119
0
        [](const TxStateUnrecognized& unrecognized) { return unrecognized.index; }
Unexecuted instantiation: wallet.cpp:_ZZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS5_E_clESA_
Unexecuted instantiation: walletdb.cpp:_ZZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEENKUlRKS5_E_clESA_
120
0
    }, state);
121
0
}
Unexecuted instantiation: fees.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: scriptpubkeyman.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: spend.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: dump.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: transaction.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: wallet.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: walletdb.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: interfaces.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: load.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: receive.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: feebumper.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: addresses.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: backup.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: coins.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: encrypt.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: signmessage.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: transactions.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: util.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
Unexecuted instantiation: init.cpp:_ZN6walletL22TxStateSerializedIndexERKSt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEE
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: _ZZN6wallet13TxStateStringISt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEERKT_ENKUlSG_E_clIS2_EEDaSG_
Unexecuted instantiation: _ZZN6wallet13TxStateStringISt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEERKT_ENKUlSG_E_clIS3_EEDaSG_
Unexecuted instantiation: _ZZN6wallet13TxStateStringISt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEERKT_ENKUlSG_E_clIS4_EEDaSG_
Unexecuted instantiation: _ZZN6wallet13TxStateStringISt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEERKT_ENKUlSG_E_clIS5_EEDaSG_
Unexecuted instantiation: _ZZN6wallet13TxStateStringISt7variantIJNS_16TxStateConfirmedENS_16TxStateInMempoolENS_22TxStateBlockConflictedENS_15TxStateInactiveENS_19TxStateUnrecognizedEEEEENSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEERKT_ENKUlSG_E_clIS6_EEDaSG_
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());
154
0
            return m_avoid_reuse_value.value();
155
0
        }
156
0
        Assert(m_all_value.has_value());
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
    // Cached value for whether the transaction spends any inputs known to the wallet
236
    mutable std::optional<bool> m_cached_from_me{std::nullopt};
237
    int64_t nOrderPos; //!< position in ordered transaction list
238
    std::multimap<int64_t, CWalletTx*>::const_iterator m_it_wtxOrdered;
239
240
    // memory only
241
    enum AmountType { DEBIT, CREDIT, AMOUNTTYPE_ENUM_ELEMENTS };
242
    mutable CachableAmount m_amounts[AMOUNTTYPE_ENUM_ELEMENTS];
243
    /**
244
     * This flag is true if all m_amounts caches are empty. This is particularly
245
     * useful in places where MarkDirty is conditionally called and the
246
     * condition can be expensive and thus can be skipped if the flag is true.
247
     * See MarkDestinationsDirty.
248
     */
249
    mutable bool m_is_cache_empty{true};
250
    mutable bool fChangeCached;
251
    mutable CAmount nChangeCached;
252
253
0
    CWalletTx(CTransactionRef tx, const TxState& state) : tx(std::move(tx)), m_state(state)
254
0
    {
255
0
        Init();
256
0
    }
257
258
    void Init()
259
0
    {
260
0
        mapValue.clear();
261
0
        vOrderForm.clear();
262
0
        nTimeReceived = 0;
263
0
        nTimeSmart = 0;
264
0
        fChangeCached = false;
265
0
        nChangeCached = 0;
266
0
        nOrderPos = -1;
267
0
    }
268
269
    CTransactionRef tx;
270
    TxState m_state;
271
272
    // Set of mempool transactions that conflict
273
    // directly with the transaction, or that conflict
274
    // with an ancestor transaction. This set will be
275
    // empty if state is InMempool or Confirmed, but
276
    // can be nonempty if state is Inactive or
277
    // BlockConflicted.
278
    std::set<Txid> mempool_conflicts;
279
280
    // Track v3 mempool tx that spends from this tx
281
    // so that we don't try to create another unconfirmed child
282
    std::optional<Txid> truc_child_in_mempool;
283
284
    template<typename Stream>
285
    void Serialize(Stream& s) const
286
0
    {
287
0
        mapValue_t mapValueCopy = mapValue;
288
289
0
        mapValueCopy["fromaccount"] = "";
290
0
        if (nOrderPos != -1) {
291
0
            mapValueCopy["n"] = util::ToString(nOrderPos);
292
0
        }
293
0
        if (nTimeSmart) {
294
0
            mapValueCopy["timesmart"] = strprintf("%u", nTimeSmart);
295
0
        }
296
297
0
        std::vector<uint8_t> dummy_vector1; //!< Used to be vMerkleBranch
298
0
        std::vector<uint8_t> dummy_vector2; //!< Used to be vtxPrev
299
0
        bool dummy_bool = false; //!< Used to be fFromMe, and fSpent
300
0
        uint32_t dummy_int = 0; // Used to be fTimeReceivedIsTxTime
301
0
        uint256 serializedHash = TxStateSerializedBlockHash(m_state);
302
0
        int serializedIndex = TxStateSerializedIndex(m_state);
303
0
        s << TX_WITH_WITNESS(tx) << serializedHash << dummy_vector1 << serializedIndex << dummy_vector2 << mapValueCopy << vOrderForm << dummy_int << nTimeReceived << dummy_bool << dummy_bool;
304
0
    }
305
306
    template<typename Stream>
307
    void Unserialize(Stream& s)
308
0
    {
309
0
        Init();
310
311
0
        std::vector<uint256> dummy_vector1; //!< Used to be vMerkleBranch
312
0
        std::vector<CMerkleTx> dummy_vector2; //!< Used to be vtxPrev
313
0
        bool dummy_bool; //! Used to be fFromMe, and fSpent
314
0
        uint32_t dummy_int; // Used to be fTimeReceivedIsTxTime
315
0
        uint256 serialized_block_hash;
316
0
        int serializedIndex;
317
0
        s >> TX_WITH_WITNESS(tx) >> serialized_block_hash >> dummy_vector1 >> serializedIndex >> dummy_vector2 >> mapValue >> vOrderForm >> dummy_int >> nTimeReceived >> dummy_bool >> dummy_bool;
318
319
0
        m_state = TxStateInterpretSerialized({serialized_block_hash, serializedIndex});
320
321
0
        const auto it_op = mapValue.find("n");
322
0
        nOrderPos = (it_op != mapValue.end()) ? LocaleIndependentAtoi<int64_t>(it_op->second) : -1;
323
0
        const auto it_ts = mapValue.find("timesmart");
324
0
        nTimeSmart = (it_ts != mapValue.end()) ? static_cast<unsigned int>(LocaleIndependentAtoi<int64_t>(it_ts->second)) : 0;
325
326
0
        mapValue.erase("fromaccount");
327
0
        mapValue.erase("spent");
328
0
        mapValue.erase("n");
329
0
        mapValue.erase("timesmart");
330
0
    }
331
332
    void SetTx(CTransactionRef arg)
333
0
    {
334
0
        tx = std::move(arg);
335
0
    }
336
337
    //! make sure balances are recalculated
338
    void MarkDirty()
339
0
    {
340
0
        m_amounts[DEBIT].Reset();
341
0
        m_amounts[CREDIT].Reset();
342
0
        fChangeCached = false;
343
0
        m_is_cache_empty = true;
344
0
        m_cached_from_me = std::nullopt;
345
0
    }
346
347
    /** True if only scriptSigs are different */
348
    bool IsEquivalentTo(const CWalletTx& tx) const;
349
350
    bool InMempool() const;
351
352
    int64_t GetTxTime() const;
353
354
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
355
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
356
357
    //! Update transaction state when attaching to a chain, filling in heights
358
    //! of conflicted and confirmed blocks
359
    void updateState(interfaces::Chain& chain);
360
361
0
    bool isAbandoned() const { return state<TxStateInactive>() && state<TxStateInactive>()->abandoned; }
362
0
    bool isMempoolConflicted() const { return !mempool_conflicts.empty(); }
363
0
    bool isBlockConflicted() const { return state<TxStateBlockConflicted>(); }
364
0
    bool isInactive() const { return state<TxStateInactive>(); }
365
0
    bool isUnconfirmed() const { return !isAbandoned() && !isBlockConflicted() && !isMempoolConflicted() && !isConfirmed(); }
366
0
    bool isConfirmed() const { return state<TxStateConfirmed>(); }
367
0
    const Txid& GetHash() const LIFETIMEBOUND { return tx->GetHash(); }
368
0
    const Wtxid& GetWitnessHash() const LIFETIMEBOUND { return tx->GetWitnessHash(); }
369
0
    bool IsCoinBase() const { return tx->IsCoinBase(); }
370
371
private:
372
    // Disable copying of CWalletTx objects to prevent bugs where instances get
373
    // copied in and out of the mapWallet map, and fields are updated in the
374
    // wrong copy.
375
    CWalletTx(const CWalletTx&) = default;
376
0
    CWalletTx& operator=(const CWalletTx&) = default;
377
public:
378
    // Instead have an explicit copy function
379
    void CopyFrom(const CWalletTx&);
380
};
381
382
struct WalletTxOrderComparator {
383
    bool operator()(const CWalletTx* a, const CWalletTx* b) const
384
0
    {
385
0
        return a->nOrderPos < b->nOrderPos;
386
0
    }
387
};
388
389
class WalletTXO
390
{
391
private:
392
    const CWalletTx& m_wtx;
393
    const CTxOut& m_output;
394
395
public:
396
    WalletTXO(const CWalletTx& wtx, const CTxOut& output)
397
0
    : m_wtx(wtx),
398
0
    m_output(output)
399
0
    {
400
0
        Assume(std::ranges::find(wtx.tx->vout, output) != wtx.tx->vout.end());
401
0
    }
402
403
0
    const CWalletTx& GetWalletTx() const { return m_wtx; }
404
405
0
    const CTxOut& GetTxOut() const { return m_output; }
406
};
407
} // namespace wallet
408
409
#endif // BITCOIN_WALLET_TRANSACTION_H