rpcserver: add SendToRoute methods
rpcserver: add SendToRoute handler
This commit is contained in:
parent
dac62e812c
commit
6ddd7b4d0d
278
rpcserver.go
278
rpcserver.go
@ -231,6 +231,14 @@ var (
|
|||||||
Entity: "offchain",
|
Entity: "offchain",
|
||||||
Action: "write",
|
Action: "write",
|
||||||
}},
|
}},
|
||||||
|
"/lnrpc.Lightning/SendToRoute": {{
|
||||||
|
Entity: "offchain",
|
||||||
|
Action: "write",
|
||||||
|
}},
|
||||||
|
"/lnrpc.Lightning/SendToRouteSync": {{
|
||||||
|
Entity: "offchain",
|
||||||
|
Action: "write",
|
||||||
|
}},
|
||||||
"/lnrpc.Lightning/AddInvoice": {{
|
"/lnrpc.Lightning/AddInvoice": {{
|
||||||
Entity: "invoices",
|
Entity: "invoices",
|
||||||
Action: "write",
|
Action: "write",
|
||||||
@ -1740,11 +1748,82 @@ func validatePayReqExpiry(payReq *zpay32.Invoice) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// paymentStream enables different types of payment stream, such as:
|
||||||
|
// lnrpc.Lightning_SendPaymentServer
|
||||||
|
// lnrpc.Lightning_SendToRouteServer
|
||||||
|
// to execute sendPayment.
|
||||||
|
type paymentStream struct {
|
||||||
|
recv func() (*paymentRequest, error)
|
||||||
|
send func(*lnrpc.SendResponse) error
|
||||||
|
}
|
||||||
|
|
||||||
|
// paymentRequest wraps lnrpc.SendRequest so that routes from
|
||||||
|
// lnrpc.SendToRouteRequest can be passed to sendPayment
|
||||||
|
type paymentRequest struct {
|
||||||
|
*lnrpc.SendRequest
|
||||||
|
routes []*routing.Route
|
||||||
|
}
|
||||||
|
|
||||||
// SendPayment dispatches a bi-directional streaming RPC for sending payments
|
// SendPayment dispatches a bi-directional streaming RPC for sending payments
|
||||||
// through the Lightning Network. A single RPC invocation creates a persistent
|
// through the Lightning Network. A single RPC invocation creates a persistent
|
||||||
// bi-directional stream allowing clients to rapidly send payments through the
|
// bi-directional stream allowing clients to rapidly send payments through the
|
||||||
// Lightning Network with a single persistent connection.
|
// Lightning Network with a single persistent connection.
|
||||||
func (r *rpcServer) SendPayment(paymentStream lnrpc.Lightning_SendPaymentServer) error {
|
func (r *rpcServer) SendPayment(stream lnrpc.Lightning_SendPaymentServer) error {
|
||||||
|
return r.sendPayment(&paymentStream{
|
||||||
|
recv: func() (*paymentRequest, error) {
|
||||||
|
req, err := stream.Recv()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return &paymentRequest{
|
||||||
|
SendRequest: req,
|
||||||
|
}, nil
|
||||||
|
},
|
||||||
|
send: stream.Send,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// SendToRoute dispatches a bi-directional streaming RPC for sending payments
|
||||||
|
// through the Lightning Network via predefined routes passed in. A single RPC
|
||||||
|
// invocation creates a persistent bi-directional stream allowing clients to
|
||||||
|
// rapidly send payments through the Lightning Network with a single
|
||||||
|
// persistent connection.
|
||||||
|
func (r *rpcServer) SendToRoute(stream lnrpc.Lightning_SendToRouteServer) error {
|
||||||
|
return r.sendPayment(&paymentStream{
|
||||||
|
recv: func() (*paymentRequest, error) {
|
||||||
|
req, err := stream.Recv()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
graph := r.server.chanDB.ChannelGraph()
|
||||||
|
|
||||||
|
if len(req.Routes) == 0 {
|
||||||
|
return nil, fmt.Errorf("unable to send, no routes provided")
|
||||||
|
}
|
||||||
|
|
||||||
|
routes := make([]*routing.Route, len(req.Routes))
|
||||||
|
for i, rpcroute := range req.Routes {
|
||||||
|
route, err := unmarshallRoute(rpcroute, graph)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
routes[i] = route
|
||||||
|
}
|
||||||
|
|
||||||
|
return &paymentRequest{
|
||||||
|
SendRequest: &lnrpc.SendRequest{
|
||||||
|
PaymentHash: req.PaymentHash,
|
||||||
|
},
|
||||||
|
routes: routes,
|
||||||
|
}, nil
|
||||||
|
},
|
||||||
|
send: stream.Send,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *rpcServer) sendPayment(stream *paymentStream) error {
|
||||||
// For each payment we need to know the msat amount, the destination
|
// For each payment we need to know the msat amount, the destination
|
||||||
// public key, the payment hash, and the optional route hints.
|
// public key, the payment hash, and the optional route hints.
|
||||||
type payment struct {
|
type payment struct {
|
||||||
@ -1753,6 +1832,7 @@ func (r *rpcServer) SendPayment(paymentStream lnrpc.Lightning_SendPaymentServer)
|
|||||||
pHash []byte
|
pHash []byte
|
||||||
cltvDelta uint16
|
cltvDelta uint16
|
||||||
routeHints [][]routing.HopHint
|
routeHints [][]routing.HopHint
|
||||||
|
routes []*routing.Route
|
||||||
}
|
}
|
||||||
payChan := make(chan *payment)
|
payChan := make(chan *payment)
|
||||||
errChan := make(chan error, 1)
|
errChan := make(chan error, 1)
|
||||||
@ -1799,7 +1879,7 @@ func (r *rpcServer) SendPayment(paymentStream lnrpc.Lightning_SendPaymentServer)
|
|||||||
// stream sent by the client. If we read the
|
// stream sent by the client. If we read the
|
||||||
// EOF sentinel, then the client has closed the
|
// EOF sentinel, then the client has closed the
|
||||||
// stream, and we can exit normally.
|
// stream, and we can exit normally.
|
||||||
nextPayment, err := paymentStream.Recv()
|
nextPayment, err := stream.recv()
|
||||||
if err == io.EOF {
|
if err == io.EOF {
|
||||||
errChan <- nil
|
errChan <- nil
|
||||||
return
|
return
|
||||||
@ -1817,6 +1897,7 @@ func (r *rpcServer) SendPayment(paymentStream lnrpc.Lightning_SendPaymentServer)
|
|||||||
// fields.
|
// fields.
|
||||||
p := &payment{}
|
p := &payment{}
|
||||||
|
|
||||||
|
if len(nextPayment.routes) == 0 {
|
||||||
// If the payment request field isn't blank,
|
// If the payment request field isn't blank,
|
||||||
// then the details of the invoice are encoded
|
// then the details of the invoice are encoded
|
||||||
// entirely within the encoded payReq. So we'll
|
// entirely within the encoded payReq. So we'll
|
||||||
@ -1876,6 +1957,10 @@ func (r *rpcServer) SendPayment(paymentStream lnrpc.Lightning_SendPaymentServer)
|
|||||||
p.pHash = nextPayment.PaymentHash
|
p.pHash = nextPayment.PaymentHash
|
||||||
p.cltvDelta = uint16(nextPayment.FinalCltvDelta)
|
p.cltvDelta = uint16(nextPayment.FinalCltvDelta)
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
p.pHash = nextPayment.PaymentHash
|
||||||
|
p.routes = nextPayment.routes
|
||||||
|
}
|
||||||
|
|
||||||
select {
|
select {
|
||||||
case payChan <- p:
|
case payChan <- p:
|
||||||
@ -1902,7 +1987,7 @@ func (r *rpcServer) SendPayment(paymentStream lnrpc.Lightning_SendPaymentServer)
|
|||||||
"large, max payment allowed is %v",
|
"large, max payment allowed is %v",
|
||||||
p.msat, maxPaymentMSat)
|
p.msat, maxPaymentMSat)
|
||||||
|
|
||||||
if err := paymentStream.Send(&lnrpc.SendResponse{
|
if err := stream.send(&lnrpc.SendResponse{
|
||||||
PaymentError: pErr.Error(),
|
PaymentError: pErr.Error(),
|
||||||
}); err != nil {
|
}); err != nil {
|
||||||
return err
|
return err
|
||||||
@ -1912,9 +1997,13 @@ func (r *rpcServer) SendPayment(paymentStream lnrpc.Lightning_SendPaymentServer)
|
|||||||
|
|
||||||
// Parse the details of the payment which include the
|
// Parse the details of the payment which include the
|
||||||
// pubkey of the destination and the payment amount.
|
// pubkey of the destination and the payment amount.
|
||||||
destNode, err := btcec.ParsePubKey(p.dest, btcec.S256())
|
var destNode *btcec.PublicKey
|
||||||
if err != nil {
|
var pubKeyErr error
|
||||||
return err
|
if len(p.routes) == 0 {
|
||||||
|
destNode, pubKeyErr = btcec.ParsePubKey(p.dest, btcec.S256())
|
||||||
|
if pubKeyErr != nil {
|
||||||
|
return pubKeyErr
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we're in debug HTLC mode, then all outgoing HTLCs
|
// If we're in debug HTLC mode, then all outgoing HTLCs
|
||||||
@ -1944,6 +2033,13 @@ func (r *rpcServer) SendPayment(paymentStream lnrpc.Lightning_SendPaymentServer)
|
|||||||
// successful, the route chosen will be
|
// successful, the route chosen will be
|
||||||
// returned. Otherwise, we'll get a non-nil
|
// returned. Otherwise, we'll get a non-nil
|
||||||
// error.
|
// error.
|
||||||
|
var (
|
||||||
|
preImage [32]byte
|
||||||
|
route *routing.Route
|
||||||
|
routerErr error
|
||||||
|
)
|
||||||
|
|
||||||
|
if len(p.routes) == 0 {
|
||||||
payment := &routing.LightningPayment{
|
payment := &routing.LightningPayment{
|
||||||
Target: destNode,
|
Target: destNode,
|
||||||
Amount: p.msat,
|
Amount: p.msat,
|
||||||
@ -1953,13 +2049,21 @@ func (r *rpcServer) SendPayment(paymentStream lnrpc.Lightning_SendPaymentServer)
|
|||||||
if p.cltvDelta != 0 {
|
if p.cltvDelta != 0 {
|
||||||
payment.FinalCLTVDelta = &p.cltvDelta
|
payment.FinalCLTVDelta = &p.cltvDelta
|
||||||
}
|
}
|
||||||
preImage, route, err := r.server.chanRouter.SendPayment(payment)
|
preImage, route, routerErr = r.server.chanRouter.SendPayment(payment)
|
||||||
if err != nil {
|
} else {
|
||||||
|
payment := &routing.LightningPayment{
|
||||||
|
PaymentHash: rHash,
|
||||||
|
}
|
||||||
|
|
||||||
|
preImage, route, routerErr = r.server.chanRouter.SendToRoute(p.routes, payment)
|
||||||
|
}
|
||||||
|
|
||||||
|
if routerErr != nil {
|
||||||
// If we receive payment error than,
|
// If we receive payment error than,
|
||||||
// instead of terminating the stream,
|
// instead of terminating the stream,
|
||||||
// send error response to the user.
|
// send error response to the user.
|
||||||
err := paymentStream.Send(&lnrpc.SendResponse{
|
err := stream.send(&lnrpc.SendResponse{
|
||||||
PaymentError: err.Error(),
|
PaymentError: routerErr.Error(),
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
errChan <- err
|
errChan <- err
|
||||||
@ -1967,14 +2071,23 @@ func (r *rpcServer) SendPayment(paymentStream lnrpc.Lightning_SendPaymentServer)
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Calculate the original amount of funds which was sent through routes
|
||||||
|
// without total fee.
|
||||||
|
var amt lnwire.MilliSatoshi
|
||||||
|
if len(p.routes) > 0 {
|
||||||
|
amt = route.TotalAmount - route.TotalFees
|
||||||
|
} else {
|
||||||
|
amt = p.msat
|
||||||
|
}
|
||||||
|
|
||||||
// Save the completed payment to the database
|
// Save the completed payment to the database
|
||||||
// for record keeping purposes.
|
// for record keeping purposes.
|
||||||
if err := r.savePayment(route, p.msat, preImage[:]); err != nil {
|
if err := r.savePayment(route, amt, preImage[:]); err != nil {
|
||||||
errChan <- err
|
errChan <- err
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
err = paymentStream.Send(&lnrpc.SendResponse{
|
err := stream.send(&lnrpc.SendResponse{
|
||||||
PaymentPreimage: preImage[:],
|
PaymentPreimage: preImage[:],
|
||||||
PaymentRoute: marshallRoute(route),
|
PaymentRoute: marshallRoute(route),
|
||||||
})
|
})
|
||||||
@ -1994,6 +2107,44 @@ func (r *rpcServer) SendPayment(paymentStream lnrpc.Lightning_SendPaymentServer)
|
|||||||
func (r *rpcServer) SendPaymentSync(ctx context.Context,
|
func (r *rpcServer) SendPaymentSync(ctx context.Context,
|
||||||
nextPayment *lnrpc.SendRequest) (*lnrpc.SendResponse, error) {
|
nextPayment *lnrpc.SendRequest) (*lnrpc.SendResponse, error) {
|
||||||
|
|
||||||
|
return r.sendPaymentSync(ctx, &paymentRequest{
|
||||||
|
SendRequest: nextPayment,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// SendToRouteSync is the synchronous non-streaming version of SendToRoute.
|
||||||
|
// This RPC is intended to be consumed by clients of the REST proxy.
|
||||||
|
// Additionally, this RPC expects the payment hash (if any) to be encoded
|
||||||
|
// as hex strings.
|
||||||
|
func (r *rpcServer) SendToRouteSync(ctx context.Context,
|
||||||
|
req *lnrpc.SendToRouteRequest) (*lnrpc.SendResponse, error) {
|
||||||
|
|
||||||
|
if len(req.Routes) == 0 {
|
||||||
|
return nil, fmt.Errorf("unable to send, no routes provided")
|
||||||
|
}
|
||||||
|
|
||||||
|
graph := r.server.chanDB.ChannelGraph()
|
||||||
|
|
||||||
|
routes := make([]*routing.Route, len(req.Routes))
|
||||||
|
for i, route := range req.Routes {
|
||||||
|
route, err := unmarshallRoute(route, graph)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
routes[i] = route
|
||||||
|
}
|
||||||
|
|
||||||
|
return r.sendPaymentSync(ctx, &paymentRequest{
|
||||||
|
SendRequest: &lnrpc.SendRequest{
|
||||||
|
PaymentHashString: req.PaymentHashString,
|
||||||
|
},
|
||||||
|
routes: routes,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *rpcServer) sendPaymentSync(ctx context.Context,
|
||||||
|
nextPayment *paymentRequest) (*lnrpc.SendResponse, error) {
|
||||||
|
|
||||||
// TODO(roasbeef): enforce fee limits, pass into router, ditch if exceed limit
|
// TODO(roasbeef): enforce fee limits, pass into router, ditch if exceed limit
|
||||||
// * limit either a %, or absolute, or iff more than sending
|
// * limit either a %, or absolute, or iff more than sending
|
||||||
|
|
||||||
@ -2013,6 +2164,7 @@ func (r *rpcServer) SendPaymentSync(ctx context.Context,
|
|||||||
routeHints [][]routing.HopHint
|
routeHints [][]routing.HopHint
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if len(nextPayment.routes) == 0 {
|
||||||
// If the proto request has an encoded payment request, then we we'll
|
// If the proto request has an encoded payment request, then we we'll
|
||||||
// use that solely to dispatch the payment.
|
// use that solely to dispatch the payment.
|
||||||
if nextPayment.PaymentRequest != "" {
|
if nextPayment.PaymentRequest != "" {
|
||||||
@ -2075,6 +2227,21 @@ func (r *rpcServer) SendPaymentSync(ctx context.Context,
|
|||||||
btcutil.Amount(nextPayment.Amt),
|
btcutil.Amount(nextPayment.Amt),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
// If we're in debug HTLC mode, then all outgoing HTLCs will
|
||||||
|
// pay to the same debug rHash. Otherwise, we pay to the rHash
|
||||||
|
// specified within the RPC request.
|
||||||
|
if cfg.DebugHTLC && nextPayment.PaymentHashString == "" {
|
||||||
|
rHash = debugHash
|
||||||
|
} else {
|
||||||
|
paymentHash, err := hex.DecodeString(nextPayment.PaymentHashString)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
copy(rHash[:], paymentHash)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Currently, within the bootstrap phase of the network, we limit the
|
// Currently, within the bootstrap phase of the network, we limit the
|
||||||
// largest payment size allotted to (2^32) - 1 mSAT or 4.29 million
|
// largest payment size allotted to (2^32) - 1 mSAT or 4.29 million
|
||||||
@ -2090,6 +2257,13 @@ func (r *rpcServer) SendPaymentSync(ctx context.Context,
|
|||||||
// Finally, send a payment request to the channel router. If the
|
// Finally, send a payment request to the channel router. If the
|
||||||
// payment succeeds, then the returned route will be that was used
|
// payment succeeds, then the returned route will be that was used
|
||||||
// successfully within the payment.
|
// successfully within the payment.
|
||||||
|
var (
|
||||||
|
preImage [32]byte
|
||||||
|
route *routing.Route
|
||||||
|
routerErr error
|
||||||
|
)
|
||||||
|
|
||||||
|
if len(nextPayment.routes) == 0 {
|
||||||
payment := &routing.LightningPayment{
|
payment := &routing.LightningPayment{
|
||||||
Target: destPub,
|
Target: destPub,
|
||||||
Amount: amtMSat,
|
Amount: amtMSat,
|
||||||
@ -2099,16 +2273,33 @@ func (r *rpcServer) SendPaymentSync(ctx context.Context,
|
|||||||
if cltvDelta != 0 {
|
if cltvDelta != 0 {
|
||||||
payment.FinalCLTVDelta = &cltvDelta
|
payment.FinalCLTVDelta = &cltvDelta
|
||||||
}
|
}
|
||||||
preImage, route, err := r.server.chanRouter.SendPayment(payment)
|
preImage, route, routerErr = r.server.chanRouter.SendPayment(payment)
|
||||||
if err != nil {
|
} else {
|
||||||
|
payment := &routing.LightningPayment{
|
||||||
|
PaymentHash: rHash,
|
||||||
|
}
|
||||||
|
preImage, route, routerErr = r.server.chanRouter.SendToRoute(
|
||||||
|
nextPayment.routes, payment)
|
||||||
|
}
|
||||||
|
|
||||||
|
if routerErr != nil {
|
||||||
return &lnrpc.SendResponse{
|
return &lnrpc.SendResponse{
|
||||||
PaymentError: err.Error(),
|
PaymentError: routerErr.Error(),
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Calculate the original amount of funds which was sent through routes
|
||||||
|
// without total fee.
|
||||||
|
var amt lnwire.MilliSatoshi
|
||||||
|
if len(nextPayment.routes) > 0 {
|
||||||
|
amt = route.TotalAmount - route.TotalFees
|
||||||
|
} else {
|
||||||
|
amt = amtMSat
|
||||||
|
}
|
||||||
|
|
||||||
// With the payment completed successfully, we now ave the details of
|
// With the payment completed successfully, we now ave the details of
|
||||||
// the completed payment to the database for historical record keeping.
|
// the completed payment to the database for historical record keeping.
|
||||||
if err := r.savePayment(route, amtMSat, preImage[:]); err != nil {
|
if err := r.savePayment(route, amt, preImage[:]); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2949,6 +3140,61 @@ func marshallRoute(route *routing.Route) *lnrpc.Route {
|
|||||||
return resp
|
return resp
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func unmarshallRoute(rpcroute *lnrpc.Route,
|
||||||
|
graph *channeldb.ChannelGraph) (*routing.Route, error) {
|
||||||
|
|
||||||
|
rpcsLog.Infof("rpcroute: %v", spew.Sdump(rpcroute))
|
||||||
|
|
||||||
|
route := &routing.Route{
|
||||||
|
TotalTimeLock: rpcroute.TotalTimeLock,
|
||||||
|
TotalFees: lnwire.MilliSatoshi(rpcroute.TotalFeesMsat),
|
||||||
|
TotalAmount: lnwire.MilliSatoshi(rpcroute.TotalAmtMsat),
|
||||||
|
Hops: make([]*routing.Hop, len(rpcroute.Hops)),
|
||||||
|
}
|
||||||
|
|
||||||
|
node, err := graph.SourceNode()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("unable to fetch source node from graph "+
|
||||||
|
"while unmarshaling route. %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, hop := range rpcroute.Hops {
|
||||||
|
edgeInfo, c1, c2, err := graph.FetchChannelEdgesByID(hop.ChanId)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("unable to fetch channel edges by "+
|
||||||
|
"channel ID for hop (%d): %v", i, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var channelEdgePolicy *channeldb.ChannelEdgePolicy
|
||||||
|
|
||||||
|
switch {
|
||||||
|
case bytes.Equal(node.PubKeyBytes[:], c1.Node.PubKeyBytes[:]):
|
||||||
|
channelEdgePolicy = c2
|
||||||
|
node = c2.Node
|
||||||
|
case bytes.Equal(node.PubKeyBytes[:], c2.Node.PubKeyBytes[:]):
|
||||||
|
channelEdgePolicy = c1
|
||||||
|
node = c1.Node
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("could not find channel edge for hop=%d", i)
|
||||||
|
}
|
||||||
|
|
||||||
|
routingHop := &routing.ChannelHop{
|
||||||
|
ChannelEdgePolicy: channelEdgePolicy,
|
||||||
|
Capacity: btcutil.Amount(hop.ChanCapacity),
|
||||||
|
Chain: edgeInfo.ChainHash,
|
||||||
|
}
|
||||||
|
|
||||||
|
route.Hops[i] = &routing.Hop{
|
||||||
|
Channel: routingHop,
|
||||||
|
OutgoingTimeLock: hop.Expiry,
|
||||||
|
AmtToForward: lnwire.MilliSatoshi(hop.AmtToForwardMsat),
|
||||||
|
Fee: lnwire.MilliSatoshi(hop.FeeMsat),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return route, nil
|
||||||
|
}
|
||||||
|
|
||||||
// GetNetworkInfo returns some basic stats about the known channel graph from
|
// GetNetworkInfo returns some basic stats about the known channel graph from
|
||||||
// the PoV of the node.
|
// the PoV of the node.
|
||||||
func (r *rpcServer) GetNetworkInfo(ctx context.Context,
|
func (r *rpcServer) GetNetworkInfo(ctx context.Context,
|
||||||
|
Loading…
Reference in New Issue
Block a user