mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-25 02:33:24 -03:00
Merge bitcoin/bitcoin#31096: Package validation: accept packages of size 1
32fc59796f
rpc: Allow single transaction through submitpackage (glozow) Pull request description: There's no particular reason to restrict single transaction submissions with submitpackage. This change relaxes the RPC checks as enables the `AcceptPackage` flow to accept packages of a single transaction. Resolves #31085 ACKs for top commit: naumenkogs: ACK32fc59796f
achow101: ACK32fc59796f
glozow: ACK32fc59796f
Tree-SHA512: ffed353bfdca610ffcfd53b40b76da05ffc26df6bac4b0421492e067bede930380e03399d2e2d1d17f0e88fb91cd8eb376e3aabebbabcc724590bf068d09807c
This commit is contained in:
commit
c9a7418a8d
5 changed files with 155 additions and 54 deletions
|
@ -74,7 +74,7 @@ The following rules are only enforced for packages to be submitted to the mempoo
|
||||||
enforced for test accepts):
|
enforced for test accepts):
|
||||||
|
|
||||||
* Packages must be child-with-unconfirmed-parents packages. This also means packages must contain at
|
* Packages must be child-with-unconfirmed-parents packages. This also means packages must contain at
|
||||||
least 2 transactions. (#22674)
|
least 1 transaction. (#31096)
|
||||||
|
|
||||||
- *Rationale*: This allows for fee-bumping by CPFP. Allowing multiple parents makes it possible
|
- *Rationale*: This allows for fee-bumping by CPFP. Allowing multiple parents makes it possible
|
||||||
to fee-bump a batch of transactions. Restricting packages to a defined topology is easier to
|
to fee-bump a batch of transactions. Restricting packages to a defined topology is easier to
|
||||||
|
|
|
@ -926,7 +926,7 @@ static RPCHelpMan submitpackage()
|
||||||
,
|
,
|
||||||
{
|
{
|
||||||
{"package", RPCArg::Type::ARR, RPCArg::Optional::NO, "An array of raw transactions.\n"
|
{"package", RPCArg::Type::ARR, RPCArg::Optional::NO, "An array of raw transactions.\n"
|
||||||
"The package must solely consist of a child and its parents. None of the parents may depend on each other.\n"
|
"The package must solely consist of a child transaction and all of its unconfirmed parents, if any. None of the parents may depend on each other.\n"
|
||||||
"The package must be topologically sorted, with the child being the last element in the array.",
|
"The package must be topologically sorted, with the child being the last element in the array.",
|
||||||
{
|
{
|
||||||
{"rawtx", RPCArg::Type::STR_HEX, RPCArg::Optional::OMITTED, ""},
|
{"rawtx", RPCArg::Type::STR_HEX, RPCArg::Optional::OMITTED, ""},
|
||||||
|
@ -968,15 +968,15 @@ static RPCHelpMan submitpackage()
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
RPCExamples{
|
RPCExamples{
|
||||||
HelpExampleRpc("submitpackage", R"(["rawtx1", "rawtx2"])") +
|
HelpExampleRpc("submitpackage", R"(["raw-parent-tx-1", "raw-parent-tx-2", "raw-child-tx"])") +
|
||||||
HelpExampleCli("submitpackage", R"('["rawtx1", "rawtx2"]')")
|
HelpExampleCli("submitpackage", R"('["raw-tx-without-unconfirmed-parents"]')")
|
||||||
},
|
},
|
||||||
[&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
|
[&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue
|
||||||
{
|
{
|
||||||
const UniValue raw_transactions = request.params[0].get_array();
|
const UniValue raw_transactions = request.params[0].get_array();
|
||||||
if (raw_transactions.size() < 2 || raw_transactions.size() > MAX_PACKAGE_COUNT) {
|
if (raw_transactions.empty() || raw_transactions.size() > MAX_PACKAGE_COUNT) {
|
||||||
throw JSONRPCError(RPC_INVALID_PARAMETER,
|
throw JSONRPCError(RPC_INVALID_PARAMETER,
|
||||||
"Array must contain between 2 and " + ToString(MAX_PACKAGE_COUNT) + " transactions.");
|
"Array must contain between 1 and " + ToString(MAX_PACKAGE_COUNT) + " transactions.");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fee check needs to be run with chainstate and package context
|
// Fee check needs to be run with chainstate and package context
|
||||||
|
@ -1007,7 +1007,8 @@ static RPCHelpMan submitpackage()
|
||||||
|
|
||||||
txns.emplace_back(MakeTransactionRef(std::move(mtx)));
|
txns.emplace_back(MakeTransactionRef(std::move(mtx)));
|
||||||
}
|
}
|
||||||
if (!IsChildWithParentsTree(txns)) {
|
CHECK_NONFATAL(!txns.empty());
|
||||||
|
if (txns.size() > 1 && !IsChildWithParentsTree(txns)) {
|
||||||
throw JSONRPCTransactionError(TransactionError::INVALID_PACKAGE, "package topology disallowed. not child-with-parents or parents depend on each other.");
|
throw JSONRPCTransactionError(TransactionError::INVALID_PACKAGE, "package topology disallowed. not child-with-parents or parents depend on each other.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -283,6 +283,8 @@ BOOST_AUTO_TEST_CASE(noncontextual_package_tests)
|
||||||
BOOST_CHECK(GetPackageHash({tx_parent}) != GetPackageHash({tx_child}));
|
BOOST_CHECK(GetPackageHash({tx_parent}) != GetPackageHash({tx_child}));
|
||||||
BOOST_CHECK(GetPackageHash({tx_child, tx_child}) != GetPackageHash({tx_child}));
|
BOOST_CHECK(GetPackageHash({tx_child, tx_child}) != GetPackageHash({tx_child}));
|
||||||
BOOST_CHECK(GetPackageHash({tx_child, tx_parent}) != GetPackageHash({tx_child, tx_child}));
|
BOOST_CHECK(GetPackageHash({tx_child, tx_parent}) != GetPackageHash({tx_child, tx_child}));
|
||||||
|
BOOST_CHECK(!IsChildWithParents({}));
|
||||||
|
BOOST_CHECK(!IsChildWithParentsTree({}));
|
||||||
}
|
}
|
||||||
|
|
||||||
// 24 Parents and 1 Child
|
// 24 Parents and 1 Child
|
||||||
|
@ -492,6 +494,97 @@ BOOST_AUTO_TEST_CASE(package_submission_tests)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Tests for packages containing a single transaction
|
||||||
|
BOOST_AUTO_TEST_CASE(package_single_tx)
|
||||||
|
{
|
||||||
|
// Mine blocks to mature coinbases.
|
||||||
|
mineBlocks(3);
|
||||||
|
LOCK(cs_main);
|
||||||
|
auto expected_pool_size{m_node.mempool->size()};
|
||||||
|
|
||||||
|
const CAmount high_fee{1000};
|
||||||
|
|
||||||
|
// No unconfirmed parents
|
||||||
|
CKey single_key = GenerateRandomKey();
|
||||||
|
CScript single_locking_script = GetScriptForDestination(PKHash(single_key.GetPubKey()));
|
||||||
|
auto mtx_single = CreateValidMempoolTransaction(/*input_transaction=*/m_coinbase_txns[0], /*input_vout=*/0,
|
||||||
|
/*input_height=*/0, /*input_signing_key=*/coinbaseKey,
|
||||||
|
/*output_destination=*/single_locking_script,
|
||||||
|
/*output_amount=*/CAmount(49 * COIN), /*submit=*/false);
|
||||||
|
CTransactionRef tx_single = MakeTransactionRef(mtx_single);
|
||||||
|
Package package_tx_single{tx_single};
|
||||||
|
const auto result_single_tx = ProcessNewPackage(m_node.chainman->ActiveChainstate(), *m_node.mempool,
|
||||||
|
package_tx_single, /*test_accept=*/false, /*client_maxfeerate=*/{});
|
||||||
|
expected_pool_size += 1;
|
||||||
|
BOOST_CHECK_MESSAGE(result_single_tx.m_state.IsValid(),
|
||||||
|
"Package validation unexpectedly failed: " << result_single_tx.m_state.ToString());
|
||||||
|
BOOST_CHECK_EQUAL(m_node.mempool->size(), expected_pool_size);
|
||||||
|
|
||||||
|
// Parent and Child. Both submitted by themselves through the ProcessNewPackage interface.
|
||||||
|
CKey parent_key = GenerateRandomKey();
|
||||||
|
CScript parent_locking_script = GetScriptForDestination(WitnessV0KeyHash(parent_key.GetPubKey()));
|
||||||
|
auto mtx_parent = CreateValidMempoolTransaction(/*input_transaction=*/m_coinbase_txns[1], /*input_vout=*/0,
|
||||||
|
/*input_height=*/0, /*input_signing_key=*/coinbaseKey,
|
||||||
|
/*output_destination=*/parent_locking_script,
|
||||||
|
/*output_amount=*/CAmount(50 * COIN) - high_fee, /*submit=*/false);
|
||||||
|
CTransactionRef tx_parent = MakeTransactionRef(mtx_parent);
|
||||||
|
Package package_just_parent{tx_parent};
|
||||||
|
const auto result_just_parent = ProcessNewPackage(m_node.chainman->ActiveChainstate(), *m_node.mempool, package_just_parent, /*test_accept=*/false, /*client_maxfeerate=*/{});
|
||||||
|
if (auto err_parent_child{CheckPackageMempoolAcceptResult(package_just_parent, result_just_parent, /*expect_valid=*/true, nullptr)}) {
|
||||||
|
BOOST_ERROR(err_parent_child.value());
|
||||||
|
} else {
|
||||||
|
auto it_parent = result_just_parent.m_tx_results.find(tx_parent->GetWitnessHash());
|
||||||
|
BOOST_CHECK_MESSAGE(it_parent->second.m_state.IsValid(), it_parent->second.m_state.ToString());
|
||||||
|
BOOST_CHECK(it_parent->second.m_effective_feerate.value().GetFee(GetVirtualTransactionSize(*tx_parent)) == high_fee);
|
||||||
|
BOOST_CHECK_EQUAL(it_parent->second.m_wtxids_fee_calculations.value().size(), 1);
|
||||||
|
BOOST_CHECK_EQUAL(it_parent->second.m_wtxids_fee_calculations.value().front(), tx_parent->GetWitnessHash());
|
||||||
|
}
|
||||||
|
expected_pool_size += 1;
|
||||||
|
BOOST_CHECK_EQUAL(m_node.mempool->size(), expected_pool_size);
|
||||||
|
|
||||||
|
CKey child_key = GenerateRandomKey();
|
||||||
|
CScript child_locking_script = GetScriptForDestination(WitnessV0KeyHash(child_key.GetPubKey()));
|
||||||
|
auto mtx_child = CreateValidMempoolTransaction(/*input_transaction=*/tx_parent, /*input_vout=*/0,
|
||||||
|
/*input_height=*/101, /*input_signing_key=*/parent_key,
|
||||||
|
/*output_destination=*/child_locking_script,
|
||||||
|
/*output_amount=*/CAmount(50 * COIN) - 2 * high_fee, /*submit=*/false);
|
||||||
|
CTransactionRef tx_child = MakeTransactionRef(mtx_child);
|
||||||
|
Package package_just_child{tx_child};
|
||||||
|
const auto result_just_child = ProcessNewPackage(m_node.chainman->ActiveChainstate(), *m_node.mempool, package_just_child, /*test_accept=*/false, /*client_maxfeerate=*/{});
|
||||||
|
if (auto err_parent_child{CheckPackageMempoolAcceptResult(package_just_child, result_just_child, /*expect_valid=*/true, nullptr)}) {
|
||||||
|
BOOST_ERROR(err_parent_child.value());
|
||||||
|
} else {
|
||||||
|
auto it_child = result_just_child.m_tx_results.find(tx_child->GetWitnessHash());
|
||||||
|
BOOST_CHECK_MESSAGE(it_child->second.m_state.IsValid(), it_child->second.m_state.ToString());
|
||||||
|
BOOST_CHECK(it_child->second.m_effective_feerate.value().GetFee(GetVirtualTransactionSize(*tx_child)) == high_fee);
|
||||||
|
BOOST_CHECK_EQUAL(it_child->second.m_wtxids_fee_calculations.value().size(), 1);
|
||||||
|
BOOST_CHECK_EQUAL(it_child->second.m_wtxids_fee_calculations.value().front(), tx_child->GetWitnessHash());
|
||||||
|
}
|
||||||
|
expected_pool_size += 1;
|
||||||
|
BOOST_CHECK_EQUAL(m_node.mempool->size(), expected_pool_size);
|
||||||
|
|
||||||
|
// Too-low fee to RBF tx_single
|
||||||
|
auto mtx_single_low_fee = CreateValidMempoolTransaction(/*input_transaction=*/m_coinbase_txns[0], /*input_vout=*/0,
|
||||||
|
/*input_height=*/0, /*input_signing_key=*/coinbaseKey,
|
||||||
|
/*output_destination=*/single_locking_script,
|
||||||
|
/*output_amount=*/CAmount(49 * COIN - 1), /*submit=*/false);
|
||||||
|
CTransactionRef tx_single_low_fee = MakeTransactionRef(mtx_single_low_fee);
|
||||||
|
Package package_tx_single_low_fee{tx_single_low_fee};
|
||||||
|
const auto result_single_tx_low_fee = ProcessNewPackage(m_node.chainman->ActiveChainstate(), *m_node.mempool,
|
||||||
|
package_tx_single_low_fee, /*test_accept=*/false, /*client_maxfeerate=*/{});
|
||||||
|
|
||||||
|
BOOST_CHECK_EQUAL(m_node.mempool->size(), expected_pool_size);
|
||||||
|
|
||||||
|
BOOST_CHECK(!result_single_tx_low_fee.m_state.IsValid());
|
||||||
|
BOOST_CHECK_EQUAL(result_single_tx_low_fee.m_state.GetResult(), PackageValidationResult::PCKG_TX);
|
||||||
|
auto it_low_fee = result_single_tx_low_fee.m_tx_results.find(tx_single_low_fee->GetWitnessHash());
|
||||||
|
BOOST_CHECK_EQUAL(it_low_fee->second.m_state.GetResult(), TxValidationResult::TX_RECONSIDERABLE);
|
||||||
|
if (auto err_single{CheckPackageMempoolAcceptResult(package_tx_single_low_fee, result_single_tx_low_fee, /*expect_valid=*/false, m_node.mempool.get())}) {
|
||||||
|
BOOST_ERROR(err_single.value());
|
||||||
|
}
|
||||||
|
BOOST_CHECK_EQUAL(m_node.mempool->size(), expected_pool_size);
|
||||||
|
}
|
||||||
|
|
||||||
// Tests for packages containing transactions that have same-txid-different-witness equivalents in
|
// Tests for packages containing transactions that have same-txid-different-witness equivalents in
|
||||||
// the mempool.
|
// the mempool.
|
||||||
BOOST_AUTO_TEST_CASE(package_witness_swap_tests)
|
BOOST_AUTO_TEST_CASE(package_witness_swap_tests)
|
||||||
|
|
|
@ -1685,10 +1685,14 @@ PackageMempoolAcceptResult MemPoolAccept::AcceptSubPackage(const std::vector<CTr
|
||||||
|
|
||||||
PackageMempoolAcceptResult MemPoolAccept::AcceptPackage(const Package& package, ATMPArgs& args)
|
PackageMempoolAcceptResult MemPoolAccept::AcceptPackage(const Package& package, ATMPArgs& args)
|
||||||
{
|
{
|
||||||
|
Assert(!package.empty());
|
||||||
AssertLockHeld(cs_main);
|
AssertLockHeld(cs_main);
|
||||||
// Used if returning a PackageMempoolAcceptResult directly from this function.
|
// Used if returning a PackageMempoolAcceptResult directly from this function.
|
||||||
PackageValidationState package_state_quit_early;
|
PackageValidationState package_state_quit_early;
|
||||||
|
|
||||||
|
// There are two topologies we are able to handle through this function:
|
||||||
|
// (1) A single transaction
|
||||||
|
// (2) A child-with-unconfirmed-parents package.
|
||||||
// Check that the package is well-formed. If it isn't, we won't try to validate any of the
|
// Check that the package is well-formed. If it isn't, we won't try to validate any of the
|
||||||
// transactions and thus won't return any MempoolAcceptResults, just a package-wide error.
|
// transactions and thus won't return any MempoolAcceptResults, just a package-wide error.
|
||||||
|
|
||||||
|
@ -1697,6 +1701,7 @@ PackageMempoolAcceptResult MemPoolAccept::AcceptPackage(const Package& package,
|
||||||
return PackageMempoolAcceptResult(package_state_quit_early, {});
|
return PackageMempoolAcceptResult(package_state_quit_early, {});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (package.size() > 1) {
|
||||||
// All transactions in the package must be a parent of the last transaction. This is just an
|
// All transactions in the package must be a parent of the last transaction. This is just an
|
||||||
// opportunity for us to fail fast on a context-free check without taking the mempool lock.
|
// opportunity for us to fail fast on a context-free check without taking the mempool lock.
|
||||||
if (!IsChildWithParents(package)) {
|
if (!IsChildWithParents(package)) {
|
||||||
|
@ -1739,6 +1744,7 @@ PackageMempoolAcceptResult MemPoolAccept::AcceptPackage(const Package& package,
|
||||||
// Protect against bugs where we pull more inputs from disk that miss being added to
|
// Protect against bugs where we pull more inputs from disk that miss being added to
|
||||||
// coins_to_uncache. The backend will be connected again when needed in PreChecks.
|
// coins_to_uncache. The backend will be connected again when needed in PreChecks.
|
||||||
m_view.SetBackend(m_dummy);
|
m_view.SetBackend(m_dummy);
|
||||||
|
}
|
||||||
|
|
||||||
LOCK(m_pool.cs);
|
LOCK(m_pool.cs);
|
||||||
// Stores results from which we will create the returned PackageMempoolAcceptResult.
|
// Stores results from which we will create the returned PackageMempoolAcceptResult.
|
||||||
|
@ -1748,6 +1754,7 @@ PackageMempoolAcceptResult MemPoolAccept::AcceptPackage(const Package& package,
|
||||||
// this transaction. "Nonfinal" because if a transaction fails by itself but succeeds later
|
// this transaction. "Nonfinal" because if a transaction fails by itself but succeeds later
|
||||||
// (i.e. when evaluated with a fee-bumping child), the result in this map may be discarded.
|
// (i.e. when evaluated with a fee-bumping child), the result in this map may be discarded.
|
||||||
std::map<uint256, MempoolAcceptResult> individual_results_nonfinal;
|
std::map<uint256, MempoolAcceptResult> individual_results_nonfinal;
|
||||||
|
// Tracks whether we think package submission could result in successful entry to the mempool
|
||||||
bool quit_early{false};
|
bool quit_early{false};
|
||||||
std::vector<CTransactionRef> txns_package_eval;
|
std::vector<CTransactionRef> txns_package_eval;
|
||||||
for (const auto& tx : package) {
|
for (const auto& tx : package) {
|
||||||
|
@ -1789,8 +1796,9 @@ PackageMempoolAcceptResult MemPoolAccept::AcceptPackage(const Package& package,
|
||||||
// in package validation, because its fees should only be "used" once.
|
// in package validation, because its fees should only be "used" once.
|
||||||
assert(m_pool.exists(GenTxid::Wtxid(wtxid)));
|
assert(m_pool.exists(GenTxid::Wtxid(wtxid)));
|
||||||
results_final.emplace(wtxid, single_res);
|
results_final.emplace(wtxid, single_res);
|
||||||
} else if (single_res.m_state.GetResult() != TxValidationResult::TX_RECONSIDERABLE &&
|
} else if (package.size() == 1 || // If there is only one transaction, no need to retry it "as a package"
|
||||||
single_res.m_state.GetResult() != TxValidationResult::TX_MISSING_INPUTS) {
|
(single_res.m_state.GetResult() != TxValidationResult::TX_RECONSIDERABLE &&
|
||||||
|
single_res.m_state.GetResult() != TxValidationResult::TX_MISSING_INPUTS)) {
|
||||||
// Package validation policy only differs from individual policy in its evaluation
|
// Package validation policy only differs from individual policy in its evaluation
|
||||||
// of feerate. For example, if a transaction fails here due to violation of a
|
// of feerate. For example, if a transaction fails here due to violation of a
|
||||||
// consensus rule, the result will not change when it is submitted as part of a
|
// consensus rule, the result will not change when it is submitted as part of a
|
||||||
|
|
|
@ -377,8 +377,8 @@ class RPCPackagesTest(BitcoinTestFramework):
|
||||||
assert txid_list[0] not in node.getrawmempool()
|
assert txid_list[0] not in node.getrawmempool()
|
||||||
assert txid_list[1] not in node.getrawmempool()
|
assert txid_list[1] not in node.getrawmempool()
|
||||||
|
|
||||||
self.log.info("Submitpackage valid packages with 1 child and some number of parents")
|
self.log.info("Submitpackage valid packages with 1 child and some number of parents (or none)")
|
||||||
for num_parents in [1, 2, 24]:
|
for num_parents in [0, 1, 2, 24]:
|
||||||
self.test_submit_child_with_parents(num_parents, False)
|
self.test_submit_child_with_parents(num_parents, False)
|
||||||
self.test_submit_child_with_parents(num_parents, True)
|
self.test_submit_child_with_parents(num_parents, True)
|
||||||
|
|
||||||
|
@ -389,10 +389,9 @@ class RPCPackagesTest(BitcoinTestFramework):
|
||||||
assert_raises_rpc_error(-25, "package topology disallowed", node.submitpackage, chain_hex)
|
assert_raises_rpc_error(-25, "package topology disallowed", node.submitpackage, chain_hex)
|
||||||
assert_equal(legacy_pool, node.getrawmempool())
|
assert_equal(legacy_pool, node.getrawmempool())
|
||||||
|
|
||||||
assert_raises_rpc_error(-8, f"Array must contain between 2 and {MAX_PACKAGE_COUNT} transactions.", node.submitpackage, [])
|
assert_raises_rpc_error(-8, f"Array must contain between 1 and {MAX_PACKAGE_COUNT} transactions.", node.submitpackage, [])
|
||||||
assert_raises_rpc_error(-8, f"Array must contain between 2 and {MAX_PACKAGE_COUNT} transactions.", node.submitpackage, [chain_hex[0]] * 1)
|
|
||||||
assert_raises_rpc_error(
|
assert_raises_rpc_error(
|
||||||
-8, f"Array must contain between 2 and {MAX_PACKAGE_COUNT} transactions.",
|
-8, f"Array must contain between 1 and {MAX_PACKAGE_COUNT} transactions.",
|
||||||
node.submitpackage, [chain_hex[0]] * (MAX_PACKAGE_COUNT + 1)
|
node.submitpackage, [chain_hex[0]] * (MAX_PACKAGE_COUNT + 1)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue