Skip to content

Commit 2b45eaa

Browse files
authored
feat: wire custom coin type and get bech32 prefix (#4569)
1 parent 0eb0923 commit 2b45eaa

24 files changed

+408
-39
lines changed

changelog.md

+4
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,10 @@
66

77
- [#4509](https://github.com/ignite/cli/pull/4509) Upgrade to Go 1.24. Running `ignite doctor` migrates the scaffolded `tools.go` to the tool directive in the go.mod
88

9+
### Changes
10+
11+
- [#4569](https://github.com/ignite/cli/pull/4569) Add flags to set coin type on commands. Add getters for bech32 prefix and coin type.
12+
913
## [`v29.0.0-beta.1`](https://github.com/ignite/cli/releases/tag/v29.0.0-beta.1)
1014

1115
### Features

go.mod

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ module github.com/ignite/cli/v29
33
go 1.24.0
44

55
require (
6+
cosmossdk.io/core v0.11.2
67
cosmossdk.io/math v1.5.0
78
dario.cat/mergo v1.0.1
89
github.com/99designs/keyring v1.2.2
@@ -80,7 +81,6 @@ require (
8081
connectrpc.com/otelconnect v0.7.1 // indirect
8182
cosmossdk.io/api v0.7.6 // indirect
8283
cosmossdk.io/collections v0.4.0 // indirect
83-
cosmossdk.io/core v0.11.2 // indirect
8484
cosmossdk.io/depinject v1.1.0 // indirect
8585
cosmossdk.io/errors v1.0.1 // indirect
8686
cosmossdk.io/log v1.4.1 // indirect

ignite/cmd/account.go

+12
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ import (
1313

1414
const (
1515
flagAddressPrefix = "address-prefix"
16+
flagCoinType = "coin-type"
1617
flagPassphrase = "passphrase"
1718
flagNonInteractive = "non-interactive"
1819
flagKeyringBackend = "keyring-backend"
@@ -102,6 +103,17 @@ func getAddressPrefix(cmd *cobra.Command) string {
102103
return prefix
103104
}
104105

106+
func flagSetCoinType() *flag.FlagSet {
107+
fs := flag.NewFlagSet("", flag.ContinueOnError)
108+
fs.Uint32(flagCoinType, cosmosaccount.CoinTypeCosmos, "coin type to use for the account")
109+
return fs
110+
}
111+
112+
func getCoinType(cmd *cobra.Command) uint32 {
113+
coinType, _ := cmd.Flags().GetUint32(flagCoinType)
114+
return coinType
115+
}
116+
105117
func flagSetAccountImport() *flag.FlagSet {
106118
fs := flag.NewFlagSet("", flag.ContinueOnError)
107119
fs.Bool(flagNonInteractive, false, "do not enter into interactive mode")

ignite/cmd/account_create.go

+3
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,8 @@ func NewAccountCreate() *cobra.Command {
1616
RunE: accountCreateHandler,
1717
}
1818

19+
c.Flags().AddFlagSet(flagSetCoinType())
20+
1921
return c
2022
}
2123

@@ -29,6 +31,7 @@ func accountCreateHandler(cmd *cobra.Command, args []string) error {
2931
ca, err := cosmosaccount.New(
3032
cosmosaccount.WithKeyringBackend(getKeyringBackend(cmd)),
3133
cosmosaccount.WithHome(getKeyringDir(cmd)),
34+
cosmosaccount.WithCoinType(getCoinType(cmd)),
3235
)
3336
if err != nil {
3437
return errors.Errorf("unable to create registry: %w", err)

ignite/cmd/account_import.go

+2
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ func NewAccountImport() *cobra.Command {
2424

2525
c.Flags().String(flagSecret, "", "Your mnemonic or path to your private key (use interactive mode instead to securely pass your mnemonic)")
2626
c.Flags().AddFlagSet(flagSetAccountImport())
27+
c.Flags().AddFlagSet(flagSetCoinType())
2728

2829
return c
2930
}
@@ -64,6 +65,7 @@ func accountImportHandler(cmd *cobra.Command, args []string) error {
6465
ca, err := cosmosaccount.New(
6566
cosmosaccount.WithKeyringBackend(getKeyringBackend(cmd)),
6667
cosmosaccount.WithHome(getKeyringDir(cmd)),
68+
cosmosaccount.WithCoinType(getCoinType(cmd)),
6769
)
6870
if err != nil {
6971
return err

ignite/cmd/account_list.go

+1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ func accountListHandler(cmd *cobra.Command, _ []string) error {
2222
ca, err := cosmosaccount.New(
2323
cosmosaccount.WithKeyringBackend(getKeyringBackend(cmd)),
2424
cosmosaccount.WithHome(getKeyringDir(cmd)),
25+
cosmosaccount.WithBech32Prefix(getAddressPrefix(cmd)),
2526
)
2627
if err != nil {
2728
return err

ignite/cmd/account_show.go

+1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ func accountShowHandler(cmd *cobra.Command, args []string) error {
2525
ca, err := cosmosaccount.New(
2626
cosmosaccount.WithKeyringBackend(getKeyringBackend(cmd)),
2727
cosmosaccount.WithHome(getKeyringDir(cmd)),
28+
cosmosaccount.WithBech32Prefix(getAddressPrefix(cmd)),
2829
)
2930
if err != nil {
3031
return err

ignite/cmd/scaffold_chain.go

+3
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,7 @@ about Cosmos SDK on https://docs.cosmos.network
8080

8181
flagSetClearCache(c)
8282
c.Flags().AddFlagSet(flagSetAccountPrefixes())
83+
c.Flags().AddFlagSet(flagSetCoinType())
8384
c.Flags().StringP(flagPath, "p", "", "create a project in a specific path")
8485
c.Flags().Bool(flagNoDefaultModule, false, "create a project without a default module")
8586
c.Flags().StringSlice(flagParams, []string{}, "add default module parameters")
@@ -102,6 +103,7 @@ func scaffoldChainHandler(cmd *cobra.Command, args []string) error {
102103
var (
103104
name = args[0]
104105
addressPrefix = getAddressPrefix(cmd)
106+
coinType = getCoinType(cmd)
105107
appPath = flagGetPath(cmd)
106108

107109
noDefaultModule, _ = cmd.Flags().GetBool(flagNoDefaultModule)
@@ -133,6 +135,7 @@ func scaffoldChainHandler(cmd *cobra.Command, args []string) error {
133135
appPath,
134136
name,
135137
addressPrefix,
138+
coinType,
136139
protoDir,
137140
noDefaultModule,
138141
minimal,

ignite/cmd/scaffold_chain_registry.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,7 @@ func scaffoldChainRegistryFiles(cmd *cobra.Command, _ []string) error {
5555
return err
5656
}
5757

58-
if err = sc.AddChainRegistryFiles(c, cfg); err != nil {
58+
if err = sc.CreateChainRegistryFiles(c, cfg); err != nil {
5959
return err
6060
}
6161

ignite/cmd/testnet_inplace.go

+13-3
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,8 @@ We can create a testnet from the local network state and mint additional coins f
3030
c.Flags().AddFlagSet(flagSetCheckDependencies())
3131
c.Flags().AddFlagSet(flagSetSkipProto())
3232
c.Flags().AddFlagSet(flagSetVerbose())
33+
c.Flags().AddFlagSet(flagSetAccountPrefixes())
34+
c.Flags().AddFlagSet(flagSetCoinType())
3335

3436
c.Flags().Bool(flagQuitOnFail, false, "quit program if the app fails to start")
3537
return c
@@ -71,17 +73,27 @@ func testnetInplace(cmd *cobra.Command, session *cliui.Session) error {
7173
if err != nil {
7274
return err
7375
}
76+
7477
home, err := c.Home()
7578
if err != nil {
7679
return err
7780
}
81+
7882
keyringBackend, err := c.KeyringBackend()
7983
if err != nil {
8084
return err
8185
}
86+
87+
prefix := getAddressPrefix(cmd)
88+
addressCodec := address.NewBech32Codec(prefix)
89+
valAddressCodec := address.NewBech32Codec(prefix + "valoper")
90+
coinType := getCoinType(cmd)
91+
8292
ca, err := cosmosaccount.New(
8393
cosmosaccount.WithKeyringBackend(cosmosaccount.KeyringBackend(keyringBackend)),
8494
cosmosaccount.WithHome(home),
95+
cosmosaccount.WithBech32Prefix(prefix),
96+
cosmosaccount.WithCoinType(coinType),
8597
)
8698
if err != nil {
8799
return err
@@ -93,9 +105,6 @@ func testnetInplace(cmd *cobra.Command, session *cliui.Session) error {
93105
accErr *cosmosaccount.AccountDoesNotExistError
94106
)
95107

96-
prefix := getAddressPrefix(cmd)
97-
addressCodec := address.NewBech32Codec(prefix)
98-
valAddressCodec := address.NewBech32Codec(prefix + "valoper")
99108
for _, acc := range cfg.Accounts {
100109
sdkAcc, err := ca.GetByName(acc.Name)
101110
if errors.As(err, &accErr) {
@@ -138,5 +147,6 @@ func testnetInplace(cmd *cobra.Command, session *cliui.Session) error {
138147
NewOperatorAddress: operatorAddressStr,
139148
AccountsToFund: accounts,
140149
}
150+
141151
return c.TestnetInPlace(cmd.Context(), args)
142152
}

ignite/pkg/chainregistry/chain.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ type Chain struct {
1919
DaemonName string `json:"daemon_name"`
2020
NodeHome string `json:"node_home"`
2121
KeyAlgos []KeyAlgos `json:"key_algos"`
22-
Slip44 int `json:"slip44"`
22+
Slip44 uint32 `json:"slip44"`
2323
Fees Fees `json:"fees"`
2424
Staking Staking `json:"staking"`
2525
Codebase Codebase `json:"codebase"`

ignite/pkg/cosmosaccount/cosmosaccount.go

+31-14
Original file line numberDiff line numberDiff line change
@@ -9,21 +9,23 @@ import (
99
dkeyring "github.com/99designs/keyring"
1010
"github.com/cosmos/go-bip39"
1111

12+
addresscodec "cosmossdk.io/core/address"
13+
1214
"github.com/cosmos/cosmos-sdk/codec"
15+
"github.com/cosmos/cosmos-sdk/codec/address"
1316
"github.com/cosmos/cosmos-sdk/codec/types"
1417
cryptocodec "github.com/cosmos/cosmos-sdk/crypto/codec"
1518
"github.com/cosmos/cosmos-sdk/crypto/hd"
1619
"github.com/cosmos/cosmos-sdk/crypto/keyring"
1720
sdktypes "github.com/cosmos/cosmos-sdk/types"
18-
"github.com/cosmos/cosmos-sdk/types/bech32"
1921
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
2022

2123
"github.com/ignite/cli/v29/ignite/pkg/errors"
2224
)
2325

2426
const (
2527
// KeyringServiceName used for the name of keyring in OS backend.
26-
KeyringServiceName = "starport"
28+
KeyringServiceName = "ignite"
2729

2830
// DefaultAccount is the name of the default account.
2931
DefaultAccount = "default"
@@ -35,6 +37,7 @@ var KeyringHome = os.ExpandEnv("$HOME/.ignite/accounts")
3537
var ErrAccountExists = errors.New("account already exists")
3638

3739
const (
40+
CoinTypeCosmos = sdktypes.CoinType
3841
AccountPrefixCosmos = "cosmos"
3942
)
4043

@@ -59,6 +62,8 @@ type Registry struct {
5962
homePath string
6063
keyringServiceName string
6164
keyringBackend KeyringBackend
65+
addressCodec addresscodec.Codec
66+
coinType uint32
6267

6368
Keyring keyring.Keyring
6469
}
@@ -84,12 +89,26 @@ func WithKeyringBackend(backend KeyringBackend) Option {
8489
}
8590
}
8691

92+
func WithBech32Prefix(prefix string) Option {
93+
return func(c *Registry) {
94+
c.addressCodec = address.NewBech32Codec(prefix)
95+
}
96+
}
97+
98+
func WithCoinType(coinType uint32) Option {
99+
return func(c *Registry) {
100+
c.coinType = coinType
101+
}
102+
}
103+
87104
// New creates a new registry to manage accounts.
88105
func New(options ...Option) (Registry, error) {
89106
r := Registry{
90107
keyringServiceName: sdktypes.KeyringServiceName(),
91108
keyringBackend: KeyringTest,
92109
homePath: KeyringHome,
110+
addressCodec: address.NewBech32Codec(AccountPrefixCosmos),
111+
coinType: CoinTypeCosmos,
93112
}
94113

95114
for _, apply := range options {
@@ -146,7 +165,13 @@ func (a Account) Address(accPrefix string) (string, error) {
146165
return "", err
147166
}
148167

149-
return toBech32(accPrefix, pk.Address())
168+
addressCodec := address.NewBech32Codec(accPrefix)
169+
addr, err := addressCodec.BytesToString(pk.Address())
170+
if err != nil {
171+
return "", err
172+
}
173+
174+
return addr, nil
150175
}
151176

152177
// PubKey returns a public key for account.
@@ -159,14 +184,6 @@ func (a Account) PubKey() (string, error) {
159184
return pk.String(), nil
160185
}
161186

162-
func toBech32(prefix string, addr []byte) (string, error) {
163-
bech32Addr, err := bech32.ConvertAndEncode(prefix, addr)
164-
if err != nil {
165-
return "", err
166-
}
167-
return bech32Addr, nil
168-
}
169-
170187
// EnsureDefaultAccount ensures that default account exists.
171188
func (r Registry) EnsureDefaultAccount() error {
172189
_, err := r.GetByName(DefaultAccount)
@@ -289,11 +306,11 @@ func (r Registry) GetByName(name string) (Account, error) {
289306

290307
// GetByAddress returns an account by its address.
291308
func (r Registry) GetByAddress(address string) (Account, error) {
292-
sdkAddr, err := sdktypes.AccAddressFromBech32(address)
309+
sdkAddr, err := r.addressCodec.StringToBytes(address)
293310
if err != nil {
294311
return Account{}, err
295312
}
296-
record, err := r.Keyring.KeyByAddress(sdkAddr)
313+
record, err := r.Keyring.KeyByAddress(sdktypes.AccAddress(sdkAddr))
297314
if errors.Is(err, dkeyring.ErrKeyNotFound) || errors.Is(err, sdkerrors.ErrKeyNotFound) {
298315
return Account{}, &AccountDoesNotExistError{address}
299316
}
@@ -335,7 +352,7 @@ func (r Registry) DeleteByName(name string) error {
335352
}
336353

337354
func (r Registry) hdPath() string {
338-
return hd.CreateHDPath(sdktypes.CoinType, 0, 0).String()
355+
return hd.CreateHDPath(r.coinType, 0, 0).String()
339356
}
340357

341358
func (r Registry) algo() (keyring.SignatureAlgo, error) {

ignite/pkg/cosmosclient/bank.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ func (c Client) BankBalances(ctx context.Context, address string, pagination *qu
2626
}
2727

2828
func (c Client) BankSendTx(ctx context.Context, fromAccount cosmosaccount.Account, toAddress string, amount sdk.Coins) (TxService, error) {
29-
addr, err := fromAccount.Address(c.addressPrefix)
29+
addr, err := fromAccount.Address(c.bech32Prefix)
3030
if err != nil {
3131
return TxService{}, err
3232
}

0 commit comments

Comments
 (0)