net.cpp
complexity and adds documentation about design goals about different connection types.
[net] Cleanup connection types- followups #19724
pull amitiuttarwar wants to merge 10 commits into bitcoin:master from amitiuttarwar:2020-08-conn-refactor-followups changing 8 files +139 −75-
amitiuttarwar commented at 10:23 pm on August 14, 2020: contributorThis PR addresses outstanding review comments from #19316. It further simplifies
-
amitiuttarwar commented at 10:32 pm on August 14, 2020: contributor
I’m not 100% convinced about replacing
IsAddrRelayPeer()
withRelayAddrsWithConn()
9345f9c9d47a88987606d1421823914a4c48dbf9.Pros:
- It is more direct to check the connection type instead of a (missing) data structure based on the connection type.
- Easy to update logic in the future
- Less mental overhead, less code touch points to glean intention.
Cons:
- We are now checking the same conditional in two places- [1] & [2], and need to manually ensure they do not fall out of sync.
Curious to hear what other reviewers think is preferable.
-
amitiuttarwar renamed this:
[net] Cleanup connection type- followups
[net] Cleanup connection types- followups
on Aug 14, 2020 -
DrahtBot added the label P2P on Aug 14, 2020
-
DrahtBot added the label Tests on Aug 14, 2020
-
amitiuttarwar force-pushed on Aug 14, 2020
-
sipa commented at 11:53 pm on August 14, 2020: member@amitiuttarwar I think it makes sense to make the addr-relay decision depend on the connection type. That should be the most authoritative information about the connection. Sure, it’s duplication, but if we accidentally forget to create the necessary data structures, test will fail; if we accidentally create them if they’re not needed, at worst we’ve wasted a bit of memory we’re already ok with for other nodes.
-
MarcoFalke removed the label Tests on Aug 15, 2020
-
in src/net.cpp:1844 in b672f8d84f outdated
1840@@ -1841,11 +1841,11 @@ void CConnman::ThreadOpenConnections(const std::vector<std::string> connect) 1841 // but inbound and manual peers do not use our outbound slots. Inbound peers 1842 // also have the added issue that they could be attacker controlled and used 1843 // to prevent us from connecting to particular hosts if we used them here. 1844- switch(pnode->m_conn_type){ 1845+ switch (pnode->m_conn_type){
MarcoFalke commented at 7:38 am on August 15, 2020:style-nit:
0 switch (pnode->m_conn_type) {
if you decide to do style-fixups, it would probably be good to do all of them in one go or not at all. Maybe with clang-format-diff? https://github.com/bitcoin/bitcoin/tree/master/contrib/devtools#clang-format-diffpy
amitiuttarwar commented at 3:30 am on August 21, 2020:heh. thanks for the nudge. I’ve updated my tooling so I’ll pay better attention to the clang-format-diff moving forward. I think I’ve now cleaned everything up properly in this PR? I’m not sure what exactly you mean by “all in one go or not at all”…
(disclaimer: I didn’t take the suggestions to not IndentCaseBlocks. I would much prefer if we turned that format option to true.)
in src/net_processing.cpp:2436 in b672f8d84f outdated
2432@@ -2433,9 +2433,8 @@ void ProcessMessage( 2433 UpdatePreferredDownload(pfrom, State(pfrom.GetId())); 2434 } 2435 2436- if (!pfrom.IsInboundConn() && pfrom.IsAddrRelayPeer()) 2437+ if (pfrom.AdvertiseAddressToConn())
jnewbery commented at 10:16 am on August 15, 2020:If we’re cleaning this up, we should really clean it up and move the
connman.MarkAddressGood()
out of this if statement, since it’s not related to address advertising at all. Having it in the same conditional is a trap for developers to fall into.(specifically, a developer might think “we don’t need to advertise our address to FEELER connections, since we disconnect before we actually send the ADDR or GETADDR messages, so we can remove FEELER from AdvertiseAddressToConn”, but doing that breaks our FEELER connection logic, because
MarkAddressGood()
needs to be called to mark the address as good)cc @sdaftuar
sdaftuar commented at 6:08 pm on August 15, 2020:I was looking at this more and I believe we should leave the MarkAddressGood call in the if-block, though it needs an explanation— the reason (I think) is that we don’t want block-relay connections to leave a trace in the addrman that could be leaked by an attacker sniffing our addr responses.
So this just needs a comment explaining why it’s in here.
jnewbery commented at 9:38 am on August 16, 2020:Can you explain this a bit? I think you’re saying that if we connect out to a peer that we use for block-relay-only we don’t want to mark it as good (essentially move it from New to Tried and update the last successful connect time) because then another peer might be able to tell that we’ve connected to them?
My understanding of how we respond to
getaddr
messages is that we’ll respond with (up to) 1000 records taken at random from all records, regardless of whether they’re in New or Tried, so marking a peer as good won’t change whether we include it inaddr
responses.
sdaftuar commented at 4:29 pm on August 16, 2020:I don’t have a great understanding of the details of how addrman works or how we interact with it – so possibly the fear was unfounded. It used to be the case that you could trivially download a listening node’s addrman though, and it also used to be the case (perhaps only long ago?) that we’d leak information about current connections based on time stamps stored in our addrman, so when I worked on #15759 I tried to leave the addrman state unchanged when making block-relay-only connections.
I’m not sure that I fully succeeded in not leaking information, or that the concern is correct to begin with given how addrman already works. So it’s possible that with some analysis we could make improvements here. Maybe not appropriate for this pr though…
jnewbery commented at 10:42 am on August 18, 2020:I believe there are three ways that a peer’s record is updated in addrman when we’re connected to them:
- when we receive a
version
message from them, we’ll mark their address good. That moves the address from the new table to the tried table, and updates thenLastSuccess
/nLastTry
/nAttempts
fields. We only do this if the peer isn’t inbound and isn’t block-relay-only. I don’t believe this change can easily be detected by a peer sending usgetaddr
messages. - if a peer self-advertises to us, we’ll call
AddAddressKnown()
, which could update thenTime
, which is included in the address record that we send to a peer in response to agetaddr
request. - when we disconnect from the peer, we’ll update the
nTime
, as long as the peer was marked asfCurrentlyConnected
. That happens when we receive averack
from a peer which is not inbound. This doesn’t include feeler peers, since we disconnect from them as soon as we receive aversion
message from them. ThenTime
is only updated during disconnection so spies can’t find out who we’re currently connected to (https://github.com/bitcoin/bitcoin/pull/5860#issuecomment-78397140).
In summary, I think block-relay-only peers should have their address marked good when we connect to them (1). If we don’t, then we’ll update the
nLastTry
andnAttempts
fields when we attempt to connect to them, but not update thenLastSuccess
.The reason it’s important in this PR is that placing
MarkAddressGood()
behind a conditionalif (AdvertiseAddressToConn())
is very confusing for anyone reading the code, and makes it easy to introduce subtle bugs.
amitiuttarwar commented at 3:24 am on August 21, 2020:I agree that having
MarkAddressGood
in this conditional is surprising & that changing toAdvertiseAddressToConn
augments the confusion.For this PR- I’ve updated the conditional to be
!pfrom.IsInboundConn() && !pfrom.IsBlockOnlyConn()
(aka removed the misleading function), and added a comment.For future work- I’ve noted this down. I’ve taken an initial look and @jnewbery I believe what you’re saying makes sense, but I’ll have to spend more time with it to fully convince myself. The fact that I missed this behavior in my initial pass affirms that its easy to miss or confuse. I’d like to either (1) understand why it should be in this conditional & comment or (2) understand why it shouldn’t be & change it. But I’m leaving this for another PR :)
ariard commented at 11:32 pm on August 22, 2020:I agree with John that block-relay-only peers should have their address marked as good when we receive their
version
. To answergetaddr
we fetch randomly a list of address from bothvvNew/vvTried
without record bias as far as a quick read ofCAddrMan::GetAddr_
let it think. So moving address from a table to another doesn’t change the ability of an attacker to learn about a block-relay-only address.Maybe we could improve block-relay-only peers unobservability in future works around AddrMan by filtering out
block-relay-only
address fromgetaddr
replies, without introducing an oracle for the original sender of the used address ? @naumenkogs
sdaftuar commented at 5:56 pm on August 25, 2020:My instinct is that it’d be best if the addrman were completely unchanged as a result of block-relay connections, at least while we are connected to the peer. So if there are places where block-relay connections cause some kind of update in addrman, then those should be bugs that we fix.
Here’s my thinking: (a) it’s not clear to me that there’s not some way to sniff out things in the tried table from things in the new table. For instance, I think the new table tends to be much bigger than the tried table, yet I think when we respond to getaddr requests, we sample 50% from the tried table and 50% from the new table. This creates a bias that, if you had the ability to read someone’s addrman, might leak information about who someone might be connected to.
(b) Even if we have limited the ability to read someone’s addrman today (eg with fixes like #18991), I think it’s helpful to separate the policies around addrman from being too tightly interwoven into other features like block-relay peers. The easiest way to reason about this stuff is to say that block-relay peers have no effect on addrman, and then we know that any changes in addrman / addr relay can’t possibly leak information about our block-relay peers.
The only potential hesitation I have advancing this point of view is if there are side effects from not updating addrman that might be detrimental. For instance, it does seem like it would be terrible if an existing block-relay peer that we are successfully peered with for some time could be evicted from the addrman altogether, and then we forget about it. But maybe it’s enough to either update addrman at the time of disconnection, or save the peer as is suggested in #17428?
At any rate, this discussion over the best behavior for the interaction between block-relay peers and addrman deserves its own issue. I do agree with the current version of the proposed code which checks for the peer being a block-relay peer.
naumenkogs commented at 12:20 pm on September 1, 2020:I think we’re taking the wrong approach in the PR as it is right now. At least that condition should be split in 2 blocks:
I think the current code change here is not useful, but only distracts: I) The comment
Advertise our address
is removed for no reason II) The comment(unrelated to advertising our address)
is very confusing. I managed to understand it only after reading this discussion III) The commentMoves address from New to Tried table in Addrman
is not sufficient IV) The change ofIsAddrRelayPeer
toIsBlockOnlyConn
here is very confusing, the two blocks should be split into 2 very separate blocks according to the following reasoning:- We self-announce to every peer that supports self-announcement AND it’s a good idea: 1a) not useful to self-announce to feelers as we won’t actually send ADDR to them (i think?) 1b) bad to self-announce to block-relay-only (maybe? wrt privacy)
- We MarkAddressGood according to the logic we have
naumenkogs commented at 12:23 pm on September 1, 2020:w.r.t block-relay-only peers and AddrMan, I agree it’s a separate issue, so i’d prefer to preserve the approach status quo here.
The current PR currently satisfies it, but I’m still unhappy with the code changes in this particular area (see the previous comment).
sdaftuar commented at 5:39 pm on September 1, 2020:@naumenkogs Would it be enough to just add some more comments to this block? I think it’s fine to leave as one block until we actually need to split it, so I’d propose something like this:
0diff --git a/src/net_processing.cpp b/src/net_processing.cpp 1index 7cd4f7c4cb5..4285e8ba898 100644 2--- a/src/net_processing.cpp 3+++ b/src/net_processing.cpp 4@@ -2427,6 +2427,21 @@ void PeerLogicValidation::ProcessMessage(CNode& pfrom, const std::string& msg_ty 5 } 6 7 if (!pfrom.IsInboundConn() && !pfrom.IsBlockOnlyConn()) { 8+ // For outbound peers, we try to relay our address (so that other 9+ // nodes can try to find us more quickly, as we have no guarantee 10+ // that an outbound peer is even aware of how to reach us) and do a 11+ // one-time address fetch (to help populate/update our addrman). If 12+ // we're starting up for the first time, our addrman may be pretty 13+ // empty and no one will know who we are. 14+ // 15+ // We also will update the addrman to record connection success for 16+ // these peers (which include OUTBOUND_FULL_RELAY peers and FEELER 17+ // connections) so that addrman will have an up-to-date notion of 18+ // which peers are online and available. 19+ // 20+ // We skip these operations for BLOCK_RELAY peers to avoid 21+ // potentially leaking information about our BLOCK_RELAY 22+ // connections via the addrman or address relay. 23 if (fListen && !::ChainstateActive().IsInitialBlockDownload()) 24 { 25 CAddress addr = GetLocalAddress(&pfrom.addr, pfrom.GetLocalServices()); 26@@ -2446,8 +2461,8 @@ void PeerLogicValidation::ProcessMessage(CNode& pfrom, const std::string& msg_ty 27 m_connman.PushMessage(&pfrom, CNetMsgMaker(nSendVersion).Make(NetMsgType::GETADDR)); 28 pfrom.fGetAddr = true; 29 30- // Moves address from New to Tried table in Addrman 31- // (unrelated to advertising our address) 32+ // Moves address from New to Tried table in Addrman, resolves 33+ // tried-table collisions, etc. 34 m_connman.MarkAddressGood(pfrom.addr); 35 }
naumenkogs commented at 7:46 am on September 2, 2020:I’d suggest making
If we're starting up for the first time, our addrman may be pretty empty and no one will know who we are.
less ambiguous:If we're starting up for the first time, it would improve our peering chances. Otherwise, our addrman may be pretty empty, and also no one will know who we are.
But generally, I think this comment:
- makes it much more clear
- actually convinces me that it’s an improvement over the existent code.
amitiuttarwar commented at 0:08 am on September 3, 2020:incorporated these comments into the changes,
I agree this should be further improved, but I’m hoping this is sufficient for this PR. @naumenkogs are you comfortable with the current state?
jnewbery commented at 10:18 am on August 15, 2020: memberStrong strong concept ACK.
I think it’d be good to extract everything in the
while (!interruptNet)
inThreadOpenConnections()
into its own function since deeply nested while/for/if blocks obscure control flow and are very often the sources of bugs. Doing that might be scope creep for this PR though.in src/net.h:902 in b672f8d84f outdated
898+ return m_conn_type != ConnectionType::BLOCK_RELAY; 899+ } 900+ 901+ /* Whether we attempt to advertise our own address to the peer when 902+ * processing their VERSION message */ 903+ bool AdvertiseAddressToConn() const {
jnewbery commented at 9:06 am on August 20, 2020:I really dislike these functions being insideCNode
(same as #19316 (review)). This is very specific net_processing logic, rather than a general property of the CNode. Having the logic inside net.h rather than at the point it’s needed makes it less readable.
amitiuttarwar commented at 3:16 am on August 21, 2020:yeah, I agree. this is why I initially opted for havingm_conn_type
as a public member var, but feedback on #19316 led me to switching to private. anyway, given the current state, its a bit unfortunate we don’t get the additional guarantee of a switch, but I agree with you & incorporated this feedback so it can be more understandable from the call site.in src/net.h:841 in b672f8d84f outdated
837@@ -838,37 +838,89 @@ class CNode 838 assert(false); 839 } 840 841+ /* These are the default connections that we use to connect with the
jnewbery commented at 9:18 am on August 20, 2020:Format these as doxygen comments (start with/**
) https://www.doxygen.nl/manual/docblocks.html
jnewbery commented at 9:21 am on August 20, 2020:These are great comments. Thank you! Should they be merged with the comments above on the enum members?
amitiuttarwar commented at 5:40 pm on August 20, 2020:yeah I felt very indecisive when trying to figure out the best spots for the comments. here what I’ve gone for is having the enum comments be a more concise description of the nature of the connection, and the function provide more context around design goals, capacity, etc.
I don’t think the split is super clear though, so yeah, maybe it would be more helpful to consolidate it all into the enum member comments. is that what you’re advocating for? or just trying to weigh like me? ⚖️ 😛
jnewbery commented at 6:22 pm on August 20, 2020:yes, consolidate them to one place. I think the enum is the right place for them.
amitiuttarwar commented at 3:09 am on August 21, 2020:done!
amitiuttarwar commented at 3:11 am on August 21, 2020:oof. so many options and yet I still missed the mark 😅 . moved the comments to the enum, but fixed this problem. thanks!jnewbery commented at 9:26 am on August 20, 2020: memberACK everything except the
AdvertiseAddressToConn()
change. I’d much rather leave the conditional asif (!pfrom.IsInboundConn() && !pfrom.IsBlockOnlyConn())
and add a large comment to above the
connman.MarkAddressGood(pfrom.addr);
to say that this isn’t anything to do with sending an addr or getaddr message.DrahtBot commented at 8:24 pm on August 20, 2020: memberThe following sections might be updated with supplementary metadata relevant to reviewers and maintainers.
Conflicts
Reviewers, this pull request conflicts with the following ones:
- #19860 (Improve diversification of new connections: privacy and stability by naumenkogs)
- #19858 (Periodically make block-relay connections and sync headers by sdaftuar)
- #19843 (Refactoring and minor improvement for self-advertisements by naumenkogs)
- #19829 ([net processing] Move block inventory state to net_processing by jnewbery)
- #19794 (p2p: Remove fGetAddr flag by mzumsande)
- #19763 (net: don’t try to relay to the address’ originator by vasild)
- #19725 ([RPC] Add connection type to getpeerinfo, improve logs by amitiuttarwar)
- #17785 (p2p: Unify Send and Receive protocol versions by hebasto)
- #17428 (p2p: Try to preserve outbound block-relay-only connections during restart by hebasto)
- #17194 (p2p: Avoid forwarding ADDR messages to SPV nodes by naumenkogs)
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.
amitiuttarwar force-pushed on Aug 21, 2020amitiuttarwar commented at 3:40 am on August 21, 2020: contributorthanks for the feedback! all comments should be addressed. @sipa I agree with this:
make the addr-relay decision depend on the connection type. That should be the most authoritative information about the connection.
But unfortunately I don’t think this is true:
but if we accidentally forget to create the necessary data structures, test will fail;
I updated
RelayAddrsWithConn
to always return true, recompiled & rantest_runner.py
& unit tests. Nothing failed. My concern would be addressed if it did :) I think with #19315, we should be able to add functional tests that would fail if a discrepancy occurs?Currently I think the downside is probably acceptable, but want to highlight since I don’t think we have any automatic protection in place. (unless I’m missing something?)
@jnewbery, RE:
I think it’d be good to extract everything in the while (!interruptNet) in ThreadOpenConnections() into its own function since deeply nested while/for/if blocks obscure control flow and are very often the sources of bugs. Doing that might be scope creep for this PR though.
I 100% agree. I spent so many hours trying to interpret the exact logic around what types of connections are open, and have gotten very confused by the deeply nested logic flows & nuances of things like where a break / continue that’s 4 levels deep will pop you back out to. I’ve taken a first pass at extracting the outer
while (!interruptNet)
loop here (yes, there are two levels nested): https://github.com/amitiuttarwar/bitcoin/commit/eecc5877c08744c55200846bb7b3b70c922b75ec. However, it needs to be done with great care and I would need to spend a lot more time / build a lot more confidence with these changes before I’d feel comfortable proposing them. Feel free to take a look or further the branch if you’re interested :)jnewbery commented at 10:43 am on August 21, 2020: memberutACK c99b26010eaf4d446eb5118e38dbcc03fabba11c
have gotten very confused by the deeply nested logic flows & nuances of things like where a break / continue that’s 4 levels deep will pop you back out to.
Yes, deeply nested if/while/switch statements hide bugs. If it’s possible to factor them out to clarify control flow, we should aim to do that.
I’ve taken a first pass at extracting the outer while (!interruptNet) loop here
I’ve had a very quick skim of that branch and it looks like the right direction to me.
I would need to spend a lot more time / build a lot more confidence with these changes before I’d feel comfortable proposing them. Feel free to take a look or further the branch if you’re interested :)
I think you’re very well positioned to open this PR if you want to. You’ve already spent hours getting to grips with the logic here :slightly_smiling_face: . I’ve already got my quota of refactor PRs open, but I can definitely commit to reviewing a PR if you open it. (Equally, we don’t need to fix this all at once. We can always leave this for another time).
in src/net_processing.cpp:2021 in 5dba7ab241 outdated
1981@@ -1982,7 +1982,7 @@ static void ProcessHeadersMessage(CNode& pfrom, CConnman& connman, ChainstateMan 1982 } 1983 } 1984 1985- if (!pfrom.fDisconnect && pfrom.IsOutboundOrBlockRelayConn() && nodestate->pindexBestKnownBlock != nullptr && pfrom.m_tx_relay != nullptr) { 1986+ if (!pfrom.fDisconnect && pfrom.IsFullOutboundConn() && nodestate->pindexBestKnownBlock != nullptr) {
ariard commented at 10:26 pm on August 22, 2020:EDIT: We may have already a bug here. I think comment “Note that block-relay-only peers are already implicitly protected so we only consider setting m_protect for the full-relay peers” is wrong as block-relay-only peers seems to not be protected from eviction by lagging chain logic and as such we should modify this check to scope them in setting
m_protect==true
(or either modifyConsiderEviction
’s first check)AFAICT, we have 2 different logics to evict outbound peers : lagging-chain (
ConsiderEviction
) and (CheckForStaleTipAndEvictPeers
). The former to sanitize out lazy/buggy peers who have never sent us a valid header and are always staying behind our tip, the latter triggered in case of stale tip due to block variance to seek a better chain by rotating our outbound peers if we already reach an outbound limit (EvictExtraOutboundPeers
).With regards to stale-tip, outbound block-relay-only are protected based on
m_tx_relay==nullptr
: https://github.com/bitcoin/bitcoin/blob/197450f80868fe752c6107955e5da80704212b34/src/net_processing.cpp#L3977With regards to lagging-chain, outbound block-relay-only are protected based on
m_protect==true
: https://github.com/bitcoin/bitcoin/blob/197450f80868fe752c6107955e5da80704212b34/src/net_processing.cpp#L3909So we may evict them if due to block variance they fulfilled the lagging chain conditions, contrary to the intent of unconditionally conserve them once they announced to us a header.
amitiuttarwar commented at 10:39 pm on August 26, 2020:ok, I’ve looked into this and am happy to discuss with you, but want to clarify we are on the same page that this PR doesn’t change the behavior.
I agree that the comment initially seems inconsistent with the logic in
ConsiderEviction
, but I’m still trying to wrap my head around all the mechanisms (since the chainwork check is repeated as the first conditional inConsiderEviction
, is the protection here essential?). I have many observations & questions, but I don’t think this PR is the right place to have this conversation. are you willing to open an issue?
ariard commented at 0:19 am on August 27, 2020:Yes as far as I understand this code it is neither introduced by this PR or #19316.
(since the chainwork check is repeated as the first conditional in ConsiderEviction, is the protection here essential?).
You mean the check
pindexBestKnownBlock != nullptr
? But you have a conditional above and it won’t exclude block-relay-only withm_protect=false
. I’m not sure about the exact semantic ofm_protect
but yes I should be able to write a test for this with #19315 to assert bug :)
sdaftuar commented at 7:00 pm on August 31, 2020:Looks like a documentation oversight when I changed around how this code worked while working on #15759. See #15759#pullrequestreview-281092623. I think originally I was going to make block-relay-only peers immune from all eviction, and then changed it to only be immune to outbound peer rotation-based eviction.
in src/net.cpp:1864 in 51f5181365 outdated
1879- continue; 1880- } 1881+ // Determine what type of connection to open. Opening 1882+ // OUTBOUND_FULL_RELAY connections gets the highest priority until we 1883+ // meet our full-relay capacity. Then we open BLOCK_RELAY connection 1884+ // until we hit our block-relay peer limit. GetTryNewOutboundPeer()
ariard commented at 10:30 pm on August 22, 2020:block-relay-only
to be consistent with mention of this connection type spread around the codebase.
amitiuttarwar commented at 7:49 pm on August 27, 2020:donein src/net.cpp:1868 in 51f5181365 outdated
1883+ // meet our full-relay capacity. Then we open BLOCK_RELAY connection 1884+ // until we hit our block-relay peer limit. GetTryNewOutboundPeer() 1885+ // gets set when a stale tip is detected, so we try opening an 1886+ // additional OUTBOUND_FULL_RELAY connection. If none of these 1887+ // conditions are met, check the nNextFeeler timer to decide if we 1888+ // should open a feeler.
ariard commented at 10:34 pm on August 22, 2020:s/feeler/FEELER/g
amitiuttarwar commented at 7:50 pm on August 27, 2020:donein src/net.cpp:1881 in 51f5181365 outdated
1894+ } else if (GetTryNewOutboundPeer()) { 1895+ // OUTBOUND_FULL_RELAY 1896+ } else if (nTime > nNextFeeler) { 1897+ nNextFeeler = PoissonNextSend(nTime, FEELER_INTERVAL); 1898+ conn_type = ConnectionType::FEELER; 1899+ fFeeler = true;
ariard commented at 10:43 pm on August 22, 2020:I think you can use nowconn_type.IsFeelerConn()
instead offFeeler=true
in the feeler-related following checks, semantically they are the same ?
amitiuttarwar commented at 7:52 pm on August 27, 2020:IsFeelerConn()
is a function onCNode
, notConnectionType
, so I don’t think this is possible
sdaftuar commented at 7:10 pm on August 31, 2020:It does seem a little weird to still have fFeeler as a variable after this change, but I don’t feel strongly.
naumenkogs commented at 10:14 am on September 1, 2020:Instead ofif (fFeeler)
, we can now further doif (conn_type == ConnectionType::FEELER)
, and drop thefFeeler
variable. I would prefer this.
naumenkogs commented at 10:19 am on September 1, 2020:Or, now that I see how manyif (fFeeler)
we have (a lot), I’m not so sure…
amitiuttarwar commented at 8:52 pm on September 2, 2020:yup, I left it in essentially as a local alias. we use it 6 times in the function and it didn’t seem helpful to switch those all over to the more verboseconn_type == ConnectionType::FEELER
in src/net.h:149 in a2d4fe3d2e outdated
153+ 154+ /** 155+ * Feeler connections are short lived connections used to increase the 156+ * number of connectable addresses in our AddrMan. Approximately every 157+ * FEELER_INTERVAL, we attempt to connect to a random address from the new 158+ * table. If successful, we add it to the tried table.
ariard commented at 10:47 pm on August 22, 2020:Only attempted after finishing to open other types of outbound connections
, comment on order of connections was dropped from original location inThreadOpenConnections
ariard commented at 0:25 am on August 27, 2020:Actually it does also solve bucket conflict between 2 tried tables entries, see : https://github.com/bitcoin/bitcoin/pull/9037
amitiuttarwar commented at 7:58 pm on August 27, 2020:I already added a comment in
ThreadOpenConnections
with the specifics of the ordering:0 // Determine what type of connection to open. Opening 1 // OUTBOUND_FULL_RELAY connections gets the highest priority until we 2 // meet our full-relay capacity. Then we open BLOCK_RELAY connection 3 // until we hit our block-relay-only peer limit. 4 // GetTryNewOutboundPeer() gets set when a stale tip is detected, so we 5 // try opening an additional OUTBOUND_FULL_RELAY connection. If none of 6 // these conditions are met, check the nNextFeeler timer to decide if 7 // we should open a FEELER.
and I believe that is the right place for documenting the order. I realized that the block-relay-only enum comment mentioned ordering so I’ve gone ahead and removed that. I think these comments are most valuable if they focus on the design goals and the high level description of the connection type & we look at the relevant code for more specifics. otherwise we also risk the comments falling out of date.
Actually it does also solve bucket conflict between 2 tried tables entries, see : #9037
what is this in response to? is there a suggested change?
naumenkogs commented at 11:41 am on September 1, 2020:I think these comments are most valuable if they focus on the design goals and the high level description of the connection type & we look at the relevant code for more specifics.
I agree
what is this in response to? is there a suggested change?
Antoine just mentions that there is another thing done when considering a feeler. See here. I think this part, if explained, also belongs there and not here.
in src/net.h:161 in a2d4fe3d2e outdated
162+ /** 163+ * After meeting our quota for full outbound connections, we attempt to 164+ * open MAX_BLOCK_RELAY_ONLY_CONNECTIONS to help prevent against partition 165+ * attacks. By not relaying transactions or addresses, these connections 166+ * are harder to detect by a third party, thus helping obfuscate the 167+ * network topology. Addresses are selected from AddrMan.
ariard commented at 10:57 pm on August 22, 2020:You can mark that addresses for {BLOCK_RELAY,OUTBOUND_FULL_RELAY} are drawn from tried table.
Also
block-relay network topology
.
amitiuttarwar commented at 7:59 pm on August 27, 2020:are they always? what happens if your tried table isn’t populated yet ?
ariard commented at 9:51 pm on September 3, 2020:Ah sorry, in fact they are drawn from bothnew/tried
tables at anytime. Surely being confused bySelectTriedCollision
innet.cpp
path!in src/net.h:163 in a2d4fe3d2e outdated
167+ * network topology. Addresses are selected from AddrMan. 168+ */ 169+ BLOCK_RELAY, 170+ 171+ /** 172+ * AddrFetch connections are short lived connections used to solicit
ariard commented at 11:00 pm on August 22, 2020:For both {ADDR_FETCH, FEELER} you can specify the condition suspending the connection (either VERSION or ADDR reception).
amitiuttarwar commented at 8:02 pm on August 27, 2020:feels a bit deep in the weeds for this patch, both have nuances of how disconnection is executedin src/net.h:155 in a2d4fe3d2e outdated
160+ FEELER, 161+ 162+ /** 163+ * After meeting our quota for full outbound connections, we attempt to 164+ * open MAX_BLOCK_RELAY_ONLY_CONNECTIONS to help prevent against partition 165+ * attacks. By not relaying transactions or addresses, these connections
ariard commented at 11:02 pm on August 22, 2020:transactions and addresses
to be strict
amitiuttarwar commented at 8:05 pm on August 27, 2020:hmm, seems like our minds are putting parenthesis in different places for this patch. I find “or” clearer. If you feel strongly I can try to rework the sentence. but otherwise, seems fairly trivial to mein src/net.h:156 in a2d4fe3d2e outdated
161+ 162+ /** 163+ * After meeting our quota for full outbound connections, we attempt to 164+ * open MAX_BLOCK_RELAY_ONLY_CONNECTIONS to help prevent against partition 165+ * attacks. By not relaying transactions or addresses, these connections 166+ * are harder to detect by a third party, thus helping obfuscate the
ariard commented at 11:08 pm on August 22, 2020:It’s unclear to me how do you qualify what is a “third party” in a P2P network, either another connected peer, a not-directly connected peer to local node or an infrastructure observer ? For the last one it doesn’t help at all. So maybe replace byany other peer, either directly paired with local node or not
?
amitiuttarwar commented at 8:06 pm on August 27, 2020:if two nodes are connected, a third party is any other entity. I don’t really understand your suggestion- if a node is “directly paired” with another peer, how would it be possible to obfuscate that connection to the peer? of course the peer can know what connections they have open?
ariard commented at 9:53 pm on September 3, 2020:I think the detection hardness assumption is function on how you define the third-party.
I meaned “directed paired” as Alice being connected with Bob and Caroll. Caroll guessing the connection Alice-Bob thanks to some relay leaks (like txn or addr).
in src/net_processing.cpp:2450 in 652e06117d outdated
2451@@ -2454,6 +2452,9 @@ void ProcessMessage( 2452 // Get recent addresses 2453 connman.PushMessage(&pfrom, CNetMsgMaker(nSendVersion).Make(NetMsgType::GETADDR)); 2454 pfrom.fGetAddr = true; 2455+ 2456+ // Moves address from New to Tried table in Addrman 2457+ // (unrelated to advertising our address)
ariard commented at 11:24 pm on August 22, 2020:I know this is likely temporary but comment may be made better like “Our address sanitization logic (feeler) is less prone to manipulation by outbound connections rather than potentially attacker controlled inbound. We don’t sanitize block-relay-only as it might be a leak of their presence”.
amitiuttarwar commented at 8:09 pm on August 27, 2020:I feel strongly that we should not cement opinions into docs in the codebase. since there are divergent opinions on whether or not we should have this logic here, I’d like to avoid documentation that indicates it must be here.
sdaftuar commented at 6:47 pm on August 31, 2020:nit:MarkAddressGood()
has other side effects besides moving addresses from new to tried (such as resolving collisions in the tried table?) so maybe it’d be better to update the comment to indicate that’s not all this does, eg// Moves address from New to Tried table in Addrman, among other things
)
naumenkogs commented at 8:38 am on September 1, 2020:I don’t find this comment particularly helpful… WhyMarkAddressGood
is called here, why for outbounds and non-blocks-only?
amitiuttarwar commented at 0:10 am on September 3, 2020:resolving this conversation just to close off threads, feel free to continue over hereDrahtBot added the label Needs rebase on Aug 24, 2020jonatack commented at 1:03 pm on August 25, 2020: memberConcept ACK on these changes. Late to the party but will try to review soon-ish.amitiuttarwar force-pushed on Aug 27, 2020amitiuttarwar commented at 8:10 pm on August 27, 2020: contributorDrahtBot removed the label Needs rebase on Aug 27, 2020in src/net_processing.cpp:846 in 8efe70ca9c outdated
842@@ -843,7 +843,7 @@ void PeerLogicValidation::InitializeNode(CNode *pnode) { 843 LOCK(cs_main); 844 mapNodeState.emplace_hint(mapNodeState.end(), std::piecewise_construct, std::forward_as_tuple(nodeid), std::forward_as_tuple(addr, std::move(addrName), pnode->IsInboundConn(), pnode->IsManualConn())); 845 } 846- if(!pnode->IsInboundConn()) 847+ if (!pnode->IsInboundConn())
sdaftuar commented at 6:48 pm on August 31, 2020:nit: If you’re including small style changes anyway, this could use curly braces.
amitiuttarwar commented at 11:51 pm on September 2, 2020:donesdaftuar approvedsdaftuar commented at 7:13 pm on August 31, 2020: memberutACK, just a couple nits but nothing blocking.in src/net.h:907 in f26b0ddac5 outdated
860@@ -861,6 +861,12 @@ class CNode 861 return m_conn_type == ConnectionType::INBOUND; 862 } 863 864+ /* Whether we send addr messages over this connection */ 865+ bool RelayAddrsWithConn() const
naumenkogs commented at 8:13 am on September 1, 2020:nit: not a big fan of this name… I actually thinkIsAddrRelayPeer
is much better :) I already saw some discussion around the distinction between “node” and “peer”, and now you start referring to it as a “connection”.
naumenkogs commented at 8:22 am on September 1, 2020:Also, this method should be now used here: https://github.com/bitcoin/bitcoin/blob/4c66cc0a5a99cbad51448e51d6b28e11e5232e33/src/net.cpp#L2772
The motivation is that we don’t want changing the addr condition in one place, and forgetting to change in the other place.
naumenkogs commented at 8:29 am on September 1, 2020:I think the new code should be:
0 if (conn_type_in != ConnectionType::BLOCK_RELAY) { 1 m_tx_relay = MakeUnique<TxRelay>(); 2 } 3 if (IsAddrRelayPeer()) { 4 m_addr_known = MakeUnique<CRollingBloomFilter>(5000, 0.001); 5 }
amitiuttarwar commented at 11:55 pm on September 2, 2020:ah nice. I like it! I was also concerned about the duplicated logic making room for future discrepancy. I’ve updated to include this, thanks!
RE name: seems like an issue with the whole convention around connection types? The name I’ve chosen is consistent with all the
m_conn_type
functions. Its a little late to change all that so I’m going to resolve this comment.in src/net_processing.cpp:2463 in f26b0ddac5 outdated
2425@@ -2426,9 +2426,7 @@ void PeerLogicValidation::ProcessMessage(CNode& pfrom, const std::string& msg_ty 2426 UpdatePreferredDownload(pfrom, State(pfrom.GetId())); 2427 } 2428 2429- if (!pfrom.IsInboundConn() && pfrom.IsAddrRelayPeer()) 2430- { 2431- // Advertise our address
naumenkogs commented at 8:31 am on September 1, 2020:nit: Why removing the comment? :)
naumenkogs commented at 12:11 pm on September 1, 2020:Sort of justified here, but i’m not convinced it’s the right approach. See my comment there.
amitiuttarwar commented at 11:59 pm on September 2, 2020:the comment removal was leftover from a previous iteration of this code where I updated the conditional to call a function
AdvertiseAddressToConn
or something that made the comment feel redundant.latest push includes the more in depth commenting proposed in that thread, will continue convo there
in src/net_processing.cpp:2462 in f26b0ddac5 outdated
2425@@ -2426,9 +2426,7 @@ void PeerLogicValidation::ProcessMessage(CNode& pfrom, const std::string& msg_ty 2426 UpdatePreferredDownload(pfrom, State(pfrom.GetId())); 2427 } 2428 2429- if (!pfrom.IsInboundConn() && pfrom.IsAddrRelayPeer()) 2430- { 2431- // Advertise our address 2432+ if (!pfrom.IsInboundConn() && !pfrom.IsBlockOnlyConn()) {
naumenkogs commented at 8:32 am on September 1, 2020:This is technically not a behavior change, but I don’t understand the logic behind replacingIsAddrRelayPeer
with!IsBlockOnlyConn()
. Why not replacing it withRelayAddrsWithConn
?
naumenkogs commented at 12:10 pm on September 1, 2020:I see it is justified (sort of?) here.
sdaftuar commented at 5:09 pm on September 1, 2020:The skipping of this logic for block-relay-only outbound peers was introduced in #15759 – gating it on whether the address-relay data structures are instantiated is confusing because the very last thing we do is unrelated to whether those data structures exist.
Instead, what this block has in common is that we only want to skip it for inbound or block-relay only peers, which is what this patch does. So I think this is right, and while we could split this block into two parts, one gated on whether the peer has the addr-relay data structures, and another gated on whether the peers is an outbound block-relay only connection, that seems pretty unnecessary to me. If someone adds another connection type that is outbound and non-block-relay only and only needs part of this, it can be refactored then.
jnewbery commented at 8:44 am on September 1, 2020: memberreACK 4c66cc0a5a
It looks like all of the remaining review comments are on style/documentation. Perhaps we should aim to merge this and address those in a follow-up?
in src/net.cpp:1859 in 1d89fe4ad4 outdated
1855@@ -1856,28 +1856,32 @@ void CConnman::ThreadOpenConnections(const std::vector<std::string> connect) 1856 } 1857 } 1858 1859- // Feeler Connections
naumenkogs commented at 8:56 am on September 1, 2020:Can this probably migrate to the enum?
naumenkogs commented at 11:55 am on September 1, 2020:Partially done in 630b9a5e1872ac91949b1e30f8cd7f28d1c946ab, but some of the details are lost.
amitiuttarwar commented at 0:02 am on September 3, 2020:what details are lost? the only thing not included is “Start attempting feeler connections only after node finishes making outbound connections,” but the ordering of connections is captured inThreadOpenConnections
. this seems like the same point as addressed over in this thread where you’ve agreed with me??in src/net.h:130 in 630b9a5e18 outdated
131+ */ 132+ INBOUND, 133+ 134+ /** 135+ * These are the default connections that we use to connect with the 136+ * network. There is no restriction on the inventory relayed- by default we
naumenkogs commented at 11:34 am on September 1, 2020:nit: mix of terminology, we normally don’t refer to addr as “inventory”. Let’s keep it that way :) Probably use something “types of stuff” or whatever else
amitiuttarwar commented at 0:03 am on September 3, 2020:oh, didn’t realize. fixednaumenkogs commented at 12:34 pm on September 1, 2020: memberIt looks like all of the remaining review comments are on style/documentation. Perhaps we should aim to merge this and address those in a follow-up?
Most of the changes in this commit are style/documentation. It’s not like it’s a big logic-changing PR with a lot of ACKs will have to invalidate :)
But actually, I’d be fine with ACKing this one as long as I get some feedback (code change, or at least a comment) on these two comments of mine I consider substantial: first, second.
sdaftuar commented at 1:13 pm on September 2, 2020: memberscripted-diff: Rename `OUTBOUND` ConnectionType to `OUTBOUND_FULL_RELAY`
-BEGIN VERIFY SCRIPT- sed -i 's/OUTBOUND, /OUTBOUND_FULL_RELAY, /g' src/net.h sed -i 's/ConnectionType::OUTBOUND/ConnectionType::OUTBOUND_FULL_RELAY/g' src/test/net_tests.cpp src/test/fuzz/process_message.cpp src/test/fuzz/process_messages.cpp src/net.cpp src/test/denialofservice_tests.cpp src/net.h src/test/fuzz/net.cpp -END VERIFY SCRIPT-
[net] Remove unnecessary default args on OpenNetworkConnection a6ab1e81f9amitiuttarwar force-pushed on Sep 2, 2020[refactor] Restructure logic to check for addr relay.
We previously identified if we relay addresses to the connection by checking for the existence of the m_addr_known data structure. With this commit, we answer this question based on the connection type. IsAddrRelayPeer() checked for the existence of the m_addr_known
[doc] Explain address handling logic in process messages
Co-authored-by: Suhas Daftuar <sdaftuar@gmail.com>
[trivial] Small style updates 1d74fc7df6[test] Add explicit tests that connection types get set correctly da3a0be61b[refactor] Simplify check for block-relay-only connection.
Previously we deduced it was a block-relay-only based on presence of the m_tx_relay structure. Now we have the ability to identify it directly via a connection type accessor function.
[refactor] Simplify connection type logic in ThreadOpenConnections
Consolidate the logic to determine connection type into one conditional to clarify how they are chosen.
[doc] Describe connection types in more depth. d5a57cef62[doc] Follow developer notes, add comment about missing default. eb1c5d090famitiuttarwar force-pushed on Sep 3, 2020amitiuttarwar commented at 0:43 am on September 3, 2020: contributorthank you for the reviews!
good catch on the silent merge conflict @sdaftuar, I’ve rebased to address. (turns out, there was more than one)
I’ve addressed all review comments. They were getting pretty web-like, so I’ve tried my best to resolve and focus on any outstanding conversations.
It looks like all of the remaining review comments are on style/documentation. Perhaps we should aim to merge this and address those in a follow-up?
Most of the changes in this commit are style/documentation. It’s not like it’s a big logic-changing PR with a lot of ACKs will have to invalidate :)
I’d like to get this PR merged sooner rather than later for a couple reasons:
- I’d like to get the renames on master to avoid possibility of more silent merge conflicts
- there are now two PRs that build on this (#19858 & #19860). further improvements can be addressed in parallel rather than holding up the whole sequence.
- I think its important not to discourage review / lose momentum. @jnewbery has ACKed the tip twice already & I’ve had to invalidate because of needing rebase. with 10 commits deep, it doesn’t seem trivial to reACK, so I’d rather use that review energy on smaller isolated changes that can be handled in another PR.
just trying to express that if I do get ACKs, I’m going to try to be cautious about invalidating, so might opt for starting another branch/PR if more non-essential changes are requested.
amitiuttarwar commented at 2:24 am on September 3, 2020: contributorthere’s one failing build, but I’m confused by it. nothing in the logs appears to have failed?? https://travis-ci.org/github/bitcoin/bitcoin/jobs/723629947RandyMcMillan commented at 2:36 am on September 3, 2020: contributor@amitiuttarwar - This PR is based on parent commits that aren’t passing - the fail may have nothing to do with your change. I recommend rebasing your commit on a recent passing commit.naumenkogs commented at 7:34 am on September 3, 2020: memberACK eb1c5d090f4ef844dabc1e8bf3864c1afe1c791claanwj commented at 11:27 am on September 3, 2020: memberCode review ACK eb1c5d090f4ef844dabc1e8bf3864c1afe1c791claanwj merged this on Sep 3, 2020laanwj closed this on Sep 3, 2020
ariard commented at 10:04 pm on September 3, 2020: memberpost-merge Code Review ACK eb1c5d0Fabcien referenced this in commit 7ef27e505c on Jul 13, 2021Fabcien referenced this in commit 86e28fbbbf on Jul 13, 2021Fabcien referenced this in commit 6ad16ebd6d on Jul 13, 2021Fabcien referenced this in commit 6166450c9c on Jul 13, 2021Fabcien referenced this in commit 54e18211fb on Jul 13, 2021Fabcien referenced this in commit 74e38fe222 on Jul 13, 2021Fabcien referenced this in commit 45b19beb72 on Jul 13, 2021Fabcien referenced this in commit a508b5b670 on Jul 13, 2021deadalnix referenced this in commit f4f912e5ca on Jul 13, 2021DrahtBot locked this on Feb 15, 2022
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: 2025-01-21 12:12 UTC
This site is hosted by @0xB10C
More mirrored repositories can be found on mirror.b10c.me