2016-03-19 16:58:06 -03:00
#!/usr/bin/env python3
2021-07-28 07:57:16 -04:00
# Copyright (c) 2014-2021 The Bitcoin Core developers
2014-12-29 16:17:21 -03:00
# Distributed under the MIT software license, see the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-07-30 15:42:17 -04:00
""" Test the rawtransaction RPCs.
2014-12-29 16:17:21 -03:00
2017-01-17 20:34:40 -03:00
Test the following RPCs :
2021-07-12 10:04:48 -04:00
- getrawtransaction
2017-01-17 20:34:40 -03:00
- createrawtransaction
2017-09-05 20:49:18 -03:00
- signrawtransactionwithwallet
2017-01-17 20:34:40 -03:00
- sendrawtransaction
- decoderawtransaction
2016-11-22 11:37:59 -03:00
"""
2014-12-29 16:17:21 -03:00
2017-12-11 17:24:02 -03:00
from collections import OrderedDict
2018-07-06 18:10:35 -04:00
from decimal import Decimal
2021-11-21 06:17:59 -03:00
from itertools import product
2021-05-17 10:38:19 -04:00
from test_framework . blocktools import COINBASE_MATURITY
2021-06-15 17:02:28 -04:00
from test_framework . messages import (
2022-07-12 11:28:20 -04:00
MAX_BIP125_RBF_SEQUENCE ,
2021-06-15 17:02:28 -04:00
CTransaction ,
tx_from_hex ,
)
2015-05-02 08:58:10 -03:00
from test_framework . test_framework import BitcoinTestFramework
2019-08-27 15:28:56 -04:00
from test_framework . util import (
assert_equal ,
assert_raises_rpc_error ,
2022-04-15 14:34:49 -04:00
)
from test_framework . wallet import (
getnewdestination ,
MiniWallet ,
2019-08-27 15:28:56 -04:00
)
2014-12-29 16:17:21 -03:00
2021-07-12 10:12:21 -04:00
TXID = " 1d1d4e24ed99057e84c3f80fd8fbec79ed9e1acee37da269356ecea000000000 "
2017-12-12 11:55:00 -03:00
class multidict ( dict ) :
""" Dictionary that allows duplicate keys.
Constructed with a list of ( key , value ) tuples . When dumped by the json module ,
will output invalid json with repeated keys , eg :
>> > json . dumps ( multidict ( [ ( 1 , 2 ) , ( 1 , 2 ) ] )
' { " 1 " : 2, " 1 " : 2} '
Used to test calls to rpc methods with repeated keys in the json object . """
def __init__ ( self , x ) :
dict . __init__ ( self , x )
self . x = x
def items ( self ) :
return self . x
2014-12-29 16:17:21 -03:00
class RawTransactionsTest ( BitcoinTestFramework ) :
2022-11-09 08:53:13 -03:00
def add_options ( self , parser ) :
self . add_wallet_options ( parser , descriptors = False )
2017-06-09 18:21:21 -04:00
def set_test_params ( self ) :
2016-05-14 08:01:31 -03:00
self . setup_clean_chain = True
2022-04-15 14:34:49 -04:00
self . num_nodes = 3
2019-05-24 04:16:11 -04:00
self . extra_args = [
[ " -txindex " ] ,
[ " -txindex " ] ,
2021-07-11 06:18:43 -04:00
[ ] ,
2019-05-24 04:16:11 -04:00
]
2021-07-13 09:41:16 -04:00
# whitelist all peers to speed up tx relay / mempool sync
for args in self . extra_args :
args . append ( " -whitelist=noban@127.0.0.1 " )
2019-12-06 11:37:49 -03:00
self . supports_cli = False
2014-12-29 16:17:21 -03:00
2018-12-22 13:19:50 -03:00
def setup_network ( self ) :
2017-04-03 10:34:04 -03:00
super ( ) . setup_network ( )
2020-09-17 04:46:07 -03:00
self . connect_nodes ( 0 , 2 )
2015-04-01 10:08:00 -03:00
2014-12-29 16:17:21 -03:00
def run_test ( self ) :
2022-04-15 14:34:49 -04:00
self . wallet = MiniWallet ( self . nodes [ 0 ] )
2021-07-12 09:13:29 -04:00
self . log . info ( " Prepare some coins for multiple *rawtransaction commands " )
2022-04-15 14:34:49 -04:00
self . generate ( self . wallet , 10 )
2021-08-19 11:10:24 -04:00
self . generate ( self . nodes [ 0 ] , COINBASE_MATURITY + 1 )
2014-12-29 16:17:21 -03:00
2021-07-12 10:04:48 -04:00
self . getrawtransaction_tests ( )
2021-11-21 06:17:59 -03:00
self . getrawtransaction_verbosity_tests ( )
2021-07-12 10:04:48 -04:00
self . createrawtransaction_tests ( )
self . sendrawtransaction_tests ( )
self . sendrawtransaction_testmempoolaccept_tests ( )
self . decoderawtransaction_tests ( )
self . transaction_version_number_tests ( )
2022-11-10 06:12:39 -03:00
if self . is_specified_wallet_compiled ( ) and not self . options . descriptors :
self . import_deterministic_coinbase_privkeys ( )
2021-07-12 10:04:48 -04:00
self . raw_multisig_transaction_legacy_tests ( )
def getrawtransaction_tests ( self ) :
2022-04-15 14:34:49 -04:00
tx = self . wallet . send_self_transfer ( from_node = self . nodes [ 0 ] )
2021-08-19 11:10:24 -04:00
self . generate ( self . nodes [ 0 ] , 1 )
2022-04-15 14:34:49 -04:00
txId = tx [ ' txid ' ]
2022-01-20 17:09:58 -03:00
err_msg = (
" No such mempool transaction. Use -txindex or provide a block hash to enable "
" blockchain transaction queries. Use gettransaction for wallet transactions. "
)
2021-07-12 09:42:21 -04:00
2022-04-15 14:34:49 -04:00
for n in [ 0 , 2 ] :
2021-07-12 09:42:21 -04:00
self . log . info ( f " Test getrawtransaction { ' with ' if n == 0 else ' without ' } -txindex " )
2022-01-20 17:09:58 -03:00
if n == 0 :
# With -txindex.
# 1. valid parameters - only supply txid
2022-04-15 14:34:49 -04:00
assert_equal ( self . nodes [ n ] . getrawtransaction ( txId ) , tx [ ' hex ' ] )
2022-01-20 17:09:58 -03:00
# 2. valid parameters - supply txid and 0 for non-verbose
2022-04-15 14:34:49 -04:00
assert_equal ( self . nodes [ n ] . getrawtransaction ( txId , 0 ) , tx [ ' hex ' ] )
2021-07-12 09:42:21 -04:00
2022-01-20 17:09:58 -03:00
# 3. valid parameters - supply txid and False for non-verbose
2022-04-15 14:34:49 -04:00
assert_equal ( self . nodes [ n ] . getrawtransaction ( txId , False ) , tx [ ' hex ' ] )
2021-07-12 09:42:21 -04:00
2022-01-20 17:09:58 -03:00
# 4. valid parameters - supply txid and 1 for verbose.
# We only check the "hex" field of the output so we don't need to update this test every time the output format changes.
2022-04-15 14:34:49 -04:00
assert_equal ( self . nodes [ n ] . getrawtransaction ( txId , 1 ) [ " hex " ] , tx [ ' hex ' ] )
2021-11-21 06:17:59 -03:00
assert_equal ( self . nodes [ n ] . getrawtransaction ( txId , 2 ) [ " hex " ] , tx [ ' hex ' ] )
2021-07-12 09:42:21 -04:00
2022-01-20 17:09:58 -03:00
# 5. valid parameters - supply txid and True for non-verbose
2022-04-15 14:34:49 -04:00
assert_equal ( self . nodes [ n ] . getrawtransaction ( txId , True ) [ " hex " ] , tx [ ' hex ' ] )
2022-01-20 17:09:58 -03:00
else :
# Without -txindex, expect to raise.
for verbose in [ None , 0 , False , 1 , True ] :
assert_raises_rpc_error ( - 5 , err_msg , self . nodes [ n ] . getrawtransaction , txId , verbose )
2021-07-12 09:42:21 -04:00
# 6. invalid parameters - supply txid and invalid boolean values (strings) for verbose
for value in [ " True " , " False " ] :
2021-11-21 06:17:59 -03:00
assert_raises_rpc_error ( - 3 , " not of expected type number " , self . nodes [ n ] . getrawtransaction , txid = txId , verbose = value )
assert_raises_rpc_error ( - 3 , " not of expected type number " , self . nodes [ n ] . getrawtransaction , txid = txId , verbosity = value )
2021-07-12 09:42:21 -04:00
# 7. invalid parameters - supply txid and empty array
2021-11-21 06:17:59 -03:00
assert_raises_rpc_error ( - 3 , " not of expected type number " , self . nodes [ n ] . getrawtransaction , txId , [ ] )
2021-07-12 09:42:21 -04:00
# 8. invalid parameters - supply txid and empty dict
2021-11-21 06:17:59 -03:00
assert_raises_rpc_error ( - 3 , " not of expected type number " , self . nodes [ n ] . getrawtransaction , txId , { } )
2021-07-12 09:42:21 -04:00
# Make a tx by sending, then generate 2 blocks; block1 has the tx in it
2022-04-15 14:34:49 -04:00
tx = self . wallet . send_self_transfer ( from_node = self . nodes [ 2 ] ) [ ' txid ' ]
2021-08-19 11:10:24 -04:00
block1 , block2 = self . generate ( self . nodes [ 2 ] , 2 )
2022-04-15 14:34:49 -04:00
for n in [ 0 , 2 ] :
2021-07-12 09:42:21 -04:00
self . log . info ( f " Test getrawtransaction { ' with ' if n == 0 else ' without ' } -txindex, with blockhash " )
# We should be able to get the raw transaction by providing the correct block
gottx = self . nodes [ n ] . getrawtransaction ( txid = tx , verbose = True , blockhash = block1 )
assert_equal ( gottx [ ' txid ' ] , tx )
assert_equal ( gottx [ ' in_active_chain ' ] , True )
if n == 0 :
self . log . info ( " Test getrawtransaction with -txindex, without blockhash: ' in_active_chain ' should be absent " )
2021-11-21 06:17:59 -03:00
for v in [ 1 , 2 ] :
gottx = self . nodes [ n ] . getrawtransaction ( txid = tx , verbosity = v )
assert_equal ( gottx [ ' txid ' ] , tx )
assert ' in_active_chain ' not in gottx
2021-07-12 09:42:21 -04:00
else :
self . log . info ( " Test getrawtransaction without -txindex, without blockhash: expect the call to raise " )
assert_raises_rpc_error ( - 5 , err_msg , self . nodes [ n ] . getrawtransaction , txid = tx , verbose = True )
# We should not get the tx if we provide an unrelated block
assert_raises_rpc_error ( - 5 , " No such transaction found " , self . nodes [ n ] . getrawtransaction , txid = tx , blockhash = block2 )
# An invalid block hash should raise the correct errors
2022-09-14 12:13:58 -03:00
assert_raises_rpc_error ( - 3 , " JSON value of type bool is not of expected type string " , self . nodes [ n ] . getrawtransaction , txid = tx , blockhash = True )
2021-07-12 09:42:21 -04:00
assert_raises_rpc_error ( - 8 , " parameter 3 must be of length 64 (not 6, for ' foobar ' ) " , self . nodes [ n ] . getrawtransaction , txid = tx , blockhash = " foobar " )
assert_raises_rpc_error ( - 8 , " parameter 3 must be of length 64 (not 8, for ' abcd1234 ' ) " , self . nodes [ n ] . getrawtransaction , txid = tx , blockhash = " abcd1234 " )
foo = " ZZZ0000000000000000000000000000000000000000000000000000000000000 "
assert_raises_rpc_error ( - 8 , f " parameter 3 must be hexadecimal string (not ' { foo } ' ) " , self . nodes [ n ] . getrawtransaction , txid = tx , blockhash = foo )
bar = " 0000000000000000000000000000000000000000000000000000000000000000 "
assert_raises_rpc_error ( - 5 , " Block hash not found " , self . nodes [ n ] . getrawtransaction , txid = tx , blockhash = bar )
# Undo the blocks and verify that "in_active_chain" is false.
self . nodes [ n ] . invalidateblock ( block1 )
gottx = self . nodes [ n ] . getrawtransaction ( txid = tx , verbose = True , blockhash = block1 )
assert_equal ( gottx [ ' in_active_chain ' ] , False )
self . nodes [ n ] . reconsiderblock ( block1 )
assert_equal ( self . nodes [ n ] . getbestblockhash ( ) , block2 )
2021-07-12 09:13:29 -04:00
self . log . info ( " Test getrawtransaction on genesis block coinbase returns an error " )
2018-01-27 04:10:56 -03:00
block = self . nodes [ 0 ] . getblock ( self . nodes [ 0 ] . getblockhash ( 0 ) )
assert_raises_rpc_error ( - 5 , " The genesis block coinbase is not considered an ordinary transaction " , self . nodes [ 0 ] . getrawtransaction , block [ ' merkleroot ' ] )
2021-11-21 06:17:59 -03:00
def getrawtransaction_verbosity_tests ( self ) :
tx = self . wallet . send_self_transfer ( from_node = self . nodes [ 1 ] ) [ ' txid ' ]
[ block1 ] = self . generate ( self . nodes [ 1 ] , 1 )
fields = [
' blockhash ' ,
' blocktime ' ,
' confirmations ' ,
' hash ' ,
' hex ' ,
' in_active_chain ' ,
' locktime ' ,
' size ' ,
' time ' ,
' txid ' ,
' vin ' ,
' vout ' ,
' vsize ' ,
' weight ' ,
]
prevout_fields = [
' generated ' ,
' height ' ,
' value ' ,
' scriptPubKey ' ,
]
script_pub_key_fields = [
' address ' ,
' asm ' ,
' hex ' ,
' type ' ,
]
# node 0 & 2 with verbosity 1 & 2
for n , v in product ( [ 0 , 2 ] , [ 1 , 2 ] ) :
self . log . info ( f " Test getrawtransaction_verbosity { v } { ' with ' if n == 0 else ' without ' } -txindex, with blockhash " )
gottx = self . nodes [ n ] . getrawtransaction ( txid = tx , verbosity = v , blockhash = block1 )
missing_fields = set ( fields ) . difference ( gottx . keys ( ) )
if missing_fields :
raise AssertionError ( f " fields { ' , ' . join ( missing_fields ) } are not in transaction " )
assert ( len ( gottx [ ' vin ' ] ) > 0 )
if v == 1 :
assert ( ' fee ' not in gottx )
assert ( ' prevout ' not in gottx [ ' vin ' ] [ 0 ] )
if v == 2 :
assert ( isinstance ( gottx [ ' fee ' ] , Decimal ) )
assert ( ' prevout ' in gottx [ ' vin ' ] [ 0 ] )
prevout = gottx [ ' vin ' ] [ 0 ] [ ' prevout ' ]
script_pub_key = prevout [ ' scriptPubKey ' ]
missing_fields = set ( prevout_fields ) . difference ( prevout . keys ( ) )
if missing_fields :
raise AssertionError ( f " fields { ' , ' . join ( missing_fields ) } are not in transaction " )
missing_fields = set ( script_pub_key_fields ) . difference ( script_pub_key . keys ( ) )
if missing_fields :
raise AssertionError ( f " fields { ' , ' . join ( missing_fields ) } are not in transaction " )
# check verbosity 2 without blockhash but with txindex
assert ( ' fee ' in self . nodes [ 0 ] . getrawtransaction ( txid = tx , verbosity = 2 ) )
# check that coinbase has no fee or does not throw any errors for verbosity 2
coin_base = self . nodes [ 1 ] . getblock ( block1 ) [ ' tx ' ] [ 0 ]
gottx = self . nodes [ 1 ] . getrawtransaction ( txid = coin_base , verbosity = 2 , blockhash = block1 )
assert ( ' fee ' not in gottx )
2021-07-12 10:04:48 -04:00
def createrawtransaction_tests ( self ) :
2021-07-12 09:13:29 -04:00
self . log . info ( " Test createrawtransaction " )
2017-12-12 11:56:03 -03:00
# Test `createrawtransaction` required parameters
assert_raises_rpc_error ( - 1 , " createrawtransaction " , self . nodes [ 0 ] . createrawtransaction )
assert_raises_rpc_error ( - 1 , " createrawtransaction " , self . nodes [ 0 ] . createrawtransaction , [ ] )
# Test `createrawtransaction` invalid extra parameters
assert_raises_rpc_error ( - 1 , " createrawtransaction " , self . nodes [ 0 ] . createrawtransaction , [ ] , { } , 0 , False , ' foo ' )
# Test `createrawtransaction` invalid `inputs`
2022-09-12 10:04:15 -03:00
assert_raises_rpc_error ( - 3 , " JSON value of type string is not of expected type array " , self . nodes [ 0 ] . createrawtransaction , ' foo ' , { } )
2022-09-14 12:13:58 -03:00
assert_raises_rpc_error ( - 3 , " JSON value of type string is not of expected type object " , self . nodes [ 0 ] . createrawtransaction , [ ' foo ' ] , { } )
assert_raises_rpc_error ( - 3 , " JSON value of type null is not of expected type string " , self . nodes [ 0 ] . createrawtransaction , [ { } ] , { } )
2018-06-08 14:16:07 -04:00
assert_raises_rpc_error ( - 8 , " txid must be of length 64 (not 3, for ' foo ' ) " , self . nodes [ 0 ] . createrawtransaction , [ { ' txid ' : ' foo ' } ] , { } )
2021-07-12 10:38:36 -04:00
txid = " ZZZ7bb8b1697ea987f3b223ba7819250cae33efacb068d23dc24859824a77844 "
assert_raises_rpc_error ( - 8 , f " txid must be hexadecimal string (not ' { txid } ' ) " , self . nodes [ 0 ] . createrawtransaction , [ { ' txid ' : txid } ] , { } )
2021-07-12 10:12:21 -04:00
assert_raises_rpc_error ( - 8 , " Invalid parameter, missing vout key " , self . nodes [ 0 ] . createrawtransaction , [ { ' txid ' : TXID } ] , { } )
assert_raises_rpc_error ( - 8 , " Invalid parameter, missing vout key " , self . nodes [ 0 ] . createrawtransaction , [ { ' txid ' : TXID , ' vout ' : ' foo ' } ] , { } )
assert_raises_rpc_error ( - 8 , " Invalid parameter, vout cannot be negative " , self . nodes [ 0 ] . createrawtransaction , [ { ' txid ' : TXID , ' vout ' : - 1 } ] , { } )
2021-07-12 08:48:54 -04:00
# sequence number out of range
for invalid_seq in [ - 1 , 4294967296 ] :
2021-07-12 10:12:21 -04:00
inputs = [ { ' txid ' : TXID , ' vout ' : 1 , ' sequence ' : invalid_seq } ]
2022-04-15 14:34:49 -04:00
address = getnewdestination ( ) [ 2 ]
outputs = { address : 1 }
2021-07-12 08:48:54 -04:00
assert_raises_rpc_error ( - 8 , ' Invalid parameter, sequence number is out of range ' ,
self . nodes [ 0 ] . createrawtransaction , inputs , outputs )
# with valid sequence number
for valid_seq in [ 1000 , 4294967294 ] :
2021-07-12 10:12:21 -04:00
inputs = [ { ' txid ' : TXID , ' vout ' : 1 , ' sequence ' : valid_seq } ]
2022-04-15 14:34:49 -04:00
address = getnewdestination ( ) [ 2 ]
outputs = { address : 1 }
2021-07-12 08:48:54 -04:00
rawtx = self . nodes [ 0 ] . createrawtransaction ( inputs , outputs )
decrawtx = self . nodes [ 0 ] . decoderawtransaction ( rawtx )
assert_equal ( decrawtx [ ' vin ' ] [ 0 ] [ ' sequence ' ] , valid_seq )
2017-12-12 11:56:03 -03:00
# Test `createrawtransaction` invalid `outputs`
2022-04-15 14:34:49 -04:00
address = getnewdestination ( ) [ 2 ]
2022-09-14 12:13:58 -03:00
assert_raises_rpc_error ( - 3 , " JSON value of type string is not of expected type array " , self . nodes [ 0 ] . createrawtransaction , [ ] , ' foo ' )
2017-12-11 17:24:02 -03:00
self . nodes [ 0 ] . createrawtransaction ( inputs = [ ] , outputs = { } ) # Should not throw for backwards compatibility
self . nodes [ 0 ] . createrawtransaction ( inputs = [ ] , outputs = [ ] )
2017-12-12 11:56:03 -03:00
assert_raises_rpc_error ( - 8 , " Data must be hexadecimal string " , self . nodes [ 0 ] . createrawtransaction , [ ] , { ' data ' : ' foo ' } )
assert_raises_rpc_error ( - 5 , " Invalid Bitcoin address " , self . nodes [ 0 ] . createrawtransaction , [ ] , { ' foo ' : 0 } )
assert_raises_rpc_error ( - 3 , " Invalid amount " , self . nodes [ 0 ] . createrawtransaction , [ ] , { address : ' foo ' } )
assert_raises_rpc_error ( - 3 , " Amount out of range " , self . nodes [ 0 ] . createrawtransaction , [ ] , { address : - 1 } )
assert_raises_rpc_error ( - 8 , " Invalid parameter, duplicated address: %s " % address , self . nodes [ 0 ] . createrawtransaction , [ ] , multidict ( [ ( address , 1 ) , ( address , 1 ) ] ) )
2017-12-11 17:24:02 -03:00
assert_raises_rpc_error ( - 8 , " Invalid parameter, duplicated address: %s " % address , self . nodes [ 0 ] . createrawtransaction , [ ] , [ { address : 1 } , { address : 1 } ] )
2018-12-06 18:13:53 -03:00
assert_raises_rpc_error ( - 8 , " Invalid parameter, duplicate key: data " , self . nodes [ 0 ] . createrawtransaction , [ ] , [ { " data " : ' aa ' } , { " data " : " bb " } ] )
assert_raises_rpc_error ( - 8 , " Invalid parameter, duplicate key: data " , self . nodes [ 0 ] . createrawtransaction , [ ] , multidict ( [ ( " data " , ' aa ' ) , ( " data " , " bb " ) ] ) )
2017-12-11 17:24:02 -03:00
assert_raises_rpc_error ( - 8 , " Invalid parameter, key-value pair must contain exactly one key " , self . nodes [ 0 ] . createrawtransaction , [ ] , [ { ' a ' : 1 , ' b ' : 2 } ] )
assert_raises_rpc_error ( - 8 , " Invalid parameter, key-value pair not an object as expected " , self . nodes [ 0 ] . createrawtransaction , [ ] , [ [ ' key-value pair1 ' ] , [ ' 2 ' ] ] )
2017-12-12 11:56:03 -03:00
2022-06-01 18:58:47 -04:00
# Test `createrawtransaction` mismatch between sequence number(s) and `replaceable` option
assert_raises_rpc_error ( - 8 , " Invalid parameter combination: Sequence number(s) contradict replaceable option " ,
2022-07-12 11:28:20 -04:00
self . nodes [ 0 ] . createrawtransaction , [ { ' txid ' : TXID , ' vout ' : 0 , ' sequence ' : MAX_BIP125_RBF_SEQUENCE + 1 } ] , { } , 0 , True )
2022-06-01 18:58:47 -04:00
2017-12-12 11:56:03 -03:00
# Test `createrawtransaction` invalid `locktime`
2022-09-12 10:04:15 -03:00
assert_raises_rpc_error ( - 3 , " JSON value of type string is not of expected type number " , self . nodes [ 0 ] . createrawtransaction , [ ] , { } , ' foo ' )
2017-12-12 11:56:03 -03:00
assert_raises_rpc_error ( - 8 , " Invalid parameter, locktime out of range " , self . nodes [ 0 ] . createrawtransaction , [ ] , { } , - 1 )
assert_raises_rpc_error ( - 8 , " Invalid parameter, locktime out of range " , self . nodes [ 0 ] . createrawtransaction , [ ] , { } , 4294967296 )
# Test `createrawtransaction` invalid `replaceable`
2022-09-12 10:04:15 -03:00
assert_raises_rpc_error ( - 3 , " JSON value of type string is not of expected type bool " , self . nodes [ 0 ] . createrawtransaction , [ ] , { } , 0 , ' foo ' )
2017-12-12 11:56:03 -03:00
2021-07-12 09:13:29 -04:00
# Test that createrawtransaction accepts an array and object as outputs
2017-12-11 17:24:02 -03:00
# One output
2021-07-12 10:12:21 -04:00
tx = tx_from_hex ( self . nodes [ 2 ] . createrawtransaction ( inputs = [ { ' txid ' : TXID , ' vout ' : 9 } ] , outputs = { address : 99 } ) )
2017-12-11 17:24:02 -03:00
assert_equal ( len ( tx . vout ) , 1 )
assert_equal (
2019-02-18 12:35:48 -03:00
tx . serialize ( ) . hex ( ) ,
2021-07-12 10:12:21 -04:00
self . nodes [ 2 ] . createrawtransaction ( inputs = [ { ' txid ' : TXID , ' vout ' : 9 } ] , outputs = [ { address : 99 } ] ) ,
2017-12-11 17:24:02 -03:00
)
# Two outputs
2022-04-15 14:34:49 -04:00
address2 = getnewdestination ( ) [ 2 ]
2021-07-12 10:12:21 -04:00
tx = tx_from_hex ( self . nodes [ 2 ] . createrawtransaction ( inputs = [ { ' txid ' : TXID , ' vout ' : 9 } ] , outputs = OrderedDict ( [ ( address , 99 ) , ( address2 , 99 ) ] ) ) )
2017-12-11 17:24:02 -03:00
assert_equal ( len ( tx . vout ) , 2 )
assert_equal (
2019-02-18 12:35:48 -03:00
tx . serialize ( ) . hex ( ) ,
2021-07-12 10:12:21 -04:00
self . nodes [ 2 ] . createrawtransaction ( inputs = [ { ' txid ' : TXID , ' vout ' : 9 } ] , outputs = [ { address : 99 } , { address2 : 99 } ] ) ,
2017-12-11 17:24:02 -03:00
)
# Multiple mixed outputs
2021-07-12 10:12:21 -04:00
tx = tx_from_hex ( self . nodes [ 2 ] . createrawtransaction ( inputs = [ { ' txid ' : TXID , ' vout ' : 9 } ] , outputs = multidict ( [ ( address , 99 ) , ( address2 , 99 ) , ( ' data ' , ' 99 ' ) ] ) ) )
2017-12-11 17:24:02 -03:00
assert_equal ( len ( tx . vout ) , 3 )
assert_equal (
2019-02-18 12:35:48 -03:00
tx . serialize ( ) . hex ( ) ,
2021-07-12 10:12:21 -04:00
self . nodes [ 2 ] . createrawtransaction ( inputs = [ { ' txid ' : TXID , ' vout ' : 9 } ] , outputs = [ { address : 99 } , { address2 : 99 } , { ' data ' : ' 99 ' } ] ) ,
2017-12-11 17:24:02 -03:00
)
2021-07-12 10:04:48 -04:00
def sendrawtransaction_tests ( self ) :
2021-07-12 09:13:29 -04:00
self . log . info ( " Test sendrawtransaction with missing input " )
2021-07-12 10:38:36 -04:00
inputs = [ { ' txid ' : TXID , ' vout ' : 1 } ] # won't exist
2022-04-15 14:34:49 -04:00
address = getnewdestination ( ) [ 2 ]
outputs = { address : 4.998 }
2021-07-12 10:38:36 -04:00
rawtx = self . nodes [ 2 ] . createrawtransaction ( inputs , outputs )
2022-04-15 14:34:49 -04:00
assert_raises_rpc_error ( - 25 , " bad-txns-inputs-missingorspent " , self . nodes [ 2 ] . sendrawtransaction , rawtx )
2015-04-01 10:08:00 -03:00
2021-07-12 10:04:48 -04:00
def sendrawtransaction_testmempoolaccept_tests ( self ) :
2021-07-12 09:42:21 -04:00
self . log . info ( " Test sendrawtransaction/testmempoolaccept with maxfeerate " )
2021-07-12 10:38:36 -04:00
fee_exceeds_max = " Fee exceeds maximum configured by user (e.g. -maxtxfee, maxfeerate) "
2021-07-12 09:42:21 -04:00
# Test a transaction with a small fee.
2022-04-15 14:34:49 -04:00
# Fee rate is 0.00100000 BTC/kvB
tx = self . wallet . create_self_transfer ( fee_rate = Decimal ( ' 0.00100000 ' ) )
2021-07-12 09:42:21 -04:00
# Thus, testmempoolaccept should reject
2022-04-15 14:34:49 -04:00
testres = self . nodes [ 2 ] . testmempoolaccept ( [ tx [ ' hex ' ] ] , 0.00001000 ) [ 0 ]
2021-07-12 09:42:21 -04:00
assert_equal ( testres [ ' allowed ' ] , False )
assert_equal ( testres [ ' reject-reason ' ] , ' max-fee-exceeded ' )
# and sendrawtransaction should throw
2022-04-15 14:34:49 -04:00
assert_raises_rpc_error ( - 25 , fee_exceeds_max , self . nodes [ 2 ] . sendrawtransaction , tx [ ' hex ' ] , 0.00001000 )
2021-07-12 09:42:21 -04:00
# and the following calls should both succeed
2022-04-15 14:34:49 -04:00
testres = self . nodes [ 2 ] . testmempoolaccept ( rawtxs = [ tx [ ' hex ' ] ] ) [ 0 ]
2021-07-12 09:42:21 -04:00
assert_equal ( testres [ ' allowed ' ] , True )
2022-04-15 14:34:49 -04:00
self . nodes [ 2 ] . sendrawtransaction ( hexstring = tx [ ' hex ' ] )
2021-07-12 09:42:21 -04:00
# Test a transaction with a large fee.
2022-04-15 14:34:49 -04:00
# Fee rate is 0.20000000 BTC/kvB
2022-06-21 05:25:54 -04:00
tx = self . wallet . create_self_transfer ( fee_rate = Decimal ( " 0.20000000 " ) )
2021-07-12 09:42:21 -04:00
# Thus, testmempoolaccept should reject
2022-04-15 14:34:49 -04:00
testres = self . nodes [ 2 ] . testmempoolaccept ( [ tx [ ' hex ' ] ] ) [ 0 ]
2021-07-12 09:42:21 -04:00
assert_equal ( testres [ ' allowed ' ] , False )
assert_equal ( testres [ ' reject-reason ' ] , ' max-fee-exceeded ' )
# and sendrawtransaction should throw
2022-04-15 14:34:49 -04:00
assert_raises_rpc_error ( - 25 , fee_exceeds_max , self . nodes [ 2 ] . sendrawtransaction , tx [ ' hex ' ] )
2021-07-12 09:42:21 -04:00
# and the following calls should both succeed
2022-04-15 14:34:49 -04:00
testres = self . nodes [ 2 ] . testmempoolaccept ( rawtxs = [ tx [ ' hex ' ] ] , maxfeerate = ' 0.20000000 ' ) [ 0 ]
2021-07-12 09:42:21 -04:00
assert_equal ( testres [ ' allowed ' ] , True )
2022-04-15 14:34:49 -04:00
self . nodes [ 2 ] . sendrawtransaction ( hexstring = tx [ ' hex ' ] , maxfeerate = ' 0.20000000 ' )
2021-07-12 09:42:21 -04:00
self . log . info ( " Test sendrawtransaction/testmempoolaccept with tx already in the chain " )
2021-08-19 11:10:24 -04:00
self . generate ( self . nodes [ 2 ] , 1 )
2021-07-12 09:42:21 -04:00
for node in self . nodes :
2022-04-15 14:34:49 -04:00
testres = node . testmempoolaccept ( [ tx [ ' hex ' ] ] ) [ 0 ]
2021-07-12 09:42:21 -04:00
assert_equal ( testres [ ' allowed ' ] , False )
assert_equal ( testres [ ' reject-reason ' ] , ' txn-already-known ' )
2022-04-15 14:34:49 -04:00
assert_raises_rpc_error ( - 27 , ' Transaction already in block chain ' , node . sendrawtransaction , tx [ ' hex ' ] )
2021-07-12 09:42:21 -04:00
2021-07-12 10:04:48 -04:00
def decoderawtransaction_tests ( self ) :
2021-07-12 09:42:21 -04:00
self . log . info ( " Test decoderawtransaction " )
# witness transaction
encrawtx = " 010000000001010000000000000072c1a6a246ae63f74f931e8365e15a089c68d61900000000000000000000ffffffff0100e1f50500000000000102616100000000 "
2021-07-12 10:38:36 -04:00
decrawtx = self . nodes [ 0 ] . decoderawtransaction ( encrawtx , True ) # decode as witness transaction
2021-07-12 09:42:21 -04:00
assert_equal ( decrawtx [ ' vout ' ] [ 0 ] [ ' value ' ] , Decimal ( ' 1.00000000 ' ) )
assert_raises_rpc_error ( - 22 , ' TX decode failed ' , self . nodes [ 0 ] . decoderawtransaction , encrawtx , False ) # force decode as non-witness transaction
# non-witness transaction
encrawtx = " 01000000010000000000000072c1a6a246ae63f74f931e8365e15a089c68d61900000000000000000000ffffffff0100e1f505000000000000000000 "
2021-07-12 10:38:36 -04:00
decrawtx = self . nodes [ 0 ] . decoderawtransaction ( encrawtx , False ) # decode as non-witness transaction
2021-07-12 09:42:21 -04:00
assert_equal ( decrawtx [ ' vout ' ] [ 0 ] [ ' value ' ] , Decimal ( ' 1.00000000 ' ) )
# known ambiguous transaction in the chain (see https://github.com/bitcoin/bitcoin/issues/20579)
2021-07-12 10:38:36 -04:00
coinbase = " 03c68708046ff8415c622f4254432e434f4d2ffabe6d6de1965d02c68f928e5b244ab1965115a36f56eb997633c7f690124bbf43644e23080000000ca3d3af6d005a65ff0200fd00000000 "
encrawtx = f " 020000000001010000000000000000000000000000000000000000000000000000000000000000ffffffff4b { coinbase } " \
" ffffffff03f4c1fb4b0000000016001497cfc76442fe717f2a3f0cc9c175f7561b6619970000000000000000266a24aa21a9ed957d1036a80343e0d1b659497e1b48a38ebe876a056d45965fac4a85cda84e1900000000000000002952534b424c4f434b3a8e092581ab01986cbadc84f4b43f4fa4bb9e7a2e2a0caf9b7cf64d939028e22c0120000000000000000000000000000000000000000000000000000000000000000000000000 "
2021-07-12 09:42:21 -04:00
decrawtx = self . nodes [ 0 ] . decoderawtransaction ( encrawtx )
decrawtx_wit = self . nodes [ 0 ] . decoderawtransaction ( encrawtx , True )
2021-07-12 10:38:36 -04:00
assert_raises_rpc_error ( - 22 , ' TX decode failed ' , self . nodes [ 0 ] . decoderawtransaction , encrawtx , False ) # fails to decode as non-witness transaction
assert_equal ( decrawtx , decrawtx_wit ) # the witness interpretation should be chosen
assert_equal ( decrawtx [ ' vin ' ] [ 0 ] [ ' coinbase ' ] , coinbase )
2021-07-12 09:42:21 -04:00
2021-07-12 10:04:48 -04:00
def transaction_version_number_tests ( self ) :
2021-07-12 09:42:21 -04:00
self . log . info ( " Test transaction version numbers " )
# Test the minimum transaction version number that fits in a signed 32-bit integer.
# As transaction version is unsigned, this should convert to its unsigned equivalent.
tx = CTransaction ( )
tx . nVersion = - 0x80000000
rawtx = tx . serialize ( ) . hex ( )
decrawtx = self . nodes [ 0 ] . decoderawtransaction ( rawtx )
assert_equal ( decrawtx [ ' version ' ] , 0x80000000 )
# Test the maximum transaction version number that fits in a signed 32-bit integer.
tx = CTransaction ( )
tx . nVersion = 0x7fffffff
rawtx = tx . serialize ( ) . hex ( )
decrawtx = self . nodes [ 0 ] . decoderawtransaction ( rawtx )
assert_equal ( decrawtx [ ' version ' ] , 0x7fffffff )
2017-05-08 01:24:32 -03:00
2021-07-12 10:04:48 -04:00
def raw_multisig_transaction_legacy_tests ( self ) :
self . log . info ( " Test raw multisig transactions (legacy) " )
# The traditional multisig workflow does not work with descriptor wallets so these are legacy only.
# The multisig workflow with descriptor wallets uses PSBTs and is tested elsewhere, no need to do them here.
2021-07-12 10:38:36 -04:00
2021-07-12 10:04:48 -04:00
# 2of2 test
addr1 = self . nodes [ 2 ] . getnewaddress ( )
addr2 = self . nodes [ 2 ] . getnewaddress ( )
addr1Obj = self . nodes [ 2 ] . getaddressinfo ( addr1 )
addr2Obj = self . nodes [ 2 ] . getaddressinfo ( addr2 )
# Tests for createmultisig and addmultisigaddress
assert_raises_rpc_error ( - 5 , " Invalid public key " , self . nodes [ 0 ] . createmultisig , 1 , [ " 01020304 " ] )
2021-07-12 10:38:36 -04:00
# createmultisig can only take public keys
self . nodes [ 0 ] . createmultisig ( 2 , [ addr1Obj [ ' pubkey ' ] , addr2Obj [ ' pubkey ' ] ] )
# addmultisigaddress can take both pubkeys and addresses so long as they are in the wallet, which is tested here
assert_raises_rpc_error ( - 5 , " Invalid public key " , self . nodes [ 0 ] . createmultisig , 2 , [ addr1Obj [ ' pubkey ' ] , addr1 ] )
2021-07-12 10:04:48 -04:00
mSigObj = self . nodes [ 2 ] . addmultisigaddress ( 2 , [ addr1Obj [ ' pubkey ' ] , addr1 ] ) [ ' address ' ]
2021-07-12 10:38:36 -04:00
# use balance deltas instead of absolute values
2021-07-12 10:04:48 -04:00
bal = self . nodes [ 2 ] . getbalance ( )
# send 1.2 BTC to msig adr
txId = self . nodes [ 0 ] . sendtoaddress ( mSigObj , 1.2 )
self . sync_all ( )
2021-08-19 11:10:24 -04:00
self . generate ( self . nodes [ 0 ] , 1 )
2021-07-12 10:38:36 -04:00
# node2 has both keys of the 2of2 ms addr, tx should affect the balance
assert_equal ( self . nodes [ 2 ] . getbalance ( ) , bal + Decimal ( ' 1.20000000 ' ) )
2021-07-12 10:04:48 -04:00
# 2of3 test from different nodes
bal = self . nodes [ 2 ] . getbalance ( )
addr1 = self . nodes [ 1 ] . getnewaddress ( )
addr2 = self . nodes [ 2 ] . getnewaddress ( )
addr3 = self . nodes [ 2 ] . getnewaddress ( )
addr1Obj = self . nodes [ 1 ] . getaddressinfo ( addr1 )
addr2Obj = self . nodes [ 2 ] . getaddressinfo ( addr2 )
addr3Obj = self . nodes [ 2 ] . getaddressinfo ( addr3 )
mSigObj = self . nodes [ 2 ] . addmultisigaddress ( 2 , [ addr1Obj [ ' pubkey ' ] , addr2Obj [ ' pubkey ' ] , addr3Obj [ ' pubkey ' ] ] ) [ ' address ' ]
txId = self . nodes [ 0 ] . sendtoaddress ( mSigObj , 2.2 )
decTx = self . nodes [ 0 ] . gettransaction ( txId )
rawTx = self . nodes [ 0 ] . decoderawtransaction ( decTx [ ' hex ' ] )
self . sync_all ( )
2021-08-19 11:10:24 -04:00
self . generate ( self . nodes [ 0 ] , 1 )
2021-07-12 10:04:48 -04:00
2021-07-12 10:38:36 -04:00
# THIS IS AN INCOMPLETE FEATURE
# NODE2 HAS TWO OF THREE KEYS AND THE FUNDS SHOULD BE SPENDABLE AND COUNT AT BALANCE CALCULATION
assert_equal ( self . nodes [ 2 ] . getbalance ( ) , bal ) # for now, assume the funds of a 2of3 multisig tx are not marked as spendable
2021-07-12 10:04:48 -04:00
txDetails = self . nodes [ 0 ] . gettransaction ( txId , True )
rawTx = self . nodes [ 0 ] . decoderawtransaction ( txDetails [ ' hex ' ] )
vout = next ( o for o in rawTx [ ' vout ' ] if o [ ' value ' ] == Decimal ( ' 2.20000000 ' ) )
bal = self . nodes [ 0 ] . getbalance ( )
2021-07-12 10:38:36 -04:00
inputs = [ { " txid " : txId , " vout " : vout [ ' n ' ] , " scriptPubKey " : vout [ ' scriptPubKey ' ] [ ' hex ' ] , " amount " : vout [ ' value ' ] } ]
outputs = { self . nodes [ 0 ] . getnewaddress ( ) : 2.19 }
2021-07-12 10:04:48 -04:00
rawTx = self . nodes [ 2 ] . createrawtransaction ( inputs , outputs )
rawTxPartialSigned = self . nodes [ 1 ] . signrawtransactionwithwallet ( rawTx , inputs )
2021-07-12 10:38:36 -04:00
assert_equal ( rawTxPartialSigned [ ' complete ' ] , False ) # node1 only has one key, can't comp. sign the tx
2021-07-12 10:04:48 -04:00
rawTxSigned = self . nodes [ 2 ] . signrawtransactionwithwallet ( rawTx , inputs )
2021-07-12 10:38:36 -04:00
assert_equal ( rawTxSigned [ ' complete ' ] , True ) # node2 can sign the tx compl., own two of three keys
2021-07-12 10:04:48 -04:00
self . nodes [ 2 ] . sendrawtransaction ( rawTxSigned [ ' hex ' ] )
rawTx = self . nodes [ 0 ] . decoderawtransaction ( rawTxSigned [ ' hex ' ] )
self . sync_all ( )
2021-08-19 11:10:24 -04:00
self . generate ( self . nodes [ 0 ] , 1 )
2021-07-12 10:38:36 -04:00
assert_equal ( self . nodes [ 0 ] . getbalance ( ) , bal + Decimal ( ' 50.00000000 ' ) + Decimal ( ' 2.19000000 ' ) ) # block reward + tx
2021-07-12 10:04:48 -04:00
# 2of2 test for combining transactions
bal = self . nodes [ 2 ] . getbalance ( )
addr1 = self . nodes [ 1 ] . getnewaddress ( )
addr2 = self . nodes [ 2 ] . getnewaddress ( )
addr1Obj = self . nodes [ 1 ] . getaddressinfo ( addr1 )
addr2Obj = self . nodes [ 2 ] . getaddressinfo ( addr2 )
self . nodes [ 1 ] . addmultisigaddress ( 2 , [ addr1Obj [ ' pubkey ' ] , addr2Obj [ ' pubkey ' ] ] ) [ ' address ' ]
mSigObj = self . nodes [ 2 ] . addmultisigaddress ( 2 , [ addr1Obj [ ' pubkey ' ] , addr2Obj [ ' pubkey ' ] ] ) [ ' address ' ]
mSigObjValid = self . nodes [ 2 ] . getaddressinfo ( mSigObj )
txId = self . nodes [ 0 ] . sendtoaddress ( mSigObj , 2.2 )
decTx = self . nodes [ 0 ] . gettransaction ( txId )
rawTx2 = self . nodes [ 0 ] . decoderawtransaction ( decTx [ ' hex ' ] )
self . sync_all ( )
2021-08-19 11:10:24 -04:00
self . generate ( self . nodes [ 0 ] , 1 )
2021-07-12 10:04:48 -04:00
2021-07-12 10:38:36 -04:00
assert_equal ( self . nodes [ 2 ] . getbalance ( ) , bal ) # the funds of a 2of2 multisig tx should not be marked as spendable
2021-07-12 10:04:48 -04:00
txDetails = self . nodes [ 0 ] . gettransaction ( txId , True )
rawTx2 = self . nodes [ 0 ] . decoderawtransaction ( txDetails [ ' hex ' ] )
vout = next ( o for o in rawTx2 [ ' vout ' ] if o [ ' value ' ] == Decimal ( ' 2.20000000 ' ) )
bal = self . nodes [ 0 ] . getbalance ( )
2021-07-12 10:38:36 -04:00
inputs = [ { " txid " : txId , " vout " : vout [ ' n ' ] , " scriptPubKey " : vout [ ' scriptPubKey ' ] [ ' hex ' ] , " redeemScript " : mSigObjValid [ ' hex ' ] , " amount " : vout [ ' value ' ] } ]
outputs = { self . nodes [ 0 ] . getnewaddress ( ) : 2.19 }
2021-07-12 10:04:48 -04:00
rawTx2 = self . nodes [ 2 ] . createrawtransaction ( inputs , outputs )
rawTxPartialSigned1 = self . nodes [ 1 ] . signrawtransactionwithwallet ( rawTx2 , inputs )
self . log . debug ( rawTxPartialSigned1 )
2021-07-12 10:38:36 -04:00
assert_equal ( rawTxPartialSigned1 [ ' complete ' ] , False ) # node1 only has one key, can't comp. sign the tx
2021-07-12 10:04:48 -04:00
rawTxPartialSigned2 = self . nodes [ 2 ] . signrawtransactionwithwallet ( rawTx2 , inputs )
self . log . debug ( rawTxPartialSigned2 )
2021-07-12 10:38:36 -04:00
assert_equal ( rawTxPartialSigned2 [ ' complete ' ] , False ) # node2 only has one key, can't comp. sign the tx
2021-07-12 10:04:48 -04:00
rawTxComb = self . nodes [ 2 ] . combinerawtransaction ( [ rawTxPartialSigned1 [ ' hex ' ] , rawTxPartialSigned2 [ ' hex ' ] ] )
self . log . debug ( rawTxComb )
self . nodes [ 2 ] . sendrawtransaction ( rawTxComb )
rawTx2 = self . nodes [ 0 ] . decoderawtransaction ( rawTxComb )
self . sync_all ( )
2021-08-19 11:10:24 -04:00
self . generate ( self . nodes [ 0 ] , 1 )
2021-07-12 10:38:36 -04:00
assert_equal ( self . nodes [ 0 ] . getbalance ( ) , bal + Decimal ( ' 50.00000000 ' ) + Decimal ( ' 2.19000000 ' ) ) # block reward + tx
2017-06-10 01:38:06 -04:00
2018-06-27 04:21:07 -04:00
2014-12-29 16:17:21 -03:00
if __name__ == ' __main__ ' :
RawTransactionsTest ( ) . main ( )