Add non UTXO mempool implementation
This commit is contained in:
parent
4e43f0d482
commit
d1a047c667
@ -73,7 +73,7 @@ func NewBitcoinRPC(config json.RawMessage, pushHandler func(bchain.NotificationT
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (b *BitcoinRPC) Initialize() error {
|
func (b *BitcoinRPC) Initialize() error {
|
||||||
b.Mempool = bchain.NewMempool(b)
|
b.Mempool = bchain.NewUTXOMempool(b)
|
||||||
|
|
||||||
chainName, err := b.GetBlockChainInfo()
|
chainName, err := b.GetBlockChainInfo()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@ -25,7 +25,7 @@ func NewZCashRPC(config json.RawMessage, pushHandler func(bchain.NotificationTyp
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (z *ZCashRPC) Initialize() error {
|
func (z *ZCashRPC) Initialize() error {
|
||||||
z.Mempool = bchain.NewMempool(z)
|
z.Mempool = bchain.NewUTXOMempool(z)
|
||||||
z.Parser = &ZCashBlockParser{}
|
z.Parser = &ZCashBlockParser{}
|
||||||
z.Testnet = false
|
z.Testnet = false
|
||||||
z.Network = "livenet"
|
z.Network = "livenet"
|
||||||
|
|||||||
105
bchain/mempool_nonutxo.go
Normal file
105
bchain/mempool_nonutxo.go
Normal file
@ -0,0 +1,105 @@
|
|||||||
|
package bchain
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/golang/glog"
|
||||||
|
)
|
||||||
|
|
||||||
|
// NonUTXOMempool is mempool handle of non UTXO chains
|
||||||
|
type NonUTXOMempool struct {
|
||||||
|
chain BlockChain
|
||||||
|
mux sync.Mutex
|
||||||
|
txToInputOutput map[string][]addrIndex
|
||||||
|
addrIDToTx map[string][]outpoint
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewNonUTXOMempool creates new mempool handler.
|
||||||
|
func NewNonUTXOMempool(chain BlockChain) *NonUTXOMempool {
|
||||||
|
return &NonUTXOMempool{chain: chain}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetTransactions returns slice of mempool transactions for given address
|
||||||
|
func (m *NonUTXOMempool) GetTransactions(address string) ([]string, error) {
|
||||||
|
m.mux.Lock()
|
||||||
|
defer m.mux.Unlock()
|
||||||
|
parser := m.chain.GetChainParser()
|
||||||
|
addrID, err := parser.GetAddrIDFromAddress(address)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
outpoints := m.addrIDToTx[string(addrID)]
|
||||||
|
txs := make([]string, 0, len(outpoints))
|
||||||
|
for _, o := range outpoints {
|
||||||
|
txs = append(txs, o.txid)
|
||||||
|
}
|
||||||
|
return txs, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *NonUTXOMempool) updateMappings(newTxToInputOutput map[string][]addrIndex, newAddrIDToTx map[string][]outpoint) {
|
||||||
|
m.mux.Lock()
|
||||||
|
defer m.mux.Unlock()
|
||||||
|
m.txToInputOutput = newTxToInputOutput
|
||||||
|
m.addrIDToTx = newAddrIDToTx
|
||||||
|
}
|
||||||
|
|
||||||
|
// Resync gets mempool transactions and maps outputs to transactions.
|
||||||
|
// Resync is not reentrant, it should be called from a single thread.
|
||||||
|
// Read operations (GetTransactions) are safe.
|
||||||
|
func (m *NonUTXOMempool) Resync(onNewTxAddr func(txid string, addr string)) error {
|
||||||
|
start := time.Now()
|
||||||
|
glog.V(1).Info("Mempool: resync")
|
||||||
|
txs, err := m.chain.GetMempool()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
parser := m.chain.GetChainParser()
|
||||||
|
newTxToInputOutput := make(map[string][]addrIndex, len(m.txToInputOutput)+1)
|
||||||
|
newAddrIDToTx := make(map[string][]outpoint, len(m.addrIDToTx)+1)
|
||||||
|
for _, txid := range txs {
|
||||||
|
io, exists := m.txToInputOutput[txid]
|
||||||
|
if !exists {
|
||||||
|
tx, err := m.chain.GetTransaction(txid)
|
||||||
|
if err != nil {
|
||||||
|
glog.Error("cannot get transaction ", txid, ": ", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
io = make([]addrIndex, 0, len(tx.Vout)+len(tx.Vin))
|
||||||
|
for _, output := range tx.Vout {
|
||||||
|
addrID, err := parser.GetAddrIDFromVout(&output)
|
||||||
|
if err != nil {
|
||||||
|
if err != ErrAddressMissing {
|
||||||
|
glog.Error("error in output addrID in ", txid, " ", output.N, ": ", err)
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if len(addrID) > 0 {
|
||||||
|
io = append(io, addrIndex{string(addrID), int32(output.N)})
|
||||||
|
}
|
||||||
|
if onNewTxAddr != nil && len(output.ScriptPubKey.Addresses) == 1 {
|
||||||
|
onNewTxAddr(tx.Txid, output.ScriptPubKey.Addresses[0])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, input := range tx.Vin {
|
||||||
|
for i, a := range input.Addresses {
|
||||||
|
if len(a) > 0 {
|
||||||
|
addrID, err := parser.GetAddrIDFromAddress(a)
|
||||||
|
if err != nil {
|
||||||
|
glog.Error("error in input addrID in ", txid, " ", a, ": ", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
io = append(io, addrIndex{string(addrID), int32(^i)})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
newTxToInputOutput[txid] = io
|
||||||
|
for _, si := range io {
|
||||||
|
newAddrIDToTx[si.addrID] = append(newAddrIDToTx[si.addrID], outpoint{txid, si.n})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
m.updateMappings(newTxToInputOutput, newAddrIDToTx)
|
||||||
|
glog.Info("Mempool: resync finished in ", time.Since(start), ", ", len(m.txToInputOutput), " transactions in mempool")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
@ -7,14 +7,15 @@ import (
|
|||||||
"github.com/golang/glog"
|
"github.com/golang/glog"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// addrIndex and outpoint are used also in nonutxo mempool
|
||||||
type addrIndex struct {
|
type addrIndex struct {
|
||||||
addrID string
|
addrID string
|
||||||
n uint32
|
n int32
|
||||||
}
|
}
|
||||||
|
|
||||||
type outpoint struct {
|
type outpoint struct {
|
||||||
txid string
|
txid string
|
||||||
vout uint32
|
vout int32
|
||||||
}
|
}
|
||||||
|
|
||||||
type inputOutput struct {
|
type inputOutput struct {
|
||||||
@ -32,7 +33,7 @@ type UTXOMempool struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewMempool creates new mempool handler.
|
// NewMempool creates new mempool handler.
|
||||||
func NewMempool(chain BlockChain) *UTXOMempool {
|
func NewUTXOMempool(chain BlockChain) *UTXOMempool {
|
||||||
return &UTXOMempool{chain: chain}
|
return &UTXOMempool{chain: chain}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -59,7 +60,7 @@ func (m *UTXOMempool) GetTransactions(address string) ([]string, error) {
|
|||||||
|
|
||||||
// GetSpentOutput returns transaction which spends given outpoint
|
// GetSpentOutput returns transaction which spends given outpoint
|
||||||
func (m *UTXOMempool) GetSpentOutput(outputTxid string, vout uint32) string {
|
func (m *UTXOMempool) GetSpentOutput(outputTxid string, vout uint32) string {
|
||||||
o := outpoint{txid: outputTxid, vout: vout}
|
o := outpoint{txid: outputTxid, vout: int32(vout)}
|
||||||
return m.inputs[o]
|
return m.inputs[o]
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -101,7 +102,7 @@ func (m *UTXOMempool) Resync(onNewTxAddr func(txid string, addr string)) error {
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if len(addrID) > 0 {
|
if len(addrID) > 0 {
|
||||||
io.outputs = append(io.outputs, addrIndex{string(addrID), output.N})
|
io.outputs = append(io.outputs, addrIndex{string(addrID), int32(output.N)})
|
||||||
}
|
}
|
||||||
if onNewTxAddr != nil && len(output.ScriptPubKey.Addresses) == 1 {
|
if onNewTxAddr != nil && len(output.ScriptPubKey.Addresses) == 1 {
|
||||||
onNewTxAddr(tx.Txid, output.ScriptPubKey.Addresses[0])
|
onNewTxAddr(tx.Txid, output.ScriptPubKey.Addresses[0])
|
||||||
@ -112,7 +113,7 @@ func (m *UTXOMempool) Resync(onNewTxAddr func(txid string, addr string)) error {
|
|||||||
if input.Coinbase != "" {
|
if input.Coinbase != "" {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
io.inputs = append(io.inputs, outpoint{input.Txid, input.Vout})
|
io.inputs = append(io.inputs, outpoint{input.Txid, int32(input.Vout)})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
newTxToInputOutput[txid] = io
|
newTxToInputOutput[txid] = io
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user