Skip to content

Commit 5f3848c

Browse files
committed
Merge bitcoin/bitcoin#31278: wallet, rpc: deprecate settxfee and paytxfee
2f2ab47 Release notes (Pol Espinasa) bf194c9 wallet, rpc: deprecate settxfee and paytxfee (Pol Espinasa) Pull request description: **Summary** This PR deprecates the settxfee RPC and paytxfee setting, marking it for removal in Bitcoin Core 31.0. **Motivation** The PR was initially motivated by bitcoin/bitcoin#31088. The intention was to create a new function `settxfeerate` to allow users to set a static fee rate in `sat/vB` instead of `btc/kvB`. The `settxfee` RPC allows users to set a static fee rate for all transactions created by the wallet. However, in a dynamic fee environment, this can lead to poor fee choices, either overpaying when the mempool is empty or underpaying when congestion is high. The preferred approach is to rely on fee estimation, which is designed to adapt to network conditions, and is the one by default. Same argument apply for `paytxfee` setting. During discussion the consensus was that static fee settings are a footgun and that users should instead specify the fee rate per transaction if they don't want to rely on the fee estimation. Given this, rather than introducing a `settxfeerate` alternative, this PR goes towards removing `settxfee` and `paytxfee` entirely. **Key Changes** `settxfee` and `paytxfee` is now deprecated and will be removed in Bitcoin Core 31.0. Users should rely on fee estimation or explicitly specify a fee rate when constructing transactions. **Impact on Users** If users currently use settxfee or paytxfee, they should transition to specifying fees per transaction. No immediate breakage in 30.0 (must use `-deprecatedrpc=settxfee`), but `settxfee` and `paytxfee` will be removed in 31.0. **Alternative Approaches Considered** A settxfeerate alternative (using sat/vB) was initially proposed but ultimately rejected in favor of deprecating static fee setting entirely. **Notes for removal** - When removing paytxfee we should also update txconfirmtarget startup option help text. - Get back the comment from `rpc_deprecated.py` test. [+info](bitcoin/bitcoin#31278 (comment)) ACKs for top commit: fjahr: re-ACK 2f2ab47 ismaelsadeeq: re-ACK 2f2ab47 rkrux: Concept and utACK 2f2ab47 Tree-SHA512: 0272812cbe5a519737c5d0683acc2072e67559792b4a6472bca8b23426e5ce9e88a3a1eba987feda70a082b8b474b3126893848628d7bf11e1520357b18c8d3e
2 parents 770d39a + 2f2ab47 commit 5f3848c

12 files changed

+35
-8
lines changed

