2015-12-26 19:35:15 +01:00
|
|
|
package channeldb
|
2015-12-26 00:00:53 +01:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2017-07-29 20:24:28 +02:00
|
|
|
"math/rand"
|
2017-01-23 08:31:01 +01:00
|
|
|
"net"
|
2016-06-21 06:39:50 +02:00
|
|
|
"reflect"
|
2017-11-10 05:56:04 +01:00
|
|
|
"runtime"
|
2023-02-10 09:32:09 +01:00
|
|
|
"sync/atomic"
|
2015-12-26 00:00:53 +01:00
|
|
|
"testing"
|
|
|
|
|
2022-02-23 14:48:00 +01:00
|
|
|
"github.com/btcsuite/btcd/btcec/v2"
|
|
|
|
"github.com/btcsuite/btcd/btcutil"
|
2018-06-05 03:34:16 +02:00
|
|
|
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
|
|
|
"github.com/btcsuite/btcd/wire"
|
|
|
|
_ "github.com/btcsuite/btcwallet/walletdb/bdb"
|
2018-07-18 04:10:07 +02:00
|
|
|
"github.com/davecgh/go-spew/spew"
|
2022-11-18 12:15:22 +01:00
|
|
|
"github.com/lightningnetwork/lnd/channeldb/models"
|
2020-01-20 10:57:34 +01:00
|
|
|
"github.com/lightningnetwork/lnd/clock"
|
2018-07-18 04:10:07 +02:00
|
|
|
"github.com/lightningnetwork/lnd/keychain"
|
2021-04-26 19:08:11 +02:00
|
|
|
"github.com/lightningnetwork/lnd/kvdb"
|
2023-05-19 16:58:05 +02:00
|
|
|
"github.com/lightningnetwork/lnd/lnmock"
|
2021-01-07 22:26:24 +01:00
|
|
|
"github.com/lightningnetwork/lnd/lntest/channels"
|
2018-07-18 04:10:07 +02:00
|
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
|
|
|
"github.com/lightningnetwork/lnd/shachain"
|
2023-11-06 21:36:31 +01:00
|
|
|
"github.com/lightningnetwork/lnd/tlv"
|
2022-02-07 13:58:21 +01:00
|
|
|
"github.com/stretchr/testify/require"
|
2015-12-26 00:00:53 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2017-01-05 22:56:27 +01:00
|
|
|
key = [chainhash.HashSize]byte{
|
2015-12-26 00:00:53 +01:00
|
|
|
0x81, 0xb6, 0x37, 0xd8, 0xfc, 0xd2, 0xc6, 0xda,
|
|
|
|
0x68, 0x59, 0xe6, 0x96, 0x31, 0x13, 0xa1, 0x17,
|
|
|
|
0xd, 0xe7, 0x93, 0xe4, 0xb7, 0x25, 0xb8, 0x4d,
|
|
|
|
0x1e, 0xb, 0x4c, 0xf9, 0x9e, 0xc5, 0x8c, 0xe9,
|
|
|
|
}
|
2017-01-05 22:56:27 +01:00
|
|
|
rev = [chainhash.HashSize]byte{
|
2015-12-31 07:28:00 +01:00
|
|
|
0x51, 0xb6, 0x37, 0xd8, 0xfc, 0xd2, 0xc6, 0xda,
|
|
|
|
0x48, 0x59, 0xe6, 0x96, 0x31, 0x13, 0xa1, 0x17,
|
|
|
|
0x2d, 0xe7, 0x93, 0xe4,
|
|
|
|
}
|
2022-02-23 14:48:00 +01:00
|
|
|
privKey, pubKey = btcec.PrivKeyFromBytes(key[:])
|
2018-01-31 05:19:40 +01:00
|
|
|
|
|
|
|
wireSig, _ = lnwire.NewSigFromSignature(testSig)
|
2020-01-20 10:57:34 +01:00
|
|
|
|
|
|
|
testClock = clock.NewTestClock(testNow)
|
2020-02-05 14:39:31 +01:00
|
|
|
|
|
|
|
// defaultPendingHeight is the default height at which we set
|
|
|
|
// channels to pending.
|
|
|
|
defaultPendingHeight = 100
|
|
|
|
|
|
|
|
// defaultAddr is the default address that we mark test channels pending
|
|
|
|
// with.
|
|
|
|
defaultAddr = &net.TCPAddr{
|
|
|
|
IP: net.ParseIP("127.0.0.1"),
|
|
|
|
Port: 18555,
|
|
|
|
}
|
2020-08-10 20:11:12 +02:00
|
|
|
|
2022-04-08 01:36:26 +02:00
|
|
|
// keyLocIndex is the KeyLocator Index we use for
|
|
|
|
// TestKeyLocatorEncoding.
|
2020-08-10 20:11:12 +02:00
|
|
|
keyLocIndex = uint32(2049)
|
2022-04-08 01:36:26 +02:00
|
|
|
|
|
|
|
// dummyLocalOutputIndex specifics a default value for our output index
|
|
|
|
// in this test.
|
|
|
|
dummyLocalOutputIndex = uint32(0)
|
|
|
|
|
|
|
|
// dummyRemoteOutIndex specifics a default value for their output index
|
|
|
|
// in this test.
|
|
|
|
dummyRemoteOutIndex = uint32(1)
|
2023-02-10 09:32:09 +01:00
|
|
|
|
|
|
|
// uniqueOutputIndex is used to create a unique funding outpoint.
|
|
|
|
//
|
|
|
|
// NOTE: must be incremented when used.
|
|
|
|
uniqueOutputIndex = atomic.Uint32{}
|
2015-12-26 00:00:53 +01:00
|
|
|
)
|
|
|
|
|
2020-02-05 14:39:31 +01:00
|
|
|
// testChannelParams is a struct which details the specifics of how a channel
|
|
|
|
// should be created.
|
|
|
|
type testChannelParams struct {
|
|
|
|
// channel is the channel that will be written to disk.
|
|
|
|
channel *OpenChannel
|
|
|
|
|
|
|
|
// addr is the address that the channel will be synced pending with.
|
|
|
|
addr *net.TCPAddr
|
|
|
|
|
|
|
|
// pendingHeight is the height that the channel should be recorded as
|
|
|
|
// pending.
|
|
|
|
pendingHeight uint32
|
|
|
|
|
|
|
|
// openChannel is set to true if the channel should be fully marked as
|
|
|
|
// open if this is false, the channel will be left in pending state.
|
|
|
|
openChannel bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// testChannelOption is a functional option which can be used to alter the
|
|
|
|
// default channel that is creates for testing.
|
|
|
|
type testChannelOption func(params *testChannelParams)
|
|
|
|
|
|
|
|
// pendingHeightOption is an option which can be used to set the height the
|
|
|
|
// channel is marked as pending at.
|
|
|
|
func pendingHeightOption(height uint32) testChannelOption {
|
|
|
|
return func(params *testChannelParams) {
|
|
|
|
params.pendingHeight = height
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// openChannelOption is an option which can be used to create a test channel
|
|
|
|
// that is open.
|
|
|
|
func openChannelOption() testChannelOption {
|
|
|
|
return func(params *testChannelParams) {
|
|
|
|
params.openChannel = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// localHtlcsOption is an option which allows setting of htlcs on the local
|
|
|
|
// commitment.
|
|
|
|
func localHtlcsOption(htlcs []HTLC) testChannelOption {
|
|
|
|
return func(params *testChannelParams) {
|
|
|
|
params.channel.LocalCommitment.Htlcs = htlcs
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// remoteHtlcsOption is an option which allows setting of htlcs on the remote
|
|
|
|
// commitment.
|
|
|
|
func remoteHtlcsOption(htlcs []HTLC) testChannelOption {
|
|
|
|
return func(params *testChannelParams) {
|
|
|
|
params.channel.RemoteCommitment.Htlcs = htlcs
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-22 11:56:22 +02:00
|
|
|
// loadFwdPkgs is a helper method that reads all forwarding packages for a
|
|
|
|
// particular packager.
|
|
|
|
func loadFwdPkgs(t *testing.T, db kvdb.Backend,
|
|
|
|
packager FwdPackager) []*FwdPkg {
|
|
|
|
|
|
|
|
var (
|
|
|
|
fwdPkgs []*FwdPkg
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
err = kvdb.View(db, func(tx kvdb.RTx) error {
|
|
|
|
fwdPkgs, err = packager.LoadFwdPkgs(tx)
|
|
|
|
return err
|
|
|
|
}, func() {})
|
|
|
|
require.NoError(t, err, "unable to load fwd pkgs")
|
|
|
|
|
|
|
|
return fwdPkgs
|
|
|
|
}
|
|
|
|
|
2020-02-05 14:39:31 +01:00
|
|
|
// localShutdownOption is an option which sets the local upfront shutdown
|
|
|
|
// script for the channel.
|
|
|
|
func localShutdownOption(addr lnwire.DeliveryAddress) testChannelOption {
|
|
|
|
return func(params *testChannelParams) {
|
|
|
|
params.channel.LocalShutdownScript = addr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// remoteShutdownOption is an option which sets the remote upfront shutdown
|
|
|
|
// script for the channel.
|
|
|
|
func remoteShutdownOption(addr lnwire.DeliveryAddress) testChannelOption {
|
|
|
|
return func(params *testChannelParams) {
|
|
|
|
params.channel.RemoteShutdownScript = addr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// fundingPointOption is an option which sets the funding outpoint of the
|
|
|
|
// channel.
|
|
|
|
func fundingPointOption(chanPoint wire.OutPoint) testChannelOption {
|
|
|
|
return func(params *testChannelParams) {
|
|
|
|
params.channel.FundingOutpoint = chanPoint
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-06 09:21:12 +01:00
|
|
|
// channelIDOption is an option which sets the short channel ID of the channel.
|
|
|
|
var channelIDOption = func(chanID lnwire.ShortChannelID) testChannelOption {
|
|
|
|
return func(params *testChannelParams) {
|
|
|
|
params.channel.ShortChannelID = chanID
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-05 14:39:31 +01:00
|
|
|
// createTestChannel writes a test channel to the database. It takes a set of
|
|
|
|
// functional options which can be used to overwrite the default of creating
|
|
|
|
// a pending channel that was broadcast at height 100.
|
2021-09-21 19:18:17 +02:00
|
|
|
func createTestChannel(t *testing.T, cdb *ChannelStateDB,
|
2020-02-05 14:39:31 +01:00
|
|
|
opts ...testChannelOption) *OpenChannel {
|
|
|
|
|
|
|
|
// Create a default set of parameters.
|
|
|
|
params := &testChannelParams{
|
|
|
|
channel: createTestChannelState(t, cdb),
|
|
|
|
addr: defaultAddr,
|
|
|
|
openChannel: false,
|
|
|
|
pendingHeight: uint32(defaultPendingHeight),
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply all functional options to the test channel params.
|
|
|
|
for _, o := range opts {
|
|
|
|
o(params)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mark the channel as pending.
|
|
|
|
err := params.channel.SyncPending(params.addr, params.pendingHeight)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unable to save and serialize channel "+
|
|
|
|
"state: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the parameters do not specify that we should open the channel
|
|
|
|
// fully, we return the pending channel.
|
|
|
|
if !params.openChannel {
|
|
|
|
return params.channel
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mark the channel as open with the short channel id provided.
|
|
|
|
err = params.channel.MarkAsOpen(params.channel.ShortChannelID)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to mark channel open")
|
2020-02-05 14:39:31 +01:00
|
|
|
|
|
|
|
return params.channel
|
|
|
|
}
|
|
|
|
|
2021-09-21 19:18:17 +02:00
|
|
|
func createTestChannelState(t *testing.T, cdb *ChannelStateDB) *OpenChannel {
|
2016-12-14 15:01:48 +01:00
|
|
|
// Simulate 1000 channel updates.
|
2017-02-24 20:53:49 +01:00
|
|
|
producer, err := shachain.NewRevocationProducerFromBytes(key[:])
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "could not get producer")
|
2016-12-14 15:01:48 +01:00
|
|
|
store := shachain.NewRevocationStore()
|
2017-07-29 20:24:28 +02:00
|
|
|
for i := 0; i < 1; i++ {
|
2016-12-14 15:01:48 +01:00
|
|
|
preImage, err := producer.AtIndex(uint64(i))
|
2016-03-24 06:39:52 +01:00
|
|
|
if err != nil {
|
2020-02-05 14:39:31 +01:00
|
|
|
t.Fatalf("could not get "+
|
|
|
|
"preimage: %v", err)
|
2016-03-24 06:39:52 +01:00
|
|
|
}
|
|
|
|
|
2017-12-01 05:35:48 +01:00
|
|
|
if err := store.AddNextEntry(preImage); err != nil {
|
2020-02-05 14:39:31 +01:00
|
|
|
t.Fatalf("could not add entry: %v", err)
|
2016-03-24 06:39:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-29 20:24:28 +02:00
|
|
|
localCfg := ChannelConfig{
|
|
|
|
ChannelConstraints: ChannelConstraints{
|
|
|
|
DustLimit: btcutil.Amount(rand.Int63()),
|
2017-08-22 07:51:45 +02:00
|
|
|
MaxPendingAmount: lnwire.MilliSatoshi(rand.Int63()),
|
2017-07-29 20:24:28 +02:00
|
|
|
ChanReserve: btcutil.Amount(rand.Int63()),
|
2017-08-22 07:51:45 +02:00
|
|
|
MinHTLC: lnwire.MilliSatoshi(rand.Int63()),
|
2017-07-29 20:24:28 +02:00
|
|
|
MaxAcceptedHtlcs: uint16(rand.Int31()),
|
2018-12-10 22:56:41 +01:00
|
|
|
CsvDelay: uint16(rand.Int31()),
|
2017-07-29 20:24:28 +02:00
|
|
|
},
|
2018-02-18 00:14:07 +01:00
|
|
|
MultiSigKey: keychain.KeyDescriptor{
|
|
|
|
PubKey: privKey.PubKey(),
|
|
|
|
},
|
|
|
|
RevocationBasePoint: keychain.KeyDescriptor{
|
|
|
|
PubKey: privKey.PubKey(),
|
|
|
|
},
|
|
|
|
PaymentBasePoint: keychain.KeyDescriptor{
|
|
|
|
PubKey: privKey.PubKey(),
|
|
|
|
},
|
|
|
|
DelayBasePoint: keychain.KeyDescriptor{
|
|
|
|
PubKey: privKey.PubKey(),
|
|
|
|
},
|
|
|
|
HtlcBasePoint: keychain.KeyDescriptor{
|
|
|
|
PubKey: privKey.PubKey(),
|
|
|
|
},
|
2017-07-29 20:24:28 +02:00
|
|
|
}
|
|
|
|
remoteCfg := ChannelConfig{
|
|
|
|
ChannelConstraints: ChannelConstraints{
|
|
|
|
DustLimit: btcutil.Amount(rand.Int63()),
|
2017-08-22 07:51:45 +02:00
|
|
|
MaxPendingAmount: lnwire.MilliSatoshi(rand.Int63()),
|
2017-07-29 20:24:28 +02:00
|
|
|
ChanReserve: btcutil.Amount(rand.Int63()),
|
2017-08-22 07:51:45 +02:00
|
|
|
MinHTLC: lnwire.MilliSatoshi(rand.Int63()),
|
2017-07-29 20:24:28 +02:00
|
|
|
MaxAcceptedHtlcs: uint16(rand.Int31()),
|
2018-12-10 22:56:41 +01:00
|
|
|
CsvDelay: uint16(rand.Int31()),
|
2017-07-29 20:24:28 +02:00
|
|
|
},
|
2018-02-18 00:14:07 +01:00
|
|
|
MultiSigKey: keychain.KeyDescriptor{
|
|
|
|
PubKey: privKey.PubKey(),
|
2018-08-14 04:21:37 +02:00
|
|
|
KeyLocator: keychain.KeyLocator{
|
|
|
|
Family: keychain.KeyFamilyMultiSig,
|
|
|
|
Index: 9,
|
|
|
|
},
|
2018-02-18 00:14:07 +01:00
|
|
|
},
|
|
|
|
RevocationBasePoint: keychain.KeyDescriptor{
|
|
|
|
PubKey: privKey.PubKey(),
|
2018-08-14 04:21:37 +02:00
|
|
|
KeyLocator: keychain.KeyLocator{
|
|
|
|
Family: keychain.KeyFamilyRevocationBase,
|
|
|
|
Index: 8,
|
|
|
|
},
|
2018-02-18 00:14:07 +01:00
|
|
|
},
|
|
|
|
PaymentBasePoint: keychain.KeyDescriptor{
|
|
|
|
PubKey: privKey.PubKey(),
|
2018-08-14 04:21:37 +02:00
|
|
|
KeyLocator: keychain.KeyLocator{
|
|
|
|
Family: keychain.KeyFamilyPaymentBase,
|
|
|
|
Index: 7,
|
|
|
|
},
|
2018-02-18 00:14:07 +01:00
|
|
|
},
|
|
|
|
DelayBasePoint: keychain.KeyDescriptor{
|
|
|
|
PubKey: privKey.PubKey(),
|
2018-08-14 04:21:37 +02:00
|
|
|
KeyLocator: keychain.KeyLocator{
|
|
|
|
Family: keychain.KeyFamilyDelayBase,
|
|
|
|
Index: 6,
|
|
|
|
},
|
2018-02-18 00:14:07 +01:00
|
|
|
},
|
|
|
|
HtlcBasePoint: keychain.KeyDescriptor{
|
|
|
|
PubKey: privKey.PubKey(),
|
2018-08-14 04:21:37 +02:00
|
|
|
KeyLocator: keychain.KeyLocator{
|
|
|
|
Family: keychain.KeyFamilyHtlcBase,
|
|
|
|
Index: 5,
|
|
|
|
},
|
2018-02-18 00:14:07 +01:00
|
|
|
},
|
2017-07-29 20:24:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
chanID := lnwire.NewShortChanIDFromInt(uint64(rand.Int63()))
|
|
|
|
|
2023-02-10 09:32:09 +01:00
|
|
|
// Increment the uniqueOutputIndex so we always get a unique value for
|
|
|
|
// the funding outpoint.
|
|
|
|
uniqueOutputIndex.Add(1)
|
|
|
|
op := wire.OutPoint{Hash: key, Index: uniqueOutputIndex.Load()}
|
|
|
|
|
2016-09-03 03:51:34 +02:00
|
|
|
return &OpenChannel{
|
2020-03-14 00:49:12 +01:00
|
|
|
ChanType: SingleFunderBit | FrozenBit,
|
2017-11-10 05:56:04 +01:00
|
|
|
ChainHash: key,
|
2023-02-10 09:32:09 +01:00
|
|
|
FundingOutpoint: op,
|
2018-05-02 01:27:20 +02:00
|
|
|
ShortChannelID: chanID,
|
2017-11-10 05:56:04 +01:00
|
|
|
IsInitiator: true,
|
|
|
|
IsPending: true,
|
|
|
|
IdentityPub: pubKey,
|
|
|
|
Capacity: btcutil.Amount(10000),
|
|
|
|
LocalChanCfg: localCfg,
|
|
|
|
RemoteChanCfg: remoteCfg,
|
|
|
|
TotalMSatSent: 8,
|
|
|
|
TotalMSatReceived: 2,
|
|
|
|
LocalCommitment: ChannelCommitment{
|
|
|
|
CommitHeight: 0,
|
|
|
|
LocalBalance: lnwire.MilliSatoshi(9000),
|
|
|
|
RemoteBalance: lnwire.MilliSatoshi(3000),
|
|
|
|
CommitFee: btcutil.Amount(rand.Int63()),
|
|
|
|
FeePerKw: btcutil.Amount(5000),
|
2021-01-07 22:26:24 +01:00
|
|
|
CommitTx: channels.TestFundingTx,
|
2017-11-10 05:56:04 +01:00
|
|
|
CommitSig: bytes.Repeat([]byte{1}, 71),
|
|
|
|
},
|
|
|
|
RemoteCommitment: ChannelCommitment{
|
|
|
|
CommitHeight: 0,
|
|
|
|
LocalBalance: lnwire.MilliSatoshi(3000),
|
|
|
|
RemoteBalance: lnwire.MilliSatoshi(9000),
|
|
|
|
CommitFee: btcutil.Amount(rand.Int63()),
|
|
|
|
FeePerKw: btcutil.Amount(5000),
|
2021-01-07 22:26:24 +01:00
|
|
|
CommitTx: channels.TestFundingTx,
|
2017-11-10 05:56:04 +01:00
|
|
|
CommitSig: bytes.Repeat([]byte{1}, 71),
|
|
|
|
},
|
2017-07-29 20:24:28 +02:00
|
|
|
NumConfsRequired: 4,
|
|
|
|
RemoteCurrentRevocation: privKey.PubKey(),
|
|
|
|
RemoteNextRevocation: privKey.PubKey(),
|
|
|
|
RevocationProducer: producer,
|
|
|
|
RevocationStore: store,
|
2017-08-22 07:51:45 +02:00
|
|
|
Db: cdb,
|
2018-02-23 00:02:02 +01:00
|
|
|
Packager: NewChannelPackager(chanID),
|
2021-01-07 22:26:24 +01:00
|
|
|
FundingTxn: channels.TestFundingTx,
|
2020-03-14 00:49:12 +01:00
|
|
|
ThawHeight: uint32(defaultPendingHeight),
|
2022-04-07 20:04:50 +02:00
|
|
|
InitialLocalBalance: lnwire.MilliSatoshi(9000),
|
|
|
|
InitialRemoteBalance: lnwire.MilliSatoshi(3000),
|
2020-02-05 14:39:31 +01:00
|
|
|
}
|
2016-09-03 03:51:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestOpenChannelPutGetDelete(t *testing.T) {
|
2017-06-17 00:59:20 +02:00
|
|
|
t.Parallel()
|
|
|
|
|
2022-08-27 09:04:55 +02:00
|
|
|
fullDB, err := MakeTestDB(t)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to make test database")
|
2015-12-26 00:00:53 +01:00
|
|
|
|
2021-09-21 19:18:17 +02:00
|
|
|
cdb := fullDB.ChannelStateDB()
|
|
|
|
|
2020-02-05 14:39:31 +01:00
|
|
|
// Create the test channel state, with additional htlcs on the local
|
|
|
|
// and remote commitment.
|
|
|
|
localHtlcs := []HTLC{
|
2023-05-19 16:58:05 +02:00
|
|
|
{
|
|
|
|
Signature: testSig.Serialize(),
|
2017-07-30 02:40:14 +02:00
|
|
|
Incoming: true,
|
|
|
|
Amt: 10,
|
|
|
|
RHash: key,
|
|
|
|
RefundTimeout: 1,
|
2023-05-19 16:58:05 +02:00
|
|
|
OnionBlob: lnmock.MockOnion(),
|
2017-11-10 05:56:04 +01:00
|
|
|
},
|
|
|
|
}
|
2020-02-05 14:39:31 +01:00
|
|
|
|
|
|
|
remoteHtlcs := []HTLC{
|
2017-11-10 05:56:04 +01:00
|
|
|
{
|
|
|
|
Signature: testSig.Serialize(),
|
|
|
|
Incoming: false,
|
|
|
|
Amt: 10,
|
|
|
|
RHash: key,
|
|
|
|
RefundTimeout: 1,
|
2023-05-19 16:58:05 +02:00
|
|
|
OnionBlob: lnmock.MockOnion(),
|
2016-09-07 04:17:34 +02:00
|
|
|
},
|
|
|
|
}
|
2019-09-06 13:14:38 +02:00
|
|
|
|
2020-02-05 14:39:31 +01:00
|
|
|
state := createTestChannel(
|
|
|
|
t, cdb,
|
|
|
|
remoteHtlcsOption(remoteHtlcs),
|
|
|
|
localHtlcsOption(localHtlcs),
|
|
|
|
)
|
2015-12-26 00:00:53 +01:00
|
|
|
|
2016-10-26 01:11:23 +02:00
|
|
|
openChannels, err := cdb.FetchOpenChannels(state.IdentityPub)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to fetch open channel")
|
2015-12-26 00:00:53 +01:00
|
|
|
|
2016-06-21 06:39:50 +02:00
|
|
|
newState := openChannels[0]
|
|
|
|
|
2015-12-26 00:00:53 +01:00
|
|
|
// The decoded channel state should be identical to what we stored
|
|
|
|
// above.
|
2017-07-29 20:24:28 +02:00
|
|
|
if !reflect.DeepEqual(state, newState) {
|
|
|
|
t.Fatalf("channel state doesn't match:: %v vs %v",
|
|
|
|
spew.Sdump(state), spew.Sdump(newState))
|
2015-12-26 00:00:53 +01:00
|
|
|
}
|
2016-06-23 01:15:07 +02:00
|
|
|
|
2017-07-29 20:24:28 +02:00
|
|
|
// We'll also test that the channel is properly able to hot swap the
|
|
|
|
// next revocation for the state machine. This tests the initial
|
|
|
|
// post-funding revocation exchange.
|
2022-02-23 14:48:00 +01:00
|
|
|
nextRevKey, err := btcec.NewPrivateKey()
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to create new private key")
|
2017-07-29 20:24:28 +02:00
|
|
|
if err := state.InsertNextRevocation(nextRevKey.PubKey()); err != nil {
|
|
|
|
t.Fatalf("unable to update revocation: %v", err)
|
2016-12-14 15:01:48 +01:00
|
|
|
}
|
|
|
|
|
2017-07-29 20:24:28 +02:00
|
|
|
openChannels, err = cdb.FetchOpenChannels(state.IdentityPub)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to fetch open channel")
|
2017-07-29 20:24:28 +02:00
|
|
|
updatedChan := openChannels[0]
|
|
|
|
|
|
|
|
// Ensure that the revocation was set properly.
|
|
|
|
if !nextRevKey.PubKey().IsEqual(updatedChan.RemoteNextRevocation) {
|
|
|
|
t.Fatalf("next revocation wasn't updated")
|
2016-11-16 03:51:48 +01:00
|
|
|
}
|
2016-06-30 20:39:57 +02:00
|
|
|
|
2016-06-23 01:15:07 +02:00
|
|
|
// Finally to wrap up the test, delete the state of the channel within
|
|
|
|
// the database. This involves "closing" the channel which removes all
|
|
|
|
// written state, and creates a small "summary" elsewhere within the
|
|
|
|
// database.
|
2017-05-05 00:21:56 +02:00
|
|
|
closeSummary := &ChannelCloseSummary{
|
2017-07-29 20:24:28 +02:00
|
|
|
ChanPoint: state.FundingOutpoint,
|
2017-05-15 04:02:59 +02:00
|
|
|
RemotePub: state.IdentityPub,
|
|
|
|
SettledBalance: btcutil.Amount(500),
|
|
|
|
TimeLockedBalance: btcutil.Amount(10000),
|
|
|
|
IsPending: false,
|
|
|
|
CloseType: CooperativeClose,
|
2017-05-05 00:21:56 +02:00
|
|
|
}
|
|
|
|
if err := state.CloseChannel(closeSummary); err != nil {
|
2016-06-23 01:15:07 +02:00
|
|
|
t.Fatalf("unable to close channel: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// As the channel is now closed, attempting to fetch all open channels
|
|
|
|
// for our fake node ID should return an empty slice.
|
2016-10-26 01:11:23 +02:00
|
|
|
openChans, err := cdb.FetchOpenChannels(state.IdentityPub)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to fetch open channels")
|
2017-02-08 01:27:53 +01:00
|
|
|
if len(openChans) != 0 {
|
|
|
|
t.Fatalf("all channels not deleted, found %v", len(openChans))
|
|
|
|
}
|
2016-06-23 01:15:07 +02:00
|
|
|
|
2017-02-08 01:27:53 +01:00
|
|
|
// Additionally, attempting to fetch all the open channels globally
|
|
|
|
// should yield no results.
|
|
|
|
openChans, err = cdb.FetchAllChannels()
|
|
|
|
if err != nil {
|
2016-12-14 15:01:48 +01:00
|
|
|
t.Fatal("unable to fetch all open chans")
|
2017-02-08 01:27:53 +01:00
|
|
|
}
|
2016-06-23 01:15:07 +02:00
|
|
|
if len(openChans) != 0 {
|
|
|
|
t.Fatalf("all channels not deleted, found %v", len(openChans))
|
|
|
|
}
|
2015-12-26 00:00:53 +01:00
|
|
|
}
|
|
|
|
|
2019-12-03 10:38:29 +01:00
|
|
|
// TestOptionalShutdown tests the reading and writing of channels with and
|
|
|
|
// without optional shutdown script fields.
|
|
|
|
func TestOptionalShutdown(t *testing.T) {
|
|
|
|
local := lnwire.DeliveryAddress([]byte("local shutdown script"))
|
|
|
|
remote := lnwire.DeliveryAddress([]byte("remote shutdown script"))
|
|
|
|
|
|
|
|
if _, err := rand.Read(remote); err != nil {
|
|
|
|
t.Fatalf("Could not create random script: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
2020-02-05 14:39:31 +01:00
|
|
|
localShutdown lnwire.DeliveryAddress
|
|
|
|
remoteShutdown lnwire.DeliveryAddress
|
2019-12-03 10:38:29 +01:00
|
|
|
}{
|
|
|
|
{
|
2020-02-05 14:39:31 +01:00
|
|
|
name: "no shutdown scripts",
|
|
|
|
localShutdown: nil,
|
|
|
|
remoteShutdown: nil,
|
2019-12-03 10:38:29 +01:00
|
|
|
},
|
|
|
|
{
|
2020-02-05 14:39:31 +01:00
|
|
|
name: "local shutdown script",
|
|
|
|
localShutdown: local,
|
|
|
|
remoteShutdown: nil,
|
2019-12-03 10:38:29 +01:00
|
|
|
},
|
|
|
|
{
|
2020-02-05 14:39:31 +01:00
|
|
|
name: "remote shutdown script",
|
|
|
|
localShutdown: nil,
|
|
|
|
remoteShutdown: remote,
|
2019-12-03 10:38:29 +01:00
|
|
|
},
|
|
|
|
{
|
2020-02-05 14:39:31 +01:00
|
|
|
name: "both scripts set",
|
|
|
|
localShutdown: local,
|
|
|
|
remoteShutdown: remote,
|
2019-12-03 10:38:29 +01:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
test := test
|
|
|
|
|
|
|
|
t.Run(test.name, func(t *testing.T) {
|
2022-08-27 09:04:55 +02:00
|
|
|
fullDB, err := MakeTestDB(t)
|
2019-12-03 10:38:29 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unable to make test database: %v", err)
|
|
|
|
}
|
|
|
|
|
2021-09-21 19:18:17 +02:00
|
|
|
cdb := fullDB.ChannelStateDB()
|
|
|
|
|
2020-02-05 14:39:31 +01:00
|
|
|
// Create a channel with upfront scripts set as
|
|
|
|
// specified in the test.
|
|
|
|
state := createTestChannel(
|
|
|
|
t, cdb,
|
|
|
|
localShutdownOption(test.localShutdown),
|
|
|
|
remoteShutdownOption(test.remoteShutdown),
|
|
|
|
)
|
|
|
|
|
|
|
|
openChannels, err := cdb.FetchOpenChannels(
|
|
|
|
state.IdentityPub,
|
|
|
|
)
|
2019-12-03 10:38:29 +01:00
|
|
|
if err != nil {
|
2020-02-05 14:39:31 +01:00
|
|
|
t.Fatalf("unable to fetch open"+
|
|
|
|
" channel: %v", err)
|
2019-12-03 10:38:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(openChannels) != 1 {
|
2020-02-05 14:39:31 +01:00
|
|
|
t.Fatalf("Expected one channel open,"+
|
|
|
|
" got: %v", len(openChannels))
|
2019-12-03 10:38:29 +01:00
|
|
|
}
|
|
|
|
|
2020-02-05 14:39:31 +01:00
|
|
|
if !bytes.Equal(openChannels[0].LocalShutdownScript,
|
|
|
|
test.localShutdown) {
|
|
|
|
|
|
|
|
t.Fatalf("Expected local: %x, got: %x",
|
|
|
|
test.localShutdown,
|
2019-12-03 10:38:29 +01:00
|
|
|
openChannels[0].LocalShutdownScript)
|
|
|
|
}
|
|
|
|
|
2020-02-05 14:39:31 +01:00
|
|
|
if !bytes.Equal(openChannels[0].RemoteShutdownScript,
|
|
|
|
test.remoteShutdown) {
|
|
|
|
|
|
|
|
t.Fatalf("Expected remote: %x, got: %x",
|
|
|
|
test.remoteShutdown,
|
2019-12-03 10:38:29 +01:00
|
|
|
openChannels[0].RemoteShutdownScript)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-10 05:56:04 +01:00
|
|
|
func assertCommitmentEqual(t *testing.T, a, b *ChannelCommitment) {
|
|
|
|
if !reflect.DeepEqual(a, b) {
|
|
|
|
_, _, line, _ := runtime.Caller(1)
|
|
|
|
t.Fatalf("line %v: commitments don't match: %v vs %v",
|
|
|
|
line, spew.Sdump(a), spew.Sdump(b))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-08 01:36:26 +02:00
|
|
|
// assertRevocationLogEntryEqual asserts that, for all the fields of a given
|
|
|
|
// revocation log entry, their values match those on a given ChannelCommitment.
|
|
|
|
func assertRevocationLogEntryEqual(t *testing.T, c *ChannelCommitment,
|
|
|
|
r *RevocationLog) {
|
|
|
|
|
|
|
|
// Check the common fields.
|
|
|
|
require.EqualValues(
|
|
|
|
t, r.CommitTxHash, c.CommitTx.TxHash(), "CommitTx mismatch",
|
|
|
|
)
|
|
|
|
|
|
|
|
// Now check the common fields from the HTLCs.
|
|
|
|
require.Equal(t, len(r.HTLCEntries), len(c.Htlcs), "HTLCs len mismatch")
|
|
|
|
for i, rHtlc := range r.HTLCEntries {
|
|
|
|
cHtlc := c.Htlcs[i]
|
|
|
|
require.Equal(t, rHtlc.RHash, cHtlc.RHash, "RHash mismatch")
|
|
|
|
require.Equal(t, rHtlc.Amt, cHtlc.Amt.ToSatoshis(),
|
|
|
|
"Amt mismatch")
|
|
|
|
require.Equal(t, rHtlc.RefundTimeout, cHtlc.RefundTimeout,
|
|
|
|
"RefundTimeout mismatch")
|
|
|
|
require.EqualValues(t, rHtlc.OutputIndex, cHtlc.OutputIndex,
|
|
|
|
"OutputIndex mismatch")
|
|
|
|
require.Equal(t, rHtlc.Incoming, cHtlc.Incoming,
|
|
|
|
"Incoming mismatch")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-07 04:17:34 +02:00
|
|
|
func TestChannelStateTransition(t *testing.T) {
|
2017-06-17 00:59:20 +02:00
|
|
|
t.Parallel()
|
|
|
|
|
2022-08-27 09:04:55 +02:00
|
|
|
fullDB, err := MakeTestDB(t)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to make test database")
|
2016-09-03 03:51:34 +02:00
|
|
|
|
2021-09-21 19:18:17 +02:00
|
|
|
cdb := fullDB.ChannelStateDB()
|
|
|
|
|
2016-09-03 03:51:34 +02:00
|
|
|
// First create a minimal channel, then perform a full sync in order to
|
|
|
|
// persist the data.
|
2020-02-05 14:39:31 +01:00
|
|
|
channel := createTestChannel(t, cdb)
|
2016-09-03 03:51:34 +02:00
|
|
|
|
2017-01-13 06:01:50 +01:00
|
|
|
// Add some HTLCs which were added during this new state transition.
|
|
|
|
// Half of the HTLCs are incoming, while the other half are outgoing.
|
2017-02-21 06:58:34 +01:00
|
|
|
var (
|
2017-11-10 06:00:35 +01:00
|
|
|
htlcs []HTLC
|
2017-08-22 07:51:45 +02:00
|
|
|
htlcAmt lnwire.MilliSatoshi
|
2017-02-21 06:58:34 +01:00
|
|
|
)
|
2016-09-03 03:51:34 +02:00
|
|
|
for i := uint32(0); i < 10; i++ {
|
|
|
|
var incoming bool
|
|
|
|
if i > 5 {
|
|
|
|
incoming = true
|
|
|
|
}
|
2017-11-10 06:00:35 +01:00
|
|
|
htlc := HTLC{
|
2017-07-30 02:40:14 +02:00
|
|
|
Signature: testSig.Serialize(),
|
|
|
|
Incoming: incoming,
|
|
|
|
Amt: 10,
|
|
|
|
RHash: key,
|
|
|
|
RefundTimeout: i,
|
|
|
|
OutputIndex: int32(i * 3),
|
2017-11-10 06:00:35 +01:00
|
|
|
LogIndex: uint64(i * 2),
|
|
|
|
HtlcIndex: uint64(i),
|
2016-09-03 03:51:34 +02:00
|
|
|
}
|
2023-05-19 16:58:05 +02:00
|
|
|
copy(
|
|
|
|
htlc.OnionBlob[:],
|
|
|
|
bytes.Repeat([]byte{2}, lnwire.OnionPacketSize),
|
|
|
|
)
|
2016-09-03 03:51:34 +02:00
|
|
|
htlcs = append(htlcs, htlc)
|
2017-02-21 06:58:34 +01:00
|
|
|
htlcAmt += htlc.Amt
|
2016-09-03 03:51:34 +02:00
|
|
|
}
|
|
|
|
|
2017-01-13 06:01:50 +01:00
|
|
|
// Create a new channel delta which includes the above HTLCs, some
|
2016-09-03 03:51:34 +02:00
|
|
|
// balance updates, and an increment of the current commitment height.
|
|
|
|
// Additionally, modify the signature and commitment transaction.
|
|
|
|
newSequence := uint32(129498)
|
|
|
|
newSig := bytes.Repeat([]byte{3}, 71)
|
2017-11-10 06:00:35 +01:00
|
|
|
newTx := channel.LocalCommitment.CommitTx.Copy()
|
2016-09-06 01:52:54 +02:00
|
|
|
newTx.TxIn[0].Sequence = newSequence
|
2017-11-10 06:00:35 +01:00
|
|
|
commitment := ChannelCommitment{
|
|
|
|
CommitHeight: 1,
|
|
|
|
LocalLogIndex: 2,
|
|
|
|
LocalHtlcIndex: 1,
|
|
|
|
RemoteLogIndex: 2,
|
|
|
|
RemoteHtlcIndex: 1,
|
|
|
|
LocalBalance: lnwire.MilliSatoshi(1e8),
|
|
|
|
RemoteBalance: lnwire.MilliSatoshi(1e8),
|
|
|
|
CommitFee: 55,
|
|
|
|
FeePerKw: 99,
|
|
|
|
CommitTx: newTx,
|
|
|
|
CommitSig: newSig,
|
|
|
|
Htlcs: htlcs,
|
|
|
|
}
|
|
|
|
|
|
|
|
// First update the local node's broadcastable state and also add a
|
|
|
|
// CommitDiff remote node's as well in order to simulate a proper state
|
|
|
|
// transition.
|
2020-01-03 15:53:51 +01:00
|
|
|
unsignedAckedUpdates := []LogUpdate{
|
|
|
|
{
|
|
|
|
LogIndex: 2,
|
|
|
|
UpdateMsg: &lnwire.UpdateAddHTLC{
|
2024-03-27 14:38:50 +01:00
|
|
|
ChanID: lnwire.ChannelID{1, 2, 3},
|
2020-01-03 15:53:51 +01:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2022-05-10 12:33:44 +02:00
|
|
|
_, err = channel.UpdateCommitment(&commitment, unsignedAckedUpdates)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to update commitment")
|
2016-09-03 03:51:34 +02:00
|
|
|
|
2020-01-03 15:53:51 +01:00
|
|
|
// Assert that update is correctly written to the database.
|
|
|
|
dbUnsignedAckedUpdates, err := channel.UnsignedAckedUpdates()
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to fetch dangling remote updates")
|
2020-01-03 15:53:51 +01:00
|
|
|
if len(dbUnsignedAckedUpdates) != 1 {
|
|
|
|
t.Fatalf("unexpected number of dangling remote updates")
|
|
|
|
}
|
|
|
|
if !reflect.DeepEqual(
|
|
|
|
dbUnsignedAckedUpdates[0], unsignedAckedUpdates[0],
|
|
|
|
) {
|
2022-02-07 13:58:28 +01:00
|
|
|
|
2020-01-28 02:25:36 +01:00
|
|
|
t.Fatalf("unexpected update: expected %v, got %v",
|
|
|
|
spew.Sdump(unsignedAckedUpdates[0]),
|
|
|
|
spew.Sdump(dbUnsignedAckedUpdates))
|
2020-01-03 15:53:51 +01:00
|
|
|
}
|
|
|
|
|
2017-01-13 06:01:50 +01:00
|
|
|
// The balances, new update, the HTLCs and the changes to the fake
|
2016-09-07 04:17:34 +02:00
|
|
|
// commitment transaction along with the modified signature should all
|
|
|
|
// have been updated.
|
2016-10-26 01:11:23 +02:00
|
|
|
updatedChannel, err := cdb.FetchOpenChannels(channel.IdentityPub)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to fetch updated channel")
|
2017-11-10 06:00:35 +01:00
|
|
|
assertCommitmentEqual(t, &commitment, &updatedChannel[0].LocalCommitment)
|
2016-11-28 04:10:05 +01:00
|
|
|
numDiskUpdates, err := updatedChannel[0].CommitmentHeight()
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to read commitment height from disk")
|
2017-11-10 06:00:35 +01:00
|
|
|
if numDiskUpdates != uint64(commitment.CommitHeight) {
|
2016-11-28 04:10:05 +01:00
|
|
|
t.Fatalf("num disk updates doesn't match: %v vs %v",
|
2017-11-10 06:00:35 +01:00
|
|
|
numDiskUpdates, commitment.CommitHeight)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attempting to query for a commitment diff should return
|
|
|
|
// ErrNoPendingCommit as we haven't yet created a new state for them.
|
|
|
|
_, err = channel.RemoteCommitChainTip()
|
|
|
|
if err != ErrNoPendingCommit {
|
|
|
|
t.Fatalf("expected ErrNoPendingCommit, instead got %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// To simulate us extending a new state to the remote party, we'll also
|
|
|
|
// create a new commit diff for them.
|
|
|
|
remoteCommit := commitment
|
|
|
|
remoteCommit.LocalBalance = lnwire.MilliSatoshi(2e8)
|
|
|
|
remoteCommit.RemoteBalance = lnwire.MilliSatoshi(3e8)
|
|
|
|
remoteCommit.CommitHeight = 1
|
|
|
|
commitDiff := &CommitDiff{
|
|
|
|
Commitment: remoteCommit,
|
|
|
|
CommitSig: &lnwire.CommitSig{
|
|
|
|
ChanID: lnwire.ChannelID(key),
|
2018-01-31 05:19:40 +01:00
|
|
|
CommitSig: wireSig,
|
|
|
|
HtlcSigs: []lnwire.Sig{
|
|
|
|
wireSig,
|
|
|
|
wireSig,
|
2017-11-10 06:00:35 +01:00
|
|
|
},
|
|
|
|
},
|
|
|
|
LogUpdates: []LogUpdate{
|
|
|
|
{
|
|
|
|
LogIndex: 1,
|
|
|
|
UpdateMsg: &lnwire.UpdateAddHTLC{
|
2024-03-27 14:38:50 +01:00
|
|
|
ID: 1,
|
|
|
|
Amount: lnwire.NewMSatFromSatoshis(100),
|
|
|
|
Expiry: 25,
|
2017-11-10 06:00:35 +01:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
LogIndex: 2,
|
|
|
|
UpdateMsg: &lnwire.UpdateAddHTLC{
|
2024-03-27 14:38:50 +01:00
|
|
|
ID: 2,
|
|
|
|
Amount: lnwire.NewMSatFromSatoshis(200),
|
|
|
|
Expiry: 50,
|
2017-11-10 06:00:35 +01:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2022-11-18 12:15:22 +01:00
|
|
|
OpenedCircuitKeys: []models.CircuitKey{},
|
|
|
|
ClosedCircuitKeys: []models.CircuitKey{},
|
2017-11-10 06:00:35 +01:00
|
|
|
}
|
|
|
|
copy(commitDiff.LogUpdates[0].UpdateMsg.(*lnwire.UpdateAddHTLC).PaymentHash[:],
|
|
|
|
bytes.Repeat([]byte{1}, 32))
|
|
|
|
copy(commitDiff.LogUpdates[1].UpdateMsg.(*lnwire.UpdateAddHTLC).PaymentHash[:],
|
|
|
|
bytes.Repeat([]byte{2}, 32))
|
|
|
|
if err := channel.AppendRemoteCommitChain(commitDiff); err != nil {
|
|
|
|
t.Fatalf("unable to add to commit chain: %v", err)
|
2016-09-07 04:17:34 +02:00
|
|
|
}
|
|
|
|
|
2018-04-18 04:03:27 +02:00
|
|
|
// The commitment tip should now match the commitment that we just
|
2017-11-10 06:00:35 +01:00
|
|
|
// inserted.
|
|
|
|
diskCommitDiff, err := channel.RemoteCommitChainTip()
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to fetch commit diff")
|
2017-11-10 06:00:35 +01:00
|
|
|
if !reflect.DeepEqual(commitDiff, diskCommitDiff) {
|
|
|
|
t.Fatalf("commit diffs don't match: %v vs %v", spew.Sdump(remoteCommit),
|
|
|
|
spew.Sdump(diskCommitDiff))
|
|
|
|
}
|
|
|
|
|
|
|
|
// We'll save the old remote commitment as this will be added to the
|
|
|
|
// revocation log shortly.
|
|
|
|
oldRemoteCommit := channel.RemoteCommitment
|
|
|
|
|
2016-09-07 04:17:34 +02:00
|
|
|
// Next, write to the log which tracks the necessary revocation state
|
|
|
|
// needed to rectify any fishy behavior by the remote party. Modify the
|
|
|
|
// current uncollapsed revocation state to simulate a state transition
|
|
|
|
// by the remote party.
|
2017-07-29 20:24:28 +02:00
|
|
|
channel.RemoteCurrentRevocation = channel.RemoteNextRevocation
|
2022-02-23 14:48:00 +01:00
|
|
|
newPriv, err := btcec.NewPrivateKey()
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to generate key")
|
2017-07-29 20:24:28 +02:00
|
|
|
channel.RemoteNextRevocation = newPriv.PubKey()
|
2018-02-23 00:02:02 +01:00
|
|
|
|
2018-05-02 01:27:20 +02:00
|
|
|
fwdPkg := NewFwdPkg(channel.ShortChanID(), oldRemoteCommit.CommitHeight,
|
2018-02-23 00:02:02 +01:00
|
|
|
diskCommitDiff.LogUpdates, nil)
|
|
|
|
|
2022-04-08 01:36:26 +02:00
|
|
|
err = channel.AdvanceCommitChainTail(
|
|
|
|
fwdPkg, nil, dummyLocalOutputIndex, dummyRemoteOutIndex,
|
|
|
|
)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to append to revocation log")
|
2016-09-03 03:51:34 +02:00
|
|
|
|
2018-02-07 04:11:11 +01:00
|
|
|
// At this point, the remote commit chain should be nil, and the posted
|
2017-11-10 06:00:35 +01:00
|
|
|
// remote commitment should match the one we added as a diff above.
|
|
|
|
if _, err := channel.RemoteCommitChainTip(); err != ErrNoPendingCommit {
|
|
|
|
t.Fatalf("expected ErrNoPendingCommit, instead got %v", err)
|
|
|
|
}
|
|
|
|
|
2018-02-07 04:13:07 +01:00
|
|
|
// We should be able to fetch the channel delta created above by its
|
2016-09-03 03:51:34 +02:00
|
|
|
// update number with all the state properly reconstructed.
|
2022-04-08 01:36:26 +02:00
|
|
|
diskPrevCommit, _, err := channel.FindPreviousState(
|
2017-11-10 06:00:35 +01:00
|
|
|
oldRemoteCommit.CommitHeight,
|
|
|
|
)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to fetch past delta")
|
2016-09-03 03:51:34 +02:00
|
|
|
|
2022-04-08 01:36:26 +02:00
|
|
|
// Check the output indexes are saved as expected.
|
|
|
|
require.EqualValues(
|
|
|
|
t, dummyLocalOutputIndex, diskPrevCommit.OurOutputIndex,
|
|
|
|
)
|
|
|
|
require.EqualValues(
|
|
|
|
t, dummyRemoteOutIndex, diskPrevCommit.TheirOutputIndex,
|
|
|
|
)
|
|
|
|
|
2016-09-03 03:51:34 +02:00
|
|
|
// The two deltas (the original vs the on-disk version) should
|
|
|
|
// identical, and all HTLC data should properly be retained.
|
2022-04-08 01:36:26 +02:00
|
|
|
assertRevocationLogEntryEqual(t, &oldRemoteCommit, diskPrevCommit)
|
2017-02-08 01:27:53 +01:00
|
|
|
|
2017-03-25 00:07:34 +01:00
|
|
|
// The state number recovered from the tail of the revocation log
|
|
|
|
// should be identical to this current state.
|
2022-03-16 11:15:06 +01:00
|
|
|
logTailHeight, err := channel.revocationLogTailCommitHeight()
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to retrieve log")
|
2022-03-16 11:15:06 +01:00
|
|
|
if logTailHeight != oldRemoteCommit.CommitHeight {
|
2017-03-25 00:07:34 +01:00
|
|
|
t.Fatal("update number doesn't match")
|
|
|
|
}
|
|
|
|
|
2017-11-10 06:00:35 +01:00
|
|
|
oldRemoteCommit = channel.RemoteCommitment
|
|
|
|
|
|
|
|
// Next modify the posted diff commitment slightly, then create a new
|
|
|
|
// commitment diff and advance the tail.
|
|
|
|
commitDiff.Commitment.CommitHeight = 2
|
|
|
|
commitDiff.Commitment.LocalBalance -= htlcAmt
|
|
|
|
commitDiff.Commitment.RemoteBalance += htlcAmt
|
|
|
|
commitDiff.LogUpdates = []LogUpdate{}
|
|
|
|
if err := channel.AppendRemoteCommitChain(commitDiff); err != nil {
|
2017-11-11 04:36:35 +01:00
|
|
|
t.Fatalf("unable to add to commit chain: %v", err)
|
2017-11-10 06:00:35 +01:00
|
|
|
}
|
2018-02-23 00:02:02 +01:00
|
|
|
|
2018-05-02 01:27:20 +02:00
|
|
|
fwdPkg = NewFwdPkg(channel.ShortChanID(), oldRemoteCommit.CommitHeight, nil, nil)
|
2018-02-23 00:02:02 +01:00
|
|
|
|
2022-04-08 01:36:26 +02:00
|
|
|
err = channel.AdvanceCommitChainTail(
|
|
|
|
fwdPkg, nil, dummyLocalOutputIndex, dummyRemoteOutIndex,
|
|
|
|
)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to append to revocation log")
|
2017-02-21 06:58:34 +01:00
|
|
|
|
|
|
|
// Once again, fetch the state and ensure it has been properly updated.
|
2022-04-08 01:36:26 +02:00
|
|
|
prevCommit, _, err := channel.FindPreviousState(
|
|
|
|
oldRemoteCommit.CommitHeight,
|
|
|
|
)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to fetch past delta")
|
2022-04-08 01:36:26 +02:00
|
|
|
|
|
|
|
// Check the output indexes are saved as expected.
|
|
|
|
require.EqualValues(
|
|
|
|
t, dummyLocalOutputIndex, diskPrevCommit.OurOutputIndex,
|
|
|
|
)
|
|
|
|
require.EqualValues(
|
|
|
|
t, dummyRemoteOutIndex, diskPrevCommit.TheirOutputIndex,
|
|
|
|
)
|
|
|
|
|
|
|
|
assertRevocationLogEntryEqual(t, &oldRemoteCommit, prevCommit)
|
2017-02-21 06:58:34 +01:00
|
|
|
|
2017-03-25 00:07:34 +01:00
|
|
|
// Once again, state number recovered from the tail of the revocation
|
|
|
|
// log should be identical to this current state.
|
2022-03-16 11:15:06 +01:00
|
|
|
logTailHeight, err = channel.revocationLogTailCommitHeight()
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to retrieve log")
|
2022-03-16 11:15:06 +01:00
|
|
|
if logTailHeight != oldRemoteCommit.CommitHeight {
|
2017-03-25 00:07:34 +01:00
|
|
|
t.Fatal("update number doesn't match")
|
|
|
|
}
|
|
|
|
|
2016-09-07 04:17:34 +02:00
|
|
|
// The revocation state stored on-disk should now also be identical.
|
2016-10-26 01:11:23 +02:00
|
|
|
updatedChannel, err = cdb.FetchOpenChannels(channel.IdentityPub)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to fetch updated channel")
|
2017-07-29 20:24:28 +02:00
|
|
|
if !channel.RemoteCurrentRevocation.IsEqual(updatedChannel[0].RemoteCurrentRevocation) {
|
|
|
|
t.Fatalf("revocation state was not synced")
|
|
|
|
}
|
|
|
|
if !channel.RemoteNextRevocation.IsEqual(updatedChannel[0].RemoteNextRevocation) {
|
|
|
|
t.Fatalf("revocation state was not synced")
|
2016-09-07 04:17:34 +02:00
|
|
|
}
|
2017-02-08 01:27:53 +01:00
|
|
|
|
2021-07-22 11:56:22 +02:00
|
|
|
// At this point, we should have 2 forwarding packages added.
|
2021-09-21 19:18:17 +02:00
|
|
|
fwdPkgs := loadFwdPkgs(t, cdb.backend, channel.Packager)
|
2021-07-22 11:56:22 +02:00
|
|
|
require.Len(t, fwdPkgs, 2, "wrong number of forwarding packages")
|
|
|
|
|
2017-02-08 01:27:53 +01:00
|
|
|
// Now attempt to delete the channel from the database.
|
2017-05-05 00:21:56 +02:00
|
|
|
closeSummary := &ChannelCloseSummary{
|
2017-07-29 20:24:28 +02:00
|
|
|
ChanPoint: channel.FundingOutpoint,
|
2017-05-15 04:02:59 +02:00
|
|
|
RemotePub: channel.IdentityPub,
|
|
|
|
SettledBalance: btcutil.Amount(500),
|
|
|
|
TimeLockedBalance: btcutil.Amount(10000),
|
|
|
|
IsPending: false,
|
2018-04-04 08:57:55 +02:00
|
|
|
CloseType: RemoteForceClose,
|
2017-05-05 00:21:56 +02:00
|
|
|
}
|
|
|
|
if err := updatedChannel[0].CloseChannel(closeSummary); err != nil {
|
2017-02-08 01:27:53 +01:00
|
|
|
t.Fatalf("unable to delete updated channel: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we attempt to fetch the target channel again, it shouldn't be
|
|
|
|
// found.
|
|
|
|
channels, err := cdb.FetchOpenChannels(channel.IdentityPub)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to fetch updated channels")
|
2017-02-08 01:27:53 +01:00
|
|
|
if len(channels) != 0 {
|
|
|
|
t.Fatalf("%v channels, found, but none should be",
|
|
|
|
len(channels))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attempting to find previous states on the channel should fail as the
|
|
|
|
// revocation log has been deleted.
|
2022-04-08 01:36:26 +02:00
|
|
|
_, _, err = updatedChannel[0].FindPreviousState(
|
|
|
|
oldRemoteCommit.CommitHeight,
|
|
|
|
)
|
2017-02-08 01:27:53 +01:00
|
|
|
if err == nil {
|
2018-02-07 04:11:11 +01:00
|
|
|
t.Fatal("revocation log search should have failed")
|
2017-02-08 01:27:53 +01:00
|
|
|
}
|
2021-07-22 11:56:22 +02:00
|
|
|
|
|
|
|
// All forwarding packages of this channel has been deleted too.
|
2021-09-21 19:18:17 +02:00
|
|
|
fwdPkgs = loadFwdPkgs(t, cdb.backend, channel.Packager)
|
2021-07-22 11:56:22 +02:00
|
|
|
require.Empty(t, fwdPkgs, "no forwarding packages should exist")
|
2015-12-26 00:00:53 +01:00
|
|
|
}
|
2017-01-23 08:31:01 +01:00
|
|
|
|
|
|
|
func TestFetchPendingChannels(t *testing.T) {
|
2017-06-17 00:59:20 +02:00
|
|
|
t.Parallel()
|
|
|
|
|
2022-08-27 09:04:55 +02:00
|
|
|
fullDB, err := MakeTestDB(t)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to make test database")
|
2017-01-23 08:31:01 +01:00
|
|
|
|
2021-09-21 19:18:17 +02:00
|
|
|
cdb := fullDB.ChannelStateDB()
|
|
|
|
|
2020-02-05 14:39:31 +01:00
|
|
|
// Create a pending channel that was broadcast at height 99.
|
2017-06-06 00:02:27 +02:00
|
|
|
const broadcastHeight = 99
|
2020-02-05 14:39:31 +01:00
|
|
|
createTestChannel(t, cdb, pendingHeightOption(broadcastHeight))
|
2017-01-23 08:31:01 +01:00
|
|
|
|
|
|
|
pendingChannels, err := cdb.FetchPendingChannels()
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to list pending channels")
|
2017-01-23 08:31:01 +01:00
|
|
|
|
|
|
|
if len(pendingChannels) != 1 {
|
|
|
|
t.Fatalf("incorrect number of pending channels: expecting %v,"+
|
|
|
|
"got %v", 1, len(pendingChannels))
|
|
|
|
}
|
|
|
|
|
2018-02-07 04:11:11 +01:00
|
|
|
// The broadcast height of the pending channel should have been set
|
2017-06-06 00:02:27 +02:00
|
|
|
// properly.
|
|
|
|
if pendingChannels[0].FundingBroadcastHeight != broadcastHeight {
|
|
|
|
t.Fatalf("broadcast height mismatch: expected %v, got %v",
|
|
|
|
pendingChannels[0].FundingBroadcastHeight,
|
|
|
|
broadcastHeight)
|
|
|
|
}
|
|
|
|
|
2017-06-16 22:28:26 +02:00
|
|
|
chanOpenLoc := lnwire.ShortChannelID{
|
|
|
|
BlockHeight: 5,
|
|
|
|
TxIndex: 10,
|
|
|
|
TxPosition: 15,
|
|
|
|
}
|
2017-11-10 05:56:04 +01:00
|
|
|
err = pendingChannels[0].MarkAsOpen(chanOpenLoc)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to mark channel as open")
|
2017-01-23 08:31:01 +01:00
|
|
|
|
2018-03-28 02:07:15 +02:00
|
|
|
if pendingChannels[0].IsPending {
|
|
|
|
t.Fatalf("channel marked open should no longer be pending")
|
|
|
|
}
|
|
|
|
|
2018-05-02 01:27:20 +02:00
|
|
|
if pendingChannels[0].ShortChanID() != chanOpenLoc {
|
2018-03-28 02:07:15 +02:00
|
|
|
t.Fatalf("channel opening height not updated: expected %v, "+
|
2018-05-02 01:27:20 +02:00
|
|
|
"got %v", spew.Sdump(pendingChannels[0].ShortChanID()),
|
2018-03-28 02:07:15 +02:00
|
|
|
chanOpenLoc)
|
|
|
|
}
|
|
|
|
|
2017-05-11 01:44:40 +02:00
|
|
|
// Next, we'll re-fetch the channel to ensure that the open height was
|
|
|
|
// properly set.
|
|
|
|
openChans, err := cdb.FetchAllChannels()
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to fetch channels")
|
2018-05-02 01:27:20 +02:00
|
|
|
if openChans[0].ShortChanID() != chanOpenLoc {
|
2017-05-11 01:44:40 +02:00
|
|
|
t.Fatalf("channel opening heights don't match: expected %v, "+
|
2018-05-02 01:27:20 +02:00
|
|
|
"got %v", spew.Sdump(openChans[0].ShortChanID()),
|
2017-06-16 22:28:26 +02:00
|
|
|
chanOpenLoc)
|
2017-05-11 01:44:40 +02:00
|
|
|
}
|
2017-06-06 00:02:27 +02:00
|
|
|
if openChans[0].FundingBroadcastHeight != broadcastHeight {
|
|
|
|
t.Fatalf("broadcast height mismatch: expected %v, got %v",
|
|
|
|
openChans[0].FundingBroadcastHeight,
|
|
|
|
broadcastHeight)
|
|
|
|
}
|
2017-05-11 01:44:40 +02:00
|
|
|
|
2017-01-23 08:31:01 +01:00
|
|
|
pendingChannels, err = cdb.FetchPendingChannels()
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to list pending channels")
|
2017-01-23 08:31:01 +01:00
|
|
|
|
|
|
|
if len(pendingChannels) != 0 {
|
|
|
|
t.Fatalf("incorrect number of pending channels: expecting %v,"+
|
|
|
|
"got %v", 0, len(pendingChannels))
|
|
|
|
}
|
|
|
|
}
|
2017-05-05 00:21:56 +02:00
|
|
|
|
|
|
|
func TestFetchClosedChannels(t *testing.T) {
|
2017-06-17 00:59:20 +02:00
|
|
|
t.Parallel()
|
|
|
|
|
2022-08-27 09:04:55 +02:00
|
|
|
fullDB, err := MakeTestDB(t)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to make test database")
|
2017-05-05 00:21:56 +02:00
|
|
|
|
2021-09-21 19:18:17 +02:00
|
|
|
cdb := fullDB.ChannelStateDB()
|
|
|
|
|
2020-02-05 14:39:31 +01:00
|
|
|
// Create an open channel in the database.
|
|
|
|
state := createTestChannel(t, cdb, openChannelOption())
|
2017-05-05 00:21:56 +02:00
|
|
|
|
|
|
|
// Next, close the channel by including a close channel summary in the
|
|
|
|
// database.
|
|
|
|
summary := &ChannelCloseSummary{
|
2017-07-29 20:24:28 +02:00
|
|
|
ChanPoint: state.FundingOutpoint,
|
2017-05-15 04:02:59 +02:00
|
|
|
ClosingTXID: rev,
|
|
|
|
RemotePub: state.IdentityPub,
|
|
|
|
Capacity: state.Capacity,
|
2017-11-10 05:56:04 +01:00
|
|
|
SettledBalance: state.LocalCommitment.LocalBalance.ToSatoshis(),
|
|
|
|
TimeLockedBalance: state.RemoteCommitment.LocalBalance.ToSatoshis() + 10000,
|
2018-04-04 08:57:55 +02:00
|
|
|
CloseType: RemoteForceClose,
|
2017-05-15 04:02:59 +02:00
|
|
|
IsPending: true,
|
2018-08-14 04:21:37 +02:00
|
|
|
LocalChanConfig: state.LocalChanCfg,
|
2017-05-05 00:21:56 +02:00
|
|
|
}
|
|
|
|
if err := state.CloseChannel(summary); err != nil {
|
|
|
|
t.Fatalf("unable to close channel: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Query the database to ensure that the channel has now been properly
|
|
|
|
// closed. We should get the same result whether querying for pending
|
|
|
|
// channels only, or not.
|
|
|
|
pendingClosed, err := cdb.FetchClosedChannels(true)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "failed fetching closed channels")
|
2017-05-05 00:21:56 +02:00
|
|
|
if len(pendingClosed) != 1 {
|
|
|
|
t.Fatalf("incorrect number of pending closed channels: expecting %v,"+
|
|
|
|
"got %v", 1, len(pendingClosed))
|
|
|
|
}
|
|
|
|
if !reflect.DeepEqual(summary, pendingClosed[0]) {
|
|
|
|
t.Fatalf("database summaries don't match: expected %v got %v",
|
|
|
|
spew.Sdump(summary), spew.Sdump(pendingClosed[0]))
|
|
|
|
}
|
|
|
|
closed, err := cdb.FetchClosedChannels(false)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "failed fetching all closed channels")
|
2017-05-05 00:21:56 +02:00
|
|
|
if len(closed) != 1 {
|
|
|
|
t.Fatalf("incorrect number of closed channels: expecting %v, "+
|
|
|
|
"got %v", 1, len(closed))
|
|
|
|
}
|
|
|
|
if !reflect.DeepEqual(summary, closed[0]) {
|
|
|
|
t.Fatalf("database summaries don't match: expected %v got %v",
|
|
|
|
spew.Sdump(summary), spew.Sdump(closed[0]))
|
|
|
|
}
|
|
|
|
|
2017-11-10 05:56:04 +01:00
|
|
|
// Mark the channel as fully closed.
|
2017-07-29 20:24:28 +02:00
|
|
|
err = cdb.MarkChanFullyClosed(&state.FundingOutpoint)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "failed fully closing channel")
|
2017-05-05 00:21:56 +02:00
|
|
|
|
|
|
|
// The channel should no longer be considered pending, but should still
|
|
|
|
// be retrieved when fetching all the closed channels.
|
|
|
|
closed, err = cdb.FetchClosedChannels(false)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "failed fetching closed channels")
|
2017-05-05 00:21:56 +02:00
|
|
|
if len(closed) != 1 {
|
|
|
|
t.Fatalf("incorrect number of closed channels: expecting %v, "+
|
|
|
|
"got %v", 1, len(closed))
|
|
|
|
}
|
|
|
|
pendingClose, err := cdb.FetchClosedChannels(true)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "failed fetching channels pending close")
|
2017-05-05 00:21:56 +02:00
|
|
|
if len(pendingClose) != 0 {
|
|
|
|
t.Fatalf("incorrect number of closed channels: expecting %v, "+
|
|
|
|
"got %v", 0, len(closed))
|
|
|
|
}
|
|
|
|
}
|
2018-05-02 01:55:22 +02:00
|
|
|
|
2019-01-04 02:33:34 +01:00
|
|
|
// TestFetchWaitingCloseChannels ensures that the correct channels that are
|
|
|
|
// waiting to be closed are returned.
|
|
|
|
func TestFetchWaitingCloseChannels(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
const numChannels = 2
|
|
|
|
const broadcastHeight = 99
|
|
|
|
|
|
|
|
// We'll start by creating two channels within our test database. One of
|
|
|
|
// them will have their funding transaction confirmed on-chain, while
|
|
|
|
// the other one will remain unconfirmed.
|
2022-08-27 09:04:55 +02:00
|
|
|
fullDB, err := MakeTestDB(t)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to make test database")
|
2019-01-04 02:33:34 +01:00
|
|
|
|
2021-09-21 19:18:17 +02:00
|
|
|
cdb := fullDB.ChannelStateDB()
|
|
|
|
|
2019-01-04 02:33:34 +01:00
|
|
|
channels := make([]*OpenChannel, numChannels)
|
|
|
|
for i := 0; i < numChannels; i++ {
|
2020-02-05 14:39:31 +01:00
|
|
|
// Create a pending channel in the database at the broadcast
|
|
|
|
// height.
|
|
|
|
channels[i] = createTestChannel(
|
2021-09-21 19:18:17 +02:00
|
|
|
t, cdb, pendingHeightOption(broadcastHeight),
|
2020-02-05 14:39:31 +01:00
|
|
|
)
|
2019-01-04 02:33:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// We'll only confirm the first one.
|
|
|
|
channelConf := lnwire.ShortChannelID{
|
|
|
|
BlockHeight: broadcastHeight + 1,
|
|
|
|
TxIndex: 10,
|
|
|
|
TxPosition: 15,
|
|
|
|
}
|
|
|
|
if err := channels[0].MarkAsOpen(channelConf); err != nil {
|
|
|
|
t.Fatalf("unable to mark channel as open: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Then, we'll mark the channels as if their commitments were broadcast.
|
|
|
|
// This would happen in the event of a force close and should make the
|
|
|
|
// channels enter a state of waiting close.
|
|
|
|
for _, channel := range channels {
|
2019-09-06 13:14:39 +02:00
|
|
|
closeTx := wire.NewMsgTx(2)
|
|
|
|
closeTx.AddTxIn(
|
|
|
|
&wire.TxIn{
|
|
|
|
PreviousOutPoint: channel.FundingOutpoint,
|
|
|
|
},
|
|
|
|
)
|
2019-12-04 22:30:46 +01:00
|
|
|
|
2020-02-21 12:24:23 +01:00
|
|
|
if err := channel.MarkCommitmentBroadcasted(closeTx, true); err != nil {
|
2019-01-04 02:33:34 +01:00
|
|
|
t.Fatalf("unable to mark commitment broadcast: %v", err)
|
|
|
|
}
|
2019-12-04 22:29:30 +01:00
|
|
|
|
2019-12-04 22:30:46 +01:00
|
|
|
// Now try to marking a coop close with a nil tx. This should
|
|
|
|
// succeed, but it shouldn't exit when queried.
|
2020-02-21 12:24:23 +01:00
|
|
|
if err = channel.MarkCoopBroadcasted(nil, true); err != nil {
|
2019-12-04 22:30:46 +01:00
|
|
|
t.Fatalf("unable to mark nil coop broadcast: %v", err)
|
|
|
|
}
|
|
|
|
_, err := channel.BroadcastedCooperative()
|
|
|
|
if err != ErrNoCloseTx {
|
|
|
|
t.Fatalf("expected no closing tx error, got: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Finally, modify the close tx deterministically and also mark
|
|
|
|
// it as coop closed. Later we will test that distinct
|
|
|
|
// transactions are returned for both coop and force closes.
|
2019-12-04 22:29:30 +01:00
|
|
|
closeTx.TxIn[0].PreviousOutPoint.Index ^= 1
|
2020-02-21 12:24:23 +01:00
|
|
|
if err := channel.MarkCoopBroadcasted(closeTx, true); err != nil {
|
2019-12-04 22:29:30 +01:00
|
|
|
t.Fatalf("unable to mark coop broadcast: %v", err)
|
|
|
|
}
|
2019-01-04 02:33:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Now, we'll fetch all the channels waiting to be closed from the
|
|
|
|
// database. We should expect to see both channels above, even if any of
|
|
|
|
// them haven't had their funding transaction confirm on-chain.
|
2021-09-21 19:18:17 +02:00
|
|
|
waitingCloseChannels, err := cdb.FetchWaitingCloseChannels()
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to fetch all waiting close channels")
|
2019-12-04 22:29:30 +01:00
|
|
|
if len(waitingCloseChannels) != numChannels {
|
2019-01-04 02:33:34 +01:00
|
|
|
t.Fatalf("expected %d channels waiting to be closed, got %d", 2,
|
|
|
|
len(waitingCloseChannels))
|
|
|
|
}
|
|
|
|
expectedChannels := make(map[wire.OutPoint]struct{})
|
|
|
|
for _, channel := range channels {
|
|
|
|
expectedChannels[channel.FundingOutpoint] = struct{}{}
|
|
|
|
}
|
|
|
|
for _, channel := range waitingCloseChannels {
|
|
|
|
if _, ok := expectedChannels[channel.FundingOutpoint]; !ok {
|
|
|
|
t.Fatalf("expected channel %v to be waiting close",
|
|
|
|
channel.FundingOutpoint)
|
|
|
|
}
|
2019-09-06 13:14:39 +02:00
|
|
|
|
2019-12-04 22:29:30 +01:00
|
|
|
chanPoint := channel.FundingOutpoint
|
|
|
|
|
|
|
|
// Assert that the force close transaction is retrievable.
|
|
|
|
forceCloseTx, err := channel.BroadcastedCommitment()
|
2019-09-06 13:14:39 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Unable to retrieve commitment: %v", err)
|
|
|
|
}
|
|
|
|
|
2019-12-04 22:29:30 +01:00
|
|
|
if forceCloseTx.TxIn[0].PreviousOutPoint != chanPoint {
|
|
|
|
t.Fatalf("expected outpoint %v, got %v",
|
|
|
|
chanPoint,
|
|
|
|
forceCloseTx.TxIn[0].PreviousOutPoint)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assert that the coop close transaction is retrievable.
|
|
|
|
coopCloseTx, err := channel.BroadcastedCooperative()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unable to retrieve coop close: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
chanPoint.Index ^= 1
|
|
|
|
if coopCloseTx.TxIn[0].PreviousOutPoint != chanPoint {
|
2019-09-06 13:14:39 +02:00
|
|
|
t.Fatalf("expected outpoint %v, got %v",
|
2019-12-04 22:29:30 +01:00
|
|
|
chanPoint,
|
|
|
|
coopCloseTx.TxIn[0].PreviousOutPoint)
|
2019-09-06 13:14:39 +02:00
|
|
|
}
|
2019-01-04 02:33:34 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-06 14:56:27 +01:00
|
|
|
// TestShutdownInfo tests that a channel's shutdown info can correctly be
|
|
|
|
// persisted and retrieved.
|
|
|
|
func TestShutdownInfo(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
localInit bool
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "local node initiated",
|
|
|
|
localInit: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "remote node initiated",
|
|
|
|
localInit: false,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
test := test
|
|
|
|
|
|
|
|
t.Run(test.name, func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
testShutdownInfo(t, test.localInit)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func testShutdownInfo(t *testing.T, locallyInitiated bool) {
|
|
|
|
fullDB, err := MakeTestDB(t)
|
|
|
|
require.NoError(t, err, "unable to make test database")
|
|
|
|
|
|
|
|
cdb := fullDB.ChannelStateDB()
|
|
|
|
|
|
|
|
// First a test channel.
|
|
|
|
channel := createTestChannel(t, cdb)
|
|
|
|
|
|
|
|
// We haven't persisted any shutdown info for this channel yet.
|
|
|
|
_, err = channel.ShutdownInfo()
|
|
|
|
require.Error(t, err, ErrNoShutdownInfo)
|
|
|
|
|
|
|
|
// Construct a new delivery script and create a new ShutdownInfo object.
|
|
|
|
script := []byte{1, 3, 4, 5}
|
|
|
|
|
|
|
|
// Create a ShutdownInfo struct.
|
|
|
|
shutdownInfo := NewShutdownInfo(script, locallyInitiated)
|
|
|
|
|
|
|
|
// Persist the shutdown info.
|
|
|
|
require.NoError(t, channel.MarkShutdownSent(shutdownInfo))
|
|
|
|
|
|
|
|
// We should now be able to retrieve the shutdown info.
|
|
|
|
info, err := channel.ShutdownInfo()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.True(t, info.IsSome())
|
|
|
|
|
|
|
|
// Assert that the decoded values of the shutdown info are correct.
|
|
|
|
info.WhenSome(func(info ShutdownInfo) {
|
|
|
|
require.EqualValues(t, script, info.DeliveryScript.Val)
|
|
|
|
require.Equal(t, locallyInitiated, info.LocalInitiator.Val)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-04-04 22:09:15 +02:00
|
|
|
// TestRefresh asserts that Refresh updates the in-memory state of another
|
|
|
|
// OpenChannel to reflect a preceding call to MarkOpen on a different
|
|
|
|
// OpenChannel.
|
|
|
|
func TestRefresh(t *testing.T) {
|
2018-05-02 01:55:22 +02:00
|
|
|
t.Parallel()
|
|
|
|
|
2022-08-27 09:04:55 +02:00
|
|
|
fullDB, err := MakeTestDB(t)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to make test database")
|
2018-05-02 01:55:22 +02:00
|
|
|
|
2021-09-21 19:18:17 +02:00
|
|
|
cdb := fullDB.ChannelStateDB()
|
|
|
|
|
2018-05-02 01:55:22 +02:00
|
|
|
// First create a test channel.
|
2020-02-05 14:39:31 +01:00
|
|
|
state := createTestChannel(t, cdb)
|
2018-05-02 01:55:22 +02:00
|
|
|
|
|
|
|
// Next, locate the pending channel with the database.
|
|
|
|
pendingChannels, err := cdb.FetchPendingChannels()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unable to load pending channels; %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var pendingChannel *OpenChannel
|
|
|
|
for _, channel := range pendingChannels {
|
|
|
|
if channel.FundingOutpoint == state.FundingOutpoint {
|
|
|
|
pendingChannel = channel
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if pendingChannel == nil {
|
|
|
|
t.Fatalf("unable to find pending channel with funding "+
|
|
|
|
"outpoint=%v: %v", state.FundingOutpoint, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Next, simulate the confirmation of the channel by marking it as
|
|
|
|
// pending within the database.
|
|
|
|
chanOpenLoc := lnwire.ShortChannelID{
|
|
|
|
BlockHeight: 105,
|
|
|
|
TxIndex: 10,
|
|
|
|
TxPosition: 15,
|
|
|
|
}
|
|
|
|
|
|
|
|
err = state.MarkAsOpen(chanOpenLoc)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to mark channel open")
|
2018-05-02 01:55:22 +02:00
|
|
|
|
|
|
|
// The short_chan_id of the receiver to MarkAsOpen should reflect the
|
|
|
|
// open location, but the other pending channel should remain unchanged.
|
|
|
|
if state.ShortChanID() == pendingChannel.ShortChanID() {
|
|
|
|
t.Fatalf("pending channel short_chan_ID should not have been " +
|
|
|
|
"updated before refreshing short_chan_id")
|
|
|
|
}
|
|
|
|
|
2018-06-19 14:36:12 +02:00
|
|
|
// Now that the receiver's short channel id has been updated, check to
|
|
|
|
// ensure that the channel packager's source has been updated as well.
|
|
|
|
// This ensures that the packager will read and write to buckets
|
|
|
|
// corresponding to the new short chan id, instead of the prior.
|
|
|
|
if state.Packager.(*ChannelPackager).source != chanOpenLoc {
|
|
|
|
t.Fatalf("channel packager source was not updated: want %v, "+
|
|
|
|
"got %v", chanOpenLoc,
|
|
|
|
state.Packager.(*ChannelPackager).source)
|
|
|
|
}
|
|
|
|
|
2022-04-04 22:09:15 +02:00
|
|
|
// Now, refresh the state of the pending channel.
|
|
|
|
err = pendingChannel.Refresh()
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unable to refresh short_chan_id")
|
2018-05-02 01:55:22 +02:00
|
|
|
|
|
|
|
// This should result in both OpenChannel's now having the same
|
|
|
|
// ShortChanID.
|
|
|
|
if state.ShortChanID() != pendingChannel.ShortChanID() {
|
|
|
|
t.Fatalf("expected pending channel short_chan_id to be "+
|
|
|
|
"refreshed: want %v, got %v", state.ShortChanID(),
|
|
|
|
pendingChannel.ShortChanID())
|
|
|
|
}
|
2018-06-19 14:36:12 +02:00
|
|
|
|
|
|
|
// Check to ensure that the _other_ OpenChannel channel packager's
|
|
|
|
// source has also been updated after the refresh. This ensures that the
|
|
|
|
// other packagers will read and write to buckets corresponding to the
|
|
|
|
// updated short chan id.
|
|
|
|
if pendingChannel.Packager.(*ChannelPackager).source != chanOpenLoc {
|
|
|
|
t.Fatalf("channel packager source was not updated: want %v, "+
|
|
|
|
"got %v", chanOpenLoc,
|
|
|
|
pendingChannel.Packager.(*ChannelPackager).source)
|
|
|
|
}
|
2019-11-05 08:57:38 +01:00
|
|
|
|
|
|
|
// Check to ensure that this channel is no longer pending and this field
|
|
|
|
// is up to date.
|
|
|
|
if pendingChannel.IsPending {
|
|
|
|
t.Fatalf("channel pending state wasn't updated: want false got true")
|
|
|
|
}
|
2018-05-02 01:55:22 +02:00
|
|
|
}
|
2020-02-21 12:24:23 +01:00
|
|
|
|
|
|
|
// TestCloseInitiator tests the setting of close initiator statuses for
|
|
|
|
// cooperative closes and local force closes.
|
|
|
|
func TestCloseInitiator(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
// updateChannel is called to update the channel as broadcast,
|
|
|
|
// cooperatively or not, based on the test's requirements.
|
|
|
|
updateChannel func(c *OpenChannel) error
|
|
|
|
expectedStatuses []ChannelStatus
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "local coop close",
|
|
|
|
// Mark the channel as cooperatively closed, initiated
|
|
|
|
// by the local party.
|
|
|
|
updateChannel: func(c *OpenChannel) error {
|
|
|
|
return c.MarkCoopBroadcasted(
|
|
|
|
&wire.MsgTx{}, true,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
expectedStatuses: []ChannelStatus{
|
|
|
|
ChanStatusLocalCloseInitiator,
|
|
|
|
ChanStatusCoopBroadcasted,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "remote coop close",
|
|
|
|
// Mark the channel as cooperatively closed, initiated
|
|
|
|
// by the remote party.
|
|
|
|
updateChannel: func(c *OpenChannel) error {
|
|
|
|
return c.MarkCoopBroadcasted(
|
|
|
|
&wire.MsgTx{}, false,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
expectedStatuses: []ChannelStatus{
|
|
|
|
ChanStatusRemoteCloseInitiator,
|
|
|
|
ChanStatusCoopBroadcasted,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "local force close",
|
|
|
|
// Mark the channel's commitment as broadcast with
|
|
|
|
// local initiator.
|
|
|
|
updateChannel: func(c *OpenChannel) error {
|
|
|
|
return c.MarkCommitmentBroadcasted(
|
|
|
|
&wire.MsgTx{}, true,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
expectedStatuses: []ChannelStatus{
|
|
|
|
ChanStatusLocalCloseInitiator,
|
|
|
|
ChanStatusCommitBroadcasted,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
test := test
|
|
|
|
|
|
|
|
t.Run(test.name, func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
2022-08-27 09:04:55 +02:00
|
|
|
fullDB, err := MakeTestDB(t)
|
2020-02-21 12:24:23 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unable to make test database: %v",
|
|
|
|
err)
|
|
|
|
}
|
|
|
|
|
2021-09-21 19:18:17 +02:00
|
|
|
cdb := fullDB.ChannelStateDB()
|
|
|
|
|
2020-02-21 12:24:23 +01:00
|
|
|
// Create an open channel.
|
|
|
|
channel := createTestChannel(
|
|
|
|
t, cdb, openChannelOption(),
|
|
|
|
)
|
|
|
|
|
|
|
|
err = test.updateChannel(channel)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Lookup open channels in the database.
|
|
|
|
dbChans, err := fetchChannels(
|
|
|
|
cdb, pendingChannelFilter(false),
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
if len(dbChans) != 1 {
|
|
|
|
t.Fatalf("expected 1 channel, got: %v",
|
|
|
|
len(dbChans))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that the statuses that we expect were written
|
|
|
|
// to disk.
|
|
|
|
for _, status := range test.expectedStatuses {
|
|
|
|
if !dbChans[0].HasChanStatus(status) {
|
|
|
|
t.Fatalf("expected channel to have "+
|
|
|
|
"status: %v, has status: %v",
|
|
|
|
status, dbChans[0].chanStatus)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2020-02-21 12:24:23 +01:00
|
|
|
|
|
|
|
// TestCloseChannelStatus tests setting of a channel status on the historical
|
|
|
|
// channel on channel close.
|
|
|
|
func TestCloseChannelStatus(t *testing.T) {
|
2022-08-27 09:04:55 +02:00
|
|
|
fullDB, err := MakeTestDB(t)
|
2020-02-21 12:24:23 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unable to make test database: %v",
|
|
|
|
err)
|
|
|
|
}
|
|
|
|
|
2021-09-21 19:18:17 +02:00
|
|
|
cdb := fullDB.ChannelStateDB()
|
|
|
|
|
2020-02-21 12:24:23 +01:00
|
|
|
// Create an open channel.
|
|
|
|
channel := createTestChannel(
|
|
|
|
t, cdb, openChannelOption(),
|
|
|
|
)
|
|
|
|
|
|
|
|
if err := channel.CloseChannel(
|
|
|
|
&ChannelCloseSummary{
|
|
|
|
ChanPoint: channel.FundingOutpoint,
|
|
|
|
RemotePub: channel.IdentityPub,
|
|
|
|
}, ChanStatusRemoteCloseInitiator,
|
|
|
|
); err != nil {
|
|
|
|
t.Fatalf("unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
histChan, err := channel.Db.FetchHistoricalChannel(
|
|
|
|
&channel.FundingOutpoint,
|
|
|
|
)
|
2022-05-05 22:11:50 +02:00
|
|
|
require.NoError(t, err, "unexpected error")
|
2020-02-21 12:24:23 +01:00
|
|
|
|
|
|
|
if !histChan.HasChanStatus(ChanStatusRemoteCloseInitiator) {
|
|
|
|
t.Fatalf("channel should have status")
|
|
|
|
}
|
|
|
|
}
|
2020-03-19 10:00:53 +01:00
|
|
|
|
2020-04-11 01:01:21 +02:00
|
|
|
// TestHasChanStatus asserts the behavior of HasChanStatus by checking the
|
|
|
|
// behavior of various status flags in addition to the special case of
|
|
|
|
// ChanStatusDefault which is treated like a flag in the code base even though
|
|
|
|
// it isn't.
|
|
|
|
func TestHasChanStatus(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
status ChannelStatus
|
|
|
|
expHas map[ChannelStatus]bool
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "default",
|
|
|
|
status: ChanStatusDefault,
|
|
|
|
expHas: map[ChannelStatus]bool{
|
|
|
|
ChanStatusDefault: true,
|
|
|
|
ChanStatusBorked: false,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "single flag",
|
|
|
|
status: ChanStatusBorked,
|
|
|
|
expHas: map[ChannelStatus]bool{
|
|
|
|
ChanStatusDefault: false,
|
|
|
|
ChanStatusBorked: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "multiple flags",
|
|
|
|
status: ChanStatusBorked | ChanStatusLocalDataLoss,
|
|
|
|
expHas: map[ChannelStatus]bool{
|
|
|
|
ChanStatusDefault: false,
|
|
|
|
ChanStatusBorked: true,
|
|
|
|
ChanStatusLocalDataLoss: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
test := test
|
|
|
|
|
|
|
|
t.Run(test.name, func(t *testing.T) {
|
|
|
|
c := &OpenChannel{
|
|
|
|
chanStatus: test.status,
|
|
|
|
}
|
|
|
|
|
|
|
|
for status, expHas := range test.expHas {
|
|
|
|
has := c.HasChanStatus(status)
|
|
|
|
if has == expHas {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
t.Fatalf("expected chan status to "+
|
|
|
|
"have %s? %t, got: %t",
|
|
|
|
status, expHas, has)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2020-08-10 20:11:12 +02:00
|
|
|
|
|
|
|
// TestKeyLocatorEncoding tests that we are able to serialize a given
|
|
|
|
// keychain.KeyLocator. After successfully encoding, we check that the decode
|
|
|
|
// output arrives at the same initial KeyLocator.
|
|
|
|
func TestKeyLocatorEncoding(t *testing.T) {
|
|
|
|
keyLoc := keychain.KeyLocator{
|
|
|
|
Family: keychain.KeyFamilyRevocationRoot,
|
|
|
|
Index: keyLocIndex,
|
|
|
|
}
|
|
|
|
|
|
|
|
// First, we'll encode the KeyLocator into a buffer.
|
|
|
|
var (
|
|
|
|
b bytes.Buffer
|
|
|
|
buf [8]byte
|
|
|
|
)
|
|
|
|
|
|
|
|
err := EKeyLocator(&b, &keyLoc, &buf)
|
|
|
|
require.NoError(t, err, "unable to encode key locator")
|
|
|
|
|
|
|
|
// Next, we'll attempt to decode the bytes into a new KeyLocator.
|
|
|
|
r := bytes.NewReader(b.Bytes())
|
|
|
|
var decodedKeyLoc keychain.KeyLocator
|
|
|
|
|
|
|
|
err = DKeyLocator(r, &decodedKeyLoc, &buf, 8)
|
|
|
|
require.NoError(t, err, "unable to decode key locator")
|
|
|
|
|
|
|
|
// Finally, we'll compare that the original KeyLocator and the decoded
|
|
|
|
// version are equal.
|
|
|
|
require.Equal(t, keyLoc, decodedKeyLoc)
|
|
|
|
}
|
2022-05-10 12:33:44 +02:00
|
|
|
|
|
|
|
// TestFinalHtlcs tests final htlc storage and retrieval.
|
|
|
|
func TestFinalHtlcs(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
2023-01-19 13:04:43 +01:00
|
|
|
fullDB, err := MakeTestDB(t, OptionStoreFinalHtlcResolutions(true))
|
2022-05-10 12:33:44 +02:00
|
|
|
require.NoError(t, err, "unable to make test database")
|
|
|
|
|
|
|
|
cdb := fullDB.ChannelStateDB()
|
|
|
|
|
|
|
|
chanID := lnwire.ShortChannelID{
|
|
|
|
BlockHeight: 1,
|
|
|
|
TxIndex: 2,
|
|
|
|
TxPosition: 3,
|
|
|
|
}
|
|
|
|
|
2022-08-29 13:20:52 +02:00
|
|
|
// Test unknown htlc lookup.
|
|
|
|
const unknownHtlcID = 999
|
|
|
|
|
|
|
|
_, err = cdb.LookupFinalHtlc(chanID, unknownHtlcID)
|
|
|
|
require.ErrorIs(t, err, ErrHtlcUnknown)
|
|
|
|
|
2022-05-10 12:33:44 +02:00
|
|
|
// Test offchain final htlcs.
|
|
|
|
const offchainHtlcID = 1
|
|
|
|
|
|
|
|
err = kvdb.Update(cdb.backend, func(tx kvdb.RwTx) error {
|
|
|
|
bucket, err := fetchFinalHtlcsBucketRw(
|
|
|
|
tx, chanID,
|
|
|
|
)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
return putFinalHtlc(bucket, offchainHtlcID, FinalHtlcInfo{
|
|
|
|
Settled: true,
|
|
|
|
Offchain: true,
|
|
|
|
})
|
|
|
|
}, func() {})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2022-08-29 13:20:52 +02:00
|
|
|
info, err := cdb.LookupFinalHtlc(chanID, offchainHtlcID)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.True(t, info.Settled)
|
|
|
|
require.True(t, info.Offchain)
|
|
|
|
|
2022-05-10 12:33:44 +02:00
|
|
|
// Test onchain final htlcs.
|
|
|
|
const onchainHtlcID = 2
|
|
|
|
|
|
|
|
err = cdb.PutOnchainFinalHtlcOutcome(chanID, onchainHtlcID, true)
|
|
|
|
require.NoError(t, err)
|
2022-08-29 13:20:52 +02:00
|
|
|
|
|
|
|
info, err = cdb.LookupFinalHtlc(chanID, onchainHtlcID)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.True(t, info.Settled)
|
|
|
|
require.False(t, info.Offchain)
|
|
|
|
|
|
|
|
// Test unknown htlc lookup for existing channel.
|
|
|
|
_, err = cdb.LookupFinalHtlc(chanID, unknownHtlcID)
|
|
|
|
require.ErrorIs(t, err, ErrHtlcUnknown)
|
2022-05-10 12:33:44 +02:00
|
|
|
}
|
2023-05-19 17:58:30 +02:00
|
|
|
|
|
|
|
// TestHTLCsExtraData tests serialization and deserialization of HTLCs
|
|
|
|
// combined with extra data.
|
|
|
|
func TestHTLCsExtraData(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
mockHtlc := HTLC{
|
|
|
|
Signature: testSig.Serialize(),
|
|
|
|
Incoming: false,
|
|
|
|
Amt: 10,
|
|
|
|
RHash: key,
|
|
|
|
RefundTimeout: 1,
|
|
|
|
OnionBlob: lnmock.MockOnion(),
|
|
|
|
}
|
|
|
|
|
2023-11-06 21:36:31 +01:00
|
|
|
// Add a blinding point to a htlc.
|
|
|
|
blindingPointHTLC := HTLC{
|
|
|
|
Signature: testSig.Serialize(),
|
|
|
|
Incoming: false,
|
|
|
|
Amt: 10,
|
|
|
|
RHash: key,
|
|
|
|
RefundTimeout: 1,
|
|
|
|
OnionBlob: lnmock.MockOnion(),
|
|
|
|
BlindingPoint: tlv.SomeRecordT(
|
|
|
|
tlv.NewPrimitiveRecord[lnwire.BlindingPointTlvType](
|
|
|
|
pubKey,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
}
|
|
|
|
|
2023-05-19 17:58:30 +02:00
|
|
|
testCases := []struct {
|
2023-11-06 21:36:31 +01:00
|
|
|
name string
|
|
|
|
htlcs []HTLC
|
|
|
|
blindingIdx int
|
2023-05-19 17:58:30 +02:00
|
|
|
}{
|
|
|
|
{
|
|
|
|
// Serialize multiple HLTCs with no extra data to
|
|
|
|
// assert that there is no regression for HTLCs with
|
|
|
|
// no extra data.
|
|
|
|
name: "no extra data",
|
|
|
|
htlcs: []HTLC{
|
|
|
|
mockHtlc, mockHtlc,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2023-11-06 21:36:31 +01:00
|
|
|
// Some HTLCs with extra data, some without.
|
2023-05-19 17:58:30 +02:00
|
|
|
name: "mixed extra data",
|
|
|
|
htlcs: []HTLC{
|
|
|
|
mockHtlc,
|
2023-11-06 21:36:31 +01:00
|
|
|
blindingPointHTLC,
|
2023-05-19 17:58:30 +02:00
|
|
|
mockHtlc,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, testCase := range testCases {
|
|
|
|
testCase := testCase
|
|
|
|
|
|
|
|
t.Run(testCase.name, func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
var b bytes.Buffer
|
|
|
|
err := SerializeHtlcs(&b, testCase.htlcs...)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
r := bytes.NewReader(b.Bytes())
|
|
|
|
htlcs, err := DeserializeHtlcs(r)
|
|
|
|
require.NoError(t, err)
|
2023-11-06 21:36:31 +01:00
|
|
|
|
|
|
|
require.EqualValues(t, len(testCase.htlcs), len(htlcs))
|
|
|
|
for i, htlc := range htlcs {
|
|
|
|
// We use the extra data field when we
|
|
|
|
// serialize, so we set to nil to be able to
|
|
|
|
// assert on equal for the test.
|
|
|
|
htlc.ExtraData = nil
|
|
|
|
require.Equal(t, testCase.htlcs[i], htlc)
|
|
|
|
}
|
2023-05-19 17:58:30 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestOnionBlobIncorrectLength tests HTLC deserialization in the case where
|
|
|
|
// the OnionBlob saved on disk is of an unexpected length. This error case is
|
|
|
|
// only expected in the case of database corruption (or some severe protocol
|
|
|
|
// breakdown/bug). A HTLC is manually serialized because we cannot force a
|
|
|
|
// case where we write an onion blob of incorrect length.
|
|
|
|
func TestOnionBlobIncorrectLength(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
var b bytes.Buffer
|
|
|
|
|
|
|
|
var numHtlcs uint16 = 1
|
|
|
|
require.NoError(t, WriteElement(&b, numHtlcs))
|
|
|
|
|
|
|
|
require.NoError(t, WriteElements(
|
|
|
|
&b,
|
|
|
|
// Number of HTLCs.
|
|
|
|
numHtlcs,
|
|
|
|
// Signature, incoming, amount, Rhash, Timeout.
|
|
|
|
testSig.Serialize(), false, lnwire.MilliSatoshi(10), key,
|
|
|
|
uint32(1),
|
|
|
|
// Write an onion blob that is half of our expected size.
|
|
|
|
bytes.Repeat([]byte{1}, lnwire.OnionPacketSize/2),
|
|
|
|
))
|
|
|
|
|
|
|
|
_, err := DeserializeHtlcs(&b)
|
|
|
|
require.ErrorIs(t, err, ErrOnionBlobLength)
|
|
|
|
}
|