2018-01-07 12:27:56 -08:00
|
|
|
package types
|
2018-01-06 14:22:21 -08:00
|
|
|
|
|
|
|
import (
|
2018-08-04 22:56:48 -07:00
|
|
|
"bytes"
|
2018-03-17 11:18:04 -07:00
|
|
|
"encoding/hex"
|
2018-07-09 16:06:05 -07:00
|
|
|
"encoding/json"
|
2018-03-17 11:18:04 -07:00
|
|
|
"errors"
|
2018-05-26 14:21:29 -07:00
|
|
|
"fmt"
|
2019-02-08 15:54:40 -08:00
|
|
|
"strings"
|
2018-03-17 11:18:04 -07:00
|
|
|
|
2018-06-28 17:54:47 -07:00
|
|
|
"github.com/tendermint/tendermint/crypto"
|
2019-07-22 08:26:42 -07:00
|
|
|
yaml "gopkg.in/yaml.v2"
|
2018-07-25 13:43:37 -07:00
|
|
|
|
2020-06-04 03:38:24 -07:00
|
|
|
tmamino "github.com/cosmos/cosmos-sdk/crypto/codec"
|
2020-05-12 09:53:30 -07:00
|
|
|
"github.com/cosmos/cosmos-sdk/types/bech32"
|
2018-01-06 14:22:21 -08:00
|
|
|
)
|
|
|
|
|
2018-05-28 16:27:34 -07:00
|
|
|
const (
|
2019-06-04 05:59:14 -07:00
|
|
|
// Constants defined here are the defaults value for address.
|
|
|
|
// You can use the specific values for your project.
|
|
|
|
// Add the follow lines to the `main()` of your server.
|
|
|
|
//
|
|
|
|
// config := sdk.GetConfig()
|
|
|
|
// config.SetBech32PrefixForAccount(yourBech32PrefixAccAddr, yourBech32PrefixAccPub)
|
|
|
|
// config.SetBech32PrefixForValidator(yourBech32PrefixValAddr, yourBech32PrefixValPub)
|
|
|
|
// config.SetBech32PrefixForConsensusNode(yourBech32PrefixConsAddr, yourBech32PrefixConsPub)
|
|
|
|
// config.SetCoinType(yourCoinType)
|
|
|
|
// config.SetFullFundraiserPath(yourFullFundraiserPath)
|
|
|
|
// config.Seal()
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// AddrLen defines a valid address length
|
2018-07-04 14:07:06 -07:00
|
|
|
AddrLen = 20
|
2019-02-18 13:35:08 -08:00
|
|
|
// Bech32PrefixAccAddr defines the Bech32 prefix of an account's address
|
|
|
|
Bech32MainPrefix = "cosmos"
|
|
|
|
|
2019-05-30 05:46:38 -07:00
|
|
|
// Atom in https://github.com/satoshilabs/slips/blob/master/slip-0044.md
|
|
|
|
CoinType = 118
|
|
|
|
|
2019-06-04 05:59:14 -07:00
|
|
|
// BIP44Prefix is the parts of the BIP44 HD path that are fixed by
|
2019-05-30 05:46:38 -07:00
|
|
|
// what we used during the fundraiser.
|
|
|
|
FullFundraiserPath = "44'/118'/0'/0/0"
|
|
|
|
|
2019-02-18 13:35:08 -08:00
|
|
|
// PrefixAccount is the prefix for account keys
|
|
|
|
PrefixAccount = "acc"
|
|
|
|
// PrefixValidator is the prefix for validator keys
|
|
|
|
PrefixValidator = "val"
|
|
|
|
// PrefixConsensus is the prefix for consensus keys
|
|
|
|
PrefixConsensus = "cons"
|
|
|
|
// PrefixPublic is the prefix for public keys
|
|
|
|
PrefixPublic = "pub"
|
|
|
|
// PrefixOperator is the prefix for operator keys
|
|
|
|
PrefixOperator = "oper"
|
|
|
|
|
|
|
|
// PrefixAddress is the prefix for addresses
|
|
|
|
PrefixAddress = "addr"
|
2018-07-04 14:07:06 -07:00
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// Bech32PrefixAccAddr defines the Bech32 prefix of an account's address
|
2019-02-18 13:35:08 -08:00
|
|
|
Bech32PrefixAccAddr = Bech32MainPrefix
|
2018-08-30 21:06:44 -07:00
|
|
|
// Bech32PrefixAccPub defines the Bech32 prefix of an account's public key
|
2019-02-18 13:35:08 -08:00
|
|
|
Bech32PrefixAccPub = Bech32MainPrefix + PrefixPublic
|
2018-08-30 21:06:44 -07:00
|
|
|
// Bech32PrefixValAddr defines the Bech32 prefix of a validator's operator address
|
2019-02-18 13:35:08 -08:00
|
|
|
Bech32PrefixValAddr = Bech32MainPrefix + PrefixValidator + PrefixOperator
|
2018-08-30 21:06:44 -07:00
|
|
|
// Bech32PrefixValPub defines the Bech32 prefix of a validator's operator public key
|
2019-02-18 13:35:08 -08:00
|
|
|
Bech32PrefixValPub = Bech32MainPrefix + PrefixValidator + PrefixOperator + PrefixPublic
|
2018-08-30 21:06:44 -07:00
|
|
|
// Bech32PrefixConsAddr defines the Bech32 prefix of a consensus node address
|
2019-02-18 13:35:08 -08:00
|
|
|
Bech32PrefixConsAddr = Bech32MainPrefix + PrefixValidator + PrefixConsensus
|
2018-08-30 21:06:44 -07:00
|
|
|
// Bech32PrefixConsPub defines the Bech32 prefix of a consensus node public key
|
2019-02-18 13:35:08 -08:00
|
|
|
Bech32PrefixConsPub = Bech32MainPrefix + PrefixValidator + PrefixConsensus + PrefixPublic
|
2018-05-28 16:27:34 -07:00
|
|
|
)
|
|
|
|
|
2019-01-04 14:09:39 -08:00
|
|
|
// Address is a common interface for different types of addresses used by the SDK
|
|
|
|
type Address interface {
|
|
|
|
Equals(Address) bool
|
|
|
|
Empty() bool
|
|
|
|
Marshal() ([]byte, error)
|
|
|
|
MarshalJSON() ([]byte, error)
|
|
|
|
Bytes() []byte
|
|
|
|
String() string
|
|
|
|
Format(s fmt.State, verb rune)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure that different address types implement the interface
|
|
|
|
var _ Address = AccAddress{}
|
|
|
|
var _ Address = ValAddress{}
|
|
|
|
var _ Address = ConsAddress{}
|
|
|
|
|
2019-05-31 06:14:35 -07:00
|
|
|
var _ yaml.Marshaler = AccAddress{}
|
|
|
|
var _ yaml.Marshaler = ValAddress{}
|
|
|
|
var _ yaml.Marshaler = ConsAddress{}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// account
|
|
|
|
// ----------------------------------------------------------------------------
|
2018-07-06 00:06:53 -07:00
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// AccAddress a wrapper around bytes meant to represent an account address.
|
|
|
|
// When marshaled to a string or JSON, it uses Bech32.
|
2018-07-06 00:06:53 -07:00
|
|
|
type AccAddress []byte
|
2018-07-03 21:52:48 -07:00
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// AccAddressFromHex creates an AccAddress from a hex string.
|
2018-07-09 16:06:05 -07:00
|
|
|
func AccAddressFromHex(address string) (addr AccAddress, err error) {
|
2020-03-04 09:49:59 -08:00
|
|
|
bz, err := addressBytesFromHexString(address)
|
|
|
|
return AccAddress(bz), err
|
2018-07-06 00:06:53 -07:00
|
|
|
}
|
|
|
|
|
2019-05-02 12:36:42 -07:00
|
|
|
// VerifyAddressFormat verifies that the provided bytes form a valid address
|
|
|
|
// according to the default address rules or a custom address verifier set by
|
|
|
|
// GetConfig().SetAddressVerifier()
|
|
|
|
func VerifyAddressFormat(bz []byte) error {
|
|
|
|
verifier := GetConfig().GetAddressVerifier()
|
|
|
|
if verifier != nil {
|
|
|
|
return verifier(bz)
|
2019-05-16 08:25:32 -07:00
|
|
|
}
|
|
|
|
if len(bz) != AddrLen {
|
2019-08-19 09:06:27 -07:00
|
|
|
return errors.New("incorrect address length")
|
2019-05-02 12:36:42 -07:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// AccAddressFromBech32 creates an AccAddress from a Bech32 string.
|
2018-07-09 16:06:05 -07:00
|
|
|
func AccAddressFromBech32(address string) (addr AccAddress, err error) {
|
2019-02-08 15:54:40 -08:00
|
|
|
if len(strings.TrimSpace(address)) == 0 {
|
|
|
|
return AccAddress{}, nil
|
|
|
|
}
|
|
|
|
|
2018-10-31 12:13:13 -07:00
|
|
|
bech32PrefixAccAddr := GetConfig().GetBech32AccountAddrPrefix()
|
2019-02-08 15:54:40 -08:00
|
|
|
|
2018-10-31 12:13:13 -07:00
|
|
|
bz, err := GetFromBech32(address, bech32PrefixAccAddr)
|
2018-07-06 00:06:53 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2019-05-02 12:36:42 -07:00
|
|
|
err = VerifyAddressFormat(bz)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2019-02-08 15:54:40 -08:00
|
|
|
}
|
|
|
|
|
2018-07-06 00:06:53 -07:00
|
|
|
return AccAddress(bz), nil
|
2018-07-03 21:52:48 -07:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// Returns boolean for whether two AccAddresses are Equal
|
2019-01-04 14:09:39 -08:00
|
|
|
func (aa AccAddress) Equals(aa2 Address) bool {
|
2018-08-30 21:06:44 -07:00
|
|
|
if aa.Empty() && aa2.Empty() {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2019-02-08 12:45:23 -08:00
|
|
|
return bytes.Equal(aa.Bytes(), aa2.Bytes())
|
2018-08-30 21:06:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Returns boolean for whether an AccAddress is empty
|
|
|
|
func (aa AccAddress) Empty() bool {
|
|
|
|
if aa == nil {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
aa2 := AccAddress{}
|
2019-02-08 12:45:23 -08:00
|
|
|
return bytes.Equal(aa.Bytes(), aa2.Bytes())
|
2018-05-26 14:21:29 -07:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// Marshal returns the raw address bytes. It is needed for protobuf
|
|
|
|
// compatibility.
|
|
|
|
func (aa AccAddress) Marshal() ([]byte, error) {
|
|
|
|
return aa, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unmarshal sets the address to the given data. It is needed for protobuf
|
|
|
|
// compatibility.
|
|
|
|
func (aa *AccAddress) Unmarshal(data []byte) error {
|
|
|
|
*aa = data
|
2018-07-03 21:52:48 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// MarshalJSON marshals to JSON using Bech32.
|
|
|
|
func (aa AccAddress) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(aa.String())
|
2018-07-03 21:52:48 -07:00
|
|
|
}
|
|
|
|
|
2019-05-31 06:14:35 -07:00
|
|
|
// MarshalYAML marshals to YAML using Bech32.
|
|
|
|
func (aa AccAddress) MarshalYAML() (interface{}, error) {
|
|
|
|
return aa.String(), nil
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// UnmarshalJSON unmarshals from JSON assuming Bech32 encoding.
|
|
|
|
func (aa *AccAddress) UnmarshalJSON(data []byte) error {
|
2018-07-09 16:06:05 -07:00
|
|
|
var s string
|
|
|
|
err := json.Unmarshal(data, &s)
|
|
|
|
if err != nil {
|
2018-09-01 04:14:17 -07:00
|
|
|
return err
|
2018-07-03 21:52:48 -07:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
aa2, err := AccAddressFromBech32(s)
|
2018-07-03 21:52:48 -07:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-08-30 21:06:44 -07:00
|
|
|
|
|
|
|
*aa = aa2
|
2018-07-03 21:52:48 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-05-31 06:14:35 -07:00
|
|
|
// UnmarshalYAML unmarshals from JSON assuming Bech32 encoding.
|
|
|
|
func (aa *AccAddress) UnmarshalYAML(data []byte) error {
|
|
|
|
var s string
|
|
|
|
err := yaml.Unmarshal(data, &s)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
aa2, err := AccAddressFromBech32(s)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
*aa = aa2
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// Bytes returns the raw address bytes.
|
|
|
|
func (aa AccAddress) Bytes() []byte {
|
|
|
|
return aa
|
2018-07-03 21:52:48 -07:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// String implements the Stringer interface.
|
|
|
|
func (aa AccAddress) String() string {
|
2019-02-08 15:54:40 -08:00
|
|
|
if aa.Empty() {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2018-10-31 12:13:13 -07:00
|
|
|
bech32PrefixAccAddr := GetConfig().GetBech32AccountAddrPrefix()
|
2019-02-08 15:54:40 -08:00
|
|
|
|
2018-10-31 12:13:13 -07:00
|
|
|
bech32Addr, err := bech32.ConvertAndEncode(bech32PrefixAccAddr, aa.Bytes())
|
2018-06-07 15:32:14 -07:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2018-07-03 21:52:48 -07:00
|
|
|
return bech32Addr
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// Format implements the fmt.Formatter interface.
|
2018-08-31 15:22:37 -07:00
|
|
|
// nolint: errcheck
|
2018-08-30 21:06:44 -07:00
|
|
|
func (aa AccAddress) Format(s fmt.State, verb rune) {
|
2018-07-03 21:52:48 -07:00
|
|
|
switch verb {
|
|
|
|
case 's':
|
2019-02-08 12:45:23 -08:00
|
|
|
s.Write([]byte(aa.String()))
|
2018-07-03 21:52:48 -07:00
|
|
|
case 'p':
|
2018-08-30 21:06:44 -07:00
|
|
|
s.Write([]byte(fmt.Sprintf("%p", aa)))
|
2018-07-03 21:52:48 -07:00
|
|
|
default:
|
2018-08-30 21:06:44 -07:00
|
|
|
s.Write([]byte(fmt.Sprintf("%X", []byte(aa))))
|
2018-07-03 21:52:48 -07:00
|
|
|
}
|
2018-06-07 15:32:14 -07:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// ----------------------------------------------------------------------------
|
2018-09-08 01:44:58 -07:00
|
|
|
// validator operator
|
2018-08-30 21:06:44 -07:00
|
|
|
// ----------------------------------------------------------------------------
|
2018-08-04 22:56:48 -07:00
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// ValAddress defines a wrapper around bytes meant to present a validator's
|
|
|
|
// operator. When marshaled to a string or JSON, it uses Bech32.
|
2018-07-06 00:06:53 -07:00
|
|
|
type ValAddress []byte
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// ValAddressFromHex creates a ValAddress from a hex string.
|
2018-07-09 16:06:05 -07:00
|
|
|
func ValAddressFromHex(address string) (addr ValAddress, err error) {
|
2020-03-04 09:49:59 -08:00
|
|
|
bz, err := addressBytesFromHexString(address)
|
|
|
|
return ValAddress(bz), err
|
2018-05-26 14:21:29 -07:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// ValAddressFromBech32 creates a ValAddress from a Bech32 string.
|
2018-07-09 16:06:05 -07:00
|
|
|
func ValAddressFromBech32(address string) (addr ValAddress, err error) {
|
2019-02-08 15:54:40 -08:00
|
|
|
if len(strings.TrimSpace(address)) == 0 {
|
|
|
|
return ValAddress{}, nil
|
|
|
|
}
|
|
|
|
|
2018-10-31 12:13:13 -07:00
|
|
|
bech32PrefixValAddr := GetConfig().GetBech32ValidatorAddrPrefix()
|
2019-02-08 15:54:40 -08:00
|
|
|
|
2018-10-31 12:13:13 -07:00
|
|
|
bz, err := GetFromBech32(address, bech32PrefixValAddr)
|
2018-06-07 15:32:14 -07:00
|
|
|
if err != nil {
|
2018-07-06 00:06:53 -07:00
|
|
|
return nil, err
|
2018-06-07 15:32:14 -07:00
|
|
|
}
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2019-05-02 12:36:42 -07:00
|
|
|
err = VerifyAddressFormat(bz)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2019-02-08 15:54:40 -08:00
|
|
|
}
|
|
|
|
|
2018-07-06 00:06:53 -07:00
|
|
|
return ValAddress(bz), nil
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// Returns boolean for whether two ValAddresses are Equal
|
2019-01-04 14:09:39 -08:00
|
|
|
func (va ValAddress) Equals(va2 Address) bool {
|
2018-08-30 21:06:44 -07:00
|
|
|
if va.Empty() && va2.Empty() {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2019-02-08 12:45:23 -08:00
|
|
|
return bytes.Equal(va.Bytes(), va2.Bytes())
|
2018-08-30 21:06:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Returns boolean for whether an AccAddress is empty
|
|
|
|
func (va ValAddress) Empty() bool {
|
|
|
|
if va == nil {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
va2 := ValAddress{}
|
2019-02-08 12:45:23 -08:00
|
|
|
return bytes.Equal(va.Bytes(), va2.Bytes())
|
2018-08-30 21:06:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Marshal returns the raw address bytes. It is needed for protobuf
|
|
|
|
// compatibility.
|
|
|
|
func (va ValAddress) Marshal() ([]byte, error) {
|
|
|
|
return va, nil
|
2018-06-07 15:32:14 -07:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// Unmarshal sets the address to the given data. It is needed for protobuf
|
|
|
|
// compatibility.
|
|
|
|
func (va *ValAddress) Unmarshal(data []byte) error {
|
|
|
|
*va = data
|
2018-07-06 00:06:53 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// MarshalJSON marshals to JSON using Bech32.
|
|
|
|
func (va ValAddress) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(va.String())
|
2018-05-26 14:21:29 -07:00
|
|
|
}
|
|
|
|
|
2019-05-31 06:14:35 -07:00
|
|
|
// MarshalYAML marshals to YAML using Bech32.
|
|
|
|
func (va ValAddress) MarshalYAML() (interface{}, error) {
|
|
|
|
return va.String(), nil
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// UnmarshalJSON unmarshals from JSON assuming Bech32 encoding.
|
|
|
|
func (va *ValAddress) UnmarshalJSON(data []byte) error {
|
2018-07-09 16:06:05 -07:00
|
|
|
var s string
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2018-07-09 16:06:05 -07:00
|
|
|
err := json.Unmarshal(data, &s)
|
|
|
|
if err != nil {
|
2019-03-06 03:38:18 -08:00
|
|
|
return err
|
2018-07-06 00:06:53 -07:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
va2, err := ValAddressFromBech32(s)
|
2018-06-07 15:32:14 -07:00
|
|
|
if err != nil {
|
2018-07-06 00:06:53 -07:00
|
|
|
return err
|
2018-06-07 15:32:14 -07:00
|
|
|
}
|
2018-08-30 21:06:44 -07:00
|
|
|
|
|
|
|
*va = va2
|
2018-07-06 00:06:53 -07:00
|
|
|
return nil
|
2018-06-07 15:32:14 -07:00
|
|
|
}
|
|
|
|
|
2019-05-31 06:14:35 -07:00
|
|
|
// UnmarshalYAML unmarshals from YAML assuming Bech32 encoding.
|
|
|
|
func (va *ValAddress) UnmarshalYAML(data []byte) error {
|
|
|
|
var s string
|
|
|
|
|
|
|
|
err := yaml.Unmarshal(data, &s)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
va2, err := ValAddressFromBech32(s)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
*va = va2
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// Bytes returns the raw address bytes.
|
|
|
|
func (va ValAddress) Bytes() []byte {
|
|
|
|
return va
|
2018-05-26 14:21:29 -07:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// String implements the Stringer interface.
|
|
|
|
func (va ValAddress) String() string {
|
2019-02-08 15:54:40 -08:00
|
|
|
if va.Empty() {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2018-10-31 12:13:13 -07:00
|
|
|
bech32PrefixValAddr := GetConfig().GetBech32ValidatorAddrPrefix()
|
2019-02-08 15:54:40 -08:00
|
|
|
|
2018-10-31 12:13:13 -07:00
|
|
|
bech32Addr, err := bech32.ConvertAndEncode(bech32PrefixValAddr, va.Bytes())
|
2018-06-07 15:32:14 -07:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2018-07-06 00:06:53 -07:00
|
|
|
return bech32Addr
|
2018-06-07 15:32:14 -07:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// Format implements the fmt.Formatter interface.
|
2018-08-31 15:22:37 -07:00
|
|
|
// nolint: errcheck
|
2018-08-30 21:06:44 -07:00
|
|
|
func (va ValAddress) Format(s fmt.State, verb rune) {
|
2018-07-06 00:06:53 -07:00
|
|
|
switch verb {
|
|
|
|
case 's':
|
2019-02-08 12:45:23 -08:00
|
|
|
s.Write([]byte(va.String()))
|
2018-07-06 00:06:53 -07:00
|
|
|
case 'p':
|
2018-08-30 21:06:44 -07:00
|
|
|
s.Write([]byte(fmt.Sprintf("%p", va)))
|
2018-07-06 00:06:53 -07:00
|
|
|
default:
|
2018-08-30 21:06:44 -07:00
|
|
|
s.Write([]byte(fmt.Sprintf("%X", []byte(va))))
|
2018-03-17 11:18:04 -07:00
|
|
|
}
|
|
|
|
}
|
2018-05-26 14:21:29 -07:00
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// consensus node
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
// ConsAddress defines a wrapper around bytes meant to present a consensus node.
|
|
|
|
// When marshaled to a string or JSON, it uses Bech32.
|
|
|
|
type ConsAddress []byte
|
|
|
|
|
|
|
|
// ConsAddressFromHex creates a ConsAddress from a hex string.
|
|
|
|
func ConsAddressFromHex(address string) (addr ConsAddress, err error) {
|
2020-03-04 09:49:59 -08:00
|
|
|
bz, err := addressBytesFromHexString(address)
|
|
|
|
return ConsAddress(bz), err
|
2018-08-30 21:06:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// ConsAddressFromBech32 creates a ConsAddress from a Bech32 string.
|
|
|
|
func ConsAddressFromBech32(address string) (addr ConsAddress, err error) {
|
2019-02-08 15:54:40 -08:00
|
|
|
if len(strings.TrimSpace(address)) == 0 {
|
|
|
|
return ConsAddress{}, nil
|
|
|
|
}
|
|
|
|
|
2018-10-31 12:13:13 -07:00
|
|
|
bech32PrefixConsAddr := GetConfig().GetBech32ConsensusAddrPrefix()
|
2019-02-08 15:54:40 -08:00
|
|
|
|
2018-10-31 12:13:13 -07:00
|
|
|
bz, err := GetFromBech32(address, bech32PrefixConsAddr)
|
2018-08-30 21:06:44 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-05-02 12:36:42 -07:00
|
|
|
err = VerifyAddressFormat(bz)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2019-02-08 15:54:40 -08:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
return ConsAddress(bz), nil
|
|
|
|
}
|
|
|
|
|
2018-09-24 21:09:31 -07:00
|
|
|
// get ConsAddress from pubkey
|
|
|
|
func GetConsAddress(pubkey crypto.PubKey) ConsAddress {
|
|
|
|
return ConsAddress(pubkey.Address())
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// Returns boolean for whether two ConsAddress are Equal
|
2019-01-04 14:09:39 -08:00
|
|
|
func (ca ConsAddress) Equals(ca2 Address) bool {
|
2018-08-30 21:06:44 -07:00
|
|
|
if ca.Empty() && ca2.Empty() {
|
2018-08-16 18:35:17 -07:00
|
|
|
return true
|
|
|
|
}
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2019-02-08 12:45:23 -08:00
|
|
|
return bytes.Equal(ca.Bytes(), ca2.Bytes())
|
2018-08-04 22:56:48 -07:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// Returns boolean for whether an ConsAddress is empty
|
|
|
|
func (ca ConsAddress) Empty() bool {
|
|
|
|
if ca == nil {
|
2018-08-16 18:35:17 -07:00
|
|
|
return true
|
|
|
|
}
|
2018-08-30 21:06:44 -07:00
|
|
|
|
|
|
|
ca2 := ConsAddress{}
|
2019-02-08 12:45:23 -08:00
|
|
|
return bytes.Equal(ca.Bytes(), ca2.Bytes())
|
2018-08-30 21:06:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Marshal returns the raw address bytes. It is needed for protobuf
|
|
|
|
// compatibility.
|
|
|
|
func (ca ConsAddress) Marshal() ([]byte, error) {
|
|
|
|
return ca, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unmarshal sets the address to the given data. It is needed for protobuf
|
|
|
|
// compatibility.
|
|
|
|
func (ca *ConsAddress) Unmarshal(data []byte) error {
|
|
|
|
*ca = data
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON marshals to JSON using Bech32.
|
|
|
|
func (ca ConsAddress) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(ca.String())
|
|
|
|
}
|
|
|
|
|
2019-05-31 06:14:35 -07:00
|
|
|
// MarshalYAML marshals to YAML using Bech32.
|
|
|
|
func (ca ConsAddress) MarshalYAML() (interface{}, error) {
|
|
|
|
return ca.String(), nil
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// UnmarshalJSON unmarshals from JSON assuming Bech32 encoding.
|
|
|
|
func (ca *ConsAddress) UnmarshalJSON(data []byte) error {
|
|
|
|
var s string
|
|
|
|
|
|
|
|
err := json.Unmarshal(data, &s)
|
|
|
|
if err != nil {
|
2019-03-06 03:38:18 -08:00
|
|
|
return err
|
2018-08-30 21:06:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ca2, err := ConsAddressFromBech32(s)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-05-31 06:14:35 -07:00
|
|
|
*ca = ca2
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalYAML unmarshals from YAML assuming Bech32 encoding.
|
|
|
|
func (ca *ConsAddress) UnmarshalYAML(data []byte) error {
|
|
|
|
var s string
|
|
|
|
|
|
|
|
err := yaml.Unmarshal(data, &s)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
ca2, err := ConsAddressFromBech32(s)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
*ca = ca2
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Bytes returns the raw address bytes.
|
|
|
|
func (ca ConsAddress) Bytes() []byte {
|
|
|
|
return ca
|
|
|
|
}
|
|
|
|
|
|
|
|
// String implements the Stringer interface.
|
|
|
|
func (ca ConsAddress) String() string {
|
2019-02-08 15:54:40 -08:00
|
|
|
if ca.Empty() {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2018-10-31 12:13:13 -07:00
|
|
|
bech32PrefixConsAddr := GetConfig().GetBech32ConsensusAddrPrefix()
|
2019-02-08 15:54:40 -08:00
|
|
|
|
2018-10-31 12:13:13 -07:00
|
|
|
bech32Addr, err := bech32.ConvertAndEncode(bech32PrefixConsAddr, ca.Bytes())
|
2018-08-30 21:06:44 -07:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return bech32Addr
|
|
|
|
}
|
|
|
|
|
2020-01-29 01:48:53 -08:00
|
|
|
// Bech32ifyAddressBytes returns a bech32 representation of address bytes.
|
|
|
|
// Returns an empty sting if the byte slice is 0-length. Returns an error if the bech32 conversion
|
|
|
|
// fails or the prefix is empty.
|
|
|
|
func Bech32ifyAddressBytes(prefix string, bs []byte) (string, error) {
|
|
|
|
if len(bs) == 0 {
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
if len(prefix) == 0 {
|
|
|
|
return "", errors.New("prefix cannot be empty")
|
|
|
|
}
|
|
|
|
return bech32.ConvertAndEncode(prefix, bs)
|
|
|
|
}
|
|
|
|
|
|
|
|
// MustBech32ifyAddressBytes returns a bech32 representation of address bytes.
|
|
|
|
// Returns an empty sting if the byte slice is 0-length. It panics if the bech32 conversion
|
|
|
|
// fails or the prefix is empty.
|
|
|
|
func MustBech32ifyAddressBytes(prefix string, bs []byte) string {
|
|
|
|
s, err := Bech32ifyAddressBytes(prefix, bs)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// Format implements the fmt.Formatter interface.
|
2018-08-31 15:22:37 -07:00
|
|
|
// nolint: errcheck
|
2018-08-30 21:06:44 -07:00
|
|
|
func (ca ConsAddress) Format(s fmt.State, verb rune) {
|
|
|
|
switch verb {
|
|
|
|
case 's':
|
2019-02-08 12:45:23 -08:00
|
|
|
s.Write([]byte(ca.String()))
|
2018-08-30 21:06:44 -07:00
|
|
|
case 'p':
|
|
|
|
s.Write([]byte(fmt.Sprintf("%p", ca)))
|
|
|
|
default:
|
|
|
|
s.Write([]byte(fmt.Sprintf("%X", []byte(ca))))
|
|
|
|
}
|
2018-08-04 22:56:48 -07:00
|
|
|
}
|
|
|
|
|
2018-08-30 21:06:44 -07:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// auxiliary
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
// Bech32PubKeyType defines a string type alias for a Bech32 public key type.
|
|
|
|
type Bech32PubKeyType string
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
// Bech32 conversion constants
|
|
|
|
const (
|
|
|
|
Bech32PubKeyTypeAccPub Bech32PubKeyType = "accpub"
|
|
|
|
Bech32PubKeyTypeValPub Bech32PubKeyType = "valpub"
|
|
|
|
Bech32PubKeyTypeConsPub Bech32PubKeyType = "conspub"
|
|
|
|
)
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
// Bech32ifyPubKey returns a Bech32 encoded string containing the appropriate
|
|
|
|
// prefix based on the key type provided for a given PublicKey.
|
|
|
|
func Bech32ifyPubKey(pkt Bech32PubKeyType, pubkey crypto.PubKey) (string, error) {
|
|
|
|
var bech32Prefix string
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
switch pkt {
|
|
|
|
case Bech32PubKeyTypeAccPub:
|
|
|
|
bech32Prefix = GetConfig().GetBech32AccountPubPrefix()
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
case Bech32PubKeyTypeValPub:
|
|
|
|
bech32Prefix = GetConfig().GetBech32ValidatorPubPrefix()
|
2018-07-07 14:21:00 -07:00
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
case Bech32PubKeyTypeConsPub:
|
|
|
|
bech32Prefix = GetConfig().GetBech32ConsensusPubPrefix()
|
2018-06-05 21:53:04 -07:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
return bech32.ConvertAndEncode(bech32Prefix, pubkey.Bytes())
|
2018-06-05 21:53:04 -07:00
|
|
|
}
|
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
// MustBech32ifyPubKey calls Bech32ifyPubKey except it panics on error.
|
|
|
|
func MustBech32ifyPubKey(pkt Bech32PubKeyType, pubkey crypto.PubKey) string {
|
|
|
|
res, err := Bech32ifyPubKey(pkt, pubkey)
|
2018-07-07 14:21:00 -07:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
return res
|
2018-07-07 14:21:00 -07:00
|
|
|
}
|
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
// GetPubKeyFromBech32 returns a PublicKey from a bech32-encoded PublicKey with
|
|
|
|
// a given key type.
|
|
|
|
func GetPubKeyFromBech32(pkt Bech32PubKeyType, pubkeyStr string) (crypto.PubKey, error) {
|
|
|
|
var bech32Prefix string
|
2018-05-27 05:21:15 -07:00
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
switch pkt {
|
|
|
|
case Bech32PubKeyTypeAccPub:
|
|
|
|
bech32Prefix = GetConfig().GetBech32AccountPubPrefix()
|
2018-05-27 05:21:15 -07:00
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
case Bech32PubKeyTypeValPub:
|
|
|
|
bech32Prefix = GetConfig().GetBech32ValidatorPubPrefix()
|
2018-05-28 16:27:34 -07:00
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
case Bech32PubKeyTypeConsPub:
|
|
|
|
bech32Prefix = GetConfig().GetBech32ConsensusPubPrefix()
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
}
|
2018-07-07 14:21:00 -07:00
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
bz, err := GetFromBech32(pubkeyStr, bech32Prefix)
|
2018-08-30 21:06:44 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
pk, err := tmamino.PubKeyFromBytes(bz)
|
2018-08-30 21:06:44 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return pk, nil
|
|
|
|
}
|
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
// MustGetPubKeyFromBech32 calls GetPubKeyFromBech32 except it panics on error.
|
|
|
|
func MustGetPubKeyFromBech32(pkt Bech32PubKeyType, pubkeyStr string) crypto.PubKey {
|
|
|
|
res, err := GetPubKeyFromBech32(pkt, pubkeyStr)
|
2018-08-30 21:06:44 -07:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2020-01-09 06:04:28 -08:00
|
|
|
return res
|
2018-08-30 21:06:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetFromBech32 decodes a bytestring from a Bech32 encoded string.
|
2018-06-11 13:09:29 -07:00
|
|
|
func GetFromBech32(bech32str, prefix string) ([]byte, error) {
|
2018-06-01 07:23:58 -07:00
|
|
|
if len(bech32str) == 0 {
|
2018-08-30 21:06:44 -07:00
|
|
|
return nil, errors.New("decoding Bech32 address failed: must provide an address")
|
2018-05-28 16:27:34 -07:00
|
|
|
}
|
2018-08-30 21:06:44 -07:00
|
|
|
|
2018-06-01 07:23:58 -07:00
|
|
|
hrp, bz, err := bech32.DecodeAndConvert(bech32str)
|
2018-05-28 16:27:34 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if hrp != prefix {
|
2018-08-30 21:06:44 -07:00
|
|
|
return nil, fmt.Errorf("invalid Bech32 prefix; expected %s, got %s", prefix, hrp)
|
2018-05-28 16:27:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return bz, nil
|
|
|
|
}
|
2020-03-04 09:49:59 -08:00
|
|
|
|
|
|
|
func addressBytesFromHexString(address string) ([]byte, error) {
|
|
|
|
if len(address) == 0 {
|
|
|
|
return nil, errors.New("decoding Bech32 address failed: must provide an address")
|
|
|
|
}
|
|
|
|
|
|
|
|
return hex.DecodeString(address)
|
|
|
|
}
|