chainntnfs/neutrinonotify: handle spend notification registration w/ TxNotifier
In this commit, we modify the logic within RegisterSpendNtfn for the NeutrinoNotifier to account for the recent changes made to the TxNotifier. Since it is now able to handle spend notification registration and dispatch, we can bypass all the current logic within the NeutrinoNotifier and interact directly with the TxNotifier instead. The most notable change is that now we'll only attempt a historical rescan if the TxNotifier tells us so.
This commit is contained in:
parent
74139c9a3f
commit
bfd11a251e
@ -164,6 +164,7 @@ func (n *NeutrinoNotifier) Start() error {
|
|||||||
|
|
||||||
n.txNotifier = chainntnfs.NewTxNotifier(
|
n.txNotifier = chainntnfs.NewTxNotifier(
|
||||||
n.bestHeight, reorgSafetyLimit, n.confirmHintCache,
|
n.bestHeight, reorgSafetyLimit, n.confirmHintCache,
|
||||||
|
n.spendHintCache,
|
||||||
)
|
)
|
||||||
|
|
||||||
n.chainConn = &NeutrinoChainConn{n.p2pNode}
|
n.chainConn = &NeutrinoChainConn{n.p2pNode}
|
||||||
@ -603,68 +604,6 @@ func (n *NeutrinoNotifier) handleBlockConnected(newBlock *filteredBlock) error {
|
|||||||
chainntnfs.Log.Infof("New block: height=%v, sha=%v", newBlock.height,
|
chainntnfs.Log.Infof("New block: height=%v, sha=%v", newBlock.height,
|
||||||
newBlock.hash)
|
newBlock.hash)
|
||||||
|
|
||||||
// Create a helper struct for coalescing spend notifications triggered
|
|
||||||
// by this block.
|
|
||||||
type spendNtfnBatch struct {
|
|
||||||
details *chainntnfs.SpendDetail
|
|
||||||
clients map[uint64]*spendNotification
|
|
||||||
}
|
|
||||||
|
|
||||||
// Scan over the list of relevant transactions and assemble the
|
|
||||||
// possible spend notifications we need to dispatch.
|
|
||||||
spendBatches := make(map[wire.OutPoint]spendNtfnBatch)
|
|
||||||
for _, tx := range newBlock.txns {
|
|
||||||
mtx := tx.MsgTx()
|
|
||||||
txSha := mtx.TxHash()
|
|
||||||
|
|
||||||
for i, txIn := range mtx.TxIn {
|
|
||||||
prevOut := txIn.PreviousOutPoint
|
|
||||||
|
|
||||||
// If this transaction indeed does spend an output which
|
|
||||||
// we have a registered notification for, then create a
|
|
||||||
// spend summary and add it to our batch of spend
|
|
||||||
// notifications to be delivered.
|
|
||||||
clients, ok := n.spendNotifications[prevOut]
|
|
||||||
if !ok {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
delete(n.spendNotifications, prevOut)
|
|
||||||
|
|
||||||
spendDetails := &chainntnfs.SpendDetail{
|
|
||||||
SpentOutPoint: &prevOut,
|
|
||||||
SpenderTxHash: &txSha,
|
|
||||||
SpendingTx: mtx,
|
|
||||||
SpenderInputIndex: uint32(i),
|
|
||||||
SpendingHeight: int32(newBlock.height),
|
|
||||||
}
|
|
||||||
|
|
||||||
spendBatches[prevOut] = spendNtfnBatch{
|
|
||||||
details: spendDetails,
|
|
||||||
clients: clients,
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Now, we'll update the spend height hint for all of our watched
|
|
||||||
// outpoints that have not been spent yet. This is safe to do as we do
|
|
||||||
// not watch already spent outpoints for spend notifications.
|
|
||||||
ops := make([]wire.OutPoint, 0, len(n.spendNotifications))
|
|
||||||
for op := range n.spendNotifications {
|
|
||||||
ops = append(ops, op)
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(ops) > 0 {
|
|
||||||
err := n.spendHintCache.CommitSpendHint(newBlock.height, ops...)
|
|
||||||
if err != nil {
|
|
||||||
// The error is not fatal since we are connecting a
|
|
||||||
// block, and advancing the spend hint is an optimistic
|
|
||||||
// optimization.
|
|
||||||
chainntnfs.Log.Errorf("Unable to update spend hint to "+
|
|
||||||
"%d for %v: %v", newBlock.height, ops, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// We want to set the best block before dispatching notifications
|
// We want to set the best block before dispatching notifications
|
||||||
// so if any subscribers make queries based on their received
|
// so if any subscribers make queries based on their received
|
||||||
// block epoch, our state is fully updated in time.
|
// block epoch, our state is fully updated in time.
|
||||||
@ -674,23 +613,6 @@ func (n *NeutrinoNotifier) handleBlockConnected(newBlock *filteredBlock) error {
|
|||||||
// of the block.
|
// of the block.
|
||||||
n.notifyBlockEpochs(int32(newBlock.height), &newBlock.hash)
|
n.notifyBlockEpochs(int32(newBlock.height), &newBlock.hash)
|
||||||
|
|
||||||
// Finally, send off the spend details to the notification subscribers.
|
|
||||||
for _, batch := range spendBatches {
|
|
||||||
for _, ntfn := range batch.clients {
|
|
||||||
chainntnfs.Log.Infof("Dispatching spend "+
|
|
||||||
"notification for outpoint=%v",
|
|
||||||
ntfn.targetOutpoint)
|
|
||||||
|
|
||||||
ntfn.spendChan <- batch.details
|
|
||||||
|
|
||||||
// Close spendChan to ensure that any calls to
|
|
||||||
// Cancel will not block. This is safe to do
|
|
||||||
// since the channel is buffered, and the
|
|
||||||
// message can still be read by the receiver.
|
|
||||||
close(ntfn.spendChan)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -766,86 +688,59 @@ type spendCancel struct {
|
|||||||
func (n *NeutrinoNotifier) RegisterSpendNtfn(outpoint *wire.OutPoint,
|
func (n *NeutrinoNotifier) RegisterSpendNtfn(outpoint *wire.OutPoint,
|
||||||
pkScript []byte, heightHint uint32) (*chainntnfs.SpendEvent, error) {
|
pkScript []byte, heightHint uint32) (*chainntnfs.SpendEvent, error) {
|
||||||
|
|
||||||
n.heightMtx.RLock()
|
// First, we'll construct a spend notification request and hand it off
|
||||||
currentHeight := n.bestHeight
|
// to the txNotifier.
|
||||||
n.heightMtx.RUnlock()
|
spendID := atomic.AddUint64(&n.spendClientCounter, 1)
|
||||||
|
cancel := func() {
|
||||||
// Before proceeding to register the notification, we'll query our
|
n.txNotifier.CancelSpend(*outpoint, spendID)
|
||||||
// height hint cache to determine whether a better one exists.
|
}
|
||||||
if hint, err := n.spendHintCache.QuerySpendHint(*outpoint); err == nil {
|
ntfn := &chainntnfs.SpendNtfn{
|
||||||
if hint > heightHint {
|
SpendID: spendID,
|
||||||
chainntnfs.Log.Debugf("Using height hint %d retrieved "+
|
OutPoint: *outpoint,
|
||||||
"from cache for %v", hint, outpoint)
|
Event: chainntnfs.NewSpendEvent(cancel),
|
||||||
heightHint = hint
|
HeightHint: heightHint,
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Construct a notification request for the outpoint. We'll defer
|
historicalDispatch, err := n.txNotifier.RegisterSpend(ntfn)
|
||||||
// sending it to the main event loop until after we've guaranteed that
|
if err != nil {
|
||||||
// the outpoint has not been spent.
|
return nil, err
|
||||||
ntfn := &spendNotification{
|
|
||||||
targetOutpoint: outpoint,
|
|
||||||
spendChan: make(chan *chainntnfs.SpendDetail, 1),
|
|
||||||
spendID: atomic.AddUint64(&n.spendClientCounter, 1),
|
|
||||||
heightHint: heightHint,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
spendEvent := &chainntnfs.SpendEvent{
|
// If the txNotifier didn't return any details to perform a historical
|
||||||
Spend: ntfn.spendChan,
|
// scan of the chain, then we can return early as there's nothing left
|
||||||
Cancel: func() {
|
// for us to do.
|
||||||
cancel := &spendCancel{
|
if historicalDispatch == nil {
|
||||||
op: *outpoint,
|
return ntfn.Event, nil
|
||||||
spendID: ntfn.spendID,
|
|
||||||
}
|
|
||||||
|
|
||||||
// Submit spend cancellation to notification dispatcher.
|
|
||||||
select {
|
|
||||||
case n.notificationCancels <- cancel:
|
|
||||||
// Cancellation is being handled, drain the
|
|
||||||
// spend chan until it is closed before yielding
|
|
||||||
// to the caller.
|
|
||||||
for {
|
|
||||||
select {
|
|
||||||
case _, ok := <-ntfn.spendChan:
|
|
||||||
if !ok {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
case <-n.quit:
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case <-n.quit:
|
|
||||||
}
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure that neutrino is caught up to the height hint before we
|
// Ensure that neutrino is caught up to the height hint before we
|
||||||
// attempt to fetch the utxo from the chain. If we're behind, then we
|
// attempt to fetch the UTXO from the chain. If we're behind, then we
|
||||||
// may miss a notification dispatch.
|
// may miss a notification dispatch.
|
||||||
for {
|
for {
|
||||||
n.heightMtx.RLock()
|
n.heightMtx.RLock()
|
||||||
currentHeight = n.bestHeight
|
currentHeight := n.bestHeight
|
||||||
n.heightMtx.RUnlock()
|
n.heightMtx.RUnlock()
|
||||||
|
|
||||||
if currentHeight < heightHint {
|
if currentHeight >= historicalDispatch.StartHeight {
|
||||||
time.Sleep(time.Millisecond * 200)
|
break
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
|
|
||||||
break
|
time.Sleep(time.Millisecond * 200)
|
||||||
}
|
|
||||||
|
|
||||||
inputToWatch := neutrino.InputWithScript{
|
|
||||||
OutPoint: *outpoint,
|
|
||||||
PkScript: pkScript,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Before sending off the notification request, we'll attempt to see if
|
// Before sending off the notification request, we'll attempt to see if
|
||||||
// this output is still spent or not at this point in the chain.
|
// this output is still spent or not at this point in the chain.
|
||||||
|
inputToWatch := neutrino.InputWithScript{
|
||||||
|
OutPoint: *outpoint,
|
||||||
|
PkScript: pkScript,
|
||||||
|
}
|
||||||
spendReport, err := n.p2pNode.GetUtxo(
|
spendReport, err := n.p2pNode.GetUtxo(
|
||||||
neutrino.WatchInputs(inputToWatch),
|
neutrino.WatchInputs(inputToWatch),
|
||||||
neutrino.StartBlock(&waddrmgr.BlockStamp{
|
neutrino.StartBlock(&waddrmgr.BlockStamp{
|
||||||
Height: int32(heightHint),
|
Height: int32(historicalDispatch.StartHeight),
|
||||||
|
}),
|
||||||
|
neutrino.EndBlock(&waddrmgr.BlockStamp{
|
||||||
|
Height: int32(historicalDispatch.EndHeight),
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
if err != nil && !strings.Contains(err.Error(), "not found") {
|
if err != nil && !strings.Contains(err.Error(), "not found") {
|
||||||
@ -857,30 +752,33 @@ func (n *NeutrinoNotifier) RegisterSpendNtfn(outpoint *wire.OutPoint,
|
|||||||
if spendReport != nil && spendReport.SpendingTx != nil {
|
if spendReport != nil && spendReport.SpendingTx != nil {
|
||||||
// As a result, we'll launch a goroutine to immediately
|
// As a result, we'll launch a goroutine to immediately
|
||||||
// dispatch the notification with a normal response.
|
// dispatch the notification with a normal response.
|
||||||
go func() {
|
spendingTxHash := spendReport.SpendingTx.TxHash()
|
||||||
txSha := spendReport.SpendingTx.TxHash()
|
spendDetails := &chainntnfs.SpendDetail{
|
||||||
select {
|
SpentOutPoint: outpoint,
|
||||||
case ntfn.spendChan <- &chainntnfs.SpendDetail{
|
SpenderTxHash: &spendingTxHash,
|
||||||
SpentOutPoint: outpoint,
|
SpendingTx: spendReport.SpendingTx,
|
||||||
SpenderTxHash: &txSha,
|
SpenderInputIndex: spendReport.SpendingInputIndex,
|
||||||
SpendingTx: spendReport.SpendingTx,
|
SpendingHeight: int32(spendReport.SpendingTxHeight),
|
||||||
SpenderInputIndex: spendReport.SpendingInputIndex,
|
}
|
||||||
SpendingHeight: int32(spendReport.SpendingTxHeight),
|
|
||||||
}:
|
|
||||||
case <-n.quit:
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
}()
|
err := n.txNotifier.UpdateSpendDetails(*outpoint, spendDetails)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
return spendEvent, nil
|
return ntfn.Event, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the output is still unspent, then we'll mark our historical rescan
|
||||||
|
// as complete and update our rescan's filter to watch for the spend of
|
||||||
|
// the outpoint in question.
|
||||||
|
if err := n.txNotifier.UpdateSpendDetails(*outpoint, nil); err != nil {
|
||||||
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the output is still unspent, then we'll update our rescan's
|
|
||||||
// filter, and send the request to the dispatcher goroutine.
|
|
||||||
rescanUpdate := []neutrino.UpdateOption{
|
rescanUpdate := []neutrino.UpdateOption{
|
||||||
neutrino.AddInputs(inputToWatch),
|
neutrino.AddInputs(inputToWatch),
|
||||||
neutrino.Rewind(currentHeight),
|
neutrino.Rewind(historicalDispatch.EndHeight),
|
||||||
neutrino.DisableDisconnectedNtfns(true),
|
neutrino.DisableDisconnectedNtfns(true),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -50,6 +50,7 @@ func (n *NeutrinoNotifier) UnsafeStart(bestHeight int32,
|
|||||||
|
|
||||||
n.txNotifier = chainntnfs.NewTxNotifier(
|
n.txNotifier = chainntnfs.NewTxNotifier(
|
||||||
uint32(bestHeight), reorgSafetyLimit, n.confirmHintCache,
|
uint32(bestHeight), reorgSafetyLimit, n.confirmHintCache,
|
||||||
|
n.spendHintCache,
|
||||||
)
|
)
|
||||||
|
|
||||||
n.chainConn = &NeutrinoChainConn{n.p2pNode}
|
n.chainConn = &NeutrinoChainConn{n.p2pNode}
|
||||||
|
Loading…
Reference in New Issue
Block a user