mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-01-19 05:45:21 +01:00
99339f706f
This commit updates the interface methods from `lnwallet.WalletController` to take optional input set which can be used to create the tx.
3407 lines
106 KiB
Go
3407 lines
106 KiB
Go
package lnwallettest
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/sha256"
|
|
"encoding/hex"
|
|
"fmt"
|
|
"net"
|
|
"path/filepath"
|
|
"reflect"
|
|
"runtime"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/btcsuite/btcd/blockchain"
|
|
"github.com/btcsuite/btcd/btcec/v2"
|
|
"github.com/btcsuite/btcd/btcjson"
|
|
"github.com/btcsuite/btcd/btcutil"
|
|
"github.com/btcsuite/btcd/chaincfg"
|
|
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
|
"github.com/btcsuite/btcd/integration/rpctest"
|
|
"github.com/btcsuite/btcd/mempool"
|
|
"github.com/btcsuite/btcd/rpcclient"
|
|
"github.com/btcsuite/btcd/txscript"
|
|
"github.com/btcsuite/btcd/wire"
|
|
"github.com/btcsuite/btcwallet/chain"
|
|
"github.com/btcsuite/btcwallet/wallet"
|
|
"github.com/btcsuite/btcwallet/walletdb"
|
|
_ "github.com/btcsuite/btcwallet/walletdb/bdb"
|
|
"github.com/davecgh/go-spew/spew"
|
|
"github.com/lightninglabs/neutrino"
|
|
"github.com/lightningnetwork/lnd/blockcache"
|
|
"github.com/lightningnetwork/lnd/chainntnfs"
|
|
"github.com/lightningnetwork/lnd/chainntnfs/btcdnotify"
|
|
"github.com/lightningnetwork/lnd/channeldb"
|
|
"github.com/lightningnetwork/lnd/input"
|
|
"github.com/lightningnetwork/lnd/keychain"
|
|
"github.com/lightningnetwork/lnd/kvdb"
|
|
"github.com/lightningnetwork/lnd/labels"
|
|
"github.com/lightningnetwork/lnd/lntest/unittest"
|
|
"github.com/lightningnetwork/lnd/lnwallet"
|
|
"github.com/lightningnetwork/lnd/lnwallet/btcwallet"
|
|
"github.com/lightningnetwork/lnd/lnwallet/chainfee"
|
|
"github.com/lightningnetwork/lnd/lnwallet/chanfunding"
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
var (
|
|
bobsPrivKey = []byte{
|
|
0x81, 0xb6, 0x37, 0xd8, 0xfc, 0xd2, 0xc6, 0xda,
|
|
0x63, 0x59, 0xe6, 0x96, 0x31, 0x13, 0xa1, 0x17,
|
|
0xd, 0xe7, 0x95, 0xe4, 0xb7, 0x25, 0xb8, 0x4d,
|
|
0x1e, 0xb, 0x4c, 0xfd, 0x9e, 0xc5, 0x8c, 0xe9,
|
|
}
|
|
|
|
// Use a hard-coded HD seed.
|
|
testHdSeed = chainhash.Hash{
|
|
0xb7, 0x94, 0x38, 0x5f, 0x2d, 0x1e, 0xf7, 0xab,
|
|
0x4d, 0x92, 0x73, 0xd1, 0x90, 0x63, 0x81, 0xb4,
|
|
0x4f, 0x2f, 0x6f, 0x25, 0x88, 0xa3, 0xef, 0xb9,
|
|
0x6a, 0x49, 0x18, 0x83, 0x31, 0x98, 0x47, 0x53,
|
|
}
|
|
|
|
aliceHDSeed = chainhash.Hash{
|
|
0xb7, 0x94, 0x38, 0x5f, 0x2d, 0x1e, 0xf7, 0xab,
|
|
0x4d, 0x92, 0x73, 0xd1, 0x90, 0x63, 0x81, 0xb4,
|
|
0x4f, 0x2f, 0x6f, 0x25, 0x18, 0xa3, 0xef, 0xb9,
|
|
0x64, 0x49, 0x18, 0x83, 0x31, 0x98, 0x47, 0x53,
|
|
}
|
|
bobHDSeed = chainhash.Hash{
|
|
0xb7, 0x94, 0x38, 0x5f, 0x2d, 0x1e, 0xf7, 0xab,
|
|
0x4d, 0x92, 0x73, 0xd1, 0x90, 0x63, 0x81, 0xb4,
|
|
0x4f, 0x2f, 0x6f, 0x25, 0x98, 0xa3, 0xef, 0xb9,
|
|
0x69, 0x49, 0x18, 0x83, 0x31, 0x98, 0x47, 0x53,
|
|
}
|
|
|
|
netParams = &chaincfg.RegressionNetParams
|
|
chainHash = netParams.GenesisHash
|
|
|
|
_, alicePub = btcec.PrivKeyFromBytes(testHdSeed[:])
|
|
_, bobPub = btcec.PrivKeyFromBytes(bobsPrivKey)
|
|
|
|
// The number of confirmations required to consider any created channel
|
|
// open.
|
|
numReqConfs uint16 = 1
|
|
|
|
csvDelay uint16 = 4
|
|
|
|
bobAddr, _ = net.ResolveTCPAddr("tcp", "10.0.0.2:9000")
|
|
aliceAddr, _ = net.ResolveTCPAddr("tcp", "10.0.0.3:9000")
|
|
|
|
defaultMaxLocalCsvDelay uint16 = 10000
|
|
)
|
|
|
|
// assertProperBalance asserts than the total value of the unspent outputs
|
|
// within the wallet are *exactly* amount. If unable to retrieve the current
|
|
// balance, or the assertion fails, the test will halt with a fatal error.
|
|
func assertProperBalance(t *testing.T, lw *lnwallet.LightningWallet,
|
|
numConfirms int32, amount float64) {
|
|
|
|
balance, err := lw.ConfirmedBalance(numConfirms, lnwallet.DefaultAccountName)
|
|
require.NoError(t, err, "unable to query for balance")
|
|
if balance.ToBTC() != amount {
|
|
t.Fatalf("wallet credits not properly loaded, should have 40BTC, "+
|
|
"instead have %v", balance)
|
|
}
|
|
}
|
|
|
|
func assertReservationDeleted(res *lnwallet.ChannelReservation, t *testing.T) {
|
|
if err := res.Cancel(); err == nil {
|
|
t.Fatalf("reservation wasn't deleted from wallet")
|
|
}
|
|
}
|
|
|
|
// mineAndAssertTxInBlock asserts that a transaction is included within the next
|
|
// block mined.
|
|
func mineAndAssertTxInBlock(t *testing.T, miner *rpctest.Harness,
|
|
txid chainhash.Hash) {
|
|
|
|
t.Helper()
|
|
|
|
// First, we'll wait for the transaction to arrive in the mempool.
|
|
if err := waitForMempoolTx(miner, &txid); err != nil {
|
|
t.Fatalf("unable to find %v in the mempool: %v", txid, err)
|
|
}
|
|
|
|
// We'll mined a block to confirm it.
|
|
blockHashes, err := miner.Client.Generate(1)
|
|
require.NoError(t, err, "unable to generate new block")
|
|
|
|
// Finally, we'll check it was actually mined in this block.
|
|
block, err := miner.Client.GetBlock(blockHashes[0])
|
|
if err != nil {
|
|
t.Fatalf("unable to get block %v: %v", blockHashes[0], err)
|
|
}
|
|
if len(block.Transactions) != 2 {
|
|
t.Fatalf("expected 2 transactions in block, found %d",
|
|
len(block.Transactions))
|
|
}
|
|
txHash := block.Transactions[1].TxHash()
|
|
if txHash != txid {
|
|
t.Fatalf("expected transaction %v to be mined, found %v", txid,
|
|
txHash)
|
|
}
|
|
}
|
|
|
|
// newPkScript generates a new public key script of the given address type.
|
|
func newPkScript(t *testing.T, w *lnwallet.LightningWallet,
|
|
addrType lnwallet.AddressType) []byte {
|
|
|
|
t.Helper()
|
|
|
|
addr, err := w.NewAddress(addrType, false, lnwallet.DefaultAccountName)
|
|
require.NoError(t, err, "unable to create new address")
|
|
pkScript, err := txscript.PayToAddrScript(addr)
|
|
require.NoError(t, err, "unable to create output script")
|
|
|
|
return pkScript
|
|
}
|
|
|
|
// sendCoins is a helper function that encompasses all the things needed for two
|
|
// parties to send on-chain funds to each other.
|
|
func sendCoins(t *testing.T, miner *rpctest.Harness,
|
|
sender, receiver *lnwallet.LightningWallet, output *wire.TxOut,
|
|
feeRate chainfee.SatPerKWeight, mineBlock bool, minConf int32) *wire.MsgTx { //nolint:unparam
|
|
|
|
t.Helper()
|
|
|
|
tx, err := sender.SendOutputs(
|
|
nil, []*wire.TxOut{output}, feeRate, minConf, labels.External,
|
|
sender.Cfg.CoinSelectionStrategy,
|
|
)
|
|
require.NoError(t, err, "unable to send transaction")
|
|
|
|
if mineBlock {
|
|
mineAndAssertTxInBlock(t, miner, tx.TxHash())
|
|
}
|
|
|
|
if err := waitForWalletSync(miner, sender); err != nil {
|
|
t.Fatalf("unable to sync alice: %v", err)
|
|
}
|
|
if err := waitForWalletSync(miner, receiver); err != nil {
|
|
t.Fatalf("unable to sync bob: %v", err)
|
|
}
|
|
|
|
return tx
|
|
}
|
|
|
|
// assertTxInWallet asserts that a transaction exists in the wallet with the
|
|
// expected confirmation status.
|
|
func assertTxInWallet(t *testing.T, w *lnwallet.LightningWallet,
|
|
txHash chainhash.Hash, confirmed bool) {
|
|
|
|
t.Helper()
|
|
|
|
// We'll fetch all of our transaction and go through each one until
|
|
// finding the expected transaction with its expected confirmation
|
|
// status.
|
|
txs, err := w.ListTransactionDetails(0, btcwallet.UnconfirmedHeight, "")
|
|
require.NoError(t, err, "unable to retrieve transactions")
|
|
for _, tx := range txs {
|
|
if tx.Hash != txHash {
|
|
continue
|
|
}
|
|
if tx.NumConfirmations <= 0 && confirmed {
|
|
t.Fatalf("expected transaction %v to be confirmed",
|
|
txHash)
|
|
}
|
|
if tx.NumConfirmations > 0 && !confirmed {
|
|
t.Fatalf("expected transaction %v to be unconfirmed",
|
|
txHash)
|
|
}
|
|
|
|
// We've found the transaction and it matches the desired
|
|
// confirmation status, so we can exit.
|
|
return
|
|
}
|
|
|
|
t.Fatalf("transaction %v not found", txHash)
|
|
}
|
|
|
|
func loadTestCredits(miner *rpctest.Harness, w *lnwallet.LightningWallet,
|
|
numOutputs int, btcPerOutput float64) error {
|
|
|
|
// For initial neutrino connection, wait a second.
|
|
// TODO(aakselrod): Eliminate the need for this.
|
|
switch w.BackEnd() {
|
|
case "neutrino":
|
|
time.Sleep(time.Second)
|
|
}
|
|
// Using the mining node, spend from a coinbase output numOutputs to
|
|
// give us btcPerOutput with each output.
|
|
satoshiPerOutput, err := btcutil.NewAmount(btcPerOutput)
|
|
if err != nil {
|
|
return fmt.Errorf("unable to create amt: %w", err)
|
|
}
|
|
expectedBalance, err := w.ConfirmedBalance(1, lnwallet.DefaultAccountName)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
expectedBalance += btcutil.Amount(int64(satoshiPerOutput) * int64(numOutputs))
|
|
addrs := make([]btcutil.Address, 0, numOutputs)
|
|
for i := 0; i < numOutputs; i++ {
|
|
// Grab a fresh address from the wallet to house this output.
|
|
walletAddr, err := w.NewAddress(
|
|
lnwallet.WitnessPubKey, false,
|
|
lnwallet.DefaultAccountName,
|
|
)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
script, err := txscript.PayToAddrScript(walletAddr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
addrs = append(addrs, walletAddr)
|
|
|
|
output := &wire.TxOut{
|
|
Value: int64(satoshiPerOutput),
|
|
PkScript: script,
|
|
}
|
|
if _, err := miner.SendOutputs([]*wire.TxOut{output}, 2500); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// TODO(roasbeef): shouldn't hardcode 10, use config param that dictates
|
|
// how many confs we wait before opening a channel.
|
|
// Generate 10 blocks with the mining node, this should mine all
|
|
// numOutputs transactions created above. We generate 10 blocks here
|
|
// in order to give all the outputs a "sufficient" number of confirmations.
|
|
if _, err := miner.Client.Generate(10); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Wait until the wallet has finished syncing up to the main chain.
|
|
ticker := time.NewTicker(100 * time.Millisecond)
|
|
timeout := time.After(30 * time.Second)
|
|
|
|
for range ticker.C {
|
|
balance, err := w.ConfirmedBalance(1, lnwallet.DefaultAccountName)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if balance == expectedBalance {
|
|
break
|
|
}
|
|
select {
|
|
case <-timeout:
|
|
synced, _, err := w.IsSynced()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return fmt.Errorf("timed out after 30 seconds "+
|
|
"waiting for balance %v, current balance %v, "+
|
|
"synced: %t", expectedBalance, balance, synced)
|
|
default:
|
|
}
|
|
}
|
|
ticker.Stop()
|
|
|
|
return nil
|
|
}
|
|
|
|
// createTestWallet creates a test LightningWallet will a total of 20BTC
|
|
// available for funding channels.
|
|
func createTestWallet(tempTestDir string, miningNode *rpctest.Harness,
|
|
netParams *chaincfg.Params, notifier chainntnfs.ChainNotifier,
|
|
wc lnwallet.WalletController, keyRing keychain.SecretKeyRing,
|
|
signer input.Signer, bio lnwallet.BlockChainIO) (*lnwallet.LightningWallet, error) {
|
|
|
|
dbDir := filepath.Join(tempTestDir, "cdb")
|
|
fullDB, err := channeldb.Open(dbDir)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
cfg := lnwallet.Config{
|
|
Database: fullDB.ChannelStateDB(),
|
|
Notifier: notifier,
|
|
SecretKeyRing: keyRing,
|
|
WalletController: wc,
|
|
Signer: signer,
|
|
ChainIO: bio,
|
|
FeeEstimator: chainfee.NewStaticEstimator(2500, 0),
|
|
NetParams: *netParams,
|
|
CoinSelectionStrategy: wallet.CoinSelectionLargest,
|
|
}
|
|
|
|
wallet, err := lnwallet.NewLightningWallet(cfg)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := wallet.Startup(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Load our test wallet with 20 outputs each holding 4BTC.
|
|
if err := loadTestCredits(miningNode, wallet, 20, 4); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return wallet, nil
|
|
}
|
|
|
|
func testGetRecoveryInfo(miner *rpctest.Harness,
|
|
alice, bob *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
// alice's wallet is in recovery mode
|
|
expectedRecoveryMode := true
|
|
expectedProgress := float64(1)
|
|
|
|
isRecoveryMode, progress, err := alice.GetRecoveryInfo()
|
|
require.NoError(t, err, "unable to get alice's recovery info")
|
|
|
|
require.Equal(t,
|
|
expectedRecoveryMode, isRecoveryMode, "recovery mode incorrect",
|
|
)
|
|
require.Equal(t, expectedProgress, progress, "progress incorrect")
|
|
|
|
// Generate 5 blocks and check the recovery process again.
|
|
const numBlocksMined = 5
|
|
_, err = miner.Client.Generate(numBlocksMined)
|
|
require.NoError(t, err, "unable to mine blocks")
|
|
|
|
// Check the recovery process. Once synced, the progress should be 1.
|
|
err = waitForWalletSync(miner, alice)
|
|
require.NoError(t, err, "Couldn't sync Alice's wallet")
|
|
|
|
isRecoveryMode, progress, err = alice.GetRecoveryInfo()
|
|
require.NoError(t, err, "unable to get alice's recovery info")
|
|
|
|
require.Equal(t,
|
|
expectedRecoveryMode, isRecoveryMode, "recovery mode incorrect",
|
|
)
|
|
require.Equal(t, expectedProgress, progress, "progress incorrect")
|
|
|
|
// bob's wallet is not in recovery mode
|
|
expectedRecoveryMode = false
|
|
expectedProgress = float64(0)
|
|
|
|
isRecoveryMode, progress, err = bob.GetRecoveryInfo()
|
|
require.NoError(t, err, "unable to get bob's recovery info")
|
|
|
|
require.Equal(t,
|
|
expectedRecoveryMode, isRecoveryMode, "recovery mode incorrect",
|
|
)
|
|
require.Equal(t, expectedProgress, progress, "progress incorrect")
|
|
}
|
|
|
|
func testDualFundingReservationWorkflow(miner *rpctest.Harness,
|
|
alice, bob *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
t.Skipf("dual funding isn't exposed on the p2p layer")
|
|
|
|
fundingAmount, err := btcutil.NewAmount(5)
|
|
require.NoError(t, err, "unable to create amt")
|
|
|
|
// In this scenario, we'll test a dual funder reservation, with each
|
|
// side putting in 10 BTC.
|
|
|
|
// Alice initiates a channel funded with 5 BTC for each side, so 10 BTC
|
|
// total. She also generates 2 BTC in change.
|
|
feePerKw, err := alice.Cfg.FeeEstimator.EstimateFeePerKW(1)
|
|
require.NoError(t, err, "unable to query fee estimator")
|
|
aliceReq := &lnwallet.InitFundingReserveMsg{
|
|
ChainHash: chainHash,
|
|
NodeID: bobPub,
|
|
NodeAddr: bobAddr,
|
|
LocalFundingAmt: fundingAmount,
|
|
RemoteFundingAmt: fundingAmount,
|
|
CommitFeePerKw: feePerKw,
|
|
FundingFeePerKw: feePerKw,
|
|
PushMSat: 0,
|
|
Flags: lnwire.FFAnnounceChannel,
|
|
}
|
|
aliceChanReservation, err := alice.InitChannelReservation(aliceReq)
|
|
require.NoError(t, err, "unable to initialize funding reservation")
|
|
aliceChanReservation.SetNumConfsRequired(numReqConfs)
|
|
bounds := &channeldb.ChannelStateBounds{
|
|
ChanReserve: fundingAmount / 100,
|
|
MaxPendingAmount: lnwire.NewMSatFromSatoshis(fundingAmount),
|
|
MinHTLC: 1,
|
|
MaxAcceptedHtlcs: input.MaxHTLCNumber / 2,
|
|
}
|
|
commitParams := &channeldb.CommitmentParams{
|
|
DustLimit: lnwallet.DustLimitUnknownWitness(),
|
|
CsvDelay: csvDelay,
|
|
}
|
|
err = aliceChanReservation.CommitConstraints(
|
|
bounds, commitParams, defaultMaxLocalCsvDelay, false,
|
|
)
|
|
require.NoError(t, err, "unable to verify constraints")
|
|
|
|
// The channel reservation should now be populated with a multi-sig key
|
|
// from our HD chain, a change output with 3 BTC, and 2 outputs
|
|
// selected of 4 BTC each. Additionally, the rest of the items needed
|
|
// to fulfill a funding contribution should also have been filled in.
|
|
aliceContribution := aliceChanReservation.OurContribution()
|
|
if len(aliceContribution.Inputs) != 2 {
|
|
t.Fatalf("outputs for funding tx not properly selected, have %v "+
|
|
"outputs should have 2", len(aliceContribution.Inputs))
|
|
}
|
|
assertContributionInitPopulated(t, aliceContribution)
|
|
|
|
// Bob does the same, generating his own contribution. He then also
|
|
// receives' Alice's contribution, and consumes that so we can continue
|
|
// the funding process.
|
|
bobReq := &lnwallet.InitFundingReserveMsg{
|
|
ChainHash: chainHash,
|
|
NodeID: alicePub,
|
|
NodeAddr: aliceAddr,
|
|
LocalFundingAmt: fundingAmount,
|
|
RemoteFundingAmt: fundingAmount,
|
|
CommitFeePerKw: feePerKw,
|
|
FundingFeePerKw: feePerKw,
|
|
PushMSat: 0,
|
|
Flags: lnwire.FFAnnounceChannel,
|
|
}
|
|
bobChanReservation, err := bob.InitChannelReservation(bobReq)
|
|
require.NoError(t, err, "bob unable to init channel reservation")
|
|
err = bobChanReservation.CommitConstraints(
|
|
bounds, commitParams, defaultMaxLocalCsvDelay, true,
|
|
)
|
|
require.NoError(t, err, "unable to verify constraints")
|
|
bobChanReservation.SetNumConfsRequired(numReqConfs)
|
|
|
|
assertContributionInitPopulated(t, bobChanReservation.OurContribution())
|
|
|
|
err = bobChanReservation.ProcessContribution(aliceContribution)
|
|
require.NoError(t, err, "bob unable to process alice's contribution")
|
|
assertContributionInitPopulated(t, bobChanReservation.TheirContribution())
|
|
|
|
bobContribution := bobChanReservation.OurContribution()
|
|
|
|
// Bob then sends over his contribution, which will be consumed by
|
|
// Alice. After this phase, Alice should have all the necessary
|
|
// material required to craft the funding transaction and commitment
|
|
// transactions.
|
|
err = aliceChanReservation.ProcessContribution(bobContribution)
|
|
require.NoError(t, err, "alice unable to process bob's contribution")
|
|
assertContributionInitPopulated(t, aliceChanReservation.TheirContribution())
|
|
|
|
// At this point, all Alice's signatures should be fully populated.
|
|
aliceFundingSigs, aliceCommitSig := aliceChanReservation.OurSignatures()
|
|
if aliceFundingSigs == nil {
|
|
t.Fatalf("alice's funding signatures not populated")
|
|
}
|
|
if aliceCommitSig == nil {
|
|
t.Fatalf("alice's commit signatures not populated")
|
|
}
|
|
|
|
// Additionally, Bob's signatures should also be fully populated.
|
|
bobFundingSigs, bobCommitSig := bobChanReservation.OurSignatures()
|
|
if bobFundingSigs == nil {
|
|
t.Fatalf("bob's funding signatures not populated")
|
|
}
|
|
if bobCommitSig == nil {
|
|
t.Fatalf("bob's commit signatures not populated")
|
|
}
|
|
|
|
// To conclude, we'll consume first Alice's signatures with Bob, and
|
|
// then the other way around.
|
|
_, err = aliceChanReservation.CompleteReservation(
|
|
bobFundingSigs, bobCommitSig,
|
|
)
|
|
if err != nil {
|
|
for _, in := range aliceChanReservation.FinalFundingTx().TxIn {
|
|
fmt.Println(in.PreviousOutPoint.String())
|
|
}
|
|
t.Fatalf("unable to consume alice's sigs: %v", err)
|
|
}
|
|
_, err = bobChanReservation.CompleteReservation(
|
|
aliceFundingSigs, aliceCommitSig,
|
|
)
|
|
require.NoError(t, err, "unable to consume bob's sigs")
|
|
|
|
// At this point, the funding tx should have been populated.
|
|
fundingTx := aliceChanReservation.FinalFundingTx()
|
|
if fundingTx == nil {
|
|
t.Fatalf("funding transaction never created!")
|
|
}
|
|
|
|
// The resulting active channel state should have been persisted to the
|
|
// DB.
|
|
fundingSha := fundingTx.TxHash()
|
|
aliceChannels, err := alice.Cfg.Database.FetchOpenChannels(bobPub)
|
|
require.NoError(t, err, "unable to retrieve channel from DB")
|
|
if !bytes.Equal(aliceChannels[0].FundingOutpoint.Hash[:], fundingSha[:]) {
|
|
t.Fatalf("channel state not properly saved")
|
|
}
|
|
if !aliceChannels[0].ChanType.IsDualFunder() {
|
|
t.Fatalf("channel not detected as dual funder")
|
|
}
|
|
bobChannels, err := bob.Cfg.Database.FetchOpenChannels(alicePub)
|
|
require.NoError(t, err, "unable to retrieve channel from DB")
|
|
if !bytes.Equal(bobChannels[0].FundingOutpoint.Hash[:], fundingSha[:]) {
|
|
t.Fatalf("channel state not properly saved")
|
|
}
|
|
if !bobChannels[0].ChanType.IsDualFunder() {
|
|
t.Fatalf("channel not detected as dual funder")
|
|
}
|
|
|
|
// Let Alice publish the funding transaction.
|
|
err = alice.PublishTransaction(fundingTx, "")
|
|
require.NoError(t, err, "unable to publish funding tx")
|
|
|
|
// Mine a single block, the funding transaction should be included
|
|
// within this block.
|
|
err = waitForMempoolTx(miner, &fundingSha)
|
|
require.NoError(t, err, "tx not relayed to miner")
|
|
blockHashes, err := miner.Client.Generate(1)
|
|
require.NoError(t, err, "unable to generate block")
|
|
block, err := miner.Client.GetBlock(blockHashes[0])
|
|
require.NoError(t, err, "unable to find block")
|
|
if len(block.Transactions) != 2 {
|
|
t.Fatalf("funding transaction wasn't mined: %v", err)
|
|
}
|
|
blockTx := block.Transactions[1]
|
|
if blockTx.TxHash() != fundingSha {
|
|
t.Fatalf("incorrect transaction was mined")
|
|
}
|
|
|
|
assertReservationDeleted(aliceChanReservation, t)
|
|
assertReservationDeleted(bobChanReservation, t)
|
|
|
|
// Wait for wallets to catch up to prevent issues in subsequent tests.
|
|
err = waitForWalletSync(miner, alice)
|
|
require.NoError(t, err, "unable to sync alice")
|
|
err = waitForWalletSync(miner, bob)
|
|
require.NoError(t, err, "unable to sync bob")
|
|
}
|
|
|
|
func testFundingTransactionLockedOutputs(miner *rpctest.Harness,
|
|
alice, _ *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
// Create a single channel asking for 16 BTC total.
|
|
fundingAmount, err := btcutil.NewAmount(8)
|
|
require.NoError(t, err, "unable to create amt")
|
|
feePerKw, err := alice.Cfg.FeeEstimator.EstimateFeePerKW(1)
|
|
require.NoError(t, err, "unable to query fee estimator")
|
|
req := &lnwallet.InitFundingReserveMsg{
|
|
ChainHash: chainHash,
|
|
NodeID: bobPub,
|
|
NodeAddr: bobAddr,
|
|
LocalFundingAmt: fundingAmount,
|
|
RemoteFundingAmt: 0,
|
|
CommitFeePerKw: feePerKw,
|
|
FundingFeePerKw: feePerKw,
|
|
PushMSat: 0,
|
|
Flags: lnwire.FFAnnounceChannel,
|
|
PendingChanID: [32]byte{0, 1, 2, 3},
|
|
}
|
|
if _, err := alice.InitChannelReservation(req); err != nil {
|
|
t.Fatalf("unable to initialize funding reservation 1: %v", err)
|
|
}
|
|
|
|
// Now attempt to reserve funds for another channel, this time
|
|
// requesting 900 BTC. We only have around 64BTC worth of outpoints
|
|
// that aren't locked, so this should fail.
|
|
amt, err := btcutil.NewAmount(900)
|
|
require.NoError(t, err, "unable to create amt")
|
|
failedReq := &lnwallet.InitFundingReserveMsg{
|
|
ChainHash: chainHash,
|
|
NodeID: bobPub,
|
|
NodeAddr: bobAddr,
|
|
LocalFundingAmt: amt,
|
|
RemoteFundingAmt: 0,
|
|
CommitFeePerKw: feePerKw,
|
|
FundingFeePerKw: feePerKw,
|
|
PushMSat: 0,
|
|
Flags: lnwire.FFAnnounceChannel,
|
|
PendingChanID: [32]byte{1, 2, 3, 4},
|
|
}
|
|
failedReservation, err := alice.InitChannelReservation(failedReq)
|
|
if err == nil {
|
|
t.Fatalf("not error returned, should fail on coin selection")
|
|
}
|
|
if _, ok := err.(*chanfunding.ErrInsufficientFunds); !ok {
|
|
t.Fatalf("error not coinselect error: %v", err)
|
|
}
|
|
if failedReservation != nil {
|
|
t.Fatalf("reservation should be nil")
|
|
}
|
|
}
|
|
|
|
func testFundingCancellationNotEnoughFunds(miner *rpctest.Harness,
|
|
alice, _ *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
feePerKw, err := alice.Cfg.FeeEstimator.EstimateFeePerKW(1)
|
|
require.NoError(t, err, "unable to query fee estimator")
|
|
|
|
// Create a reservation for 44 BTC.
|
|
fundingAmount, err := btcutil.NewAmount(44)
|
|
require.NoError(t, err, "unable to create amt")
|
|
req := &lnwallet.InitFundingReserveMsg{
|
|
ChainHash: chainHash,
|
|
NodeID: bobPub,
|
|
NodeAddr: bobAddr,
|
|
LocalFundingAmt: fundingAmount,
|
|
RemoteFundingAmt: 0,
|
|
CommitFeePerKw: feePerKw,
|
|
FundingFeePerKw: feePerKw,
|
|
PushMSat: 0,
|
|
Flags: lnwire.FFAnnounceChannel,
|
|
PendingChanID: [32]byte{2, 3, 4, 5},
|
|
}
|
|
chanReservation, err := alice.InitChannelReservation(req)
|
|
require.NoError(t, err, "unable to initialize funding reservation")
|
|
|
|
// Attempt to create another channel with 44 BTC, this should fail.
|
|
req.PendingChanID = [32]byte{3, 4, 5, 6}
|
|
_, err = alice.InitChannelReservation(req)
|
|
if _, ok := err.(*chanfunding.ErrInsufficientFunds); !ok {
|
|
t.Fatalf("coin selection succeeded should have insufficient funds: %v",
|
|
err)
|
|
}
|
|
|
|
// Now cancel that old reservation.
|
|
if err := chanReservation.Cancel(); err != nil {
|
|
t.Fatalf("unable to cancel reservation: %v", err)
|
|
}
|
|
|
|
// Those outpoints should no longer be locked.
|
|
lockedOutPoints := alice.LockedOutpoints()
|
|
if len(lockedOutPoints) != 0 {
|
|
t.Fatalf("outpoints still locked")
|
|
}
|
|
|
|
// Reservation ID should no longer be tracked.
|
|
numReservations := alice.ActiveReservations()
|
|
if len(alice.ActiveReservations()) != 0 {
|
|
t.Fatalf("should have 0 reservations, instead have %v",
|
|
numReservations)
|
|
}
|
|
|
|
// TODO(roasbeef): create method like Balance that ignores locked
|
|
// outpoints, will let us fail early/fast instead of querying and
|
|
// attempting coin selection.
|
|
|
|
// Request to fund a new channel should now succeed.
|
|
req.PendingChanID = [32]byte{4, 5, 6, 7, 8}
|
|
if _, err := alice.InitChannelReservation(req); err != nil {
|
|
t.Fatalf("unable to initialize funding reservation: %v", err)
|
|
}
|
|
}
|
|
|
|
func testCancelNonExistentReservation(miner *rpctest.Harness,
|
|
alice, _ *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
feePerKw, err := alice.Cfg.FeeEstimator.EstimateFeePerKW(1)
|
|
require.NoError(t, err, "unable to query fee estimator")
|
|
|
|
req := &lnwallet.InitFundingReserveMsg{
|
|
CommitFeePerKw: feePerKw,
|
|
PushMSat: 10,
|
|
Flags: lnwire.FFAnnounceChannel,
|
|
CommitType: lnwallet.CommitmentTypeTweakless,
|
|
PendingChanID: [32]byte{},
|
|
}
|
|
|
|
// Create our own reservation, give it some ID.
|
|
res, err := lnwallet.NewChannelReservation(
|
|
10000, 10000, alice, 22, &testHdSeed, 0, req,
|
|
)
|
|
require.NoError(t, err, "unable to create res")
|
|
|
|
// Attempt to cancel this reservation. This should fail, we know
|
|
// nothing of it.
|
|
if err := res.Cancel(); err == nil {
|
|
t.Fatalf("canceled non-existent reservation")
|
|
}
|
|
}
|
|
|
|
func testReservationInitiatorBalanceBelowDustCancel(miner *rpctest.Harness,
|
|
alice, _ *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
// We'll attempt to create a new reservation with an extremely high
|
|
// commitment fee rate. This should push our balance into the negative
|
|
// and result in a failure to create the reservation.
|
|
const numBTC = 4
|
|
fundingAmount, err := btcutil.NewAmount(numBTC)
|
|
require.NoError(t, err, "unable to create amt")
|
|
|
|
feePerKw := chainfee.SatPerKWeight(
|
|
numBTC * numBTC * btcutil.SatoshiPerBitcoin,
|
|
)
|
|
req := &lnwallet.InitFundingReserveMsg{
|
|
ChainHash: chainHash,
|
|
NodeID: bobPub,
|
|
NodeAddr: bobAddr,
|
|
LocalFundingAmt: fundingAmount,
|
|
RemoteFundingAmt: 0,
|
|
CommitFeePerKw: feePerKw,
|
|
FundingFeePerKw: 1000,
|
|
PushMSat: 0,
|
|
Flags: lnwire.FFAnnounceChannel,
|
|
CommitType: lnwallet.CommitmentTypeTweakless,
|
|
PendingChanID: [32]byte{1},
|
|
}
|
|
_, err = alice.InitChannelReservation(req)
|
|
switch {
|
|
case err == nil:
|
|
t.Fatalf("initialization should have failed due to " +
|
|
"insufficient local amount")
|
|
|
|
case !strings.Contains(err.Error(), "funder balance too small"):
|
|
t.Fatalf("incorrect error: %v", err)
|
|
}
|
|
}
|
|
|
|
func assertContributionInitPopulated(t *testing.T, c *lnwallet.ChannelContribution) {
|
|
_, _, line, _ := runtime.Caller(1)
|
|
|
|
if c.FirstCommitmentPoint == nil {
|
|
t.Fatalf("line #%v: commitment point not fond", line)
|
|
}
|
|
|
|
if c.CsvDelay == 0 {
|
|
t.Fatalf("line #%v: csv delay not set", line)
|
|
}
|
|
|
|
if c.MultiSigKey.PubKey == nil {
|
|
t.Fatalf("line #%v: multi-sig key not set", line)
|
|
}
|
|
if c.RevocationBasePoint.PubKey == nil {
|
|
t.Fatalf("line #%v: revocation key not set", line)
|
|
}
|
|
if c.PaymentBasePoint.PubKey == nil {
|
|
t.Fatalf("line #%v: payment key not set", line)
|
|
}
|
|
if c.DelayBasePoint.PubKey == nil {
|
|
t.Fatalf("line #%v: delay key not set", line)
|
|
}
|
|
|
|
if c.DustLimit == 0 {
|
|
t.Fatalf("line #%v: dust limit not set", line)
|
|
}
|
|
if c.MaxPendingAmount == 0 {
|
|
t.Fatalf("line #%v: max pending amt not set", line)
|
|
}
|
|
if c.ChanReserve == 0 {
|
|
t.Fatalf("line #%v: chan reserve not set", line)
|
|
}
|
|
if c.MinHTLC == 0 {
|
|
t.Fatalf("line #%v: min htlc not set", line)
|
|
}
|
|
if c.MaxAcceptedHtlcs == 0 {
|
|
t.Fatalf("line #%v: max accepted htlc's not set", line)
|
|
}
|
|
}
|
|
|
|
func testSingleFunderReservationWorkflow(miner *rpctest.Harness,
|
|
alice, bob *lnwallet.LightningWallet, t *testing.T,
|
|
commitType lnwallet.CommitmentType,
|
|
aliceChanFunder chanfunding.Assembler, fetchFundingTx func() *wire.MsgTx,
|
|
pendingChanID [32]byte, thawHeight uint32) {
|
|
|
|
// For this scenario, Alice will be the channel initiator while bob
|
|
// will act as the responder to the workflow.
|
|
|
|
// First, Alice will Initialize a reservation for a channel with 4 BTC
|
|
// funded solely by us. We'll also initially push 1 BTC of the channel
|
|
// towards Bob's side.
|
|
fundingAmt, err := btcutil.NewAmount(4)
|
|
require.NoError(t, err, "unable to create amt")
|
|
pushAmt := lnwire.NewMSatFromSatoshis(btcutil.SatoshiPerBitcoin)
|
|
feePerKw, err := alice.Cfg.FeeEstimator.EstimateFeePerKW(1)
|
|
require.NoError(t, err, "unable to query fee estimator")
|
|
aliceReq := &lnwallet.InitFundingReserveMsg{
|
|
ChainHash: chainHash,
|
|
PendingChanID: pendingChanID,
|
|
NodeID: bobPub,
|
|
NodeAddr: bobAddr,
|
|
LocalFundingAmt: fundingAmt,
|
|
RemoteFundingAmt: 0,
|
|
CommitFeePerKw: feePerKw,
|
|
FundingFeePerKw: feePerKw,
|
|
PushMSat: pushAmt,
|
|
Flags: lnwire.FFAnnounceChannel,
|
|
CommitType: commitType,
|
|
ChanFunder: aliceChanFunder,
|
|
}
|
|
aliceChanReservation, err := alice.InitChannelReservation(aliceReq)
|
|
require.NoError(t, err, "unable to init channel reservation")
|
|
aliceChanReservation.SetNumConfsRequired(numReqConfs)
|
|
bounds := &channeldb.ChannelStateBounds{
|
|
ChanReserve: fundingAmt / 100,
|
|
MaxPendingAmount: lnwire.NewMSatFromSatoshis(fundingAmt),
|
|
MinHTLC: 1,
|
|
MaxAcceptedHtlcs: input.MaxHTLCNumber / 2,
|
|
}
|
|
commitParams := &channeldb.CommitmentParams{
|
|
DustLimit: lnwallet.DustLimitUnknownWitness(),
|
|
CsvDelay: csvDelay,
|
|
}
|
|
err = aliceChanReservation.CommitConstraints(
|
|
bounds, commitParams, defaultMaxLocalCsvDelay, false,
|
|
)
|
|
require.NoError(t, err, "unable to verify constraints")
|
|
|
|
// Verify all contribution fields have been set properly, but only if
|
|
// Alice is the funder herself.
|
|
aliceContribution := aliceChanReservation.OurContribution()
|
|
if fetchFundingTx == nil {
|
|
if len(aliceContribution.Inputs) < 1 {
|
|
t.Fatalf("outputs for funding tx not properly "+
|
|
"selected, have %v outputs should at least 1",
|
|
len(aliceContribution.Inputs))
|
|
}
|
|
if len(aliceContribution.ChangeOutputs) != 1 {
|
|
t.Fatalf("coin selection failed, should have one "+
|
|
"change outputs, instead have: %v",
|
|
len(aliceContribution.ChangeOutputs))
|
|
}
|
|
}
|
|
assertContributionInitPopulated(t, aliceContribution)
|
|
|
|
// Next, Bob receives the initial request, generates a corresponding
|
|
// reservation initiation, then consume Alice's contribution.
|
|
bobReq := &lnwallet.InitFundingReserveMsg{
|
|
ChainHash: chainHash,
|
|
PendingChanID: pendingChanID,
|
|
NodeID: alicePub,
|
|
NodeAddr: aliceAddr,
|
|
LocalFundingAmt: 0,
|
|
RemoteFundingAmt: fundingAmt,
|
|
CommitFeePerKw: feePerKw,
|
|
FundingFeePerKw: feePerKw,
|
|
PushMSat: pushAmt,
|
|
Flags: lnwire.FFAnnounceChannel,
|
|
CommitType: commitType,
|
|
}
|
|
bobChanReservation, err := bob.InitChannelReservation(bobReq)
|
|
require.NoError(t, err, "unable to create bob reservation")
|
|
err = bobChanReservation.CommitConstraints(
|
|
bounds, commitParams, defaultMaxLocalCsvDelay, true,
|
|
)
|
|
require.NoError(t, err, "unable to verify constraints")
|
|
bobChanReservation.SetNumConfsRequired(numReqConfs)
|
|
|
|
// We'll ensure that Bob's contribution also gets generated properly.
|
|
bobContribution := bobChanReservation.OurContribution()
|
|
assertContributionInitPopulated(t, bobContribution)
|
|
|
|
// With his contribution generated, he can now process Alice's
|
|
// contribution.
|
|
err = bobChanReservation.ProcessSingleContribution(aliceContribution)
|
|
require.NoError(t, err, "bob unable to process alice's contribution")
|
|
assertContributionInitPopulated(t, bobChanReservation.TheirContribution())
|
|
|
|
// Bob will next send over his contribution to Alice, we simulate this
|
|
// by having Alice immediately process his contribution.
|
|
err = aliceChanReservation.ProcessContribution(bobContribution)
|
|
if err != nil {
|
|
t.Fatalf("alice unable to process bob's contribution: %v", err)
|
|
}
|
|
assertContributionInitPopulated(t, bobChanReservation.TheirContribution())
|
|
|
|
// At this point, Alice should have generated all the signatures
|
|
// required for the funding transaction, as well as Alice's commitment
|
|
// signature to bob, but only if the funding transaction was
|
|
// constructed internally.
|
|
aliceRemoteContribution := aliceChanReservation.TheirContribution()
|
|
aliceFundingSigs, aliceCommitSig := aliceChanReservation.OurSignatures()
|
|
if fetchFundingTx == nil && aliceFundingSigs == nil {
|
|
t.Fatalf("funding sigs not found")
|
|
}
|
|
if aliceCommitSig == nil {
|
|
t.Fatalf("commitment sig not found")
|
|
}
|
|
|
|
// Additionally, the funding tx and the funding outpoint should have
|
|
// been populated.
|
|
if aliceChanReservation.FinalFundingTx() == nil && fetchFundingTx == nil {
|
|
t.Fatalf("funding transaction never created!")
|
|
}
|
|
if aliceChanReservation.FundingOutpoint() == nil {
|
|
t.Fatalf("funding outpoint never created!")
|
|
}
|
|
|
|
// Their funds should also be filled in.
|
|
if len(aliceRemoteContribution.Inputs) != 0 {
|
|
t.Fatalf("bob shouldn't have any inputs, instead has %v",
|
|
len(aliceRemoteContribution.Inputs))
|
|
}
|
|
if len(aliceRemoteContribution.ChangeOutputs) != 0 {
|
|
t.Fatalf("bob shouldn't have any change outputs, instead "+
|
|
"has %v",
|
|
aliceRemoteContribution.ChangeOutputs[0].Value)
|
|
}
|
|
|
|
// Next, Alice will send over her signature for Bob's commitment
|
|
// transaction, as well as the funding outpoint.
|
|
fundingPoint := aliceChanReservation.FundingOutpoint()
|
|
_, err = bobChanReservation.CompleteReservationSingle(
|
|
fundingPoint, aliceCommitSig,
|
|
)
|
|
require.NoError(t, err, "bob unable to consume single reservation")
|
|
|
|
// Finally, we'll conclude the reservation process by sending over
|
|
// Bob's commitment signature, which is the final thing Alice needs to
|
|
// be able to safely broadcast the funding transaction.
|
|
_, bobCommitSig := bobChanReservation.OurSignatures()
|
|
if bobCommitSig == nil {
|
|
t.Fatalf("bob failed to generate commitment signature: %v", err)
|
|
}
|
|
_, err = aliceChanReservation.CompleteReservation(
|
|
nil, bobCommitSig,
|
|
)
|
|
require.NoError(t, err, "alice unable to complete reservation")
|
|
|
|
// If the caller provided an alternative way to obtain the funding tx,
|
|
// then we'll use that. Otherwise, we'll obtain it directly from Alice.
|
|
var fundingTx *wire.MsgTx
|
|
if fetchFundingTx != nil {
|
|
fundingTx = fetchFundingTx()
|
|
} else {
|
|
fundingTx = aliceChanReservation.FinalFundingTx()
|
|
}
|
|
|
|
// The resulting active channel state should have been persisted to the
|
|
// DB for both Alice and Bob.
|
|
fundingSha := fundingTx.TxHash()
|
|
aliceChannels, err := alice.Cfg.Database.FetchOpenChannels(bobPub)
|
|
require.NoError(t, err, "unable to retrieve channel from DB")
|
|
if len(aliceChannels) != 1 {
|
|
t.Fatalf("alice didn't save channel state: %v", err)
|
|
}
|
|
if !bytes.Equal(aliceChannels[0].FundingOutpoint.Hash[:], fundingSha[:]) {
|
|
t.Fatalf("channel state not properly saved: %v vs %v",
|
|
hex.EncodeToString(aliceChannels[0].FundingOutpoint.Hash[:]),
|
|
hex.EncodeToString(fundingSha[:]))
|
|
}
|
|
if !aliceChannels[0].IsInitiator {
|
|
t.Fatalf("alice not detected as channel initiator")
|
|
}
|
|
if !aliceChannels[0].ChanType.IsSingleFunder() {
|
|
t.Fatalf("channel type is incorrect, expected %v instead got %v",
|
|
channeldb.SingleFunderBit, aliceChannels[0].ChanType)
|
|
}
|
|
|
|
bobChannels, err := bob.Cfg.Database.FetchOpenChannels(alicePub)
|
|
require.NoError(t, err, "unable to retrieve channel from DB")
|
|
if len(bobChannels) != 1 {
|
|
t.Fatalf("bob didn't save channel state: %v", err)
|
|
}
|
|
if !bytes.Equal(bobChannels[0].FundingOutpoint.Hash[:], fundingSha[:]) {
|
|
t.Fatalf("channel state not properly saved: %v vs %v",
|
|
hex.EncodeToString(bobChannels[0].FundingOutpoint.Hash[:]),
|
|
hex.EncodeToString(fundingSha[:]))
|
|
}
|
|
if bobChannels[0].IsInitiator {
|
|
t.Fatalf("bob not detected as channel responder")
|
|
}
|
|
if !bobChannels[0].ChanType.IsSingleFunder() {
|
|
t.Fatalf("channel type is incorrect, expected %v instead got %v",
|
|
channeldb.SingleFunderBit, bobChannels[0].ChanType)
|
|
}
|
|
|
|
// Let Alice publish the funding transaction.
|
|
err = alice.PublishTransaction(fundingTx, "")
|
|
require.NoError(t, err, "unable to publish funding tx")
|
|
|
|
// Mine a single block, the funding transaction should be included
|
|
// within this block.
|
|
err = waitForMempoolTx(miner, &fundingSha)
|
|
require.NoError(t, err, "tx not relayed to miner")
|
|
blockHashes, err := miner.Client.Generate(1)
|
|
require.NoError(t, err, "unable to generate block")
|
|
block, err := miner.Client.GetBlock(blockHashes[0])
|
|
require.NoError(t, err, "unable to find block")
|
|
if len(block.Transactions) != 2 {
|
|
t.Fatalf("funding transaction wasn't mined: %d",
|
|
len(block.Transactions))
|
|
}
|
|
blockTx := block.Transactions[1]
|
|
if blockTx.TxHash() != fundingSha {
|
|
t.Fatalf("incorrect transaction was mined")
|
|
}
|
|
|
|
// If a frozen channel was requested, then we expect that both channel
|
|
// types show as being a frozen channel type.
|
|
aliceChanFrozen := aliceChannels[0].ChanType.IsFrozen()
|
|
bobChanFrozen := bobChannels[0].ChanType.IsFrozen()
|
|
if thawHeight != 0 && (!aliceChanFrozen || !bobChanFrozen) {
|
|
t.Fatalf("expected both alice and bob to have frozen chans: "+
|
|
"alice_frozen=%v, bob_frozen=%v", aliceChanFrozen,
|
|
bobChanFrozen)
|
|
}
|
|
if thawHeight != bobChannels[0].ThawHeight {
|
|
t.Fatalf("wrong thaw height: expected %v got %v", thawHeight,
|
|
bobChannels[0].ThawHeight)
|
|
}
|
|
if thawHeight != aliceChannels[0].ThawHeight {
|
|
t.Fatalf("wrong thaw height: expected %v got %v", thawHeight,
|
|
aliceChannels[0].ThawHeight)
|
|
}
|
|
|
|
assertReservationDeleted(aliceChanReservation, t)
|
|
assertReservationDeleted(bobChanReservation, t)
|
|
}
|
|
|
|
func testListTransactionDetails(miner *rpctest.Harness,
|
|
alice, _ *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
// Create 5 new outputs spendable by the wallet.
|
|
const numTxns = 5
|
|
const outputAmt = btcutil.SatoshiPerBitcoin
|
|
isOurAddress := make(map[string]bool)
|
|
txids := make(map[chainhash.Hash]struct{})
|
|
for i := 0; i < numTxns; i++ {
|
|
addr, err := alice.NewAddress(
|
|
lnwallet.WitnessPubKey, false,
|
|
lnwallet.DefaultAccountName,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to create new address: %v", err)
|
|
}
|
|
isOurAddress[addr.EncodeAddress()] = true
|
|
script, err := txscript.PayToAddrScript(addr)
|
|
if err != nil {
|
|
t.Fatalf("unable to create output script: %v", err)
|
|
}
|
|
|
|
output := &wire.TxOut{
|
|
Value: outputAmt,
|
|
PkScript: script,
|
|
}
|
|
txid, err := miner.SendOutputs([]*wire.TxOut{output}, 2500)
|
|
if err != nil {
|
|
t.Fatalf("unable to send coinbase: %v", err)
|
|
}
|
|
txids[*txid] = struct{}{}
|
|
}
|
|
|
|
// Get the miner's current best block height before we mine blocks.
|
|
_, startHeight, err := miner.Client.GetBestBlock()
|
|
require.NoError(t, err, "cannot get best block")
|
|
|
|
// Generate 10 blocks to mine all the transactions created above.
|
|
const numBlocksMined = 10
|
|
blocks, err := miner.Client.Generate(numBlocksMined)
|
|
require.NoError(t, err, "unable to mine blocks")
|
|
|
|
// Our new best block height should be our start height + the number of
|
|
// blocks we just mined.
|
|
chainTip := startHeight + numBlocksMined
|
|
|
|
// Next, fetch all the current transaction details. We should find all
|
|
// of our transactions between our start height before we generated
|
|
// blocks, and our end height, which is the chain tip. This query does
|
|
// not include unconfirmed transactions, since all of our transactions
|
|
// should be confirmed.
|
|
err = waitForWalletSync(miner, alice)
|
|
require.NoError(t, err, "Couldn't sync Alice's wallet")
|
|
txDetails, err := alice.ListTransactionDetails(
|
|
startHeight, chainTip, "",
|
|
)
|
|
require.NoError(t, err, "unable to fetch tx details")
|
|
|
|
// This is a mapping from:
|
|
// blockHash -> transactionHash -> transactionOutputs
|
|
blockTxOuts := make(map[chainhash.Hash]map[chainhash.Hash][]*wire.TxOut)
|
|
|
|
// Each of the transactions created above should be found with the
|
|
// proper details populated.
|
|
for _, txDetail := range txDetails {
|
|
if _, ok := txids[txDetail.Hash]; !ok {
|
|
continue
|
|
}
|
|
|
|
if txDetail.NumConfirmations != numBlocksMined {
|
|
t.Fatalf("num confs incorrect, got %v expected %v",
|
|
txDetail.NumConfirmations, numBlocksMined)
|
|
}
|
|
if txDetail.Value != outputAmt {
|
|
t.Fatalf("tx value incorrect, got %v expected %v",
|
|
txDetail.Value, outputAmt)
|
|
}
|
|
|
|
if !bytes.Equal(txDetail.BlockHash[:], blocks[0][:]) {
|
|
t.Fatalf("block hash mismatch, got %v expected %v",
|
|
txDetail.BlockHash, blocks[0])
|
|
}
|
|
|
|
// This fetches the transactions in a block so that we can compare the
|
|
// txouts stored in the mined transaction against the ones in the transaction
|
|
// details
|
|
if _, ok := blockTxOuts[*txDetail.BlockHash]; !ok {
|
|
fetchedBlock, err := alice.Cfg.ChainIO.GetBlock(txDetail.BlockHash)
|
|
if err != nil {
|
|
t.Fatalf("err fetching block: %s", err)
|
|
}
|
|
|
|
transactions := make(
|
|
map[chainhash.Hash][]*wire.TxOut,
|
|
len(fetchedBlock.Transactions),
|
|
)
|
|
for _, tx := range fetchedBlock.Transactions {
|
|
transactions[tx.TxHash()] = tx.Copy().TxOut
|
|
}
|
|
|
|
blockTxOuts[fetchedBlock.BlockHash()] = transactions
|
|
}
|
|
|
|
if txOuts, ok := blockTxOuts[*txDetail.BlockHash][txDetail.Hash]; !ok {
|
|
t.Fatalf("tx (%v) not found in block (%v)",
|
|
txDetail.Hash, txDetail.BlockHash)
|
|
} else {
|
|
var destinationOutputs []lnwallet.OutputDetail
|
|
|
|
for i, txOut := range txOuts {
|
|
sc, addrs, _, err :=
|
|
txscript.ExtractPkScriptAddrs(txOut.PkScript, &alice.Cfg.NetParams)
|
|
if err != nil {
|
|
t.Fatalf("err extract script addresses: %s", err)
|
|
}
|
|
destinationOutputs = append(destinationOutputs, lnwallet.OutputDetail{
|
|
OutputType: sc,
|
|
Addresses: addrs,
|
|
PkScript: txOut.PkScript,
|
|
OutputIndex: i,
|
|
Value: btcutil.Amount(txOut.Value),
|
|
IsOurAddress: isOurAddress[addrs[0].EncodeAddress()],
|
|
})
|
|
}
|
|
|
|
if !reflect.DeepEqual(txDetail.OutputDetails, destinationOutputs) {
|
|
t.Fatalf("destination outputs mismatch, got %v expected %v",
|
|
txDetail.OutputDetails, destinationOutputs)
|
|
}
|
|
}
|
|
|
|
delete(txids, txDetail.Hash)
|
|
}
|
|
if len(txids) != 0 {
|
|
t.Fatalf("all transactions not found in details: left=%v, "+
|
|
"returned_set=%v", spew.Sdump(txids),
|
|
spew.Sdump(txDetails))
|
|
}
|
|
|
|
// Next create a transaction paying to an output which isn't under the
|
|
// wallet's control.
|
|
minerAddr, err := miner.NewAddress()
|
|
require.NoError(t, err, "unable to generate address")
|
|
outputScript, err := txscript.PayToAddrScript(minerAddr)
|
|
require.NoError(t, err, "unable to make output script")
|
|
burnOutput := wire.NewTxOut(outputAmt, outputScript)
|
|
burnTX, err := alice.SendOutputs(
|
|
nil, []*wire.TxOut{burnOutput}, 2500, 1, labels.External,
|
|
alice.Cfg.CoinSelectionStrategy,
|
|
)
|
|
require.NoError(t, err, "unable to create burn tx")
|
|
burnTXID := burnTX.TxHash()
|
|
err = waitForMempoolTx(miner, &burnTXID)
|
|
require.NoError(t, err, "tx not relayed to miner")
|
|
|
|
// Before we mine the next block, we'll ensure that the above
|
|
// transaction shows up in the set of unconfirmed transactions returned
|
|
// by ListTransactionDetails.
|
|
err = waitForWalletSync(miner, alice)
|
|
require.NoError(t, err, "Couldn't sync Alice's wallet")
|
|
|
|
// Query our wallet for transactions from the chain tip, including
|
|
// unconfirmed transactions. The transaction above should be included
|
|
// with a confirmation height of 0, indicating that it has not been
|
|
// mined yet.
|
|
txDetails, err = alice.ListTransactionDetails(
|
|
chainTip, btcwallet.UnconfirmedHeight, "",
|
|
)
|
|
require.NoError(t, err, "unable to fetch tx details")
|
|
var mempoolTxFound bool
|
|
for _, txDetail := range txDetails {
|
|
if !bytes.Equal(txDetail.Hash[:], burnTXID[:]) {
|
|
continue
|
|
}
|
|
|
|
// Now that we've found the transaction, ensure that it has a
|
|
// negative number of confirmations to indicate that it's
|
|
// unconfirmed.
|
|
mempoolTxFound = true
|
|
if txDetail.NumConfirmations != 0 {
|
|
t.Fatalf("num confs incorrect, got %v expected %v",
|
|
txDetail.NumConfirmations, 0)
|
|
}
|
|
|
|
// We test that each txDetail has destination addresses. This ensures
|
|
// that even when we have 0 confirmation transactions, the destination
|
|
// addresses are returned.
|
|
var match bool
|
|
for _, o := range txDetail.OutputDetails {
|
|
for _, addr := range o.Addresses {
|
|
if addr.String() == minerAddr.String() {
|
|
match = true
|
|
break
|
|
}
|
|
}
|
|
}
|
|
if !match {
|
|
t.Fatalf("minerAddr: %v should have been a dest addr", minerAddr)
|
|
}
|
|
}
|
|
if !mempoolTxFound {
|
|
t.Fatalf("unable to find mempool tx in tx details!")
|
|
}
|
|
|
|
// Generate one block for our transaction to confirm in.
|
|
var numBlocks int32 = 1
|
|
burnBlock, err := miner.Client.Generate(uint32(numBlocks))
|
|
require.NoError(t, err, "unable to mine block")
|
|
|
|
// Progress our chain tip by the number of blocks we have just mined.
|
|
chainTip += numBlocks
|
|
|
|
// Fetch the transaction details again, the new transaction should be
|
|
// shown as debiting from the wallet's balance. Start and end height
|
|
// are inclusive, so we use chainTip for both parameters to get only
|
|
// transactions from the last block.
|
|
err = waitForWalletSync(miner, alice)
|
|
require.NoError(t, err, "Couldn't sync Alice's wallet")
|
|
txDetails, err = alice.ListTransactionDetails(chainTip, chainTip, "")
|
|
require.NoError(t, err, "unable to fetch tx details")
|
|
var burnTxFound bool
|
|
for _, txDetail := range txDetails {
|
|
if !bytes.Equal(txDetail.Hash[:], burnTXID[:]) {
|
|
continue
|
|
}
|
|
|
|
burnTxFound = true
|
|
if txDetail.NumConfirmations != 1 {
|
|
t.Fatalf("num confs incorrect, got %v expected %v",
|
|
txDetail.NumConfirmations, 1)
|
|
}
|
|
|
|
// We assert that the value is greater than the amount we
|
|
// attempted to send, as the wallet should have paid some amount
|
|
// of network fees.
|
|
if txDetail.Value >= -outputAmt {
|
|
fmt.Println(spew.Sdump(txDetail))
|
|
t.Fatalf("tx value incorrect, got %v expected %v",
|
|
int64(txDetail.Value), -int64(outputAmt))
|
|
}
|
|
if !bytes.Equal(txDetail.BlockHash[:], burnBlock[0][:]) {
|
|
t.Fatalf("block hash mismatch, got %v expected %v",
|
|
txDetail.BlockHash, burnBlock[0])
|
|
}
|
|
}
|
|
if !burnTxFound {
|
|
t.Fatal("tx burning btc not found")
|
|
}
|
|
|
|
// Generate a block which has no wallet transactions in it.
|
|
chainTip += numBlocks
|
|
_, err = miner.Client.Generate(uint32(numBlocks))
|
|
require.NoError(t, err, "unable to mine block")
|
|
|
|
err = waitForWalletSync(miner, alice)
|
|
require.NoError(t, err, "Couldn't sync Alice's wallet")
|
|
|
|
// Query for transactions only in the latest block. We do not expect
|
|
// any transactions to be returned.
|
|
txDetails, err = alice.ListTransactionDetails(chainTip, chainTip, "")
|
|
require.NoError(t, err, "unexpected error")
|
|
if len(txDetails) != 0 {
|
|
t.Fatalf("expected 0 transactions, got: %v", len(txDetails))
|
|
}
|
|
}
|
|
|
|
func testTransactionSubscriptions(miner *rpctest.Harness,
|
|
alice, _ *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
// First, check to see if this wallet meets the TransactionNotifier
|
|
// interface, if not then we'll skip this test for this particular
|
|
// implementation of the WalletController.
|
|
txClient, err := alice.SubscribeTransactions()
|
|
if err != nil {
|
|
t.Skipf("unable to generate tx subscription: %v", err)
|
|
}
|
|
defer txClient.Cancel()
|
|
|
|
const (
|
|
outputAmt = btcutil.SatoshiPerBitcoin
|
|
numTxns = 3
|
|
)
|
|
errCh1 := make(chan error, 1)
|
|
switch alice.BackEnd() {
|
|
case "neutrino":
|
|
// Neutrino doesn't listen for unconfirmed transactions.
|
|
default:
|
|
go func() {
|
|
for i := 0; i < numTxns; i++ {
|
|
txDetail := <-txClient.UnconfirmedTransactions()
|
|
if txDetail.NumConfirmations != 0 {
|
|
errCh1 <- fmt.Errorf("incorrect number of confs, "+
|
|
"expected %v got %v", 0,
|
|
txDetail.NumConfirmations)
|
|
return
|
|
}
|
|
if txDetail.Value != outputAmt {
|
|
errCh1 <- fmt.Errorf("incorrect output amt, "+
|
|
"expected %v got %v", outputAmt,
|
|
txDetail.Value)
|
|
return
|
|
}
|
|
if txDetail.BlockHash != nil {
|
|
errCh1 <- fmt.Errorf("block hash should be nil, "+
|
|
"is instead %v",
|
|
txDetail.BlockHash)
|
|
return
|
|
}
|
|
}
|
|
errCh1 <- nil
|
|
}()
|
|
}
|
|
|
|
// Next, fetch a fresh address from the wallet, create 3 new outputs
|
|
// with the pkScript.
|
|
for i := 0; i < numTxns; i++ {
|
|
addr, err := alice.NewAddress(
|
|
lnwallet.WitnessPubKey, false,
|
|
lnwallet.DefaultAccountName,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to create new address: %v", err)
|
|
}
|
|
script, err := txscript.PayToAddrScript(addr)
|
|
if err != nil {
|
|
t.Fatalf("unable to create output script: %v", err)
|
|
}
|
|
|
|
output := &wire.TxOut{
|
|
Value: outputAmt,
|
|
PkScript: script,
|
|
}
|
|
txid, err := miner.SendOutputs([]*wire.TxOut{output}, 2500)
|
|
if err != nil {
|
|
t.Fatalf("unable to send coinbase: %v", err)
|
|
}
|
|
err = waitForMempoolTx(miner, txid)
|
|
if err != nil {
|
|
t.Fatalf("tx not relayed to miner: %v", err)
|
|
}
|
|
}
|
|
|
|
switch alice.BackEnd() {
|
|
case "neutrino":
|
|
// Neutrino doesn't listen for on unconfirmed transactions.
|
|
default:
|
|
// We should receive a notification for all three transactions
|
|
// generated above.
|
|
select {
|
|
case <-time.After(time.Second * 10):
|
|
t.Fatalf("transactions not received after 10 seconds")
|
|
case err := <-errCh1:
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
}
|
|
|
|
errCh2 := make(chan error, 1)
|
|
go func() {
|
|
for i := 0; i < numTxns; i++ {
|
|
txDetail := <-txClient.ConfirmedTransactions()
|
|
if txDetail.NumConfirmations != 1 {
|
|
errCh2 <- fmt.Errorf("incorrect number of confs for %s, expected %v got %v",
|
|
txDetail.Hash, 1, txDetail.NumConfirmations)
|
|
return
|
|
}
|
|
if txDetail.Value != outputAmt {
|
|
errCh2 <- fmt.Errorf("incorrect output amt, expected %v got %v in txid %s",
|
|
outputAmt, txDetail.Value, txDetail.Hash)
|
|
return
|
|
}
|
|
}
|
|
errCh2 <- nil
|
|
}()
|
|
|
|
// Next mine a single block, all the transactions generated above
|
|
// should be included.
|
|
if _, err := miner.Client.Generate(1); err != nil {
|
|
t.Fatalf("unable to generate block: %v", err)
|
|
}
|
|
|
|
// We should receive a notification for all three transactions
|
|
// since they should be mined in the next block.
|
|
select {
|
|
case <-time.After(time.Second * 5):
|
|
t.Fatalf("transactions not received after 5 seconds")
|
|
case err := <-errCh2:
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
// We'll also ensure that the client is able to send our new
|
|
// notifications when we _create_ transactions ourselves that spend our
|
|
// own outputs.
|
|
addr, err := alice.NewAddress(
|
|
lnwallet.WitnessPubKey, false,
|
|
lnwallet.DefaultAccountName,
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
outputScript, err := txscript.PayToAddrScript(addr)
|
|
require.NoError(t, err)
|
|
|
|
burnOutput := wire.NewTxOut(outputAmt, outputScript)
|
|
tx, err := alice.SendOutputs(
|
|
nil, []*wire.TxOut{burnOutput}, 2500, 1, labels.External,
|
|
alice.Cfg.CoinSelectionStrategy,
|
|
)
|
|
require.NoError(t, err, "unable to create tx")
|
|
txid := tx.TxHash()
|
|
err = waitForMempoolTx(miner, &txid)
|
|
require.NoError(t, err, "tx not relayed to miner")
|
|
|
|
// Before we mine the next block, we'll ensure that the above
|
|
// transaction shows up in the set of unconfirmed transactions returned
|
|
// by ListTransactionDetails.
|
|
err = waitForWalletSync(miner, alice)
|
|
require.NoError(t, err, "Couldn't sync Alice's wallet")
|
|
|
|
// As we just sent the transaction and it was landed in the mempool, we
|
|
// should get a notification for a new unconfirmed transactions
|
|
select {
|
|
case <-time.After(time.Second * 10):
|
|
t.Fatalf("transactions not received after 10 seconds")
|
|
case unConfTx := <-txClient.UnconfirmedTransactions():
|
|
if unConfTx.Hash != txid {
|
|
t.Fatalf("wrong txn notified: expected %v got %v",
|
|
txid, unConfTx.Hash)
|
|
}
|
|
}
|
|
}
|
|
|
|
// scriptFromKey creates a P2WKH script from the given pubkey.
|
|
func scriptFromKey(pubkey *btcec.PublicKey) ([]byte, error) {
|
|
pubkeyHash := btcutil.Hash160(pubkey.SerializeCompressed())
|
|
keyAddr, err := btcutil.NewAddressWitnessPubKeyHash(
|
|
pubkeyHash, &chaincfg.RegressionNetParams,
|
|
)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to create addr: %w", err)
|
|
}
|
|
keyScript, err := txscript.PayToAddrScript(keyAddr)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to generate script: %w", err)
|
|
}
|
|
|
|
return keyScript, nil
|
|
}
|
|
|
|
// mineAndAssert mines a block and ensures the passed TX is part of that block.
|
|
func mineAndAssert(r *rpctest.Harness, tx *wire.MsgTx) error {
|
|
txid := tx.TxHash()
|
|
err := waitForMempoolTx(r, &txid)
|
|
if err != nil {
|
|
return fmt.Errorf("tx not relayed to miner: %w", err)
|
|
}
|
|
|
|
blockHashes, err := r.Client.Generate(1)
|
|
if err != nil {
|
|
return fmt.Errorf("unable to generate block: %w", err)
|
|
}
|
|
|
|
block, err := r.Client.GetBlock(blockHashes[0])
|
|
if err != nil {
|
|
return fmt.Errorf("unable to find block: %w", err)
|
|
}
|
|
|
|
if len(block.Transactions) != 2 {
|
|
return fmt.Errorf("expected 2 txs in block, got %d",
|
|
len(block.Transactions))
|
|
}
|
|
|
|
blockTx := block.Transactions[1]
|
|
if blockTx.TxHash() != tx.TxHash() {
|
|
return fmt.Errorf("incorrect transaction was mined")
|
|
}
|
|
|
|
// Sleep for a second before returning, to make sure the block has
|
|
// propagated.
|
|
time.Sleep(1 * time.Second)
|
|
return nil
|
|
}
|
|
|
|
// txFromOutput takes a tx paying to fromPubKey, and creates a new tx that
|
|
// spends the output from this tx, to an address derived from payToPubKey.
|
|
func txFromOutput(tx *wire.MsgTx, signer input.Signer, fromPubKey,
|
|
payToPubKey *btcec.PublicKey, txFee btcutil.Amount,
|
|
rbf bool) (*wire.MsgTx, error) {
|
|
|
|
// Generate the script we want to spend from.
|
|
keyScript, err := scriptFromKey(fromPubKey)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to generate script: %w", err)
|
|
}
|
|
|
|
// We assume the output was paid to the keyScript made earlier.
|
|
var outputIndex uint32
|
|
if len(tx.TxOut) == 1 || bytes.Equal(tx.TxOut[0].PkScript, keyScript) {
|
|
outputIndex = 0
|
|
} else {
|
|
outputIndex = 1
|
|
}
|
|
outputValue := tx.TxOut[outputIndex].Value
|
|
|
|
// With the index located, we can create a transaction spending the
|
|
// referenced output.
|
|
tx1 := wire.NewMsgTx(2)
|
|
|
|
// If we want to create a tx that signals replacement, set its
|
|
// sequence number to the max one that signals replacement.
|
|
// Otherwise we just use the standard max sequence.
|
|
sequence := wire.MaxTxInSequenceNum
|
|
if rbf {
|
|
sequence = mempool.MaxRBFSequence
|
|
}
|
|
|
|
tx1.AddTxIn(&wire.TxIn{
|
|
PreviousOutPoint: wire.OutPoint{
|
|
Hash: tx.TxHash(),
|
|
Index: outputIndex,
|
|
},
|
|
Sequence: sequence,
|
|
})
|
|
|
|
// Create a script to pay to.
|
|
payToScript, err := scriptFromKey(payToPubKey)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to generate script: %w", err)
|
|
}
|
|
tx1.AddTxOut(&wire.TxOut{
|
|
Value: outputValue - int64(txFee),
|
|
PkScript: payToScript,
|
|
})
|
|
|
|
// Now we can populate the sign descriptor which we'll use to generate
|
|
// the signature.
|
|
signDesc := &input.SignDescriptor{
|
|
KeyDesc: keychain.KeyDescriptor{
|
|
PubKey: fromPubKey,
|
|
},
|
|
WitnessScript: keyScript,
|
|
Output: tx.TxOut[outputIndex],
|
|
HashType: txscript.SigHashAll,
|
|
SigHashes: input.NewTxSigHashesV0Only(tx1),
|
|
InputIndex: 0, // Has only one input.
|
|
}
|
|
|
|
// With the descriptor created, we use it to generate a signature, then
|
|
// manually create a valid witness stack we'll use for signing.
|
|
spendSig, err := signer.SignOutputRaw(tx1, signDesc)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to generate signature: %w", err)
|
|
}
|
|
witness := make([][]byte, 2)
|
|
witness[0] = append(spendSig.Serialize(), byte(txscript.SigHashAll))
|
|
witness[1] = fromPubKey.SerializeCompressed()
|
|
tx1.TxIn[0].Witness = witness
|
|
|
|
// Finally, attempt to validate the completed transaction. This should
|
|
// succeed if the wallet was able to properly generate the proper
|
|
// private key.
|
|
vm, err := txscript.NewEngine(
|
|
keyScript, tx1, 0, txscript.StandardVerifyFlags, nil,
|
|
nil, outputValue, txscript.NewCannedPrevOutputFetcher(
|
|
keyScript, outputValue,
|
|
),
|
|
)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to create engine: %w", err)
|
|
}
|
|
if err := vm.Execute(); err != nil {
|
|
return nil, fmt.Errorf("spend is invalid: %w", err)
|
|
}
|
|
|
|
return tx1, nil
|
|
}
|
|
|
|
// newTx sends coins from Alice's wallet, mines this transaction, and creates a
|
|
// new, unconfirmed tx that spends this output to pubKey.
|
|
func newTx(t *testing.T, r *rpctest.Harness, pubKey *btcec.PublicKey,
|
|
alice *lnwallet.LightningWallet, rbf bool) *wire.MsgTx {
|
|
|
|
t.Helper()
|
|
|
|
keyScript, err := scriptFromKey(pubKey)
|
|
require.NoError(t, err, "unable to generate script")
|
|
|
|
// Instruct the wallet to fund the output with a newly created
|
|
// transaction.
|
|
newOutput := &wire.TxOut{
|
|
Value: btcutil.SatoshiPerBitcoin,
|
|
PkScript: keyScript,
|
|
}
|
|
tx, err := alice.SendOutputs(
|
|
nil, []*wire.TxOut{newOutput}, 2500, 1, labels.External,
|
|
alice.Cfg.CoinSelectionStrategy,
|
|
)
|
|
require.NoError(t, err, "unable to create output")
|
|
|
|
// Query for the transaction generated above so we can located the
|
|
// index of our output.
|
|
if err := mineAndAssert(r, tx); err != nil {
|
|
t.Fatalf("unable to mine tx: %v", err)
|
|
}
|
|
|
|
// Create a new unconfirmed tx that spends this output.
|
|
txFee := btcutil.Amount(0.001 * btcutil.SatoshiPerBitcoin)
|
|
tx1, err := txFromOutput(
|
|
tx, alice.Cfg.Signer, pubKey, pubKey, txFee, rbf,
|
|
)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
return tx1
|
|
}
|
|
|
|
// testGetTransactionDetails checks that GetTransactionDetails returns the
|
|
// correct amount after a transaction has been sent from alice to bob.
|
|
func testGetTransactionDetails(r *rpctest.Harness,
|
|
alice, bob *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
const txFee = int64(14500)
|
|
|
|
bobPkScript := newPkScript(t, bob, lnwallet.WitnessPubKey)
|
|
txFeeRate := chainfee.SatPerKWeight(2500)
|
|
amountSats := btcutil.Amount(btcutil.SatoshiPerBitcoin - txFee)
|
|
output := &wire.TxOut{
|
|
Value: int64(amountSats),
|
|
PkScript: bobPkScript,
|
|
}
|
|
tx := sendCoins(t, r, alice, bob, output, txFeeRate, true, 1)
|
|
txHash := tx.TxHash()
|
|
assertTxInWallet(t, alice, txHash, true)
|
|
assertTxInWallet(t, bob, txHash, true)
|
|
|
|
txDetails, err := bob.GetTransactionDetails(&txHash)
|
|
require.NoError(t, err, "unable to receive transaction details")
|
|
require.Equal(t, txDetails.Value, amountSats, "tx value")
|
|
}
|
|
|
|
// testPublishTransaction checks that PublishTransaction returns the expected
|
|
// error types in case the transaction being published conflicts with the
|
|
// current mempool or chain.
|
|
func testPublishTransaction(r *rpctest.Harness,
|
|
alice, _ *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
// Generate a pubkey, and pay-to-addr script.
|
|
keyDesc, err := alice.DeriveNextKey(keychain.KeyFamilyMultiSig)
|
|
require.NoError(t, err, "unable to obtain public key")
|
|
|
|
t.Run("no_error_on_duplicate_tx", func(t *testing.T) {
|
|
// We will first check that publishing a transaction already in
|
|
// the mempool does NOT return an error. Create the tx.
|
|
tx1 := newTx(t, r, keyDesc.PubKey, alice, false)
|
|
|
|
// Publish the transaction.
|
|
err = alice.PublishTransaction(tx1, labels.External)
|
|
require.NoError(t, err)
|
|
|
|
txid1 := tx1.TxHash()
|
|
err = waitForMempoolTx(r, &txid1)
|
|
require.NoError(t, err, "tx not relayed to miner")
|
|
|
|
// Publish the exact same transaction again. This should not
|
|
// return an error, even though the transaction is already in
|
|
// the mempool.
|
|
err = alice.PublishTransaction(tx1, labels.External)
|
|
require.NoError(t, err)
|
|
|
|
// Mine the transaction.
|
|
_, err := r.Client.Generate(1)
|
|
require.NoError(t, err)
|
|
})
|
|
|
|
t.Run("no_error_on_minged_tx", func(t *testing.T) {
|
|
// We'll now test that we don't get an error if we try to
|
|
// publish a transaction that is already mined.
|
|
//
|
|
// Create a new transaction. We must do this to properly test
|
|
// the reject messages from our peers. They might only send us
|
|
// a reject message for a given tx once, so we create a new to
|
|
// make sure it is not just immediately rejected.
|
|
tx2 := newTx(t, r, keyDesc.PubKey, alice, false)
|
|
|
|
// Publish this tx.
|
|
err = alice.PublishTransaction(tx2, labels.External)
|
|
require.NoError(t, err)
|
|
|
|
// Mine the transaction.
|
|
err := mineAndAssert(r, tx2)
|
|
require.NoError(t, err)
|
|
|
|
// Publish the transaction again. It is already mined, and we
|
|
// don't expect this to return an error.
|
|
err = alice.PublishTransaction(tx2, labels.External)
|
|
require.NoError(t, err)
|
|
})
|
|
|
|
// We'll do the next mempool check on both RBF and non-RBF
|
|
// enabled transactions.
|
|
var (
|
|
txFee = btcutil.Amount(
|
|
0.005 * btcutil.SatoshiPerBitcoin,
|
|
)
|
|
tx3, tx3Spend *wire.MsgTx
|
|
)
|
|
t.Run("rbf_tests", func(t *testing.T) {
|
|
for _, rbf := range []bool{false, true} {
|
|
// Now we'll try to double spend an output with a
|
|
// different transaction. Create a new tx and publish
|
|
// it. This is the output we'll try to double spend.
|
|
tx3 = newTx(t, r, keyDesc.PubKey, alice, false)
|
|
err := alice.PublishTransaction(tx3, labels.External)
|
|
require.NoError(t, err)
|
|
|
|
// Mine the transaction.
|
|
err = mineAndAssert(r, tx3)
|
|
require.NoError(t, err)
|
|
|
|
// Now we create a transaction that spends the output
|
|
// from the tx just mined.
|
|
tx4, err := txFromOutput(
|
|
tx3, alice.Cfg.Signer, keyDesc.PubKey,
|
|
keyDesc.PubKey, txFee, rbf,
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
// This should be accepted into the mempool.
|
|
err = alice.PublishTransaction(tx4, labels.External)
|
|
require.NoError(t, err)
|
|
|
|
// Keep track of the last successfully published tx to
|
|
// spend tx3.
|
|
tx3Spend = tx4
|
|
|
|
txid4 := tx4.TxHash()
|
|
err = waitForMempoolTx(r, &txid4)
|
|
require.NoError(t, err, "tx not relayed to miner")
|
|
|
|
// Create a new key we'll pay to, to ensure we create a
|
|
// unique transaction.
|
|
keyDesc2, err := alice.DeriveNextKey(
|
|
keychain.KeyFamilyMultiSig,
|
|
)
|
|
require.NoError(t, err, "unable to obtain public key")
|
|
|
|
// Create a new transaction that spends the output from
|
|
// tx3, and that pays to a different address.
|
|
tx5, err := txFromOutput(
|
|
tx3, alice.Cfg.Signer, keyDesc.PubKey,
|
|
keyDesc2.PubKey, txFee, rbf,
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
err = alice.PublishTransaction(tx5, labels.External)
|
|
|
|
// If RBF is not enabled, we expect this to be rejected
|
|
// because it is a double spend.
|
|
expectedErr := lnwallet.ErrDoubleSpend
|
|
|
|
// If RBF is enabled, we expect it to be rejected
|
|
// because it doesn't pay enough fees.
|
|
if rbf {
|
|
expectedErr = chain.ErrInsufficientFee
|
|
}
|
|
|
|
// Assert the expected error.
|
|
require.ErrorIsf(t, err, expectedErr, "has rbf=%v", rbf)
|
|
|
|
// Create another transaction that spends the same
|
|
// output, but has a higher fee. We expect also this tx
|
|
// to be rejected for non-RBF enabled transactions,
|
|
// while it should succeed otherwise.
|
|
pubKey3, err := alice.DeriveNextKey(
|
|
keychain.KeyFamilyMultiSig,
|
|
)
|
|
require.NoError(t, err, "unable to obtain public key")
|
|
|
|
tx6, err := txFromOutput(
|
|
tx3, alice.Cfg.Signer, keyDesc.PubKey,
|
|
pubKey3.PubKey, 2*txFee, rbf,
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
// Expect rejection in non-RBF case.
|
|
expErr := lnwallet.ErrDoubleSpend
|
|
if rbf {
|
|
// Expect success in rbf case.
|
|
expErr = nil
|
|
tx3Spend = tx6
|
|
}
|
|
err = alice.PublishTransaction(tx6, labels.External)
|
|
require.ErrorIs(t, err, expErr)
|
|
|
|
// Mine the tx spending tx3.
|
|
err = mineAndAssert(r, tx3Spend)
|
|
require.NoError(t, err)
|
|
}
|
|
})
|
|
|
|
t.Run("tx_double_spend", func(t *testing.T) {
|
|
// At last we try to spend an output already spent by a
|
|
// confirmed transaction.
|
|
//
|
|
// TODO(halseth): we currently skip this test for neutrino, as
|
|
// the backing btcd node will consider the tx being an orphan,
|
|
// and will accept it. Should look into if this is the behavior
|
|
// also for bitcoind, and update test accordingly.
|
|
if alice.BackEnd() != "neutrino" {
|
|
// Create another tx spending tx3.
|
|
pubKey4, err := alice.DeriveNextKey(
|
|
keychain.KeyFamilyMultiSig,
|
|
)
|
|
require.NoError(t, err, "unable to obtain public key")
|
|
|
|
tx7, err := txFromOutput(
|
|
tx3, alice.Cfg.Signer, keyDesc.PubKey,
|
|
pubKey4.PubKey, txFee, false,
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
// Expect rejection.
|
|
err = alice.PublishTransaction(tx7, labels.External)
|
|
require.ErrorIs(t, err, lnwallet.ErrDoubleSpend)
|
|
}
|
|
})
|
|
|
|
t.Run("test_tx_size_limit", func(t *testing.T) {
|
|
// In this test, we'll try to create a massive transaction that
|
|
// can't be mined as dictacted by widely deployed transaction
|
|
// policy.
|
|
//
|
|
// To do this, we'll take out of the prior transactions, and
|
|
// add a bunch of outputs, putting it over the max weight
|
|
// limit.
|
|
testTx := tx3.Copy()
|
|
for i := 0; i < blockchain.MaxOutputsPerBlock; i++ {
|
|
testTx.AddTxOut(&wire.TxOut{
|
|
Value: tx3.TxOut[0].Value,
|
|
PkScript: tx3.TxOut[0].PkScript,
|
|
})
|
|
}
|
|
|
|
// Now broadcast the transaction, we should get an error that
|
|
// the weight is too large.
|
|
err := alice.PublishTransaction(testTx, labels.External)
|
|
require.ErrorIs(t, err, chain.ErrOversizeTx)
|
|
})
|
|
}
|
|
|
|
func testSignOutputUsingTweaks(r *rpctest.Harness,
|
|
alice, _ *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
// We'd like to test the ability of the wallet's Signer implementation
|
|
// to be able to sign with a private key derived from tweaking the
|
|
// specific public key. This scenario exercises the case when the
|
|
// wallet needs to sign for a sweep of a revoked output, or just claim
|
|
// any output that pays to a tweaked key.
|
|
|
|
// First, generate a new public key under the control of the wallet,
|
|
// then generate a revocation key using it.
|
|
pubKey, err := alice.DeriveNextKey(
|
|
keychain.KeyFamilyMultiSig,
|
|
)
|
|
require.NoError(t, err, "unable to obtain public key")
|
|
|
|
// As we'd like to test both single tweak, and double tweak spends,
|
|
// we'll generate a commitment pre-image, then derive a revocation key
|
|
// and single tweak from that.
|
|
commitPreimage := bytes.Repeat([]byte{2}, 32)
|
|
commitSecret, commitPoint := btcec.PrivKeyFromBytes(commitPreimage)
|
|
|
|
revocationKey := input.DeriveRevocationPubkey(pubKey.PubKey, commitPoint)
|
|
commitTweak := input.SingleTweakBytes(commitPoint, pubKey.PubKey)
|
|
|
|
tweakedPub := input.TweakPubKey(pubKey.PubKey, commitPoint)
|
|
|
|
// As we'd like to test both single and double tweaks, we'll repeat
|
|
// the same set up twice. The first will use a regular single tweak,
|
|
// and the second will use a double tweak.
|
|
baseKey := pubKey
|
|
for i := 0; i < 2; i++ {
|
|
var tweakedKey *btcec.PublicKey
|
|
if i == 0 {
|
|
tweakedKey = tweakedPub
|
|
} else {
|
|
tweakedKey = revocationKey
|
|
}
|
|
|
|
// Using the given key for the current iteration, we'll
|
|
// generate a regular p2wkh from that.
|
|
pubkeyHash := btcutil.Hash160(tweakedKey.SerializeCompressed())
|
|
keyAddr, err := btcutil.NewAddressWitnessPubKeyHash(pubkeyHash,
|
|
&chaincfg.RegressionNetParams)
|
|
if err != nil {
|
|
t.Fatalf("unable to create addr: %v", err)
|
|
}
|
|
keyScript, err := txscript.PayToAddrScript(keyAddr)
|
|
if err != nil {
|
|
t.Fatalf("unable to generate script: %v", err)
|
|
}
|
|
|
|
// With the script fully assembled, instruct the wallet to fund
|
|
// the output with a newly created transaction.
|
|
newOutput := &wire.TxOut{
|
|
Value: btcutil.SatoshiPerBitcoin,
|
|
PkScript: keyScript,
|
|
}
|
|
tx, err := alice.SendOutputs(
|
|
nil, []*wire.TxOut{newOutput}, 2500, 1, labels.External,
|
|
alice.Cfg.CoinSelectionStrategy,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to create output: %v", err)
|
|
}
|
|
txid := tx.TxHash()
|
|
// Query for the transaction generated above so we can located
|
|
// the index of our output.
|
|
err = waitForMempoolTx(r, &txid)
|
|
if err != nil {
|
|
t.Fatalf("tx not relayed to miner: %v", err)
|
|
}
|
|
var outputIndex uint32
|
|
if bytes.Equal(tx.TxOut[0].PkScript, keyScript) {
|
|
outputIndex = 0
|
|
} else {
|
|
outputIndex = 1
|
|
}
|
|
|
|
// With the index located, we can create a transaction spending
|
|
// the referenced output.
|
|
sweepTx := wire.NewMsgTx(2)
|
|
sweepTx.AddTxIn(&wire.TxIn{
|
|
PreviousOutPoint: wire.OutPoint{
|
|
Hash: txid,
|
|
Index: outputIndex,
|
|
},
|
|
})
|
|
sweepTx.AddTxOut(&wire.TxOut{
|
|
Value: 1000,
|
|
PkScript: keyScript,
|
|
})
|
|
|
|
// Now we can populate the sign descriptor which we'll use to
|
|
// generate the signature. Within the descriptor we set the
|
|
// private tweak value as the key in the script is derived
|
|
// based on this tweak value and the key we originally
|
|
// generated above.
|
|
signDesc := &input.SignDescriptor{
|
|
KeyDesc: keychain.KeyDescriptor{
|
|
PubKey: baseKey.PubKey,
|
|
},
|
|
WitnessScript: keyScript,
|
|
Output: newOutput,
|
|
HashType: txscript.SigHashAll,
|
|
SigHashes: input.NewTxSigHashesV0Only(sweepTx),
|
|
InputIndex: 0,
|
|
}
|
|
|
|
// If this is the first, loop, we'll use the generated single
|
|
// tweak, otherwise, we'll use the double tweak.
|
|
if i == 0 {
|
|
signDesc.SingleTweak = commitTweak
|
|
} else {
|
|
signDesc.DoubleTweak = commitSecret
|
|
}
|
|
|
|
// With the descriptor created, we use it to generate a
|
|
// signature, then manually create a valid witness stack we'll
|
|
// use for signing.
|
|
spendSig, err := alice.Cfg.Signer.SignOutputRaw(sweepTx, signDesc)
|
|
if err != nil {
|
|
t.Fatalf("unable to generate signature: %v", err)
|
|
}
|
|
witness := make([][]byte, 2)
|
|
witness[0] = append(spendSig.Serialize(), byte(txscript.SigHashAll))
|
|
witness[1] = tweakedKey.SerializeCompressed()
|
|
sweepTx.TxIn[0].Witness = witness
|
|
|
|
// Finally, attempt to validate the completed transaction. This
|
|
// should succeed if the wallet was able to properly generate
|
|
// the proper private key.
|
|
vm, err := txscript.NewEngine(
|
|
keyScript, sweepTx, 0, txscript.StandardVerifyFlags,
|
|
nil, nil, int64(btcutil.SatoshiPerBitcoin),
|
|
txscript.NewCannedPrevOutputFetcher(
|
|
keyScript, int64(btcutil.SatoshiPerBitcoin),
|
|
),
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to create engine: %v", err)
|
|
}
|
|
if err := vm.Execute(); err != nil {
|
|
t.Fatalf("spend #%v is invalid: %v", i, err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func testReorgWalletBalance(r *rpctest.Harness, w *lnwallet.LightningWallet,
|
|
_ *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
// We first mine a few blocks to ensure any transactions still in the
|
|
// mempool confirm, and then get the original balance, before a
|
|
// reorganization that doesn't invalidate any existing transactions or
|
|
// create any new non-coinbase transactions. We'll then check if it's
|
|
// the same after the empty reorg.
|
|
_, err := r.Client.Generate(5)
|
|
require.NoError(t, err, "unable to generate blocks on passed node")
|
|
|
|
// Give wallet time to catch up.
|
|
err = waitForWalletSync(r, w)
|
|
require.NoError(t, err, "unable to sync wallet")
|
|
|
|
// Send some money from the miner to the wallet
|
|
err = loadTestCredits(r, w, 20, 4)
|
|
require.NoError(t, err, "unable to send money to lnwallet")
|
|
|
|
// Send some money from the wallet back to the miner.
|
|
// Grab a fresh address from the miner to house this output.
|
|
minerAddr, err := r.NewAddress()
|
|
require.NoError(t, err, "unable to generate address for miner")
|
|
script, err := txscript.PayToAddrScript(minerAddr)
|
|
require.NoError(t, err, "unable to create pay to addr script")
|
|
output := &wire.TxOut{
|
|
Value: 1e8,
|
|
PkScript: script,
|
|
}
|
|
tx, err := w.SendOutputs(
|
|
nil, []*wire.TxOut{output}, 2500, 1, labels.External,
|
|
w.Cfg.CoinSelectionStrategy,
|
|
)
|
|
require.NoError(t, err, "unable to send outputs")
|
|
txid := tx.TxHash()
|
|
err = waitForMempoolTx(r, &txid)
|
|
require.NoError(t, err, "tx not relayed to miner")
|
|
_, err = r.Client.Generate(50)
|
|
require.NoError(t, err, "unable to generate blocks on passed node")
|
|
|
|
// Give wallet time to catch up.
|
|
err = waitForWalletSync(r, w)
|
|
require.NoError(t, err, "unable to sync wallet")
|
|
|
|
// Get the original balance.
|
|
origBalance, err := w.ConfirmedBalance(1, lnwallet.DefaultAccountName)
|
|
require.NoError(t, err, "unable to query for balance")
|
|
|
|
// Now we cause a reorganization as follows.
|
|
// Step 1: create a new miner and start it.
|
|
r2 := unittest.NewMiner(
|
|
t, r.ActiveNet, []string{"--txindex"}, false, 0,
|
|
)
|
|
newBalance, err := w.ConfirmedBalance(1, lnwallet.DefaultAccountName)
|
|
require.NoError(t, err, "unable to query for balance")
|
|
if origBalance != newBalance {
|
|
t.Fatalf("wallet balance incorrect, should have %v, "+
|
|
"instead have %v", origBalance, newBalance)
|
|
}
|
|
|
|
// Step 2: connect the miner to the passed miner and wait for
|
|
// synchronization.
|
|
err = r2.Client.AddNode(r.P2PAddress(), rpcclient.ANAdd)
|
|
require.NoError(t, err, "unable to connect mining nodes together")
|
|
err = rpctest.JoinNodes([]*rpctest.Harness{r2, r}, rpctest.Blocks)
|
|
require.NoError(t, err, "unable to synchronize mining nodes")
|
|
|
|
// Step 3: Do a set of reorgs by disconnecting the two miners, mining
|
|
// one block on the passed miner and two on the created miner,
|
|
// connecting them, and waiting for them to sync.
|
|
for i := 0; i < 5; i++ {
|
|
// Wait for disconnection
|
|
timeout := time.After(30 * time.Second)
|
|
stillConnected := true
|
|
var peers []btcjson.GetPeerInfoResult
|
|
for stillConnected {
|
|
// Allow for timeout
|
|
time.Sleep(100 * time.Millisecond)
|
|
select {
|
|
case <-timeout:
|
|
t.Fatalf("timeout waiting for miner disconnect")
|
|
default:
|
|
}
|
|
err = r2.Client.AddNode(r.P2PAddress(), rpcclient.ANRemove)
|
|
if err != nil {
|
|
t.Fatalf("unable to disconnect mining nodes: %v",
|
|
err)
|
|
}
|
|
peers, err = r2.Client.GetPeerInfo()
|
|
if err != nil {
|
|
t.Fatalf("unable to get peer info: %v", err)
|
|
}
|
|
stillConnected = false
|
|
for _, peer := range peers {
|
|
if peer.Addr == r.P2PAddress() {
|
|
stillConnected = true
|
|
break
|
|
}
|
|
}
|
|
}
|
|
_, err = r.Client.Generate(2)
|
|
if err != nil {
|
|
t.Fatalf("unable to generate blocks on passed node: %v",
|
|
err)
|
|
}
|
|
_, err = r2.Client.Generate(3)
|
|
if err != nil {
|
|
t.Fatalf("unable to generate blocks on created node: %v",
|
|
err)
|
|
}
|
|
|
|
// Step 5: Reconnect the miners and wait for them to synchronize.
|
|
err = r2.Client.AddNode(r.P2PAddress(), rpcclient.ANAdd)
|
|
if err != nil {
|
|
switch err := err.(type) {
|
|
case *btcjson.RPCError:
|
|
if err.Code != -8 {
|
|
t.Fatalf("unable to connect mining "+
|
|
"nodes together: %v", err)
|
|
}
|
|
default:
|
|
t.Fatalf("unable to connect mining nodes "+
|
|
"together: %v", err)
|
|
}
|
|
}
|
|
err = rpctest.JoinNodes([]*rpctest.Harness{r2, r},
|
|
rpctest.Blocks)
|
|
if err != nil {
|
|
t.Fatalf("unable to synchronize mining nodes: %v", err)
|
|
}
|
|
|
|
// Give wallet time to catch up.
|
|
err = waitForWalletSync(r, w)
|
|
if err != nil {
|
|
t.Fatalf("unable to sync wallet: %v", err)
|
|
}
|
|
}
|
|
|
|
// Now we check that the wallet balance stays the same.
|
|
newBalance, err = w.ConfirmedBalance(1, lnwallet.DefaultAccountName)
|
|
require.NoError(t, err, "unable to query for balance")
|
|
if origBalance != newBalance {
|
|
t.Fatalf("wallet balance incorrect, should have %v, "+
|
|
"instead have %v", origBalance, newBalance)
|
|
}
|
|
}
|
|
|
|
// testChangeOutputSpendConfirmation ensures that when we attempt to spend a
|
|
// change output created by the wallet, the wallet receives its confirmation
|
|
// once included in the chain.
|
|
func testChangeOutputSpendConfirmation(r *rpctest.Harness,
|
|
alice, bob *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
// In order to test that we see the confirmation of a transaction that
|
|
// spends an output created by SendOutputs, we'll start by emptying
|
|
// Alice's wallet so that no other UTXOs can be picked. To do so, we'll
|
|
// generate an address for Bob, who will receive all the coins.
|
|
// Assuming a balance of 80 BTC and a transaction fee of 2500 sat/kw,
|
|
// we'll craft the following transaction so that Alice doesn't have any
|
|
// UTXOs left.
|
|
aliceBalance, err := alice.ConfirmedBalance(0, lnwallet.DefaultAccountName)
|
|
require.NoError(t, err, "unable to retrieve alice's balance")
|
|
bobPkScript := newPkScript(t, bob, lnwallet.WitnessPubKey)
|
|
|
|
// We'll use a transaction fee of 14380 satoshis, which will allow us to
|
|
// sweep all of Alice's balance in one transaction containing 1 input
|
|
// and 1 output.
|
|
//
|
|
// TODO(wilmer): replace this once SendOutputs easily supports sending
|
|
// all funds in one transaction.
|
|
txFeeRate := chainfee.SatPerKWeight(2500)
|
|
const txFee = int64(14500)
|
|
output := &wire.TxOut{
|
|
Value: int64(aliceBalance) - txFee,
|
|
PkScript: bobPkScript,
|
|
}
|
|
tx := sendCoins(t, r, alice, bob, output, txFeeRate, true, 1)
|
|
txHash := tx.TxHash()
|
|
assertTxInWallet(t, alice, txHash, true)
|
|
assertTxInWallet(t, bob, txHash, true)
|
|
|
|
// With the transaction sent and confirmed, Alice's balance should now
|
|
// be 0.
|
|
aliceBalance, err = alice.ConfirmedBalance(0, lnwallet.DefaultAccountName)
|
|
require.NoError(t, err, "unable to retrieve alice's balance")
|
|
if aliceBalance != 0 {
|
|
t.Fatalf("expected alice's balance to be 0 BTC, found %v",
|
|
aliceBalance)
|
|
}
|
|
|
|
// Now, we'll send an output back to Alice from Bob of 1 BTC.
|
|
alicePkScript := newPkScript(t, alice, lnwallet.WitnessPubKey)
|
|
output = &wire.TxOut{
|
|
Value: btcutil.SatoshiPerBitcoin,
|
|
PkScript: alicePkScript,
|
|
}
|
|
tx = sendCoins(t, r, bob, alice, output, txFeeRate, true, 1)
|
|
txHash = tx.TxHash()
|
|
assertTxInWallet(t, alice, txHash, true)
|
|
assertTxInWallet(t, bob, txHash, true)
|
|
|
|
// Alice now has an available output to spend, but it was not a change
|
|
// output, which is what the test expects. Therefore, we'll generate one
|
|
// by sending Bob back some coins.
|
|
output = &wire.TxOut{
|
|
Value: btcutil.SatoshiPerBitcent,
|
|
PkScript: bobPkScript,
|
|
}
|
|
tx = sendCoins(t, r, alice, bob, output, txFeeRate, true, 1)
|
|
txHash = tx.TxHash()
|
|
assertTxInWallet(t, alice, txHash, true)
|
|
assertTxInWallet(t, bob, txHash, true)
|
|
|
|
// Then, we'll spend the change output and ensure we see its
|
|
// confirmation come in.
|
|
tx = sendCoins(t, r, alice, bob, output, txFeeRate, true, 1)
|
|
txHash = tx.TxHash()
|
|
assertTxInWallet(t, alice, txHash, true)
|
|
assertTxInWallet(t, bob, txHash, true)
|
|
|
|
// Finally, we'll replenish Alice's wallet with some more coins to
|
|
// ensure she has enough for any following test cases.
|
|
if err := loadTestCredits(r, alice, 20, 4); err != nil {
|
|
t.Fatalf("unable to replenish alice's wallet: %v", err)
|
|
}
|
|
}
|
|
|
|
// testSpendUnconfirmed ensures that when can spend unconfirmed outputs.
|
|
func testSpendUnconfirmed(miner *rpctest.Harness,
|
|
alice, bob *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
bobPkScript := newPkScript(t, bob, lnwallet.WitnessPubKey)
|
|
alicePkScript := newPkScript(t, alice, lnwallet.WitnessPubKey)
|
|
txFeeRate := chainfee.SatPerKWeight(2500)
|
|
|
|
// First we will empty out bob's wallet, sending the entire balance
|
|
// to alice.
|
|
bobBalance, err := bob.ConfirmedBalance(0, lnwallet.DefaultAccountName)
|
|
require.NoError(t, err, "unable to retrieve bob's balance")
|
|
txFee := btcutil.Amount(28760)
|
|
output := &wire.TxOut{
|
|
Value: int64(bobBalance - txFee),
|
|
PkScript: alicePkScript,
|
|
}
|
|
tx := sendCoins(t, miner, bob, alice, output, txFeeRate, true, 1)
|
|
txHash := tx.TxHash()
|
|
assertTxInWallet(t, alice, txHash, true)
|
|
assertTxInWallet(t, bob, txHash, true)
|
|
|
|
// Verify that bob doesn't have enough balance to send coins.
|
|
output = &wire.TxOut{
|
|
Value: btcutil.SatoshiPerBitcoin * 0.5,
|
|
PkScript: alicePkScript,
|
|
}
|
|
_, err = bob.SendOutputs(
|
|
nil, []*wire.TxOut{output}, txFeeRate, 0, labels.External,
|
|
bob.Cfg.CoinSelectionStrategy,
|
|
)
|
|
if err == nil {
|
|
t.Fatalf("should have not been able to pay due to insufficient balance: %v", err)
|
|
}
|
|
|
|
// Next we will send a transaction to bob but leave it in an
|
|
// unconfirmed state.
|
|
output = &wire.TxOut{
|
|
Value: btcutil.SatoshiPerBitcoin,
|
|
PkScript: bobPkScript,
|
|
}
|
|
tx = sendCoins(t, miner, alice, bob, output, txFeeRate, false, 1)
|
|
txHash = tx.TxHash()
|
|
assertTxInWallet(t, alice, txHash, false)
|
|
assertTxInWallet(t, bob, txHash, false)
|
|
|
|
// Now, try to spend some of the unconfirmed funds from bob's wallet.
|
|
output = &wire.TxOut{
|
|
Value: btcutil.SatoshiPerBitcoin * 0.5,
|
|
PkScript: alicePkScript,
|
|
}
|
|
|
|
// First, verify that we don't have enough balance to send the coins
|
|
// using confirmed outputs only.
|
|
_, err = bob.SendOutputs(
|
|
nil, []*wire.TxOut{output}, txFeeRate, 1, labels.External,
|
|
bob.Cfg.CoinSelectionStrategy,
|
|
)
|
|
if err == nil {
|
|
t.Fatalf("should have not been able to pay due to insufficient balance: %v", err)
|
|
}
|
|
|
|
// Now try the send again using unconfirmed outputs.
|
|
tx = sendCoins(t, miner, bob, alice, output, txFeeRate, false, 0)
|
|
txHash = tx.TxHash()
|
|
assertTxInWallet(t, alice, txHash, false)
|
|
assertTxInWallet(t, bob, txHash, false)
|
|
|
|
// Mine the unconfirmed transactions.
|
|
err = waitForMempoolTx(miner, &txHash)
|
|
require.NoError(t, err, "tx not relayed to miner")
|
|
if _, err := miner.Client.Generate(1); err != nil {
|
|
t.Fatalf("unable to generate block: %v", err)
|
|
}
|
|
if err := waitForWalletSync(miner, alice); err != nil {
|
|
t.Fatalf("unable to sync alice: %v", err)
|
|
}
|
|
if err := waitForWalletSync(miner, bob); err != nil {
|
|
t.Fatalf("unable to sync bob: %v", err)
|
|
}
|
|
|
|
// Finally, send the remainder of bob's wallet balance back to him so
|
|
// that these money movements dont mess up later tests.
|
|
output = &wire.TxOut{
|
|
Value: int64(bobBalance) - (btcutil.SatoshiPerBitcoin * 0.4),
|
|
PkScript: bobPkScript,
|
|
}
|
|
tx = sendCoins(t, miner, alice, bob, output, txFeeRate, true, 1)
|
|
txHash = tx.TxHash()
|
|
assertTxInWallet(t, alice, txHash, true)
|
|
assertTxInWallet(t, bob, txHash, true)
|
|
}
|
|
|
|
// testLastUnusedAddr tests that the LastUnusedAddress returns the address if
|
|
// it isn't used, and also that once the address becomes used, then it's
|
|
// properly rotated.
|
|
func testLastUnusedAddr(miner *rpctest.Harness,
|
|
alice, bob *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
if _, err := miner.Client.Generate(1); err != nil {
|
|
t.Fatalf("unable to generate block: %v", err)
|
|
}
|
|
|
|
// We'll repeat this test for each address type to ensure they're all
|
|
// rotated properly.
|
|
addrTypes := []lnwallet.AddressType{
|
|
lnwallet.WitnessPubKey, lnwallet.NestedWitnessPubKey,
|
|
}
|
|
for _, addrType := range addrTypes {
|
|
addr1, err := alice.LastUnusedAddress(
|
|
addrType, lnwallet.DefaultAccountName,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to get addr: %v", err)
|
|
}
|
|
addr2, err := alice.LastUnusedAddress(
|
|
addrType, lnwallet.DefaultAccountName,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to get addr: %v", err)
|
|
}
|
|
|
|
// If we generate two addresses back to back, then we should
|
|
// get the same addr, as none of them have been used yet.
|
|
if addr1.String() != addr2.String() {
|
|
t.Fatalf("addresses changed w/o use: %v vs %v", addr1, addr2)
|
|
}
|
|
|
|
// Next, we'll have Bob pay to Alice's new address. This should
|
|
// trigger address rotation at the backend wallet.
|
|
addrScript, err := txscript.PayToAddrScript(addr1)
|
|
if err != nil {
|
|
t.Fatalf("unable to convert addr to script: %v", err)
|
|
}
|
|
feeRate := chainfee.SatPerKWeight(2500)
|
|
output := &wire.TxOut{
|
|
Value: 1000000,
|
|
PkScript: addrScript,
|
|
}
|
|
sendCoins(t, miner, bob, alice, output, feeRate, true, 1)
|
|
|
|
// If we make a new address, then it should be brand new, as
|
|
// the prior address has been used.
|
|
addr3, err := alice.LastUnusedAddress(
|
|
addrType, lnwallet.DefaultAccountName,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to get addr: %v", err)
|
|
}
|
|
if addr1.String() == addr3.String() {
|
|
t.Fatalf("address should have changed but didn't")
|
|
}
|
|
}
|
|
}
|
|
|
|
// testCreateSimpleTx checks that a call to CreateSimpleTx will return a
|
|
// transaction that is equal to the one that is being created by SendOutputs in
|
|
// a subsequent call.
|
|
// All test cases are doubled-up: one for testing unconfirmed inputs,
|
|
// one for testing only confirmed inputs.
|
|
func testCreateSimpleTx(r *rpctest.Harness, w *lnwallet.LightningWallet,
|
|
_ *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
// Send some money from the miner to the wallet
|
|
err := loadTestCredits(r, w, 20, 4)
|
|
require.NoError(t, err, "unable to send money to lnwallet")
|
|
|
|
// The test cases we will run through for all backends.
|
|
testCases := []struct {
|
|
outVals []int64
|
|
feeRate chainfee.SatPerKWeight
|
|
valid bool
|
|
unconfirmed bool
|
|
}{
|
|
{
|
|
outVals: []int64{},
|
|
feeRate: 2500,
|
|
valid: false, // No outputs.
|
|
unconfirmed: false,
|
|
},
|
|
{
|
|
outVals: []int64{},
|
|
feeRate: 2500,
|
|
valid: false, // No outputs.
|
|
unconfirmed: true,
|
|
},
|
|
|
|
{
|
|
outVals: []int64{200},
|
|
feeRate: 2500,
|
|
valid: false, // Dust output.
|
|
unconfirmed: false,
|
|
},
|
|
{
|
|
outVals: []int64{200},
|
|
feeRate: 2500,
|
|
valid: false, // Dust output.
|
|
unconfirmed: true,
|
|
},
|
|
|
|
{
|
|
outVals: []int64{1e8},
|
|
feeRate: 2500,
|
|
valid: true,
|
|
unconfirmed: false,
|
|
},
|
|
{
|
|
outVals: []int64{1e8},
|
|
feeRate: 2500,
|
|
valid: true,
|
|
unconfirmed: true,
|
|
},
|
|
|
|
{
|
|
outVals: []int64{1e8, 2e8, 1e8, 2e7, 3e5},
|
|
feeRate: 2500,
|
|
valid: true,
|
|
unconfirmed: false,
|
|
},
|
|
{
|
|
outVals: []int64{1e8, 2e8, 1e8, 2e7, 3e5},
|
|
feeRate: 2500,
|
|
valid: true,
|
|
unconfirmed: true,
|
|
},
|
|
|
|
{
|
|
outVals: []int64{1e8, 2e8, 1e8, 2e7, 3e5},
|
|
feeRate: 12500,
|
|
valid: true,
|
|
unconfirmed: false,
|
|
},
|
|
{
|
|
outVals: []int64{1e8, 2e8, 1e8, 2e7, 3e5},
|
|
feeRate: 12500,
|
|
valid: true,
|
|
unconfirmed: true,
|
|
},
|
|
|
|
{
|
|
outVals: []int64{1e8, 2e8, 1e8, 2e7, 3e5},
|
|
feeRate: 50000,
|
|
valid: true,
|
|
unconfirmed: false,
|
|
},
|
|
{
|
|
outVals: []int64{1e8, 2e8, 1e8, 2e7, 3e5},
|
|
feeRate: 50000,
|
|
valid: true,
|
|
unconfirmed: true,
|
|
},
|
|
|
|
{
|
|
outVals: []int64{1e8, 2e8, 1e8, 2e7, 3e5, 1e8, 2e8,
|
|
1e8, 2e7, 3e5},
|
|
feeRate: 44250,
|
|
valid: true,
|
|
unconfirmed: false,
|
|
},
|
|
{
|
|
outVals: []int64{1e8, 2e8, 1e8, 2e7, 3e5, 1e8, 2e8,
|
|
1e8, 2e7, 3e5},
|
|
feeRate: 44250,
|
|
valid: true,
|
|
unconfirmed: true,
|
|
},
|
|
}
|
|
|
|
for i, test := range testCases {
|
|
var minConfs int32 = 1
|
|
|
|
feeRate := test.feeRate
|
|
|
|
// Grab some fresh addresses from the miner that we will send
|
|
// to.
|
|
outputs := make([]*wire.TxOut, len(test.outVals))
|
|
for i, outVal := range test.outVals {
|
|
minerAddr, err := r.NewAddress()
|
|
if err != nil {
|
|
t.Fatalf("unable to generate address for "+
|
|
"miner: %v", err)
|
|
}
|
|
script, err := txscript.PayToAddrScript(minerAddr)
|
|
if err != nil {
|
|
t.Fatalf("unable to create pay to addr "+
|
|
"script: %v", err)
|
|
}
|
|
output := &wire.TxOut{
|
|
Value: outVal,
|
|
PkScript: script,
|
|
}
|
|
|
|
outputs[i] = output
|
|
}
|
|
|
|
// Now try creating a tx spending to these outputs.
|
|
createTx, createErr := w.CreateSimpleTx(
|
|
nil, outputs, feeRate, minConfs,
|
|
w.Cfg.CoinSelectionStrategy, true,
|
|
)
|
|
switch {
|
|
case test.valid && createErr != nil:
|
|
fmt.Println(spew.Sdump(createTx.Tx))
|
|
t.Fatalf("got unexpected error when creating tx: %v",
|
|
createErr)
|
|
|
|
case !test.valid && createErr == nil:
|
|
t.Fatalf("test #%v should have failed on tx "+
|
|
"creation", i)
|
|
}
|
|
|
|
// Also send to these outputs. This should result in a tx
|
|
// _very_ similar to the one we just created being sent. The
|
|
// only difference is that the dry run tx is not signed, and
|
|
// that the change output position might be different.
|
|
tx, sendErr := w.SendOutputs(
|
|
nil, outputs, feeRate, minConfs, labels.External,
|
|
w.Cfg.CoinSelectionStrategy,
|
|
)
|
|
switch {
|
|
case test.valid && sendErr != nil:
|
|
t.Fatalf("got unexpected error when sending tx: %v",
|
|
sendErr)
|
|
|
|
case !test.valid && sendErr == nil:
|
|
t.Fatalf("test #%v should fail for tx sending", i)
|
|
}
|
|
|
|
// We expected either both to not fail, or both to fail with
|
|
// the same error.
|
|
if createErr != sendErr {
|
|
t.Fatalf("error creating tx (%v) different "+
|
|
"from error sending outputs (%v)",
|
|
createErr, sendErr)
|
|
}
|
|
|
|
// If we expected the creation to fail, then this test is over.
|
|
if !test.valid {
|
|
continue
|
|
}
|
|
|
|
txid := tx.TxHash()
|
|
err = waitForMempoolTx(r, &txid)
|
|
if err != nil {
|
|
t.Fatalf("tx not relayed to miner: %v", err)
|
|
}
|
|
|
|
// Helper method to check that the two txs are similar.
|
|
assertSimilarTx := func(a, b *wire.MsgTx) error {
|
|
if a.Version != b.Version {
|
|
return fmt.Errorf("different versions: "+
|
|
"%v vs %v", a.Version, b.Version)
|
|
}
|
|
if a.LockTime != b.LockTime {
|
|
return fmt.Errorf("different locktimes: "+
|
|
"%v vs %v", a.LockTime, b.LockTime)
|
|
}
|
|
if len(a.TxIn) != len(b.TxIn) {
|
|
return fmt.Errorf("different number of "+
|
|
"inputs: %v vs %v", len(a.TxIn),
|
|
len(b.TxIn))
|
|
}
|
|
if len(a.TxOut) != len(b.TxOut) {
|
|
return fmt.Errorf("different number of "+
|
|
"outputs: %v vs %v", len(a.TxOut),
|
|
len(b.TxOut))
|
|
}
|
|
|
|
// They should be spending the same inputs.
|
|
for i := range a.TxIn {
|
|
prevA := a.TxIn[i].PreviousOutPoint
|
|
prevB := b.TxIn[i].PreviousOutPoint
|
|
if prevA != prevB {
|
|
return fmt.Errorf("different inputs: "+
|
|
"%v vs %v", spew.Sdump(prevA),
|
|
spew.Sdump(prevB))
|
|
}
|
|
}
|
|
|
|
// They should have the same outputs. Since the change
|
|
// output position gets randomized, they are not
|
|
// guaranteed to be in the same order.
|
|
for _, outA := range a.TxOut {
|
|
found := false
|
|
for _, outB := range b.TxOut {
|
|
if reflect.DeepEqual(outA, outB) {
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
if !found {
|
|
return fmt.Errorf("did not find "+
|
|
"output %v", spew.Sdump(outA))
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Assert that our "template tx" was similar to the one that
|
|
// ended up being sent.
|
|
if err := assertSimilarTx(createTx.Tx, tx); err != nil {
|
|
t.Fatalf("transactions not similar: %v", err)
|
|
}
|
|
|
|
// Now that we know both transactions were essentially
|
|
// identical, we'll make sure that a P2TR addr was used as the
|
|
// change output, which is the current default.
|
|
changeTxOut := createTx.Tx.TxOut[createTx.ChangeIndex]
|
|
changeScriptType, _, _, err := txscript.ExtractPkScriptAddrs(
|
|
changeTxOut.PkScript, &w.Cfg.NetParams,
|
|
)
|
|
require.NoError(t, err)
|
|
require.Equal(t, changeScriptType, txscript.WitnessV1TaprootTy)
|
|
}
|
|
}
|
|
|
|
// testSignOutputCreateAccount tests that we're able to properly sign for an
|
|
// output if the target account hasn't yet been created on disk. In this case,
|
|
// we'll create the account, then sign.
|
|
func testSignOutputCreateAccount(r *rpctest.Harness, w *lnwallet.LightningWallet,
|
|
_ *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
// First, we'll create a sign desc that references a non-default key
|
|
// family. Under the hood, key families are actually accounts, so this
|
|
// should force create of the account so we can sign with it.
|
|
fakeTx := wire.NewMsgTx(2)
|
|
fakeTx.AddTxIn(&wire.TxIn{
|
|
PreviousOutPoint: wire.OutPoint{
|
|
Hash: chainhash.Hash{},
|
|
Index: 0,
|
|
},
|
|
})
|
|
signDesc := &input.SignDescriptor{
|
|
KeyDesc: keychain.KeyDescriptor{
|
|
KeyLocator: keychain.KeyLocator{
|
|
Family: 99,
|
|
Index: 1,
|
|
},
|
|
},
|
|
WitnessScript: []byte{},
|
|
Output: &wire.TxOut{
|
|
Value: 1000,
|
|
},
|
|
HashType: txscript.SigHashAll,
|
|
SigHashes: input.NewTxSigHashesV0Only(fakeTx),
|
|
InputIndex: 0,
|
|
}
|
|
|
|
// We'll now sign and expect this to succeed, as even though the
|
|
// account doesn't exist atm, it should be created in order to process
|
|
// the inbound signing request.
|
|
_, err := w.Cfg.Signer.SignOutputRaw(fakeTx, signDesc)
|
|
if err != nil {
|
|
t.Fatalf("unable to sign for output with non-existent "+
|
|
"account: %v", err)
|
|
}
|
|
}
|
|
|
|
type walletTestCase struct {
|
|
name string
|
|
test func(miner *rpctest.Harness, alice, bob *lnwallet.LightningWallet,
|
|
test *testing.T)
|
|
}
|
|
|
|
var walletTests = []walletTestCase{
|
|
{
|
|
// TODO(wilmer): this test should remain first until the wallet
|
|
// can properly craft a transaction that spends all of its
|
|
// on-chain funds.
|
|
name: "change output spend confirmation",
|
|
test: testChangeOutputSpendConfirmation,
|
|
},
|
|
{
|
|
// TODO(guggero): this test should remain second until dual
|
|
// funding can properly exchange full UTXO information and we
|
|
// can use P2TR change outputs as the funding inputs for a dual
|
|
// funded channel.
|
|
name: "dual funder workflow",
|
|
test: testDualFundingReservationWorkflow,
|
|
},
|
|
{
|
|
name: "spend unconfirmed outputs",
|
|
test: testSpendUnconfirmed,
|
|
},
|
|
{
|
|
name: "insane fee reject",
|
|
test: testReservationInitiatorBalanceBelowDustCancel,
|
|
},
|
|
{
|
|
name: "single funding workflow",
|
|
test: func(miner *rpctest.Harness, alice,
|
|
bob *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
testSingleFunderReservationWorkflow(
|
|
miner, alice, bob, t,
|
|
lnwallet.CommitmentTypeLegacy, nil,
|
|
nil, [32]byte{1}, 0,
|
|
)
|
|
},
|
|
},
|
|
{
|
|
name: "single funding workflow tweakless",
|
|
test: func(miner *rpctest.Harness, alice,
|
|
bob *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
testSingleFunderReservationWorkflow(
|
|
miner, alice, bob, t,
|
|
lnwallet.CommitmentTypeTweakless, nil,
|
|
nil, [32]byte{1}, 0,
|
|
)
|
|
},
|
|
},
|
|
{
|
|
name: "single funding workflow musig2",
|
|
test: func(miner *rpctest.Harness, alice,
|
|
bob *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
testSingleFunderReservationWorkflow(
|
|
miner, alice, bob, t,
|
|
lnwallet.CommitmentTypeSimpleTaproot, nil,
|
|
nil, [32]byte{1}, 0,
|
|
)
|
|
},
|
|
},
|
|
// TODO(roasbeef): add musig2 external funding
|
|
{
|
|
name: "single funding workflow external funding tx",
|
|
test: testSingleFunderExternalFundingTx,
|
|
},
|
|
{
|
|
name: "output locking",
|
|
test: testFundingTransactionLockedOutputs,
|
|
},
|
|
{
|
|
name: "reservation insufficient funds",
|
|
test: testFundingCancellationNotEnoughFunds,
|
|
},
|
|
{
|
|
name: "transaction subscriptions",
|
|
test: testTransactionSubscriptions,
|
|
},
|
|
{
|
|
name: "transaction details",
|
|
test: testListTransactionDetails,
|
|
},
|
|
{
|
|
name: "get transaction details",
|
|
test: testGetTransactionDetails,
|
|
},
|
|
{
|
|
name: "publish transaction",
|
|
test: testPublishTransaction,
|
|
},
|
|
{
|
|
name: "signed with tweaked pubkeys",
|
|
test: testSignOutputUsingTweaks,
|
|
},
|
|
{
|
|
name: "test cancel non-existent reservation",
|
|
test: testCancelNonExistentReservation,
|
|
},
|
|
{
|
|
name: "last unused addr",
|
|
test: testLastUnusedAddr,
|
|
},
|
|
{
|
|
name: "reorg wallet balance",
|
|
test: testReorgWalletBalance,
|
|
},
|
|
{
|
|
name: "create simple tx",
|
|
test: testCreateSimpleTx,
|
|
},
|
|
{
|
|
name: "test sign create account",
|
|
test: testSignOutputCreateAccount,
|
|
},
|
|
{
|
|
name: "test get recovery info",
|
|
test: testGetRecoveryInfo,
|
|
},
|
|
}
|
|
|
|
func clearWalletStates(a, b *lnwallet.LightningWallet) error {
|
|
a.ResetReservations()
|
|
b.ResetReservations()
|
|
|
|
if err := a.Cfg.Database.GetParentDB().Wipe(); err != nil {
|
|
return err
|
|
}
|
|
|
|
return b.Cfg.Database.GetParentDB().Wipe()
|
|
}
|
|
|
|
func waitForMempoolTx(r *rpctest.Harness, txid *chainhash.Hash) error {
|
|
var found bool
|
|
var tx *btcutil.Tx
|
|
var err error
|
|
timeout := time.After(30 * time.Second)
|
|
for !found {
|
|
// Do a short wait
|
|
select {
|
|
case <-timeout:
|
|
return fmt.Errorf("timeout after 10s")
|
|
default:
|
|
}
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
// Check for the harness' knowledge of the txid
|
|
tx, err = r.Client.GetRawTransaction(txid)
|
|
if err != nil {
|
|
switch e := err.(type) {
|
|
case *btcjson.RPCError:
|
|
if e.Code == btcjson.ErrRPCNoTxInfo {
|
|
continue
|
|
}
|
|
default:
|
|
}
|
|
return err
|
|
}
|
|
if tx != nil && tx.MsgTx().TxHash() == *txid {
|
|
found = true
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func waitForWalletSync(r *rpctest.Harness, w *lnwallet.LightningWallet) error {
|
|
var (
|
|
synced bool
|
|
err error
|
|
bestHash, knownHash *chainhash.Hash
|
|
bestHeight, knownHeight int32
|
|
)
|
|
timeout := time.After(10 * time.Second)
|
|
for !synced {
|
|
// Do a short wait
|
|
select {
|
|
case <-timeout:
|
|
return fmt.Errorf("timeout after 30s")
|
|
case <-time.Tick(100 * time.Millisecond):
|
|
}
|
|
|
|
// Check whether the chain source of the wallet is caught up to
|
|
// the harness it's supposed to be catching up to.
|
|
bestHash, bestHeight, err = r.Client.GetBestBlock()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
knownHash, knownHeight, err = w.Cfg.ChainIO.GetBestBlock()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if knownHeight != bestHeight {
|
|
continue
|
|
}
|
|
if *knownHash != *bestHash {
|
|
return fmt.Errorf("hash at height %d doesn't match: "+
|
|
"expected %s, got %s", bestHeight, bestHash,
|
|
knownHash)
|
|
}
|
|
|
|
// Check for synchronization.
|
|
synced, _, err = w.IsSynced()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// testSingleFunderExternalFundingTx tests that the wallet is able to properly
|
|
// carry out a funding flow backed by a channel point that has been crafted
|
|
// outside the wallet.
|
|
func testSingleFunderExternalFundingTx(miner *rpctest.Harness,
|
|
alice, bob *lnwallet.LightningWallet, t *testing.T) {
|
|
|
|
// Define a filter function without any restrictions.
|
|
allowUtxo := func(lnwallet.Utxo) bool {
|
|
return true
|
|
}
|
|
|
|
// First, we'll obtain multi-sig keys from both Alice and Bob which
|
|
// simulates them exchanging keys on a higher level.
|
|
aliceFundingKey, err := alice.DeriveNextKey(keychain.KeyFamilyMultiSig)
|
|
require.NoError(t, err, "unable to obtain alice funding key")
|
|
bobFundingKey, err := bob.DeriveNextKey(keychain.KeyFamilyMultiSig)
|
|
require.NoError(t, err, "unable to obtain bob funding key")
|
|
|
|
// We'll now set up for them to open a 4 BTC channel, with 1 BTC pushed
|
|
// to Bob's side.
|
|
chanAmt := 4 * btcutil.SatoshiPerBitcoin
|
|
|
|
// Simulating external funding negotiation, we'll now create the
|
|
// funding transaction for both parties. Utilizing existing tools,
|
|
// we'll create a new chanfunding.Assembler hacked by Alice's wallet.
|
|
aliceChanFunder := chanfunding.NewWalletAssembler(
|
|
chanfunding.WalletConfig{
|
|
CoinSource: lnwallet.NewCoinSource(
|
|
alice, allowUtxo,
|
|
),
|
|
CoinSelectLocker: alice,
|
|
CoinLeaser: alice,
|
|
Signer: alice.Cfg.Signer,
|
|
DustLimit: 600,
|
|
CoinSelectionStrategy: wallet.CoinSelectionLargest,
|
|
},
|
|
)
|
|
|
|
// With the chan funder created, we'll now provision a funding intent,
|
|
// bind the keys we obtained above, and finally obtain our funding
|
|
// transaction and outpoint.
|
|
fundingIntent, err := aliceChanFunder.ProvisionChannel(
|
|
&chanfunding.Request{
|
|
LocalAmt: btcutil.Amount(chanAmt),
|
|
MinConfs: 1,
|
|
FeeRate: 253,
|
|
ChangeAddr: func() (btcutil.Address, error) {
|
|
return alice.NewAddress(
|
|
lnwallet.WitnessPubKey, true,
|
|
lnwallet.DefaultAccountName,
|
|
)
|
|
},
|
|
},
|
|
)
|
|
require.NoError(t, err, "unable to perform coin selection")
|
|
|
|
// With our intent created, we'll instruct it to finalize the funding
|
|
// transaction, and also hand us the outpoint so we can simulate
|
|
// external crafting of the funding transaction.
|
|
var (
|
|
fundingTx *wire.MsgTx
|
|
chanPoint *wire.OutPoint
|
|
)
|
|
if fullIntent, ok := fundingIntent.(*chanfunding.FullIntent); ok {
|
|
fullIntent.BindKeys(&aliceFundingKey, bobFundingKey.PubKey)
|
|
|
|
fundingTx, err = fullIntent.CompileFundingTx(nil, nil)
|
|
if err != nil {
|
|
t.Fatalf("unable to compile funding tx: %v", err)
|
|
}
|
|
chanPoint, err = fullIntent.ChanPoint()
|
|
if err != nil {
|
|
t.Fatalf("unable to obtain chan point: %v", err)
|
|
}
|
|
} else {
|
|
t.Fatalf("expected full intent, instead got: %T", fullIntent)
|
|
}
|
|
|
|
// Now that we have the fully constructed funding transaction, we'll
|
|
// create a new shim external funder out of it for Alice, and prep a
|
|
// shim intent for Bob.
|
|
thawHeight := uint32(200)
|
|
aliceExternalFunder := chanfunding.NewCannedAssembler(
|
|
thawHeight, *chanPoint, btcutil.Amount(chanAmt), &aliceFundingKey,
|
|
bobFundingKey.PubKey, true, false,
|
|
)
|
|
bobShimIntent, err := chanfunding.NewCannedAssembler(
|
|
thawHeight, *chanPoint, btcutil.Amount(chanAmt), &bobFundingKey,
|
|
aliceFundingKey.PubKey, false, false,
|
|
).ProvisionChannel(&chanfunding.Request{
|
|
LocalAmt: btcutil.Amount(chanAmt),
|
|
MinConfs: 1,
|
|
FeeRate: 253,
|
|
ChangeAddr: func() (btcutil.Address, error) {
|
|
return bob.NewAddress(
|
|
lnwallet.WitnessPubKey, true,
|
|
lnwallet.DefaultAccountName,
|
|
)
|
|
},
|
|
})
|
|
require.NoError(t, err, "unable to create shim intent for bob")
|
|
|
|
// At this point, we have everything we need to carry out our test, so
|
|
// we'll being the funding flow between Alice and Bob.
|
|
//
|
|
// However, before we do so, we'll register a new shim intent for Bob,
|
|
// so he knows what keys to use when he receives the funding request
|
|
// from Alice.
|
|
pendingChanID := testHdSeed
|
|
err = bob.RegisterFundingIntent(pendingChanID, bobShimIntent)
|
|
require.NoError(t, err, "unable to register intent")
|
|
|
|
// Now we can carry out the single funding flow as normal, we'll
|
|
// specify our external funder and funding transaction, as well as the
|
|
// pending channel ID generated above to allow Alice and Bob to track
|
|
// the funding flow externally.
|
|
testSingleFunderReservationWorkflow(
|
|
miner, alice, bob, t, lnwallet.CommitmentTypeTweakless,
|
|
aliceExternalFunder, func() *wire.MsgTx {
|
|
return fundingTx
|
|
}, pendingChanID, thawHeight,
|
|
)
|
|
}
|
|
|
|
// TestInterfaces tests all registered interfaces with a unified set of tests
|
|
// which exercise each of the required methods found within the WalletController
|
|
// interface.
|
|
//
|
|
// NOTE: In the future, when additional implementations of the WalletController
|
|
// interface have been implemented, in order to ensure the new concrete
|
|
// implementation is automatically tested, two steps must be undertaken. First,
|
|
// one needs add a "non-captured" (_) import from the new sub-package. This
|
|
// import should trigger an init() method within the package which registers
|
|
// the interface. Second, an additional case in the switch within the main loop
|
|
// below needs to be added which properly initializes the interface.
|
|
//
|
|
// TODO(roasbeef): purge bobNode in favor of dual lnwallet's
|
|
func TestLightningWallet(t *testing.T, targetBackEnd string) {
|
|
t.Parallel()
|
|
|
|
// Initialize the harness around a btcd node which will serve as our
|
|
// dedicated miner to generate blocks, cause re-orgs, etc. We'll set
|
|
// up this node with a chain length of 125, so we have plenty of BTC
|
|
// to play around with.
|
|
miningNode := unittest.NewMiner(
|
|
t, netParams, []string{"--txindex"}, true, 25,
|
|
)
|
|
|
|
// Next mine enough blocks in order for segwit and the CSV package
|
|
// soft-fork to activate on RegNet.
|
|
numBlocks := netParams.MinerConfirmationWindow * 2
|
|
if _, err := miningNode.Client.Generate(numBlocks); err != nil {
|
|
t.Fatalf("unable to generate blocks: %v", err)
|
|
}
|
|
|
|
rpcConfig := miningNode.RPCConfig()
|
|
|
|
tempDir := t.TempDir()
|
|
db, err := channeldb.Open(tempDir)
|
|
require.NoError(t, err, "unable to create db")
|
|
testCfg := channeldb.CacheConfig{
|
|
QueryDisable: false,
|
|
}
|
|
hintCache, err := channeldb.NewHeightHintCache(testCfg, db.Backend)
|
|
require.NoError(t, err, "unable to create height hint cache")
|
|
blockCache := blockcache.NewBlockCache(10000)
|
|
chainNotifier, err := btcdnotify.New(
|
|
&rpcConfig, netParams, hintCache, hintCache, blockCache,
|
|
)
|
|
require.NoError(t, err, "unable to create notifier")
|
|
if err := chainNotifier.Start(); err != nil {
|
|
t.Fatalf("unable to start notifier: %v", err)
|
|
}
|
|
|
|
for _, walletDriver := range lnwallet.RegisteredWallets() {
|
|
for _, backEnd := range walletDriver.BackEnds() {
|
|
if backEnd != targetBackEnd {
|
|
continue
|
|
}
|
|
|
|
if !runTests(t, walletDriver, backEnd, miningNode,
|
|
rpcConfig, chainNotifier) {
|
|
|
|
return
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// runTests runs all of the tests for a single interface implementation and
|
|
// chain back-end combination. This makes it easier to use `defer` as well as
|
|
// factoring out the test logic from the loop which cycles through the
|
|
// interface implementations.
|
|
func runTests(t *testing.T, walletDriver *lnwallet.WalletDriver,
|
|
backEnd string, miningNode *rpctest.Harness,
|
|
rpcConfig rpcclient.ConnConfig,
|
|
chainNotifier chainntnfs.ChainNotifier) bool {
|
|
|
|
var (
|
|
bio lnwallet.BlockChainIO
|
|
|
|
aliceSigner input.Signer
|
|
bobSigner input.Signer
|
|
|
|
aliceKeyRing keychain.SecretKeyRing
|
|
bobKeyRing keychain.SecretKeyRing
|
|
|
|
aliceWalletController lnwallet.WalletController
|
|
bobWalletController lnwallet.WalletController
|
|
|
|
err error
|
|
)
|
|
|
|
tempTestDirAlice := t.TempDir()
|
|
tempTestDirBob := t.TempDir()
|
|
|
|
blockCache := blockcache.NewBlockCache(10000)
|
|
|
|
walletType := walletDriver.WalletType
|
|
switch walletType {
|
|
case "btcwallet":
|
|
var aliceClient, bobClient chain.Interface
|
|
switch backEnd {
|
|
case "btcd":
|
|
aliceClient, err = chain.NewRPCClient(
|
|
netParams, rpcConfig.Host, rpcConfig.User,
|
|
rpcConfig.Pass, rpcConfig.Certificates, false,
|
|
20,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to make chain rpc: %v", err)
|
|
}
|
|
bobClient, err = chain.NewRPCClient(
|
|
netParams, rpcConfig.Host, rpcConfig.User,
|
|
rpcConfig.Pass, rpcConfig.Certificates, false,
|
|
20,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to make chain rpc: %v", err)
|
|
}
|
|
|
|
case "neutrino":
|
|
// Set some package-level variable to speed up
|
|
// operation for tests.
|
|
neutrino.BanDuration = time.Millisecond * 100
|
|
neutrino.QueryTimeout = time.Millisecond * 500
|
|
neutrino.QueryNumRetries = 1
|
|
|
|
// Start Alice - open a database, start a neutrino
|
|
// instance, and initialize a btcwallet driver for it.
|
|
aliceDB, err := walletdb.Create(
|
|
"bdb", tempTestDirAlice+"/neutrino.db", true,
|
|
kvdb.DefaultDBTimeout,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to create DB: %v", err)
|
|
}
|
|
defer aliceDB.Close()
|
|
aliceChain, err := neutrino.NewChainService(
|
|
neutrino.Config{
|
|
DataDir: tempTestDirAlice,
|
|
Database: aliceDB,
|
|
ChainParams: *netParams,
|
|
ConnectPeers: []string{
|
|
miningNode.P2PAddress(),
|
|
},
|
|
},
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to make neutrino: %v", err)
|
|
}
|
|
aliceChain.Start()
|
|
defer aliceChain.Stop()
|
|
aliceClient = chain.NewNeutrinoClient(
|
|
netParams, aliceChain,
|
|
)
|
|
|
|
// Start Bob - open a database, start a neutrino
|
|
// instance, and initialize a btcwallet driver for it.
|
|
bobDB, err := walletdb.Create(
|
|
"bdb", tempTestDirBob+"/neutrino.db", true,
|
|
kvdb.DefaultDBTimeout,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to create DB: %v", err)
|
|
}
|
|
defer bobDB.Close()
|
|
bobChain, err := neutrino.NewChainService(
|
|
neutrino.Config{
|
|
DataDir: tempTestDirBob,
|
|
Database: bobDB,
|
|
ChainParams: *netParams,
|
|
ConnectPeers: []string{
|
|
miningNode.P2PAddress(),
|
|
},
|
|
},
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to make neutrino: %v", err)
|
|
}
|
|
bobChain.Start()
|
|
defer bobChain.Stop()
|
|
bobClient = chain.NewNeutrinoClient(
|
|
netParams, bobChain,
|
|
)
|
|
|
|
case "bitcoind":
|
|
// Start a bitcoind instance.
|
|
chainConn := unittest.NewBitcoindBackend(
|
|
t, unittest.NetParams, miningNode.P2PAddress(),
|
|
true, false,
|
|
)
|
|
|
|
// Create a btcwallet bitcoind client for both Alice and
|
|
// Bob.
|
|
aliceClient = chainConn.NewBitcoindClient()
|
|
bobClient = chainConn.NewBitcoindClient()
|
|
|
|
case "bitcoind-rpc-polling":
|
|
// Start a bitcoind instance.
|
|
chainConn := unittest.NewBitcoindBackend(
|
|
t, unittest.NetParams, miningNode.P2PAddress(),
|
|
true, true,
|
|
)
|
|
|
|
// Create a btcwallet bitcoind client for both Alice and
|
|
// Bob.
|
|
aliceClient = chainConn.NewBitcoindClient()
|
|
bobClient = chainConn.NewBitcoindClient()
|
|
|
|
default:
|
|
t.Fatalf("unknown chain driver: %v", backEnd)
|
|
}
|
|
|
|
aliceSeed := sha256.New()
|
|
aliceSeed.Write([]byte(backEnd))
|
|
aliceSeed.Write(aliceHDSeed[:])
|
|
aliceSeedBytes := aliceSeed.Sum(nil)
|
|
|
|
aliceWalletConfig := &btcwallet.Config{
|
|
PrivatePass: []byte("alice-pass"),
|
|
HdSeed: aliceSeedBytes,
|
|
NetParams: netParams,
|
|
ChainSource: aliceClient,
|
|
CoinType: keychain.CoinTypeTestnet,
|
|
// wallet starts in recovery mode
|
|
RecoveryWindow: 2,
|
|
LoaderOptions: []btcwallet.LoaderOption{
|
|
btcwallet.LoaderWithLocalWalletDB(
|
|
tempTestDirAlice, false, time.Minute,
|
|
),
|
|
},
|
|
}
|
|
aliceWalletController, err = walletDriver.New(
|
|
aliceWalletConfig, blockCache,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to create btcwallet: %v", err)
|
|
}
|
|
aliceSigner = aliceWalletController.(*btcwallet.BtcWallet)
|
|
aliceKeyRing = keychain.NewBtcWalletKeyRing(
|
|
aliceWalletController.(*btcwallet.BtcWallet).InternalWallet(),
|
|
keychain.CoinTypeTestnet,
|
|
)
|
|
|
|
bobSeed := sha256.New()
|
|
bobSeed.Write([]byte(backEnd))
|
|
bobSeed.Write(bobHDSeed[:])
|
|
bobSeedBytes := bobSeed.Sum(nil)
|
|
|
|
bobWalletConfig := &btcwallet.Config{
|
|
PrivatePass: []byte("bob-pass"),
|
|
HdSeed: bobSeedBytes,
|
|
NetParams: netParams,
|
|
ChainSource: bobClient,
|
|
CoinType: keychain.CoinTypeTestnet,
|
|
// wallet starts without recovery mode
|
|
RecoveryWindow: 0,
|
|
LoaderOptions: []btcwallet.LoaderOption{
|
|
btcwallet.LoaderWithLocalWalletDB(
|
|
tempTestDirBob, false, time.Minute,
|
|
),
|
|
},
|
|
}
|
|
bobWalletController, err = walletDriver.New(
|
|
bobWalletConfig, blockCache,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("unable to create btcwallet: %v", err)
|
|
}
|
|
bobSigner = bobWalletController.(*btcwallet.BtcWallet)
|
|
bobKeyRing = keychain.NewBtcWalletKeyRing(
|
|
bobWalletController.(*btcwallet.BtcWallet).InternalWallet(),
|
|
keychain.CoinTypeTestnet,
|
|
)
|
|
bio = bobWalletController.(*btcwallet.BtcWallet)
|
|
default:
|
|
t.Fatalf("unknown wallet driver: %v", walletType)
|
|
}
|
|
|
|
// Funding via 20 outputs with 4BTC each.
|
|
alice, err := createTestWallet(
|
|
tempTestDirAlice, miningNode, netParams,
|
|
chainNotifier, aliceWalletController, aliceKeyRing,
|
|
aliceSigner, bio,
|
|
)
|
|
require.NoError(t, err, "unable to create test ln wallet")
|
|
defer alice.Shutdown()
|
|
|
|
bob, err := createTestWallet(
|
|
tempTestDirBob, miningNode, netParams,
|
|
chainNotifier, bobWalletController, bobKeyRing, bobSigner, bio,
|
|
)
|
|
require.NoError(t, err, "unable to create test ln wallet")
|
|
defer bob.Shutdown()
|
|
|
|
// Both wallets should now have 80BTC available for
|
|
// spending.
|
|
assertProperBalance(t, alice, 1, 80)
|
|
assertProperBalance(t, bob, 1, 80)
|
|
|
|
// Execute every test, clearing possibly mutated
|
|
// wallet state after each step.
|
|
for _, walletTest := range walletTests {
|
|
|
|
walletTest := walletTest
|
|
|
|
testName := fmt.Sprintf("%v/%v:%v", walletType, backEnd,
|
|
walletTest.name)
|
|
success := t.Run(testName, func(t *testing.T) {
|
|
if backEnd == "neutrino" &&
|
|
strings.Contains(walletTest.name, "dual funder") {
|
|
|
|
t.Skip("skipping dual funder tests for neutrino")
|
|
}
|
|
if backEnd == "neutrino" &&
|
|
strings.Contains(walletTest.name, "spend unconfirmed") {
|
|
|
|
t.Skip("skipping spend unconfirmed tests for neutrino")
|
|
}
|
|
|
|
walletTest.test(miningNode, alice, bob, t)
|
|
})
|
|
if !success {
|
|
return false
|
|
}
|
|
|
|
// TODO(roasbeef): possible reset mining
|
|
// node's chainstate to initial level, cleanly
|
|
// wipe buckets
|
|
if err := clearWalletStates(alice, bob); err !=
|
|
nil && err != kvdb.ErrBucketNotFound {
|
|
|
|
t.Fatalf("unable to wipe wallet state: %v", err)
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|