This PR refactors the eth test suite to make it more readable and easier to use. Some notable differences: - A new file helpers.go stores all of the methods used between both eth66 and eth65 and below tests, as well as methods shared among many test functions. - suite.go now contains all of the test functions for both eth65 tests and eth66 tests. - The utesting.T object doesn't get passed through to other helper methods, but is instead only used within the scope of the test function, whereas helper methods return errors, so only the test function itself can fatal out in the case of an error. - The full test suite now only takes 13.5 seconds to run.
		
			
				
	
	
		
			199 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			199 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2020 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 ethtest
 | |
| 
 | |
| import (
 | |
| 	"compress/gzip"
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"io/ioutil"
 | |
| 	"math/big"
 | |
| 	"os"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/ethereum/go-ethereum/core"
 | |
| 	"github.com/ethereum/go-ethereum/core/forkid"
 | |
| 	"github.com/ethereum/go-ethereum/core/types"
 | |
| 	"github.com/ethereum/go-ethereum/params"
 | |
| 	"github.com/ethereum/go-ethereum/rlp"
 | |
| )
 | |
| 
 | |
| type Chain struct {
 | |
| 	genesis     core.Genesis
 | |
| 	blocks      []*types.Block
 | |
| 	chainConfig *params.ChainConfig
 | |
| }
 | |
| 
 | |
| func (c *Chain) WriteTo(writer io.Writer) error {
 | |
| 	for _, block := range c.blocks {
 | |
| 		if err := rlp.Encode(writer, block); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // Len returns the length of the chain.
 | |
| func (c *Chain) Len() int {
 | |
| 	return len(c.blocks)
 | |
| }
 | |
| 
 | |
| // TD calculates the total difficulty of the chain at the
 | |
| // chain head.
 | |
| func (c *Chain) TD() *big.Int {
 | |
| 	sum := big.NewInt(0)
 | |
| 	for _, block := range c.blocks[:c.Len()] {
 | |
| 		sum.Add(sum, block.Difficulty())
 | |
| 	}
 | |
| 	return sum
 | |
| }
 | |
| 
 | |
| // TotalDifficultyAt calculates the total difficulty of the chain
 | |
| // at the given block height.
 | |
| func (c *Chain) TotalDifficultyAt(height int) *big.Int {
 | |
| 	sum := big.NewInt(0)
 | |
| 	if height >= c.Len() {
 | |
| 		return sum
 | |
| 	}
 | |
| 	for _, block := range c.blocks[:height+1] {
 | |
| 		sum.Add(sum, block.Difficulty())
 | |
| 	}
 | |
| 	return sum
 | |
| }
 | |
| 
 | |
| // ForkID gets the fork id of the chain.
 | |
| func (c *Chain) ForkID() forkid.ID {
 | |
| 	return forkid.NewID(c.chainConfig, c.blocks[0].Hash(), uint64(c.Len()))
 | |
| }
 | |
| 
 | |
| // Shorten returns a copy chain of a desired height from the imported
 | |
| func (c *Chain) Shorten(height int) *Chain {
 | |
| 	blocks := make([]*types.Block, height)
 | |
| 	copy(blocks, c.blocks[:height])
 | |
| 
 | |
| 	config := *c.chainConfig
 | |
| 	return &Chain{
 | |
| 		blocks:      blocks,
 | |
| 		chainConfig: &config,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Head returns the chain head.
 | |
| func (c *Chain) Head() *types.Block {
 | |
| 	return c.blocks[c.Len()-1]
 | |
| }
 | |
| 
 | |
| func (c *Chain) GetHeaders(req GetBlockHeaders) (BlockHeaders, error) {
 | |
| 	if req.Amount < 1 {
 | |
| 		return nil, fmt.Errorf("no block headers requested")
 | |
| 	}
 | |
| 
 | |
| 	headers := make(BlockHeaders, req.Amount)
 | |
| 	var blockNumber uint64
 | |
| 
 | |
| 	// range over blocks to check if our chain has the requested header
 | |
| 	for _, block := range c.blocks {
 | |
| 		if block.Hash() == req.Origin.Hash || block.Number().Uint64() == req.Origin.Number {
 | |
| 			headers[0] = block.Header()
 | |
| 			blockNumber = block.Number().Uint64()
 | |
| 		}
 | |
| 	}
 | |
| 	if headers[0] == nil {
 | |
| 		return nil, fmt.Errorf("no headers found for given origin number %v, hash %v", req.Origin.Number, req.Origin.Hash)
 | |
| 	}
 | |
| 
 | |
| 	if req.Reverse {
 | |
| 		for i := 1; i < int(req.Amount); i++ {
 | |
| 			blockNumber -= (1 - req.Skip)
 | |
| 			headers[i] = c.blocks[blockNumber].Header()
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 		return headers, nil
 | |
| 	}
 | |
| 
 | |
| 	for i := 1; i < int(req.Amount); i++ {
 | |
| 		blockNumber += (1 + req.Skip)
 | |
| 		headers[i] = c.blocks[blockNumber].Header()
 | |
| 	}
 | |
| 
 | |
| 	return headers, nil
 | |
| }
 | |
| 
 | |
| // loadChain takes the given chain.rlp file, and decodes and returns
 | |
| // the blocks from the file.
 | |
| func loadChain(chainfile string, genesis string) (*Chain, error) {
 | |
| 	gen, err := loadGenesis(genesis)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	gblock := gen.ToBlock(nil)
 | |
| 
 | |
| 	blocks, err := blocksFromFile(chainfile, gblock)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	c := &Chain{genesis: gen, blocks: blocks, chainConfig: gen.Config}
 | |
| 	return c, nil
 | |
| }
 | |
| 
 | |
| func loadGenesis(genesisFile string) (core.Genesis, error) {
 | |
| 	chainConfig, err := ioutil.ReadFile(genesisFile)
 | |
| 	if err != nil {
 | |
| 		return core.Genesis{}, err
 | |
| 	}
 | |
| 	var gen core.Genesis
 | |
| 	if err := json.Unmarshal(chainConfig, &gen); err != nil {
 | |
| 		return core.Genesis{}, err
 | |
| 	}
 | |
| 	return gen, nil
 | |
| }
 | |
| 
 | |
| func blocksFromFile(chainfile string, gblock *types.Block) ([]*types.Block, error) {
 | |
| 	// Load chain.rlp.
 | |
| 	fh, err := os.Open(chainfile)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	defer fh.Close()
 | |
| 	var reader io.Reader = fh
 | |
| 	if strings.HasSuffix(chainfile, ".gz") {
 | |
| 		if reader, err = gzip.NewReader(reader); err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 	}
 | |
| 	stream := rlp.NewStream(reader, 0)
 | |
| 	var blocks = make([]*types.Block, 1)
 | |
| 	blocks[0] = gblock
 | |
| 	for i := 0; ; i++ {
 | |
| 		var b types.Block
 | |
| 		if err := stream.Decode(&b); err == io.EOF {
 | |
| 			break
 | |
| 		} else if err != nil {
 | |
| 			return nil, fmt.Errorf("at block index %d: %v", i, err)
 | |
| 		}
 | |
| 		if b.NumberU64() != uint64(i+1) {
 | |
| 			return nil, fmt.Errorf("block at index %d has wrong number %d", i, b.NumberU64())
 | |
| 		}
 | |
| 		blocks = append(blocks, &b)
 | |
| 	}
 | |
| 	return blocks, nil
 | |
| }
 |