wallet: Properly rebroadcast unconfirmed transaction chains #25768

pull achow101 wants to merge 2 commits into bitcoin:master from achow101:unify-resend-reaccept changing 7 files +111 −64
  1. achow101 commented at 0:45 am on August 2, 2022: member

    Currently ResendWalletTransactions (used for normal rebroadcasts) will attempt to rebroadcast all of the transactions in the wallet in the order they are stored in mapWallet. This ends up being random as mapWallet is a std::unordered_map. However ReacceptWalletTransactions (used for adding to the mempool on loading) first sorts the txs by wallet insertion order, then submits them. The result is that ResendWalletTranactions will fail to rebroadcast child transactions if their txids happen to be lexicographically less than their parent’s txid. This PR resolves this issue by combining ReacceptWalletTransactions and ResendWalletTransactions into a new ResubmitWalletTransactions so that the iteration code and basic checks are shared.

    A test has also been added that checks that such transaction chains are rebroadcast correctly.

  2. DrahtBot added the label Wallet on Aug 2, 2022
  3. DrahtBot commented at 2:59 am on August 2, 2022: contributor

    The following sections might be updated with supplementary metadata relevant to reviewers and maintainers.

    Conflicts

    Reviewers, this pull request conflicts with the following ones:

    • #24683 (Add and use ForEachWallet by promag)
    • #20892 (tests: Run both descriptor and legacy tests within a single test invocation by achow101)

    If you consider this pull request important, please also help to review the conflicting pull requests. Ideally, start with the one that should be merged first.

  4. glozow commented at 4:17 pm on August 2, 2022: member
    Concept ACK
  5. achow101 force-pushed on Aug 2, 2022
  6. in test/functional/wallet_resendwallettransactions.py:76 in b11cceaf16 outdated
    78@@ -74,6 +79,41 @@ def run_test(self):
    79         node.setmocktime(now + 36 * 60 * 60 + 600)
    80         peer_second.wait_for_broadcast([txid])
    81 
    82+        self.log.info("Chain of unconfirmed not-in-mempool txs are rebroadcast")
    


    glozow commented at 3:52 pm on August 8, 2022:
    Question: Is this test meant to fail without the first commit?

    achow101 commented at 4:04 pm on August 8, 2022:
    Yes

    glozow commented at 3:06 pm on August 18, 2022:
    It’s passing for me on master :thinking:

    achow101 commented at 3:17 pm on August 18, 2022:
    Since master has mapWallet as a std::unsorted_map, this will now only fail randomly on it. It should fail half of the time, on average. I don’t know how we can guarantee that the child will come before the parent in an unsorted map.

    achow101 commented at 3:26 pm on August 18, 2022:
    I’ve removed the “make a spend that will come before” and updated the comment to explain that.

    glozow commented at 3:39 pm on August 18, 2022:
    Hm, I see what you’re saying about std::unsorted_map and it makes sense to me, but I’ve run the test >20 times and haven’t gotten a failure yet…

    achow101 commented at 4:26 pm on August 18, 2022:
    Found a bug in the test. It should fail intermittently on master now

    glozow commented at 4:34 pm on August 18, 2022:
    Aha, fails intermittently on master for me now :) thanks!

    kevkevinpal commented at 5:48 pm on August 18, 2022:
    I’m seeing the test fail intermittently on commit e40ddf3 does that sound right or should the test only fail on the master branch

    achow101 commented at 6:20 pm on August 18, 2022:
    It should only fail on master. It should not fail on this PR. Make sure that you’ve compiled this PR before running the test.

    kevkevinpal commented at 9:04 pm on August 18, 2022:

    hmm I think I’ve compiled it correctly (still new to the repo so might be doing it incorrectly), this is what I did

    0git fetch origin pull/25768/head:PR25768
    1git checkout  PR25768
    2./autogen.sh
    3./configure
    4make clean
    5make -j "$(($(sysctl -n hw.physicalcpu)+1))"
    6./test/functional/wallet_resendwallettransactions.py
    

    This is the error I see after I run the test, sometimes it passes

     02022-08-18T20:38:25.914000Z TestFramework (INFO): Initializing test directory /var/folders/9g/cvx014yx4dq5lwl_x5zvn_j80000gn/T/bitcoin_func_test_43xe3o56
     12022-08-18T20:38:28.879000Z TestFramework (INFO): Create a new transaction and wait until it's broadcast
     22022-08-18T20:38:29.627000Z TestFramework (INFO): Create a block
     32022-08-18T20:38:29.634000Z TestFramework (INFO): Bump time & check that transaction is rebroadcast
     42022-08-18T20:38:29.905000Z TestFramework (INFO): Chain of unconfirmed not-in-mempool txs are rebroadcast
     52022-08-18T20:38:31.037000Z TestFramework (ERROR): Assertion failed
     6Traceback (most recent call last):
     7  File "/Users/kevkevin/DEVDIR/bitcoin/test/functional/test_framework/test_framework.py", line 133, in main
     8    self.run_test()
     9  File "/Users/kevkevin/DEVDIR/bitcoin/./test/functional/wallet_resendwallettransactions.py", line 107, in run_test
    10    assert_raises_rpc_error(-5, "Transaction not in mempool", node.getmempoolentry, txid)
    11  File "/Users/kevkevin/DEVDIR/bitcoin/test/functional/test_framework/util.py", line 126, in assert_raises_rpc_error
    12    assert try_rpc(code, message, fun, *args, **kwds), "No exception raised"
    13AssertionError: No exception raised
    142022-08-18T20:38:31.089000Z TestFramework (INFO): Stopping nodes
    152022-08-18T20:38:31.508000Z TestFramework (WARNING): Not cleaning up dir /var/folders/9g/cvx014yx4dq5lwl_x5zvn_j80000gn/T/bitcoin_func_test_43xe3o56
    162022-08-18T20:38:31.508000Z TestFramework (ERROR): Test failed. Test logging available at /var/folders/9g/cvx014yx4dq5lwl_x5zvn_j80000gn/T/bitcoin_func_test_43xe3o56/test_framework.log
    172022-08-18T20:38:31.508000Z TestFramework (ERROR):
    182022-08-18T20:38:31.509000Z TestFramework (ERROR): Hint: Call /Users/kevkevin/DEVDIR/bitcoin/test/functional/combine_logs.py '/var/folders/9g/cvx014yx4dq5lwl_x5zvn_j80000gn/T/bitcoin_func_test_43xe3o56' to consolidate all logs
    192022-08-18T20:38:31.509000Z TestFramework (ERROR):
    202022-08-18T20:38:31.509000Z TestFramework (ERROR): If this failure happened unexpectedly or intermittently, please file a bug and provide a link or upload of the combined log.
    212022-08-18T20:38:31.509000Z TestFramework (ERROR): https://github.com/bitcoin/bitcoin/issues
    222022-08-18T20:38:31.509000Z TestFramework (ERROR):
    

    running on MacOs Monterey Version 12.3.1


    achow101 commented at 9:40 pm on August 18, 2022:
    I’m not seeing that error even when running on a loop.

    maflcko commented at 8:10 am on August 19, 2022:

    git checkout PR25768

    Can you check that git log -1 prints the correct hash, or alternatively just checkout the correct hash directly? Also, make sure that the working tree is clean.

    0git fetch origin e40ddf36bed81bdf28d386eb961c9ed22b69e207
    1git checkout e40ddf36bed81bdf28d386eb961c9ed22b69e207
    2git reset --hard HEAD # warning!! wipes unstaged changes in the tree if there still are some
    

    kevkevinpal commented at 0:33 am on August 21, 2022:
    still getting the error and I am able to see I’m on the correct commit with git log -1 and tried thee commands you posted aswell, I’ll try and dig deeper to see what may be the issue

    kevkevinpal commented at 1:50 am on August 21, 2022:
    not exactly sure what happened but after rebuilding master then rebuilding this PR’s branch twice the test passes now sorry for any confusion
  7. in src/wallet/wallet.cpp:1831 in b11cceaf16 outdated
    1826+    for (auto& [wtxid, wtx] : mapWallet) {
    1827+        assert(wtx.GetHash() == wtxid);
    1828+        sorted_txs.insert(std::make_pair(wtx.nOrderPos, &wtx));
    1829+    }
    1830+    std::vector<CWalletTx*> out;
    1831+    for (auto [pos, wtx] : sorted_txs) {
    


    kevkevinpal commented at 1:20 am on August 17, 2022:
    nit: not sure if we need pos here doesn’t seem to be used

    achow101 commented at 3:27 pm on August 18, 2022:
    Done
  8. kevkevinpal commented at 1:22 am on August 17, 2022: contributor
    Concept ACK was able to run the functional test on my local and it ran fine, still want to test on regtest but still new to the project so still learning
  9. achow101 force-pushed on Aug 18, 2022
  10. achow101 force-pushed on Aug 18, 2022
  11. in test/functional/wallet_resendwallettransactions.py:34 in e40ddf36be outdated
    28@@ -27,7 +29,10 @@ def run_test(self):
    29         peer_first = node.add_p2p_connection(P2PTxInvStore())
    30 
    31         self.log.info("Create a new transaction and wait until it's broadcast")
    32-        txid = node.sendtoaddress(node.getnewaddress(), 1)
    33+        unspent = node.listunspent()
    34+        parent_utxo = unspent[0]
    35+        indep_utxo = unspent[1]
    


    glozow commented at 2:18 pm on August 19, 2022:
    Perhaps it’s worth doing the test with, say, 10 parent-child pairs? That would catch this bug more reliably without needing to run the test multiple times.

    stickies-v commented at 1:22 pm on August 24, 2022:

    nit: could use a one-liner which imo is equally readable?

    0        parent_utxo, indep_utxo = node.listunspent()[:2]
    

    achow101 commented at 9:56 pm on August 24, 2022:
    I’ve reimplemented the test so that it always fails on master.

    achow101 commented at 8:41 pm on August 25, 2022:
    Done
  12. in src/wallet/wallet.cpp:1827 in e40ddf36be outdated
    1822+{
    1823+    AssertLockHeld(cs_wallet);
    1824+    // Sort pending transactions based on their initial wallet insertion order
    1825+    std::map<int64_t, CWalletTx*> sorted_txs;
    1826+    for (auto& [wtxid, wtx] : mapWallet) {
    1827+        assert(wtx.GetHash() == wtxid);
    


    Riahiamirreza commented at 5:37 pm on August 20, 2022:
    Sorry, I don’t understand the need for assert(wtx.GetHash() == wtxid);. In which case can this assert fails? Only when the data is corrupted?

    sipa commented at 6:54 pm on August 20, 2022:

    There is never a need for an assert. Assertions verify assumptions about the codebase, and should only fail if something you believe to be impossible in the codebase to be the case. It should never trigger, unless there is a bug in the code.

    By writing this, the code author signals to readers of the code “here you can assume that wtxid is always equal to wtx.GetHash()”. Further, this is verified during all tests, so we know it is also true for all scenarios any test case runs through.


    glozow commented at 8:59 am on August 22, 2022:

    Wait, I see this is just moved and not new, but is this key wtxid or txid? Or does the “w” mean wallet instead of witness?

    If it’s wtxid, this assertion could fail when wtxid != txid since CWalletTx::GetHash() calls tx->GetHash(). But AFAICT it is a txid, so should be called txid instead? https://github.com/bitcoin/bitcoin/blob/02aefa169a9e6ed12c7bd8f3392adcd073d8d56b/src/wallet/wallet.cpp#L956-L971


    achow101 commented at 4:04 pm on August 22, 2022:
    The “w” refers to “wallet”.

    naumenkogs commented at 8:18 am on August 24, 2022:
    Is it good time to rename it to wallet_txid for clarification? Without @glozow comment, i would just assume w stands for witness without any doubt…

    hernanmarino commented at 3:06 pm on August 24, 2022:
    The variable name also confused me , +1 to renaming it

    pablomartin4btc commented at 3:06 pm on August 24, 2022:
    I got the same confusion before reading the entire thread, thought it came from witness…

    achow101 commented at 3:56 pm on August 24, 2022:
    The variable names wtxid and wtx are used very frequently throughout the wallet code to refer to wallet transactions and their txids. If people find this confusing, then I think it would be more appropriate to have a scripted-diff PR rename all of those rather than renaming each individual instance that occurs in a PR.

    achow101 commented at 9:56 pm on August 24, 2022:
    Renamed to txid.
  13. Riahiamirreza commented at 4:27 pm on August 21, 2022: contributor
    Would you explain what is the difference between ResendWalletTransactions and ReacceptWalletTransactions? As you said ResendWalletTransactions is used for normal rebroadcasting and ReacceptWalletTransactions is used for rebroadcasting on loading. I’m not sure what do you mean by “loading”, you mean something like reindex or reorg?
  14. achow101 commented at 4:36 pm on August 21, 2022: member

    I’m not sure what do you mean by “loading”, you mean something like reindex or reorg?

    “Loading” as in when the wallet is read from disk and loaded into memory.

  15. Riahiamirreza commented at 4:55 pm on August 21, 2022: contributor
    So what is the difference between rebroadcasting when loading and in normal cases?
  16. achow101 commented at 5:04 pm on August 21, 2022: member
    There are differences in which transactions will be considered for rebroadcasting. On load, all non-abandoned transactions will be rebroadcast. The periodic rebroadcast will only rebroadcast those that are more than 5 minutes older than the most recent block.
  17. in test/functional/wallet_resendwallettransactions.py:38 in e40ddf36be outdated
    34+        parent_utxo = unspent[0]
    35+        indep_utxo = unspent[1]
    36+        txid = node.send(outputs=[{node.getnewaddress(): 1}], options={"inputs": [parent_utxo]})["txid"]
    37 
    38         # Wallet rebroadcast is first scheduled 1 sec after startup (see
    39         # nNextResend in ResendWalletTransactions()). Tell scheduler to call
    


    rajarshimaitra commented at 8:29 am on August 24, 2022:

    This comment on scheduling the resending logic at 1 sec after the startup seems to be visible in the StartWallets() function, rather than in ResendWalletTransactions()’s nNextResend timepoint setting.

    So maybe this comment should point to this part of the code instead?

    https://github.com/bitcoin/bitcoin/blob/713ea7a4181f411949a3e9bb9a358533388b1159/src/wallet/load.cpp#L154


    achow101 commented at 9:53 pm on August 24, 2022:
    The comment is removed now that this behavior is no longer needed.
  18. in src/wallet/wallet.cpp:1847 in e40ddf36be outdated
    1852+    for (auto wtx : GetSortedTxs()) {
    1853+        int nDepth = GetTxDepthInMainChain(*wtx);
    1854 
    1855-        if (!wtx.IsCoinBase() && (nDepth == 0 && !wtx.isAbandoned())) {
    1856-            mapSorted.insert(std::make_pair(wtx.nOrderPos, &wtx));
    1857+        if (!wtx->IsCoinBase() && (nDepth == 0 && !wtx->isAbandoned())) {
    


    stickies-v commented at 8:29 am on August 24, 2022:

    When looking at CWallet::GetTxDepthInMainChain, if nDepth == 0, isn’t that equivalent to !wtx->IsConfirmed() && !wtx->IsConflicted(), which in turn in combination with !wtx->isAbandoned() is equivalent to !wtx->IsUnconfirmed() - am I missing something? (I know this is moved code)

    0        if (!wtx->IsCoinBase() && !wtx->IsUnconfirmed()) {
    

    achow101 commented at 4:09 pm on August 24, 2022:
    Yes, those should be equivalent. Will do if I need to re-push.

    achow101 commented at 9:53 pm on August 24, 2022:
    Done

    stickies-v commented at 4:05 pm on August 25, 2022:
    Good catch that !IsCoinBase() is redundant in combination with !IsUnconfirmed() 👍
  19. in src/wallet/wallet.cpp:1821 in 9a556564e9 outdated
    1817@@ -1818,32 +1818,37 @@ CWallet::ScanResult CWallet::ScanForWalletTransactions(const uint256& start_bloc
    1818     return result;
    1819 }
    1820 
    1821+std::vector<CWalletTx*> CWallet::GetSortedTxs()
    


    naumenkogs commented at 8:33 am on August 24, 2022:
    Should we specify in variable name what kind of sorting is applied?

    stickies-v commented at 8:43 am on August 24, 2022:
    I was about to add the same suggestion 👍

    achow101 commented at 9:55 pm on August 24, 2022:
    The function has ended up being removed.
  20. in src/wallet/wallet.cpp:1922 in 9a556564e9 outdated
    1918@@ -1914,15 +1919,14 @@ void CWallet::ResendWalletTransactions()
    1919     { // cs_wallet scope
    1920         LOCK(cs_wallet);
    1921 
    1922-        // Relay transactions
    1923-        for (std::pair<const uint256, CWalletTx>& item : mapWallet) {
    1924-            CWalletTx& wtx = item.second;
    1925+        // Try to add wallet transactions to memory pool
    


    naumenkogs commented at 8:39 am on August 24, 2022:
    “and relay”?

    pablomartin4btc commented at 4:05 pm on August 24, 2022:
    you meant the variable? the change is on void CWallet::ResendWalletTransactions().

    achow101 commented at 9:55 pm on August 24, 2022:
    Done
  21. ghost commented at 9:03 am on August 24, 2022: none

    This PR resolves this issue by refactoring ReacceptWalletTransactions sorting code into a separate function, then using that function in both ReacceptWalletTransactions and ResendWalletTransactions so that both functions will sort the transactions in the same order before processing for rebroadcast.

    Is it possible to shuffle transactions before SubmitTxMemoryPoolAndRelay rebroadcasting using std::shuffle? Since only wallet txs are rebroadcasted which is a separate privacy issue, we could at least hide the order in which they were inserted in wallet db.

  22. in test/functional/wallet_resendwallettransactions.py:88 in e40ddf36be outdated
    83+        # Ideally we would add a transaction that would spend the existing transaction and also be
    84+        # in mapWallet before it. However mapWallet is a std::unsorted_map which means that it is
    85+        # hard for us to externally construct such a transaction. So we will ignore that fact and
    86+        # rely on the fact that because there are only two items in mapWallet, 50% of the time the
    87+        # child will be placed before the parent in mapWallet. The incorrect behavior that this
    88+        # test tries to catch should thus fail intermittently at a rate of 50%.
    


    rajarshimaitra commented at 10:05 am on August 24, 2022:

    I am a bit confused on the intermittent test failure occurring here, maybe I am missing something.

    Isn’t sorting the order by wallet insertion index is what this PR fixes? So why would it still fail to reboadcast if child.txid() < parent.txid()?

    Also, does it make sense to have some unit testing to ensure this behavior of the GetSorted() function itself?


    achow101 commented at 4:16 pm on August 24, 2022:

    Isn’t sorting the order by wallet insertion index is what this PR fixes? So why would it still fail to reboadcast if child.txid() < parent.txid()?

    The comment is talking about the code before this PR and how the test tries to catch it, not that this test should fail intermittently after the PR.

    Also, does it make sense to have some unit testing to ensure this behavior of the GetSorted() function itself?

    There are already tests for the underlying wtxOrdered itself (or tests that end up testing that behavior, although perhaps not specifically).


    rajarshimaitra commented at 4:49 pm on August 24, 2022:
    So that means the code should not fail intermittently?? I will double check again that I am testing the correct binary, but I thoughts that’s expected behavior from the comment..

    achow101 commented at 4:54 pm on August 24, 2022:
    It should only fail intermittently on master.
  23. rajarshimaitra commented at 10:07 am on August 24, 2022: contributor

    Concept + tACK e40ddf36bed81bdf28d386eb961c9ed22b69e207

    • Verified that the tests are intermittently failing.
    • Failure occurring at debug log assertion ResendWalletTransactions: resubmit 2 unconfirmed transactions

    Below are just one nit comment and a conceptual question.

  24. in test/functional/wallet_resendwallettransactions.py:109 in e40ddf36be outdated
     97+        block.solve()
     98+        node.submitblock(block.serialize().hex())
     99+        node.syncwithvalidationinterfacequeue()
    100+
    101+        # Evict these txs from the mempool
    102+        evict_time = block_time + 60 * 60 * 336 + 5
    


    stickies-v commented at 1:52 pm on August 24, 2022:
    nit: DEFAULT_MEMPOOL_EXPIRY_HOURS is defined in test/functional/mempool_expiry.py. Similarly to #25810, could move that to messages.py and have the benefit of using a named var instead of a less obvious literal?

    achow101 commented at 8:41 pm on August 25, 2022:
    Done
  25. in test/functional/wallet_resendwallettransactions.py:119 in e40ddf36be outdated
    107+        assert_raises_rpc_error(-5, "Transaction not in mempool", node.getmempoolentry, txid)
    108+        assert_raises_rpc_error(-5, "Transaction not in mempool", node.getmempoolentry, child_txid)
    109+
    110+        # Rebroadcast and check that parent and child are both in the mempool
    111+        with node.assert_debug_log(['ResendWalletTransactions: resubmit 2 unconfirmed transactions']):
    112+            node.setmocktime(evict_time + 36 * 60 * 60)
    


    stickies-v commented at 2:42 pm on August 24, 2022:

    nit: perhaps useful to explain the 36 literal?

    0            node.setmocktime(evict_time + 36 * 60 * 60)  # 36h is the upper limit in CWallet::ResendWalletTransactions()`
    

    achow101 commented at 8:42 pm on August 25, 2022:
    Done
  26. achow101 commented at 3:54 pm on August 24, 2022: member

    Is it possible to shuffle transactions before SubmitTxMemoryPoolAndRelay rebroadcasting using std::shuffle? Since only wallet txs are rebroadcasted which is a separate privacy issue, we could at least hide the order in which they were inserted in wallet db.

    No. Shuffling is the reason we have this issue in the first place (there is no order in an unsorted map so iterating it is basically the same as iterating a shuffled list). I don’t think the insertion order is a privacy leak since it’s the same order as when the transactions were originally broadcast which is public information to anyone that’s connected to the network. Additionally, this is not a unilateral broadcast - it is up to the mempool relay code to determine when, how, and which transactions are actually rebroadcast. Only transactions that are not already in the mempool will be rebroadcast.

  27. ghost commented at 4:22 pm on August 24, 2022: none

    No. Shuffling is the reason we have this issue in the first place (there is no order in an unsorted map so iterating it is basically the same as iterating a shuffled list).

    I was suggesting to shuffle the sorted list before rebroadcast. So we have all the transactions we need but do not relay in that order.

    I don’t think the insertion order is a privacy leak since it’s the same order as when the transactions were originally broadcast which is public information to anyone that’s connected to the network.

    In that case, feel free to ignore my suggestion.

  28. ghost commented at 4:23 pm on August 24, 2022: none
    Concept ACK
  29. achow101 commented at 4:34 pm on August 24, 2022: member

    I was suggesting to shuffle the sorted list before rebroadcast. So we have all the transactions we need but do not relay in that order.

    Shuffling after sorting would just undo the sorting. Sorting is not used for filtering; it is used because we need to have the transactions in order. Specifically parent transactions must be rebroadcast before child transactions, and that specific order can be obtained through the insertion order.

    While we could do something more complicated like actually building packages so that the actually order dependent things are grouped together, and then shuffling the packages, but I don’t see the benefit in doing that. That might actually harm privacy because we would be broadcasting transactions that are related to each other at the same time, but only those transactions that are related to our wallet. Since a package that the wallet builds may exclude other transactions in the package that the mempool would know about, we could essentially be telling other nodes some of the scripts that our wallet is watching for as they could then look at what the rebroadcast package has that the rest of the package in the mempool does not.

  30. achow101 force-pushed on Aug 24, 2022
  31. achow101 commented at 9:52 pm on August 24, 2022: member

    Based on some discussions during the PR review club today, I’ve made several changes to this PR. The first is that ResendWalletTransaction and ReacceptWalletTransactions are now combined into a single function ResubmitWalletTranasctions. There were some subtle differences between the two functions that are being preserved by using two optional parameters. The first difference is that ReacceptWalletTransactions would not actually ask the mempool to relay the transactions (there is a parameter for relay in SubmitTxMemoryPoolAndRelay that was different in both). There is a parameter relay with a default of true to handle that. The second is that ResendWalletTransactions would look chain status (is ibd, reindexing, etc.) and transaction time in order to determine whether to try submitting a transaction, whereas ReacceptWalletTransactions would not. To match the ReaccpetWalletTransactions behavior, a force option with a default of false is added that will bypass those checks.

    Additionally, nNextResend was initialized the first time ResendWalletTransactions was run. As the new ResubmitWalletTransactions is always run on start, I’ve removed from the test the need to force it to be run as well as the handling around not actually trying to resend during that first run (we want to do the reaccept behavior in that first run). I’ve also removed GetSortedTxs and have instead taken the suggestion of filtering then sorting the transactions. This is done by inserting the CWalletTxs that we want to submit into a std::set with a custom comparator that uses CWalletTx.nOrderPos to determine the sort order.

    Lastly I’ve updated the test to reliably fail on master. I found that listreceivedbyaddress and related RPCs will output txids in the order that they are found in mapWallet. The new test will use that to check if the child is positioned before the parent, and if it is not, it will remake the child until it does.

  32. achow101 force-pushed on Aug 24, 2022
  33. unknown approved
  34. hernanmarino approved
  35. hernanmarino commented at 1:18 am on August 25, 2022: contributor

    ACK fee0dea

    Now the test consistently fails on master, as it should.

  36. in src/wallet/wallet.cpp:1931 in bec0579357 outdated
    1878 {
    1879+    // Don't attempt to resubmit if the wallet is configured to not broadcast,
    1880+    // even if forcing.
    1881+    if (!fBroadcastTransactions) return;
    1882+
    1883     // During reindex, importing and IBD, old wallet transactions become
    


    naumenkogs commented at 7:27 am on August 25, 2022:

    I’d say this comment is now slightly outdated. My first thought was “so all those calls with force=true will spam now?”. Then I realized they all have relay=false.

    I think extending this comment would improve readability. Might as well document what force does exactly (in broader terms is fine).


    achow101 commented at 3:23 pm on August 25, 2022:
    I’ve expanded the comment and also added relay to the check. This way we guarantee that we won’t spam if we want to relay, regardless of force.
  37. in src/wallet/wallet.cpp:1949 in bec0579357 outdated
    1890-    if (GetTime() < nNextResend || !fBroadcastTransactions) return;
    1891-    bool fFirst = (nNextResend == 0);
    1892+    if (!force && GetTime() < nNextResend) return;
    1893     // resend 12-36 hours from now, ~1 day on average.
    1894     nNextResend = GetTime() + (12 * 60 * 60) + GetRand(24 * 60 * 60);
    1895-    if (fFirst) return;
    


    naumenkogs commented at 7:31 am on August 25, 2022:
    It took me some time to realize why it’s safe. I think it’s worth expanding On startup, we will do this for all unconfirmed // transactions but will not ask the mempool to relay them. saying it protects privacy as a side effect?

    achow101 commented at 3:23 pm on August 25, 2022:
    Done.
  38. in src/wallet/wallet.cpp:1956 in bec0579357 outdated
    1914+            if (!wtx.isUnconfirmed()) continue;
    1915+
    1916+            // For periodic rebroadcast (force == false),
    1917+            // attempt to rebroadcast all txes more than 5 minutes older than
    1918+            // the last block.
    1919+            if (!force && wtx.nTimeReceived > m_best_block_time - 5 * 60) continue;
    


    naumenkogs commented at 7:38 am on August 25, 2022:
    Slightly off-topic, is this code prone to privacy leaks? Imagine a fork appearing: one (empty) block with a timestamp in the past. Would many own transactions get rebroadcast? If yes, i assume other nodes won’t forward them because they had them already, and it becomes apparent who is the source?

    achow101 commented at 2:48 pm on August 25, 2022:
    I don’t think so. Your own transactions would be submitted to your node’s mempool first, and only if it does not know about those transactions will it try to broadcast them.

    naumenkogs commented at 9:22 am on August 26, 2022:
    Right, thank you.
  39. in src/wallet/wallet.cpp:1962 in fee0dea1b1 outdated
    1921+        }
    1922+        // Now try submitting the transactions to the memory pool and (optionally) relay them.
    1923+        for (auto wtx : to_submit) {
    1924             std::string unused_err_string;
    1925-            if (SubmitTxMemoryPoolAndRelay(wtx, unused_err_string, true)) ++submitted_tx_count;
    1926+            if (SubmitTxMemoryPoolAndRelay(*wtx, unused_err_string, relay)) ++submitted_tx_count;
    


    rajarshimaitra commented at 2:44 pm on August 25, 2022:
    I might be wrong, but I thought we write bool variables with an f prefix, so should this be frelay?

    achow101 commented at 3:24 pm on August 25, 2022:
    No. We have had a new style in effect for the past several years. It only applies to new code, so a bunch of the old code written under the old style is still around.
  40. in test/functional/wallet_resendwallettransactions.py:73 in fee0dea1b1 outdated
    70+        with node.assert_debug_log(['resubmit 1 unconfirmed transactions']):
    71             node.setmocktime(now + 36 * 60 * 60)
    72             # Tell scheduler to call MaybeResendWalletTxn now.
    73             node.mockscheduler(1)
    74         # Give some time for trickle to occur
    75         node.setmocktime(now + 36 * 60 * 60 + 600)
    


    rajarshimaitra commented at 2:55 pm on August 25, 2022:
    Maybe this is simpler when written as mockscheduler(600)?

    achow101 commented at 3:25 pm on August 25, 2022:
    mockscheduler does not change the mocked time.
  41. rajarshimaitra approved
  42. rajarshimaitra commented at 3:09 pm on August 25, 2022: contributor

    tACK fee0dea1b1e32832dc5efca3621cfc8ba7c6c8f9

    Verified the test is failing reliably in master..

    Just two more nit comments (questions)..

  43. achow101 force-pushed on Aug 25, 2022
  44. in src/wallet/wallet.cpp:1925 in 71b7709fa0 outdated
    1884+// The `force` option results in all unconfirmed transactions being submitted to
    1885+// the mempool. This does not necessarily result in those transactions being relayed,
    1886+// that depends on the `relay` option. Periodic rebroadcast uses the pattern
    1887+// relay=true force=false (also the default values), while loading into the mempool
    1888+// (on start, or after import) uses relay=false force=true.
    1889+void CWallet::ResubmitWalletTransactions(bool relay, bool force)
    


    stickies-v commented at 4:07 pm on August 25, 2022:

    nit: intuitively I’d expect a force parameter to not respect flags like fBroadcastTransactions. I think something like force_ready would make it more obvious that this is not an absolute force. Probably personal preference?

    0void CWallet::ResubmitWalletTransactions(bool relay, bool force_ready)
    

    achow101 commented at 8:37 pm on August 25, 2022:
    I agree that force implies that more will happen than actually does, but I couldn’t think of anything better to call it. I don’t think force_ready is any clearer.
  45. in src/wallet/wallet.cpp:1917 in 71b7709fa0 outdated
    1926+            // Only rebroadcast unconfirmed txs
    1927+            if (!wtx.isUnconfirmed()) continue;
    1928+
    1929+            // For periodic rebroadcast (force == false),
    1930+            // attempt to rebroadcast all txes more than 5 minutes older than
    1931+            // the last block.
    


    stickies-v commented at 4:20 pm on August 25, 2022:

    nit: I think this comment is confusing, from just this I’d think the below code only executes for periodic rebroadcast.

    0            // Attempt to rebroadcast all txes more than 5 minutes older than
    1            // the last block, or all txes if force is true.
    

    achow101 commented at 8:42 pm on August 25, 2022:
    Done
  46. in src/wallet/wallet.cpp:1874 in 71b7709fa0 outdated
    1871+//
    1872+// Otherwise this function is called periodically in order to relay our unconfirmed txs.
    1873+// We do this on a random timer to slightly obfuscate which transactions
    1874+// come from our wallet.
    1875 //
    1876 // Ideally, we'd only resend transactions that we think should have been
    


    stickies-v commented at 4:27 pm on August 25, 2022:
    nit: while you’re touching this docstring and fn, worth slapping in a TODO here for increased visibility?

    achow101 commented at 8:42 pm on August 25, 2022:
    Done
  47. in src/wallet/transaction.h:309 in 71b7709fa0 outdated
    304@@ -305,6 +305,13 @@ class CWalletTx
    305     CWalletTx(CWalletTx const &) = delete;
    306     void operator=(CWalletTx const &x) = delete;
    307 };
    308+
    309+struct WalletTxOrderComparator {
    


    stickies-v commented at 5:13 pm on August 25, 2022:

    What do you think about a lambda function instead? Quite a bit less code, and imo about equally readable? As long as we’re not using this comparator anywhere else the lambda would have my preference, but I’m okay with the current implementation too.

     0diff --git a/src/wallet/transaction.h b/src/wallet/transaction.h
     1index 27983e356..c36cbeb23 100644
     2--- a/src/wallet/transaction.h
     3+++ b/src/wallet/transaction.h
     4@@ -306,12 +306,6 @@ public:
     5     void operator=(CWalletTx const &x) = delete;
     6 };
     7 
     8-struct WalletTxOrderComparator {
     9-    bool operator()(const CWalletTx* a, const CWalletTx* b) const
    10-    {
    11-        return a->nOrderPos < b->nOrderPos;
    12-    }
    13-};
    14 } // namespace wallet
    15 
    16 #endif // BITCOIN_WALLET_TRANSACTION_H
    17diff --git a/src/wallet/wallet.cpp b/src/wallet/wallet.cpp
    18index 69957ed9e..ad20f332a 100644
    19--- a/src/wallet/wallet.cpp
    20+++ b/src/wallet/wallet.cpp
    21@@ -1906,8 +1906,9 @@ void CWallet::ResubmitWalletTransactions(bool relay, bool force)
    22         LOCK(cs_wallet);
    23 
    24         // First filter for the transactions we want to rebroadcast.
    25-        // We use a set with WalletTxOrderComparator so that rebroadcasting occurs in insertion order
    26-        std::set<CWalletTx*, WalletTxOrderComparator> to_submit;
    27+        // We use a set ordered by nOrderPos so that rebroadcasting occurs in insertion order
    28+        auto cmp = [](const CWalletTx* a, const CWalletTx* b) { return a->nOrderPos < b->nOrderPos; };
    29+        std::set<CWalletTx*, decltype(cmp)> to_submit(cmp);
    30         for (auto& [txid, wtx] : mapWallet) {
    31             // Only rebroadcast unconfirmed txs
    32             if (!wtx.isUnconfirmed()) continue;
    

    achow101 commented at 5:38 pm on August 25, 2022:
    I think the comparator might actually be useful in other places too.
  48. stickies-v approved
  49. stickies-v commented at 5:16 pm on August 25, 2022: contributor

    utACK 71b7709fa075dc4c9b55bb07304687724f67f0cb

    The refactoring into ResubmitWalletTransactions() makes a lot of sense and improves understanding the codebase imo.

  50. in src/wallet/wallet.cpp:1976 in 71b7709fa0 outdated
    1934@@ -1936,7 +1935,7 @@ void CWallet::ResendWalletTransactions()
    1935 void MaybeResendWalletTxs(WalletContext& context)
    1936 {
    1937     for (const std::shared_ptr<CWallet>& pwallet : GetWallets(context)) {
    1938-        pwallet->ResendWalletTransactions();
    1939+        pwallet->ResubmitWalletTransactions();
    


    LarryRuane commented at 7:32 pm on August 25, 2022:

    nit, this is the only call site that takes advantage of default arguments, so making this change would allow this method to not need default arguments. (Personally, I dislike default arguments unless there’s a good reason for them.)

    0        pwallet->ResubmitWalletTransactions(/*relay=*/true, /*force=*/false);
    

    achow101 commented at 4:42 pm on August 29, 2022:
    Removed the default arguments.
  51. achow101 force-pushed on Aug 25, 2022
  52. achow101 force-pushed on Aug 25, 2022
  53. achow101 force-pushed on Aug 25, 2022
  54. in test/functional/wallet_resendwallettransactions.py:77 in 789b78430b outdated
    72@@ -68,6 +73,53 @@ def run_test(self):
    73         node.setmocktime(now + 36 * 60 * 60 + 600)
    74         peer_second.wait_for_broadcast([txid])
    75 
    76+        self.log.info("Chain of unconfirmed not-in-mempool txs are rebroadcast")
    77+        # This tests that the node broadcasts the parent transaction before the parent transaction.
    


    naumenkogs commented at 9:55 am on August 26, 2022:
    the parent transaction before the parent transaction?

    achow101 commented at 4:07 pm on August 26, 2022:
    Oops, fixed.
  55. in src/wallet/wallet.cpp:1901 in 789b78430b outdated
    1895@@ -1924,43 +1896,69 @@ std::set<uint256> CWallet::GetTxConflicts(const CWalletTx& wtx) const
    1896     return result;
    1897 }
    1898 
    1899-// Rebroadcast transactions from the wallet. We do this on a random timer
    1900-// to slightly obfuscate which transactions come from our wallet.
    1901+// Resubmit transactions from the wallet to the mempool, optionally asking the
    1902+// mempool to relay them. On startup, we will do this for all unconfirmed
    


    stickies-v commented at 3:13 pm on August 26, 2022:
    nit: perhaps this first sentence should live in wallet.h?
  56. achow101 force-pushed on Aug 26, 2022
  57. DrahtBot added the label Needs rebase on Aug 26, 2022
  58. achow101 force-pushed on Aug 27, 2022
  59. DrahtBot removed the label Needs rebase on Aug 27, 2022
  60. in test/functional/wallet_resendwallettransactions.py:78 in 748d5c75d6 outdated
    72@@ -68,6 +73,53 @@ def run_test(self):
    73         node.setmocktime(now + 36 * 60 * 60 + 600)
    74         peer_second.wait_for_broadcast([txid])
    75 
    76+        self.log.info("Chain of unconfirmed not-in-mempool txs are rebroadcast")
    77+        # This tests that the node broadcasts the parent transaction before the child transaction.
    78+        # The previous behavior is that the rebroadcasting would simply iterate mapWallet. As
    


    naumenkogs commented at 6:29 am on August 29, 2022:
    nit: I think the previous behavior stuff belongs to the commit message, not to the file content.

    achow101 commented at 4:41 pm on August 29, 2022:
    Done
  61. LarryRuane approved
  62. LarryRuane commented at 3:44 pm on August 29, 2022: contributor
    ACK 748d5c75d60505a3d84d36782f5cf4949224e103
  63. wallet: Deduplicate Resend and ReacceptWalletTransactions
    Both of these functions do almost the exact same thing. They can be
    deduplicated so that their behavior matches except for the filtering
    aspect. As this function will now always be called on wallet loading,
    nNextResend will also always be initialized, so
    wallet_resendwallettransactions.py is updated to account for that.
    
    This also resolves a bug where ResendWalletTransactions would fail to
    rebroadcast txs in insertion order thereby potentially rebroadcasting a
    child transaction before its parent and causing the child to not
    actually get rebroadcast.
    
    Also names the combined function to ResubmitWalletTransactions as the
    function just submits the transactions to the mempool rather than doing
    any sending by itself.
    10d91c5abe
  64. test: Test that an unconfirmed not-in-mempool chain is rebroadcast
    The test checks that parent txs are broadcast before child txs.
    
    The previous behavior is that the rebroadcasting would simply iterate mapWallet. As
    mapWallet is a std::unsorted_map, the child can sometimes come before the parent and thus
    be rebroadcast in the wrong order and fail the test.
    3405f3eed5
  65. achow101 force-pushed on Aug 29, 2022
  66. unknown approved
  67. naumenkogs commented at 6:19 am on August 30, 2022: member
    utACK 3405f3eed5cf841b23a569b64a376c2e5b5026cd
  68. in src/wallet/wallet.cpp:1938 in 10d91c5abe outdated
    1940 
    1941     // Do this infrequently and randomly to avoid giving away
    1942     // that these are our transactions.
    1943-    if (GetTime() < nNextResend || !fBroadcastTransactions) return;
    1944-    bool fFirst = (nNextResend == 0);
    1945+    if (!force && GetTime() < nNextResend) return;
    


    furszy commented at 12:38 pm on August 30, 2022:

    At this point of the review, this is just a minor nit but code would be a bit clearer in this way:

    0if (!force) {
    1    // During reindex, importing and IBD, old wallet transactions become
    2    // unconfirmed. Don't resend them as that would spam other nodes.
    3    // We only allow forcing mempool submission when not relaying to avoid this spam.
    4    if (relay && !chain().isReadyToBroadcast()) return;
    5
    6    // Do this infrequently and randomly to avoid giving away
    7    // that these are our transactions.
    8    if (GetTime() < nNextResend) return;
    9}
    

    stickies-v commented at 11:59 am on September 5, 2022:

    I think both these checks would be more appropriate in MaybeResendWalletTxs() since they’re more scheduling-like in nature. It would also address furszy’s comment them since we don’t need to check for force anymore (it’s always True there). If you do this, you probably also want to duplicate if (!fBroadcastTransactions) there since it’s a very cheap check and we could avoid iterating over the wallets.

    Unfortunately, we still need the force parameter when iterating over the transactions, otherwise ResubmitWalletTransactions()’s signature could have been simplified.

  69. furszy approved
  70. furszy commented at 12:56 pm on August 30, 2022: member

    Late code review ACK 3405f3ee

    The only orthogonal question that arises from my side is why are we trying to resubmit txes after every import RPC command. Can anything change at the, already created, wallet txes level after an import?

  71. in test/functional/wallet_resendwallettransactions.py:107 in 3405f3eed5
    103+
    104+        # Evict these txs from the mempool
    105+        evict_time = block_time + 60 * 60 * DEFAULT_MEMPOOL_EXPIRY_HOURS + 5
    106+        node.setmocktime(evict_time)
    107+        indep_send = node.send(outputs=[{node.getnewaddress(): 1}], options={"inputs": [indep_utxo]})
    108+        node.syncwithvalidationinterfacequeue()
    


    maflcko commented at 2:31 pm on August 30, 2022:
    Can you explain why this is needed? Generally this is only needed when forcing the wallet to receive all in-flight mempool notifications. However, there is no wallet call that follows. Also, there is no m_best_block_time update.

    achow101 commented at 5:22 pm on August 30, 2022:

    Also, there is no m_best_block_time update.

    Why not? IIRC this was done in order to update m_best_block_time so that the transaction would be properly evicted later.


    maflcko commented at 5:24 pm on August 30, 2022:

    The transaction is evicted from the mempool, not the wallet. (Only the wallet has an m_best_block_time)

    Also, there is no new best block, so m_best_block_time won’t be updated either way.


    achow101 commented at 5:43 pm on August 30, 2022:
    Oh, wrong line. I think this syncwithvalidationinterfacequeue is probably unneeded.
  72. in test/functional/wallet_resendwallettransactions.py:101 in 3405f3eed5
     97+        block_time = entry_time + 6 * 60
     98+        node.setmocktime(block_time)
     99+        block = create_block(int(node.getbestblockhash(), 16), create_coinbase(node.getblockcount() + 1), block_time)
    100+        block.solve()
    101+        node.submitblock(block.serialize().hex())
    102+        node.syncwithvalidationinterfacequeue()
    


    maflcko commented at 2:35 pm on August 30, 2022:
    Can you explain why this is needed? Maybe duplicate the comment “Set correct m_best_block_time” from above?
  73. in src/wallet/wallet.cpp:1923 in 3405f3eed5
    1922-void CWallet::ResendWalletTransactions()
    1923+//
    1924+// The `force` option results in all unconfirmed transactions being submitted to
    1925+// the mempool. This does not necessarily result in those transactions being relayed,
    1926+// that depends on the `relay` option. Periodic rebroadcast uses the pattern
    1927+// relay=true force=false (also the default values), while loading into the mempool
    


    maflcko commented at 2:39 pm on August 30, 2022:
    0// relay=true force=false, while loading into the mempool
    
  74. in src/wallet/wallet.cpp:1954 in 3405f3eed5
    1964+        std::set<CWalletTx*, WalletTxOrderComparator> to_submit;
    1965+        for (auto& [txid, wtx] : mapWallet) {
    1966+            // Only rebroadcast unconfirmed txs
    1967+            if (!wtx.isUnconfirmed()) continue;
    1968+
    1969+            // attempt to rebroadcast all txes more than 5 minutes older than
    


    maflcko commented at 2:40 pm on August 30, 2022:
    0            // Attempt to rebroadcast all txes more than 5 minutes older than
    

    Any reason that this A changed into a?

  75. maflcko commented at 2:40 pm on August 30, 2022: member
    left some minor nits, which can be addressed the next time this code is touched
  76. glozow requested review from stickies-v on Sep 1, 2022
  77. stickies-v commented at 12:00 pm on September 5, 2022: contributor
    ACK 3405f3eed5cf841b23a569b64a376c2e5b5026cd
  78. maflcko added this to the milestone 24.0 on Sep 5, 2022
  79. glozow merged this on Sep 5, 2022
  80. glozow closed this on Sep 5, 2022

  81. sidhujag referenced this in commit 66165db695 on Sep 5, 2022
  82. bitcoin deleted a comment on Sep 5, 2022
  83. maflcko referenced this in commit fa44670f9a on Sep 14, 2022
  84. maflcko referenced this in commit faa4916529 on Sep 14, 2022
  85. in src/wallet/rpc/backup.cpp:297 in 3405f3eed5
    294@@ -295,7 +295,7 @@ RPCHelpMan importaddress()
    295         RescanWallet(*pwallet, reserver);
    296         {
    297             LOCK(pwallet->cs_wallet);
    


    maflcko commented at 10:09 am on September 20, 2022:
    The locks are no longer needed and can be removed
  86. maflcko commented at 4:40 pm on September 20, 2022: member
    If a user calls importaddress every 6 hours, this will block any p2p relay from ever happening?
  87. stickies-v commented at 5:05 pm on September 29, 2022: contributor

    If a user calls importaddress every 6 hours, this will block any p2p relay from ever happening?

    I addressed that in #26205

  88. fanquake referenced this in commit aa6fb37acc on Oct 13, 2022
  89. sidhujag referenced this in commit 15e781feaf on Oct 23, 2022
  90. janus referenced this in commit d23ce23a94 on Jan 20, 2023
  91. bitcoin locked this on Nov 4, 2023

github-metadata-mirror

This is a metadata mirror of the GitHub repository bitcoin/bitcoin. This site is not affiliated with GitHub. Content is generated from a GitHub metadata backup.
generated: 2025-01-22 03:12 UTC

This site is hosted by @0xB10C
More mirrored repositories can be found on mirror.b10c.me