|
|
@ -4,6 +4,7 @@
|
|
|
|
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
"""Test mempool acceptance of raw transactions."""
|
|
|
|
"""Test mempool acceptance of raw transactions."""
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
from copy import deepcopy
|
|
|
|
from decimal import Decimal
|
|
|
|
from decimal import Decimal
|
|
|
|
import math
|
|
|
|
import math
|
|
|
|
|
|
|
|
|
|
|
@ -34,6 +35,7 @@ from test_framework.util import (
|
|
|
|
assert_equal,
|
|
|
|
assert_equal,
|
|
|
|
assert_raises_rpc_error,
|
|
|
|
assert_raises_rpc_error,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
from test_framework.wallet import MiniWallet
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
@ -44,9 +46,6 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
]] * self.num_nodes
|
|
|
|
]] * self.num_nodes
|
|
|
|
self.supports_cli = False
|
|
|
|
self.supports_cli = False
|
|
|
|
|
|
|
|
|
|
|
|
def skip_test_if_missing_module(self):
|
|
|
|
|
|
|
|
self.skip_if_no_wallet()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def check_mempool_result(self, result_expected, *args, **kwargs):
|
|
|
|
def check_mempool_result(self, result_expected, *args, **kwargs):
|
|
|
|
"""Wrapper to check result of testmempoolaccept on node_0's mempool"""
|
|
|
|
"""Wrapper to check result of testmempoolaccept on node_0's mempool"""
|
|
|
|
result_test = self.nodes[0].testmempoolaccept(*args, **kwargs)
|
|
|
|
result_test = self.nodes[0].testmempoolaccept(*args, **kwargs)
|
|
|
@ -57,12 +56,13 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
|
|
|
|
|
|
|
|
def run_test(self):
|
|
|
|
def run_test(self):
|
|
|
|
node = self.nodes[0]
|
|
|
|
node = self.nodes[0]
|
|
|
|
|
|
|
|
self.wallet = MiniWallet(node)
|
|
|
|
|
|
|
|
self.wallet.rescan_utxos()
|
|
|
|
|
|
|
|
|
|
|
|
self.log.info('Start with empty mempool, and 200 blocks')
|
|
|
|
self.log.info('Start with empty mempool, and 200 blocks')
|
|
|
|
self.mempool_size = 0
|
|
|
|
self.mempool_size = 0
|
|
|
|
assert_equal(node.getblockcount(), 200)
|
|
|
|
assert_equal(node.getblockcount(), 200)
|
|
|
|
assert_equal(node.getmempoolinfo()['size'], self.mempool_size)
|
|
|
|
assert_equal(node.getmempoolinfo()['size'], self.mempool_size)
|
|
|
|
coins = node.listunspent()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
self.log.info('Should not accept garbage to testmempoolaccept')
|
|
|
|
self.log.info('Should not accept garbage to testmempoolaccept')
|
|
|
|
assert_raises_rpc_error(-3, 'Expected type array, got string', lambda: node.testmempoolaccept(rawtxs='ff00baar'))
|
|
|
|
assert_raises_rpc_error(-3, 'Expected type array, got string', lambda: node.testmempoolaccept(rawtxs='ff00baar'))
|
|
|
@ -71,12 +71,12 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
assert_raises_rpc_error(-22, 'TX decode failed', lambda: node.testmempoolaccept(rawtxs=['ff00baar']))
|
|
|
|
assert_raises_rpc_error(-22, 'TX decode failed', lambda: node.testmempoolaccept(rawtxs=['ff00baar']))
|
|
|
|
|
|
|
|
|
|
|
|
self.log.info('A transaction already in the blockchain')
|
|
|
|
self.log.info('A transaction already in the blockchain')
|
|
|
|
coin = coins.pop() # Pick a random coin(base) to spend
|
|
|
|
tx = self.wallet.create_self_transfer()['tx'] # Pick a random coin(base) to spend
|
|
|
|
raw_tx_in_block = node.signrawtransactionwithwallet(node.createrawtransaction(
|
|
|
|
tx.vout.append(deepcopy(tx.vout[0]))
|
|
|
|
inputs=[{'txid': coin['txid'], 'vout': coin['vout']}],
|
|
|
|
tx.vout[0].nValue = int(0.3 * COIN)
|
|
|
|
outputs=[{node.getnewaddress(): 0.3}, {node.getnewaddress(): 49}],
|
|
|
|
tx.vout[1].nValue = int(49 * COIN)
|
|
|
|
))['hex']
|
|
|
|
raw_tx_in_block = tx.serialize().hex()
|
|
|
|
txid_in_block = node.sendrawtransaction(hexstring=raw_tx_in_block, maxfeerate=0)
|
|
|
|
txid_in_block = self.wallet.sendrawtransaction(from_node=node, tx_hex=raw_tx_in_block, maxfeerate=0)
|
|
|
|
self.generate(node, 1)
|
|
|
|
self.generate(node, 1)
|
|
|
|
self.mempool_size = 0
|
|
|
|
self.mempool_size = 0
|
|
|
|
self.check_mempool_result(
|
|
|
|
self.check_mempool_result(
|
|
|
@ -86,11 +86,10 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
|
|
|
|
|
|
|
|
self.log.info('A transaction not in the mempool')
|
|
|
|
self.log.info('A transaction not in the mempool')
|
|
|
|
fee = Decimal('0.000007')
|
|
|
|
fee = Decimal('0.000007')
|
|
|
|
raw_tx_0 = node.signrawtransactionwithwallet(node.createrawtransaction(
|
|
|
|
utxo_to_spend = self.wallet.get_utxo(txid=txid_in_block) # use 0.3 BTC UTXO
|
|
|
|
inputs=[{"txid": txid_in_block, "vout": 0, "sequence": BIP125_SEQUENCE_NUMBER}], # RBF is used later
|
|
|
|
tx = self.wallet.create_self_transfer(utxo_to_spend=utxo_to_spend, sequence=BIP125_SEQUENCE_NUMBER)['tx']
|
|
|
|
outputs=[{node.getnewaddress(): Decimal('0.3') - fee}],
|
|
|
|
tx.vout[0].nValue = int((Decimal('0.3') - fee) * COIN)
|
|
|
|
))['hex']
|
|
|
|
raw_tx_0 = tx.serialize().hex()
|
|
|
|
tx = tx_from_hex(raw_tx_0)
|
|
|
|
|
|
|
|
txid_0 = tx.rehash()
|
|
|
|
txid_0 = tx.rehash()
|
|
|
|
self.check_mempool_result(
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': txid_0, 'allowed': True, 'vsize': tx.get_vsize(), 'fees': {'base': fee}}],
|
|
|
|
result_expected=[{'txid': txid_0, 'allowed': True, 'vsize': tx.get_vsize(), 'fees': {'base': fee}}],
|
|
|
@ -98,15 +97,15 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
self.log.info('A final transaction not in the mempool')
|
|
|
|
self.log.info('A final transaction not in the mempool')
|
|
|
|
coin = coins.pop() # Pick a random coin(base) to spend
|
|
|
|
|
|
|
|
output_amount = Decimal('0.025')
|
|
|
|
output_amount = Decimal('0.025')
|
|
|
|
raw_tx_final = node.signrawtransactionwithwallet(node.createrawtransaction(
|
|
|
|
tx = self.wallet.create_self_transfer(
|
|
|
|
inputs=[{'txid': coin['txid'], 'vout': coin['vout'], "sequence": SEQUENCE_FINAL}],
|
|
|
|
sequence=SEQUENCE_FINAL,
|
|
|
|
outputs=[{node.getnewaddress(): output_amount}],
|
|
|
|
|
|
|
|
locktime=node.getblockcount() + 2000, # Can be anything
|
|
|
|
locktime=node.getblockcount() + 2000, # Can be anything
|
|
|
|
))['hex']
|
|
|
|
)['tx']
|
|
|
|
|
|
|
|
tx.vout[0].nValue = int(output_amount * COIN)
|
|
|
|
|
|
|
|
raw_tx_final = tx.serialize().hex()
|
|
|
|
tx = tx_from_hex(raw_tx_final)
|
|
|
|
tx = tx_from_hex(raw_tx_final)
|
|
|
|
fee_expected = coin['amount'] - output_amount
|
|
|
|
fee_expected = Decimal('50.0') - output_amount
|
|
|
|
self.check_mempool_result(
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': True, 'vsize': tx.get_vsize(), 'fees': {'base': fee_expected}}],
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': True, 'vsize': tx.get_vsize(), 'fees': {'base': fee_expected}}],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
@ -127,8 +126,7 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
tx = tx_from_hex(raw_tx_0)
|
|
|
|
tx = tx_from_hex(raw_tx_0)
|
|
|
|
tx.vout[0].nValue -= int(fee * COIN) # Double the fee
|
|
|
|
tx.vout[0].nValue -= int(fee * COIN) # Double the fee
|
|
|
|
tx.vin[0].nSequence = BIP125_SEQUENCE_NUMBER + 1 # Now, opt out of RBF
|
|
|
|
tx.vin[0].nSequence = BIP125_SEQUENCE_NUMBER + 1 # Now, opt out of RBF
|
|
|
|
raw_tx_0 = node.signrawtransactionwithwallet(tx.serialize().hex())['hex']
|
|
|
|
raw_tx_0 = tx.serialize().hex()
|
|
|
|
tx = tx_from_hex(raw_tx_0)
|
|
|
|
|
|
|
|
txid_0 = tx.rehash()
|
|
|
|
txid_0 = tx.rehash()
|
|
|
|
self.check_mempool_result(
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': txid_0, 'allowed': True, 'vsize': tx.get_vsize(), 'fees': {'base': (2 * fee)}}],
|
|
|
|
result_expected=[{'txid': txid_0, 'allowed': True, 'vsize': tx.get_vsize(), 'fees': {'base': (2 * fee)}}],
|
|
|
@ -141,7 +139,6 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
# take original raw_tx_0
|
|
|
|
# take original raw_tx_0
|
|
|
|
tx = tx_from_hex(raw_tx_0)
|
|
|
|
tx = tx_from_hex(raw_tx_0)
|
|
|
|
tx.vout[0].nValue -= int(4 * fee * COIN) # Set more fee
|
|
|
|
tx.vout[0].nValue -= int(4 * fee * COIN) # Set more fee
|
|
|
|
# skip re-signing the tx
|
|
|
|
|
|
|
|
self.check_mempool_result(
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'txn-mempool-conflict'}],
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'txn-mempool-conflict'}],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
@ -151,7 +148,6 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
self.log.info('A transaction with missing inputs, that never existed')
|
|
|
|
self.log.info('A transaction with missing inputs, that never existed')
|
|
|
|
tx = tx_from_hex(raw_tx_0)
|
|
|
|
tx = tx_from_hex(raw_tx_0)
|
|
|
|
tx.vin[0].prevout = COutPoint(hash=int('ff' * 32, 16), n=14)
|
|
|
|
tx.vin[0].prevout = COutPoint(hash=int('ff' * 32, 16), n=14)
|
|
|
|
# skip re-signing the tx
|
|
|
|
|
|
|
|
self.check_mempool_result(
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'missing-inputs'}],
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'missing-inputs'}],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
@ -160,17 +156,17 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
self.log.info('A transaction with missing inputs, that existed once in the past')
|
|
|
|
self.log.info('A transaction with missing inputs, that existed once in the past')
|
|
|
|
tx = tx_from_hex(raw_tx_0)
|
|
|
|
tx = tx_from_hex(raw_tx_0)
|
|
|
|
tx.vin[0].prevout.n = 1 # Set vout to 1, to spend the other outpoint (49 coins) of the in-chain-tx we want to double spend
|
|
|
|
tx.vin[0].prevout.n = 1 # Set vout to 1, to spend the other outpoint (49 coins) of the in-chain-tx we want to double spend
|
|
|
|
raw_tx_1 = node.signrawtransactionwithwallet(tx.serialize().hex())['hex']
|
|
|
|
raw_tx_1 = tx.serialize().hex()
|
|
|
|
txid_1 = node.sendrawtransaction(hexstring=raw_tx_1, maxfeerate=0)
|
|
|
|
txid_1 = node.sendrawtransaction(hexstring=raw_tx_1, maxfeerate=0)
|
|
|
|
# Now spend both to "clearly hide" the outputs, ie. remove the coins from the utxo set by spending them
|
|
|
|
# Now spend both to "clearly hide" the outputs, ie. remove the coins from the utxo set by spending them
|
|
|
|
raw_tx_spend_both = node.signrawtransactionwithwallet(node.createrawtransaction(
|
|
|
|
tx = self.wallet.create_self_transfer()['tx']
|
|
|
|
inputs=[
|
|
|
|
tx.vin.append(deepcopy(tx.vin[0]))
|
|
|
|
{'txid': txid_0, 'vout': 0},
|
|
|
|
tx.wit.vtxinwit.append(deepcopy(tx.wit.vtxinwit[0]))
|
|
|
|
{'txid': txid_1, 'vout': 0},
|
|
|
|
tx.vin[0].prevout = COutPoint(hash=int(txid_0, 16), n=0)
|
|
|
|
],
|
|
|
|
tx.vin[1].prevout = COutPoint(hash=int(txid_1, 16), n=0)
|
|
|
|
outputs=[{node.getnewaddress(): 0.1}]
|
|
|
|
tx.vout[0].nValue = int(0.1 * COIN)
|
|
|
|
))['hex']
|
|
|
|
raw_tx_spend_both = tx.serialize().hex()
|
|
|
|
txid_spend_both = node.sendrawtransaction(hexstring=raw_tx_spend_both, maxfeerate=0)
|
|
|
|
txid_spend_both = self.wallet.sendrawtransaction(from_node=node, tx_hex=raw_tx_spend_both, maxfeerate=0)
|
|
|
|
self.generate(node, 1)
|
|
|
|
self.generate(node, 1)
|
|
|
|
self.mempool_size = 0
|
|
|
|
self.mempool_size = 0
|
|
|
|
# Now see if we can add the coins back to the utxo set by sending the exact txs again
|
|
|
|
# Now see if we can add the coins back to the utxo set by sending the exact txs again
|
|
|
@ -183,12 +179,11 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
rawtxs=[raw_tx_1],
|
|
|
|
rawtxs=[raw_tx_1],
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
self.log.info('Create a signed "reference" tx for later use')
|
|
|
|
self.log.info('Create a "reference" tx for later use')
|
|
|
|
raw_tx_reference = node.signrawtransactionwithwallet(node.createrawtransaction(
|
|
|
|
utxo_to_spend = self.wallet.get_utxo(txid=txid_spend_both)
|
|
|
|
inputs=[{'txid': txid_spend_both, 'vout': 0}],
|
|
|
|
tx = self.wallet.create_self_transfer(utxo_to_spend=utxo_to_spend, sequence=SEQUENCE_FINAL)['tx']
|
|
|
|
outputs=[{node.getnewaddress(): 0.05}],
|
|
|
|
tx.vout[0].nValue = int(0.05 * COIN)
|
|
|
|
))['hex']
|
|
|
|
raw_tx_reference = tx.serialize().hex()
|
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
|
|
|
|
|
|
|
# Reference tx should be valid on itself
|
|
|
|
# Reference tx should be valid on itself
|
|
|
|
self.check_mempool_result(
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': True, 'vsize': tx.get_vsize(), 'fees': { 'base': Decimal('0.1') - Decimal('0.05')}}],
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': True, 'vsize': tx.get_vsize(), 'fees': { 'base': Decimal('0.1') - Decimal('0.05')}}],
|
|
|
@ -199,8 +194,6 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
self.log.info('A transaction with no outputs')
|
|
|
|
self.log.info('A transaction with no outputs')
|
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
|
|
|
tx.vout = []
|
|
|
|
tx.vout = []
|
|
|
|
# Skip re-signing the transaction for context independent checks from now on
|
|
|
|
|
|
|
|
# tx = tx_from_hex(node.signrawtransactionwithwallet(tx.serialize().hex())['hex'])
|
|
|
|
|
|
|
|
self.check_mempool_result(
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'bad-txns-vout-empty'}],
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'bad-txns-vout-empty'}],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
@ -257,7 +250,7 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
self.log.info('A coinbase transaction')
|
|
|
|
self.log.info('A coinbase transaction')
|
|
|
|
# Pick the input of the first tx we signed, so it has to be a coinbase tx
|
|
|
|
# Pick the input of the first tx we created, so it has to be a coinbase tx
|
|
|
|
raw_tx_coinbase_spent = node.getrawtransaction(txid=node.decoderawtransaction(hexstring=raw_tx_in_block)['vin'][0]['txid'])
|
|
|
|
raw_tx_coinbase_spent = node.getrawtransaction(txid=node.decoderawtransaction(hexstring=raw_tx_in_block)['vin'][0]['txid'])
|
|
|
|
tx = tx_from_hex(raw_tx_coinbase_spent)
|
|
|
|
tx = tx_from_hex(raw_tx_coinbase_spent)
|
|
|
|
self.check_mempool_result(
|
|
|
|
self.check_mempool_result(
|
|
|
@ -334,7 +327,6 @@ class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
self.log.info('A transaction that is locked by BIP68 sequence logic')
|
|
|
|
self.log.info('A transaction that is locked by BIP68 sequence logic')
|
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
|
|
|
tx.vin[0].nSequence = 2 # We could include it in the second block mined from now, but not the very next one
|
|
|
|
tx.vin[0].nSequence = 2 # We could include it in the second block mined from now, but not the very next one
|
|
|
|
# Can skip re-signing the tx because of early rejection
|
|
|
|
|
|
|
|
self.check_mempool_result(
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'non-BIP68-final'}],
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'non-BIP68-final'}],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
|