mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-18 08:38:46 +00:00
436493350e
1. changes sentinel to use an http-like interface 2. moves hexutil, crypto/blake2b, metrics packages to erigon-lib
370 lines
13 KiB
Go
370 lines
13 KiB
Go
// Copyright 2015 The go-ethereum Authors
|
|
// This file is part of the go-ethereum library.
|
|
//
|
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// The go-ethereum library is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
// Package tests implements execution of Ethereum JSON tests.
|
|
package tests
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/hex"
|
|
"encoding/json"
|
|
"fmt"
|
|
"github.com/ledgerwatch/erigon-lib/common/hexutil"
|
|
"math/big"
|
|
"reflect"
|
|
"testing"
|
|
|
|
"github.com/holiman/uint256"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/chain"
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/erigon-lib/common/hexutility"
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
"github.com/ledgerwatch/erigon/turbo/services"
|
|
"github.com/ledgerwatch/erigon/turbo/stages/mock"
|
|
|
|
"github.com/ledgerwatch/log/v3"
|
|
|
|
"github.com/ledgerwatch/erigon/common"
|
|
"github.com/ledgerwatch/erigon/common/math"
|
|
"github.com/ledgerwatch/erigon/core"
|
|
"github.com/ledgerwatch/erigon/core/rawdb"
|
|
"github.com/ledgerwatch/erigon/core/state"
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
"github.com/ledgerwatch/erigon/eth/ethconsensusconfig"
|
|
"github.com/ledgerwatch/erigon/rlp"
|
|
)
|
|
|
|
// A BlockTest checks handling of entire blocks.
|
|
type BlockTest struct {
|
|
json btJSON
|
|
br services.FullBlockReader
|
|
}
|
|
|
|
// UnmarshalJSON implements json.Unmarshaler interface.
|
|
func (bt *BlockTest) UnmarshalJSON(in []byte) error {
|
|
return json.Unmarshal(in, &bt.json)
|
|
}
|
|
|
|
type btJSON struct {
|
|
Blocks []btBlock `json:"blocks"`
|
|
Genesis btHeader `json:"genesisBlockHeader"`
|
|
Pre types.GenesisAlloc `json:"pre"`
|
|
Post types.GenesisAlloc `json:"postState"`
|
|
BestBlock common.UnprefixedHash `json:"lastblockhash"`
|
|
Network string `json:"network"`
|
|
SealEngine string `json:"sealEngine"`
|
|
}
|
|
|
|
type btBlock struct {
|
|
BlockHeader *btHeader
|
|
ExpectException string
|
|
Rlp string
|
|
UncleHeaders []*btHeader
|
|
}
|
|
|
|
//go:generate gencodec -type btHeader -field-override btHeaderMarshaling -out gen_btheader.go
|
|
|
|
type btHeader struct {
|
|
Bloom types.Bloom
|
|
Coinbase libcommon.Address
|
|
MixHash libcommon.Hash
|
|
Nonce types.BlockNonce
|
|
Number *big.Int
|
|
Hash libcommon.Hash
|
|
ParentHash libcommon.Hash
|
|
ReceiptTrie libcommon.Hash
|
|
StateRoot libcommon.Hash
|
|
TransactionsTrie libcommon.Hash
|
|
UncleHash libcommon.Hash
|
|
ExtraData []byte
|
|
Difficulty *big.Int
|
|
GasLimit uint64
|
|
GasUsed uint64
|
|
Timestamp uint64
|
|
BaseFeePerGas *big.Int
|
|
WithdrawalsRoot *libcommon.Hash
|
|
BlobGasUsed *uint64
|
|
ExcessBlobGas *uint64
|
|
ParentBeaconBlockRoot *libcommon.Hash
|
|
}
|
|
|
|
type btHeaderMarshaling struct {
|
|
ExtraData hexutility.Bytes
|
|
Number *math.HexOrDecimal256
|
|
Difficulty *math.HexOrDecimal256
|
|
GasLimit math.HexOrDecimal64
|
|
GasUsed math.HexOrDecimal64
|
|
Timestamp math.HexOrDecimal64
|
|
BaseFeePerGas *math.HexOrDecimal256
|
|
BlobGasUsed *math.HexOrDecimal64
|
|
ExcessBlobGas *math.HexOrDecimal64
|
|
}
|
|
|
|
func (bt *BlockTest) Run(t *testing.T, checkStateRoot bool) error {
|
|
config, ok := Forks[bt.json.Network]
|
|
if !ok {
|
|
return UnsupportedForkError{bt.json.Network}
|
|
}
|
|
engine := ethconsensusconfig.CreateConsensusEngineBareBones(context.Background(), config, log.New())
|
|
m := mock.MockWithGenesisEngine(t, bt.genesis(config), engine, false, checkStateRoot)
|
|
defer m.Close()
|
|
|
|
bt.br = m.BlockReader
|
|
// import pre accounts & construct test genesis block & state root
|
|
if m.Genesis.Hash() != bt.json.Genesis.Hash {
|
|
return fmt.Errorf("genesis block hash doesn't match test: computed=%x, test=%x", m.Genesis.Hash().Bytes()[:6], bt.json.Genesis.Hash[:6])
|
|
}
|
|
if m.Genesis.Root() != bt.json.Genesis.StateRoot {
|
|
return fmt.Errorf("genesis block state root does not match test: computed=%x, test=%x", m.Genesis.Root().Bytes()[:6], bt.json.Genesis.StateRoot[:6])
|
|
}
|
|
|
|
validBlocks, err := bt.insertBlocks(m)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
tx, err := m.DB.BeginRw(m.Ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer tx.Rollback()
|
|
|
|
cmlast := rawdb.ReadHeadBlockHash(tx)
|
|
if libcommon.Hash(bt.json.BestBlock) != cmlast {
|
|
return fmt.Errorf("last block hash validation mismatch: want: %x, have: %x", bt.json.BestBlock, cmlast)
|
|
}
|
|
newDB := state.New(m.NewStateReader(tx))
|
|
if err := bt.validatePostState(newDB); err != nil {
|
|
return fmt.Errorf("post state validation failed: %w", err)
|
|
}
|
|
return bt.validateImportedHeaders(tx, validBlocks, m)
|
|
}
|
|
|
|
func (bt *BlockTest) genesis(config *chain.Config) *types.Genesis {
|
|
return &types.Genesis{
|
|
Config: config,
|
|
Nonce: bt.json.Genesis.Nonce.Uint64(),
|
|
Timestamp: bt.json.Genesis.Timestamp,
|
|
ParentHash: bt.json.Genesis.ParentHash,
|
|
ExtraData: bt.json.Genesis.ExtraData,
|
|
GasLimit: bt.json.Genesis.GasLimit,
|
|
GasUsed: bt.json.Genesis.GasUsed,
|
|
Difficulty: bt.json.Genesis.Difficulty,
|
|
Mixhash: bt.json.Genesis.MixHash,
|
|
Coinbase: bt.json.Genesis.Coinbase,
|
|
Alloc: bt.json.Pre,
|
|
BaseFee: bt.json.Genesis.BaseFeePerGas,
|
|
BlobGasUsed: bt.json.Genesis.BlobGasUsed,
|
|
ExcessBlobGas: bt.json.Genesis.ExcessBlobGas,
|
|
ParentBeaconBlockRoot: bt.json.Genesis.ParentBeaconBlockRoot,
|
|
}
|
|
}
|
|
|
|
/*
|
|
See https://github.com/ethereum/tests/wiki/Blockchain-Tests-II
|
|
|
|
Whether a block is valid or not is a bit subtle, it's defined by presence of
|
|
blockHeader, transactions and uncleHeaders fields. If they are missing, the block is
|
|
invalid and we must verify that we do not accept it.
|
|
|
|
Since some tests mix valid and invalid blocks we need to check this for every block.
|
|
|
|
If a block is invalid it does not necessarily fail the test, if it's invalidness is
|
|
expected we are expected to ignore it and continue processing and then validate the
|
|
post state.
|
|
*/
|
|
func (bt *BlockTest) insertBlocks(m *mock.MockSentry) ([]btBlock, error) {
|
|
validBlocks := make([]btBlock, 0)
|
|
// insert the test blocks, which will execute all transaction
|
|
for bi, b := range bt.json.Blocks {
|
|
cb, err := b.decode()
|
|
if err != nil {
|
|
if b.BlockHeader == nil {
|
|
continue // OK - block is supposed to be invalid, continue with next block
|
|
} else {
|
|
return nil, fmt.Errorf("block RLP decoding failed when expected to succeed: %w", err)
|
|
}
|
|
}
|
|
// RLP decoding worked, try to insert into chain:
|
|
chain := &core.ChainPack{Blocks: []*types.Block{cb}, Headers: []*types.Header{cb.Header()}, TopBlock: cb}
|
|
|
|
err1 := m.InsertChain(chain)
|
|
if err1 != nil {
|
|
if b.BlockHeader == nil {
|
|
continue // OK - block is supposed to be invalid, continue with next block
|
|
} else {
|
|
return nil, fmt.Errorf("block #%v insertion into chain failed: %w", cb.Number(), err1)
|
|
}
|
|
} else if b.BlockHeader == nil {
|
|
roTx, err := m.DB.BeginRo(m.Ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer roTx.Rollback()
|
|
canonical, cErr := bt.br.CanonicalHash(context.Background(), roTx, cb.NumberU64())
|
|
if cErr != nil {
|
|
return nil, cErr
|
|
}
|
|
if canonical == cb.Hash() {
|
|
return nil, fmt.Errorf("block (index %d) insertion should have failed due to: %v", bi, b.ExpectException)
|
|
}
|
|
}
|
|
if b.BlockHeader == nil {
|
|
continue
|
|
}
|
|
// validate RLP decoding by checking all values against test file JSON
|
|
if err = validateHeader(b.BlockHeader, cb.Header()); err != nil {
|
|
return nil, fmt.Errorf("deserialised block header validation failed: %w", err)
|
|
}
|
|
validBlocks = append(validBlocks, b)
|
|
}
|
|
return validBlocks, nil
|
|
}
|
|
|
|
func validateHeader(h *btHeader, h2 *types.Header) error {
|
|
if h == nil {
|
|
return fmt.Errorf("validateHeader: h == nil")
|
|
}
|
|
if h2 == nil {
|
|
return fmt.Errorf("validateHeader: h2 == nil")
|
|
}
|
|
if h.Bloom != h2.Bloom {
|
|
return fmt.Errorf("bloom: want: %x have: %x", h.Bloom, h2.Bloom)
|
|
}
|
|
if h.Coinbase != h2.Coinbase {
|
|
return fmt.Errorf("coinbase: want: %x have: %x", h.Coinbase, h2.Coinbase)
|
|
}
|
|
if h.MixHash != h2.MixDigest {
|
|
return fmt.Errorf("MixHash: want: %x have: %x", h.MixHash, h2.MixDigest)
|
|
}
|
|
if h.Nonce != h2.Nonce {
|
|
return fmt.Errorf("nonce: want: %x have: %x", h.Nonce, h2.Nonce)
|
|
}
|
|
if h.Number.Cmp(h2.Number) != 0 {
|
|
return fmt.Errorf("number: want: %v have: %v", h.Number, h2.Number)
|
|
}
|
|
if h.ParentHash != h2.ParentHash {
|
|
return fmt.Errorf("parent hash: want: %x have: %x", h.ParentHash, h2.ParentHash)
|
|
}
|
|
if h.ReceiptTrie != h2.ReceiptHash {
|
|
return fmt.Errorf("receipt hash: want: %x have: %x", h.ReceiptTrie, h2.ReceiptHash)
|
|
}
|
|
if h.TransactionsTrie != h2.TxHash {
|
|
return fmt.Errorf("txn hash: want: %x have: %x", h.TransactionsTrie, h2.TxHash)
|
|
}
|
|
if h.StateRoot != h2.Root {
|
|
return fmt.Errorf("state hash: want: %x have: %x", h.StateRoot, h2.Root)
|
|
}
|
|
if h.UncleHash != h2.UncleHash {
|
|
return fmt.Errorf("uncle hash: want: %x have: %x", h.UncleHash, h2.UncleHash)
|
|
}
|
|
if !bytes.Equal(h.ExtraData, h2.Extra) {
|
|
return fmt.Errorf("extra data: want: %x have: %x", h.ExtraData, h2.Extra)
|
|
}
|
|
if h.Difficulty.Cmp(h2.Difficulty) != 0 {
|
|
return fmt.Errorf("difficulty: want: %v have: %v", h.Difficulty, h2.Difficulty)
|
|
}
|
|
if h.GasLimit != h2.GasLimit {
|
|
return fmt.Errorf("gasLimit: want: %d have: %d", h.GasLimit, h2.GasLimit)
|
|
}
|
|
if h.GasUsed != h2.GasUsed {
|
|
return fmt.Errorf("gasUsed: want: %d have: %d", h.GasUsed, h2.GasUsed)
|
|
}
|
|
if h.Timestamp != h2.Time {
|
|
return fmt.Errorf("timestamp: want: %v have: %v", h.Timestamp, h2.Time)
|
|
}
|
|
if !reflect.DeepEqual(h.BaseFeePerGas, h2.BaseFee) {
|
|
return fmt.Errorf("baseFeePerGas: want: %v have: %v", h.BaseFeePerGas, h2.BaseFee)
|
|
}
|
|
if !reflect.DeepEqual(h.WithdrawalsRoot, h2.WithdrawalsHash) {
|
|
return fmt.Errorf("withdrawalsRoot: want: %v have: %v", h.WithdrawalsRoot, h2.WithdrawalsHash)
|
|
}
|
|
if !reflect.DeepEqual(h.BlobGasUsed, h2.BlobGasUsed) {
|
|
return fmt.Errorf("blobGasUsed: want: %v have: %v", h.BlobGasUsed, h2.BlobGasUsed)
|
|
}
|
|
if !reflect.DeepEqual(h.ExcessBlobGas, h2.ExcessBlobGas) {
|
|
return fmt.Errorf("excessBlobGas: want: %v have: %v", h.ExcessBlobGas, h2.ExcessBlobGas)
|
|
}
|
|
if !reflect.DeepEqual(h.ParentBeaconBlockRoot, h2.ParentBeaconBlockRoot) {
|
|
return fmt.Errorf("parentBeaconBlockRoot: want: %v have: %v", h.ParentBeaconBlockRoot, h2.ParentBeaconBlockRoot)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (bt *BlockTest) validatePostState(statedb *state.IntraBlockState) error {
|
|
// validate post state accounts in test file against what we have in state db
|
|
for addr, acct := range bt.json.Post {
|
|
// address is indirectly verified by the other fields, as it's the db key
|
|
code2 := statedb.GetCode(addr)
|
|
balance2 := statedb.GetBalance(addr)
|
|
nonce2 := statedb.GetNonce(addr)
|
|
if !bytes.Equal(code2, acct.Code) {
|
|
return fmt.Errorf("account code mismatch for addr: %x want: %v have: %s", addr, acct.Code, hex.EncodeToString(code2))
|
|
}
|
|
if balance2.ToBig().Cmp(acct.Balance) != 0 {
|
|
return fmt.Errorf("account balance mismatch for addr: %x, want: %d, have: %d", addr, acct.Balance, balance2)
|
|
}
|
|
if nonce2 != acct.Nonce {
|
|
return fmt.Errorf("account nonce mismatch for addr: %x want: %d have: %d", addr, acct.Nonce, nonce2)
|
|
}
|
|
for loc, val := range acct.Storage {
|
|
val1 := uint256.NewInt(0).SetBytes(val.Bytes())
|
|
val2 := uint256.NewInt(0)
|
|
statedb.GetState(addr, &loc, val2)
|
|
if !val1.Eq(val2) {
|
|
return fmt.Errorf("storage mismatch for addr: %x loc: %x want: %d have: %d", addr, loc, val1, val2)
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (bt *BlockTest) validateImportedHeaders(tx kv.Tx, validBlocks []btBlock, m *mock.MockSentry) error {
|
|
// to get constant lookup when verifying block headers by hash (some tests have many blocks)
|
|
bmap := make(map[libcommon.Hash]btBlock, len(bt.json.Blocks))
|
|
for _, b := range validBlocks {
|
|
bmap[b.BlockHeader.Hash] = b
|
|
}
|
|
// iterate over blocks backwards from HEAD and validate imported
|
|
// headers vs test file. some tests have reorgs, and we import
|
|
// block-by-block, so we can only validate imported headers after
|
|
// all blocks have been processed by BlockChain, as they may not
|
|
// be part of the longest chain until last block is imported.
|
|
for b, _ := m.BlockReader.CurrentBlock(tx); b != nil && b.NumberU64() != 0; {
|
|
if err := validateHeader(bmap[b.Hash()].BlockHeader, b.Header()); err != nil {
|
|
return fmt.Errorf("imported block header validation failed: %w", err)
|
|
}
|
|
number := rawdb.ReadHeaderNumber(tx, b.ParentHash())
|
|
if number == nil {
|
|
break
|
|
}
|
|
b, _, _ = m.BlockReader.BlockWithSenders(m.Ctx, tx, b.ParentHash(), *number)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (bb *btBlock) decode() (*types.Block, error) {
|
|
data, err := hexutil.Decode(bb.Rlp)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var b types.Block
|
|
err = rlp.DecodeBytes(data, &b)
|
|
return &b, err
|
|
}
|