Skip to content

Commit

Permalink
Merge pull request #313 from ethstorage/integratiointest
Browse files Browse the repository at this point in the history
Create action to run Integration test regularly
  • Loading branch information
ping-ke authored Sep 30, 2024
2 parents 85774d3 + 8d709fa commit 396dd13
Show file tree
Hide file tree
Showing 8 changed files with 583 additions and 5 deletions.
65 changes: 65 additions & 0 deletions .github/workflows/integration-test.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
name: Integration Test
on:
schedule:
- cron: '1 8 * * 0,2,4'

jobs:
es-node-integration-test:
runs-on: self-hosted
timeout-minutes: 2880

steps:
- uses: actions/checkout@v3

- name: Set up Go
uses: actions/setup-go@v4
with:
go-version: '1.21'

- name: Use Node.js
uses: actions/setup-node@v3
with:
node-version: '20.x'

- name: Install Foundry
uses: foundry-rs/foundry-toolchain@v1
with:
version: nightly

- name: Deploy Contracts
run: |
cd ../storage-contracts-v1
git checkout main
git pull
npm install
git submodule init && git submodule update
npx hardhat run scripts/deployL2-it.js --network qkc_testnet >> deploy.log
echo ES_NODE_CONTRACT_ADDRESS=`cat .caddr` >> "$GITHUB_ENV"
- name: Build and Run Bootnode Node
run: |
cd ../es-node
make
./run-l2-it-rpc.sh > es-node-it-bootnode.log&
- name: Set ENV Parameters
run: |
echo ES_NODE_UPLOADER_PRIVATE_KEY=`cat ../uploader.key` >> "$GITHUB_ENV"
echo ES_NODE_SIGNER_PRIVATE_KEY=`cat ../private.key` >> "$GITHUB_ENV"
echo ES_NODE_STORAGE_MINER=0x5C935469C5592Aeeac3372e922d9bCEabDF8830d >> "$GITHUB_ENV"
- name: Upload Blobs
run: |
cd ./integration_tests/scripts
npm install --force
node ituploader.js 10800 true > upload.log
cp .data ../../cmd/integration-test-server/.data
- name: Test
run: |
./run-l2-it.sh > es-node-it.log&
cd ./integration_tests/scripts
node ituploader.js 12288 false > upload2.log&
cd ../../cmd/integration-test-server
go build
./integration-test-server --contract_addr $ES_NODE_CONTRACT_ADDRESS > itserver.log
317 changes: 317 additions & 0 deletions cmd/integration-test-server/main.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,317 @@
// Copyright 2022-2023, EthStorage.
// For license information, see https://github.com/ethstorage/es-node/blob/main/LICENSE

package main

import (
"bufio"
"bytes"
"encoding/json"
"errors"
"flag"
"fmt"
"io"
"math"
"net/http"
"os"
"time"

"github.com/crate-crypto/go-proto-danksharding-crypto/eth"
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/common/hexutil"
"github.com/ethereum/go-ethereum/crypto/kzg4844"
"github.com/ethereum/go-ethereum/log"
"github.com/ethereum/go-ethereum/rpc"
"github.com/ethstorage/go-ethstorage/cmd/es-utils/utils"
es "github.com/ethstorage/go-ethstorage/ethstorage"
"github.com/ethstorage/go-ethstorage/ethstorage/node"
prv "github.com/ethstorage/go-ethstorage/ethstorage/prover"
)

const (
expectedSaidHelloTime = 10 * time.Minute
expectedStateRefreshTime = 5 * time.Minute
executionTime = 2 * time.Hour

kvEntries = 8192
kvSize = 32 * 4096
dataSize = 31 * 4096

rpcEndpoint = "http://127.0.0.1:9595"
uploadedDataFile = ".data"
shardFile0 = "../../es-data-it/shard-0.dat"
shardFile1 = "../../es-data-it/shard-1.dat"
)

var (
portFlag = flag.Int("port", 9096, "Listener port for the es-node to report node status")
contractAddr = flag.String("contract_addr", "", "EthStorage contract address")
)

var (
errorMessages = make([]string, 0)
lastQueryTime = time.Now()
lastRecord *node.NodeState
hasConnectedPeer = false
testLog = log.New("IntegrationTest")
prover = prv.NewKZGProver(testLog)
contractAddress = common.Address{}
)

func HelloHandler(w http.ResponseWriter, r *http.Request) {
body, err := io.ReadAll(r.Body)
if err != nil {
addErrorMessage(fmt.Sprintf("Read Hello body failed with error %s", err.Error()))
return
}
log.Info("Get hello from node", "id", string(body))

if time.Since(lastQueryTime) > expectedSaidHelloTime {
addErrorMessage(fmt.Sprintf("Get Hello message later then expect time %v real value %v", expectedSaidHelloTime, time.Since(lastQueryTime)))
}
lastQueryTime = time.Now()
log.Info("Get Hello request from es-node", "id", string(body))

answer := `{"status":"ok"}`
w.Write([]byte(answer))
}

