2018-01-22 05:44:24 -08:00
|
|
|
package auth
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"reflect"
|
|
|
|
|
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
2018-03-02 01:24:07 -08:00
|
|
|
wire "github.com/cosmos/cosmos-sdk/wire"
|
2018-01-22 05:44:24 -08:00
|
|
|
)
|
|
|
|
|
2018-03-12 15:25:55 -07:00
|
|
|
var _ sdk.AccountMapper = (*accountMapper)(nil)
|
|
|
|
|
2018-01-22 05:44:24 -08:00
|
|
|
// Implements sdk.AccountMapper.
|
|
|
|
// This AccountMapper encodes/decodes accounts using the
|
2018-04-06 17:25:08 -07:00
|
|
|
// go-amino (binary) encoding/decoding library.
|
2018-01-22 05:44:24 -08:00
|
|
|
type accountMapper struct {
|
|
|
|
|
|
|
|
// The (unexposed) key used to access the store from the Context.
|
|
|
|
key sdk.StoreKey
|
|
|
|
|
|
|
|
// The prototypical sdk.Account concrete type.
|
|
|
|
proto sdk.Account
|
|
|
|
|
|
|
|
// The wire codec for binary encoding/decoding of accounts.
|
|
|
|
cdc *wire.Codec
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewAccountMapper returns a new sdk.AccountMapper that
|
2018-04-06 17:25:08 -07:00
|
|
|
// uses go-amino to (binary) encode and decode concrete sdk.Accounts.
|
2018-04-18 21:49:24 -07:00
|
|
|
// nolint
|
2018-04-07 00:02:00 -07:00
|
|
|
func NewAccountMapper(cdc *wire.Codec, key sdk.StoreKey, proto sdk.Account) accountMapper {
|
2018-01-22 05:44:24 -08:00
|
|
|
return accountMapper{
|
|
|
|
key: key,
|
|
|
|
proto: proto,
|
|
|
|
cdc: cdc,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Implements sdk.AccountMapper.
|
2018-03-01 23:49:07 -08:00
|
|
|
func (am accountMapper) NewAccountWithAddress(ctx sdk.Context, addr sdk.Address) sdk.Account {
|
2018-01-22 05:44:24 -08:00
|
|
|
acc := am.clonePrototype()
|
|
|
|
acc.SetAddress(addr)
|
|
|
|
return acc
|
|
|
|
}
|
|
|
|
|
|
|
|
// Implements sdk.AccountMapper.
|
2018-03-01 23:49:07 -08:00
|
|
|
func (am accountMapper) GetAccount(ctx sdk.Context, addr sdk.Address) sdk.Account {
|
2018-01-22 05:44:24 -08:00
|
|
|
store := ctx.KVStore(am.key)
|
|
|
|
bz := store.Get(addr)
|
|
|
|
if bz == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
acc := am.decodeAccount(bz)
|
|
|
|
return acc
|
|
|
|
}
|
|
|
|
|
|
|
|
// Implements sdk.AccountMapper.
|
|
|
|
func (am accountMapper) SetAccount(ctx sdk.Context, acc sdk.Account) {
|
|
|
|
addr := acc.GetAddress()
|
|
|
|
store := ctx.KVStore(am.key)
|
|
|
|
bz := am.encodeAccount(acc)
|
|
|
|
store.Set(addr, bz)
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------
|
|
|
|
// misc.
|
|
|
|
|
|
|
|
// Creates a new struct (or pointer to struct) from am.proto.
|
|
|
|
func (am accountMapper) clonePrototype() sdk.Account {
|
|
|
|
protoRt := reflect.TypeOf(am.proto)
|
|
|
|
if protoRt.Kind() == reflect.Ptr {
|
|
|
|
protoCrt := protoRt.Elem()
|
|
|
|
if protoCrt.Kind() != reflect.Struct {
|
|
|
|
panic("accountMapper requires a struct proto sdk.Account, or a pointer to one")
|
|
|
|
}
|
|
|
|
protoRv := reflect.New(protoCrt)
|
|
|
|
clone, ok := protoRv.Interface().(sdk.Account)
|
|
|
|
if !ok {
|
|
|
|
panic(fmt.Sprintf("accountMapper requires a proto sdk.Account, but %v doesn't implement sdk.Account", protoRt))
|
|
|
|
}
|
|
|
|
return clone
|
|
|
|
}
|
2018-04-18 21:49:24 -07:00
|
|
|
|
|
|
|
protoRv := reflect.New(protoRt).Elem()
|
|
|
|
clone, ok := protoRv.Interface().(sdk.Account)
|
|
|
|
if !ok {
|
|
|
|
panic(fmt.Sprintf("accountMapper requires a proto sdk.Account, but %v doesn't implement sdk.Account", protoRt))
|
|
|
|
}
|
|
|
|
return clone
|
2018-01-22 05:44:24 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (am accountMapper) encodeAccount(acc sdk.Account) []byte {
|
2018-04-06 17:25:08 -07:00
|
|
|
bz, err := am.cdc.MarshalBinaryBare(acc)
|
2018-01-22 05:44:24 -08:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return bz
|
|
|
|
}
|
|
|
|
|
2018-04-06 17:25:08 -07:00
|
|
|
func (am accountMapper) decodeAccount(bz []byte) (acc sdk.Account) {
|
|
|
|
err := am.cdc.UnmarshalBinaryBare(bz, &acc)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
2018-01-22 05:44:24 -08:00
|
|
|
}
|
2018-04-06 17:25:08 -07:00
|
|
|
return
|
2018-01-22 05:44:24 -08:00
|
|
|
}
|