Skip to content

Commit

Permalink
add testcase for offchain evm backward compatibilities
Browse files Browse the repository at this point in the history
  • Loading branch information
zhangchiqing committed Nov 21, 2024
1 parent e94c022 commit dffadd8
Show file tree
Hide file tree
Showing 2 changed files with 429 additions and 3 deletions.
375 changes: 375 additions & 0 deletions fvm/evm/offchain/utils/collection_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,375 @@
package utils_test

import (
"bufio"
"encoding/gob"
"encoding/hex"
"encoding/json"
"fmt"
"os"
"path/filepath"
"strings"
"testing"

"github.com/rs/zerolog"
"github.com/rs/zerolog/log"
"github.com/stretchr/testify/require"

"github.com/onflow/cadence"
"github.com/onflow/cadence/encoding/ccf"

"github.com/onflow/flow-go/fvm/environment"
"github.com/onflow/flow-go/fvm/evm"
"github.com/onflow/flow-go/fvm/evm/events"
"github.com/onflow/flow-go/fvm/evm/offchain/blocks"
"github.com/onflow/flow-go/fvm/evm/offchain/storage"
"github.com/onflow/flow-go/fvm/evm/offchain/sync"
"github.com/onflow/flow-go/fvm/evm/offchain/utils"
. "github.com/onflow/flow-go/fvm/evm/testutils"
"github.com/onflow/flow-go/model/flow"
)

func TestTestnetBackwardCompatibility(t *testing.T) {
t.Skip("TIME CONSUMING TESTS. Enable the tests with the events files saved in local")
// how to run this tests
// Note: this is a time consuming tests, so please run it in local
//
// 1) run the following cli to get the events files across different sporks

// flow events get A.8c5303eaa26202d6.EVM.TransactionExecuted A.8c5303eaa26202d6.EVM.BlockExecuted
// --start 211176670 --end 211176770 --network testnet --host access-001.devnet51.nodes.onflow.org:9000
// > ~/Downloads/events_devnet51_1.jsonl
// ...
//
// 2) comment the above t.Skip, and update the events file paths and checkpoint dir
// to run the tests
BackwardCompatibleSinceEVMGenesisBlock(
t, flow.Testnet, []string{
"~/Downloads/events_devnet51_1.jsonl",
"~/Downloads/events_devnet51_2.jsonl",
},
"~/Downloads/",
0,
)
}

// BackwardCompatibilityTestSinceEVMGenesisBlock verifies that the offchain package
// is able to read EVM events from the given file paths and replay blocks since the
// EVM genesis block and derive a consistant state as the latest onchain EVM state.
// the eventsFilePaths is a list of file paths that contain ordered EVM events in JSONL format.
// The EVM events file can be queried by flow cli query, for instance:
//
// flow events get A.8c5303eaa26202d6.EVM.TransactionExecuted A.8c5303eaa26202d6.EVM.BlockExecuted
// --start 211176670 --end 211176770 --network testnet --host access-001.devnet51.nodes.onflow.org:9000
//
// After replaying with each event json file, it will generate a values_<height>.gob and
// allocators_<height>.gob files as checkpoint, such that when the checkpoint exists, it will loaded
// and skil replaying the coresponding event json files.

// backwardCompatibilityTestSinceEVMGenesisBlock ensures that the offchain package
// can read EVM events from the provided file paths, replay blocks starting from
// the EVM genesis block, and derive a consistent state matching the latest on-chain EVM state.
//
// The parameter `eventsFilePaths` is a list of file paths containing ordered EVM events in JSONL format.
// These EVM event files can be generated using the Flow CLI query command, for example:
//
// flow events get A.8c5303eaa26202d6.EVM.TransactionExecuted A.8c5303eaa26202d6.EVM.BlockExecuted
//
// --start 211176670 --end 211176770 --network testnet --host access-001.devnet51.nodes.onflow.org:9000
//
// During the replay process, it will generate `values_<height>.gob` and
// `allocators_<height>.gob` checkpoint files for each height. If these checkpoint files exist,
// the corresponding event JSON files will be skipped to optimize replay.
func BackwardCompatibleSinceEVMGenesisBlock(
t *testing.T,
chainID flow.ChainID,
eventsFilePaths []string, // ordered EVM events in JSONL format
checkpointDir string,
checkpointEndHeight uint64, // EVM height of an EVM state that a checkpoint was created for
) {
// ensure that checkpoints are not more than the event files
require.True(t, len(eventsFilePaths) > 0)

log.Info().Msgf("replaying EVM events from %v to %v, with checkpoints in %s, and checkpointEndHeight: %v",
eventsFilePaths[0], eventsFilePaths[len(eventsFilePaths)-1],
checkpointDir, checkpointEndHeight)

store, checkpointEndHeightOrZero := initStorageWithCheckpoints(t, chainID, checkpointDir, checkpointEndHeight)

// the events to replay
nextHeight := checkpointEndHeightOrZero + 1

// replay each event files
for _, eventsFilePath := range eventsFilePaths {
log.Info().Msgf("replaying events from %v, nextHeight: %v", eventsFilePath, nextHeight)

checkpointEndHeight := replayEvents(t, chainID, store, eventsFilePath, checkpointDir, nextHeight)
nextHeight = checkpointEndHeight + 1
}

log.Info().
Msgf("succhessfully replayed all events and state changes are consistent with onchain state change. nextHeight: %v", nextHeight)
}