doc/release-notes-31278.md

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
RPC and Startup Option
2+
---
3+
The `-paytxfee` startup option and the `settxfee` RPC are now deprecated and will be removed in Bitcoin Core 31.0. They allowed the user to set a static fee rate for wallet transactions, which could potentially lead to overpaying or underpaying. Users should instead rely on fee estimation or specify a fee rate per transaction using the `fee_rate` argument in RPCs such as `fundrawtransaction`, `sendtoaddress`, `send`, `sendall`, and `sendmany`. (#31278)

src/wallet/init.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -66,7 +66,7 @@ void WalletInit::AddWalletOptions(ArgsManager& argsman) const
6666
CURRENCY_UNIT, FormatMoney(DEFAULT_TRANSACTION_MAXFEE)), ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
6767
argsman.AddArg("-mintxfee=<amt>", strprintf("Fee rates (in %s/kvB) smaller than this are considered zero fee for transaction creation (default: %s)",
6868
CURRENCY_UNIT, FormatMoney(DEFAULT_TRANSACTION_MINFEE)), ArgsManager::ALLOW_ANY, OptionsCategory::WALLET);
69-
argsman.AddArg("-paytxfee=<amt>", strprintf("Fee rate (in %s/kvB) to add to transactions you send (default: %s)",
69+
argsman.AddArg("-paytxfee=<amt>", strprintf("(DEPRECATED) Fee rate (in %s/kvB) to add to transactions you send (default: %s)",
7070
CURRENCY_UNIT, FormatMoney(CFeeRate{DEFAULT_PAY_TX_FEE}.GetFeePerK())), ArgsManager::ALLOW_ANY, OptionsCategory::WALLET);
7171
#ifdef ENABLE_EXTERNAL_SIGNER
7272
argsman.AddArg("-signer=<cmd>", "External signing tool, see doc/external-signer.md", ArgsManager::ALLOW_ANY, OptionsCategory::WALLET);

src/wallet/rpc/spend.cpp

+6-1
Original file line numberDiff line numberDiff line change
@@ -418,7 +418,7 @@ RPCHelpMan sendmany()
418418
RPCHelpMan settxfee()
419419
{
420420
return RPCHelpMan{"settxfee",
421-
"\nSet the transaction fee rate in " + CURRENCY_UNIT + "/kvB for this wallet. Overrides the global -paytxfee command line parameter.\n"
421+
"\n(DEPRECATED) Set the transaction fee rate in " + CURRENCY_UNIT + "/kvB for this wallet. Overrides the global -paytxfee command line parameter.\n"
422422
"Can be deactivated by passing 0 as the fee. In that case automatic fee selection will be used by default.\n",
423423
{
424424
{"amount", RPCArg::Type::AMOUNT, RPCArg::Optional::NO, "The transaction fee rate in " + CURRENCY_UNIT + "/kvB"},
@@ -437,6 +437,11 @@ RPCHelpMan settxfee()
437437

438438
LOCK(pwallet->cs_wallet);
439439

440+
if (!pwallet->chain().rpcEnableDeprecated("settxfee")) {
441+
throw JSONRPCError(RPC_METHOD_DEPRECATED, "settxfee is deprecated and will be fully removed in v31.0."
442+
"\nTo use settxfee restart bitcoind with -deprecatedrpc=settxfee.");
443+
}
444+
440445
CAmount nAmount = AmountFromValue(request.params[0]);
441446
CFeeRate tx_fee_rate(nAmount, 1000);
442447
CFeeRate max_tx_fee_rate(pwallet->m_default_max_tx_fee, 1000);

src/wallet/wallet.cpp

+2
Original file line numberDiff line numberDiff line change
@@ -3181,6 +3181,8 @@ std::shared_ptr<CWallet> CWallet::Create(WalletContext& context, const std::stri
31813181
}
31823182

31833183
if (args.IsArgSet("-paytxfee")) {
3184+
warnings.push_back(_("-paytxfee is deprecated and will be fully removed in v31.0."));
3185+
31843186
std::optional<CAmount> pay_tx_fee = ParseMoney(args.GetArg("-paytxfee", ""));
31853187
if (!pay_tx_fee) {
31863188
error = AmountErrMsg("paytxfee", args.GetArg("-paytxfee", ""));

test/functional/rpc_deprecated.py

+10-1
Original file line numberDiff line numberDiff line change
@@ -4,13 +4,21 @@
44
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
55
"""Test deprecation of RPC calls."""
66
from test_framework.test_framework import BitcoinTestFramework
7+
from test_framework.util import assert_raises_rpc_error
78

89
class DeprecatedRpcTest(BitcoinTestFramework):
10+
def add_options(self, parser):
11+
self.add_wallet_options(parser)
12+
913
def set_test_params(self):
1014
self.num_nodes = 1
1115
self.setup_clean_chain = True
1216
self.extra_args = [[]]
1317

18+
19+
def skip_test_if_missing_module(self):
20+
self.skip_if_no_wallet()
21+
1422
def run_test(self):
1523
# This test should be used to verify the errors of the currently
1624
# deprecated RPC methods (without the -deprecatedrpc flag) until
@@ -23,7 +31,8 @@ def run_test(self):
2331
# at least one other functional test that still tests the RPCs
2432
# functionality using the respective -deprecatedrpc flag.
2533

26-
self.log.info("Currently no tests for deprecated RPC methods")
34+
self.log.info("Test settxfee RPC")
35+
assert_raises_rpc_error(-32, 'settxfee is deprecated and will be fully removed in v31.0.', self.nodes[0].rpc.settxfee, 0.01)
2736

2837
if __name__ == '__main__':
2938
DeprecatedRpcTest(__file__).main()

test/functional/test_framework/test_framework.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -641,9 +641,9 @@ def stop_nodes(self, wait=0):
641641
# Wait for nodes to stop
642642
node.wait_until_stopped()
643643

644-
def restart_node(self, i, extra_args=None, clear_addrman=False):
644+
def restart_node(self, i, extra_args=None, clear_addrman=False, *, expected_stderr=''):
645645
"""Stop and start a test node"""
646-
self.stop_node(i)
646+
self.stop_node(i, expected_stderr=expected_stderr)
647647
if clear_addrman:
648648
peers_dat = self.nodes[i].chain_path / "peers.dat"
649649
os.remove(peers_dat)

test/functional/wallet_basic.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,7 @@ def set_test_params(self):
3535
# whitelist peers to speed up tx relay / mempool sync
3636
self.noban_tx_relay = True
3737
self.extra_args = [[
38-
"-dustrelayfee=0", "-walletrejectlongchains=0"
38+
"-dustrelayfee=0", "-walletrejectlongchains=0", "-deprecatedrpc=settxfee"
3939
]] * self.num_nodes
4040
self.setup_clean_chain = True
4141
self.supports_cli = False

test/functional/wallet_bumpfee.py

+1
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,7 @@ def set_test_params(self):
6161
"-walletrbf={}".format(i),
6262
"-mintxfee=0.00002",
6363
"-addresstype=bech32",
64+
"-deprecatedrpc=settxfee"
6465
] for i in range(self.num_nodes)]
6566

6667
def skip_test_if_missing_module(self):

test/functional/wallet_create_tx.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ def add_options(self, parser):
2323
def set_test_params(self):
2424
self.setup_clean_chain = True
2525
self.num_nodes = 1
26+
self.extra_args = [["-deprecatedrpc=settxfee"]]
2627

2728
def skip_test_if_missing_module(self):
2829
self.skip_if_no_wallet()
@@ -71,7 +72,7 @@ def test_tx_size_too_large(self):
7172
)
7273

7374
self.log.info('Check maxtxfee in combination with settxfee')
74-
self.restart_node(0)
75+
self.restart_node(0, expected_stderr='Warning: -paytxfee is deprecated and will be fully removed in v31.0.')
7576
self.nodes[0].settxfee(0.01)
7677
assert_raises_rpc_error(
7778
-6,

test/functional/wallet_fundrawtransaction.py

+3
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,9 @@ def add_options(self, parser):
4444

4545
def set_test_params(self):
4646
self.num_nodes = 4
47+
self.extra_args = [[
48+
"-deprecatedrpc=settxfee"
49+
] for i in range(self.num_nodes)]
4750
self.setup_clean_chain = True
4851
# whitelist peers to speed up tx relay / mempool sync
4952
self.noban_tx_relay = True

test/functional/wallet_multiwallet.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,7 @@ def set_test_params(self):
4646
self.setup_clean_chain = True
4747
self.num_nodes = 2
4848
self.rpc_timeout = 120
49-
self.extra_args = [["-nowallet"], []]
49+
self.extra_args = [["-nowallet", "-deprecatedrpc=settxfee"], []]
5050

5151
def skip_test_if_missing_module(self):
5252
self.skip_if_no_wallet()

test/functional/wallet_txn_clone.py

+3
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,9 @@ class TxnMallTest(BitcoinTestFramework):
1818
def set_test_params(self):
1919
self.num_nodes = 3
2020
self.supports_cli = False
21+
self.extra_args = [[
22+
"-deprecatedrpc=settxfee"
23+
] for i in range(self.num_nodes)]
2124

2225
def skip_test_if_missing_module(self):
2326
self.skip_if_no_wallet()

0 commit comments

Comments
 (0)