Skip to content

Commit c45b9fb

Browse files
committed
net: correctly ban before the handshake is complete
7a8c251 made a change to avoid getting into SendMessages() until the version handshake (VERSION + VERACK) is complete. That was done to avoid leaking out messages to nodes who could connect, but never bothered sending us their version/verack. Unfortunately, the ban tally and possible disconnect are done as part of SendMessages(). So after 7a8c251, if a peer managed to do something bannable before completing the handshake (say send 100 non-version messages before their version), they wouldn't actually end up getting disconnected/banned. That's fixed here by checking the banscore as part of ProcessMessages() in addition to SendMessages().
1 parent d304fef commit c45b9fb

File tree

1 file changed

+37
-23
lines changed

1 file changed

+37
-23
lines changed

src/net_processing.cpp

Lines changed: 37 additions & 23 deletions
Original file line numberDiff line numberDiff line change
@@ -2596,6 +2596,36 @@ bool static ProcessMessage(CNode* pfrom, const std::string& strCommand, CDataStr
25962596
return true;
25972597
}
25982598

2599+
static bool SendRejectsAndCheckIfBanned(CNode* pnode, CConnman& connman)
2600+
{
2601+
AssertLockHeld(cs_main);
2602+
CNodeState &state = *State(pnode->GetId());
2603+
2604+
BOOST_FOREACH(const CBlockReject& reject, state.rejects) {
2605+
connman.PushMessage(pnode, CNetMsgMaker(INIT_PROTO_VERSION).Make(NetMsgType::REJECT, (std::string)NetMsgType::BLOCK, reject.chRejectCode, reject.strRejectReason, reject.hashBlock));
2606+
}
2607+
state.rejects.clear();
2608+
2609+
if (state.fShouldBan) {
2610+
state.fShouldBan = false;
2611+
if (pnode->fWhitelisted)
2612+
LogPrintf("Warning: not punishing whitelisted peer %s!\n", pnode->addr.ToString());
2613+
else if (pnode->fAddnode)
2614+
LogPrintf("Warning: not punishing addnoded peer %s!\n", pnode->addr.ToString());
2615+
else {
2616+
pnode->fDisconnect = true;
2617+
if (pnode->addr.IsLocal())
2618+
LogPrintf("Warning: not banning local peer %s!\n", pnode->addr.ToString());
2619+
else
2620+
{
2621+
connman.Ban(pnode->addr, BanReasonNodeMisbehaving);
2622+
}
2623+
}
2624+
return true;
2625+
}
2626+
return false;
2627+
}
2628+
25992629
bool ProcessMessages(CNode* pfrom, CConnman& connman, const std::atomic<bool>& interruptMsgProc)
26002630
{
26012631
const CChainParams& chainparams = Params();
@@ -2706,8 +2736,12 @@ bool ProcessMessages(CNode* pfrom, CConnman& connman, const std::atomic<bool>& i
27062736
PrintExceptionContinue(NULL, "ProcessMessages()");
27072737
}
27082738

2709-
if (!fRet)
2739+
if (!fRet) {
27102740
LogPrintf("%s(%s, %u bytes) FAILED peer=%d\n", __func__, SanitizeString(strCommand), nMessageSize, pfrom->id);
2741+
}
2742+
2743+
LOCK(cs_main);
2744+
SendRejectsAndCheckIfBanned(pfrom, connman);
27112745

27122746
return fMoreWork;
27132747
}
@@ -2773,30 +2807,10 @@ bool SendMessages(CNode* pto, CConnman& connman, const std::atomic<bool>& interr
27732807
if (!lockMain)
27742808
return true;
27752809

2810+
if (SendRejectsAndCheckIfBanned(pto, connman))
2811+
return true;
27762812
CNodeState &state = *State(pto->GetId());
27772813

2778-
BOOST_FOREACH(const CBlockReject& reject, state.rejects)
2779-
connman.PushMessage(pto, msgMaker.Make(NetMsgType::REJECT, (std::string)NetMsgType::BLOCK, reject.chRejectCode, reject.strRejectReason, reject.hashBlock));
2780-
state.rejects.clear();
2781-
2782-
if (state.fShouldBan) {
2783-
state.fShouldBan = false;
2784-
if (pto->fWhitelisted)
2785-
LogPrintf("Warning: not punishing whitelisted peer %s!\n", pto->addr.ToString());
2786-
else if (pto->fAddnode)
2787-
LogPrintf("Warning: not punishing addnoded peer %s!\n", pto->addr.ToString());
2788-
else {
2789-
pto->fDisconnect = true;
2790-
if (pto->addr.IsLocal())
2791-
LogPrintf("Warning: not banning local peer %s!\n", pto->addr.ToString());
2792-
else
2793-
{
2794-
connman.Ban(pto->addr, BanReasonNodeMisbehaving);
2795-
}
2796-
return true;
2797-
}
2798-
}
2799-
28002814
// Address refresh broadcast
28012815
int64_t nNow = GetTimeMicros();
28022816
if (!IsInitialBlockDownload() && pto->nNextLocalAddrSend < nNow) {

0 commit comments

Comments
 (0)