func initStorageWithCheckpoints(t *testing.T, chainID flow.ChainID, checkpointDir string, checkpointEndHeight uint64) (
*TestValueStore, uint64,
) {
rootAddr := evm.StorageAccountAddress(chainID)

// if there is no checkpoint, create a empty store and initialize the account status,
// return 0 as the genesis height
if checkpointEndHeight == 0 {
store := GetSimpleValueStore()
as := environment.NewAccountStatus()
require.NoError(t, store.SetValue(rootAddr[:], []byte(flow.AccountStatusKey), as.ToBytes()))

return store, 0
}

valueFileName, allocatorFileName := checkpointFileNamesByEndHeight(checkpointDir, checkpointEndHeight)
values, err := deserialize(valueFileName)
require.NoError(t, err)
allocators, err := deserializeAllocator(allocatorFileName)
require.NoError(t, err)
store := GetSimpleValueStorePopulated(values, allocators)
return store, checkpointEndHeight
}

func replayEvents(
t *testing.T,
chainID flow.ChainID,
store *TestValueStore, eventsFilePath string, checkpointDir string, initialNextHeight uint64) uint64 {

rootAddr := evm.StorageAccountAddress(chainID)

bpStorage := storage.NewEphemeralStorage(store)
bp, err := blocks.NewBasicProvider(chainID, bpStorage, rootAddr)
require.NoError(t, err)

nextHeight := initialNextHeight

scanEventFilesAndRun(t, eventsFilePath,
func(blockEventPayload *events.BlockEventPayload, txEvents []events.TransactionEventPayload) error {
if blockEventPayload.Height != nextHeight {
return fmt.Errorf(
"expected height for next block event to be %v, but got %v",
nextHeight, blockEventPayload.Height)
}

err = bp.OnBlockReceived(blockEventPayload)
require.NoError(t, err)

sp := NewTestStorageProvider(store, blockEventPayload.Height)
cr := sync.NewReplayer(chainID, rootAddr, sp, bp, zerolog.Logger{}, nil, true)
res, err := cr.ReplayBlock(txEvents, blockEventPayload)
require.NoError(t, err)

// commit all changes
for k, v := range res.StorageRegisterUpdates() {
err = store.SetValue([]byte(k.Owner), []byte(k.Key), v)
require.NoError(t, err)
}

err = bp.OnBlockExecuted(blockEventPayload.Height, res)
require.NoError(t, err)

// commit all block hash list changes
for k, v := range bpStorage.StorageRegisterUpdates() {
err = store.SetValue([]byte(k.Owner), []byte(k.Key), v)
require.NoError(t, err)
}

// verify the block height is sequential without gap
nextHeight++

return nil
})

checkpointEndHeight := nextHeight - 1

log.Info().Msgf("finished replaying events from %v to %v, creating checkpoint", initialNextHeight, checkpointEndHeight)
valuesFile, allocatorsFile := dumpCheckpoint(t, store, checkpointDir, checkpointEndHeight)
log.Info().Msgf("checkpoint created: %v, %v", valuesFile, allocatorsFile)

return checkpointEndHeight
}

func checkpointFileNamesByEndHeight(dir string, endHeight uint64) (string, string) {
return filepath.Join(dir, fmt.Sprintf("values_%d.gob", endHeight)),
filepath.Join(dir, fmt.Sprintf("allocators_%d.gob", endHeight))
}

