mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-03-04 09:48:19 +01:00
lnwallet/chancloser: add tests for the new max fee behavior
This commit is contained in:
parent
106912b015
commit
9eb1e8721a
1 changed files with 174 additions and 0 deletions
|
@ -2,8 +2,14 @@ package chancloser
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"crypto/rand"
|
"crypto/rand"
|
||||||
|
"fmt"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/btcsuite/btcd/btcutil"
|
||||||
|
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
||||||
|
"github.com/btcsuite/btcd/wire"
|
||||||
|
"github.com/lightningnetwork/lnd/input"
|
||||||
|
"github.com/lightningnetwork/lnd/lnwallet/chainfee"
|
||||||
"github.com/lightningnetwork/lnd/lnwire"
|
"github.com/lightningnetwork/lnd/lnwire"
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
@ -23,6 +29,8 @@ func randDeliveryAddress(t *testing.T) lnwire.DeliveryAddress {
|
||||||
// when an upfront shutdown script is set and the script provided does not
|
// when an upfront shutdown script is set and the script provided does not
|
||||||
// match, and does not error in any other case.
|
// match, and does not error in any other case.
|
||||||
func TestMaybeMatchScript(t *testing.T) {
|
func TestMaybeMatchScript(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
addr1 := randDeliveryAddress(t)
|
addr1 := randDeliveryAddress(t)
|
||||||
addr2 := randDeliveryAddress(t)
|
addr2 := randDeliveryAddress(t)
|
||||||
|
|
||||||
|
@ -62,6 +70,8 @@ func TestMaybeMatchScript(t *testing.T) {
|
||||||
test := test
|
test := test
|
||||||
|
|
||||||
t.Run(test.name, func(t *testing.T) {
|
t.Run(test.name, func(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
err := maybeMatchScript(
|
err := maybeMatchScript(
|
||||||
func() error { return nil }, test.upfrontScript,
|
func() error { return nil }, test.upfrontScript,
|
||||||
test.shutdownScript,
|
test.shutdownScript,
|
||||||
|
@ -73,3 +83,167 @@ func TestMaybeMatchScript(t *testing.T) {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type mockChannel struct {
|
||||||
|
absoluteFee btcutil.Amount
|
||||||
|
chanPoint wire.OutPoint
|
||||||
|
initiator bool
|
||||||
|
scid lnwire.ShortChannelID
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *mockChannel) CalcFee(chainfee.SatPerKWeight) btcutil.Amount {
|
||||||
|
return m.absoluteFee
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *mockChannel) ChannelPoint() *wire.OutPoint {
|
||||||
|
return &m.chanPoint
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *mockChannel) MarkCoopBroadcasted(*wire.MsgTx, bool) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *mockChannel) IsInitiator() bool {
|
||||||
|
return m.initiator
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *mockChannel) ShortChanID() lnwire.ShortChannelID {
|
||||||
|
return m.scid
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *mockChannel) AbsoluteThawHeight() (uint32, error) {
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *mockChannel) RemoteUpfrontShutdownScript() lnwire.DeliveryAddress {
|
||||||
|
return lnwire.DeliveryAddress{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *mockChannel) CreateCloseProposal(fee btcutil.Amount,
|
||||||
|
localScript, remoteScript []byte,
|
||||||
|
) (input.Signature, *chainhash.Hash, btcutil.Amount, error) {
|
||||||
|
|
||||||
|
return nil, nil, 0, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *mockChannel) CompleteCooperativeClose(localSig,
|
||||||
|
remoteSig input.Signature, localScript, remoteScript []byte,
|
||||||
|
proposedFee btcutil.Amount) (*wire.MsgTx, btcutil.Amount, error) {
|
||||||
|
|
||||||
|
return nil, 0, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestMaxFeeClamp tests that if a max fee is specified, then it's used instead
|
||||||
|
// of the default max fee multiplier.
|
||||||
|
func TestMaxFeeClamp(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
const absoluteFee = btcutil.Amount(1000)
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
|
||||||
|
idealFee chainfee.SatPerKWeight
|
||||||
|
inputMaxFee chainfee.SatPerKWeight
|
||||||
|
|
||||||
|
maxFee btcutil.Amount
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
// No max fee specified, we should see 3x the ideal fee.
|
||||||
|
name: "no max fee",
|
||||||
|
|
||||||
|
idealFee: chainfee.SatPerKWeight(253),
|
||||||
|
maxFee: absoluteFee * defaultMaxFeeMultiplier,
|
||||||
|
}, {
|
||||||
|
// Max fee specified, this should be used in place.
|
||||||
|
name: "max fee clamp",
|
||||||
|
|
||||||
|
idealFee: chainfee.SatPerKWeight(253),
|
||||||
|
inputMaxFee: chainfee.SatPerKWeight(2530),
|
||||||
|
|
||||||
|
// Our mock just returns the canned absolute fee here.
|
||||||
|
maxFee: absoluteFee,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, test := range tests {
|
||||||
|
t.Run(test.name, func(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
channel := mockChannel{
|
||||||
|
absoluteFee: absoluteFee,
|
||||||
|
}
|
||||||
|
|
||||||
|
chanCloser := NewChanCloser(
|
||||||
|
ChanCloseCfg{
|
||||||
|
Channel: &channel,
|
||||||
|
MaxFee: test.inputMaxFee,
|
||||||
|
}, nil, test.idealFee, 0, nil, false,
|
||||||
|
)
|
||||||
|
|
||||||
|
require.Equal(t, test.maxFee, chanCloser.maxFee)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestMaxFeeBailOut tests that once the negotiated fee rate rises above our
|
||||||
|
// maximum fee, we'll return an error and refuse to process a co-op close
|
||||||
|
// message.
|
||||||
|
func TestMaxFeeBailOut(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
const (
|
||||||
|
absoluteFee = btcutil.Amount(1000)
|
||||||
|
idealFee = chainfee.SatPerKWeight(253)
|
||||||
|
)
|
||||||
|
|
||||||
|
for _, isInitiator := range []bool{true, false} {
|
||||||
|
t.Run(fmt.Sprintf("initiator=%v", isInitiator), func(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
// First, we'll make our mock channel, and use that to
|
||||||
|
// instantiate our channel closer.
|
||||||
|
closeCfg := ChanCloseCfg{
|
||||||
|
Channel: &mockChannel{
|
||||||
|
absoluteFee: absoluteFee,
|
||||||
|
initiator: isInitiator,
|
||||||
|
},
|
||||||
|
MaxFee: idealFee * 2,
|
||||||
|
}
|
||||||
|
chanCloser := NewChanCloser(
|
||||||
|
closeCfg, nil, idealFee, 0, nil, false,
|
||||||
|
)
|
||||||
|
|
||||||
|
// We'll now force the channel state into the
|
||||||
|
// closeFeeNegotiation state so we can skip straight to
|
||||||
|
// the juicy part. We'll also set our last fee sent so
|
||||||
|
// we'll attempt to actually "negotiate" here.
|
||||||
|
chanCloser.state = closeFeeNegotiation
|
||||||
|
chanCloser.lastFeeProposal = absoluteFee
|
||||||
|
|
||||||
|
// Next, we'll make a ClosingSigned message that
|
||||||
|
// proposes a fee that's above the specified max fee.
|
||||||
|
//
|
||||||
|
// NOTE: We use the absoluteFee here since our mock
|
||||||
|
// always returns this fee for the CalcFee method which
|
||||||
|
// is used to translate a fee rate
|
||||||
|
// into an absolute fee amount in sats.
|
||||||
|
closeMsg := &lnwire.ClosingSigned{
|
||||||
|
FeeSatoshis: absoluteFee * 2,
|
||||||
|
}
|
||||||
|
|
||||||
|
_, _, err := chanCloser.ProcessCloseMsg(closeMsg)
|
||||||
|
|
||||||
|
switch isInitiator {
|
||||||
|
// If we're the initiator, then we expect an error at
|
||||||
|
// this point.
|
||||||
|
case true:
|
||||||
|
require.ErrorIs(t, err, ErrProposalExeceedsMaxFee)
|
||||||
|
|
||||||
|
// Otherwise, we expect things to fail for some other
|
||||||
|
// reason (invalid sig, etc).
|
||||||
|
case false:
|
||||||
|
require.NotErrorIs(t, err, ErrProposalExeceedsMaxFee)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue