Get rid of ambiguous OutputType::NONE value
Based on suggestion by Pieter Wuille <pieter.wuille@gmail.com> at https://github.com/bitcoin/bitcoin/pull/12119#issuecomment-357982763 After #12119, the NONE output type was overloaded to refer to either an output type that couldn't be parsed, or to an automatic change output mode. This change drops the NONE enum and uses a simple bool indicate parse failure, and a new CHANGE_AUTO enum to refer the change output type. This change is almost a pure refactoring except it makes RPCs reject empty string ("") address types instead of treating them like they were unset. This simplifies the parsing code a little bit and could prevent RPC usage mistakes. It's noted in the release notes.
This commit is contained in:
parent
5f0c6a7b0e
commit
1e46d8ae89
7 changed files with 41 additions and 31 deletions
|
@ -104,6 +104,11 @@ Low-level RPC changes
|
||||||
now the empty string `""` instead of `"wallet.dat"`. If bitcoin is started
|
now the empty string `""` instead of `"wallet.dat"`. If bitcoin is started
|
||||||
with any `-wallet=<path>` options, there is no change in behavior, and the
|
with any `-wallet=<path>` options, there is no change in behavior, and the
|
||||||
name of any wallet is just its `<path>` string.
|
name of any wallet is just its `<path>` string.
|
||||||
|
- Passing an empty string (`""`) as the `address_type` parameter to
|
||||||
|
`getnewaddress`, `getrawchangeaddress`, `addmultisigaddress`,
|
||||||
|
`fundrawtransaction` RPCs is now an error. Previously, this would fall back
|
||||||
|
to using the default address type. It is still possible to pass null or leave
|
||||||
|
the parameter unset to use the default address type.
|
||||||
|
|
||||||
### Logging
|
### Logging
|
||||||
|
|
||||||
|
|
|
@ -648,7 +648,7 @@ void PaymentServer::fetchPaymentACK(WalletModel* walletModel, const SendCoinsRec
|
||||||
// use for change. Despite an actual payment and not change, this is a close match:
|
// use for change. Despite an actual payment and not change, this is a close match:
|
||||||
// it's the output type we use subject to privacy issues, but not restricted by what
|
// it's the output type we use subject to privacy issues, but not restricted by what
|
||||||
// other software supports.
|
// other software supports.
|
||||||
const OutputType change_type = walletModel->wallet().getDefaultChangeType() != OutputType::NONE ? walletModel->wallet().getDefaultChangeType() : walletModel->wallet().getDefaultAddressType();
|
const OutputType change_type = walletModel->wallet().getDefaultChangeType() != OutputType::CHANGE_AUTO ? walletModel->wallet().getDefaultChangeType() : walletModel->wallet().getDefaultAddressType();
|
||||||
walletModel->wallet().learnRelatedScripts(newKey, change_type);
|
walletModel->wallet().learnRelatedScripts(newKey, change_type);
|
||||||
CTxDestination dest = GetDestinationForKey(newKey, change_type);
|
CTxDestination dest = GetDestinationForKey(newKey, change_type);
|
||||||
std::string label = tr("Refund from %1").arg(recipient.authenticatedMerchant).toStdString();
|
std::string label = tr("Refund from %1").arg(recipient.authenticatedMerchant).toStdString();
|
||||||
|
|
|
@ -164,8 +164,7 @@ UniValue getnewaddress(const JSONRPCRequest& request)
|
||||||
|
|
||||||
OutputType output_type = pwallet->m_default_address_type;
|
OutputType output_type = pwallet->m_default_address_type;
|
||||||
if (!request.params[1].isNull()) {
|
if (!request.params[1].isNull()) {
|
||||||
output_type = ParseOutputType(request.params[1].get_str(), pwallet->m_default_address_type);
|
if (!ParseOutputType(request.params[1].get_str(), output_type)) {
|
||||||
if (output_type == OutputType::NONE) {
|
|
||||||
throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("Unknown address type '%s'", request.params[1].get_str()));
|
throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("Unknown address type '%s'", request.params[1].get_str()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -259,10 +258,9 @@ UniValue getrawchangeaddress(const JSONRPCRequest& request)
|
||||||
pwallet->TopUpKeyPool();
|
pwallet->TopUpKeyPool();
|
||||||
}
|
}
|
||||||
|
|
||||||
OutputType output_type = pwallet->m_default_change_type != OutputType::NONE ? pwallet->m_default_change_type : pwallet->m_default_address_type;
|
OutputType output_type = pwallet->m_default_change_type != OutputType::CHANGE_AUTO ? pwallet->m_default_change_type : pwallet->m_default_address_type;
|
||||||
if (!request.params[0].isNull()) {
|
if (!request.params[0].isNull()) {
|
||||||
output_type = ParseOutputType(request.params[0].get_str(), output_type);
|
if (!ParseOutputType(request.params[0].get_str(), output_type)) {
|
||||||
if (output_type == OutputType::NONE) {
|
|
||||||
throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("Unknown address type '%s'", request.params[0].get_str()));
|
throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("Unknown address type '%s'", request.params[0].get_str()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1226,8 +1224,7 @@ UniValue addmultisigaddress(const JSONRPCRequest& request)
|
||||||
|
|
||||||
OutputType output_type = pwallet->m_default_address_type;
|
OutputType output_type = pwallet->m_default_address_type;
|
||||||
if (!request.params[3].isNull()) {
|
if (!request.params[3].isNull()) {
|
||||||
output_type = ParseOutputType(request.params[3].get_str(), output_type);
|
if (!ParseOutputType(request.params[3].get_str(), output_type)) {
|
||||||
if (output_type == OutputType::NONE) {
|
|
||||||
throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("Unknown address type '%s'", request.params[3].get_str()));
|
throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("Unknown address type '%s'", request.params[3].get_str()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3180,8 +3177,8 @@ UniValue fundrawtransaction(const JSONRPCRequest& request)
|
||||||
if (options.exists("changeAddress")) {
|
if (options.exists("changeAddress")) {
|
||||||
throw JSONRPCError(RPC_INVALID_PARAMETER, "Cannot specify both changeAddress and address_type options");
|
throw JSONRPCError(RPC_INVALID_PARAMETER, "Cannot specify both changeAddress and address_type options");
|
||||||
}
|
}
|
||||||
coinControl.m_change_type = ParseOutputType(options["change_type"].get_str(), pwallet->m_default_change_type);
|
coinControl.m_change_type = pwallet->m_default_change_type;
|
||||||
if (coinControl.m_change_type == OutputType::NONE) {
|
if (!ParseOutputType(options["change_type"].get_str(), *coinControl.m_change_type)) {
|
||||||
throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("Unknown change type '%s'", options["change_type"].get_str()));
|
throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("Unknown change type '%s'", options["change_type"].get_str()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2648,7 +2648,7 @@ bool CWallet::FundTransaction(CMutableTransaction& tx, CAmount& nFeeRet, int& nC
|
||||||
OutputType CWallet::TransactionChangeType(OutputType change_type, const std::vector<CRecipient>& vecSend)
|
OutputType CWallet::TransactionChangeType(OutputType change_type, const std::vector<CRecipient>& vecSend)
|
||||||
{
|
{
|
||||||
// If -changetype is specified, always use that change type.
|
// If -changetype is specified, always use that change type.
|
||||||
if (change_type != OutputType::NONE) {
|
if (change_type != OutputType::CHANGE_AUTO) {
|
||||||
return change_type;
|
return change_type;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4022,16 +4022,12 @@ CWallet* CWallet::CreateWalletFromFile(const std::string& name, const fs::path&
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
walletInstance->m_default_address_type = ParseOutputType(gArgs.GetArg("-addresstype", ""), DEFAULT_ADDRESS_TYPE);
|
if (!gArgs.GetArg("-addresstype", "").empty() && !ParseOutputType(gArgs.GetArg("-addresstype", ""), walletInstance->m_default_address_type)) {
|
||||||
if (walletInstance->m_default_address_type == OutputType::NONE) {
|
|
||||||
InitError(strprintf("Unknown address type '%s'", gArgs.GetArg("-addresstype", "")));
|
InitError(strprintf("Unknown address type '%s'", gArgs.GetArg("-addresstype", "")));
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
// If changetype is set in config file or parameter, check that it's valid.
|
if (!gArgs.GetArg("-changetype", "").empty() && !ParseOutputType(gArgs.GetArg("-changetype", ""), walletInstance->m_default_change_type)) {
|
||||||
// Default to OutputType::NONE if not set.
|
|
||||||
walletInstance->m_default_change_type = ParseOutputType(gArgs.GetArg("-changetype", ""), OutputType::NONE);
|
|
||||||
if (walletInstance->m_default_change_type == OutputType::NONE && !gArgs.GetArg("-changetype", "").empty()) {
|
|
||||||
InitError(strprintf("Unknown change type '%s'", gArgs.GetArg("-changetype", "")));
|
InitError(strprintf("Unknown change type '%s'", gArgs.GetArg("-changetype", "")));
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
@ -4219,19 +4215,19 @@ static const std::string OUTPUT_TYPE_STRING_LEGACY = "legacy";
|
||||||
static const std::string OUTPUT_TYPE_STRING_P2SH_SEGWIT = "p2sh-segwit";
|
static const std::string OUTPUT_TYPE_STRING_P2SH_SEGWIT = "p2sh-segwit";
|
||||||
static const std::string OUTPUT_TYPE_STRING_BECH32 = "bech32";
|
static const std::string OUTPUT_TYPE_STRING_BECH32 = "bech32";
|
||||||
|
|
||||||
OutputType ParseOutputType(const std::string& type, OutputType default_type)
|
bool ParseOutputType(const std::string& type, OutputType& output_type)
|
||||||
{
|
{
|
||||||
if (type.empty()) {
|
if (type == OUTPUT_TYPE_STRING_LEGACY) {
|
||||||
return default_type;
|
output_type = OutputType::LEGACY;
|
||||||
} else if (type == OUTPUT_TYPE_STRING_LEGACY) {
|
return true;
|
||||||
return OutputType::LEGACY;
|
|
||||||
} else if (type == OUTPUT_TYPE_STRING_P2SH_SEGWIT) {
|
} else if (type == OUTPUT_TYPE_STRING_P2SH_SEGWIT) {
|
||||||
return OutputType::P2SH_SEGWIT;
|
output_type = OutputType::P2SH_SEGWIT;
|
||||||
|
return true;
|
||||||
} else if (type == OUTPUT_TYPE_STRING_BECH32) {
|
} else if (type == OUTPUT_TYPE_STRING_BECH32) {
|
||||||
return OutputType::BECH32;
|
output_type = OutputType::BECH32;
|
||||||
} else {
|
return true;
|
||||||
return OutputType::NONE;
|
|
||||||
}
|
}
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::string& FormatOutputType(OutputType type)
|
const std::string& FormatOutputType(OutputType type)
|
||||||
|
|
|
@ -99,15 +99,24 @@ enum WalletFeature
|
||||||
};
|
};
|
||||||
|
|
||||||
enum class OutputType {
|
enum class OutputType {
|
||||||
NONE,
|
|
||||||
LEGACY,
|
LEGACY,
|
||||||
P2SH_SEGWIT,
|
P2SH_SEGWIT,
|
||||||
BECH32,
|
BECH32,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Special output type for change outputs only. Automatically choose type
|
||||||
|
* based on address type setting and the types other of non-change outputs
|
||||||
|
* (see -changetype option documentation and implementation in
|
||||||
|
* CWallet::TransactionChangeType for details).
|
||||||
|
*/
|
||||||
|
CHANGE_AUTO,
|
||||||
};
|
};
|
||||||
|
|
||||||
//! Default for -addresstype
|
//! Default for -addresstype
|
||||||
constexpr OutputType DEFAULT_ADDRESS_TYPE{OutputType::P2SH_SEGWIT};
|
constexpr OutputType DEFAULT_ADDRESS_TYPE{OutputType::P2SH_SEGWIT};
|
||||||
|
|
||||||
|
//! Default for -changetype
|
||||||
|
constexpr OutputType DEFAULT_CHANGE_TYPE{OutputType::CHANGE_AUTO};
|
||||||
|
|
||||||
/** A key pool entry */
|
/** A key pool entry */
|
||||||
class CKeyPool
|
class CKeyPool
|
||||||
|
@ -988,7 +997,7 @@ public:
|
||||||
static CFeeRate fallbackFee;
|
static CFeeRate fallbackFee;
|
||||||
static CFeeRate m_discard_rate;
|
static CFeeRate m_discard_rate;
|
||||||
OutputType m_default_address_type{DEFAULT_ADDRESS_TYPE};
|
OutputType m_default_address_type{DEFAULT_ADDRESS_TYPE};
|
||||||
OutputType m_default_change_type{OutputType::NONE}; // Default to OutputType::NONE if not set by -changetype
|
OutputType m_default_change_type{DEFAULT_CHANGE_TYPE};
|
||||||
|
|
||||||
bool NewKeyPool();
|
bool NewKeyPool();
|
||||||
size_t KeypoolCountExternalKeys();
|
size_t KeypoolCountExternalKeys();
|
||||||
|
@ -1232,7 +1241,7 @@ public:
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
OutputType ParseOutputType(const std::string& str, OutputType default_type);
|
bool ParseOutputType(const std::string& str, OutputType& output_type);
|
||||||
const std::string& FormatOutputType(OutputType type);
|
const std::string& FormatOutputType(OutputType type);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -224,7 +224,7 @@ class RawTransactionsTest(BitcoinTestFramework):
|
||||||
outputs = { self.nodes[0].getnewaddress() : Decimal(4.0) }
|
outputs = { self.nodes[0].getnewaddress() : Decimal(4.0) }
|
||||||
rawtx = self.nodes[2].createrawtransaction(inputs, outputs)
|
rawtx = self.nodes[2].createrawtransaction(inputs, outputs)
|
||||||
assert_raises_rpc_error(-1, "JSON value is not a string as expected", self.nodes[2].fundrawtransaction, rawtx, {'change_type': None})
|
assert_raises_rpc_error(-1, "JSON value is not a string as expected", self.nodes[2].fundrawtransaction, rawtx, {'change_type': None})
|
||||||
assert_raises_rpc_error(-5, "Unknown change type", self.nodes[2].fundrawtransaction, rawtx, {'change_type': ''})
|
assert_raises_rpc_error(-5, "Unknown change type ''", self.nodes[2].fundrawtransaction, rawtx, {'change_type': ''})
|
||||||
rawtx = self.nodes[2].fundrawtransaction(rawtx, {'change_type': 'bech32'})
|
rawtx = self.nodes[2].fundrawtransaction(rawtx, {'change_type': 'bech32'})
|
||||||
dec_tx = self.nodes[2].decoderawtransaction(rawtx['hex'])
|
dec_tx = self.nodes[2].decoderawtransaction(rawtx['hex'])
|
||||||
assert_equal('witness_v0_keyhash', dec_tx['vout'][rawtx['changepos']]['scriptPubKey']['type'])
|
assert_equal('witness_v0_keyhash', dec_tx['vout'][rawtx['changepos']]['scriptPubKey']['type'])
|
||||||
|
|
|
@ -280,7 +280,10 @@ class AddressTypeTest(BitcoinTestFramework):
|
||||||
self.log.info('getrawchangeaddress defaults to addresstype if -changetype is not set and argument is absent')
|
self.log.info('getrawchangeaddress defaults to addresstype if -changetype is not set and argument is absent')
|
||||||
self.test_address(3, self.nodes[3].getrawchangeaddress(), multisig=False, typ='bech32')
|
self.test_address(3, self.nodes[3].getrawchangeaddress(), multisig=False, typ='bech32')
|
||||||
|
|
||||||
self.log.info('getrawchangeaddress fails with invalid changetype argument')
|
self.log.info('test invalid address type arguments')
|
||||||
|
assert_raises_rpc_error(-5, "Unknown address type ''", self.nodes[3].addmultisigaddress, 2, [compressed_1, compressed_2], None, '')
|
||||||
|
assert_raises_rpc_error(-5, "Unknown address type ''", self.nodes[3].getnewaddress, None, '')
|
||||||
|
assert_raises_rpc_error(-5, "Unknown address type ''", self.nodes[3].getrawchangeaddress, '')
|
||||||
assert_raises_rpc_error(-5, "Unknown address type 'bech23'", self.nodes[3].getrawchangeaddress, 'bech23')
|
assert_raises_rpc_error(-5, "Unknown address type 'bech23'", self.nodes[3].getrawchangeaddress, 'bech23')
|
||||||
|
|
||||||
self.log.info("Nodes with changetype=p2sh-segwit never use a P2WPKH change output")
|
self.log.info("Nodes with changetype=p2sh-segwit never use a P2WPKH change output")
|
||||||
|
|
Loading…
Add table
Reference in a new issue