15 const bool inserted{m_transactions.try_emplace(tx).second};
23 const auto handle{m_transactions.extract(tx)};
25 const auto p{DerivePriority(handle.mapped().send_statuses)};
26 return p.num_confirmed;
36 const auto it{std::ranges::max_element(
38 [](
const auto& a,
const auto& b) {
return a < b; },
39 [](
const auto& el) {
return DerivePriority(el.second.send_statuses); })};
41 if (it != m_transactions.end()) {
42 auto& [tx, state]{*it};
43 state.send_statuses.emplace_back(will_send_to_nodeid, will_send_to_address,
NodeClock::now());
54 const auto tx_and_status{GetSendStatusByNode(nodeid)};
55 if (tx_and_status.has_value()) {
56 return tx_and_status.value().tx;
65 const auto tx_and_status{GetSendStatusByNode(nodeid)};
66 if (tx_and_status.has_value()) {
75 const auto tx_and_status{GetSendStatusByNode(nodeid)};
76 if (tx_and_status.has_value()) {
77 return tx_and_status.value().send_status.confirmed.has_value();
86 return !m_transactions.empty();
94 std::vector<CTransactionRef> stale;
95 for (
const auto& [tx, state] : m_transactions) {
97 if (p.num_confirmed == 0) {
110 std::vector<TxBroadcastInfo> entries;
111 entries.reserve(m_transactions.size());
113 for (
const auto& [tx, state] : m_transactions) {
114 std::vector<PeerSendInfo> peers;
115 peers.reserve(state.send_statuses.size());
116 for (
const auto& status : state.send_statuses) {
117 peers.emplace_back(
PeerSendInfo{.
address = status.address, .sent = status.picked, .received = status.confirmed});
119 entries.emplace_back(
TxBroadcastInfo{.
tx = tx, .time_added = state.time_added, .peers = std::move(peers)});
129 for (
const auto& send_status : sent_to) {
131 if (send_status.confirmed.has_value()) {
143 for (
auto& [tx, state] : m_transactions) {
144 for (
auto& send_status : state.send_statuses) {
145 if (send_status.nodeid == nodeid) {
A combination of a network address (CNetAddr) and a (TCP) port.
static Priority DerivePriority(const std::vector< SendStatus > &sent_to)
Derive the sending priority of a transaction.
void NodeConfirmedReception(const NodeId &nodeid) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Mark that the node has confirmed reception of the transaction we sent it by responding with PONG to o...
std::vector< TxBroadcastInfo > GetBroadcastInfo() const EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Get stats about all transactions currently being privately broadcast.
bool HavePendingTransactions() EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Check if there are transactions that need to be broadcast.
bool DidNodeConfirmReception(const NodeId &nodeid) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Check if the node has confirmed reception of the transaction.
static constexpr auto STALE_DURATION
If a transaction is not received back from the network for this duration after it is broadcast,...
std::optional< size_t > Remove(const CTransactionRef &tx) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Forget a transaction.
std::optional< CTransactionRef > PickTxForSend(const NodeId &will_send_to_nodeid, const CService &will_send_to_address) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Pick the transaction with the fewest send attempts, and confirmations, and oldest send/confirm times.
std::optional< TxAndSendStatusForNode > GetSendStatusByNode(const NodeId &nodeid) EXCLUSIVE_LOCKS_REQUIRED(m_mutex)
Find which transaction we sent to a given node (marked by PickTxForSend()).
std::optional< CTransactionRef > GetTxForNode(const NodeId &nodeid) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Get the transaction that was picked for sending to a given node by PickTxForSend().
bool Add(const CTransactionRef &tx) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Add a transaction to the storage.
std::vector< CTransactionRef > GetStale() const EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Get the transactions that have not been broadcast recently.
static constexpr auto INITIAL_STALE_DURATION
If a transaction is not sent to any peer for this duration, then we consider it stale / for rebroadca...
std::shared_ptr< const CTransaction > CTransactionRef
static time_point now() noexcept
Return current system time or mocked time, if set.
Cumulative stats from all the send attempts for a transaction. Used to prioritize transactions.
size_t num_picked
Number of times the transaction was picked for sending.
NodeClock::time_point last_confirmed
The most recent time when the transaction was confirmed.
size_t num_confirmed
Number of nodes that have confirmed reception of a transaction (by PONG).
NodeClock::time_point last_picked
The most recent time when the transaction was picked for sending.
A pair of a transaction and a sent status for a given node. Convenience return type of GetSendStatusB...
const CTransactionRef & tx
#define EXCLUSIVE_LOCKS_REQUIRED(...)