Skip to content

Commit c495731

Browse files
committed
fuzz: wallet: add target for CreateTransaction
1 parent 3db68e2 commit c495731

File tree

2 files changed

+103
-0
lines changed

2 files changed

+103
-0
lines changed

src/wallet/test/fuzz/CMakeLists.txt

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ target_sources(fuzz
1111
$<$<BOOL:${USE_SQLITE}>:${CMAKE_CURRENT_LIST_DIR}/notifications.cpp>
1212
parse_iso8601.cpp
1313
$<$<BOOL:${USE_SQLITE}>:${CMAKE_CURRENT_LIST_DIR}/scriptpubkeyman.cpp>
14+
spend.cpp
1415
wallet_bdb_parser.cpp
1516
)
1617
target_link_libraries(fuzz bitcoin_wallet)

src/wallet/test/fuzz/spend.cpp

Lines changed: 102 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,102 @@
1+
// Copyright (c) 2024-present The Bitcoin Core developers
2+
// Distributed under the MIT software license, see the accompanying
3+
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
4+
5+
#include <test/fuzz/FuzzedDataProvider.h>
6+
#include <test/fuzz/fuzz.h>
7+
#include <test/fuzz/util.h>
8+
#include <test/fuzz/util/wallet.h>
9+
#include <test/util/random.h>
10+
#include <test/util/setup_common.h>
11+
#include <wallet/coincontrol.h>
12+
#include <wallet/context.h>
13+
#include <wallet/spend.h>
14+
#include <wallet/test/util.h>
15+
#include <wallet/wallet.h>
16+
#include <validation.h>
17+
#include <addresstype.h>
18+
19+
using util::ToString;
20+
21+
namespace wallet {
22+
namespace {
23+
const TestingSetup* g_setup;
24+
25+
void initialize_setup()
26+
{
27+
static const auto testing_setup = MakeNoLogFileContext<const TestingSetup>();
28+
g_setup = testing_setup.get();
29+
}
30+
31+
FUZZ_TARGET(wallet_create_transaction, .init = initialize_setup)
32+
{
33+
SeedRandomStateForTest(SeedRand::ZEROS);
34+
FuzzedDataProvider fuzzed_data_provider{buffer.data(), buffer.size()};
35+
const auto& node = g_setup->m_node;
36+
Chainstate& chainstate{node.chainman->ActiveChainstate()};
37+
ArgsManager& args = *node.args;
38+
args.ForceSetArg("-dustrelayfee", ToString(fuzzed_data_provider.ConsumeIntegralInRange<CAmount>(0, MAX_MONEY)));
39+
FuzzedWallet fuzzed_wallet{
40+
*g_setup->m_node.chain,
41+
"fuzzed_wallet_a",
42+
"tprv8ZgxMBicQKsPd1QwsGgzfu2pcPYbBosZhJknqreRHgsWx32nNEhMjGQX2cgFL8n6wz9xdDYwLcs78N4nsCo32cxEX8RBtwGsEGgybLiQJfk",
43+
};
44+
45+
CCoinControl coin_control;
46+
if (fuzzed_data_provider.ConsumeBool()) coin_control.m_version = fuzzed_data_provider.ConsumeIntegral<unsigned int>();
47+
coin_control.m_avoid_partial_spends = fuzzed_data_provider.ConsumeBool();
48+
coin_control.m_include_unsafe_inputs = fuzzed_data_provider.ConsumeBool();
49+
if (fuzzed_data_provider.ConsumeBool()) coin_control.m_confirm_target = fuzzed_data_provider.ConsumeIntegral<unsigned int>();
50+
coin_control.destChange = fuzzed_data_provider.ConsumeBool() ? fuzzed_wallet.GetDestination(fuzzed_data_provider) : ConsumeTxDestination(fuzzed_data_provider);
51+
if (fuzzed_data_provider.ConsumeBool()) coin_control.m_change_type = fuzzed_data_provider.PickValueInArray(OUTPUT_TYPES);
52+
if (fuzzed_data_provider.ConsumeBool()) coin_control.m_feerate = CFeeRate(ConsumeMoney(fuzzed_data_provider, /*max=*/COIN));
53+
coin_control.m_allow_other_inputs = fuzzed_data_provider.ConsumeBool();
54+
coin_control.m_locktime = fuzzed_data_provider.ConsumeIntegral<unsigned int>();
55+
coin_control.fOverrideFeeRate = fuzzed_data_provider.ConsumeBool();
56+
57+
int next_locktime{0};
58+
CAmount all_values{0};
59+
LIMITED_WHILE(fuzzed_data_provider.ConsumeBool(), 10000)
60+
{
61+
CMutableTransaction tx;
62+
tx.nLockTime = next_locktime++;
63+
tx.vout.resize(1);
64+
CAmount n_value{ConsumeMoney(fuzzed_data_provider)};
65+
all_values += n_value;
66+
if (all_values > MAX_MONEY) return;
67+
tx.vout[0].nValue = n_value;
68+
tx.vout[0].scriptPubKey = GetScriptForDestination(fuzzed_wallet.GetDestination(fuzzed_data_provider));
69+
LOCK(fuzzed_wallet.wallet->cs_wallet);
70+
auto txid{tx.GetHash()};
71+
auto ret{fuzzed_wallet.wallet->mapWallet.emplace(std::piecewise_construct, std::forward_as_tuple(txid), std::forward_as_tuple(MakeTransactionRef(std::move(tx)), TxStateConfirmed{chainstate.m_chain.Tip()->GetBlockHash(), chainstate.m_chain.Height(), /*index=*/0}))};
72+
assert(ret.second);
73+
}
74+
75+
std::vector<CRecipient> recipients;
76+
LIMITED_WHILE(fuzzed_data_provider.ConsumeBool(), 100) {
77+
CTxDestination destination;
78+
CallOneOf(
79+
fuzzed_data_provider,
80+
[&] {
81+
destination = fuzzed_wallet.GetDestination(fuzzed_data_provider);
82+
},
83+
[&] {
84+
CScript script;
85+
script << OP_RETURN;
86+
destination = CNoDestination{script};
87+
},
88+
[&] {
89+
destination = ConsumeTxDestination(fuzzed_data_provider);
90+
}
91+
);
92+
recipients.push_back({destination,
93+
/*nAmount=*/ConsumeMoney(fuzzed_data_provider),
94+
/*fSubtractFeeFromAmount=*/fuzzed_data_provider.ConsumeBool()});
95+
}
96+
97+
std::optional<unsigned int> change_pos;
98+
if (fuzzed_data_provider.ConsumeBool()) change_pos = fuzzed_data_provider.ConsumeIntegral<unsigned int>();
99+
(void)CreateTransaction(*fuzzed_wallet.wallet, recipients, change_pos, coin_control);
100+
}
101+
} // namespace
102+
} // namespace wallet

0 commit comments

Comments
 (0)