server: eliminate possible initial connection churn by ignoring duplicate inbound conns

This commit fixes a bug that was introduced when the connection
handling was re-worked to properly handle the case of concurrent
connections being made. In certain cases after a successful initial
connection, a peer’s stray goroutine would still attempt to establish a
second outbound connection even though a connection had already been
established. This was properly handled by the connecting peer, but not
he receiving peer. This commit adds the additional logic to the
receiving peer to ensure that we properly handle this case.
This commit is contained in:
Olaoluwa Osuntokun 2017-05-10 17:42:53 -07:00
parent e05ec619ca
commit 5fe3a5631d
No known key found for this signature in database
GPG Key ID: 9CC5B105D03521A2

@ -757,15 +757,23 @@ func (s *server) inboundPeerConnected(conn net.Conn) {
s.peersMtx.Lock() s.peersMtx.Lock()
defer s.peersMtx.Unlock() defer s.peersMtx.Unlock()
// If we already have an inbound connection to this peer, then ignore
// this new connection.
nodePub := conn.(*brontide.Conn).RemotePub()
pubStr := string(nodePub.SerializeCompressed())
if _, ok := s.inboundPeers[pubStr]; ok {
srvrLog.Debugf("Ignoring duplicate inbound connection")
conn.Close()
return
}
srvrLog.Infof("New inbound connection from %v", conn.RemoteAddr()) srvrLog.Infof("New inbound connection from %v", conn.RemoteAddr())
localPub := s.identityPriv.PubKey() localPub := s.identityPriv.PubKey()
nodePub := conn.(*brontide.Conn).RemotePub()
// Check to see if we should drop our connection, if not, then we'll // Check to see if we should drop our connection, if not, then we'll
// close out this connection with the remote peer. This // close out this connection with the remote peer. This
// prevents us from having duplicate connections, or none. // prevents us from having duplicate connections, or none.
pubStr := string(nodePub.SerializeCompressed())
if connectedPeer, ok := s.peersByPub[pubStr]; ok { if connectedPeer, ok := s.peersByPub[pubStr]; ok {
// If the connection we've already established should be kept, // If the connection we've already established should be kept,
// then we'll close out this connection s.t there's only a // then we'll close out this connection s.t there's only a