2016-03-19 20:58:06 +01:00
#!/usr/bin/env python3
2022-12-24 23:49:50 +00:00
# Copyright (c) 2014-2022 The Bitcoin Core developers
2014-10-23 09:48:19 +08:00
# Distributed under the MIT software license, see the accompanying
2014-06-16 14:45:32 +02:00
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-01-17 18:34:40 -05:00
""" Test the wallet keypool and interaction with wallet encryption/locking. """
2014-06-16 14:45:32 +02:00
2018-07-07 00:10:35 +02:00
import time
2019-10-23 15:21:50 +02:00
from decimal import Decimal
2018-07-07 00:10:35 +02:00
2015-11-15 17:58:01 +01:00
from test_framework . test_framework import BitcoinTestFramework
2018-07-07 00:10:35 +02:00
from test_framework . util import assert_equal , assert_raises_rpc_error
2023-10-09 15:30:40 +03:00
from test_framework . wallet_util import WalletUnlock
2014-06-16 14:45:32 +02:00
2015-11-15 17:58:01 +01:00
class KeyPoolTest ( BitcoinTestFramework ) :
2022-11-09 12:53:13 +01:00
def add_options ( self , parser ) :
self . add_wallet_options ( parser )
2017-06-09 18:21:21 -04:00
def set_test_params ( self ) :
self . num_nodes = 1
2015-11-15 17:58:01 +01:00
2018-09-09 13:32:37 -04:00
def skip_test_if_missing_module ( self ) :
self . skip_if_no_wallet ( )
2015-11-15 17:58:01 +01:00
def run_test ( self ) :
nodes = self . nodes
2016-07-21 21:19:02 +02:00
addr_before_encrypting = nodes [ 0 ] . getnewaddress ( )
2018-02-09 11:12:27 -05:00
addr_before_encrypting_data = nodes [ 0 ] . getaddressinfo ( addr_before_encrypting )
2016-07-21 21:19:02 +02:00
wallet_info_old = nodes [ 0 ] . getwalletinfo ( )
2019-07-16 15:33:35 -04:00
if not self . options . descriptors :
assert addr_before_encrypting_data [ ' hdseedid ' ] == wallet_info_old [ ' hdseedid ' ]
2018-04-16 11:13:07 +02:00
2015-11-15 18:48:18 +01:00
# Encrypt wallet and wait to terminate
2018-02-20 16:09:51 -05:00
nodes [ 0 ] . encryptwallet ( ' test ' )
2019-07-16 15:33:35 -04:00
if self . options . descriptors :
# Import hardened derivation only descriptors
nodes [ 0 ] . walletpassphrase ( ' test ' , 10 )
nodes [ 0 ] . importdescriptors ( [
{
" desc " : " wpkh(tprv8ZgxMBicQKsPd7Uf69XL1XwhmjHopUGep8GuEiJDZmbQz6o58LninorQAfcKZWARbtRtfnLcJ5MQ2AtHcQJCCRUcMRvmDUjyEmNUWwx8UbK/0h/*h)#y4dfsj7n " ,
" timestamp " : " now " ,
" range " : [ 0 , 0 ] ,
" active " : True
} ,
{
" desc " : " pkh(tprv8ZgxMBicQKsPd7Uf69XL1XwhmjHopUGep8GuEiJDZmbQz6o58LninorQAfcKZWARbtRtfnLcJ5MQ2AtHcQJCCRUcMRvmDUjyEmNUWwx8UbK/1h/*h)#a0nyvl0k " ,
" timestamp " : " now " ,
" range " : [ 0 , 0 ] ,
" active " : True
} ,
{
" desc " : " sh(wpkh(tprv8ZgxMBicQKsPd7Uf69XL1XwhmjHopUGep8GuEiJDZmbQz6o58LninorQAfcKZWARbtRtfnLcJ5MQ2AtHcQJCCRUcMRvmDUjyEmNUWwx8UbK/2h/*h))#lmeu2axg " ,
" timestamp " : " now " ,
" range " : [ 0 , 0 ] ,
" active " : True
} ,
{
" desc " : " wpkh(tprv8ZgxMBicQKsPd7Uf69XL1XwhmjHopUGep8GuEiJDZmbQz6o58LninorQAfcKZWARbtRtfnLcJ5MQ2AtHcQJCCRUcMRvmDUjyEmNUWwx8UbK/3h/*h)#jkl636gm " ,
" timestamp " : " now " ,
" range " : [ 0 , 0 ] ,
" active " : True ,
" internal " : True
} ,
{
" desc " : " pkh(tprv8ZgxMBicQKsPd7Uf69XL1XwhmjHopUGep8GuEiJDZmbQz6o58LninorQAfcKZWARbtRtfnLcJ5MQ2AtHcQJCCRUcMRvmDUjyEmNUWwx8UbK/4h/*h)#l3crwaus " ,
" timestamp " : " now " ,
" range " : [ 0 , 0 ] ,
" active " : True ,
" internal " : True
} ,
{
" desc " : " sh(wpkh(tprv8ZgxMBicQKsPd7Uf69XL1XwhmjHopUGep8GuEiJDZmbQz6o58LninorQAfcKZWARbtRtfnLcJ5MQ2AtHcQJCCRUcMRvmDUjyEmNUWwx8UbK/5h/*h))#qg8wa75f " ,
" timestamp " : " now " ,
" range " : [ 0 , 0 ] ,
" active " : True ,
" internal " : True
}
] )
nodes [ 0 ] . walletlock ( )
2015-11-15 18:48:18 +01:00
# Keep creating keys
2014-06-16 14:45:32 +02:00
addr = nodes [ 0 ] . getnewaddress ( )
2018-02-09 11:12:27 -05:00
addr_data = nodes [ 0 ] . getaddressinfo ( addr )
2016-07-21 21:19:02 +02:00
wallet_info = nodes [ 0 ] . getwalletinfo ( )
2019-07-16 15:33:35 -04:00
assert addr_before_encrypting_data [ ' hdmasterfingerprint ' ] != addr_data [ ' hdmasterfingerprint ' ]
if not self . options . descriptors :
assert addr_data [ ' hdseedid ' ] == wallet_info [ ' hdseedid ' ]
2017-07-12 10:33:46 -04:00
assert_raises_rpc_error ( - 12 , " Error: Keypool ran out, please call keypoolrefill first " , nodes [ 0 ] . getnewaddress )
2015-11-15 18:48:18 +01:00
2017-01-17 08:55:30 +01:00
# put six (plus 2) new keys in the keypool (100% external-, +100% internal-keys, 1 in min)
2023-10-09 15:30:40 +03:00
with WalletUnlock ( nodes [ 0 ] , ' test ' ) :
nodes [ 0 ] . keypoolrefill ( 6 )
2017-01-10 16:45:30 +01:00
wi = nodes [ 0 ] . getwalletinfo ( )
2019-07-16 15:33:35 -04:00
if self . options . descriptors :
2021-06-23 16:11:51 -04:00
assert_equal ( wi [ ' keypoolsize_hd_internal ' ] , 24 )
assert_equal ( wi [ ' keypoolsize ' ] , 24 )
2019-07-16 15:33:35 -04:00
else :
assert_equal ( wi [ ' keypoolsize_hd_internal ' ] , 6 )
assert_equal ( wi [ ' keypoolsize ' ] , 6 )
2015-11-15 18:48:18 +01:00
2017-01-10 16:45:30 +01:00
# drain the internal keys
nodes [ 0 ] . getrawchangeaddress ( )
nodes [ 0 ] . getrawchangeaddress ( )
2017-01-17 08:55:30 +01:00
nodes [ 0 ] . getrawchangeaddress ( )
nodes [ 0 ] . getrawchangeaddress ( )
nodes [ 0 ] . getrawchangeaddress ( )
nodes [ 0 ] . getrawchangeaddress ( )
2015-11-15 18:48:18 +01:00
addr = set ( )
# the next one should fail
2017-07-12 10:33:46 -04:00
assert_raises_rpc_error ( - 12 , " Keypool ran out " , nodes [ 0 ] . getrawchangeaddress )
2015-11-15 18:48:18 +01:00
2017-01-10 16:45:30 +01:00
# drain the external keys
2019-10-23 15:21:50 +02:00
addr . add ( nodes [ 0 ] . getnewaddress ( address_type = " bech32 " ) )
addr . add ( nodes [ 0 ] . getnewaddress ( address_type = " bech32 " ) )
addr . add ( nodes [ 0 ] . getnewaddress ( address_type = " bech32 " ) )
addr . add ( nodes [ 0 ] . getnewaddress ( address_type = " bech32 " ) )
addr . add ( nodes [ 0 ] . getnewaddress ( address_type = " bech32 " ) )
addr . add ( nodes [ 0 ] . getnewaddress ( address_type = " bech32 " ) )
2019-02-19 17:43:44 -05:00
assert len ( addr ) == 6
2017-01-10 16:45:30 +01:00
# the next one should fail
2017-07-12 10:33:46 -04:00
assert_raises_rpc_error ( - 12 , " Error: Keypool ran out, please call keypoolrefill first " , nodes [ 0 ] . getnewaddress )
2017-01-10 16:45:30 +01:00
2015-11-15 18:48:18 +01:00
# refill keypool with three new addresses
2016-01-08 13:12:16 +01:00
nodes [ 0 ] . walletpassphrase ( ' test ' , 1 )
2015-11-15 18:48:18 +01:00
nodes [ 0 ] . keypoolrefill ( 3 )
2017-01-10 16:45:30 +01:00
2016-01-08 13:12:16 +01:00
# test walletpassphrase timeout
time . sleep ( 1.1 )
assert_equal ( nodes [ 0 ] . getwalletinfo ( ) [ " unlocked_until " ] , 0 )
2015-11-15 18:48:18 +01:00
2018-10-12 00:04:06 +09:00
# drain the keypool
for _ in range ( 3 ) :
nodes [ 0 ] . getnewaddress ( )
assert_raises_rpc_error ( - 12 , " Keypool ran out " , nodes [ 0 ] . getnewaddress )
2015-11-15 18:48:18 +01:00
2023-10-09 15:30:40 +03:00
with WalletUnlock ( nodes [ 0 ] , ' test ' ) :
nodes [ 0 ] . keypoolrefill ( 100 )
wi = nodes [ 0 ] . getwalletinfo ( )
if self . options . descriptors :
assert_equal ( wi [ ' keypoolsize_hd_internal ' ] , 400 )
assert_equal ( wi [ ' keypoolsize ' ] , 400 )
else :
assert_equal ( wi [ ' keypoolsize_hd_internal ' ] , 100 )
assert_equal ( wi [ ' keypoolsize ' ] , 100 )
if not self . options . descriptors :
# Check that newkeypool entirely flushes the keypool
start_keypath = nodes [ 0 ] . getaddressinfo ( nodes [ 0 ] . getnewaddress ( ) ) [ ' hdkeypath ' ]
start_change_keypath = nodes [ 0 ] . getaddressinfo ( nodes [ 0 ] . getrawchangeaddress ( ) ) [ ' hdkeypath ' ]
# flush keypool and get new addresses
nodes [ 0 ] . newkeypool ( )
end_keypath = nodes [ 0 ] . getaddressinfo ( nodes [ 0 ] . getnewaddress ( ) ) [ ' hdkeypath ' ]
end_change_keypath = nodes [ 0 ] . getaddressinfo ( nodes [ 0 ] . getrawchangeaddress ( ) ) [ ' hdkeypath ' ]
# The new keypath index should be 100 more than the old one
new_index = int ( start_keypath . rsplit ( ' / ' , 1 ) [ 1 ] [ : - 1 ] ) + 100
new_change_index = int ( start_change_keypath . rsplit ( ' / ' , 1 ) [ 1 ] [ : - 1 ] ) + 100
assert_equal ( end_keypath , " m/0 ' /0 ' / " + str ( new_index ) + " ' " )
assert_equal ( end_change_keypath , " m/0 ' /1 ' / " + str ( new_change_index ) + " ' " )
2021-09-26 13:43:55 +13:00
2019-10-23 15:21:50 +02:00
# create a blank wallet
2019-07-16 15:33:35 -04:00
nodes [ 0 ] . createwallet ( wallet_name = ' w2 ' , blank = True , disable_private_keys = True )
2019-10-23 15:21:50 +02:00
w2 = nodes [ 0 ] . get_wallet_rpc ( ' w2 ' )
# refer to initial wallet as w1
2020-09-28 20:24:06 -04:00
w1 = nodes [ 0 ] . get_wallet_rpc ( self . default_wallet_name )
2019-10-23 15:21:50 +02:00
# import private key and fund it
address = addr . pop ( )
2019-07-16 15:33:35 -04:00
desc = w1 . getaddressinfo ( address ) [ ' desc ' ]
if self . options . descriptors :
res = w2 . importdescriptors ( [ { ' desc ' : desc , ' timestamp ' : ' now ' } ] )
else :
res = w2 . importmulti ( [ { ' desc ' : desc , ' timestamp ' : ' now ' } ] )
2019-10-23 15:21:50 +02:00
assert_equal ( res [ 0 ] [ ' success ' ] , True )
2023-10-09 15:30:40 +03:00
with WalletUnlock ( w1 , ' test ' ) :
res = w1 . sendtoaddress ( address = address , amount = 0.00010000 )
2021-08-19 17:10:24 +02:00
self . generate ( nodes [ 0 ] , 1 )
2019-10-23 15:21:50 +02:00
destination = addr . pop ( )
# Using a fee rate (10 sat / byte) well above the minimum relay rate
# creating a 5,000 sat transaction with change should not be possible
2022-11-10 12:04:07 -05:00
assert_raises_rpc_error ( - 4 , " Transaction needs a change address, but we can ' t generate it. " , w2 . walletcreatefundedpsbt , inputs = [ ] , outputs = [ { addr . pop ( ) : 0.00005000 } ] , subtractFeeFromOutputs = [ 0 ] , feeRate = 0.00010 )
2019-10-23 15:21:50 +02:00
# creating a 10,000 sat transaction without change, with a manual input, should still be possible
2022-11-10 12:04:07 -05:00
res = w2 . walletcreatefundedpsbt ( inputs = w2 . listunspent ( ) , outputs = [ { destination : 0.00010000 } ] , subtractFeeFromOutputs = [ 0 ] , feeRate = 0.00010 )
2019-10-23 15:21:50 +02:00
assert_equal ( " psbt " in res , True )
# creating a 10,000 sat transaction without change should still be possible
2022-11-10 12:04:07 -05:00
res = w2 . walletcreatefundedpsbt ( inputs = [ ] , outputs = [ { destination : 0.00010000 } ] , subtractFeeFromOutputs = [ 0 ] , feeRate = 0.00010 )
2019-10-23 15:21:50 +02:00
assert_equal ( " psbt " in res , True )
# should work without subtractFeeFromOutputs if the exact fee is subtracted from the amount
2022-11-10 12:04:07 -05:00
res = w2 . walletcreatefundedpsbt ( inputs = [ ] , outputs = [ { destination : 0.00008900 } ] , feeRate = 0.00010 )
2019-10-23 15:21:50 +02:00
assert_equal ( " psbt " in res , True )
# dust change should be removed
2022-11-10 12:04:07 -05:00
res = w2 . walletcreatefundedpsbt ( inputs = [ ] , outputs = [ { destination : 0.00008800 } ] , feeRate = 0.00010 )
2019-10-23 15:21:50 +02:00
assert_equal ( " psbt " in res , True )
# create a transaction without change at the maximum fee rate, such that the output is still spendable:
2022-11-10 12:04:07 -05:00
res = w2 . walletcreatefundedpsbt ( inputs = [ ] , outputs = [ { destination : 0.00010000 } ] , subtractFeeFromOutputs = [ 0 ] , feeRate = 0.0008823 )
2019-10-23 15:21:50 +02:00
assert_equal ( " psbt " in res , True )
assert_equal ( res [ " fee " ] , Decimal ( " 0.00009706 " ) )
# creating a 10,000 sat transaction with a manual change address should be possible
2022-11-10 12:04:07 -05:00
res = w2 . walletcreatefundedpsbt ( inputs = [ ] , outputs = [ { destination : 0.00010000 } ] , subtractFeeFromOutputs = [ 0 ] , feeRate = 0.00010 , changeAddress = addr . pop ( ) )
2019-10-23 15:21:50 +02:00
assert_equal ( " psbt " in res , True )
2022-12-18 17:21:46 -03:00
if not self . options . descriptors :
msg = " Error: Private keys are disabled for this wallet "
assert_raises_rpc_error ( - 4 , msg , w2 . keypoolrefill , 100 )
2019-10-23 15:21:50 +02:00
2014-06-16 14:45:32 +02:00
if __name__ == ' __main__ ' :
2015-11-15 17:58:01 +01:00
KeyPoolTest ( ) . main ( )