2016-12-07 18:46:22 +03:00
|
|
|
package lnwire
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2016-12-22 23:24:48 +03:00
|
|
|
"io"
|
|
|
|
"net"
|
|
|
|
|
2016-12-07 18:46:22 +03:00
|
|
|
"github.com/go-errors/errors"
|
|
|
|
"github.com/roasbeef/btcd/btcec"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
startPort uint16 = 1024
|
|
|
|
endPort uint16 = 49151
|
|
|
|
aliasSpecLen = 21
|
|
|
|
)
|
|
|
|
|
|
|
|
// RGB is used to represent the color.
|
|
|
|
type RGB struct {
|
|
|
|
red uint8
|
|
|
|
green uint8
|
|
|
|
blue uint8
|
|
|
|
}
|
|
|
|
|
2016-12-25 01:55:31 +03:00
|
|
|
// Alias a hex encoded UTF-8 string that may be displayed as an alternative to
|
|
|
|
// the node's ID. Notice that aliases are not unique and may be freely chosen
|
|
|
|
// by the node operators.
|
|
|
|
type Alias struct {
|
|
|
|
data [32]byte
|
|
|
|
aliasLen int
|
|
|
|
}
|
2016-12-07 18:46:22 +03:00
|
|
|
|
|
|
|
// NewAlias create the alias from string and also checks spec requirements.
|
|
|
|
func NewAlias(s string) (Alias, error) {
|
|
|
|
data := []byte(s)
|
2016-12-25 01:55:31 +03:00
|
|
|
return newAlias(data)
|
|
|
|
}
|
|
|
|
|
|
|
|
func newAlias(data []byte) (Alias, error) {
|
|
|
|
var a [32]byte
|
|
|
|
|
2017-03-09 01:21:50 +03:00
|
|
|
rawAlias := data
|
|
|
|
if len(data) > aliasSpecLen {
|
|
|
|
rawAlias = data[:aliasSpecLen]
|
|
|
|
}
|
|
|
|
|
|
|
|
aliasEnd := len(rawAlias)
|
|
|
|
for rawAlias[aliasEnd-1] == 0 && aliasEnd > 0 {
|
2016-12-25 01:55:31 +03:00
|
|
|
aliasEnd--
|
2016-12-07 18:46:22 +03:00
|
|
|
}
|
|
|
|
|
2017-03-09 01:21:50 +03:00
|
|
|
copy(a[:aliasEnd], rawAlias[:aliasEnd])
|
2016-12-25 01:55:31 +03:00
|
|
|
|
|
|
|
return Alias{
|
|
|
|
data: a,
|
|
|
|
aliasLen: aliasEnd,
|
|
|
|
}, nil
|
2016-12-07 18:46:22 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Alias) String() string {
|
2016-12-25 01:55:31 +03:00
|
|
|
return string(a.data[:a.aliasLen])
|
2016-12-07 18:46:22 +03:00
|
|
|
}
|
|
|
|
|
2016-12-25 01:55:31 +03:00
|
|
|
// Validate check that alias data length is lower than spec size.
|
2016-12-07 18:46:22 +03:00
|
|
|
func (a *Alias) Validate() error {
|
2016-12-25 01:55:31 +03:00
|
|
|
nonzero := len(a.data)
|
|
|
|
for a.data[nonzero-1] == 0 && nonzero > 0 {
|
2016-12-07 18:46:22 +03:00
|
|
|
nonzero--
|
|
|
|
}
|
|
|
|
|
|
|
|
if nonzero > aliasSpecLen {
|
|
|
|
return errors.New("alias should be less then 21 bytes")
|
|
|
|
}
|
2016-12-25 01:55:31 +03:00
|
|
|
|
2016-12-07 18:46:22 +03:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-03-06 07:43:34 +03:00
|
|
|
// NodeAnnouncement message is used to announce the presence of a Lightning
|
|
|
|
// node and also to signal that the node is accepting incoming connections.
|
|
|
|
// Each NodeAnnouncement authenticating the advertised information within the
|
|
|
|
// announcement via a signature using the advertised node pubkey.
|
2016-12-07 18:46:22 +03:00
|
|
|
type NodeAnnouncement struct {
|
|
|
|
// Signature is used to prove the ownership of node id.
|
|
|
|
Signature *btcec.Signature
|
|
|
|
|
2017-03-06 07:43:34 +03:00
|
|
|
// Timestamp allows ordering in the case of multiple
|
|
|
|
// announcements.
|
2016-12-07 18:46:22 +03:00
|
|
|
Timestamp uint32
|
|
|
|
|
2017-03-06 07:43:34 +03:00
|
|
|
// Address includes two specification fields: 'ipv6' and
|
|
|
|
// 'port' on which the node is accepting incoming connections.
|
2016-12-07 18:46:22 +03:00
|
|
|
Address *net.TCPAddr
|
|
|
|
|
2017-03-06 07:43:34 +03:00
|
|
|
// NodeID is a public key which is used as node
|
|
|
|
// identification.
|
2016-12-07 18:46:22 +03:00
|
|
|
NodeID *btcec.PublicKey
|
|
|
|
|
2017-03-06 07:43:34 +03:00
|
|
|
// RGBColor is used to customize their node's appearance in
|
|
|
|
// maps and graphs
|
2016-12-07 18:46:22 +03:00
|
|
|
RGBColor RGB
|
|
|
|
|
2017-03-09 01:21:50 +03:00
|
|
|
// TODO(roasbeef): add the global features here
|
|
|
|
|
2017-03-06 07:43:34 +03:00
|
|
|
// pad is used to reserve to additional bytes for future
|
|
|
|
// usage.
|
2016-12-07 18:46:22 +03:00
|
|
|
pad uint16
|
|
|
|
|
2017-03-06 07:43:34 +03:00
|
|
|
// Alias is used to customize their node's appearance in maps
|
|
|
|
// and graphs
|
2016-12-07 18:46:22 +03:00
|
|
|
Alias Alias
|
|
|
|
}
|
|
|
|
|
|
|
|
// A compile time check to ensure NodeAnnouncement implements the
|
|
|
|
// lnwire.Message interface.
|
|
|
|
var _ Message = (*NodeAnnouncement)(nil)
|
|
|
|
|
|
|
|
// Validate performs any necessary sanity checks to ensure all fields present
|
|
|
|
// on the NodeAnnouncement are valid.
|
|
|
|
//
|
|
|
|
// This is part of the lnwire.Message interface.
|
|
|
|
func (a *NodeAnnouncement) Validate() error {
|
2016-12-24 23:58:26 +03:00
|
|
|
// TODO(roasbeef): move validation to discovery service
|
|
|
|
return nil
|
2016-12-07 18:46:22 +03:00
|
|
|
}
|
|
|
|
|
2017-03-06 07:43:34 +03:00
|
|
|
// Decode deserializes a serialized NodeAnnouncement stored in the passed
|
|
|
|
// io.Reader observing the specified protocol version.
|
2016-12-07 18:46:22 +03:00
|
|
|
//
|
|
|
|
// This is part of the lnwire.Message interface.
|
|
|
|
func (c *NodeAnnouncement) Decode(r io.Reader, pver uint32) error {
|
|
|
|
err := readElements(r,
|
|
|
|
&c.Signature,
|
|
|
|
&c.Timestamp,
|
|
|
|
&c.Address,
|
|
|
|
&c.NodeID,
|
|
|
|
&c.RGBColor,
|
|
|
|
&c.pad,
|
|
|
|
&c.Alias,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-03-06 07:43:34 +03:00
|
|
|
// Encode serializes the target NodeAnnouncement into the passed io.Writer
|
|
|
|
// observing the protocol version specified.
|
2016-12-07 18:46:22 +03:00
|
|
|
//
|
|
|
|
// This is part of the lnwire.Message interface.
|
|
|
|
func (c *NodeAnnouncement) Encode(w io.Writer, pver uint32) error {
|
|
|
|
err := writeElements(w,
|
|
|
|
c.Signature,
|
|
|
|
c.Timestamp,
|
|
|
|
c.Address,
|
|
|
|
c.NodeID,
|
|
|
|
c.RGBColor,
|
|
|
|
c.pad,
|
|
|
|
c.Alias,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command returns the integer uniquely identifying this message type on the
|
|
|
|
// wire.
|
|
|
|
//
|
|
|
|
// This is part of the lnwire.Message interface.
|
|
|
|
func (c *NodeAnnouncement) Command() uint32 {
|
|
|
|
return CmdNodeAnnoucmentMessage
|
|
|
|
}
|
|
|
|
|
|
|
|
// MaxPayloadLength returns the maximum allowed payload size for this message
|
|
|
|
// observing the specified protocol version.
|
|
|
|
//
|
|
|
|
// This is part of the lnwire.Message interface.
|
|
|
|
func (c *NodeAnnouncement) MaxPayloadLength(pver uint32) uint32 {
|
|
|
|
var length uint32
|
|
|
|
|
|
|
|
// Signature - 64 bytes
|
|
|
|
length += 64
|
|
|
|
|
|
|
|
// Timestamp - 4 bytes
|
|
|
|
length += 4
|
|
|
|
|
|
|
|
// Ipv6 - 16 bytes
|
|
|
|
length += 16
|
|
|
|
|
2016-12-24 23:46:42 +03:00
|
|
|
// Port - 4 bytes
|
|
|
|
length += 4
|
2016-12-07 18:46:22 +03:00
|
|
|
|
2016-12-24 23:46:42 +03:00
|
|
|
// NodeID - 33 bytes
|
|
|
|
length += 33
|
2016-12-07 18:46:22 +03:00
|
|
|
|
|
|
|
// RGBColor - 3 bytes
|
|
|
|
length += 3
|
|
|
|
|
|
|
|
// pad - 2 bytes
|
|
|
|
length += 2
|
|
|
|
|
|
|
|
// Alias - 32 bytes
|
|
|
|
length += 32
|
|
|
|
|
2016-12-24 23:46:42 +03:00
|
|
|
// 158
|
2016-12-07 18:46:22 +03:00
|
|
|
return length
|
|
|
|
}
|
|
|
|
|
|
|
|
// dataToSign...
|
|
|
|
func (c *NodeAnnouncement) DataToSign() ([]byte, error) {
|
|
|
|
|
|
|
|
// We should not include the signatures itself.
|
|
|
|
var w bytes.Buffer
|
|
|
|
err := writeElements(&w,
|
|
|
|
c.Timestamp,
|
|
|
|
c.Address,
|
|
|
|
c.NodeID,
|
|
|
|
c.RGBColor,
|
|
|
|
c.pad,
|
|
|
|
c.Alias,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return w.Bytes(), nil
|
|
|
|
}
|