2017-06-14 15:27:24 +02:00
#!/usr/bin/env python3
2019-12-31 18:35:41 +01:00
# Copyright (c) 2016-2019 The Bitcoin Core developers
2017-06-14 15:27:24 +02:00
# Distributed under the MIT software license, see the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
""" Test Wallet encryption """
import time
2017-09-06 20:02:08 +02:00
from test_framework . test_framework import BitcoinTestFramework
2017-06-14 15:27:24 +02:00
from test_framework . util import (
assert_equal ,
2019-09-25 11:34:51 +02:00
assert_raises_rpc_error ,
2018-01-17 12:15:45 +01:00
assert_greater_than ,
assert_greater_than_or_equal ,
2017-06-14 15:27:24 +02:00
)
2020-07-14 17:08:16 +02:00
2017-06-14 15:27:24 +02:00
class WalletEncryptionTest ( BitcoinTestFramework ) :
2017-09-01 18:47:13 +02:00
def set_test_params ( self ) :
2017-06-14 15:27:24 +02:00
self . setup_clean_chain = True
self . num_nodes = 1
2018-09-13 12:33:15 +02:00
def skip_test_if_missing_module ( self ) :
self . skip_if_no_wallet ( )
2017-06-14 15:27:24 +02:00
def run_test ( self ) :
passphrase = " WalletPassphrase "
passphrase2 = " SecondWalletPassphrase "
# Make sure the wallet isn't encrypted first
address = self . nodes [ 0 ] . getnewaddress ( )
privkey = self . nodes [ 0 ] . dumpprivkey ( address )
assert_equal ( privkey [ : 1 ] , " c " )
assert_equal ( len ( privkey ) , 52 )
2021-09-24 11:56:10 +02:00
assert_raises_rpc_error ( - 15 , " Error: running with an unencrypted wallet, but walletpassphrase was called " , self . nodes [ 0 ] . walletpassphrase , ' ff ' , 1 )
assert_raises_rpc_error ( - 15 , " Error: running with an unencrypted wallet, but walletpassphrasechange was called. " , self . nodes [ 0 ] . walletpassphrasechange , ' ff ' , ' ff ' )
2017-06-14 15:27:24 +02:00
# Encrypt the wallet
2021-09-24 11:56:10 +02:00
assert_raises_rpc_error ( - 8 , " passphrase can not be empty " , self . nodes [ 0 ] . encryptwallet , ' ' )
2018-09-14 10:28:27 +02:00
self . nodes [ 0 ] . encryptwallet ( passphrase )
2017-06-14 15:27:24 +02:00
# Test that the wallet is encrypted
2019-09-25 11:34:51 +02:00
assert_raises_rpc_error ( - 13 , " Please enter the wallet passphrase with walletpassphrase first " , self . nodes [ 0 ] . dumpprivkey , address )
2021-09-24 11:56:10 +02:00
assert_raises_rpc_error ( - 15 , " Error: running with an encrypted wallet, but encryptwallet was called. " , self . nodes [ 0 ] . encryptwallet , ' ff ' )
assert_raises_rpc_error ( - 8 , " passphrase can not be empty " , self . nodes [ 0 ] . walletpassphrase , ' ' , 1 )
assert_raises_rpc_error ( - 8 , " passphrase can not be empty " , self . nodes [ 0 ] . walletpassphrasechange , ' ' , ' ff ' )
2017-06-14 15:27:24 +02:00
# Check that walletpassphrase works
self . nodes [ 0 ] . walletpassphrase ( passphrase , 2 )
assert_equal ( privkey , self . nodes [ 0 ] . dumpprivkey ( address ) )
# Check that the timeout is right
2020-01-02 18:36:17 +01:00
time . sleep ( 4 ) # Wait a little bit longer to make sure wallet gets locked
2019-09-25 11:34:51 +02:00
assert_raises_rpc_error ( - 13 , " Please enter the wallet passphrase with walletpassphrase first " , self . nodes [ 0 ] . dumpprivkey , address )
2017-06-14 15:27:24 +02:00
# Test wrong passphrase
2019-09-25 11:34:51 +02:00
assert_raises_rpc_error ( - 14 , " wallet passphrase entered was incorrect " , self . nodes [ 0 ] . walletpassphrase , passphrase + " wrong " , 10 )
2017-06-14 15:27:24 +02:00
# Test walletlock
self . nodes [ 0 ] . walletpassphrase ( passphrase , 84600 )
assert_equal ( privkey , self . nodes [ 0 ] . dumpprivkey ( address ) )
self . nodes [ 0 ] . walletlock ( )
2019-09-25 11:34:51 +02:00
assert_raises_rpc_error ( - 13 , " Please enter the wallet passphrase with walletpassphrase first " , self . nodes [ 0 ] . dumpprivkey , address )
2017-06-14 15:27:24 +02:00
# Test passphrase changes
self . nodes [ 0 ] . walletpassphrasechange ( passphrase , passphrase2 )
2019-09-25 11:34:51 +02:00
assert_raises_rpc_error ( - 14 , " wallet passphrase entered was incorrect " , self . nodes [ 0 ] . walletpassphrase , passphrase , 10 )
2017-06-14 15:27:24 +02:00
self . nodes [ 0 ] . walletpassphrase ( passphrase2 , 10 )
assert_equal ( privkey , self . nodes [ 0 ] . dumpprivkey ( address ) )
2018-01-17 12:15:45 +01:00
self . nodes [ 0 ] . walletlock ( )
# Test timeout bounds
assert_raises_rpc_error ( - 8 , " Timeout cannot be negative. " , self . nodes [ 0 ] . walletpassphrase , passphrase2 , - 10 )
2020-07-14 17:08:16 +02:00
self . log . info ( ' Check a timeout less than the limit ' )
2018-04-08 18:58:42 +02:00
MAX_VALUE = 100000000
2020-02-29 20:48:18 +01:00
expected_time = self . mocktime + MAX_VALUE - 600
2018-04-08 18:58:42 +02:00
self . nodes [ 0 ] . walletpassphrase ( passphrase2 , MAX_VALUE - 600 )
2020-07-14 17:08:16 +02:00
self . bump_mocktime ( 1 )
# give buffer for walletpassphrase, since it iterates over all crypted keys
expected_time_with_buffer = self . mocktime + MAX_VALUE - 600
2018-01-17 12:15:45 +01:00
actual_time = self . nodes [ 0 ] . getwalletinfo ( ) [ ' unlocked_until ' ]
assert_greater_than_or_equal ( actual_time , expected_time )
2020-07-14 17:08:16 +02:00
assert_greater_than ( expected_time_with_buffer , actual_time )
self . log . info ( ' Check a timeout greater than the limit ' )
2020-02-29 20:48:18 +01:00
expected_time = self . mocktime + MAX_VALUE - 1
2018-04-08 18:58:42 +02:00
self . nodes [ 0 ] . walletpassphrase ( passphrase2 , MAX_VALUE + 1000 )
2020-07-14 17:08:16 +02:00
self . bump_mocktime ( 1 )
expected_time_with_buffer = self . mocktime + MAX_VALUE
2018-01-17 12:15:45 +01:00
actual_time = self . nodes [ 0 ] . getwalletinfo ( ) [ ' unlocked_until ' ]
assert_greater_than_or_equal ( actual_time , expected_time )
2020-07-14 17:08:16 +02:00
assert_greater_than ( expected_time_with_buffer , actual_time )
2017-06-14 15:27:24 +02:00
if __name__ == ' __main__ ' :
WalletEncryptionTest ( ) . main ( )