Added a `Difference` method to `types.Transactions` which sets the receiver to the difference of a to b (NOTE: not a **and** b). Transaction pool subscribes to RemovedTransactionEvent adding back to those potential missing from the chain. When a chain re-org occurs remove any transactions that were removed from the canonical chain during the re-org as well as the receipts that were generated in the process. Closes #1746
		
			
				
	
	
		
			183 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			183 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2015 The go-ethereum Authors
 | 
						|
// This file is part of the go-ethereum library.
 | 
						|
//
 | 
						|
// The go-ethereum library is free software: you can redistribute it and/or modify
 | 
						|
// it under the terms of the GNU Lesser General Public License as published by
 | 
						|
// the Free Software Foundation, either version 3 of the License, or
 | 
						|
// (at your option) any later version.
 | 
						|
//
 | 
						|
// The go-ethereum library is distributed in the hope that it will be useful,
 | 
						|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
						|
// GNU Lesser General Public License for more details.
 | 
						|
//
 | 
						|
// You should have received a copy of the GNU Lesser General Public License
 | 
						|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
 | 
						|
 | 
						|
package core
 | 
						|
 | 
						|
import (
 | 
						|
	"github.com/ethereum/go-ethereum/common"
 | 
						|
	"github.com/ethereum/go-ethereum/core/types"
 | 
						|
	"github.com/ethereum/go-ethereum/ethdb"
 | 
						|
	"github.com/ethereum/go-ethereum/logger"
 | 
						|
	"github.com/ethereum/go-ethereum/logger/glog"
 | 
						|
	"github.com/ethereum/go-ethereum/rlp"
 | 
						|
	"github.com/syndtr/goleveldb/leveldb"
 | 
						|
)
 | 
						|
 | 
						|
var (
 | 
						|
	receiptsPre      = []byte("receipts-")
 | 
						|
	blockReceiptsPre = []byte("receipts-block-")
 | 
						|
)
 | 
						|
 | 
						|
// PutTransactions stores the transactions in the given database
 | 
						|
