mirror of
https://github.com/0glabs/0g-chain.git
synced 2025-01-27 07:25:17 +00:00
247 lines
6.9 KiB
Go
247 lines
6.9 KiB
Go
// Copyright 2016 All in Bits, inc
|
|
// Modifications copyright 2018 Kava Labs
|
|
|
|
package app
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/cosmos/cosmos-sdk/client"
|
|
"github.com/spf13/pflag"
|
|
"github.com/tendermint/tendermint/crypto"
|
|
tmtypes "github.com/tendermint/tendermint/types"
|
|
|
|
"github.com/cosmos/cosmos-sdk/server"
|
|
"github.com/cosmos/cosmos-sdk/server/config"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
"github.com/cosmos/cosmos-sdk/wire"
|
|
"github.com/cosmos/cosmos-sdk/x/auth"
|
|
"github.com/cosmos/cosmos-sdk/x/stake"
|
|
)
|
|
|
|
const defaultPassword = "password"
|
|
|
|
var (
|
|
// Tokens given to genesis validators and accounts
|
|
numStartingTokensValidators = int64(1000)
|
|
numStartingTokensAccounts = int64(99000)
|
|
)
|
|
|
|
// Initial app state to be written to (and read from) genesis file
|
|
type GenesisState struct {
|
|
Accounts []GenesisAccount `json:"accounts"`
|
|
StakeData stake.GenesisState `json:"stake"`
|
|
}
|
|
|
|
// A simplified version of a normal account. It doesn't have pubkey or sequence.
|
|
type GenesisAccount struct {
|
|
Address sdk.AccAddress `json:"address"`
|
|
Coins sdk.Coins `json:"coins"`
|
|
}
|
|
|
|
// TODO remove?
|
|
func NewGenesisAccount(acc *auth.BaseAccount) GenesisAccount {
|
|
return GenesisAccount{
|
|
Address: acc.Address,
|
|
Coins: acc.Coins,
|
|
}
|
|
}
|
|
|
|
// TODO remove?
|
|
func NewGenesisAccountI(acc auth.Account) GenesisAccount {
|
|
return GenesisAccount{
|
|
Address: acc.GetAddress(),
|
|
Coins: acc.GetCoins(),
|
|
}
|
|
}
|
|
|
|
// Converts a GenesisAccount to auth.BaseAccount TODO rename
|
|
func (ga *GenesisAccount) ToAccount() (acc *auth.BaseAccount) {
|
|
return &auth.BaseAccount{
|
|
Address: ga.Address,
|
|
Coins: ga.Coins.Sort(),
|
|
}
|
|
}
|
|
|
|
// Create the appInit struct for server init command
|
|
func KavaAppInit() server.AppInit {
|
|
fsAppGenState := pflag.NewFlagSet("", pflag.ContinueOnError)
|
|
fsAppGenTx := pflag.NewFlagSet("", pflag.ContinueOnError)
|
|
|
|
fsAppGenTx.String(server.FlagName, "", "validator moniker, required")
|
|
fsAppGenTx.String(server.FlagClientHome, DefaultCLIHome,
|
|
"home directory for the client, used for key generation")
|
|
fsAppGenTx.Bool(server.FlagOWK, false, "overwrite the accounts created")
|
|
|
|
return server.AppInit{
|
|
FlagsAppGenState: fsAppGenState,
|
|
FlagsAppGenTx: fsAppGenTx,
|
|
AppGenTx: KavaAppGenTx,
|
|
AppGenState: KavaAppGenStateJSON,
|
|
}
|
|
}
|
|
|
|
// Define format for GenTx json
|
|
type KavaGenTx struct {
|
|
Name string `json:"name"`
|
|
Address sdk.AccAddress `json:"address"`
|
|
PubKey string `json:"pub_key"`
|
|
}
|
|
|
|
// Generate a genesis transaction
|
|
func KavaAppGenTx(cdc *wire.Codec, pk crypto.PubKey, genTxConfig config.GenTx) (
|
|
appGenTx, cliPrint json.RawMessage, validator tmtypes.GenesisValidator, err error) {
|
|
|
|
// check a name for the new validator has been specified
|
|
if genTxConfig.Name == "" {
|
|
return nil, nil, tmtypes.GenesisValidator{}, errors.New("Must specify --name (validator moniker)")
|
|
}
|
|
|
|
// get a new password for the new account
|
|
buf := client.BufferStdin()
|
|
prompt := fmt.Sprintf("Password for account '%s' (default %s):", genTxConfig.Name, defaultPassword)
|
|
keyPass, err := client.GetPassword(prompt, buf)
|
|
if err != nil && keyPass != "" {
|
|
// An error was returned that either failed to read the password from
|
|
// STDIN or the given password is not empty but failed to meet minimum
|
|
// length requirements.
|
|
return appGenTx, cliPrint, validator, err
|
|
}
|
|
if keyPass == "" {
|
|
keyPass = defaultPassword
|
|
}
|
|
|
|
// generate a new address and secret key
|
|
addr, secret, err := server.GenerateSaveCoinKey(
|
|
genTxConfig.CliRoot,
|
|
genTxConfig.Name,
|
|
defaultPassword,
|
|
genTxConfig.Overwrite,
|
|
)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
// Create message string to print out
|
|
mm := map[string]string{"secret": secret}
|
|
var bz []byte
|
|
bz, err = cdc.MarshalJSON(mm)
|
|
if err != nil {
|
|
return
|
|
}
|
|
cliPrint = json.RawMessage(bz)
|
|
|
|
// Create genTx and validator
|
|
appGenTx, _, validator, err = KavaAppGenTxNF(cdc, pk, addr, genTxConfig.Name)
|
|
|
|
return
|
|
}
|
|
|
|
// TODO combine with KavaAppGenTx
|
|
func KavaAppGenTxNF(cdc *wire.Codec, pk crypto.PubKey, addr sdk.AccAddress, name string) (
|
|
appGenTx, cliPrint json.RawMessage, validator tmtypes.GenesisValidator, err error) {
|
|
|
|
// Create the gentx
|
|
var bz []byte
|
|
genTx := KavaGenTx{
|
|
Name: name,
|
|
Address: addr,
|
|
PubKey: sdk.MustBech32ifyAccPub(pk),
|
|
}
|
|
bz, err = wire.MarshalJSONIndent(cdc, genTx)
|
|
if err != nil {
|
|
return
|
|
}
|
|
appGenTx = json.RawMessage(bz)
|
|
|
|
// Create the validator
|
|
validator = tmtypes.GenesisValidator{
|
|
PubKey: pk,
|
|
Power: numStartingTokensValidators,
|
|
}
|
|
return
|
|
}
|
|
|
|
// Create the core parameters for genesis initialization
|
|
// note that the pubkey input is this machines pubkey
|
|
func KavaAppGenState(cdc *wire.Codec, appGenTxs []json.RawMessage) (genesisState GenesisState, err error) {
|
|
|
|
if len(appGenTxs) == 0 {
|
|
err = errors.New("must provide at least 1 genesis transaction")
|
|
return
|
|
}
|
|
|
|
// start with the default staking genesis state
|
|
stakeData := stake.DefaultGenesisState()
|
|
// change denom of staking coin
|
|
stakeData.Params.BondDenom = "KVA"
|
|
// drastically shorten unbonding time for test purposes
|
|
stakeData.Params.UnbondingTime = time.Second * 60 * 5 // 5 minutes
|
|
|
|
// get genesis flag account information
|
|
genaccs := make([]GenesisAccount, len(appGenTxs))
|
|
for i, appGenTx := range appGenTxs {
|
|
|
|
var genTx KavaGenTx
|
|
err = cdc.UnmarshalJSON(appGenTx, &genTx)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
// create the genesis account
|
|
accAuth := auth.NewBaseAccountWithAddress(genTx.Address)
|
|
accAuth.Coins = sdk.Coins{
|
|
{"KVA", sdk.NewInt(numStartingTokensAccounts)},
|
|
}
|
|
acc := NewGenesisAccount(&accAuth)
|
|
genaccs[i] = acc
|
|
stakeData.Pool.LooseTokens = stakeData.Pool.LooseTokens.Add(sdk.NewRat(numStartingTokensAccounts)) // increase the supply
|
|
|
|
// add the validator
|
|
if len(genTx.Name) > 0 {
|
|
desc := stake.NewDescription(genTx.Name, "", "", "")
|
|
validator := stake.NewValidator(genTx.Address,
|
|
sdk.MustGetAccPubKeyBech32(genTx.PubKey), desc)
|
|
|
|
stakeData.Pool.LooseTokens = stakeData.Pool.LooseTokens.Add(sdk.NewRat(numStartingTokensValidators)) // increase the supply
|
|
|
|
// add some new shares to the validator
|
|
var issuedDelShares sdk.Rat
|
|
validator, stakeData.Pool, issuedDelShares = validator.AddTokensFromDel(stakeData.Pool, numStartingTokensValidators)
|
|
stakeData.Validators = append(stakeData.Validators, validator)
|
|
|
|
// create the self-delegation from the issuedDelShares
|
|
delegation := stake.Delegation{
|
|
DelegatorAddr: validator.Owner,
|
|
ValidatorAddr: validator.Owner,
|
|
Shares: issuedDelShares,
|
|
Height: 0,
|
|
}
|
|
|
|
stakeData.Bonds = append(stakeData.Bonds, delegation)
|
|
}
|
|
}
|
|
|
|
// create the final app state
|
|
genesisState = GenesisState{
|
|
Accounts: genaccs,
|
|
StakeData: stakeData,
|
|
}
|
|
return
|
|
}
|
|
|
|
// Run KavaAppGenState then convert to JSON
|
|
func KavaAppGenStateJSON(cdc *wire.Codec, appGenTxs []json.RawMessage) (appState json.RawMessage, err error) {
|
|
|
|
// create the final app state
|
|
genesisState, err := KavaAppGenState(cdc, appGenTxs)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
appState, err = wire.MarshalJSONIndent(cdc, genesisState)
|
|
return
|
|
}
|