chainntnfs/bitcoindnotify: always call UpdateSpendDetails

This commit fixes a bug that would cause the
notifier not to commit spend hints for items
that are not found. This is done by calling
UpdateSpendDetails with a nil detail, permitting
the notifier to begin updating the spend hints
with new blocks that arrive at tip. The change
is designed to mimic the behavior for historical
confirmation dispatch.

The symptom of this bug is needing to do many
long rescans on startup, even if new blocks
arrive after the rescan had completed. With
this change, nodes will have to do the scans
once more before their hints will be properly
updated. Restarts from then on should not
have this behavior.
This commit is contained in:
Conner Fromknecht 2019-04-01 18:20:52 -07:00
parent c37ea68ba6
commit 7a08825b1e
No known key found for this signature in database
GPG Key ID: E7D737B67FA592C7

@ -25,13 +25,6 @@ const (
notifierType = "bitcoind" notifierType = "bitcoind"
) )
var (
// ErrTransactionNotFound is an error returned when we attempt to find a
// transaction by manually scanning the chain within a specific range
// but it is not found.
ErrTransactionNotFound = errors.New("transaction not found within range")
)
// chainUpdate encapsulates an update to the current main chain. This struct is // chainUpdate encapsulates an update to the current main chain. This struct is
// used as an element within an unbounded queue in order to avoid blocking the // used as an element within an unbounded queue in order to avoid blocking the
// main rpc dispatch rule. // main rpc dispatch rule.
@ -264,7 +257,10 @@ out:
go func() { go func() {
defer b.wg.Done() defer b.wg.Done()
err := b.dispatchSpendDetailsManually(msg) spendDetails, err := b.historicalSpendDetails(
msg.SpendRequest,
msg.StartHeight, msg.EndHeight,
)
if err != nil { if err != nil {
chainntnfs.Log.Errorf("Rescan to "+ chainntnfs.Log.Errorf("Rescan to "+
"determine the spend "+ "determine the spend "+
@ -273,6 +269,24 @@ out:
msg.SpendRequest, msg.SpendRequest,
msg.StartHeight, msg.StartHeight,
msg.EndHeight, err) msg.EndHeight, err)
return
}
// If the historical dispatch finished
// without error, we will invoke
// UpdateSpendDetails even if none were
// found. This allows the notifier to
// begin safely updating the height hint
// cache at tip, since any pending
// rescans have now completed.
err = b.txNotifier.UpdateSpendDetails(
msg.SpendRequest, spendDetails,
)
if err != nil {
chainntnfs.Log.Errorf("Unable "+
"to update spend "+
"details of %v: %v",
msg.SpendRequest, err)
} }
}() }()
@ -823,16 +837,13 @@ func (b *BitcoindNotifier) RegisterSpendNtfn(outpoint *wire.OutPoint,
return ntfn.Event, nil return ntfn.Event, nil
} }
// disaptchSpendDetailsManually attempts to manually scan the chain within the // historicalSpendDetails attempts to manually scan the chain within the given
// given height range for a transaction that spends the given outpoint/output // height range for a transaction that spends the given outpoint/output script.
// script. If one is found, it's spending details are sent to the TxNotifier, // If one is found, the spend details are assembled and returned to the caller.
// which will then dispatch the notification to all of its clients. // If the spend is not found, a nil spend detail will be returned.
func (b *BitcoindNotifier) dispatchSpendDetailsManually( func (b *BitcoindNotifier) historicalSpendDetails(
historicalDispatchDetails *chainntnfs.HistoricalSpendDispatch) error { spendRequest chainntnfs.SpendRequest, startHeight, endHeight uint32) (
*chainntnfs.SpendDetail, error) {
spendRequest := historicalDispatchDetails.SpendRequest
startHeight := historicalDispatchDetails.StartHeight
endHeight := historicalDispatchDetails.EndHeight
// Begin scanning blocks at every height to determine if the outpoint // Begin scanning blocks at every height to determine if the outpoint
// was spent. // was spent.
@ -841,20 +852,20 @@ func (b *BitcoindNotifier) dispatchSpendDetailsManually(
// processing the next height. // processing the next height.
select { select {
case <-b.quit: case <-b.quit:
return chainntnfs.ErrChainNotifierShuttingDown return nil, chainntnfs.ErrChainNotifierShuttingDown
default: default:
} }
// First, we'll fetch the block for the current height. // First, we'll fetch the block for the current height.
blockHash, err := b.chainConn.GetBlockHash(int64(height)) blockHash, err := b.chainConn.GetBlockHash(int64(height))
if err != nil { if err != nil {
return fmt.Errorf("unable to retrieve hash for block "+ return nil, fmt.Errorf("unable to retrieve hash for "+
"with height %d: %v", height, err) "block with height %d: %v", height, err)
} }
block, err := b.chainConn.GetBlock(blockHash) block, err := b.chainConn.GetBlock(blockHash)
if err != nil { if err != nil {
return fmt.Errorf("unable to retrieve block with hash "+ return nil, fmt.Errorf("unable to retrieve block "+
"%v: %v", blockHash, err) "with hash %v: %v", blockHash, err)
} }
// Then, we'll manually go over every input in every transaction // Then, we'll manually go over every input in every transaction
@ -863,29 +874,24 @@ func (b *BitcoindNotifier) dispatchSpendDetailsManually(
for _, tx := range block.Transactions { for _, tx := range block.Transactions {
matches, inputIdx, err := spendRequest.MatchesTx(tx) matches, inputIdx, err := spendRequest.MatchesTx(tx)
if err != nil { if err != nil {
return err return nil, err
} }
if !matches { if !matches {
continue continue
} }
txHash := tx.TxHash() txHash := tx.TxHash()
details := &chainntnfs.SpendDetail{ return &chainntnfs.SpendDetail{
SpentOutPoint: &tx.TxIn[inputIdx].PreviousOutPoint, SpentOutPoint: &tx.TxIn[inputIdx].PreviousOutPoint,
SpenderTxHash: &txHash, SpenderTxHash: &txHash,
SpendingTx: tx, SpendingTx: tx,
SpenderInputIndex: inputIdx, SpenderInputIndex: inputIdx,
SpendingHeight: int32(height), SpendingHeight: int32(height),
} }, nil
return b.txNotifier.UpdateSpendDetails(
historicalDispatchDetails.SpendRequest,
details,
)
} }
} }
return ErrTransactionNotFound return nil, nil
} }
// RegisterConfirmationsNtfn registers an intent to be notified once the target // RegisterConfirmationsNtfn registers an intent to be notified once the target