Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: create-validator now takes a json file as arg #14864

Merged
merged 43 commits into from
Feb 22, 2023
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
Show all changes
43 commits
Select commit Hold shift + click to select a range
4cc3059
wip: take json file as arg for create-validator cmd
likhita-809 Feb 1, 2023
53ab328
wip: cleanup
likhita-809 Feb 1, 2023
98caf65
wip: more cleanup
likhita-809 Feb 1, 2023
545471b
remove unncessary fields from validator struct
likhita-809 Feb 1, 2023
2b2d84c
wip: more cleanup
likhita-809 Feb 1, 2023
0aabe92
wip: refactor code
likhita-809 Feb 1, 2023
c358a48
Merge branch 'main' of https://github.com/cosmos/cosmos-sdk into likh…
likhita-809 Feb 1, 2023
540c20e
add tests
likhita-809 Feb 1, 2023
434e8e9
Merge branch 'main' of https://github.com/cosmos/cosmos-sdk into likh…
likhita-809 Feb 1, 2023
6eeabda
remove unused flags from create-validator cmd
likhita-809 Feb 2, 2023
50e335b
wip: address review comments and add changelog
likhita-809 Feb 2, 2023
f766120
fix relevant tests
likhita-809 Feb 2, 2023
c2b85aa
Merge branch 'main' of https://github.com/cosmos/cosmos-sdk into likh…
likhita-809 Feb 2, 2023
1cf4d15
add optional json fields and remove flags
likhita-809 Feb 2, 2023
1a4ca2e
fix warnings and tests
likhita-809 Feb 2, 2023
139f3d5
Merge branch 'main' of https://github.com/cosmos/cosmos-sdk into likh…
likhita-809 Feb 2, 2023
56861aa
fix tests
likhita-809 Feb 2, 2023
edb60c1
Merge branch 'main' of https://github.com/cosmos/cosmos-sdk into likh…
likhita-809 Feb 2, 2023
09c41c5
use explicit Example cobra field in create-validator cmd
likhita-809 Feb 3, 2023
2f2a81d
Merge branch 'main' of https://github.com/cosmos/cosmos-sdk into likh…
likhita-809 Feb 3, 2023
a769166
wip: use interface for pubkey
likhita-809 Feb 3, 2023
348b947
wip: use json.RawMessage for pubkey type
likhita-809 Feb 6, 2023
47bcc81
Merge branch 'main' of https://github.com/cosmos/cosmos-sdk into likh…
likhita-809 Feb 6, 2023
2bbba8f
fix tests
likhita-809 Feb 6, 2023
e346655
address review comments
likhita-809 Feb 6, 2023
631cf02
Merge branch 'main' of https://github.com/cosmos/cosmos-sdk into likh…
likhita-809 Feb 6, 2023
97ef612
Merge branch 'main' of https://github.com/cosmos/cosmos-sdk into likh…
likhita-809 Feb 6, 2023
bd3a134
Merge branch 'main' into likhita/val-json-wip
atheeshp Feb 6, 2023
0481b64
Merge branch 'main' into likhita/val-json-wip
likhita-809 Feb 6, 2023
1ba91f6
Merge branch 'main' into likhita/val-json-wip
likhita-809 Feb 7, 2023
b50b47b
Merge branch 'main' into likhita/val-json-wip
likhita-809 Feb 7, 2023
41b2a84
Merge branch 'main' into likhita/val-json-wip
likhita-809 Feb 8, 2023
e4b67f4
Merge branch 'main' into likhita/val-json-wip
atheeshp Feb 9, 2023
42f921b
Merge branch 'main' into likhita/val-json-wip
likhita-809 Feb 10, 2023
36eceeb
Merge branch 'main' into likhita/val-json-wip
atheeshp Feb 13, 2023
7299331
Merge branch 'main' into likhita/val-json-wip
likhita-809 Feb 14, 2023
28c516b
Merge branch 'main' of https://github.com/cosmos/cosmos-sdk into likh…
likhita-809 Feb 15, 2023
4a5fc22
Merge branch 'main' into likhita/val-json-wip
likhita-809 Feb 16, 2023
c825b2f
Merge branch 'main' of https://github.com/cosmos/cosmos-sdk into likh…
likhita-809 Feb 20, 2023
3f67566
Merge branch 'main' into likhita/val-json-wip
likhita-809 Feb 21, 2023
34f340d
fix tests
likhita-809 Feb 21, 2023
10204a4
Merge branch 'main' into likhita/val-json-wip
atheeshp Feb 21, 2023
cb31f4b
Merge branch 'main' into likhita/val-json-wip
likhita-809 Feb 22, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
60 changes: 35 additions & 25 deletions x/staking/client/cli/tx.go
Original file line number Diff line number Diff line change
Expand Up @@ -56,17 +56,44 @@ func NewTxCmd() *cobra.Command {
// NewCreateValidatorCmd returns a CLI command handler for creating a MsgCreateValidator transaction.
func NewCreateValidatorCmd() *cobra.Command {
cmd := &cobra.Command{
Use: "create-validator",
Use: "create-validator [path/to/validator.json]",
Short: "create new validator initialized with a self-delegation to it",
Args: cobra.ExactArgs(1),
Long: strings.TrimSpace(
fmt.Sprintf(`Submit a JSON file with the new validator details.
Example:
likhita-809 marked this conversation as resolved.
Show resolved Hide resolved
$ %s tx staking create-validator path/to/validator.json

Where validator.json contains:

{
"from": "key2",
"pubkey": "{\"@type\":\"/cosmos.crypto.ed25519.PubKey\",\"key\":\"oWg2ISpLF405Jcm2vXV+2v4fnjodh6aafuIdeoW+rUw=\"}",
"amount": "1000000stake",
"moniker": "myvalidator",
"commission-rate": "0.1",
"commission-max-rate": "0.2",
"commission-max-change-rate": "0.01",
"min-self-delegation": "1"
}
`,
version.AppName,
),
),
RunE: func(cmd *cobra.Command, args []string) error {
clientCtx, err := client.GetClientTxContext(cmd)
if err != nil {
return err
}

validator, err := parseValidatorJSON(clientCtx.Codec, args[0])
if err != nil {
return err
}

txf := tx.NewFactoryCLI(clientCtx, cmd.Flags()).
WithTxConfig(clientCtx.TxConfig).WithAccountRetriever(clientCtx.AccountRetriever)
txf, msg, err := newBuildCreateValidatorMsg(clientCtx, txf, cmd.Flags())
txf, msg, err := newBuildCreateValidatorMsg(clientCtx, txf, cmd.Flags(), validator)
if err != nil {
return err
}
Expand All @@ -86,9 +113,6 @@ func NewCreateValidatorCmd() *cobra.Command {
flags.AddTxFlagsToCmd(cmd)

_ = cmd.MarkFlagRequired(flags.FlagFrom)
_ = cmd.MarkFlagRequired(FlagAmount)
_ = cmd.MarkFlagRequired(FlagPubKey)
_ = cmd.MarkFlagRequired(FlagMoniker)

return cmd
}
Expand Down Expand Up @@ -337,51 +361,37 @@ $ %s tx staking cancel-unbond %s1gghjut3ccd8ay0zduzj64hwre2fxs9ldmqhffj 100stake
return cmd
}

func newBuildCreateValidatorMsg(clientCtx client.Context, txf tx.Factory, fs *flag.FlagSet) (tx.Factory, *types.MsgCreateValidator, error) {
fAmount, _ := fs.GetString(FlagAmount)
amount, err := sdk.ParseCoinNormalized(fAmount)
func newBuildCreateValidatorMsg(clientCtx client.Context, txf tx.Factory, fs *flag.FlagSet, val validator) (tx.Factory, *types.MsgCreateValidator, error) {
amount, err := sdk.ParseCoinNormalized(val.Amount)
if err != nil {
return txf, nil, err
}

valAddr := clientCtx.GetFromAddress()
pkStr, err := fs.GetString(FlagPubKey)
if err != nil {
return txf, nil, err
}

var pk cryptotypes.PubKey
if err := clientCtx.Codec.UnmarshalInterfaceJSON([]byte(pkStr), &pk); err != nil {
if err := clientCtx.Codec.UnmarshalInterfaceJSON([]byte(val.PubKey), &pk); err != nil {
likhita-809 marked this conversation as resolved.
Show resolved Hide resolved
return txf, nil, err
}

moniker, _ := fs.GetString(FlagMoniker)
identity, _ := fs.GetString(FlagIdentity)
website, _ := fs.GetString(FlagWebsite)
likhita-809 marked this conversation as resolved.
Show resolved Hide resolved
security, _ := fs.GetString(FlagSecurityContact)
details, _ := fs.GetString(FlagDetails)
description := types.NewDescription(
moniker,
val.Moniker,
identity,
website,
security,
details,
)

// get the initial validator commission parameters
rateStr, _ := fs.GetString(FlagCommissionRate)
maxRateStr, _ := fs.GetString(FlagCommissionMaxRate)
maxChangeRateStr, _ := fs.GetString(FlagCommissionMaxChangeRate)

commissionRates, err := buildCommissionRates(rateStr, maxRateStr, maxChangeRateStr)
commissionRates, err := buildCommissionRates(val.CommissionRate, val.CommissionMaxRate, val.CommissionMaxChange)
if err != nil {
return txf, nil, err
}

// get the initial validator min self delegation
msbStr, _ := fs.GetString(FlagMinSelfDelegation)

minSelfDelegation, ok := sdk.NewIntFromString(msbStr)
minSelfDelegation, ok := sdk.NewIntFromString(val.MinSelfDelegation)
if !ok {
return txf, nil, sdkerrors.Wrap(sdkerrors.ErrInvalidRequest, "minimum self delegation must be a positive integer")
}
Expand Down
65 changes: 65 additions & 0 deletions x/staking/client/cli/utils.go
Original file line number Diff line number Diff line change
@@ -1,12 +1,77 @@
package cli

import (
"encoding/json"
"errors"
"fmt"
"os"

"github.com/cosmos/cosmos-sdk/codec"
sdk "github.com/cosmos/cosmos-sdk/types"
"github.com/cosmos/cosmos-sdk/x/staking/types"
)

// validator struct to define the fields of the validator
type validator struct {
likhita-809 marked this conversation as resolved.
Show resolved Hide resolved
Amount string `json:"amount"`
From string `json:"from"`
PubKey string `json:"pubkey"`
likhita-809 marked this conversation as resolved.
Show resolved Hide resolved
Moniker string `json:"moniker"`
CommissionRate string `json:"commission-rate"`
CommissionMaxRate string `json:"commission-max-rate"`
CommissionMaxChange string `json:"commission-max-change-rate"`
MinSelfDelegation string `json:"min-self-delegation"`
}

func parseValidatorJSON(cdc codec.Codec, path string) (validator, error) {
var val validator

contents, err := os.ReadFile(path)
Fixed Show fixed Hide fixed
Fixed Show fixed Hide fixed

Check failure

Code scanning / gosec

Potential file inclusion via variable

Potential file inclusion via variable
if err != nil {
return validator{}, err
}

err = json.Unmarshal(contents, &val)
if err != nil {
return validator{}, err
}

if err := val.validate(); err != nil {
return validator{}, err
}

return val, nil
}

// validate checks that the required fields are present in the validator json.
func (v validator) validate() error {
if v.Amount == "" {
return fmt.Errorf("must specify amount of coins to bond")
}
if v.From == "" {
return fmt.Errorf("must specify name or address of from key")
}
if v.PubKey == "" {
return fmt.Errorf("must specify the JSON encoded pubkey")
}
if v.Moniker == "" {
return fmt.Errorf("must specify the moniker name")
}
if v.CommissionRate == "" {
return fmt.Errorf("must specify initial commission rate percentage")
}
if v.CommissionMaxRate == "" {
return fmt.Errorf("must specify maximum commission rate percentage")
}
if v.CommissionMaxChange == "" {
return fmt.Errorf("must specify maximum commission change rate percentage (per day)")
}
if v.MinSelfDelegation == "" {
return fmt.Errorf("minimum self delegation must be a positive integer")
}
return nil
}

func buildCommissionRates(rateStr, maxRateStr, maxChangeRateStr string) (commission types.CommissionRates, err error) {
if rateStr == "" || maxRateStr == "" || maxChangeRateStr == "" {
return commission, errors.New("must specify all validator commission parameters")
Expand Down