2019-08-08 01:04:45 +03:00
|
|
|
package tlv_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"errors"
|
|
|
|
"io"
|
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/btcsuite/btcd/btcec"
|
|
|
|
"github.com/lightningnetwork/lnd/tlv"
|
|
|
|
)
|
|
|
|
|
|
|
|
type nodeAmts struct {
|
|
|
|
nodeID *btcec.PublicKey
|
|
|
|
amt1 uint64
|
|
|
|
amt2 uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
func ENodeAmts(w io.Writer, val interface{}, buf *[8]byte) error {
|
|
|
|
if t, ok := val.(*nodeAmts); ok {
|
|
|
|
if err := tlv.EPubKey(w, &t.nodeID, buf); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := tlv.EUint64T(w, t.amt1, buf); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return tlv.EUint64T(w, t.amt2, buf)
|
|
|
|
}
|
|
|
|
return tlv.NewTypeForEncodingErr(val, "nodeAmts")
|
|
|
|
}
|
|
|
|
|
|
|
|
func DNodeAmts(r io.Reader, val interface{}, buf *[8]byte, l uint64) error {
|
|
|
|
if t, ok := val.(*nodeAmts); ok && l == 49 {
|
|
|
|
if err := tlv.DPubKey(r, &t.nodeID, buf, 33); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := tlv.DUint64(r, &t.amt1, buf, 8); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return tlv.DUint64(r, &t.amt2, buf, 8)
|
|
|
|
}
|
|
|
|
return tlv.NewTypeForDecodingErr(val, "nodeAmts", l, 49)
|
|
|
|
}
|
|
|
|
|
|
|
|
type N1 struct {
|
|
|
|
amt uint64
|
|
|
|
scid uint64
|
|
|
|
nodeAmts nodeAmts
|
|
|
|
cltvDelta uint16
|
|
|
|
|
2019-08-22 02:30:05 +03:00
|
|
|
alias []byte
|
|
|
|
|
2019-08-08 01:04:45 +03:00
|
|
|
stream *tlv.Stream
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *N1) sizeAmt() uint64 {
|
|
|
|
return tlv.SizeTUint64(n.amt)
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewN1() *N1 {
|
|
|
|
n := new(N1)
|
|
|
|
|
|
|
|
n.stream = tlv.MustNewStream(
|
|
|
|
tlv.MakeDynamicRecord(
|
|
|
|
1, &n.amt, n.sizeAmt, tlv.ETUint64, tlv.DTUint64,
|
|
|
|
),
|
|
|
|
tlv.MakePrimitiveRecord(2, &n.scid),
|
|
|
|
tlv.MakeStaticRecord(3, &n.nodeAmts, 49, ENodeAmts, DNodeAmts),
|
|
|
|
tlv.MakePrimitiveRecord(254, &n.cltvDelta),
|
2019-08-22 02:30:05 +03:00
|
|
|
tlv.MakePrimitiveRecord(401, &n.alias),
|
2019-08-08 01:04:45 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
return n
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *N1) Encode(w io.Writer) error {
|
|
|
|
return n.stream.Encode(w)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *N1) Decode(r io.Reader) error {
|
|
|
|
return n.stream.Decode(r)
|
|
|
|
}
|
|
|
|
|
|
|
|
type N2 struct {
|
|
|
|
amt uint64
|
|
|
|
cltvExpiry uint32
|
|
|
|
|
|
|
|
stream *tlv.Stream
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *N2) sizeAmt() uint64 {
|
|
|
|
return tlv.SizeTUint64(n.amt)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *N2) sizeCltv() uint64 {
|
|
|
|
return tlv.SizeTUint32(n.cltvExpiry)
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewN2() *N2 {
|
|
|
|
n := new(N2)
|
|
|
|
|
|
|
|
n.stream = tlv.MustNewStream(
|
|
|
|
tlv.MakeDynamicRecord(
|
|
|
|
0, &n.amt, n.sizeAmt, tlv.ETUint64, tlv.DTUint64,
|
|
|
|
),
|
|
|
|
tlv.MakeDynamicRecord(
|
|
|
|
11, &n.cltvExpiry, n.sizeCltv, tlv.ETUint32, tlv.DTUint32,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
return n
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *N2) Encode(w io.Writer) error {
|
|
|
|
return n.stream.Encode(w)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *N2) Decode(r io.Reader) error {
|
|
|
|
return n.stream.Decode(r)
|
|
|
|
}
|
|
|
|
|
|
|
|
var tlvDecodingFailureTests = []struct {
|
|
|
|
name string
|
|
|
|
bytes []byte
|
|
|
|
expErr error
|
|
|
|
|
|
|
|
// skipN2 if true, will cause the test to only be executed on N1.
|
|
|
|
skipN2 bool
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "type truncated",
|
|
|
|
bytes: []byte{0xfd},
|
|
|
|
expErr: io.ErrUnexpectedEOF,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "type truncated",
|
|
|
|
bytes: []byte{0xfd, 0x01},
|
|
|
|
expErr: io.ErrUnexpectedEOF,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "not minimally encoded type",
|
|
|
|
bytes: []byte{0xfd, 0x00, 0x01}, // spec has trailing 0x00
|
|
|
|
expErr: tlv.ErrVarIntNotCanonical,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing length",
|
|
|
|
bytes: []byte{0xfd, 0x01, 0x01},
|
|
|
|
expErr: io.ErrUnexpectedEOF,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "length truncated",
|
|
|
|
bytes: []byte{0x0f, 0xfd},
|
|
|
|
expErr: io.ErrUnexpectedEOF,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "length truncated",
|
|
|
|
bytes: []byte{0x0f, 0xfd, 0x26},
|
|
|
|
expErr: io.ErrUnexpectedEOF,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "missing value",
|
|
|
|
bytes: []byte{0x0f, 0xfd, 0x26, 0x02},
|
|
|
|
expErr: io.ErrUnexpectedEOF,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "not minimally encoded length",
|
|
|
|
bytes: []byte{0x0f, 0xfd, 0x00, 0x01}, // spec has trailing 0x00
|
|
|
|
expErr: tlv.ErrVarIntNotCanonical,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "value truncated",
|
|
|
|
bytes: []byte{0x0f, 0xfd, 0x02, 0x01,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
},
|
|
|
|
expErr: io.ErrUnexpectedEOF,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "greater than encoding length for n1's amt",
|
|
|
|
bytes: []byte{0x01, 0x09, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff},
|
|
|
|
expErr: tlv.NewTypeForDecodingErr(new(uint64), "uint64", 9, 8),
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "encoding for n1's amt is not minimal",
|
|
|
|
bytes: []byte{0x01, 0x01, 0x00},
|
|
|
|
expErr: tlv.ErrTUintNotMinimal,
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "encoding for n1's amt is not minimal",
|
|
|
|
bytes: []byte{0x01, 0x02, 0x00, 0x01},
|
|
|
|
expErr: tlv.ErrTUintNotMinimal,
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "encoding for n1's amt is not minimal",
|
|
|
|
bytes: []byte{0x01, 0x03, 0x00, 0x01, 0x00},
|
|
|
|
expErr: tlv.ErrTUintNotMinimal,
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "encoding for n1's amt is not minimal",
|
|
|
|
bytes: []byte{0x01, 0x04, 0x00, 0x01, 0x00, 0x00},
|
|
|
|
expErr: tlv.ErrTUintNotMinimal,
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "encoding for n1's amt is not minimal",
|
|
|
|
bytes: []byte{0x01, 0x05, 0x00, 0x01, 0x00, 0x00, 0x00},
|
|
|
|
expErr: tlv.ErrTUintNotMinimal,
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "encoding for n1's amt is not minimal",
|
|
|
|
bytes: []byte{0x01, 0x06, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00},
|
|
|
|
expErr: tlv.ErrTUintNotMinimal,
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "encoding for n1's amt is not minimal",
|
|
|
|
bytes: []byte{0x01, 0x07, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00},
|
|
|
|
expErr: tlv.ErrTUintNotMinimal,
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "encoding for n1's amt is not minimal",
|
|
|
|
bytes: []byte{0x01, 0x08, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
|
|
|
|
expErr: tlv.ErrTUintNotMinimal,
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "less than encoding length for n1's scid",
|
|
|
|
bytes: []byte{0x02, 0x07, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01},
|
|
|
|
expErr: tlv.NewTypeForDecodingErr(new(uint64), "uint64", 7, 8),
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "less than encoding length for n1's scid",
|
|
|
|
bytes: []byte{0x02, 0x09, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01, 0x01},
|
|
|
|
expErr: tlv.NewTypeForDecodingErr(new(uint64), "uint64", 9, 8),
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "less than encoding length for n1's nodeAmts",
|
|
|
|
bytes: []byte{0x03, 0x29,
|
|
|
|
0x02, 0x3d, 0xa0, 0x92, 0xf6, 0x98, 0x0e, 0x58, 0xd2,
|
|
|
|
0xc0, 0x37, 0x17, 0x31, 0x80, 0xe9, 0xa4, 0x65, 0x47,
|
|
|
|
0x60, 0x26, 0xee, 0x50, 0xf9, 0x66, 0x95, 0x96, 0x3e,
|
|
|
|
0x8e, 0xfe, 0x43, 0x6f, 0x54, 0xeb, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x01,
|
|
|
|
},
|
|
|
|
expErr: tlv.NewTypeForDecodingErr(new(nodeAmts), "nodeAmts", 41, 49),
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "less than encoding length for n1's nodeAmts",
|
|
|
|
bytes: []byte{0x03, 0x30,
|
|
|
|
0x02, 0x3d, 0xa0, 0x92, 0xf6, 0x98, 0x0e, 0x58, 0xd2,
|
|
|
|
0xc0, 0x37, 0x17, 0x31, 0x80, 0xe9, 0xa4, 0x65, 0x47,
|
|
|
|
0x60, 0x26, 0xee, 0x50, 0xf9, 0x66, 0x95, 0x96, 0x3e,
|
|
|
|
0x8e, 0xfe, 0x43, 0x6f, 0x54, 0xeb, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x01,
|
|
|
|
},
|
|
|
|
expErr: tlv.NewTypeForDecodingErr(new(nodeAmts), "nodeAmts", 48, 49),
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "n1's node_id is not a valid point",
|
|
|
|
bytes: []byte{0x03, 0x31,
|
|
|
|
0x04, 0x3d, 0xa0, 0x92, 0xf6, 0x98, 0x0e, 0x58, 0xd2,
|
|
|
|
0xc0, 0x37, 0x17, 0x31, 0x80, 0xe9, 0xa4, 0x65, 0x47,
|
|
|
|
0x60, 0x26, 0xee, 0x50, 0xf9, 0x66, 0x95, 0x96, 0x3e,
|
|
|
|
0x8e, 0xfe, 0x43, 0x6f, 0x54, 0xeb, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x02,
|
|
|
|
},
|
|
|
|
expErr: errors.New("invalid magic in compressed pubkey string: 4"),
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "greater than encoding length for n1's nodeAmts",
|
|
|
|
bytes: []byte{0x03, 0x32,
|
|
|
|
0x02, 0x3d, 0xa0, 0x92, 0xf6, 0x98, 0x0e, 0x58, 0xd2,
|
|
|
|
0xc0, 0x37, 0x17, 0x31, 0x80, 0xe9, 0xa4, 0x65, 0x47,
|
|
|
|
0x60, 0x26, 0xee, 0x50, 0xf9, 0x66, 0x95, 0x96, 0x3e,
|
|
|
|
0x8e, 0xfe, 0x43, 0x6f, 0x54, 0xeb, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x01,
|
|
|
|
},
|
|
|
|
expErr: tlv.NewTypeForDecodingErr(new(nodeAmts), "nodeAmts", 50, 49),
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "less than encoding length for n1's cltvDelta",
|
|
|
|
bytes: []byte{0xfd, 0x00, 0x0fe, 0x00},
|
|
|
|
expErr: tlv.NewTypeForDecodingErr(new(uint16), "uint16", 0, 2),
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "less than encoding length for n1's cltvDelta",
|
|
|
|
bytes: []byte{0xfd, 0x00, 0xfe, 0x01, 0x01},
|
|
|
|
expErr: tlv.NewTypeForDecodingErr(new(uint16), "uint16", 1, 2),
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "greater than encoding length for n1's cltvDelta",
|
|
|
|
bytes: []byte{0xfd, 0x00, 0xfe, 0x03, 0x01, 0x01, 0x01},
|
|
|
|
expErr: tlv.NewTypeForDecodingErr(new(uint16), "uint16", 3, 2),
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "valid records but invalid ordering",
|
|
|
|
bytes: []byte{0x02, 0x08,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x26, 0x01,
|
|
|
|
0x01, 0x2a,
|
|
|
|
},
|
|
|
|
expErr: tlv.ErrStreamNotCanonical,
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "duplicate tlv type",
|
|
|
|
bytes: []byte{0x02, 0x08,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x31, 0x02,
|
|
|
|
0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x04, 0x51,
|
|
|
|
},
|
|
|
|
expErr: tlv.ErrStreamNotCanonical,
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "duplicate ignored tlv type",
|
|
|
|
bytes: []byte{0x1f, 0x00, 0x1f, 0x01, 0x2a},
|
|
|
|
expErr: tlv.ErrStreamNotCanonical,
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "type wraparound",
|
|
|
|
bytes: []byte{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, 0x00},
|
|
|
|
expErr: tlv.ErrStreamNotCanonical,
|
|
|
|
},
|
2019-08-22 02:30:05 +03:00
|
|
|
{
|
|
|
|
name: "absurd record length",
|
|
|
|
bytes: []byte{0xfd, 0x01, 0x91, 0xfe, 0xff, 0xff, 0xff, 0xff},
|
|
|
|
expErr: tlv.ErrRecordTooLarge,
|
|
|
|
skipN2: true,
|
|
|
|
},
|
2019-08-08 01:04:45 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// TestTLVDecodingSuccess asserts that the TLV parser fails to decode invalid
|
|
|
|
// TLV streams.
|
|
|
|
func TestTLVDecodingFailures(t *testing.T) {
|
|
|
|
for _, test := range tlvDecodingFailureTests {
|
|
|
|
t.Run(test.name, func(t *testing.T) {
|
|
|
|
n1 := NewN1()
|
|
|
|
r := bytes.NewReader(test.bytes)
|
|
|
|
|
|
|
|
err := n1.Decode(r)
|
|
|
|
if !reflect.DeepEqual(err, test.expErr) {
|
|
|
|
t.Fatalf("expected N1 decoding failure: %v, "+
|
|
|
|
"got: %v", test.expErr, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if test.skipN2 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
n2 := NewN2()
|
|
|
|
r = bytes.NewReader(test.bytes)
|
|
|
|
|
|
|
|
err = n2.Decode(r)
|
|
|
|
if !reflect.DeepEqual(err, test.expErr) {
|
|
|
|
t.Fatalf("expected N2 decoding failure: %v, "+
|
|
|
|
"got: %v", test.expErr, err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var tlvDecodingSuccessTests = []struct {
|
|
|
|
name string
|
|
|
|
bytes []byte
|
|
|
|
skipN2 bool
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "empty",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "unknown odd type",
|
|
|
|
bytes: []byte{0x21, 0x00},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "unknown odd type",
|
|
|
|
bytes: []byte{0xfd, 0x02, 0x01, 0x00},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "unknown odd type",
|
|
|
|
bytes: []byte{0xfd, 0x00, 0xfd, 0x00},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "unknown odd type",
|
|
|
|
bytes: []byte{0xfd, 0x00, 0xff, 0x00},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "unknown odd type",
|
|
|
|
bytes: []byte{0xfe, 0x02, 0x00, 0x00, 0x01, 0x00},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "unknown odd type",
|
|
|
|
bytes: []byte{0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "N1 amt=0",
|
|
|
|
bytes: []byte{0x01, 0x00},
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "N1 amt=1",
|
|
|
|
bytes: []byte{0x01, 0x01, 0x01},
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "N1 amt=256",
|
|
|
|
bytes: []byte{0x01, 0x02, 0x01, 0x00},
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "N1 amt=65536",
|
|
|
|
bytes: []byte{0x01, 0x03, 0x01, 0x00, 0x00},
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "N1 amt=16777216",
|
|
|
|
bytes: []byte{0x01, 0x04, 0x01, 0x00, 0x00, 0x00},
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "N1 amt=4294967296",
|
|
|
|
bytes: []byte{0x01, 0x05, 0x01, 0x00, 0x00, 0x00, 0x00},
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "N1 amt=1099511627776",
|
|
|
|
bytes: []byte{0x01, 0x06, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00},
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "N1 amt=281474976710656",
|
|
|
|
bytes: []byte{0x01, 0x07, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "N1 amt=72057594037927936",
|
|
|
|
bytes: []byte{0x01, 0x08, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "N1 scid=0x0x550",
|
|
|
|
bytes: []byte{0x02, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x26},
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "N1 node_id=023da092f6980e58d2c037173180e9a465476026ee50f96695963e8efe436f54eb amount_msat_1=1 amount_msat_2=2",
|
|
|
|
bytes: []byte{0x03, 0x31,
|
|
|
|
0x02, 0x3d, 0xa0, 0x92, 0xf6, 0x98, 0x0e, 0x58, 0xd2,
|
|
|
|
0xc0, 0x37, 0x17, 0x31, 0x80, 0xe9, 0xa4, 0x65, 0x47,
|
|
|
|
0x60, 0x26, 0xee, 0x50, 0xf9, 0x66, 0x95, 0x96, 0x3e,
|
|
|
|
0x8e, 0xfe, 0x43, 0x6f, 0x54, 0xeb, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x02},
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "N1 cltv_delta=550",
|
|
|
|
bytes: []byte{0xfd, 0x00, 0xfe, 0x02, 0x02, 0x26},
|
|
|
|
skipN2: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestTLVDecodingSuccess asserts that the TLV parser is able to successfully
|
|
|
|
// decode valid TLV streams.
|
|
|
|
func TestTLVDecodingSuccess(t *testing.T) {
|
|
|
|
for _, test := range tlvDecodingSuccessTests {
|
|
|
|
t.Run(test.name, func(t *testing.T) {
|
|
|
|
n1 := NewN1()
|
|
|
|
r := bytes.NewReader(test.bytes)
|
|
|
|
|
|
|
|
err := n1.Decode(r)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("expected N1 decoding success, got: %v",
|
|
|
|
err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if test.skipN2 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
n2 := NewN2()
|
|
|
|
r = bytes.NewReader(test.bytes)
|
|
|
|
|
|
|
|
err = n2.Decode(r)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("expected N2 decoding succes, got: %v",
|
|
|
|
err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|