func ReportStateHandler(w http.ResponseWriter, r *http.Request) {
if time.Since(lastQueryTime) > expectedStateRefreshTime*2 {
addErrorMessage(fmt.Sprintf("Get Hello message later then expect time %v real value %v",
expectedStateRefreshTime*2, time.Since(lastQueryTime)))
}
lastQueryTime = time.Now()

body, err := io.ReadAll(r.Body)
if err != nil {
addErrorMessage(fmt.Sprintf("Read ReportState body failed with error %s", err.Error()))
return
}

state := &node.NodeState{}
err = json.Unmarshal(body, state)
if err != nil {
addErrorMessage(fmt.Sprintf("Parse node state failed with error %s", err.Error()))
w.Write([]byte(fmt.Sprintf(`{"status":"error", "err message":"%s"}`, err.Error())))
return
}

// If no state updated
log.Info("Get state from peer", "peer id", state.Id, "Version", state.Version)
if lastRecord != nil {
checkState(lastRecord, state)
}

lastRecord = state

w.Write([]byte(`{"status":"ok"}`))
}

func checkState(oldState, newState *node.NodeState) {
if len(oldState.Shards) != len(newState.Shards) {
addErrorMessage(fmt.Sprintf("shards count mismatch between two state, new %d, old %d", len(newState.Shards), len(oldState.Shards)))
return
}

for _, shardState := range newState.Shards {
check := false
for _, oldShardState := range oldState.Shards {
if shardState.ShardId != oldShardState.ShardId {
continue
}
check = true
if shardState.SyncState.PeerCount > 0 {
hasConnectedPeer = true
}

if oldShardState.SyncState.SyncProgress < 10000 &&
(shardState.SyncState.BlobsSynced <= oldShardState.SyncState.BlobsSynced ||
shardState.SyncState.SyncProgress <= oldShardState.SyncState.SyncProgress) {
addErrorMessage(fmt.Sprintf("es-node sync progress do not increase in %f minutes, "+
"old synced: %d, new synced %d; old progress: %d, new progress: %d", expectedStateRefreshTime.Minutes(), oldShardState.SyncState.BlobsSynced,
shardState.SyncState.BlobsSynced, oldShardState.SyncState.SyncProgress, shardState.SyncState.SyncProgress))
}
if oldShardState.SyncState.FillEmptyProgress < 10000 &&
(shardState.SyncState.EmptyFilled < oldShardState.SyncState.EmptyFilled ||
shardState.SyncState.FillEmptyProgress < oldShardState.SyncState.FillEmptyProgress) {
addErrorMessage(fmt.Sprintf("es-node fill empty progress do not increase in %f minutes, "+
"old filled: %d, new filled %d; old progress: %d, new progress: %d", expectedStateRefreshTime.Minutes(), oldShardState.SyncState.EmptyFilled,
shardState.SyncState.EmptyFilled, oldShardState.SyncState.FillEmptyProgress, shardState.SyncState.FillEmptyProgress))
}

if oldShardState.SyncState.FillEmptyProgress == 10000 && oldShardState.SyncState.SyncProgress == 10000 &&
(shardState.MiningState.MiningPower == 0 || shardState.MiningState.SamplingTime == 0) {
addErrorMessage("Mining should be start after sync done.")
}
}
if !check {
addErrorMessage(fmt.Sprintf("Shard %d in the new state do not exist in the old state", shardState.ShardId))
}
}
}

func checkFinalState(state *node.NodeState) {
if state == nil {
addErrorMessage("No state submitted during the test")
return
}
if !hasConnectedPeer {
addErrorMessage("es-node peer count should larger than 0")
}

log.Info("Final state", "id", state.Id, "version", state.Version)
for _, shardState := range state.Shards {
if shardState.SyncState.SyncProgress != 10000 {
addErrorMessage("Sync should be finished during the test")
}
if shardState.SyncState.FillEmptyProgress != 10000 {
addErrorMessage("Fill should be finished during the test")
}
if shardState.MiningState.SamplingTime == 0 || shardState.MiningState.MiningPower == 0 {
addErrorMessage("Mining should be start after sync done.")
}
if shardState.SubmissionState.LastSucceededTime == 0 || shardState.SubmissionState.Succeeded == 0 {
addErrorMessage("At lease one block should be mined successfully during the test.")
}
if shardState.SubmissionState.Failed > 0 {
addErrorMessage(fmt.Sprintf("%d submission failed during the test.", shardState.SubmissionState.Failed))
}
log.Info("Final state", "id", state.Id, "shard", shardState.ShardId, "miner", shardState.Miner, "sync progress",
shardState.SyncState.SyncProgress, "fill progress", shardState.SyncState.FillEmptyProgress, "mining power",
shardState.MiningState.MiningPower, "sampling time", shardState.MiningState.SamplingTime, "succeeded submission",
shardState.SubmissionState.Succeeded, "failed submission", shardState.SubmissionState.Failed, "dropped submission",
shardState.SubmissionState.Dropped, "last succeeded time", shardState.SubmissionState.LastSucceededTime)
}
}

