itest/test: add test for expired hold invoices
This commit adds a test for a hold invoice which is accepted off-chain, and held by the recipient until it expired and the payer force-closes the channel. With this test we demonstrate two bugs in our handling of hold invoice state in the invoice registry when we expire on chain: - Htlcs not updated: even when we've timed out, we don't update the htlc state accordingly. - Invoice can be settled: the invoice can be settled even though it's expired on chain.
This commit is contained in:
parent
16373d3879
commit
2e39edd6bd
131
lntest/itest/lnd_hold_invoice_force_test.go
Normal file
131
lntest/itest/lnd_hold_invoice_force_test.go
Normal file
@ -0,0 +1,131 @@
|
|||||||
|
package itest
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/lightningnetwork/lnd/lncfg"
|
||||||
|
"github.com/lightningnetwork/lnd/lnrpc"
|
||||||
|
"github.com/lightningnetwork/lnd/lnrpc/invoicesrpc"
|
||||||
|
"github.com/lightningnetwork/lnd/lnrpc/routerrpc"
|
||||||
|
"github.com/lightningnetwork/lnd/lntest"
|
||||||
|
"github.com/lightningnetwork/lnd/lntest/wait"
|
||||||
|
"github.com/lightningnetwork/lnd/lntypes"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
// testHoldInvoiceForceClose demonstrates that recipients of hold invoices
|
||||||
|
// will not release active htlcs for their own invoices when they expire,
|
||||||
|
// resulting in a force close of their channel.
|
||||||
|
func testHoldInvoiceForceClose(net *lntest.NetworkHarness, t *harnessTest) {
|
||||||
|
ctxb, cancel := context.WithCancel(context.Background())
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
// Open a channel between alice and bob.
|
||||||
|
chanReq := lntest.OpenChannelParams{
|
||||||
|
Amt: 300000,
|
||||||
|
}
|
||||||
|
|
||||||
|
ctxt, _ := context.WithTimeout(ctxb, channelOpenTimeout)
|
||||||
|
chanPoint := openChannelAndAssert(ctxt, t, net, net.Alice, net.Bob, chanReq)
|
||||||
|
|
||||||
|
// Create a non-dust hold invoice for bob.
|
||||||
|
var (
|
||||||
|
preimage = lntypes.Preimage{1, 2, 3}
|
||||||
|
payHash = preimage.Hash()
|
||||||
|
)
|
||||||
|
invoiceReq := &invoicesrpc.AddHoldInvoiceRequest{
|
||||||
|
Value: 30000,
|
||||||
|
CltvExpiry: 40,
|
||||||
|
Hash: payHash[:],
|
||||||
|
}
|
||||||
|
|
||||||
|
ctxt, _ = context.WithTimeout(ctxb, defaultTimeout)
|
||||||
|
bobInvoice, err := net.Bob.AddHoldInvoice(ctxt, invoiceReq)
|
||||||
|
require.NoError(t.t, err)
|
||||||
|
|
||||||
|
// Pay this invoice from Alice -> Bob, we should achieve this with a
|
||||||
|
// single htlc.
|
||||||
|
_, err = net.Alice.RouterClient.SendPaymentV2(
|
||||||
|
ctxb, &routerrpc.SendPaymentRequest{
|
||||||
|
PaymentRequest: bobInvoice.PaymentRequest,
|
||||||
|
TimeoutSeconds: 60,
|
||||||
|
FeeLimitMsat: noFeeLimitMsat,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
require.NoError(t.t, err)
|
||||||
|
|
||||||
|
waitForInvoiceAccepted(t, net.Bob, payHash)
|
||||||
|
|
||||||
|
// Once the HTLC has cleared, alice and bob should both have a single
|
||||||
|
// htlc locked in.
|
||||||
|
nodes := []*lntest.HarnessNode{net.Alice, net.Bob}
|
||||||
|
err = wait.NoError(func() error {
|
||||||
|
return assertActiveHtlcs(nodes, payHash[:])
|
||||||
|
}, defaultTimeout)
|
||||||
|
require.NoError(t.t, err)
|
||||||
|
|
||||||
|
// Get our htlc expiry height and current block height so that we
|
||||||
|
// can mine the exact number of blocks required to expire the htlc.
|
||||||
|
chans, err := net.Alice.ListChannels(ctxb, &lnrpc.ListChannelsRequest{})
|
||||||
|
require.NoError(t.t, err)
|
||||||
|
require.Len(t.t, chans.Channels, 1)
|
||||||
|
require.Len(t.t, chans.Channels[0].PendingHtlcs, 1)
|
||||||
|
activeHtlc := chans.Channels[0].PendingHtlcs[0]
|
||||||
|
|
||||||
|
info, err := net.Alice.GetInfo(ctxb, &lnrpc.GetInfoRequest{})
|
||||||
|
require.NoError(t.t, err)
|
||||||
|
|
||||||
|
// Now we will mine blocks until the htlc expires, and wait for each
|
||||||
|
// node to sync to our latest height. Sanity check that we won't
|
||||||
|
// underflow.
|
||||||
|
require.Greater(t.t, activeHtlc.ExpirationHeight, info.BlockHeight,
|
||||||
|
"expected expiry after current height")
|
||||||
|
blocksTillExpiry := activeHtlc.ExpirationHeight - info.BlockHeight
|
||||||
|
|
||||||
|
// Alice will go to chain with some delta, sanity check that we won't
|
||||||
|
// underflow and subtract this from our mined blocks.
|
||||||
|
require.Greater(t.t, blocksTillExpiry,
|
||||||
|
uint32(lncfg.DefaultOutgoingBroadcastDelta))
|
||||||
|
blocksTillForce := blocksTillExpiry - lncfg.DefaultOutgoingBroadcastDelta
|
||||||
|
|
||||||
|
mineBlocks(t, net, blocksTillForce, 0)
|
||||||
|
|
||||||
|
require.NoError(t.t, net.Alice.WaitForBlockchainSync(ctxb))
|
||||||
|
require.NoError(t.t, net.Bob.WaitForBlockchainSync(ctxb))
|
||||||
|
|
||||||
|
// Alice should have a waiting-close channel because she has force
|
||||||
|
// closed to time out the htlc.
|
||||||
|
assertNumPendingChannels(t, net.Alice, 1, 0)
|
||||||
|
|
||||||
|
// We should have our force close tx in the mempool.
|
||||||
|
mineBlocks(t, net, 1, 1)
|
||||||
|
|
||||||
|
// Ensure alice and bob are synced to chain after we've mined our force
|
||||||
|
// close.
|
||||||
|
require.NoError(t.t, net.Alice.WaitForBlockchainSync(ctxb))
|
||||||
|
require.NoError(t.t, net.Bob.WaitForBlockchainSync(ctxb))
|
||||||
|
|
||||||
|
// At this point, Bob's channel should be resolved because his htlc is
|
||||||
|
// expired, so no further action is required. Alice will still have a
|
||||||
|
// pending force close channel because she needs to resolve the htlc.
|
||||||
|
assertNumPendingChannels(t, net.Alice, 0, 1)
|
||||||
|
assertNumPendingChannels(t, net.Bob, 0, 0)
|
||||||
|
|
||||||
|
ctxt, _ = context.WithTimeout(ctxb, defaultTimeout)
|
||||||
|
err = waitForNumChannelPendingForceClose(ctxt, net.Alice, 1,
|
||||||
|
func(channel *lnrpcForceCloseChannel) error {
|
||||||
|
numHtlcs := len(channel.PendingHtlcs)
|
||||||
|
if numHtlcs != 1 {
|
||||||
|
return fmt.Errorf("expected 1 htlc, got: "+
|
||||||
|
"%v", numHtlcs)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
)
|
||||||
|
require.NoError(t.t, err)
|
||||||
|
|
||||||
|
// Cleanup Alice's force close.
|
||||||
|
cleanupForceClose(t, net, net.Alice, chanPoint)
|
||||||
|
}
|
@ -440,8 +440,9 @@ func waitForNumChannelPendingForceClose(ctx context.Context,
|
|||||||
|
|
||||||
forceCloseChans := resp.PendingForceClosingChannels
|
forceCloseChans := resp.PendingForceClosingChannels
|
||||||
if len(forceCloseChans) != expectedNum {
|
if len(forceCloseChans) != expectedNum {
|
||||||
return fmt.Errorf("bob should have %d pending "+
|
return fmt.Errorf("%v should have %d pending "+
|
||||||
"force close channels but has %d", expectedNum,
|
"force close channels but has %d",
|
||||||
|
node.Cfg.Name, expectedNum,
|
||||||
len(forceCloseChans))
|
len(forceCloseChans))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -230,6 +230,10 @@ var allTestCases = []*testCase{
|
|||||||
name: "hold invoice sender persistence",
|
name: "hold invoice sender persistence",
|
||||||
test: testHoldInvoicePersistence,
|
test: testHoldInvoicePersistence,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
name: "hold invoice force close",
|
||||||
|
test: testHoldInvoiceForceClose,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
name: "cpfp",
|
name: "cpfp",
|
||||||
test: testCPFP,
|
test: testCPFP,
|
||||||
|
Loading…
Reference in New Issue
Block a user