Merge pull request #5465 from Crypt-iQ/spend_height_0702

chainntnfs: populate spendsByHeight during historical dispatch
This commit is contained in:
Olaoluwa Osuntokun 2021-09-13 11:58:49 -07:00 committed by GitHub
commit b0f3a08f2d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 107 additions and 0 deletions

View File

@ -1325,6 +1325,19 @@ func (n *TxNotifier) dispatchSpendDetails(ntfn *SpendNtfn, details *SpendDetail)
return ErrTxNotifierExiting
}
spendHeight := uint32(details.SpendingHeight)
// We also add to spendsByHeight to notify on chain reorgs.
reorgSafeHeight := spendHeight + n.reorgSafetyLimit
if reorgSafeHeight > n.currentHeight {
txSet, exists := n.spendsByHeight[spendHeight]
if !exists {
txSet = make(map[SpendRequest]struct{})
n.spendsByHeight[spendHeight] = txSet
}
txSet[ntfn.SpendRequest] = struct{}{}
}
return nil
}

View File

@ -10,6 +10,7 @@ import (
"github.com/btcsuite/btcd/wire"
"github.com/btcsuite/btcutil"
"github.com/lightningnetwork/lnd/chainntnfs"
"github.com/stretchr/testify/require"
)
var (
@ -1843,6 +1844,97 @@ func TestTxNotifierSpendReorg(t *testing.T) {
}
}
// TestTxNotifierUpdateSpendReorg tests that a call to RegisterSpend after the
// spend has been confirmed, and then UpdateSpendDetails (called by historical
// dispatch), followed by a chain re-org will notify on the Reorg channel. This
// was not always the case and has since been fixed.
func TestTxNotifierSpendReorgMissed(t *testing.T) {
t.Parallel()
const startingHeight = 10
hintCache := newMockHintCache()
n := chainntnfs.NewTxNotifier(
startingHeight, chainntnfs.ReorgSafetyLimit, hintCache,
hintCache,
)
// We'll create a spending transaction that spends the outpoint we'll
// watch.
op := wire.OutPoint{Index: 1}
spendTx := wire.NewMsgTx(2)
spendTx.AddTxIn(&wire.TxIn{
PreviousOutPoint: op,
SignatureScript: testSigScript,
})
spendTxHash := spendTx.TxHash()
// Create the spend details that we'll call UpdateSpendDetails with.
spendDetails := &chainntnfs.SpendDetail{
SpentOutPoint: &op,
SpenderTxHash: &spendTxHash,
SpendingTx: spendTx,
SpenderInputIndex: 0,
SpendingHeight: startingHeight + 1,
}
// Now confirm the spending transaction.
block := btcutil.NewBlock(&wire.MsgBlock{
Transactions: []*wire.MsgTx{spendTx},
})
err := n.ConnectTip(block.Hash(), startingHeight+1, block.Transactions())
if err != nil {
t.Fatalf("unable to connect block: %v", err)
}
if err := n.NotifyHeight(startingHeight + 1); err != nil {
t.Fatalf("unable to dispatch notifications: %v", err)
}
// We register for the spend now and will not get a spend notification
// until we call UpdateSpendDetails.
ntfn, err := n.RegisterSpend(&op, testRawScript, 1)
if err != nil {
t.Fatalf("unable to register spend: %v", err)
}
// Assert that the HistoricalDispatch variable is non-nil. We'll use
// the SpendRequest member to update the spend details.
require.NotEmpty(t, ntfn.HistoricalDispatch)
select {
case <-ntfn.Event.Spend:
t.Fatalf("did not expect to receive spend ntfn")
default:
}
// We now call UpdateSpendDetails with our generated spend details to
// simulate a historical spend dispatch being performed. This should
// result in a notification being received on the Spend channel.
err = n.UpdateSpendDetails(
ntfn.HistoricalDispatch.SpendRequest, spendDetails,
)
require.Empty(t, err)
// Assert that we receive a Spend notification.
select {
case <-ntfn.Event.Spend:
default:
t.Fatalf("expected to receive spend ntfn")
}
// We will now re-org the spending transaction out of the chain, and we
// should receive a notification on the Reorg channel.
err = n.DisconnectTip(startingHeight + 1)
require.Empty(t, err)
select {
case <-ntfn.Event.Spend:
t.Fatalf("received unexpected spend ntfn")
case <-ntfn.Event.Reorg:
default:
t.Fatalf("expected spend reorg ntfn")
}
}
// TestTxNotifierConfirmHintCache ensures that the height hints for transactions
// are kept track of correctly with each new block connected/disconnected. This
// test also asserts that the height hints are not updated until the simulated

View File

@ -265,6 +265,8 @@ mode](https://github.com/lightningnetwork/lnd/pull/5564).
[A bug has been fixed with Neutrino's `RegisterConfirmationsNtfn` and `RegisterSpendNtfn` calls that would cause notifications to be missed.](https://github.com/lightningnetwork/lnd/pull/5453)
[A bug has been fixed when registering for spend notifications in the `txnotifier`. A re-org notification would previously not be dispatched in certain scenarios.](https://github.com/lightningnetwork/lnd/pull/5465)
## Documentation
The [code contribution guidelines have been updated to mention the new