mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-01-18 21:35:24 +01:00
chainntnfs: remove subscriptions when the relevant tx is confirmed
This commit removes the subscribed inputs from mempool notifier when the relevant transaction is confirmed.
This commit is contained in:
parent
556c38fefd
commit
e9c10bf0a1
@ -455,23 +455,19 @@ out:
|
|||||||
case chain.RelevantTx:
|
case chain.RelevantTx:
|
||||||
tx := btcutil.NewTx(&item.TxRecord.MsgTx)
|
tx := btcutil.NewTx(&item.TxRecord.MsgTx)
|
||||||
|
|
||||||
// If this is a mempool spend, we'll ask the
|
// Init values.
|
||||||
// mempool notifier to hanlde it.
|
isMempool := false
|
||||||
|
height := uint32(0)
|
||||||
|
|
||||||
|
// Unwrap values.
|
||||||
if item.Block == nil {
|
if item.Block == nil {
|
||||||
b.memNotifier.ProcessRelevantSpendTx(tx)
|
isMempool = true
|
||||||
continue
|
} else {
|
||||||
|
height = uint32(item.Block.Height)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise this is a confirmed spend, and
|
// Handle the transaction.
|
||||||
// we'll ask the tx notifier to handle it.
|
b.handleRelevantTx(tx, isMempool, height)
|
||||||
err := b.txNotifier.ProcessRelevantSpendTx(
|
|
||||||
tx, uint32(item.Block.Height),
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
chainntnfs.Log.Errorf("Unable to "+
|
|
||||||
"process transaction %v: %v",
|
|
||||||
tx.Hash(), err)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case <-b.quit:
|
case <-b.quit:
|
||||||
@ -480,6 +476,39 @@ out:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// handleRelevantTx handles a new transaction that has been seen either in a
|
||||||
|
// block or in the mempool. If in mempool, it will ask the mempool notifier to
|
||||||
|
// handle it. If in a block, it will ask the txNotifier to handle it, and
|
||||||
|
// cancel any relevant subscriptions made in the mempool.
|
||||||
|
func (b *BitcoindNotifier) handleRelevantTx(tx *btcutil.Tx,
|
||||||
|
mempool bool, height uint32) {
|
||||||
|
|
||||||
|
// If this is a mempool spend, we'll ask the mempool notifier to hanlde
|
||||||
|
// it.
|
||||||
|
if mempool {
|
||||||
|
b.memNotifier.ProcessRelevantSpendTx(tx)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Otherwise this is a confirmed spend, and we'll ask the tx notifier
|
||||||
|
// to handle it.
|
||||||
|
err := b.txNotifier.ProcessRelevantSpendTx(tx, height)
|
||||||
|
if err != nil {
|
||||||
|
chainntnfs.Log.Errorf("Unable to process transaction %v: %v",
|
||||||
|
tx.Hash(), err)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Once the tx is processed, we will ask the memNotifier to unsubscribe
|
||||||
|
// the input.
|
||||||
|
//
|
||||||
|
// NOTE(yy): we could build it into txNotifier.ProcessRelevantSpendTx,
|
||||||
|
// but choose to implement it here so we can easily decouple the two
|
||||||
|
// notifiers in the future.
|
||||||
|
b.memNotifier.UnsubsribeConfirmedSpentTx(tx)
|
||||||
|
}
|
||||||
|
|
||||||
// historicalConfDetails looks up whether a confirmation request (txid/output
|
// historicalConfDetails looks up whether a confirmation request (txid/output
|
||||||
// script) has already been included in a block in the active chain and, if so,
|
// script) has already been included in a block in the active chain and, if so,
|
||||||
// returns details about said block.
|
// returns details about said block.
|
||||||
|
@ -505,23 +505,19 @@ out:
|
|||||||
newSpend := item.(*txUpdate)
|
newSpend := item.(*txUpdate)
|
||||||
tx := newSpend.tx
|
tx := newSpend.tx
|
||||||
|
|
||||||
// If this is a mempool spend, we'll ask the mempool
|
// Init values.
|
||||||
// notifier to hanlde it.
|
isMempool := false
|
||||||
|
height := uint32(0)
|
||||||
|
|
||||||
|
// Unwrap values.
|
||||||
if newSpend.details == nil {
|
if newSpend.details == nil {
|
||||||
b.memNotifier.ProcessRelevantSpendTx(tx)
|
isMempool = true
|
||||||
continue
|
} else {
|
||||||
|
height = uint32(newSpend.details.Height)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise this is a confirmed spend, and we'll ask
|
// Handle the transaction.
|
||||||
// the tx notifier to handle it.
|
b.handleRelevantTx(tx, isMempool, height)
|
||||||
err := b.txNotifier.ProcessRelevantSpendTx(
|
|
||||||
tx, uint32(newSpend.details.Height),
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
chainntnfs.Log.Errorf("Unable to process "+
|
|
||||||
"transaction %v: %v",
|
|
||||||
newSpend.tx.Hash(), err)
|
|
||||||
}
|
|
||||||
|
|
||||||
case <-b.quit:
|
case <-b.quit:
|
||||||
break out
|
break out
|
||||||
@ -529,6 +525,39 @@ out:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// handleRelevantTx handles a new transaction that has been seen either in a
|
||||||
|
// block or in the mempool. If in mempool, it will ask the mempool notifier to
|
||||||
|
// handle it. If in a block, it will ask the txNotifier to handle it, and
|
||||||
|
// cancel any relevant subscriptions made in the mempool.
|
||||||
|
func (b *BtcdNotifier) handleRelevantTx(tx *btcutil.Tx,
|
||||||
|
mempool bool, height uint32) {
|
||||||
|
|
||||||
|
// If this is a mempool spend, we'll ask the mempool notifier to hanlde
|
||||||
|
// it.
|
||||||
|
if mempool {
|
||||||
|
b.memNotifier.ProcessRelevantSpendTx(tx)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Otherwise this is a confirmed spend, and we'll ask the tx notifier
|
||||||
|
// to handle it.
|
||||||
|
err := b.txNotifier.ProcessRelevantSpendTx(tx, height)
|
||||||
|
if err != nil {
|
||||||
|
chainntnfs.Log.Errorf("Unable to process transaction %v: %v",
|
||||||
|
tx.Hash(), err)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Once the tx is processed, we will ask the memNotifier to unsubscribe
|
||||||
|
// the input.
|
||||||
|
//
|
||||||
|
// NOTE(yy): we could build it into txNotifier.ProcessRelevantSpendTx,
|
||||||
|
// but choose to implement it here so we can easily decouple the two
|
||||||
|
// notifiers in the future.
|
||||||
|
b.memNotifier.UnsubsribeConfirmedSpentTx(tx)
|
||||||
|
}
|
||||||
|
|
||||||
// historicalConfDetails looks up whether a confirmation request (txid/output
|
// historicalConfDetails looks up whether a confirmation request (txid/output
|
||||||
// script) has already been included in a block in the active chain and, if so,
|
// script) has already been included in a block in the active chain and, if so,
|
||||||
// returns details about said block.
|
// returns details about said block.
|
||||||
|
@ -140,6 +140,23 @@ func (m *MempoolNotifier) UnsubscribeEvent(sub *MempoolSpendEvent) {
|
|||||||
clients.Delete(sub.id)
|
clients.Delete(sub.id)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UnsubsribeConfirmedSpentTx takes a transaction and removes the subscriptions
|
||||||
|
// identified using its inputs.
|
||||||
|
func (m *MempoolNotifier) UnsubsribeConfirmedSpentTx(tx *btcutil.Tx) {
|
||||||
|
Log.Tracef("Unsubscribe confirmed tx %s", tx.Hash())
|
||||||
|
|
||||||
|
// Get the spent inputs of interest.
|
||||||
|
spentInputs := m.findRelevantInputs(tx)
|
||||||
|
|
||||||
|
// Unsubscribe the subscribers.
|
||||||
|
for outpoint := range spentInputs {
|
||||||
|
m.UnsubscribeInput(outpoint)
|
||||||
|
}
|
||||||
|
|
||||||
|
Log.Tracef("Finished unsubscribing confirmed tx %s, found %d inputs",
|
||||||
|
tx.Hash(), len(spentInputs))
|
||||||
|
}
|
||||||
|
|
||||||
// ProcessRelevantSpendTx takes a transaction and checks whether it spends any
|
// ProcessRelevantSpendTx takes a transaction and checks whether it spends any
|
||||||
// of the subscribed inputs. If so, spend notifications are sent to the
|
// of the subscribed inputs. If so, spend notifications are sent to the
|
||||||
// relevant subscribers.
|
// relevant subscribers.
|
||||||
|
@ -342,3 +342,46 @@ func TestMempoolNotifySpentCancel(t *testing.T) {
|
|||||||
// Expected
|
// Expected
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TestMempoolUnsubscribeConfirmedSpentTx tests that the subscriptions for a
|
||||||
|
// confirmed tx are removed when calling the method.
|
||||||
|
func TestMempoolUnsubscribeConfirmedSpentTx(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
// Create a new mempool notifier instance.
|
||||||
|
notifier := NewMempoolNotifier()
|
||||||
|
|
||||||
|
// Create two inputs and subscribe to them.
|
||||||
|
input1 := wire.OutPoint{Hash: [32]byte{1}, Index: 0}
|
||||||
|
input2 := wire.OutPoint{Hash: [32]byte{2}, Index: 0}
|
||||||
|
|
||||||
|
// sub1 and sub2 are subscribed to the same input.
|
||||||
|
notifier.SubscribeInput(input1)
|
||||||
|
notifier.SubscribeInput(input1)
|
||||||
|
|
||||||
|
// sub3 is subscribed to a different input.
|
||||||
|
sub3 := notifier.SubscribeInput(input2)
|
||||||
|
|
||||||
|
// Create a transaction that spends input1.
|
||||||
|
msgTx := &wire.MsgTx{
|
||||||
|
TxIn: []*wire.TxIn{
|
||||||
|
{PreviousOutPoint: input1},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
tx := btcutil.NewTx(msgTx)
|
||||||
|
|
||||||
|
// Unsubscribe the relevant transaction.
|
||||||
|
notifier.UnsubsribeConfirmedSpentTx(tx)
|
||||||
|
|
||||||
|
// Verify that the sub1 and sub2 are removed from the notifier.
|
||||||
|
_, loaded := notifier.subscribedInputs.Load(input1)
|
||||||
|
require.False(t, loaded)
|
||||||
|
|
||||||
|
// Verify that the sub3 is not affected.
|
||||||
|
subs, loaded := notifier.subscribedInputs.Load(input2)
|
||||||
|
require.True(t, loaded)
|
||||||
|
|
||||||
|
// sub3 should still be found.
|
||||||
|
_, loaded = subs.Load(sub3.id)
|
||||||
|
require.True(t, loaded)
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user