-
Notifications
You must be signed in to change notification settings - Fork 115
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
go/oasis-node/cmd/debug/consim: Initial import
- Loading branch information
Showing
9 changed files
with
891 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
go/oasis-node/cmd/debug/consim: Initial import | ||
|
||
Add the ability to exercise some backends without tendermint, while | ||
attempting to preserve some of the semantics. | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,233 @@ | ||
// Package consim implements the mock consensus simulator. | ||
package consim | ||
|
||
import ( | ||
"context" | ||
"crypto" | ||
"encoding/hex" | ||
"fmt" | ||
"math/rand" | ||
"path/filepath" | ||
|
||
"github.com/spf13/cobra" | ||
flag "github.com/spf13/pflag" | ||
"github.com/spf13/viper" | ||
"github.com/tendermint/tendermint/abci/types" | ||
tmtypes "github.com/tendermint/tendermint/types" | ||
|
||
"github.com/oasislabs/oasis-core/go/common/crypto/drbg" | ||
"github.com/oasislabs/oasis-core/go/common/crypto/mathrand" | ||
"github.com/oasislabs/oasis-core/go/common/logging" | ||
"github.com/oasislabs/oasis-core/go/consensus/tendermint/abci" | ||
registryApp "github.com/oasislabs/oasis-core/go/consensus/tendermint/apps/registry" | ||
stakingApp "github.com/oasislabs/oasis-core/go/consensus/tendermint/apps/staking" | ||
genesisFile "github.com/oasislabs/oasis-core/go/genesis/file" | ||
cmdCommon "github.com/oasislabs/oasis-core/go/oasis-node/cmd/common" | ||
cmdFlags "github.com/oasislabs/oasis-core/go/oasis-node/cmd/common/flags" | ||
) | ||
|
||
const ( | ||
cfgWorkload = "consim.workload" | ||
cfgWorkloadSeed = "consim.workload.seed" | ||
) | ||
|
||
var ( | ||
logger = logging.GetLogger("cmd/consim") | ||
|
||
flagsConsim = flag.NewFlagSet("", flag.ContinueOnError) | ||
|
||
conSimCmd = &cobra.Command{ | ||
Use: "consim", | ||
Short: "mock consensus simulator", | ||
RunE: doRun, | ||
} | ||
) | ||
|
||
func doRun(cmd *cobra.Command, args []string) error { | ||
cmd.SilenceUsage = true | ||
|
||
if err := cmdCommon.Init(); err != nil { | ||
cmdCommon.EarlyLogAndExit(err) | ||
} | ||
|
||
dataDir := cmdCommon.DataDir() | ||
if dataDir == "" { | ||
return fmt.Errorf("datadir is mandatory") | ||
} | ||
|
||
ctx, cancelFn := context.WithCancel(context.Background()) | ||
defer cancelFn() | ||
|
||
// Load the genesis document. | ||
genesisProvider, err := genesisFile.DefaultFileProvider() | ||
if err != nil { | ||
logger.Error("failed to initialize genesis provider", | ||
"err", err, | ||
) | ||
return err | ||
} | ||
genesisDoc, err := genesisProvider.GetGenesisDocument() | ||
if err != nil { | ||
logger.Error("failed to get genesis document", | ||
"err", err, | ||
) | ||
return err | ||
} | ||
genesisDoc.SetChainContext() | ||
tmChainID := genesisDoc.ChainContext()[:tmtypes.MaxChainIDLen] | ||
|
||
// Initialize the DRBG and workload. | ||
rngSrc, err := drbg.New(crypto.SHA512, []byte(viper.GetString(cfgWorkloadSeed)), nil, []byte("consim workload generator")) | ||
if err != nil { | ||
logger.Error("failed to initialize DRBG", | ||
"err", err, | ||
) | ||
return err | ||
} | ||
|
||
workload, err := newWorkload(rand.New(mathrand.New(rngSrc))) | ||
if err != nil { | ||
logger.Error("failed to create workload", | ||
"err", err, | ||
) | ||
return err | ||
} | ||
defer workload.Cleanup() | ||
|
||
if err = workload.Init(genesisDoc); err != nil { | ||
logger.Error("failed to initialize workload", | ||
"err", err, | ||
) | ||
return err | ||
} | ||
|
||
// Initialize the mock chain backend. | ||
txAuthApp := stakingApp.New() | ||
cfg := &mockChainCfg{ | ||
dataDir: dataDir, | ||
apps: []abci.Application{ | ||
registryApp.New(), | ||
txAuthApp, // This is the staking app. | ||
}, | ||
genesisDoc: genesisDoc, | ||
tmChainID: tmChainID, | ||
txAuthHandler: txAuthApp.(abci.TransactionAuthHandler), | ||
} | ||
mockChain, err := initMockChain(ctx, cfg) | ||
if err != nil { | ||
logger.Error("failed to initialize mock chain backend", | ||
"err", err, | ||
) | ||
return err | ||
} | ||
defer mockChain.close() | ||
chainState, err := mockChain.stateToGenesis(ctx) | ||
if err != nil { | ||
logger.Error("failed to obtain chain state", | ||
"err", err, | ||
) | ||
return err | ||
} | ||
|
||
// Start the workload. | ||
cancelCh, errCh := make(chan struct{}), make(chan error) | ||
defer close(cancelCh) | ||
txVecCh, err := workload.Start(chainState, cancelCh, errCh) | ||
if err != nil { | ||
logger.Error("failed to start workload", | ||
"err", err, | ||
) | ||
return err | ||
} | ||
|
||
// Emulate the tendermint block generation loop. | ||
txLoop: | ||
for { | ||
var ( | ||
txVec []BlockTx | ||
ok bool | ||
) | ||
select { | ||
case err = <-errCh: | ||
logger.Error("workload error", | ||
"err", err, | ||
) | ||
return err | ||
case txVec, ok = <-txVecCh: | ||
if !ok { | ||
break txLoop | ||
} | ||
} | ||
|
||
mockChain.beginBlock() | ||
|
||
// CheckTx all the pending transactions for this block. | ||
var toDeliver []BlockTx | ||
for _, v := range txVec { | ||
txCode := mockChain.checkTx(v.Tx) | ||
if txCode != v.Code { | ||
logger.Error("CheckTx response code mismatch", | ||
"tx", hex.EncodeToString(v.Tx), | ||
"code", txCode, | ||
) | ||
return fmt.Errorf("consim: CheckTx response code mismatch") | ||
} else if v.Code == types.CodeTypeOK { | ||
toDeliver = append(toDeliver, v) | ||
} | ||
} | ||
|
||
// DeliverTx all the pending transactions for this block. | ||
for _, v := range toDeliver { | ||
txCode := mockChain.deliverTx(v.Tx) | ||
if txCode != types.CodeTypeOK { | ||
logger.Error("DeliverTx failed", | ||
"tx", hex.EncodeToString(v.Tx), | ||
"code", txCode, | ||
) | ||
return fmt.Errorf("consim: DeliverTx response code mismatch") | ||
} | ||
} | ||
|
||
mockChain.endBlock() | ||
} | ||
|
||
// Dump the final state to a JSON document. | ||
finalGenesis, err := mockChain.stateToGenesis(ctx) | ||
if err != nil { | ||
logger.Error("failed to obtain state dump", | ||
"err", err, | ||
) | ||
return err | ||
} | ||
|
||
if err = workload.Finalize(finalGenesis); err != nil { | ||
logger.Error("failed to finalize workload", | ||
"err", err, | ||
) | ||
return err | ||
} | ||
|
||
if err = finalGenesis.WriteFileJSON(filepath.Join(dataDir, "dump.json")); err != nil { | ||
logger.Error("failed to write state dump", | ||
"err", err, | ||
) | ||
return err | ||
} | ||
|
||
return nil | ||
} | ||
|
||
// Register registers the consim sub-command. | ||
func Register(parentCmd *cobra.Command) { | ||
conSimCmd.Flags().AddFlagSet(cmdFlags.GenesisFileFlags) | ||
conSimCmd.Flags().AddFlagSet(flagsConsim) | ||
conSimCmd.Flags().AddFlagSet(fileTxsFlag) | ||
conSimCmd.Flags().AddFlagSet(xferFlags) | ||
parentCmd.AddCommand(conSimCmd) | ||
} | ||
|
||
func init() { | ||
flagsConsim.String(cfgWorkload, fileWorkloadName, "workload to execute") | ||
flagsConsim.String(cfgWorkloadSeed, "seeeeeeeeeeeeeeeeeeeeeeeeeeeeeed", "DRBG seed for workloads") | ||
_ = viper.BindPFlags(flagsConsim) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,86 @@ | ||
package consim | ||
|
||
import ( | ||
"bufio" | ||
"encoding/json" | ||
"fmt" | ||
"math/rand" | ||
"os" | ||
|
||
flag "github.com/spf13/pflag" | ||
"github.com/spf13/viper" | ||
|
||
genesis "github.com/oasislabs/oasis-core/go/genesis/api" | ||
) | ||
|
||
const ( | ||
cfgFileTxs = "consim.workload.file.txs" | ||
fileWorkloadName = "file" | ||
) | ||
|
||
var fileTxsFlag = flag.NewFlagSet("", flag.ContinueOnError) | ||
|
||
type fileWorkload struct { | ||
ch chan []BlockTx | ||
dec *json.Decoder | ||
|
||
f *os.File | ||
} | ||
|
||
func (w *fileWorkload) Init(doc *genesis.Document) error { | ||
return nil | ||
} | ||
|
||
func (w *fileWorkload) Start(doc *genesis.Document, cancelCh <-chan struct{}, errCh chan<- error) (<-chan []BlockTx, error) { | ||
if _, err := w.dec.Token(); err != nil { | ||
return nil, fmt.Errorf("consim/workload/file: failed to find opening delimiter: %w", err) | ||
} | ||
w.ch = make(chan []BlockTx) | ||
|
||
go func() { | ||
defer close(w.ch) | ||
for w.dec.More() { | ||
var txVec []BlockTx | ||
if err := w.dec.Decode(&txVec); err != nil { | ||
errCh <- fmt.Errorf("consim/workload/file: failed to decode block tx: %w", err) | ||
return | ||
} | ||
select { | ||
case <-cancelCh: | ||
return | ||
case w.ch <- txVec: | ||
} | ||
} | ||
}() | ||
|
||
return w.ch, nil | ||
} | ||
|
||
func (w *fileWorkload) Finalize(*genesis.Document) error { | ||
if _, err := w.dec.Token(); err != nil { | ||
return fmt.Errorf("consim/workload/file: failed to find closing delimiter: %w", err) | ||
} | ||
|
||
return nil | ||
} | ||
|
||
func (w *fileWorkload) Cleanup() { | ||
_ = w.f.Close() | ||
} | ||
|
||
func newFileWorkload(rng *rand.Rand) (Workload, error) { | ||
f, err := os.Open(viper.GetString(cfgFileTxs)) | ||
if err != nil { | ||
return nil, fmt.Errorf("consim/workload/file: failed to open transaction file: %w", err) | ||
} | ||
|
||
return &fileWorkload{ | ||
dec: json.NewDecoder(bufio.NewReader(f)), | ||
f: f, | ||
}, nil | ||
} | ||
|
||
func init() { | ||
fileTxsFlag.String(cfgFileTxs, "transactions.json", "path to transactions document") | ||
_ = viper.BindPFlags(fileTxsFlag) | ||
} |
Oops, something went wrong.