c7402f3462
* Re-orgs are still unhanded. * RegisterSpendNtfn should perhaps also register directly with the rpc client instead of pushing the responsibility to the caller.
269 lines
7.1 KiB
Go
269 lines
7.1 KiB
Go
package btcdnotify
|
|
|
|
import (
|
|
"container/heap"
|
|
"fmt"
|
|
"sync"
|
|
"sync/atomic"
|
|
|
|
"github.com/btcsuite/btcd/wire"
|
|
"github.com/btcsuite/btcwallet/chain"
|
|
"github.com/btcsuite/btcwallet/wtxmgr"
|
|
"github.com/lightningnetwork/lnd/chainntfs"
|
|
)
|
|
|
|
// BtcdNotifier...
|
|
type BtcdNotifier struct {
|
|
started int32 // To be used atomically
|
|
stopped int32 // To be used atomically
|
|
|
|
// TODO(roasbeef): refactor to use the new NotificationServer
|
|
conn ChainConnection
|
|
|
|
notificationRegistry chan interface{}
|
|
|
|
// TODO(roasbeef): make map point to slices? Would allow for multiple
|
|
// clients to listen for same spend. Would we ever need this?
|
|
spendNotifications map[wire.OutPoint]*spendNotification
|
|
confNotifications map[wire.ShaHash]*confirmationsNotification
|
|
confHeap *confirmationHeap
|
|
|
|
connectedBlocks <-chan wtxmgr.BlockMeta
|
|
disconnectedBlocks <-chan wtxmgr.BlockMeta
|
|
relevantTxs <-chan chain.RelevantTx
|
|
|
|
rpcConnected chan struct{}
|
|
|
|
wg sync.WaitGroup
|
|
quit chan struct{}
|
|
}
|
|
|
|
var _ chainntnfs.ChainNotifier = (*BtcdNotifier)(nil)
|
|
|
|
// NewBtcdNotifier...
|
|
func NewBtcdNotifier(c ChainConnection) (*BtcdNotifier, error) {
|
|
// TODO(roasbeef): take client also in order to get notifications?
|
|
return &BtcdNotifier{
|
|
conn: c,
|
|
notificationRegistry: make(chan interface{}),
|
|
|
|
spendNotifications: make(map[wire.OutPoint]*spendNotification),
|
|
confNotifications: make(map[wire.ShaHash]*confirmationsNotification),
|
|
confHeap: newConfirmationHeap(),
|
|
|
|
connectedBlocks: make(chan wtxmgr.BlockMeta),
|
|
disconnectedBlocks: make(chan wtxmgr.BlockMeta),
|
|
relevantTxs: make(chan chain.RelevantTx),
|
|
|
|
rpcConnected: make(chan struct{}, 1),
|
|
|
|
quit: make(chan struct{}),
|
|
}, nil
|
|
}
|
|
|
|
// Start...
|
|
func (b *BtcdNotifier) Start() error {
|
|
// Already started?
|
|
if atomic.AddInt32(&b.started, 1) != 1 {
|
|
return nil
|
|
}
|
|
|
|
b.wg.Add(1)
|
|
go b.notificationDispatcher()
|
|
b.rpcConnected <- struct{}{} // TODO(roasbeef) why?
|
|
|
|
return nil
|
|
}
|
|
|
|
// Stop...
|
|
func (b *BtcdNotifier) Stop() error {
|
|
// Already shutting down?
|
|
if atomic.AddInt32(&b.stopped, 1) != 1 {
|
|
return nil
|
|
}
|
|
|
|
close(b.quit)
|
|
b.wg.Wait()
|
|
|
|
return nil
|
|
}
|
|
|
|
// notificationDispatcher...
|
|
func (b *BtcdNotifier) notificationDispatcher() {
|
|
out:
|
|
for {
|
|
select {
|
|
case <-b.rpcConnected:
|
|
err := b.initAllNotifications()
|
|
fmt.Println(err)
|
|
case registerMsg := <-b.notificationRegistry:
|
|
switch msg := registerMsg.(type) {
|
|
case *spendNotification:
|
|
b.spendNotifications[*msg.targetOutpoint] = msg
|
|
case *confirmationsNotification:
|
|
b.confNotifications[*msg.txid] = msg
|
|
}
|
|
case txNtfn := <-b.relevantTxs:
|
|
tx := txNtfn.TxRecord.MsgTx
|
|
txMined := txNtfn.Block != nil
|
|
|
|
// First, check if this transaction spends an output
|
|
// that has an existing spend notification for it.
|
|
for i, txIn := range tx.TxIn {
|
|
prevOut := txIn.PreviousOutPoint
|
|
|
|
// If this transaction indeed does spend an
|
|
// output which we have a registered notification
|
|
// for, then create a spend summary, finally
|
|
// sending off the details to the notification
|
|
// subscriber.
|
|
if ntfn, ok := b.spendNotifications[prevOut]; ok {
|
|
spenderSha := tx.TxSha()
|
|
spendDetails := &chainntnfs.SpendDetail{
|
|
SpentOutPoint: ntfn.targetOutpoint,
|
|
SpenderTxHash: &spenderSha,
|
|
// TODO(roasbeef): copy tx?
|
|
SpendingTx: &tx,
|
|
SpenderInputIndex: uint32(i),
|
|
}
|
|
|
|
ntfn.spendChan <- spendDetails
|
|
delete(b.spendNotifications, prevOut)
|
|
}
|
|
}
|
|
|
|
// If the transaction has been mined, then we check if
|
|
// a notification for the confirmation of this txid has
|
|
// been registered previously. Otherwise, we're done,
|
|
// for now.
|
|
if !txMined {
|
|
break
|
|
}
|
|
|
|
// If a confirmation notification has been registered
|
|
// for this txid, then either trigger a notification
|
|
// event if only a single confirmation notification was
|
|
// requested, or place the notification on the
|
|
// confirmation heap for future usage.
|
|
if confNtfn, ok := b.confNotifications[tx.TxSha()]; ok {
|
|
if confNtfn.numConfirmations == 1 {
|
|
confNtfn.finConf <- struct{}{}
|
|
break
|
|
}
|
|
|
|
// The registered notification requires more
|
|
// than one confirmation before triggering. So
|
|
// we create a heapConf entry for this notification.
|
|
// The heapConf allows us to easily keep track of
|
|
// which notification(s) we should fire off with
|
|
// each incoming block.
|
|
confNtfn.initialConfirmHeight = uint32(txNtfn.Block.Height)
|
|
heapEntry := &confEntry{
|
|
confNtfn,
|
|
confNtfn.initialConfirmHeight + confNtfn.numConfirmations,
|
|
}
|
|
heap.Push(b.confHeap, heapEntry)
|
|
}
|
|
case blockNtfn := <-b.connectedBlocks:
|
|
blockHeight := uint32(blockNtfn.Height)
|
|
|
|
// Traverse our confirmation heap. The heap is a
|
|
// min-heap, so the confirmation notification which requires
|
|
// the smallest block-height will always be at the top
|
|
// of the heap. If a confirmation notification is eligible
|
|
// for triggering, then fire it off, and check if another
|
|
// is eligible until there are no more eligible entries.
|
|
nextConf := heap.Pop(b.confHeap).(*confEntry)
|
|
for nextConf.triggerHeight <= blockHeight {
|
|
nextConf.finConf <- struct{}{}
|
|
|
|
nextConf = heap.Pop(b.confHeap).(*confEntry)
|
|
}
|
|
|
|
heap.Push(b.confHeap, nextConf)
|
|
case delBlockNtfn := <-b.disconnectedBlocks:
|
|
// TODO(roasbeef): re-orgs
|
|
// * second channel to notify of confirmation decrementing
|
|
// re-org?
|
|
// * notify of negative confirmations
|
|
fmt.Println(delBlockNtfn)
|
|
case <-b.quit:
|
|
break out
|
|
}
|
|
}
|
|
}
|
|
|
|
// initAllNotifications...
|
|
func (b *BtcdNotifier) initAllNotifications() error {
|
|
var err error
|
|
|
|
b.connectedBlocks, err = b.conn.ListenConnectedBlocks()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.disconnectedBlocks, err = b.conn.ListenDisconnectedBlocks()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.relevantTxs, err = b.conn.ListenRelevantTxs()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// spendNotification....
|
|
type spendNotification struct {
|
|
targetOutpoint *wire.OutPoint
|
|
|
|
spendChan chan *chainntnfs.SpendDetail
|
|
}
|
|
|
|
// RegisterSpendNotification...
|
|
// NOTE: eventChan MUST be buffered
|
|
func (b *BtcdNotifier) RegisterSpendNtfn(outpoint *wire.OutPoint) (*chainntnfs.SpendEvent, error) {
|
|
|
|
// TODO(roasbeef): also register with rpc client? bool?
|
|
|
|
ntfn := &spendNotification{
|
|
targetOutpoint: outpoint,
|
|
spendChan: make(chan *chainntnfs.SpendDetail, 1),
|
|
}
|
|
|
|
b.notificationRegistry <- ntfn
|
|
|
|
return &chainntnfs.SpendEvent{ntfn.spendChan}, nil
|
|
}
|
|
|
|
// confirmationNotification...
|
|
// TODO(roasbeef): re-org funny business
|
|
type confirmationsNotification struct {
|
|
txid *wire.ShaHash
|
|
|
|
initialConfirmHeight uint32
|
|
numConfirmations uint32
|
|
|
|
finConf chan struct{}
|
|
negativeConf chan uint32
|
|
}
|
|
|
|
// RegisterConfirmationsNotification...
|
|
func (b *BtcdNotifier) RegisterConfirmationsNtfn(txid *wire.ShaHash,
|
|
numConfs uint32) (*chainntnfs.ConfirmationEvent, error) {
|
|
|
|
ntfn := &confirmationsNotification{
|
|
txid: txid,
|
|
numConfirmations: numConfs,
|
|
finConf: make(chan struct{}, 1),
|
|
negativeConf: make(chan uint32, 1),
|
|
}
|
|
|
|
b.notificationRegistry <- ntfn
|
|
|
|
return &chainntnfs.ConfirmationEvent{
|
|
Confirmed: ntfn.finConf,
|
|
NegativeConf: ntfn.negativeConf,
|
|
}, nil
|
|
}
|