mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-01-19 05:45:21 +01:00
677f2c390a
To reflect the new sweeping behavior, also makes it easier to be used as we need to method to quickly cleanup force closes without concerning the details when we are not testing the force close behavior.
2163 lines
70 KiB
Go
2163 lines
70 KiB
Go
package lntest
|
|
|
|
import (
|
|
"context"
|
|
"encoding/hex"
|
|
"fmt"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/btcsuite/btcd/blockchain"
|
|
"github.com/btcsuite/btcd/btcutil"
|
|
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
|
"github.com/btcsuite/btcd/txscript"
|
|
"github.com/btcsuite/btcd/wire"
|
|
"github.com/go-errors/errors"
|
|
"github.com/lightningnetwork/lnd/fn"
|
|
"github.com/lightningnetwork/lnd/kvdb/etcd"
|
|
"github.com/lightningnetwork/lnd/lnrpc"
|
|
"github.com/lightningnetwork/lnd/lnrpc/routerrpc"
|
|
"github.com/lightningnetwork/lnd/lnrpc/walletrpc"
|
|
"github.com/lightningnetwork/lnd/lntest/miner"
|
|
"github.com/lightningnetwork/lnd/lntest/node"
|
|
"github.com/lightningnetwork/lnd/lntest/rpc"
|
|
"github.com/lightningnetwork/lnd/lntest/wait"
|
|
"github.com/lightningnetwork/lnd/lntypes"
|
|
"github.com/lightningnetwork/lnd/lnwallet/chainfee"
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
const (
|
|
// defaultMinerFeeRate specifies the fee rate in sats when sending
|
|
// outputs from the miner.
|
|
defaultMinerFeeRate = 7500
|
|
|
|
// numBlocksSendOutput specifies the number of blocks to mine after
|
|
// sending outputs from the miner.
|
|
numBlocksSendOutput = 2
|
|
|
|
// numBlocksOpenChannel specifies the number of blocks mined when
|
|
// opening a channel.
|
|
numBlocksOpenChannel = 6
|
|
|
|
// lndErrorChanSize specifies the buffer size used to receive errors
|
|
// from lnd process.
|
|
lndErrorChanSize = 10
|
|
|
|
// maxBlocksAllowed specifies the max allowed value to be used when
|
|
// mining blocks.
|
|
maxBlocksAllowed = 100
|
|
)
|
|
|
|
// TestCase defines a test case that's been used in the integration test.
|
|
type TestCase struct {
|
|
// Name specifies the test name.
|
|
Name string
|
|
|
|
// TestFunc is the test case wrapped in a function.
|
|
TestFunc func(t *HarnessTest)
|
|
}
|
|
|
|
// standbyNodes are a list of nodes which are created during the initialization
|
|
// of the test and used across all test cases.
|
|
type standbyNodes struct {
|
|
// Alice and Bob are the initial seeder nodes that are automatically
|
|
// created to be the initial participants of the test network.
|
|
Alice *node.HarnessNode
|
|
Bob *node.HarnessNode
|
|
}
|
|
|
|
// HarnessTest builds on top of a testing.T with enhanced error detection. It
|
|
// is responsible for managing the interactions among different nodes, and
|
|
// providing easy-to-use assertions.
|
|
type HarnessTest struct {
|
|
*testing.T
|
|
|
|
// Embed the standbyNodes so we can easily access them via `ht.Alice`.
|
|
standbyNodes
|
|
|
|
// miner is a reference to a running full node that can be used to
|
|
// create new blocks on the network.
|
|
miner *miner.HarnessMiner
|
|
|
|
// manager handles the start and stop of a given node.
|
|
manager *nodeManager
|
|
|
|
// feeService is a web service that provides external fee estimates to
|
|
// lnd.
|
|
feeService WebFeeService
|
|
|
|
// Channel for transmitting stderr output from failed lightning node
|
|
// to main process.
|
|
lndErrorChan chan error
|
|
|
|
// runCtx is a context with cancel method. It's used to signal when the
|
|
// node needs to quit, and used as the parent context when spawning
|
|
// children contexts for RPC requests.
|
|
runCtx context.Context //nolint:containedctx
|
|
cancel context.CancelFunc
|
|
|
|
// stopChainBackend points to the cleanup function returned by the
|
|
// chainBackend.
|
|
stopChainBackend func()
|
|
|
|
// cleaned specifies whether the cleanup has been applied for the
|
|
// current HarnessTest.
|
|
cleaned bool
|
|
|
|
// currentHeight is the current height of the chain backend.
|
|
currentHeight uint32
|
|
}
|
|
|
|
// harnessOpts contains functional option to modify the behavior of the various
|
|
// harness calls.
|
|
type harnessOpts struct {
|
|
useAMP bool
|
|
}
|
|
|
|
// defaultHarnessOpts returns a new instance of the harnessOpts with default
|
|
// values specified.
|
|
func defaultHarnessOpts() harnessOpts {
|
|
return harnessOpts{
|
|
useAMP: false,
|
|
}
|
|
}
|
|
|
|
// HarnessOpt is a functional option that can be used to modify the behavior of
|
|
// harness functionality.
|
|
type HarnessOpt func(*harnessOpts)
|
|
|
|
// WithAMP is a functional option that can be used to enable the AMP feature
|
|
// for sending payments.
|
|
func WithAMP() HarnessOpt {
|
|
return func(h *harnessOpts) {
|
|
h.useAMP = true
|
|
}
|
|
}
|
|
|
|
// NewHarnessTest creates a new instance of a harnessTest from a regular
|
|
// testing.T instance.
|
|
func NewHarnessTest(t *testing.T, lndBinary string, feeService WebFeeService,
|
|
dbBackend node.DatabaseBackend, nativeSQL bool) *HarnessTest {
|
|
|
|
t.Helper()
|
|
|
|
// Create the run context.
|
|
ctxt, cancel := context.WithCancel(context.Background())
|
|
|
|
manager := newNodeManager(lndBinary, dbBackend, nativeSQL)
|
|
|
|
return &HarnessTest{
|
|
T: t,
|
|
manager: manager,
|
|
feeService: feeService,
|
|
runCtx: ctxt,
|
|
cancel: cancel,
|
|
// We need to use buffered channel here as we don't want to
|
|
// block sending errors.
|
|
lndErrorChan: make(chan error, lndErrorChanSize),
|
|
}
|
|
}
|
|
|
|
// Start will assemble the chain backend and the miner for the HarnessTest. It
|
|
// also starts the fee service and watches lnd process error.
|
|
func (h *HarnessTest) Start(chain node.BackendConfig,
|
|
miner *miner.HarnessMiner) {
|
|
|
|
// Spawn a new goroutine to watch for any fatal errors that any of the
|
|
// running lnd processes encounter. If an error occurs, then the test
|
|
// case should naturally as a result and we log the server error here
|
|
// to help debug.
|
|
go func() {
|
|
select {
|
|
case err, more := <-h.lndErrorChan:
|
|
if !more {
|
|
return
|
|
}
|
|
h.Logf("lnd finished with error (stderr):\n%v", err)
|
|
|
|
case <-h.runCtx.Done():
|
|
return
|
|
}
|
|
}()
|
|
|
|
// Start the fee service.
|
|
err := h.feeService.Start()
|
|
require.NoError(h, err, "failed to start fee service")
|
|
|
|
// Assemble the node manager with chainBackend and feeServiceURL.
|
|
h.manager.chainBackend = chain
|
|
h.manager.feeServiceURL = h.feeService.URL()
|
|
|
|
// Assemble the miner.
|
|
h.miner = miner
|
|
}
|
|
|
|
// ChainBackendName returns the chain backend name used in the test.
|
|
func (h *HarnessTest) ChainBackendName() string {
|
|
return h.manager.chainBackend.Name()
|
|
}
|
|
|
|
// Context returns the run context used in this test. Usaually it should be
|
|
// managed by the test itself otherwise undefined behaviors will occur. It can
|
|
// be used, however, when a test needs to have its own context being managed
|
|
// differently. In that case, instead of using a background context, the run
|
|
// context should be used such that the test context scope can be fully
|
|
// controlled.
|
|
func (h *HarnessTest) Context() context.Context {
|
|
return h.runCtx
|
|
}
|
|
|
|
// setupWatchOnlyNode initializes a node with the watch-only accounts of an
|
|
// associated remote signing instance.
|
|
func (h *HarnessTest) setupWatchOnlyNode(name string,
|
|
signerNode *node.HarnessNode, password []byte) *node.HarnessNode {
|
|
|
|
// Prepare arguments for watch-only node connected to the remote signer.
|
|
remoteSignerArgs := []string{
|
|
"--remotesigner.enable",
|
|
fmt.Sprintf("--remotesigner.rpchost=localhost:%d",
|
|
signerNode.Cfg.RPCPort),
|
|
fmt.Sprintf("--remotesigner.tlscertpath=%s",
|
|
signerNode.Cfg.TLSCertPath),
|
|
fmt.Sprintf("--remotesigner.macaroonpath=%s",
|
|
signerNode.Cfg.AdminMacPath),
|
|
}
|
|
|
|
// Fetch watch-only accounts from the signer node.
|
|
resp := signerNode.RPC.ListAccounts(&walletrpc.ListAccountsRequest{})
|
|
watchOnlyAccounts, err := walletrpc.AccountsToWatchOnly(resp.Accounts)
|
|
require.NoErrorf(h, err, "unable to find watch only accounts for %s",
|
|
name)
|
|
|
|
// Create a new watch-only node with remote signer configuration.
|
|
return h.NewNodeRemoteSigner(
|
|
name, remoteSignerArgs, password,
|
|
&lnrpc.WatchOnly{
|
|
MasterKeyBirthdayTimestamp: 0,
|
|
MasterKeyFingerprint: nil,
|
|
Accounts: watchOnlyAccounts,
|
|
},
|
|
)
|
|
}
|
|
|
|
// createAndSendOutput send amt satoshis from the internal mining node to the
|
|
// targeted lightning node using a P2WKH address. No blocks are mined so
|
|
// transactions will sit unconfirmed in mempool.
|
|
func (h *HarnessTest) createAndSendOutput(target *node.HarnessNode,
|
|
amt btcutil.Amount, addrType lnrpc.AddressType) {
|
|
|
|
req := &lnrpc.NewAddressRequest{Type: addrType}
|
|
resp := target.RPC.NewAddress(req)
|
|
addr := h.DecodeAddress(resp.Address)
|
|
addrScript := h.PayToAddrScript(addr)
|
|
|
|
output := &wire.TxOut{
|
|
PkScript: addrScript,
|
|
Value: int64(amt),
|
|
}
|
|
h.miner.SendOutput(output, defaultMinerFeeRate)
|
|
}
|
|
|
|
// SetupRemoteSigningStandbyNodes starts the initial seeder nodes within the
|
|
// test harness in a remote signing configuration. The initial node's wallets
|
|
// will be funded wallets with 100x1 BTC outputs each.
|
|
func (h *HarnessTest) SetupRemoteSigningStandbyNodes() {
|
|
h.Log("Setting up standby nodes Alice and Bob with remote " +
|
|
"signing configurations...")
|
|
defer h.Log("Finished the setup, now running tests...")
|
|
|
|
password := []byte("itestpassword")
|
|
|
|
// Setup remote signing nodes for Alice and Bob.
|
|
signerAlice := h.NewNode("SignerAlice", nil)
|
|
signerBob := h.NewNode("SignerBob", nil)
|
|
|
|
// Setup watch-only nodes for Alice and Bob, each configured with their
|
|
// own remote signing instance.
|
|
h.Alice = h.setupWatchOnlyNode("Alice", signerAlice, password)
|
|
h.Bob = h.setupWatchOnlyNode("Bob", signerBob, password)
|
|
|
|
// Fund each node with 100 BTC (using 100 separate transactions).
|
|
const fundAmount = 1 * btcutil.SatoshiPerBitcoin
|
|
const numOutputs = 100
|
|
const totalAmount = fundAmount * numOutputs
|
|
for _, node := range []*node.HarnessNode{h.Alice, h.Bob} {
|
|
h.manager.standbyNodes[node.Cfg.NodeID] = node
|
|
for i := 0; i < numOutputs; i++ {
|
|
h.createAndSendOutput(
|
|
node, fundAmount,
|
|
lnrpc.AddressType_WITNESS_PUBKEY_HASH,
|
|
)
|
|
}
|
|
}
|
|
|
|
// We generate several blocks in order to give the outputs created
|
|
// above a good number of confirmations.
|
|
const totalTxes = 200
|
|
h.MineBlocksAndAssertNumTxes(numBlocksSendOutput, totalTxes)
|
|
|
|
// Now we want to wait for the nodes to catch up.
|
|
h.WaitForBlockchainSync(h.Alice)
|
|
h.WaitForBlockchainSync(h.Bob)
|
|
|
|
// Now block until both wallets have fully synced up.
|
|
h.WaitForBalanceConfirmed(h.Alice, totalAmount)
|
|
h.WaitForBalanceConfirmed(h.Bob, totalAmount)
|
|
}
|
|
|
|
// SetUp starts the initial seeder nodes within the test harness. The initial
|
|
// node's wallets will be funded wallets with 10x10 BTC outputs each.
|
|
func (h *HarnessTest) SetupStandbyNodes() {
|
|
h.Log("Setting up standby nodes Alice and Bob...")
|
|
defer h.Log("Finished the setup, now running tests...")
|
|
|
|
lndArgs := []string{
|
|
"--default-remote-max-htlcs=483",
|
|
"--dust-threshold=5000000",
|
|
}
|
|
|
|
// Start the initial seeder nodes within the test network.
|
|
h.Alice = h.NewNode("Alice", lndArgs)
|
|
h.Bob = h.NewNode("Bob", lndArgs)
|
|
|
|
// Load up the wallets of the seeder nodes with 100 outputs of 1 BTC
|
|
// each.
|
|
const fundAmount = 1 * btcutil.SatoshiPerBitcoin
|
|
const numOutputs = 100
|
|
const totalAmount = fundAmount * numOutputs
|
|
for _, node := range []*node.HarnessNode{h.Alice, h.Bob} {
|
|
h.manager.standbyNodes[node.Cfg.NodeID] = node
|
|
for i := 0; i < numOutputs; i++ {
|
|
h.createAndSendOutput(
|
|
node, fundAmount,
|
|
lnrpc.AddressType_WITNESS_PUBKEY_HASH,
|
|
)
|
|
}
|
|
}
|
|
|
|
// We generate several blocks in order to give the outputs created
|
|
// above a good number of confirmations.
|
|
const totalTxes = 200
|
|
h.MineBlocksAndAssertNumTxes(numBlocksSendOutput, totalTxes)
|
|
|
|
// Now we want to wait for the nodes to catch up.
|
|
h.WaitForBlockchainSync(h.Alice)
|
|
h.WaitForBlockchainSync(h.Bob)
|
|
|
|
// Now block until both wallets have fully synced up.
|
|
h.WaitForBalanceConfirmed(h.Alice, totalAmount)
|
|
h.WaitForBalanceConfirmed(h.Bob, totalAmount)
|
|
}
|
|
|
|
// Stop stops the test harness.
|
|
func (h *HarnessTest) Stop() {
|
|
// Do nothing if it's not started.
|
|
if h.runCtx == nil {
|
|
h.Log("HarnessTest is not started")
|
|
return
|
|
}
|
|
|
|
close(h.lndErrorChan)
|
|
|
|
// Stop the fee service.
|
|
err := h.feeService.Stop()
|
|
require.NoError(h, err, "failed to stop fee service")
|
|
|
|
// Stop the chainBackend.
|
|
h.stopChainBackend()
|
|
|
|
// Stop the miner.
|
|
h.miner.Stop()
|
|
}
|
|
|
|
// RunTestCase executes a harness test case. Any errors or panics will be
|
|
// represented as fatal.
|
|
func (h *HarnessTest) RunTestCase(testCase *TestCase) {
|
|
defer func() {
|
|
if err := recover(); err != nil {
|
|
description := errors.Wrap(err, 2).ErrorStack()
|
|
h.Fatalf("Failed: (%v) panic with: \n%v",
|
|
testCase.Name, description)
|
|
}
|
|
}()
|
|
|
|
testCase.TestFunc(h)
|
|
}
|
|
|
|
// resetStandbyNodes resets all standby nodes by attaching the new testing.T
|
|
// and restarting them with the original config.
|
|
func (h *HarnessTest) resetStandbyNodes(t *testing.T) {
|
|
t.Helper()
|
|
|
|
for _, hn := range h.manager.standbyNodes {
|
|
// Inherit the testing.T.
|
|
h.T = t
|
|
|
|
// Reset the config so the node will be using the default
|
|
// config for the coming test. This will also inherit the
|
|
// test's running context.
|
|
h.RestartNodeWithExtraArgs(hn, hn.Cfg.OriginalExtraArgs)
|
|
|
|
hn.AddToLogf("Finished test case %v", h.manager.currentTestCase)
|
|
}
|
|
}
|
|
|
|
// Subtest creates a child HarnessTest, which inherits the harness net and
|
|
// stand by nodes created by the parent test. It will return a cleanup function
|
|
// which resets all the standby nodes' configs back to its original state and
|
|
// create snapshots of each nodes' internal state.
|
|
func (h *HarnessTest) Subtest(t *testing.T) *HarnessTest {
|
|
t.Helper()
|
|
|
|
st := &HarnessTest{
|
|
T: t,
|
|
manager: h.manager,
|
|
miner: h.miner,
|
|
standbyNodes: h.standbyNodes,
|
|
feeService: h.feeService,
|
|
lndErrorChan: make(chan error, lndErrorChanSize),
|
|
}
|
|
|
|
// Inherit context from the main test.
|
|
st.runCtx, st.cancel = context.WithCancel(h.runCtx)
|
|
|
|
// Inherit the subtest for the miner.
|
|
st.miner.T = st.T
|
|
|
|
// Reset the standby nodes.
|
|
st.resetStandbyNodes(t)
|
|
|
|
// Reset fee estimator.
|
|
st.feeService.Reset()
|
|
|
|
// Record block height.
|
|
h.updateCurrentHeight()
|
|
startHeight := int32(h.CurrentHeight())
|
|
|
|
st.Cleanup(func() {
|
|
_, endHeight := h.GetBestBlock()
|
|
|
|
st.Logf("finished test: %s, start height=%d, end height=%d, "+
|
|
"mined blocks=%d", st.manager.currentTestCase,
|
|
startHeight, endHeight, endHeight-startHeight)
|
|
|
|
// Don't bother run the cleanups if the test is failed.
|
|
if st.Failed() {
|
|
st.Log("test failed, skipped cleanup")
|
|
st.shutdownAllNodes()
|
|
return
|
|
}
|
|
|
|
// Don't run cleanup if it's already done. This can happen if
|
|
// we have multiple level inheritance of the parent harness
|
|
// test. For instance, a `Subtest(st)`.
|
|
if st.cleaned {
|
|
st.Log("test already cleaned, skipped cleanup")
|
|
return
|
|
}
|
|
|
|
// When we finish the test, reset the nodes' configs and take a
|
|
// snapshot of each of the nodes' internal states.
|
|
for _, node := range st.manager.standbyNodes {
|
|
st.cleanupStandbyNode(node)
|
|
}
|
|
|
|
// If found running nodes, shut them down.
|
|
st.shutdownNonStandbyNodes()
|
|
|
|
// We require the mempool to be cleaned from the test.
|
|
require.Empty(st, st.miner.GetRawMempool(), "mempool not "+
|
|
"cleaned, please mine blocks to clean them all.")
|
|
|
|
// Finally, cancel the run context. We have to do it here
|
|
// because we need to keep the context alive for the above
|
|
// assertions used in cleanup.
|
|
st.cancel()
|
|
|
|
// We now want to mark the parent harness as cleaned to avoid
|
|
// running cleanup again since its internal state has been
|
|
// cleaned up by its child harness tests.
|
|
h.cleaned = true
|
|
})
|
|
|
|
return st
|
|
}
|
|
|
|
// shutdownNonStandbyNodes will shutdown any non-standby nodes.
|
|
func (h *HarnessTest) shutdownNonStandbyNodes() {
|
|
h.shutdownNodes(true)
|
|
}
|
|
|
|
// shutdownAllNodes will shutdown all running nodes.
|
|
func (h *HarnessTest) shutdownAllNodes() {
|
|
h.shutdownNodes(false)
|
|
}
|
|
|
|
// shutdownNodes will shutdown any non-standby nodes. If skipStandby is false,
|
|
// all the standby nodes will be shutdown too.
|
|
func (h *HarnessTest) shutdownNodes(skipStandby bool) {
|
|
for nid, node := range h.manager.activeNodes {
|
|
// If it's a standby node, skip.
|
|
_, ok := h.manager.standbyNodes[nid]
|
|
if ok && skipStandby {
|
|
continue
|
|
}
|
|
|
|
// The process may not be in a state to always shutdown
|
|
// immediately, so we'll retry up to a hard limit to ensure we
|
|
// eventually shutdown.
|
|
err := wait.NoError(func() error {
|
|
return h.manager.shutdownNode(node)
|
|
}, DefaultTimeout)
|
|
|
|
if err == nil {
|
|
continue
|
|
}
|
|
|
|
// Instead of returning the error, we will log it instead. This
|
|
// is needed so other nodes can continue their shutdown
|
|
// processes.
|
|
h.Logf("unable to shutdown %s, got err: %v", node.Name(), err)
|
|
}
|
|
}
|
|
|
|
// cleanupStandbyNode is a function should be called with defer whenever a
|
|
// subtest is created. It will reset the standby nodes configs, snapshot the
|
|
// states, and validate the node has a clean state.
|
|
func (h *HarnessTest) cleanupStandbyNode(hn *node.HarnessNode) {
|
|
// Remove connections made from this test.
|
|
h.removeConnectionns(hn)
|
|
|
|
// Delete all payments made from this test.
|
|
hn.RPC.DeleteAllPayments()
|
|
|
|
// Check the node's current state with timeout.
|
|
//
|
|
// NOTE: we need to do this in a `wait` because it takes some time for
|
|
// the node to update its internal state. Once the RPCs are synced we
|
|
// can then remove this wait.
|
|
err := wait.NoError(func() error {
|
|
// Update the node's internal state.
|
|
hn.UpdateState()
|
|
|
|
// Check the node is in a clean state for the following tests.
|
|
return h.validateNodeState(hn)
|
|
}, wait.DefaultTimeout)
|
|
require.NoError(h, err, "timeout checking node's state")
|
|
}
|
|
|
|
// removeConnectionns will remove all connections made on the standby nodes
|
|
// expect the connections between Alice and Bob.
|
|
func (h *HarnessTest) removeConnectionns(hn *node.HarnessNode) {
|
|
resp := hn.RPC.ListPeers()
|
|
for _, peer := range resp.Peers {
|
|
// Skip disconnecting Alice and Bob.
|
|
switch peer.PubKey {
|
|
case h.Alice.PubKeyStr:
|
|
continue
|
|
case h.Bob.PubKeyStr:
|
|
continue
|
|
}
|
|
|
|
hn.RPC.DisconnectPeer(peer.PubKey)
|
|
}
|
|
}
|
|
|
|
// SetTestName set the test case name.
|
|
func (h *HarnessTest) SetTestName(name string) {
|
|
h.manager.currentTestCase = name
|
|
|
|
// Overwrite the old log filename so we can create new log files.
|
|
for _, node := range h.manager.standbyNodes {
|
|
node.Cfg.LogFilenamePrefix = name
|
|
}
|
|
}
|
|
|
|
// NewNode creates a new node and asserts its creation. The node is guaranteed
|
|
// to have finished its initialization and all its subservers are started.
|
|
func (h *HarnessTest) NewNode(name string,
|
|
extraArgs []string) *node.HarnessNode {
|
|
|
|
node, err := h.manager.newNode(h.T, name, extraArgs, nil, false)
|
|
require.NoErrorf(h, err, "unable to create new node for %s", name)
|
|
|
|
// Start the node.
|
|
err = node.Start(h.runCtx)
|
|
require.NoError(h, err, "failed to start node %s", node.Name())
|
|
|
|
return node
|
|
}
|
|
|
|
// Shutdown shuts down the given node and asserts that no errors occur.
|
|
func (h *HarnessTest) Shutdown(node *node.HarnessNode) {
|
|
// The process may not be in a state to always shutdown immediately, so
|
|
// we'll retry up to a hard limit to ensure we eventually shutdown.
|
|
err := wait.NoError(func() error {
|
|
return h.manager.shutdownNode(node)
|
|
}, DefaultTimeout)
|
|
|
|
require.NoErrorf(h, err, "unable to shutdown %v in %v", node.Name(),
|
|
h.manager.currentTestCase)
|
|
}
|
|
|
|
// SuspendNode stops the given node and returns a callback that can be used to
|
|
// start it again.
|
|
func (h *HarnessTest) SuspendNode(node *node.HarnessNode) func() error {
|
|
err := node.Stop()
|
|
require.NoErrorf(h, err, "failed to stop %s", node.Name())
|
|
|
|
// Remove the node from active nodes.
|
|
delete(h.manager.activeNodes, node.Cfg.NodeID)
|
|
|
|
return func() error {
|
|
h.manager.registerNode(node)
|
|
|
|
if err := node.Start(h.runCtx); err != nil {
|
|
return err
|
|
}
|
|
h.WaitForBlockchainSync(node)
|
|
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// RestartNode restarts a given node, unlocks it and asserts it's successfully
|
|
// started.
|
|
func (h *HarnessTest) RestartNode(hn *node.HarnessNode) {
|
|
err := h.manager.restartNode(h.runCtx, hn, nil)
|
|
require.NoErrorf(h, err, "failed to restart node %s", hn.Name())
|
|
|
|
err = h.manager.unlockNode(hn)
|
|
require.NoErrorf(h, err, "failed to unlock node %s", hn.Name())
|
|
|
|
if !hn.Cfg.SkipUnlock {
|
|
// Give the node some time to catch up with the chain before we
|
|
// continue with the tests.
|
|
h.WaitForBlockchainSync(hn)
|
|
}
|
|
}
|
|
|
|
// RestartNodeNoUnlock restarts a given node without unlocking its wallet.
|
|
func (h *HarnessTest) RestartNodeNoUnlock(hn *node.HarnessNode) {
|
|
err := h.manager.restartNode(h.runCtx, hn, nil)
|
|
require.NoErrorf(h, err, "failed to restart node %s", hn.Name())
|
|
}
|
|
|
|
// RestartNodeWithChanBackups restarts a given node with the specified channel
|
|
// backups.
|
|
func (h *HarnessTest) RestartNodeWithChanBackups(hn *node.HarnessNode,
|
|
chanBackups ...*lnrpc.ChanBackupSnapshot) {
|
|
|
|
err := h.manager.restartNode(h.runCtx, hn, nil)
|
|
require.NoErrorf(h, err, "failed to restart node %s", hn.Name())
|
|
|
|
err = h.manager.unlockNode(hn, chanBackups...)
|
|
require.NoErrorf(h, err, "failed to unlock node %s", hn.Name())
|
|
|
|
// Give the node some time to catch up with the chain before we
|
|
// continue with the tests.
|
|
h.WaitForBlockchainSync(hn)
|
|
}
|
|
|
|
// RestartNodeWithExtraArgs updates the node's config and restarts it.
|
|
func (h *HarnessTest) RestartNodeWithExtraArgs(hn *node.HarnessNode,
|
|
extraArgs []string) {
|
|
|
|
hn.SetExtraArgs(extraArgs)
|
|
h.RestartNode(hn)
|
|
}
|
|
|
|
// NewNodeWithSeed fully initializes a new HarnessNode after creating a fresh
|
|
// aezeed. The provided password is used as both the aezeed password and the
|
|
// wallet password. The generated mnemonic is returned along with the
|
|
// initialized harness node.
|
|
func (h *HarnessTest) NewNodeWithSeed(name string,
|
|
extraArgs []string, password []byte,
|
|
statelessInit bool) (*node.HarnessNode, []string, []byte) {
|
|
|
|
// Create a request to generate a new aezeed. The new seed will have
|
|
// the same password as the internal wallet.
|
|
req := &lnrpc.GenSeedRequest{
|
|
AezeedPassphrase: password,
|
|
SeedEntropy: nil,
|
|
}
|
|
|
|
return h.newNodeWithSeed(name, extraArgs, req, statelessInit)
|
|
}
|
|
|
|
// newNodeWithSeed creates and initializes a new HarnessNode such that it'll be
|
|
// ready to accept RPC calls. A `GenSeedRequest` is needed to generate the
|
|
// seed.
|
|
func (h *HarnessTest) newNodeWithSeed(name string,
|
|
extraArgs []string, req *lnrpc.GenSeedRequest,
|
|
statelessInit bool) (*node.HarnessNode, []string, []byte) {
|
|
|
|
node, err := h.manager.newNode(
|
|
h.T, name, extraArgs, req.AezeedPassphrase, true,
|
|
)
|
|
require.NoErrorf(h, err, "unable to create new node for %s", name)
|
|
|
|
// Start the node with seed only, which will only create the `State`
|
|
// and `WalletUnlocker` clients.
|
|
err = node.StartWithNoAuth(h.runCtx)
|
|
require.NoErrorf(h, err, "failed to start node %s", node.Name())
|
|
|
|
// Generate a new seed.
|
|
genSeedResp := node.RPC.GenSeed(req)
|
|
|
|
// With the seed created, construct the init request to the node,
|
|
// including the newly generated seed.
|
|
initReq := &lnrpc.InitWalletRequest{
|
|
WalletPassword: req.AezeedPassphrase,
|
|
CipherSeedMnemonic: genSeedResp.CipherSeedMnemonic,
|
|
AezeedPassphrase: req.AezeedPassphrase,
|
|
StatelessInit: statelessInit,
|
|
}
|
|
|
|
// Pass the init request via rpc to finish unlocking the node. This
|
|
// will also initialize the macaroon-authenticated LightningClient.
|
|
adminMac, err := h.manager.initWalletAndNode(node, initReq)
|
|
require.NoErrorf(h, err, "failed to unlock and init node %s",
|
|
node.Name())
|
|
|
|
// In stateless initialization mode we get a macaroon back that we have
|
|
// to return to the test, otherwise gRPC calls won't be possible since
|
|
// there are no macaroon files created in that mode.
|
|
// In stateful init the admin macaroon will just be nil.
|
|
return node, genSeedResp.CipherSeedMnemonic, adminMac
|
|
}
|
|
|
|
// RestoreNodeWithSeed fully initializes a HarnessNode using a chosen mnemonic,
|
|
// password, recovery window, and optionally a set of static channel backups.
|
|
// After providing the initialization request to unlock the node, this method
|
|
// will finish initializing the LightningClient such that the HarnessNode can
|
|
// be used for regular rpc operations.
|
|
func (h *HarnessTest) RestoreNodeWithSeed(name string, extraArgs []string,
|
|
password []byte, mnemonic []string, rootKey string,
|
|
recoveryWindow int32,
|
|
chanBackups *lnrpc.ChanBackupSnapshot) *node.HarnessNode {
|
|
|
|
n, err := h.manager.newNode(h.T, name, extraArgs, password, true)
|
|
require.NoErrorf(h, err, "unable to create new node for %s", name)
|
|
|
|
// Start the node with seed only, which will only create the `State`
|
|
// and `WalletUnlocker` clients.
|
|
err = n.StartWithNoAuth(h.runCtx)
|
|
require.NoErrorf(h, err, "failed to start node %s", n.Name())
|
|
|
|
// Create the wallet.
|
|
initReq := &lnrpc.InitWalletRequest{
|
|
WalletPassword: password,
|
|
CipherSeedMnemonic: mnemonic,
|
|
AezeedPassphrase: password,
|
|
ExtendedMasterKey: rootKey,
|
|
RecoveryWindow: recoveryWindow,
|
|
ChannelBackups: chanBackups,
|
|
}
|
|
_, err = h.manager.initWalletAndNode(n, initReq)
|
|
require.NoErrorf(h, err, "failed to unlock and init node %s",
|
|
n.Name())
|
|
|
|
return n
|
|
}
|
|
|
|
// NewNodeEtcd starts a new node with seed that'll use an external etcd
|
|
// database as its storage. The passed cluster flag indicates that we'd like
|
|
// the node to join the cluster leader election. We won't wait until RPC is
|
|
// available (this is useful when the node is not expected to become the leader
|
|
// right away).
|
|
func (h *HarnessTest) NewNodeEtcd(name string, etcdCfg *etcd.Config,
|
|
password []byte, cluster bool,
|
|
leaderSessionTTL int) *node.HarnessNode {
|
|
|
|
// We don't want to use the embedded etcd instance.
|
|
h.manager.dbBackend = node.BackendBbolt
|
|
|
|
extraArgs := node.ExtraArgsEtcd(
|
|
etcdCfg, name, cluster, leaderSessionTTL,
|
|
)
|
|
node, err := h.manager.newNode(h.T, name, extraArgs, password, true)
|
|
require.NoError(h, err, "failed to create new node with etcd")
|
|
|
|
// Start the node daemon only.
|
|
err = node.StartLndCmd(h.runCtx)
|
|
require.NoError(h, err, "failed to start node %s", node.Name())
|
|
|
|
return node
|
|
}
|
|
|
|
// NewNodeWithSeedEtcd starts a new node with seed that'll use an external etcd
|
|
// database as its storage. The passed cluster flag indicates that we'd like
|
|
// the node to join the cluster leader election.
|
|
func (h *HarnessTest) NewNodeWithSeedEtcd(name string, etcdCfg *etcd.Config,
|
|
password []byte, statelessInit, cluster bool,
|
|
leaderSessionTTL int) (*node.HarnessNode, []string, []byte) {
|
|
|
|
// We don't want to use the embedded etcd instance.
|
|
h.manager.dbBackend = node.BackendBbolt
|
|
|
|
// Create a request to generate a new aezeed. The new seed will have
|
|
// the same password as the internal wallet.
|
|
req := &lnrpc.GenSeedRequest{
|
|
AezeedPassphrase: password,
|
|
SeedEntropy: nil,
|
|
}
|
|
|
|
extraArgs := node.ExtraArgsEtcd(
|
|
etcdCfg, name, cluster, leaderSessionTTL,
|
|
)
|
|
|
|
return h.newNodeWithSeed(name, extraArgs, req, statelessInit)
|
|
}
|
|
|
|
// NewNodeRemoteSigner creates a new remote signer node and asserts its
|
|
// creation.
|
|
func (h *HarnessTest) NewNodeRemoteSigner(name string, extraArgs []string,
|
|
password []byte, watchOnly *lnrpc.WatchOnly) *node.HarnessNode {
|
|
|
|
hn, err := h.manager.newNode(h.T, name, extraArgs, password, true)
|
|
require.NoErrorf(h, err, "unable to create new node for %s", name)
|
|
|
|
err = hn.StartWithNoAuth(h.runCtx)
|
|
require.NoError(h, err, "failed to start node %s", name)
|
|
|
|
// With the seed created, construct the init request to the node,
|
|
// including the newly generated seed.
|
|
initReq := &lnrpc.InitWalletRequest{
|
|
WalletPassword: password,
|
|
WatchOnly: watchOnly,
|
|
}
|
|
|
|
// Pass the init request via rpc to finish unlocking the node. This
|
|
// will also initialize the macaroon-authenticated LightningClient.
|
|
_, err = h.manager.initWalletAndNode(hn, initReq)
|
|
require.NoErrorf(h, err, "failed to init node %s", name)
|
|
|
|
return hn
|
|
}
|
|
|
|
// KillNode kills the node (but won't wait for the node process to stop).
|
|
func (h *HarnessTest) KillNode(hn *node.HarnessNode) {
|
|
require.NoErrorf(h, hn.Kill(), "%s: kill got error", hn.Name())
|
|
delete(h.manager.activeNodes, hn.Cfg.NodeID)
|
|
}
|
|
|
|
// SetFeeEstimate sets a fee rate to be returned from fee estimator.
|
|
//
|
|
// NOTE: this method will set the fee rate for a conf target of 1, which is the
|
|
// fallback fee rate for a `WebAPIEstimator` if a higher conf target's fee rate
|
|
// is not set. This means if the fee rate for conf target 6 is set, the fee
|
|
// estimator will use that value instead.
|
|
func (h *HarnessTest) SetFeeEstimate(fee chainfee.SatPerKWeight) {
|
|
h.feeService.SetFeeRate(fee, 1)
|
|
}
|
|
|
|
// SetFeeEstimateWithConf sets a fee rate of a specified conf target to be
|
|
// returned from fee estimator.
|
|
func (h *HarnessTest) SetFeeEstimateWithConf(
|
|
fee chainfee.SatPerKWeight, conf uint32) {
|
|
|
|
h.feeService.SetFeeRate(fee, conf)
|
|
}
|
|
|
|
// SetMinRelayFeerate sets a min relay fee rate to be returned from fee
|
|
// estimator.
|
|
func (h *HarnessTest) SetMinRelayFeerate(fee chainfee.SatPerKVByte) {
|
|
h.feeService.SetMinRelayFeerate(fee)
|
|
}
|
|
|
|
// validateNodeState checks that the node doesn't have any uncleaned states
|
|
// which will affect its following tests.
|
|
func (h *HarnessTest) validateNodeState(hn *node.HarnessNode) error {
|
|
errStr := func(subject string) error {
|
|
return fmt.Errorf("%s: found %s channels, please close "+
|
|
"them properly", hn.Name(), subject)
|
|
}
|
|
// If the node still has open channels, it's most likely that the
|
|
// current test didn't close it properly.
|
|
if hn.State.OpenChannel.Active != 0 {
|
|
return errStr("active")
|
|
}
|
|
if hn.State.OpenChannel.Public != 0 {
|
|
return errStr("public")
|
|
}
|
|
if hn.State.OpenChannel.Private != 0 {
|
|
return errStr("private")
|
|
}
|
|
if hn.State.OpenChannel.Pending != 0 {
|
|
return errStr("pending open")
|
|
}
|
|
|
|
// The number of pending force close channels should be zero.
|
|
if hn.State.CloseChannel.PendingForceClose != 0 {
|
|
return errStr("pending force")
|
|
}
|
|
|
|
// The number of waiting close channels should be zero.
|
|
if hn.State.CloseChannel.WaitingClose != 0 {
|
|
return errStr("waiting close")
|
|
}
|
|
|
|
// Ths number of payments should be zero.
|
|
if hn.State.Payment.Total != 0 {
|
|
return fmt.Errorf("%s: found uncleaned payments, please "+
|
|
"delete all of them properly", hn.Name())
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// GetChanPointFundingTxid takes a channel point and converts it into a chain
|
|
// hash.
|
|
func (h *HarnessTest) GetChanPointFundingTxid(
|
|
cp *lnrpc.ChannelPoint) *chainhash.Hash {
|
|
|
|
txid, err := lnrpc.GetChanPointFundingTxid(cp)
|
|
require.NoError(h, err, "unable to get txid")
|
|
|
|
return txid
|
|
}
|
|
|
|
// OutPointFromChannelPoint creates an outpoint from a given channel point.
|
|
func (h *HarnessTest) OutPointFromChannelPoint(
|
|
cp *lnrpc.ChannelPoint) wire.OutPoint {
|
|
|
|
txid := h.GetChanPointFundingTxid(cp)
|
|
return wire.OutPoint{
|
|
Hash: *txid,
|
|
Index: cp.OutputIndex,
|
|
}
|
|
}
|
|
|
|
// OpenChannelParams houses the params to specify when opening a new channel.
|
|
type OpenChannelParams struct {
|
|
// Amt is the local amount being put into the channel.
|
|
Amt btcutil.Amount
|
|
|
|
// PushAmt is the amount that should be pushed to the remote when the
|
|
// channel is opened.
|
|
PushAmt btcutil.Amount
|
|
|
|
// Private is a boolan indicating whether the opened channel should be
|
|
// private.
|
|
Private bool
|
|
|
|
// SpendUnconfirmed is a boolean indicating whether we can utilize
|
|
// unconfirmed outputs to fund the channel.
|
|
SpendUnconfirmed bool
|
|
|
|
// MinHtlc is the htlc_minimum_msat value set when opening the channel.
|
|
MinHtlc lnwire.MilliSatoshi
|
|
|
|
// RemoteMaxHtlcs is the remote_max_htlcs value set when opening the
|
|
// channel, restricting the number of concurrent HTLCs the remote party
|
|
// can add to a commitment.
|
|
RemoteMaxHtlcs uint16
|
|
|
|
// FundingShim is an optional funding shim that the caller can specify
|
|
// in order to modify the channel funding workflow.
|
|
FundingShim *lnrpc.FundingShim
|
|
|
|
// SatPerVByte is the amount of satoshis to spend in chain fees per
|
|
// virtual byte of the transaction.
|
|
SatPerVByte btcutil.Amount
|
|
|
|
// ConfTarget is the number of blocks that the funding transaction
|
|
// should be confirmed in.
|
|
ConfTarget fn.Option[int32]
|
|
|
|
// CommitmentType is the commitment type that should be used for the
|
|
// channel to be opened.
|
|
CommitmentType lnrpc.CommitmentType
|
|
|
|
// ZeroConf is used to determine if the channel will be a zero-conf
|
|
// channel. This only works if the explicit negotiation is used with
|
|
// anchors or script enforced leases.
|
|
ZeroConf bool
|
|
|
|
// ScidAlias denotes whether the channel will be an option-scid-alias
|
|
// channel type negotiation.
|
|
ScidAlias bool
|
|
|
|
// BaseFee is the channel base fee applied during the channel
|
|
// announcement phase.
|
|
BaseFee uint64
|
|
|
|
// FeeRate is the channel fee rate in ppm applied during the channel
|
|
// announcement phase.
|
|
FeeRate uint64
|
|
|
|
// UseBaseFee, if set, instructs the downstream logic to apply the
|
|
// user-specified channel base fee to the channel update announcement.
|
|
// If set to false it avoids applying a base fee of 0 and instead
|
|
// activates the default configured base fee.
|
|
UseBaseFee bool
|
|
|
|
// UseFeeRate, if set, instructs the downstream logic to apply the
|
|
// user-specified channel fee rate to the channel update announcement.
|
|
// If set to false it avoids applying a fee rate of 0 and instead
|
|
// activates the default configured fee rate.
|
|
UseFeeRate bool
|
|
|
|
// FundMax is a boolean indicating whether the channel should be funded
|
|
// with the maximum possible amount from the wallet.
|
|
FundMax bool
|
|
|
|
// An optional note-to-self containing some useful information about the
|
|
// channel. This is stored locally only, and is purely for reference. It
|
|
// has no bearing on the channel's operation. Max allowed length is 500
|
|
// characters.
|
|
Memo string
|
|
|
|
// Outpoints is a list of client-selected outpoints that should be used
|
|
// for funding a channel. If Amt is specified then this amount is
|
|
// allocated from the sum of outpoints towards funding. If the
|
|
// FundMax flag is specified the entirety of selected funds is
|
|
// allocated towards channel funding.
|
|
Outpoints []*lnrpc.OutPoint
|
|
|
|
// CloseAddress sets the upfront_shutdown_script parameter during
|
|
// channel open. It is expected to be encoded as a bitcoin address.
|
|
CloseAddress string
|
|
}
|
|
|
|
// prepareOpenChannel waits for both nodes to be synced to chain and returns an
|
|
// OpenChannelRequest.
|
|
func (h *HarnessTest) prepareOpenChannel(srcNode, destNode *node.HarnessNode,
|
|
p OpenChannelParams) *lnrpc.OpenChannelRequest {
|
|
|
|
// Wait until srcNode and destNode have the latest chain synced.
|
|
// Otherwise, we may run into a check within the funding manager that
|
|
// prevents any funding workflows from being kicked off if the chain
|
|
// isn't yet synced.
|
|
h.WaitForBlockchainSync(srcNode)
|
|
h.WaitForBlockchainSync(destNode)
|
|
|
|
// Specify the minimal confirmations of the UTXOs used for channel
|
|
// funding.
|
|
minConfs := int32(1)
|
|
if p.SpendUnconfirmed {
|
|
minConfs = 0
|
|
}
|
|
|
|
// Get the requested conf target. If not set, default to 6.
|
|
confTarget := p.ConfTarget.UnwrapOr(6)
|
|
|
|
// If there's fee rate set, unset the conf target.
|
|
if p.SatPerVByte != 0 {
|
|
confTarget = 0
|
|
}
|
|
|
|
// Prepare the request.
|
|
return &lnrpc.OpenChannelRequest{
|
|
NodePubkey: destNode.PubKey[:],
|
|
LocalFundingAmount: int64(p.Amt),
|
|
PushSat: int64(p.PushAmt),
|
|
Private: p.Private,
|
|
TargetConf: confTarget,
|
|
MinConfs: minConfs,
|
|
SpendUnconfirmed: p.SpendUnconfirmed,
|
|
MinHtlcMsat: int64(p.MinHtlc),
|
|
RemoteMaxHtlcs: uint32(p.RemoteMaxHtlcs),
|
|
FundingShim: p.FundingShim,
|
|
SatPerVbyte: uint64(p.SatPerVByte),
|
|
CommitmentType: p.CommitmentType,
|
|
ZeroConf: p.ZeroConf,
|
|
ScidAlias: p.ScidAlias,
|
|
BaseFee: p.BaseFee,
|
|
FeeRate: p.FeeRate,
|
|
UseBaseFee: p.UseBaseFee,
|
|
UseFeeRate: p.UseFeeRate,
|
|
FundMax: p.FundMax,
|
|
Memo: p.Memo,
|
|
Outpoints: p.Outpoints,
|
|
CloseAddress: p.CloseAddress,
|
|
}
|
|
}
|
|
|
|
// OpenChannelAssertPending attempts to open a channel between srcNode and
|
|
// destNode with the passed channel funding parameters. Once the `OpenChannel`
|
|
// is called, it will consume the first event it receives from the open channel
|
|
// client and asserts it's a channel pending event.
|
|
func (h *HarnessTest) openChannelAssertPending(srcNode,
|
|
destNode *node.HarnessNode,
|
|
p OpenChannelParams) (*lnrpc.PendingUpdate, rpc.OpenChanClient) {
|
|
|
|
// Prepare the request and open the channel.
|
|
openReq := h.prepareOpenChannel(srcNode, destNode, p)
|
|
respStream := srcNode.RPC.OpenChannel(openReq)
|
|
|
|
// Consume the "channel pending" update. This waits until the node
|
|
// notifies us that the final message in the channel funding workflow
|
|
// has been sent to the remote node.
|
|
resp := h.ReceiveOpenChannelUpdate(respStream)
|
|
|
|
// Check that the update is channel pending.
|
|
update, ok := resp.Update.(*lnrpc.OpenStatusUpdate_ChanPending)
|
|
require.Truef(h, ok, "expected channel pending: update, instead got %v",
|
|
resp)
|
|
|
|
return update.ChanPending, respStream
|
|
}
|
|
|
|
// OpenChannelAssertPending attempts to open a channel between srcNode and
|
|
// destNode with the passed channel funding parameters. Once the `OpenChannel`
|
|
// is called, it will consume the first event it receives from the open channel
|
|
// client and asserts it's a channel pending event. It returns the
|
|
// `PendingUpdate`.
|
|
func (h *HarnessTest) OpenChannelAssertPending(srcNode,
|
|
destNode *node.HarnessNode, p OpenChannelParams) *lnrpc.PendingUpdate {
|
|
|
|
resp, _ := h.openChannelAssertPending(srcNode, destNode, p)
|
|
return resp
|
|
}
|
|
|
|
// OpenChannelAssertStream attempts to open a channel between srcNode and
|
|
// destNode with the passed channel funding parameters. Once the `OpenChannel`
|
|
// is called, it will consume the first event it receives from the open channel
|
|
// client and asserts it's a channel pending event. It returns the open channel
|
|
// stream.
|
|
func (h *HarnessTest) OpenChannelAssertStream(srcNode,
|
|
destNode *node.HarnessNode, p OpenChannelParams) rpc.OpenChanClient {
|
|
|
|
_, stream := h.openChannelAssertPending(srcNode, destNode, p)
|
|
return stream
|
|
}
|
|
|
|
// OpenChannel attempts to open a channel with the specified parameters
|
|
// extended from Alice to Bob. Additionally, for public channels, it will mine
|
|
// extra blocks so they are announced to the network. In specific, the
|
|
// following items are asserted,
|
|
// - for non-zero conf channel, 1 blocks will be mined to confirm the funding
|
|
// tx.
|
|
// - both nodes should see the channel edge update in their network graph.
|
|
// - both nodes can report the status of the new channel from ListChannels.
|
|
// - extra blocks are mined if it's a public channel.
|
|
func (h *HarnessTest) OpenChannel(alice, bob *node.HarnessNode,
|
|
p OpenChannelParams) *lnrpc.ChannelPoint {
|
|
|
|
// First, open the channel without announcing it.
|
|
cp := h.OpenChannelNoAnnounce(alice, bob, p)
|
|
|
|
// If this is a private channel, there's no need to mine extra blocks
|
|
// since it will never be announced to the network.
|
|
if p.Private {
|
|
return cp
|
|
}
|
|
|
|
// Mine extra blocks to announce the channel.
|
|
if p.ZeroConf {
|
|
// For a zero-conf channel, no blocks have been mined so we
|
|
// need to mine 6 blocks.
|
|
//
|
|
// Mine 1 block to confirm the funding transaction.
|
|
h.MineBlocksAndAssertNumTxes(numBlocksOpenChannel, 1)
|
|
} else {
|
|
// For a regular channel, 1 block has already been mined to
|
|
// confirm the funding transaction, so we mine 5 blocks.
|
|
h.MineBlocks(numBlocksOpenChannel - 1)
|
|
}
|
|
|
|
return cp
|
|
}
|
|
|
|
// OpenChannelNoAnnounce attempts to open a channel with the specified
|
|
// parameters extended from Alice to Bob without mining the necessary blocks to
|
|
// announce the channel. Additionally, the following items are asserted,
|
|
// - for non-zero conf channel, 1 blocks will be mined to confirm the funding
|
|
// tx.
|
|
// - both nodes should see the channel edge update in their network graph.
|
|
// - both nodes can report the status of the new channel from ListChannels.
|
|
func (h *HarnessTest) OpenChannelNoAnnounce(alice, bob *node.HarnessNode,
|
|
p OpenChannelParams) *lnrpc.ChannelPoint {
|
|
|
|
chanOpenUpdate := h.OpenChannelAssertStream(alice, bob, p)
|
|
|
|
// Open a zero conf channel.
|
|
if p.ZeroConf {
|
|
return h.openChannelZeroConf(alice, bob, chanOpenUpdate)
|
|
}
|
|
|
|
// Open a non-zero conf channel.
|
|
return h.openChannel(alice, bob, chanOpenUpdate)
|
|
}
|
|
|
|
// openChannel attempts to open a channel with the specified parameters
|
|
// extended from Alice to Bob. Additionally, the following items are asserted,
|
|
// - 1 block is mined and the funding transaction should be found in it.
|
|
// - both nodes should see the channel edge update in their network graph.
|
|
// - both nodes can report the status of the new channel from ListChannels.
|
|
func (h *HarnessTest) openChannel(alice, bob *node.HarnessNode,
|
|
stream rpc.OpenChanClient) *lnrpc.ChannelPoint {
|
|
|
|
// Mine 1 block to confirm the funding transaction.
|
|
block := h.MineBlocksAndAssertNumTxes(1, 1)[0]
|
|
|
|
// Wait for the channel open event.
|
|
fundingChanPoint := h.WaitForChannelOpenEvent(stream)
|
|
|
|
// Check that the funding tx is found in the first block.
|
|
fundingTxID := h.GetChanPointFundingTxid(fundingChanPoint)
|
|
h.AssertTxInBlock(block, fundingTxID)
|
|
|
|
// Check that both alice and bob have seen the channel from their
|
|
// network topology.
|
|
h.AssertTopologyChannelOpen(alice, fundingChanPoint)
|
|
h.AssertTopologyChannelOpen(bob, fundingChanPoint)
|
|
|
|
// Check that the channel can be seen in their ListChannels.
|
|
h.AssertChannelExists(alice, fundingChanPoint)
|
|
h.AssertChannelExists(bob, fundingChanPoint)
|
|
|
|
return fundingChanPoint
|
|
}
|
|
|
|
// openChannelZeroConf attempts to open a channel with the specified parameters
|
|
// extended from Alice to Bob. Additionally, the following items are asserted,
|
|
// - both nodes should see the channel edge update in their network graph.
|
|
// - both nodes can report the status of the new channel from ListChannels.
|
|
func (h *HarnessTest) openChannelZeroConf(alice, bob *node.HarnessNode,
|
|
stream rpc.OpenChanClient) *lnrpc.ChannelPoint {
|
|
|
|
// Wait for the channel open event.
|
|
fundingChanPoint := h.WaitForChannelOpenEvent(stream)
|
|
|
|
// Check that both alice and bob have seen the channel from their
|
|
// network topology.
|
|
h.AssertTopologyChannelOpen(alice, fundingChanPoint)
|
|
h.AssertTopologyChannelOpen(bob, fundingChanPoint)
|
|
|
|
// Finally, check that the channel can be seen in their ListChannels.
|
|
h.AssertChannelExists(alice, fundingChanPoint)
|
|
h.AssertChannelExists(bob, fundingChanPoint)
|
|
|
|
return fundingChanPoint
|
|
}
|
|
|
|
// OpenChannelAssertErr opens a channel between node srcNode and destNode,
|
|
// asserts that the expected error is returned from the channel opening.
|
|
func (h *HarnessTest) OpenChannelAssertErr(srcNode, destNode *node.HarnessNode,
|
|
p OpenChannelParams, expectedErr error) {
|
|
|
|
// Prepare the request and open the channel.
|
|
openReq := h.prepareOpenChannel(srcNode, destNode, p)
|
|
respStream := srcNode.RPC.OpenChannel(openReq)
|
|
|
|
// Receive an error to be sent from the stream.
|
|
_, err := h.receiveOpenChannelUpdate(respStream)
|
|
require.NotNil(h, err, "expected channel opening to fail")
|
|
|
|
// Use string comparison here as we haven't codified all the RPC errors
|
|
// yet.
|
|
require.Containsf(h, err.Error(), expectedErr.Error(), "unexpected "+
|
|
"error returned, want %v, got %v", expectedErr, err)
|
|
}
|
|
|
|
// CloseChannelAssertPending attempts to close the channel indicated by the
|
|
// passed channel point, initiated by the passed node. Once the CloseChannel
|
|
// rpc is called, it will consume one event and assert it's a close pending
|
|
// event. In addition, it will check that the closing tx can be found in the
|
|
// mempool.
|
|
func (h *HarnessTest) CloseChannelAssertPending(hn *node.HarnessNode,
|
|
cp *lnrpc.ChannelPoint,
|
|
force bool) (rpc.CloseChanClient, *chainhash.Hash) {
|
|
|
|
// Calls the rpc to close the channel.
|
|
closeReq := &lnrpc.CloseChannelRequest{
|
|
ChannelPoint: cp,
|
|
Force: force,
|
|
NoWait: true,
|
|
}
|
|
|
|
// For coop close, we use a default confg target of 6.
|
|
if !force {
|
|
closeReq.TargetConf = 6
|
|
}
|
|
|
|
var (
|
|
stream rpc.CloseChanClient
|
|
event *lnrpc.CloseStatusUpdate
|
|
err error
|
|
)
|
|
|
|
// Consume the "channel close" update in order to wait for the closing
|
|
// transaction to be broadcast, then wait for the closing tx to be seen
|
|
// within the network.
|
|
stream = hn.RPC.CloseChannel(closeReq)
|
|
_, err = h.ReceiveCloseChannelUpdate(stream)
|
|
require.NoError(h, err, "close channel update got error: %v", err)
|
|
|
|
event, err = h.ReceiveCloseChannelUpdate(stream)
|
|
if err != nil {
|
|
h.Logf("Test: %s, close channel got error: %v",
|
|
h.manager.currentTestCase, err)
|
|
}
|
|
require.NoError(h, err, "retry closing channel failed")
|
|
|
|
pendingClose, ok := event.Update.(*lnrpc.CloseStatusUpdate_ClosePending)
|
|
require.Truef(h, ok, "expected channel close update, instead got %v",
|
|
pendingClose)
|
|
|
|
closeTxid, err := chainhash.NewHash(pendingClose.ClosePending.Txid)
|
|
require.NoErrorf(h, err, "unable to decode closeTxid: %v",
|
|
pendingClose.ClosePending.Txid)
|
|
|
|
// Assert the closing tx is in the mempool.
|
|
h.miner.AssertTxInMempool(closeTxid)
|
|
|
|
return stream, closeTxid
|
|
}
|
|
|
|
// CloseChannel attempts to coop close a non-anchored channel identified by the
|
|
// passed channel point owned by the passed harness node. The following items
|
|
// are asserted,
|
|
// 1. a close pending event is sent from the close channel client.
|
|
// 2. the closing tx is found in the mempool.
|
|
// 3. the node reports the channel being waiting to close.
|
|
// 4. a block is mined and the closing tx should be found in it.
|
|
// 5. the node reports zero waiting close channels.
|
|
// 6. the node receives a topology update regarding the channel close.
|
|
func (h *HarnessTest) CloseChannel(hn *node.HarnessNode,
|
|
cp *lnrpc.ChannelPoint) *chainhash.Hash {
|
|
|
|
stream, _ := h.CloseChannelAssertPending(hn, cp, false)
|
|
|
|
return h.AssertStreamChannelCoopClosed(hn, cp, false, stream)
|
|
}
|
|
|
|
// ForceCloseChannel attempts to force close a non-anchored channel identified
|
|
// by the passed channel point owned by the passed harness node. The following
|
|
// items are asserted,
|
|
// 1. a close pending event is sent from the close channel client.
|
|
// 2. the closing tx is found in the mempool.
|
|
// 3. the node reports the channel being waiting to close.
|
|
// 4. a block is mined and the closing tx should be found in it.
|
|
// 5. the node reports zero waiting close channels.
|
|
// 6. the node receives a topology update regarding the channel close.
|
|
// 7. mine DefaultCSV-1 blocks.
|
|
// 8. the node reports zero pending force close channels.
|
|
func (h *HarnessTest) ForceCloseChannel(hn *node.HarnessNode,
|
|
cp *lnrpc.ChannelPoint) *chainhash.Hash {
|
|
|
|
stream, _ := h.CloseChannelAssertPending(hn, cp, true)
|
|
|
|
closingTxid := h.AssertStreamChannelForceClosed(hn, cp, false, stream)
|
|
|
|
// Cleanup the force close.
|
|
h.CleanupForceClose(hn)
|
|
|
|
return closingTxid
|
|
}
|
|
|
|
// CloseChannelAssertErr closes the given channel and asserts an error
|
|
// returned.
|
|
func (h *HarnessTest) CloseChannelAssertErr(hn *node.HarnessNode,
|
|
cp *lnrpc.ChannelPoint, force bool) error {
|
|
|
|
// Calls the rpc to close the channel.
|
|
closeReq := &lnrpc.CloseChannelRequest{
|
|
ChannelPoint: cp,
|
|
Force: force,
|
|
}
|
|
stream := hn.RPC.CloseChannel(closeReq)
|
|
|
|
// Consume the "channel close" update in order to wait for the closing
|
|
// transaction to be broadcast, then wait for the closing tx to be seen
|
|
// within the network.
|
|
_, err := h.ReceiveCloseChannelUpdate(stream)
|
|
require.Errorf(h, err, "%s: expect close channel to return an error",
|
|
hn.Name())
|
|
|
|
return err
|
|
}
|
|
|
|
// IsNeutrinoBackend returns a bool indicating whether the node is using a
|
|
// neutrino as its backend. This is useful when we want to skip certain tests
|
|
// which cannot be done with a neutrino backend.
|
|
func (h *HarnessTest) IsNeutrinoBackend() bool {
|
|
return h.manager.chainBackend.Name() == NeutrinoBackendName
|
|
}
|
|
|
|
// fundCoins attempts to send amt satoshis from the internal mining node to the
|
|
// targeted lightning node. The confirmed boolean indicates whether the
|
|
// transaction that pays to the target should confirm. For neutrino backend,
|
|
// the `confirmed` param is ignored.
|
|
func (h *HarnessTest) fundCoins(amt btcutil.Amount, target *node.HarnessNode,
|
|
addrType lnrpc.AddressType, confirmed bool) {
|
|
|
|
initialBalance := target.RPC.WalletBalance()
|
|
|
|
// First, obtain an address from the target lightning node, preferring
|
|
// to receive a p2wkh address s.t the output can immediately be used as
|
|
// an input to a funding transaction.
|
|
req := &lnrpc.NewAddressRequest{Type: addrType}
|
|
resp := target.RPC.NewAddress(req)
|
|
addr := h.DecodeAddress(resp.Address)
|
|
addrScript := h.PayToAddrScript(addr)
|
|
|
|
// Generate a transaction which creates an output to the target
|
|
// pkScript of the desired amount.
|
|
output := &wire.TxOut{
|
|
PkScript: addrScript,
|
|
Value: int64(amt),
|
|
}
|
|
h.miner.SendOutput(output, defaultMinerFeeRate)
|
|
|
|
// Encode the pkScript in hex as this the format that it will be
|
|
// returned via rpc.
|
|
expPkScriptStr := hex.EncodeToString(addrScript)
|
|
|
|
// Now, wait for ListUnspent to show the unconfirmed transaction
|
|
// containing the correct pkscript.
|
|
//
|
|
// Since neutrino doesn't support unconfirmed outputs, skip this check.
|
|
if !h.IsNeutrinoBackend() {
|
|
utxos := h.AssertNumUTXOsUnconfirmed(target, 1)
|
|
|
|
// Assert that the lone unconfirmed utxo contains the same
|
|
// pkscript as the output generated above.
|
|
pkScriptStr := utxos[0].PkScript
|
|
require.Equal(h, pkScriptStr, expPkScriptStr,
|
|
"pkscript mismatch")
|
|
|
|
expectedBalance := btcutil.Amount(
|
|
initialBalance.UnconfirmedBalance,
|
|
) + amt
|
|
h.WaitForBalanceUnconfirmed(target, expectedBalance)
|
|
}
|
|
|
|
// If the transaction should remain unconfirmed, then we'll wait until
|
|
// the target node's unconfirmed balance reflects the expected balance
|
|
// and exit.
|
|
if !confirmed {
|
|
return
|
|
}
|
|
|
|
// Otherwise, we'll generate 1 new blocks to ensure the output gains a
|
|
// sufficient number of confirmations and wait for the balance to
|
|
// reflect what's expected.
|
|
h.MineBlocksAndAssertNumTxes(1, 1)
|
|
|
|
expectedBalance := btcutil.Amount(initialBalance.ConfirmedBalance) + amt
|
|
h.WaitForBalanceConfirmed(target, expectedBalance)
|
|
}
|
|
|
|
// FundCoins attempts to send amt satoshis from the internal mining node to the
|
|
// targeted lightning node using a P2WKH address. 2 blocks are mined after in
|
|
// order to confirm the transaction.
|
|
func (h *HarnessTest) FundCoins(amt btcutil.Amount, hn *node.HarnessNode) {
|
|
h.fundCoins(amt, hn, lnrpc.AddressType_WITNESS_PUBKEY_HASH, true)
|
|
}
|
|
|
|
// FundCoinsUnconfirmed attempts to send amt satoshis from the internal mining
|
|
// node to the targeted lightning node using a P2WKH address. No blocks are
|
|
// mined after and the UTXOs are unconfirmed.
|
|
func (h *HarnessTest) FundCoinsUnconfirmed(amt btcutil.Amount,
|
|
hn *node.HarnessNode) {
|
|
|
|
h.fundCoins(amt, hn, lnrpc.AddressType_WITNESS_PUBKEY_HASH, false)
|
|
}
|
|
|
|
// FundCoinsNP2WKH attempts to send amt satoshis from the internal mining node
|
|
// to the targeted lightning node using a NP2WKH address.
|
|
func (h *HarnessTest) FundCoinsNP2WKH(amt btcutil.Amount,
|
|
target *node.HarnessNode) {
|
|
|
|
h.fundCoins(amt, target, lnrpc.AddressType_NESTED_PUBKEY_HASH, true)
|
|
}
|
|
|
|
// FundCoinsP2TR attempts to send amt satoshis from the internal mining node to
|
|
// the targeted lightning node using a P2TR address.
|
|
func (h *HarnessTest) FundCoinsP2TR(amt btcutil.Amount,
|
|
target *node.HarnessNode) {
|
|
|
|
h.fundCoins(amt, target, lnrpc.AddressType_TAPROOT_PUBKEY, true)
|
|
}
|
|
|
|
// completePaymentRequestsAssertStatus sends payments from a node to complete
|
|
// all payment requests. This function does not return until all payments
|
|
// have reached the specified status.
|
|
func (h *HarnessTest) completePaymentRequestsAssertStatus(hn *node.HarnessNode,
|
|
paymentRequests []string, status lnrpc.Payment_PaymentStatus,
|
|
opts ...HarnessOpt) {
|
|
|
|
payOpts := defaultHarnessOpts()
|
|
for _, opt := range opts {
|
|
opt(&payOpts)
|
|
}
|
|
|
|
// Create a buffered chan to signal the results.
|
|
results := make(chan rpc.PaymentClient, len(paymentRequests))
|
|
|
|
// send sends a payment and asserts if it doesn't succeeded.
|
|
send := func(payReq string) {
|
|
req := &routerrpc.SendPaymentRequest{
|
|
PaymentRequest: payReq,
|
|
TimeoutSeconds: int32(wait.PaymentTimeout.Seconds()),
|
|
FeeLimitMsat: noFeeLimitMsat,
|
|
Amp: payOpts.useAMP,
|
|
}
|
|
stream := hn.RPC.SendPayment(req)
|
|
|
|
// Signal sent succeeded.
|
|
results <- stream
|
|
}
|
|
|
|
// Launch all payments simultaneously.
|
|
for _, payReq := range paymentRequests {
|
|
payReqCopy := payReq
|
|
go send(payReqCopy)
|
|
}
|
|
|
|
// Wait for all payments to report the expected status.
|
|
timer := time.After(wait.PaymentTimeout)
|
|
select {
|
|
case stream := <-results:
|
|
h.AssertPaymentStatusFromStream(stream, status)
|
|
|
|
case <-timer:
|
|
require.Fail(h, "timeout", "waiting payment results timeout")
|
|
}
|
|
}
|
|
|
|
// CompletePaymentRequests sends payments from a node to complete all payment
|
|
// requests. This function does not return until all payments successfully
|
|
// complete without errors.
|
|
func (h *HarnessTest) CompletePaymentRequests(hn *node.HarnessNode,
|
|
paymentRequests []string, opts ...HarnessOpt) {
|
|
|
|
h.completePaymentRequestsAssertStatus(
|
|
hn, paymentRequests, lnrpc.Payment_SUCCEEDED, opts...,
|
|
)
|
|
}
|
|
|
|
// CompletePaymentRequestsNoWait sends payments from a node to complete all
|
|
// payment requests without waiting for the results. Instead, it checks the
|
|
// number of updates in the specified channel has increased.
|
|
func (h *HarnessTest) CompletePaymentRequestsNoWait(hn *node.HarnessNode,
|
|
paymentRequests []string, chanPoint *lnrpc.ChannelPoint) {
|
|
|
|
// We start by getting the current state of the client's channels. This
|
|
// is needed to ensure the payments actually have been committed before
|
|
// we return.
|
|
oldResp := h.GetChannelByChanPoint(hn, chanPoint)
|
|
|
|
// Send payments and assert they are in-flight.
|
|
h.completePaymentRequestsAssertStatus(
|
|
hn, paymentRequests, lnrpc.Payment_IN_FLIGHT,
|
|
)
|
|
|
|
// We are not waiting for feedback in the form of a response, but we
|
|
// should still wait long enough for the server to receive and handle
|
|
// the send before cancelling the request. We wait for the number of
|
|
// updates to one of our channels has increased before we return.
|
|
err := wait.NoError(func() error {
|
|
newResp := h.GetChannelByChanPoint(hn, chanPoint)
|
|
|
|
// If this channel has an increased number of updates, we
|
|
// assume the payments are committed, and we can return.
|
|
if newResp.NumUpdates > oldResp.NumUpdates {
|
|
return nil
|
|
}
|
|
|
|
// Otherwise return an error as the NumUpdates are not
|
|
// increased.
|
|
return fmt.Errorf("%s: channel:%v not updated after sending "+
|
|
"payments, old updates: %v, new updates: %v", hn.Name(),
|
|
chanPoint, oldResp.NumUpdates, newResp.NumUpdates)
|
|
}, DefaultTimeout)
|
|
require.NoError(h, err, "timeout while checking for channel updates")
|
|
}
|
|
|
|
// OpenChannelPsbt attempts to open a channel between srcNode and destNode with
|
|
// the passed channel funding parameters. It will assert if the expected step
|
|
// of funding the PSBT is not received from the source node.
|
|
func (h *HarnessTest) OpenChannelPsbt(srcNode, destNode *node.HarnessNode,
|
|
p OpenChannelParams) (rpc.OpenChanClient, []byte) {
|
|
|
|
// Wait until srcNode and destNode have the latest chain synced.
|
|
// Otherwise, we may run into a check within the funding manager that
|
|
// prevents any funding workflows from being kicked off if the chain
|
|
// isn't yet synced.
|
|
h.WaitForBlockchainSync(srcNode)
|
|
h.WaitForBlockchainSync(destNode)
|
|
|
|
// Send the request to open a channel to the source node now. This will
|
|
// open a long-lived stream where we'll receive status updates about
|
|
// the progress of the channel.
|
|
// respStream := h.OpenChannelStreamAndAssert(srcNode, destNode, p)
|
|
req := &lnrpc.OpenChannelRequest{
|
|
NodePubkey: destNode.PubKey[:],
|
|
LocalFundingAmount: int64(p.Amt),
|
|
PushSat: int64(p.PushAmt),
|
|
Private: p.Private,
|
|
SpendUnconfirmed: p.SpendUnconfirmed,
|
|
MinHtlcMsat: int64(p.MinHtlc),
|
|
FundingShim: p.FundingShim,
|
|
CommitmentType: p.CommitmentType,
|
|
}
|
|
respStream := srcNode.RPC.OpenChannel(req)
|
|
|
|
// Consume the "PSBT funding ready" update. This waits until the node
|
|
// notifies us that the PSBT can now be funded.
|
|
resp := h.ReceiveOpenChannelUpdate(respStream)
|
|
upd, ok := resp.Update.(*lnrpc.OpenStatusUpdate_PsbtFund)
|
|
require.Truef(h, ok, "expected PSBT funding update, got %v", resp)
|
|
|
|
// Make sure the channel funding address has the correct type for the
|
|
// given commitment type.
|
|
fundingAddr, err := btcutil.DecodeAddress(
|
|
upd.PsbtFund.FundingAddress, miner.HarnessNetParams,
|
|
)
|
|
require.NoError(h, err)
|
|
|
|
switch p.CommitmentType {
|
|
case lnrpc.CommitmentType_SIMPLE_TAPROOT:
|
|
require.IsType(h, &btcutil.AddressTaproot{}, fundingAddr)
|
|
|
|
default:
|
|
require.IsType(
|
|
h, &btcutil.AddressWitnessScriptHash{}, fundingAddr,
|
|
)
|
|
}
|
|
|
|
return respStream, upd.PsbtFund.Psbt
|
|
}
|
|
|
|
// CleanupForceClose mines blocks to clean up the force close process. This is
|
|
// used for tests that are not asserting the expected behavior is found during
|
|
// the force close process, e.g., num of sweeps, etc. Instead, it provides a
|
|
// shortcut to move the test forward with a clean mempool.
|
|
func (h *HarnessTest) CleanupForceClose(hn *node.HarnessNode) {
|
|
// Wait for the channel to be marked pending force close.
|
|
h.AssertNumPendingForceClose(hn, 1)
|
|
|
|
// Mine blocks to get any second level HTLC resolved. If there are no
|
|
// HTLCs, this will behave like h.AssertNumPendingCloseChannels.
|
|
h.mineTillForceCloseResolved(hn)
|
|
}
|
|
|
|
// CreatePayReqs is a helper method that will create a slice of payment
|
|
// requests for the given node.
|
|
func (h *HarnessTest) CreatePayReqs(hn *node.HarnessNode,
|
|
paymentAmt btcutil.Amount, numInvoices int,
|
|
routeHints ...*lnrpc.RouteHint) ([]string, [][]byte, []*lnrpc.Invoice) {
|
|
|
|
payReqs := make([]string, numInvoices)
|
|
rHashes := make([][]byte, numInvoices)
|
|
invoices := make([]*lnrpc.Invoice, numInvoices)
|
|
for i := 0; i < numInvoices; i++ {
|
|
preimage := h.Random32Bytes()
|
|
|
|
invoice := &lnrpc.Invoice{
|
|
Memo: "testing",
|
|
RPreimage: preimage,
|
|
Value: int64(paymentAmt),
|
|
RouteHints: routeHints,
|
|
}
|
|
resp := hn.RPC.AddInvoice(invoice)
|
|
|
|
// Set the payment address in the invoice so the caller can
|
|
// properly use it.
|
|
invoice.PaymentAddr = resp.PaymentAddr
|
|
|
|
payReqs[i] = resp.PaymentRequest
|
|
rHashes[i] = resp.RHash
|
|
invoices[i] = invoice
|
|
}
|
|
|
|
return payReqs, rHashes, invoices
|
|
}
|
|
|
|
// BackupDB creates a backup of the current database. It will stop the node
|
|
// first, copy the database files, and restart the node.
|
|
func (h *HarnessTest) BackupDB(hn *node.HarnessNode) {
|
|
restart := h.SuspendNode(hn)
|
|
|
|
err := hn.BackupDB()
|
|
require.NoErrorf(h, err, "%s: failed to backup db", hn.Name())
|
|
|
|
err = restart()
|
|
require.NoErrorf(h, err, "%s: failed to restart", hn.Name())
|
|
}
|
|
|
|
// RestartNodeAndRestoreDB restarts a given node with a callback to restore the
|
|
// db.
|
|
func (h *HarnessTest) RestartNodeAndRestoreDB(hn *node.HarnessNode) {
|
|
cb := func() error { return hn.RestoreDB() }
|
|
err := h.manager.restartNode(h.runCtx, hn, cb)
|
|
require.NoErrorf(h, err, "failed to restart node %s", hn.Name())
|
|
|
|
err = h.manager.unlockNode(hn)
|
|
require.NoErrorf(h, err, "failed to unlock node %s", hn.Name())
|
|
|
|
// Give the node some time to catch up with the chain before we
|
|
// continue with the tests.
|
|
h.WaitForBlockchainSync(hn)
|
|
}
|
|
|
|
// CleanShutDown is used to quickly end a test by shutting down all non-standby
|
|
// nodes and mining blocks to empty the mempool.
|
|
//
|
|
// NOTE: this method provides a faster exit for a test that involves force
|
|
// closures as the caller doesn't need to mine all the blocks to make sure the
|
|
// mempool is empty.
|
|
func (h *HarnessTest) CleanShutDown() {
|
|
// First, shutdown all non-standby nodes to prevent new transactions
|
|
// being created and fed into the mempool.
|
|
h.shutdownNonStandbyNodes()
|
|
|
|
// Now mine blocks till the mempool is empty.
|
|
h.cleanMempool()
|
|
}
|
|
|
|
// QueryChannelByChanPoint tries to find a channel matching the channel point
|
|
// and asserts. It returns the channel found.
|
|
func (h *HarnessTest) QueryChannelByChanPoint(hn *node.HarnessNode,
|
|
chanPoint *lnrpc.ChannelPoint,
|
|
opts ...ListChannelOption) *lnrpc.Channel {
|
|
|
|
channel, err := h.findChannel(hn, chanPoint, opts...)
|
|
require.NoError(h, err, "failed to query channel")
|
|
|
|
return channel
|
|
}
|
|
|
|
// SendPaymentAndAssertStatus sends a payment from the passed node and asserts
|
|
// the desired status is reached.
|
|
func (h *HarnessTest) SendPaymentAndAssertStatus(hn *node.HarnessNode,
|
|
req *routerrpc.SendPaymentRequest,
|
|
status lnrpc.Payment_PaymentStatus) *lnrpc.Payment {
|
|
|
|
stream := hn.RPC.SendPayment(req)
|
|
return h.AssertPaymentStatusFromStream(stream, status)
|
|
}
|
|
|
|
// SendPaymentAssertFail sends a payment from the passed node and asserts the
|
|
// payment is failed with the specified failure reason .
|
|
func (h *HarnessTest) SendPaymentAssertFail(hn *node.HarnessNode,
|
|
req *routerrpc.SendPaymentRequest,
|
|
reason lnrpc.PaymentFailureReason) *lnrpc.Payment {
|
|
|
|
payment := h.SendPaymentAndAssertStatus(hn, req, lnrpc.Payment_FAILED)
|
|
require.Equal(h, reason, payment.FailureReason,
|
|
"payment failureReason not matched")
|
|
|
|
return payment
|
|
}
|
|
|
|
// SendPaymentAssertSettled sends a payment from the passed node and asserts the
|
|
// payment is settled.
|
|
func (h *HarnessTest) SendPaymentAssertSettled(hn *node.HarnessNode,
|
|
req *routerrpc.SendPaymentRequest) *lnrpc.Payment {
|
|
|
|
return h.SendPaymentAndAssertStatus(hn, req, lnrpc.Payment_SUCCEEDED)
|
|
}
|
|
|
|
// SendPaymentAssertInflight sends a payment from the passed node and asserts
|
|
// the payment is inflight.
|
|
func (h *HarnessTest) SendPaymentAssertInflight(hn *node.HarnessNode,
|
|
req *routerrpc.SendPaymentRequest) *lnrpc.Payment {
|
|
|
|
return h.SendPaymentAndAssertStatus(hn, req, lnrpc.Payment_IN_FLIGHT)
|
|
}
|
|
|
|
// OpenChannelRequest is used to open a channel using the method
|
|
// OpenMultiChannelsAsync.
|
|
type OpenChannelRequest struct {
|
|
// Local is the funding node.
|
|
Local *node.HarnessNode
|
|
|
|
// Remote is the receiving node.
|
|
Remote *node.HarnessNode
|
|
|
|
// Param is the open channel params.
|
|
Param OpenChannelParams
|
|
|
|
// stream is the client created after calling OpenChannel RPC.
|
|
stream rpc.OpenChanClient
|
|
|
|
// result is a channel used to send the channel point once the funding
|
|
// has succeeded.
|
|
result chan *lnrpc.ChannelPoint
|
|
}
|
|
|
|
// OpenMultiChannelsAsync takes a list of OpenChannelRequest and opens them in
|
|
// batch. The channel points are returned in same the order of the requests
|
|
// once all of the channel open succeeded.
|
|
//
|
|
// NOTE: compared to open multiple channel sequentially, this method will be
|
|
// faster as it doesn't need to mine 6 blocks for each channel open. However,
|
|
// it does make debugging the logs more difficult as messages are intertwined.
|
|
func (h *HarnessTest) OpenMultiChannelsAsync(
|
|
reqs []*OpenChannelRequest) []*lnrpc.ChannelPoint {
|
|
|
|
// openChannel opens a channel based on the request.
|
|
openChannel := func(req *OpenChannelRequest) {
|
|
stream := h.OpenChannelAssertStream(
|
|
req.Local, req.Remote, req.Param,
|
|
)
|
|
req.stream = stream
|
|
}
|
|
|
|
// assertChannelOpen is a helper closure that asserts a channel is
|
|
// open.
|
|
assertChannelOpen := func(req *OpenChannelRequest) {
|
|
// Wait for the channel open event from the stream.
|
|
cp := h.WaitForChannelOpenEvent(req.stream)
|
|
|
|
if !req.Param.Private {
|
|
// Check that both alice and bob have seen the channel
|
|
// from their channel watch request.
|
|
h.AssertTopologyChannelOpen(req.Local, cp)
|
|
h.AssertTopologyChannelOpen(req.Remote, cp)
|
|
}
|
|
|
|
// Finally, check that the channel can be seen in their
|
|
// ListChannels.
|
|
h.AssertChannelExists(req.Local, cp)
|
|
h.AssertChannelExists(req.Remote, cp)
|
|
|
|
req.result <- cp
|
|
}
|
|
|
|
// Go through the requests and make the OpenChannel RPC call.
|
|
for _, r := range reqs {
|
|
openChannel(r)
|
|
}
|
|
|
|
// Mine one block to confirm all the funding transactions.
|
|
h.MineBlocksAndAssertNumTxes(1, len(reqs))
|
|
|
|
// Mine 5 more blocks so all the public channels are announced to the
|
|
// network.
|
|
h.MineBlocks(numBlocksOpenChannel - 1)
|
|
|
|
// Once the blocks are mined, we fire goroutines for each of the
|
|
// request to watch for the channel openning.
|
|
for _, r := range reqs {
|
|
r.result = make(chan *lnrpc.ChannelPoint, 1)
|
|
go assertChannelOpen(r)
|
|
}
|
|
|
|
// Finally, collect the results.
|
|
channelPoints := make([]*lnrpc.ChannelPoint, 0)
|
|
for _, r := range reqs {
|
|
select {
|
|
case cp := <-r.result:
|
|
channelPoints = append(channelPoints, cp)
|
|
|
|
case <-time.After(wait.ChannelOpenTimeout):
|
|
require.Failf(h, "timeout", "wait channel point "+
|
|
"timeout for channel %s=>%s", r.Local.Name(),
|
|
r.Remote.Name())
|
|
}
|
|
}
|
|
|
|
// Assert that we have the expected num of channel points.
|
|
require.Len(h, channelPoints, len(reqs),
|
|
"returned channel points not match")
|
|
|
|
return channelPoints
|
|
}
|
|
|
|
// ReceiveInvoiceUpdate waits until a message is received on the subscribe
|
|
// invoice stream or the timeout is reached.
|
|
func (h *HarnessTest) ReceiveInvoiceUpdate(
|
|
stream rpc.InvoiceUpdateClient) *lnrpc.Invoice {
|
|
|
|
chanMsg := make(chan *lnrpc.Invoice)
|
|
errChan := make(chan error)
|
|
go func() {
|
|
// Consume one message. This will block until the message is
|
|
// received.
|
|
resp, err := stream.Recv()
|
|
if err != nil {
|
|
errChan <- err
|
|
return
|
|
}
|
|
chanMsg <- resp
|
|
}()
|
|
|
|
select {
|
|
case <-time.After(DefaultTimeout):
|
|
require.Fail(h, "timeout", "timeout receiving invoice update")
|
|
|
|
case err := <-errChan:
|
|
require.Failf(h, "err from stream",
|
|
"received err from stream: %v", err)
|
|
|
|
case updateMsg := <-chanMsg:
|
|
return updateMsg
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// CalculateTxFee retrieves parent transactions and reconstructs the fee paid.
|
|
func (h *HarnessTest) CalculateTxFee(tx *wire.MsgTx) btcutil.Amount {
|
|
var balance btcutil.Amount
|
|
for _, in := range tx.TxIn {
|
|
parentHash := in.PreviousOutPoint.Hash
|
|
rawTx := h.miner.GetRawTransaction(&parentHash)
|
|
parent := rawTx.MsgTx()
|
|
value := parent.TxOut[in.PreviousOutPoint.Index].Value
|
|
|
|
balance += btcutil.Amount(value)
|
|
}
|
|
|
|
for _, out := range tx.TxOut {
|
|
balance -= btcutil.Amount(out.Value)
|
|
}
|
|
|
|
return balance
|
|
}
|
|
|
|
// CalculateTxWeight calculates the weight for a given tx.
|
|
//
|
|
// TODO(yy): use weight estimator to get more accurate result.
|
|
func (h *HarnessTest) CalculateTxWeight(tx *wire.MsgTx) lntypes.WeightUnit {
|
|
utx := btcutil.NewTx(tx)
|
|
return lntypes.WeightUnit(blockchain.GetTransactionWeight(utx))
|
|
}
|
|
|
|
// CalculateTxFeeRate calculates the fee rate for a given tx.
|
|
func (h *HarnessTest) CalculateTxFeeRate(
|
|
tx *wire.MsgTx) chainfee.SatPerKWeight {
|
|
|
|
w := h.CalculateTxWeight(tx)
|
|
fee := h.CalculateTxFee(tx)
|
|
|
|
return chainfee.NewSatPerKWeight(fee, w)
|
|
}
|
|
|
|
// CalculateTxesFeeRate takes a list of transactions and estimates the fee rate
|
|
// used to sweep them.
|
|
//
|
|
// NOTE: only used in current test file.
|
|
func (h *HarnessTest) CalculateTxesFeeRate(txns []*wire.MsgTx) int64 {
|
|
const scale = 1000
|
|
|
|
var totalWeight, totalFee int64
|
|
for _, tx := range txns {
|
|
utx := btcutil.NewTx(tx)
|
|
totalWeight += blockchain.GetTransactionWeight(utx)
|
|
|
|
fee := h.CalculateTxFee(tx)
|
|
totalFee += int64(fee)
|
|
}
|
|
feeRate := totalFee * scale / totalWeight
|
|
|
|
return feeRate
|
|
}
|
|
|
|
// AssertSweepFound looks up a sweep in a nodes list of broadcast sweeps and
|
|
// asserts it's found.
|
|
//
|
|
// NOTE: Does not account for node's internal state.
|
|
func (h *HarnessTest) AssertSweepFound(hn *node.HarnessNode,
|
|
sweep string, verbose bool, startHeight int32) {
|
|
|
|
err := wait.NoError(func() error {
|
|
// List all sweeps that alice's node had broadcast.
|
|
sweepResp := hn.RPC.ListSweeps(verbose, startHeight)
|
|
|
|
var found bool
|
|
if verbose {
|
|
found = findSweepInDetails(h, sweep, sweepResp)
|
|
} else {
|
|
found = findSweepInTxids(h, sweep, sweepResp)
|
|
}
|
|
|
|
if found {
|
|
return nil
|
|
}
|
|
|
|
return fmt.Errorf("sweep tx %v not found", sweep)
|
|
}, wait.DefaultTimeout)
|
|
require.NoError(h, err, "%s: timeout checking sweep tx", hn.Name())
|
|
}
|
|
|
|
func findSweepInTxids(ht *HarnessTest, sweepTxid string,
|
|
sweepResp *walletrpc.ListSweepsResponse) bool {
|
|
|
|
sweepTxIDs := sweepResp.GetTransactionIds()
|
|
require.NotNil(ht, sweepTxIDs, "expected transaction ids")
|
|
require.Nil(ht, sweepResp.GetTransactionDetails())
|
|
|
|
// Check that the sweep tx we have just produced is present.
|
|
for _, tx := range sweepTxIDs.TransactionIds {
|
|
if tx == sweepTxid {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func findSweepInDetails(ht *HarnessTest, sweepTxid string,
|
|
sweepResp *walletrpc.ListSweepsResponse) bool {
|
|
|
|
sweepDetails := sweepResp.GetTransactionDetails()
|
|
require.NotNil(ht, sweepDetails, "expected transaction details")
|
|
require.Nil(ht, sweepResp.GetTransactionIds())
|
|
|
|
for _, tx := range sweepDetails.Transactions {
|
|
if tx.TxHash == sweepTxid {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// QueryRoutesAndRetry attempts to keep querying a route until timeout is
|
|
// reached.
|
|
//
|
|
// NOTE: when a channel is opened, we may need to query multiple times to get
|
|
// it in our QueryRoutes RPC. This happens even after we check the channel is
|
|
// heard by the node using ht.AssertChannelOpen. Deep down, this is because our
|
|
// GraphTopologySubscription and QueryRoutes give different results regarding a
|
|
// specific channel, with the formal reporting it being open while the latter
|
|
// not, resulting GraphTopologySubscription acting "faster" than QueryRoutes.
|
|
// TODO(yy): make sure related subsystems share the same view on a given
|
|
// channel.
|
|
func (h *HarnessTest) QueryRoutesAndRetry(hn *node.HarnessNode,
|
|
req *lnrpc.QueryRoutesRequest) *lnrpc.QueryRoutesResponse {
|
|
|
|
var routes *lnrpc.QueryRoutesResponse
|
|
err := wait.NoError(func() error {
|
|
ctxt, cancel := context.WithCancel(h.runCtx)
|
|
defer cancel()
|
|
|
|
resp, err := hn.RPC.LN.QueryRoutes(ctxt, req)
|
|
if err != nil {
|
|
return fmt.Errorf("%s: failed to query route: %w",
|
|
hn.Name(), err)
|
|
}
|
|
|
|
routes = resp
|
|
|
|
return nil
|
|
}, DefaultTimeout)
|
|
|
|
require.NoError(h, err, "timeout querying routes")
|
|
|
|
return routes
|
|
}
|
|
|
|
// ReceiveHtlcInterceptor waits until a message is received on the htlc
|
|
// interceptor stream or the timeout is reached.
|
|
func (h *HarnessTest) ReceiveHtlcInterceptor(
|
|
stream rpc.InterceptorClient) *routerrpc.ForwardHtlcInterceptRequest {
|
|
|
|
chanMsg := make(chan *routerrpc.ForwardHtlcInterceptRequest)
|
|
errChan := make(chan error)
|
|
go func() {
|
|
// Consume one message. This will block until the message is
|
|
// received.
|
|
resp, err := stream.Recv()
|
|
if err != nil {
|
|
errChan <- err
|
|
return
|
|
}
|
|
chanMsg <- resp
|
|
}()
|
|
|
|
select {
|
|
case <-time.After(DefaultTimeout):
|
|
require.Fail(h, "timeout", "timeout intercepting htlc")
|
|
|
|
case err := <-errChan:
|
|
require.Failf(h, "err from stream",
|
|
"received err from stream: %v", err)
|
|
|
|
case updateMsg := <-chanMsg:
|
|
return updateMsg
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// ReceiveChannelEvent waits until a message is received from the
|
|
// ChannelEventsClient stream or the timeout is reached.
|
|
func (h *HarnessTest) ReceiveChannelEvent(
|
|
stream rpc.ChannelEventsClient) *lnrpc.ChannelEventUpdate {
|
|
|
|
chanMsg := make(chan *lnrpc.ChannelEventUpdate)
|
|
errChan := make(chan error)
|
|
go func() {
|
|
// Consume one message. This will block until the message is
|
|
// received.
|
|
resp, err := stream.Recv()
|
|
if err != nil {
|
|
errChan <- err
|
|
return
|
|
}
|
|
chanMsg <- resp
|
|
}()
|
|
|
|
select {
|
|
case <-time.After(DefaultTimeout):
|
|
require.Fail(h, "timeout", "timeout intercepting htlc")
|
|
|
|
case err := <-errChan:
|
|
require.Failf(h, "err from stream",
|
|
"received err from stream: %v", err)
|
|
|
|
case updateMsg := <-chanMsg:
|
|
return updateMsg
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// GetOutputIndex returns the output index of the given address in the given
|
|
// transaction.
|
|
func (h *HarnessTest) GetOutputIndex(txid *chainhash.Hash, addr string) int {
|
|
// We'll then extract the raw transaction from the mempool in order to
|
|
// determine the index of the p2tr output.
|
|
tx := h.miner.GetRawTransaction(txid)
|
|
|
|
p2trOutputIndex := -1
|
|
for i, txOut := range tx.MsgTx().TxOut {
|
|
_, addrs, _, err := txscript.ExtractPkScriptAddrs(
|
|
txOut.PkScript, h.miner.ActiveNet,
|
|
)
|
|
require.NoError(h, err)
|
|
|
|
if addrs[0].String() == addr {
|
|
p2trOutputIndex = i
|
|
}
|
|
}
|
|
require.Greater(h, p2trOutputIndex, -1)
|
|
|
|
return p2trOutputIndex
|
|
}
|
|
|
|
// SendCoins sends a coin from node A to node B with the given amount, returns
|
|
// the sending tx.
|
|
func (h *HarnessTest) SendCoins(a, b *node.HarnessNode,
|
|
amt btcutil.Amount) *wire.MsgTx {
|
|
|
|
// Create an address for Bob receive the coins.
|
|
req := &lnrpc.NewAddressRequest{
|
|
Type: lnrpc.AddressType_TAPROOT_PUBKEY,
|
|
}
|
|
resp := b.RPC.NewAddress(req)
|
|
|
|
// Send the coins from Alice to Bob. We should expect a tx to be
|
|
// broadcast and seen in the mempool.
|
|
sendReq := &lnrpc.SendCoinsRequest{
|
|
Addr: resp.Address,
|
|
Amount: int64(amt),
|
|
TargetConf: 6,
|
|
}
|
|
a.RPC.SendCoins(sendReq)
|
|
tx := h.GetNumTxsFromMempool(1)[0]
|
|
|
|
return tx
|
|
}
|