2017-11-17 12:54:39 -05:00
|
|
|
#!/usr/bin/env python3
|
2021-07-28 13:57:16 +02:00
|
|
|
# Copyright (c) 2017-2021 The Bitcoin Core developers
|
2017-11-17 12:54:39 -05:00
|
|
|
# Distributed under the MIT software license, see the accompanying
|
|
|
|
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
"""Test mempool acceptance of raw transactions."""
|
|
|
|
|
2022-01-11 03:30:23 +01:00
|
|
|
from copy import deepcopy
|
2020-05-24 18:28:56 +05:30
|
|
|
from decimal import Decimal
|
2018-11-27 17:54:05 +00:00
|
|
|
import math
|
2018-12-11 13:30:50 -05:00
|
|
|
|
2017-11-17 12:54:39 -05:00
|
|
|
from test_framework.test_framework import BitcoinTestFramework
|
2019-11-20 21:40:53 +01:00
|
|
|
from test_framework.key import ECKey
|
2017-11-17 12:54:39 -05:00
|
|
|
from test_framework.messages import (
|
2022-07-12 17:28:20 +02:00
|
|
|
MAX_BIP125_RBF_SEQUENCE,
|
2017-11-17 12:54:39 -05:00
|
|
|
COIN,
|
|
|
|
COutPoint,
|
2022-10-05 14:50:30 -04:00
|
|
|
CTransaction,
|
2021-07-05 23:06:06 +02:00
|
|
|
CTxIn,
|
2022-10-05 14:50:30 -04:00
|
|
|
CTxInWitness,
|
2017-11-17 12:54:39 -05:00
|
|
|
CTxOut,
|
2021-07-01 01:43:45 +02:00
|
|
|
MAX_BLOCK_WEIGHT,
|
2020-03-03 12:48:49 -05:00
|
|
|
MAX_MONEY,
|
2022-01-11 02:08:01 +01:00
|
|
|
SEQUENCE_FINAL,
|
2021-06-15 23:02:28 +02:00
|
|
|
tx_from_hex,
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
from test_framework.script import (
|
|
|
|
CScript,
|
|
|
|
OP_0,
|
|
|
|
OP_HASH160,
|
|
|
|
OP_RETURN,
|
2022-10-05 14:50:30 -04:00
|
|
|
OP_TRUE,
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
2021-06-27 23:52:38 +02:00
|
|
|
from test_framework.script_util import (
|
2022-10-05 14:50:30 -04:00
|
|
|
DUMMY_MIN_OP_RETURN_SCRIPT,
|
2021-10-18 22:25:04 +02:00
|
|
|
keys_to_multisig_script,
|
2022-10-05 14:50:30 -04:00
|
|
|
MIN_PADDING,
|
|
|
|
MIN_STANDARD_TX_NONWITNESS_SIZE,
|
2021-06-27 23:52:38 +02:00
|
|
|
script_to_p2sh_script,
|
2022-10-05 14:50:30 -04:00
|
|
|
script_to_p2wsh_script,
|
2021-06-27 23:52:38 +02:00
|
|
|
)
|
2017-11-17 12:54:39 -05:00
|
|
|
from test_framework.util import (
|
|
|
|
assert_equal,
|
2022-10-05 14:50:30 -04:00
|
|
|
assert_greater_than,
|
2017-11-17 12:54:39 -05:00
|
|
|
assert_raises_rpc_error,
|
|
|
|
)
|
2022-01-11 03:30:23 +01:00
|
|
|
from test_framework.wallet import MiniWallet
|
2017-11-17 12:54:39 -05:00
|
|
|
|
|
|
|
|
|
|
|
class MempoolAcceptanceTest(BitcoinTestFramework):
|
|
|
|
def set_test_params(self):
|
|
|
|
self.num_nodes = 1
|
|
|
|
self.extra_args = [[
|
2019-11-20 21:40:53 +01:00
|
|
|
'-txindex','-permitbaremultisig=0',
|
2017-11-17 12:54:39 -05:00
|
|
|
]] * self.num_nodes
|
2019-12-06 14:37:49 +00:00
|
|
|
self.supports_cli = False
|
2017-11-17 12:54:39 -05:00
|
|
|
|
|
|
|
def check_mempool_result(self, result_expected, *args, **kwargs):
|
|
|
|
"""Wrapper to check result of testmempoolaccept on node_0's mempool"""
|
|
|
|
result_test = self.nodes[0].testmempoolaccept(*args, **kwargs)
|
2021-01-07 16:47:35 +01:00
|
|
|
for r in result_test:
|
|
|
|
r.pop('wtxid') # Skip check for now
|
2017-11-17 12:54:39 -05:00
|
|
|
assert_equal(result_expected, result_test)
|
|
|
|
assert_equal(self.nodes[0].getmempoolinfo()['size'], self.mempool_size) # Must not change mempool state
|
|
|
|
|
|
|
|
def run_test(self):
|
|
|
|
node = self.nodes[0]
|
2022-01-11 03:30:23 +01:00
|
|
|
self.wallet = MiniWallet(node)
|
|
|
|
self.wallet.rescan_utxos()
|
2017-11-17 12:54:39 -05:00
|
|
|
|
|
|
|
self.log.info('Start with empty mempool, and 200 blocks')
|
|
|
|
self.mempool_size = 0
|
2019-02-11 13:56:19 -05:00
|
|
|
assert_equal(node.getblockcount(), 200)
|
2017-11-17 12:54:39 -05:00
|
|
|
assert_equal(node.getmempoolinfo()['size'], self.mempool_size)
|
|
|
|
|
|
|
|
self.log.info('Should not accept garbage to testmempoolaccept')
|
2022-09-12 10:04:15 -03:00
|
|
|
assert_raises_rpc_error(-3, 'JSON value of type string is not of expected type array', lambda: node.testmempoolaccept(rawtxs='ff00baar'))
|
2021-02-11 10:50:45 -08:00
|
|
|
assert_raises_rpc_error(-8, 'Array must contain between 1 and 25 transactions.', lambda: node.testmempoolaccept(rawtxs=['ff22']*26))
|
|
|
|
assert_raises_rpc_error(-8, 'Array must contain between 1 and 25 transactions.', lambda: node.testmempoolaccept(rawtxs=[]))
|
2017-11-17 12:54:39 -05:00
|
|
|
assert_raises_rpc_error(-22, 'TX decode failed', lambda: node.testmempoolaccept(rawtxs=['ff00baar']))
|
|
|
|
|
|
|
|
self.log.info('A transaction already in the blockchain')
|
2022-01-11 03:30:23 +01:00
|
|
|
tx = self.wallet.create_self_transfer()['tx'] # Pick a random coin(base) to spend
|
|
|
|
tx.vout.append(deepcopy(tx.vout[0]))
|
|
|
|
tx.vout[0].nValue = int(0.3 * COIN)
|
|
|
|
tx.vout[1].nValue = int(49 * COIN)
|
|
|
|
raw_tx_in_block = tx.serialize().hex()
|
2022-06-01 17:05:13 +02:00
|
|
|
txid_in_block = self.wallet.sendrawtransaction(from_node=node, tx_hex=raw_tx_in_block)
|
2021-08-19 17:10:24 +02:00
|
|
|
self.generate(node, 1)
|
2018-12-11 13:30:50 -05:00
|
|
|
self.mempool_size = 0
|
2017-11-17 12:54:39 -05:00
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': txid_in_block, 'allowed': False, 'reject-reason': 'txn-already-known'}],
|
2017-11-17 12:54:39 -05:00
|
|
|
rawtxs=[raw_tx_in_block],
|
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('A transaction not in the mempool')
|
2020-09-29 23:11:58 +05:30
|
|
|
fee = Decimal('0.000007')
|
2022-01-11 03:30:23 +01:00
|
|
|
utxo_to_spend = self.wallet.get_utxo(txid=txid_in_block) # use 0.3 BTC UTXO
|
2022-07-12 17:28:20 +02:00
|
|
|
tx = self.wallet.create_self_transfer(utxo_to_spend=utxo_to_spend, sequence=MAX_BIP125_RBF_SEQUENCE)['tx']
|
2022-01-11 03:30:23 +01:00
|
|
|
tx.vout[0].nValue = int((Decimal('0.3') - fee) * COIN)
|
|
|
|
raw_tx_0 = tx.serialize().hex()
|
2017-11-17 12:54:39 -05:00
|
|
|
txid_0 = tx.rehash()
|
|
|
|
self.check_mempool_result(
|
2020-09-29 23:11:58 +05:30
|
|
|
result_expected=[{'txid': txid_0, 'allowed': True, 'vsize': tx.get_vsize(), 'fees': {'base': fee}}],
|
2017-11-17 12:54:39 -05:00
|
|
|
rawtxs=[raw_tx_0],
|
|
|
|
)
|
|
|
|
|
2018-12-11 13:30:50 -05:00
|
|
|
self.log.info('A final transaction not in the mempool')
|
2020-09-29 23:11:58 +05:30
|
|
|
output_amount = Decimal('0.025')
|
2022-01-11 03:30:23 +01:00
|
|
|
tx = self.wallet.create_self_transfer(
|
|
|
|
sequence=SEQUENCE_FINAL,
|
2018-12-11 13:30:50 -05:00
|
|
|
locktime=node.getblockcount() + 2000, # Can be anything
|
2022-01-11 03:30:23 +01:00
|
|
|
)['tx']
|
|
|
|
tx.vout[0].nValue = int(output_amount * COIN)
|
|
|
|
raw_tx_final = tx.serialize().hex()
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_final)
|
2022-01-11 03:30:23 +01:00
|
|
|
fee_expected = Decimal('50.0') - output_amount
|
2018-12-11 13:30:50 -05:00
|
|
|
self.check_mempool_result(
|
2020-09-29 23:11:58 +05:30
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': True, 'vsize': tx.get_vsize(), 'fees': {'base': fee_expected}}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2018-06-27 17:21:07 +09:00
|
|
|
maxfeerate=0,
|
2018-12-11 13:30:50 -05:00
|
|
|
)
|
2018-06-27 17:21:07 +09:00
|
|
|
node.sendrawtransaction(hexstring=raw_tx_final, maxfeerate=0)
|
2018-12-11 13:30:50 -05:00
|
|
|
self.mempool_size += 1
|
|
|
|
|
2017-11-17 12:54:39 -05:00
|
|
|
self.log.info('A transaction in the mempool')
|
|
|
|
node.sendrawtransaction(hexstring=raw_tx_0)
|
2018-12-11 13:30:50 -05:00
|
|
|
self.mempool_size += 1
|
2017-11-17 12:54:39 -05:00
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': txid_0, 'allowed': False, 'reject-reason': 'txn-already-in-mempool'}],
|
2017-11-17 12:54:39 -05:00
|
|
|
rawtxs=[raw_tx_0],
|
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('A transaction that replaces a mempool transaction')
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_0)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vout[0].nValue -= int(fee * COIN) # Double the fee
|
2022-07-12 17:28:20 +02:00
|
|
|
tx.vin[0].nSequence = MAX_BIP125_RBF_SEQUENCE + 1 # Now, opt out of RBF
|
2022-01-11 03:30:23 +01:00
|
|
|
raw_tx_0 = tx.serialize().hex()
|
2017-11-17 12:54:39 -05:00
|
|
|
txid_0 = tx.rehash()
|
|
|
|
self.check_mempool_result(
|
2020-09-29 23:11:58 +05:30
|
|
|
result_expected=[{'txid': txid_0, 'allowed': True, 'vsize': tx.get_vsize(), 'fees': {'base': (2 * fee)}}],
|
2017-11-17 12:54:39 -05:00
|
|
|
rawtxs=[raw_tx_0],
|
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('A transaction that conflicts with an unconfirmed tx')
|
|
|
|
# Send the transaction that replaces the mempool transaction and opts out of replaceability
|
2018-06-27 17:21:07 +09:00
|
|
|
node.sendrawtransaction(hexstring=tx.serialize().hex(), maxfeerate=0)
|
2017-11-17 12:54:39 -05:00
|
|
|
# take original raw_tx_0
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_0)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vout[0].nValue -= int(4 * fee * COIN) # Set more fee
|
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'txn-mempool-conflict'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2018-06-27 17:21:07 +09:00
|
|
|
maxfeerate=0,
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('A transaction with missing inputs, that never existed')
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_0)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vin[0].prevout = COutPoint(hash=int('ff' * 32, 16), n=14)
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'missing-inputs'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('A transaction with missing inputs, that existed once in the past')
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_0)
|
2017-11-17 12:54:39 -05:00
|
|
|
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
|
2022-01-11 03:30:23 +01:00
|
|
|
raw_tx_1 = tx.serialize().hex()
|
2018-06-27 17:21:07 +09:00
|
|
|
txid_1 = node.sendrawtransaction(hexstring=raw_tx_1, maxfeerate=0)
|
2017-11-17 12:54:39 -05:00
|
|
|
# Now spend both to "clearly hide" the outputs, ie. remove the coins from the utxo set by spending them
|
2022-01-11 03:30:23 +01:00
|
|
|
tx = self.wallet.create_self_transfer()['tx']
|
|
|
|
tx.vin.append(deepcopy(tx.vin[0]))
|
|
|
|
tx.wit.vtxinwit.append(deepcopy(tx.wit.vtxinwit[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)
|
|
|
|
tx.vout[0].nValue = int(0.1 * COIN)
|
|
|
|
raw_tx_spend_both = tx.serialize().hex()
|
2022-06-01 17:05:13 +02:00
|
|
|
txid_spend_both = self.wallet.sendrawtransaction(from_node=node, tx_hex=raw_tx_spend_both)
|
2021-08-19 17:10:24 +02:00
|
|
|
self.generate(node, 1)
|
2017-11-17 12:54:39 -05:00
|
|
|
self.mempool_size = 0
|
|
|
|
# Now see if we can add the coins back to the utxo set by sending the exact txs again
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': txid_0, 'allowed': False, 'reject-reason': 'missing-inputs'}],
|
|
|
|
rawtxs=[raw_tx_0],
|
|
|
|
)
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': txid_1, 'allowed': False, 'reject-reason': 'missing-inputs'}],
|
|
|
|
rawtxs=[raw_tx_1],
|
|
|
|
)
|
|
|
|
|
2022-01-11 03:30:23 +01:00
|
|
|
self.log.info('Create a "reference" tx for later use')
|
|
|
|
utxo_to_spend = self.wallet.get_utxo(txid=txid_spend_both)
|
|
|
|
tx = self.wallet.create_self_transfer(utxo_to_spend=utxo_to_spend, sequence=SEQUENCE_FINAL)['tx']
|
|
|
|
tx.vout[0].nValue = int(0.05 * COIN)
|
|
|
|
raw_tx_reference = tx.serialize().hex()
|
2017-11-17 12:54:39 -05:00
|
|
|
# Reference tx should be valid on itself
|
|
|
|
self.check_mempool_result(
|
2020-09-29 23:11:58 +05:30
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': True, 'vsize': tx.get_vsize(), 'fees': { 'base': Decimal('0.1') - Decimal('0.05')}}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2019-08-02 02:03:01 +09:00
|
|
|
maxfeerate=0,
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('A transaction with no outputs')
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vout = []
|
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'bad-txns-vout-empty'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('A really large transaction')
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2021-07-01 01:43:45 +02:00
|
|
|
tx.vin = [tx.vin[0]] * math.ceil(MAX_BLOCK_WEIGHT // 4 / len(tx.vin[0].serialize()))
|
2017-11-17 12:54:39 -05:00
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'bad-txns-oversize'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('A transaction with negative output value')
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vout[0].nValue *= -1
|
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'bad-txns-vout-negative'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
2018-11-10 09:11:22 -08:00
|
|
|
# The following two validations prevent overflow of the output amounts (see CVE-2010-5139).
|
2017-11-17 12:54:39 -05:00
|
|
|
self.log.info('A transaction with too large output value')
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2020-03-03 12:48:49 -05:00
|
|
|
tx.vout[0].nValue = MAX_MONEY + 1
|
2017-11-17 12:54:39 -05:00
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'bad-txns-vout-toolarge'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('A transaction with too large sum of output values')
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vout = [tx.vout[0]] * 2
|
2020-03-03 12:48:49 -05:00
|
|
|
tx.vout[0].nValue = MAX_MONEY
|
2017-11-17 12:54:39 -05:00
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'bad-txns-txouttotal-toolarge'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('A transaction with duplicate inputs')
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vin = [tx.vin[0]] * 2
|
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'bad-txns-inputs-duplicate'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
2021-07-05 23:06:06 +02:00
|
|
|
self.log.info('A non-coinbase transaction with coinbase-like outpoint')
|
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
|
|
|
tx.vin.append(CTxIn(COutPoint(hash=0, n=0xffffffff)))
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'bad-txns-prevout-null'}],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
|
)
|
|
|
|
|
2017-11-17 12:54:39 -05:00
|
|
|
self.log.info('A coinbase transaction')
|
2022-01-11 03:30:23 +01:00
|
|
|
# Pick the input of the first tx we created, so it has to be a coinbase tx
|
2017-11-17 12:54:39 -05:00
|
|
|
raw_tx_coinbase_spent = node.getrawtransaction(txid=node.decoderawtransaction(hexstring=raw_tx_in_block)['vin'][0]['txid'])
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_coinbase_spent)
|
2017-11-17 12:54:39 -05:00
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'coinbase'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('Some nonstandard transactions')
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.nVersion = 3 # A version currently non-standard
|
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'version'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vout[0].scriptPubKey = CScript([OP_0]) # Some non-standard script
|
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'scriptpubkey'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2019-11-20 21:40:53 +01:00
|
|
|
)
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2019-11-20 21:40:53 +01:00
|
|
|
key = ECKey()
|
|
|
|
key.generate()
|
|
|
|
pubkey = key.get_pubkey().get_bytes()
|
2021-10-18 22:25:04 +02:00
|
|
|
tx.vout[0].scriptPubKey = keys_to_multisig_script([pubkey] * 3, k=2) # Some bare multisig script (2-of-3)
|
2019-11-20 21:40:53 +01:00
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'bare-multisig'}],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vin[0].scriptSig = CScript([OP_HASH160]) # Some not-pushonly scriptSig
|
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'scriptsig-not-pushonly'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2019-11-20 10:59:44 +01:00
|
|
|
tx.vin[0].scriptSig = CScript([b'a' * 1648]) # Some too large scriptSig (>1650 bytes)
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'scriptsig-size'}],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
|
)
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2021-06-27 23:52:38 +02:00
|
|
|
output_p2sh_burn = CTxOut(nValue=540, scriptPubKey=script_to_p2sh_script(b'burn'))
|
2017-11-17 12:54:39 -05:00
|
|
|
num_scripts = 100000 // len(output_p2sh_burn.serialize()) # Use enough outputs to make the tx too large for our policy
|
|
|
|
tx.vout = [output_p2sh_burn] * num_scripts
|
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'tx-size'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vout[0] = output_p2sh_burn
|
|
|
|
tx.vout[0].nValue -= 1 # Make output smaller, such that it is dust for our policy
|
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'dust'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vout[0].scriptPubKey = CScript([OP_RETURN, b'\xff'])
|
|
|
|
tx.vout = [tx.vout[0]] * 2
|
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'multi-op-return'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('A timelocked transaction')
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vin[0].nSequence -= 1 # Should be non-max, so locktime is not ignored
|
|
|
|
tx.nLockTime = node.getblockcount() + 1
|
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'non-final'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('A transaction that is locked by BIP68 sequence logic')
|
2021-06-15 23:02:28 +02:00
|
|
|
tx = tx_from_hex(raw_tx_reference)
|
2017-11-17 12:54:39 -05:00
|
|
|
tx.vin[0].nSequence = 2 # We could include it in the second block mined from now, but not the very next one
|
|
|
|
self.check_mempool_result(
|
2019-10-10 11:19:42 -04:00
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'non-BIP68-final'}],
|
2019-02-18 10:35:48 -05:00
|
|
|
rawtxs=[tx.serialize().hex()],
|
2018-06-27 17:21:07 +09:00
|
|
|
maxfeerate=0,
|
2017-11-17 12:54:39 -05:00
|
|
|
)
|
|
|
|
|
2022-10-05 14:50:30 -04:00
|
|
|
# Prep for tiny-tx tests with wsh(OP_TRUE) output
|
|
|
|
seed_tx = self.wallet.send_to(from_node=node, scriptPubKey=script_to_p2wsh_script(CScript([OP_TRUE])), amount=COIN)
|
|
|
|
self.generate(node, 1)
|
|
|
|
|
|
|
|
self.log.info('A tiny transaction(in non-witness bytes) that is disallowed')
|
|
|
|
tx = CTransaction()
|
|
|
|
tx.vin.append(CTxIn(COutPoint(int(seed_tx[0], 16), seed_tx[1]), b"", SEQUENCE_FINAL))
|
|
|
|
tx.wit.vtxinwit = [CTxInWitness()]
|
|
|
|
tx.wit.vtxinwit[0].scriptWitness.stack = [CScript([OP_TRUE])]
|
|
|
|
tx.vout.append(CTxOut(0, CScript([OP_RETURN] + ([OP_0] * (MIN_PADDING - 2)))))
|
|
|
|
# Note it's only non-witness size that matters!
|
|
|
|
assert_equal(len(tx.serialize_without_witness()), 64)
|
|
|
|
assert_equal(MIN_STANDARD_TX_NONWITNESS_SIZE - 1, 64)
|
|
|
|
assert_greater_than(len(tx.serialize()), 64)
|
|
|
|
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': False, 'reject-reason': 'tx-size-small'}],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
|
maxfeerate=0,
|
|
|
|
)
|
|
|
|
|
|
|
|
self.log.info('Minimally-small transaction(in non-witness bytes) that is allowed')
|
|
|
|
tx.vout[0] = CTxOut(COIN - 1000, DUMMY_MIN_OP_RETURN_SCRIPT)
|
|
|
|
assert_equal(len(tx.serialize_without_witness()), MIN_STANDARD_TX_NONWITNESS_SIZE)
|
|
|
|
self.check_mempool_result(
|
|
|
|
result_expected=[{'txid': tx.rehash(), 'allowed': True, 'vsize': tx.get_vsize(), 'fees': { 'base': Decimal('0.00001000')}}],
|
|
|
|
rawtxs=[tx.serialize().hex()],
|
|
|
|
maxfeerate=0,
|
|
|
|
)
|
2017-11-17 12:54:39 -05:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
MempoolAcceptanceTest().main()
|