core: improved chainDb using sequential keys
This commit is contained in:
		
							parent
							
								
									5a458da42a
								
							
						
					
					
						commit
						f9917c8c7b
					
				| @ -170,7 +170,7 @@ func dump(ctx *cli.Context) { | |||||||
| 	for _, arg := range ctx.Args() { | 	for _, arg := range ctx.Args() { | ||||||
| 		var block *types.Block | 		var block *types.Block | ||||||
| 		if hashish(arg) { | 		if hashish(arg) { | ||||||
| 			block = chain.GetBlock(common.HexToHash(arg)) | 			block = chain.GetBlockByHash(common.HexToHash(arg)) | ||||||
| 		} else { | 		} else { | ||||||
| 			num, _ := strconv.Atoi(arg) | 			num, _ := strconv.Atoi(arg) | ||||||
| 			block = chain.GetBlockByNumber(uint64(num)) | 			block = chain.GetBlockByNumber(uint64(num)) | ||||||
|  | |||||||
| @ -806,7 +806,7 @@ func MustMakeChainConfig(ctx *cli.Context) *core.ChainConfig { | |||||||
| 
 | 
 | ||||||
| // MustMakeChainConfigFromDb reads the chain configuration from the given database.
 | // MustMakeChainConfigFromDb reads the chain configuration from the given database.
 | ||||||
| func MustMakeChainConfigFromDb(ctx *cli.Context, db ethdb.Database) *core.ChainConfig { | func MustMakeChainConfigFromDb(ctx *cli.Context, db ethdb.Database) *core.ChainConfig { | ||||||
| 	genesis := core.GetBlock(db, core.GetCanonicalHash(db, 0)) | 	genesis := core.GetBlock(db, core.GetCanonicalHash(db, 0), 0) | ||||||
| 
 | 
 | ||||||
| 	if genesis != nil { | 	if genesis != nil { | ||||||
| 		// Existing genesis block, use stored config if available.
 | 		// Existing genesis block, use stored config if available.
 | ||||||
|  | |||||||
| @ -231,11 +231,11 @@ func makeChainForBench(db ethdb.Database, full bool, count uint64) { | |||||||
| 		hash = header.Hash() | 		hash = header.Hash() | ||||||
| 		WriteHeader(db, header) | 		WriteHeader(db, header) | ||||||
| 		WriteCanonicalHash(db, hash, n) | 		WriteCanonicalHash(db, hash, n) | ||||||
| 		WriteTd(db, hash, big.NewInt(int64(n+1))) | 		WriteTd(db, hash, n, big.NewInt(int64(n+1))) | ||||||
| 		if full || n == 0 { | 		if full || n == 0 { | ||||||
| 			block := types.NewBlockWithHeader(header) | 			block := types.NewBlockWithHeader(header) | ||||||
| 			WriteBody(db, hash, block.Body()) | 			WriteBody(db, hash, n, block.Body()) | ||||||
| 			WriteBlockReceipts(db, hash, nil) | 			WriteBlockReceipts(db, hash, n, nil) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -287,8 +287,8 @@ func benchReadChain(b *testing.B, full bool, count uint64) { | |||||||
| 			header := chain.GetHeaderByNumber(n) | 			header := chain.GetHeaderByNumber(n) | ||||||
| 			if full { | 			if full { | ||||||
| 				hash := header.Hash() | 				hash := header.Hash() | ||||||
| 				GetBody(db, hash) | 				GetBody(db, hash, n) | ||||||
| 				GetBlockReceipts(db, hash) | 				GetBlockReceipts(db, hash, n) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -72,7 +72,7 @@ func (v *BlockValidator) ValidateBlock(block *types.Block) error { | |||||||
| 			return &KnownBlockError{block.Number(), block.Hash()} | 			return &KnownBlockError{block.Number(), block.Hash()} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	parent := v.bc.GetBlock(block.ParentHash()) | 	parent := v.bc.GetBlock(block.ParentHash(), block.NumberU64()-1) | ||||||
| 	if parent == nil { | 	if parent == nil { | ||||||
| 		return ParentError(block.ParentHash()) | 		return ParentError(block.ParentHash()) | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -54,9 +54,7 @@ var ( | |||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| const ( | const ( | ||||||
| 	headerCacheLimit    = 512 |  | ||||||
| 	bodyCacheLimit      = 256 | 	bodyCacheLimit      = 256 | ||||||
| 	tdCacheLimit        = 1024 |  | ||||||
| 	blockCacheLimit     = 256 | 	blockCacheLimit     = 256 | ||||||
| 	maxFutureBlocks     = 256 | 	maxFutureBlocks     = 256 | ||||||
| 	maxTimeFutureBlocks = 30 | 	maxTimeFutureBlocks = 30 | ||||||
| @ -151,7 +149,7 @@ func NewBlockChain(chainDb ethdb.Database, config *ChainConfig, pow pow.PoW, mux | |||||||
| 	} | 	} | ||||||
| 	// Check the current state of the block hashes and make sure that we do not have any of the bad blocks in our chain
 | 	// Check the current state of the block hashes and make sure that we do not have any of the bad blocks in our chain
 | ||||||
| 	for hash, _ := range BadHashes { | 	for hash, _ := range BadHashes { | ||||||
| 		if header := bc.GetHeader(hash); header != nil { | 		if header := bc.GetHeaderByHash(hash); header != nil { | ||||||
| 			glog.V(logger.Error).Infof("Found bad hash, rewinding chain to block #%d [%x…]", header.Number, header.ParentHash[:4]) | 			glog.V(logger.Error).Infof("Found bad hash, rewinding chain to block #%d [%x…]", header.Number, header.ParentHash[:4]) | ||||||
| 			bc.SetHead(header.Number.Uint64() - 1) | 			bc.SetHead(header.Number.Uint64() - 1) | ||||||
| 			glog.V(logger.Error).Infoln("Chain rewind was successful, resuming normal operation") | 			glog.V(logger.Error).Infoln("Chain rewind was successful, resuming normal operation") | ||||||
| @ -175,7 +173,7 @@ func (self *BlockChain) loadLastState() error { | |||||||
| 		// Corrupt or empty database, init from scratch
 | 		// Corrupt or empty database, init from scratch
 | ||||||
| 		self.Reset() | 		self.Reset() | ||||||
| 	} else { | 	} else { | ||||||
| 		if block := self.GetBlock(head); block != nil { | 		if block := self.GetBlockByHash(head); block != nil { | ||||||
| 			// Block found, set as the current head
 | 			// Block found, set as the current head
 | ||||||
| 			self.currentBlock = block | 			self.currentBlock = block | ||||||
| 		} else { | 		} else { | ||||||
| @ -186,7 +184,7 @@ func (self *BlockChain) loadLastState() error { | |||||||
| 	// Restore the last known head header
 | 	// Restore the last known head header
 | ||||||
| 	currentHeader := self.currentBlock.Header() | 	currentHeader := self.currentBlock.Header() | ||||||
| 	if head := GetHeadHeaderHash(self.chainDb); head != (common.Hash{}) { | 	if head := GetHeadHeaderHash(self.chainDb); head != (common.Hash{}) { | ||||||
| 		if header := self.GetHeader(head); header != nil { | 		if header := self.GetHeaderByHash(head); header != nil { | ||||||
| 			currentHeader = header | 			currentHeader = header | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -194,16 +192,16 @@ func (self *BlockChain) loadLastState() error { | |||||||
| 	// Restore the last known head fast block
 | 	// Restore the last known head fast block
 | ||||||
| 	self.currentFastBlock = self.currentBlock | 	self.currentFastBlock = self.currentBlock | ||||||
| 	if head := GetHeadFastBlockHash(self.chainDb); head != (common.Hash{}) { | 	if head := GetHeadFastBlockHash(self.chainDb); head != (common.Hash{}) { | ||||||
| 		if block := self.GetBlock(head); block != nil { | 		if block := self.GetBlockByHash(head); block != nil { | ||||||
| 			self.currentFastBlock = block | 			self.currentFastBlock = block | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	// Issue a status log and return
 | 	// Issue a status log and return
 | ||||||
| 	headerTd := self.GetTd(self.hc.CurrentHeader().Hash()) | 	headerTd := self.GetTd(currentHeader.Hash(), currentHeader.Number.Uint64()) | ||||||
| 	blockTd := self.GetTd(self.currentBlock.Hash()) | 	blockTd := self.GetTd(self.currentBlock.Hash(), self.currentBlock.NumberU64()) | ||||||
| 	fastTd := self.GetTd(self.currentFastBlock.Hash()) | 	fastTd := self.GetTd(self.currentFastBlock.Hash(), self.currentFastBlock.NumberU64()) | ||||||
| 
 | 
 | ||||||
| 	glog.V(logger.Info).Infof("Last header: #%d [%x…] TD=%v", self.hc.CurrentHeader().Number, self.hc.CurrentHeader().Hash().Bytes()[:4], headerTd) | 	glog.V(logger.Info).Infof("Last header: #%d [%x…] TD=%v", currentHeader.Number, currentHeader.Hash().Bytes()[:4], headerTd) | ||||||
| 	glog.V(logger.Info).Infof("Last block: #%d [%x…] TD=%v", self.currentBlock.Number(), self.currentBlock.Hash().Bytes()[:4], blockTd) | 	glog.V(logger.Info).Infof("Last block: #%d [%x…] TD=%v", self.currentBlock.Number(), self.currentBlock.Hash().Bytes()[:4], blockTd) | ||||||
| 	glog.V(logger.Info).Infof("Fast block: #%d [%x…] TD=%v", self.currentFastBlock.Number(), self.currentFastBlock.Hash().Bytes()[:4], fastTd) | 	glog.V(logger.Info).Infof("Fast block: #%d [%x…] TD=%v", self.currentFastBlock.Number(), self.currentFastBlock.Hash().Bytes()[:4], fastTd) | ||||||
| 
 | 
 | ||||||
| @ -218,8 +216,8 @@ func (bc *BlockChain) SetHead(head uint64) { | |||||||
| 	bc.mu.Lock() | 	bc.mu.Lock() | ||||||
| 	defer bc.mu.Unlock() | 	defer bc.mu.Unlock() | ||||||
| 
 | 
 | ||||||
| 	delFn := func(hash common.Hash) { | 	delFn := func(hash common.Hash, num uint64) { | ||||||
| 		DeleteBody(bc.chainDb, hash) | 		DeleteBody(bc.chainDb, hash, num) | ||||||
| 	} | 	} | ||||||
| 	bc.hc.SetHead(head, delFn) | 	bc.hc.SetHead(head, delFn) | ||||||
| 
 | 
 | ||||||
| @ -230,11 +228,12 @@ func (bc *BlockChain) SetHead(head uint64) { | |||||||
| 	bc.futureBlocks.Purge() | 	bc.futureBlocks.Purge() | ||||||
| 
 | 
 | ||||||
| 	// Update all computed fields to the new head
 | 	// Update all computed fields to the new head
 | ||||||
| 	if bc.currentBlock != nil && bc.hc.CurrentHeader().Number.Uint64() < bc.currentBlock.NumberU64() { | 	currentHeader := bc.hc.CurrentHeader() | ||||||
| 		bc.currentBlock = bc.GetBlock(bc.hc.CurrentHeader().Hash()) | 	if bc.currentBlock != nil && currentHeader.Number.Uint64() < bc.currentBlock.NumberU64() { | ||||||
|  | 		bc.currentBlock = bc.GetBlock(currentHeader.Hash(), currentHeader.Number.Uint64()) | ||||||
| 	} | 	} | ||||||
| 	if bc.currentFastBlock != nil && bc.hc.CurrentHeader().Number.Uint64() < bc.currentFastBlock.NumberU64() { | 	if bc.currentFastBlock != nil && currentHeader.Number.Uint64() < bc.currentFastBlock.NumberU64() { | ||||||
| 		bc.currentFastBlock = bc.GetBlock(bc.hc.CurrentHeader().Hash()) | 		bc.currentFastBlock = bc.GetBlock(currentHeader.Hash(), currentHeader.Number.Uint64()) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if bc.currentBlock == nil { | 	if bc.currentBlock == nil { | ||||||
| @ -257,7 +256,7 @@ func (bc *BlockChain) SetHead(head uint64) { | |||||||
| // irrelevant what the chain contents were prior.
 | // irrelevant what the chain contents were prior.
 | ||||||
| func (self *BlockChain) FastSyncCommitHead(hash common.Hash) error { | func (self *BlockChain) FastSyncCommitHead(hash common.Hash) error { | ||||||
| 	// Make sure that both the block as well at its state trie exists
 | 	// Make sure that both the block as well at its state trie exists
 | ||||||
| 	block := self.GetBlock(hash) | 	block := self.GetBlockByHash(hash) | ||||||
| 	if block == nil { | 	if block == nil { | ||||||
| 		return fmt.Errorf("non existent block [%x…]", hash[:4]) | 		return fmt.Errorf("non existent block [%x…]", hash[:4]) | ||||||
| 	} | 	} | ||||||
| @ -313,7 +312,7 @@ func (self *BlockChain) Status() (td *big.Int, currentBlock common.Hash, genesis | |||||||
| 	self.mu.RLock() | 	self.mu.RLock() | ||||||
| 	defer self.mu.RUnlock() | 	defer self.mu.RUnlock() | ||||||
| 
 | 
 | ||||||
| 	return self.GetTd(self.currentBlock.Hash()), self.currentBlock.Hash(), self.genesisBlock.Hash() | 	return self.GetTd(self.currentBlock.Hash(), self.currentBlock.NumberU64()), self.currentBlock.Hash(), self.genesisBlock.Hash() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // SetProcessor sets the processor required for making state modifications.
 | // SetProcessor sets the processor required for making state modifications.
 | ||||||
| @ -367,7 +366,7 @@ func (bc *BlockChain) ResetWithGenesisBlock(genesis *types.Block) { | |||||||
| 	defer bc.mu.Unlock() | 	defer bc.mu.Unlock() | ||||||
| 
 | 
 | ||||||
| 	// Prepare the genesis block and reinitialise the chain
 | 	// Prepare the genesis block and reinitialise the chain
 | ||||||
| 	if err := bc.hc.WriteTd(genesis.Hash(), genesis.Difficulty()); err != nil { | 	if err := bc.hc.WriteTd(genesis.Hash(), genesis.NumberU64(), genesis.Difficulty()); err != nil { | ||||||
| 		glog.Fatalf("failed to write genesis block TD: %v", err) | 		glog.Fatalf("failed to write genesis block TD: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if err := WriteBlock(bc.chainDb, genesis); err != nil { | 	if err := WriteBlock(bc.chainDb, genesis); err != nil { | ||||||
| @ -457,7 +456,7 @@ func (self *BlockChain) GetBody(hash common.Hash) *types.Body { | |||||||
| 		body := cached.(*types.Body) | 		body := cached.(*types.Body) | ||||||
| 		return body | 		return body | ||||||
| 	} | 	} | ||||||
| 	body := GetBody(self.chainDb, hash) | 	body := GetBody(self.chainDb, hash, self.hc.GetBlockNumber(hash)) | ||||||
| 	if body == nil { | 	if body == nil { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| @ -473,7 +472,7 @@ func (self *BlockChain) GetBodyRLP(hash common.Hash) rlp.RawValue { | |||||||
| 	if cached, ok := self.bodyRLPCache.Get(hash); ok { | 	if cached, ok := self.bodyRLPCache.Get(hash); ok { | ||||||
| 		return cached.(rlp.RawValue) | 		return cached.(rlp.RawValue) | ||||||
| 	} | 	} | ||||||
| 	body := GetBodyRLP(self.chainDb, hash) | 	body := GetBodyRLP(self.chainDb, hash, self.hc.GetBlockNumber(hash)) | ||||||
| 	if len(body) == 0 { | 	if len(body) == 0 { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| @ -485,14 +484,14 @@ func (self *BlockChain) GetBodyRLP(hash common.Hash) rlp.RawValue { | |||||||
| // HasBlock checks if a block is fully present in the database or not, caching
 | // HasBlock checks if a block is fully present in the database or not, caching
 | ||||||
| // it if present.
 | // it if present.
 | ||||||
| func (bc *BlockChain) HasBlock(hash common.Hash) bool { | func (bc *BlockChain) HasBlock(hash common.Hash) bool { | ||||||
| 	return bc.GetBlock(hash) != nil | 	return bc.GetBlockByHash(hash) != nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // HasBlockAndState checks if a block and associated state trie is fully present
 | // HasBlockAndState checks if a block and associated state trie is fully present
 | ||||||
| // in the database or not, caching it if present.
 | // in the database or not, caching it if present.
 | ||||||
| func (bc *BlockChain) HasBlockAndState(hash common.Hash) bool { | func (bc *BlockChain) HasBlockAndState(hash common.Hash) bool { | ||||||
| 	// Check first that the block itself is known
 | 	// Check first that the block itself is known
 | ||||||
| 	block := bc.GetBlock(hash) | 	block := bc.GetBlockByHash(hash) | ||||||
| 	if block == nil { | 	if block == nil { | ||||||
| 		return false | 		return false | ||||||
| 	} | 	} | ||||||
| @ -501,13 +500,14 @@ func (bc *BlockChain) HasBlockAndState(hash common.Hash) bool { | |||||||
| 	return err == nil | 	return err == nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetBlock retrieves a block from the database by hash, caching it if found.
 | // GetBlock retrieves a block from the database by hash and number,
 | ||||||
| func (self *BlockChain) GetBlock(hash common.Hash) *types.Block { | // caching it if found.
 | ||||||
|  | func (self *BlockChain) GetBlock(hash common.Hash, number uint64) *types.Block { | ||||||
| 	// Short circuit if the block's already in the cache, retrieve otherwise
 | 	// Short circuit if the block's already in the cache, retrieve otherwise
 | ||||||
| 	if block, ok := self.blockCache.Get(hash); ok { | 	if block, ok := self.blockCache.Get(hash); ok { | ||||||
| 		return block.(*types.Block) | 		return block.(*types.Block) | ||||||
| 	} | 	} | ||||||
| 	block := GetBlock(self.chainDb, hash) | 	block := GetBlock(self.chainDb, hash, number) | ||||||
| 	if block == nil { | 	if block == nil { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| @ -516,6 +516,11 @@ func (self *BlockChain) GetBlock(hash common.Hash) *types.Block { | |||||||
| 	return block | 	return block | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // GetBlockByHash retrieves a block from the database by hash, caching it if found.
 | ||||||
|  | func (self *BlockChain) GetBlockByHash(hash common.Hash) *types.Block { | ||||||
|  | 	return self.GetBlock(hash, self.hc.GetBlockNumber(hash)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // GetBlockByNumber retrieves a block from the database by number, caching it
 | // GetBlockByNumber retrieves a block from the database by number, caching it
 | ||||||
| // (associated with its hash) if found.
 | // (associated with its hash) if found.
 | ||||||
| func (self *BlockChain) GetBlockByNumber(number uint64) *types.Block { | func (self *BlockChain) GetBlockByNumber(number uint64) *types.Block { | ||||||
| @ -523,19 +528,21 @@ func (self *BlockChain) GetBlockByNumber(number uint64) *types.Block { | |||||||
| 	if hash == (common.Hash{}) { | 	if hash == (common.Hash{}) { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	return self.GetBlock(hash) | 	return self.GetBlock(hash, number) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // [deprecated by eth/62]
 | // [deprecated by eth/62]
 | ||||||
| // GetBlocksFromHash returns the block corresponding to hash and up to n-1 ancestors.
 | // GetBlocksFromHash returns the block corresponding to hash and up to n-1 ancestors.
 | ||||||
| func (self *BlockChain) GetBlocksFromHash(hash common.Hash, n int) (blocks []*types.Block) { | func (self *BlockChain) GetBlocksFromHash(hash common.Hash, n int) (blocks []*types.Block) { | ||||||
|  | 	number := self.hc.GetBlockNumber(hash) | ||||||
| 	for i := 0; i < n; i++ { | 	for i := 0; i < n; i++ { | ||||||
| 		block := self.GetBlock(hash) | 		block := self.GetBlock(hash, number) | ||||||
| 		if block == nil { | 		if block == nil { | ||||||
| 			break | 			break | ||||||
| 		} | 		} | ||||||
| 		blocks = append(blocks, block) | 		blocks = append(blocks, block) | ||||||
| 		hash = block.ParentHash() | 		hash = block.ParentHash() | ||||||
|  | 		number-- | ||||||
| 	} | 	} | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
| @ -546,7 +553,7 @@ func (self *BlockChain) GetUnclesInChain(block *types.Block, length int) []*type | |||||||
| 	uncles := []*types.Header{} | 	uncles := []*types.Header{} | ||||||
| 	for i := 0; block != nil && i < length; i++ { | 	for i := 0; block != nil && i < length; i++ { | ||||||
| 		uncles = append(uncles, block.Uncles()...) | 		uncles = append(uncles, block.Uncles()...) | ||||||
| 		block = self.GetBlock(block.ParentHash()) | 		block = self.GetBlock(block.ParentHash(), block.NumberU64()-1) | ||||||
| 	} | 	} | ||||||
| 	return uncles | 	return uncles | ||||||
| } | } | ||||||
| @ -596,15 +603,16 @@ func (self *BlockChain) Rollback(chain []common.Hash) { | |||||||
| 	for i := len(chain) - 1; i >= 0; i-- { | 	for i := len(chain) - 1; i >= 0; i-- { | ||||||
| 		hash := chain[i] | 		hash := chain[i] | ||||||
| 
 | 
 | ||||||
| 		if self.hc.CurrentHeader().Hash() == hash { | 		currentHeader := self.hc.CurrentHeader() | ||||||
| 			self.hc.SetCurrentHeader(self.GetHeader(self.hc.CurrentHeader().ParentHash)) | 		if currentHeader.Hash() == hash { | ||||||
|  | 			self.hc.SetCurrentHeader(self.GetHeader(currentHeader.ParentHash, currentHeader.Number.Uint64()-1)) | ||||||
| 		} | 		} | ||||||
| 		if self.currentFastBlock.Hash() == hash { | 		if self.currentFastBlock.Hash() == hash { | ||||||
| 			self.currentFastBlock = self.GetBlock(self.currentFastBlock.ParentHash()) | 			self.currentFastBlock = self.GetBlock(self.currentFastBlock.ParentHash(), self.currentFastBlock.NumberU64()-1) | ||||||
| 			WriteHeadFastBlockHash(self.chainDb, self.currentFastBlock.Hash()) | 			WriteHeadFastBlockHash(self.chainDb, self.currentFastBlock.Hash()) | ||||||
| 		} | 		} | ||||||
| 		if self.currentBlock.Hash() == hash { | 		if self.currentBlock.Hash() == hash { | ||||||
| 			self.currentBlock = self.GetBlock(self.currentBlock.ParentHash()) | 			self.currentBlock = self.GetBlock(self.currentBlock.ParentHash(), self.currentBlock.NumberU64()-1) | ||||||
| 			WriteHeadBlockHash(self.chainDb, self.currentBlock.Hash()) | 			WriteHeadBlockHash(self.chainDb, self.currentBlock.Hash()) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -678,13 +686,13 @@ func (self *BlockChain) InsertReceiptChain(blockChain types.Blocks, receiptChain | |||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			// Write all the data out into the database
 | 			// Write all the data out into the database
 | ||||||
| 			if err := WriteBody(self.chainDb, block.Hash(), block.Body()); err != nil { | 			if err := WriteBody(self.chainDb, block.Hash(), block.NumberU64(), block.Body()); err != nil { | ||||||
| 				errs[index] = fmt.Errorf("failed to write block body: %v", err) | 				errs[index] = fmt.Errorf("failed to write block body: %v", err) | ||||||
| 				atomic.AddInt32(&failed, 1) | 				atomic.AddInt32(&failed, 1) | ||||||
| 				glog.Fatal(errs[index]) | 				glog.Fatal(errs[index]) | ||||||
| 				return | 				return | ||||||
| 			} | 			} | ||||||
| 			if err := WriteBlockReceipts(self.chainDb, block.Hash(), receipts); err != nil { | 			if err := WriteBlockReceipts(self.chainDb, block.Hash(), block.NumberU64(), receipts); err != nil { | ||||||
| 				errs[index] = fmt.Errorf("failed to write block receipts: %v", err) | 				errs[index] = fmt.Errorf("failed to write block receipts: %v", err) | ||||||
| 				atomic.AddInt32(&failed, 1) | 				atomic.AddInt32(&failed, 1) | ||||||
| 				glog.Fatal(errs[index]) | 				glog.Fatal(errs[index]) | ||||||
| @ -737,7 +745,7 @@ func (self *BlockChain) InsertReceiptChain(blockChain types.Blocks, receiptChain | |||||||
| 	// Update the head fast sync block if better
 | 	// Update the head fast sync block if better
 | ||||||
| 	self.mu.Lock() | 	self.mu.Lock() | ||||||
| 	head := blockChain[len(errs)-1] | 	head := blockChain[len(errs)-1] | ||||||
| 	if self.GetTd(self.currentFastBlock.Hash()).Cmp(self.GetTd(head.Hash())) < 0 { | 	if self.GetTd(self.currentFastBlock.Hash(), self.currentFastBlock.NumberU64()).Cmp(self.GetTd(head.Hash(), head.NumberU64())) < 0 { | ||||||
| 		if err := WriteHeadFastBlockHash(self.chainDb, head.Hash()); err != nil { | 		if err := WriteHeadFastBlockHash(self.chainDb, head.Hash()); err != nil { | ||||||
| 			glog.Fatalf("failed to update head fast block hash: %v", err) | 			glog.Fatalf("failed to update head fast block hash: %v", err) | ||||||
| 		} | 		} | ||||||
| @ -759,12 +767,12 @@ func (self *BlockChain) WriteBlock(block *types.Block) (status WriteStatus, err | |||||||
| 	defer self.wg.Done() | 	defer self.wg.Done() | ||||||
| 
 | 
 | ||||||
| 	// Calculate the total difficulty of the block
 | 	// Calculate the total difficulty of the block
 | ||||||
| 	ptd := self.GetTd(block.ParentHash()) | 	ptd := self.GetTd(block.ParentHash(), block.NumberU64()-1) | ||||||
| 	if ptd == nil { | 	if ptd == nil { | ||||||
| 		return NonStatTy, ParentError(block.ParentHash()) | 		return NonStatTy, ParentError(block.ParentHash()) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	localTd := self.GetTd(self.currentBlock.Hash()) | 	localTd := self.GetTd(self.currentBlock.Hash(), self.currentBlock.NumberU64()) | ||||||
| 	externTd := new(big.Int).Add(block.Difficulty(), ptd) | 	externTd := new(big.Int).Add(block.Difficulty(), ptd) | ||||||
| 
 | 
 | ||||||
| 	// Make sure no inconsistent state is leaked during insertion
 | 	// Make sure no inconsistent state is leaked during insertion
 | ||||||
| @ -788,7 +796,7 @@ func (self *BlockChain) WriteBlock(block *types.Block) (status WriteStatus, err | |||||||
| 		status = SideStatTy | 		status = SideStatTy | ||||||
| 	} | 	} | ||||||
| 	// Irrelevant of the canonical status, write the block itself to the database
 | 	// Irrelevant of the canonical status, write the block itself to the database
 | ||||||
| 	if err := self.hc.WriteTd(block.Hash(), externTd); err != nil { | 	if err := self.hc.WriteTd(block.Hash(), block.NumberU64(), externTd); err != nil { | ||||||
| 		glog.Fatalf("failed to write block total difficulty: %v", err) | 		glog.Fatalf("failed to write block total difficulty: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if err := WriteBlock(self.chainDb, block); err != nil { | 	if err := WriteBlock(self.chainDb, block); err != nil { | ||||||
| @ -887,7 +895,7 @@ func (self *BlockChain) InsertChain(chain types.Blocks) (int, error) { | |||||||
| 		// Create a new statedb using the parent block and report an
 | 		// Create a new statedb using the parent block and report an
 | ||||||
| 		// error if it fails.
 | 		// error if it fails.
 | ||||||
| 		if statedb == nil { | 		if statedb == nil { | ||||||
| 			statedb, err = state.New(self.GetBlock(block.ParentHash()).Root(), self.chainDb) | 			statedb, err = state.New(self.GetBlock(block.ParentHash(), block.NumberU64()-1).Root(), self.chainDb) | ||||||
| 		} else { | 		} else { | ||||||
| 			err = statedb.Reset(chain[i-1].Root()) | 			err = statedb.Reset(chain[i-1].Root()) | ||||||
| 		} | 		} | ||||||
| @ -902,7 +910,7 @@ func (self *BlockChain) InsertChain(chain types.Blocks) (int, error) { | |||||||
| 			return i, err | 			return i, err | ||||||
| 		} | 		} | ||||||
| 		// Validate the state using the default validator
 | 		// Validate the state using the default validator
 | ||||||
| 		err = self.Validator().ValidateState(block, self.GetBlock(block.ParentHash()), statedb, receipts, usedGas) | 		err = self.Validator().ValidateState(block, self.GetBlock(block.ParentHash(), block.NumberU64()-1), statedb, receipts, usedGas) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			reportBlock(block, err) | 			reportBlock(block, err) | ||||||
| 			return i, err | 			return i, err | ||||||
| @ -916,7 +924,7 @@ func (self *BlockChain) InsertChain(chain types.Blocks) (int, error) { | |||||||
| 		// coalesce logs for later processing
 | 		// coalesce logs for later processing
 | ||||||
| 		coalescedLogs = append(coalescedLogs, logs...) | 		coalescedLogs = append(coalescedLogs, logs...) | ||||||
| 
 | 
 | ||||||
| 		if err := WriteBlockReceipts(self.chainDb, block.Hash(), receipts); err != nil { | 		if err := WriteBlockReceipts(self.chainDb, block.Hash(), block.NumberU64(), receipts); err != nil { | ||||||
| 			return i, err | 			return i, err | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| @ -986,7 +994,7 @@ func (self *BlockChain) reorg(oldBlock, newBlock *types.Block) error { | |||||||
| 		// These logs are later announced as deleted.
 | 		// These logs are later announced as deleted.
 | ||||||
| 		collectLogs = func(h common.Hash) { | 		collectLogs = func(h common.Hash) { | ||||||
| 			// Coalesce logs
 | 			// Coalesce logs
 | ||||||
| 			receipts := GetBlockReceipts(self.chainDb, h) | 			receipts := GetBlockReceipts(self.chainDb, h, self.hc.GetBlockNumber(h)) | ||||||
| 			for _, receipt := range receipts { | 			for _, receipt := range receipts { | ||||||
| 				deletedLogs = append(deletedLogs, receipt.Logs...) | 				deletedLogs = append(deletedLogs, receipt.Logs...) | ||||||
| 
 | 
 | ||||||
| @ -998,7 +1006,7 @@ func (self *BlockChain) reorg(oldBlock, newBlock *types.Block) error { | |||||||
| 	// first reduce whoever is higher bound
 | 	// first reduce whoever is higher bound
 | ||||||
| 	if oldBlock.NumberU64() > newBlock.NumberU64() { | 	if oldBlock.NumberU64() > newBlock.NumberU64() { | ||||||
| 		// reduce old chain
 | 		// reduce old chain
 | ||||||
| 		for ; oldBlock != nil && oldBlock.NumberU64() != newBlock.NumberU64(); oldBlock = self.GetBlock(oldBlock.ParentHash()) { | 		for ; oldBlock != nil && oldBlock.NumberU64() != newBlock.NumberU64(); oldBlock = self.GetBlock(oldBlock.ParentHash(), oldBlock.NumberU64()-1) { | ||||||
| 			oldChain = append(oldChain, oldBlock) | 			oldChain = append(oldChain, oldBlock) | ||||||
| 			deletedTxs = append(deletedTxs, oldBlock.Transactions()...) | 			deletedTxs = append(deletedTxs, oldBlock.Transactions()...) | ||||||
| 
 | 
 | ||||||
| @ -1006,7 +1014,7 @@ func (self *BlockChain) reorg(oldBlock, newBlock *types.Block) error { | |||||||
| 		} | 		} | ||||||
| 	} else { | 	} else { | ||||||
| 		// reduce new chain and append new chain blocks for inserting later on
 | 		// reduce new chain and append new chain blocks for inserting later on
 | ||||||
| 		for ; newBlock != nil && newBlock.NumberU64() != oldBlock.NumberU64(); newBlock = self.GetBlock(newBlock.ParentHash()) { | 		for ; newBlock != nil && newBlock.NumberU64() != oldBlock.NumberU64(); newBlock = self.GetBlock(newBlock.ParentHash(), newBlock.NumberU64()-1) { | ||||||
| 			newChain = append(newChain, newBlock) | 			newChain = append(newChain, newBlock) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -1029,7 +1037,7 @@ func (self *BlockChain) reorg(oldBlock, newBlock *types.Block) error { | |||||||
| 		deletedTxs = append(deletedTxs, oldBlock.Transactions()...) | 		deletedTxs = append(deletedTxs, oldBlock.Transactions()...) | ||||||
| 		collectLogs(oldBlock.Hash()) | 		collectLogs(oldBlock.Hash()) | ||||||
| 
 | 
 | ||||||
| 		oldBlock, newBlock = self.GetBlock(oldBlock.ParentHash()), self.GetBlock(newBlock.ParentHash()) | 		oldBlock, newBlock = self.GetBlock(oldBlock.ParentHash(), oldBlock.NumberU64()-1), self.GetBlock(newBlock.ParentHash(), newBlock.NumberU64()-1) | ||||||
| 		if oldBlock == nil { | 		if oldBlock == nil { | ||||||
| 			return fmt.Errorf("Invalid old chain") | 			return fmt.Errorf("Invalid old chain") | ||||||
| 		} | 		} | ||||||
| @ -1052,7 +1060,7 @@ func (self *BlockChain) reorg(oldBlock, newBlock *types.Block) error { | |||||||
| 		if err := WriteTransactions(self.chainDb, block); err != nil { | 		if err := WriteTransactions(self.chainDb, block); err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		receipts := GetBlockReceipts(self.chainDb, block.Hash()) | 		receipts := GetBlockReceipts(self.chainDb, block.Hash(), block.NumberU64()) | ||||||
| 		// write receipts
 | 		// write receipts
 | ||||||
| 		if err := WriteReceipts(self.chainDb, receipts); err != nil { | 		if err := WriteReceipts(self.chainDb, receipts); err != nil { | ||||||
| 			return err | 			return err | ||||||
| @ -1187,15 +1195,27 @@ func (self *BlockChain) CurrentHeader() *types.Header { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetTd retrieves a block's total difficulty in the canonical chain from the
 | // GetTd retrieves a block's total difficulty in the canonical chain from the
 | ||||||
| // database by hash, caching it if found.
 | // database by hash and number, caching it if found.
 | ||||||
| func (self *BlockChain) GetTd(hash common.Hash) *big.Int { | func (self *BlockChain) GetTd(hash common.Hash, number uint64) *big.Int { | ||||||
| 	return self.hc.GetTd(hash) | 	return self.hc.GetTd(hash, number) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetHeader retrieves a block header from the database by hash, caching it if
 | // GetTdByHash retrieves a block's total difficulty in the canonical chain from the
 | ||||||
|  | // database by hash, caching it if found.
 | ||||||
|  | func (self *BlockChain) GetTdByHash(hash common.Hash) *big.Int { | ||||||
|  | 	return self.hc.GetTdByHash(hash) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // GetHeader retrieves a block header from the database by hash and number,
 | ||||||
|  | // caching it if found.
 | ||||||
|  | func (self *BlockChain) GetHeader(hash common.Hash, number uint64) *types.Header { | ||||||
|  | 	return self.hc.GetHeader(hash, number) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // GetHeaderByHash retrieves a block header from the database by hash, caching it if
 | ||||||
| // found.
 | // found.
 | ||||||
| func (self *BlockChain) GetHeader(hash common.Hash) *types.Header { | func (self *BlockChain) GetHeaderByHash(hash common.Hash) *types.Header { | ||||||
| 	return self.hc.GetHeader(hash) | 	return self.hc.GetHeaderByHash(hash) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // HasHeader checks if a block header is present in the database or not, caching
 | // HasHeader checks if a block header is present in the database or not, caching
 | ||||||
|  | |||||||
| @ -102,17 +102,17 @@ func testFork(t *testing.T, blockchain *BlockChain, i, n int, full bool, compara | |||||||
| 	var tdPre, tdPost *big.Int | 	var tdPre, tdPost *big.Int | ||||||
| 
 | 
 | ||||||
| 	if full { | 	if full { | ||||||
| 		tdPre = blockchain.GetTd(blockchain.CurrentBlock().Hash()) | 		tdPre = blockchain.GetTdByHash(blockchain.CurrentBlock().Hash()) | ||||||
| 		if err := testBlockChainImport(blockChainB, blockchain); err != nil { | 		if err := testBlockChainImport(blockChainB, blockchain); err != nil { | ||||||
| 			t.Fatalf("failed to import forked block chain: %v", err) | 			t.Fatalf("failed to import forked block chain: %v", err) | ||||||
| 		} | 		} | ||||||
| 		tdPost = blockchain.GetTd(blockChainB[len(blockChainB)-1].Hash()) | 		tdPost = blockchain.GetTdByHash(blockChainB[len(blockChainB)-1].Hash()) | ||||||
| 	} else { | 	} else { | ||||||
| 		tdPre = blockchain.GetTd(blockchain.CurrentHeader().Hash()) | 		tdPre = blockchain.GetTdByHash(blockchain.CurrentHeader().Hash()) | ||||||
| 		if err := testHeaderChainImport(headerChainB, blockchain); err != nil { | 		if err := testHeaderChainImport(headerChainB, blockchain); err != nil { | ||||||
| 			t.Fatalf("failed to import forked header chain: %v", err) | 			t.Fatalf("failed to import forked header chain: %v", err) | ||||||
| 		} | 		} | ||||||
| 		tdPost = blockchain.GetTd(headerChainB[len(headerChainB)-1].Hash()) | 		tdPost = blockchain.GetTdByHash(headerChainB[len(headerChainB)-1].Hash()) | ||||||
| 	} | 	} | ||||||
| 	// Compare the total difficulties of the chains
 | 	// Compare the total difficulties of the chains
 | ||||||
| 	comparator(tdPre, tdPost) | 	comparator(tdPre, tdPost) | ||||||
| @ -137,7 +137,7 @@ func testBlockChainImport(chain types.Blocks, blockchain *BlockChain) error { | |||||||
| 			} | 			} | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		statedb, err := state.New(blockchain.GetBlock(block.ParentHash()).Root(), blockchain.chainDb) | 		statedb, err := state.New(blockchain.GetBlockByHash(block.ParentHash()).Root(), blockchain.chainDb) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| @ -146,13 +146,13 @@ func testBlockChainImport(chain types.Blocks, blockchain *BlockChain) error { | |||||||
| 			reportBlock(block, err) | 			reportBlock(block, err) | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		err = blockchain.Validator().ValidateState(block, blockchain.GetBlock(block.ParentHash()), statedb, receipts, usedGas) | 		err = blockchain.Validator().ValidateState(block, blockchain.GetBlockByHash(block.ParentHash()), statedb, receipts, usedGas) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			reportBlock(block, err) | 			reportBlock(block, err) | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		blockchain.mu.Lock() | 		blockchain.mu.Lock() | ||||||
| 		WriteTd(blockchain.chainDb, block.Hash(), new(big.Int).Add(block.Difficulty(), blockchain.GetTd(block.ParentHash()))) | 		WriteTd(blockchain.chainDb, block.Hash(), block.NumberU64(), new(big.Int).Add(block.Difficulty(), blockchain.GetTdByHash(block.ParentHash()))) | ||||||
| 		WriteBlock(blockchain.chainDb, block) | 		WriteBlock(blockchain.chainDb, block) | ||||||
| 		statedb.Commit() | 		statedb.Commit() | ||||||
| 		blockchain.mu.Unlock() | 		blockchain.mu.Unlock() | ||||||
| @ -165,12 +165,12 @@ func testBlockChainImport(chain types.Blocks, blockchain *BlockChain) error { | |||||||
| func testHeaderChainImport(chain []*types.Header, blockchain *BlockChain) error { | func testHeaderChainImport(chain []*types.Header, blockchain *BlockChain) error { | ||||||
| 	for _, header := range chain { | 	for _, header := range chain { | ||||||
| 		// Try and validate the header
 | 		// Try and validate the header
 | ||||||
| 		if err := blockchain.Validator().ValidateHeader(header, blockchain.GetHeader(header.ParentHash), false); err != nil { | 		if err := blockchain.Validator().ValidateHeader(header, blockchain.GetHeaderByHash(header.ParentHash), false); err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		// Manually insert the header into the database, but don't reorganise (allows subsequent testing)
 | 		// Manually insert the header into the database, but don't reorganise (allows subsequent testing)
 | ||||||
| 		blockchain.mu.Lock() | 		blockchain.mu.Lock() | ||||||
| 		WriteTd(blockchain.chainDb, header.Hash(), new(big.Int).Add(header.Difficulty, blockchain.GetTd(header.ParentHash))) | 		WriteTd(blockchain.chainDb, header.Hash(), header.Number.Uint64(), new(big.Int).Add(header.Difficulty, blockchain.GetTdByHash(header.ParentHash))) | ||||||
| 		WriteHeader(blockchain.chainDb, header) | 		WriteHeader(blockchain.chainDb, header) | ||||||
| 		blockchain.mu.Unlock() | 		blockchain.mu.Unlock() | ||||||
| 	} | 	} | ||||||
| @ -543,11 +543,11 @@ func testReorg(t *testing.T, first, second []int, td int64, full bool) { | |||||||
| 	// Make sure the chain total difficulty is the correct one
 | 	// Make sure the chain total difficulty is the correct one
 | ||||||
| 	want := new(big.Int).Add(genesis.Difficulty(), big.NewInt(td)) | 	want := new(big.Int).Add(genesis.Difficulty(), big.NewInt(td)) | ||||||
| 	if full { | 	if full { | ||||||
| 		if have := bc.GetTd(bc.CurrentBlock().Hash()); have.Cmp(want) != 0 { | 		if have := bc.GetTdByHash(bc.CurrentBlock().Hash()); have.Cmp(want) != 0 { | ||||||
| 			t.Errorf("total difficulty mismatch: have %v, want %v", have, want) | 			t.Errorf("total difficulty mismatch: have %v, want %v", have, want) | ||||||
| 		} | 		} | ||||||
| 	} else { | 	} else { | ||||||
| 		if have := bc.GetTd(bc.CurrentHeader().Hash()); have.Cmp(want) != 0 { | 		if have := bc.GetTdByHash(bc.CurrentHeader().Hash()); have.Cmp(want) != 0 { | ||||||
| 			t.Errorf("total difficulty mismatch: have %v, want %v", have, want) | 			t.Errorf("total difficulty mismatch: have %v, want %v", have, want) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -758,20 +758,20 @@ func TestFastVsFullChains(t *testing.T) { | |||||||
| 	for i := 0; i < len(blocks); i++ { | 	for i := 0; i < len(blocks); i++ { | ||||||
| 		num, hash := blocks[i].NumberU64(), blocks[i].Hash() | 		num, hash := blocks[i].NumberU64(), blocks[i].Hash() | ||||||
| 
 | 
 | ||||||
| 		if ftd, atd := fast.GetTd(hash), archive.GetTd(hash); ftd.Cmp(atd) != 0 { | 		if ftd, atd := fast.GetTdByHash(hash), archive.GetTdByHash(hash); ftd.Cmp(atd) != 0 { | ||||||
| 			t.Errorf("block #%d [%x]: td mismatch: have %v, want %v", num, hash, ftd, atd) | 			t.Errorf("block #%d [%x]: td mismatch: have %v, want %v", num, hash, ftd, atd) | ||||||
| 		} | 		} | ||||||
| 		if fheader, aheader := fast.GetHeader(hash), archive.GetHeader(hash); fheader.Hash() != aheader.Hash() { | 		if fheader, aheader := fast.GetHeaderByHash(hash), archive.GetHeaderByHash(hash); fheader.Hash() != aheader.Hash() { | ||||||
| 			t.Errorf("block #%d [%x]: header mismatch: have %v, want %v", num, hash, fheader, aheader) | 			t.Errorf("block #%d [%x]: header mismatch: have %v, want %v", num, hash, fheader, aheader) | ||||||
| 		} | 		} | ||||||
| 		if fblock, ablock := fast.GetBlock(hash), archive.GetBlock(hash); fblock.Hash() != ablock.Hash() { | 		if fblock, ablock := fast.GetBlockByHash(hash), archive.GetBlockByHash(hash); fblock.Hash() != ablock.Hash() { | ||||||
| 			t.Errorf("block #%d [%x]: block mismatch: have %v, want %v", num, hash, fblock, ablock) | 			t.Errorf("block #%d [%x]: block mismatch: have %v, want %v", num, hash, fblock, ablock) | ||||||
| 		} else if types.DeriveSha(fblock.Transactions()) != types.DeriveSha(ablock.Transactions()) { | 		} else if types.DeriveSha(fblock.Transactions()) != types.DeriveSha(ablock.Transactions()) { | ||||||
| 			t.Errorf("block #%d [%x]: transactions mismatch: have %v, want %v", num, hash, fblock.Transactions(), ablock.Transactions()) | 			t.Errorf("block #%d [%x]: transactions mismatch: have %v, want %v", num, hash, fblock.Transactions(), ablock.Transactions()) | ||||||
| 		} else if types.CalcUncleHash(fblock.Uncles()) != types.CalcUncleHash(ablock.Uncles()) { | 		} else if types.CalcUncleHash(fblock.Uncles()) != types.CalcUncleHash(ablock.Uncles()) { | ||||||
| 			t.Errorf("block #%d [%x]: uncles mismatch: have %v, want %v", num, hash, fblock.Uncles(), ablock.Uncles()) | 			t.Errorf("block #%d [%x]: uncles mismatch: have %v, want %v", num, hash, fblock.Uncles(), ablock.Uncles()) | ||||||
| 		} | 		} | ||||||
| 		if freceipts, areceipts := GetBlockReceipts(fastDb, hash), GetBlockReceipts(archiveDb, hash); types.DeriveSha(freceipts) != types.DeriveSha(areceipts) { | 		if freceipts, areceipts := GetBlockReceipts(fastDb, hash, GetBlockNumber(fastDb, hash)), GetBlockReceipts(archiveDb, hash, GetBlockNumber(archiveDb, hash)); types.DeriveSha(freceipts) != types.DeriveSha(areceipts) { | ||||||
| 			t.Errorf("block #%d [%x]: receipts mismatch: have %v, want %v", num, hash, freceipts, areceipts) | 			t.Errorf("block #%d [%x]: receipts mismatch: have %v, want %v", num, hash, freceipts, areceipts) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -36,34 +36,72 @@ var ( | |||||||
| 	headBlockKey  = []byte("LastBlock") | 	headBlockKey  = []byte("LastBlock") | ||||||
| 	headFastKey   = []byte("LastFast") | 	headFastKey   = []byte("LastFast") | ||||||
| 
 | 
 | ||||||
| 	blockPrefix    = []byte("block-") | 	headerPrefix        = []byte("h") // headerPrefix + num (uint64 big endian) + hash -> header
 | ||||||
| 	blockNumPrefix = []byte("block-num-") | 	tdSuffix            = []byte("t") // headerPrefix + num (uint64 big endian) + hash + tdSuffix -> td
 | ||||||
|  | 	numSuffix           = []byte("n") // headerPrefix + num (uint64 big endian) + numSuffix -> hash
 | ||||||
|  | 	blockHashPrefix     = []byte("H") // blockHashPrefix + hash -> num (uint64 big endian)
 | ||||||
|  | 	bodyPrefix          = []byte("b") // bodyPrefix + num (uint64 big endian) + hash -> block body
 | ||||||
|  | 	blockReceiptsPrefix = []byte("r") // blockReceiptsPrefix + num (uint64 big endian) + hash -> block receipts
 | ||||||
| 
 | 
 | ||||||
| 	headerSuffix = []byte("-header") | 	txMetaSuffix   = []byte{0x01} | ||||||
| 	bodySuffix   = []byte("-body") | 	receiptsPrefix = []byte("receipts-") | ||||||
| 	tdSuffix     = []byte("-td") |  | ||||||
| 
 |  | ||||||
| 	txMetaSuffix        = []byte{0x01} |  | ||||||
| 	receiptsPrefix      = []byte("receipts-") |  | ||||||
| 	blockReceiptsPrefix = []byte("receipts-block-") |  | ||||||
| 
 | 
 | ||||||
| 	mipmapPre    = []byte("mipmap-log-bloom-") | 	mipmapPre    = []byte("mipmap-log-bloom-") | ||||||
| 	MIPMapLevels = []uint64{1000000, 500000, 100000, 50000, 1000} | 	MIPMapLevels = []uint64{1000000, 500000, 100000, 50000, 1000} | ||||||
| 
 | 
 | ||||||
| 	blockHashPrefix = []byte("block-hash-") // [deprecated by the header/block split, remove eventually]
 |  | ||||||
| 
 |  | ||||||
| 	configPrefix = []byte("ethereum-config-") // config prefix for the db
 | 	configPrefix = []byte("ethereum-config-") // config prefix for the db
 | ||||||
|  | 
 | ||||||
|  | 	// used by old (non-sequential keys) db, now only used for conversion
 | ||||||
|  | 	oldBlockPrefix         = []byte("block-") | ||||||
|  | 	oldHeaderSuffix        = []byte("-header") | ||||||
|  | 	oldTdSuffix            = []byte("-td") // headerPrefix + num (uint64 big endian) + hash + tdSuffix -> td
 | ||||||
|  | 	oldBodySuffix          = []byte("-body") | ||||||
|  | 	oldBlockNumPrefix      = []byte("block-num-") | ||||||
|  | 	oldBlockReceiptsPrefix = []byte("receipts-block-") | ||||||
|  | 	oldBlockHashPrefix     = []byte("block-hash-") // [deprecated by the header/block split, remove eventually]
 | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // encodeBlockNumber encodes a block number as big endian uint64
 | ||||||
|  | func encodeBlockNumber(number uint64) []byte { | ||||||
|  | 	enc := make([]byte, 8) | ||||||
|  | 	binary.BigEndian.PutUint64(enc, number) | ||||||
|  | 	return enc | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // GetCanonicalHash retrieves a hash assigned to a canonical block number.
 | // GetCanonicalHash retrieves a hash assigned to a canonical block number.
 | ||||||
| func GetCanonicalHash(db ethdb.Database, number uint64) common.Hash { | func GetCanonicalHash(db ethdb.Database, number uint64) common.Hash { | ||||||
| 	data, _ := db.Get(append(blockNumPrefix, big.NewInt(int64(number)).Bytes()...)) | 	data, _ := db.Get(append(append(headerPrefix, encodeBlockNumber(number)...), numSuffix...)) | ||||||
| 	if len(data) == 0 { | 	if len(data) == 0 { | ||||||
| 		return common.Hash{} | 		data, _ = db.Get(append(oldBlockNumPrefix, big.NewInt(int64(number)).Bytes()...)) | ||||||
|  | 		if len(data) == 0 { | ||||||
|  | 			return common.Hash{} | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	return common.BytesToHash(data) | 	return common.BytesToHash(data) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // missingNumber is returned by GetBlockNumber if no header with the
 | ||||||
|  | // given block hash has been stored in the database
 | ||||||
|  | const missingNumber = uint64(0xffffffffffffffff) | ||||||
|  | 
 | ||||||
|  | // GetBlockNumber returns the block number assigned to a block hash
 | ||||||
|  | // if the corresponding header is present in the database
 | ||||||
|  | func GetBlockNumber(db ethdb.Database, hash common.Hash) uint64 { | ||||||
|  | 	data, _ := db.Get(append(blockHashPrefix, hash.Bytes()...)) | ||||||
|  | 	if len(data) != 8 { | ||||||
|  | 		data, _ := db.Get(append(append(oldBlockPrefix, hash.Bytes()...), oldHeaderSuffix...)) | ||||||
|  | 		if len(data) == 0 { | ||||||
|  | 			return missingNumber | ||||||
|  | 		} | ||||||
|  | 		header := new(types.Header) | ||||||
|  | 		if err := rlp.Decode(bytes.NewReader(data), header); err != nil { | ||||||
|  | 			glog.Fatalf("failed to decode block header: %v", err) | ||||||
|  | 		} | ||||||
|  | 		return header.Number.Uint64() | ||||||
|  | 	} | ||||||
|  | 	return binary.BigEndian.Uint64(data) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // GetHeadHeaderHash retrieves the hash of the current canonical head block's
 | // GetHeadHeaderHash retrieves the hash of the current canonical head block's
 | ||||||
| // header. The difference between this and GetHeadBlockHash is that whereas the
 | // header. The difference between this and GetHeadBlockHash is that whereas the
 | ||||||
| // last block hash is only updated upon a full block import, the last header
 | // last block hash is only updated upon a full block import, the last header
 | ||||||
| @ -100,15 +138,18 @@ func GetHeadFastBlockHash(db ethdb.Database) common.Hash { | |||||||
| 
 | 
 | ||||||
| // GetHeaderRLP retrieves a block header in its raw RLP database encoding, or nil
 | // GetHeaderRLP retrieves a block header in its raw RLP database encoding, or nil
 | ||||||
| // if the header's not found.
 | // if the header's not found.
 | ||||||
| func GetHeaderRLP(db ethdb.Database, hash common.Hash) rlp.RawValue { | func GetHeaderRLP(db ethdb.Database, hash common.Hash, number uint64) rlp.RawValue { | ||||||
| 	data, _ := db.Get(append(append(blockPrefix, hash[:]...), headerSuffix...)) | 	data, _ := db.Get(append(append(headerPrefix, encodeBlockNumber(number)...), hash.Bytes()...)) | ||||||
|  | 	if len(data) == 0 { | ||||||
|  | 		data, _ = db.Get(append(append(oldBlockPrefix, hash.Bytes()...), oldHeaderSuffix...)) | ||||||
|  | 	} | ||||||
| 	return data | 	return data | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetHeader retrieves the block header corresponding to the hash, nil if none
 | // GetHeader retrieves the block header corresponding to the hash, nil if none
 | ||||||
| // found.
 | // found.
 | ||||||
| func GetHeader(db ethdb.Database, hash common.Hash) *types.Header { | func GetHeader(db ethdb.Database, hash common.Hash, number uint64) *types.Header { | ||||||
| 	data := GetHeaderRLP(db, hash) | 	data := GetHeaderRLP(db, hash, number) | ||||||
| 	if len(data) == 0 { | 	if len(data) == 0 { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| @ -121,15 +162,18 @@ func GetHeader(db ethdb.Database, hash common.Hash) *types.Header { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetBodyRLP retrieves the block body (transactions and uncles) in RLP encoding.
 | // GetBodyRLP retrieves the block body (transactions and uncles) in RLP encoding.
 | ||||||
| func GetBodyRLP(db ethdb.Database, hash common.Hash) rlp.RawValue { | func GetBodyRLP(db ethdb.Database, hash common.Hash, number uint64) rlp.RawValue { | ||||||
| 	data, _ := db.Get(append(append(blockPrefix, hash[:]...), bodySuffix...)) | 	data, _ := db.Get(append(append(bodyPrefix, encodeBlockNumber(number)...), hash.Bytes()...)) | ||||||
|  | 	if len(data) == 0 { | ||||||
|  | 		data, _ = db.Get(append(append(oldBlockPrefix, hash.Bytes()...), oldBodySuffix...)) | ||||||
|  | 	} | ||||||
| 	return data | 	return data | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetBody retrieves the block body (transactons, uncles) corresponding to the
 | // GetBody retrieves the block body (transactons, uncles) corresponding to the
 | ||||||
| // hash, nil if none found.
 | // hash, nil if none found.
 | ||||||
| func GetBody(db ethdb.Database, hash common.Hash) *types.Body { | func GetBody(db ethdb.Database, hash common.Hash, number uint64) *types.Body { | ||||||
| 	data := GetBodyRLP(db, hash) | 	data := GetBodyRLP(db, hash, number) | ||||||
| 	if len(data) == 0 { | 	if len(data) == 0 { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| @ -143,10 +187,13 @@ func GetBody(db ethdb.Database, hash common.Hash) *types.Body { | |||||||
| 
 | 
 | ||||||
| // GetTd retrieves a block's total difficulty corresponding to the hash, nil if
 | // GetTd retrieves a block's total difficulty corresponding to the hash, nil if
 | ||||||
| // none found.
 | // none found.
 | ||||||
| func GetTd(db ethdb.Database, hash common.Hash) *big.Int { | func GetTd(db ethdb.Database, hash common.Hash, number uint64) *big.Int { | ||||||
| 	data, _ := db.Get(append(append(blockPrefix, hash.Bytes()...), tdSuffix...)) | 	data, _ := db.Get(append(append(append(headerPrefix, encodeBlockNumber(number)...), hash[:]...), tdSuffix...)) | ||||||
| 	if len(data) == 0 { | 	if len(data) == 0 { | ||||||
| 		return nil | 		data, _ = db.Get(append(append(oldBlockPrefix, hash.Bytes()...), oldTdSuffix...)) | ||||||
|  | 		if len(data) == 0 { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	td := new(big.Int) | 	td := new(big.Int) | ||||||
| 	if err := rlp.Decode(bytes.NewReader(data), td); err != nil { | 	if err := rlp.Decode(bytes.NewReader(data), td); err != nil { | ||||||
| @ -158,13 +205,13 @@ func GetTd(db ethdb.Database, hash common.Hash) *big.Int { | |||||||
| 
 | 
 | ||||||
| // GetBlock retrieves an entire block corresponding to the hash, assembling it
 | // GetBlock retrieves an entire block corresponding to the hash, assembling it
 | ||||||
| // back from the stored header and body.
 | // back from the stored header and body.
 | ||||||
| func GetBlock(db ethdb.Database, hash common.Hash) *types.Block { | func GetBlock(db ethdb.Database, hash common.Hash, number uint64) *types.Block { | ||||||
| 	// Retrieve the block header and body contents
 | 	// Retrieve the block header and body contents
 | ||||||
| 	header := GetHeader(db, hash) | 	header := GetHeader(db, hash, number) | ||||||
| 	if header == nil { | 	if header == nil { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	body := GetBody(db, hash) | 	body := GetBody(db, hash, number) | ||||||
| 	if body == nil { | 	if body == nil { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| @ -174,10 +221,13 @@ func GetBlock(db ethdb.Database, hash common.Hash) *types.Block { | |||||||
| 
 | 
 | ||||||
| // GetBlockReceipts retrieves the receipts generated by the transactions included
 | // GetBlockReceipts retrieves the receipts generated by the transactions included
 | ||||||
| // in a block given by its hash.
 | // in a block given by its hash.
 | ||||||
| func GetBlockReceipts(db ethdb.Database, hash common.Hash) types.Receipts { | func GetBlockReceipts(db ethdb.Database, hash common.Hash, number uint64) types.Receipts { | ||||||
| 	data, _ := db.Get(append(blockReceiptsPrefix, hash[:]...)) | 	data, _ := db.Get(append(append(blockReceiptsPrefix, encodeBlockNumber(number)...), hash[:]...)) | ||||||
| 	if len(data) == 0 { | 	if len(data) == 0 { | ||||||
| 		return nil | 		data, _ = db.Get(append(oldBlockReceiptsPrefix, hash.Bytes()...)) | ||||||
|  | 		if len(data) == 0 { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	storageReceipts := []*types.ReceiptForStorage{} | 	storageReceipts := []*types.ReceiptForStorage{} | ||||||
| 	if err := rlp.DecodeBytes(data, &storageReceipts); err != nil { | 	if err := rlp.DecodeBytes(data, &storageReceipts); err != nil { | ||||||
| @ -235,10 +285,9 @@ func GetReceipt(db ethdb.Database, txHash common.Hash) *types.Receipt { | |||||||
| 
 | 
 | ||||||
| // WriteCanonicalHash stores the canonical hash for the given block number.
 | // WriteCanonicalHash stores the canonical hash for the given block number.
 | ||||||
| func WriteCanonicalHash(db ethdb.Database, hash common.Hash, number uint64) error { | func WriteCanonicalHash(db ethdb.Database, hash common.Hash, number uint64) error { | ||||||
| 	key := append(blockNumPrefix, big.NewInt(int64(number)).Bytes()...) | 	key := append(append(headerPrefix, encodeBlockNumber(number)...), numSuffix...) | ||||||
| 	if err := db.Put(key, hash.Bytes()); err != nil { | 	if err := db.Put(key, hash.Bytes()); err != nil { | ||||||
| 		glog.Fatalf("failed to store number to hash mapping into database: %v", err) | 		glog.Fatalf("failed to store number to hash mapping into database: %v", err) | ||||||
| 		return err |  | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| @ -247,7 +296,6 @@ func WriteCanonicalHash(db ethdb.Database, hash common.Hash, number uint64) erro | |||||||
| func WriteHeadHeaderHash(db ethdb.Database, hash common.Hash) error { | func WriteHeadHeaderHash(db ethdb.Database, hash common.Hash) error { | ||||||
| 	if err := db.Put(headHeaderKey, hash.Bytes()); err != nil { | 	if err := db.Put(headHeaderKey, hash.Bytes()); err != nil { | ||||||
| 		glog.Fatalf("failed to store last header's hash into database: %v", err) | 		glog.Fatalf("failed to store last header's hash into database: %v", err) | ||||||
| 		return err |  | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| @ -256,7 +304,6 @@ func WriteHeadHeaderHash(db ethdb.Database, hash common.Hash) error { | |||||||
| func WriteHeadBlockHash(db ethdb.Database, hash common.Hash) error { | func WriteHeadBlockHash(db ethdb.Database, hash common.Hash) error { | ||||||
| 	if err := db.Put(headBlockKey, hash.Bytes()); err != nil { | 	if err := db.Put(headBlockKey, hash.Bytes()); err != nil { | ||||||
| 		glog.Fatalf("failed to store last block's hash into database: %v", err) | 		glog.Fatalf("failed to store last block's hash into database: %v", err) | ||||||
| 		return err |  | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| @ -265,7 +312,6 @@ func WriteHeadBlockHash(db ethdb.Database, hash common.Hash) error { | |||||||
| func WriteHeadFastBlockHash(db ethdb.Database, hash common.Hash) error { | func WriteHeadFastBlockHash(db ethdb.Database, hash common.Hash) error { | ||||||
| 	if err := db.Put(headFastKey, hash.Bytes()); err != nil { | 	if err := db.Put(headFastKey, hash.Bytes()); err != nil { | ||||||
| 		glog.Fatalf("failed to store last fast block's hash into database: %v", err) | 		glog.Fatalf("failed to store last fast block's hash into database: %v", err) | ||||||
| 		return err |  | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| @ -276,40 +322,44 @@ func WriteHeader(db ethdb.Database, header *types.Header) error { | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	key := append(append(blockPrefix, header.Hash().Bytes()...), headerSuffix...) | 	hash := header.Hash().Bytes() | ||||||
|  | 	num := header.Number.Uint64() | ||||||
|  | 	encNum := encodeBlockNumber(num) | ||||||
|  | 	key := append(blockHashPrefix, hash...) | ||||||
|  | 	if err := db.Put(key, encNum); err != nil { | ||||||
|  | 		glog.Fatalf("failed to store hash to number mapping into database: %v", err) | ||||||
|  | 	} | ||||||
|  | 	key = append(append(headerPrefix, encNum...), hash...) | ||||||
| 	if err := db.Put(key, data); err != nil { | 	if err := db.Put(key, data); err != nil { | ||||||
| 		glog.Fatalf("failed to store header into database: %v", err) | 		glog.Fatalf("failed to store header into database: %v", err) | ||||||
| 		return err |  | ||||||
| 	} | 	} | ||||||
| 	glog.V(logger.Debug).Infof("stored header #%v [%x…]", header.Number, header.Hash().Bytes()[:4]) | 	glog.V(logger.Debug).Infof("stored header #%v [%x…]", header.Number, hash[:4]) | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // WriteBody serializes the body of a block into the database.
 | // WriteBody serializes the body of a block into the database.
 | ||||||
| func WriteBody(db ethdb.Database, hash common.Hash, body *types.Body) error { | func WriteBody(db ethdb.Database, hash common.Hash, number uint64, body *types.Body) error { | ||||||
| 	data, err := rlp.EncodeToBytes(body) | 	data, err := rlp.EncodeToBytes(body) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	key := append(append(blockPrefix, hash.Bytes()...), bodySuffix...) | 	key := append(append(bodyPrefix, encodeBlockNumber(number)...), hash.Bytes()...) | ||||||
| 	if err := db.Put(key, data); err != nil { | 	if err := db.Put(key, data); err != nil { | ||||||
| 		glog.Fatalf("failed to store block body into database: %v", err) | 		glog.Fatalf("failed to store block body into database: %v", err) | ||||||
| 		return err |  | ||||||
| 	} | 	} | ||||||
| 	glog.V(logger.Debug).Infof("stored block body [%x…]", hash.Bytes()[:4]) | 	glog.V(logger.Debug).Infof("stored block body [%x…]", hash.Bytes()[:4]) | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // WriteTd serializes the total difficulty of a block into the database.
 | // WriteTd serializes the total difficulty of a block into the database.
 | ||||||
| func WriteTd(db ethdb.Database, hash common.Hash, td *big.Int) error { | func WriteTd(db ethdb.Database, hash common.Hash, number uint64, td *big.Int) error { | ||||||
| 	data, err := rlp.EncodeToBytes(td) | 	data, err := rlp.EncodeToBytes(td) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	key := append(append(blockPrefix, hash.Bytes()...), tdSuffix...) | 	key := append(append(append(headerPrefix, encodeBlockNumber(number)...), hash.Bytes()...), tdSuffix...) | ||||||
| 	if err := db.Put(key, data); err != nil { | 	if err := db.Put(key, data); err != nil { | ||||||
| 		glog.Fatalf("failed to store block total difficulty into database: %v", err) | 		glog.Fatalf("failed to store block total difficulty into database: %v", err) | ||||||
| 		return err |  | ||||||
| 	} | 	} | ||||||
| 	glog.V(logger.Debug).Infof("stored block total difficulty [%x…]: %v", hash.Bytes()[:4], td) | 	glog.V(logger.Debug).Infof("stored block total difficulty [%x…]: %v", hash.Bytes()[:4], td) | ||||||
| 	return nil | 	return nil | ||||||
| @ -318,7 +368,7 @@ func WriteTd(db ethdb.Database, hash common.Hash, td *big.Int) error { | |||||||
| // WriteBlock serializes a block into the database, header and body separately.
 | // WriteBlock serializes a block into the database, header and body separately.
 | ||||||
| func WriteBlock(db ethdb.Database, block *types.Block) error { | func WriteBlock(db ethdb.Database, block *types.Block) error { | ||||||
| 	// Store the body first to retain database consistency
 | 	// Store the body first to retain database consistency
 | ||||||
| 	if err := WriteBody(db, block.Hash(), block.Body()); err != nil { | 	if err := WriteBody(db, block.Hash(), block.NumberU64(), block.Body()); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	// Store the header too, signaling full block ownership
 | 	// Store the header too, signaling full block ownership
 | ||||||
| @ -331,7 +381,7 @@ func WriteBlock(db ethdb.Database, block *types.Block) error { | |||||||
| // WriteBlockReceipts stores all the transaction receipts belonging to a block
 | // WriteBlockReceipts stores all the transaction receipts belonging to a block
 | ||||||
| // as a single receipt slice. This is used during chain reorganisations for
 | // as a single receipt slice. This is used during chain reorganisations for
 | ||||||
| // rescheduling dropped transactions.
 | // rescheduling dropped transactions.
 | ||||||
| func WriteBlockReceipts(db ethdb.Database, hash common.Hash, receipts types.Receipts) error { | func WriteBlockReceipts(db ethdb.Database, hash common.Hash, number uint64, receipts types.Receipts) error { | ||||||
| 	// Convert the receipts into their storage form and serialize them
 | 	// Convert the receipts into their storage form and serialize them
 | ||||||
| 	storageReceipts := make([]*types.ReceiptForStorage, len(receipts)) | 	storageReceipts := make([]*types.ReceiptForStorage, len(receipts)) | ||||||
| 	for i, receipt := range receipts { | 	for i, receipt := range receipts { | ||||||
| @ -342,9 +392,9 @@ func WriteBlockReceipts(db ethdb.Database, hash common.Hash, receipts types.Rece | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	// Store the flattened receipt slice
 | 	// Store the flattened receipt slice
 | ||||||
| 	if err := db.Put(append(blockReceiptsPrefix, hash.Bytes()...), bytes); err != nil { | 	key := append(append(blockReceiptsPrefix, encodeBlockNumber(number)...), hash.Bytes()...) | ||||||
|  | 	if err := db.Put(key, bytes); err != nil { | ||||||
| 		glog.Fatalf("failed to store block receipts into database: %v", err) | 		glog.Fatalf("failed to store block receipts into database: %v", err) | ||||||
| 		return err |  | ||||||
| 	} | 	} | ||||||
| 	glog.V(logger.Debug).Infof("stored block receipts [%x…]", hash.Bytes()[:4]) | 	glog.V(logger.Debug).Infof("stored block receipts [%x…]", hash.Bytes()[:4]) | ||||||
| 	return nil | 	return nil | ||||||
| @ -388,7 +438,6 @@ func WriteTransactions(db ethdb.Database, block *types.Block) error { | |||||||
| 	// Write the scheduled data into the database
 | 	// Write the scheduled data into the database
 | ||||||
| 	if err := batch.Write(); err != nil { | 	if err := batch.Write(); err != nil { | ||||||
| 		glog.Fatalf("failed to store transactions into database: %v", err) | 		glog.Fatalf("failed to store transactions into database: %v", err) | ||||||
| 		return err |  | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| @ -411,42 +460,42 @@ func WriteReceipts(db ethdb.Database, receipts types.Receipts) error { | |||||||
| 	// Write the scheduled data into the database
 | 	// Write the scheduled data into the database
 | ||||||
| 	if err := batch.Write(); err != nil { | 	if err := batch.Write(); err != nil { | ||||||
| 		glog.Fatalf("failed to store receipts into database: %v", err) | 		glog.Fatalf("failed to store receipts into database: %v", err) | ||||||
| 		return err |  | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // DeleteCanonicalHash removes the number to hash canonical mapping.
 | // DeleteCanonicalHash removes the number to hash canonical mapping.
 | ||||||
| func DeleteCanonicalHash(db ethdb.Database, number uint64) { | func DeleteCanonicalHash(db ethdb.Database, number uint64) { | ||||||
| 	db.Delete(append(blockNumPrefix, big.NewInt(int64(number)).Bytes()...)) | 	db.Delete(append(append(headerPrefix, encodeBlockNumber(number)...), numSuffix...)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // DeleteHeader removes all block header data associated with a hash.
 | // DeleteHeader removes all block header data associated with a hash.
 | ||||||
| func DeleteHeader(db ethdb.Database, hash common.Hash) { | func DeleteHeader(db ethdb.Database, hash common.Hash, number uint64) { | ||||||
| 	db.Delete(append(append(blockPrefix, hash.Bytes()...), headerSuffix...)) | 	db.Delete(append(blockHashPrefix, hash.Bytes()...)) | ||||||
|  | 	db.Delete(append(append(headerPrefix, encodeBlockNumber(number)...), hash.Bytes()...)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // DeleteBody removes all block body data associated with a hash.
 | // DeleteBody removes all block body data associated with a hash.
 | ||||||
| func DeleteBody(db ethdb.Database, hash common.Hash) { | func DeleteBody(db ethdb.Database, hash common.Hash, number uint64) { | ||||||
| 	db.Delete(append(append(blockPrefix, hash.Bytes()...), bodySuffix...)) | 	db.Delete(append(append(bodyPrefix, encodeBlockNumber(number)...), hash.Bytes()...)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // DeleteTd removes all block total difficulty data associated with a hash.
 | // DeleteTd removes all block total difficulty data associated with a hash.
 | ||||||
| func DeleteTd(db ethdb.Database, hash common.Hash) { | func DeleteTd(db ethdb.Database, hash common.Hash, number uint64) { | ||||||
| 	db.Delete(append(append(blockPrefix, hash.Bytes()...), tdSuffix...)) | 	db.Delete(append(append(append(headerPrefix, encodeBlockNumber(number)...), hash.Bytes()...), tdSuffix...)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // DeleteBlock removes all block data associated with a hash.
 | // DeleteBlock removes all block data associated with a hash.
 | ||||||
| func DeleteBlock(db ethdb.Database, hash common.Hash) { | func DeleteBlock(db ethdb.Database, hash common.Hash, number uint64) { | ||||||
| 	DeleteBlockReceipts(db, hash) | 	DeleteBlockReceipts(db, hash, number) | ||||||
| 	DeleteHeader(db, hash) | 	DeleteHeader(db, hash, number) | ||||||
| 	DeleteBody(db, hash) | 	DeleteBody(db, hash, number) | ||||||
| 	DeleteTd(db, hash) | 	DeleteTd(db, hash, number) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // DeleteBlockReceipts removes all receipt data associated with a block hash.
 | // DeleteBlockReceipts removes all receipt data associated with a block hash.
 | ||||||
| func DeleteBlockReceipts(db ethdb.Database, hash common.Hash) { | func DeleteBlockReceipts(db ethdb.Database, hash common.Hash, number uint64) { | ||||||
| 	db.Delete(append(blockReceiptsPrefix, hash.Bytes()...)) | 	db.Delete(append(append(blockReceiptsPrefix, encodeBlockNumber(number)...), hash.Bytes()...)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // DeleteTransaction removes all transaction data associated with a hash.
 | // DeleteTransaction removes all transaction data associated with a hash.
 | ||||||
| @ -466,7 +515,7 @@ func DeleteReceipt(db ethdb.Database, hash common.Hash) { | |||||||
| // access the old combined block representation. It will be dropped after the
 | // access the old combined block representation. It will be dropped after the
 | ||||||
| // network transitions to eth/63.
 | // network transitions to eth/63.
 | ||||||
| func GetBlockByHashOld(db ethdb.Database, hash common.Hash) *types.Block { | func GetBlockByHashOld(db ethdb.Database, hash common.Hash) *types.Block { | ||||||
| 	data, _ := db.Get(append(blockHashPrefix, hash[:]...)) | 	data, _ := db.Get(append(oldBlockHashPrefix, hash[:]...)) | ||||||
| 	if len(data) == 0 { | 	if len(data) == 0 { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -89,20 +89,20 @@ func TestHeaderStorage(t *testing.T) { | |||||||
| 	db, _ := ethdb.NewMemDatabase() | 	db, _ := ethdb.NewMemDatabase() | ||||||
| 
 | 
 | ||||||
| 	// Create a test header to move around the database and make sure it's really new
 | 	// Create a test header to move around the database and make sure it's really new
 | ||||||
| 	header := &types.Header{Extra: []byte("test header")} | 	header := &types.Header{Number: big.NewInt(42), Extra: []byte("test header")} | ||||||
| 	if entry := GetHeader(db, header.Hash()); entry != nil { | 	if entry := GetHeader(db, header.Hash(), header.Number.Uint64()); entry != nil { | ||||||
| 		t.Fatalf("Non existent header returned: %v", entry) | 		t.Fatalf("Non existent header returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| 	// Write and verify the header in the database
 | 	// Write and verify the header in the database
 | ||||||
| 	if err := WriteHeader(db, header); err != nil { | 	if err := WriteHeader(db, header); err != nil { | ||||||
| 		t.Fatalf("Failed to write header into database: %v", err) | 		t.Fatalf("Failed to write header into database: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetHeader(db, header.Hash()); entry == nil { | 	if entry := GetHeader(db, header.Hash(), header.Number.Uint64()); entry == nil { | ||||||
| 		t.Fatalf("Stored header not found") | 		t.Fatalf("Stored header not found") | ||||||
| 	} else if entry.Hash() != header.Hash() { | 	} else if entry.Hash() != header.Hash() { | ||||||
| 		t.Fatalf("Retrieved header mismatch: have %v, want %v", entry, header) | 		t.Fatalf("Retrieved header mismatch: have %v, want %v", entry, header) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetHeaderRLP(db, header.Hash()); entry == nil { | 	if entry := GetHeaderRLP(db, header.Hash(), header.Number.Uint64()); entry == nil { | ||||||
| 		t.Fatalf("Stored header RLP not found") | 		t.Fatalf("Stored header RLP not found") | ||||||
| 	} else { | 	} else { | ||||||
| 		hasher := sha3.NewKeccak256() | 		hasher := sha3.NewKeccak256() | ||||||
| @ -113,8 +113,8 @@ func TestHeaderStorage(t *testing.T) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	// Delete the header and verify the execution
 | 	// Delete the header and verify the execution
 | ||||||
| 	DeleteHeader(db, header.Hash()) | 	DeleteHeader(db, header.Hash(), header.Number.Uint64()) | ||||||
| 	if entry := GetHeader(db, header.Hash()); entry != nil { | 	if entry := GetHeader(db, header.Hash(), header.Number.Uint64()); entry != nil { | ||||||
| 		t.Fatalf("Deleted header returned: %v", entry) | 		t.Fatalf("Deleted header returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -130,19 +130,19 @@ func TestBodyStorage(t *testing.T) { | |||||||
| 	rlp.Encode(hasher, body) | 	rlp.Encode(hasher, body) | ||||||
| 	hash := common.BytesToHash(hasher.Sum(nil)) | 	hash := common.BytesToHash(hasher.Sum(nil)) | ||||||
| 
 | 
 | ||||||
| 	if entry := GetBody(db, hash); entry != nil { | 	if entry := GetBody(db, hash, 0); entry != nil { | ||||||
| 		t.Fatalf("Non existent body returned: %v", entry) | 		t.Fatalf("Non existent body returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| 	// Write and verify the body in the database
 | 	// Write and verify the body in the database
 | ||||||
| 	if err := WriteBody(db, hash, body); err != nil { | 	if err := WriteBody(db, hash, 0, body); err != nil { | ||||||
| 		t.Fatalf("Failed to write body into database: %v", err) | 		t.Fatalf("Failed to write body into database: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetBody(db, hash); entry == nil { | 	if entry := GetBody(db, hash, 0); entry == nil { | ||||||
| 		t.Fatalf("Stored body not found") | 		t.Fatalf("Stored body not found") | ||||||
| 	} else if types.DeriveSha(types.Transactions(entry.Transactions)) != types.DeriveSha(types.Transactions(body.Transactions)) || types.CalcUncleHash(entry.Uncles) != types.CalcUncleHash(body.Uncles) { | 	} else if types.DeriveSha(types.Transactions(entry.Transactions)) != types.DeriveSha(types.Transactions(body.Transactions)) || types.CalcUncleHash(entry.Uncles) != types.CalcUncleHash(body.Uncles) { | ||||||
| 		t.Fatalf("Retrieved body mismatch: have %v, want %v", entry, body) | 		t.Fatalf("Retrieved body mismatch: have %v, want %v", entry, body) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetBodyRLP(db, hash); entry == nil { | 	if entry := GetBodyRLP(db, hash, 0); entry == nil { | ||||||
| 		t.Fatalf("Stored body RLP not found") | 		t.Fatalf("Stored body RLP not found") | ||||||
| 	} else { | 	} else { | ||||||
| 		hasher := sha3.NewKeccak256() | 		hasher := sha3.NewKeccak256() | ||||||
| @ -153,8 +153,8 @@ func TestBodyStorage(t *testing.T) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	// Delete the body and verify the execution
 | 	// Delete the body and verify the execution
 | ||||||
| 	DeleteBody(db, hash) | 	DeleteBody(db, hash, 0) | ||||||
| 	if entry := GetBody(db, hash); entry != nil { | 	if entry := GetBody(db, hash, 0); entry != nil { | ||||||
| 		t.Fatalf("Deleted body returned: %v", entry) | 		t.Fatalf("Deleted body returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -170,43 +170,43 @@ func TestBlockStorage(t *testing.T) { | |||||||
| 		TxHash:      types.EmptyRootHash, | 		TxHash:      types.EmptyRootHash, | ||||||
| 		ReceiptHash: types.EmptyRootHash, | 		ReceiptHash: types.EmptyRootHash, | ||||||
| 	}) | 	}) | ||||||
| 	if entry := GetBlock(db, block.Hash()); entry != nil { | 	if entry := GetBlock(db, block.Hash(), block.NumberU64()); entry != nil { | ||||||
| 		t.Fatalf("Non existent block returned: %v", entry) | 		t.Fatalf("Non existent block returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetHeader(db, block.Hash()); entry != nil { | 	if entry := GetHeader(db, block.Hash(), block.NumberU64()); entry != nil { | ||||||
| 		t.Fatalf("Non existent header returned: %v", entry) | 		t.Fatalf("Non existent header returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetBody(db, block.Hash()); entry != nil { | 	if entry := GetBody(db, block.Hash(), block.NumberU64()); entry != nil { | ||||||
| 		t.Fatalf("Non existent body returned: %v", entry) | 		t.Fatalf("Non existent body returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| 	// Write and verify the block in the database
 | 	// Write and verify the block in the database
 | ||||||
| 	if err := WriteBlock(db, block); err != nil { | 	if err := WriteBlock(db, block); err != nil { | ||||||
| 		t.Fatalf("Failed to write block into database: %v", err) | 		t.Fatalf("Failed to write block into database: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetBlock(db, block.Hash()); entry == nil { | 	if entry := GetBlock(db, block.Hash(), block.NumberU64()); entry == nil { | ||||||
| 		t.Fatalf("Stored block not found") | 		t.Fatalf("Stored block not found") | ||||||
| 	} else if entry.Hash() != block.Hash() { | 	} else if entry.Hash() != block.Hash() { | ||||||
| 		t.Fatalf("Retrieved block mismatch: have %v, want %v", entry, block) | 		t.Fatalf("Retrieved block mismatch: have %v, want %v", entry, block) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetHeader(db, block.Hash()); entry == nil { | 	if entry := GetHeader(db, block.Hash(), block.NumberU64()); entry == nil { | ||||||
| 		t.Fatalf("Stored header not found") | 		t.Fatalf("Stored header not found") | ||||||
| 	} else if entry.Hash() != block.Header().Hash() { | 	} else if entry.Hash() != block.Header().Hash() { | ||||||
| 		t.Fatalf("Retrieved header mismatch: have %v, want %v", entry, block.Header()) | 		t.Fatalf("Retrieved header mismatch: have %v, want %v", entry, block.Header()) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetBody(db, block.Hash()); entry == nil { | 	if entry := GetBody(db, block.Hash(), block.NumberU64()); entry == nil { | ||||||
| 		t.Fatalf("Stored body not found") | 		t.Fatalf("Stored body not found") | ||||||
| 	} else if types.DeriveSha(types.Transactions(entry.Transactions)) != types.DeriveSha(block.Transactions()) || types.CalcUncleHash(entry.Uncles) != types.CalcUncleHash(block.Uncles()) { | 	} else if types.DeriveSha(types.Transactions(entry.Transactions)) != types.DeriveSha(block.Transactions()) || types.CalcUncleHash(entry.Uncles) != types.CalcUncleHash(block.Uncles()) { | ||||||
| 		t.Fatalf("Retrieved body mismatch: have %v, want %v", entry, block.Body()) | 		t.Fatalf("Retrieved body mismatch: have %v, want %v", entry, block.Body()) | ||||||
| 	} | 	} | ||||||
| 	// Delete the block and verify the execution
 | 	// Delete the block and verify the execution
 | ||||||
| 	DeleteBlock(db, block.Hash()) | 	DeleteBlock(db, block.Hash(), block.NumberU64()) | ||||||
| 	if entry := GetBlock(db, block.Hash()); entry != nil { | 	if entry := GetBlock(db, block.Hash(), block.NumberU64()); entry != nil { | ||||||
| 		t.Fatalf("Deleted block returned: %v", entry) | 		t.Fatalf("Deleted block returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetHeader(db, block.Hash()); entry != nil { | 	if entry := GetHeader(db, block.Hash(), block.NumberU64()); entry != nil { | ||||||
| 		t.Fatalf("Deleted header returned: %v", entry) | 		t.Fatalf("Deleted header returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetBody(db, block.Hash()); entry != nil { | 	if entry := GetBody(db, block.Hash(), block.NumberU64()); entry != nil { | ||||||
| 		t.Fatalf("Deleted body returned: %v", entry) | 		t.Fatalf("Deleted body returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -224,28 +224,28 @@ func TestPartialBlockStorage(t *testing.T) { | |||||||
| 	if err := WriteHeader(db, block.Header()); err != nil { | 	if err := WriteHeader(db, block.Header()); err != nil { | ||||||
| 		t.Fatalf("Failed to write header into database: %v", err) | 		t.Fatalf("Failed to write header into database: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetBlock(db, block.Hash()); entry != nil { | 	if entry := GetBlock(db, block.Hash(), block.NumberU64()); entry != nil { | ||||||
| 		t.Fatalf("Non existent block returned: %v", entry) | 		t.Fatalf("Non existent block returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| 	DeleteHeader(db, block.Hash()) | 	DeleteHeader(db, block.Hash(), block.NumberU64()) | ||||||
| 
 | 
 | ||||||
| 	// Store a body and check that it's not recognized as a block
 | 	// Store a body and check that it's not recognized as a block
 | ||||||
| 	if err := WriteBody(db, block.Hash(), block.Body()); err != nil { | 	if err := WriteBody(db, block.Hash(), block.NumberU64(), block.Body()); err != nil { | ||||||
| 		t.Fatalf("Failed to write body into database: %v", err) | 		t.Fatalf("Failed to write body into database: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetBlock(db, block.Hash()); entry != nil { | 	if entry := GetBlock(db, block.Hash(), block.NumberU64()); entry != nil { | ||||||
| 		t.Fatalf("Non existent block returned: %v", entry) | 		t.Fatalf("Non existent block returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| 	DeleteBody(db, block.Hash()) | 	DeleteBody(db, block.Hash(), block.NumberU64()) | ||||||
| 
 | 
 | ||||||
| 	// Store a header and a body separately and check reassembly
 | 	// Store a header and a body separately and check reassembly
 | ||||||
| 	if err := WriteHeader(db, block.Header()); err != nil { | 	if err := WriteHeader(db, block.Header()); err != nil { | ||||||
| 		t.Fatalf("Failed to write header into database: %v", err) | 		t.Fatalf("Failed to write header into database: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if err := WriteBody(db, block.Hash(), block.Body()); err != nil { | 	if err := WriteBody(db, block.Hash(), block.NumberU64(), block.Body()); err != nil { | ||||||
| 		t.Fatalf("Failed to write body into database: %v", err) | 		t.Fatalf("Failed to write body into database: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetBlock(db, block.Hash()); entry == nil { | 	if entry := GetBlock(db, block.Hash(), block.NumberU64()); entry == nil { | ||||||
| 		t.Fatalf("Stored block not found") | 		t.Fatalf("Stored block not found") | ||||||
| 	} else if entry.Hash() != block.Hash() { | 	} else if entry.Hash() != block.Hash() { | ||||||
| 		t.Fatalf("Retrieved block mismatch: have %v, want %v", entry, block) | 		t.Fatalf("Retrieved block mismatch: have %v, want %v", entry, block) | ||||||
| @ -258,21 +258,21 @@ func TestTdStorage(t *testing.T) { | |||||||
| 
 | 
 | ||||||
| 	// Create a test TD to move around the database and make sure it's really new
 | 	// Create a test TD to move around the database and make sure it's really new
 | ||||||
| 	hash, td := common.Hash{}, big.NewInt(314) | 	hash, td := common.Hash{}, big.NewInt(314) | ||||||
| 	if entry := GetTd(db, hash); entry != nil { | 	if entry := GetTd(db, hash, 0); entry != nil { | ||||||
| 		t.Fatalf("Non existent TD returned: %v", entry) | 		t.Fatalf("Non existent TD returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| 	// Write and verify the TD in the database
 | 	// Write and verify the TD in the database
 | ||||||
| 	if err := WriteTd(db, hash, td); err != nil { | 	if err := WriteTd(db, hash, 0, td); err != nil { | ||||||
| 		t.Fatalf("Failed to write TD into database: %v", err) | 		t.Fatalf("Failed to write TD into database: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if entry := GetTd(db, hash); entry == nil { | 	if entry := GetTd(db, hash, 0); entry == nil { | ||||||
| 		t.Fatalf("Stored TD not found") | 		t.Fatalf("Stored TD not found") | ||||||
| 	} else if entry.Cmp(td) != 0 { | 	} else if entry.Cmp(td) != 0 { | ||||||
| 		t.Fatalf("Retrieved TD mismatch: have %v, want %v", entry, td) | 		t.Fatalf("Retrieved TD mismatch: have %v, want %v", entry, td) | ||||||
| 	} | 	} | ||||||
| 	// Delete the TD and verify the execution
 | 	// Delete the TD and verify the execution
 | ||||||
| 	DeleteTd(db, hash) | 	DeleteTd(db, hash, 0) | ||||||
| 	if entry := GetTd(db, hash); entry != nil { | 	if entry := GetTd(db, hash, 0); entry != nil { | ||||||
| 		t.Fatalf("Deleted TD returned: %v", entry) | 		t.Fatalf("Deleted TD returned: %v", entry) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -473,14 +473,14 @@ func TestBlockReceiptStorage(t *testing.T) { | |||||||
| 
 | 
 | ||||||
| 	// Check that no receipt entries are in a pristine database
 | 	// Check that no receipt entries are in a pristine database
 | ||||||
| 	hash := common.BytesToHash([]byte{0x03, 0x14}) | 	hash := common.BytesToHash([]byte{0x03, 0x14}) | ||||||
| 	if rs := GetBlockReceipts(db, hash); len(rs) != 0 { | 	if rs := GetBlockReceipts(db, hash, 0); len(rs) != 0 { | ||||||
| 		t.Fatalf("non existent receipts returned: %v", rs) | 		t.Fatalf("non existent receipts returned: %v", rs) | ||||||
| 	} | 	} | ||||||
| 	// Insert the receipt slice into the database and check presence
 | 	// Insert the receipt slice into the database and check presence
 | ||||||
| 	if err := WriteBlockReceipts(db, hash, receipts); err != nil { | 	if err := WriteBlockReceipts(db, hash, 0, receipts); err != nil { | ||||||
| 		t.Fatalf("failed to write block receipts: %v", err) | 		t.Fatalf("failed to write block receipts: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if rs := GetBlockReceipts(db, hash); len(rs) == 0 { | 	if rs := GetBlockReceipts(db, hash, 0); len(rs) == 0 { | ||||||
| 		t.Fatalf("no receipts returned") | 		t.Fatalf("no receipts returned") | ||||||
| 	} else { | 	} else { | ||||||
| 		for i := 0; i < len(receipts); i++ { | 		for i := 0; i < len(receipts); i++ { | ||||||
| @ -493,8 +493,8 @@ func TestBlockReceiptStorage(t *testing.T) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	// Delete the receipt slice and check purge
 | 	// Delete the receipt slice and check purge
 | ||||||
| 	DeleteBlockReceipts(db, hash) | 	DeleteBlockReceipts(db, hash, 0) | ||||||
| 	if rs := GetBlockReceipts(db, hash); len(rs) != 0 { | 	if rs := GetBlockReceipts(db, hash, 0); len(rs) != 0 { | ||||||
| 		t.Fatalf("deleted receipts returned: %v", rs) | 		t.Fatalf("deleted receipts returned: %v", rs) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -597,7 +597,7 @@ func TestMipmapChain(t *testing.T) { | |||||||
| 		if err := WriteHeadBlockHash(db, block.Hash()); err != nil { | 		if err := WriteHeadBlockHash(db, block.Hash()); err != nil { | ||||||
| 			t.Fatalf("failed to insert block number: %v", err) | 			t.Fatalf("failed to insert block number: %v", err) | ||||||
| 		} | 		} | ||||||
| 		if err := WriteBlockReceipts(db, block.Hash(), receipts[i]); err != nil { | 		if err := WriteBlockReceipts(db, block.Hash(), block.NumberU64(), receipts[i]); err != nil { | ||||||
| 			t.Fatal("error writing block receipts:", err) | 			t.Fatal("error writing block receipts:", err) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -88,7 +88,7 @@ func WriteGenesisBlock(chainDb ethdb.Database, reader io.Reader) (*types.Block, | |||||||
| 		Root:       root, | 		Root:       root, | ||||||
| 	}, nil, nil, nil) | 	}, nil, nil, nil) | ||||||
| 
 | 
 | ||||||
| 	if block := GetBlock(chainDb, block.Hash()); block != nil { | 	if block := GetBlock(chainDb, block.Hash(), block.NumberU64()); block != nil { | ||||||
| 		glog.V(logger.Info).Infoln("Genesis block already in chain. Writing canonical number") | 		glog.V(logger.Info).Infoln("Genesis block already in chain. Writing canonical number") | ||||||
| 		err := WriteCanonicalHash(chainDb, block.Hash(), block.NumberU64()) | 		err := WriteCanonicalHash(chainDb, block.Hash(), block.NumberU64()) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -100,13 +100,13 @@ func WriteGenesisBlock(chainDb ethdb.Database, reader io.Reader) (*types.Block, | |||||||
| 	if err := stateBatch.Write(); err != nil { | 	if err := stateBatch.Write(); err != nil { | ||||||
| 		return nil, fmt.Errorf("cannot write state: %v", err) | 		return nil, fmt.Errorf("cannot write state: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if err := WriteTd(chainDb, block.Hash(), difficulty); err != nil { | 	if err := WriteTd(chainDb, block.Hash(), block.NumberU64(), difficulty); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if err := WriteBlock(chainDb, block); err != nil { | 	if err := WriteBlock(chainDb, block); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if err := WriteBlockReceipts(chainDb, block.Hash(), nil); err != nil { | 	if err := WriteBlockReceipts(chainDb, block.Hash(), block.NumberU64(), nil); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	if err := WriteCanonicalHash(chainDb, block.Hash(), block.NumberU64()); err != nil { | 	if err := WriteCanonicalHash(chainDb, block.Hash(), block.NumberU64()); err != nil { | ||||||
|  | |||||||
| @ -35,6 +35,12 @@ import ( | |||||||
| 	"github.com/hashicorp/golang-lru" | 	"github.com/hashicorp/golang-lru" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | const ( | ||||||
|  | 	headerCacheLimit = 512 | ||||||
|  | 	tdCacheLimit     = 1024 | ||||||
|  | 	numberCacheLimit = 2048 | ||||||
|  | ) | ||||||
|  | 
 | ||||||
| // HeaderChain implements the basic block header chain logic that is shared by
 | // HeaderChain implements the basic block header chain logic that is shared by
 | ||||||
| // core.BlockChain and light.LightChain. It is not usable in itself, only as
 | // core.BlockChain and light.LightChain. It is not usable in itself, only as
 | ||||||
| // a part of either structure.
 | // a part of either structure.
 | ||||||
| @ -51,6 +57,7 @@ type HeaderChain struct { | |||||||
| 
 | 
 | ||||||
| 	headerCache *lru.Cache // Cache for the most recent block headers
 | 	headerCache *lru.Cache // Cache for the most recent block headers
 | ||||||
| 	tdCache     *lru.Cache // Cache for the most recent block total difficulties
 | 	tdCache     *lru.Cache // Cache for the most recent block total difficulties
 | ||||||
|  | 	numberCache *lru.Cache // Cache for the most recent block numbers
 | ||||||
| 
 | 
 | ||||||
| 	procInterrupt func() bool | 	procInterrupt func() bool | ||||||
| 
 | 
 | ||||||
| @ -68,6 +75,7 @@ type getHeaderValidatorFn func() HeaderValidator | |||||||
| func NewHeaderChain(chainDb ethdb.Database, config *ChainConfig, getValidator getHeaderValidatorFn, procInterrupt func() bool) (*HeaderChain, error) { | func NewHeaderChain(chainDb ethdb.Database, config *ChainConfig, getValidator getHeaderValidatorFn, procInterrupt func() bool) (*HeaderChain, error) { | ||||||
| 	headerCache, _ := lru.New(headerCacheLimit) | 	headerCache, _ := lru.New(headerCacheLimit) | ||||||
| 	tdCache, _ := lru.New(tdCacheLimit) | 	tdCache, _ := lru.New(tdCacheLimit) | ||||||
|  | 	numberCache, _ := lru.New(numberCacheLimit) | ||||||
| 
 | 
 | ||||||
| 	// Seed a fast but crypto originating random generator
 | 	// Seed a fast but crypto originating random generator
 | ||||||
| 	seed, err := crand.Int(crand.Reader, big.NewInt(math.MaxInt64)) | 	seed, err := crand.Int(crand.Reader, big.NewInt(math.MaxInt64)) | ||||||
| @ -80,6 +88,7 @@ func NewHeaderChain(chainDb ethdb.Database, config *ChainConfig, getValidator ge | |||||||
| 		chainDb:       chainDb, | 		chainDb:       chainDb, | ||||||
| 		headerCache:   headerCache, | 		headerCache:   headerCache, | ||||||
| 		tdCache:       tdCache, | 		tdCache:       tdCache, | ||||||
|  | 		numberCache:   numberCache, | ||||||
| 		procInterrupt: procInterrupt, | 		procInterrupt: procInterrupt, | ||||||
| 		rand:          mrand.New(mrand.NewSource(seed.Int64())), | 		rand:          mrand.New(mrand.NewSource(seed.Int64())), | ||||||
| 		getValidator:  getValidator, | 		getValidator:  getValidator, | ||||||
| @ -97,7 +106,7 @@ func NewHeaderChain(chainDb ethdb.Database, config *ChainConfig, getValidator ge | |||||||
| 
 | 
 | ||||||
| 	hc.currentHeader = hc.genesisHeader | 	hc.currentHeader = hc.genesisHeader | ||||||
| 	if head := GetHeadBlockHash(chainDb); head != (common.Hash{}) { | 	if head := GetHeadBlockHash(chainDb); head != (common.Hash{}) { | ||||||
| 		if chead := hc.GetHeader(head); chead != nil { | 		if chead := hc.GetHeaderByHash(head); chead != nil { | ||||||
| 			hc.currentHeader = chead | 			hc.currentHeader = chead | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -106,6 +115,19 @@ func NewHeaderChain(chainDb ethdb.Database, config *ChainConfig, getValidator ge | |||||||
| 	return hc, nil | 	return hc, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // GetBlockNumber retrieves the block number belonging to the given hash
 | ||||||
|  | // from the cache or database
 | ||||||
|  | func (hc *HeaderChain) GetBlockNumber(hash common.Hash) uint64 { | ||||||
|  | 	if cached, ok := hc.numberCache.Get(hash); ok { | ||||||
|  | 		return cached.(uint64) | ||||||
|  | 	} | ||||||
|  | 	number := GetBlockNumber(hc.chainDb, hash) | ||||||
|  | 	if number != missingNumber { | ||||||
|  | 		hc.numberCache.Add(hash, number) | ||||||
|  | 	} | ||||||
|  | 	return number | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // WriteHeader writes a header into the local chain, given that its parent is
 | // WriteHeader writes a header into the local chain, given that its parent is
 | ||||||
| // already known. If the total difficulty of the newly inserted header becomes
 | // already known. If the total difficulty of the newly inserted header becomes
 | ||||||
| // greater than the current known TD, the canonical chain is re-routed.
 | // greater than the current known TD, the canonical chain is re-routed.
 | ||||||
| @ -122,11 +144,11 @@ func (hc *HeaderChain) WriteHeader(header *types.Header) (status WriteStatus, er | |||||||
| 		number = header.Number.Uint64() | 		number = header.Number.Uint64() | ||||||
| 	) | 	) | ||||||
| 	// Calculate the total difficulty of the header
 | 	// Calculate the total difficulty of the header
 | ||||||
| 	ptd := hc.GetTd(header.ParentHash) | 	ptd := hc.GetTd(header.ParentHash, number-1) | ||||||
| 	if ptd == nil { | 	if ptd == nil { | ||||||
| 		return NonStatTy, ParentError(header.ParentHash) | 		return NonStatTy, ParentError(header.ParentHash) | ||||||
| 	} | 	} | ||||||
| 	localTd := hc.GetTd(hc.currentHeaderHash) | 	localTd := hc.GetTd(hc.currentHeaderHash, hc.currentHeader.Number.Uint64()) | ||||||
| 	externTd := new(big.Int).Add(header.Difficulty, ptd) | 	externTd := new(big.Int).Add(header.Difficulty, ptd) | ||||||
| 
 | 
 | ||||||
| 	// If the total difficulty is higher than our known, add it to the canonical chain
 | 	// If the total difficulty is higher than our known, add it to the canonical chain
 | ||||||
| @ -134,21 +156,25 @@ func (hc *HeaderChain) WriteHeader(header *types.Header) (status WriteStatus, er | |||||||
| 	// Please refer to http://www.cs.cornell.edu/~ie53/publications/btcProcFC.pdf
 | 	// Please refer to http://www.cs.cornell.edu/~ie53/publications/btcProcFC.pdf
 | ||||||
| 	if externTd.Cmp(localTd) > 0 || (externTd.Cmp(localTd) == 0 && mrand.Float64() < 0.5) { | 	if externTd.Cmp(localTd) > 0 || (externTd.Cmp(localTd) == 0 && mrand.Float64() < 0.5) { | ||||||
| 		// Delete any canonical number assignments above the new head
 | 		// Delete any canonical number assignments above the new head
 | ||||||
| 		for i := number + 1; GetCanonicalHash(hc.chainDb, i) != (common.Hash{}); i++ { | 		for i := number + 1; ; i++ { | ||||||
|  | 			hash := GetCanonicalHash(hc.chainDb, i) | ||||||
|  | 			if hash == (common.Hash{}) { | ||||||
|  | 				break | ||||||
|  | 			} | ||||||
| 			DeleteCanonicalHash(hc.chainDb, i) | 			DeleteCanonicalHash(hc.chainDb, i) | ||||||
| 		} | 		} | ||||||
| 		// Overwrite any stale canonical number assignments
 | 		// Overwrite any stale canonical number assignments
 | ||||||
| 		var ( | 		var ( | ||||||
| 			headHash   = header.ParentHash | 			headHash   = header.ParentHash | ||||||
| 			headHeader = hc.GetHeader(headHash) | 			headNumber = header.Number.Uint64() - 1 | ||||||
| 			headNumber = headHeader.Number.Uint64() | 			headHeader = hc.GetHeader(headHash, headNumber) | ||||||
| 		) | 		) | ||||||
| 		for GetCanonicalHash(hc.chainDb, headNumber) != headHash { | 		for GetCanonicalHash(hc.chainDb, headNumber) != headHash { | ||||||
| 			WriteCanonicalHash(hc.chainDb, headHash, headNumber) | 			WriteCanonicalHash(hc.chainDb, headHash, headNumber) | ||||||
| 
 | 
 | ||||||
| 			headHash = headHeader.ParentHash | 			headHash = headHeader.ParentHash | ||||||
| 			headHeader = hc.GetHeader(headHash) | 			headNumber = headHeader.Number.Uint64() - 1 | ||||||
| 			headNumber = headHeader.Number.Uint64() | 			headHeader = hc.GetHeader(headHash, headNumber) | ||||||
| 		} | 		} | ||||||
| 		// Extend the canonical chain with the new header
 | 		// Extend the canonical chain with the new header
 | ||||||
| 		if err := WriteCanonicalHash(hc.chainDb, hash, number); err != nil { | 		if err := WriteCanonicalHash(hc.chainDb, hash, number); err != nil { | ||||||
| @ -164,13 +190,14 @@ func (hc *HeaderChain) WriteHeader(header *types.Header) (status WriteStatus, er | |||||||
| 		status = SideStatTy | 		status = SideStatTy | ||||||
| 	} | 	} | ||||||
| 	// Irrelevant of the canonical status, write the header itself to the database
 | 	// Irrelevant of the canonical status, write the header itself to the database
 | ||||||
| 	if err := hc.WriteTd(hash, externTd); err != nil { | 	if err := hc.WriteTd(hash, number, externTd); err != nil { | ||||||
| 		glog.Fatalf("failed to write header total difficulty: %v", err) | 		glog.Fatalf("failed to write header total difficulty: %v", err) | ||||||
| 	} | 	} | ||||||
| 	if err := WriteHeader(hc.chainDb, header); err != nil { | 	if err := WriteHeader(hc.chainDb, header); err != nil { | ||||||
| 		glog.Fatalf("failed to write header contents: %v", err) | 		glog.Fatalf("failed to write header contents: %v", err) | ||||||
| 	} | 	} | ||||||
| 	hc.headerCache.Add(hash, header) | 	hc.headerCache.Add(hash, header) | ||||||
|  | 	hc.numberCache.Add(hash, number) | ||||||
| 
 | 
 | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
| @ -239,7 +266,7 @@ func (hc *HeaderChain) InsertHeaderChain(chain []*types.Header, checkFreq int, w | |||||||
| 
 | 
 | ||||||
| 			var err error | 			var err error | ||||||
| 			if index == 0 { | 			if index == 0 { | ||||||
| 				err = hc.getValidator().ValidateHeader(header, hc.GetHeader(header.ParentHash), checkPow) | 				err = hc.getValidator().ValidateHeader(header, hc.GetHeader(header.ParentHash, header.Number.Uint64()-1), checkPow) | ||||||
| 			} else { | 			} else { | ||||||
| 				err = hc.getValidator().ValidateHeader(header, chain[index-1], checkPow) | 				err = hc.getValidator().ValidateHeader(header, chain[index-1], checkPow) | ||||||
| 			} | 			} | ||||||
| @ -300,7 +327,7 @@ func (hc *HeaderChain) InsertHeaderChain(chain []*types.Header, checkFreq int, w | |||||||
| // hash, fetching towards the genesis block.
 | // hash, fetching towards the genesis block.
 | ||||||
| func (hc *HeaderChain) GetBlockHashesFromHash(hash common.Hash, max uint64) []common.Hash { | func (hc *HeaderChain) GetBlockHashesFromHash(hash common.Hash, max uint64) []common.Hash { | ||||||
| 	// Get the origin header from which to fetch
 | 	// Get the origin header from which to fetch
 | ||||||
| 	header := hc.GetHeader(hash) | 	header := hc.GetHeaderByHash(hash) | ||||||
| 	if header == nil { | 	if header == nil { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| @ -308,7 +335,7 @@ func (hc *HeaderChain) GetBlockHashesFromHash(hash common.Hash, max uint64) []co | |||||||
| 	chain := make([]common.Hash, 0, max) | 	chain := make([]common.Hash, 0, max) | ||||||
| 	for i := uint64(0); i < max; i++ { | 	for i := uint64(0); i < max; i++ { | ||||||
| 		next := header.ParentHash | 		next := header.ParentHash | ||||||
| 		if header = hc.GetHeader(next); header == nil { | 		if header = hc.GetHeader(next, header.Number.Uint64()-1); header == nil { | ||||||
| 			break | 			break | ||||||
| 		} | 		} | ||||||
| 		chain = append(chain, next) | 		chain = append(chain, next) | ||||||
| @ -320,13 +347,13 @@ func (hc *HeaderChain) GetBlockHashesFromHash(hash common.Hash, max uint64) []co | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetTd retrieves a block's total difficulty in the canonical chain from the
 | // GetTd retrieves a block's total difficulty in the canonical chain from the
 | ||||||
| // database by hash, caching it if found.
 | // database by hash and number, caching it if found.
 | ||||||
| func (hc *HeaderChain) GetTd(hash common.Hash) *big.Int { | func (hc *HeaderChain) GetTd(hash common.Hash, number uint64) *big.Int { | ||||||
| 	// Short circuit if the td's already in the cache, retrieve otherwise
 | 	// Short circuit if the td's already in the cache, retrieve otherwise
 | ||||||
| 	if cached, ok := hc.tdCache.Get(hash); ok { | 	if cached, ok := hc.tdCache.Get(hash); ok { | ||||||
| 		return cached.(*big.Int) | 		return cached.(*big.Int) | ||||||
| 	} | 	} | ||||||
| 	td := GetTd(hc.chainDb, hash) | 	td := GetTd(hc.chainDb, hash, number) | ||||||
| 	if td == nil { | 	if td == nil { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| @ -335,24 +362,30 @@ func (hc *HeaderChain) GetTd(hash common.Hash) *big.Int { | |||||||
| 	return td | 	return td | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // GetTdByHash retrieves a block's total difficulty in the canonical chain from the
 | ||||||
|  | // database by hash, caching it if found.
 | ||||||
|  | func (hc *HeaderChain) GetTdByHash(hash common.Hash) *big.Int { | ||||||
|  | 	return hc.GetTd(hash, hc.GetBlockNumber(hash)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // WriteTd stores a block's total difficulty into the database, also caching it
 | // WriteTd stores a block's total difficulty into the database, also caching it
 | ||||||
| // along the way.
 | // along the way.
 | ||||||
| func (hc *HeaderChain) WriteTd(hash common.Hash, td *big.Int) error { | func (hc *HeaderChain) WriteTd(hash common.Hash, number uint64, td *big.Int) error { | ||||||
| 	if err := WriteTd(hc.chainDb, hash, td); err != nil { | 	if err := WriteTd(hc.chainDb, hash, number, td); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	hc.tdCache.Add(hash, new(big.Int).Set(td)) | 	hc.tdCache.Add(hash, new(big.Int).Set(td)) | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetHeader retrieves a block header from the database by hash, caching it if
 | // GetHeader retrieves a block header from the database by hash and number,
 | ||||||
| // found.
 | // caching it if found.
 | ||||||
| func (hc *HeaderChain) GetHeader(hash common.Hash) *types.Header { | func (hc *HeaderChain) GetHeader(hash common.Hash, number uint64) *types.Header { | ||||||
| 	// Short circuit if the header's already in the cache, retrieve otherwise
 | 	// Short circuit if the header's already in the cache, retrieve otherwise
 | ||||||
| 	if header, ok := hc.headerCache.Get(hash); ok { | 	if header, ok := hc.headerCache.Get(hash); ok { | ||||||
| 		return header.(*types.Header) | 		return header.(*types.Header) | ||||||
| 	} | 	} | ||||||
| 	header := GetHeader(hc.chainDb, hash) | 	header := GetHeader(hc.chainDb, hash, number) | ||||||
| 	if header == nil { | 	if header == nil { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| @ -361,10 +394,16 @@ func (hc *HeaderChain) GetHeader(hash common.Hash) *types.Header { | |||||||
| 	return header | 	return header | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // GetHeaderByHash retrieves a block header from the database by hash, caching it if
 | ||||||
|  | // found.
 | ||||||
|  | func (hc *HeaderChain) GetHeaderByHash(hash common.Hash) *types.Header { | ||||||
|  | 	return hc.GetHeader(hash, hc.GetBlockNumber(hash)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // HasHeader checks if a block header is present in the database or not, caching
 | // HasHeader checks if a block header is present in the database or not, caching
 | ||||||
| // it if present.
 | // it if present.
 | ||||||
| func (hc *HeaderChain) HasHeader(hash common.Hash) bool { | func (hc *HeaderChain) HasHeader(hash common.Hash) bool { | ||||||
| 	return hc.GetHeader(hash) != nil | 	return hc.GetHeaderByHash(hash) != nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetHeaderByNumber retrieves a block header from the database by number,
 | // GetHeaderByNumber retrieves a block header from the database by number,
 | ||||||
| @ -374,7 +413,7 @@ func (hc *HeaderChain) GetHeaderByNumber(number uint64) *types.Header { | |||||||
| 	if hash == (common.Hash{}) { | 	if hash == (common.Hash{}) { | ||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
| 	return hc.GetHeader(hash) | 	return hc.GetHeader(hash, number) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // CurrentHeader retrieves the current head header of the canonical chain. The
 | // CurrentHeader retrieves the current head header of the canonical chain. The
 | ||||||
| @ -394,7 +433,7 @@ func (hc *HeaderChain) SetCurrentHeader(head *types.Header) { | |||||||
| 
 | 
 | ||||||
| // DeleteCallback is a callback function that is called by SetHead before
 | // DeleteCallback is a callback function that is called by SetHead before
 | ||||||
| // each header is deleted.
 | // each header is deleted.
 | ||||||
| type DeleteCallback func(common.Hash) | type DeleteCallback func(common.Hash, uint64) | ||||||
| 
 | 
 | ||||||
| // SetHead rewinds the local chain to a new head. Everything above the new head
 | // SetHead rewinds the local chain to a new head. Everything above the new head
 | ||||||
| // will be deleted and the new one set.
 | // will be deleted and the new one set.
 | ||||||
| @ -406,12 +445,13 @@ func (hc *HeaderChain) SetHead(head uint64, delFn DeleteCallback) { | |||||||
| 
 | 
 | ||||||
| 	for hc.currentHeader != nil && hc.currentHeader.Number.Uint64() > head { | 	for hc.currentHeader != nil && hc.currentHeader.Number.Uint64() > head { | ||||||
| 		hash := hc.currentHeader.Hash() | 		hash := hc.currentHeader.Hash() | ||||||
|  | 		num := hc.currentHeader.Number.Uint64() | ||||||
| 		if delFn != nil { | 		if delFn != nil { | ||||||
| 			delFn(hash) | 			delFn(hash, num) | ||||||
| 		} | 		} | ||||||
| 		DeleteHeader(hc.chainDb, hash) | 		DeleteHeader(hc.chainDb, hash, num) | ||||||
| 		DeleteTd(hc.chainDb, hash) | 		DeleteTd(hc.chainDb, hash, num) | ||||||
| 		hc.currentHeader = hc.GetHeader(hc.currentHeader.ParentHash) | 		hc.currentHeader = hc.GetHeader(hc.currentHeader.ParentHash, hc.currentHeader.Number.Uint64()-1) | ||||||
| 	} | 	} | ||||||
| 	// Roll back the canonical chain numbering
 | 	// Roll back the canonical chain numbering
 | ||||||
| 	for i := height; i > head; i-- { | 	for i := height; i > head; i-- { | ||||||
| @ -420,6 +460,7 @@ func (hc *HeaderChain) SetHead(head uint64, delFn DeleteCallback) { | |||||||
| 	// Clear out any stale content from the caches
 | 	// Clear out any stale content from the caches
 | ||||||
| 	hc.headerCache.Purge() | 	hc.headerCache.Purge() | ||||||
| 	hc.tdCache.Purge() | 	hc.tdCache.Purge() | ||||||
|  | 	hc.numberCache.Purge() | ||||||
| 
 | 
 | ||||||
| 	if hc.currentHeader == nil { | 	if hc.currentHeader == nil { | ||||||
| 		hc.currentHeader = hc.genesisHeader | 		hc.currentHeader = hc.genesisHeader | ||||||
|  | |||||||
| @ -30,7 +30,7 @@ import ( | |||||||
| // to query for information.
 | // to query for information.
 | ||||||
| func GetHashFn(ref common.Hash, chain *BlockChain) func(n uint64) common.Hash { | func GetHashFn(ref common.Hash, chain *BlockChain) func(n uint64) common.Hash { | ||||||
| 	return func(n uint64) common.Hash { | 	return func(n uint64) common.Hash { | ||||||
| 		for block := chain.GetBlock(ref); block != nil; block = chain.GetBlock(block.ParentHash()) { | 		for block := chain.GetBlockByHash(ref); block != nil; block = chain.GetBlock(block.ParentHash(), block.NumberU64()-1) { | ||||||
| 			if block.NumberU64() == n { | 			if block.NumberU64() == n { | ||||||
| 				return block.Hash() | 				return block.Hash() | ||||||
| 			} | 			} | ||||||
|  | |||||||
							
								
								
									
										26
									
								
								eth/api.go
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								eth/api.go
									
									
									
									
									
								
							| @ -594,7 +594,7 @@ func (s *PublicBlockChainAPI) GetBlockByNumber(blockNr rpc.BlockNumber, fullTx b | |||||||
| // GetBlockByHash returns the requested block. When fullTx is true all transactions in the block are returned in full
 | // GetBlockByHash returns the requested block. When fullTx is true all transactions in the block are returned in full
 | ||||||
| // detail, otherwise only the transaction hash is returned.
 | // detail, otherwise only the transaction hash is returned.
 | ||||||
| func (s *PublicBlockChainAPI) GetBlockByHash(blockHash common.Hash, fullTx bool) (map[string]interface{}, error) { | func (s *PublicBlockChainAPI) GetBlockByHash(blockHash common.Hash, fullTx bool) (map[string]interface{}, error) { | ||||||
| 	if block := s.bc.GetBlock(blockHash); block != nil { | 	if block := s.bc.GetBlockByHash(blockHash); block != nil { | ||||||
| 		return s.rpcOutputBlock(block, true, fullTx) | 		return s.rpcOutputBlock(block, true, fullTx) | ||||||
| 	} | 	} | ||||||
| 	return nil, nil | 	return nil, nil | ||||||
| @ -618,7 +618,7 @@ func (s *PublicBlockChainAPI) GetUncleByBlockNumberAndIndex(blockNr rpc.BlockNum | |||||||
| // GetUncleByBlockHashAndIndex returns the uncle block for the given block hash and index. When fullTx is true
 | // GetUncleByBlockHashAndIndex returns the uncle block for the given block hash and index. When fullTx is true
 | ||||||
| // all transactions in the block are returned in full detail, otherwise only the transaction hash is returned.
 | // all transactions in the block are returned in full detail, otherwise only the transaction hash is returned.
 | ||||||
| func (s *PublicBlockChainAPI) GetUncleByBlockHashAndIndex(blockHash common.Hash, index rpc.HexNumber) (map[string]interface{}, error) { | func (s *PublicBlockChainAPI) GetUncleByBlockHashAndIndex(blockHash common.Hash, index rpc.HexNumber) (map[string]interface{}, error) { | ||||||
| 	if block := s.bc.GetBlock(blockHash); block != nil { | 	if block := s.bc.GetBlockByHash(blockHash); block != nil { | ||||||
| 		uncles := block.Uncles() | 		uncles := block.Uncles() | ||||||
| 		if index.Int() < 0 || index.Int() >= len(uncles) { | 		if index.Int() < 0 || index.Int() >= len(uncles) { | ||||||
| 			glog.V(logger.Debug).Infof("uncle block on index %d not found for block %s", index.Int(), blockHash.Hex()) | 			glog.V(logger.Debug).Infof("uncle block on index %d not found for block %s", index.Int(), blockHash.Hex()) | ||||||
| @ -640,7 +640,7 @@ func (s *PublicBlockChainAPI) GetUncleCountByBlockNumber(blockNr rpc.BlockNumber | |||||||
| 
 | 
 | ||||||
| // GetUncleCountByBlockHash returns number of uncles in the block for the given block hash
 | // GetUncleCountByBlockHash returns number of uncles in the block for the given block hash
 | ||||||
| func (s *PublicBlockChainAPI) GetUncleCountByBlockHash(blockHash common.Hash) *rpc.HexNumber { | func (s *PublicBlockChainAPI) GetUncleCountByBlockHash(blockHash common.Hash) *rpc.HexNumber { | ||||||
| 	if block := s.bc.GetBlock(blockHash); block != nil { | 	if block := s.bc.GetBlockByHash(blockHash); block != nil { | ||||||
| 		return rpc.NewHexNumber(len(block.Uncles())) | 		return rpc.NewHexNumber(len(block.Uncles())) | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| @ -814,7 +814,7 @@ func (s *PublicBlockChainAPI) rpcOutputBlock(b *types.Block, inclTx bool, fullTx | |||||||
| 		"stateRoot":        b.Root(), | 		"stateRoot":        b.Root(), | ||||||
| 		"miner":            b.Coinbase(), | 		"miner":            b.Coinbase(), | ||||||
| 		"difficulty":       rpc.NewHexNumber(b.Difficulty()), | 		"difficulty":       rpc.NewHexNumber(b.Difficulty()), | ||||||
| 		"totalDifficulty":  rpc.NewHexNumber(s.bc.GetTd(b.Hash())), | 		"totalDifficulty":  rpc.NewHexNumber(s.bc.GetTd(b.Hash(), b.NumberU64())), | ||||||
| 		"extraData":        fmt.Sprintf("0x%x", b.Extra()), | 		"extraData":        fmt.Sprintf("0x%x", b.Extra()), | ||||||
| 		"size":             rpc.NewHexNumber(b.Size().Int64()), | 		"size":             rpc.NewHexNumber(b.Size().Int64()), | ||||||
| 		"gasLimit":         rpc.NewHexNumber(b.GasLimit()), | 		"gasLimit":         rpc.NewHexNumber(b.GasLimit()), | ||||||
| @ -1004,7 +1004,7 @@ func (s *PublicTransactionPoolAPI) GetBlockTransactionCountByNumber(blockNr rpc. | |||||||
| 
 | 
 | ||||||
| // GetBlockTransactionCountByHash returns the number of transactions in the block with the given hash.
 | // GetBlockTransactionCountByHash returns the number of transactions in the block with the given hash.
 | ||||||
| func (s *PublicTransactionPoolAPI) GetBlockTransactionCountByHash(blockHash common.Hash) *rpc.HexNumber { | func (s *PublicTransactionPoolAPI) GetBlockTransactionCountByHash(blockHash common.Hash) *rpc.HexNumber { | ||||||
| 	if block := s.bc.GetBlock(blockHash); block != nil { | 	if block := s.bc.GetBlockByHash(blockHash); block != nil { | ||||||
| 		return rpc.NewHexNumber(len(block.Transactions())) | 		return rpc.NewHexNumber(len(block.Transactions())) | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| @ -1020,7 +1020,7 @@ func (s *PublicTransactionPoolAPI) GetTransactionByBlockNumberAndIndex(blockNr r | |||||||
| 
 | 
 | ||||||
| // GetTransactionByBlockHashAndIndex returns the transaction for the given block hash and index.
 | // GetTransactionByBlockHashAndIndex returns the transaction for the given block hash and index.
 | ||||||
| func (s *PublicTransactionPoolAPI) GetTransactionByBlockHashAndIndex(blockHash common.Hash, index rpc.HexNumber) (*RPCTransaction, error) { | func (s *PublicTransactionPoolAPI) GetTransactionByBlockHashAndIndex(blockHash common.Hash, index rpc.HexNumber) (*RPCTransaction, error) { | ||||||
| 	if block := s.bc.GetBlock(blockHash); block != nil { | 	if block := s.bc.GetBlockByHash(blockHash); block != nil { | ||||||
| 		return newRPCTransactionFromBlockIndex(block, index.Int()) | 		return newRPCTransactionFromBlockIndex(block, index.Int()) | ||||||
| 	} | 	} | ||||||
| 	return nil, nil | 	return nil, nil | ||||||
| @ -1080,7 +1080,7 @@ func (s *PublicTransactionPoolAPI) GetTransactionByHash(txHash common.Hash) (*RP | |||||||
| 		return nil, nil | 		return nil, nil | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if block := s.bc.GetBlock(blockHash); block != nil { | 	if block := s.bc.GetBlockByHash(blockHash); block != nil { | ||||||
| 		return newRPCTransaction(block, txHash) | 		return newRPCTransaction(block, txHash) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -1705,7 +1705,7 @@ func (api *PrivateDebugAPI) TraceBlockByNumber(number uint64, config *vm.Config) | |||||||
| // TraceBlockByHash processes the block by hash.
 | // TraceBlockByHash processes the block by hash.
 | ||||||
| func (api *PrivateDebugAPI) TraceBlockByHash(hash common.Hash, config *vm.Config) BlockTraceResult { | func (api *PrivateDebugAPI) TraceBlockByHash(hash common.Hash, config *vm.Config) BlockTraceResult { | ||||||
| 	// Fetch the block that we aim to reprocess
 | 	// Fetch the block that we aim to reprocess
 | ||||||
| 	block := api.eth.BlockChain().GetBlock(hash) | 	block := api.eth.BlockChain().GetBlockByHash(hash) | ||||||
| 	if block == nil { | 	if block == nil { | ||||||
| 		return BlockTraceResult{Error: fmt.Sprintf("block #%x not found", hash)} | 		return BlockTraceResult{Error: fmt.Sprintf("block #%x not found", hash)} | ||||||
| 	} | 	} | ||||||
| @ -1745,10 +1745,10 @@ func (api *PrivateDebugAPI) traceBlock(block *types.Block, config *vm.Config) (b | |||||||
| 	config.Debug = true // make sure debug is set.
 | 	config.Debug = true // make sure debug is set.
 | ||||||
| 	config.Logger.Collector = collector | 	config.Logger.Collector = collector | ||||||
| 
 | 
 | ||||||
| 	if err := core.ValidateHeader(api.config, blockchain.AuxValidator(), block.Header(), blockchain.GetHeader(block.ParentHash()), true, false); err != nil { | 	if err := core.ValidateHeader(api.config, blockchain.AuxValidator(), block.Header(), blockchain.GetHeader(block.ParentHash(), block.NumberU64()-1), true, false); err != nil { | ||||||
| 		return false, collector.traces, err | 		return false, collector.traces, err | ||||||
| 	} | 	} | ||||||
| 	statedb, err := state.New(blockchain.GetBlock(block.ParentHash()).Root(), api.eth.ChainDb()) | 	statedb, err := state.New(blockchain.GetBlock(block.ParentHash(), block.NumberU64()-1).Root(), api.eth.ChainDb()) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return false, collector.traces, err | 		return false, collector.traces, err | ||||||
| 	} | 	} | ||||||
| @ -1757,7 +1757,7 @@ func (api *PrivateDebugAPI) traceBlock(block *types.Block, config *vm.Config) (b | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return false, collector.traces, err | 		return false, collector.traces, err | ||||||
| 	} | 	} | ||||||
| 	if err := validator.ValidateState(block, blockchain.GetBlock(block.ParentHash()), statedb, receipts, usedGas); err != nil { | 	if err := validator.ValidateState(block, blockchain.GetBlock(block.ParentHash(), block.NumberU64()-1), statedb, receipts, usedGas); err != nil { | ||||||
| 		return false, collector.traces, err | 		return false, collector.traces, err | ||||||
| 	} | 	} | ||||||
| 	return true, collector.traces, nil | 	return true, collector.traces, nil | ||||||
| @ -1841,12 +1841,12 @@ func (api *PrivateDebugAPI) TraceTransaction(txHash common.Hash, logger *vm.LogC | |||||||
| 	if tx == nil { | 	if tx == nil { | ||||||
| 		return nil, fmt.Errorf("transaction %x not found", txHash) | 		return nil, fmt.Errorf("transaction %x not found", txHash) | ||||||
| 	} | 	} | ||||||
| 	block := api.eth.BlockChain().GetBlock(blockHash) | 	block := api.eth.BlockChain().GetBlockByHash(blockHash) | ||||||
| 	if block == nil { | 	if block == nil { | ||||||
| 		return nil, fmt.Errorf("block %x not found", blockHash) | 		return nil, fmt.Errorf("block %x not found", blockHash) | ||||||
| 	} | 	} | ||||||
| 	// Create the state database to mutate and eventually trace
 | 	// Create the state database to mutate and eventually trace
 | ||||||
| 	parent := api.eth.BlockChain().GetBlock(block.ParentHash()) | 	parent := api.eth.BlockChain().GetBlock(block.ParentHash(), block.NumberU64()-1) | ||||||
| 	if parent == nil { | 	if parent == nil { | ||||||
| 		return nil, fmt.Errorf("block parent %x not found", block.ParentHash()) | 		return nil, fmt.Errorf("block parent %x not found", block.ParentHash()) | ||||||
| 	} | 	} | ||||||
|  | |||||||
							
								
								
									
										118
									
								
								eth/backend.go
									
									
									
									
									
								
							
							
						
						
									
										118
									
								
								eth/backend.go
									
									
									
									
									
								
							| @ -18,7 +18,6 @@ | |||||||
| package eth | package eth | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"bytes" |  | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"math/big" | 	"math/big" | ||||||
| @ -47,7 +46,6 @@ import ( | |||||||
| 	"github.com/ethereum/go-ethereum/miner" | 	"github.com/ethereum/go-ethereum/miner" | ||||||
| 	"github.com/ethereum/go-ethereum/node" | 	"github.com/ethereum/go-ethereum/node" | ||||||
| 	"github.com/ethereum/go-ethereum/p2p" | 	"github.com/ethereum/go-ethereum/p2p" | ||||||
| 	"github.com/ethereum/go-ethereum/rlp" |  | ||||||
| 	"github.com/ethereum/go-ethereum/rpc" | 	"github.com/ethereum/go-ethereum/rpc" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -104,9 +102,9 @@ type Config struct { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type Ethereum struct { | type Ethereum struct { | ||||||
| 	chainConfig *core.ChainConfig | 	chainConfig   *core.ChainConfig | ||||||
| 	// Channel for shutting down the ethereum
 | 	shutdownChan  chan bool // Channel for shutting down the ethereum
 | ||||||
| 	shutdownChan chan bool | 	stopDbUpgrade func()    // stop chain db sequential key upgrade
 | ||||||
| 
 | 
 | ||||||
| 	// DB interfaces
 | 	// DB interfaces
 | ||||||
| 	chainDb ethdb.Database // Block chain database
 | 	chainDb ethdb.Database // Block chain database
 | ||||||
| @ -161,6 +159,7 @@ func New(ctx *node.ServiceContext, config *Config) (*Ethereum, error) { | |||||||
| 	if err := addMipmapBloomBins(chainDb); err != nil { | 	if err := addMipmapBloomBins(chainDb); err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 	stopDbUpgrade := upgradeSequentialKeys(chainDb) | ||||||
| 
 | 
 | ||||||
| 	dappDb, err := ctx.OpenDatabase("dapp", config.DatabaseCache, config.DatabaseHandles) | 	dappDb, err := ctx.OpenDatabase("dapp", config.DatabaseCache, config.DatabaseHandles) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -185,7 +184,7 @@ func New(ctx *node.ServiceContext, config *Config) (*Ethereum, error) { | |||||||
| 		chainDb = config.TestGenesisState | 		chainDb = config.TestGenesisState | ||||||
| 	} | 	} | ||||||
| 	if config.TestGenesisBlock != nil { | 	if config.TestGenesisBlock != nil { | ||||||
| 		core.WriteTd(chainDb, config.TestGenesisBlock.Hash(), config.TestGenesisBlock.Difficulty()) | 		core.WriteTd(chainDb, config.TestGenesisBlock.Hash(), config.TestGenesisBlock.NumberU64(), config.TestGenesisBlock.Difficulty()) | ||||||
| 		core.WriteBlock(chainDb, config.TestGenesisBlock) | 		core.WriteBlock(chainDb, config.TestGenesisBlock) | ||||||
| 		core.WriteCanonicalHash(chainDb, config.TestGenesisBlock.Hash(), config.TestGenesisBlock.NumberU64()) | 		core.WriteCanonicalHash(chainDb, config.TestGenesisBlock.Hash(), config.TestGenesisBlock.NumberU64()) | ||||||
| 		core.WriteHeadBlockHash(chainDb, config.TestGenesisBlock.Hash()) | 		core.WriteHeadBlockHash(chainDb, config.TestGenesisBlock.Hash()) | ||||||
| @ -202,6 +201,7 @@ func New(ctx *node.ServiceContext, config *Config) (*Ethereum, error) { | |||||||
| 
 | 
 | ||||||
| 	eth := &Ethereum{ | 	eth := &Ethereum{ | ||||||
| 		shutdownChan:            make(chan bool), | 		shutdownChan:            make(chan bool), | ||||||
|  | 		stopDbUpgrade:           stopDbUpgrade, | ||||||
| 		chainDb:                 chainDb, | 		chainDb:                 chainDb, | ||||||
| 		dappDb:                  dappDb, | 		dappDb:                  dappDb, | ||||||
| 		eventMux:                ctx.EventMux, | 		eventMux:                ctx.EventMux, | ||||||
| @ -238,7 +238,7 @@ func New(ctx *node.ServiceContext, config *Config) (*Ethereum, error) { | |||||||
| 
 | 
 | ||||||
| 	// load the genesis block or write a new one if no genesis
 | 	// load the genesis block or write a new one if no genesis
 | ||||||
| 	// block is prenent in the database.
 | 	// block is prenent in the database.
 | ||||||
| 	genesis := core.GetBlock(chainDb, core.GetCanonicalHash(chainDb, 0)) | 	genesis := core.GetBlock(chainDb, core.GetCanonicalHash(chainDb, 0), 0) | ||||||
| 	if genesis == nil { | 	if genesis == nil { | ||||||
| 		genesis, err = core.WriteDefaultGenesisBlock(chainDb) | 		genesis, err = core.WriteDefaultGenesisBlock(chainDb) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @ -415,6 +415,9 @@ func (s *Ethereum) Start(srvr *p2p.Server) error { | |||||||
| // Stop implements node.Service, terminating all internal goroutines used by the
 | // Stop implements node.Service, terminating all internal goroutines used by the
 | ||||||
| // Ethereum protocol.
 | // Ethereum protocol.
 | ||||||
| func (s *Ethereum) Stop() error { | func (s *Ethereum) Stop() error { | ||||||
|  | 	if s.stopDbUpgrade != nil { | ||||||
|  | 		s.stopDbUpgrade() | ||||||
|  | 	} | ||||||
| 	s.blockchain.Stop() | 	s.blockchain.Stop() | ||||||
| 	s.protocolManager.Stop() | 	s.protocolManager.Stop() | ||||||
| 	s.txPool.Stop() | 	s.txPool.Stop() | ||||||
| @ -526,104 +529,3 @@ func dagFiles(epoch uint64) (string, string) { | |||||||
| 	dag := fmt.Sprintf("full-R%d-%x", ethashRevision, seedHash[:8]) | 	dag := fmt.Sprintf("full-R%d-%x", ethashRevision, seedHash[:8]) | ||||||
| 	return dag, "full-R" + dag | 	return dag, "full-R" + dag | ||||||
| } | } | ||||||
| 
 |  | ||||||
| // upgradeChainDatabase ensures that the chain database stores block split into
 |  | ||||||
| // separate header and body entries.
 |  | ||||||
| func upgradeChainDatabase(db ethdb.Database) error { |  | ||||||
| 	// Short circuit if the head block is stored already as separate header and body
 |  | ||||||
| 	data, err := db.Get([]byte("LastBlock")) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	head := common.BytesToHash(data) |  | ||||||
| 
 |  | ||||||
| 	if block := core.GetBlockByHashOld(db, head); block == nil { |  | ||||||
| 		return nil |  | ||||||
| 	} |  | ||||||
| 	// At least some of the database is still the old format, upgrade (skip the head block!)
 |  | ||||||
| 	glog.V(logger.Info).Info("Old database detected, upgrading...") |  | ||||||
| 
 |  | ||||||
| 	if db, ok := db.(*ethdb.LDBDatabase); ok { |  | ||||||
| 		blockPrefix := []byte("block-hash-") |  | ||||||
| 		for it := db.NewIterator(); it.Next(); { |  | ||||||
| 			// Skip anything other than a combined block
 |  | ||||||
| 			if !bytes.HasPrefix(it.Key(), blockPrefix) { |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 			// Skip the head block (merge last to signal upgrade completion)
 |  | ||||||
| 			if bytes.HasSuffix(it.Key(), head.Bytes()) { |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 			// Load the block, split and serialize (order!)
 |  | ||||||
| 			block := core.GetBlockByHashOld(db, common.BytesToHash(bytes.TrimPrefix(it.Key(), blockPrefix))) |  | ||||||
| 
 |  | ||||||
| 			if err := core.WriteTd(db, block.Hash(), block.DeprecatedTd()); err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 			if err := core.WriteBody(db, block.Hash(), block.Body()); err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 			if err := core.WriteHeader(db, block.Header()); err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 			if err := db.Delete(it.Key()); err != nil { |  | ||||||
| 				return err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 		// Lastly, upgrade the head block, disabling the upgrade mechanism
 |  | ||||||
| 		current := core.GetBlockByHashOld(db, head) |  | ||||||
| 
 |  | ||||||
| 		if err := core.WriteTd(db, current.Hash(), current.DeprecatedTd()); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		if err := core.WriteBody(db, current.Hash(), current.Body()); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 		if err := core.WriteHeader(db, current.Header()); err != nil { |  | ||||||
| 			return err |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| func addMipmapBloomBins(db ethdb.Database) (err error) { |  | ||||||
| 	const mipmapVersion uint = 2 |  | ||||||
| 
 |  | ||||||
| 	// check if the version is set. We ignore data for now since there's
 |  | ||||||
| 	// only one version so we can easily ignore it for now
 |  | ||||||
| 	var data []byte |  | ||||||
| 	data, _ = db.Get([]byte("setting-mipmap-version")) |  | ||||||
| 	if len(data) > 0 { |  | ||||||
| 		var version uint |  | ||||||
| 		if err := rlp.DecodeBytes(data, &version); err == nil && version == mipmapVersion { |  | ||||||
| 			return nil |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	defer func() { |  | ||||||
| 		if err == nil { |  | ||||||
| 			var val []byte |  | ||||||
| 			val, err = rlp.EncodeToBytes(mipmapVersion) |  | ||||||
| 			if err == nil { |  | ||||||
| 				err = db.Put([]byte("setting-mipmap-version"), val) |  | ||||||
| 			} |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
| 	latestBlock := core.GetBlock(db, core.GetHeadBlockHash(db)) |  | ||||||
| 	if latestBlock == nil { // clean database
 |  | ||||||
| 		return |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	tstart := time.Now() |  | ||||||
| 	glog.V(logger.Info).Infoln("upgrading db log bloom bins") |  | ||||||
| 	for i := uint64(0); i <= latestBlock.NumberU64(); i++ { |  | ||||||
| 		hash := core.GetCanonicalHash(db, i) |  | ||||||
| 		if (hash == common.Hash{}) { |  | ||||||
| 			return fmt.Errorf("chain db corrupted. Could not find block %d.", i) |  | ||||||
| 		} |  | ||||||
| 		core.WriteMipmapBloom(db, i, core.GetBlockReceipts(db, hash)) |  | ||||||
| 	} |  | ||||||
| 	glog.V(logger.Info).Infoln("upgrade completed in", time.Since(tstart)) |  | ||||||
| 	return nil |  | ||||||
| } |  | ||||||
|  | |||||||
| @ -61,7 +61,7 @@ func TestMipmapUpgrade(t *testing.T) { | |||||||
| 		if err := core.WriteHeadBlockHash(db, block.Hash()); err != nil { | 		if err := core.WriteHeadBlockHash(db, block.Hash()); err != nil { | ||||||
| 			t.Fatalf("failed to insert block number: %v", err) | 			t.Fatalf("failed to insert block number: %v", err) | ||||||
| 		} | 		} | ||||||
| 		if err := core.WriteBlockReceipts(db, block.Hash(), receipts[i]); err != nil { | 		if err := core.WriteBlockReceipts(db, block.Hash(), block.NumberU64(), receipts[i]); err != nil { | ||||||
| 			t.Fatal("error writing block receipts:", err) | 			t.Fatal("error writing block receipts:", err) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | |||||||
							
								
								
									
										343
									
								
								eth/db_upgrade.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										343
									
								
								eth/db_upgrade.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,343 @@ | |||||||
|  | // Copyright 2014 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 eth implements the Ethereum protocol.
 | ||||||
|  | package eth | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"bytes" | ||||||
|  | 	"encoding/binary" | ||||||
|  | 	"fmt" | ||||||
|  | 	"math/big" | ||||||
|  | 	"time" | ||||||
|  | 
 | ||||||
|  | 	"github.com/ethereum/go-ethereum/common" | ||||||
|  | 	"github.com/ethereum/go-ethereum/core" | ||||||
|  | 	"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" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | var useSequentialKeys = []byte("dbUpgrade_20160530sequentialKeys") | ||||||
|  | 
 | ||||||
|  | // upgradeSequentialKeys checks the chain database version and
 | ||||||
|  | // starts a background process to make upgrades if necessary.
 | ||||||
|  | // Returns a stop function that blocks until the process has
 | ||||||
|  | // been safely stopped.
 | ||||||
|  | func upgradeSequentialKeys(db ethdb.Database) (stopFn func()) { | ||||||
|  | 	data, _ := db.Get(useSequentialKeys) | ||||||
|  | 	if len(data) > 0 && data[0] == 42 { | ||||||
|  | 		return nil // already converted
 | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if data, _ := db.Get([]byte("LastHeader")); len(data) == 0 { | ||||||
|  | 		db.Put(useSequentialKeys, []byte{42}) | ||||||
|  | 		return nil // empty database, nothing to do
 | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	glog.V(logger.Info).Infof("Upgrading chain database to use sequential keys") | ||||||
|  | 
 | ||||||
|  | 	stopChn := make(chan struct{}) | ||||||
|  | 	stoppedChn := make(chan struct{}) | ||||||
|  | 
 | ||||||
|  | 	go func() { | ||||||
|  | 		stopFn := func() bool { | ||||||
|  | 			select { | ||||||
|  | 			case <-time.After(time.Microsecond * 100): // make sure other processes don't get starved
 | ||||||
|  | 			case <-stopChn: | ||||||
|  | 				return true | ||||||
|  | 			} | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		err, stopped := upgradeSequentialCanonicalNumbers(db, stopFn) | ||||||
|  | 		if err == nil && !stopped { | ||||||
|  | 			err, stopped = upgradeSequentialBlocks(db, stopFn) | ||||||
|  | 		} | ||||||
|  | 		if err == nil && !stopped { | ||||||
|  | 			err, stopped = upgradeSequentialOrphanedReceipts(db, stopFn) | ||||||
|  | 		} | ||||||
|  | 		if err == nil && !stopped { | ||||||
|  | 			glog.V(logger.Info).Infof("Database conversion successful") | ||||||
|  | 			db.Put(useSequentialKeys, []byte{42}) | ||||||
|  | 		} | ||||||
|  | 		if err != nil { | ||||||
|  | 			glog.V(logger.Error).Infof("Database conversion failed: %v", err) | ||||||
|  | 		} | ||||||
|  | 		close(stoppedChn) | ||||||
|  | 	}() | ||||||
|  | 
 | ||||||
|  | 	return func() { | ||||||
|  | 		close(stopChn) | ||||||
|  | 		<-stoppedChn | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // upgradeSequentialCanonicalNumbers reads all old format canonical numbers from
 | ||||||
|  | // the database, writes them in new format and deletes the old ones if successful.
 | ||||||
|  | func upgradeSequentialCanonicalNumbers(db ethdb.Database, stopFn func() bool) (error, bool) { | ||||||
|  | 	prefix := []byte("block-num-") | ||||||
|  | 	it := db.(*ethdb.LDBDatabase).NewIterator() | ||||||
|  | 	it.Seek(prefix) | ||||||
|  | 	cnt := 0 | ||||||
|  | 	for bytes.HasPrefix(it.Key(), prefix) { | ||||||
|  | 		keyPtr := it.Key() | ||||||
|  | 		if len(keyPtr) < 20 { | ||||||
|  | 			cnt++ | ||||||
|  | 			if cnt%100000 == 0 { | ||||||
|  | 				glog.V(logger.Info).Infof("converting %d canonical numbers...", cnt) | ||||||
|  | 			} | ||||||
|  | 			number := big.NewInt(0).SetBytes(keyPtr[10:]).Uint64() | ||||||
|  | 			newKey := []byte("h12345678n") | ||||||
|  | 			binary.BigEndian.PutUint64(newKey[1:9], number) | ||||||
|  | 			if err := db.Put(newKey, it.Value()); err != nil { | ||||||
|  | 				return err, false | ||||||
|  | 			} | ||||||
|  | 			if err := db.Delete(keyPtr); err != nil { | ||||||
|  | 				return err, false | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if stopFn() { | ||||||
|  | 			return nil, true | ||||||
|  | 		} | ||||||
|  | 		it.Next() | ||||||
|  | 	} | ||||||
|  | 	if cnt > 0 { | ||||||
|  | 		glog.V(logger.Info).Infof("converted %d canonical numbers...", cnt) | ||||||
|  | 	} | ||||||
|  | 	return nil, false | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // upgradeSequentialBlocks reads all old format block headers, bodies, TDs and block
 | ||||||
|  | // receipts from the database, writes them in new format and deletes the old ones
 | ||||||
|  | // if successful.
 | ||||||
|  | func upgradeSequentialBlocks(db ethdb.Database, stopFn func() bool) (error, bool) { | ||||||
|  | 	prefix := []byte("block-") | ||||||
|  | 	it := db.(*ethdb.LDBDatabase).NewIterator() | ||||||
|  | 	it.Seek(prefix) | ||||||
|  | 	cnt := 0 | ||||||
|  | 	for bytes.HasPrefix(it.Key(), prefix) { | ||||||
|  | 		keyPtr := it.Key() | ||||||
|  | 		if len(keyPtr) >= 38 { | ||||||
|  | 			cnt++ | ||||||
|  | 			if cnt%10000 == 0 { | ||||||
|  | 				glog.V(logger.Info).Infof("converting %d blocks...", cnt) | ||||||
|  | 			} | ||||||
|  | 			// convert header, body, td and block receipts
 | ||||||
|  | 			var keyPrefix [38]byte | ||||||
|  | 			copy(keyPrefix[:], keyPtr[0:38]) | ||||||
|  | 			hash := keyPrefix[6:38] | ||||||
|  | 			if err := upgradeSequentialBlockData(db, hash); err != nil { | ||||||
|  | 				return err, false | ||||||
|  | 			} | ||||||
|  | 			// delete old db entries belonging to this hash
 | ||||||
|  | 			for bytes.HasPrefix(it.Key(), keyPrefix[:]) { | ||||||
|  | 				if err := db.Delete(it.Key()); err != nil { | ||||||
|  | 					return err, false | ||||||
|  | 				} | ||||||
|  | 				it.Next() | ||||||
|  | 			} | ||||||
|  | 			if err := db.Delete(append([]byte("receipts-block-"), hash...)); err != nil { | ||||||
|  | 				return err, false | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			it.Next() | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if stopFn() { | ||||||
|  | 			return nil, true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if cnt > 0 { | ||||||
|  | 		glog.V(logger.Info).Infof("converted %d blocks...", cnt) | ||||||
|  | 	} | ||||||
|  | 	return nil, false | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // upgradeSequentialOrphanedReceipts removes any old format block receipts from the
 | ||||||
|  | // database that did not have a corresponding block
 | ||||||
|  | func upgradeSequentialOrphanedReceipts(db ethdb.Database, stopFn func() bool) (error, bool) { | ||||||
|  | 	prefix := []byte("receipts-block-") | ||||||
|  | 	it := db.(*ethdb.LDBDatabase).NewIterator() | ||||||
|  | 	it.Seek(prefix) | ||||||
|  | 	cnt := 0 | ||||||
|  | 	for bytes.HasPrefix(it.Key(), prefix) { | ||||||
|  | 		// phase 2 already converted receipts belonging to existing
 | ||||||
|  | 		// blocks, just remove if there's anything left
 | ||||||
|  | 		cnt++ | ||||||
|  | 		if err := db.Delete(it.Key()); err != nil { | ||||||
|  | 			return err, false | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if stopFn() { | ||||||
|  | 			return nil, true | ||||||
|  | 		} | ||||||
|  | 		it.Next() | ||||||
|  | 	} | ||||||
|  | 	if cnt > 0 { | ||||||
|  | 		glog.V(logger.Info).Infof("removed %d orphaned block receipts...", cnt) | ||||||
|  | 	} | ||||||
|  | 	return nil, false | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // upgradeSequentialBlockData upgrades the header, body, td and block receipts
 | ||||||
|  | // database entries belonging to a single hash (doesn't delete old data).
 | ||||||
|  | func upgradeSequentialBlockData(db ethdb.Database, hash []byte) error { | ||||||
|  | 	// get old chain data and block number
 | ||||||
|  | 	headerRLP, _ := db.Get(append(append([]byte("block-"), hash...), []byte("-header")...)) | ||||||
|  | 	if len(headerRLP) == 0 { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	header := new(types.Header) | ||||||
|  | 	if err := rlp.Decode(bytes.NewReader(headerRLP), header); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	number := header.Number.Uint64() | ||||||
|  | 	bodyRLP, _ := db.Get(append(append([]byte("block-"), hash...), []byte("-body")...)) | ||||||
|  | 	tdRLP, _ := db.Get(append(append([]byte("block-"), hash...), []byte("-td")...)) | ||||||
|  | 	receiptsRLP, _ := db.Get(append([]byte("receipts-block-"), hash...)) | ||||||
|  | 	// store new hash -> number association
 | ||||||
|  | 	encNum := make([]byte, 8) | ||||||
|  | 	binary.BigEndian.PutUint64(encNum, number) | ||||||
|  | 	if err := db.Put(append([]byte("H"), hash...), encNum); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	// store new chain data
 | ||||||
|  | 	if err := db.Put(append(append([]byte("h"), encNum...), hash...), headerRLP); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	if len(tdRLP) != 0 { | ||||||
|  | 		if err := db.Put(append(append(append([]byte("h"), encNum...), hash...), []byte("t")...), tdRLP); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if len(bodyRLP) != 0 { | ||||||
|  | 		if err := db.Put(append(append([]byte("b"), encNum...), hash...), bodyRLP); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if len(receiptsRLP) != 0 { | ||||||
|  | 		if err := db.Put(append(append([]byte("r"), encNum...), hash...), receiptsRLP); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // upgradeChainDatabase ensures that the chain database stores block split into
 | ||||||
|  | // separate header and body entries.
 | ||||||
|  | func upgradeChainDatabase(db ethdb.Database) error { | ||||||
|  | 	// Short circuit if the head block is stored already as separate header and body
 | ||||||
|  | 	data, err := db.Get([]byte("LastBlock")) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	head := common.BytesToHash(data) | ||||||
|  | 
 | ||||||
|  | 	if block := core.GetBlockByHashOld(db, head); block == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	// At least some of the database is still the old format, upgrade (skip the head block!)
 | ||||||
|  | 	glog.V(logger.Info).Info("Old database detected, upgrading...") | ||||||
|  | 
 | ||||||
|  | 	if db, ok := db.(*ethdb.LDBDatabase); ok { | ||||||
|  | 		blockPrefix := []byte("block-hash-") | ||||||
|  | 		for it := db.NewIterator(); it.Next(); { | ||||||
|  | 			// Skip anything other than a combined block
 | ||||||
|  | 			if !bytes.HasPrefix(it.Key(), blockPrefix) { | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			// Skip the head block (merge last to signal upgrade completion)
 | ||||||
|  | 			if bytes.HasSuffix(it.Key(), head.Bytes()) { | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			// Load the block, split and serialize (order!)
 | ||||||
|  | 			block := core.GetBlockByHashOld(db, common.BytesToHash(bytes.TrimPrefix(it.Key(), blockPrefix))) | ||||||
|  | 
 | ||||||
|  | 			if err := core.WriteTd(db, block.Hash(), block.NumberU64(), block.DeprecatedTd()); err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 			if err := core.WriteBody(db, block.Hash(), block.NumberU64(), block.Body()); err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 			if err := core.WriteHeader(db, block.Header()); err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 			if err := db.Delete(it.Key()); err != nil { | ||||||
|  | 				return err | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		// Lastly, upgrade the head block, disabling the upgrade mechanism
 | ||||||
|  | 		current := core.GetBlockByHashOld(db, head) | ||||||
|  | 
 | ||||||
|  | 		if err := core.WriteTd(db, current.Hash(), current.NumberU64(), current.DeprecatedTd()); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		if err := core.WriteBody(db, current.Hash(), current.NumberU64(), current.Body()); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		if err := core.WriteHeader(db, current.Header()); err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func addMipmapBloomBins(db ethdb.Database) (err error) { | ||||||
|  | 	const mipmapVersion uint = 2 | ||||||
|  | 
 | ||||||
|  | 	// check if the version is set. We ignore data for now since there's
 | ||||||
|  | 	// only one version so we can easily ignore it for now
 | ||||||
|  | 	var data []byte | ||||||
|  | 	data, _ = db.Get([]byte("setting-mipmap-version")) | ||||||
|  | 	if len(data) > 0 { | ||||||
|  | 		var version uint | ||||||
|  | 		if err := rlp.DecodeBytes(data, &version); err == nil && version == mipmapVersion { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	defer func() { | ||||||
|  | 		if err == nil { | ||||||
|  | 			var val []byte | ||||||
|  | 			val, err = rlp.EncodeToBytes(mipmapVersion) | ||||||
|  | 			if err == nil { | ||||||
|  | 				err = db.Put([]byte("setting-mipmap-version"), val) | ||||||
|  | 			} | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	}() | ||||||
|  | 	latestHash := core.GetHeadBlockHash(db) | ||||||
|  | 	latestBlock := core.GetBlock(db, latestHash, core.GetBlockNumber(db, latestHash)) | ||||||
|  | 	if latestBlock == nil { // clean database
 | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	tstart := time.Now() | ||||||
|  | 	glog.V(logger.Info).Infoln("upgrading db log bloom bins") | ||||||
|  | 	for i := uint64(0); i <= latestBlock.NumberU64(); i++ { | ||||||
|  | 		hash := core.GetCanonicalHash(db, i) | ||||||
|  | 		if (hash == common.Hash{}) { | ||||||
|  | 			return fmt.Errorf("chain db corrupted. Could not find block %d.", i) | ||||||
|  | 		} | ||||||
|  | 		core.WriteMipmapBloom(db, i, core.GetBlockReceipts(db, hash, i)) | ||||||
|  | 	} | ||||||
|  | 	glog.V(logger.Info).Infoln("upgrade completed in", time.Since(tstart)) | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
| @ -361,7 +361,7 @@ func (args *NewFilterArgs) UnmarshalJSON(data []byte) error { | |||||||
| 		if len(raw) >= 2 && raw[0] == '0' && (raw[1] == 'x' || raw[1] == 'X') { | 		if len(raw) >= 2 && raw[0] == '0' && (raw[1] == 'x' || raw[1] == 'X') { | ||||||
| 			raw = raw[2:] | 			raw = raw[2:] | ||||||
| 		} | 		} | ||||||
| 		if len(raw) != 2 * common.HashLength { | 		if len(raw) != 2*common.HashLength { | ||||||
| 			return common.Hash{}, errors.New("invalid topic(s)") | 			return common.Hash{}, errors.New("invalid topic(s)") | ||||||
| 		} | 		} | ||||||
| 		if decAddr, err := hex.DecodeString(raw); err == nil { | 		if decAddr, err := hex.DecodeString(raw); err == nil { | ||||||
|  | |||||||
| @ -72,7 +72,8 @@ func (self *Filter) SetTopics(topics [][]common.Hash) { | |||||||
| 
 | 
 | ||||||
| // Run filters logs with the current parameters set
 | // Run filters logs with the current parameters set
 | ||||||
| func (self *Filter) Find() vm.Logs { | func (self *Filter) Find() vm.Logs { | ||||||
| 	latestBlock := core.GetBlock(self.db, core.GetHeadBlockHash(self.db)) | 	latestHash := core.GetHeadBlockHash(self.db) | ||||||
|  | 	latestBlock := core.GetBlock(self.db, latestHash, core.GetBlockNumber(self.db, latestHash)) | ||||||
| 	var beginBlockNo uint64 = uint64(self.begin) | 	var beginBlockNo uint64 = uint64(self.begin) | ||||||
| 	if self.begin == -1 { | 	if self.begin == -1 { | ||||||
| 		beginBlockNo = latestBlock.NumberU64() | 		beginBlockNo = latestBlock.NumberU64() | ||||||
| @ -127,7 +128,7 @@ func (self *Filter) getLogs(start, end uint64) (logs vm.Logs) { | |||||||
| 	for i := start; i <= end; i++ { | 	for i := start; i <= end; i++ { | ||||||
| 		hash := core.GetCanonicalHash(self.db, i) | 		hash := core.GetCanonicalHash(self.db, i) | ||||||
| 		if hash != (common.Hash{}) { | 		if hash != (common.Hash{}) { | ||||||
| 			block = core.GetBlock(self.db, hash) | 			block = core.GetBlock(self.db, hash, i) | ||||||
| 		} else { // block not found
 | 		} else { // block not found
 | ||||||
| 			return logs | 			return logs | ||||||
| 		} | 		} | ||||||
| @ -137,7 +138,7 @@ func (self *Filter) getLogs(start, end uint64) (logs vm.Logs) { | |||||||
| 		if self.bloomFilter(block) { | 		if self.bloomFilter(block) { | ||||||
| 			// Get the logs of the block
 | 			// Get the logs of the block
 | ||||||
| 			var ( | 			var ( | ||||||
| 				receipts   = core.GetBlockReceipts(self.db, block.Hash()) | 				receipts   = core.GetBlockReceipts(self.db, block.Hash(), i) | ||||||
| 				unfiltered vm.Logs | 				unfiltered vm.Logs | ||||||
| 			) | 			) | ||||||
| 			for _, receipt := range receipts { | 			for _, receipt := range receipts { | ||||||
|  | |||||||
| @ -94,7 +94,7 @@ func BenchmarkMipmaps(b *testing.B) { | |||||||
| 		if err := core.WriteHeadBlockHash(db, block.Hash()); err != nil { | 		if err := core.WriteHeadBlockHash(db, block.Hash()); err != nil { | ||||||
| 			b.Fatalf("failed to insert block number: %v", err) | 			b.Fatalf("failed to insert block number: %v", err) | ||||||
| 		} | 		} | ||||||
| 		if err := core.WriteBlockReceipts(db, block.Hash(), receipts[i]); err != nil { | 		if err := core.WriteBlockReceipts(db, block.Hash(), block.NumberU64(), receipts[i]); err != nil { | ||||||
| 			b.Fatal("error writing block receipts:", err) | 			b.Fatal("error writing block receipts:", err) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -196,7 +196,7 @@ func TestFilters(t *testing.T) { | |||||||
| 		if err := core.WriteHeadBlockHash(db, block.Hash()); err != nil { | 		if err := core.WriteHeadBlockHash(db, block.Hash()); err != nil { | ||||||
| 			t.Fatalf("failed to insert block number: %v", err) | 			t.Fatalf("failed to insert block number: %v", err) | ||||||
| 		} | 		} | ||||||
| 		if err := core.WriteBlockReceipts(db, block.Hash(), receipts[i]); err != nil { | 		if err := core.WriteBlockReceipts(db, block.Hash(), block.NumberU64(), receipts[i]); err != nil { | ||||||
| 			t.Fatal("error writing block receipts:", err) | 			t.Fatal("error writing block receipts:", err) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -166,7 +166,7 @@ func (self *GasPriceOracle) processBlock(block *types.Block) { | |||||||
| func (self *GasPriceOracle) lowestPrice(block *types.Block) *big.Int { | func (self *GasPriceOracle) lowestPrice(block *types.Block) *big.Int { | ||||||
| 	gasUsed := big.NewInt(0) | 	gasUsed := big.NewInt(0) | ||||||
| 
 | 
 | ||||||
| 	receipts := core.GetBlockReceipts(self.eth.ChainDb(), block.Hash()) | 	receipts := core.GetBlockReceipts(self.eth.ChainDb(), block.Hash(), block.NumberU64()) | ||||||
| 	if len(receipts) > 0 { | 	if len(receipts) > 0 { | ||||||
| 		if cgu := receipts[len(receipts)-1].CumulativeGasUsed; cgu != nil { | 		if cgu := receipts[len(receipts)-1].CumulativeGasUsed; cgu != nil { | ||||||
| 			gasUsed = receipts[len(receipts)-1].CumulativeGasUsed | 			gasUsed = receipts[len(receipts)-1].CumulativeGasUsed | ||||||
|  | |||||||
| @ -152,9 +152,9 @@ func NewProtocolManager(config *core.ChainConfig, fastSync bool, networkId int, | |||||||
| 		return nil, errIncompatibleConfig | 		return nil, errIncompatibleConfig | ||||||
| 	} | 	} | ||||||
| 	// Construct the different synchronisation mechanisms
 | 	// Construct the different synchronisation mechanisms
 | ||||||
| 	manager.downloader = downloader.New(chaindb, manager.eventMux, blockchain.HasHeader, blockchain.HasBlockAndState, blockchain.GetHeader, | 	manager.downloader = downloader.New(chaindb, manager.eventMux, blockchain.HasHeader, blockchain.HasBlockAndState, blockchain.GetHeaderByHash, | ||||||
| 		blockchain.GetBlock, blockchain.CurrentHeader, blockchain.CurrentBlock, blockchain.CurrentFastBlock, blockchain.FastSyncCommitHead, | 		blockchain.GetBlockByHash, blockchain.CurrentHeader, blockchain.CurrentBlock, blockchain.CurrentFastBlock, blockchain.FastSyncCommitHead, | ||||||
| 		blockchain.GetTd, blockchain.InsertHeaderChain, manager.insertChain, blockchain.InsertReceiptChain, blockchain.Rollback, | 		blockchain.GetTdByHash, blockchain.InsertHeaderChain, manager.insertChain, blockchain.InsertReceiptChain, blockchain.Rollback, | ||||||
| 		manager.removePeer) | 		manager.removePeer) | ||||||
| 
 | 
 | ||||||
| 	validator := func(block *types.Block, parent *types.Block) error { | 	validator := func(block *types.Block, parent *types.Block) error { | ||||||
| @ -167,7 +167,7 @@ func NewProtocolManager(config *core.ChainConfig, fastSync bool, networkId int, | |||||||
| 		atomic.StoreUint32(&manager.synced, 1) // Mark initial sync done on any fetcher import
 | 		atomic.StoreUint32(&manager.synced, 1) // Mark initial sync done on any fetcher import
 | ||||||
| 		return manager.insertChain(blocks) | 		return manager.insertChain(blocks) | ||||||
| 	} | 	} | ||||||
| 	manager.fetcher = fetcher.New(blockchain.GetBlock, validator, manager.BroadcastBlock, heighter, inserter, manager.removePeer) | 	manager.fetcher = fetcher.New(blockchain.GetBlockByHash, validator, manager.BroadcastBlock, heighter, inserter, manager.removePeer) | ||||||
| 
 | 
 | ||||||
| 	if blockchain.Genesis().Hash().Hex() == defaultGenesisHash && networkId == 1 { | 	if blockchain.Genesis().Hash().Hex() == defaultGenesisHash && networkId == 1 { | ||||||
| 		glog.V(logger.Debug).Infoln("Bad Block Reporting is enabled") | 		glog.V(logger.Debug).Infoln("Bad Block Reporting is enabled") | ||||||
| @ -382,7 +382,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { | |||||||
| 				return errResp(ErrDecode, "msg %v: %v", msg, err) | 				return errResp(ErrDecode, "msg %v: %v", msg, err) | ||||||
| 			} | 			} | ||||||
| 			// Retrieve the requested block, stopping if enough was found
 | 			// Retrieve the requested block, stopping if enough was found
 | ||||||
| 			if block := pm.blockchain.GetBlock(hash); block != nil { | 			if block := pm.blockchain.GetBlockByHash(hash); block != nil { | ||||||
| 				blocks = append(blocks, block) | 				blocks = append(blocks, block) | ||||||
| 				bytes += block.Size() | 				bytes += block.Size() | ||||||
| 			} | 			} | ||||||
| @ -425,13 +425,14 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { | |||||||
| 			// Retrieve the next header satisfying the query
 | 			// Retrieve the next header satisfying the query
 | ||||||
| 			var origin *types.Header | 			var origin *types.Header | ||||||
| 			if hashMode { | 			if hashMode { | ||||||
| 				origin = pm.blockchain.GetHeader(query.Origin.Hash) | 				origin = pm.blockchain.GetHeaderByHash(query.Origin.Hash) | ||||||
| 			} else { | 			} else { | ||||||
| 				origin = pm.blockchain.GetHeaderByNumber(query.Origin.Number) | 				origin = pm.blockchain.GetHeaderByNumber(query.Origin.Number) | ||||||
| 			} | 			} | ||||||
| 			if origin == nil { | 			if origin == nil { | ||||||
| 				break | 				break | ||||||
| 			} | 			} | ||||||
|  | 			number := origin.Number.Uint64() | ||||||
| 			headers = append(headers, origin) | 			headers = append(headers, origin) | ||||||
| 			bytes += estHeaderRlpSize | 			bytes += estHeaderRlpSize | ||||||
| 
 | 
 | ||||||
| @ -440,8 +441,9 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { | |||||||
| 			case query.Origin.Hash != (common.Hash{}) && query.Reverse: | 			case query.Origin.Hash != (common.Hash{}) && query.Reverse: | ||||||
| 				// Hash based traversal towards the genesis block
 | 				// Hash based traversal towards the genesis block
 | ||||||
| 				for i := 0; i < int(query.Skip)+1; i++ { | 				for i := 0; i < int(query.Skip)+1; i++ { | ||||||
| 					if header := pm.blockchain.GetHeader(query.Origin.Hash); header != nil { | 					if header := pm.blockchain.GetHeader(query.Origin.Hash, number); header != nil { | ||||||
| 						query.Origin.Hash = header.ParentHash | 						query.Origin.Hash = header.ParentHash | ||||||
|  | 						number-- | ||||||
| 					} else { | 					} else { | ||||||
| 						unknown = true | 						unknown = true | ||||||
| 						break | 						break | ||||||
| @ -602,9 +604,9 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { | |||||||
| 				return errResp(ErrDecode, "msg %v: %v", msg, err) | 				return errResp(ErrDecode, "msg %v: %v", msg, err) | ||||||
| 			} | 			} | ||||||
| 			// Retrieve the requested block's receipts, skipping if unknown to us
 | 			// Retrieve the requested block's receipts, skipping if unknown to us
 | ||||||
| 			results := core.GetBlockReceipts(pm.chaindb, hash) | 			results := core.GetBlockReceipts(pm.chaindb, hash, core.GetBlockNumber(pm.chaindb, hash)) | ||||||
| 			if results == nil { | 			if results == nil { | ||||||
| 				if header := pm.blockchain.GetHeader(hash); header == nil || header.ReceiptHash != types.EmptyRootHash { | 				if header := pm.blockchain.GetHeaderByHash(hash); header == nil || header.ReceiptHash != types.EmptyRootHash { | ||||||
| 					continue | 					continue | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| @ -697,7 +699,8 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { | |||||||
| 		// Update the peers total difficulty if needed, schedule a download if gapped
 | 		// Update the peers total difficulty if needed, schedule a download if gapped
 | ||||||
| 		if request.TD.Cmp(p.Td()) > 0 { | 		if request.TD.Cmp(p.Td()) > 0 { | ||||||
| 			p.SetTd(request.TD) | 			p.SetTd(request.TD) | ||||||
| 			td := pm.blockchain.GetTd(pm.blockchain.CurrentBlock().Hash()) | 			currentBlock := pm.blockchain.CurrentBlock() | ||||||
|  | 			td := pm.blockchain.GetTd(currentBlock.Hash(), currentBlock.NumberU64()) | ||||||
| 			if request.TD.Cmp(new(big.Int).Add(td, request.Block.Difficulty())) > 0 { | 			if request.TD.Cmp(new(big.Int).Add(td, request.Block.Difficulty())) > 0 { | ||||||
| 				go pm.synchronise(p) | 				go pm.synchronise(p) | ||||||
| 			} | 			} | ||||||
| @ -738,8 +741,8 @@ func (pm *ProtocolManager) BroadcastBlock(block *types.Block, propagate bool) { | |||||||
| 	if propagate { | 	if propagate { | ||||||
| 		// Calculate the TD of the block (it's not imported yet, so block.Td is not valid)
 | 		// Calculate the TD of the block (it's not imported yet, so block.Td is not valid)
 | ||||||
| 		var td *big.Int | 		var td *big.Int | ||||||
| 		if parent := pm.blockchain.GetBlock(block.ParentHash()); parent != nil { | 		if parent := pm.blockchain.GetBlock(block.ParentHash(), block.NumberU64()-1); parent != nil { | ||||||
| 			td = new(big.Int).Add(block.Difficulty(), pm.blockchain.GetTd(block.ParentHash())) | 			td = new(big.Int).Add(block.Difficulty(), pm.blockchain.GetTd(block.ParentHash(), block.NumberU64()-1)) | ||||||
| 		} else { | 		} else { | ||||||
| 			glog.V(logger.Error).Infof("propagating dangling block #%d [%x]", block.NumberU64(), hash[:4]) | 			glog.V(logger.Error).Infof("propagating dangling block #%d [%x]", block.NumberU64(), hash[:4]) | ||||||
| 			return | 			return | ||||||
| @ -807,10 +810,11 @@ type EthNodeInfo struct { | |||||||
| 
 | 
 | ||||||
| // NodeInfo retrieves some protocol metadata about the running host node.
 | // NodeInfo retrieves some protocol metadata about the running host node.
 | ||||||
| func (self *ProtocolManager) NodeInfo() *EthNodeInfo { | func (self *ProtocolManager) NodeInfo() *EthNodeInfo { | ||||||
|  | 	currentBlock := self.blockchain.CurrentBlock() | ||||||
| 	return &EthNodeInfo{ | 	return &EthNodeInfo{ | ||||||
| 		Network:    self.networkId, | 		Network:    self.networkId, | ||||||
| 		Difficulty: self.blockchain.GetTd(self.blockchain.CurrentBlock().Hash()), | 		Difficulty: self.blockchain.GetTd(currentBlock.Hash(), currentBlock.NumberU64()), | ||||||
| 		Genesis:    self.blockchain.Genesis().Hash(), | 		Genesis:    self.blockchain.Genesis().Hash(), | ||||||
| 		Head:       self.blockchain.CurrentBlock().Hash(), | 		Head:       currentBlock.Hash(), | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | |||||||
| @ -91,7 +91,7 @@ func testGetBlockHashes(t *testing.T, protocol int) { | |||||||
| 		// Assemble the hash response we would like to receive
 | 		// Assemble the hash response we would like to receive
 | ||||||
| 		resp := make([]common.Hash, tt.result) | 		resp := make([]common.Hash, tt.result) | ||||||
| 		if len(resp) > 0 { | 		if len(resp) > 0 { | ||||||
| 			from := pm.blockchain.GetBlock(tt.origin).NumberU64() - 1 | 			from := pm.blockchain.GetBlockByHash(tt.origin).NumberU64() - 1 | ||||||
| 			for j := 0; j < len(resp); j++ { | 			for j := 0; j < len(resp); j++ { | ||||||
| 				resp[j] = pm.blockchain.GetBlockByNumber(uint64(int(from) - j)).Hash() | 				resp[j] = pm.blockchain.GetBlockByNumber(uint64(int(from) - j)).Hash() | ||||||
| 			} | 			} | ||||||
| @ -204,7 +204,7 @@ func testGetBlocks(t *testing.T, protocol int) { | |||||||
| 		for j, hash := range tt.explicit { | 		for j, hash := range tt.explicit { | ||||||
| 			hashes = append(hashes, hash) | 			hashes = append(hashes, hash) | ||||||
| 			if tt.available[j] && len(blocks) < tt.expected { | 			if tt.available[j] && len(blocks) < tt.expected { | ||||||
| 				blocks = append(blocks, pm.blockchain.GetBlock(hash)) | 				blocks = append(blocks, pm.blockchain.GetBlockByHash(hash)) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		// Send the hash request and verify the response
 | 		// Send the hash request and verify the response
 | ||||||
| @ -339,7 +339,7 @@ func testGetBlockHeaders(t *testing.T, protocol int) { | |||||||
| 		// Collect the headers to expect in the response
 | 		// Collect the headers to expect in the response
 | ||||||
| 		headers := []*types.Header{} | 		headers := []*types.Header{} | ||||||
| 		for _, hash := range tt.expect { | 		for _, hash := range tt.expect { | ||||||
| 			headers = append(headers, pm.blockchain.GetBlock(hash).Header()) | 			headers = append(headers, pm.blockchain.GetBlockByHash(hash).Header()) | ||||||
| 		} | 		} | ||||||
| 		// Send the hash request and verify the response
 | 		// Send the hash request and verify the response
 | ||||||
| 		p2p.Send(peer.app, 0x03, tt.query) | 		p2p.Send(peer.app, 0x03, tt.query) | ||||||
| @ -420,7 +420,7 @@ func testGetBlockBodies(t *testing.T, protocol int) { | |||||||
| 		for j, hash := range tt.explicit { | 		for j, hash := range tt.explicit { | ||||||
| 			hashes = append(hashes, hash) | 			hashes = append(hashes, hash) | ||||||
| 			if tt.available[j] && len(bodies) < tt.expected { | 			if tt.available[j] && len(bodies) < tt.expected { | ||||||
| 				block := pm.blockchain.GetBlock(hash) | 				block := pm.blockchain.GetBlockByHash(hash) | ||||||
| 				bodies = append(bodies, &blockBody{Transactions: block.Transactions(), Uncles: block.Uncles()}) | 				bodies = append(bodies, &blockBody{Transactions: block.Transactions(), Uncles: block.Uncles()}) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| @ -572,7 +572,7 @@ func testGetReceipt(t *testing.T, protocol int) { | |||||||
| 		block := pm.blockchain.GetBlockByNumber(i) | 		block := pm.blockchain.GetBlockByNumber(i) | ||||||
| 
 | 
 | ||||||
| 		hashes = append(hashes, block.Hash()) | 		hashes = append(hashes, block.Hash()) | ||||||
| 		receipts = append(receipts, core.GetBlockReceipts(pm.chaindb, block.Hash())) | 		receipts = append(receipts, core.GetBlockReceipts(pm.chaindb, block.Hash(), block.NumberU64())) | ||||||
| 	} | 	} | ||||||
| 	// Send the hash request and verify the response
 | 	// Send the hash request and verify the response
 | ||||||
| 	p2p.Send(peer.app, 0x0f, hashes) | 	p2p.Send(peer.app, 0x0f, hashes) | ||||||
|  | |||||||
| @ -162,7 +162,8 @@ func (pm *ProtocolManager) synchronise(peer *peer) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	// Make sure the peer's TD is higher than our own. If not drop.
 | 	// Make sure the peer's TD is higher than our own. If not drop.
 | ||||||
| 	td := pm.blockchain.GetTd(pm.blockchain.CurrentBlock().Hash()) | 	currentBlock := pm.blockchain.CurrentBlock() | ||||||
|  | 	td := pm.blockchain.GetTd(currentBlock.Hash(), currentBlock.NumberU64()) | ||||||
| 	if peer.Td().Cmp(td) <= 0 { | 	if peer.Td().Cmp(td) <= 0 { | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -272,7 +272,7 @@ func (self *worker) wait() { | |||||||
| 				go self.mux.Post(core.NewMinedBlockEvent{Block: block}) | 				go self.mux.Post(core.NewMinedBlockEvent{Block: block}) | ||||||
| 			} else { | 			} else { | ||||||
| 				work.state.Commit() | 				work.state.Commit() | ||||||
| 				parent := self.chain.GetBlock(block.ParentHash()) | 				parent := self.chain.GetBlock(block.ParentHash(), block.NumberU64()-1) | ||||||
| 				if parent == nil { | 				if parent == nil { | ||||||
| 					glog.V(logger.Error).Infoln("Invalid block found during mining") | 					glog.V(logger.Error).Infoln("Invalid block found during mining") | ||||||
| 					continue | 					continue | ||||||
| @ -319,7 +319,7 @@ func (self *worker) wait() { | |||||||
| 						self.mux.Post(core.ChainHeadEvent{Block: block}) | 						self.mux.Post(core.ChainHeadEvent{Block: block}) | ||||||
| 						self.mux.Post(logs) | 						self.mux.Post(logs) | ||||||
| 					} | 					} | ||||||
| 					if err := core.WriteBlockReceipts(self.chainDb, block.Hash(), receipts); err != nil { | 					if err := core.WriteBlockReceipts(self.chainDb, block.Hash(), block.NumberU64(), receipts); err != nil { | ||||||
| 						glog.V(logger.Warn).Infoln("error writing block receipts:", err) | 						glog.V(logger.Warn).Infoln("error writing block receipts:", err) | ||||||
| 					} | 					} | ||||||
| 				}(block, work.state.Logs(), work.receipts) | 				}(block, work.state.Logs(), work.receipts) | ||||||
|  | |||||||
| @ -164,7 +164,7 @@ func runBlockTest(homesteadBlock *big.Int, test *BlockTest) error { | |||||||
| 		return fmt.Errorf("InsertPreState: %v", err) | 		return fmt.Errorf("InsertPreState: %v", err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	core.WriteTd(db, test.Genesis.Hash(), test.Genesis.Difficulty()) | 	core.WriteTd(db, test.Genesis.Hash(), 0, test.Genesis.Difficulty()) | ||||||
| 	core.WriteBlock(db, test.Genesis) | 	core.WriteBlock(db, test.Genesis) | ||||||
| 	core.WriteCanonicalHash(db, test.Genesis.Hash(), test.Genesis.NumberU64()) | 	core.WriteCanonicalHash(db, test.Genesis.Hash(), test.Genesis.NumberU64()) | ||||||
| 	core.WriteHeadBlockHash(db, test.Genesis.Hash()) | 	core.WriteHeadBlockHash(db, test.Genesis.Hash()) | ||||||
| @ -412,7 +412,7 @@ func (test *BlockTest) ValidateImportedHeaders(cm *core.BlockChain, validBlocks | |||||||
| 	// block-by-block, so we can only validate imported headers after
 | 	// block-by-block, so we can only validate imported headers after
 | ||||||
| 	// all blocks have been processed by ChainManager, as they may not
 | 	// all blocks have been processed by ChainManager, as they may not
 | ||||||
| 	// be part of the longest chain until last block is imported.
 | 	// be part of the longest chain until last block is imported.
 | ||||||
| 	for b := cm.CurrentBlock(); b != nil && b.NumberU64() != 0; b = cm.GetBlock(b.Header().ParentHash) { | 	for b := cm.CurrentBlock(); b != nil && b.NumberU64() != 0; b = cm.GetBlockByHash(b.Header().ParentHash) { | ||||||
| 		bHash := common.Bytes2Hex(b.Hash().Bytes()) // hex without 0x prefix
 | 		bHash := common.Bytes2Hex(b.Hash().Bytes()) // hex without 0x prefix
 | ||||||
| 		if err := validateHeader(bmap[bHash].BlockHeader, b.Header()); err != nil { | 		if err := validateHeader(bmap[bHash].BlockHeader, b.Header()); err != nil { | ||||||
| 			return fmt.Errorf("Imported block header validation failed: %v", err) | 			return fmt.Errorf("Imported block header validation failed: %v", err) | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user