go-ethereum/core/transaction_pool.go

180 lines
3.8 KiB
Go
Raw Normal View History

2014-12-04 11:28:02 +02:00
package core
2014-02-15 00:56:09 +02:00
import (
"fmt"
"math/big"
2014-12-04 11:28:02 +02:00
"github.com/ethereum/go-ethereum/core/types"
"github.com/ethereum/go-ethereum/event"
2014-10-31 13:56:05 +02:00
"github.com/ethereum/go-ethereum/logger"
"gopkg.in/fatih/set.v0"
2014-02-15 00:56:09 +02:00
)
2014-10-31 13:56:05 +02:00
var txplogger = logger.NewLogger("TXP")
const txPoolQueueSize = 50
2014-02-15 00:56:09 +02:00
type TxPoolHook chan *types.Transaction
type TxMsg struct {
Tx *types.Transaction
}
2014-02-25 12:22:27 +02:00
const (
minGasPrice = 1000000
2014-02-25 12:22:27 +02:00
)
2014-02-23 02:57:04 +02:00
type TxProcessor interface {
ProcessTransaction(tx *types.Transaction)
2014-02-23 02:57:04 +02:00
}
2014-02-15 00:56:09 +02:00
// The tx pool a thread safe transaction pool handler. In order to
// guarantee a non blocking pool we use a queue channel which can be
// independently read without needing access to the actual pool.
2014-02-15 00:56:09 +02:00
type TxPool struct {
// Queueing channel for reading and writing incoming
// transactions to
queueChan chan *types.Transaction
2014-02-15 00:56:09 +02:00
// Quiting channel
quit chan bool
// The actual pool
//pool *list.List
pool *set.Set
2014-02-15 00:56:09 +02:00
2014-02-23 02:57:04 +02:00
SecondaryProcessor TxProcessor
2014-02-25 12:22:27 +02:00
subscribers []chan TxMsg
stateQuery StateQuery
eventMux *event.TypeMux
2014-02-15 00:56:09 +02:00
}
func NewTxPool(stateQuery StateQuery, eventMux *event.TypeMux) *TxPool {
2014-02-15 00:56:09 +02:00
return &TxPool{
pool: set.New(),
queueChan: make(chan *types.Transaction, txPoolQueueSize),
quit: make(chan bool),
stateQuery: stateQuery,
eventMux: eventMux,
2014-02-15 00:56:09 +02:00
}
}
func (pool *TxPool) addTransaction(tx *types.Transaction) {
2014-05-21 13:38:56 +03:00
pool.pool.Add(tx)
2014-02-15 00:56:09 +02:00
// Broadcast the transaction to the rest of the peers
2014-12-18 14:22:59 +02:00
pool.eventMux.Post(TxPreEvent{tx})
2014-02-15 00:56:09 +02:00
}
func (pool *TxPool) ValidateTransaction(tx *types.Transaction) error {
if len(tx.To()) != 0 && len(tx.To()) != 20 {
return fmt.Errorf("Invalid recipient. len = %d", len(tx.To()))
2014-12-02 01:14:34 +02:00
}
2014-12-03 15:05:19 +02:00
v, _, _ := tx.Curve()
if v > 28 || v < 27 {
2014-12-02 01:14:34 +02:00
return fmt.Errorf("tx.v != (28 || 27)")
2014-06-12 11:07:27 +03:00
}
2014-02-15 00:56:09 +02:00
// Get the sender
senderAddr := tx.From()
2014-11-29 02:39:20 +02:00
if senderAddr == nil {
return fmt.Errorf("invalid sender")
2014-11-29 02:39:20 +02:00
}
sender := pool.stateQuery.GetAccount(senderAddr)
2014-02-15 00:56:09 +02:00
totAmount := new(big.Int).Set(tx.Value())
2014-02-15 00:56:09 +02:00
// Make sure there's enough in the sender's account. Having insufficient
// funds won't invalidate this transaction but simple ignores it.
if sender.Balance().Cmp(totAmount) < 0 {
return fmt.Errorf("Insufficient amount in sender's (%x) account", tx.From())
}
2014-02-15 00:56:09 +02:00
return nil
}
2014-12-02 12:52:56 +02:00
func (self *TxPool) Add(tx *types.Transaction) error {
hash := tx.Hash()
if self.pool.Has(tx) {
return fmt.Errorf("Known transaction (%x)", hash[0:4])
}
err := self.ValidateTransaction(tx)
if err != nil {
return err
}
self.addTransaction(tx)
txplogger.Debugf("(t) %x => %x (%v) %x\n", tx.From()[:4], tx.To()[:4], tx.Value, tx.Hash())
// Notify the subscribers
go self.eventMux.Post(TxPreEvent{tx})
return nil
}
func (self *TxPool) Size() int {
return self.pool.Size()
}
func (self *TxPool) AddTransactions(txs []*types.Transaction) {
for _, tx := range txs {
if err := self.Add(tx); err != nil {
txplogger.Infoln(err)
} else {
txplogger.Infof("tx %x\n", tx.Hash()[0:4])
}
}
}
func (pool *TxPool) GetTransactions() []*types.Transaction {
txList := make([]*types.Transaction, pool.Size())
2014-02-15 00:56:09 +02:00
i := 0
pool.pool.Each(func(v interface{}) bool {
txList[i] = v.(*types.Transaction)
2014-02-15 00:56:09 +02:00
i++
return true
})
2014-02-15 00:56:09 +02:00
return txList
}
func (pool *TxPool) RemoveInvalid(query StateQuery) {
var removedTxs types.Transactions
pool.pool.Each(func(v interface{}) bool {
tx := v.(*types.Transaction)
sender := query.GetAccount(tx.From())
err := pool.ValidateTransaction(tx)
if err != nil || sender.Nonce >= tx.Nonce() {
removedTxs = append(removedTxs, tx)
}
return true
})
pool.RemoveSet(removedTxs)
}
func (self *TxPool) RemoveSet(txs types.Transactions) {
for _, tx := range txs {
self.pool.Remove(tx)
}
}
func (pool *TxPool) Flush() []*types.Transaction {
txList := pool.GetTransactions()
pool.pool.Clear()
2014-02-15 00:56:09 +02:00
return txList
}
func (pool *TxPool) Start() {
}
func (pool *TxPool) Stop() {
pool.Flush()
2014-03-17 11:33:03 +02:00
txplogger.Infoln("Stopped")
2014-02-15 00:56:09 +02:00
}