From adaecdeb786c32a495d0d4a755749db3968c5113 Mon Sep 17 00:00:00 2001 From: Pavel Zbitskiy <65323360+algorandskiy@users.noreply.github.com> Date: Fri, 26 Jul 2024 15:17:56 -0400 Subject: [PATCH] p2p: fix connection deduplication in hybrid mode (#6082) --- network/p2pNetwork.go | 1 + network/p2pNetwork_test.go | 55 +++++++++++++++++++++++++++++++++++++- 2 files changed, 55 insertions(+), 1 deletion(-) diff --git a/network/p2pNetwork.go b/network/p2pNetwork.go index 9f5448c0ea..9417b641ae 100644 --- a/network/p2pNetwork.go +++ b/network/p2pNetwork.go @@ -797,6 +797,7 @@ func (n *P2PNetwork) wsStreamHandler(ctx context.Context, p2pPeer peer.ID, strea networkPeerIdentityDisconnect.Inc(nil) n.log.With("remote", addr).With("local", localAddr).Warn("peer deduplicated before adding because the identity is already known") stream.Close() + return } wsp.init(n.config, outgoingMessagesBufferSize) diff --git a/network/p2pNetwork_test.go b/network/p2pNetwork_test.go index 0eac398431..9fc00774fb 100644 --- a/network/p2pNetwork_test.go +++ b/network/p2pNetwork_test.go @@ -31,6 +31,7 @@ import ( "time" "github.com/algorand/go-algorand/config" + algocrypto "github.com/algorand/go-algorand/crypto" "github.com/algorand/go-algorand/logging" "github.com/algorand/go-algorand/network/limitcaller" "github.com/algorand/go-algorand/network/p2p" @@ -1081,7 +1082,7 @@ func TestP2PWantTXGossip(t *testing.T) { require.True(t, net.wantTXGossip.Load()) } -func TestMergeP2PAddrInfoResolvedAddresses(t *testing.T) { +func TestP2PMergeAddrInfoResolvedAddresses(t *testing.T) { partitiontest.PartitionTest(t) t.Parallel() @@ -1154,3 +1155,55 @@ func TestMergeP2PAddrInfoResolvedAddresses(t *testing.T) { }) } } + +// TestP2PwsStreamHandlerDedup checks that the wsStreamHandler detects duplicate connections +// and does not add a new wePeer for it. +func TestP2PwsStreamHandlerDedup(t *testing.T) { + partitiontest.PartitionTest(t) + + cfg := config.GetDefaultLocal() + cfg.DNSBootstrapID = "" // disable DNS lookups since the test uses phonebook addresses + cfg.NetAddress = "127.0.0.1:0" + log := logging.TestingLog(t) + netA, err := NewP2PNetwork(log, cfg, "", nil, genesisID, config.Devtestnet, &nopeNodeInfo{}, &identityOpts{tracker: NewIdentityTracker()}) + require.NoError(t, err) + err = netA.Start() + require.NoError(t, err) + defer netA.Stop() + + peerInfoA := netA.service.AddrInfo() + addrsA, err := peer.AddrInfoToP2pAddrs(&peerInfoA) + require.NoError(t, err) + require.NotZero(t, addrsA[0]) + + multiAddrStr := addrsA[0].String() + phoneBookAddresses := []string{multiAddrStr} + netB, err := NewP2PNetwork(log, cfg, "", phoneBookAddresses, genesisID, config.Devtestnet, &nopeNodeInfo{}, &identityOpts{tracker: NewIdentityTracker()}) + require.NoError(t, err) + + // now say netA's identity tracker knows about netB's peerID + var netIdentPeerID algocrypto.PublicKey + p2pPeerPubKey, err := netB.service.ID().ExtractPublicKey() + require.NoError(t, err) + + b, err := p2pPeerPubKey.Raw() + require.NoError(t, err) + netIdentPeerID = algocrypto.PublicKey(b) + wsp := &wsPeer{ + identity: netIdentPeerID, + } + netA.identityTracker.setIdentity(wsp) + networkPeerIdentityDisconnectInitial := networkPeerIdentityDisconnect.GetUint64Value() + + // start network and ensure dedup happens + err = netB.Start() + require.NoError(t, err) + defer netB.Stop() + + require.Eventually(t, func() bool { + return networkPeerIdentityDisconnect.GetUint64Value() == networkPeerIdentityDisconnectInitial+1 + }, 2*time.Second, 50*time.Millisecond) + + require.False(t, netA.hasPeers()) + require.False(t, netB.hasPeers()) +}