func createShardManager() (*es.ShardManager, error) {
sm := es.NewShardManager(contractAddress, kvSize, kvEntries, kvSize)
df0, err := es.OpenDataFile(shardFile0)
if err != nil {
return nil, err
}
err = sm.AddDataFileAndShard(df0)
if err != nil {
return nil, err
}

df1, err := es.OpenDataFile(shardFile1)
if err != nil {
return nil, err
}
err = sm.AddDataFileAndShard(df1)
if err != nil {
return nil, err
}

return sm, nil
}

func verifyData() error {
file, err := os.OpenFile(uploadedDataFile, os.O_RDONLY, 0755)
if err != nil {
return err
}
defer file.Close()

fileScanner := bufio.NewScanner(file)
fileScanner.Buffer(make([]byte, dataSize*2), kvSize*2)
fileScanner.Split(bufio.ScanLines)

sm, err := createShardManager()
if err != nil {
return err
}

client, err := rpc.DialHTTP(rpcEndpoint)
if err != nil {
return err
}
defer client.Close()

i := uint64(0)
for fileScanner.Scan() {
expectedData := common.Hex2Bytes(fileScanner.Text())
blob := utils.EncodeBlobs(expectedData)[0]
commit, _, _ := sm.TryReadMeta(i)
data, _, err := sm.TryRead(i, kvSize, common.BytesToHash(commit))
if err != nil {
return errors.New(fmt.Sprintf("read %d from shard fail with err: %s", i, err.Error()))
}
if bytes.Compare(blob[:], data) != 0 {
return errors.New(fmt.Sprintf("compare shard data %d fail, expected data %s; data: %s",
i, common.Bytes2Hex(blob[:64]), common.Bytes2Hex(data[:64])))
}

rpcdata, err := downloadBlobFromRPC(client, i, common.BytesToHash(commit))
if err != nil {
return errors.New(fmt.Sprintf("get data %d from rpc fail with err: %s", i, err.Error()))
}
if bytes.Compare(blob[:], rpcdata) != 0 {
return errors.New(fmt.Sprintf("compare rpc data %d fail, expected data %s; data: %s",
i, common.Bytes2Hex(blob[:64]), common.Bytes2Hex(rpcdata[:64])))
}
i++
}
return nil
}

func downloadBlobFromRPC(client *rpc.Client, kvIndex uint64, hash common.Hash) ([]byte, error) {
var result hexutil.Bytes
err := client.Call(&result, "es_getBlob", kvIndex, hash, 0, 0, 4096*32)
if err != nil {
return nil, err
}

var blob kzg4844.Blob
copy(blob[:], result)
commit, err := kzg4844.BlobToCommitment(blob)
if err != nil {
return nil, fmt.Errorf("blobToCommitment failed: %w", err)
}
cmt := common.Hash(eth.KZGToVersionedHash(commit))
if bytes.Compare(cmt[:es.HashSizeInContract], hash[:es.HashSizeInContract]) != 0 {
return nil, fmt.Errorf("invalid blob for %d hash: %s, commit: %s", kvIndex, hash, cmt)
}

return result, nil
}

func addErrorMessage(errMessage string) {
log.Warn("Add error message", "msg", errMessage)
errorMessages = append(errorMessages, errMessage+"\n")
}

func listenAndServe(port int) error {
http.HandleFunc("/hello", HelloHandler)
http.HandleFunc("/reportstate", ReportStateHandler)
return http.ListenAndServe(fmt.Sprintf(":%d", port), nil)
}

func main() {
// Parse the flags and set up the logger to print everything requested
flag.Parse()
log.Root().SetHandler(log.LvlFilterHandler(log.Lvl(3), log.StreamHandler(os.Stderr, log.TerminalFormat(true))))

if *portFlag < 0 || *portFlag > math.MaxUint16 {
log.Crit("Invalid port")
}
if *contractAddr == "" {
log.Crit("Invalid contract address")
} else {
contractAddress = common.HexToAddress(*contractAddr)
}

go listenAndServe(*portFlag)

time.Sleep(executionTime)
checkFinalState(lastRecord)
if err := verifyData(); err != nil {
addErrorMessage(err.Error())
}

if len(errorMessages) > 0 {
log.Crit(fmt.Sprintf("integration test fail %v", errorMessages))
}
}
3 changes: 0 additions & 3 deletions ethstorage/p2p/protocol/syncclient.go
Original file line number Diff line number Diff line change
Expand Up @@ -895,12 +895,9 @@ func (s *SyncClient) assignFillEmptyBlobTasks() {
s.notifyUpdate()
s.wg.Done()
}()
t := time.Now()
next, err := s.FillFileWithEmptyBlob(start, limit)
if err != nil {
log.Warn("Fill in empty fail", "err", err.Error())
} else {
log.Debug("Fill in empty done", "time", time.Now().Sub(t).Seconds())
}
filled := next - start

Expand Down
Loading

0 comments on commit 396dd13

Please sign in to comment.