routing: path finding test refactored

This commit is contained in:
Joost Jager 2018-06-07 11:00:58 +02:00 committed by Olaoluwa Osuntokun
parent 6c918a1806
commit e52d829168
2 changed files with 221 additions and 185 deletions

@ -558,6 +558,53 @@ func TestFindLowestFeePath(t *testing.T) {
} }
} }
type expectedHop struct {
alias string
fee lnwire.MilliSatoshi
fwdAmount lnwire.MilliSatoshi
timeLock uint32
}
type basicGraphPathFindingTestCase struct {
target string
paymentAmt btcutil.Amount
totalAmt lnwire.MilliSatoshi
totalTimeLock uint32
expectedHops []expectedHop
}
var basicGraphPathFindingTests = []basicGraphPathFindingTestCase{
// Basic route with one intermediate hop
{target: "sophon", paymentAmt: 100, totalTimeLock: 102, totalAmt: 100110,
expectedHops: []expectedHop{
{alias: "songoku", fwdAmount: 100000, fee: 110, timeLock: 101},
{alias: "sophon", fwdAmount: 100000, fee: 0, timeLock: 101},
}},
// Basic direct (one hop) route
{target: "luoji", paymentAmt: 100, totalTimeLock: 101, totalAmt: 100000,
expectedHops: []expectedHop{
{alias: "luoji", fwdAmount: 100000, fee: 0, timeLock: 101},
}},
// Three hop route where fees need to be added in to the forwarding amount.
// The high fee hop phamnewun should be avoided
{target: "elst", paymentAmt: 50000, totalTimeLock: 103, totalAmt: 50050210,
expectedHops: []expectedHop{
{alias: "songoku", fwdAmount: 50000200, fee: 50010, timeLock: 102},
{alias: "sophon", fwdAmount: 50000000, fee: 200, timeLock: 101},
{alias: "elst", fwdAmount: 50000000, fee: 0, timeLock: 101},
}},
// Three hop route where fees need to be added in to the forwarding amount.
// However this time the fwdAmount becomes too large for the roasbeef <->
// songoku channel. Then there is no other option than to choose the
// expensive phamnuwen channel. This test case was failing before
// the route search was executed backwards.
{target: "elst", paymentAmt: 100000, totalTimeLock: 103, totalAmt: 110010220,
expectedHops: []expectedHop{
{alias: "phamnuwen", fwdAmount: 100000200, fee: 10010020, timeLock: 102},
{alias: "sophon", fwdAmount: 100000000, fee: 200, timeLock: 101},
{alias: "elst", fwdAmount: 100000000, fee: 0, timeLock: 101},
}}}
func TestBasicGraphPathFinding(t *testing.T) { func TestBasicGraphPathFinding(t *testing.T) {
t.Parallel() t.Parallel()
@ -567,6 +614,24 @@ func TestBasicGraphPathFinding(t *testing.T) {
t.Fatalf("unable to create graph: %v", err) t.Fatalf("unable to create graph: %v", err)
} }
// With the test graph loaded, we'll test some basic path finding using
// the pre-generated graph. Consult the testdata/basic_graph.json file
// to follow along with the assumptions we'll use to test the path
// finding.
for _, testCase := range basicGraphPathFindingTests {
t.Run(testCase.target, func(subT *testing.T) {
testBasicGraphPathFindingCase(subT, graph, aliases, &testCase)
})
}
}
func testBasicGraphPathFindingCase(t *testing.T, graph *channeldb.ChannelGraph,
aliases aliasMap, test *basicGraphPathFindingTestCase) {
expectedHops := test.expectedHops
expectedHopCount := len(expectedHops)
sourceNode, err := graph.SourceNode() sourceNode, err := graph.SourceNode()
if err != nil { if err != nil {
t.Fatalf("unable to fetch source node: %v", err) t.Fatalf("unable to fetch source node: %v", err)
@ -576,17 +641,13 @@ func TestBasicGraphPathFinding(t *testing.T) {
ignoredEdges := make(map[uint64]struct{}) ignoredEdges := make(map[uint64]struct{})
ignoredVertexes := make(map[Vertex]struct{}) ignoredVertexes := make(map[Vertex]struct{})
// With the test graph loaded, we'll test some basic path finding using
// the pre-generated graph. Consult the testdata/basic_graph.json file
// to follow along with the assumptions we'll use to test the path
// finding.
const ( const (
startingHeight = 100 startingHeight = 100
finalHopCLTV = 1 finalHopCLTV = 1
) )
paymentAmt := lnwire.NewMSatFromSatoshis(100) paymentAmt := lnwire.NewMSatFromSatoshis(test.paymentAmt)
target := aliases["sophon"] target := aliases[test.target]
path, err := findPath( path, err := findPath(
nil, graph, nil, sourceNode, target, ignoredVertexes, nil, graph, nil, sourceNode, target, ignoredVertexes,
ignoredEdges, paymentAmt, nil, ignoredEdges, paymentAmt, nil,
@ -603,171 +664,116 @@ func TestBasicGraphPathFinding(t *testing.T) {
t.Fatalf("unable to create path: %v", err) t.Fatalf("unable to create path: %v", err)
} }
// The length of the route selected should be of exactly length two. if len(route.Hops) != len(expectedHops) {
if len(route.Hops) != 2 { t.Fatalf("route is of incorrect length, expected %v got %v",
t.Fatalf("route is of incorrect length, expected %v got %v", 2, expectedHopCount, len(route.Hops))
len(route.Hops))
} }
// As each hop only decrements a single block from the time-lock, the // Check hop nodes
// total time lock value should two more than our starting block for i := 0; i < len(expectedHops); i++ {
// height. if !bytes.Equal(route.Hops[i].Channel.Node.PubKeyBytes[:],
if route.TotalTimeLock != 102 { aliases[expectedHops[i].alias].SerializeCompressed()) {
t.Fatalf("expected time lock of %v, instead have %v", 2,
route.TotalTimeLock) t.Fatalf("%v-th hop should be %v, is instead: %v",
i, expectedHops[i], route.Hops[i].Channel.Node.Alias)
} }
// The first hop in the path should be an edge from roasbeef to goku.
if !bytes.Equal(route.Hops[0].Channel.Node.PubKeyBytes[:],
aliases["songoku"].SerializeCompressed()) {
t.Fatalf("first hop should be goku, is instead: %v",
route.Hops[0].Channel.Node.Alias)
}
// The second hop should be from goku to sophon.
if !bytes.Equal(route.Hops[1].Channel.Node.PubKeyBytes[:],
aliases["sophon"].SerializeCompressed()) {
t.Fatalf("second hop should be sophon, is instead: %v",
route.Hops[0].Channel.Node.Alias)
} }
// Next, we'll assert that the "next hop" field in each route payload // Next, we'll assert that the "next hop" field in each route payload
// properly points to the channel ID that the HTLC should be forwarded // properly points to the channel ID that the HTLC should be forwarded
// along. // along.
hopPayloads := route.ToHopPayloads() hopPayloads := route.ToHopPayloads()
if len(hopPayloads) != 2 { if len(hopPayloads) != expectedHopCount {
t.Fatalf("incorrect number of hop payloads: expected %v, got %v", t.Fatalf("incorrect number of hop payloads: expected %v, got %v",
2, len(hopPayloads)) expectedHopCount, len(hopPayloads))
} }
// The first hop should point to the second hop. // Hops should point to the next hop
for i := 0; i < len(expectedHops)-1; i++ {
var expectedHop [8]byte var expectedHop [8]byte
binary.BigEndian.PutUint64(expectedHop[:], route.Hops[1].Channel.ChannelID) binary.BigEndian.PutUint64(expectedHop[:], route.Hops[i+1].Channel.ChannelID)
if !bytes.Equal(hopPayloads[0].NextAddress[:], expectedHop[:]) { if !bytes.Equal(hopPayloads[i].NextAddress[:], expectedHop[:]) {
t.Fatalf("first hop has incorrect next hop: expected %x, got %x", t.Fatalf("first hop has incorrect next hop: expected %x, got %x",
expectedHop[:], hopPayloads[0].NextAddress) expectedHop[:], hopPayloads[i].NextAddress)
}
} }
// The second hop should have a next hop value of all zeroes in order // The final hop should have a next hop value of all zeroes in order
// to indicate it's the exit hop. // to indicate it's the exit hop.
var exitHop [8]byte var exitHop [8]byte
if !bytes.Equal(hopPayloads[1].NextAddress[:], exitHop[:]) { lastHopIndex := len(expectedHops) - 1
if !bytes.Equal(hopPayloads[lastHopIndex].NextAddress[:], exitHop[:]) {
t.Fatalf("first hop has incorrect next hop: expected %x, got %x", t.Fatalf("first hop has incorrect next hop: expected %x, got %x",
exitHop[:], hopPayloads[0].NextAddress) exitHop[:], hopPayloads[lastHopIndex].NextAddress)
} }
// We'll also assert that the outgoing CLTV value for each hop was set var expectedTotalFee lnwire.MilliSatoshi = 0
// accordingly. for i := 0; i < expectedHopCount; i++ {
if route.Hops[0].OutgoingTimeLock != 101 { // We'll ensure that the amount to forward, and fees
t.Fatalf("expected outgoing time-lock of %v, instead have %v",
1, route.Hops[0].OutgoingTimeLock)
}
if route.Hops[1].OutgoingTimeLock != 101 {
t.Fatalf("outgoing time-lock for final hop is incorrect: "+
"expected %v, got %v", 1, route.Hops[1].OutgoingTimeLock)
}
// Additionally, we'll ensure that the amount to forward, and fees
// computed for each hop are correct. // computed for each hop are correct.
firstHopFee := computeFee(
paymentAmt, route.Hops[1].Channel.ChannelEdgePolicy, if route.Hops[i].Fee != expectedHops[i].fee {
) t.Fatalf("fee incorrect for hop %v: expected %v, got %v",
if route.Hops[0].Fee != firstHopFee { i, expectedHops[i].fee, route.Hops[i].Fee)
t.Fatalf("first hop fee incorrect: expected %v, got %v",
firstHopFee, route.Hops[0].Fee)
} }
if route.TotalAmount != paymentAmt+firstHopFee { if route.Hops[i].AmtToForward != expectedHops[i].fwdAmount {
t.Fatalf("first hop forwarding amount incorrect: expected %v, got %v", t.Fatalf("forwarding amount for hop %v incorrect: "+
paymentAmt+firstHopFee, route.TotalAmount) "expected %v, got %v",
} i, expectedHops[i].fwdAmount,
if route.Hops[1].Fee != 0 { route.Hops[i].AmtToForward)
t.Fatalf("first hop fee incorrect: expected %v, got %v",
firstHopFee, 0)
} }
if route.Hops[1].AmtToForward != paymentAmt { // We'll also assert that the outgoing CLTV value for each
t.Fatalf("second hop forwarding amount incorrect: expected %v, got %v", // hop was set accordingly.
paymentAmt+firstHopFee, route.Hops[1].AmtToForward) if route.Hops[i].OutgoingTimeLock != expectedHops[i].timeLock {
t.Fatalf("outgoing time-lock for hop %v is incorrect: "+
"expected %v, got %v", i,
expectedHops[i].timeLock,
route.Hops[i].OutgoingTimeLock)
} }
// Finally, the next and prev hop maps should be properly set. expectedTotalFee += expectedHops[i].fee
//
// The previous hop from goku should be the channel from roasbeef, and
// the next hop should be the channel to sophon.
gokuPrevChan, ok := route.prevHopChannel(aliases["songoku"])
if !ok {
t.Fatalf("goku didn't have next chan but should have")
}
if gokuPrevChan.ChannelID != route.Hops[0].Channel.ChannelID {
t.Fatalf("incorrect prev chan: expected %v, got %v",
gokuPrevChan.ChannelID, route.Hops[0].Channel.ChannelID)
}
gokuNextChan, ok := route.nextHopChannel(aliases["songoku"])
if !ok {
t.Fatalf("goku didn't have prev chan but should have")
}
if gokuNextChan.ChannelID != route.Hops[1].Channel.ChannelID {
t.Fatalf("incorrect prev chan: expected %v, got %v",
gokuNextChan.ChannelID, route.Hops[1].Channel.ChannelID)
} }
// Sophon shouldn't have a next chan, but she should have a prev chan. if route.TotalAmount != test.totalAmt {
if _, ok := route.nextHopChannel(aliases["sophon"]); ok { t.Fatalf("total amount incorrect: "+
t.Fatalf("incorrect next hop map, no vertexes should " + "expected %v, got %v",
"be after sophon") test.totalAmt, route.TotalAmount)
}
sophonPrevEdge, ok := route.prevHopChannel(aliases["sophon"])
if !ok {
t.Fatalf("sophon didn't have prev chan but should have")
}
if sophonPrevEdge.ChannelID != route.Hops[1].Channel.ChannelID {
t.Fatalf("incorrect prev chan: expected %v, got %v",
sophonPrevEdge.ChannelID, route.Hops[1].Channel.ChannelID)
} }
// Next, attempt to query for a path to Luo Ji for 100 satoshis, there if route.TotalTimeLock != test.totalTimeLock {
// exist two possible paths in the graph, but the shorter (1 hop) path t.Fatalf("expected time lock of %v, instead have %v", 2,
// should be selected.
target = aliases["luoji"]
path, err = findPath(
nil, graph, nil, sourceNode, target, ignoredVertexes,
ignoredEdges, paymentAmt, nil,
)
if err != nil {
t.Fatalf("unable to find route: %v", err)
}
route, err = newRoute(
paymentAmt, noFeeLimit, sourceVertex, path, startingHeight,
finalHopCLTV,
)
if err != nil {
t.Fatalf("unable to create path: %v", err)
}
// The length of the path should be exactly one hop as it's the
// "shortest" known path in the graph.
if len(route.Hops) != 1 {
t.Fatalf("shortest path not selected, should be of length 1, "+
"is instead: %v", len(route.Hops))
}
// As we have a direct path, the total time lock value should be
// exactly the current block height plus one.
if route.TotalTimeLock != 101 {
t.Fatalf("expected time lock of %v, instead have %v", 1,
route.TotalTimeLock) route.TotalTimeLock)
} }
// Additionally, since this is a single-hop payment, we shouldn't have // The next and prev hop maps should be properly set.
// to pay any fees in total, so the total amount should be the payment for i := 0; i < expectedHopCount; i++ {
// amount. prevChan, ok := route.prevHopChannel(aliases[expectedHops[i].alias])
if route.TotalAmount != paymentAmt { if !ok {
t.Fatalf("incorrect total amount, expected %v got %v", t.Fatalf("hop didn't have prev chan but should have")
paymentAmt, route.TotalAmount) }
if prevChan.ChannelID != route.Hops[i].Channel.ChannelID {
t.Fatalf("incorrect prev chan: expected %v, got %v",
prevChan.ChannelID, route.Hops[i].Channel.ChannelID)
}
}
for i := 0; i < expectedHopCount-1; i++ {
nextChan, ok := route.nextHopChannel(aliases[expectedHops[i].alias])
if !ok {
t.Fatalf("hop didn't have prev chan but should have")
}
if nextChan.ChannelID != route.Hops[i+1].Channel.ChannelID {
t.Fatalf("incorrect prev chan: expected %v, got %v",
nextChan.ChannelID, route.Hops[i+1].Channel.ChannelID)
}
}
// Final hop shouldn't have a next chan
if _, ok := route.nextHopChannel(aliases[expectedHops[lastHopIndex].alias]); ok {
t.Fatalf("incorrect next hop map, no vertexes should " +
"be after sophon")
} }
} }
@ -1289,10 +1295,10 @@ func TestRouteFailDisabledEdge(t *testing.T) {
ignoredEdges := make(map[uint64]struct{}) ignoredEdges := make(map[uint64]struct{})
ignoredVertexes := make(map[Vertex]struct{}) ignoredVertexes := make(map[Vertex]struct{})
// First, we'll try to route from roasbeef -> songoku. This should // First, we'll try to route from roasbeef -> sophon. This should
// succeed without issue, and return a single path. // succeed without issue, and return a single path via phamnuwen
target := aliases["songoku"] target := aliases["sophon"]
payAmt := lnwire.NewMSatFromSatoshis(10000) payAmt := lnwire.NewMSatFromSatoshis(120000)
_, err = findPath( _, err = findPath(
nil, graph, nil, sourceNode, target, ignoredVertexes, nil, graph, nil, sourceNode, target, ignoredVertexes,
ignoredEdges, payAmt, nil, ignoredEdges, payAmt, nil,
@ -1301,14 +1307,14 @@ func TestRouteFailDisabledEdge(t *testing.T) {
t.Fatalf("unable to find path: %v", err) t.Fatalf("unable to find path: %v", err)
} }
// First, we'll modify the edge from roasbeef -> songoku, to read that // First, we'll modify the edge from roasbeef -> phamnuwen, to read that
// it's disabled. // it's disabled.
_, gokuEdge, _, err := graph.FetchChannelEdgesByID(12345) _, _, phamnuwenEdge, err := graph.FetchChannelEdgesByID(999991)
if err != nil { if err != nil {
t.Fatalf("unable to fetch goku's edge: %v", err) t.Fatalf("unable to fetch goku's edge: %v", err)
} }
gokuEdge.Flags = lnwire.ChanUpdateDisabled phamnuwenEdge.Flags = lnwire.ChanUpdateDisabled | lnwire.ChanUpdateDirection
if err := graph.UpdateEdgePolicy(gokuEdge); err != nil { if err := graph.UpdateEdgePolicy(phamnuwenEdge); err != nil {
t.Fatalf("unable to update edge: %v", err) t.Fatalf("unable to update edge: %v", err)
} }

@ -4,30 +4,31 @@
"", "",
" 50k satoshis ┌──────┐ ", " 50k satoshis ┌──────┐ ",
" ┌───────────────────▶│luo ji│◀─┐ ", " ┌───────────────────▶│luo ji│◀─┐ ",
" │ └──────┘ │ ", " │ └──────┘ │ ┌──────┐ ",
" │ │ ", " │ │ | elst | ",
" │ │ ", " │ │ └──────┘ ",
" │ │ ", " │ │ ",
" │ │ ", " │ │ | 100k sat ",
" │ │ ", " │ │ ",
" ▼ │ ┌──────┐ ", " ▼ │ ┌──────┐ ",
" ┌────────┐ │ │sophon│◀┐ ", " ┌────────┐ │ │sophon│◀┐ ",
" │satoshi │ │ └──────┘ │ ", " │satoshi │ │ └──────┘ │ ",
" └────────┘ │ │ ", " └────────┘ │ │ ",
" ▲ │ │ 500 satoshis ", " ▲ │ | │ 110k satoshis ",
" │ ┌───────────────────┘ │ ", " │ ┌───────────────────┘ | │ ",
" │ │ 100k satoshis │ ", " │ │ 100k satoshis | │ ",
" │ │ │ ", " │ │ | │ ",
" │ │ │ ┌────────┐ ", " │ │ 120k sat | │ ┌────────┐ ",
" └──────────┤ └─▶│son goku│ ", " └──────────┤ (hi fee) ▼ └─▶│son goku│ ",
" 10k satoshis │ └────────┘ ", " 10k satoshis │ ┌────────────┐ └────────┘ ",
" │ ▲ ", " │ | pham nuwen | ▲ ",
" │ │ ", " │ └────────────┘ │ ",
" │ │ ", " │ │ ",
" ▼ │ ", " ▼ | 120k sat (hi fee) │ ",
" ┌──────────┐ │ ", " ┌──────────┐ | │ ",
" │ roasbeef │◀────────────────────────────────────┘ ", " │ roasbeef │◀────────────────────────────────────┘ ",
" └──────────┘ 100k satoshis ", " └──────────┘ 100k satoshis ",
" the graph also includes a channel from roasbeef to sophon via pham nuwen" " the graph also includes a channel from roasbeef to sophon via pham nuwen"
], ],
"nodes": [ "nodes": [
@ -60,9 +61,38 @@
"source": false, "source": false,
"pubkey": "02a1d2856be336a58af08989aea0d8c41e072ccc392c46f8ce0e6e069f002035f3", "pubkey": "02a1d2856be336a58af08989aea0d8c41e072ccc392c46f8ce0e6e069f002035f3",
"alias": "phamnuwen" "alias": "phamnuwen"
},
{
"source": false,
"pubkey": "02a4b236b69b09b8efe6ccf822fa95ee95a0196451f4d066a450b7489e2e354a64",
"alias": "elst"
} }
], ],
"edges": [ "edges": [
{
"node_1": "02a4b236b69b09b8efe6ccf822fa95ee95a0196451f4d066a450b7489e2e354a64",
"node_2": "036264734b40c9e91d3d990a8cdfbbe23b5b0b7ad3cd0e080a25dcd05d39eeb7eb",
"channel_id": 15433,
"channel_point": "33bd5d49a50e284221561b91e781f1fca0d60341c9f9dd785b5e379a6d88af3d:0",
"flags": 1,
"expiry": 1,
"min_htlc": 1000,
"fee_base_msat": 200,
"fee_rate": 0,
"capacity": 100000
},
{
"node_1": "02a4b236b69b09b8efe6ccf822fa95ee95a0196451f4d066a450b7489e2e354a64",
"node_2": "036264734b40c9e91d3d990a8cdfbbe23b5b0b7ad3cd0e080a25dcd05d39eeb7eb",
"channel_id": 15433,
"channel_point": "33bd5d49a50e284221561b91e781f1fca0d60341c9f9dd785b5e379a6d88af3d:0",
"flags": 0,
"expiry": 1,
"min_htlc": 1000,
"fee_base_msat": 200,
"fee_rate": 0,
"capacity": 100000
},
{ {
"node_1": "02a1d2856be336a58af08989aea0d8c41e072ccc392c46f8ce0e6e069f002035f3", "node_1": "02a1d2856be336a58af08989aea0d8c41e072ccc392c46f8ce0e6e069f002035f3",
"node_2": "0367cec75158a4129177bfb8b269cb586efe93d751b43800d456485e81c2620ca6", "node_2": "0367cec75158a4129177bfb8b269cb586efe93d751b43800d456485e81c2620ca6",
@ -72,8 +102,8 @@
"expiry": 1, "expiry": 1,
"min_htlc": 1000, "min_htlc": 1000,
"fee_base_msat": 10000, "fee_base_msat": 10000,
"fee_rate": 1000000, "fee_rate": 100000,
"capacity": 100000 "capacity": 120000
}, },
{ {
"node_1": "02a1d2856be336a58af08989aea0d8c41e072ccc392c46f8ce0e6e069f002035f3", "node_1": "02a1d2856be336a58af08989aea0d8c41e072ccc392c46f8ce0e6e069f002035f3",
@ -84,8 +114,8 @@
"expiry": 1, "expiry": 1,
"min_htlc": 1000, "min_htlc": 1000,
"fee_base_msat": 10000, "fee_base_msat": 10000,
"fee_rate": 1000000, "fee_rate": 100000,
"capacity": 100000 "capacity": 120000
}, },
{ {
"node_1": "02a1d2856be336a58af08989aea0d8c41e072ccc392c46f8ce0e6e069f002035f3", "node_1": "02a1d2856be336a58af08989aea0d8c41e072ccc392c46f8ce0e6e069f002035f3",
@ -96,8 +126,8 @@
"expiry": 1, "expiry": 1,
"min_htlc": 1000, "min_htlc": 1000,
"fee_base_msat": 10000, "fee_base_msat": 10000,
"fee_rate": 1000000, "fee_rate": 100000,
"capacity": 100000 "capacity": 120000
}, },
{ {
"node_1": "02a1d2856be336a58af08989aea0d8c41e072ccc392c46f8ce0e6e069f002035f3", "node_1": "02a1d2856be336a58af08989aea0d8c41e072ccc392c46f8ce0e6e069f002035f3",
@ -108,8 +138,8 @@
"expiry": 1, "expiry": 1,
"min_htlc": 1000, "min_htlc": 1000,
"fee_base_msat": 10000, "fee_base_msat": 10000,
"fee_rate": 1000000, "fee_rate": 100000,
"capacity": 100000 "capacity": 120000
}, },
{ {
"node_1": "0367cec75158a4129177bfb8b269cb586efe93d751b43800d456485e81c2620ca6", "node_1": "0367cec75158a4129177bfb8b269cb586efe93d751b43800d456485e81c2620ca6",
@ -145,7 +175,7 @@
"min_htlc": 1, "min_htlc": 1,
"fee_base_msat": 10, "fee_base_msat": 10,
"fee_rate": 1000, "fee_rate": 1000,
"capacity": 500 "capacity": 110000
}, },
{ {
"node_1": "032b480de5d002f1a8fd1fe1bbf0a0f1b07760f65f052e66d56f15d71097c01add", "node_1": "032b480de5d002f1a8fd1fe1bbf0a0f1b07760f65f052e66d56f15d71097c01add",
@ -157,7 +187,7 @@
"min_htlc": 1, "min_htlc": 1,
"fee_base_msat": 10, "fee_base_msat": 10,
"fee_rate": 1000, "fee_rate": 1000,
"capacity": 500 "capacity": 110000
}, },
{ {
"node_1": "0367cec75158a4129177bfb8b269cb586efe93d751b43800d456485e81c2620ca6", "node_1": "0367cec75158a4129177bfb8b269cb586efe93d751b43800d456485e81c2620ca6",