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
 | |
| }
 |