func dumpCheckpoint(t *testing.T, store *TestValueStore, dir string, checkpointEndHeight uint64) (string, string) {
valuesFileName, allocatorsFileName := checkpointFileNamesByEndHeight(dir, checkpointEndHeight)
values, allocators := store.Dump()

require.NoError(t, serialize(valuesFileName, values))
require.NoError(t, serializeAllocator(allocatorsFileName, allocators))
return valuesFileName, allocatorsFileName
}

const resume_height = 6559268

func decodeFullKey(encoded string) ([]byte, []byte, error) {
// Split the encoded string at the first occurrence of "~"
parts := strings.SplitN(encoded, "~", 2)
if len(parts) != 2 {
return nil, nil, fmt.Errorf("invalid encoded key: no delimiter found")
}

// Convert the split parts back to byte slices
owner := []byte(parts[0])
key := []byte(parts[1])
return owner, key, nil
}

type Subscription[T any] struct {
ch chan T
err error
}

func NewSubscription[T any]() *Subscription[T] {
return &Subscription[T]{
ch: make(chan T),
}
}

func (s *Subscription[T]) Channel() <-chan T {
return s.ch
}

func (s *Subscription[T]) Err() error {
return s.err
}

// scanEventFilesAndRun
func scanEventFilesAndRun(
t *testing.T,
filePath string,
handler func(*events.BlockEventPayload, []events.TransactionEventPayload) error,
) {
file, err := os.Open(filePath)
require.NoError(t, err)
defer file.Close()

scanner := bufio.NewScanner(file)

buf := make([]byte, 0, 64*1024)
scanner.Buffer(buf, 1024*1024)

txEvents := make([]events.TransactionEventPayload, 0)

for scanner.Scan() {
data := scanner.Bytes()
var e utils.Event
err := json.Unmarshal(data, &e)
require.NoError(t, err)
if strings.Contains(e.EventType, "BlockExecuted") {
temp, err := hex.DecodeString(e.EventPayload)
require.NoError(t, err)
ev, err := ccf.Decode(nil, temp)
require.NoError(t, err)
blockEventPayload, err := events.DecodeBlockEventPayload(ev.(cadence.Event))
require.NoError(t, err)

require.NoError(t, handler(blockEventPayload, txEvents), fmt.Sprintf("fail to handle block at height %d",
blockEventPayload.Height))

txEvents = make([]events.TransactionEventPayload, 0)
continue
}

temp, err := hex.DecodeString(e.EventPayload)
require.NoError(t, err)
ev, err := ccf.Decode(nil, temp)
require.NoError(t, err)
txEv, err := events.DecodeTransactionEventPayload(ev.(cadence.Event))
require.NoError(t, err)
txEvents = append(txEvents, *txEv)
}
if err := scanner.Err(); err != nil {
t.Fatal(err)
}
}

// Serialize function: saves map data to a file
func serialize(filename string, data map[string][]byte) error {
// Create a file to save data
file, err := os.Create(filename)
if err != nil {
return err
}
defer file.Close()

// Use gob to encode data
encoder := gob.NewEncoder(file)
err = encoder.Encode(data)
if err != nil {
return err
}

return nil
}

// Deserialize function: reads map data from a file
func deserialize(filename string) (map[string][]byte, error) {
// Open the file for reading
file, err := os.Open(filename)
if err != nil {
return nil, err
}
defer file.Close()

// Prepare the map to store decoded data
var data map[string][]byte

// Use gob to decode data
decoder := gob.NewDecoder(file)
err = decoder.Decode(&data)
if err != nil {
return nil, err
}

return data, nil
}

// Serialize function: saves map data to a file
func serializeAllocator(filename string, data map[string]uint64) error {
// Create a file to save data
file, err := os.Create(filename)
if err != nil {
return err
}
defer file.Close()

// Use gob to encode data
encoder := gob.NewEncoder(file)
err = encoder.Encode(data)
if err != nil {
return err
}

return nil
}

// Deserialize function: reads map data from a file
func deserializeAllocator(filename string) (map[string]uint64, error) {
// Open the file for reading
file, err := os.Open(filename)
if err != nil {
return nil, err
}
defer file.Close()

// Prepare the map to store decoded data
var data map[string]uint64

// Use gob to decode data
decoder := gob.NewDecoder(file)
err = decoder.Decode(&data)
if err != nil {
return nil, err
}

return data, nil
}
Loading

0 comments on commit dffadd8

Please sign in to comment.