Coverage Report

Created: 2026-04-29 19:21

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/tmp/bitcoin/src/private_broadcast.h
Line
Count
Source
1
// Copyright (c) 2023-present The Bitcoin Core developers
2
// Distributed under the MIT software license, see the accompanying
3
// file COPYING or https://opensource.org/license/mit/.
4
5
#ifndef BITCOIN_PRIVATE_BROADCAST_H
6
#define BITCOIN_PRIVATE_BROADCAST_H
7
8
#include <net.h>
9
#include <primitives/transaction.h>
10
#include <primitives/transaction_identifier.h>
11
#include <sync.h>
12
#include <util/time.h>
13
14
#include <optional>
15
#include <tuple>
16
#include <unordered_map>
17
#include <vector>
18
19
/**
20
 * Store a list of transactions to be broadcast privately. Supports the following operations:
21
 * - Add a new transaction
22
 * - Remove a transaction
23
 * - Pick a transaction for sending to one recipient
24
 * - Query which transaction has been picked for sending to a given recipient node
25
 * - Mark that a given recipient node has confirmed receipt of a transaction
26
 * - Query whether a given recipient node has confirmed reception
27
 * - Query whether any transactions that need sending are currently on the list
28
 */
29
class PrivateBroadcast
30
{
31
public:
32
33
    /// If a transaction is not sent to any peer for this duration,
34
    /// then we consider it stale / for rebroadcasting.
35
    static constexpr auto INITIAL_STALE_DURATION{5min};
36
37
    /// If a transaction is not received back from the network for this duration
38
    /// after it is broadcast, then we consider it stale / for rebroadcasting.
39
    static constexpr auto STALE_DURATION{1min};
40
41
    struct PeerSendInfo {
42
        CService address;
43
        NodeClock::time_point sent;
44
        std::optional<NodeClock::time_point> received;
45
    };
46
47
    struct TxBroadcastInfo {
48
        CTransactionRef tx;
49
        NodeClock::time_point time_added;
50
        std::vector<PeerSendInfo> peers;
51
    };
52
53
    /**
54
     * Add a transaction to the storage.
55
     * @param[in] tx The transaction to add.
56
     * @retval true The transaction was added.
57
     * @retval false The transaction was already present.
58
     */
59
    bool Add(const CTransactionRef& tx)
60
        EXCLUSIVE_LOCKS_REQUIRED(!m_mutex);
61
62
    /**
63
     * Forget a transaction.
64
     * @param[in] tx Transaction to forget.
65
     * @retval !nullopt The number of times the transaction was sent and confirmed
66
     * by the recipient (if the transaction existed and was removed).
67
     * @retval nullopt The transaction was not in the storage.
68
     */
69
    std::optional<size_t> Remove(const CTransactionRef& tx)
70
        EXCLUSIVE_LOCKS_REQUIRED(!m_mutex);
71
72
    /**
73
     * Pick the transaction with the fewest send attempts, and confirmations,
74
     * and oldest send/confirm times.
75
     * @param[in] will_send_to_nodeid Will remember that the returned transaction
76
     * was picked for sending to this node.
77
     * @param[in] will_send_to_address Address of the peer to which this transaction
78
     * will be sent.
79
     * @return Most urgent transaction or nullopt if there are no transactions.
80
     */
81
    std::optional<CTransactionRef> PickTxForSend(const NodeId& will_send_to_nodeid, const CService& will_send_to_address)
82
        EXCLUSIVE_LOCKS_REQUIRED(!m_mutex);
83
84
    /**
85
     * Get the transaction that was picked for sending to a given node by PickTxForSend().
86
     * @param[in] nodeid Node to which a transaction is being (or was) sent.
87
     * @return Transaction or nullopt if the nodeid is unknown.
88
     */
89
    std::optional<CTransactionRef> GetTxForNode(const NodeId& nodeid)
90
        EXCLUSIVE_LOCKS_REQUIRED(!m_mutex);
91
92
    /**
93
     * Mark that the node has confirmed reception of the transaction we sent it by
94
     * responding with `PONG` to our `PING` message.
95
     * @param[in] nodeid Node that we sent a transaction to.
96
     */
97
    void NodeConfirmedReception(const NodeId& nodeid)
98
        EXCLUSIVE_LOCKS_REQUIRED(!m_mutex);
99
100
    /**
101
     * Check if the node has confirmed reception of the transaction.
102
     * @retval true Node has confirmed, `NodeConfirmedReception()` has been called.
103
     * @retval false Node has not confirmed, `NodeConfirmedReception()` has not been called.
104
     */
105
    bool DidNodeConfirmReception(const NodeId& nodeid)
106
        EXCLUSIVE_LOCKS_REQUIRED(!m_mutex);
107
108
    /**
109
     * Check if there are transactions that need to be broadcast.
110
     */
111
    bool HavePendingTransactions()
112
        EXCLUSIVE_LOCKS_REQUIRED(!m_mutex);
113
114
    /**
115
     * Get the transactions that have not been broadcast recently.
116
     */
117
    std::vector<CTransactionRef> GetStale() const
118
        EXCLUSIVE_LOCKS_REQUIRED(!m_mutex);
119
120
    /**
121
     * Get stats about all transactions currently being privately broadcast.
122
     */
123
    std::vector<TxBroadcastInfo> GetBroadcastInfo() const
124
        EXCLUSIVE_LOCKS_REQUIRED(!m_mutex);
125
126
private:
127
    /// Status of a transaction sent to a given node.
128
    struct SendStatus {
129
        /// Node to which the transaction will be sent (or was sent).
130
        const NodeId nodeid;
131
        /// Address of the node.
132
        const CService address;
133
        /// When was the transaction picked for sending to the node.
134
        const NodeClock::time_point picked;
135
        /// When was the transaction reception confirmed by the node (by PONG).
136
        std::optional<NodeClock::time_point> confirmed;
137
138
12
        SendStatus(const NodeId& nodeid, const CService& address, const NodeClock::time_point& picked) : nodeid{nodeid}, address{address}, picked{picked} {}
139
    };
140
141
    /// Cumulative stats from all the send attempts for a transaction. Used to prioritize transactions.
142
    struct Priority {
143
        size_t num_picked{0}; ///< Number of times the transaction was picked for sending.
144
        NodeClock::time_point last_picked{}; ///< The most recent time when the transaction was picked for sending.
145
        size_t num_confirmed{0}; ///< Number of nodes that have confirmed reception of a transaction (by PONG).
146
        NodeClock::time_point last_confirmed{}; ///< The most recent time when the transaction was confirmed.
147
148
        auto operator<=>(const Priority& other) const
149
5
        {
150
            // Invert `other` and `this` in the comparison because smaller num_picked, num_confirmed or
151
            // earlier times mean greater priority. In other words, if this.num_picked < other.num_picked
152
            // then this > other.
153
5
            return std::tie(other.num_picked, other.num_confirmed, other.last_picked, other.last_confirmed) <=>
154
5
                   std::tie(num_picked, num_confirmed, last_picked, last_confirmed);
155
5
        }
156
    };
157
158
    /// A pair of a transaction and a sent status for a given node. Convenience return type of GetSendStatusByNode().
159
    struct TxAndSendStatusForNode {
160
        const CTransactionRef& tx;
161
        SendStatus& send_status;
162
    };
163
164
    // No need for salted hasher because we are going to store just a bunch of locally originating transactions.
165
166
    struct CTransactionRefHash {
167
        size_t operator()(const CTransactionRef& tx) const
168
16.2k
        {
169
16.2k
            return static_cast<size_t>(tx->GetWitnessHash().ToUint256().GetUint64(0));
170
16.2k
        }
171
    };
172
173
    struct CTransactionRefComp {
174
        bool operator()(const CTransactionRef& a, const CTransactionRef& b) const
175
7
        {
176
7
            return a->GetWitnessHash() == b->GetWitnessHash(); // If wtxid equals, then txid also equals.
177
7
        }
178
    };
179
180
    /**
181
     * Derive the sending priority of a transaction.
182
     * @param[in] sent_to List of nodes that the transaction has been sent to.
183
     */
184
    static Priority DerivePriority(const std::vector<SendStatus>& sent_to);
185
186
    /**
187
     * Find which transaction we sent to a given node (marked by PickTxForSend()).
188
     * @return That transaction together with the send status or nullopt if we did not
189
     * send any transaction to the given node.
190
     */
191
    std::optional<TxAndSendStatusForNode> GetSendStatusByNode(const NodeId& nodeid)
192
        EXCLUSIVE_LOCKS_REQUIRED(m_mutex);
193
    struct TxSendStatus {
194
        const NodeClock::time_point time_added{NodeClock::now()};
195
        std::vector<SendStatus> send_statuses;
196
    };
197
    mutable Mutex m_mutex;
198
    std::unordered_map<CTransactionRef, TxSendStatus, CTransactionRefHash, CTransactionRefComp>
199
        m_transactions GUARDED_BY(m_mutex);
200
};
201
202
#endif // BITCOIN_PRIVATE_BROADCAST_H