2017-03-19 21:40:25 +03:00
|
|
|
package discovery
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
prand "math/rand"
|
|
|
|
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"math/big"
|
|
|
|
|
|
|
|
"time"
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
"github.com/go-errors/errors"
|
2017-03-19 21:40:25 +03:00
|
|
|
"github.com/lightningnetwork/lnd/chainntnfs"
|
|
|
|
"github.com/lightningnetwork/lnd/channeldb"
|
2017-03-27 20:25:44 +03:00
|
|
|
"github.com/lightningnetwork/lnd/lnwallet"
|
2017-03-19 21:40:25 +03:00
|
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
|
|
|
"github.com/lightningnetwork/lnd/routing"
|
|
|
|
"github.com/roasbeef/btcd/btcec"
|
|
|
|
"github.com/roasbeef/btcd/chaincfg/chainhash"
|
|
|
|
"github.com/roasbeef/btcd/wire"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
testAddr = &net.TCPAddr{IP: (net.IP)([]byte{0xA, 0x0, 0x0, 0x1}),
|
|
|
|
Port: 9000}
|
|
|
|
testAddrs = []net.Addr{testAddr}
|
|
|
|
testFeatures = lnwire.NewFeatureVector([]lnwire.Feature{})
|
|
|
|
testSig = &btcec.Signature{
|
|
|
|
R: new(big.Int),
|
|
|
|
S: new(big.Int),
|
|
|
|
}
|
|
|
|
_, _ = testSig.R.SetString("63724406601629180062774974542967536251589935445068131219452686511677818569431", 10)
|
|
|
|
_, _ = testSig.S.SetString("18801056069249825825291287104931333862866033135609736119018462340006816851118", 10)
|
2017-03-28 22:08:14 +03:00
|
|
|
|
|
|
|
inputStr = "147caa76786596590baa4e98f5d9f48b86c7765e489f7a6ff3360fe5c674360b"
|
|
|
|
sha, _ = chainhash.NewHashFromStr(inputStr)
|
|
|
|
outpoint = wire.NewOutPoint(sha, 0)
|
|
|
|
|
|
|
|
bitcoinKeyPriv1, _ = btcec.NewPrivateKey(btcec.S256())
|
|
|
|
bitcoinKeyPub1 = bitcoinKeyPriv1.PubKey()
|
|
|
|
|
|
|
|
nodeKeyPriv1, _ = btcec.NewPrivateKey(btcec.S256())
|
|
|
|
nodeKeyPub1 = nodeKeyPriv1.PubKey()
|
|
|
|
|
|
|
|
bitcoinKeyPriv2, _ = btcec.NewPrivateKey(btcec.S256())
|
|
|
|
bitcoinKeyPub2 = bitcoinKeyPriv2.PubKey()
|
|
|
|
|
|
|
|
nodeKeyPriv2, _ = btcec.NewPrivateKey(btcec.S256())
|
|
|
|
nodeKeyPub2 = nodeKeyPriv2.PubKey()
|
|
|
|
|
|
|
|
trickleDelay = time.Millisecond * 300
|
|
|
|
proofMatureDelta uint32
|
2017-03-19 21:40:25 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
type mockGraphSource struct {
|
|
|
|
nodes []*channeldb.LightningNode
|
2017-03-28 22:08:14 +03:00
|
|
|
infos map[uint64]*channeldb.ChannelEdgeInfo
|
|
|
|
edges map[uint64][]*channeldb.ChannelEdgePolicy
|
2017-03-19 21:40:25 +03:00
|
|
|
bestHeight uint32
|
|
|
|
}
|
|
|
|
|
|
|
|
func newMockRouter(height uint32) *mockGraphSource {
|
|
|
|
return &mockGraphSource{
|
|
|
|
bestHeight: height,
|
2017-03-28 22:08:14 +03:00
|
|
|
infos: make(map[uint64]*channeldb.ChannelEdgeInfo),
|
|
|
|
edges: make(map[uint64][]*channeldb.ChannelEdgePolicy),
|
2017-03-19 21:40:25 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ routing.ChannelGraphSource = (*mockGraphSource)(nil)
|
|
|
|
|
|
|
|
func (r *mockGraphSource) AddNode(node *channeldb.LightningNode) error {
|
|
|
|
r.nodes = append(r.nodes, node)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
func (r *mockGraphSource) AddEdge(info *channeldb.ChannelEdgeInfo) error {
|
|
|
|
if _, ok := r.infos[info.ChannelID]; ok {
|
|
|
|
return errors.New("info already exist")
|
|
|
|
}
|
|
|
|
r.infos[info.ChannelID] = info
|
2017-03-19 21:40:25 +03:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
func (r *mockGraphSource) UpdateEdge(edge *channeldb.ChannelEdgePolicy) error {
|
|
|
|
r.edges[edge.ChannelID] = append(
|
|
|
|
r.edges[edge.ChannelID],
|
|
|
|
edge,
|
|
|
|
)
|
2017-03-19 21:40:25 +03:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *mockGraphSource) SelfEdges() ([]*channeldb.ChannelEdgePolicy, error) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *mockGraphSource) CurrentBlockHeight() (uint32, error) {
|
|
|
|
return r.bestHeight, nil
|
|
|
|
}
|
|
|
|
|
2017-03-27 20:00:38 +03:00
|
|
|
func (r *mockGraphSource) AddProof(chanID lnwire.ShortChannelID,
|
|
|
|
proof *channeldb.ChannelAuthProof) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-03-19 21:40:25 +03:00
|
|
|
func (r *mockGraphSource) ForEachNode(func(node *channeldb.LightningNode) error) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *mockGraphSource) ForAllOutgoingChannels(cb func(c *channeldb.ChannelEdgePolicy) error) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *mockGraphSource) ForEachChannel(func(chanInfo *channeldb.ChannelEdgeInfo,
|
|
|
|
e1, e2 *channeldb.ChannelEdgePolicy) error) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
func (r *mockGraphSource) GetChannelByID(chanID lnwire.ShortChannelID) (
|
|
|
|
*channeldb.ChannelEdgeInfo,
|
|
|
|
*channeldb.ChannelEdgePolicy,
|
|
|
|
*channeldb.ChannelEdgePolicy, error) {
|
|
|
|
|
|
|
|
chanInfo, ok := r.infos[chanID.ToUint64()]
|
|
|
|
if !ok {
|
|
|
|
return nil, nil, nil, errors.New("can't find channel info")
|
|
|
|
}
|
|
|
|
|
|
|
|
edges := r.edges[chanID.ToUint64()]
|
|
|
|
if len(edges) == 0 {
|
|
|
|
return chanInfo, nil, nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(edges) == 1 {
|
|
|
|
return chanInfo, edges[0], nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return chanInfo, edges[0], edges[1], nil
|
|
|
|
}
|
|
|
|
|
2017-03-19 21:40:25 +03:00
|
|
|
type mockNotifier struct {
|
|
|
|
clientCounter uint32
|
|
|
|
epochClients map[uint32]chan *chainntnfs.BlockEpoch
|
|
|
|
|
|
|
|
sync.RWMutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func newMockNotifier() *mockNotifier {
|
|
|
|
return &mockNotifier{
|
|
|
|
epochClients: make(map[uint32]chan *chainntnfs.BlockEpoch),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockNotifier) RegisterConfirmationsNtfn(txid *chainhash.Hash,
|
|
|
|
numConfs uint32) (*chainntnfs.ConfirmationEvent, error) {
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockNotifier) RegisterSpendNtfn(outpoint *wire.OutPoint) (*chainntnfs.SpendEvent, error) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockNotifier) notifyBlock(hash chainhash.Hash, height uint32) {
|
|
|
|
m.RLock()
|
|
|
|
defer m.RUnlock()
|
|
|
|
|
|
|
|
for _, client := range m.epochClients {
|
|
|
|
client <- &chainntnfs.BlockEpoch{
|
|
|
|
Height: int32(height),
|
|
|
|
Hash: &hash,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockNotifier) RegisterBlockEpochNtfn() (*chainntnfs.BlockEpochEvent, error) {
|
|
|
|
m.RLock()
|
|
|
|
defer m.RUnlock()
|
|
|
|
|
|
|
|
epochChan := make(chan *chainntnfs.BlockEpoch)
|
|
|
|
clientID := m.clientCounter
|
|
|
|
m.clientCounter++
|
|
|
|
m.epochClients[clientID] = epochChan
|
|
|
|
|
|
|
|
return &chainntnfs.BlockEpochEvent{
|
|
|
|
Epochs: epochChan,
|
|
|
|
Cancel: func() {},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockNotifier) Start() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *mockNotifier) Stop() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
type annBatch struct {
|
|
|
|
nodeAnn1 *lnwire.NodeAnnouncement
|
|
|
|
nodeAnn2 *lnwire.NodeAnnouncement
|
|
|
|
localChanAnn *lnwire.ChannelAnnouncement
|
|
|
|
remoteChanAnn *lnwire.ChannelAnnouncement
|
|
|
|
chanUpdAnn *lnwire.ChannelUpdateAnnouncement
|
|
|
|
localProofAnn *lnwire.AnnounceSignatures
|
|
|
|
remoteProofAnn *lnwire.AnnounceSignatures
|
|
|
|
}
|
|
|
|
|
|
|
|
func createAnnouncements(blockHeight uint32) (*annBatch, error) {
|
|
|
|
var err error
|
|
|
|
var batch annBatch
|
|
|
|
|
|
|
|
batch.nodeAnn1, err = createNodeAnnouncement(nodeKeyPriv1)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
batch.nodeAnn2, err = createNodeAnnouncement(nodeKeyPriv2)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
batch.remoteChanAnn, err = createRemoteChannelAnnouncement(blockHeight)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
batch.localProofAnn = &lnwire.AnnounceSignatures{
|
|
|
|
NodeSignature: batch.remoteChanAnn.NodeSig1,
|
|
|
|
BitcoinSignature: batch.remoteChanAnn.BitcoinSig1,
|
|
|
|
}
|
|
|
|
|
|
|
|
batch.remoteProofAnn = &lnwire.AnnounceSignatures{
|
|
|
|
NodeSignature: batch.remoteChanAnn.NodeSig2,
|
|
|
|
BitcoinSignature: batch.remoteChanAnn.BitcoinSig2,
|
|
|
|
}
|
|
|
|
|
|
|
|
batch.localChanAnn, err = createRemoteChannelAnnouncement(blockHeight)
|
2017-03-19 21:40:25 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-03-28 22:08:14 +03:00
|
|
|
batch.localChanAnn.BitcoinSig1 = nil
|
|
|
|
batch.localChanAnn.BitcoinSig2 = nil
|
|
|
|
batch.localChanAnn.NodeSig1 = nil
|
|
|
|
batch.localChanAnn.NodeSig2 = nil
|
2017-03-19 21:40:25 +03:00
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
batch.chanUpdAnn, err = createUpdateAnnouncement(blockHeight)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-03-27 20:25:44 +03:00
|
|
|
batch.localChanAnn.BitcoinSig1 = nil
|
|
|
|
batch.localChanAnn.BitcoinSig2 = nil
|
|
|
|
batch.localChanAnn.NodeSig1 = nil
|
|
|
|
batch.localChanAnn.NodeSig2 = nil
|
2017-03-19 21:40:25 +03:00
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
return &batch, nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func createNodeAnnouncement(priv *btcec.PrivateKey) (*lnwire.NodeAnnouncement,
|
|
|
|
error) {
|
2017-03-27 20:25:44 +03:00
|
|
|
var err error
|
2017-03-28 22:08:14 +03:00
|
|
|
|
|
|
|
alias, err := lnwire.NewAlias("kek" + string(priv.Serialize()))
|
2017-03-19 21:40:25 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
a := &lnwire.NodeAnnouncement{
|
2017-03-19 21:40:25 +03:00
|
|
|
Timestamp: uint32(prand.Int31()),
|
|
|
|
Addresses: testAddrs,
|
|
|
|
NodeID: priv.PubKey(),
|
|
|
|
Alias: alias,
|
|
|
|
Features: testFeatures,
|
2017-03-28 22:08:14 +03:00
|
|
|
}
|
|
|
|
|
2017-03-27 20:25:44 +03:00
|
|
|
signer := lnwallet.NewMessageSigner(nodeKeyPriv1)
|
|
|
|
if a.Signature, err = SignAnnouncement(signer, a); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
return a, nil
|
2017-03-19 21:40:25 +03:00
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
func createUpdateAnnouncement(blockHeight uint32) (*lnwire.ChannelUpdateAnnouncement,
|
|
|
|
error) {
|
2017-03-27 20:25:44 +03:00
|
|
|
var err error
|
2017-03-28 22:08:14 +03:00
|
|
|
|
|
|
|
a := &lnwire.ChannelUpdateAnnouncement{
|
2017-03-27 18:22:37 +03:00
|
|
|
ShortChannelID: lnwire.ShortChannelID{
|
2017-03-19 21:40:25 +03:00
|
|
|
BlockHeight: blockHeight,
|
|
|
|
},
|
|
|
|
Timestamp: uint32(prand.Int31()),
|
|
|
|
TimeLockDelta: uint16(prand.Int63()),
|
|
|
|
HtlcMinimumMsat: uint32(prand.Int31()),
|
|
|
|
FeeBaseMsat: uint32(prand.Int31()),
|
|
|
|
FeeProportionalMillionths: uint32(prand.Int31()),
|
|
|
|
}
|
2017-03-28 22:08:14 +03:00
|
|
|
|
2017-03-27 20:25:44 +03:00
|
|
|
signer := lnwallet.NewMessageSigner(nodeKeyPriv1)
|
|
|
|
if a.Signature, err = SignAnnouncement(signer, a); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
return a, nil
|
2017-03-19 21:40:25 +03:00
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
func createRemoteChannelAnnouncement(blockHeight uint32) (*lnwire.ChannelAnnouncement,
|
|
|
|
error) {
|
2017-03-27 20:25:44 +03:00
|
|
|
var err error
|
2017-03-19 21:40:25 +03:00
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
a := &lnwire.ChannelAnnouncement{
|
|
|
|
ShortChannelID: lnwire.ShortChannelID{
|
|
|
|
BlockHeight: blockHeight,
|
|
|
|
TxIndex: 0,
|
|
|
|
TxPosition: 0,
|
|
|
|
},
|
|
|
|
NodeID1: nodeKeyPub1,
|
|
|
|
NodeID2: nodeKeyPub2,
|
|
|
|
BitcoinKey1: bitcoinKeyPub1,
|
|
|
|
BitcoinKey2: bitcoinKeyPub2,
|
2017-03-27 20:25:44 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
signer := lnwallet.NewMessageSigner(nodeKeyPriv1)
|
|
|
|
if a.NodeSig1, err = SignAnnouncement(signer, a); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
signer = lnwallet.NewMessageSigner(nodeKeyPriv2)
|
|
|
|
if a.NodeSig2, err = SignAnnouncement(signer, a); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-03-28 22:08:14 +03:00
|
|
|
|
2017-03-27 20:25:44 +03:00
|
|
|
hash := chainhash.DoubleHashB(nodeKeyPub1.SerializeCompressed())
|
|
|
|
a.BitcoinSig1, err = bitcoinKeyPriv1.Sign(hash)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
hash = chainhash.DoubleHashB(nodeKeyPub2.SerializeCompressed())
|
|
|
|
a.BitcoinSig2, err = bitcoinKeyPriv2.Sign(hash)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2017-03-19 21:40:25 +03:00
|
|
|
}
|
2017-03-28 22:08:14 +03:00
|
|
|
|
|
|
|
return a, nil
|
2017-03-19 21:40:25 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
type testCtx struct {
|
2017-03-28 22:08:14 +03:00
|
|
|
discovery *Discovery
|
|
|
|
router *mockGraphSource
|
|
|
|
notifier *mockNotifier
|
2017-03-19 21:40:25 +03:00
|
|
|
broadcastedMessage chan lnwire.Message
|
|
|
|
}
|
|
|
|
|
|
|
|
func createTestCtx(startHeight uint32) (*testCtx, func(), error) {
|
|
|
|
// Next we'll initialize an instance of the channel router with mock
|
|
|
|
// versions of the chain and channel notifier. As we don't need to test
|
|
|
|
// any p2p functionality, the peer send and switch send,
|
|
|
|
// broadcast functions won't be populated.
|
|
|
|
notifier := newMockNotifier()
|
|
|
|
router := newMockRouter(startHeight)
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
broadcastedMessage := make(chan lnwire.Message, 10)
|
2017-03-19 21:40:25 +03:00
|
|
|
discovery, err := New(Config{
|
|
|
|
Notifier: notifier,
|
|
|
|
Broadcast: func(_ *btcec.PublicKey, msgs ...lnwire.Message) error {
|
|
|
|
for _, msg := range msgs {
|
|
|
|
broadcastedMessage <- msg
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
},
|
2017-03-28 22:08:14 +03:00
|
|
|
SendToPeer: func(target *btcec.PublicKey, msg ...lnwire.Message) error {
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
Router: router,
|
|
|
|
TrickleDelay: trickleDelay,
|
|
|
|
ProofMatureDelta: proofMatureDelta,
|
2017-03-19 21:40:25 +03:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, fmt.Errorf("unable to create router %v", err)
|
|
|
|
}
|
|
|
|
if err := discovery.Start(); err != nil {
|
|
|
|
return nil, nil, fmt.Errorf("unable to start router: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanUp := func() {
|
|
|
|
discovery.Stop()
|
|
|
|
}
|
|
|
|
|
|
|
|
return &testCtx{
|
|
|
|
router: router,
|
|
|
|
notifier: notifier,
|
|
|
|
discovery: discovery,
|
|
|
|
broadcastedMessage: broadcastedMessage,
|
|
|
|
}, cleanUp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestProcessAnnouncement checks that mature announcements are propagated to
|
|
|
|
// the router subsystem.
|
|
|
|
func TestProcessAnnouncement(t *testing.T) {
|
|
|
|
ctx, cleanup, err := createTestCtx(0)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("can't create context: %v", err)
|
|
|
|
}
|
|
|
|
defer cleanup()
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
// Create node valid, signed announcement, process it with with
|
|
|
|
// discovery service, check that valid announcement have been
|
|
|
|
// propagated farther into the lightning network, and check that we
|
|
|
|
// added new node into router.
|
|
|
|
na, err := createNodeAnnouncement(nodeKeyPriv1)
|
2017-03-19 21:40:25 +03:00
|
|
|
if err != nil {
|
2017-03-28 22:08:14 +03:00
|
|
|
t.Fatalf("can't create node announcement: %v", err)
|
2017-03-19 21:40:25 +03:00
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
err = <-ctx.discovery.ProcessRemoteAnnouncement(na, na.NodeID)
|
2017-03-19 21:40:25 +03:00
|
|
|
if err != nil {
|
2017-03-28 22:08:14 +03:00
|
|
|
t.Fatalf("can't process remote announcement: %v", err)
|
2017-03-19 21:40:25 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
select {
|
|
|
|
case <-ctx.broadcastedMessage:
|
2017-03-28 22:08:14 +03:00
|
|
|
case <-time.After(2 * trickleDelay):
|
2017-03-19 21:40:25 +03:00
|
|
|
t.Fatal("announcememt wasn't proceeded")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(ctx.router.nodes) != 1 {
|
|
|
|
t.Fatalf("node wasn't added to router: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
// Pretending that we receive the valid channel announcement from
|
|
|
|
// remote side, and check that we broadcasted it to the our network,
|
|
|
|
// and added channel info in the router.
|
|
|
|
ca, err := createRemoteChannelAnnouncement(0)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("can't create channel announcement: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = <-ctx.discovery.ProcessRemoteAnnouncement(ca, na.NodeID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("can't process remote announcement: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-03-19 21:40:25 +03:00
|
|
|
select {
|
|
|
|
case <-ctx.broadcastedMessage:
|
2017-03-28 22:08:14 +03:00
|
|
|
case <-time.After(2 * trickleDelay):
|
2017-03-19 21:40:25 +03:00
|
|
|
t.Fatal("announcememt wasn't proceeded")
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
if len(ctx.router.infos) != 1 {
|
2017-03-19 21:40:25 +03:00
|
|
|
t.Fatalf("edge wasn't added to router: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
// Pretending that we received valid channel policy update from remote
|
|
|
|
// side, and check that we broadcasted it to the other network, and
|
|
|
|
// added updates to the router.
|
|
|
|
ua, err := createUpdateAnnouncement(0)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("can't create update announcement: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = <-ctx.discovery.ProcessRemoteAnnouncement(ua, na.NodeID)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("can't process remote announcement: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-03-19 21:40:25 +03:00
|
|
|
select {
|
|
|
|
case <-ctx.broadcastedMessage:
|
2017-03-28 22:08:14 +03:00
|
|
|
case <-time.After(2 * trickleDelay):
|
2017-03-19 21:40:25 +03:00
|
|
|
t.Fatal("announcememt wasn't proceeded")
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
if len(ctx.router.edges) != 1 {
|
2017-03-19 21:40:25 +03:00
|
|
|
t.Fatalf("edge update wasn't added to router: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
// TestPrematureAnnouncement checks that premature announcements are
|
2017-03-19 21:40:25 +03:00
|
|
|
// not propagated to the router subsystem until block with according
|
|
|
|
// block height received.
|
|
|
|
func TestPrematureAnnouncement(t *testing.T) {
|
|
|
|
ctx, cleanup, err := createTestCtx(0)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("can't create context: %v", err)
|
|
|
|
}
|
|
|
|
defer cleanup()
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
na, err := createNodeAnnouncement(nodeKeyPriv1)
|
2017-03-19 21:40:25 +03:00
|
|
|
if err != nil {
|
2017-03-28 22:08:14 +03:00
|
|
|
t.Fatalf("can't create node announcement: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pretending that we receive the valid channel announcement from
|
|
|
|
// remote side, but block height of this announcement is greater than
|
|
|
|
// highest know to us, for that reason it should be added to the
|
|
|
|
// repeat/premature batch.
|
|
|
|
ca, err := createRemoteChannelAnnouncement(1)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("can't create channel announcement: %v", err)
|
2017-03-19 21:40:25 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
select {
|
2017-03-28 22:08:14 +03:00
|
|
|
case <-ctx.discovery.ProcessRemoteAnnouncement(ca, na.NodeID):
|
2017-03-19 21:40:25 +03:00
|
|
|
t.Fatal("announcement was proceeded")
|
|
|
|
case <-time.After(100 * time.Millisecond):
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
if len(ctx.router.infos) != 0 {
|
2017-03-19 21:40:25 +03:00
|
|
|
t.Fatal("edge was added to router")
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
// Pretending that we receive the valid channel update announcement from
|
|
|
|
// remote side, but block height of this announcement is greater than
|
|
|
|
// highest know to us, for that reason it should be added to the
|
|
|
|
// repeat/premature batch.
|
|
|
|
ua, err := createUpdateAnnouncement(1)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("can't create update announcement: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-03-19 21:40:25 +03:00
|
|
|
select {
|
2017-03-28 22:08:14 +03:00
|
|
|
case <-ctx.discovery.ProcessRemoteAnnouncement(ua, na.NodeID):
|
2017-03-19 21:40:25 +03:00
|
|
|
t.Fatal("announcement was proceeded")
|
|
|
|
case <-time.After(100 * time.Millisecond):
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
if len(ctx.router.edges) != 0 {
|
2017-03-19 21:40:25 +03:00
|
|
|
t.Fatal("edge update was added to router")
|
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
// Generate new block and waiting the previously added announcements
|
|
|
|
// to be proceeded.
|
2017-03-19 21:40:25 +03:00
|
|
|
newBlock := &wire.MsgBlock{}
|
|
|
|
ctx.notifier.notifyBlock(newBlock.Header.BlockHash(), 1)
|
|
|
|
|
|
|
|
select {
|
|
|
|
case <-ctx.broadcastedMessage:
|
2017-03-28 22:08:14 +03:00
|
|
|
case <-time.After(2 * trickleDelay):
|
|
|
|
t.Fatal("announcememt wasn't broadcasted")
|
2017-03-19 21:40:25 +03:00
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
if len(ctx.router.infos) != 1 {
|
2017-03-19 21:40:25 +03:00
|
|
|
t.Fatalf("edge was't added to router: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
select {
|
|
|
|
case <-ctx.broadcastedMessage:
|
2017-03-28 22:08:14 +03:00
|
|
|
case <-time.After(2 * trickleDelay):
|
|
|
|
t.Fatal("announcememt wasn't broadcasted")
|
2017-03-19 21:40:25 +03:00
|
|
|
}
|
|
|
|
|
2017-03-28 22:08:14 +03:00
|
|
|
if len(ctx.router.edges) != 1 {
|
2017-03-19 21:40:25 +03:00
|
|
|
t.Fatalf("edge update wasn't added to router: %v", err)
|
|
|
|
}
|
|
|
|
}
|
2017-03-28 22:08:14 +03:00
|
|
|
|
|
|
|
// TestSignatureAnnouncement....
|
|
|
|
func TestSignatureAnnouncement(t *testing.T) {
|
|
|
|
ctx, cleanup, err := createTestCtx(proofMatureDelta)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("can't create context: %v", err)
|
|
|
|
}
|
|
|
|
defer cleanup()
|
|
|
|
|
|
|
|
batch, err := createAnnouncements(0)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("can't generate announcements: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
localKey := batch.nodeAnn1.NodeID
|
|
|
|
remoteKey := batch.nodeAnn2.NodeID
|
|
|
|
|
|
|
|
// Recreate lightning network topology. Initialize router with
|
|
|
|
// channel between two nodes.
|
|
|
|
err = <-ctx.discovery.ProcessLocalAnnouncement(batch.localChanAnn, localKey)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unable to process :%v", err)
|
|
|
|
}
|
|
|
|
select {
|
|
|
|
case <-ctx.broadcastedMessage:
|
|
|
|
t.Fatal("channel announcement was broadcasted")
|
|
|
|
case <-time.After(2 * trickleDelay):
|
|
|
|
}
|
|
|
|
|
|
|
|
err = <-ctx.discovery.ProcessLocalAnnouncement(batch.chanUpdAnn, localKey)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unable to process :%v", err)
|
|
|
|
}
|
|
|
|
select {
|
|
|
|
case <-ctx.broadcastedMessage:
|
|
|
|
t.Fatal("channel update announcement was broadcasted")
|
|
|
|
case <-time.After(2 * trickleDelay):
|
|
|
|
}
|
|
|
|
|
|
|
|
err = <-ctx.discovery.ProcessRemoteAnnouncement(batch.chanUpdAnn, remoteKey)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unable to process :%v", err)
|
|
|
|
}
|
|
|
|
select {
|
|
|
|
case <-ctx.broadcastedMessage:
|
|
|
|
t.Fatal("channel update announcement was broadcasted")
|
|
|
|
case <-time.After(2 * trickleDelay):
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pretending that we receive local channel announcement from funding
|
|
|
|
// manager, thereby kick off the announcement exchange process.
|
|
|
|
err = <-ctx.discovery.ProcessLocalAnnouncement(batch.localProofAnn, localKey)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unable to process :%v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
select {
|
|
|
|
case <-ctx.broadcastedMessage:
|
|
|
|
t.Fatal("announcements were broadcasted")
|
|
|
|
case <-time.After(2 * trickleDelay):
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(ctx.discovery.waitingProofs) != 1 {
|
|
|
|
t.Fatal("local proof annoucement should be stored")
|
|
|
|
}
|
|
|
|
|
|
|
|
err = <-ctx.discovery.ProcessRemoteAnnouncement(batch.remoteProofAnn, remoteKey)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unable to process :%v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < 3; i++ {
|
|
|
|
select {
|
|
|
|
case <-ctx.broadcastedMessage:
|
|
|
|
case <-time.After(time.Second):
|
|
|
|
t.Fatal("announcement wasn't broadcasted")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|