func PutTransactions(db ethdb.Database, block *types.Block, txs types.Transactions) {
 | 
						|
	batch := new(leveldb.Batch)
 | 
						|
	_, batchWrite := db.(*ethdb.LDBDatabase)
 | 
						|
 | 
						|
	for i, tx := range block.Transactions() {
 | 
						|
		rlpEnc, err := rlp.EncodeToBytes(tx)
 | 
						|
		if err != nil {
 | 
						|
			glog.V(logger.Debug).Infoln("Failed encoding tx", err)
 | 
						|
			return
 | 
						|
		}
 | 
						|
 | 
						|
		if batchWrite {
 | 
						|
			batch.Put(tx.Hash().Bytes(), rlpEnc)
 | 
						|
		} else {
 | 
						|
			db.Put(tx.Hash().Bytes(), rlpEnc)
 | 
						|
		}
 | 
						|
 | 
						|
		var txExtra struct {
 | 
						|
			BlockHash  common.Hash
 | 
						|
			BlockIndex uint64
 | 
						|
			Index      uint64
 | 
						|
		}
 | 
						|
		txExtra.BlockHash = block.Hash()
 | 
						|
		txExtra.BlockIndex = block.NumberU64()
 | 
						|
		txExtra.Index = uint64(i)
 | 
						|
		rlpMeta, err := rlp.EncodeToBytes(txExtra)
 | 
						|
		if err != nil {
 | 
						|
			glog.V(logger.Debug).Infoln("Failed encoding tx meta data", err)
 | 
						|
			return
 | 
						|
		}
 | 
						|
 | 
						|
		if batchWrite {
 | 
						|
			batch.Put(append(tx.Hash().Bytes(), 0x0001), rlpMeta)
 | 
						|
		} else {
 | 
						|
			db.Put(append(tx.Hash().Bytes(), 0x0001), rlpMeta)
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if db, ok := db.(*ethdb.LDBDatabase); ok {
 | 
						|
		if err := db.LDB().Write(batch, nil); err != nil {
 | 
						|
			glog.V(logger.Error).Infoln("db write err:", err)
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func DeleteTransaction(db ethdb.Database, txHash common.Hash) {
 | 
						|
	db.Delete(txHash[:])
 | 
						|
}
 | 
						|
 | 
						|
func GetTransaction(db ethdb.Database, txhash common.Hash) *types.Transaction {
 | 
						|
	data, _ := db.Get(txhash[:])
 | 
						|
	if len(data) != 0 {
 | 
						|
		var tx types.Transaction
 | 
						|
		if err := rlp.DecodeBytes(data, &tx); err != nil {
 | 
						|
			return nil
 | 
						|
		}
 | 
						|
		return &tx
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// PutReceipts stores the receipts in the current database
 | 
						|
func PutReceipts(db ethdb.Database, receipts types.Receipts) error {
 | 
						|
	batch := new(leveldb.Batch)
 | 
						|
	_, batchWrite := db.(*ethdb.LDBDatabase)
 | 
						|
 | 
						|
	for _, receipt := range receipts {
 | 
						|
		storageReceipt := (*types.ReceiptForStorage)(receipt)
 | 
						|
		bytes, err := rlp.EncodeToBytes(storageReceipt)
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		if batchWrite {
 | 
						|
			batch.Put(append(receiptsPre, receipt.TxHash[:]...), bytes)
 | 
						|
		} else {
 | 
						|
			err = db.Put(append(receiptsPre, receipt.TxHash[:]...), bytes)
 | 
						|
			if err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if db, ok := db.(*ethdb.LDBDatabase); ok {
 | 
						|
		if err := db.LDB().Write(batch, nil); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// Delete a receipts from the database
 | 
						|
func DeleteReceipt(db ethdb.Database, txHash common.Hash) {
 | 
						|
	db.Delete(append(receiptsPre, txHash[:]...))
 | 
						|
}
 | 
						|
 | 
						|
// GetReceipt returns a receipt by hash
 | 
						|
func GetReceipt(db ethdb.Database, txHash common.Hash) *types.Receipt {
 | 
						|
	data, _ := db.Get(append(receiptsPre, txHash[:]...))
 | 
						|
	if len(data) == 0 {
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
 | 
						|
	var receipt types.Receipt
 | 
						|
	err := rlp.DecodeBytes(data, &receipt)
 | 
						|
	if err != nil {
 | 
						|
		glog.V(logger.Core).Infoln("GetReceipt err:", err)
 | 
						|
	}
 | 
						|
	return &receipt
 | 
						|
}
 | 
						|
 | 
						|
// GetBlockReceipts returns the receipts generated by the transactions
 | 
						|
// included in block's given hash.
 | 
						|
func GetBlockReceipts(db ethdb.Database, hash common.Hash) types.Receipts {
 | 
						|
	data, _ := db.Get(append(blockReceiptsPre, hash[:]...))
 | 
						|
	if len(data) == 0 {
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
 | 
						|
	var receipts types.Receipts
 | 
						|
	err := rlp.DecodeBytes(data, &receipts)
 | 
						|
	if err != nil {
 | 
						|
		glog.V(logger.Core).Infoln("GetReceiptse err", err)
 | 
						|
	}
 | 
						|
	return receipts
 | 
						|
}
 | 
						|
 | 
						|
// PutBlockReceipts stores the block's transactions associated receipts
 | 
						|
// and stores them by block hash in a single slice. This is required for
 | 
						|
// forks and chain reorgs
 | 
						|
func PutBlockReceipts(db ethdb.Database, block *types.Block, receipts types.Receipts) error {
 | 
						|
	rs := make([]*types.ReceiptForStorage, len(receipts))
 | 
						|
	for i, receipt := range receipts {
 | 
						|
		rs[i] = (*types.ReceiptForStorage)(receipt)
 | 
						|
	}
 | 
						|
	bytes, err := rlp.EncodeToBytes(rs)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	hash := block.Hash()
 | 
						|
	err = db.Put(append(blockReceiptsPre, hash[:]...), bytes)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 |