routing: modify SendPayment loop to be lazy, iterative, and use missionControl
In this commit we modify the SendPayment loop to optimize for time-to-first-payment-success-or-failure. The prior logic would first attempt to find at least 100 routes to the destination, then iteratively prune them away as errors were encountered. In this commit, we modify this approach to instead take a lazy approach: we first find the current “best” path, attempt to send to that, and if an error occurs we prune a section of the graph by reporting to missionControl, then continue. With this new approach, if the first known path has sufficient capacity, and is available, then the payment speed is greatly improved from the PoV of users. Additionally, we avoid the excessive computation of crawling most of the graph in the k-shortest paths loop. With the decay on missionControl, all routes will now feed information into the central knowledge hung, allowing all payments to iteratively find out the inactive portions of the payment graph.
This commit is contained in:
parent
276f2e467b
commit
8ef829ed80
@ -198,6 +198,15 @@ type ChannelRouter struct {
|
|||||||
// existing client.
|
// existing client.
|
||||||
ntfnClientUpdates chan *topologyClientUpdate
|
ntfnClientUpdates chan *topologyClientUpdate
|
||||||
|
|
||||||
|
// missionControl is a shared memory of sorts that executions of
|
||||||
|
// payment path finding use in order to remember which vertexes/edges
|
||||||
|
// were pruned from prior attempts. During SendPayment execution,
|
||||||
|
// errors sent by nodes are mapped into a vertex or edge to be pruned.
|
||||||
|
// Each run will then take into account this set of pruned
|
||||||
|
// vertexes/edges to reduce route failure and pass on graph information
|
||||||
|
// gained to the next execution.
|
||||||
|
missionControl *missionControl
|
||||||
|
|
||||||
sync.RWMutex
|
sync.RWMutex
|
||||||
|
|
||||||
quit chan struct{}
|
quit chan struct{}
|
||||||
@ -225,6 +234,7 @@ func New(cfg Config) (*ChannelRouter, error) {
|
|||||||
networkUpdates: make(chan *routingMsg),
|
networkUpdates: make(chan *routingMsg),
|
||||||
topologyClients: make(map[uint64]*topologyClient),
|
topologyClients: make(map[uint64]*topologyClient),
|
||||||
ntfnClientUpdates: make(chan *topologyClientUpdate),
|
ntfnClientUpdates: make(chan *topologyClientUpdate),
|
||||||
|
missionControl: newMissionControl(),
|
||||||
routeCache: make(map[routeTuple][]*Route),
|
routeCache: make(map[routeTuple][]*Route),
|
||||||
quit: make(chan struct{}),
|
quit: make(chan struct{}),
|
||||||
}, nil
|
}, nil
|
||||||
@ -950,6 +960,8 @@ func pruneChannelFromRoutes(routes []*Route, skipChan uint64) []*Route {
|
|||||||
func (r *ChannelRouter) FindRoutes(target *btcec.PublicKey,
|
func (r *ChannelRouter) FindRoutes(target *btcec.PublicKey,
|
||||||
amt lnwire.MilliSatoshi) ([]*Route, error) {
|
amt lnwire.MilliSatoshi) ([]*Route, error) {
|
||||||
|
|
||||||
|
// TODO(roasbeef): make num routes a param
|
||||||
|
|
||||||
dest := target.SerializeCompressed()
|
dest := target.SerializeCompressed()
|
||||||
log.Debugf("Searching for path to %x, sending %v", dest, amt)
|
log.Debugf("Searching for path to %x, sending %v", dest, amt)
|
||||||
|
|
||||||
@ -1168,20 +1180,51 @@ func (r *ChannelRouter) SendPayment(payment *LightningPayment) ([32]byte, *Route
|
|||||||
sendError error
|
sendError error
|
||||||
)
|
)
|
||||||
|
|
||||||
// First, we'll kick off the payment attempt by attempting to query the
|
// We'll also fetch the current block height so we can properly
|
||||||
// known channel graph for a route to the destination.
|
// calculate the required HTLC time locks within the route.
|
||||||
routes, err := r.FindRoutes(payment.Target, payment.Amount)
|
_, currentHeight, err := r.cfg.Chain.GetBestBlock()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return preImage, nil, err
|
return preImage, nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// For each eligible path, we'll attempt to successfully send our
|
// We'll continue until either our payment succeeds, or we encounter a
|
||||||
// target payment using the multi-hop route. We'll try each route
|
// critical error during path finding.
|
||||||
// serially until either once succeeds, or we've exhausted our set of
|
sourceVertex := newVertex(r.selfNode.PubKey)
|
||||||
// available paths.
|
for {
|
||||||
sendLoop:
|
// First, we'll query mission control for it's current
|
||||||
for i := 0; i < len(routes); i++ {
|
// recommendation on the edges/vertexes to ignore during path
|
||||||
route := routes[i]
|
// finding.
|
||||||
|
pruneView := r.missionControl.GraphPruneView()
|
||||||
|
|
||||||
|
// Taking into account this prune view, we'll attempt to locate
|
||||||
|
// a path to our destination, respecting the recommendations
|
||||||
|
// from missionControl.
|
||||||
|
path, err := findPath(nil, r.cfg.Graph, r.selfNode,
|
||||||
|
payment.Target, pruneView.vertexes, pruneView.edges,
|
||||||
|
payment.Amount)
|
||||||
|
if err != nil {
|
||||||
|
// If we're unable to successfully make a payment using
|
||||||
|
// any of the routes we've found, then return an error.
|
||||||
|
if sendError != nil {
|
||||||
|
return [32]byte{}, nil, fmt.Errorf("unable to "+
|
||||||
|
"route payment to destination: %v",
|
||||||
|
sendError)
|
||||||
|
}
|
||||||
|
|
||||||
|
return preImage, nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// With the next candiate path found, we'll attempt to turn
|
||||||
|
// this into a route by applying the time-lock and fee
|
||||||
|
// requirements.
|
||||||
|
route, err := newRoute(payment.Amount, sourceVertex, path,
|
||||||
|
uint32(currentHeight))
|
||||||
|
if err != nil {
|
||||||
|
// TODO(roasbeef): return which edge/vertex didn't work
|
||||||
|
// out
|
||||||
|
return preImage, nil, err
|
||||||
|
}
|
||||||
|
|
||||||
log.Tracef("Attempting to send payment %x, using route: %v",
|
log.Tracef("Attempting to send payment %x, using route: %v",
|
||||||
payment.PaymentHash, newLogClosure(func() string {
|
payment.PaymentHash, newLogClosure(func() string {
|
||||||
return spew.Sdump(route)
|
return spew.Sdump(route)
|
||||||
@ -1332,27 +1375,23 @@ sendLoop:
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the channel was found, then we'll filter
|
// If the channel was found, then we'll inform
|
||||||
// the rest of the routes to exclude any path
|
// mission control of this failure so future
|
||||||
// that includes this channel, and restart the
|
// attempts avoid this link temporarily.
|
||||||
// loop.
|
r.missionControl.ReportChannelFailure(badChan)
|
||||||
//
|
continue
|
||||||
// TODO(roasbeef): should actually be
|
|
||||||
// directional?
|
|
||||||
routes = pruneChannelFromRoutes(routes,
|
|
||||||
badChan)
|
|
||||||
goto sendLoop
|
|
||||||
|
|
||||||
// If the send fail due to a node not having the
|
// If the send fail due to a node not having the
|
||||||
// required features, then we'll note this error and
|
// required features, then we'll note this error and
|
||||||
// continue
|
// continue.
|
||||||
|
//
|
||||||
// TODO(roasbeef): remove node from path
|
// TODO(roasbeef): remove node from path
|
||||||
case *lnwire.FailRequiredNodeFeatureMissing:
|
case *lnwire.FailRequiredNodeFeatureMissing:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
// If the send fail due to a node not having the
|
// If the send fail due to a node not having the
|
||||||
// required features, then we'll note this error and
|
// required features, then we'll note this error and
|
||||||
// continue
|
// continue.
|
||||||
//
|
//
|
||||||
// TODO(roasbeef): remove channel from path
|
// TODO(roasbeef): remove channel from path
|
||||||
case *lnwire.FailRequiredChannelFeatureMissing:
|
case *lnwire.FailRequiredChannelFeatureMissing:
|
||||||
@ -1371,17 +1410,22 @@ sendLoop:
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
// Once we've located the vertex, we'll prune
|
// Once we've located the vertex, we'll report
|
||||||
// it out fromthe rest of the routes, and
|
// this failure to missionControl and restart
|
||||||
// restart path finding.
|
// path finding.
|
||||||
routes = pruneNodeFromRoutes(routes,
|
r.missionControl.ReportVertexFailure(missingNode)
|
||||||
missingNode)
|
continue
|
||||||
goto sendLoop
|
|
||||||
|
|
||||||
// If the node wasn't able to forward for which ever
|
// If the node wasn't able to forward for which ever
|
||||||
// reason, then we'll note this and continue with the
|
// reason, then we'll note this and continue with the
|
||||||
// routes.
|
// routes.
|
||||||
case *lnwire.FailTemporaryNodeFailure:
|
case *lnwire.FailTemporaryNodeFailure:
|
||||||
|
missingNode, ok := route.nextHopVertex(errSource)
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
r.missionControl.ReportVertexFailure(missingNode)
|
||||||
continue
|
continue
|
||||||
|
|
||||||
// If we get a permanent channel or node failure, then
|
// If we get a permanent channel or node failure, then
|
||||||
@ -1401,11 +1445,6 @@ sendLoop:
|
|||||||
|
|
||||||
return preImage, route, nil
|
return preImage, route, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we're unable to successfully make a payment using any of the
|
|
||||||
// routes we've found, then return an error.
|
|
||||||
return [32]byte{}, nil, fmt.Errorf("unable to route payment to "+
|
|
||||||
"destination: %v", sendError)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// applyChannelUpdate applies a channel update directly to the database,
|
// applyChannelUpdate applies a channel update directly to the database,
|
||||||
|
Loading…
Reference in New Issue
Block a user