mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-04-03 17:40:38 +02:00
lntest: fix possible race condition re asserting channel propagation
In this commit, we attempt to fix a race condition that may occur in the current AMP and MPP tests. It appears the following scenario is possible: * The `mppTestContext` [is used to create 6 channels back to back](https://github.com/lightningnetwork/lnd/blob/master/lntest/itest/lnd_amp_test.go#L43) * The method used to create the channel ends up calling [`openChannelAndAssert`](edd4152682/lntest/itest/lnd_mpp_test.go (L300)
) which'll open the channel, mine 6 blocks, [then ensure that the channel gets advertised](edd4152682/lntest/itest/assertions.go (L78)
) * Later on, [we wait for all nodes to hear about all channels on the network level](https://github.com/lightningnetwork/lnd/blob/master/lntest/itest/lnd_amp_test.go#L62) I think the issue here is that we'll potentially already have mined 30 or so blocks before getting to the final nodes, and those nodes may have already heard about the channel already. This may then cause their [`lightningNetworkWatcher`](edd4152682/lntest/node.go (L1213)
) goroutine to not properly dispatch this, since it's assumed that the channel hasn't been announced (or notified) when the method is called. One solution here is to just check if the channel is already in the node's graph or not, when we go to register the notification. If we do this in the same state machine as the watcher, then we ensure if the channel is already known, the client is immediately notified. One thing that can help us debug this more in the future is adding additional logging in some of these helper goroutines so we can more easily track the control flow. This commit implements this solution by checking to ensure that the channel isn't already known in our channel graph before attempting to wait for its notification, as we may already have missed the notification before this registration request came in.
This commit is contained in:
parent
ea7ea33e24
commit
6cd981420a
@ -1210,6 +1210,27 @@ func getChanPointFundingTxid(chanPoint *lnrpc.ChannelPoint) ([]byte, error) {
|
||||
return txid, nil
|
||||
}
|
||||
|
||||
func checkChanPointInGraph(ctx context.Context,
|
||||
node *HarnessNode, chanPoint wire.OutPoint) bool {
|
||||
|
||||
ctxt, cancel := context.WithTimeout(ctx, DefaultTimeout)
|
||||
defer cancel()
|
||||
chanGraph, err := node.DescribeGraph(ctxt, &lnrpc.ChannelGraphRequest{})
|
||||
if err != nil {
|
||||
return false
|
||||
}
|
||||
|
||||
targetChanPoint := chanPoint.String()
|
||||
for _, chanEdge := range chanGraph.Edges {
|
||||
candidateChanPoint := chanEdge.ChanPoint
|
||||
if targetChanPoint == candidateChanPoint {
|
||||
return true
|
||||
}
|
||||
}
|
||||
|
||||
return false
|
||||
}
|
||||
|
||||
// lightningNetworkWatcher is a goroutine which is able to dispatch
|
||||
// notifications once it has been observed that a target channel has been
|
||||
// closed or opened within the network. In order to dispatch these
|
||||
@ -1322,6 +1343,21 @@ func (hn *HarnessNode) lightningNetworkWatcher(subscribed chan error) {
|
||||
continue
|
||||
}
|
||||
|
||||
// Before we add the channel to our set of open
|
||||
// clients, we'll check to see if the channel
|
||||
// is already in the channel graph of the
|
||||
// target node. This lets us handle the case
|
||||
// where a node has already seen a channel
|
||||
// before a notification has been requested,
|
||||
// causing us to miss it.
|
||||
chanFound := checkChanPointInGraph(
|
||||
context.Background(), hn, targetChan,
|
||||
)
|
||||
if chanFound {
|
||||
close(watchRequest.eventChan)
|
||||
continue
|
||||
}
|
||||
|
||||
// Otherwise, we'll add this to the list of
|
||||
// watch open clients for this out point.
|
||||
hn.openClients[targetChan] = append(
|
||||
|
Loading…
x
Reference in New Issue
Block a user