2020-07-09 05:21:20 -07:00
|
|
|
package network
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
2020-09-23 06:44:40 -07:00
|
|
|
"context"
|
2020-06-26 09:30:49 -07:00
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
2021-01-18 10:25:22 -08:00
|
|
|
"net/http"
|
2020-06-26 09:30:49 -07:00
|
|
|
"net/url"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2021-06-29 03:41:55 -07:00
|
|
|
"strings"
|
2020-06-26 09:30:49 -07:00
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2021-11-16 11:24:38 -08:00
|
|
|
"github.com/rs/zerolog"
|
2021-06-29 03:41:55 -07:00
|
|
|
"github.com/spf13/cobra"
|
2021-11-16 11:24:38 -08:00
|
|
|
"github.com/tendermint/tendermint/config"
|
2020-06-26 09:30:49 -07:00
|
|
|
tmrand "github.com/tendermint/tendermint/libs/rand"
|
2021-11-16 11:24:38 -08:00
|
|
|
"github.com/tendermint/tendermint/libs/service"
|
2020-06-26 09:30:49 -07:00
|
|
|
tmclient "github.com/tendermint/tendermint/rpc/client"
|
|
|
|
dbm "github.com/tendermint/tm-db"
|
2020-07-27 10:57:15 -07:00
|
|
|
"google.golang.org/grpc"
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
"github.com/cosmos/cosmos-sdk/baseapp"
|
|
|
|
"github.com/cosmos/cosmos-sdk/client"
|
2020-07-29 15:33:42 -07:00
|
|
|
"github.com/cosmos/cosmos-sdk/client/tx"
|
2020-06-30 13:59:21 -07:00
|
|
|
"github.com/cosmos/cosmos-sdk/codec"
|
2020-08-13 07:22:16 -07:00
|
|
|
codectypes "github.com/cosmos/cosmos-sdk/codec/types"
|
2020-08-12 01:34:10 -07:00
|
|
|
"github.com/cosmos/cosmos-sdk/crypto/hd"
|
2020-06-26 09:30:49 -07:00
|
|
|
"github.com/cosmos/cosmos-sdk/crypto/keyring"
|
2020-11-09 08:01:43 -08:00
|
|
|
cryptotypes "github.com/cosmos/cosmos-sdk/crypto/types"
|
2020-06-26 09:30:49 -07:00
|
|
|
"github.com/cosmos/cosmos-sdk/server"
|
|
|
|
"github.com/cosmos/cosmos-sdk/server/api"
|
|
|
|
srvconfig "github.com/cosmos/cosmos-sdk/server/config"
|
2020-07-27 10:57:15 -07:00
|
|
|
servertypes "github.com/cosmos/cosmos-sdk/server/types"
|
2020-06-26 09:30:49 -07:00
|
|
|
"github.com/cosmos/cosmos-sdk/simapp"
|
2020-10-08 03:44:11 -07:00
|
|
|
"github.com/cosmos/cosmos-sdk/simapp/params"
|
2020-06-26 09:30:49 -07:00
|
|
|
storetypes "github.com/cosmos/cosmos-sdk/store/types"
|
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
|
|
authtypes "github.com/cosmos/cosmos-sdk/x/auth/types"
|
|
|
|
banktypes "github.com/cosmos/cosmos-sdk/x/bank/types"
|
|
|
|
"github.com/cosmos/cosmos-sdk/x/genutil"
|
|
|
|
stakingtypes "github.com/cosmos/cosmos-sdk/x/staking/types"
|
|
|
|
)
|
|
|
|
|
2020-06-30 13:59:21 -07:00
|
|
|
// package-wide network lock to only allow one test network at a time
|
|
|
|
var lock = new(sync.Mutex)
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
// AppConstructor defines a function which accepts a network configuration and
|
|
|
|
// creates an ABCI Application to provide to Tendermint.
|
2020-07-27 10:57:15 -07:00
|
|
|
type AppConstructor = func(val Validator) servertypes.Application
|
2020-06-26 09:30:49 -07:00
|
|
|
|
2020-10-08 03:44:11 -07:00
|
|
|
// NewAppConstructor returns a new simapp AppConstructor
|
|
|
|
func NewAppConstructor(encodingCfg params.EncodingConfig) AppConstructor {
|
|
|
|
return func(val Validator) servertypes.Application {
|
|
|
|
return simapp.NewSimApp(
|
|
|
|
val.Ctx.Logger, dbm.NewMemDB(), nil, true, make(map[int64]bool), val.Ctx.Config.RootDir, 0,
|
|
|
|
encodingCfg,
|
2020-11-02 11:10:14 -08:00
|
|
|
simapp.EmptyAppOptions{},
|
2020-10-08 03:44:11 -07:00
|
|
|
baseapp.SetPruning(storetypes.NewPruningOptionsFromString(val.AppConfig.Pruning)),
|
|
|
|
baseapp.SetMinGasPrices(val.AppConfig.MinGasPrices),
|
|
|
|
)
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Config defines the necessary configuration used to bootstrap and start an
|
|
|
|
// in-process local testing network.
|
|
|
|
type Config struct {
|
2021-04-29 03:46:22 -07:00
|
|
|
Codec codec.Codec
|
2020-08-13 07:22:16 -07:00
|
|
|
LegacyAmino *codec.LegacyAmino // TODO: Remove!
|
|
|
|
InterfaceRegistry codectypes.InterfaceRegistry
|
2020-08-17 05:47:31 -07:00
|
|
|
|
|
|
|
TxConfig client.TxConfig
|
|
|
|
AccountRetriever client.AccountRetriever
|
|
|
|
AppConstructor AppConstructor // the ABCI application constructor
|
|
|
|
GenesisState map[string]json.RawMessage // custom gensis state to provide
|
|
|
|
TimeoutCommit time.Duration // the consensus commitment timeout
|
|
|
|
ChainID string // the network chain-id
|
|
|
|
NumValidators int // the total number of validators to create and bond
|
|
|
|
BondDenom string // the staking bond denomination
|
|
|
|
MinGasPrices string // the minimum gas prices each validator will accept
|
|
|
|
AccountTokens sdk.Int // the amount of unique validator tokens (e.g. 1000node0)
|
|
|
|
StakingTokens sdk.Int // the amount of tokens each validator has available to stake
|
|
|
|
BondedTokens sdk.Int // the amount of tokens each validator stakes
|
|
|
|
PruningStrategy string // the pruning strategy each validator will have
|
2021-06-29 03:41:55 -07:00
|
|
|
EnableTMLogging bool // enable Tendermint logging to STDOUT
|
2020-08-17 05:47:31 -07:00
|
|
|
CleanupDir bool // remove base temporary directory during cleanup
|
|
|
|
SigningAlgo string // signing algorithm for keys
|
2021-06-29 03:41:55 -07:00
|
|
|
KeyringOptions []keyring.Option // keyring configuration options
|
|
|
|
RPCAddress string // RPC listen address (including port)
|
|
|
|
APIAddress string // REST API listen address (including port)
|
|
|
|
GRPCAddress string // GRPC server listen address (including port)
|
|
|
|
PrintMnemonic bool // print the mnemonic of first validator as log output for testing
|
2020-06-26 09:30:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// DefaultConfig returns a sane default configuration suitable for nearly all
|
|
|
|
// testing requirements.
|
|
|
|
func DefaultConfig() Config {
|
2020-10-27 04:33:48 -07:00
|
|
|
encCfg := simapp.MakeTestEncodingConfig()
|
2020-06-30 13:59:21 -07:00
|
|
|
|
2020-06-26 09:30:49 -07:00
|
|
|
return Config{
|
2021-07-01 01:52:38 -07:00
|
|
|
Codec: encCfg.Codec,
|
2020-08-13 07:22:16 -07:00
|
|
|
TxConfig: encCfg.TxConfig,
|
|
|
|
LegacyAmino: encCfg.Amino,
|
|
|
|
InterfaceRegistry: encCfg.InterfaceRegistry,
|
|
|
|
AccountRetriever: authtypes.AccountRetriever{},
|
2020-10-08 03:44:11 -07:00
|
|
|
AppConstructor: NewAppConstructor(encCfg),
|
2021-07-01 01:52:38 -07:00
|
|
|
GenesisState: simapp.ModuleBasics.DefaultGenesis(encCfg.Codec),
|
2020-08-13 07:22:16 -07:00
|
|
|
TimeoutCommit: 2 * time.Second,
|
2021-11-16 11:24:38 -08:00
|
|
|
ChainID: "chain-" + tmrand.Str(6),
|
2020-08-13 07:22:16 -07:00
|
|
|
NumValidators: 4,
|
|
|
|
BondDenom: sdk.DefaultBondDenom,
|
|
|
|
MinGasPrices: fmt.Sprintf("0.000006%s", sdk.DefaultBondDenom),
|
2021-01-27 02:24:00 -08:00
|
|
|
AccountTokens: sdk.TokensFromConsensusPower(1000, sdk.DefaultPowerReduction),
|
|
|
|
StakingTokens: sdk.TokensFromConsensusPower(500, sdk.DefaultPowerReduction),
|
|
|
|
BondedTokens: sdk.TokensFromConsensusPower(100, sdk.DefaultPowerReduction),
|
2020-08-13 07:22:16 -07:00
|
|
|
PruningStrategy: storetypes.PruningOptionNothing,
|
|
|
|
CleanupDir: true,
|
|
|
|
SigningAlgo: string(hd.Secp256k1Type),
|
|
|
|
KeyringOptions: []keyring.Option{},
|
2021-06-29 03:41:55 -07:00
|
|
|
PrintMnemonic: false,
|
2020-06-26 09:30:49 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type (
|
|
|
|
// Network defines a local in-process testing network using SimApp. It can be
|
|
|
|
// configured to start any number of validators, each with its own RPC and API
|
|
|
|
// clients. Typically, this test network would be used in client and integration
|
|
|
|
// testing where user input is expected.
|
|
|
|
//
|
|
|
|
// Note, due to Tendermint constraints in regards to RPC functionality, there
|
|
|
|
// may only be one test network running at a time. Thus, any caller must be
|
|
|
|
// sure to Cleanup after testing is finished in order to allow other tests
|
|
|
|
// to create networks. In addition, only the first validator will have a valid
|
|
|
|
// RPC and API server/client.
|
|
|
|
Network struct {
|
2021-06-29 03:41:55 -07:00
|
|
|
Logger Logger
|
2020-06-26 09:30:49 -07:00
|
|
|
BaseDir string
|
|
|
|
Validators []*Validator
|
|
|
|
|
2020-07-27 10:57:15 -07:00
|
|
|
Config Config
|
2020-06-26 09:30:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Validator defines an in-process Tendermint validator node. Through this object,
|
|
|
|
// a client can make RPC and API calls and interact with any client command
|
|
|
|
// or handler.
|
|
|
|
Validator struct {
|
|
|
|
AppConfig *srvconfig.Config
|
|
|
|
ClientCtx client.Context
|
|
|
|
Ctx *server.Context
|
|
|
|
Dir string
|
|
|
|
NodeID string
|
2020-11-09 08:01:43 -08:00
|
|
|
PubKey cryptotypes.PubKey
|
2020-06-26 09:30:49 -07:00
|
|
|
Moniker string
|
|
|
|
APIAddress string
|
|
|
|
RPCAddress string
|
|
|
|
P2PAddress string
|
|
|
|
Address sdk.AccAddress
|
|
|
|
ValAddress sdk.ValAddress
|
|
|
|
RPCClient tmclient.Client
|
|
|
|
|
2021-11-16 11:24:38 -08:00
|
|
|
tmNode service.Service
|
2021-01-18 10:25:22 -08:00
|
|
|
api *api.Server
|
|
|
|
grpc *grpc.Server
|
|
|
|
grpcWeb *http.Server
|
2020-06-26 09:30:49 -07:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2021-06-29 03:41:55 -07:00
|
|
|
// Logger is a network logger interface that exposes testnet-level Log() methods for an in-process testing network
|
|
|
|
// This is not to be confused with logging that may happen at an individual node or validator level
|
|
|
|
type Logger interface {
|
|
|
|
Log(args ...interface{})
|
|
|
|
Logf(format string, args ...interface{})
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ Logger = (*testing.T)(nil)
|
|
|
|
var _ Logger = (*CLILogger)(nil)
|
|
|
|
|
|
|
|
type CLILogger struct {
|
|
|
|
cmd *cobra.Command
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s CLILogger) Log(args ...interface{}) {
|
|
|
|
s.cmd.Println(args...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s CLILogger) Logf(format string, args ...interface{}) {
|
|
|
|
s.cmd.Printf(format, args...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewCLILogger(cmd *cobra.Command) CLILogger {
|
|
|
|
return CLILogger{cmd}
|
|
|
|
}
|
|
|
|
|
|
|
|
// New creates a new Network for integration tests or in-process testnets run via the CLI
|
|
|
|
func New(l Logger, baseDir string, cfg Config) (*Network, error) {
|
2020-06-26 09:30:49 -07:00
|
|
|
// only one caller/test can create and use a network at a time
|
2021-06-29 03:41:55 -07:00
|
|
|
l.Log("acquiring test network lock")
|
2020-06-26 09:30:49 -07:00
|
|
|
lock.Lock()
|
|
|
|
|
|
|
|
network := &Network{
|
2021-06-29 03:41:55 -07:00
|
|
|
Logger: l,
|
2020-06-26 09:30:49 -07:00
|
|
|
BaseDir: baseDir,
|
|
|
|
Validators: make([]*Validator, cfg.NumValidators),
|
2020-07-27 10:57:15 -07:00
|
|
|
Config: cfg,
|
2020-06-26 09:30:49 -07:00
|
|
|
}
|
|
|
|
|
2021-06-29 03:41:55 -07:00
|
|
|
l.Logf("preparing test network with chain-id \"%s\"\n", cfg.ChainID)
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
monikers := make([]string, cfg.NumValidators)
|
|
|
|
nodeIDs := make([]string, cfg.NumValidators)
|
2020-11-09 08:01:43 -08:00
|
|
|
valPubKeys := make([]cryptotypes.PubKey, cfg.NumValidators)
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
var (
|
|
|
|
genAccounts []authtypes.GenesisAccount
|
|
|
|
genBalances []banktypes.Balance
|
|
|
|
genFiles []string
|
|
|
|
)
|
|
|
|
|
|
|
|
buf := bufio.NewReader(os.Stdin)
|
|
|
|
|
|
|
|
// generate private keys, node IDs, and initial transactions
|
|
|
|
for i := 0; i < cfg.NumValidators; i++ {
|
|
|
|
appCfg := srvconfig.DefaultConfig()
|
|
|
|
appCfg.Pruning = cfg.PruningStrategy
|
|
|
|
appCfg.MinGasPrices = cfg.MinGasPrices
|
|
|
|
appCfg.API.Enable = true
|
|
|
|
appCfg.API.Swagger = false
|
|
|
|
appCfg.Telemetry.Enabled = false
|
|
|
|
|
|
|
|
ctx := server.NewDefaultContext()
|
|
|
|
tmCfg := ctx.Config
|
|
|
|
tmCfg.Consensus.TimeoutCommit = cfg.TimeoutCommit
|
2021-11-16 11:24:38 -08:00
|
|
|
tmCfg.Mode = config.ModeValidator
|
2020-06-26 09:30:49 -07:00
|
|
|
|
2020-08-05 08:45:22 -07:00
|
|
|
// Only allow the first validator to expose an RPC, API and gRPC
|
|
|
|
// server/client due to Tendermint in-process constraints.
|
2020-06-26 09:30:49 -07:00
|
|
|
apiAddr := ""
|
|
|
|
tmCfg.RPC.ListenAddress = ""
|
2020-08-05 08:45:22 -07:00
|
|
|
appCfg.GRPC.Enable = false
|
2021-01-18 10:25:22 -08:00
|
|
|
appCfg.GRPCWeb.Enable = false
|
2021-06-29 03:41:55 -07:00
|
|
|
apiListenAddr := ""
|
2020-06-26 09:30:49 -07:00
|
|
|
if i == 0 {
|
2021-06-29 03:41:55 -07:00
|
|
|
if cfg.APIAddress != "" {
|
|
|
|
apiListenAddr = cfg.APIAddress
|
|
|
|
} else {
|
|
|
|
var err error
|
|
|
|
apiListenAddr, _, err = server.FreeTCPAddr()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
|
2021-06-29 03:41:55 -07:00
|
|
|
appCfg.API.Address = apiListenAddr
|
2020-06-26 09:30:49 -07:00
|
|
|
apiURL, err := url.Parse(apiListenAddr)
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
apiAddr = fmt.Sprintf("http://%s:%s", apiURL.Hostname(), apiURL.Port())
|
|
|
|
|
2021-06-29 03:41:55 -07:00
|
|
|
if cfg.RPCAddress != "" {
|
|
|
|
tmCfg.RPC.ListenAddress = cfg.RPCAddress
|
|
|
|
} else {
|
|
|
|
rpcAddr, _, err := server.FreeTCPAddr()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
tmCfg.RPC.ListenAddress = rpcAddr
|
|
|
|
}
|
2020-07-27 10:57:15 -07:00
|
|
|
|
2021-06-29 03:41:55 -07:00
|
|
|
if cfg.GRPCAddress != "" {
|
|
|
|
appCfg.GRPC.Address = cfg.GRPCAddress
|
|
|
|
} else {
|
|
|
|
_, grpcPort, err := server.FreeTCPAddr()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
appCfg.GRPC.Address = fmt.Sprintf("0.0.0.0:%s", grpcPort)
|
|
|
|
}
|
2020-07-27 10:57:15 -07:00
|
|
|
appCfg.GRPC.Enable = true
|
2021-01-18 10:25:22 -08:00
|
|
|
|
|
|
|
_, grpcWebPort, err := server.FreeTCPAddr()
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-01-18 10:25:22 -08:00
|
|
|
appCfg.GRPCWeb.Address = fmt.Sprintf("0.0.0.0:%s", grpcWebPort)
|
|
|
|
appCfg.GRPCWeb.Enable = true
|
2020-06-26 09:30:49 -07:00
|
|
|
}
|
|
|
|
|
2021-11-16 11:24:38 -08:00
|
|
|
logger := server.ZeroLogWrapper{Logger: zerolog.Nop()}
|
2021-06-29 03:41:55 -07:00
|
|
|
if cfg.EnableTMLogging {
|
2021-11-16 11:24:38 -08:00
|
|
|
logWriter := zerolog.ConsoleWriter{Out: os.Stderr}
|
|
|
|
logger = server.ZeroLogWrapper{Logger: zerolog.New(logWriter).Level(zerolog.InfoLevel).With().Timestamp().Logger()}
|
2020-06-26 09:30:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ctx.Logger = logger
|
|
|
|
|
|
|
|
nodeDirName := fmt.Sprintf("node%d", i)
|
|
|
|
nodeDir := filepath.Join(network.BaseDir, nodeDirName, "simd")
|
|
|
|
clientDir := filepath.Join(network.BaseDir, nodeDirName, "simcli")
|
|
|
|
gentxsDir := filepath.Join(network.BaseDir, "gentxs")
|
|
|
|
|
2021-06-29 03:41:55 -07:00
|
|
|
err := os.MkdirAll(filepath.Join(nodeDir, "config"), 0755)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = os.MkdirAll(clientDir, 0755)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
tmCfg.SetRoot(nodeDir)
|
|
|
|
tmCfg.Moniker = nodeDirName
|
|
|
|
monikers[i] = nodeDirName
|
|
|
|
|
|
|
|
proxyAddr, _, err := server.FreeTCPAddr()
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
tmCfg.ProxyApp = proxyAddr
|
|
|
|
|
|
|
|
p2pAddr, _, err := server.FreeTCPAddr()
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
tmCfg.P2P.ListenAddress = p2pAddr
|
|
|
|
tmCfg.P2P.AddrBookStrict = false
|
|
|
|
tmCfg.P2P.AllowDuplicateIP = true
|
|
|
|
|
|
|
|
nodeID, pubKey, err := genutil.InitializeNodeValidatorFiles(tmCfg)
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
nodeIDs[i] = nodeID
|
|
|
|
valPubKeys[i] = pubKey
|
|
|
|
|
2021-09-20 05:02:15 -07:00
|
|
|
kb, err := keyring.New(sdk.KeyringServiceName(), keyring.BackendTest, clientDir, buf, cfg.Codec, cfg.KeyringOptions...)
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
|
2020-08-12 01:34:10 -07:00
|
|
|
keyringAlgos, _ := kb.SupportedAlgorithms()
|
|
|
|
algo, err := keyring.NewSigningAlgoFromString(cfg.SigningAlgo, keyringAlgos)
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-08-12 01:34:10 -07:00
|
|
|
|
|
|
|
addr, secret, err := server.GenerateSaveCoinKey(kb, nodeDirName, true, algo)
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// if PrintMnemonic is set to true, we print the first validator node's secret to the network's logger
|
|
|
|
// for debugging and manual testing
|
|
|
|
if cfg.PrintMnemonic && i == 0 {
|
|
|
|
printMnemonic(l, secret)
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
info := map[string]string{"secret": secret}
|
|
|
|
infoBz, err := json.Marshal(info)
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
// save private key seed words
|
2021-06-29 03:41:55 -07:00
|
|
|
err = writeFile(fmt.Sprintf("%v.json", "key_seed"), clientDir, infoBz)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
balances := sdk.NewCoins(
|
|
|
|
sdk.NewCoin(fmt.Sprintf("%stoken", nodeDirName), cfg.AccountTokens),
|
|
|
|
sdk.NewCoin(cfg.BondDenom, cfg.StakingTokens),
|
|
|
|
)
|
|
|
|
|
|
|
|
genFiles = append(genFiles, tmCfg.GenesisFile())
|
2020-09-25 03:25:37 -07:00
|
|
|
genBalances = append(genBalances, banktypes.Balance{Address: addr.String(), Coins: balances.Sort()})
|
2020-06-26 09:30:49 -07:00
|
|
|
genAccounts = append(genAccounts, authtypes.NewBaseAccount(addr, nil, 0, 0))
|
|
|
|
|
2020-07-10 07:45:46 -07:00
|
|
|
commission, err := sdk.NewDecFromStr("0.5")
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-07-10 07:45:46 -07:00
|
|
|
|
2020-10-19 06:04:55 -07:00
|
|
|
createValMsg, err := stakingtypes.NewMsgCreateValidator(
|
2020-06-26 09:30:49 -07:00
|
|
|
sdk.ValAddress(addr),
|
|
|
|
valPubKeys[i],
|
2020-12-24 06:44:05 -08:00
|
|
|
sdk.NewCoin(cfg.BondDenom, cfg.BondedTokens),
|
2020-06-26 09:30:49 -07:00
|
|
|
stakingtypes.NewDescription(nodeDirName, "", "", "", ""),
|
2020-07-10 07:45:46 -07:00
|
|
|
stakingtypes.NewCommissionRates(commission, sdk.OneDec(), sdk.OneDec()),
|
2020-06-26 09:30:49 -07:00
|
|
|
sdk.OneInt(),
|
|
|
|
)
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
p2pURL, err := url.Parse(p2pAddr)
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
memo := fmt.Sprintf("%s@%s:%s", nodeIDs[i], p2pURL.Hostname(), p2pURL.Port())
|
2020-09-03 03:11:46 -07:00
|
|
|
fee := sdk.NewCoins(sdk.NewCoin(fmt.Sprintf("%stoken", nodeDirName), sdk.NewInt(0)))
|
2020-07-25 01:10:04 -07:00
|
|
|
txBuilder := cfg.TxConfig.NewTxBuilder()
|
2021-06-29 03:41:55 -07:00
|
|
|
err = txBuilder.SetMsgs(createValMsg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-09-03 03:11:46 -07:00
|
|
|
txBuilder.SetFeeAmount(fee) // Arbitrary fee
|
|
|
|
txBuilder.SetGasLimit(1000000) // Need at least 100386
|
2020-07-25 01:10:04 -07:00
|
|
|
txBuilder.SetMemo(memo)
|
|
|
|
|
|
|
|
txFactory := tx.Factory{}
|
|
|
|
txFactory = txFactory.
|
2020-06-26 09:30:49 -07:00
|
|
|
WithChainID(cfg.ChainID).
|
|
|
|
WithMemo(memo).
|
2020-07-25 01:10:04 -07:00
|
|
|
WithKeybase(kb).
|
|
|
|
WithTxConfig(cfg.TxConfig)
|
2020-06-26 09:30:49 -07:00
|
|
|
|
2020-12-14 13:44:15 -08:00
|
|
|
err = tx.Sign(txFactory, nodeDirName, txBuilder, true)
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
|
2020-07-25 01:10:04 -07:00
|
|
|
txBz, err := cfg.TxConfig.TxJSONEncoder()(txBuilder.GetTx())
|
2021-06-29 03:41:55 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
err = writeFile(fmt.Sprintf("%v.json", nodeDirName), gentxsDir, txBz)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-11-16 11:24:38 -08:00
|
|
|
srvconfig.WriteConfigFile(filepath.Join(nodeDir, "config", "app.toml"), appCfg)
|
2020-06-26 09:30:49 -07:00
|
|
|
|
2020-06-30 13:59:21 -07:00
|
|
|
clientCtx := client.Context{}.
|
2021-03-31 03:04:33 -07:00
|
|
|
WithKeyringDir(clientDir).
|
2020-06-30 13:59:21 -07:00
|
|
|
WithKeyring(kb).
|
|
|
|
WithHomeDir(tmCfg.RootDir).
|
|
|
|
WithChainID(cfg.ChainID).
|
2020-08-17 05:47:31 -07:00
|
|
|
WithInterfaceRegistry(cfg.InterfaceRegistry).
|
2021-06-11 04:49:39 -07:00
|
|
|
WithCodec(cfg.Codec).
|
2020-08-10 12:41:21 -07:00
|
|
|
WithLegacyAmino(cfg.LegacyAmino).
|
2020-07-20 05:30:12 -07:00
|
|
|
WithTxConfig(cfg.TxConfig).
|
2020-08-17 05:47:31 -07:00
|
|
|
WithAccountRetriever(cfg.AccountRetriever)
|
2020-06-30 13:59:21 -07:00
|
|
|
|
2020-06-26 09:30:49 -07:00
|
|
|
network.Validators[i] = &Validator{
|
|
|
|
AppConfig: appCfg,
|
2020-06-30 13:59:21 -07:00
|
|
|
ClientCtx: clientCtx,
|
2020-06-26 09:30:49 -07:00
|
|
|
Ctx: ctx,
|
|
|
|
Dir: filepath.Join(network.BaseDir, nodeDirName),
|
|
|
|
NodeID: nodeID,
|
|
|
|
PubKey: pubKey,
|
|
|
|
Moniker: nodeDirName,
|
|
|
|
RPCAddress: tmCfg.RPC.ListenAddress,
|
|
|
|
P2PAddress: tmCfg.P2P.ListenAddress,
|
|
|
|
APIAddress: apiAddr,
|
|
|
|
Address: addr,
|
|
|
|
ValAddress: sdk.ValAddress(addr),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-29 03:41:55 -07:00
|
|
|
err := initGenFiles(cfg, genAccounts, genBalances, genFiles)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
err = collectGenFiles(cfg, network.Validators, network.BaseDir)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
|
2021-06-29 03:41:55 -07:00
|
|
|
l.Log("starting test network...")
|
2021-11-16 11:24:38 -08:00
|
|
|
for idx, v := range network.Validators {
|
2021-06-29 03:41:55 -07:00
|
|
|
err := startInProcess(cfg, v)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-11-16 11:24:38 -08:00
|
|
|
l.Log("started validator", idx)
|
|
|
|
}
|
|
|
|
|
|
|
|
height, err := network.LatestHeight()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2020-06-26 09:30:49 -07:00
|
|
|
}
|
|
|
|
|
2021-11-16 11:24:38 -08:00
|
|
|
l.Log("started test network at height:", height)
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
// Ensure we cleanup incase any test was abruptly halted (e.g. SIGINT) as any
|
|
|
|
// defer in a test would not be called.
|
|
|
|
server.TrapSignal(network.Cleanup)
|
|
|
|
|
2021-06-29 03:41:55 -07:00
|
|
|
return network, nil
|
2020-06-26 09:30:49 -07:00
|
|
|
}
|
|
|
|
|
2020-07-10 07:45:46 -07:00
|
|
|
// LatestHeight returns the latest height of the network or an error if the
|
|
|
|
// query fails or no validators exist.
|
|
|
|
func (n *Network) LatestHeight() (int64, error) {
|
|
|
|
if len(n.Validators) == 0 {
|
|
|
|
return 0, errors.New("no validators available")
|
|
|
|
}
|
|
|
|
|
2020-09-23 06:44:40 -07:00
|
|
|
status, err := n.Validators[0].RPCClient.Status(context.Background())
|
2020-07-10 07:45:46 -07:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return status.SyncInfo.LatestBlockHeight, nil
|
|
|
|
}
|
|
|
|
|
2020-07-21 06:54:07 -07:00
|
|
|
// WaitForHeight performs a blocking check where it waits for a block to be
|
|
|
|
// committed after a given block. If that height is not reached within a timeout,
|
|
|
|
// an error is returned. Regardless, the latest height queried is returned.
|
|
|
|
func (n *Network) WaitForHeight(h int64) (int64, error) {
|
|
|
|
return n.WaitForHeightWithTimeout(h, 10*time.Second)
|
|
|
|
}
|
|
|
|
|
2020-06-26 09:30:49 -07:00
|
|
|
// WaitForHeightWithTimeout is the same as WaitForHeight except the caller can
|
|
|
|
// provide a custom timeout.
|
|
|
|
func (n *Network) WaitForHeightWithTimeout(h int64, t time.Duration) (int64, error) {
|
|
|
|
ticker := time.NewTicker(time.Second)
|
2021-11-16 11:24:38 -08:00
|
|
|
defer ticker.Stop()
|
|
|
|
|
|
|
|
timeout := time.NewTimer(t)
|
|
|
|
defer timeout.Stop()
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
if len(n.Validators) == 0 {
|
|
|
|
return 0, errors.New("no validators available")
|
|
|
|
}
|
|
|
|
|
|
|
|
var latestHeight int64
|
|
|
|
val := n.Validators[0]
|
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
2021-11-16 11:24:38 -08:00
|
|
|
case <-timeout.C:
|
2020-06-26 09:30:49 -07:00
|
|
|
return latestHeight, errors.New("timeout exceeded waiting for block")
|
|
|
|
case <-ticker.C:
|
2020-09-23 06:44:40 -07:00
|
|
|
status, err := val.RPCClient.Status(context.Background())
|
2020-06-26 09:30:49 -07:00
|
|
|
if err == nil && status != nil {
|
|
|
|
latestHeight = status.SyncInfo.LatestBlockHeight
|
|
|
|
if latestHeight >= h {
|
|
|
|
return latestHeight, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-21 06:54:07 -07:00
|
|
|
// WaitForNextBlock waits for the next block to be committed, returning an error
|
|
|
|
// upon failure.
|
|
|
|
func (n *Network) WaitForNextBlock() error {
|
|
|
|
lastBlock, err := n.LatestHeight()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err = n.WaitForHeight(lastBlock + 1)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-06-26 09:30:49 -07:00
|
|
|
// Cleanup removes the root testing (temporary) directory and stops both the
|
|
|
|
// Tendermint and API services. It allows other callers to create and start
|
|
|
|
// test networks. This method must be called when a test is finished, typically
|
|
|
|
// in a defer.
|
|
|
|
func (n *Network) Cleanup() {
|
|
|
|
defer func() {
|
|
|
|
lock.Unlock()
|
2021-06-29 03:41:55 -07:00
|
|
|
n.Logger.Log("released test network lock")
|
2020-06-26 09:30:49 -07:00
|
|
|
}()
|
|
|
|
|
2021-06-29 03:41:55 -07:00
|
|
|
n.Logger.Log("cleaning up test network...")
|
2020-06-26 09:30:49 -07:00
|
|
|
|
|
|
|
for _, v := range n.Validators {
|
|
|
|
if v.tmNode != nil && v.tmNode.IsRunning() {
|
|
|
|
_ = v.tmNode.Stop()
|
|
|
|
}
|
|
|
|
|
|
|
|
if v.api != nil {
|
|
|
|
_ = v.api.Close()
|
|
|
|
}
|
2020-08-05 08:45:22 -07:00
|
|
|
|
|
|
|
if v.grpc != nil {
|
|
|
|
v.grpc.Stop()
|
2021-01-18 10:25:22 -08:00
|
|
|
if v.grpcWeb != nil {
|
|
|
|
_ = v.grpcWeb.Close()
|
|
|
|
}
|
2020-08-05 08:45:22 -07:00
|
|
|
}
|
2020-06-26 09:30:49 -07:00
|
|
|
}
|
|
|
|
|
2020-07-27 10:57:15 -07:00
|
|
|
if n.Config.CleanupDir {
|
2020-06-26 09:30:49 -07:00
|
|
|
_ = os.RemoveAll(n.BaseDir)
|
|
|
|
}
|
|
|
|
|
2021-06-29 03:41:55 -07:00
|
|
|
n.Logger.Log("finished cleaning up test network")
|
|
|
|
}
|
|
|
|
|
|
|
|
// printMnemonic prints a provided mnemonic seed phrase on a network logger
|
|
|
|
// for debugging and manual testing
|
|
|
|
func printMnemonic(l Logger, secret string) {
|
|
|
|
lines := []string{
|
|
|
|
"THIS MNEMONIC IS FOR TESTING PURPOSES ONLY",
|
|
|
|
"DO NOT USE IN PRODUCTION",
|
|
|
|
"",
|
|
|
|
strings.Join(strings.Fields(secret)[0:8], " "),
|
|
|
|
strings.Join(strings.Fields(secret)[8:16], " "),
|
|
|
|
strings.Join(strings.Fields(secret)[16:24], " "),
|
|
|
|
}
|
|
|
|
|
|
|
|
lineLengths := make([]int, len(lines))
|
|
|
|
for i, line := range lines {
|
|
|
|
lineLengths[i] = len(line)
|
|
|
|
}
|
|
|
|
|
|
|
|
maxLineLength := 0
|
|
|
|
for _, lineLen := range lineLengths {
|
|
|
|
if lineLen > maxLineLength {
|
|
|
|
maxLineLength = lineLen
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
l.Log("\n")
|
|
|
|
l.Log(strings.Repeat("+", maxLineLength+8))
|
|
|
|
for _, line := range lines {
|
|
|
|
l.Logf("++ %s ++\n", centerText(line, maxLineLength))
|
|
|
|
}
|
|
|
|
l.Log(strings.Repeat("+", maxLineLength+8))
|
|
|
|
l.Log("\n")
|
|
|
|
}
|
|
|
|
|
|
|
|
// centerText centers text across a fixed width, filling either side with whitespace buffers
|
|
|
|
func centerText(text string, width int) string {
|
|
|
|
textLen := len(text)
|
|
|
|
leftBuffer := strings.Repeat(" ", (width-textLen)/2)
|
|
|
|
rightBuffer := strings.Repeat(" ", (width-textLen)/2+(width-textLen)%2)
|
|
|
|
|
|
|
|
return fmt.Sprintf("%s%s%s", leftBuffer, text, rightBuffer)
|
2020-06-26 09:30:49 -07:00
|
|
|
}
|