From b579c23310b49d1d8d76737b3c054b92ede56fec Mon Sep 17 00:00:00 2001 From: Wilmer Paulino Date: Thu, 6 Dec 2018 21:13:41 -0800 Subject: [PATCH] chainntnfs/txnotifier: support registration of scripts for spends --- chainntnfs/txnotifier.go | 220 +++++++++++++++++++++------------------ 1 file changed, 117 insertions(+), 103 deletions(-) diff --git a/chainntnfs/txnotifier.go b/chainntnfs/txnotifier.go index 2323b151c..6d6776e24 100644 --- a/chainntnfs/txnotifier.go +++ b/chainntnfs/txnotifier.go @@ -101,20 +101,22 @@ func newConfNtfnSet() *confNtfnSet { } } -// spendNtfnSet holds all known, registered spend notifications for an outpoint. -// If duplicate notifications are requested, only one historical dispatch will -// be spawned to ensure redundant scans are not permitted. +// spendNtfnSet holds all known, registered spend notifications for a spend +// request (outpoint/output script). If duplicate notifications are requested, +// only one historical dispatch will be spawned to ensure redundant scans are +// not permitted. type spendNtfnSet struct { // ntfns keeps tracks of all the active client notification requests for - // an outpoint. + // an outpoint/output script. ntfns map[uint64]*SpendNtfn - // rescanStatus represents the current rescan state for the outpoint. + // rescanStatus represents the current rescan state for the spend + // request (outpoint/output script). rescanStatus rescanState - // details serves as a cache of the spend details for an outpoint that - // we'll use to determine if an outpoint has already been spent at the - // time of registration. + // details serves as a cache of the spend details for an outpoint/output + // script that we'll use to determine if it has already been spent at + // the time of registration. details *SpendDetail } @@ -238,7 +240,8 @@ type SpendRequest struct { // _script_, rather than the outpoint. OutPoint wire.OutPoint - // PkScript is the script of the outpoint. + // PkScript is the script of the outpoint. If a zero outpoint is set, + // then this can be an arbitrary script. PkScript txscript.PkScript } @@ -288,29 +291,25 @@ func (r SpendRequest) SpendHintKey() ([]byte, error) { } // SpendNtfn represents a client's request to receive a notification once an -// outpoint has been spent on-chain. The client is asynchronously notified via -// the SpendEvent channels. +// outpoint/output script has been spent on-chain. The client is asynchronously +// notified via the SpendEvent channels. type SpendNtfn struct { // SpendID uniquely identies the spend notification request for the - // specified outpoint. + // specified outpoint/output script. SpendID uint64 - // OutPoint is the outpoint for which a client has requested a spend - // notification for. - OutPoint wire.OutPoint - - // PkScript is the script of the outpoint. This is needed in order to - // match compact filters when attempting a historical rescan to - // determine if the outpoint has already been spent. - PkScript []byte + // SpendRequest represents either the outpoint or script we should + // detect the spend of. + SpendRequest // Event contains references to the channels that the notifications are // to be sent over. Event *SpendEvent // HeightHint is the earliest height in the chain that we expect to find - // the spending transaction of the specified outpoint. This value will - // be overridden by the spend hint cache if it contains an entry for it. + // the spending transaction of the specified outpoint/output script. + // This value will be overridden by the spend hint cache if it contains + // an entry for it. HeightHint uint32 // dispatched signals whether a spend notification has been disptached @@ -319,15 +318,13 @@ type SpendNtfn struct { } // HistoricalSpendDispatch parameterizes a manual rescan to determine the -// spending details (if any) of an outpoint. The parameters include the start -// and end block heights specifying the range of blocks to scan. +// spending details (if any) of an outpoint/output script. The parameters +// include the start and end block heights specifying the range of blocks to +// scan. type HistoricalSpendDispatch struct { - // OutPoint is the outpoint which we should attempt to find the spending - OutPoint wire.OutPoint - - // PkScript is the script of the outpoint. This is needed in order to - // match compact filters when attempting a historical rescan. - PkScript []byte + // SpendRequest represents either the outpoint or script we should + // detect the spend of. + SpendRequest // StartHeight specified the block height at which to begin the // historical rescan. @@ -378,14 +375,14 @@ type TxNotifier struct { ntfnsByConfirmHeight map[uint32]map[*ConfNtfn]struct{} // spendNotifications is an index of all active notification requests - // per outpoint. - spendNotifications map[wire.OutPoint]*spendNtfnSet + // per outpoint/output script. + spendNotifications map[SpendRequest]*spendNtfnSet - // opsBySpendHeight is an index that keeps tracks of the spending height - // of an outpoint we are currently tracking notifications for. This is - // used in order to recover from the spending transaction of an outpoint + // spendsByHeight is an index that keeps tracks of the spending height + // of outpoints/output scripts we are currently tracking notifications + // for. This is used in order to recover from spending transactions // being reorged out of the chain. - opsBySpendHeight map[uint32]map[wire.OutPoint]struct{} + spendsByHeight map[uint32]map[SpendRequest]struct{} // confirmHintCache is a cache used to maintain the latest height hints // for transactions/output scripts. Each height hint represents the @@ -394,8 +391,8 @@ type TxNotifier struct { confirmHintCache ConfirmHintCache // spendHintCache is a cache used to maintain the latest height hints - // for outpoints. Each height hint represents the earliest height at - // which the outpoints could have been spent within the chain. + // for outpoints/output scripts. Each height hint represents the + // earliest height at which they could have been spent within the chain. spendHintCache SpendHintCache // quit is closed in order to signal that the notifier is gracefully @@ -419,8 +416,8 @@ func NewTxNotifier(startHeight uint32, reorgSafetyLimit uint32, confNotifications: make(map[ConfRequest]*confNtfnSet), confsByInitialHeight: make(map[uint32]map[ConfRequest]struct{}), ntfnsByConfirmHeight: make(map[uint32]map[*ConfNtfn]struct{}), - spendNotifications: make(map[wire.OutPoint]*spendNtfnSet), - opsBySpendHeight: make(map[uint32]map[wire.OutPoint]struct{}), + spendNotifications: make(map[SpendRequest]*spendNtfnSet), + spendsByHeight: make(map[uint32]map[SpendRequest]struct{}), confirmHintCache: confirmHintCache, spendHintCache: spendHintCache, quit: make(chan struct{}), @@ -726,53 +723,57 @@ func (n *TxNotifier) dispatchConfDetails( } // RegisterSpend handles a new spend notification request. The client will be -// notified once the outpoint is detected as spent within the chain. +// notified once the outpoint/output script is detected as spent within the +// chain. // // The registration succeeds if no error is returned. If the returned // HistoricalSpendDisaptch is non-nil, the caller is responsible for attempting -// to determine whether the outpoint has been spent between the start and end -// heights. +// to determine whether the outpoint/output script has been spent between the +// start and end heights. The notifier's current height is also returned so that +// backends can request to be notified of spends from this point forwards. // -// NOTE: If the outpoint has already been spent within the chain before the -// notifier's current tip, the spend details must be provided with the -// UpdateSpendDetails method, otherwise we will wait for the outpoint to -// be spent at tip, even though it already has. -func (n *TxNotifier) RegisterSpend(ntfn *SpendNtfn) (*HistoricalSpendDispatch, error) { +// NOTE: If the outpoint/output script has already been spent within the chain +// before the notifier's current tip, the spend details must be provided with +// the UpdateSpendDetails method, otherwise we will wait for the outpoint/output +// script to be spent at tip, even though it already has. +func (n *TxNotifier) RegisterSpend(ntfn *SpendNtfn) (*HistoricalSpendDispatch, + uint32, error) { + select { case <-n.quit: - return nil, ErrTxNotifierExiting + return nil, 0, ErrTxNotifierExiting default: } // Before proceeding to register the notification, we'll query our spend // hint cache to determine whether a better one exists. startHeight := ntfn.HeightHint - hint, err := n.spendHintCache.QuerySpendHint(ntfn.OutPoint) + hint, err := n.spendHintCache.QuerySpendHint(ntfn.SpendRequest) if err == nil { if hint > startHeight { Log.Debugf("Using height hint %d retrieved from cache "+ - "for %v", startHeight, ntfn.OutPoint) + "for %v", startHeight, ntfn.SpendRequest) startHeight = hint } } else if err != ErrSpendHintNotFound { Log.Errorf("Unable to query spend hint for %v: %v", - ntfn.OutPoint, err) + ntfn.SpendRequest, err) } n.Lock() defer n.Unlock() - Log.Infof("New spend subscription: spend_id=%d, outpoint=%v, "+ - "height_hint=%d", ntfn.SpendID, ntfn.OutPoint, ntfn.HeightHint) + Log.Infof("New spend subscription: spend_id=%d, %v, height_hint=%d", + ntfn.SpendID, ntfn.SpendRequest, ntfn.HeightHint) // Keep track of the notification request so that we can properly // dispatch a spend notification later on. - spendSet, ok := n.spendNotifications[ntfn.OutPoint] + spendSet, ok := n.spendNotifications[ntfn.SpendRequest] if !ok { - // If this is the first registration for the outpoint, we'll + // If this is the first registration for the request, we'll // construct a spendNtfnSet to coalesce all notifications. spendSet = newSpendNtfnSet() - n.spendNotifications[ntfn.OutPoint] = spendSet + n.spendNotifications[ntfn.SpendRequest] = spendSet } spendSet.ntfns[ntfn.SpendID] = ntfn @@ -780,19 +781,28 @@ func (n *TxNotifier) RegisterSpend(ntfn *SpendNtfn) (*HistoricalSpendDispatch, e // depending on the current rescan status. switch spendSet.rescanStatus { - // If the spending details for this outpoint have already been - // determined and cached, then we can use them to immediately dispatch - // the spend notification to the client. + // If the spending details for this request have already been determined + // and cached, then we can use them to immediately dispatch the spend + // notification to the client. case rescanComplete: - return nil, n.dispatchSpendDetails(ntfn, spendSet.details) + Log.Debugf("Attempting to dispatch spend for %v on "+ + "registration since rescan has finished", + ntfn.SpendRequest) - // If there is an active rescan to determine whether the outpoint has + return nil, n.currentHeight, n.dispatchSpendDetails( + ntfn, spendSet.details, + ) + + // If there is an active rescan to determine whether the request has // been spent, then we won't trigger another one. case rescanPending: - return nil, nil + Log.Debugf("Waiting for pending rescan to finish before "+ + "notifying %v at tip", ntfn.SpendRequest) + + return nil, n.currentHeight, nil // Otherwise, we'll fall through and let the caller know that a rescan - // should be dispatched to determine whether the outpoint has already + // should be dispatched to determine whether the request has already // been spent. case rescanNotStarted: } @@ -803,30 +813,32 @@ func (n *TxNotifier) RegisterSpend(ntfn *SpendNtfn) (*HistoricalSpendDispatch, e // historical rescan and wait for the spend to come in at tip. if startHeight > n.currentHeight { Log.Debugf("Spend hint of %d for %v is above current height %d", - startHeight, ntfn.OutPoint, n.currentHeight) + startHeight, ntfn.SpendRequest, n.currentHeight) // We'll also set the rescan status as complete to ensure that - // spend hints for this outpoint get updated upon + // spend hints for this request get updated upon // connected/disconnected blocks. spendSet.rescanStatus = rescanComplete - return nil, nil + return nil, n.currentHeight, nil } // We'll set the rescan status to pending to ensure subsequent // notifications don't also attempt a historical dispatch. spendSet.rescanStatus = rescanPending + Log.Debugf("Dispatching historical spend rescan for %v", + ntfn.SpendRequest) + return &HistoricalSpendDispatch{ - OutPoint: ntfn.OutPoint, - PkScript: ntfn.PkScript, - StartHeight: startHeight, - EndHeight: n.currentHeight, - }, nil + SpendRequest: ntfn.SpendRequest, + StartHeight: startHeight, + EndHeight: n.currentHeight, + }, n.currentHeight, nil } // CancelSpend cancels an existing request for a spend notification of an -// outpoint. The request is identified by its spend ID. -func (n *TxNotifier) CancelSpend(op wire.OutPoint, spendID uint64) { +// outpoint/output script. The request is identified by its spend ID. +func (n *TxNotifier) CancelSpend(spendRequest SpendRequest, spendID uint64) { select { case <-n.quit: return @@ -836,10 +848,10 @@ func (n *TxNotifier) CancelSpend(op wire.OutPoint, spendID uint64) { n.Lock() defer n.Unlock() - Log.Infof("Canceling spend notification: spend_id=%d, outpoint=%v", - spendID, op) + Log.Infof("Canceling spend notification: spend_id=%d, %v", spendID, + spendRequest) - spendSet, ok := n.spendNotifications[op] + spendSet, ok := n.spendNotifications[spendRequest] if !ok { return } @@ -904,13 +916,13 @@ func (n *TxNotifier) ProcessRelevantSpendTx(tx *wire.MsgTx, txHeight int32) erro } // UpdateSpendDetails attempts to update the spend details for all active spend -// notification requests for an outpoint. This method should be used once a -// historical scan of the chain has finished. If the historical scan did not -// find a spending transaction for the outpoint, the spend details may be nil. +// notification requests for an outpoint/output script. This method should be +// used once a historical scan of the chain has finished. If the historical scan +// did not find a spending transaction for it, the spend details may be nil. // -// NOTE: A notification request for the outpoint must be registered first to -// ensure notifications are delivered. -func (n *TxNotifier) UpdateSpendDetails(op wire.OutPoint, +// NOTE: A notification request for the outpoint/output script must be +// registered first to ensure notifications are delivered. +func (n *TxNotifier) UpdateSpendDetails(spendRequest SpendRequest, details *SpendDetail) error { select { @@ -924,24 +936,24 @@ func (n *TxNotifier) UpdateSpendDetails(op wire.OutPoint, n.Lock() defer n.Unlock() - return n.updateSpendDetails(op, details) + return n.updateSpendDetails(spendRequest, details) } // updateSpendDetails attempts to update the spend details for all active spend -// notification requests for an outpoint. This method should be used once a -// historical scan of the chain has finished. If the historical scan did not -// find a spending transaction for the outpoint, the spend details may be nil. +// notification requests for an outpoint/output script. This method should be +// used once a historical scan of the chain has finished. If the historical scan +// did not find a spending transaction for it, the spend details may be nil. // // NOTE: This method must be called with the TxNotifier's lock held. -func (n *TxNotifier) updateSpendDetails(op wire.OutPoint, +func (n *TxNotifier) updateSpendDetails(spendRequest SpendRequest, details *SpendDetail) error { - // Mark the ongoing historical rescan for this outpoint as finished. - // This will allow us to update the spend hints for this outpoint at - // tip. - spendSet, ok := n.spendNotifications[op] + // Mark the ongoing historical rescan for this request as finished. This + // will allow us to update the spend hints for it at tip. + spendSet, ok := n.spendNotifications[spendRequest] if !ok { - return fmt.Errorf("no notifications found for outpoint %v", op) + return fmt.Errorf("spend notification for %v not found", + spendRequest) } // If the spend details have already been found either at tip, then the @@ -951,31 +963,33 @@ func (n *TxNotifier) updateSpendDetails(op wire.OutPoint, return nil } - // Since the historical rescan has completed for this outpoint, we'll + // Since the historical rescan has completed for this request, we'll // mark its rescan status as complete in order to ensure that the // TxNotifier can properly update its spend hints upon // connected/disconnected blocks. spendSet.rescanStatus = rescanComplete // If the historical rescan was not able to find a spending transaction - // for this outpoint, then we can track the spend at tip. + // for this request, then we can track the spend at tip. if details == nil { // We'll commit the current height as the spend hint to prevent // another potentially long rescan if we restart before a new // block comes in. - err := n.spendHintCache.CommitSpendHint(n.currentHeight, op) + err := n.spendHintCache.CommitSpendHint( + n.currentHeight, spendRequest, + ) if err != nil { // The error is not fatal as this is an optimistic // optimization, so we'll avoid returning an error. Log.Debugf("Unable to update spend hint to %d for %v: %v", - n.currentHeight, op, err) + n.currentHeight, spendRequest, err) } return nil } // If the historical rescan found the spending transaction for this - // outpoint, but it's at a later height than the notifier (this can + // request, but it's at a later height than the notifier (this can // happen due to latency with the backend during a reorg), then we'll // defer handling the notification until the notifier has caught up to // such height. @@ -983,17 +997,17 @@ func (n *TxNotifier) updateSpendDetails(op wire.OutPoint, return nil } - // Now that we've determined the outpoint has been spent, we'll commit + // Now that we've determined the request has been spent, we'll commit // its spending height as its hint in the cache and dispatch // notifications to all of its respective clients. err := n.spendHintCache.CommitSpendHint( - uint32(details.SpendingHeight), op, + uint32(details.SpendingHeight), spendRequest, ) if err != nil { // The error is not fatal as this is an optimistic optimization, // so we'll avoid returning an error. Log.Debugf("Unable to update spend hint to %d for %v: %v", - details.SpendingHeight, op, err) + details.SpendingHeight, spendRequest, err) } spendSet.details = details @@ -1017,8 +1031,8 @@ func (n *TxNotifier) dispatchSpendDetails(ntfn *SpendNtfn, details *SpendDetail) return nil } - Log.Infof("Dispatching spend notification for outpoint=%v at height=%d", - ntfn.OutPoint, n.currentHeight) + Log.Infof("Dispatching confirmed spend notification for %v at height=%d", + ntfn.SpendRequest, n.currentHeight) select { case ntfn.Event.Spend <- details: