Some checks failed
Build SimApp / build (amd64) (push) Waiting to run
Build SimApp / build (arm64) (push) Waiting to run
CodeQL / Analyze (push) Waiting to run
Build & Push / build (push) Waiting to run
Run Gosec / Gosec (push) Waiting to run
Lint / golangci-lint (push) Waiting to run
Checks dependencies and mocks generation / Check go mod tidy (push) Waiting to run
Checks dependencies and mocks generation / Check up to date mocks (push) Waiting to run
System Tests / setup (push) Waiting to run
System Tests / test-system (push) Blocked by required conditions
System Tests / test-system-legacy (push) Blocked by required conditions
Tests / Code Coverage / split-test-files (push) Waiting to run
Tests / Code Coverage / tests (00) (push) Blocked by required conditions
Tests / Code Coverage / tests (01) (push) Blocked by required conditions
Tests / Code Coverage / tests (02) (push) Blocked by required conditions
Tests / Code Coverage / tests (03) (push) Blocked by required conditions
Tests / Code Coverage / test-integration (push) Waiting to run
Tests / Code Coverage / test-e2e (push) Waiting to run
Tests / Code Coverage / repo-analysis (push) Blocked by required conditions
Tests / Code Coverage / test-sim-nondeterminism (push) Waiting to run
Tests / Code Coverage / test-clientv2 (push) Waiting to run
Tests / Code Coverage / test-core (push) Waiting to run
Tests / Code Coverage / test-depinject (push) Waiting to run
Tests / Code Coverage / test-errors (push) Waiting to run
Tests / Code Coverage / test-math (push) Waiting to run
Tests / Code Coverage / test-schema (push) Waiting to run
Tests / Code Coverage / test-collections (push) Waiting to run
Tests / Code Coverage / test-cosmovisor (push) Waiting to run
Tests / Code Coverage / test-confix (push) Waiting to run
Tests / Code Coverage / test-store (push) Waiting to run
Tests / Code Coverage / test-log (push) Waiting to run
Tests / Code Coverage / test-x-tx (push) Waiting to run
Tests / Code Coverage / test-x-nft (push) Waiting to run
Tests / Code Coverage / test-x-circuit (push) Waiting to run
Tests / Code Coverage / test-x-feegrant (push) Waiting to run
Tests / Code Coverage / test-x-evidence (push) Waiting to run
Tests / Code Coverage / test-x-upgrade (push) Waiting to run
Tests / Code Coverage / test-tools-benchmark (push) Waiting to run
Build & Push SDK Proto Builder / build (push) Has been cancelled
293 lines
9.5 KiB
Go
293 lines
9.5 KiB
Go
package genutil
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
|
|
"cosmossdk.io/core/address"
|
|
|
|
"github.com/cosmos/cosmos-sdk/codec"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
authtypes "github.com/cosmos/cosmos-sdk/x/auth/types"
|
|
authvesting "github.com/cosmos/cosmos-sdk/x/auth/vesting/types"
|
|
banktypes "github.com/cosmos/cosmos-sdk/x/bank/types"
|
|
genutiltypes "github.com/cosmos/cosmos-sdk/x/genutil/types"
|
|
)
|
|
|
|
// AddGenesisAccount adds a genesis account to the genesis state.
|
|
// Where `cdc` is client codec, `genesisFileUrl` is the path/url of current genesis file,
|
|
// `accAddr` is the address to be added to the genesis state, `amountStr` is the list of initial coins
|
|
// to be added for the account, `appendAcct` updates the account if already exists.
|
|
// `vestingStart, vestingEnd and vestingAmtStr` respectively are the schedule start time, end time (unix epoch)
|
|
// `moduleName“ is the module name for which the account is being created
|
|
// and coins to be appended to the account already in the genesis.json file.
|
|
func AddGenesisAccount(
|
|
cdc codec.Codec,
|
|
accAddr sdk.AccAddress,
|
|
appendAcct bool,
|
|
genesisFileURL, amountStr, vestingAmtStr string,
|
|
vestingStart, vestingEnd int64,
|
|
moduleName string,
|
|
) error {
|
|
coins, err := sdk.ParseCoinsNormalized(amountStr)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to parse coins: %w", err)
|
|
}
|
|
|
|
vestingAmt, err := sdk.ParseCoinsNormalized(vestingAmtStr)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to parse vesting amount: %w", err)
|
|
}
|
|
|
|
// create concrete account type based on input parameters
|
|
var genAccount authtypes.GenesisAccount
|
|
|
|
balances := banktypes.Balance{Address: accAddr.String(), Coins: coins.Sort()}
|
|
baseAccount := authtypes.NewBaseAccount(accAddr, nil, 0, 0)
|
|
|
|
if !vestingAmt.IsZero() {
|
|
baseVestingAccount, err := authvesting.NewBaseVestingAccount(baseAccount, vestingAmt.Sort(), vestingEnd)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create base vesting account: %w", err)
|
|
}
|
|
|
|
if (balances.Coins.IsZero() && !baseVestingAccount.OriginalVesting.IsZero()) ||
|
|
baseVestingAccount.OriginalVesting.IsAnyGT(balances.Coins) {
|
|
return errors.New("vesting amount cannot be greater than total amount")
|
|
}
|
|
|
|
switch {
|
|
case vestingStart != 0 && vestingEnd != 0:
|
|
genAccount = authvesting.NewContinuousVestingAccountRaw(baseVestingAccount, vestingStart)
|
|
|
|
case vestingEnd != 0:
|
|
genAccount = authvesting.NewDelayedVestingAccountRaw(baseVestingAccount)
|
|
|
|
default:
|
|
return errors.New("invalid vesting parameters; must supply start and end time or end time")
|
|
}
|
|
} else if moduleName != "" {
|
|
genAccount = authtypes.NewEmptyModuleAccount(moduleName, authtypes.Burner, authtypes.Minter)
|
|
} else {
|
|
genAccount = baseAccount
|
|
}
|
|
|
|
if err := genAccount.Validate(); err != nil {
|
|
return fmt.Errorf("failed to validate new genesis account: %w", err)
|
|
}
|
|
|
|
appState, appGenesis, err := genutiltypes.GenesisStateFromGenFile(genesisFileURL)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to unmarshal genesis state: %w", err)
|
|
}
|
|
|
|
authGenState := authtypes.GetGenesisStateFromAppState(cdc, appState)
|
|
|
|
accs, err := authtypes.UnpackAccounts(authGenState.Accounts)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to get accounts from any: %w", err)
|
|
}
|
|
|
|
bankGenState := banktypes.GetGenesisStateFromAppState(cdc, appState)
|
|
if accs.Contains(accAddr) {
|
|
if !appendAcct {
|
|
return fmt.Errorf(" Account %s already exists\nUse `append` flag to append account at existing address", accAddr)
|
|
}
|
|
|
|
genesisB := banktypes.GetGenesisStateFromAppState(cdc, appState)
|
|
for idx, acc := range genesisB.Balances {
|
|
if acc.Address != accAddr.String() {
|
|
continue
|
|
}
|
|
|
|
updatedCoins := acc.Coins.Add(coins...)
|
|
bankGenState.Balances[idx] = banktypes.Balance{Address: accAddr.String(), Coins: updatedCoins.Sort()}
|
|
break
|
|
}
|
|
} else {
|
|
// Add the new account to the set of genesis accounts and sanitize the accounts afterwards.
|
|
accs = append(accs, genAccount)
|
|
accs = authtypes.SanitizeGenesisAccounts(accs)
|
|
|
|
genAccs, err := authtypes.PackAccounts(accs)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to convert accounts into any's: %w", err)
|
|
}
|
|
authGenState.Accounts = genAccs
|
|
|
|
authGenStateBz, err := cdc.MarshalJSON(&authGenState)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshal auth genesis state: %w", err)
|
|
}
|
|
appState[authtypes.ModuleName] = authGenStateBz
|
|
|
|
bankGenState.Balances = append(bankGenState.Balances, balances)
|
|
}
|
|
|
|
bankGenState.Balances = banktypes.SanitizeGenesisBalances(bankGenState.Balances)
|
|
|
|
bankGenState.Supply = bankGenState.Supply.Add(balances.Coins...)
|
|
|
|
bankGenStateBz, err := cdc.MarshalJSON(bankGenState)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshal bank genesis state: %w", err)
|
|
}
|
|
appState[banktypes.ModuleName] = bankGenStateBz
|
|
|
|
appStateJSON, err := json.Marshal(appState)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshal application genesis state: %w", err)
|
|
}
|
|
|
|
appGenesis.AppState = appStateJSON
|
|
return ExportGenesisFile(appGenesis, genesisFileURL)
|
|
}
|
|
|
|
type GenesisAccount struct {
|
|
// Base
|
|
Address string `json:"address"`
|
|
Coins sdk.Coins `json:"coins"`
|
|
|
|
// Vesting
|
|
VestingAmt sdk.Coins `json:"vesting_amt,omitempty"`
|
|
VestingStart int64 `json:"vesting_start,omitempty"`
|
|
VestingEnd int64 `json:"vesting_end,omitempty"`
|
|
|
|
// Module
|
|
ModuleName string `json:"module_name,omitempty"`
|
|
}
|
|
|
|
// AddGenesisAccounts adds genesis accounts to the genesis state.
|
|
// Where `cdc` is the client codec, `accounts` are the genesis accounts to add,
|
|
// `appendAcct` updates the account if already exists, and `genesisFileURL` is the path/url of the current genesis file.
|
|
func AddGenesisAccounts(
|
|
cdc codec.Codec,
|
|
ac address.Codec,
|
|
accounts []GenesisAccount,
|
|
appendAcct bool,
|
|
genesisFileURL string,
|
|
) error {
|
|
appState, appGenesis, err := genutiltypes.GenesisStateFromGenFile(genesisFileURL)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to unmarshal genesis state: %w", err)
|
|
}
|
|
|
|
authGenState := authtypes.GetGenesisStateFromAppState(cdc, appState)
|
|
bankGenState := banktypes.GetGenesisStateFromAppState(cdc, appState)
|
|
|
|
accs, err := authtypes.UnpackAccounts(authGenState.Accounts)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to get accounts from any: %w", err)
|
|
}
|
|
|
|
newSupplyCoinsCache := sdk.NewCoins()
|
|
balanceCache := make(map[string]banktypes.Balance)
|
|
for _, acc := range accs {
|
|
for _, balance := range bankGenState.GetBalances() {
|
|
if balance.Address == acc.GetAddress().String() {
|
|
balanceCache[acc.GetAddress().String()] = balance
|
|
}
|
|
}
|
|
}
|
|
|
|
for _, acc := range accounts {
|
|
addr := acc.Address
|
|
coins := acc.Coins
|
|
|
|
accAddr, err := ac.StringToBytes(addr)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to parse account address %s: %w", addr, err)
|
|
}
|
|
|
|
// create concrete account type based on input parameters
|
|
var genAccount authtypes.GenesisAccount
|
|
|
|
balances := banktypes.Balance{Address: addr, Coins: coins.Sort()}
|
|
baseAccount := authtypes.NewBaseAccount(accAddr, nil, 0, 0)
|
|
|
|
vestingAmt := acc.VestingAmt
|
|
if !vestingAmt.IsZero() {
|
|
vestingStart := acc.VestingStart
|
|
vestingEnd := acc.VestingEnd
|
|
|
|
baseVestingAccount, err := authvesting.NewBaseVestingAccount(baseAccount, vestingAmt.Sort(), vestingEnd)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create base vesting account: %w", err)
|
|
}
|
|
|
|
if (balances.Coins.IsZero() && !baseVestingAccount.OriginalVesting.IsZero()) ||
|
|
baseVestingAccount.OriginalVesting.IsAnyGT(balances.Coins) {
|
|
return errors.New("vesting amount cannot be greater than total amount")
|
|
}
|
|
|
|
switch {
|
|
case vestingStart != 0 && vestingEnd != 0:
|
|
genAccount = authvesting.NewContinuousVestingAccountRaw(baseVestingAccount, vestingStart)
|
|
|
|
case vestingEnd != 0:
|
|
genAccount = authvesting.NewDelayedVestingAccountRaw(baseVestingAccount)
|
|
|
|
default:
|
|
return errors.New("invalid vesting parameters; must supply start and end time or end time")
|
|
}
|
|
} else if acc.ModuleName != "" {
|
|
genAccount = authtypes.NewEmptyModuleAccount(acc.ModuleName, authtypes.Burner, authtypes.Minter)
|
|
} else {
|
|
genAccount = baseAccount
|
|
}
|
|
|
|
if err := genAccount.Validate(); err != nil {
|
|
return fmt.Errorf("failed to validate new genesis account: %w", err)
|
|
}
|
|
|
|
if _, ok := balanceCache[addr]; ok {
|
|
if !appendAcct {
|
|
return fmt.Errorf(" Account %s already exists\nUse `append` flag to append account at existing address", accAddr)
|
|
}
|
|
|
|
for idx, acc := range bankGenState.Balances {
|
|
if acc.Address != addr {
|
|
continue
|
|
}
|
|
|
|
updatedCoins := acc.Coins.Add(coins...)
|
|
bankGenState.Balances[idx] = banktypes.Balance{Address: addr, Coins: updatedCoins.Sort()}
|
|
break
|
|
}
|
|
} else {
|
|
accs = append(accs, genAccount)
|
|
bankGenState.Balances = append(bankGenState.Balances, balances)
|
|
}
|
|
|
|
newSupplyCoinsCache = newSupplyCoinsCache.Add(coins...)
|
|
}
|
|
|
|
accs = authtypes.SanitizeGenesisAccounts(accs)
|
|
|
|
authGenState.Accounts, err = authtypes.PackAccounts(accs)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to convert accounts into any's: %w", err)
|
|
}
|
|
|
|
appState[authtypes.ModuleName], err = cdc.MarshalJSON(&authGenState)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshal auth genesis state: %w", err)
|
|
}
|
|
|
|
bankGenState.Balances = banktypes.SanitizeGenesisBalances(bankGenState.Balances)
|
|
bankGenState.Supply = bankGenState.Supply.Add(newSupplyCoinsCache...)
|
|
|
|
appState[banktypes.ModuleName], err = cdc.MarshalJSON(bankGenState)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshal bank genesis state: %w", err)
|
|
}
|
|
|
|
appStateJSON, err := json.Marshal(appState)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshal application genesis state: %w", err)
|
|
}
|
|
|
|
appGenesis.AppState = appStateJSON
|
|
return ExportGenesisFile(appGenesis, genesisFileURL)
|
|
}
|