chainntnfs/neutrinonotify: fix early historical confirmation dispatch

In this commit, we fix an existing bug within the logic of the neutrino
notifier. Rather than properly dispatching only once a transaction had
reached the expected number of confirmations, the historical dispatch
logic would trigger as soon as the transaction reached a single
confirmation.

This was due to the fact that we were using the scanHeight variable
which would be set to zero to calculate the number of confirmations.
The value would end up being the current height, which is generally
always greater than the number of expected confirmations. To remedy
this, we’ll now properly use the block height the transaction was
originally confirmed in to know when to dispatch.

This also applies a fix that was discovered in
93981a85c0b47622a3a5e7089b8bca9b80b834c5.
This commit is contained in:
Olaoluwa Osuntokun 2017-12-07 19:08:40 -08:00
parent d329502e8d
commit 36c299c1d8
No known key found for this signature in database
GPG Key ID: 964EA263DD637C21

@ -419,10 +419,7 @@ func (n *NeutrinoNotifier) attemptHistoricalDispatch(msg *confirmationsNotificat
targetHash := msg.txid targetHash := msg.txid
var ( var confDetails *chainntnfs.TxConfirmation
confDetails *chainntnfs.TxConfirmation
scanHeight uint32
)
chainntnfs.Log.Infof("Attempting to trigger dispatch for %v from "+ chainntnfs.Log.Infof("Attempting to trigger dispatch for %v from "+
"historical chain", msg.txid) "historical chain", msg.txid)
@ -503,20 +500,21 @@ chainScan:
// Otherwise, we'll calculate the number of confirmations that the // Otherwise, we'll calculate the number of confirmations that the
// transaction has so we can decide if it has reached the desired // transaction has so we can decide if it has reached the desired
// number of confirmations or not. // number of confirmations or not.
txConfs := currentHeight - scanHeight txConfs := currentHeight - confDetails.BlockHeight + 1
// If the transaction has more that enough confirmations, then we can // If the transaction has more that enough confirmations, then we can
// dispatch it immediately after obtaining for information w.r.t // dispatch it immediately after obtaining for information w.r.t
// exactly *when* if got all its confirmations. // exactly *when* if got all its confirmations.
if uint32(txConfs) >= msg.numConfirmations { if uint32(txConfs) >= msg.numConfirmations {
chainntnfs.Log.Infof("Dispatching %v conf notification, "+
"height=%v", msg.numConfirmations, currentHeight)
msg.finConf <- confDetails msg.finConf <- confDetails
return true return true
} }
// Otherwise, the transaction has only been *partially* confirmed, so // Otherwise, the transaction has only been *partially* confirmed, so
// we need to insert it into the confirmation heap. // we need to insert it into the confirmation heap.
confsLeft := msg.numConfirmations - uint32(txConfs) confHeight := confDetails.BlockHeight + msg.numConfirmations - 1
confHeight := uint32(currentHeight) + confsLeft
heapEntry := &confEntry{ heapEntry := &confEntry{
msg, msg,
confDetails, confDetails,
@ -524,7 +522,7 @@ chainScan:
} }
heap.Push(n.confHeap, heapEntry) heap.Push(n.confHeap, heapEntry)
return false return true
} }
// notifyBlockEpochs notifies all registered block epoch clients of the newly // notifyBlockEpochs notifies all registered block epoch clients of the newly
@ -574,6 +572,9 @@ func (n *NeutrinoNotifier) notifyConfs(newBlockHeight int32) {
nextConf := heap.Pop(n.confHeap).(*confEntry) nextConf := heap.Pop(n.confHeap).(*confEntry)
for nextConf.triggerHeight <= uint32(newBlockHeight) { for nextConf.triggerHeight <= uint32(newBlockHeight) {
chainntnfs.Log.Infof("Dispatching %v conf notification, "+
"height=%v", nextConf.numConfirmations, newBlockHeight)
nextConf.finConf <- nextConf.initialConfDetails nextConf.finConf <- nextConf.initialConfDetails
if n.confHeap.Len() == 0 { if n.confHeap.Len() == 0 {