mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-25 02:33:24 -03:00
Move CalculateChunksForRBF() to the mempool changeset
This commit is contained in:
parent
284a1d33f1
commit
d7dc9fd2f7
7 changed files with 147 additions and 73 deletions
|
@ -184,14 +184,10 @@ std::optional<std::string> PaysForRBF(CAmount original_fees,
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::optional<std::pair<DiagramCheckError, std::string>> ImprovesFeerateDiagram(CTxMemPool& pool,
|
std::optional<std::pair<DiagramCheckError, std::string>> ImprovesFeerateDiagram(CTxMemPool::ChangeSet& changeset)
|
||||||
const CTxMemPool::setEntries& direct_conflicts,
|
|
||||||
const CTxMemPool::setEntries& all_conflicts,
|
|
||||||
CAmount replacement_fees,
|
|
||||||
int64_t replacement_vsize)
|
|
||||||
{
|
{
|
||||||
// Require that the replacement strictly improves the mempool's feerate diagram.
|
// Require that the replacement strictly improves the mempool's feerate diagram.
|
||||||
const auto chunk_results{pool.CalculateChunksForRBF(replacement_fees, replacement_vsize, all_conflicts, all_conflicts)};
|
const auto chunk_results{changeset.CalculateChunksForRBF()};
|
||||||
|
|
||||||
if (!chunk_results.has_value()) {
|
if (!chunk_results.has_value()) {
|
||||||
return std::make_pair(DiagramCheckError::UNCALCULABLE, util::ErrorString(chunk_results).original);
|
return std::make_pair(DiagramCheckError::UNCALCULABLE, util::ErrorString(chunk_results).original);
|
||||||
|
|
|
@ -117,19 +117,9 @@ std::optional<std::string> PaysForRBF(CAmount original_fees,
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The replacement transaction must improve the feerate diagram of the mempool.
|
* The replacement transaction must improve the feerate diagram of the mempool.
|
||||||
* @param[in] pool The mempool.
|
* @param[in] changeset The changeset containing proposed additions/removals
|
||||||
* @param[in] direct_conflicts Set of in-mempool txids corresponding to the direct conflicts i.e.
|
|
||||||
* input double-spends with the proposed transaction
|
|
||||||
* @param[in] all_conflicts Set of mempool entries corresponding to all transactions to be evicted
|
|
||||||
* @param[in] replacement_fees Fees of proposed replacement package
|
|
||||||
* @param[in] replacement_vsize Size of proposed replacement package
|
|
||||||
* @returns error type and string if mempool diagram doesn't improve, otherwise std::nullopt.
|
* @returns error type and string if mempool diagram doesn't improve, otherwise std::nullopt.
|
||||||
*/
|
*/
|
||||||
std::optional<std::pair<DiagramCheckError, std::string>> ImprovesFeerateDiagram(CTxMemPool& pool,
|
std::optional<std::pair<DiagramCheckError, std::string>> ImprovesFeerateDiagram(CTxMemPool::ChangeSet& changeset);
|
||||||
const CTxMemPool::setEntries& direct_conflicts,
|
|
||||||
const CTxMemPool::setEntries& all_conflicts,
|
|
||||||
CAmount replacement_fees,
|
|
||||||
int64_t replacement_vsize)
|
|
||||||
EXCLUSIVE_LOCKS_REQUIRED(pool.cs);
|
|
||||||
|
|
||||||
#endif // BITCOIN_POLICY_RBF_H
|
#endif // BITCOIN_POLICY_RBF_H
|
||||||
|
|
|
@ -104,10 +104,18 @@ FUZZ_TARGET(package_rbf, .init = initialize_package_rbf)
|
||||||
std::vector<CTransaction> mempool_txs;
|
std::vector<CTransaction> mempool_txs;
|
||||||
size_t iter{0};
|
size_t iter{0};
|
||||||
|
|
||||||
int32_t replacement_vsize = fuzzed_data_provider.ConsumeIntegralInRange<int32_t>(1, 1000000);
|
|
||||||
|
|
||||||
// Keep track of the total vsize of CTxMemPoolEntry's being added to the mempool to avoid overflow
|
// Keep track of the total vsize of CTxMemPoolEntry's being added to the mempool to avoid overflow
|
||||||
// Add replacement_vsize since this is added to new diagram during RBF check
|
// Add replacement_vsize since this is added to new diagram during RBF check
|
||||||
|
std::optional<CMutableTransaction> replacement_tx = ConsumeDeserializable<CMutableTransaction>(fuzzed_data_provider, TX_WITH_WITNESS);
|
||||||
|
if (!replacement_tx) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
assert(iter <= g_outpoints.size());
|
||||||
|
replacement_tx->vin.resize(1);
|
||||||
|
replacement_tx->vin[0].prevout = g_outpoints[iter++];
|
||||||
|
CTransaction replacement_tx_final{*replacement_tx};
|
||||||
|
auto replacement_entry = ConsumeTxMemPoolEntry(fuzzed_data_provider, replacement_tx_final);
|
||||||
|
int32_t replacement_vsize = replacement_entry.GetTxSize();
|
||||||
int64_t running_vsize_total{replacement_vsize};
|
int64_t running_vsize_total{replacement_vsize};
|
||||||
|
|
||||||
LOCK2(cs_main, pool.cs);
|
LOCK2(cs_main, pool.cs);
|
||||||
|
@ -162,9 +170,17 @@ FUZZ_TARGET(package_rbf, .init = initialize_package_rbf)
|
||||||
pool.CalculateDescendants(txiter, all_conflicts);
|
pool.CalculateDescendants(txiter, all_conflicts);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calculate the chunks for a replacement.
|
|
||||||
CAmount replacement_fees = ConsumeMoney(fuzzed_data_provider);
|
CAmount replacement_fees = ConsumeMoney(fuzzed_data_provider);
|
||||||
auto calc_results{pool.CalculateChunksForRBF(replacement_fees, replacement_vsize, direct_conflicts, all_conflicts)};
|
auto changeset = pool.GetChangeSet();
|
||||||
|
for (auto& txiter : all_conflicts) {
|
||||||
|
changeset->StageRemoval(txiter);
|
||||||
|
}
|
||||||
|
changeset->StageAddition(replacement_entry.GetSharedTx(), replacement_fees,
|
||||||
|
replacement_entry.GetTime().count(), replacement_entry.GetHeight(),
|
||||||
|
replacement_entry.GetSequence(), replacement_entry.GetSpendsCoinbase(),
|
||||||
|
replacement_entry.GetSigOpCost(), replacement_entry.GetLockPoints());
|
||||||
|
// Calculate the chunks for a replacement.
|
||||||
|
auto calc_results{changeset->CalculateChunksForRBF()};
|
||||||
|
|
||||||
if (calc_results.has_value()) {
|
if (calc_results.has_value()) {
|
||||||
// Sanity checks on the chunks.
|
// Sanity checks on the chunks.
|
||||||
|
@ -192,7 +208,7 @@ FUZZ_TARGET(package_rbf, .init = initialize_package_rbf)
|
||||||
}
|
}
|
||||||
|
|
||||||
// If internals report error, wrapper should too
|
// If internals report error, wrapper should too
|
||||||
auto err_tuple{ImprovesFeerateDiagram(pool, direct_conflicts, all_conflicts, replacement_fees, replacement_vsize)};
|
auto err_tuple{ImprovesFeerateDiagram(*changeset)};
|
||||||
if (!calc_results.has_value()) {
|
if (!calc_results.has_value()) {
|
||||||
assert(err_tuple.value().first == DiagramCheckError::UNCALCULABLE);
|
assert(err_tuple.value().first == DiagramCheckError::UNCALCULABLE);
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -369,44 +369,79 @@ BOOST_FIXTURE_TEST_CASE(improves_feerate, TestChain100Setup)
|
||||||
|
|
||||||
const auto entry1 = pool.GetIter(tx1->GetHash()).value();
|
const auto entry1 = pool.GetIter(tx1->GetHash()).value();
|
||||||
const auto tx1_fee = entry1->GetModifiedFee();
|
const auto tx1_fee = entry1->GetModifiedFee();
|
||||||
const auto tx1_size = entry1->GetTxSize();
|
|
||||||
const auto entry2 = pool.GetIter(tx2->GetHash()).value();
|
const auto entry2 = pool.GetIter(tx2->GetHash()).value();
|
||||||
const auto tx2_fee = entry2->GetModifiedFee();
|
const auto tx2_fee = entry2->GetModifiedFee();
|
||||||
const auto tx2_size = entry2->GetTxSize();
|
|
||||||
|
|
||||||
// conflicting transactions
|
// conflicting transactions
|
||||||
const auto tx1_conflict = make_tx(/*inputs=*/ {m_coinbase_txns[0], m_coinbase_txns[2]}, /*output_values=*/ {10 * COIN});
|
const auto tx1_conflict = make_tx(/*inputs=*/ {m_coinbase_txns[0], m_coinbase_txns[2]}, /*output_values=*/ {10 * COIN});
|
||||||
|
const auto tx3 = make_tx(/*inputs=*/ {tx1_conflict}, /*output_values=*/ {995 * CENT});
|
||||||
|
auto entry3 = entry.FromTx(tx3);
|
||||||
|
|
||||||
// Now test ImprovesFeerateDiagram with various levels of "package rbf" feerates
|
// Now test ImprovesFeerateDiagram with various levels of "package rbf" feerates
|
||||||
|
|
||||||
// It doesn't improve itself
|
// It doesn't improve itself
|
||||||
const auto res1 = ImprovesFeerateDiagram(pool, {entry1}, {entry1, entry2}, tx1_fee + tx2_fee, tx1_size + tx2_size);
|
auto changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(entry1);
|
||||||
|
changeset->StageRemoval(entry2);
|
||||||
|
changeset->StageAddition(tx1_conflict, tx1_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
changeset->StageAddition(tx3, tx2_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
const auto res1 = ImprovesFeerateDiagram(*changeset);
|
||||||
BOOST_CHECK(res1.has_value());
|
BOOST_CHECK(res1.has_value());
|
||||||
BOOST_CHECK(res1.value().first == DiagramCheckError::FAILURE);
|
BOOST_CHECK(res1.value().first == DiagramCheckError::FAILURE);
|
||||||
BOOST_CHECK(res1.value().second == "insufficient feerate: does not improve feerate diagram");
|
BOOST_CHECK(res1.value().second == "insufficient feerate: does not improve feerate diagram");
|
||||||
|
|
||||||
// With one more satoshi it does
|
// With one more satoshi it does
|
||||||
BOOST_CHECK(ImprovesFeerateDiagram(pool, {entry1}, {entry1, entry2}, tx1_fee + tx2_fee + 1, tx1_size + tx2_size) == std::nullopt);
|
changeset.reset();
|
||||||
|
changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(entry1);
|
||||||
|
changeset->StageRemoval(entry2);
|
||||||
|
changeset->StageAddition(tx1_conflict, tx1_fee+1, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
changeset->StageAddition(tx3, tx2_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
BOOST_CHECK(ImprovesFeerateDiagram(*changeset) == std::nullopt);
|
||||||
|
|
||||||
|
changeset.reset();
|
||||||
// With prioritisation of in-mempool conflicts, it affects the results of the comparison using the same args as just above
|
// With prioritisation of in-mempool conflicts, it affects the results of the comparison using the same args as just above
|
||||||
pool.PrioritiseTransaction(entry1->GetSharedTx()->GetHash(), /*nFeeDelta=*/1);
|
pool.PrioritiseTransaction(entry1->GetSharedTx()->GetHash(), /*nFeeDelta=*/1);
|
||||||
const auto res2 = ImprovesFeerateDiagram(pool, {entry1}, {entry1, entry2}, tx1_fee + tx2_fee + 1, tx1_size + tx2_size);
|
changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(entry1);
|
||||||
|
changeset->StageRemoval(entry2);
|
||||||
|
changeset->StageAddition(tx1_conflict, tx1_fee+1, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
changeset->StageAddition(tx3, tx2_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
const auto res2 = ImprovesFeerateDiagram(*changeset);
|
||||||
BOOST_CHECK(res2.has_value());
|
BOOST_CHECK(res2.has_value());
|
||||||
BOOST_CHECK(res2.value().first == DiagramCheckError::FAILURE);
|
BOOST_CHECK(res2.value().first == DiagramCheckError::FAILURE);
|
||||||
BOOST_CHECK(res2.value().second == "insufficient feerate: does not improve feerate diagram");
|
BOOST_CHECK(res2.value().second == "insufficient feerate: does not improve feerate diagram");
|
||||||
|
changeset.reset();
|
||||||
|
|
||||||
pool.PrioritiseTransaction(entry1->GetSharedTx()->GetHash(), /*nFeeDelta=*/-1);
|
pool.PrioritiseTransaction(entry1->GetSharedTx()->GetHash(), /*nFeeDelta=*/-1);
|
||||||
|
|
||||||
// With one less vB it does
|
// With fewer vbytes it does
|
||||||
BOOST_CHECK(ImprovesFeerateDiagram(pool, {entry1}, {entry1, entry2}, tx1_fee + tx2_fee, tx1_size + tx2_size - 1) == std::nullopt);
|
CMutableTransaction tx4{entry3.GetTx()};
|
||||||
|
tx4.vin[0].scriptWitness = CScriptWitness(); // Clear out the witness, to reduce size
|
||||||
|
auto entry4 = entry.FromTx(MakeTransactionRef(tx4));
|
||||||
|
changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(entry1);
|
||||||
|
changeset->StageRemoval(entry2);
|
||||||
|
changeset->StageAddition(tx1_conflict, tx1_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
changeset->StageAddition(entry4.GetSharedTx(), tx2_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
BOOST_CHECK(ImprovesFeerateDiagram(*changeset) == std::nullopt);
|
||||||
|
changeset.reset();
|
||||||
|
|
||||||
// Adding a grandchild makes the cluster size 3, which is uncalculable
|
// Adding a grandchild makes the cluster size 3, which is uncalculable
|
||||||
const auto tx3 = make_tx(/*inputs=*/ {tx2}, /*output_values=*/ {995 * CENT});
|
const auto tx5 = make_tx(/*inputs=*/ {tx2}, /*output_values=*/ {995 * CENT});
|
||||||
AddToMempool(pool, entry.Fee(normal_fee).FromTx(tx3));
|
AddToMempool(pool, entry.Fee(normal_fee).FromTx(tx5));
|
||||||
const auto res3 = ImprovesFeerateDiagram(pool, {entry1}, {entry1, entry2}, tx1_fee + tx2_fee + 1, tx1_size + tx2_size);
|
const auto entry5 = pool.GetIter(tx5->GetHash()).value();
|
||||||
|
|
||||||
|
changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(entry1);
|
||||||
|
changeset->StageRemoval(entry2);
|
||||||
|
changeset->StageRemoval(entry5);
|
||||||
|
changeset->StageAddition(tx1_conflict, tx1_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
changeset->StageAddition(entry4.GetSharedTx(), tx2_fee + entry5->GetModifiedFee() + 1, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
const auto res3 = ImprovesFeerateDiagram(*changeset);
|
||||||
BOOST_CHECK(res3.has_value());
|
BOOST_CHECK(res3.has_value());
|
||||||
BOOST_CHECK(res3.value().first == DiagramCheckError::UNCALCULABLE);
|
BOOST_CHECK(res3.value().first == DiagramCheckError::UNCALCULABLE);
|
||||||
BOOST_CHECK(res3.value().second == strprintf("%s has both ancestor and descendant, exceeding cluster limit of 2", tx2->GetHash().GetHex()));
|
BOOST_CHECK(res3.value().second == strprintf("%s has 2 ancestors, max 1 allowed", tx5->GetHash().GetHex()));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
BOOST_FIXTURE_TEST_CASE(calc_feerate_diagram_rbf, TestChain100Setup)
|
BOOST_FIXTURE_TEST_CASE(calc_feerate_diagram_rbf, TestChain100Setup)
|
||||||
|
@ -427,19 +462,28 @@ BOOST_FIXTURE_TEST_CASE(calc_feerate_diagram_rbf, TestChain100Setup)
|
||||||
const auto entry_low = pool.GetIter(low_tx->GetHash()).value();
|
const auto entry_low = pool.GetIter(low_tx->GetHash()).value();
|
||||||
const auto low_size = entry_low->GetTxSize();
|
const auto low_size = entry_low->GetTxSize();
|
||||||
|
|
||||||
|
const auto replacement_tx = make_tx(/*inputs=*/ {m_coinbase_txns[0]}, /*output_values=*/ {9 * COIN});
|
||||||
|
auto entry_replacement = entry.FromTx(replacement_tx);
|
||||||
|
|
||||||
// Replacement of size 1
|
// Replacement of size 1
|
||||||
{
|
{
|
||||||
const auto replace_one{pool.CalculateChunksForRBF(/*replacement_fees=*/0, /*replacement_vsize=*/1, {entry_low}, {entry_low})};
|
auto changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(entry_low);
|
||||||
|
changeset->StageAddition(replacement_tx, 0, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
const auto replace_one{changeset->CalculateChunksForRBF()};
|
||||||
BOOST_CHECK(replace_one.has_value());
|
BOOST_CHECK(replace_one.has_value());
|
||||||
std::vector<FeeFrac> expected_old_chunks{{low_fee, low_size}};
|
std::vector<FeeFrac> expected_old_chunks{{low_fee, low_size}};
|
||||||
BOOST_CHECK(replace_one->first == expected_old_chunks);
|
BOOST_CHECK(replace_one->first == expected_old_chunks);
|
||||||
std::vector<FeeFrac> expected_new_chunks{{0, 1}};
|
std::vector<FeeFrac> expected_new_chunks{{0, int32_t(entry_replacement.GetTxSize())}};
|
||||||
BOOST_CHECK(replace_one->second == expected_new_chunks);
|
BOOST_CHECK(replace_one->second == expected_new_chunks);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Non-zero replacement fee/size
|
// Non-zero replacement fee/size
|
||||||
{
|
{
|
||||||
const auto replace_one_fee{pool.CalculateChunksForRBF(/*replacement_fees=*/high_fee, /*replacement_vsize=*/low_size, {entry_low}, {entry_low})};
|
auto changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(entry_low);
|
||||||
|
changeset->StageAddition(replacement_tx, high_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
const auto replace_one_fee{changeset->CalculateChunksForRBF()};
|
||||||
BOOST_CHECK(replace_one_fee.has_value());
|
BOOST_CHECK(replace_one_fee.has_value());
|
||||||
std::vector<FeeFrac> expected_old_diagram{{low_fee, low_size}};
|
std::vector<FeeFrac> expected_old_diagram{{low_fee, low_size}};
|
||||||
BOOST_CHECK(replace_one_fee->first == expected_old_diagram);
|
BOOST_CHECK(replace_one_fee->first == expected_old_diagram);
|
||||||
|
@ -454,7 +498,11 @@ BOOST_FIXTURE_TEST_CASE(calc_feerate_diagram_rbf, TestChain100Setup)
|
||||||
const auto high_size = entry_high->GetTxSize();
|
const auto high_size = entry_high->GetTxSize();
|
||||||
|
|
||||||
{
|
{
|
||||||
const auto replace_single_chunk{pool.CalculateChunksForRBF(/*replacement_fees=*/high_fee, /*replacement_vsize=*/low_size, {entry_low}, {entry_low, entry_high})};
|
auto changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(entry_low);
|
||||||
|
changeset->StageRemoval(entry_high);
|
||||||
|
changeset->StageAddition(replacement_tx, high_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
const auto replace_single_chunk{changeset->CalculateChunksForRBF()};
|
||||||
BOOST_CHECK(replace_single_chunk.has_value());
|
BOOST_CHECK(replace_single_chunk.has_value());
|
||||||
std::vector<FeeFrac> expected_old_chunks{{low_fee + high_fee, low_size + high_size}};
|
std::vector<FeeFrac> expected_old_chunks{{low_fee + high_fee, low_size + high_size}};
|
||||||
BOOST_CHECK(replace_single_chunk->first == expected_old_chunks);
|
BOOST_CHECK(replace_single_chunk->first == expected_old_chunks);
|
||||||
|
@ -464,7 +512,10 @@ BOOST_FIXTURE_TEST_CASE(calc_feerate_diagram_rbf, TestChain100Setup)
|
||||||
|
|
||||||
// Conflict with the 2nd tx, resulting in new diagram with three entries
|
// Conflict with the 2nd tx, resulting in new diagram with three entries
|
||||||
{
|
{
|
||||||
const auto replace_cpfp_child{pool.CalculateChunksForRBF(/*replacement_fees=*/high_fee, /*replacement_vsize=*/low_size, {entry_high}, {entry_high})};
|
auto changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(entry_high);
|
||||||
|
changeset->StageAddition(replacement_tx, high_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
const auto replace_cpfp_child{changeset->CalculateChunksForRBF()};
|
||||||
BOOST_CHECK(replace_cpfp_child.has_value());
|
BOOST_CHECK(replace_cpfp_child.has_value());
|
||||||
std::vector<FeeFrac> expected_old_chunks{{low_fee + high_fee, low_size + high_size}};
|
std::vector<FeeFrac> expected_old_chunks{{low_fee + high_fee, low_size + high_size}};
|
||||||
BOOST_CHECK(replace_cpfp_child->first == expected_old_chunks);
|
BOOST_CHECK(replace_cpfp_child->first == expected_old_chunks);
|
||||||
|
@ -476,12 +527,16 @@ BOOST_FIXTURE_TEST_CASE(calc_feerate_diagram_rbf, TestChain100Setup)
|
||||||
const auto normal_tx = make_tx(/*inputs=*/ {high_tx}, /*output_values=*/ {995 * CENT});
|
const auto normal_tx = make_tx(/*inputs=*/ {high_tx}, /*output_values=*/ {995 * CENT});
|
||||||
AddToMempool(pool, entry.Fee(normal_fee).FromTx(normal_tx));
|
AddToMempool(pool, entry.Fee(normal_fee).FromTx(normal_tx));
|
||||||
const auto entry_normal = pool.GetIter(normal_tx->GetHash()).value();
|
const auto entry_normal = pool.GetIter(normal_tx->GetHash()).value();
|
||||||
const auto normal_size = entry_normal->GetTxSize();
|
|
||||||
|
|
||||||
{
|
{
|
||||||
const auto replace_too_large{pool.CalculateChunksForRBF(/*replacement_fees=*/normal_fee, /*replacement_vsize=*/normal_size, {entry_low}, {entry_low, entry_high, entry_normal})};
|
auto changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(entry_low);
|
||||||
|
changeset->StageRemoval(entry_high);
|
||||||
|
changeset->StageRemoval(entry_normal);
|
||||||
|
changeset->StageAddition(replacement_tx, high_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
const auto replace_too_large{changeset->CalculateChunksForRBF()};
|
||||||
BOOST_CHECK(!replace_too_large.has_value());
|
BOOST_CHECK(!replace_too_large.has_value());
|
||||||
BOOST_CHECK_EQUAL(util::ErrorString(replace_too_large).original, strprintf("%s has 2 descendants, max 1 allowed", low_tx->GetHash().GetHex()));
|
BOOST_CHECK_EQUAL(util::ErrorString(replace_too_large).original, strprintf("%s has 2 ancestors, max 1 allowed", normal_tx->GetHash().GetHex()));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make a size 2 cluster that is itself two chunks; evict both txns
|
// Make a size 2 cluster that is itself two chunks; evict both txns
|
||||||
|
@ -496,7 +551,11 @@ BOOST_FIXTURE_TEST_CASE(calc_feerate_diagram_rbf, TestChain100Setup)
|
||||||
const auto low_size_2 = entry_low_2->GetTxSize();
|
const auto low_size_2 = entry_low_2->GetTxSize();
|
||||||
|
|
||||||
{
|
{
|
||||||
const auto replace_two_chunks_single_cluster{pool.CalculateChunksForRBF(/*replacement_fees=*/high_fee, /*replacement_vsize=*/low_size, {entry_high_2}, {entry_high_2, entry_low_2})};
|
auto changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(entry_high_2);
|
||||||
|
changeset->StageRemoval(entry_low_2);
|
||||||
|
changeset->StageAddition(replacement_tx, high_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
const auto replace_two_chunks_single_cluster{changeset->CalculateChunksForRBF()};
|
||||||
BOOST_CHECK(replace_two_chunks_single_cluster.has_value());
|
BOOST_CHECK(replace_two_chunks_single_cluster.has_value());
|
||||||
std::vector<FeeFrac> expected_old_chunks{{high_fee, high_size_2}, {low_fee, low_size_2}};
|
std::vector<FeeFrac> expected_old_chunks{{high_fee, high_size_2}, {low_fee, low_size_2}};
|
||||||
BOOST_CHECK(replace_two_chunks_single_cluster->first == expected_old_chunks);
|
BOOST_CHECK(replace_two_chunks_single_cluster->first == expected_old_chunks);
|
||||||
|
@ -518,7 +577,12 @@ BOOST_FIXTURE_TEST_CASE(calc_feerate_diagram_rbf, TestChain100Setup)
|
||||||
const auto conflict_3_entry = pool.GetIter(conflict_3->GetHash()).value();
|
const auto conflict_3_entry = pool.GetIter(conflict_3->GetHash()).value();
|
||||||
|
|
||||||
{
|
{
|
||||||
const auto replace_multiple_clusters{pool.CalculateChunksForRBF(/*replacement_fees=*/high_fee, /*replacement_vsize=*/low_size, {conflict_1_entry, conflict_2_entry, conflict_3_entry}, {conflict_1_entry, conflict_2_entry, conflict_3_entry})};
|
auto changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(conflict_1_entry);
|
||||||
|
changeset->StageRemoval(conflict_2_entry);
|
||||||
|
changeset->StageRemoval(conflict_3_entry);
|
||||||
|
changeset->StageAddition(replacement_tx, high_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
const auto replace_multiple_clusters{changeset->CalculateChunksForRBF()};
|
||||||
BOOST_CHECK(replace_multiple_clusters.has_value());
|
BOOST_CHECK(replace_multiple_clusters.has_value());
|
||||||
BOOST_CHECK(replace_multiple_clusters->first.size() == 3);
|
BOOST_CHECK(replace_multiple_clusters->first.size() == 3);
|
||||||
BOOST_CHECK(replace_multiple_clusters->second.size() == 1);
|
BOOST_CHECK(replace_multiple_clusters->second.size() == 1);
|
||||||
|
@ -530,7 +594,13 @@ BOOST_FIXTURE_TEST_CASE(calc_feerate_diagram_rbf, TestChain100Setup)
|
||||||
const auto conflict_1_child_entry = pool.GetIter(conflict_1_child->GetHash()).value();
|
const auto conflict_1_child_entry = pool.GetIter(conflict_1_child->GetHash()).value();
|
||||||
|
|
||||||
{
|
{
|
||||||
const auto replace_multiple_clusters_2{pool.CalculateChunksForRBF(/*replacement_fees=*/high_fee, /*replacement_vsize=*/low_size, {conflict_1_entry, conflict_2_entry, conflict_3_entry}, {conflict_1_entry, conflict_2_entry, conflict_3_entry, conflict_1_child_entry})};
|
auto changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(conflict_1_entry);
|
||||||
|
changeset->StageRemoval(conflict_2_entry);
|
||||||
|
changeset->StageRemoval(conflict_3_entry);
|
||||||
|
changeset->StageRemoval(conflict_1_child_entry);
|
||||||
|
changeset->StageAddition(replacement_tx, high_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
const auto replace_multiple_clusters_2{changeset->CalculateChunksForRBF()};
|
||||||
|
|
||||||
BOOST_CHECK(replace_multiple_clusters_2.has_value());
|
BOOST_CHECK(replace_multiple_clusters_2.has_value());
|
||||||
BOOST_CHECK(replace_multiple_clusters_2->first.size() == 4);
|
BOOST_CHECK(replace_multiple_clusters_2->first.size() == 4);
|
||||||
|
@ -543,10 +613,17 @@ BOOST_FIXTURE_TEST_CASE(calc_feerate_diagram_rbf, TestChain100Setup)
|
||||||
const auto conflict_1_grand_child_entry = pool.GetIter(conflict_1_child->GetHash()).value();
|
const auto conflict_1_grand_child_entry = pool.GetIter(conflict_1_child->GetHash()).value();
|
||||||
|
|
||||||
{
|
{
|
||||||
const auto replace_cluster_size_3{pool.CalculateChunksForRBF(/*replacement_fees=*/high_fee, /*replacement_vsize=*/low_size, {conflict_1_entry, conflict_2_entry, conflict_3_entry}, {conflict_1_entry, conflict_2_entry, conflict_3_entry, conflict_1_child_entry, conflict_1_grand_child_entry})};
|
auto changeset = pool.GetChangeSet();
|
||||||
|
changeset->StageRemoval(conflict_1_entry);
|
||||||
|
changeset->StageRemoval(conflict_2_entry);
|
||||||
|
changeset->StageRemoval(conflict_3_entry);
|
||||||
|
changeset->StageRemoval(conflict_1_child_entry);
|
||||||
|
changeset->StageRemoval(conflict_1_grand_child_entry);
|
||||||
|
changeset->StageAddition(replacement_tx, high_fee, 0, 1, 0, false, 4, LockPoints());
|
||||||
|
const auto replace_cluster_size_3{changeset->CalculateChunksForRBF()};
|
||||||
|
|
||||||
BOOST_CHECK(!replace_cluster_size_3.has_value());
|
BOOST_CHECK(!replace_cluster_size_3.has_value());
|
||||||
BOOST_CHECK_EQUAL(util::ErrorString(replace_cluster_size_3).original, strprintf("%s has 2 descendants, max 1 allowed", conflict_1->GetHash().GetHex()));
|
BOOST_CHECK_EQUAL(util::ErrorString(replace_cluster_size_3).original, strprintf("%s has both ancestor and descendant, exceeding cluster limit of 2", conflict_1_child->GetHash().GetHex()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1312,11 +1312,15 @@ std::optional<std::string> CTxMemPool::CheckConflictTopology(const setEntries& d
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
|
||||||
util::Result<std::pair<std::vector<FeeFrac>, std::vector<FeeFrac>>> CTxMemPool::CalculateChunksForRBF(CAmount replacement_fees, int64_t replacement_vsize, const setEntries& direct_conflicts, const setEntries& all_conflicts)
|
util::Result<std::pair<std::vector<FeeFrac>, std::vector<FeeFrac>>> CTxMemPool::ChangeSet::CalculateChunksForRBF()
|
||||||
{
|
{
|
||||||
Assume(replacement_vsize > 0);
|
LOCK(m_pool->cs);
|
||||||
|
FeeFrac replacement_feerate{0, 0};
|
||||||
|
for (auto it : m_entry_vec) {
|
||||||
|
replacement_feerate += {it->GetModifiedFee(), it->GetTxSize()};
|
||||||
|
}
|
||||||
|
|
||||||
auto err_string{CheckConflictTopology(direct_conflicts)};
|
auto err_string{m_pool->CheckConflictTopology(m_to_remove)};
|
||||||
if (err_string.has_value()) {
|
if (err_string.has_value()) {
|
||||||
// Unsupported topology for calculating a feerate diagram
|
// Unsupported topology for calculating a feerate diagram
|
||||||
return util::Error{Untranslated(err_string.value())};
|
return util::Error{Untranslated(err_string.value())};
|
||||||
|
@ -1338,7 +1342,7 @@ util::Result<std::pair<std::vector<FeeFrac>, std::vector<FeeFrac>>> CTxMemPool::
|
||||||
// they have a strict topology of 1 or two connected transactions.
|
// they have a strict topology of 1 or two connected transactions.
|
||||||
|
|
||||||
// OLD: Compute existing chunks from all affected clusters
|
// OLD: Compute existing chunks from all affected clusters
|
||||||
for (auto txiter : all_conflicts) {
|
for (auto txiter : m_to_remove) {
|
||||||
// Does this transaction have descendants?
|
// Does this transaction have descendants?
|
||||||
if (txiter->GetCountWithDescendants() > 1) {
|
if (txiter->GetCountWithDescendants() > 1) {
|
||||||
// Consider this tx when we consider the descendant.
|
// Consider this tx when we consider the descendant.
|
||||||
|
@ -1378,13 +1382,13 @@ util::Result<std::pair<std::vector<FeeFrac>, std::vector<FeeFrac>>> CTxMemPool::
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// OLD - CON: Add any parents of direct conflicts that are not conflicted themselves
|
// OLD - CON: Add any parents of direct conflicts that are not conflicted themselves
|
||||||
for (auto direct_conflict : direct_conflicts) {
|
for (auto direct_conflict : m_to_remove) {
|
||||||
// If a direct conflict has an ancestor that is not in all_conflicts,
|
// If a direct conflict has an ancestor that is not in all_conflicts,
|
||||||
// it can be affected by the replacement of the child.
|
// it can be affected by the replacement of the child.
|
||||||
if (direct_conflict->GetMemPoolParentsConst().size() > 0) {
|
if (direct_conflict->GetMemPoolParentsConst().size() > 0) {
|
||||||
// Grab the parent.
|
// Grab the parent.
|
||||||
const CTxMemPoolEntry& parent = direct_conflict->GetMemPoolParentsConst().begin()->get();
|
const CTxMemPoolEntry& parent = direct_conflict->GetMemPoolParentsConst().begin()->get();
|
||||||
if (!all_conflicts.count(mapTx.iterator_to(parent))) {
|
if (!m_to_remove.contains(m_pool->mapTx.iterator_to(parent))) {
|
||||||
// This transaction would be left over, so add to the NEW
|
// This transaction would be left over, so add to the NEW
|
||||||
// diagram.
|
// diagram.
|
||||||
new_chunks.emplace_back(parent.GetModifiedFee(), parent.GetTxSize());
|
new_chunks.emplace_back(parent.GetModifiedFee(), parent.GetTxSize());
|
||||||
|
@ -1392,7 +1396,7 @@ util::Result<std::pair<std::vector<FeeFrac>, std::vector<FeeFrac>>> CTxMemPool::
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// + CNK: Add the proposed chunk itself
|
// + CNK: Add the proposed chunk itself
|
||||||
new_chunks.emplace_back(replacement_fees, int32_t(replacement_vsize));
|
new_chunks.emplace_back(replacement_feerate);
|
||||||
|
|
||||||
// No topology restrictions post-chunking; sort
|
// No topology restrictions post-chunking; sort
|
||||||
std::sort(new_chunks.begin(), new_chunks.end(), std::greater());
|
std::sort(new_chunks.begin(), new_chunks.end(), std::greater());
|
||||||
|
|
|
@ -706,23 +706,6 @@ public:
|
||||||
return m_sequence_number;
|
return m_sequence_number;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Calculate the sorted chunks for the old and new mempool relating to the
|
|
||||||
* clusters that would be affected by a potential replacement transaction.
|
|
||||||
* (replacement_fees, replacement_vsize) values are gathered from a
|
|
||||||
* proposed set of replacement transactions that are considered as a single
|
|
||||||
* chunk, and represent their complete cluster. In other words, they have no
|
|
||||||
* in-mempool ancestors.
|
|
||||||
*
|
|
||||||
* @param[in] replacement_fees Package fees
|
|
||||||
* @param[in] replacement_vsize Package size (must be greater than 0)
|
|
||||||
* @param[in] direct_conflicts All transactions that would be removed directly by
|
|
||||||
* having a conflicting input with a proposed transaction
|
|
||||||
* @param[in] all_conflicts All transactions that would be removed
|
|
||||||
* @return old and new diagram pair respectively, or an error string if the conflicts don't match a calculable topology
|
|
||||||
*/
|
|
||||||
util::Result<std::pair<std::vector<FeeFrac>, std::vector<FeeFrac>>> CalculateChunksForRBF(CAmount replacement_fees, int64_t replacement_vsize, const setEntries& direct_conflicts, const setEntries& all_conflicts) EXCLUSIVE_LOCKS_REQUIRED(cs);
|
|
||||||
|
|
||||||
/* Check that all direct conflicts are in a cluster size of two or less. Each
|
/* Check that all direct conflicts are in a cluster size of two or less. Each
|
||||||
* direct conflict may be in a separate cluster.
|
* direct conflict may be in a separate cluster.
|
||||||
*/
|
*/
|
||||||
|
@ -844,6 +827,14 @@ public:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Calculate the sorted chunks for the old and new mempool relating to the
|
||||||
|
* clusters that would be affected by a potential replacement transaction.
|
||||||
|
*
|
||||||
|
* @return old and new diagram pair respectively, or an error string if the conflicts don't match a calculable topology
|
||||||
|
*/
|
||||||
|
util::Result<std::pair<std::vector<FeeFrac>, std::vector<FeeFrac>>> CalculateChunksForRBF();
|
||||||
|
|
||||||
size_t GetTxCount() const { return m_entry_vec.size(); }
|
size_t GetTxCount() const { return m_entry_vec.size(); }
|
||||||
const CTransaction& GetAddedTxn(size_t index) const { return m_entry_vec.at(index)->GetTx(); }
|
const CTransaction& GetAddedTxn(size_t index) const { return m_entry_vec.at(index)->GetTx(); }
|
||||||
|
|
||||||
|
|
|
@ -1207,7 +1207,7 @@ bool MemPoolAccept::PackageMempoolChecks(const std::vector<CTransactionRef>& txn
|
||||||
|
|
||||||
// Check if it's economically rational to mine this package rather than the ones it replaces.
|
// Check if it's economically rational to mine this package rather than the ones it replaces.
|
||||||
// This takes the place of ReplacementChecks()'s PaysMoreThanConflicts() in the package RBF setting.
|
// This takes the place of ReplacementChecks()'s PaysMoreThanConflicts() in the package RBF setting.
|
||||||
if (const auto err_tup{ImprovesFeerateDiagram(m_pool, direct_conflict_iters, m_subpackage.m_all_conflicts, m_subpackage.m_total_modified_fees, m_subpackage.m_total_vsize)}) {
|
if (const auto err_tup{ImprovesFeerateDiagram(*m_subpackage.m_changeset)}) {
|
||||||
return package_state.Invalid(PackageValidationResult::PCKG_POLICY,
|
return package_state.Invalid(PackageValidationResult::PCKG_POLICY,
|
||||||
"package RBF failed: " + err_tup.value().second, "");
|
"package RBF failed: " + err_tup.value().second, "");
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue