2023-05-31 21:28:09 +02:00
|
|
|
package routing
|
|
|
|
|
|
|
|
import (
|
2023-11-20 17:54:37 +01:00
|
|
|
"bytes"
|
2023-05-31 21:28:09 +02:00
|
|
|
"testing"
|
|
|
|
|
2022-12-15 22:48:56 +01:00
|
|
|
"github.com/btcsuite/btcd/btcec/v2"
|
2023-05-31 21:28:09 +02:00
|
|
|
sphinx "github.com/lightningnetwork/lightning-onion"
|
2024-05-15 15:52:17 +02:00
|
|
|
"github.com/lightningnetwork/lnd/fn"
|
2024-10-22 12:57:00 +02:00
|
|
|
"github.com/lightningnetwork/lnd/graph/db/models"
|
2022-12-15 22:48:56 +01:00
|
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
|
|
|
"github.com/lightningnetwork/lnd/routing/route"
|
2023-05-31 21:28:09 +02:00
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
// TestBlindedPathValidation tests validation of blinded paths.
|
|
|
|
func TestBlindedPathValidation(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
payment *BlindedPayment
|
|
|
|
err error
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "no path",
|
|
|
|
payment: &BlindedPayment{},
|
|
|
|
err: ErrNoBlindedPath,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "insufficient hops",
|
|
|
|
payment: &BlindedPayment{
|
|
|
|
BlindedPath: &sphinx.BlindedPath{
|
|
|
|
BlindedHops: []*sphinx.BlindedHopInfo{},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
err: ErrInsufficientBlindedHops,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "maximum < minimum",
|
|
|
|
payment: &BlindedPayment{
|
|
|
|
BlindedPath: &sphinx.BlindedPath{
|
|
|
|
BlindedHops: []*sphinx.BlindedHopInfo{
|
|
|
|
{},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
HtlcMaximum: 10,
|
|
|
|
HtlcMinimum: 20,
|
|
|
|
},
|
|
|
|
err: ErrHTLCRestrictions,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "valid",
|
|
|
|
payment: &BlindedPayment{
|
|
|
|
BlindedPath: &sphinx.BlindedPath{
|
|
|
|
BlindedHops: []*sphinx.BlindedHopInfo{
|
|
|
|
{},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
HtlcMaximum: 15,
|
|
|
|
HtlcMinimum: 5,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, testCase := range tests {
|
|
|
|
testCase := testCase
|
|
|
|
|
|
|
|
t.Run(testCase.name, func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
err := testCase.payment.Validate()
|
|
|
|
require.ErrorIs(t, err, testCase.err)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2022-12-15 22:48:56 +01:00
|
|
|
|
|
|
|
// TestBlindedPaymentToHints tests conversion of a blinded path to a chain of
|
|
|
|
// route hints. As our function assumes that the blinded payment has already
|
|
|
|
// been validated.
|
|
|
|
func TestBlindedPaymentToHints(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
var (
|
|
|
|
_, pk1 = btcec.PrivKeyFromBytes([]byte{1})
|
|
|
|
_, pkb1 = btcec.PrivKeyFromBytes([]byte{2})
|
|
|
|
_, pkb2 = btcec.PrivKeyFromBytes([]byte{3})
|
|
|
|
_, pkb3 = btcec.PrivKeyFromBytes([]byte{4})
|
|
|
|
|
|
|
|
v1 = route.NewVertex(pk1)
|
|
|
|
vb2 = route.NewVertex(pkb2)
|
|
|
|
vb3 = route.NewVertex(pkb3)
|
|
|
|
|
|
|
|
baseFee uint32 = 1000
|
|
|
|
ppmFee uint32 = 500
|
|
|
|
cltvDelta uint16 = 140
|
|
|
|
htlcMin uint64 = 100
|
|
|
|
htlcMax uint64 = 100_000_000
|
|
|
|
|
2023-11-20 17:54:37 +01:00
|
|
|
sizeEncryptedData = 100
|
|
|
|
cipherText = bytes.Repeat(
|
|
|
|
[]byte{1}, sizeEncryptedData,
|
|
|
|
)
|
|
|
|
_, blindedPoint = btcec.PrivKeyFromBytes([]byte{5})
|
|
|
|
|
2022-12-15 22:48:56 +01:00
|
|
|
rawFeatures = lnwire.NewRawFeatureVector(
|
|
|
|
lnwire.AMPOptional,
|
|
|
|
)
|
|
|
|
|
|
|
|
features = lnwire.NewFeatureVector(
|
|
|
|
rawFeatures, lnwire.Features,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
// Create a blinded payment that's just to the introduction node and
|
|
|
|
// assert that we get nil hints.
|
|
|
|
blindedPayment := &BlindedPayment{
|
|
|
|
BlindedPath: &sphinx.BlindedPath{
|
|
|
|
IntroductionPoint: pk1,
|
2023-11-20 17:54:37 +01:00
|
|
|
BlindingPoint: blindedPoint,
|
2022-12-15 22:48:56 +01:00
|
|
|
BlindedHops: []*sphinx.BlindedHopInfo{
|
|
|
|
{},
|
|
|
|
},
|
|
|
|
},
|
2024-02-15 20:13:51 +01:00
|
|
|
BaseFee: baseFee,
|
|
|
|
ProportionalFeeRate: ppmFee,
|
|
|
|
CltvExpiryDelta: cltvDelta,
|
|
|
|
HtlcMinimum: htlcMin,
|
|
|
|
HtlcMaximum: htlcMax,
|
|
|
|
Features: features,
|
2022-12-15 22:48:56 +01:00
|
|
|
}
|
2024-05-15 15:52:17 +02:00
|
|
|
hints, err := blindedPayment.toRouteHints(fn.None[*btcec.PublicKey]())
|
2024-05-13 13:44:56 +02:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Nil(t, hints)
|
2022-12-15 22:48:56 +01:00
|
|
|
|
|
|
|
// Populate the blinded payment with hops.
|
|
|
|
blindedPayment.BlindedPath.BlindedHops = []*sphinx.BlindedHopInfo{
|
|
|
|
{
|
|
|
|
BlindedNodePub: pkb1,
|
2023-11-20 17:54:37 +01:00
|
|
|
CipherText: cipherText,
|
2022-12-15 22:48:56 +01:00
|
|
|
},
|
|
|
|
{
|
|
|
|
BlindedNodePub: pkb2,
|
2023-11-20 17:54:37 +01:00
|
|
|
CipherText: cipherText,
|
2022-12-15 22:48:56 +01:00
|
|
|
},
|
|
|
|
{
|
|
|
|
BlindedNodePub: pkb3,
|
2023-11-20 17:54:37 +01:00
|
|
|
CipherText: cipherText,
|
2022-12-15 22:48:56 +01:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2024-05-13 13:44:56 +02:00
|
|
|
policy1 := &models.CachedEdgePolicy{
|
|
|
|
TimeLockDelta: cltvDelta,
|
|
|
|
MinHTLC: lnwire.MilliSatoshi(htlcMin),
|
|
|
|
MaxHTLC: lnwire.MilliSatoshi(htlcMax),
|
|
|
|
FeeBaseMSat: lnwire.MilliSatoshi(baseFee),
|
|
|
|
FeeProportionalMillionths: lnwire.MilliSatoshi(
|
|
|
|
ppmFee,
|
|
|
|
),
|
|
|
|
ToNodePubKey: func() route.Vertex {
|
|
|
|
return vb2
|
|
|
|
},
|
|
|
|
ToNodeFeatures: features,
|
|
|
|
}
|
|
|
|
policy2 := &models.CachedEdgePolicy{
|
|
|
|
ToNodePubKey: func() route.Vertex {
|
|
|
|
return vb3
|
|
|
|
},
|
|
|
|
ToNodeFeatures: features,
|
|
|
|
}
|
|
|
|
|
|
|
|
blindedEdge1, err := NewBlindedEdge(policy1, blindedPayment, 0)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
blindedEdge2, err := NewBlindedEdge(policy2, blindedPayment, 1)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2022-12-15 22:48:56 +01:00
|
|
|
expected := RouteHints{
|
|
|
|
v1: {
|
2024-05-13 13:44:56 +02:00
|
|
|
blindedEdge1,
|
2022-12-15 22:48:56 +01:00
|
|
|
},
|
|
|
|
vb2: {
|
2024-05-13 13:44:56 +02:00
|
|
|
blindedEdge2,
|
2022-12-15 22:48:56 +01:00
|
|
|
},
|
|
|
|
}
|
2023-11-20 17:54:37 +01:00
|
|
|
|
2024-05-15 15:52:17 +02:00
|
|
|
actual, err := blindedPayment.toRouteHints(fn.None[*btcec.PublicKey]())
|
2024-05-13 13:44:56 +02:00
|
|
|
require.NoError(t, err)
|
2022-12-15 22:48:56 +01:00
|
|
|
|
|
|
|
require.Equal(t, len(expected), len(actual))
|
|
|
|
for vertex, expectedHint := range expected {
|
|
|
|
actualHint, ok := actual[vertex]
|
|
|
|
require.True(t, ok, "node not found: %v", vertex)
|
|
|
|
|
|
|
|
require.Len(t, expectedHint, 1)
|
|
|
|
require.Len(t, actualHint, 1)
|
|
|
|
|
|
|
|
// We can't assert that our functions are equal, so we check
|
2023-11-20 17:54:37 +01:00
|
|
|
// their output and then mark them as nil so that we can use
|
2022-12-15 22:48:56 +01:00
|
|
|
// require.Equal for all our other fields.
|
2023-11-20 17:54:37 +01:00
|
|
|
require.Equal(t, expectedHint[0].EdgePolicy().ToNodePubKey(),
|
|
|
|
actualHint[0].EdgePolicy().ToNodePubKey())
|
|
|
|
|
|
|
|
actualHint[0].EdgePolicy().ToNodePubKey = nil
|
|
|
|
expectedHint[0].EdgePolicy().ToNodePubKey = nil
|
|
|
|
|
|
|
|
// The arguments we use for the payload do not matter as long as
|
|
|
|
// both functions return the same payload.
|
|
|
|
expectedPayloadSize := expectedHint[0].IntermediatePayloadSize(
|
2024-05-29 22:14:46 +02:00
|
|
|
0, 0, 0,
|
2023-11-20 17:54:37 +01:00
|
|
|
)
|
|
|
|
actualPayloadSize := actualHint[0].IntermediatePayloadSize(
|
2024-05-29 22:14:46 +02:00
|
|
|
0, 0, 0,
|
2023-11-20 17:54:37 +01:00
|
|
|
)
|
2022-12-15 22:48:56 +01:00
|
|
|
|
2023-11-20 17:54:37 +01:00
|
|
|
require.Equal(t, expectedPayloadSize, actualPayloadSize)
|
2022-12-15 22:48:56 +01:00
|
|
|
|
|
|
|
require.Equal(t, expectedHint[0], actualHint[0])
|
|
|
|
}
|
|
|
|
}
|