• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

lightningnetwork / lnd / 16911773184

12 Aug 2025 02:21PM UTC coverage: 57.471% (-9.4%) from 66.9%
16911773184

Pull #10103

github

web-flow
Merge d64a1234d into f3e1f2f35
Pull Request #10103: Rate limit outgoing gossip bandwidth by peer

57 of 77 new or added lines in 5 files covered. (74.03%)

28294 existing lines in 457 files now uncovered.

99110 of 172451 relevant lines covered (57.47%)

1.78 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

10.26
/autopilot/prefattach.go
1
package autopilot
2

3
import (
4
        "context"
5
        prand "math/rand"
6
        "time"
7

8
        "github.com/btcsuite/btcd/btcec/v2"
9
        "github.com/btcsuite/btcd/btcutil"
10
)
11

12
// minMedianChanSizeFraction determines the minimum size a channel must have to
13
// count positively when calculating the scores using preferential attachment.
14
// The minimum channel size is calculated as median/minMedianChanSizeFraction,
15
// where median is the median channel size of the entire graph.
16
const minMedianChanSizeFraction = 4
17

18
// PrefAttachment is an implementation of the AttachmentHeuristic interface
19
// that implement a non-linear preferential attachment heuristic. This means
20
// that given a threshold to allocate to automatic channel establishment, the
21
// heuristic will attempt to favor connecting to nodes which already have a set
22
// amount of links, selected by sampling from a power law distribution. The
23
// attachment is non-linear in that it favors nodes with a higher in-degree but
24
// less so than regular linear preferential attachment. As a result, this
25
// creates smaller and less clusters than regular linear preferential
26
// attachment.
27
//
28
// TODO(roasbeef): BA, with k=-3
29
type PrefAttachment struct {
30
}
31

32
// NewPrefAttachment creates a new instance of a PrefAttachment heuristic.
33
func NewPrefAttachment() *PrefAttachment {
3✔
34
        prand.Seed(time.Now().Unix())
3✔
35
        return &PrefAttachment{}
3✔
36
}
3✔
37

38
// A compile time assertion to ensure PrefAttachment meets the
39
// AttachmentHeuristic interface.
40
var _ AttachmentHeuristic = (*PrefAttachment)(nil)
41

42
// NodeID is a simple type that holds an EC public key serialized in compressed
43
// format.
44
type NodeID [33]byte
45

46
// NewNodeID creates a new nodeID from a passed public key.
47
func NewNodeID(pub *btcec.PublicKey) NodeID {
3✔
48
        var n NodeID
3✔
49
        copy(n[:], pub.SerializeCompressed())
3✔
50
        return n
3✔
51
}
3✔
52

53
// Name returns the name of this heuristic.
54
//
55
// NOTE: This is a part of the AttachmentHeuristic interface.
56
func (p *PrefAttachment) Name() string {
3✔
57
        return "preferential"
3✔
58
}
3✔
59

