lnrpc: extract hop unmarshall code

This commit is contained in:
Joost Jager 2018-10-29 13:07:08 +01:00
parent 68d96488cc
commit e55a678eda
No known key found for this signature in database
GPG Key ID: AE6B0D042C8E38D9

@ -3516,50 +3516,71 @@ func (r *rpcServer) marshallRoute(route *routing.Route) *lnrpc.Route {
return resp return resp
} }
func (r *rpcServer) unmarshallRoute(rpcroute *lnrpc.Route, // unmarshallHopByChannelLookup unmarshalls an rpc hop for which the pub key is
graph *channeldb.ChannelGraph) (*routing.Route, error) { // not known. This function will query the channel graph with channel id to
// retrieve both endpoints and determine the hop pubkey using the previous hop
// pubkey. If the channel is unknown, an error is returned.
func unmarshallHopByChannelLookup(graph *channeldb.ChannelGraph, hop *lnrpc.Hop,
prevPubKeyBytes []byte) (*routing.Hop, error) {
node, err := graph.SourceNode()
if err != nil {
return nil, fmt.Errorf("unable to fetch source node from graph "+
"while unmarshaling route. %v", err)
}
nodePubKeyBytes := node.PubKeyBytes[:]
hops := make([]*routing.Hop, len(rpcroute.Hops))
for i, hop := range rpcroute.Hops {
// Discard edge policies, because they may be nil. // Discard edge policies, because they may be nil.
edgeInfo, _, _, err := graph.FetchChannelEdgesByID(hop.ChanId) edgeInfo, _, _, err := graph.FetchChannelEdgesByID(hop.ChanId)
if err != nil { if err != nil {
return nil, fmt.Errorf("unable to fetch channel edges by "+ return nil, fmt.Errorf("unable to fetch channel edges by "+
"channel ID for hop (%d): %v", i, err) "channel ID %d: %v", hop.ChanId, err)
} }
var pubKeyBytes [33]byte var pubKeyBytes [33]byte
switch { switch {
case bytes.Equal(nodePubKeyBytes[:], edgeInfo.NodeKey1Bytes[:]): case bytes.Equal(prevPubKeyBytes, edgeInfo.NodeKey1Bytes[:]):
pubKeyBytes = edgeInfo.NodeKey2Bytes pubKeyBytes = edgeInfo.NodeKey2Bytes
case bytes.Equal(nodePubKeyBytes[:], edgeInfo.NodeKey2Bytes[:]): case bytes.Equal(prevPubKeyBytes, edgeInfo.NodeKey2Bytes[:]):
pubKeyBytes = edgeInfo.NodeKey1Bytes pubKeyBytes = edgeInfo.NodeKey1Bytes
default: default:
return nil, fmt.Errorf("channel edge does not match expected node") return nil, fmt.Errorf("channel edge does not match expected node")
} }
hops[i] = &routing.Hop{ return &routing.Hop{
OutgoingTimeLock: hop.Expiry, OutgoingTimeLock: hop.Expiry,
AmtToForward: lnwire.MilliSatoshi(hop.AmtToForwardMsat), AmtToForward: lnwire.MilliSatoshi(hop.AmtToForwardMsat),
PubKeyBytes: pubKeyBytes, PubKeyBytes: pubKeyBytes,
ChannelID: edgeInfo.ChannelID, ChannelID: edgeInfo.ChannelID,
}, nil
} }
nodePubKeyBytes = pubKeyBytes[:] // unmarshallRoute unmarshalls an rpc route. For hops that don't specify a
// pubkey, the channel graph is queried.
func (r *rpcServer) unmarshallRoute(rpcroute *lnrpc.Route,
graph *channeldb.ChannelGraph) (*routing.Route, error) {
sourceNode, err := graph.SourceNode()
if err != nil {
return nil, fmt.Errorf("unable to fetch source node from graph "+
"while unmarshaling route. %v", err)
}
prevNodePubKey := sourceNode.PubKeyBytes[:]
hops := make([]*routing.Hop, len(rpcroute.Hops))
for i, hop := range rpcroute.Hops {
// No hop pub key is given, so the local channel graph needs to
// be queried to complete the information necessary for routing.
routeHop, err := unmarshallHopByChannelLookup(graph,
hop, prevNodePubKey)
if err != nil {
return nil, err
}
hops[i] = routeHop
prevNodePubKey = routeHop.PubKeyBytes[:]
} }
route := routing.NewRouteFromHops( route := routing.NewRouteFromHops(
lnwire.MilliSatoshi(rpcroute.TotalAmtMsat), lnwire.MilliSatoshi(rpcroute.TotalAmtMsat),
rpcroute.TotalTimeLock, rpcroute.TotalTimeLock,
node.PubKeyBytes, sourceNode.PubKeyBytes,
hops, hops,
) )