2019-06-18 18:30:56 +02:00
|
|
|
package routing
|
|
|
|
|
|
|
|
import (
|
2022-02-23 14:48:00 +01:00
|
|
|
"github.com/btcsuite/btcd/btcec/v2"
|
2019-06-18 18:30:56 +02:00
|
|
|
"github.com/lightningnetwork/lnd/channeldb"
|
2023-11-08 10:18:45 +01:00
|
|
|
"github.com/lightningnetwork/lnd/channeldb/models"
|
2024-04-19 13:49:45 +02:00
|
|
|
"github.com/lightningnetwork/lnd/fn"
|
2019-06-18 18:30:56 +02:00
|
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
|
|
|
"github.com/lightningnetwork/lnd/routing/route"
|
2024-04-19 13:49:45 +02:00
|
|
|
"github.com/lightningnetwork/lnd/tlv"
|
2019-06-18 18:30:56 +02:00
|
|
|
"github.com/lightningnetwork/lnd/zpay32"
|
|
|
|
)
|
|
|
|
|
2024-08-13 18:52:16 +02:00
|
|
|
// A compile time assertion to ensure SessionSource meets the
|
2019-06-18 18:30:56 +02:00
|
|
|
// PaymentSessionSource interface.
|
|
|
|
var _ PaymentSessionSource = (*SessionSource)(nil)
|
|
|
|
|
|
|
|
// SessionSource defines a source for the router to retrieve new payment
|
|
|
|
// sessions.
|
|
|
|
type SessionSource struct {
|
2024-06-26 04:58:57 +02:00
|
|
|
// GraphSessionFactory can be used to gain access to a Graph session.
|
|
|
|
// If the backing DB allows it, this will mean that a read transaction
|
|
|
|
// is being held during the use of the session.
|
|
|
|
GraphSessionFactory GraphSessionFactory
|
2021-10-21 13:55:22 +02:00
|
|
|
|
|
|
|
// SourceNode is the graph's source node.
|
|
|
|
SourceNode *channeldb.LightningNode
|
2019-06-18 18:30:56 +02:00
|
|
|
|
2021-10-19 09:37:44 +02:00
|
|
|
// GetLink is a method that allows querying the lower link layer
|
2019-06-18 18:30:56 +02:00
|
|
|
// to determine the up to date available bandwidth at a prospective link
|
|
|
|
// to be traversed. If the link isn't available, then a value of zero
|
|
|
|
// should be returned. Otherwise, the current up to date knowledge of
|
|
|
|
// the available bandwidth of the link should be returned.
|
2021-10-19 09:37:44 +02:00
|
|
|
GetLink getLinkQuery
|
2019-06-18 18:30:56 +02:00
|
|
|
|
|
|
|
// 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 payment 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.
|
2024-08-13 18:52:16 +02:00
|
|
|
MissionControl MissionControlQuerier
|
2019-06-18 18:30:56 +02:00
|
|
|
|
2019-06-20 12:03:45 +02:00
|
|
|
// PathFindingConfig defines global parameters that control the
|
2023-09-20 17:37:32 +02:00
|
|
|
// trade-off in path finding between fees and probability.
|
2019-06-20 12:03:45 +02:00
|
|
|
PathFindingConfig PathFindingConfig
|
2019-06-18 18:30:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewPaymentSession creates a new payment session backed by the latest prune
|
|
|
|
// view from Mission Control. An optional set of routing hints can be provided
|
|
|
|
// in order to populate additional edges to explore when finding a path to the
|
|
|
|
// payment's destination.
|
2024-04-19 13:49:45 +02:00
|
|
|
func (m *SessionSource) NewPaymentSession(p *LightningPayment,
|
|
|
|
firstHopBlob fn.Option[tlv.Blob],
|
|
|
|
trafficShaper fn.Option[TlvTrafficShaper]) (PaymentSession, error) {
|
2019-06-18 18:30:56 +02:00
|
|
|
|
2024-06-26 04:22:00 +02:00
|
|
|
getBandwidthHints := func(graph Graph) (bandwidthHints, error) {
|
2021-10-21 13:55:22 +02:00
|
|
|
return newBandwidthManager(
|
|
|
|
graph, m.SourceNode.PubKeyBytes, m.GetLink,
|
2024-04-19 13:49:45 +02:00
|
|
|
firstHopBlob, trafficShaper,
|
2021-10-21 13:55:22 +02:00
|
|
|
)
|
2020-01-14 11:19:52 +01:00
|
|
|
}
|
|
|
|
|
2020-04-16 15:20:23 +02:00
|
|
|
session, err := newPaymentSession(
|
2024-06-15 00:47:15 +02:00
|
|
|
p, m.SourceNode.PubKeyBytes, getBandwidthHints,
|
2024-06-26 04:58:57 +02:00
|
|
|
m.GraphSessionFactory, m.MissionControl, m.PathFindingConfig,
|
2020-04-16 15:20:23 +02:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return session, nil
|
2020-01-14 11:19:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewPaymentSessionEmpty creates a new paymentSession instance that is empty,
|
|
|
|
// and will be exhausted immediately. Used for failure reporting to
|
|
|
|
// missioncontrol for resumed payment we don't want to make more attempts for.
|
|
|
|
func (m *SessionSource) NewPaymentSessionEmpty() PaymentSession {
|
|
|
|
return &paymentSession{
|
2020-03-17 11:32:07 +01:00
|
|
|
empty: true,
|
2020-01-14 11:19:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// RouteHintsToEdges converts a list of invoice route hints to an edge map that
|
|
|
|
// can be passed into pathfinding.
|
|
|
|
func RouteHintsToEdges(routeHints [][]zpay32.HopHint, target route.Vertex) (
|
2023-11-20 17:54:37 +01:00
|
|
|
map[route.Vertex][]AdditionalEdge, error) {
|
2020-01-14 11:19:52 +01:00
|
|
|
|
2023-11-20 17:54:37 +01:00
|
|
|
edges := make(map[route.Vertex][]AdditionalEdge)
|
2019-06-18 18:30:56 +02:00
|
|
|
|
|
|
|
// Traverse through all of the available hop hints and include them in
|
|
|
|
// our edges map, indexed by the public key of the channel's starting
|
|
|
|
// node.
|
|
|
|
for _, routeHint := range routeHints {
|
|
|
|
// If multiple hop hints are provided within a single route
|
|
|
|
// hint, we'll assume they must be chained together and sorted
|
|
|
|
// in forward order in order to reach the target successfully.
|
|
|
|
for i, hopHint := range routeHint {
|
|
|
|
// In order to determine the end node of this hint,
|
|
|
|
// we'll need to look at the next hint's start node. If
|
|
|
|
// we've reached the end of the hints list, we can
|
|
|
|
// assume we've reached the destination.
|
|
|
|
endNode := &channeldb.LightningNode{}
|
|
|
|
if i != len(routeHint)-1 {
|
|
|
|
endNode.AddPubKey(routeHint[i+1].NodeID)
|
|
|
|
} else {
|
|
|
|
targetPubKey, err := btcec.ParsePubKey(
|
2022-02-23 14:48:00 +01:00
|
|
|
target[:],
|
2019-06-18 18:30:56 +02:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
endNode.AddPubKey(targetPubKey)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Finally, create the channel edge from the hop hint
|
|
|
|
// and add it to list of edges corresponding to the node
|
|
|
|
// at the start of the channel.
|
2023-11-20 17:54:37 +01:00
|
|
|
edgePolicy := &models.CachedEdgePolicy{
|
2021-09-21 19:18:22 +02:00
|
|
|
ToNodePubKey: func() route.Vertex {
|
|
|
|
return endNode.PubKeyBytes
|
|
|
|
},
|
|
|
|
ToNodeFeatures: lnwire.EmptyFeatureVector(),
|
|
|
|
ChannelID: hopHint.ChannelID,
|
2019-06-18 18:30:56 +02:00
|
|
|
FeeBaseMSat: lnwire.MilliSatoshi(
|
|
|
|
hopHint.FeeBaseMSat,
|
|
|
|
),
|
|
|
|
FeeProportionalMillionths: lnwire.MilliSatoshi(
|
|
|
|
hopHint.FeeProportionalMillionths,
|
|
|
|
),
|
|
|
|
TimeLockDelta: hopHint.CLTVExpiryDelta,
|
|
|
|
}
|
|
|
|
|
2023-11-20 17:54:37 +01:00
|
|
|
edge := &PrivateEdge{
|
|
|
|
policy: edgePolicy,
|
|
|
|
}
|
|
|
|
|
2019-06-18 18:30:56 +02:00
|
|
|
v := route.NewVertex(hopHint.NodeID)
|
|
|
|
edges[v] = append(edges[v], edge)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-14 11:19:52 +01:00
|
|
|
return edges, nil
|
2019-06-18 18:30:56 +02:00
|
|
|
}
|