[policy] check ancestor feerate in RBF, remove BIP125 Rule2 #23121

pull glozow wants to merge 7 commits into bitcoin:master from glozow:ancestorscore-remove-bip1252 changing 8 files +252 −60
  1. glozow commented at 5:32 pm on September 28, 2021: member

    When RBF was originally implemented in #6871, the mempool did not keep track of ancestor scores yet. It was pointed out that ancestor scores would be the most accurate metric, and suggested to temporarily “Require that any new inputs that show up in the replacing transaction be already confirmed. In the future, if we do merge something like ancestor package tracking and better mining code, we could update this…”

    #7594 added ancestor tracking to the mempool. It enables mempool validation and mining code to directly access a mempool entry’s ancestor score (total modified fees divided by total vsize for the transaction and all of its mempool ancestors). As such, we now have the ability to do what was desired originally.

    This PR does 2 things:

    (1) When evaluating a transaction for RBF, requires the replacement transaction to have a higher ancestor score than that of each original transaction. This boils down to “the replacement transaction is a better candidate for mining (by feerate) than all of the transactions it would replace.”

    (2) Removes enforcement of BIP125 Rule #2, allowing replacement transactions to spend unconfirmed inputs not spent by the original transactions.

  2. glozow commented at 5:33 pm on September 28, 2021: member

    Further motivation:

    (1) The restriction of only allowing confirmed UTXOs for funding a fee-bump doesn’t necessarily help our mempool validation logic, but hurts users trying to fee-bump their transactions. If the original transaction’s output value isn’t sufficient to fund a fee-bump and/or all of the user’s other UTXOs are unconfirmed, they might not be able to fee bump. Wallet developers also need to treat self-owned unconfirmed UTXOs as unusable for fee-bumping, which is an unnecessary complication.

    (2) BIP125#2 can also be bypassed: @jnewbery recently pointed out to me that an attacker can simply split a 1-input 1-output transaction off from the replacement transaction, then broadcast the transaction as is. This can always be done, and quite cheaply.

    To illustrate, Example L shows how BIP125#2 can be bypassed in all cases where the replacement transaction has enough fees to be split into multiple transactions. Example L1 is blocked by BIP125#2 because C is not allowed to spend an output of B. The owner of transaction C can bypass BIP125#2 as shown in example L2. Simply create a 1-input 1-output transaction, C*, that spends the output from B, and replace it with C.

    image

    This might be a good hack for people who need to RBF transactions and only have unconfirmed UTXOs available, like scenario M, where we want to replace A (a 100vB transaction paying 1000sats), but our only UTXOs are in unconfirmed transactions B and C. We can simply create D*, an intermediary transaction spending the outputs we want from B and C, and then replace that with D. This just means that D needs to pay some additional fees for replacing D* (which can be at the mempool minimum feerate).

    image

    However, Example N shows how this strategy can cause us to accept an replacement transaction that is actually less economical to mine than the original. Assume all transactions have a vsize of 100vB. A user wants to replace A, which has an ancestor score of 10sat/vB, with transaction C. Suppose they want to spend an unconfirmed output from transaction B, which has an ancestor score of 1sat/vB (maybe their wallet doesn’t have enough funds to provide a higher fee using only confirmed inputs). BIP125#2 prevents scenario N1, where the inclusion of another unconfirmed input means C has an ancestor score of 8sat/vB and thus less economical to mine than A. However, it does not prevent scenario M2, where the user splits off a 1-input 1-output transaction, C*, in order to be able to include the output from B. This causes us to incorrectly accept C (7.5sat/vB including its parent B) in favor of A (10sat/vB).

    image

    Again, credit to @jnewbery for this discovery.

    (3) I believe package RBF requires removal of BIP125#2, (explained here if you’re interested). I know this is not a super robust argument, but it would be a much better interface if the two sets of RBF rules are the same.

  3. DrahtBot added the label Build system on Sep 28, 2021
  4. DrahtBot added the label TX fees and policy on Sep 28, 2021
  5. DrahtBot added the label Validation on Sep 28, 2021
  6. glozow removed the label Build system on Sep 28, 2021
  7. DrahtBot commented at 6:05 pm on September 28, 2021: member

    The following sections might be updated with supplementary metadata relevant to reviewers and maintainers.

    Conflicts

    Reviewers, this pull request conflicts with the following ones:

    • #23711 (docs: RBF policy and mempool limit exemptions by glozow)
    • #22867 (test: Extend test coverage of BIP125 and document confusing/inconsistent behavior by mjdietzx)

    If you consider this pull request important, please also help to review the conflicting pull requests. Ideally, start with the one that should be merged first.

  8. ariard commented at 0:07 am on September 29, 2021: member

    @glozow @jnewbery If you think this discovery has safety implications, have you done a responsible disclosure of this defect not only to the maintainers of this project but also potentially affected ecosystem stakeholders e.g maintainers of other full-nodes implementations ?


    Good finding! IIUC, the bypass trick relies on extending the set of replaced transactions with a 1 input/1 output “bridge” transaction spending a targeted new unconfirmed intput. A new replacement candidate tries to replace the bridge and original (the setConflicts computed L620). As HasNoNewUnconfirmed considers the parent set of iters_conflicting any parent of the bridge and original are qualified of old inputs and thus allowed to be spend by the replacement candidate. I think the bridge transaction can be extended to N new unconfirmed inputs within package limits ? The bypass trick isn’t zero-cost as the Rule#4 penalty must still be paid.

    I think this Rule#2 bypass doesn’t present obvious safety implications for LN, even with “anchor output” channel type. If an adversary needs currently unconfirmed UTXOs to be used as fee-bumping ones to succeed an attack, she can waits to have them confirmed before to initiate any malicious operation. That’s an adversary “first-move” advantage anyway. In a pinch, unconfirmed UTXOs could be used to increase the rate of attacks launched sequentially if the UTXOs resources are limited…

  9. darosior commented at 6:59 am on September 29, 2021: member

    Concept ACK

    The disclosed fee decrease (good catch) is a breach against the fee-bumping protocol for Revault and really any other protocol that broadcasts ANYONECANPAY-signed inputs. Sure you can always re-RBF it but in the end we can’t assume honest node to engage in (and win) a race for fee-bumping.

  10. glozow commented at 2:16 pm on September 29, 2021: member

    If you think this discovery has safety implications, have you done a responsible disclosure of this defect not only to the maintainers of this project but also potentially affected ecosystem stakeholders e.g maintainers of other full-nodes implementations ? @ariard I didn’t think this is a safety issue for Bitcoin nodes, just a limitation that we no longer have. The “trick” isn’t completely cost-free and the replacement transaction still needs to be higher fees due to the other BIP125 rules, it’s just lower priority for mining than the mempool code thinks it is. Worst case scenario is if the original transaction was going to be in the next block and the replacement transaction isn’t. That’s unfortunate for the miner of this block since they might lose a bit on fees depending on the next-highest feerate packages, and not incentive-compatible behavior, so I think we should fix this.

    The disclosed fee decrease (good catch) is a breach against the fee-bumping protocol for Revault and really any other protocol that broadcasts ANYONECANPAY-signed inputs. Sure you can always re-RBF it but in the end we can’t assume honest node to engage in (and win) a race for fee-bumping. @darosior Attempting to summarize, please correct me if I’m misunderstanding: when a transaction has all inputs signed ANYONECANPAY and RBF opt-in, any attacker can create a replacement transaction by adding any input they want. Coupled with the strategy mentioned above, they can reduce the ancestor score of this transaction, thereby pinning it. You can re-RBF it, but might need to pay more fees. :scream:

  11. darosior commented at 2:27 pm on September 29, 2021: member

    Coupled with the strategy mentioned above, they can reduce the ancestor score of this transaction, thereby pinning it

    Yes.

    You can re-RBF it, but might need to pay more fees. scream

    Yes, and they can do that again and again (paying more fee at each iteration though so it’s a limiting factor). Since nodes will accept any of the low-feerate package or the higher-feerate one it’s a game of who will be able to connect to the miners’ nodes and maintain their version of the package: at this game we can expect an attacker running a custom software has a much higher success probability than a honest watchtower trying to feebump.

  12. glozow commented at 2:51 pm on September 29, 2021: member
    I don’t suppose it’s reasonable to recommend opting out of RBF for now? In any case, I think we need to add this ancestor score checking before we try to do full RBF, since it would leave all ANYONECANPAY-signed transactions vulnerable to this pinning attack.
  13. t-bast commented at 7:12 am on September 30, 2021: member
    Concept ACK
  14. ghost commented at 10:14 am on September 30, 2021: none

    @glozow @jnewbery If you think this discovery has safety implications, have you done a responsible disclosure of this defect not only to the maintainers of this project but also potentially affected ecosystem stakeholders e.g maintainers of other full-nodes implementations ?

    @ariard I didn’t think this is a safety issue for Bitcoin nodes, just slightly incentive-incompatible behavior. The trick isn’t completely cost-free and the replacement transaction still needs to be higher fees due to the other BIP125 rules, it’s just lower priority for mining than the mempool code thinks it is. Worst case scenario is if the original transaction was going to be in the next block and the replacement transaction isn’t. That’s unfortunate for the miner of this block since they might lose a bit on fees, and definitely incentive-incompatible mempool behavior, so I think we should fix this. @glozow @ariard even if this doesn’t involve any security issues we should inform others who are not following each pull request in this repository every day. This looks like an important bug which needs a fix. Although I am still trying to understand everything involved.

    And thanks for the diagrams, they are helpful.

  15. ariard commented at 5:29 pm on September 30, 2021: member

    @glozow

    For public records, before to open this PR, have you informed Bitcoin Core maintainers of your discovery yes or no ?


    I think by the past this project have always been careful about leveling the field in the miners block race (e.g ASICBOOST) and I think defects or bugs in our mempool code could be silently exploited by a miner to gain an advantage on the rest of the ecosystem. Of course, I understand to miss a potential safety impact on the mining ecosystem. Really, no one has a complete overview of the base-layer threats model, without even mentioning the layers above.

    A responsible disclosure in the Bitcoin ecosystem is a really complex or hard task. If you are not aware about it, I would invite you to read the informative pieces written by MIT DCI. Going further, as soon as we’re talking about security disclosures it’s a grey area. I believe it’s not a matter of who is wrong or right as an individual contributor. Instead, I believe the leading principle should be to follow a process minimizing the risks of harm inflicted to Bitcoin stakeholders. Where “harm” should be interpreted in the large sense as negative consequences, especially when those consequences are significant and unjust.

    Even if this software is released as an “experimental” digital currency, and that users should be fully-aware it’s a FOSS project and we don’t engage our responsibility as contributors, I think we should keep in mind that our acts could provoke irrecoverable disruptions in the life of real people, even indirectly or unintentionally.

    I think that’s why we should be careful. I hold the belief that communicating to a crypto-currency project maintainers a vulnerability is one piece of such harm-minimization process and helpful to fully assess the safety risks and what can be done to remedy. A lightweight bug could reveal itself as a serious vulnerability affecting far more users than the initial thoughts of the discoverer let it imagine. For sure, if you communicate a vulnerability to a project maintainers and they don’t reply back or take any action to patch a vulnerability, I believe it’s a good behavior to disclose it to the wider public, instead of letting propagate a false sense of security and safety-critical problems to worsen.

    That said, we all know we’re working in a highly-dynamic environment, that our resources are limited and that 100% of “security” certainty is a lure. And that’s okay to do mistakes, as long as we’re learning from then. From my experience working on Lightning, I can testify I’ve done few of them by lack of patience.

    I think it’s good to time to reflect on how we do security disclosures in this ecosystem, notably with having more and more funds backed by newer “L2” security models. I would say the “unseen” impact of this current breach on Revault is a good proof of that. Even if I think the project philosophy is fluctuating about what components are safety-critical and thus require special care, what we do when we find vulnerabilities send a strong signal towards downstream project maintainers if Bitcoin Core and the community around are reliable for their use-cases or not.

    Further, let’s remember, one of the most nasty bug in the recent years (CVE-2018-17144) have been discovered by a developer of another full-node implementation. Lacking considerations towards the safety and reliability of their projects is likely to damage their good-will and kindness to share back future vulnerabilities that they might discover first and also affecting us.

    IMHO, those are good reasons to express high-standards of behaviors among us, as otherwise we might as a group pay the indelicacies of one contributor.

    I think it’s important to have a safety-first mindset to contribute to Bitcoin protocols development.

  16. luke-jr commented at 6:36 pm on October 3, 2021: member
    1. Policy is a per-node and per-miner decision. What they choose should never be a security concern to anyone else - if it is, the “someone else” has the security flaw, not the node/miner imposing the policy. The “something else” in question should be fixed regardless of what policy nodes/miners choose.
    2. If policies were based on what is good for the network, Bitcoin Core wouldn’t make blocks over ~300k. Rather, unfortunately the policy here has been what certain devs expect to make miners the most short-term profit at the expense of the rest of the network and long-term.

    I don’t think @glozow did anything wrong by opening this PR here.

  17. glozow commented at 1:33 pm on October 4, 2021: member

    @glozow For public records, before to open this PR, have you informed Bitcoin Core maintainers of your discovery yes or no ?

    No, because this is not a security vulnerability or even a bug in Bitcoin Core. A disclosure that boils down to “if your transaction is signed with ANYONECANPAY and opts in to RBF, anyone can pay to RBF it. Our mempool evaluates its fees according to the rules specified in BIP125” is unnecessary.

    Thank you for the education on disclosing security issues delicately. If I find an issue that could cause mempool resource exhaustion, deviation from p2p protocol or consensus, or acceptance of uneconomical replacements, I’ll keep it in mind.

    As described in the OP, this PR implements a TODO item documented in the code. Code review and testing is welcome.

  18. ariard commented at 9:00 pm on October 4, 2021: member

    @luke-jr

    Sure, I think it’s one stand to recommend users or downstream projects to minimize assumptions on node policies, as it’s ultimately not (and can’t be) uniform on the p2p network. But I would say it’s another thinking when we find a defect or bug in our code behavior, of which correctness signaling is present (AFAICT bip125, including the rule 2, is present in doc/bips.md). Without going as far as safety implications, a mempool behavior bug might disrupt the availability of some bitcoin services providers or even alter their economic units, as this knowledge could constitute a competitive advantage. E.g being able to fee-bump your transactions thanks to unconfirmed inputs to deliver withdraws faster than your competitors.

    As your second point seems to indicate, policies design philosophy is a subject of discussions among devs. Advocating my position, I think we should aim to guarantee a high quality of service to upper layers and downstream projects, aligned with Bitcoin principles. And aiming to this end, promote good software engineering practices such as documenting our transaction relay behavior (see #22806).

    As the project philosophy is still fluctuating w.r.t if and how we should handle newer security models, I agree that @glozow isn’t wrong by opening this PR. @glozow

    No, because this is not a security vulnerability or even a bug in Bitcoin Core. A disclosure that boils down to “if your transaction is signed with ANYONECANPAY and opts in to RBF, anyone can pay to RBF it. Our mempool evaluates its fees according to the rules specified in BIP125” is unnecessary.

    Well, I think that qualifies as a breach of the fee-bumping protocol of any protocols that broadcasts ANYONECANPAY-signed inputs ? As of today, I would say we’re lucky it’s not deployed (yet) in production :)

    But in the coming years, if we have billions of funds secured by assumptions around mempool behavior, I think it’s better to have a more defined security process than failing in a really noisy ecosystem drama. Better safe than sorry :/

    If I find an issue that could cause mempool resource exhaustion, deviation from p2p protocol or consensus, or acceptance of uneconomical replacements, I’ll keep it in mind.

    Yeah, even if you have an established security model well in-mind, the public perception of how you handle potentially security-sensitive information is another thing. The question of “Oh shit we can lose money that way ?” can turn quickly as a not-that-clear one in Bitcoin. E.g, before the 2013 LevelDB bug (CVE-2013-3220), it wasn’t really in developers common knowledge that a database change could provoke a consensus failure. So sharing the information with project maintainers or other regular contributors is a way to protect your reputation in case of after-the-fact issues.

    Overall, thanks to John and you for this discovery, please find more of them and Concept ACK on this PR.

  19. meshcollider commented at 7:47 am on October 5, 2021: contributor

    User @prayank23 is having difficulty commenting on this PR for some reason, so I am relaying his comments/questions from IRC:

    Two questions for PR author:

    1. Unconfirmed UTXO are discouraged in Core. Does this PR change default behaviour? Context: https://github.com/bitcoin-core/gui/issues/242
    2. Why do you think it’s not a bug if attacker can confirm a tx with less fee rate by replacing? It can be exploited if we had on chain DeFi but I am not sure what all projects are currently impacted
  20. MarcoFalke removed the label Validation on Oct 5, 2021
  21. t-bast commented at 7:39 am on October 19, 2021: member

    ACK https://github.com/bitcoin/bitcoin/pull/23121/commits/cc5cbc93b58d41d99faacf88fcd4ca51509766e1

    I don’t have much knowledge of the subtleties of the mempool code so take my ACK lightly, but the code changes look simple enough and tests are looking good.

  22. unknown commented at 9:44 pm on October 21, 2021: none

    Approach NACK

    Reason:

    This PR does 2 things: (1) When evaluating a transaction for RBF, requires the replacement transaction to have a higher ancestor score than that of each original transaction. This boils down to “the replacement transaction is a better candidate for mining (by feerate) than all of the transactions it would replace.” (2) Removes enforcement of BIP125 Rule #2, allowing replacement transactions to spend unconfirmed inputs not spent by the original transactions.

    1 should be fixed which is a BUG as described in the initial few comments by PR author.

    2 should not be bundled with this fix and needs more discussion considering security of all projects that use Bitcoin Core

    My second question shared above by meshcollider was because of MEV related issues: https://bitcoin.stackexchange.com/questions/107787/front-running-in-bitcoin/


    This is my last review in any PR by this author for few reasons:

    Lies to justify blocking which prevented me from commenting earlier here. is:pr reviewed-by:prayank23 author:glozow shows 1 PR in which I asked one question and said nothing crazy.

    Mailing list post had no opinions: https://lists.linuxfoundation.org/pipermail/bitcoin-dev/2021-October/019504.html it was just to highlight the importance of this PR, get more people to review and make everyone aware about the issue in community. I had initially said the same thing here: #23121 (comment). Also this pull request is public, already mentioned in another email on mailing list by PR author: https://lists.linuxfoundation.org/pipermail/bitcoin-dev/2021-September/019496.html

  23. DrahtBot added the label Needs rebase on Oct 22, 2021
  24. glozow force-pushed on Oct 22, 2021
  25. glozow commented at 1:50 pm on October 22, 2021: member
    rebased and added documentation for our RBF policy + difference with BIP125
  26. DrahtBot removed the label Needs rebase on Oct 22, 2021
  27. DrahtBot added the label Needs rebase on Nov 9, 2021
  28. [validation] require replacement tx to have higher ancestor score 258f184a6a
  29. [unit test] for CheckAncestorScores 8afab375be
  30. [validation] gate CMPA relaxation by HasNoNewUnconfirmedInputs 4d45f5350b
  31. [policy] remove BIP125 Rule 2
    Now that we check the ancestor scores of replacement transactions, this
    rule is no longer needed. The HasNoNewUnconfirmedInputs check still
    needs to be done for CMPA relaxation in PreChecks.
    677e884aea
  32. [functional test] RBF with new unconfirmed input but low ancestor scores 8cc41100d3
  33. change HasNoNewUnconfirmed to return bool c5d0b3898b
  34. [doc] document RBF policy, note differences with BIP125 3847e41786
  35. glozow force-pushed on Nov 9, 2021
  36. glozow commented at 8:37 pm on November 9, 2021: member
    Rebased
  37. DrahtBot removed the label Needs rebase on Nov 9, 2021
  38. in src/test/rbf_tests.cpp:27 in 3847e41786
    22+    CMutableTransaction tx = CMutableTransaction();
    23+    tx.vin.resize(inputs.size());
    24+    tx.vout.resize(output_values.size());
    25+    for (size_t i = 0; i < inputs.size(); ++i) {
    26+        tx.vin[i].prevout.hash = inputs[i]->GetHash();
    27+        tx.vin[i].prevout.n = input_indices.size() > i ? input_indices[i] : 0;
    


    t-bast commented at 10:49 am on November 10, 2021:

    nit:

    0        tx.vin[i].prevout.n = input_indices[i];
    

    I’m not sure it’s worth having this mechanism of fallback to 0 but only for the last inputs, it’s a bit confusing. Maybe just making input_indices completely optional (in which case you’d use 0 for all inputs) is better? And if it’s provided then it should be provided for all inputs. In that case it would be:

    0        tx.vin[i].prevout.n = input_indices.size() > 0 ? input_indices[i] : 0;
    

    Feel free to ignore though if you disagree.

  39. in src/validation.cpp:766 in 3847e41786
    762@@ -763,7 +763,7 @@ bool MemPoolAccept::PreChecks(ATMPArgs& args, Workspace& ws)
    763 
    764     ws.m_iters_conflicting = m_pool.GetIterSet(ws.m_conflicts);
    765     // Calculate in-mempool ancestors, up to a limit.
    766-    if (ws.m_conflicts.size() == 1) {
    767+    if (ws.m_conflicts.size() == 1 && HasNoNewUnconfirmed(tx, m_pool, ws.m_iters_conflicting)) {
    


    t-bast commented at 10:55 am on November 10, 2021:
    Would it be worth documenting the CPFP carve-out rule (in the doc folder, similarly to RBF)? Especially now that it does require no new unconfirmed inputs which RBF doesn’t? It can be done in a separate PR though.

    glozow commented at 8:56 am on December 9, 2021:
    Sorry it took me so long to get to this - added in #23711
  40. DrahtBot added the label Needs rebase on Dec 20, 2021
  41. DrahtBot commented at 9:43 am on December 20, 2021: member

    🐙 This pull request conflicts with the target branch and needs rebase.

    Want to unsubscribe from rebase notifications on this pull request? Just convert this pull request to a “draft”.

  42. in src/policy/rbf.cpp:171 in 258f184a6a outdated
    166@@ -174,3 +167,35 @@ std::optional<std::string> PaysForRBF(CAmount original_fees,
    167     }
    168     return std::nullopt;
    169 }
    170+
    171+std::optional<std::string> CheckAncestorScores(CAmount replacement_fees,
    


    sdaftuar commented at 7:04 pm on January 1, 2022:
    In this function, we should be thinking about the “ancestor score” of a transaction as the minimum of its ancestor feerate and its own feerate. A transaction should never sort higher in the ancestor scoring comparison than its own feerate, because the parent transactions can be included without the child.
  43. in src/policy/rbf.cpp:185 in 258f184a6a outdated
    180+    for (CTxMemPool::txiter it : ancestors) {
    181+        replacement_fees += it->GetModifiedFee();
    182+        replacement_vsize += it->GetTxSize();
    183+    }
    184+    const CFeeRate replacement_ancestor_score(replacement_fees, replacement_vsize);
    185+    for (CTxMemPool::txiter it : all_conflicts) {
    


    sdaftuar commented at 7:12 pm on January 1, 2022:

    If we want to fix the issue where a replacement transaction might have a worse mining score than some transactions it replaces, then I think the conservative thing to do would be to compare the ancestor score of the replacement transaction with the actual feerates of the conflicted transactions themselves (ie not the ancestor feerate of the conflicted transactions).

    The rationale is that you never know if some sibling transaction might exist in the mempool that would cause the ancestors of a target transaction to be mined, allowing the target transaction to be considered using its own feerate (rather than being weighed down by low feerate parents).

    This may be overly conservative; I think the real question that we’re trying to answer with the feerate checks is something like “are the fees expected to be paid in the next (N?) blocks higher or lower if we process this transaction”, which is not very easy to answer with the tools we have at our disposal. If we’re okay being conservative here for the use cases that we’re trying to support, then I think comparing ancestor score to actual feerates would be an easy fix.

  44. sdaftuar commented at 7:23 pm on January 1, 2022: member
    The feerate comparison that we do in PaysMoreThanConflicts (comparing the feerate of the new transaction with the feerates of the transactions that it directly conflicts) was put in place as a proxy for enforcing that the the mining incentives for the new transaction should be at least as good as the ones it was replacing, back when our block construction logic was simpler and didn’t take ancestor scores into account. I think we could just update that function to compare the ancestor score of the new transaction (that is, min(feerate, feerate-with-ancestors) for the new transaction) to the individual feerates of all the transactions to be replaced, requiring that the new transaction’s ancestor score is higher. This would be a conservative way of ensuring that the new transaction is more likely to be mined than any of the transactions that would be removed.
  45. glozow commented at 12:38 pm on January 4, 2022: member

    Thanks for the review @sdaftuar!

    If we want to fix the issue where a replacement transaction might have a worse mining score than some transactions it replaces, then I think the conservative thing to do would be to compare the ancestor score of the replacement transaction with the actual feerates of the conflicted transactions themselves (ie not the ancestor feerate of the conflicted transactions).

    The rationale is that you never know if some sibling transaction might exist in the mempool that would cause the ancestors of a target transaction to be mined, allowing the target transaction to be considered using its own feerate (rather than being weighed down by low feerate parents).

    My understanding is: When a future block template is built, a transaction’s mining score may include none, some, or all of its current mempool ancestors, e.g. because of a high feerate sibling. To ensure that our replacement transaction is not a worse mining candidate than the conflicting transactions in a conservative manner, we’ll require worst-case replacement tx feerate >= best-case conflicting tx feerate for every conflicting tx (best-case meaning highest feerate, and worst-case meaning lowest feerate).

    ~Since individual feerate always >= ancestor feerate for any tx, this means we require replacement tx ancestor feerate >= conflicting tx individual feerate for every conflicting transaction.~

    Edit: It’s not true that individual feerate always >= ancestor feerate. We’d require replacement tx min(ancestor feerate, individual feerate) > conflicting tx max(ancestor feerate, individual feerate)

    To clarify, we would need to do this comparison with every conflicting transaction and all their descendants (not just direct conflicts), since they may have high-feerate descendants? Or do we only consider directly conflicting transactions, and perhaps the other RBF rules are sufficient to ensure we are being incentive-compatible (based on this comment)?

    If we compare with all conflicts, this could be too conservative, at least for package RBF. For example:

    Alice (adversary) and Bob (honest) have a LN channel, where Alice has commitment tx A1, Bob has commitment tx B1.

    Alice broadcasts A1 <- A2 <- A3, where A2 is a very large child of A1, and A3 is a very small child of A2. A2 has a low feerate and A3 has an extremely high feerate. This package, as a whole, has a low ancestor feerate.

    Bob broadcasts a package, B1 <- B2, where B2 is a fee-bumping child of B1. Perhaps Bob’s package is higher feerate than Alice’s package, but since B2’s ancestor feerate must be higher than A3’s individual feerate, Bob’s package is rejected.

    ===

    I think the real question that we’re trying to answer with the feerate checks is something like “are the fees expected to be paid in the next (N?) blocks higher or lower if we process this transaction”, which is not very easy to answer with the tools we have at our disposal.

    I suppose if N=1, we can just compare ancestor feerates? Only when N>1, the transaction’s future mining score may be different from its current ancestor feerate.

  46. sdaftuar commented at 1:55 pm on January 4, 2022: member

    <snip summary of block construction and feerate scores, which all sounds correct to me>

    To clarify, we would need to do this comparison with every conflicting transaction and all their descendants (not just direct conflicts), since they may have high-feerate descendants? Or do we only consider directly conflicting transactions, and perhaps the other RBF rules are sufficient to ensure we are being incentive-compatible (based on this comment)?

    So if we only compare ancestor feerates to the direct conflicts’ actual feerates, then we might very well replace a transaction that would be included in the next block:

    Consider a set of transactions: A1 <- A2 where A2 is a child of A1, and suppose that A1 is small and very low fee, and A2 is small and very high fee. We can do this so that (A1, A2) would be a candidate for inclusion in the next block.

    Then construct a transaction B that conflicts with A1, and is much larger than A1 and A2 and pays a low feerate (and greater total fee than A1+A2). It could satisfy the RBF total fee requirement and the requirement that its ancestor feerate exceed A1’s actual feerate, without being a candidate for inclusion in any blocks in the near future, even though (A1, A2) might already be in our set of transactions to include in the next block.

    So that seems like it wouldn’t be incentive compatible.

    If we compare with all conflicts, this could be too conservative, at least for package RBF. For example:

    Alice (adversary) and Bob (honest) have a LN channel, where Alice has commitment tx A1, Bob has commitment tx B1.

    Alice broadcasts A1 <- A2 <- A3, where A2 is a very large child of A1, and A3 is a very small child of A2. A2 has a low feerate and A3 has an extremely high feerate. This package, as a whole, has a low ancestor feerate.

    Bob broadcasts a package, B1 <- B2, where B2 is a fee-bumping child of B1. Perhaps Bob’s package is higher feerate than Alice’s package, but since B2’s ancestor feerate must be higher than A3’s individual feerate, Bob’s package is rejected.

    Thanks for bringing up a concrete example to consider. (I’m just taking it as an assumption that any other issues around package RBF have already been solved – I haven’t thought it all through in a while.). In this example, since A3 is very small and B2+B1 already have pay enough total fee to exceed A1+A2+A3, isn’t it a pretty small economic difference to further require that B2 have an ancestor feerate greater than A3’s actual feerate?

    I think the underlying issue is that we may already want to be relaxing the total fee requirement somehow, because transaction pinning with our current RBF rules – ie, requiring that B1+B2 actually pay more total fee than A1+A2+A3, even though that package is low value to miners – has been a problem for some time, and I think people would like a solution to that. However it’s a bit tough to consider what the RBF feerate requirements should be in the absence of a concrete proposal of what we might do differently for the total fees as well. (If we relaxed both, then I think it’d be easy to construct examples where miners are reducing their fees in the next block, which wouldn’t be incentive compatible.)

    From a practical perspective, it seems to me like we could “solve” the issues around incentive compatibility with our current RBF rules (along the lines of what I described above of requiring ancestor feerates to be greater than actual feerates of all transactions which would be replaced), and then later address the pinning problems. However, I don’t know if this is actually good for the ecosystem as a whole, since other Bitcoin transaction creators might care a lot about the precise relay behavior here, so making multiple changes – and particularly, making transaction replacement worse somehow in the interim by becoming more conservative – might be pretty unwelcome. So maybe it would be better to craft a new proposal that addresses both problems, try to solicit feedback about whether it would solve the issues people are running into, and then fix both?

  47. glozow commented at 11:23 am on January 7, 2022: member

    From a practical perspective, it seems to me like we could “solve” the issues around incentive compatibility with our current RBF rules (along the lines of what I described above of requiring ancestor feerates to be greater than actual feerates of all transactions which would be replaced), and then later address the pinning problems. However, I don’t know if this is actually good for the ecosystem as a whole, since other Bitcoin transaction creators might care a lot about the precise relay behavior here, so making multiple changes – and particularly, making transaction replacement worse somehow in the interim by becoming more conservative – might be pretty unwelcome. So maybe it would be better to craft a new proposal that addresses both problems, try to solicit feedback about whether it would solve the issues people are running into, and then fix both?

    I agree it’s not good to try to think about the multiple fee/feerate rules in isolation. I’ve also heard about issues with the current absolute fee rule, and definitely don’t want to make things worse. Thanks a lot for your feedback! I’ll come back with a new RBF proposal to address both issues.

  48. glozow closed this on Jan 7, 2022

  49. DrahtBot locked this on Jan 11, 2023

github-metadata-mirror

This is a metadata mirror of the GitHub repository bitcoin/bitcoin. This site is not affiliated with GitHub. Content is generated from a GitHub metadata backup.
generated: 2024-11-23 06:12 UTC

This site is hosted by @0xB10C
More mirrored repositories can be found on mirror.b10c.me