Keep track of recently rejected transactions #6450

pull morcos wants to merge 1 commits into bitcoin:master from morcos:recentRejects changing 2 files +29 −7
  1. morcos commented at 8:56 pm on July 16, 2015: member

    Nodes can have divergent policies on which transactions they will accept and relay. This can cause you to repeatedly request and reject the same tx after its inved to you from various peers which have accepted it.

    This commit puts a 15 min timer on rerequesting the same tx hash if you just rejected it. I think the proper value for this can be debated, it might be almost as effective with a significantly smaller timer like 1 min.

  2. Keep track of recently rejected transactions
    Nodes can have divergent policies on which transactions they will accept and relay.  This can cause you to repeatedly request and reject the same tx after its inved to you from various peers which have accepted it.  This commit puts a 15 min timer on rerequesting the same tx hash if you just rejected it.
    d7b2800e0d
  3. petertodd commented at 6:56 am on July 17, 2015: contributor

    I’ve noticed this issue too on my full-RBF nodes, doubly so on the full-RBF nodes that have a higher min-relay fee set. In some cases that resulted in huge amounts of bandwidth being wasted, as those nodes are otherwise well connected with nearly the full number of peers. So strong ACK on the intent.

    I think we could make the table quite a bit bigger though. Each entry is about 40 bytes, so 2000 replacements is only about 80k. Equally, for the purpose of rejecting txs it’d be perfectly acceptable to only store 64bit partial txids if we first hash them with a per-node nonce, so that gets us 16bytes/entry.

    If someone is trying to do a bandwidth attack by flooding the network, right now the re-requests are a pretty big multiplier - if half your ~100 incoming peers relay a tx and half don’t, you’ve increased your incoming bandwidth used by the flood by about 50x. With the patch that’s reduced drastically.

    Now, to defeat this protection you basically need to relay enough txs to overflow the limitedmap, before all your peers send you the associated INVs. (modulo cases where the attacker is directy connected to you) The minimum tx size is about 60 bytes, so at 1MB/second you might receive 16k txs in a really extreme case - but that overflows the map multiple times over.

    Spending more like, say, 5MB of RAM to mitigate that kind of bandwidth attack seems reasonable to me. w/ the existing design, that’s a 125,000 entry limitedmap; if you implement the 64-bit hash idea that’s a 312,000 entry limited map.

    Also, would the attack be harder to pull off if dropping entries from the map was probabalistic? I suspect yes.

  4. petertodd commented at 7:30 am on July 17, 2015: contributor

    One more idea: How about discarding the entire recent rejects map every time the best block hash changes? I suspect the most likely scenario where a previously rejected transaction is now again valid will probably be validity changes due to new/reorged blocks. For instance the main reason why my #6216 pull-req causes no issues w/ relaying is that if a node receives a transaction that it rejects due to the tx being non-final, it will be accepted again later as it winds its way around the network; this pull-req might break that behavior.

    If you do this it’d probably be fine to remove the time expiration entirely, saving RAM, and implement the map with a simple set with txids dropped from it randomly.

  5. petertodd commented at 7:32 am on July 17, 2015: contributor
    (or for that matter, a simple bloom filter with a bit randomly cleared on each insertion to keep the fp rate steady)
  6. in src/main.cpp: in d7b2800e0d
    4302+                mapRecentRejects.insert(make_pair(tx.GetHash(), GetTime()));
    4303+            if (pfrom->fWhitelisted) {
    4304+                // Always relay transactions received from whitelisted peers, even
    4305+                // if they are already in the mempool (allowing the node to function
    4306+                // as a gateway for nodes hidden behind it).
    4307+                RelayTransaction(tx);
    


    petertodd commented at 9:08 am on July 17, 2015:
    Actually, this will relay invalid transactions from whitelisted peers as well.
  7. petertodd commented at 10:56 am on July 17, 2015: contributor
    Created #6452 with a bloom filter instead of limitedmap. Thoughts?
  8. morcos commented at 12:09 pm on July 17, 2015: member
    @ajweiss had suggested a bloom filter when I created this, but I was concerned about false positives. I hadn’t thought it through that much. I’m not tied to a specific approach. I was looking at this less as a DoS attack prevention and more as a response to the unfortunate p2p feedback loop of different acceptance policies, but I guess you’re right its important to make sure it still works under load. I do like the idea of clearing when a block comes in.
  9. petertodd commented at 12:15 pm on July 17, 2015: contributor
    Yeah, it’s quite cheap to get extremely low fp rates - my bloom filter approach has a one in a million FP rate. The (bloom filter) code also already exists and is well tested, as the netaddr stuff uses it.
  10. laanwj added the label P2P on Jul 17, 2015
  11. laanwj commented at 8:08 am on July 18, 2015: member
    Concept ACK. Using a bloom filter is slightly more risky, but if we make sure that at least every node uses an (unpredictable) tweak the impact of false positives should be minimized.
  12. morcos commented at 9:25 pm on July 19, 2015: member
    Closing in favor of #6452
  13. morcos closed this on Jul 19, 2015

  14. DrahtBot locked this on Sep 8, 2021

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-12-24 09:12 UTC

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