From 33af14b62e47d8ef145575882b3efe1a7d7aa9a8 Mon Sep 17 00:00:00 2001 From: tdb3 <106488469+tdb3@users.noreply.github.com> Date: Sat, 5 Oct 2024 13:00:23 -0400 Subject: [PATCH 1/2] test: reduce assert_debug_log reliance p2p_orphan_handling now uses tx_in_orphanage to more directly check for inclusion/exclusion in the orphanage. --- test/functional/p2p_orphan_handling.py | 27 +++++++++++++------------- 1 file changed, 14 insertions(+), 13 deletions(-) diff --git a/test/functional/p2p_orphan_handling.py b/test/functional/p2p_orphan_handling.py index 9ec23e9d6eb..db7f079d6d8 100755 --- a/test/functional/p2p_orphan_handling.py +++ b/test/functional/p2p_orphan_handling.py @@ -210,9 +210,9 @@ class OrphanHandlingTest(BitcoinTestFramework): # Relay the child. It should not be accepted because it has missing inputs. # Its parent should not be requested because its hash (txid == wtxid) has been added to the rejection filter. - with node.assert_debug_log(['not keeping orphan with rejected parents {}'.format(child_nonsegwit["txid"])]): - self.relay_transaction(peer2, child_nonsegwit["tx"]) + self.relay_transaction(peer2, child_nonsegwit["tx"]) assert child_nonsegwit["txid"] not in node.getrawmempool() + assert not tx_in_orphanage(node, child_nonsegwit["tx"]) # No parents are requested. self.nodes[0].bumpmocktime(GETDATA_TX_INTERVAL) @@ -402,15 +402,15 @@ class OrphanHandlingTest(BitcoinTestFramework): # Relay the child. It should be rejected for having missing parents, and this rejection is # cached by txid and wtxid. - with node.assert_debug_log(['not keeping orphan with rejected parents {}'.format(child["txid"])]): - self.relay_transaction(peer1, child["tx"]) + self.relay_transaction(peer1, child["tx"]) assert_equal(0, len(node.getrawmempool())) + assert not tx_in_orphanage(node, child["tx"]) peer1.assert_never_requested(parent_low_fee_nonsegwit["txid"]) # Grandchild should also not be kept in orphanage because its parent has been rejected. - with node.assert_debug_log(['not keeping orphan with rejected parents {}'.format(grandchild["txid"])]): - self.relay_transaction(peer2, grandchild["tx"]) + self.relay_transaction(peer2, grandchild["tx"]) assert_equal(0, len(node.getrawmempool())) + assert not tx_in_orphanage(node, grandchild["tx"]) peer2.assert_never_requested(child["txid"]) peer2.assert_never_requested(child["tx"].getwtxid()) @@ -446,8 +446,9 @@ class OrphanHandlingTest(BitcoinTestFramework): # 3. Honest peer relays the real child, which is also missing parents and should be placed # in the orphanage. - with node.assert_debug_log(["missingorspent", "stored orphan tx"]): + with node.assert_debug_log(["missingorspent"]): honest_peer.send_and_ping(msg_tx(tx_child["tx"])) + assert tx_in_orphanage(node, tx_child["tx"]) # Time out the previous request for the parent (node will not request the same transaction # from multiple nodes at the same time) @@ -496,16 +497,16 @@ class OrphanHandlingTest(BitcoinTestFramework): # 3. Honest peer relays the grandchild, which is missing a parent. The parent by txid already # exists in orphanage, but should be re-requested because the node shouldn't assume that the # witness data is the same. In this case, a same-txid-different-witness transaction exists! - with node.assert_debug_log(["stored orphan tx"]): - honest_peer.send_and_ping(msg_tx(tx_grandchild["tx"])) + honest_peer.send_and_ping(msg_tx(tx_grandchild["tx"])) + assert tx_in_orphanage(node, tx_grandchild["tx"]) middle_txid_int = int(tx_middle["txid"], 16) node.bumpmocktime(NONPREF_PEER_TX_DELAY + TXID_RELAY_DELAY) honest_peer.wait_for_getdata([middle_txid_int]) # 4. Honest peer relays the real child, which is also missing parents and should be placed # in the orphanage. - with node.assert_debug_log(["stored orphan tx"]): - honest_peer.send_and_ping(msg_tx(tx_middle["tx"])) + honest_peer.send_and_ping(msg_tx(tx_middle["tx"])) + assert tx_in_orphanage(node, tx_middle["tx"]) assert_equal(len(node.getrawmempool()), 0) # 5. Honest peer sends tx_grandparent @@ -550,8 +551,8 @@ class OrphanHandlingTest(BitcoinTestFramework): # 4. The child is requested. Honest peer sends it. node.bumpmocktime(TXREQUEST_TIME_SKIP) honest_peer.wait_for_getdata([child_txid_int]) - with node.assert_debug_log(["stored orphan tx"]): - honest_peer.send_and_ping(msg_tx(tx_child["tx"])) + honest_peer.send_and_ping(msg_tx(tx_child["tx"])) + assert tx_in_orphanage(node, tx_child["tx"]) # 5. After first parent request times out, the node sends another one for the missing parent # of the real orphan child. From 9de9c858d5aa412e1c6086e564fbe85984a4f2d5 Mon Sep 17 00:00:00 2001 From: tdb3 <106488469+tdb3@users.noreply.github.com> Date: Sat, 5 Oct 2024 13:01:04 -0400 Subject: [PATCH 2/2] test: enhance p2p_orphan_handling Increases test robustness by adding checks for orphanage size and presence of orphans in the orphanage --- test/functional/p2p_orphan_handling.py | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/test/functional/p2p_orphan_handling.py b/test/functional/p2p_orphan_handling.py index db7f079d6d8..bcb4dd7f8db 100755 --- a/test/functional/p2p_orphan_handling.py +++ b/test/functional/p2p_orphan_handling.py @@ -231,6 +231,7 @@ class OrphanHandlingTest(BitcoinTestFramework): # Relay the child. It should not be accepted because it has missing inputs. self.relay_transaction(peer2, child_low_fee["tx"]) assert child_low_fee["txid"] not in node.getrawmempool() + assert tx_in_orphanage(node, child_low_fee["tx"]) # The parent should be requested because even though the txid commits to the fee, it doesn't # commit to the feerate. Delayed because it's by txid and this is not a preferred relay peer. @@ -250,6 +251,7 @@ class OrphanHandlingTest(BitcoinTestFramework): # Relay the child. It should not be accepted because it has missing inputs. self.relay_transaction(peer2, child_invalid_witness["tx"]) assert child_invalid_witness["txid"] not in node.getrawmempool() + assert tx_in_orphanage(node, child_invalid_witness["tx"]) # The parent should be requested since the unstripped wtxid would differ. Delayed because # it's by txid and this is not a preferred relay peer. @@ -298,6 +300,7 @@ class OrphanHandlingTest(BitcoinTestFramework): self.relay_transaction(peer, orphan["tx"]) self.nodes[0].bumpmocktime(NONPREF_PEER_TX_DELAY + TXID_RELAY_DELAY) peer.sync_with_ping() + assert tx_in_orphanage(node, orphan["tx"]) assert_equal(len(peer.last_message["getdata"].inv), 2) peer.wait_for_parent_requests([int(txid_conf_old, 16), int(missing_tx["txid"], 16)]) @@ -347,6 +350,7 @@ class OrphanHandlingTest(BitcoinTestFramework): # Relay orphan child_A self.relay_transaction(peer_orphans, child_A["tx"]) self.nodes[0].bumpmocktime(NONPREF_PEER_TX_DELAY + TXID_RELAY_DELAY) + assert tx_in_orphanage(node, child_A["tx"]) # There are 3 missing parents. missing_parent_A and missing_parent_AB should be requested. # But inflight_parent_AB should not, because there is already an in-flight request for it. peer_orphans.wait_for_parent_requests([int(missing_parent_A["txid"], 16), int(missing_parent_AB["txid"], 16)]) @@ -355,6 +359,7 @@ class OrphanHandlingTest(BitcoinTestFramework): # Relay orphan child_B self.relay_transaction(peer_orphans, child_B["tx"]) self.nodes[0].bumpmocktime(NONPREF_PEER_TX_DELAY + TXID_RELAY_DELAY) + assert tx_in_orphanage(node, child_B["tx"]) # Only missing_parent_B should be requested. Not inflight_parent_AB or missing_parent_AB # because they are already being requested from peer_txrequest and peer_orphans respectively. peer_orphans.wait_for_parent_requests([int(missing_parent_B["txid"], 16)]) @@ -374,12 +379,14 @@ class OrphanHandlingTest(BitcoinTestFramework): # The node should put missing_parent_orphan into the orphanage and request missing_grandparent self.relay_transaction(peer, missing_parent_orphan["tx"]) self.nodes[0].bumpmocktime(NONPREF_PEER_TX_DELAY + TXID_RELAY_DELAY) + assert tx_in_orphanage(node, missing_parent_orphan["tx"]) peer.wait_for_parent_requests([int(missing_grandparent["txid"], 16)]) # The node should put the orphan into the orphanage and request missing_parent, skipping # missing_parent_orphan because it already has it in the orphanage. self.relay_transaction(peer, orphan["tx"]) self.nodes[0].bumpmocktime(NONPREF_PEER_TX_DELAY + TXID_RELAY_DELAY) + assert tx_in_orphanage(node, orphan["tx"]) peer.wait_for_parent_requests([int(missing_parent["txid"], 16)]) @cleanup @@ -399,6 +406,7 @@ class OrphanHandlingTest(BitcoinTestFramework): # Relay the parent. It should be rejected because it pays 0 fees. self.relay_transaction(peer1, parent_low_fee_nonsegwit["tx"]) + assert parent_low_fee_nonsegwit["txid"] not in node.getrawmempool() # Relay the child. It should be rejected for having missing parents, and this rejection is # cached by txid and wtxid. @@ -438,6 +446,7 @@ class OrphanHandlingTest(BitcoinTestFramework): # 1. Fake orphan is received first. It is missing an input. bad_peer.send_and_ping(msg_tx(tx_orphan_bad_wit)) + assert tx_in_orphanage(node, tx_orphan_bad_wit) # 2. Node requests the missing parent by txid. parent_txid_int = int(tx_parent["txid"], 16) @@ -488,6 +497,7 @@ class OrphanHandlingTest(BitcoinTestFramework): # 1. Fake orphan is received first. It is missing an input. bad_peer.send_and_ping(msg_tx(tx_orphan_bad_wit)) + assert tx_in_orphanage(node, tx_orphan_bad_wit) # 2. Node requests missing tx_grandparent by txid. grandparent_txid_int = int(tx_grandparent["txid"], 16) @@ -519,6 +529,7 @@ class OrphanHandlingTest(BitcoinTestFramework): assert tx_middle["txid"] in node_mempool assert tx_grandchild["txid"] in node_mempool assert_equal(node.getmempoolentry(tx_middle["txid"])["wtxid"], tx_middle["wtxid"]) + assert_equal(len(node.getorphantxs()), 0) @cleanup def test_orphan_txid_inv(self): @@ -537,6 +548,7 @@ class OrphanHandlingTest(BitcoinTestFramework): # 1. Fake orphan is received first. It is missing an input. bad_peer.send_and_ping(msg_tx(tx_orphan_bad_wit)) + assert tx_in_orphanage(node, tx_orphan_bad_wit) # 2. Node requests the missing parent by txid. parent_txid_int = int(tx_parent["txid"], 16) @@ -569,6 +581,7 @@ class OrphanHandlingTest(BitcoinTestFramework): assert tx_parent["txid"] in node_mempool assert tx_child["txid"] in node_mempool assert_equal(node.getmempoolentry(tx_child["txid"])["wtxid"], tx_child["wtxid"]) + assert_equal(len(node.getorphantxs()), 0) @cleanup def test_max_orphan_amount(self):