|
|
- package main
-
- import (
- "bytes"
- "encoding/base64"
- "errors"
- "fmt"
- "os"
- "path/filepath"
- "strconv"
-
- "github.com/tendermint/tendermint/abci/example/code"
- abci "github.com/tendermint/tendermint/abci/types"
- "github.com/tendermint/tendermint/libs/log"
- "github.com/tendermint/tendermint/version"
- )
-
- // Application is an ABCI application for use by end-to-end tests. It is a
- // simple key/value store for strings, storing data in memory and persisting
- // to disk as JSON, taking state sync snapshots if requested.
- type Application struct {
- abci.BaseApplication
- logger log.Logger
- state *State
- snapshots *SnapshotStore
- cfg *Config
- restoreSnapshot *abci.Snapshot
- restoreChunks [][]byte
- }
-
- // NewApplication creates the application.
- func NewApplication(cfg *Config) (*Application, error) {
- state, err := NewState(filepath.Join(cfg.Dir, "state.json"), cfg.PersistInterval)
- if err != nil {
- return nil, err
- }
- snapshots, err := NewSnapshotStore(filepath.Join(cfg.Dir, "snapshots"))
- if err != nil {
- return nil, err
- }
- return &Application{
- logger: log.NewTMLogger(log.NewSyncWriter(os.Stdout)),
- state: state,
- snapshots: snapshots,
- cfg: cfg,
- }, nil
- }
-
- // Info implements ABCI.
- func (app *Application) Info(req abci.RequestInfo) abci.ResponseInfo {
- return abci.ResponseInfo{
- Version: version.ABCIVersion,
- AppVersion: 1,
- LastBlockHeight: int64(app.state.Height),
- LastBlockAppHash: app.state.Hash,
- }
- }
-
- // Info implements ABCI.
- func (app *Application) InitChain(req abci.RequestInitChain) abci.ResponseInitChain {
- var err error
- app.state.initialHeight = uint64(req.InitialHeight)
- if len(req.AppStateBytes) > 0 {
- err = app.state.Import(0, req.AppStateBytes)
- if err != nil {
- panic(err)
- }
- }
- resp := abci.ResponseInitChain{
- AppHash: app.state.Hash,
- }
- if resp.Validators, err = app.validatorUpdates(0); err != nil {
- panic(err)
- }
- return resp
- }
-
- // CheckTx implements ABCI.
- func (app *Application) CheckTx(req abci.RequestCheckTx) abci.ResponseCheckTx {
- _, _, err := parseTx(req.Tx)
- if err != nil {
- return abci.ResponseCheckTx{
- Code: code.CodeTypeEncodingError,
- Log: err.Error(),
- }
- }
- return abci.ResponseCheckTx{Code: code.CodeTypeOK, GasWanted: 1}
- }
-
- // DeliverTx implements ABCI.
- func (app *Application) DeliverTx(req abci.RequestDeliverTx) abci.ResponseDeliverTx {
- key, value, err := parseTx(req.Tx)
- if err != nil {
- panic(err) // shouldn't happen since we verified it in CheckTx
- }
- app.state.Set(key, value)
- return abci.ResponseDeliverTx{Code: code.CodeTypeOK}
- }
-
- // EndBlock implements ABCI.
- func (app *Application) EndBlock(req abci.RequestEndBlock) abci.ResponseEndBlock {
- valUpdates, err := app.validatorUpdates(uint64(req.Height))
- if err != nil {
- panic(err)
- }
-
- return abci.ResponseEndBlock{
- ValidatorUpdates: valUpdates,
- Events: []abci.Event{
- {
- Type: "val_updates",
- Attributes: []abci.EventAttribute{
- {
- Key: []byte("size"),
- Value: []byte(strconv.Itoa(valUpdates.Len())),
- },
- {
- Key: []byte("height"),
- Value: []byte(strconv.Itoa(int(req.Height))),
- },
- },
- },
- },
- }
- }
-
- // Commit implements ABCI.
- func (app *Application) Commit() abci.ResponseCommit {
- height, hash, err := app.state.Commit()
- if err != nil {
- panic(err)
- }
- if app.cfg.SnapshotInterval > 0 && height%app.cfg.SnapshotInterval == 0 {
- snapshot, err := app.snapshots.Create(app.state)
- if err != nil {
- panic(err)
- }
- logger.Info("Created state sync snapshot", "height", snapshot.Height)
- }
- retainHeight := int64(0)
- if app.cfg.RetainBlocks > 0 {
- retainHeight = int64(height - app.cfg.RetainBlocks + 1)
- }
- return abci.ResponseCommit{
- Data: hash,
- RetainHeight: retainHeight,
- }
- }
-
- // Query implements ABCI.
- func (app *Application) Query(req abci.RequestQuery) abci.ResponseQuery {
- return abci.ResponseQuery{
- Height: int64(app.state.Height),
- Key: req.Data,
- Value: []byte(app.state.Get(string(req.Data))),
- }
- }
-
- // ListSnapshots implements ABCI.
- func (app *Application) ListSnapshots(req abci.RequestListSnapshots) abci.ResponseListSnapshots {
- snapshots, err := app.snapshots.List()
- if err != nil {
- panic(err)
- }
- return abci.ResponseListSnapshots{Snapshots: snapshots}
- }
-
- // LoadSnapshotChunk implements ABCI.
- func (app *Application) LoadSnapshotChunk(req abci.RequestLoadSnapshotChunk) abci.ResponseLoadSnapshotChunk {
- chunk, err := app.snapshots.LoadChunk(req.Height, req.Format, req.Chunk)
- if err != nil {
- panic(err)
- }
- return abci.ResponseLoadSnapshotChunk{Chunk: chunk}
- }
-
- // OfferSnapshot implements ABCI.
- func (app *Application) OfferSnapshot(req abci.RequestOfferSnapshot) abci.ResponseOfferSnapshot {
- if app.restoreSnapshot != nil {
- panic("A snapshot is already being restored")
- }
- app.restoreSnapshot = req.Snapshot
- app.restoreChunks = [][]byte{}
- return abci.ResponseOfferSnapshot{Result: abci.ResponseOfferSnapshot_ACCEPT}
- }
-
- // ApplySnapshotChunk implements ABCI.
- func (app *Application) ApplySnapshotChunk(req abci.RequestApplySnapshotChunk) abci.ResponseApplySnapshotChunk {
- if app.restoreSnapshot == nil {
- panic("No restore in progress")
- }
- app.restoreChunks = append(app.restoreChunks, req.Chunk)
- if len(app.restoreChunks) == int(app.restoreSnapshot.Chunks) {
- bz := []byte{}
- for _, chunk := range app.restoreChunks {
- bz = append(bz, chunk...)
- }
- err := app.state.Import(app.restoreSnapshot.Height, bz)
- if err != nil {
- panic(err)
- }
- app.restoreSnapshot = nil
- app.restoreChunks = nil
- }
- return abci.ResponseApplySnapshotChunk{Result: abci.ResponseApplySnapshotChunk_ACCEPT}
- }
-
- // validatorUpdates generates a validator set update.
- func (app *Application) validatorUpdates(height uint64) (abci.ValidatorUpdates, error) {
- updates := app.cfg.ValidatorUpdates[fmt.Sprintf("%v", height)]
- if len(updates) == 0 {
- return nil, nil
- }
-
- valUpdates := abci.ValidatorUpdates{}
- for keyString, power := range updates {
-
- keyBytes, err := base64.StdEncoding.DecodeString(keyString)
- if err != nil {
- return nil, fmt.Errorf("invalid base64 pubkey value %q: %w", keyString, err)
- }
- valUpdates = append(valUpdates, abci.UpdateValidator(keyBytes, int64(power), app.cfg.KeyType))
- }
- return valUpdates, nil
- }
-
- // parseTx parses a tx in 'key=value' format into a key and value.
- func parseTx(tx []byte) (string, string, error) {
- parts := bytes.Split(tx, []byte("="))
- if len(parts) != 2 {
- return "", "", fmt.Errorf("invalid tx format: %q", string(tx))
- }
- if len(parts[0]) == 0 {
- return "", "", errors.New("key cannot be empty")
- }
- return string(parts[0]), string(parts[1]), nil
- }
|