2020-04-16 13:14:08 -04:00
// Copyright (c) 2017-2020 The Bitcoin Core developers
2018-03-10 01:18:18 -03:00
// Distributed under the MIT software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2018-05-14 03:51:03 -04:00
# include <amount.h>
2020-04-16 13:11:54 -04:00
# include <node/context.h>
2018-05-14 03:51:03 -04:00
# include <primitives/transaction.h>
# include <random.h>
2019-11-05 17:18:59 -03:00
# include <test/util/setup_common.h>
2020-04-16 13:11:54 -04:00
# include <wallet/coincontrol.h>
# include <wallet/coinselection.h>
2018-05-14 03:51:03 -04:00
# include <wallet/test/wallet_test_fixture.h>
2020-04-16 13:11:54 -04:00
# include <wallet/wallet.h>
2018-03-10 01:18:18 -03:00
# include <boost/test/unit_test.hpp>
# include <random>
2018-03-18 14:52:30 -03:00
BOOST_FIXTURE_TEST_SUITE ( coinselector_tests , WalletTestingSetup )
2018-03-10 01:18:18 -03:00
2018-03-10 02:39:29 -03:00
// how many times to run all the tests to have a chance to catch errors that only show up with particular random shuffles
# define RUN_TESTS 100
// some tests fail 1% of the time due to bad luck.
// we repeat those tests this many times and only complain if all iterations of the test fail
# define RANDOM_REPEATS 5
2018-03-10 01:18:18 -03:00
typedef std : : set < CInputCoin > CoinSet ;
static std : : vector < COutput > vCoins ;
2019-09-17 19:28:03 -03:00
static NodeContext testNode ;
static auto testChain = interfaces : : MakeChain ( testNode ) ;
2019-03-22 01:24:55 -03:00
static CWallet testWallet ( testChain . get ( ) , WalletLocation ( ) , WalletDatabase : : CreateDummy ( ) ) ;
2018-03-05 18:42:49 -03:00
static CAmount balance = 0 ;
2018-03-10 01:18:18 -03:00
2018-03-12 20:43:32 -03:00
CoinEligibilityFilter filter_standard ( 1 , 6 , 0 ) ;
CoinEligibilityFilter filter_confirmed ( 1 , 1 , 0 ) ;
CoinEligibilityFilter filter_standard_extra ( 6 , 6 , 0 ) ;
CoinSelectionParams coin_selection_params ( false , 0 , 0 , CFeeRate ( 0 ) , 0 ) ;
2018-03-10 01:18:18 -03:00
static void add_coin ( const CAmount & nValue , int nInput , std : : vector < CInputCoin > & set )
{
CMutableTransaction tx ;
tx . vout . resize ( nInput + 1 ) ;
tx . vout [ nInput ] . nValue = nValue ;
set . emplace_back ( MakeTransactionRef ( tx ) , nInput ) ;
}
static void add_coin ( const CAmount & nValue , int nInput , CoinSet & set )
{
CMutableTransaction tx ;
tx . vout . resize ( nInput + 1 ) ;
tx . vout [ nInput ] . nValue = nValue ;
set . emplace ( MakeTransactionRef ( tx ) , nInput ) ;
}
2019-10-25 19:10:13 -03:00
static void add_coin ( CWallet & wallet , const CAmount & nValue , int nAge = 6 * 24 , bool fIsFromMe = false , int nInput = 0 , bool spendable = false )
2018-03-10 02:39:29 -03:00
{
2018-03-05 18:42:49 -03:00
balance + = nValue ;
2018-03-10 02:39:29 -03:00
static int nextLockTime = 0 ;
CMutableTransaction tx ;
tx . nLockTime = nextLockTime + + ; // so all transactions get different hashes
tx . vout . resize ( nInput + 1 ) ;
tx . vout [ nInput ] . nValue = nValue ;
2019-10-25 19:10:13 -03:00
if ( spendable ) {
CTxDestination dest ;
std : : string error ;
assert ( wallet . GetNewDestination ( OutputType : : BECH32 , " " , dest , error ) ) ;
tx . vout [ nInput ] . scriptPubKey = GetScriptForDestination ( dest ) ;
}
2018-03-10 02:39:29 -03:00
if ( fIsFromMe ) {
// IsFromMe() returns (GetDebit() > 0), and GetDebit() is 0 if vin.empty(),
// so stop vin being empty, and cache a non-zero Debit to fake out IsFromMe()
tx . vin . resize ( 1 ) ;
}
2016-12-19 13:25:15 -03:00
CWalletTx * wtx = wallet . AddToWallet ( MakeTransactionRef ( std : : move ( tx ) ) , /* confirm= */ { } ) ;
2018-03-10 02:39:29 -03:00
if ( fIsFromMe )
{
2019-04-09 22:40:35 -04:00
wtx - > m_amounts [ CWalletTx : : DEBIT ] . Set ( ISMINE_SPENDABLE , 1 ) ;
2020-01-07 10:47:20 -03:00
wtx - > m_is_cache_empty = false ;
2018-03-10 02:39:29 -03:00
}
2016-12-19 13:25:15 -03:00
COutput output ( wtx , nInput , nAge , true /* spendable */ , true /* solvable */ , true /* safe */ ) ;
2018-03-10 02:39:29 -03:00
vCoins . push_back ( output ) ;
}
2019-10-25 19:10:13 -03:00
static void add_coin ( const CAmount & nValue , int nAge = 6 * 24 , bool fIsFromMe = false , int nInput = 0 , bool spendable = false )
{
add_coin ( testWallet , nValue , nAge , fIsFromMe , nInput , spendable ) ;
}
2018-03-10 02:39:29 -03:00
static void empty_wallet ( void )
{
vCoins . clear ( ) ;
2018-03-05 18:42:49 -03:00
balance = 0 ;
2018-03-10 02:39:29 -03:00
}
2018-03-10 01:18:18 -03:00
static bool equal_sets ( CoinSet a , CoinSet b )
{
std : : pair < CoinSet : : iterator , CoinSet : : iterator > ret = mismatch ( a . begin ( ) , a . end ( ) , b . begin ( ) ) ;
return ret . first = = a . end ( ) & & ret . second = = b . end ( ) ;
}
static CAmount make_hard_case ( int utxos , std : : vector < CInputCoin > & utxo_pool )
{
utxo_pool . clear ( ) ;
CAmount target = 0 ;
for ( int i = 0 ; i < utxos ; + + i ) {
target + = ( CAmount ) 1 < < ( utxos + i ) ;
add_coin ( ( CAmount ) 1 < < ( utxos + i ) , 2 * i , utxo_pool ) ;
add_coin ( ( ( CAmount ) 1 < < ( utxos + i ) ) + ( ( CAmount ) 1 < < ( utxos - 1 - i ) ) , 2 * i + 1 , utxo_pool ) ;
}
return target ;
}
2018-07-18 22:45:26 -04:00
inline std : : vector < OutputGroup > & GroupCoins ( const std : : vector < CInputCoin > & coins )
{
static std : : vector < OutputGroup > static_groups ;
static_groups . clear ( ) ;
for ( auto & coin : coins ) static_groups . emplace_back ( coin , 0 , true , 0 , 0 ) ;
return static_groups ;
}
inline std : : vector < OutputGroup > & GroupCoins ( const std : : vector < COutput > & coins )
{
static std : : vector < OutputGroup > static_groups ;
static_groups . clear ( ) ;
2019-04-09 22:40:35 -04:00
for ( auto & coin : coins ) static_groups . emplace_back ( coin . GetInputCoin ( ) , coin . nDepth , coin . tx - > m_amounts [ CWalletTx : : DEBIT ] . m_cached [ ISMINE_SPENDABLE ] & & coin . tx - > m_amounts [ CWalletTx : : DEBIT ] . m_value [ ISMINE_SPENDABLE ] = = 1 /* HACK: we can't figure out the is_me flag so we use the conditions defined above; perhaps set safe to false for !fIsFromMe in add_coin() */ , 0 , 0 ) ;
2018-07-18 22:45:26 -04:00
return static_groups ;
}
2018-03-10 01:18:18 -03:00
// Branch and bound coin selection tests
BOOST_AUTO_TEST_CASE ( bnb_search_test )
{
LOCK ( testWallet . cs_wallet ) ;
2019-10-07 15:11:34 -03:00
testWallet . SetupLegacyScriptPubKeyMan ( ) ;
2018-03-10 01:18:18 -03:00
// Setup
std : : vector < CInputCoin > utxo_pool ;
CoinSet selection ;
CoinSet actual_selection ;
CAmount value_ret = 0 ;
CAmount not_input_fees = 0 ;
/////////////////////////
// Known Outcome tests //
/////////////////////////
// Empty utxo pool
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( ! SelectCoinsBnB ( GroupCoins ( utxo_pool ) , 1 * CENT , 0.5 * CENT , selection , value_ret , not_input_fees ) ) ;
2018-03-10 01:18:18 -03:00
selection . clear ( ) ;
// Add utxos
add_coin ( 1 * CENT , 1 , utxo_pool ) ;
add_coin ( 2 * CENT , 2 , utxo_pool ) ;
add_coin ( 3 * CENT , 3 , utxo_pool ) ;
add_coin ( 4 * CENT , 4 , utxo_pool ) ;
// Select 1 Cent
add_coin ( 1 * CENT , 1 , actual_selection ) ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( SelectCoinsBnB ( GroupCoins ( utxo_pool ) , 1 * CENT , 0.5 * CENT , selection , value_ret , not_input_fees ) ) ;
2018-03-10 01:18:18 -03:00
BOOST_CHECK ( equal_sets ( selection , actual_selection ) ) ;
2018-05-12 00:17:55 -03:00
BOOST_CHECK_EQUAL ( value_ret , 1 * CENT ) ;
2018-03-10 01:18:18 -03:00
actual_selection . clear ( ) ;
selection . clear ( ) ;
// Select 2 Cent
add_coin ( 2 * CENT , 2 , actual_selection ) ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( SelectCoinsBnB ( GroupCoins ( utxo_pool ) , 2 * CENT , 0.5 * CENT , selection , value_ret , not_input_fees ) ) ;
2018-03-10 01:18:18 -03:00
BOOST_CHECK ( equal_sets ( selection , actual_selection ) ) ;
2018-05-12 00:17:55 -03:00
BOOST_CHECK_EQUAL ( value_ret , 2 * CENT ) ;
2018-03-10 01:18:18 -03:00
actual_selection . clear ( ) ;
selection . clear ( ) ;
// Select 5 Cent
2020-03-04 18:35:15 -03:00
add_coin ( 4 * CENT , 4 , actual_selection ) ;
add_coin ( 1 * CENT , 1 , actual_selection ) ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( SelectCoinsBnB ( GroupCoins ( utxo_pool ) , 5 * CENT , 0.5 * CENT , selection , value_ret , not_input_fees ) ) ;
2018-03-10 01:18:18 -03:00
BOOST_CHECK ( equal_sets ( selection , actual_selection ) ) ;
2018-05-12 00:17:55 -03:00
BOOST_CHECK_EQUAL ( value_ret , 5 * CENT ) ;
2018-03-10 01:18:18 -03:00
actual_selection . clear ( ) ;
selection . clear ( ) ;
// Select 11 Cent, not possible
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( ! SelectCoinsBnB ( GroupCoins ( utxo_pool ) , 11 * CENT , 0.5 * CENT , selection , value_ret , not_input_fees ) ) ;
2018-03-10 01:18:18 -03:00
actual_selection . clear ( ) ;
selection . clear ( ) ;
2020-02-20 17:59:22 -03:00
// Cost of change is greater than the difference between target value and utxo sum
add_coin ( 1 * CENT , 1 , actual_selection ) ;
BOOST_CHECK ( SelectCoinsBnB ( GroupCoins ( utxo_pool ) , 0.9 * CENT , 0.5 * CENT , selection , value_ret , not_input_fees ) ) ;
BOOST_CHECK_EQUAL ( value_ret , 1 * CENT ) ;
BOOST_CHECK ( equal_sets ( selection , actual_selection ) ) ;
actual_selection . clear ( ) ;
selection . clear ( ) ;
// Cost of change is less than the difference between target value and utxo sum
BOOST_CHECK ( ! SelectCoinsBnB ( GroupCoins ( utxo_pool ) , 0.9 * CENT , 0 , selection , value_ret , not_input_fees ) ) ;
actual_selection . clear ( ) ;
selection . clear ( ) ;
2018-03-10 01:18:18 -03:00
// Select 10 Cent
add_coin ( 5 * CENT , 5 , utxo_pool ) ;
2020-03-04 18:35:15 -03:00
add_coin ( 5 * CENT , 5 , actual_selection ) ;
2018-03-10 01:18:18 -03:00
add_coin ( 4 * CENT , 4 , actual_selection ) ;
add_coin ( 1 * CENT , 1 , actual_selection ) ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( SelectCoinsBnB ( GroupCoins ( utxo_pool ) , 10 * CENT , 0.5 * CENT , selection , value_ret , not_input_fees ) ) ;
2018-03-10 01:18:18 -03:00
BOOST_CHECK ( equal_sets ( selection , actual_selection ) ) ;
2018-05-12 00:17:55 -03:00
BOOST_CHECK_EQUAL ( value_ret , 10 * CENT ) ;
2018-03-10 01:18:18 -03:00
actual_selection . clear ( ) ;
selection . clear ( ) ;
// Negative effective value
// Select 10 Cent but have 1 Cent not be possible because too small
add_coin ( 5 * CENT , 5 , actual_selection ) ;
add_coin ( 3 * CENT , 3 , actual_selection ) ;
add_coin ( 2 * CENT , 2 , actual_selection ) ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( SelectCoinsBnB ( GroupCoins ( utxo_pool ) , 10 * CENT , 5000 , selection , value_ret , not_input_fees ) ) ;
2018-05-12 00:17:55 -03:00
BOOST_CHECK_EQUAL ( value_ret , 10 * CENT ) ;
// FIXME: this test is redundant with the above, because 1 Cent is selected, not "too small"
// BOOST_CHECK(equal_sets(selection, actual_selection));
2018-03-10 01:18:18 -03:00
// Select 0.25 Cent, not possible
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( ! SelectCoinsBnB ( GroupCoins ( utxo_pool ) , 0.25 * CENT , 0.5 * CENT , selection , value_ret , not_input_fees ) ) ;
2018-03-10 01:18:18 -03:00
actual_selection . clear ( ) ;
selection . clear ( ) ;
// Iteration exhaustion test
CAmount target = make_hard_case ( 17 , utxo_pool ) ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( ! SelectCoinsBnB ( GroupCoins ( utxo_pool ) , target , 0 , selection , value_ret , not_input_fees ) ) ; // Should exhaust
2018-03-10 01:18:18 -03:00
target = make_hard_case ( 14 , utxo_pool ) ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( SelectCoinsBnB ( GroupCoins ( utxo_pool ) , target , 0 , selection , value_ret , not_input_fees ) ) ; // Should not exhaust
2018-03-10 01:18:18 -03:00
// Test same value early bailout optimization
2018-05-12 00:17:55 -03:00
utxo_pool . clear ( ) ;
2018-03-10 01:18:18 -03:00
add_coin ( 7 * CENT , 7 , actual_selection ) ;
add_coin ( 7 * CENT , 7 , actual_selection ) ;
add_coin ( 7 * CENT , 7 , actual_selection ) ;
add_coin ( 7 * CENT , 7 , actual_selection ) ;
add_coin ( 2 * CENT , 7 , actual_selection ) ;
add_coin ( 7 * CENT , 7 , utxo_pool ) ;
add_coin ( 7 * CENT , 7 , utxo_pool ) ;
add_coin ( 7 * CENT , 7 , utxo_pool ) ;
add_coin ( 7 * CENT , 7 , utxo_pool ) ;
add_coin ( 2 * CENT , 7 , utxo_pool ) ;
for ( int i = 0 ; i < 50000 ; + + i ) {
add_coin ( 5 * CENT , 7 , utxo_pool ) ;
}
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( SelectCoinsBnB ( GroupCoins ( utxo_pool ) , 30 * CENT , 5000 , selection , value_ret , not_input_fees ) ) ;
2018-05-12 00:17:55 -03:00
BOOST_CHECK_EQUAL ( value_ret , 30 * CENT ) ;
BOOST_CHECK ( equal_sets ( selection , actual_selection ) ) ;
2018-03-10 01:18:18 -03:00
////////////////////
// Behavior tests //
////////////////////
// Select 1 Cent with pool of only greater than 5 Cent
utxo_pool . clear ( ) ;
for ( int i = 5 ; i < = 20 ; + + i ) {
add_coin ( i * CENT , i , utxo_pool ) ;
}
// Run 100 times, to make sure it is never finding a solution
for ( int i = 0 ; i < 100 ; + + i ) {
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( ! SelectCoinsBnB ( GroupCoins ( utxo_pool ) , 1 * CENT , 2 * CENT , selection , value_ret , not_input_fees ) ) ;
2018-03-10 01:18:18 -03:00
}
2018-03-12 20:43:32 -03:00
// Make sure that effective value is working in SelectCoinsMinConf when BnB is used
CoinSelectionParams coin_selection_params_bnb ( true , 0 , 0 , CFeeRate ( 3000 ) , 0 ) ;
CoinSet setCoinsRet ;
CAmount nValueRet ;
bool bnb_used ;
empty_wallet ( ) ;
add_coin ( 1 ) ;
vCoins . at ( 0 ) . nInputBytes = 40 ; // Make sure that it has a negative effective value. The next check should assert if this somehow got through. Otherwise it will fail
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( ! testWallet . SelectCoinsMinConf ( 1 * CENT , filter_standard , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params_bnb , bnb_used ) ) ;
2018-03-15 12:25:50 -03:00
2019-10-25 19:45:58 -03:00
// Test fees subtracted from output:
empty_wallet ( ) ;
add_coin ( 1 * CENT ) ;
vCoins . at ( 0 ) . nInputBytes = 40 ;
BOOST_CHECK ( ! testWallet . SelectCoinsMinConf ( 1 * CENT , filter_standard , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params_bnb , bnb_used ) ) ;
coin_selection_params_bnb . m_subtract_fee_outputs = true ;
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 1 * CENT , filter_standard , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params_bnb , bnb_used ) ) ;
BOOST_CHECK_EQUAL ( nValueRet , 1 * CENT ) ;
2019-10-25 19:10:13 -03:00
// Make sure that can use BnB when there are preset inputs
2018-03-15 12:25:50 -03:00
empty_wallet ( ) ;
2019-10-25 19:10:13 -03:00
{
std : : unique_ptr < CWallet > wallet = MakeUnique < CWallet > ( m_chain . get ( ) , WalletLocation ( ) , WalletDatabase : : CreateMock ( ) ) ;
bool firstRun ;
wallet - > LoadWallet ( firstRun ) ;
2019-10-07 15:11:34 -03:00
wallet - > SetupLegacyScriptPubKeyMan ( ) ;
2019-10-25 19:10:13 -03:00
LOCK ( wallet - > cs_wallet ) ;
add_coin ( * wallet , 5 * CENT , 6 * 24 , false , 0 , true ) ;
add_coin ( * wallet , 3 * CENT , 6 * 24 , false , 0 , true ) ;
add_coin ( * wallet , 2 * CENT , 6 * 24 , false , 0 , true ) ;
CCoinControl coin_control ;
coin_control . fAllowOtherInputs = true ;
coin_control . Select ( COutPoint ( vCoins . at ( 0 ) . tx - > GetHash ( ) , vCoins . at ( 0 ) . i ) ) ;
coin_selection_params_bnb . effective_fee = CFeeRate ( 0 ) ;
BOOST_CHECK ( wallet - > SelectCoins ( vCoins , 10 * CENT , setCoinsRet , nValueRet , coin_control , coin_selection_params_bnb , bnb_used ) ) ;
BOOST_CHECK ( bnb_used ) ;
BOOST_CHECK ( coin_selection_params_bnb . use_bnb ) ;
}
2018-03-12 20:43:32 -03:00
}
2018-03-10 02:39:29 -03:00
BOOST_AUTO_TEST_CASE ( knapsack_solver_test )
{
CoinSet setCoinsRet , setCoinsRet2 ;
CAmount nValueRet ;
2018-03-05 18:42:49 -03:00
bool bnb_used ;
2018-03-10 02:39:29 -03:00
LOCK ( testWallet . cs_wallet ) ;
2019-10-07 15:11:34 -03:00
testWallet . SetupLegacyScriptPubKeyMan ( ) ;
2018-03-10 02:39:29 -03:00
// test multiple times to allow for differences in the shuffle order
for ( int i = 0 ; i < RUN_TESTS ; i + + )
{
empty_wallet ( ) ;
// with an empty wallet we can't even pay one cent
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( ! testWallet . SelectCoinsMinConf ( 1 * CENT , filter_standard , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
add_coin ( 1 * CENT , 4 ) ; // add a new 1 cent coin
// with a new 1 cent coin, we still can't find a mature 1 cent
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( ! testWallet . SelectCoinsMinConf ( 1 * CENT , filter_standard , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
// but we can find a new 1 cent
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 1 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 1 * CENT ) ;
add_coin ( 2 * CENT ) ; // add a mature 2 cent coin
// we can't make 3 cents of mature coins
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( ! testWallet . SelectCoinsMinConf ( 3 * CENT , filter_standard , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
// we can make 3 cents of new coins
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 3 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 3 * CENT ) ;
add_coin ( 5 * CENT ) ; // add a mature 5 cent coin,
add_coin ( 10 * CENT , 3 , true ) ; // a new 10 cent coin sent from one of our own addresses
add_coin ( 20 * CENT ) ; // and a mature 20 cent coin
// now we have new: 1+10=11 (of which 10 was self-sent), and mature: 2+5+20=27. total = 38
// we can't make 38 cents only if we disallow new coins:
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( ! testWallet . SelectCoinsMinConf ( 38 * CENT , filter_standard , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
// we can't even make 37 cents if we don't allow new coins even if they're from us
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( ! testWallet . SelectCoinsMinConf ( 38 * CENT , filter_standard_extra , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
// but we can make 37 cents if we accept new coins from ourself
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 37 * CENT , filter_standard , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 37 * CENT ) ;
// and we can make 38 cents if we accept all new coins
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 38 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 38 * CENT ) ;
// try making 34 cents from 1,2,5,10,20 - we can't do it exactly
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 34 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 35 * CENT ) ; // but 35 cents is closest
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 3U ) ; // the best should be 20+10+5. it's incredibly unlikely the 1 or 2 got included (but possible)
// when we try making 7 cents, the smaller coins (1,2,5) are enough. We should see just 2+5
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 7 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 7 * CENT ) ;
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 2U ) ;
// when we try making 8 cents, the smaller coins (1,2,5) are exactly enough.
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 8 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK ( nValueRet = = 8 * CENT ) ;
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 3U ) ;
// when we try making 9 cents, no subset of smaller coins is enough, and we get the next bigger coin (10)
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 9 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 10 * CENT ) ;
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 1U ) ;
// now clear out the wallet and start again to test choosing between subsets of smaller coins and the next biggest coin
empty_wallet ( ) ;
add_coin ( 6 * CENT ) ;
add_coin ( 7 * CENT ) ;
add_coin ( 8 * CENT ) ;
add_coin ( 20 * CENT ) ;
add_coin ( 30 * CENT ) ; // now we have 6+7+8+20+30 = 71 cents total
// check that we have 71 and not 72
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 71 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
BOOST_CHECK ( ! testWallet . SelectCoinsMinConf ( 72 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
// now try making 16 cents. the best smaller coins can do is 6+7+8 = 21; not as good at the next biggest coin, 20
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 16 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 20 * CENT ) ; // we should get 20 in one coin
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 1U ) ;
add_coin ( 5 * CENT ) ; // now we have 5+6+7+8+20+30 = 75 cents total
// now if we try making 16 cents again, the smaller coins can make 5+6+7 = 18 cents, better than the next biggest coin, 20
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 16 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 18 * CENT ) ; // we should get 18 in 3 coins
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 3U ) ;
add_coin ( 18 * CENT ) ; // now we have 5+6+7+8+18+20+30
// and now if we try making 16 cents again, the smaller coins can make 5+6+7 = 18 cents, the same as the next biggest coin, 18
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 16 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 18 * CENT ) ; // we should get 18 in 1 coin
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 1U ) ; // because in the event of a tie, the biggest coin wins
// now try making 11 cents. we should get 5+6
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 11 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 11 * CENT ) ;
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 2U ) ;
// check that the smallest bigger coin is used
add_coin ( 1 * COIN ) ;
add_coin ( 2 * COIN ) ;
add_coin ( 3 * COIN ) ;
add_coin ( 4 * COIN ) ; // now we have 5+6+7+8+18+20+30+100+200+300+400 = 1094 cents
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 95 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 1 * COIN ) ; // we should get 1 BTC in 1 coin
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 1U ) ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 195 * CENT , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 2 * COIN ) ; // we should get 2 BTC in 1 coin
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 1U ) ;
// empty the wallet and start again, now with fractions of a cent, to test small change avoidance
empty_wallet ( ) ;
add_coin ( MIN_CHANGE * 1 / 10 ) ;
add_coin ( MIN_CHANGE * 2 / 10 ) ;
add_coin ( MIN_CHANGE * 3 / 10 ) ;
add_coin ( MIN_CHANGE * 4 / 10 ) ;
add_coin ( MIN_CHANGE * 5 / 10 ) ;
// try making 1 * MIN_CHANGE from the 1.5 * MIN_CHANGE
// we'll get change smaller than MIN_CHANGE whatever happens, so can expect MIN_CHANGE exactly
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( MIN_CHANGE , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , MIN_CHANGE ) ;
// but if we add a bigger coin, small change is avoided
add_coin ( 1111 * MIN_CHANGE ) ;
// try making 1 from 0.1 + 0.2 + 0.3 + 0.4 + 0.5 + 1111 = 1112.5
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 1 * MIN_CHANGE , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 1 * MIN_CHANGE ) ; // we should get the exact amount
// if we add more small coins:
add_coin ( MIN_CHANGE * 6 / 10 ) ;
add_coin ( MIN_CHANGE * 7 / 10 ) ;
// and try again to make 1.0 * MIN_CHANGE
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 1 * MIN_CHANGE , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 1 * MIN_CHANGE ) ; // we should get the exact amount
// run the 'mtgox' test (see http://blockexplorer.com/tx/29a3efd3ef04f9153d47a990bd7b048a4b2d213daaa5fb8ed670fb85f13bdbcf)
// they tried to consolidate 10 50k coins into one 500k coin, and ended up with 50k in change
empty_wallet ( ) ;
for ( int j = 0 ; j < 20 ; j + + )
add_coin ( 50000 * COIN ) ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 500000 * COIN , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 500000 * COIN ) ; // we should get the exact amount
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 10U ) ; // in ten coins
// if there's not enough in the smaller coins to make at least 1 * MIN_CHANGE change (0.5+0.6+0.7 < 1.0+1.0),
// we need to try finding an exact subset anyway
// sometimes it will fail, and so we use the next biggest coin:
empty_wallet ( ) ;
add_coin ( MIN_CHANGE * 5 / 10 ) ;
add_coin ( MIN_CHANGE * 6 / 10 ) ;
add_coin ( MIN_CHANGE * 7 / 10 ) ;
add_coin ( 1111 * MIN_CHANGE ) ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 1 * MIN_CHANGE , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 1111 * MIN_CHANGE ) ; // we get the bigger coin
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 1U ) ;
// but sometimes it's possible, and we use an exact subset (0.4 + 0.6 = 1.0)
empty_wallet ( ) ;
add_coin ( MIN_CHANGE * 4 / 10 ) ;
add_coin ( MIN_CHANGE * 6 / 10 ) ;
add_coin ( MIN_CHANGE * 8 / 10 ) ;
add_coin ( 1111 * MIN_CHANGE ) ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( MIN_CHANGE , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , MIN_CHANGE ) ; // we should get the exact amount
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 2U ) ; // in two coins 0.4+0.6
// test avoiding small change
empty_wallet ( ) ;
add_coin ( MIN_CHANGE * 5 / 100 ) ;
add_coin ( MIN_CHANGE * 1 ) ;
add_coin ( MIN_CHANGE * 100 ) ;
// trying to make 100.01 from these three coins
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( MIN_CHANGE * 10001 / 100 , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , MIN_CHANGE * 10105 / 100 ) ; // we should get all coins
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 3U ) ;
// but if we try to make 99.9, we should take the bigger of the two small coins to avoid small change
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( MIN_CHANGE * 9990 / 100 , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 101 * MIN_CHANGE ) ;
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 2U ) ;
2018-05-12 13:06:38 -03:00
}
2018-03-10 02:39:29 -03:00
2018-05-12 13:06:38 -03:00
// test with many inputs
for ( CAmount amt = 1500 ; amt < COIN ; amt * = 10 ) {
empty_wallet ( ) ;
// Create 676 inputs (= (old MAX_STANDARD_TX_SIZE == 100000) / 148 bytes per input)
for ( uint16_t j = 0 ; j < 676 ; j + + )
add_coin ( amt ) ;
// We only create the wallet once to save time, but we still run the coin selection RUN_TESTS times.
for ( int i = 0 ; i < RUN_TESTS ; i + + ) {
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 2000 , filter_confirmed , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-05-12 13:06:38 -03:00
2018-03-10 02:39:29 -03:00
if ( amt - 2000 < MIN_CHANGE ) {
// needs more than one input:
uint16_t returnSize = std : : ceil ( ( 2000.0 + MIN_CHANGE ) / amt ) ;
CAmount returnValue = amt * returnSize ;
BOOST_CHECK_EQUAL ( nValueRet , returnValue ) ;
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , returnSize ) ;
} else {
// one input is sufficient:
BOOST_CHECK_EQUAL ( nValueRet , amt ) ;
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 1U ) ;
}
2018-05-12 13:06:38 -03:00
}
}
2018-03-10 02:39:29 -03:00
2018-05-12 13:06:38 -03:00
// test randomness
{
empty_wallet ( ) ;
for ( int i2 = 0 ; i2 < 100 ; i2 + + )
add_coin ( COIN ) ;
2018-03-10 02:39:29 -03:00
2018-05-12 13:06:38 -03:00
// Again, we only create the wallet once to save time, but we still run the coin selection RUN_TESTS times.
for ( int i = 0 ; i < RUN_TESTS ; i + + ) {
2018-03-10 02:39:29 -03:00
// picking 50 from 100 coins doesn't depend on the shuffle,
// but does depend on randomness in the stochastic approximation code
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 50 * COIN , filter_standard , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 50 * COIN , filter_standard , GroupCoins ( vCoins ) , setCoinsRet2 , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK ( ! equal_sets ( setCoinsRet , setCoinsRet2 ) ) ;
int fails = 0 ;
for ( int j = 0 ; j < RANDOM_REPEATS ; j + + )
{
// selecting 1 from 100 identical coins depends on the shuffle; this test will fail 1% of the time
// run the test RANDOM_REPEATS times and only complain if all of them fail
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( COIN , filter_standard , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( COIN , filter_standard , GroupCoins ( vCoins ) , setCoinsRet2 , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
if ( equal_sets ( setCoinsRet , setCoinsRet2 ) )
fails + + ;
}
BOOST_CHECK_NE ( fails , RANDOM_REPEATS ) ;
2018-05-12 13:06:38 -03:00
}
// add 75 cents in small change. not enough to make 90 cents,
// then try making 90 cents. there are multiple competing "smallest bigger" coins,
// one of which should be picked at random
add_coin ( 5 * CENT ) ;
add_coin ( 10 * CENT ) ;
add_coin ( 15 * CENT ) ;
add_coin ( 20 * CENT ) ;
add_coin ( 25 * CENT ) ;
for ( int i = 0 ; i < RUN_TESTS ; i + + ) {
int fails = 0 ;
2018-03-10 02:39:29 -03:00
for ( int j = 0 ; j < RANDOM_REPEATS ; j + + )
{
// selecting 1 from 100 identical coins depends on the shuffle; this test will fail 1% of the time
// run the test RANDOM_REPEATS times and only complain if all of them fail
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 90 * CENT , filter_standard , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 90 * CENT , filter_standard , GroupCoins ( vCoins ) , setCoinsRet2 , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
if ( equal_sets ( setCoinsRet , setCoinsRet2 ) )
fails + + ;
}
BOOST_CHECK_NE ( fails , RANDOM_REPEATS ) ;
2018-05-12 13:06:38 -03:00
}
}
2018-03-10 02:39:29 -03:00
empty_wallet ( ) ;
}
BOOST_AUTO_TEST_CASE ( ApproximateBestSubset )
{
CoinSet setCoinsRet ;
CAmount nValueRet ;
2018-03-05 18:42:49 -03:00
bool bnb_used ;
2018-03-10 02:39:29 -03:00
LOCK ( testWallet . cs_wallet ) ;
2019-10-07 15:11:34 -03:00
testWallet . SetupLegacyScriptPubKeyMan ( ) ;
2018-03-10 02:39:29 -03:00
empty_wallet ( ) ;
// Test vValue sort order
for ( int i = 0 ; i < 1000 ; i + + )
add_coin ( 1000 * COIN ) ;
add_coin ( 3 * COIN ) ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( 1003 * COIN , filter_standard , GroupCoins ( vCoins ) , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2018-03-10 02:39:29 -03:00
BOOST_CHECK_EQUAL ( nValueRet , 1003 * COIN ) ;
BOOST_CHECK_EQUAL ( setCoinsRet . size ( ) , 2U ) ;
empty_wallet ( ) ;
}
2018-03-05 18:42:49 -03:00
// Tests that with the ideal conditions, the coin selector will always be able to find a solution that can pay the target value
BOOST_AUTO_TEST_CASE ( SelectCoins_test )
{
2019-10-07 15:11:34 -03:00
testWallet . SetupLegacyScriptPubKeyMan ( ) ;
2018-03-05 18:42:49 -03:00
// Random generator stuff
std : : default_random_engine generator ;
std : : exponential_distribution < double > distribution ( 100 ) ;
FastRandomContext rand ;
// Run this test 100 times
for ( int i = 0 ; i < 100 ; + + i )
{
empty_wallet ( ) ;
// Make a wallet with 1000 exponentially distributed random inputs
for ( int j = 0 ; j < 1000 ; + + j )
{
add_coin ( ( CAmount ) ( distribution ( generator ) * 10000000 ) ) ;
}
// Generate a random fee rate in the range of 100 - 400
CFeeRate rate ( rand . randrange ( 300 ) + 100 ) ;
// Generate a random target value between 1000 and wallet balance
2018-04-12 03:25:45 -03:00
CAmount target = rand . randrange ( balance - 1000 ) + 1000 ;
2018-03-05 18:42:49 -03:00
// Perform selection
CoinSelectionParams coin_selection_params_knapsack ( false , 34 , 148 , CFeeRate ( 0 ) , 0 ) ;
CoinSelectionParams coin_selection_params_bnb ( true , 34 , 148 , CFeeRate ( 0 ) , 0 ) ;
2018-04-12 03:25:45 -03:00
CoinSet out_set ;
CAmount out_value = 0 ;
bool bnb_used = false ;
2018-07-18 22:45:26 -04:00
BOOST_CHECK ( testWallet . SelectCoinsMinConf ( target , filter_standard , GroupCoins ( vCoins ) , out_set , out_value , coin_selection_params_bnb , bnb_used ) | |
testWallet . SelectCoinsMinConf ( target , filter_standard , GroupCoins ( vCoins ) , out_set , out_value , coin_selection_params_knapsack , bnb_used ) ) ;
2018-03-05 18:42:49 -03:00
BOOST_CHECK_GE ( out_value , target ) ;
}
}
2018-03-10 01:18:18 -03:00
BOOST_AUTO_TEST_SUITE_END ( )