Skip to content

Commit

Permalink
e2e: allow running of single node using the e2e app (backport) (#7024)
Browse files Browse the repository at this point in the history
Co-authored-by:    Callum Waters <[email protected]>
  • Loading branch information
tnasu and cmwaters committed Feb 10, 2022
1 parent ecd020e commit e52953e
Show file tree
Hide file tree
Showing 13 changed files with 128 additions and 17 deletions.
4 changes: 1 addition & 3 deletions cmd/ostracon/commands/run_node.go
Original file line number Diff line number Diff line change
Expand Up @@ -46,9 +46,7 @@ func AddNodeFlags(cmd *cobra.Command) {
"proxy_app",
config.ProxyApp,
"proxy app address, or one of: 'kvstore',"+
" 'persistent_kvstore',"+
" 'counter',"+
" 'counter_serial' or 'noop' for local testing.")
" 'persistent_kvstore', 'counter', 'e2e' or 'noop' for local testing.")
cmd.Flags().String("abci", config.ABCI, "specify abci transport (socket | grpc)")

// rpc flags
Expand Down
7 changes: 7 additions & 0 deletions proxy/client.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import (
"github.com/line/ostracon/abci/example/kvstore"
"github.com/line/ostracon/abci/types"
tmsync "github.com/line/ostracon/libs/sync"
e2e "github.com/line/ostracon/test/e2e/app"
)

