From 2911944e5a426d2f991353a14cfb7272e253216f Mon Sep 17 00:00:00 2001 From: "Johan T. Halseth" Date: Tue, 30 Oct 2018 10:00:08 +0100 Subject: [PATCH 1/2] discovery/gossiper test: define unknown node announcement as stal To mimic the current behaviour of the router's IsStaleNode, we make the mockGraphSource consider a unknown node with no channels in the graph as stale. --- discovery/gossiper_test.go | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/discovery/gossiper_test.go b/discovery/gossiper_test.go index d9499a4a..96c3714f 100644 --- a/discovery/gossiper_test.go +++ b/discovery/gossiper_test.go @@ -206,7 +206,18 @@ func (r *mockGraphSource) IsStaleNode(nodePub routing.Vertex, timestamp time.Tim } } - return false + // If we did not find the node among our existing graph nodes, we + // require the node to already have a channel in the graph to not be + // considered stale. + for _, info := range r.infos { + if info.NodeKey1Bytes == nodePub { + return false + } + if info.NodeKey2Bytes == nodePub { + return false + } + } + return true } // IsPublicNode determines whether the given vertex is seen as a public node in From 13b33d79b3877d613e116b35fdf164f898776215 Mon Sep 17 00:00:00 2001 From: "Johan T. Halseth" Date: Tue, 30 Oct 2018 10:01:11 +0100 Subject: [PATCH 2/2] discover/gossiper test: add TestNodeAnnouncementNoChannels This commit adds a new test TestNodeAnnouncementNoChannels that ensures a node announcement for a unknown node is not forwarded by the gossiper. --- discovery/gossiper_test.go | 103 +++++++++++++++++++++++++++++++++++++ 1 file changed, 103 insertions(+) diff --git a/discovery/gossiper_test.go b/discovery/gossiper_test.go index 96c3714f..50384d95 100644 --- a/discovery/gossiper_test.go +++ b/discovery/gossiper_test.go @@ -2429,6 +2429,109 @@ func TestExtraDataNodeAnnouncementValidation(t *testing.T) { } } +// TestNodeAnnouncementNoChannels tests that NodeAnnouncements for nodes with +// no existing channels in the graph do not get forwarded. +func TestNodeAnnouncementNoChannels(t *testing.T) { + t.Parallel() + + ctx, cleanup, err := createTestCtx(0) + if err != nil { + t.Fatalf("can't create context: %v", err) + } + defer cleanup() + + batch, err := createAnnouncements(0) + if err != nil { + t.Fatalf("can't generate announcements: %v", err) + } + + remoteKey, err := btcec.ParsePubKey(batch.nodeAnn2.NodeID[:], + btcec.S256()) + if err != nil { + t.Fatalf("unable to parse pubkey: %v", err) + } + remotePeer := &mockPeer{remoteKey, nil, nil} + + // Process the remote node announcement. + select { + case err = <-ctx.gossiper.ProcessRemoteAnnouncement(batch.nodeAnn2, + remotePeer): + case <-time.After(2 * time.Second): + t.Fatal("did not process remote announcement") + } + if err != nil { + t.Fatalf("unable to process announcement: %v", err) + } + + // Since no channels or node announcements were already in the graph, + // the node announcement should be ignored, and not forwarded. + select { + case <-ctx.broadcastedMessage: + t.Fatal("node announcement was broadcast") + case <-time.After(2 * trickleDelay): + } + + // Now add the node's channel to the graph by processing the channel + // announement and channel update. + select { + case err = <-ctx.gossiper.ProcessRemoteAnnouncement(batch.remoteChanAnn, + remotePeer): + case <-time.After(2 * time.Second): + t.Fatal("did not process remote announcement") + } + if err != nil { + t.Fatalf("unable to process announcement: %v", err) + } + + select { + case err = <-ctx.gossiper.ProcessRemoteAnnouncement(batch.chanUpdAnn2, + remotePeer): + case <-time.After(2 * time.Second): + t.Fatal("did not process remote announcement") + } + if err != nil { + t.Fatalf("unable to process announcement: %v", err) + } + + // Now process the node announcement again. + select { + case err = <-ctx.gossiper.ProcessRemoteAnnouncement(batch.nodeAnn2, remotePeer): + case <-time.After(2 * time.Second): + t.Fatal("did not process remote announcement") + } + if err != nil { + t.Fatalf("unable to process announcement: %v", err) + } + + // This time the node announcement should be forwarded. The same should + // the channel announcement and update be. + for i := 0; i < 3; i++ { + select { + case <-ctx.broadcastedMessage: + case <-time.After(time.Second): + t.Fatal("announcement wasn't broadcast") + } + } + + // Processing the same node announement again should be ignored, as it + // is stale. + select { + case err = <-ctx.gossiper.ProcessRemoteAnnouncement(batch.nodeAnn2, + remotePeer): + case <-time.After(2 * time.Second): + t.Fatal("did not process remote announcement") + } + if err != nil { + t.Fatalf("unable to process announcement: %v", err) + } + + select { + case <-ctx.broadcastedMessage: + t.Fatal("node announcement was broadcast") + case <-time.After(2 * trickleDelay): + } +} + // mockPeer implements the lnpeer.Peer interface and is used to test the // gossiper's interaction with peers. type mockPeer struct {