60
// NodeScores is a method that given the current channel graph and current set
61
// of local channels, scores the given nodes according to the preference of
62
// opening a channel of the given size with them. The returned channel
63
// candidates maps the NodeID to a NodeScore for the node.
64
//
65
// The heuristic employed by this method is one that attempts to promote a
66
// scale-free network globally, via local attachment preferences for new nodes
67
// joining the network with an amount of available funds to be allocated to
68
// channels. Specifically, we consider the degree of each node (and the flow
69
// in/out of the node available via its open channels) and utilize the
70
// Barabási–Albert model to drive our recommended attachment heuristics. If
71
// implemented globally for each new participant, this results in a channel
72
// graph that is scale-free and follows a power law distribution with k=-3.
73
//
74
// To avoid assigning a high score to nodes with a large number of small
75
// channels, we only count channels at least as large as a given fraction of
76
// the graph's median channel size.
77
//
78
// The returned scores will be in the range [0.0, 1.0], where higher scores are
79
// given to nodes already having high connectivity in the graph.
80
//
81
// NOTE: This is a part of the AttachmentHeuristic interface.
82
func (p *PrefAttachment) NodeScores(ctx context.Context, g ChannelGraph,
83
        chans []LocalChannel, chanSize btcutil.Amount,
UNCOV
84
        nodes map[NodeID]struct{}) (map[NodeID]*NodeScore, error) {
×
UNCOV
85

×
UNCOV
86
        // We first run though the graph once in order to find the median
×
UNCOV
87
        // channel size.
×
UNCOV
88
        var (
×
UNCOV
89
                allChans  []btcutil.Amount
×
UNCOV
90
                seenChans = make(map[uint64]struct{})
×
UNCOV
91
        )
×
UNCOV
92
        err := g.ForEachNodesChannels(
×
UNCOV
93
                ctx, func(_ context.Context, node Node,
×
UNCOV
94
                        channels []*ChannelEdge) error {
×
UNCOV
95

×
UNCOV
96
                        for _, e := range channels {
×
UNCOV
97
                                if _, ok := seenChans[e.ChanID.ToUint64()]; ok {
×
UNCOV
98
                                        continue
×
99
                                }
UNCOV
100
                                seenChans[e.ChanID.ToUint64()] = struct{}{}
×
UNCOV
101
                                allChans = append(allChans, e.Capacity)
×
102
                        }
103

UNCOV
104
                        return nil
×
105
                },
UNCOV
106
                func() {
×
UNCOV
107
                        allChans = nil
×
UNCOV
108
                        clear(seenChans)
×
UNCOV
109
                },
×
110
        )
UNCOV
111
        if err != nil {
×
112
                return nil, err
×
113
        }
×
114

UNCOV
115
        medianChanSize := Median(allChans)
×
UNCOV
116
        log.Tracef("Found channel median %v for preferential score heuristic",
×
UNCOV
117
                medianChanSize)
×
UNCOV
118

×
UNCOV
119
        // Count the number of large-ish channels for each particular node in
×
UNCOV
120
        // the graph.
×
UNCOV
121
        var maxChans int
×
UNCOV
122
        nodeChanNum := make(map[NodeID]int)
×
UNCOV
123
        err = g.ForEachNodesChannels(
×
UNCOV
124
                ctx, func(ctx context.Context, node Node,
×
UNCOV
125
                        edges []*ChannelEdge) error {
×
UNCOV
126

×
UNCOV
127
                        var nodeChans int
×
UNCOV
128
                        for _, e := range edges {
×
UNCOV
129
                                // Since connecting to nodes with a lot of small
×
UNCOV
130
                                // channels actually worsens our connectivity in
×
UNCOV
131
                                // the graph (we will potentially waste time
×
UNCOV
132
                                // trying to use these useless channels in path
×
UNCOV
133
                                // finding), we decrease the counter for such
×
UNCOV
134
                                // channels.
×
UNCOV
135
                                //
×
UNCOV
136
                                //nolint:ll
×
UNCOV
137
                                if e.Capacity <
×
UNCOV
138
                                        medianChanSize/minMedianChanSizeFraction {
×
139

×
140
                                        nodeChans--
×
141

×
142
                                        continue
×
143
                                }
144

145
                                // Larger channels we count.
UNCOV
146
                                nodeChans++
×
147
                        }
148

149
                        // We keep track of the highest-degree node we've seen,
150
                        // as this will be given the max score.
UNCOV
151
                        if nodeChans > maxChans {
×
UNCOV
152
                                maxChans = nodeChans
×
UNCOV
153
                        }
×
154

155
                        // If this node is not among our nodes to score, we can
156
                        // return early.
UNCOV
157
                        nID := NodeID(node.PubKey())
×
UNCOV
158
                        if _, ok := nodes[nID]; !ok {
×
159
                                log.Tracef("Node %x not among nodes to score, "+
×
160
                                        "ignoring", nID[:])
×
161
                                return nil
×
162
                        }
×
163

164
                        // Otherwise we'll record the number of channels.
UNCOV
165
                        nodeChanNum[nID] = nodeChans
×
UNCOV
166
                        log.Tracef("Counted %v channels for node %x", nodeChans,
×
UNCOV
167
                                nID[:])
×
UNCOV
168

×
UNCOV
169
                        return nil
×
UNCOV
170
                }, func() {
×
UNCOV
171
                        maxChans = 0
×
UNCOV
172
                        clear(nodeChanNum)
×
UNCOV
173
                },
×
174
        )
UNCOV
175
        if err != nil {
×
176
                return nil, err
×
177
        }
×
178

179
        // If there are no channels in the graph we cannot determine any
180
        // preferences, so we return, indicating all candidates get a score of
181
        // zero.
UNCOV
182
        if maxChans == 0 {
×
UNCOV
183
                log.Tracef("No channels in the graph")
×
UNCOV
184
                return nil, nil
×
UNCOV
185
        }
×
186

UNCOV
187
        existingPeers := make(map[NodeID]struct{})
×
UNCOV
188
        for _, c := range chans {
×
UNCOV
189
                existingPeers[c.Node] = struct{}{}
×
UNCOV
190
        }
×
191

192
        // For each node in the set of nodes, count their fraction of channels
193
        // in the graph, and use that as the score.
UNCOV
194
        candidates := make(map[NodeID]*NodeScore)
×
UNCOV
195
        for nID, nodeChans := range nodeChanNum {
×
UNCOV
196

×
UNCOV
197
                // If the node is among or existing channel peers, we don't
×
UNCOV
198
                // need another channel.
×
UNCOV
199
                if _, ok := existingPeers[nID]; ok {
×
UNCOV
200
                        log.Tracef("Node %x among existing peers for pref "+
×
UNCOV
201
                                "attach heuristic, giving zero score", nID[:])
×
UNCOV
202
                        continue
×
203
                }
204

205
                // If the node had no large channels, we skip it, since it
206
                // would have gotten a zero score anyway.
UNCOV
207
                if nodeChans <= 0 {
×
UNCOV
208
                        log.Tracef("Skipping node %x with channel count %v",
×
UNCOV
209
                                nID[:], nodeChans)
×
UNCOV
210
                        continue
×
211
                }
212

213
                // Otherwise we score the node according to its fraction of
214
                // channels in the graph, scaled such that the highest-degree
215
                // node will be given a score of 1.0.
UNCOV
216
                score := float64(nodeChans) / float64(maxChans)
×
UNCOV
217
                log.Tracef("Giving node %x a pref attach score of %v",
×
UNCOV
218
                        nID[:], score)
×
UNCOV
219

×
UNCOV
220
                candidates[nID] = &NodeScore{
×
UNCOV
221
                        NodeID: nID,
×
UNCOV
222
                        Score:  score,
×
UNCOV
223
                }
×
224
        }
225

UNCOV
226
        return candidates, nil
×
227
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2025 Coveralls, Inc