// ClientCreator creates new ABCI clients.
Expand Down Expand Up @@ -79,6 +80,12 @@ func DefaultClientCreator(addr, transport, dbDir string) ClientCreator {
return NewLocalClientCreator(kvstore.NewApplication())
case "persistent_kvstore":
return NewLocalClientCreator(kvstore.NewPersistentKVStoreApplication(dbDir))
case "e2e":
app, err := e2e.NewApplication(e2e.DefaultConfig(dbDir))
if err != nil {
panic(err)
}
return NewLocalClientCreator(app)
case "noop":
return NewLocalClientCreator(types.NewBaseApplication())
default:
Expand Down
2 changes: 1 addition & 1 deletion rpc/openapi/openapi.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -335,7 +335,7 @@ paths:
https://godoc.org/github.com/line/ostracon/libs/pubsub/query.
```go
import rpchttp "github.com/tendermint/rpc/client/http"
import rpchttp "github.com/line/ostracon/rpc/client/http"
import "github.com/line/ostracon/types"
client := rpchttp.New("tcp:0.0.0.0:26657", "/websocket")
Expand Down
6 changes: 3 additions & 3 deletions test/e2e/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,8 @@ docker:
# We need to build support for database backends into the app in
# order to build a binary with an Ostracon node in it (for built-in
# ABCI testing).
app:
go build -o build/app -tags libsodium,badgerdb,boltdb,cleveldb,rocksdb ./app
node:
go build -o build/node -tags libsodium,badgerdb,boltdb,cleveldb,rocksdb ./node

# To be used primarily by the e2e docker instance. If you want to produce this binary
# elsewhere, then run go build in the maverick directory.
Expand All @@ -20,4 +20,4 @@ generator:
runner:
go build -o build/runner ./runner

.PHONY: all app docker generator maverick runner
.PHONY: all node docker generator maverick runner
34 changes: 33 additions & 1 deletion test/e2e/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -122,10 +122,42 @@ Docker does not enable IPv6 by default. To do so, enter the following in
}
```

## Benchmarking testnets
## Benchmarking Testnets

It is also possible to run a simple benchmark on a testnet. This is done through the `benchmark` command. This manages the entire process: setting up the environment, starting the test net, waiting for a considerable amount of blocks to be used (currently 100), and then returning the following metrics from the sample of the blockchain:

- Average time to produce a block
- Standard deviation of producing a block
- Minimum and maximum time to produce a block

## Running Individual Nodes

The E2E test harness is designed to run several nodes of varying configurations within docker. It is also possible to run a single node in the case of running larger, geographically-dispersed testnets. To run a single node you can either run:

**Built-in**

```bash
make node
ostracon init validator
OCHOME=$HOME/.ostracon ./build/node ./node/built-in.toml
```

To make things simpler the e2e application can also be run in the ostracon binary
by running

```bash
ostracon start --proxy-app e2e
```

However this won't offer the same level of configurability of the application.

**Socket**

```bash
make node
ostracon init validator
ostracon start
./build/node ./node.socket.toml
```

Check `node/config.go` to see how the settings of the test application can be tweaked.
51 changes: 50 additions & 1 deletion test/e2e/app/app.go
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
package main
package app

import (
"bytes"
Expand Down Expand Up @@ -31,6 +31,55 @@ type Application struct {
restoreChunks [][]byte
}

// Config allows for the setting of high level parameters for running the e2e Application
// KeyType and ValidatorUpdates must be the same for all nodes running the same application.
type Config struct {
// The directory with which state.json will be persisted in. Usually $HOME/.ostracon/data
Dir string `toml:"dir"`

// SnapshotInterval specifies the height interval at which the application
// will take state sync snapshots. Defaults to 0 (disabled).
SnapshotInterval uint64 `toml:"snapshot_interval"`

// RetainBlocks specifies the number of recent blocks to retain. Defaults to
// 0, which retains all blocks. Must be greater that PersistInterval,
// SnapshotInterval and EvidenceAgeHeight.
RetainBlocks uint64 `toml:"retain_blocks"`

// KeyType sets the curve that will be used by validators.
// Options are ed25519 & secp256k1
KeyType string `toml:"key_type"`

// PersistInterval specifies the height interval at which the application
// will persist state to disk. Defaults to 1 (every height), setting this to
// 0 disables state persistence.
PersistInterval uint64 `toml:"persist_interval"`

// ValidatorUpdates is a map of heights to validator names and their power,
// and will be returned by the ABCI application. For example, the following
// changes the power of validator01 and validator02 at height 1000:
//
// [validator_update.1000]
// validator01 = 20
// validator02 = 10
//
// Specifying height 0 returns the validator update during InitChain. The
// application returns the validator updates as-is, i.e. removing a
// validator must be done by returning it with power 0, and any validators
// not specified are not changed.
//
// height <-> pubkey <-> voting power
ValidatorUpdates map[string]map[string]uint8 `toml:"validator_update"`
}

func DefaultConfig(dir string) *Config {
return &Config{
PersistInterval: 1,
SnapshotInterval: 100,
Dir: dir,
}
}

// NewApplication creates the application.
func NewApplication(cfg *Config) (*Application, error) {
state, err := NewState(filepath.Join(cfg.Dir, "state.json"), cfg.PersistInterval)
Expand Down
2 changes: 1 addition & 1 deletion test/e2e/app/snapshots.go
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
// nolint: gosec
package main
package app

import (
"encoding/json"
Expand Down
2 changes: 1 addition & 1 deletion test/e2e/app/state.go
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
//nolint: gosec
package main
package app

import (
"crypto/sha256"
Expand Down
2 changes: 1 addition & 1 deletion test/e2e/docker/Dockerfile
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ COPY . .
RUN make build && cp build/ostracon /usr/bin/ostracon
COPY test/e2e/docker/entrypoint* /usr/bin/
RUN cd test/e2e && make maverick && cp build/maverick /usr/bin/maverick
RUN cd test/e2e && make app && cp build/app /usr/bin/app
RUN cd test/e2e && make node && cp build/node /usr/bin/app

# Set up runtime directory. We don't use a separate runtime image since we need
# e.g. leveldb and rocksdb which are already installed in the build image.
Expand Down
4 changes: 4 additions & 0 deletions test/e2e/node/built-in.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
snapshot_interval = 100
persist_interval = 1
chain_id = "test-chain"
protocol = "builtin"
15 changes: 15 additions & 0 deletions test/e2e/app/config.go → test/e2e/node/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@ import (
"fmt"

"github.com/BurntSushi/toml"

"github.com/line/ostracon/test/e2e/app"
)

// Config is the application configuration.
Expand All @@ -23,6 +25,19 @@ type Config struct {
PrivValKey string `toml:"privval_key"`
PrivValState string `toml:"privval_state"`
Misbehaviors map[string]string `toml:"misbehaviors"`
KeyType string `toml:"key_type"`
}

// App extracts out the application specific configuration parameters
func (cfg *Config) App() *app.Config {
return &app.Config{
Dir: cfg.Dir,
SnapshotInterval: cfg.SnapshotInterval,
RetainBlocks: cfg.RetainBlocks,
KeyType: cfg.KeyType,
ValidatorUpdates: cfg.ValidatorUpdates,
PersistInterval: cfg.PersistInterval,
}
}

// LoadConfig loads the configuration from disk.
Expand Down
11 changes: 6 additions & 5 deletions test/e2e/app/main.go → test/e2e/node/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ import (
"github.com/line/ostracon/privval"
"github.com/line/ostracon/proxy"
rpcserver "github.com/line/ostracon/rpc/jsonrpc/server"
"github.com/line/ostracon/test/e2e/app"
e2e "github.com/line/ostracon/test/e2e/pkg"
mcs "github.com/line/ostracon/test/maverick/consensus"
maverick "github.com/line/ostracon/test/maverick/node"
Expand Down Expand Up @@ -100,7 +101,7 @@ func run(configFile string) error {

// startApp starts the application server, listening for connections from Ostracon.
func startApp(cfg *Config) error {
app, err := NewApplication(cfg)
app, err := app.NewApplication(cfg.App())
if err != nil {
return err
}
Expand All @@ -121,7 +122,7 @@ func startApp(cfg *Config) error {
//
// FIXME There is no way to simply load the configuration from a file, so we need to pull in Viper.
func startNode(cfg *Config) error {
app, err := NewApplication(cfg)
app, err := app.NewApplication(cfg.App())
if err != nil {
return err
}
Expand Down Expand Up @@ -213,10 +214,10 @@ func startLightClient(cfg *Config) error {
}

// FIXME: Temporarily disconnected maverick until it is redesigned
// startMaverick starts a Maverick node that runs the application directly. It assumes the Tendermint
// configuration is in $TMHOME/config/tendermint.toml.
// startMaverick starts a Maverick node that runs the application directly. It assumes the Ostracon
// configuration is in $OCHOME/config/ostracon.toml.
func startMaverick(cfg *Config) error {
app, err := NewApplication(cfg)
app, err := app.NewApplication(cfg.App())
if err != nil {
return err
}
Expand Down
5 changes: 5 additions & 0 deletions test/e2e/node/socket.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
snapshot_interval = 100
persist_interval = 1
chain_id = "test-chain"
protocol = "socket"
listen = "tcp://127.0.0.1:26658"

0 comments on commit e52953e

Please sign in to comment.