2019-09-05 15:05:38 +02:00
|
|
|
package hop_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2022-10-26 16:57:37 +02:00
|
|
|
"encoding/hex"
|
2019-09-05 15:05:38 +02:00
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
|
2022-10-26 16:57:37 +02:00
|
|
|
"github.com/btcsuite/btcd/btcec/v2"
|
2019-09-05 15:05:38 +02:00
|
|
|
"github.com/lightningnetwork/lnd/htlcswitch/hop"
|
2019-11-05 00:10:00 +01:00
|
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
2019-09-05 15:05:38 +02:00
|
|
|
"github.com/lightningnetwork/lnd/record"
|
2021-03-25 03:47:58 +01:00
|
|
|
"github.com/stretchr/testify/require"
|
2019-09-05 15:05:38 +02:00
|
|
|
)
|
|
|
|
|
2022-10-26 16:57:37 +02:00
|
|
|
var (
|
|
|
|
//nolint:lll
|
|
|
|
testPrivKeyBytes, _ = hex.DecodeString("e126f68f7eafcc8b74f54d269fe206be715000f94dac067d1c04a8ca3b2db734")
|
|
|
|
_, testPubKey = btcec.PrivKeyFromBytes(testPrivKeyBytes)
|
|
|
|
)
|
|
|
|
|
2022-01-11 14:15:23 +01:00
|
|
|
const testUnknownRequiredType = 0x80
|
2021-03-25 03:47:33 +01:00
|
|
|
|
2019-09-05 15:05:38 +02:00
|
|
|
type decodePayloadTest struct {
|
2022-01-11 14:15:23 +01:00
|
|
|
name string
|
|
|
|
payload []byte
|
2023-11-01 15:39:33 +01:00
|
|
|
isFinalHop bool
|
2022-01-11 14:15:23 +01:00
|
|
|
expErr error
|
|
|
|
expCustomRecords map[uint64][]byte
|
|
|
|
shouldHaveMPP bool
|
|
|
|
shouldHaveAMP bool
|
2022-10-26 16:57:37 +02:00
|
|
|
shouldHaveEncData bool
|
|
|
|
shouldHaveBlinding bool
|
2022-01-11 14:15:23 +01:00
|
|
|
shouldHaveMetadata bool
|
2022-10-26 16:57:37 +02:00
|
|
|
shouldHaveTotalAmt bool
|
2019-09-05 15:05:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var decodePayloadTests = []decodePayloadTest{
|
2019-10-31 05:19:08 +01:00
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "final hop valid",
|
|
|
|
isFinalHop: true,
|
|
|
|
payload: []byte{0x02, 0x00, 0x04, 0x00},
|
2019-10-31 05:19:08 +01:00
|
|
|
},
|
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "intermediate hop valid",
|
|
|
|
isFinalHop: false,
|
2019-10-31 05:19:08 +01:00
|
|
|
payload: []byte{0x02, 0x00, 0x04, 0x00, 0x06, 0x08, 0x01, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
},
|
|
|
|
},
|
2019-09-05 15:05:38 +02:00
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "final hop no amount",
|
|
|
|
payload: []byte{0x04, 0x00},
|
|
|
|
isFinalHop: true,
|
2019-09-05 15:05:38 +02:00
|
|
|
expErr: hop.ErrInvalidPayload{
|
2019-10-31 05:19:08 +01:00
|
|
|
Type: record.AmtOnionType,
|
|
|
|
Violation: hop.OmittedViolation,
|
|
|
|
FinalHop: true,
|
2019-09-05 15:05:38 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "intermediate hop no amount",
|
|
|
|
isFinalHop: false,
|
2019-09-05 15:05:38 +02:00
|
|
|
payload: []byte{0x04, 0x00, 0x06, 0x08, 0x01, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00,
|
|
|
|
},
|
|
|
|
expErr: hop.ErrInvalidPayload{
|
2019-10-31 05:19:08 +01:00
|
|
|
Type: record.AmtOnionType,
|
|
|
|
Violation: hop.OmittedViolation,
|
|
|
|
FinalHop: false,
|
2019-09-05 15:05:38 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "final hop no expiry",
|
|
|
|
isFinalHop: true,
|
|
|
|
payload: []byte{0x02, 0x00},
|
2019-09-05 15:05:38 +02:00
|
|
|
expErr: hop.ErrInvalidPayload{
|
2019-10-31 05:19:08 +01:00
|
|
|
Type: record.LockTimeOnionType,
|
|
|
|
Violation: hop.OmittedViolation,
|
|
|
|
FinalHop: true,
|
2019-09-05 15:05:38 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "intermediate hop no expiry",
|
|
|
|
isFinalHop: false,
|
2019-09-05 15:05:38 +02:00
|
|
|
payload: []byte{0x02, 0x00, 0x06, 0x08, 0x01, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00,
|
|
|
|
},
|
|
|
|
expErr: hop.ErrInvalidPayload{
|
2019-10-31 05:19:08 +01:00
|
|
|
Type: record.LockTimeOnionType,
|
|
|
|
Violation: hop.OmittedViolation,
|
|
|
|
FinalHop: false,
|
2019-09-05 15:05:38 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "final hop next sid present",
|
|
|
|
isFinalHop: true,
|
2019-09-05 15:05:38 +02:00
|
|
|
payload: []byte{0x02, 0x00, 0x04, 0x00, 0x06, 0x08, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
},
|
|
|
|
expErr: hop.ErrInvalidPayload{
|
2019-10-31 05:19:08 +01:00
|
|
|
Type: record.NextHopOnionType,
|
|
|
|
Violation: hop.IncludedViolation,
|
|
|
|
FinalHop: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "required type after omitted hop id",
|
|
|
|
isFinalHop: true,
|
2021-03-25 03:47:33 +01:00
|
|
|
payload: []byte{
|
|
|
|
0x02, 0x00, 0x04, 0x00,
|
|
|
|
testUnknownRequiredType, 0x00,
|
|
|
|
},
|
2019-10-31 05:19:08 +01:00
|
|
|
expErr: hop.ErrInvalidPayload{
|
2021-03-25 03:47:33 +01:00
|
|
|
Type: testUnknownRequiredType,
|
2019-10-31 05:19:08 +01:00
|
|
|
Violation: hop.RequiredViolation,
|
|
|
|
FinalHop: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "required type after included hop id",
|
|
|
|
isFinalHop: false,
|
2021-03-25 03:47:33 +01:00
|
|
|
payload: []byte{
|
|
|
|
0x02, 0x00, 0x04, 0x00, 0x06, 0x08, 0x01, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
testUnknownRequiredType, 0x00,
|
2019-10-31 05:19:08 +01:00
|
|
|
},
|
|
|
|
expErr: hop.ErrInvalidPayload{
|
2021-03-25 03:47:33 +01:00
|
|
|
Type: testUnknownRequiredType,
|
2019-10-31 05:19:08 +01:00
|
|
|
Violation: hop.RequiredViolation,
|
|
|
|
FinalHop: false,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "required type zero final hop",
|
|
|
|
isFinalHop: true,
|
|
|
|
payload: []byte{0x00, 0x00, 0x02, 0x00, 0x04, 0x00},
|
2019-10-31 05:19:08 +01:00
|
|
|
expErr: hop.ErrInvalidPayload{
|
|
|
|
Type: 0,
|
|
|
|
Violation: hop.RequiredViolation,
|
|
|
|
FinalHop: true,
|
2019-09-05 15:05:38 +02:00
|
|
|
},
|
|
|
|
},
|
2019-10-31 05:20:49 +01:00
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "required type zero final hop zero sid",
|
|
|
|
isFinalHop: true,
|
2019-10-31 05:21:10 +01:00
|
|
|
payload: []byte{0x00, 0x00, 0x02, 0x00, 0x04, 0x00, 0x06, 0x08,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
2019-10-31 05:20:49 +01:00
|
|
|
},
|
|
|
|
expErr: hop.ErrInvalidPayload{
|
2019-11-05 00:10:00 +01:00
|
|
|
Type: record.NextHopOnionType,
|
2019-10-31 05:21:10 +01:00
|
|
|
Violation: hop.IncludedViolation,
|
2019-10-31 05:20:49 +01:00
|
|
|
FinalHop: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "required type zero intermediate hop",
|
|
|
|
isFinalHop: false,
|
2019-10-31 05:21:10 +01:00
|
|
|
payload: []byte{0x00, 0x00, 0x02, 0x00, 0x04, 0x00, 0x06, 0x08,
|
|
|
|
0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
2019-10-31 05:20:49 +01:00
|
|
|
},
|
|
|
|
expErr: hop.ErrInvalidPayload{
|
|
|
|
Type: 0,
|
|
|
|
Violation: hop.RequiredViolation,
|
|
|
|
FinalHop: false,
|
|
|
|
},
|
|
|
|
},
|
2019-11-11 10:37:24 +01:00
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "required type in custom range",
|
|
|
|
isFinalHop: false,
|
2023-11-01 20:34:31 +01:00
|
|
|
payload: []byte{
|
|
|
|
// amount
|
|
|
|
0x02, 0x00,
|
|
|
|
// cltv
|
|
|
|
0x04, 0x00,
|
|
|
|
// next hop id
|
|
|
|
0x06, 0x08,
|
|
|
|
0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
// custom
|
2019-11-19 12:32:56 +01:00
|
|
|
0xfe, 0x00, 0x01, 0x00, 0x00, 0x02, 0x10, 0x11,
|
|
|
|
},
|
|
|
|
expCustomRecords: map[uint64][]byte{
|
|
|
|
65536: {0x10, 0x11},
|
2019-11-11 10:37:24 +01:00
|
|
|
},
|
|
|
|
},
|
2019-11-05 00:10:00 +01:00
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "valid intermediate hop",
|
|
|
|
isFinalHop: false,
|
2019-11-05 00:10:00 +01:00
|
|
|
payload: []byte{0x02, 0x00, 0x04, 0x00, 0x06, 0x08, 0x01, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
},
|
|
|
|
expErr: nil,
|
|
|
|
},
|
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "valid final hop",
|
|
|
|
isFinalHop: true,
|
|
|
|
payload: []byte{0x02, 0x00, 0x04, 0x00},
|
|
|
|
expErr: nil,
|
2019-11-05 00:10:00 +01:00
|
|
|
},
|
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "intermediate hop with mpp",
|
|
|
|
isFinalHop: false,
|
2019-11-05 00:10:00 +01:00
|
|
|
payload: []byte{
|
|
|
|
// amount
|
|
|
|
0x02, 0x00,
|
|
|
|
// cltv
|
|
|
|
0x04, 0x00,
|
|
|
|
// next hop id
|
|
|
|
0x06, 0x08,
|
|
|
|
0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
// mpp
|
|
|
|
0x08, 0x21,
|
|
|
|
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
|
|
|
|
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
|
|
|
|
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
|
|
|
|
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
|
|
|
|
0x08,
|
|
|
|
},
|
|
|
|
expErr: hop.ErrInvalidPayload{
|
|
|
|
Type: record.MPPOnionType,
|
|
|
|
Violation: hop.IncludedViolation,
|
|
|
|
FinalHop: false,
|
|
|
|
},
|
|
|
|
},
|
2021-03-25 03:47:58 +01:00
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "intermediate hop with amp",
|
|
|
|
isFinalHop: false,
|
2021-03-25 03:47:58 +01:00
|
|
|
payload: []byte{
|
|
|
|
// amount
|
|
|
|
0x02, 0x00,
|
|
|
|
// cltv
|
|
|
|
0x04, 0x00,
|
|
|
|
// next hop id
|
|
|
|
0x06, 0x08,
|
|
|
|
0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
// amp
|
|
|
|
0x0e, 0x41,
|
|
|
|
// amp.root_share
|
|
|
|
0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12,
|
|
|
|
0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12,
|
|
|
|
0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12,
|
|
|
|
0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12,
|
|
|
|
// amp.set_id
|
|
|
|
0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
|
|
|
|
0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
|
|
|
|
0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
|
|
|
|
0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
|
|
|
|
// amp.child_index
|
|
|
|
0x09,
|
|
|
|
},
|
|
|
|
expErr: hop.ErrInvalidPayload{
|
|
|
|
Type: record.AMPOnionType,
|
|
|
|
Violation: hop.IncludedViolation,
|
|
|
|
FinalHop: false,
|
|
|
|
},
|
|
|
|
},
|
2022-10-26 16:57:37 +02:00
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "intermediate hop with encrypted data",
|
|
|
|
isFinalHop: false,
|
2022-10-26 16:57:37 +02:00
|
|
|
payload: []byte{
|
|
|
|
// encrypted data
|
|
|
|
0x0a, 0x03, 0x03, 0x02, 0x01,
|
|
|
|
},
|
|
|
|
shouldHaveEncData: true,
|
|
|
|
},
|
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "intermediate hop with blinding point",
|
|
|
|
isFinalHop: false,
|
2022-10-26 16:57:37 +02:00
|
|
|
payload: append([]byte{
|
2023-11-01 20:20:04 +01:00
|
|
|
// encrypted data
|
|
|
|
0x0a, 0x03, 0x03, 0x02, 0x01,
|
2022-10-26 16:57:37 +02:00
|
|
|
// blinding point (type / length)
|
|
|
|
0x0c, 0x21,
|
|
|
|
},
|
|
|
|
// blinding point (value)
|
|
|
|
testPubKey.SerializeCompressed()...,
|
|
|
|
),
|
|
|
|
shouldHaveBlinding: true,
|
2023-11-01 20:20:04 +01:00
|
|
|
shouldHaveEncData: true,
|
2022-10-26 16:57:37 +02:00
|
|
|
},
|
2019-11-05 00:10:00 +01:00
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "final hop with mpp",
|
|
|
|
isFinalHop: true,
|
2019-11-05 00:10:00 +01:00
|
|
|
payload: []byte{
|
|
|
|
// amount
|
|
|
|
0x02, 0x00,
|
|
|
|
// cltv
|
|
|
|
0x04, 0x00,
|
|
|
|
// mpp
|
|
|
|
0x08, 0x21,
|
|
|
|
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
|
|
|
|
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
|
|
|
|
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
|
|
|
|
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
|
|
|
|
0x08,
|
|
|
|
},
|
|
|
|
expErr: nil,
|
|
|
|
shouldHaveMPP: true,
|
|
|
|
},
|
2021-03-25 03:47:58 +01:00
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "final hop with amp",
|
|
|
|
isFinalHop: true,
|
2021-03-25 03:47:58 +01:00
|
|
|
payload: []byte{
|
|
|
|
// amount
|
|
|
|
0x02, 0x00,
|
|
|
|
// cltv
|
|
|
|
0x04, 0x00,
|
|
|
|
// amp
|
|
|
|
0x0e, 0x41,
|
|
|
|
// amp.root_share
|
|
|
|
0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12,
|
|
|
|
0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12,
|
|
|
|
0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12,
|
|
|
|
0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12,
|
|
|
|
// amp.set_id
|
|
|
|
0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
|
|
|
|
0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
|
|
|
|
0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
|
|
|
|
0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
|
|
|
|
// amp.child_index
|
|
|
|
0x09,
|
|
|
|
},
|
|
|
|
shouldHaveAMP: true,
|
|
|
|
},
|
2022-01-11 14:15:23 +01:00
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "final hop with metadata",
|
|
|
|
isFinalHop: true,
|
2022-01-11 14:15:23 +01:00
|
|
|
payload: []byte{
|
|
|
|
// amount
|
|
|
|
0x02, 0x00,
|
|
|
|
// cltv
|
|
|
|
0x04, 0x00,
|
|
|
|
// metadata
|
|
|
|
0x10, 0x03, 0x01, 0x02, 0x03,
|
|
|
|
},
|
|
|
|
shouldHaveMetadata: true,
|
|
|
|
},
|
2022-10-26 16:57:37 +02:00
|
|
|
{
|
2023-11-01 15:39:33 +01:00
|
|
|
name: "final hop with total amount",
|
|
|
|
isFinalHop: true,
|
2022-10-26 16:57:37 +02:00
|
|
|
payload: []byte{
|
|
|
|
// amount
|
|
|
|
0x02, 0x00,
|
|
|
|
// cltv
|
|
|
|
0x04, 0x00,
|
|
|
|
// total amount
|
|
|
|
0x12, 0x01, 0x01,
|
|
|
|
},
|
|
|
|
shouldHaveTotalAmt: true,
|
|
|
|
},
|
2019-09-05 15:05:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// TestDecodeHopPayloadRecordValidation asserts that parsing the payloads in the
|
|
|
|
// tests yields the expected errors depending on whether the proper fields were
|
|
|
|
// included or omitted.
|
|
|
|
func TestDecodeHopPayloadRecordValidation(t *testing.T) {
|
|
|
|
for _, test := range decodePayloadTests {
|
|
|
|
t.Run(test.name, func(t *testing.T) {
|
|
|
|
testDecodeHopPayloadValidation(t, test)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func testDecodeHopPayloadValidation(t *testing.T, test decodePayloadTest) {
|
2019-11-05 00:10:00 +01:00
|
|
|
var (
|
|
|
|
testTotalMsat = lnwire.MilliSatoshi(8)
|
|
|
|
testAddr = [32]byte{
|
|
|
|
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
|
|
|
|
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
|
|
|
|
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
|
|
|
|
0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11, 0x11,
|
|
|
|
}
|
2021-03-25 03:47:58 +01:00
|
|
|
|
|
|
|
testRootShare = [32]byte{
|
|
|
|
0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12,
|
|
|
|
0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12,
|
|
|
|
0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12,
|
|
|
|
0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12, 0x12,
|
|
|
|
}
|
|
|
|
testSetID = [32]byte{
|
|
|
|
0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
|
|
|
|
0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
|
|
|
|
0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
|
|
|
|
0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
|
|
|
|
}
|
2022-10-26 16:57:37 +02:00
|
|
|
testEncData = []byte{3, 2, 1}
|
2022-01-11 14:15:23 +01:00
|
|
|
testMetadata = []byte{1, 2, 3}
|
2021-03-25 03:47:58 +01:00
|
|
|
testChildIndex = uint32(9)
|
2019-11-05 00:10:00 +01:00
|
|
|
)
|
|
|
|
|
2023-11-01 15:39:33 +01:00
|
|
|
p, err := hop.NewPayloadFromReader(
|
|
|
|
bytes.NewReader(test.payload), test.isFinalHop,
|
|
|
|
)
|
2019-09-05 15:05:38 +02:00
|
|
|
if !reflect.DeepEqual(test.expErr, err) {
|
|
|
|
t.Fatalf("expected error mismatch, want: %v, got: %v",
|
|
|
|
test.expErr, err)
|
|
|
|
}
|
2019-11-05 00:10:00 +01:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assert MPP fields if we expect them.
|
|
|
|
if test.shouldHaveMPP {
|
|
|
|
if p.MPP == nil {
|
|
|
|
t.Fatalf("payload should have MPP record")
|
|
|
|
}
|
|
|
|
if p.MPP.TotalMsat() != testTotalMsat {
|
|
|
|
t.Fatalf("invalid total msat")
|
|
|
|
}
|
|
|
|
if p.MPP.PaymentAddr() != testAddr {
|
|
|
|
t.Fatalf("invalid payment addr")
|
|
|
|
}
|
|
|
|
} else if p.MPP != nil {
|
|
|
|
t.Fatalf("unexpected MPP payload")
|
|
|
|
}
|
2019-11-19 12:32:56 +01:00
|
|
|
|
2021-03-25 03:47:58 +01:00
|
|
|
if test.shouldHaveAMP {
|
|
|
|
if p.AMP == nil {
|
|
|
|
t.Fatalf("payload should have AMP record")
|
|
|
|
}
|
|
|
|
require.Equal(t, testRootShare, p.AMP.RootShare())
|
|
|
|
require.Equal(t, testSetID, p.AMP.SetID())
|
|
|
|
require.Equal(t, testChildIndex, p.AMP.ChildIndex())
|
|
|
|
} else if p.AMP != nil {
|
|
|
|
t.Fatalf("unexpected AMP payload")
|
|
|
|
}
|
|
|
|
|
2022-01-11 14:15:23 +01:00
|
|
|
if test.shouldHaveMetadata {
|
|
|
|
if p.Metadata() == nil {
|
|
|
|
t.Fatalf("payload should have metadata")
|
|
|
|
}
|
|
|
|
require.Equal(t, testMetadata, p.Metadata())
|
|
|
|
} else if p.Metadata() != nil {
|
|
|
|
t.Fatalf("unexpected metadata")
|
|
|
|
}
|
|
|
|
|
2022-10-26 16:57:37 +02:00
|
|
|
if test.shouldHaveEncData {
|
|
|
|
require.NotNil(t, p.EncryptedData(),
|
|
|
|
"payment should have encrypted data")
|
|
|
|
|
|
|
|
require.Equal(t, testEncData, p.EncryptedData())
|
|
|
|
} else {
|
|
|
|
require.Nil(t, p.EncryptedData())
|
|
|
|
}
|
|
|
|
|
|
|
|
if test.shouldHaveBlinding {
|
|
|
|
require.NotNil(t, p.BlindingPoint())
|
|
|
|
|
|
|
|
require.Equal(t, testPubKey, p.BlindingPoint())
|
|
|
|
} else {
|
|
|
|
require.Nil(t, p.BlindingPoint())
|
|
|
|
}
|
|
|
|
|
|
|
|
if test.shouldHaveTotalAmt {
|
|
|
|
require.NotZero(t, p.TotalAmtMsat())
|
|
|
|
} else {
|
|
|
|
require.Zero(t, p.TotalAmtMsat())
|
|
|
|
}
|
|
|
|
|
2019-11-19 12:32:56 +01:00
|
|
|
// Convert expected nil map to empty map, because we always expect an
|
|
|
|
// initiated map from the payload.
|
2019-12-12 00:01:55 +01:00
|
|
|
expCustomRecords := make(record.CustomSet)
|
2019-11-19 12:32:56 +01:00
|
|
|
if test.expCustomRecords != nil {
|
|
|
|
expCustomRecords = test.expCustomRecords
|
|
|
|
}
|
|
|
|
if !reflect.DeepEqual(expCustomRecords, p.CustomRecords()) {
|
|
|
|
t.Fatalf("invalid custom records")
|
|
|
|
}
|
2019-09-05 15:05:38 +02:00
|
|
|
}
|