From f39b7fe625e6167eacba7d516f0efb7a7340ab13 Mon Sep 17 00:00:00 2001 From: Julien Robert Date: Thu, 9 Feb 2023 15:16:07 +0100 Subject: [PATCH] refactor!: extract `AppStateFn` out of simapp (#14977) (cherry picked from commit 4f13b5b3197572bf03a2d2743b0dec2605eaf12e) # Conflicts: # CHANGELOG.md # crypto/keys/secp256k1/secp256k1_test.go # testutil/sims/state_helpers.go --- CHANGELOG.md | 19 ++++ crypto/keys/secp256k1/secp256k1_test.go | 86 +++++++++++++++++++ simapp/app_v2.go | 2 +- simapp/params/params.go | 7 -- simapp/sim_bench_test.go | 4 +- simapp/sim_test.go | 10 +-- .../sims/state_helpers.go | 77 ++++++++++++----- 7 files changed, 167 insertions(+), 38 deletions(-) delete mode 100644 simapp/params/params.go rename simapp/state.go => testutil/sims/state_helpers.go (74%) diff --git a/CHANGELOG.md b/CHANGELOG.md index 81e8797df02..680e475885a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -240,6 +240,25 @@ Ref: https://keepachangelog.com/en/1.0.0/ ### API Breaking Changes +<<<<<<< HEAD +======= +* (simapp) [#14977](https://github.com/cosmos/cosmos-sdk/pull/14977) Move simulation helpers functions (`AppStateFn` and `AppStateRandomizedFn`) to `testutil/sims`. These takes an extra genesisState argument which is the default state of the app. +* (x/gov) [#14720](https://github.com/cosmos/cosmos-sdk/pull/14720) Add an expedited field in the gov v1 proposal and `MsgNewMsgProposal`. +* [#14847](https://github.com/cosmos/cosmos-sdk/pull/14847) App and ModuleManager methods `InitGenesis`, `ExportGenesis`, `BeginBlock` and `EndBlock` now also return an error. +* (simulation) [#14728](https://github.com/cosmos/cosmos-sdk/pull/14728) Rename the `ParamChanges` field to `LegacyParamChange` and `Contents` to `LegacyProposalContents` in `simulation.SimulationState`. Additionally it adds a `ProposalMsgs` field to `simulation.SimulationState`. +* (x/upgrade) [#14764](https://github.com/cosmos/cosmos-sdk/pull/14764) The `x/upgrade` module is extracted to have a separate go.mod file which allows it to be a standalone module. +* (x/gov) [#14782](https://github.com/cosmos/cosmos-sdk/pull/14782) Move the `metadata` argument in `govv1.NewProposal` alongside `title` and `summary`. +* (store) [#14746](https://github.com/cosmos/cosmos-sdk/pull/14746) Extract Store in its own go.mod and rename the package to `cosmossdk.io/store`. +* (simulation) [#14751](https://github.com/cosmos/cosmos-sdk/pull/14751) Remove the `MsgType` field from `simulation.OperationInput` struct. +* (crypto/keyring) [#13734](https://github.com/cosmos/cosmos-sdk/pull/13834) The keyring's `Sign` method now takes a new `signMode` argument. It is only used if the signing key is a Ledger hardware device. You can set it to 0 in all other cases. +* (x/evidence) [14724](https://github.com/cosmos/cosmos-sdk/pull/14724) Extract Evidence in its own go.mod and rename the package to `cosmossdk.io/x/evidence`. +* (x/nft) [#14725](https://github.com/cosmos/cosmos-sdk/pull/14725) Extract NFT in its own go.mod and rename the package to `cosmossdk.io/x/nft`. +* (tx) [#14634](https://github.com/cosmos/cosmos-sdk/pull/14634) Move the `tx` go module to `x/tx`. +* (snapshots) [#14597](https://github.com/cosmos/cosmos-sdk/pull/14597) Move `snapshots` to `store/snapshots`, rename and bump proto package to v1. +* (crypto/keyring) [#14151](https://github.com/cosmos/cosmos-sdk/pull/14151) Move keys presentation from `crypto/keyring` to `client/keys` +* (modules) [#13850](https://github.com/cosmos/cosmos-sdk/pull/13850) and [#14046](https://github.com/cosmos/cosmos-sdk/pull/14046) Remove gogoproto stringer annotations. This removes the custom `String()` methods on all types that were using the annotations. +* (x/auth) [#13850](https://github.com/cosmos/cosmos-sdk/pull/13850/) Remove `MarshalYAML` methods from module (`x/...`) types. +>>>>>>> 4f13b5b31 (refactor!: extract `AppStateFn` out of simapp (#14977)) * (x/auth) [#13877](https://github.com/cosmos/cosmos-sdk/pull/13877) Rename `AccountKeeper`'s `GetNextAccountNumber` to `NextAccountNumber`. * (x/evidence) [#13740](https://github.com/cosmos/cosmos-sdk/pull/13740) The `NewQueryEvidenceRequest` function now takes `hash` as a HEX encoded `string`. * (server) [#13485](https://github.com/cosmos/cosmos-sdk/pull/13485) The `Application` service now requires the `RegisterNodeService` method to be implemented. diff --git a/crypto/keys/secp256k1/secp256k1_test.go b/crypto/keys/secp256k1/secp256k1_test.go index 651665ad4e8..076046c9f86 100644 --- a/crypto/keys/secp256k1/secp256k1_test.go +++ b/crypto/keys/secp256k1/secp256k1_test.go @@ -27,6 +27,92 @@ type keyData struct { addr string } +<<<<<<< HEAD +======= +/* +The following code snippet has been used to generate test vectors. The purpose of these vectors are to check our +implementation of secp256k1 against go-ethereum's one. It has been commented to avoid dependencies. + + github.com/btcsuite/btcutil v1.0.2 + github.com/ethereum/go-ethereum v1.10.26 + golang.org/x/crypto v0.0.0-20210921155107-089bfa567519 + +--- + + import ( + "crypto/ecdsa" + "crypto/sha256" + "encoding/hex" + "fmt" + "github.com/btcsuite/btcutil/base58" + "github.com/ethereum/go-ethereum/crypto" + "golang.org/x/crypto/ripemd160" + ) + + func ethereumKeys() keyData { + // Generate private key with the go-ethereum + priv, err := crypto.GenerateKey() + if err != nil { + panic(err) + } + encPriv := make([]byte, len(priv.D.Bytes())*2) + hex.Encode(encPriv, priv.D.Bytes()) + + // Get go-ethereum public key + ethPub, ok := priv.Public().(*ecdsa.PublicKey) + if !ok { + panic(err) + } + ethPublicKeyBytes := crypto.FromECDSAPub(ethPub) + + // Format byte depending on the oddness of the Y coordinate. + format := 0x02 + if ethPub.Y.Bit(0) != 0 { + format = 0x03 + } + + // Public key in the 33-byte compressed format. + pub := ethPublicKeyBytes[:33] + encPub := make([]byte, len(pub)*2) + pub[0] = byte(format) + hex.Encode(encPub, pub) + + // Bitcoin style addresses + sha := sha256.Sum256(pub) + hasherRIPEMD160 := ripemd160.New() + hasherRIPEMD160.Write(sha[:]) + addr := hasherRIPEMD160.Sum(nil) + return keyData{ + priv: string(encPriv), + pub: string(encPub), + addr: base58.CheckEncode(addr[:], 0), + } + } +*/ + +/* +generateKeyForCheckingConsistency was used to create test vectors that matches consistency against prior versions. +Here are the specific versions used to generate the vectors. + +github.com/cosmos/btcutil v1.0.5 +github.com/cosmos/cosmos-sdk v0.46.8 +*/ +var _ = func() keyData { + priv := secp256k1.GenPrivKey() + encPriv := make([]byte, len(priv.Key)*2) + hex.Encode(encPriv, priv.Key) + pub := priv.PubKey() + encPub := make([]byte, len(pub.Bytes())*2) + hex.Encode(encPub, pub.Bytes()) + addr := pub.Address() + return keyData{ + priv: string(encPriv), + pub: string(encPub), + addr: base58.CheckEncode(addr, 0), + } +} + +>>>>>>> 4f13b5b31 (refactor!: extract `AppStateFn` out of simapp (#14977)) var secpDataTable = []keyData{ { priv: "a96e62ed3955e65be32703f12d87b6b5cf26039ecfa948dc5107a495418e5330", diff --git a/simapp/app_v2.go b/simapp/app_v2.go index 99f93d53163..4be43f0e19c 100644 --- a/simapp/app_v2.go +++ b/simapp/app_v2.go @@ -203,7 +203,7 @@ func NewSimApp( // For providing a custom a base account type add it below. // By default the auth module uses authtypes.ProtoBaseAccount(). // - // func() authtypes.AccountI { return authtypes.ProtoBaseAccount() }, + // func() sdk.AccountI { return authtypes.ProtoBaseAccount() }, // // MINT diff --git a/simapp/params/params.go b/simapp/params/params.go deleted file mode 100644 index b6aa5fb55e0..00000000000 --- a/simapp/params/params.go +++ /dev/null @@ -1,7 +0,0 @@ -package params - -// Simulation parameter constants -const ( - StakePerAccount = "stake_per_account" - InitiallyBondedValidators = "initially_bonded_validators" -) diff --git a/simapp/sim_bench_test.go b/simapp/sim_bench_test.go index 8a7b46a2b70..17ab7098e79 100644 --- a/simapp/sim_bench_test.go +++ b/simapp/sim_bench_test.go @@ -49,7 +49,7 @@ func BenchmarkFullAppSimulation(b *testing.B) { b, os.Stdout, app.BaseApp, - AppStateFn(app.AppCodec(), app.SimulationManager()), + simtestutil.AppStateFn(app.AppCodec(), app.SimulationManager(), app.DefaultGenesis()), simtypes.RandomAccounts, // Replace with own random account function if using keys other than secp256k1 simtestutil.SimulationOperations(app, app.AppCodec(), config), BlockedAddresses(), @@ -104,7 +104,7 @@ func BenchmarkInvariants(b *testing.B) { b, os.Stdout, app.BaseApp, - AppStateFn(app.AppCodec(), app.SimulationManager()), + simtestutil.AppStateFn(app.AppCodec(), app.SimulationManager(), app.DefaultGenesis()), simtypes.RandomAccounts, // Replace with own random account function if using keys other than secp256k1 simtestutil.SimulationOperations(app, app.AppCodec(), config), BlockedAddresses(), diff --git a/simapp/sim_test.go b/simapp/sim_test.go index 5192c92eeea..e3106b875cf 100644 --- a/simapp/sim_test.go +++ b/simapp/sim_test.go @@ -92,7 +92,7 @@ func TestFullAppSimulation(t *testing.T) { t, os.Stdout, app.BaseApp, - AppStateFn(app.AppCodec(), app.SimulationManager()), + simtestutil.AppStateFn(app.AppCodec(), app.SimulationManager(), app.DefaultGenesis()), simtypes.RandomAccounts, // Replace with own random account function if using keys other than secp256k1 simtestutil.SimulationOperations(app, app.AppCodec(), config), BlockedAddresses(), @@ -137,7 +137,7 @@ func TestAppImportExport(t *testing.T) { t, os.Stdout, app.BaseApp, - AppStateFn(app.AppCodec(), app.SimulationManager()), + simtestutil.AppStateFn(app.AppCodec(), app.SimulationManager(), app.DefaultGenesis()), simtypes.RandomAccounts, // Replace with own random account function if using keys other than secp256k1 simtestutil.SimulationOperations(app, app.AppCodec(), config), BlockedAddresses(), @@ -253,7 +253,7 @@ func TestAppSimulationAfterImport(t *testing.T) { t, os.Stdout, app.BaseApp, - AppStateFn(app.AppCodec(), app.SimulationManager()), + simtestutil.AppStateFn(app.AppCodec(), app.SimulationManager(), app.DefaultGenesis()), simtypes.RandomAccounts, // Replace with own random account function if using keys other than secp256k1 simtestutil.SimulationOperations(app, app.AppCodec(), config), BlockedAddresses(), @@ -301,7 +301,7 @@ func TestAppSimulationAfterImport(t *testing.T) { t, os.Stdout, newApp.BaseApp, - AppStateFn(app.AppCodec(), app.SimulationManager()), + simtestutil.AppStateFn(app.AppCodec(), app.SimulationManager(), app.DefaultGenesis()), simtypes.RandomAccounts, // Replace with own random account function if using keys other than secp256k1 simtestutil.SimulationOperations(newApp, newApp.AppCodec(), config), BlockedAddresses(), @@ -356,7 +356,7 @@ func TestAppStateDeterminism(t *testing.T) { t, os.Stdout, app.BaseApp, - AppStateFn(app.AppCodec(), app.SimulationManager()), + simtestutil.AppStateFn(app.AppCodec(), app.SimulationManager(), app.DefaultGenesis()), simtypes.RandomAccounts, // Replace with own random account function if using keys other than secp256k1 simtestutil.SimulationOperations(app, app.AppCodec(), config), BlockedAddresses(), diff --git a/simapp/state.go b/testutil/sims/state_helpers.go similarity index 74% rename from simapp/state.go rename to testutil/sims/state_helpers.go index ee1836ea767..b0bb9fba5a3 100644 --- a/simapp/state.go +++ b/testutil/sims/state_helpers.go @@ -1,4 +1,4 @@ -package simapp +package sims import ( "encoding/json" @@ -8,11 +8,18 @@ import ( "os" "time" +<<<<<<< HEAD:simapp/state.go tmjson "github.com/tendermint/tendermint/libs/json" tmtypes "github.com/tendermint/tendermint/types" "cosmossdk.io/math" simappparams "cosmossdk.io/simapp/params" +======= + "cosmossdk.io/math" + cmtjson "github.com/cometbft/cometbft/libs/json" + cmttypes "github.com/cometbft/cometbft/types" + +>>>>>>> 4f13b5b31 (refactor!: extract `AppStateFn` out of simapp (#14977)):testutil/sims/state_helpers.go "github.com/cosmos/cosmos-sdk/codec" "github.com/cosmos/cosmos-sdk/crypto/keys/secp256k1" sdk "github.com/cosmos/cosmos-sdk/types" @@ -24,11 +31,21 @@ import ( stakingtypes "github.com/cosmos/cosmos-sdk/x/staking/types" ) +// Simulation parameter constants +const ( + StakePerAccount = "stake_per_account" + InitiallyBondedValidators = "initially_bonded_validators" +) + // AppStateFn returns the initial application state using a genesis or the simulation parameters. // It panics if the user provides files for both of them. // If a file is not given for the genesis or the sim params, it creates a randomized one. -func AppStateFn(cdc codec.JSONCodec, simManager *module.SimulationManager) simtypes.AppStateFn { - return func(r *rand.Rand, accs []simtypes.Account, config simtypes.Config, +// genesisState is the default genesis state of the whole app. +func AppStateFn(cdc codec.JSONCodec, simManager *module.SimulationManager, genesisState map[string]json.RawMessage) simtypes.AppStateFn { + return func( + r *rand.Rand, + accs []simtypes.Account, + config simtypes.Config, ) (appState json.RawMessage, simAccs []simtypes.Account, chainID string, genesisTimestamp time.Time) { if simcli.FlagGenesisTimeValue == 0 { genesisTimestamp = simtypes.RandTimestamp(r) @@ -43,7 +60,10 @@ func AppStateFn(cdc codec.JSONCodec, simManager *module.SimulationManager) simty case config.GenesisFile != "": // override the default chain-id from simapp to set it later to the config - genesisDoc, accounts := AppStateFromGenesisFileFn(r, cdc, config.GenesisFile) + genesisDoc, accounts, err := AppStateFromGenesisFileFn(r, cdc, config.GenesisFile) + if err != nil { + panic(err) + } if simcli.FlagGenesisTimeValue == 0 { // use genesis timestamp if no custom timestamp is provided (i.e no random timestamp) @@ -65,11 +85,11 @@ func AppStateFn(cdc codec.JSONCodec, simManager *module.SimulationManager) simty if err != nil { panic(err) } - appState, simAccs = AppStateRandomizedFn(simManager, r, cdc, accs, genesisTimestamp, appParams) + appState, simAccs = AppStateRandomizedFn(simManager, r, cdc, accs, genesisTimestamp, appParams, genesisState) default: appParams := make(simtypes.AppParams) - appState, simAccs = AppStateRandomizedFn(simManager, r, cdc, accs, genesisTimestamp, appParams) + appState, simAccs = AppStateRandomizedFn(simManager, r, cdc, accs, genesisTimestamp, appParams, genesisState) } rawState := make(map[string]json.RawMessage) @@ -84,8 +104,7 @@ func AppStateFn(cdc codec.JSONCodec, simManager *module.SimulationManager) simty } stakingState := new(stakingtypes.GenesisState) - err = cdc.UnmarshalJSON(stakingStateBz, stakingState) - if err != nil { + if err = cdc.UnmarshalJSON(stakingStateBz, stakingState); err != nil { panic(err) } // compute not bonded balance @@ -104,8 +123,7 @@ func AppStateFn(cdc codec.JSONCodec, simManager *module.SimulationManager) simty panic("bank genesis state is missing") } bankState := new(banktypes.GenesisState) - err = cdc.UnmarshalJSON(bankStateBz, bankState) - if err != nil { + if err = cdc.UnmarshalJSON(bankStateBz, bankState); err != nil { panic(err) } @@ -140,12 +158,15 @@ func AppStateFn(cdc codec.JSONCodec, simManager *module.SimulationManager) simty // AppStateRandomizedFn creates calls each module's GenesisState generator function // and creates the simulation params func AppStateRandomizedFn( - simManager *module.SimulationManager, r *rand.Rand, cdc codec.JSONCodec, - accs []simtypes.Account, genesisTimestamp time.Time, appParams simtypes.AppParams, + simManager *module.SimulationManager, + r *rand.Rand, + cdc codec.JSONCodec, + accs []simtypes.Account, + genesisTimestamp time.Time, + appParams simtypes.AppParams, + genesisState map[string]json.RawMessage, ) (json.RawMessage, []simtypes.Account) { numAccs := int64(len(accs)) - genesisState := ModuleBasics.DefaultGenesis(cdc) - // generate a random amount of initial stake coins and a random initial // number of bonded accounts var ( @@ -153,11 +174,11 @@ func AppStateRandomizedFn( initialStake math.Int ) appParams.GetOrGenerate( - cdc, simappparams.StakePerAccount, &initialStake, r, + cdc, StakePerAccount, &initialStake, r, func(r *rand.Rand) { initialStake = math.NewInt(r.Int63n(1e12)) }, ) appParams.GetOrGenerate( - cdc, simappparams.InitiallyBondedValidators, &numInitiallyBonded, r, + cdc, InitiallyBondedValidators, &numInitiallyBonded, r, func(r *rand.Rand) { numInitiallyBonded = int64(r.Intn(300)) }, ) @@ -197,23 +218,33 @@ func AppStateRandomizedFn( // AppStateFromGenesisFileFn util function to generate the genesis AppState // from a genesis.json file. +<<<<<<< HEAD:simapp/state.go func AppStateFromGenesisFileFn(r io.Reader, cdc codec.JSONCodec, genesisFile string) (tmtypes.GenesisDoc, []simtypes.Account) { +======= +func AppStateFromGenesisFileFn(r io.Reader, cdc codec.JSONCodec, genesisFile string) (cmttypes.GenesisDoc, []simtypes.Account, error) { +>>>>>>> 4f13b5b31 (refactor!: extract `AppStateFn` out of simapp (#14977)):testutil/sims/state_helpers.go bytes, err := os.ReadFile(genesisFile) if err != nil { panic(err) } +<<<<<<< HEAD:simapp/state.go var genesis tmtypes.GenesisDoc // NOTE: Tendermint uses a custom JSON decoder for GenesisDoc err = tmjson.Unmarshal(bytes, &genesis) if err != nil { panic(err) +======= + var genesis cmttypes.GenesisDoc + // NOTE: CometBFT uses a custom JSON decoder for GenesisDoc + if err = cmtjson.Unmarshal(bytes, &genesis); err != nil { + return genesis, nil, err +>>>>>>> 4f13b5b31 (refactor!: extract `AppStateFn` out of simapp (#14977)):testutil/sims/state_helpers.go } - var appState GenesisState - err = json.Unmarshal(genesis.AppState, &appState) - if err != nil { - panic(err) + var appState map[string]json.RawMessage + if err = json.Unmarshal(genesis.AppState, &appState); err != nil { + return genesis, nil, err } var authGenesis authtypes.GenesisState @@ -233,9 +264,9 @@ func AppStateFromGenesisFileFn(r io.Reader, cdc codec.JSONCodec, genesisFile str privKey := secp256k1.GenPrivKeyFromSecret(privkeySeed) - a, ok := acc.GetCachedValue().(authtypes.AccountI) + a, ok := acc.GetCachedValue().(sdk.AccountI) if !ok { - panic("expected account") + return genesis, nil, fmt.Errorf("expected account") } // create simulator accounts @@ -243,5 +274,5 @@ func AppStateFromGenesisFileFn(r io.Reader, cdc codec.JSONCodec, genesisFile str newAccs[i] = simAcc } - return genesis, newAccs + return genesis, newAccs, nil }