mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-25 02:33:24 -03:00
9de9c858d5
Increases test robustness by adding checks for orphanage size and presence of orphans in the orphanage
647 lines
32 KiB
Python
Executable file
647 lines
32 KiB
Python
Executable file
#!/usr/bin/env python3
|
|
# Copyright (c) 2023 The Bitcoin Core developers
|
|
# Distributed under the MIT software license, see the accompanying
|
|
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
import time
|
|
|
|
from test_framework.mempool_util import tx_in_orphanage
|
|
from test_framework.messages import (
|
|
CInv,
|
|
CTxInWitness,
|
|
MSG_TX,
|
|
MSG_WITNESS_TX,
|
|
MSG_WTX,
|
|
msg_getdata,
|
|
msg_inv,
|
|
msg_notfound,
|
|
msg_tx,
|
|
tx_from_hex,
|
|
)
|
|
from test_framework.p2p import (
|
|
GETDATA_TX_INTERVAL,
|
|
NONPREF_PEER_TX_DELAY,
|
|
OVERLOADED_PEER_TX_DELAY,
|
|
p2p_lock,
|
|
P2PInterface,
|
|
P2PTxInvStore,
|
|
TXID_RELAY_DELAY,
|
|
)
|
|
from test_framework.util import (
|
|
assert_equal,
|
|
)
|
|
from test_framework.test_framework import BitcoinTestFramework
|
|
from test_framework.wallet import (
|
|
MiniWallet,
|
|
MiniWalletMode,
|
|
)
|
|
|
|
# Time to bump forward (using setmocktime) before waiting for the node to send getdata(tx) in response
|
|
# to an inv(tx), in seconds. This delay includes all possible delays + 1, so it should only be used
|
|
# when the value of the delay is not interesting. If we want to test that the node waits x seconds
|
|
# for one peer and y seconds for another, use specific values instead.
|
|
TXREQUEST_TIME_SKIP = NONPREF_PEER_TX_DELAY + TXID_RELAY_DELAY + OVERLOADED_PEER_TX_DELAY + 1
|
|
|
|
DEFAULT_MAX_ORPHAN_TRANSACTIONS = 100
|
|
|
|
def cleanup(func):
|
|
# Time to fastfoward (using setmocktime) in between subtests to ensure they do not interfere with
|
|
# one another, in seconds. Equal to 12 hours, which is enough to expire anything that may exist
|
|
# (though nothing should since state should be cleared) in p2p data structures.
|
|
LONG_TIME_SKIP = 12 * 60 * 60
|
|
|
|
def wrapper(self):
|
|
try:
|
|
func(self)
|
|
finally:
|
|
# Clear mempool
|
|
self.generate(self.nodes[0], 1)
|
|
self.nodes[0].disconnect_p2ps()
|
|
self.nodes[0].bumpmocktime(LONG_TIME_SKIP)
|
|
return wrapper
|
|
|
|
class PeerTxRelayer(P2PTxInvStore):
|
|
"""A P2PTxInvStore that also remembers all of the getdata and tx messages it receives."""
|
|
def __init__(self):
|
|
super().__init__()
|
|
self._tx_received = []
|
|
self._getdata_received = []
|
|
|
|
@property
|
|
def tx_received(self):
|
|
with p2p_lock:
|
|
return self._tx_received
|
|
|
|
@property
|
|
def getdata_received(self):
|
|
with p2p_lock:
|
|
return self._getdata_received
|
|
|
|
def on_tx(self, message):
|
|
self._tx_received.append(message)
|
|
|
|
def on_getdata(self, message):
|
|
self._getdata_received.append(message)
|
|
|
|
def wait_for_parent_requests(self, txids):
|
|
"""Wait for requests for missing parents by txid with witness data (MSG_WITNESS_TX or
|
|
WitnessTx). Requires that the getdata message match these txids exactly; all txids must be
|
|
requested and no additional requests are allowed."""
|
|
def test_function():
|
|
last_getdata = self.last_message.get('getdata')
|
|
if not last_getdata:
|
|
return False
|
|
return len(last_getdata.inv) == len(txids) and all([item.type == MSG_WITNESS_TX and item.hash in txids for item in last_getdata.inv])
|
|
self.wait_until(test_function, timeout=10)
|
|
|
|
def assert_no_immediate_response(self, message):
|
|
"""Check that the node does not immediately respond to this message with any of getdata,
|
|
inv, tx. The node may respond later.
|
|
"""
|
|
prev_lastmessage = self.last_message
|
|
self.send_and_ping(message)
|
|
after_lastmessage = self.last_message
|
|
for msgtype in ["getdata", "inv", "tx"]:
|
|
if msgtype not in prev_lastmessage:
|
|
assert msgtype not in after_lastmessage
|
|
else:
|
|
assert_equal(prev_lastmessage[msgtype], after_lastmessage[msgtype])
|
|
|
|
def assert_never_requested(self, txhash):
|
|
"""Check that the node has never sent us a getdata for this hash (int type)"""
|
|
for getdata in self.getdata_received:
|
|
for request in getdata.inv:
|
|
assert request.hash != txhash
|
|
|
|
class OrphanHandlingTest(BitcoinTestFramework):
|
|
def set_test_params(self):
|
|
self.num_nodes = 1
|
|
self.extra_args = [[]]
|
|
|
|
def create_parent_and_child(self):
|
|
"""Create package with 1 parent and 1 child, normal fees (no cpfp)."""
|
|
parent = self.wallet.create_self_transfer()
|
|
child = self.wallet.create_self_transfer(utxo_to_spend=parent['new_utxo'])
|
|
return child["tx"].getwtxid(), child["tx"], parent["tx"]
|
|
|
|
def relay_transaction(self, peer, tx):
|
|
"""Relay transaction using MSG_WTX"""
|
|
wtxid = int(tx.getwtxid(), 16)
|
|
peer.send_and_ping(msg_inv([CInv(t=MSG_WTX, h=wtxid)]))
|
|
self.nodes[0].bumpmocktime(TXREQUEST_TIME_SKIP)
|
|
peer.wait_for_getdata([wtxid])
|
|
peer.send_and_ping(msg_tx(tx))
|
|
|
|
def create_malleated_version(self, tx):
|
|
"""
|
|
Create a malleated version of the tx where the witness is replaced with garbage data.
|
|
Returns a CTransaction object.
|
|
"""
|
|
tx_bad_wit = tx_from_hex(tx["hex"])
|
|
tx_bad_wit.wit.vtxinwit = [CTxInWitness()]
|
|
# Add garbage data to witness 0. We cannot simply strip the witness, as the node would
|
|
# classify it as a transaction in which the witness was missing rather than wrong.
|
|
tx_bad_wit.wit.vtxinwit[0].scriptWitness.stack = [b'garbage']
|
|
|
|
assert_equal(tx["txid"], tx_bad_wit.rehash())
|
|
assert tx["wtxid"] != tx_bad_wit.getwtxid()
|
|
|
|
return tx_bad_wit
|
|
|
|
@cleanup
|
|
def test_arrival_timing_orphan(self):
|
|
self.log.info("Test missing parents that arrive during delay are not requested")
|
|
node = self.nodes[0]
|
|
tx_parent_arrives = self.wallet.create_self_transfer()
|
|
tx_parent_doesnt_arrive = self.wallet.create_self_transfer()
|
|
# Fake orphan spends nonexistent outputs of the two parents
|
|
tx_fake_orphan = self.wallet.create_self_transfer_multi(utxos_to_spend=[
|
|
{"txid": tx_parent_doesnt_arrive["txid"], "vout": 10, "value": tx_parent_doesnt_arrive["new_utxo"]["value"]},
|
|
{"txid": tx_parent_arrives["txid"], "vout": 10, "value": tx_parent_arrives["new_utxo"]["value"]}
|
|
])
|
|
|
|
peer_spy = node.add_p2p_connection(PeerTxRelayer())
|
|
peer_normal = node.add_p2p_connection(PeerTxRelayer())
|
|
# This transaction is an orphan because it is missing inputs. It is a "fake" orphan that the
|
|
# spy peer has crafted to learn information about tx_parent_arrives even though it isn't
|
|
# able to spend a real output of it, but it could also just be a normal, real child tx.
|
|
# The node should not immediately respond with a request for orphan parents.
|
|
# Also, no request should be sent later because it will be resolved by
|
|
# the time the request is scheduled to be sent.
|
|
peer_spy.assert_no_immediate_response(msg_tx(tx_fake_orphan["tx"]))
|
|
|
|
# Node receives transaction. It attempts to obfuscate the exact timing at which this
|
|
# transaction entered its mempool. Send unsolicited because otherwise we need to wait for
|
|
# request delays.
|
|
peer_normal.send_and_ping(msg_tx(tx_parent_arrives["tx"]))
|
|
assert tx_parent_arrives["txid"] in node.getrawmempool()
|
|
|
|
# Spy peer should not be able to query the node for the parent yet, since it hasn't been
|
|
# announced / insufficient time has elapsed.
|
|
parent_inv = CInv(t=MSG_WTX, h=int(tx_parent_arrives["tx"].getwtxid(), 16))
|
|
assert_equal(len(peer_spy.get_invs()), 0)
|
|
peer_spy.assert_no_immediate_response(msg_getdata([parent_inv]))
|
|
|
|
# Request would be scheduled with this delay because it is not a preferred relay peer.
|
|
self.nodes[0].bumpmocktime(NONPREF_PEER_TX_DELAY)
|
|
peer_spy.assert_never_requested(int(tx_parent_arrives["txid"], 16))
|
|
peer_spy.assert_never_requested(int(tx_parent_doesnt_arrive["txid"], 16))
|
|
# Request would be scheduled with this delay because it is by txid.
|
|
self.nodes[0].bumpmocktime(TXID_RELAY_DELAY)
|
|
peer_spy.wait_for_parent_requests([int(tx_parent_doesnt_arrive["txid"], 16)])
|
|
peer_spy.assert_never_requested(int(tx_parent_arrives["txid"], 16))
|
|
|
|
@cleanup
|
|
def test_orphan_rejected_parents_exceptions(self):
|
|
node = self.nodes[0]
|
|
peer1 = node.add_p2p_connection(PeerTxRelayer())
|
|
peer2 = node.add_p2p_connection(PeerTxRelayer())
|
|
|
|
self.log.info("Test orphan handling when a nonsegwit parent is known to be invalid")
|
|
parent_low_fee_nonsegwit = self.wallet_nonsegwit.create_self_transfer(fee_rate=0)
|
|
assert_equal(parent_low_fee_nonsegwit["txid"], parent_low_fee_nonsegwit["tx"].getwtxid())
|
|
parent_other = self.wallet_nonsegwit.create_self_transfer()
|
|
child_nonsegwit = self.wallet_nonsegwit.create_self_transfer_multi(
|
|
utxos_to_spend=[parent_other["new_utxo"], parent_low_fee_nonsegwit["new_utxo"]])
|
|
|
|
# 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 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.
|
|
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)
|
|
peer1.assert_never_requested(int(parent_other["txid"], 16))
|
|
peer2.assert_never_requested(int(parent_other["txid"], 16))
|
|
peer2.assert_never_requested(int(parent_low_fee_nonsegwit["txid"], 16))
|
|
|
|
self.log.info("Test orphan handling when a segwit parent was invalid but may be retried with another witness")
|
|
parent_low_fee = self.wallet.create_self_transfer(fee_rate=0)
|
|
child_low_fee = self.wallet.create_self_transfer(utxo_to_spend=parent_low_fee["new_utxo"])
|
|
|
|
# Relay the low fee parent. It should not be accepted.
|
|
self.relay_transaction(peer1, parent_low_fee["tx"])
|
|
assert parent_low_fee["txid"] not in node.getrawmempool()
|
|
|
|
# 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.
|
|
self.nodes[0].bumpmocktime(NONPREF_PEER_TX_DELAY + TXID_RELAY_DELAY)
|
|
peer2.wait_for_getdata([int(parent_low_fee["tx"].rehash(), 16)])
|
|
|
|
self.log.info("Test orphan handling when a parent was previously downloaded with witness stripped")
|
|
parent_normal = self.wallet.create_self_transfer()
|
|
parent1_witness_stripped = tx_from_hex(parent_normal["tx"].serialize_without_witness().hex())
|
|
child_invalid_witness = self.wallet.create_self_transfer(utxo_to_spend=parent_normal["new_utxo"])
|
|
|
|
# Relay the parent with witness stripped. It should not be accepted.
|
|
self.relay_transaction(peer1, parent1_witness_stripped)
|
|
assert_equal(parent_normal["txid"], parent1_witness_stripped.rehash())
|
|
assert parent1_witness_stripped.rehash() not in node.getrawmempool()
|
|
|
|
# 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.
|
|
self.nodes[0].bumpmocktime(NONPREF_PEER_TX_DELAY + TXID_RELAY_DELAY)
|
|
peer2.wait_for_getdata([int(parent_normal["tx"].rehash(), 16)])
|
|
|
|
# parent_normal can be relayed again even though parent1_witness_stripped was rejected
|
|
self.relay_transaction(peer1, parent_normal["tx"])
|
|
assert_equal(set(node.getrawmempool()), set([parent_normal["txid"], child_invalid_witness["txid"]]))
|
|
|
|
@cleanup
|
|
def test_orphan_multiple_parents(self):
|
|
node = self.nodes[0]
|
|
peer = node.add_p2p_connection(PeerTxRelayer())
|
|
|
|
self.log.info("Test orphan parent requests with a mixture of confirmed, in-mempool and missing parents")
|
|
# This UTXO confirmed a long time ago.
|
|
utxo_conf_old = self.wallet.send_self_transfer(from_node=node)["new_utxo"]
|
|
txid_conf_old = utxo_conf_old["txid"]
|
|
self.generate(self.wallet, 10)
|
|
|
|
# Create a fake reorg to trigger BlockDisconnected, which resets the rolling bloom filter.
|
|
# The alternative is to mine thousands of transactions to push it out of the filter.
|
|
last_block = node.getbestblockhash()
|
|
node.invalidateblock(last_block)
|
|
node.preciousblock(last_block)
|
|
node.syncwithvalidationinterfacequeue()
|
|
|
|
# This UTXO confirmed recently.
|
|
utxo_conf_recent = self.wallet.send_self_transfer(from_node=node)["new_utxo"]
|
|
self.generate(node, 1)
|
|
|
|
# This UTXO is unconfirmed and in the mempool.
|
|
assert_equal(len(node.getrawmempool()), 0)
|
|
mempool_tx = self.wallet.send_self_transfer(from_node=node)
|
|
utxo_unconf_mempool = mempool_tx["new_utxo"]
|
|
|
|
# This UTXO is unconfirmed and missing.
|
|
missing_tx = self.wallet.create_self_transfer()
|
|
utxo_unconf_missing = missing_tx["new_utxo"]
|
|
assert missing_tx["txid"] not in node.getrawmempool()
|
|
|
|
orphan = self.wallet.create_self_transfer_multi(utxos_to_spend=[utxo_conf_old,
|
|
utxo_conf_recent, utxo_unconf_mempool, utxo_unconf_missing])
|
|
|
|
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)])
|
|
|
|
# Even though the peer would send a notfound for the "old" confirmed transaction, the node
|
|
# doesn't give up on the orphan. Once all of the missing parents are received, it should be
|
|
# submitted to mempool.
|
|
peer.send_message(msg_notfound(vec=[CInv(MSG_WITNESS_TX, int(txid_conf_old, 16))]))
|
|
# Sync with ping to ensure orphans are reconsidered
|
|
peer.send_and_ping(msg_tx(missing_tx["tx"]))
|
|
assert_equal(node.getmempoolentry(orphan["txid"])["ancestorcount"], 3)
|
|
|
|
@cleanup
|
|
def test_orphans_overlapping_parents(self):
|
|
node = self.nodes[0]
|
|
# In the process of relaying inflight_parent_AB
|
|
peer_txrequest = node.add_p2p_connection(PeerTxRelayer())
|
|
# Sends the orphans
|
|
peer_orphans = node.add_p2p_connection(PeerTxRelayer())
|
|
|
|
confirmed_utxos = [self.wallet_nonsegwit.get_utxo() for _ in range(4)]
|
|
assert all([utxo["confirmations"] > 0 for utxo in confirmed_utxos])
|
|
self.log.info("Test handling of multiple orphans with missing parents that are already being requested")
|
|
# Parent of child_A only
|
|
missing_parent_A = self.wallet_nonsegwit.create_self_transfer(utxo_to_spend=confirmed_utxos[0])
|
|
# Parents of child_A and child_B
|
|
missing_parent_AB = self.wallet_nonsegwit.create_self_transfer(utxo_to_spend=confirmed_utxos[1])
|
|
inflight_parent_AB = self.wallet_nonsegwit.create_self_transfer(utxo_to_spend=confirmed_utxos[2])
|
|
# Parent of child_B only
|
|
missing_parent_B = self.wallet_nonsegwit.create_self_transfer(utxo_to_spend=confirmed_utxos[3])
|
|
child_A = self.wallet_nonsegwit.create_self_transfer_multi(
|
|
utxos_to_spend=[missing_parent_A["new_utxo"], missing_parent_AB["new_utxo"], inflight_parent_AB["new_utxo"]]
|
|
)
|
|
child_B = self.wallet_nonsegwit.create_self_transfer_multi(
|
|
utxos_to_spend=[missing_parent_B["new_utxo"], missing_parent_AB["new_utxo"], inflight_parent_AB["new_utxo"]]
|
|
)
|
|
|
|
# The wtxid and txid need to be the same for the node to recognize that the missing input
|
|
# and in-flight request for inflight_parent_AB are the same transaction.
|
|
assert_equal(inflight_parent_AB["txid"], inflight_parent_AB["tx"].getwtxid())
|
|
|
|
# Announce inflight_parent_AB and wait for getdata
|
|
peer_txrequest.send_and_ping(msg_inv([CInv(t=MSG_WTX, h=int(inflight_parent_AB["tx"].getwtxid(), 16))]))
|
|
self.nodes[0].bumpmocktime(NONPREF_PEER_TX_DELAY)
|
|
peer_txrequest.wait_for_getdata([int(inflight_parent_AB["tx"].getwtxid(), 16)])
|
|
|
|
self.log.info("Test that the node does not request a parent if it has an in-flight txrequest")
|
|
# 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)])
|
|
|
|
self.log.info("Test that the node does not request a parent if it has an in-flight orphan parent request")
|
|
# 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)])
|
|
peer_orphans.assert_never_requested(int(inflight_parent_AB["txid"], 16))
|
|
|
|
@cleanup
|
|
def test_orphan_of_orphan(self):
|
|
node = self.nodes[0]
|
|
peer = node.add_p2p_connection(PeerTxRelayer())
|
|
|
|
self.log.info("Test handling of an orphan with a parent who is another orphan")
|
|
missing_grandparent = self.wallet_nonsegwit.create_self_transfer()
|
|
missing_parent_orphan = self.wallet_nonsegwit.create_self_transfer(utxo_to_spend=missing_grandparent["new_utxo"])
|
|
missing_parent = self.wallet_nonsegwit.create_self_transfer()
|
|
orphan = self.wallet_nonsegwit.create_self_transfer_multi(utxos_to_spend=[missing_parent["new_utxo"], missing_parent_orphan["new_utxo"]])
|
|
|
|
# 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
|
|
def test_orphan_inherit_rejection(self):
|
|
node = self.nodes[0]
|
|
peer1 = node.add_p2p_connection(PeerTxRelayer())
|
|
peer2 = node.add_p2p_connection(PeerTxRelayer())
|
|
peer3 = node.add_p2p_connection(PeerTxRelayer())
|
|
|
|
self.log.info("Test that an orphan with rejected parents, along with any descendants, cannot be retried with an alternate witness")
|
|
parent_low_fee_nonsegwit = self.wallet_nonsegwit.create_self_transfer(fee_rate=0)
|
|
assert_equal(parent_low_fee_nonsegwit["txid"], parent_low_fee_nonsegwit["tx"].getwtxid())
|
|
child = self.wallet.create_self_transfer(utxo_to_spend=parent_low_fee_nonsegwit["new_utxo"])
|
|
grandchild = self.wallet.create_self_transfer(utxo_to_spend=child["new_utxo"])
|
|
assert child["txid"] != child["tx"].getwtxid()
|
|
assert grandchild["txid"] != grandchild["tx"].getwtxid()
|
|
|
|
# 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.
|
|
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.
|
|
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())
|
|
|
|
# The child should never be requested, even if announced again with potentially different witness.
|
|
# Sync with ping to ensure orphans are reconsidered
|
|
peer3.send_and_ping(msg_inv([CInv(t=MSG_TX, h=int(child["txid"], 16))]))
|
|
self.nodes[0].bumpmocktime(TXREQUEST_TIME_SKIP)
|
|
peer3.assert_never_requested(child["txid"])
|
|
|
|
@cleanup
|
|
def test_same_txid_orphan(self):
|
|
self.log.info("Check what happens when orphan with same txid is already in orphanage")
|
|
node = self.nodes[0]
|
|
|
|
tx_parent = self.wallet.create_self_transfer()
|
|
|
|
# Create the real child
|
|
tx_child = self.wallet.create_self_transfer(utxo_to_spend=tx_parent["new_utxo"])
|
|
|
|
# Create a fake version of the child
|
|
tx_orphan_bad_wit = self.create_malleated_version(tx_child)
|
|
|
|
bad_peer = node.add_p2p_connection(P2PInterface())
|
|
honest_peer = node.add_p2p_connection(P2PInterface())
|
|
|
|
# 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)
|
|
node.bumpmocktime(NONPREF_PEER_TX_DELAY + TXID_RELAY_DELAY)
|
|
bad_peer.wait_for_getdata([parent_txid_int])
|
|
|
|
# 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"]):
|
|
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)
|
|
node.bumpmocktime(GETDATA_TX_INTERVAL)
|
|
|
|
# 4. The parent is requested. Honest peer sends it.
|
|
honest_peer.wait_for_getdata([parent_txid_int])
|
|
# Sync with ping to ensure orphans are reconsidered
|
|
honest_peer.send_and_ping(msg_tx(tx_parent["tx"]))
|
|
|
|
# 5. After parent is accepted, orphans should be reconsidered.
|
|
# The real child should be accepted and the fake one rejected.
|
|
node_mempool = node.getrawmempool()
|
|
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"])
|
|
|
|
@cleanup
|
|
def test_same_txid_orphan_of_orphan(self):
|
|
self.log.info("Check what happens when orphan's parent with same txid is already in orphanage")
|
|
node = self.nodes[0]
|
|
|
|
tx_grandparent = self.wallet.create_self_transfer()
|
|
|
|
# Create middle tx (both parent and child) which will be in orphanage.
|
|
tx_middle = self.wallet.create_self_transfer(utxo_to_spend=tx_grandparent["new_utxo"])
|
|
|
|
# Create a fake version of the middle tx
|
|
tx_orphan_bad_wit = self.create_malleated_version(tx_middle)
|
|
|
|
# Create grandchild spending from tx_middle (and spending from tx_orphan_bad_wit since they
|
|
# have the same txid).
|
|
tx_grandchild = self.wallet.create_self_transfer(utxo_to_spend=tx_middle["new_utxo"])
|
|
|
|
bad_peer = node.add_p2p_connection(P2PInterface())
|
|
honest_peer = node.add_p2p_connection(P2PInterface())
|
|
|
|
# 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)
|
|
node.bumpmocktime(NONPREF_PEER_TX_DELAY + TXID_RELAY_DELAY)
|
|
bad_peer.wait_for_getdata([grandparent_txid_int])
|
|
|
|
# 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!
|
|
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.
|
|
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
|
|
honest_peer.send_and_ping(msg_tx(tx_grandparent["tx"]))
|
|
|
|
# 6. After parent is accepted, orphans should be reconsidered.
|
|
# The real child should be accepted and the fake one rejected.
|
|
node_mempool = node.getrawmempool()
|
|
assert tx_grandparent["txid"] in node_mempool
|
|
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):
|
|
self.log.info("Check node does not ignore announcement with same txid as tx in orphanage")
|
|
node = self.nodes[0]
|
|
|
|
tx_parent = self.wallet.create_self_transfer()
|
|
|
|
# Create the real child and fake version
|
|
tx_child = self.wallet.create_self_transfer(utxo_to_spend=tx_parent["new_utxo"])
|
|
tx_orphan_bad_wit = self.create_malleated_version(tx_child)
|
|
|
|
bad_peer = node.add_p2p_connection(PeerTxRelayer())
|
|
# Must not send wtxidrelay because otherwise the inv(TX) will be ignored later
|
|
honest_peer = node.add_p2p_connection(P2PInterface(wtxidrelay=False))
|
|
|
|
# 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)
|
|
node.bumpmocktime(NONPREF_PEER_TX_DELAY + TXID_RELAY_DELAY)
|
|
bad_peer.wait_for_getdata([parent_txid_int])
|
|
|
|
# 3. Honest peer announces the real child, by txid (this isn't common but the node should
|
|
# still keep track of it).
|
|
child_txid_int = int(tx_child["txid"], 16)
|
|
honest_peer.send_and_ping(msg_inv([CInv(t=MSG_TX, h=child_txid_int)]))
|
|
|
|
# 4. The child is requested. Honest peer sends it.
|
|
node.bumpmocktime(TXREQUEST_TIME_SKIP)
|
|
honest_peer.wait_for_getdata([child_txid_int])
|
|
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.
|
|
node.bumpmocktime(GETDATA_TX_INTERVAL)
|
|
honest_peer.wait_for_getdata([parent_txid_int])
|
|
honest_peer.send_and_ping(msg_tx(tx_parent["tx"]))
|
|
|
|
# 6. After parent is accepted, orphans should be reconsidered.
|
|
# The real child should be accepted and the fake one rejected. This may happen in either
|
|
# order since the message-processing is randomized. If tx_orphan_bad_wit is validated first,
|
|
# its consensus error leads to disconnection of bad_peer. If tx_child is validated first,
|
|
# tx_orphan_bad_wit is rejected for txn-same-nonwitness-data-in-mempool (no punishment).
|
|
node_mempool = node.getrawmempool()
|
|
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):
|
|
self.log.info("Check that we never exceed our storage limits for orphans")
|
|
|
|
node = self.nodes[0]
|
|
self.generate(self.wallet, 1)
|
|
peer_1 = node.add_p2p_connection(P2PInterface())
|
|
|
|
self.log.info("Check that orphanage is empty on start of test")
|
|
assert len(node.getorphantxs()) == 0
|
|
|
|
self.log.info("Filling up orphanage with " + str(DEFAULT_MAX_ORPHAN_TRANSACTIONS) + "(DEFAULT_MAX_ORPHAN_TRANSACTIONS) orphans")
|
|
orphans = []
|
|
parent_orphans = []
|
|
for _ in range(DEFAULT_MAX_ORPHAN_TRANSACTIONS):
|
|
tx_parent_1 = self.wallet.create_self_transfer()
|
|
tx_child_1 = self.wallet.create_self_transfer(utxo_to_spend=tx_parent_1["new_utxo"])
|
|
parent_orphans.append(tx_parent_1["tx"])
|
|
orphans.append(tx_child_1["tx"])
|
|
peer_1.send_message(msg_tx(tx_child_1["tx"]))
|
|
|
|
peer_1.sync_with_ping()
|
|
orphanage = node.getorphantxs()
|
|
assert_equal(len(orphanage), DEFAULT_MAX_ORPHAN_TRANSACTIONS)
|
|
|
|
for orphan in orphans:
|
|
assert tx_in_orphanage(node, orphan)
|
|
|
|
self.log.info("Check that we do not add more than the max orphan amount")
|
|
tx_parent_1 = self.wallet.create_self_transfer()
|
|
tx_child_1 = self.wallet.create_self_transfer(utxo_to_spend=tx_parent_1["new_utxo"])
|
|
peer_1.send_and_ping(msg_tx(tx_child_1["tx"]))
|
|
parent_orphans.append(tx_parent_1["tx"])
|
|
orphanage = node.getorphantxs()
|
|
assert_equal(len(orphanage), DEFAULT_MAX_ORPHAN_TRANSACTIONS)
|
|
|
|
self.log.info("Clearing the orphanage")
|
|
for index, parent_orphan in enumerate(parent_orphans):
|
|
peer_1.send_and_ping(msg_tx(parent_orphan))
|
|
assert_equal(len(node.getorphantxs()),0)
|
|
|
|
|
|
def run_test(self):
|
|
self.nodes[0].setmocktime(int(time.time()))
|
|
self.wallet_nonsegwit = MiniWallet(self.nodes[0], mode=MiniWalletMode.RAW_P2PK)
|
|
self.generate(self.wallet_nonsegwit, 10)
|
|
self.wallet = MiniWallet(self.nodes[0])
|
|
self.generate(self.wallet, 160)
|
|
self.test_arrival_timing_orphan()
|
|
self.test_orphan_rejected_parents_exceptions()
|
|
self.test_orphan_multiple_parents()
|
|
self.test_orphans_overlapping_parents()
|
|
self.test_orphan_of_orphan()
|
|
self.test_orphan_inherit_rejection()
|
|
self.test_same_txid_orphan()
|
|
self.test_same_txid_orphan_of_orphan()
|
|
self.test_orphan_txid_inv()
|
|
self.test_max_orphan_amount()
|
|
|
|
|
|
if __name__ == '__main__':
|
|
OrphanHandlingTest(__file__).main()
|