2020-08-07 17:36:36 +02:00
#!/usr/bin/env python3
2022-12-24 23:49:50 +00:00
# Copyright (c) 2020-2022 The Bitcoin Core developers
2020-08-07 17:36:36 +02:00
# Distributed under the MIT software license, see the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
""" Test the send RPC command. """
from decimal import Decimal , getcontext
2020-11-05 06:34:01 +01:00
from itertools import product
2020-08-07 17:36:36 +02:00
from test_framework . authproxy import JSONRPCException
2020-11-06 15:34:41 -05:00
from test_framework . descriptors import descsum_create
2020-08-07 17:36:36 +02:00
from test_framework . test_framework import BitcoinTestFramework
from test_framework . util import (
assert_equal ,
assert_fee_amount ,
assert_greater_than ,
2020-11-05 06:34:01 +01:00
assert_raises_rpc_error ,
2022-02-02 17:52:05 +01:00
count_bytes ,
2020-08-07 17:36:36 +02:00
)
2024-04-01 14:03:35 +02:00
from test_framework . wallet_util import (
calculate_input_weight ,
generate_keypair ,
)
2023-05-23 23:38:31 +02:00
2020-08-07 17:36:36 +02:00
class WalletSendTest ( BitcoinTestFramework ) :
2022-11-09 12:53:13 +01:00
def add_options ( self , parser ) :
self . add_wallet_options ( parser )
2020-08-07 17:36:36 +02:00
def set_test_params ( self ) :
self . num_nodes = 2
2023-03-14 10:18:47 -03:00
# whitelist peers to speed up tx relay / mempool sync
self . noban_tx_relay = True
2020-08-07 17:36:36 +02:00
self . extra_args = [
2023-03-14 10:18:47 -03:00
[ " -walletrbf=1 " ] ,
[ " -walletrbf=1 " ]
2020-08-07 17:36:36 +02:00
]
getcontext ( ) . prec = 8 # Satoshi precision for Decimal
def skip_test_if_missing_module ( self ) :
self . skip_if_no_wallet ( )
def test_send ( self , from_wallet , to_wallet = None , amount = None , data = None ,
2020-11-04 13:13:17 +01:00
arg_conf_target = None , arg_estimate_mode = None , arg_fee_rate = None ,
conf_target = None , estimate_mode = None , fee_rate = None , add_to_wallet = None , psbt = None ,
2021-03-10 15:37:18 +01:00
inputs = None , add_inputs = None , include_unsafe = None , change_address = None , change_position = None , change_type = None ,
2020-09-17 15:28:59 +02:00
include_watching = None , locktime = None , lock_unspents = None , replaceable = None , subtract_fee_from_outputs = None ,
2021-05-24 10:22:10 -03:00
expect_error = None , solving_data = None , minconf = None ) :
2020-08-07 17:36:36 +02:00
assert ( amount is None ) != ( data is None )
2021-03-10 15:37:18 +01:00
from_balance_before = from_wallet . getbalances ( ) [ " mine " ] [ " trusted " ]
if include_unsafe :
from_balance_before + = from_wallet . getbalances ( ) [ " mine " ] [ " untrusted_pending " ]
2020-08-07 17:36:36 +02:00
if to_wallet is None :
assert amount is None
else :
to_untrusted_pending_before = to_wallet . getbalances ( ) [ " mine " ] [ " untrusted_pending " ]
if amount :
dest = to_wallet . getnewaddress ( )
outputs = { dest : amount }
else :
outputs = { " data " : data }
# Construct options dictionary
options = { }
if add_to_wallet is not None :
options [ " add_to_wallet " ] = add_to_wallet
else :
if psbt :
add_to_wallet = False
else :
add_to_wallet = from_wallet . getwalletinfo ( ) [ " private_keys_enabled " ] # Default value
if psbt is not None :
options [ " psbt " ] = psbt
if conf_target is not None :
options [ " conf_target " ] = conf_target
if estimate_mode is not None :
options [ " estimate_mode " ] = estimate_mode
2020-11-04 13:13:17 +01:00
if fee_rate is not None :
options [ " fee_rate " ] = fee_rate
2020-08-07 17:36:36 +02:00
if inputs is not None :
options [ " inputs " ] = inputs
if add_inputs is not None :
options [ " add_inputs " ] = add_inputs
2021-03-10 15:37:18 +01:00
if include_unsafe is not None :
options [ " include_unsafe " ] = include_unsafe
2020-08-07 17:36:36 +02:00
if change_address is not None :
options [ " change_address " ] = change_address
if change_position is not None :
options [ " change_position " ] = change_position
if change_type is not None :
options [ " change_type " ] = change_type
if include_watching is not None :
options [ " include_watching " ] = include_watching
if locktime is not None :
options [ " locktime " ] = locktime
if lock_unspents is not None :
options [ " lock_unspents " ] = lock_unspents
if replaceable is None :
replaceable = True # default
else :
options [ " replaceable " ] = replaceable
if subtract_fee_from_outputs is not None :
options [ " subtract_fee_from_outputs " ] = subtract_fee_from_outputs
2021-01-12 15:17:29 -05:00
if solving_data is not None :
options [ " solving_data " ] = solving_data
2021-05-24 10:22:10 -03:00
if minconf is not None :
options [ " minconf " ] = minconf
2020-08-07 17:36:36 +02:00
if len ( options . keys ( ) ) == 0 :
options = None
if expect_error is None :
2020-11-04 13:13:17 +01:00
res = from_wallet . send ( outputs = outputs , conf_target = arg_conf_target , estimate_mode = arg_estimate_mode , fee_rate = arg_fee_rate , options = options )
2020-08-07 17:36:36 +02:00
else :
try :
2020-09-17 15:28:59 +02:00
assert_raises_rpc_error ( expect_error [ 0 ] , expect_error [ 1 ] , from_wallet . send ,
2020-11-04 13:13:17 +01:00
outputs = outputs , conf_target = arg_conf_target , estimate_mode = arg_estimate_mode , fee_rate = arg_fee_rate , options = options )
2020-08-07 17:36:36 +02:00
except AssertionError :
# Provide debug info if the test fails
self . log . error ( " Unexpected successful result: " )
2020-10-25 20:53:38 +01:00
self . log . error ( arg_conf_target )
self . log . error ( arg_estimate_mode )
2020-11-04 13:13:17 +01:00
self . log . error ( arg_fee_rate )
2020-08-07 17:36:36 +02:00
self . log . error ( options )
2020-11-04 13:13:17 +01:00
res = from_wallet . send ( outputs = outputs , conf_target = arg_conf_target , estimate_mode = arg_estimate_mode , fee_rate = arg_fee_rate , options = options )
2020-08-07 17:36:36 +02:00
self . log . error ( res )
if " txid " in res and add_to_wallet :
self . log . error ( " Transaction details: " )
try :
tx = from_wallet . gettransaction ( res [ " txid " ] )
self . log . error ( tx )
self . log . error ( " testmempoolaccept (transaction may already be in mempool): " )
self . log . error ( from_wallet . testmempoolaccept ( [ tx [ " hex " ] ] ) )
except JSONRPCException as exc :
self . log . error ( exc )
raise
return
if locktime :
return res
if from_wallet . getwalletinfo ( ) [ " private_keys_enabled " ] and not include_watching :
assert_equal ( res [ " complete " ] , True )
assert " txid " in res
else :
assert_equal ( res [ " complete " ] , False )
assert not " txid " in res
assert " psbt " in res
2021-03-10 15:37:18 +01:00
from_balance = from_wallet . getbalances ( ) [ " mine " ] [ " trusted " ]
if include_unsafe :
from_balance + = from_wallet . getbalances ( ) [ " mine " ] [ " untrusted_pending " ]
2020-08-07 17:36:36 +02:00
if add_to_wallet and not include_watching :
# Ensure transaction exists in the wallet:
tx = from_wallet . gettransaction ( res [ " txid " ] )
assert tx
assert_equal ( tx [ " bip125-replaceable " ] , " yes " if replaceable else " no " )
# Ensure transaction exists in the mempool:
2020-09-17 15:28:59 +02:00
tx = from_wallet . getrawtransaction ( res [ " txid " ] , True )
2020-08-07 17:36:36 +02:00
assert tx
if amount :
if subtract_fee_from_outputs :
2021-03-10 15:37:18 +01:00
assert_equal ( from_balance_before - from_balance , amount )
2020-08-07 17:36:36 +02:00
else :
2021-03-10 15:37:18 +01:00
assert_greater_than ( from_balance_before - from_balance , amount )
2020-08-07 17:36:36 +02:00
else :
assert next ( ( out for out in tx [ " vout " ] if out [ " scriptPubKey " ] [ " asm " ] == " OP_RETURN 35 " ) , None )
else :
2021-03-10 15:37:18 +01:00
assert_equal ( from_balance_before , from_balance )
2020-08-07 17:36:36 +02:00
if to_wallet :
self . sync_mempools ( )
if add_to_wallet :
if not subtract_fee_from_outputs :
assert_equal ( to_wallet . getbalances ( ) [ " mine " ] [ " untrusted_pending " ] , to_untrusted_pending_before + Decimal ( amount if amount else 0 ) )
else :
assert_equal ( to_wallet . getbalances ( ) [ " mine " ] [ " untrusted_pending " ] , to_untrusted_pending_before )
return res
def run_test ( self ) :
self . log . info ( " Setup wallets... " )
# w0 is a wallet with coinbase rewards
2020-09-28 20:24:06 -04:00
w0 = self . nodes [ 0 ] . get_wallet_rpc ( self . default_wallet_name )
2020-08-07 17:36:36 +02:00
# w1 is a regular wallet
self . nodes [ 1 ] . createwallet ( wallet_name = " w1 " )
w1 = self . nodes [ 1 ] . get_wallet_rpc ( " w1 " )
# w2 contains the private keys for w3
2020-11-06 15:34:41 -05:00
self . nodes [ 1 ] . createwallet ( wallet_name = " w2 " , blank = True )
2020-08-07 17:36:36 +02:00
w2 = self . nodes [ 1 ] . get_wallet_rpc ( " w2 " )
2020-11-06 15:34:41 -05:00
xpriv = " tprv8ZgxMBicQKsPfHCsTwkiM1KT56RXbGGTqvc2hgqzycpwbHqqpcajQeMRZoBD35kW4RtyCemu6j34Ku5DEspmgjKdt2qe4SvRch5Kk8B8A2v "
xpub = " tpubD6NzVbkrYhZ4YkEfMbRJkQyZe7wTkbTNRECozCtJPtdLRn6cT1QKb8yHjwAPcAr26eHBFYs5iLiFFnCbwPRsncCKUKCfubHDMGKzMVcN1Jg "
if self . options . descriptors :
w2 . importdescriptors ( [ {
" desc " : descsum_create ( " wpkh( " + xpriv + " /0/0/*) " ) ,
" timestamp " : " now " ,
" range " : [ 0 , 100 ] ,
" active " : True
} , {
" desc " : descsum_create ( " wpkh( " + xpriv + " /0/1/*) " ) ,
" timestamp " : " now " ,
" range " : [ 0 , 100 ] ,
" active " : True ,
" internal " : True
} ] )
else :
w2 . sethdseed ( True )
2020-08-07 17:36:36 +02:00
# w3 is a watch-only wallet, based on w2
2020-09-17 15:28:59 +02:00
self . nodes [ 1 ] . createwallet ( wallet_name = " w3 " , disable_private_keys = True )
2020-08-07 17:36:36 +02:00
w3 = self . nodes [ 1 ] . get_wallet_rpc ( " w3 " )
2020-11-06 15:34:41 -05:00
if self . options . descriptors :
# Match the privkeys in w2 for descriptors
res = w3 . importdescriptors ( [ {
" desc " : descsum_create ( " wpkh( " + xpub + " /0/0/*) " ) ,
2020-08-07 17:36:36 +02:00
" timestamp " : " now " ,
2020-11-06 15:34:41 -05:00
" range " : [ 0 , 100 ] ,
2020-08-07 17:36:36 +02:00
" keypool " : True ,
2020-11-06 15:34:41 -05:00
" active " : True ,
2020-08-07 17:36:36 +02:00
" watchonly " : True
} , {
2020-11-06 15:34:41 -05:00
" desc " : descsum_create ( " wpkh( " + xpub + " /0/1/*) " ) ,
2020-08-07 17:36:36 +02:00
" timestamp " : " now " ,
2020-11-06 15:34:41 -05:00
" range " : [ 0 , 100 ] ,
2020-08-07 17:36:36 +02:00
" keypool " : True ,
2020-11-06 15:34:41 -05:00
" active " : True ,
2020-08-07 17:36:36 +02:00
" internal " : True ,
" watchonly " : True
} ] )
assert_equal ( res , [ { " success " : True } , { " success " : True } ] )
for _ in range ( 3 ) :
a2_receive = w2 . getnewaddress ( )
2020-11-06 15:34:41 -05:00
if not self . options . descriptors :
# Because legacy wallets use exclusively hardened derivation, we can't do a ranged import like we do for descriptors
a2_change = w2 . getrawchangeaddress ( ) # doesn't actually use change derivation
res = w3 . importmulti ( [ {
" desc " : w2 . getaddressinfo ( a2_receive ) [ " desc " ] ,
" timestamp " : " now " ,
" keypool " : True ,
" watchonly " : True
} , {
" desc " : w2 . getaddressinfo ( a2_change ) [ " desc " ] ,
" timestamp " : " now " ,
" keypool " : True ,
" internal " : True ,
" watchonly " : True
} ] )
assert_equal ( res , [ { " success " : True } , { " success " : True } ] )
2020-08-07 17:36:36 +02:00
2020-11-06 15:34:41 -05:00
w0 . sendtoaddress ( a2_receive , 10 ) # fund w3
2021-08-19 17:10:24 +02:00
self . generate ( self . nodes [ 0 ] , 1 )
2020-08-07 17:36:36 +02:00
2020-11-06 15:34:41 -05:00
if not self . options . descriptors :
# w4 has private keys enabled, but only contains watch-only keys (from w2)
# This is legacy wallet behavior only as descriptor wallets don't allow watchonly and non-watchonly things in the same wallet.
self . nodes [ 1 ] . createwallet ( wallet_name = " w4 " , disable_private_keys = False )
w4 = self . nodes [ 1 ] . get_wallet_rpc ( " w4 " )
for _ in range ( 3 ) :
a2_receive = w2 . getnewaddress ( )
res = w4 . importmulti ( [ {
" desc " : w2 . getaddressinfo ( a2_receive ) [ " desc " ] ,
" timestamp " : " now " ,
" keypool " : False ,
" watchonly " : True
} ] )
assert_equal ( res , [ { " success " : True } ] )
w0 . sendtoaddress ( a2_receive , 10 ) # fund w4
2021-08-19 17:10:24 +02:00
self . generate ( self . nodes [ 0 ] , 1 )
2020-11-06 15:34:41 -05:00
2020-08-07 17:36:36 +02:00
self . log . info ( " Send to address... " )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , add_to_wallet = True )
self . log . info ( " Don ' t broadcast... " )
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , add_to_wallet = False )
2022-10-04 15:18:42 +02:00
assert res [ " hex " ]
2020-08-07 17:36:36 +02:00
self . log . info ( " Return PSBT... " )
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , psbt = True )
2022-10-04 15:18:42 +02:00
assert res [ " psbt " ]
2020-08-07 17:36:36 +02:00
self . log . info ( " Create transaction that spends to address, but don ' t broadcast... " )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , add_to_wallet = False )
# conf_target & estimate_mode can be set as argument or option
res1 = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_conf_target = 1 , arg_estimate_mode = " economical " , add_to_wallet = False )
res2 = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , conf_target = 1 , estimate_mode = " economical " , add_to_wallet = False )
assert_equal ( self . nodes [ 1 ] . decodepsbt ( res1 [ " psbt " ] ) [ " fee " ] ,
self . nodes [ 1 ] . decodepsbt ( res2 [ " psbt " ] ) [ " fee " ] )
# but not at the same time
2020-11-04 13:13:17 +01:00
for mode in [ " unset " , " economical " , " conservative " ] :
2020-10-25 20:53:38 +01:00
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_conf_target = 1 , arg_estimate_mode = " economical " ,
conf_target = 1 , estimate_mode = mode , add_to_wallet = False ,
2020-11-04 13:13:17 +01:00
expect_error = ( - 8 , " Pass conf_target and estimate_mode either as arguments or in the options object, but not both " ) )
2020-08-07 17:36:36 +02:00
self . log . info ( " Create PSBT from watch-only wallet w3, sign with w2... " )
res = self . test_send ( from_wallet = w3 , to_wallet = w1 , amount = 1 )
res = w2 . walletprocesspsbt ( res [ " psbt " ] )
assert res [ " complete " ]
2020-11-06 15:34:41 -05:00
if not self . options . descriptors :
# Descriptor wallets do not allow mixed watch-only and non-watch-only things in the same wallet.
# This is specifically testing that w4 ignores its own private keys and creates a psbt with send
# which is not something that needs to be tested in descriptor wallets.
self . log . info ( " Create PSBT from wallet w4 with watch-only keys, sign with w2... " )
self . test_send ( from_wallet = w4 , to_wallet = w1 , amount = 1 , expect_error = ( - 4 , " Insufficient funds " ) )
res = self . test_send ( from_wallet = w4 , to_wallet = w1 , amount = 1 , include_watching = True , add_to_wallet = False )
res = w2 . walletprocesspsbt ( res [ " psbt " ] )
assert res [ " complete " ]
2020-08-07 17:36:36 +02:00
self . log . info ( " Create OP_RETURN... " )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 )
self . test_send ( from_wallet = w0 , data = " Hello World " , expect_error = ( - 8 , " Data must be hexadecimal string (not ' Hello World ' ) " ) )
self . test_send ( from_wallet = w0 , data = " 23 " )
res = self . test_send ( from_wallet = w3 , data = " 23 " )
res = w2 . walletprocesspsbt ( res [ " psbt " ] )
assert res [ " complete " ]
2020-10-25 20:53:38 +01:00
self . log . info ( " Test setting explicit fee rate " )
2020-12-04 11:20:39 +01:00
res1 = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_fee_rate = " 1 " , add_to_wallet = False )
res2 = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , fee_rate = " 1 " , add_to_wallet = False )
2020-10-25 20:53:38 +01:00
assert_equal ( self . nodes [ 1 ] . decodepsbt ( res1 [ " psbt " ] ) [ " fee " ] , self . nodes [ 1 ] . decodepsbt ( res2 [ " psbt " ] ) [ " fee " ] )
2020-11-17 20:08:30 +01:00
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , fee_rate = 7 , add_to_wallet = False )
2020-10-25 20:53:38 +01:00
fee = self . nodes [ 1 ] . decodepsbt ( res [ " psbt " ] ) [ " fee " ]
2022-02-02 17:52:05 +01:00
assert_fee_amount ( fee , count_bytes ( res [ " hex " ] ) , Decimal ( " 0.00007 " ) )
2020-10-25 20:53:38 +01:00
2020-11-17 20:08:30 +01:00
# "unset" and None are treated the same for estimate_mode
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , fee_rate = 2 , estimate_mode = " unset " , add_to_wallet = False )
2020-08-07 17:36:36 +02:00
fee = self . nodes [ 1 ] . decodepsbt ( res [ " psbt " ] ) [ " fee " ]
2022-02-02 17:52:05 +01:00
assert_fee_amount ( fee , count_bytes ( res [ " hex " ] ) , Decimal ( " 0.00002 " ) )
2020-10-25 20:53:38 +01:00
2020-11-17 20:08:30 +01:00
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_fee_rate = 4.531 , add_to_wallet = False )
2020-10-25 20:53:38 +01:00
fee = self . nodes [ 1 ] . decodepsbt ( res [ " psbt " ] ) [ " fee " ]
2022-02-02 17:52:05 +01:00
assert_fee_amount ( fee , count_bytes ( res [ " hex " ] ) , Decimal ( " 0.00004531 " ) )
2020-10-25 20:53:38 +01:00
2020-11-04 13:13:17 +01:00
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_fee_rate = 3 , add_to_wallet = False )
2020-10-25 20:53:38 +01:00
fee = self . nodes [ 1 ] . decodepsbt ( res [ " psbt " ] ) [ " fee " ]
2022-02-02 17:52:05 +01:00
assert_fee_amount ( fee , count_bytes ( res [ " hex " ] ) , Decimal ( " 0.00003 " ) )
2020-10-25 20:53:38 +01:00
2020-11-04 13:13:17 +01:00
# Test that passing fee_rate as both an argument and an option raises.
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_fee_rate = 1 , fee_rate = 1 , add_to_wallet = False ,
expect_error = ( - 8 , " Pass the fee_rate either as an argument, or in the options object, but not both " ) )
assert_raises_rpc_error ( - 8 , " Use fee_rate (sat/vB) instead of feeRate " , w0 . send , { w1 . getnewaddress ( ) : 1 } , 6 , " conservative " , 1 , { " feeRate " : 0.01 } )
assert_raises_rpc_error ( - 3 , " Unexpected key totalFee " , w0 . send , { w1 . getnewaddress ( ) : 1 } , 6 , " conservative " , 1 , { " totalFee " : 0.01 } )
2020-11-05 06:34:01 +01:00
for target , mode in product ( [ - 1 , 0 , 1009 ] , [ " economical " , " conservative " ] ) :
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , conf_target = target , estimate_mode = mode ,
2020-11-04 13:13:17 +01:00
expect_error = ( - 8 , " Invalid conf_target, must be between 1 and 1008 " ) ) # max value of 1008 per src/policy/fees.h
2020-11-10 12:29:01 +01:00
msg = ' Invalid estimate_mode parameter, must be one of: " unset " , " economical " , " conservative " '
2020-11-04 13:13:17 +01:00
for target , mode in product ( [ - 1 , 0 ] , [ " btc/kb " , " sat/b " ] ) :
2020-11-10 12:29:01 +01:00
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , conf_target = target , estimate_mode = mode , expect_error = ( - 8 , msg ) )
2020-11-04 13:13:17 +01:00
for mode in [ " " , " foo " , Decimal ( " 3.141592 " ) ] :
2020-11-10 12:29:01 +01:00
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , conf_target = 0.1 , estimate_mode = mode , expect_error = ( - 8 , msg ) )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_conf_target = 0.1 , arg_estimate_mode = mode , expect_error = ( - 8 , msg ) )
assert_raises_rpc_error ( - 8 , msg , w0 . send , { w1 . getnewaddress ( ) : 1 } , 0.1 , mode )
2020-11-05 06:34:01 +01:00
2020-12-04 12:51:34 +01:00
for mode in [ " economical " , " conservative " ] :
for k , v in { " string " : " true " , " bool " : True , " object " : { " foo " : " bar " } } . items ( ) :
2020-11-05 06:34:01 +01:00
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , conf_target = v , estimate_mode = mode ,
2022-10-04 13:51:24 -03:00
expect_error = ( - 3 , f " JSON value of type { k } for field conf_target is not of expected type number " ) )
2020-10-25 20:53:38 +01:00
2020-12-04 11:28:47 +01:00
# Test setting explicit fee rate just below the minimum of 1 sat/vB.
2020-11-04 13:13:17 +01:00
self . log . info ( " Explicit fee rate raises RPC error ' fee rate too low ' if fee_rate of 0.99999999 is passed " )
2021-04-27 11:04:10 +02:00
msg = " Fee rate (0.999 sat/vB) is lower than the minimum fee rate setting (1.000 sat/vB) "
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , fee_rate = 0.999 , expect_error = ( - 4 , msg ) )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_fee_rate = 0.999 , expect_error = ( - 4 , msg ) )
self . log . info ( " Explicit fee rate raises if invalid fee_rate is passed " )
2020-12-04 11:28:47 +01:00
# Test fee_rate with zero values.
msg = " Fee rate (0.000 sat/vB) is lower than the minimum fee rate setting (1.000 sat/vB) "
for zero_value in [ 0 , 0.000 , 0.00000000 , " 0 " , " 0.000 " , " 0.00000000 " ] :
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , fee_rate = zero_value , expect_error = ( - 4 , msg ) )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_fee_rate = zero_value , expect_error = ( - 4 , msg ) )
2020-12-03 10:55:15 +01:00
msg = " Invalid amount "
# Test fee_rate values that don't pass fixed-point parsing checks.
for invalid_value in [ " " , 0.000000001 , 1e-09 , 1.111111111 , 1111111111111111 , " 31.999999999999999999999 " ] :
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , fee_rate = invalid_value , expect_error = ( - 3 , msg ) )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_fee_rate = invalid_value , expect_error = ( - 3 , msg ) )
2020-12-02 13:55:26 +01:00
# Test fee_rate values that cannot be represented in sat/vB.
2023-07-26 19:14:19 -06:00
for invalid_value in [ 0.0001 , 0.00000001 , 0.00099999 , 31.99999999 ] :
2020-12-02 13:55:26 +01:00
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , fee_rate = invalid_value , expect_error = ( - 3 , msg ) )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_fee_rate = invalid_value , expect_error = ( - 3 , msg ) )
2020-12-04 12:51:34 +01:00
# Test fee_rate out of range (negative number).
msg = " Amount out of range "
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , fee_rate = - 1 , expect_error = ( - 3 , msg ) )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_fee_rate = - 1 , expect_error = ( - 3 , msg ) )
# Test type error.
msg = " Amount is not a number or string "
for invalid_value in [ True , { " foo " : " bar " } ] :
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , fee_rate = invalid_value , expect_error = ( - 3 , msg ) )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , arg_fee_rate = invalid_value , expect_error = ( - 3 , msg ) )
2020-08-07 17:36:36 +02:00
# TODO: Return hex if fee rate is below -maxmempool
# res = self.test_send(from_wallet=w0, to_wallet=w1, amount=1, conf_target=0.1, estimate_mode="sat/b", add_to_wallet=False)
# assert res["hex"]
# hex = res["hex"]
# res = self.nodes[0].testmempoolaccept([hex])
# assert not res[0]["allowed"]
# assert_equal(res[0]["reject-reason"], "...") # low fee
# assert_fee_amount(fee, Decimal(len(res["hex"]) / 2), Decimal("0.000001"))
self . log . info ( " If inputs are specified, do not automatically add more... " )
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 51 , inputs = [ ] , add_to_wallet = False )
assert res [ " complete " ]
utxo1 = w0 . listunspent ( ) [ 0 ]
assert_equal ( utxo1 [ " amount " ] , 50 )
2022-12-07 14:35:46 -03:00
ERR_NOT_ENOUGH_PRESET_INPUTS = " The preselected coins total amount does not cover the transaction target. " \
" Please allow other inputs to be automatically selected or include more coins manually "
2020-08-07 17:36:36 +02:00
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 51 , inputs = [ utxo1 ] ,
2022-12-07 14:35:46 -03:00
expect_error = ( - 4 , ERR_NOT_ENOUGH_PRESET_INPUTS ) )
2020-08-07 17:36:36 +02:00
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 51 , inputs = [ utxo1 ] , add_inputs = False ,
2022-12-07 14:35:46 -03:00
expect_error = ( - 4 , ERR_NOT_ENOUGH_PRESET_INPUTS ) )
2020-08-07 17:36:36 +02:00
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 51 , inputs = [ utxo1 ] , add_inputs = True , add_to_wallet = False )
assert res [ " complete " ]
self . log . info ( " Manual change address and position... " )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , change_address = " not an address " ,
expect_error = ( - 5 , " Change address must be a valid bitcoin address " ) )
change_address = w0 . getnewaddress ( )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , add_to_wallet = False , change_address = change_address )
assert res [ " complete " ]
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , add_to_wallet = False , change_address = change_address , change_position = 0 )
assert res [ " complete " ]
2021-02-01 09:52:07 -06:00
assert_equal ( self . nodes [ 0 ] . decodepsbt ( res [ " psbt " ] ) [ " tx " ] [ " vout " ] [ 0 ] [ " scriptPubKey " ] [ " address " ] , change_address )
2020-08-07 17:36:36 +02:00
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , add_to_wallet = False , change_type = " legacy " , change_position = 0 )
assert res [ " complete " ]
2021-02-01 09:52:07 -06:00
change_address = self . nodes [ 0 ] . decodepsbt ( res [ " psbt " ] ) [ " tx " ] [ " vout " ] [ 0 ] [ " scriptPubKey " ] [ " address " ]
2020-08-07 17:36:36 +02:00
assert change_address [ 0 ] == " m " or change_address [ 0 ] == " n "
self . log . info ( " Set lock time... " )
height = self . nodes [ 0 ] . getblockchaininfo ( ) [ " blocks " ]
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , locktime = height + 1 )
assert res [ " complete " ]
assert res [ " txid " ]
txid = res [ " txid " ]
# Although the wallet finishes the transaction, it can't be added to the mempool yet:
hex = self . nodes [ 0 ] . gettransaction ( res [ " txid " ] ) [ " hex " ]
res = self . nodes [ 0 ] . testmempoolaccept ( [ hex ] )
assert not res [ 0 ] [ " allowed " ]
assert_equal ( res [ 0 ] [ " reject-reason " ] , " non-final " )
# It shouldn't be confirmed in the next block
2021-08-19 17:10:24 +02:00
self . generate ( self . nodes [ 0 ] , 1 )
2020-08-07 17:36:36 +02:00
assert_equal ( self . nodes [ 0 ] . gettransaction ( txid ) [ " confirmations " ] , 0 )
# The mempool should allow it now:
res = self . nodes [ 0 ] . testmempoolaccept ( [ hex ] )
assert res [ 0 ] [ " allowed " ]
# Don't wait for wallet to add it to the mempool:
res = self . nodes [ 0 ] . sendrawtransaction ( hex )
2021-08-19 17:10:24 +02:00
self . generate ( self . nodes [ 0 ] , 1 )
2020-08-07 17:36:36 +02:00
assert_equal ( self . nodes [ 0 ] . gettransaction ( txid ) [ " confirmations " ] , 1 )
self . log . info ( " Lock unspents... " )
utxo1 = w0 . listunspent ( ) [ 0 ]
assert_greater_than ( utxo1 [ " amount " ] , 1 )
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , inputs = [ utxo1 ] , add_to_wallet = False , lock_unspents = True )
assert res [ " complete " ]
locked_coins = w0 . listlockunspent ( )
assert_equal ( len ( locked_coins ) , 1 )
# Locked coins are automatically unlocked when manually selected
2020-09-17 15:28:59 +02:00
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , inputs = [ utxo1 ] , add_to_wallet = False )
assert res [ " complete " ]
2020-08-07 17:36:36 +02:00
self . log . info ( " Replaceable... " )
2020-09-17 20:57:51 +02:00
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , add_to_wallet = True , replaceable = True )
assert res [ " complete " ]
assert_equal ( self . nodes [ 0 ] . gettransaction ( res [ " txid " ] ) [ " bip125-replaceable " ] , " yes " )
res = self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , add_to_wallet = True , replaceable = False )
assert res [ " complete " ]
assert_equal ( self . nodes [ 0 ] . gettransaction ( res [ " txid " ] ) [ " bip125-replaceable " ] , " no " )
2020-08-07 17:36:36 +02:00
self . log . info ( " Subtract fee from output " )
self . test_send ( from_wallet = w0 , to_wallet = w1 , amount = 1 , subtract_fee_from_outputs = [ 0 ] )
2021-03-10 15:37:18 +01:00
self . log . info ( " Include unsafe inputs " )
self . nodes [ 1 ] . createwallet ( wallet_name = " w5 " )
w5 = self . nodes [ 1 ] . get_wallet_rpc ( " w5 " )
self . test_send ( from_wallet = w0 , to_wallet = w5 , amount = 2 )
self . test_send ( from_wallet = w5 , to_wallet = w0 , amount = 1 , expect_error = ( - 4 , " Insufficient funds " ) )
res = self . test_send ( from_wallet = w5 , to_wallet = w0 , amount = 1 , include_unsafe = True )
assert res [ " complete " ]
2021-05-24 10:22:10 -03:00
self . log . info ( " Minconf " )
self . nodes [ 1 ] . createwallet ( wallet_name = " minconfw " )
minconfw = self . nodes [ 1 ] . get_wallet_rpc ( " minconfw " )
self . test_send ( from_wallet = w0 , to_wallet = minconfw , amount = 2 )
self . generate ( self . nodes [ 0 ] , 3 )
self . test_send ( from_wallet = minconfw , to_wallet = w0 , amount = 1 , minconf = 4 , expect_error = ( - 4 , " Insufficient funds " ) )
self . test_send ( from_wallet = minconfw , to_wallet = w0 , amount = 1 , minconf = - 4 , expect_error = ( - 8 , " Negative minconf " ) )
res = self . test_send ( from_wallet = minconfw , to_wallet = w0 , amount = 1 , minconf = 3 )
assert res [ " complete " ]
2021-01-12 15:17:29 -05:00
self . log . info ( " External outputs " )
2023-05-23 23:38:31 +02:00
privkey , _ = generate_keypair ( wif = True )
2021-01-12 15:17:29 -05:00
self . nodes [ 1 ] . createwallet ( " extsend " )
ext_wallet = self . nodes [ 1 ] . get_wallet_rpc ( " extsend " )
self . nodes [ 1 ] . createwallet ( " extfund " )
ext_fund = self . nodes [ 1 ] . get_wallet_rpc ( " extfund " )
2022-03-24 11:50:02 -04:00
# Make a weird but signable script. sh(wsh(pkh())) descriptor accomplishes this
desc = descsum_create ( " sh(wsh(pkh( {} ))) " . format ( privkey ) )
2021-01-12 15:17:29 -05:00
if self . options . descriptors :
res = ext_fund . importdescriptors ( [ { " desc " : desc , " timestamp " : " now " } ] )
else :
res = ext_fund . importmulti ( [ { " desc " : desc , " timestamp " : " now " } ] )
assert res [ 0 ] [ " success " ]
addr = self . nodes [ 0 ] . deriveaddresses ( desc ) [ 0 ]
addr_info = ext_fund . getaddressinfo ( addr )
self . nodes [ 0 ] . sendtoaddress ( addr , 10 )
self . nodes [ 0 ] . sendtoaddress ( ext_wallet . getnewaddress ( ) , 10 )
2021-10-06 12:18:33 +13:00
self . generate ( self . nodes [ 0 ] , 6 )
2021-01-12 15:17:29 -05:00
ext_utxo = ext_fund . listunspent ( addresses = [ addr ] ) [ 0 ]
# An external input without solving data should result in an error
2022-08-05 12:51:44 -03:00
self . test_send ( from_wallet = ext_wallet , to_wallet = self . nodes [ 0 ] , amount = 15 , inputs = [ ext_utxo ] , add_inputs = True , psbt = True , include_watching = True , expect_error = ( - 4 , " Not solvable pre-selected input COutPoint( %s , %s ) " % ( ext_utxo [ " txid " ] [ 0 : 10 ] , ext_utxo [ " vout " ] ) ) )
2021-01-12 15:17:29 -05:00
# But funding should work when the solving data is provided
2022-03-24 11:50:02 -04:00
res = self . test_send ( from_wallet = ext_wallet , to_wallet = self . nodes [ 0 ] , amount = 15 , inputs = [ ext_utxo ] , add_inputs = True , psbt = True , include_watching = True , solving_data = { " pubkeys " : [ addr_info [ ' pubkey ' ] ] , " scripts " : [ addr_info [ " embedded " ] [ " scriptPubKey " ] , addr_info [ " embedded " ] [ " embedded " ] [ " scriptPubKey " ] ] } )
2021-01-12 15:17:29 -05:00
signed = ext_wallet . walletprocesspsbt ( res [ " psbt " ] )
signed = ext_fund . walletprocesspsbt ( res [ " psbt " ] )
assert signed [ " complete " ]
res = self . test_send ( from_wallet = ext_wallet , to_wallet = self . nodes [ 0 ] , amount = 15 , inputs = [ ext_utxo ] , add_inputs = True , psbt = True , include_watching = True , solving_data = { " descriptors " : [ desc ] } )
signed = ext_wallet . walletprocesspsbt ( res [ " psbt " ] )
signed = ext_fund . walletprocesspsbt ( res [ " psbt " ] )
assert signed [ " complete " ]
2020-08-07 17:36:36 +02:00
2021-10-05 16:45:10 -04:00
dec = self . nodes [ 0 ] . decodepsbt ( signed [ " psbt " ] )
for i , txin in enumerate ( dec [ " tx " ] [ " vin " ] ) :
if txin [ " txid " ] == ext_utxo [ " txid " ] and txin [ " vout " ] == ext_utxo [ " vout " ] :
input_idx = i
break
psbt_in = dec [ " inputs " ] [ input_idx ]
2024-04-01 14:03:35 +02:00
scriptsig_hex = psbt_in [ " final_scriptSig " ] [ " hex " ] if " final_scriptSig " in psbt_in else " "
witness_stack_hex = psbt_in [ " final_scriptwitness " ] if " final_scriptwitness " in psbt_in else None
input_weight = calculate_input_weight ( scriptsig_hex , witness_stack_hex )
2021-10-05 16:45:10 -04:00
# Input weight error conditions
assert_raises_rpc_error (
- 8 ,
" Input weights should be specified in inputs rather than in options. " ,
ext_wallet . send ,
outputs = { self . nodes [ 0 ] . getnewaddress ( ) : 15 } ,
options = { " inputs " : [ ext_utxo ] , " input_weights " : [ { " txid " : ext_utxo [ " txid " ] , " vout " : ext_utxo [ " vout " ] , " weight " : 1000 } ] }
)
2024-03-01 11:44:21 +00:00
target_fee_rate_sat_vb = 10
2021-10-05 16:45:10 -04:00
# Funding should also work when input weights are provided
res = self . test_send (
from_wallet = ext_wallet ,
to_wallet = self . nodes [ 0 ] ,
amount = 15 ,
inputs = [ { " txid " : ext_utxo [ " txid " ] , " vout " : ext_utxo [ " vout " ] , " weight " : input_weight } ] ,
add_inputs = True ,
psbt = True ,
include_watching = True ,
2024-03-01 11:44:21 +00:00
fee_rate = target_fee_rate_sat_vb
2021-10-05 16:45:10 -04:00
)
signed = ext_wallet . walletprocesspsbt ( res [ " psbt " ] )
signed = ext_fund . walletprocesspsbt ( res [ " psbt " ] )
assert signed [ " complete " ]
2023-09-05 17:10:21 +01:00
testres = self . nodes [ 0 ] . testmempoolaccept ( [ signed [ " hex " ] ] ) [ 0 ]
2021-10-05 16:45:10 -04:00
assert_equal ( testres [ " allowed " ] , True )
2024-03-01 11:44:21 +00:00
actual_fee_rate_sat_vb = Decimal ( testres [ " fees " ] [ " base " ] ) * Decimal ( 1e8 ) / Decimal ( testres [ " vsize " ] )
# Due to ECDSA signatures not always being the same length, the actual fee rate may be slightly different
# but rounded to nearest integer, it should be the same as the target fee rate
assert_equal ( round ( actual_fee_rate_sat_vb ) , target_fee_rate_sat_vb )
2021-10-05 16:45:10 -04:00
2024-06-19 14:27:35 -03:00
# Check tx creation size limits
self . test_weight_limits ( )
def test_weight_limits ( self ) :
self . log . info ( " Test weight limits " )
self . nodes [ 1 ] . createwallet ( " test_weight_limits " )
wallet = self . nodes [ 1 ] . get_wallet_rpc ( " test_weight_limits " )
# Generate future inputs; 272 WU per input (273 when high-s).
# Picking 1471 inputs will exceed the max standard tx weight.
outputs = [ ]
for _ in range ( 1472 ) :
outputs . append ( { wallet . getnewaddress ( address_type = " legacy " ) : 0.1 } )
self . nodes [ 0 ] . send ( outputs = outputs )
self . generate ( self . nodes [ 0 ] , 1 )
# 1) Try to fund transaction only using the preset inputs
inputs = wallet . listunspent ( )
assert_raises_rpc_error ( - 4 , " Transaction too large " ,
wallet . send , outputs = [ { wallet . getnewaddress ( ) : 0.1 * 1471 } ] , options = { " inputs " : inputs , " add_inputs " : False } )
# 2) Let the wallet fund the transaction
assert_raises_rpc_error ( - 4 , " The inputs size exceeds the maximum weight. Please try sending a smaller amount or manually consolidating your wallet ' s UTXOs " ,
wallet . send , outputs = [ { wallet . getnewaddress ( ) : 0.1 * 1471 } ] )
# 3) Pre-select some inputs and let the wallet fill-up the remaining amount
inputs = inputs [ 0 : 1000 ]
assert_raises_rpc_error ( - 4 , " The combination of the pre-selected inputs and the wallet automatic inputs selection exceeds the transaction maximum weight. Please try sending a smaller amount or manually consolidating your wallet ' s UTXOs " ,
wallet . send , outputs = [ { wallet . getnewaddress ( ) : 0.1 * 1471 } ] , options = { " inputs " : inputs , " add_inputs " : True } )
self . nodes [ 1 ] . unloadwallet ( " test_weight_limits " )
2020-08-07 17:36:36 +02:00
if __name__ == ' __main__ ' :
2024-07-16 22:05:14 +01:00
WalletSendTest ( __file__ ) . main ( )