routing: also fail pairs for node-level failures

This commit modifies the interpretation of node-level failures.
Previously only the failing node was marked. With this commit, also the
incoming and outgoing connections involved in the route are marked as
failed.

The change prepares for the removal of node-level failures in mission
control probability estimation.
This commit is contained in:
Joost Jager 2019-09-04 16:26:18 +02:00
parent 843c87a203
commit 559d3c0910
No known key found for this signature in database
GPG Key ID: A61B9D4C393C59C7
3 changed files with 33 additions and 6 deletions

@ -165,8 +165,8 @@ func TestMissionControl(t *testing.T) {
ctx.restartMc() ctx.restartMc()
ctx.expectP(1000, 0.4) ctx.expectP(1000, 0.4)
// A node level failure should bring probability of every channel back // A node level failure should bring probability of all known channels
// to zero. // back to zero.
ctx.reportFailure(0, lnwire.NewExpiryTooSoon(lnwire.ChannelUpdate{})) ctx.reportFailure(0, lnwire.NewExpiryTooSoon(lnwire.ChannelUpdate{}))
ctx.expectP(1000, 0) ctx.expectP(1000, 0)
@ -177,8 +177,8 @@ func TestMissionControl(t *testing.T) {
len(history.Nodes)) len(history.Nodes))
} }
if len(history.Pairs) != 2 { if len(history.Pairs) != 3 {
t.Fatalf("expected 2 pairs, but got %v", len(history.Pairs)) t.Fatalf("expected 3 pairs, but got %v", len(history.Pairs))
} }
// Test reporting a success. // Test reporting a success.

@ -378,10 +378,30 @@ func (i *interpretedResult) processPaymentOutcomeUnknown(route *route.Route) {
i.failPairRange(route, 0, n-1) i.failPairRange(route, 0, n-1)
} }
// failNode marks the node indicated by idx in the route as failed. This // failNode marks the node indicated by idx in the route as failed. It also
// function intentionally panics when the self node is failed. // marks the incoming and outgoing channels of the node as failed. This function
// intentionally panics when the self node is failed.
func (i *interpretedResult) failNode(rt *route.Route, idx int) { func (i *interpretedResult) failNode(rt *route.Route, idx int) {
// Mark the node as failing.
i.nodeFailure = &rt.Hops[idx-1].PubKeyBytes i.nodeFailure = &rt.Hops[idx-1].PubKeyBytes
// Mark the incoming connection as failed for the node. We intent to
// penalize as much as we can for a node level failure, including future
// outgoing traffic for this connection. The pair as it is returned by
// getPair is directed towards the failed node. Therefore we first
// reverse the pair. We don't want to affect the score of the node
// sending towards the failing node.
incomingChannelIdx := idx - 1
inPair, _ := getPair(rt, incomingChannelIdx)
i.pairResults[inPair.Reverse()] = failPairResult(0)
// If not the ultimate node, mark the outgoing connection as failed for
// the node.
if idx < len(rt.Hops) {
outgoingChannelIdx := idx
outPair, _ := getPair(rt, outgoingChannelIdx)
i.pairResults[outPair] = failPairResult(0)
}
} }
// failPairRange marks the node pairs from node fromIdx to node toIdx as failed // failPairRange marks the node pairs from node fromIdx to node toIdx as failed

@ -146,6 +146,10 @@ var resultTestCases = []resultTestCase{
expectedResult: &interpretedResult{ expectedResult: &interpretedResult{
nodeFailure: &hops[1], nodeFailure: &hops[1],
pairResults: map[DirectedNodePair]pairResult{
getTestPair(1, 0): failPairResult(0),
getTestPair(1, 2): failPairResult(0),
},
}, },
}, },
@ -160,6 +164,9 @@ var resultTestCases = []resultTestCase{
expectedResult: &interpretedResult{ expectedResult: &interpretedResult{
finalFailureReason: &reasonError, finalFailureReason: &reasonError,
nodeFailure: &hops[1], nodeFailure: &hops[1],
pairResults: map[DirectedNodePair]pairResult{
getTestPair(1, 0): failPairResult(0),
},
}, },
}, },
} }