Merge pull request #1164 from cosmos/cwgoes/bech32ify

Use bech32 encoding for msg.GetSignBytes()
This commit is contained in:
Rigel 2018-06-07 21:52:29 -07:00 committed by GitHub
commit 27365a43e5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 213 additions and 40 deletions

View File

@ -1,6 +1,7 @@
# Changelog # Changelog
BREAKING CHANGES BREAKING CHANGES
* msg.GetSignBytes() now returns bech32-encoded addresses in all cases
FEATURES FEATURES

View File

@ -26,21 +26,57 @@ func Bech32ifyAcc(addr Address) (string, error) {
return bech32.ConvertAndEncode(Bech32PrefixAccAddr, addr.Bytes()) return bech32.ConvertAndEncode(Bech32PrefixAccAddr, addr.Bytes())
} }
// MustBech32ifyAcc panics on bech32-encoding failure
func MustBech32ifyAcc(addr Address) string {
enc, err := Bech32ifyAcc(addr)
if err != nil {
panic(err)
}
return enc
}
// Bech32ifyAccPub takes AccountPubKey and returns the bech32 encoded string // Bech32ifyAccPub takes AccountPubKey and returns the bech32 encoded string
func Bech32ifyAccPub(pub crypto.PubKey) (string, error) { func Bech32ifyAccPub(pub crypto.PubKey) (string, error) {
return bech32.ConvertAndEncode(Bech32PrefixAccPub, pub.Bytes()) return bech32.ConvertAndEncode(Bech32PrefixAccPub, pub.Bytes())
} }
// MustBech32ifyAccPub panics on bech32-encoding failure
func MustBech32ifyAccPub(pub crypto.PubKey) string {
enc, err := Bech32ifyAccPub(pub)
if err != nil {
panic(err)
}
return enc
}
// Bech32ifyVal returns the bech32 encoded string for a validator address // Bech32ifyVal returns the bech32 encoded string for a validator address
func Bech32ifyVal(addr Address) (string, error) { func Bech32ifyVal(addr Address) (string, error) {
return bech32.ConvertAndEncode(Bech32PrefixValAddr, addr.Bytes()) return bech32.ConvertAndEncode(Bech32PrefixValAddr, addr.Bytes())
} }
// MustBech32ifyVal panics on bech32-encoding failure
func MustBech32ifyVal(addr Address) string {
enc, err := Bech32ifyVal(addr)
if err != nil {
panic(err)
}
return enc
}
// Bech32ifyValPub returns the bech32 encoded string for a validator pubkey // Bech32ifyValPub returns the bech32 encoded string for a validator pubkey
func Bech32ifyValPub(pub crypto.PubKey) (string, error) { func Bech32ifyValPub(pub crypto.PubKey) (string, error) {
return bech32.ConvertAndEncode(Bech32PrefixValPub, pub.Bytes()) return bech32.ConvertAndEncode(Bech32PrefixValPub, pub.Bytes())
} }
// MustBech32ifyValPub pancis on bech32-encoding failure
func MustBech32ifyValPub(pub crypto.PubKey) string {
enc, err := Bech32ifyValPub(pub)
if err != nil {
panic(err)
}
return enc
}
// create an Address from a string // create an Address from a string
func GetAccAddressHex(address string) (addr Address, err error) { func GetAccAddressHex(address string) (addr Address, err error) {
if len(address) == 0 { if len(address) == 0 {
@ -98,7 +134,7 @@ func GetValAddressBech32(address string) (addr Address, err error) {
return Address(bz), nil return Address(bz), nil
} }
//Decode a validator publickey into a public key // decode a validator public key into a PubKey
func GetValPubKeyBech32(pubkey string) (pk crypto.PubKey, err error) { func GetValPubKeyBech32(pubkey string) (pk crypto.PubKey, err error) {
bz, err := getFromBech32(pubkey, Bech32PrefixValPub) bz, err := getFromBech32(pubkey, Bech32PrefixValPub)
if err != nil { if err != nil {

View File

@ -1,6 +1,8 @@
package auth package auth
import ( import (
"encoding/json"
sdk "github.com/cosmos/cosmos-sdk/types" sdk "github.com/cosmos/cosmos-sdk/types"
crypto "github.com/tendermint/go-crypto" crypto "github.com/tendermint/go-crypto"
) )
@ -83,11 +85,11 @@ func (fee StdFee) Bytes() []byte {
// and the Sequence numbers for each signature (prevent // and the Sequence numbers for each signature (prevent
// inchain replay and enforce tx ordering per account). // inchain replay and enforce tx ordering per account).
type StdSignDoc struct { type StdSignDoc struct {
ChainID string `json:"chain_id"` ChainID string `json:"chain_id"`
Sequences []int64 `json:"sequences"` Sequences []int64 `json:"sequences"`
FeeBytes []byte `json:"fee_bytes"` FeeBytes json.RawMessage `json:"fee_bytes"`
MsgBytes []byte `json:"msg_bytes"` MsgBytes json.RawMessage `json:"msg_bytes"`
AltBytes []byte `json:"alt_bytes"` AltBytes json.RawMessage `json:"alt_bytes"`
} }
// StdSignBytes returns the bytes to sign for a transaction. // StdSignBytes returns the bytes to sign for a transaction.
@ -96,8 +98,8 @@ func StdSignBytes(chainID string, sequences []int64, fee StdFee, msg sdk.Msg) []
bz, err := msgCdc.MarshalJSON(StdSignDoc{ bz, err := msgCdc.MarshalJSON(StdSignDoc{
ChainID: chainID, ChainID: chainID,
Sequences: sequences, Sequences: sequences,
FeeBytes: fee.Bytes(), FeeBytes: json.RawMessage(fee.Bytes()),
MsgBytes: msg.GetSignBytes(), MsgBytes: json.RawMessage(msg.GetSignBytes()),
}) })
if err != nil { if err != nil {
panic(err) panic(err)

View File

@ -1,6 +1,8 @@
package bank package bank
import ( import (
"encoding/json"
sdk "github.com/cosmos/cosmos-sdk/types" sdk "github.com/cosmos/cosmos-sdk/types"
) )
@ -53,7 +55,20 @@ func (msg MsgSend) ValidateBasic() sdk.Error {
// Implements Msg. // Implements Msg.
func (msg MsgSend) GetSignBytes() []byte { func (msg MsgSend) GetSignBytes() []byte {
b, err := msgCdc.MarshalJSON(msg) // XXX: ensure some canonical form var inputs, outputs []json.RawMessage
for _, input := range msg.Inputs {
inputs = append(inputs, input.GetSignBytes())
}
for _, output := range msg.Outputs {
outputs = append(outputs, output.GetSignBytes())
}
b, err := msgCdc.MarshalJSON(struct {
Inputs []json.RawMessage `json:"inputs"`
Outputs []json.RawMessage `json:"outputs"`
}{
Inputs: inputs,
Outputs: outputs,
})
if err != nil { if err != nil {
panic(err) panic(err)
} }
@ -78,6 +93,8 @@ type MsgIssue struct {
Outputs []Output `json:"outputs"` Outputs []Output `json:"outputs"`
} }
var _ sdk.Msg = MsgIssue{}
// NewMsgIssue - construct arbitrary multi-in, multi-out send msg. // NewMsgIssue - construct arbitrary multi-in, multi-out send msg.
func NewMsgIssue(banker sdk.Address, out []Output) MsgIssue { func NewMsgIssue(banker sdk.Address, out []Output) MsgIssue {
return MsgIssue{Banker: banker, Outputs: out} return MsgIssue{Banker: banker, Outputs: out}
@ -102,7 +119,17 @@ func (msg MsgIssue) ValidateBasic() sdk.Error {
// Implements Msg. // Implements Msg.
func (msg MsgIssue) GetSignBytes() []byte { func (msg MsgIssue) GetSignBytes() []byte {
b, err := msgCdc.MarshalJSON(msg) // XXX: ensure some canonical form var outputs []json.RawMessage
for _, output := range msg.Outputs {
outputs = append(outputs, output.GetSignBytes())
}
b, err := msgCdc.MarshalJSON(struct {
Banker string `json:"banker"`
Outputs []json.RawMessage `json:"outputs"`
}{
Banker: sdk.MustBech32ifyAcc(msg.Banker),
Outputs: outputs,
})
if err != nil { if err != nil {
panic(err) panic(err)
} }
@ -123,6 +150,21 @@ type Input struct {
Coins sdk.Coins `json:"coins"` Coins sdk.Coins `json:"coins"`
} }
// Return bytes to sign for Input
func (in Input) GetSignBytes() []byte {
bin, err := msgCdc.MarshalJSON(struct {
Address string `json:"address"`
Coins sdk.Coins `json:"coins"`
}{
Address: sdk.MustBech32ifyAcc(in.Address),
Coins: in.Coins,
})
if err != nil {
panic(err)
}
return bin
}
// ValidateBasic - validate transaction input // ValidateBasic - validate transaction input
func (in Input) ValidateBasic() sdk.Error { func (in Input) ValidateBasic() sdk.Error {
if len(in.Address) == 0 { if len(in.Address) == 0 {
@ -155,6 +197,21 @@ type Output struct {
Coins sdk.Coins `json:"coins"` Coins sdk.Coins `json:"coins"`
} }
// Return bytes to sign for Output
func (out Output) GetSignBytes() []byte {
bin, err := msgCdc.MarshalJSON(struct {
Address string `json:"address"`
Coins sdk.Coins `json:"coins"`
}{
Address: sdk.MustBech32ifyAcc(out.Address),
Coins: out.Coins,
})
if err != nil {
panic(err)
}
return bin
}
// ValidateBasic - validate transaction output // ValidateBasic - validate transaction output
func (out Output) ValidateBasic() sdk.Error { func (out Output) ValidateBasic() sdk.Error {
if len(out.Address) == 0 { if len(out.Address) == 0 {

View File

@ -187,12 +187,7 @@ func TestMsgSendGetSignBytes(t *testing.T) {
} }
res := msg.GetSignBytes() res := msg.GetSignBytes()
unmarshaledMsg := &MsgSend{} expected := `{"inputs":[{"address":"cosmosaccaddr1d9h8qat5e4ehc5","coins":[{"denom":"atom","amount":10}]}],"outputs":[{"address":"cosmosaccaddr1da6hgur4wse3jx32","coins":[{"denom":"atom","amount":10}]}]}`
msgCdc.UnmarshalJSON(res, unmarshaledMsg)
assert.Equal(t, &msg, unmarshaledMsg)
// TODO bad results
expected := `{"type":"EAFDE32A2C87F8","value":{"inputs":[{"address":"696E707574","coins":[{"denom":"atom","amount":10}]}],"outputs":[{"address":"6F7574707574","coins":[{"denom":"atom","amount":10}]}]}}`
assert.Equal(t, expected, string(res)) assert.Equal(t, expected, string(res))
} }
@ -262,12 +257,7 @@ func TestMsgIssueGetSignBytes(t *testing.T) {
} }
res := msg.GetSignBytes() res := msg.GetSignBytes()
unmarshaledMsg := &MsgIssue{} expected := `{"banker":"cosmosaccaddr1d9h8qat5e4ehc5","outputs":[{"address":"cosmosaccaddr1d3hkzm3dveex7mfdvfsku6cwsauqd","coins":[{"denom":"atom","amount":10}]}]}`
msgCdc.UnmarshalJSON(res, unmarshaledMsg)
assert.Equal(t, &msg, unmarshaledMsg)
// TODO bad results
expected := `{"type":"72E617C06ABAD0","value":{"banker":"696E707574","outputs":[{"address":"6C6F616E2D66726F6D2D62616E6B","coins":[{"denom":"atom","amount":10}]}]}}`
assert.Equal(t, expected, string(res)) assert.Equal(t, expected, string(res))
} }

View File

@ -1,11 +1,20 @@
package ibc package ibc
import ( import (
sdk "github.com/cosmos/cosmos-sdk/types" "encoding/json"
sdk "github.com/cosmos/cosmos-sdk/types"
wire "github.com/cosmos/cosmos-sdk/wire" wire "github.com/cosmos/cosmos-sdk/wire"
) )
var (
msgCdc *wire.Codec
)
func init() {
msgCdc = wire.NewCodec()
}
// ------------------------------ // ------------------------------
// IBCPacket // IBCPacket
@ -32,12 +41,33 @@ func NewIBCPacket(srcAddr sdk.Address, destAddr sdk.Address, coins sdk.Coins,
} }
} }
//nolint
func (p IBCPacket) GetSignBytes() []byte {
b, err := msgCdc.MarshalJSON(struct {
SrcAddr string
DestAddr string
Coins sdk.Coins
SrcChain string
DestChain string
}{
SrcAddr: sdk.MustBech32ifyAcc(p.SrcAddr),
DestAddr: sdk.MustBech32ifyAcc(p.DestAddr),
Coins: p.Coins,
SrcChain: p.SrcChain,
DestChain: p.DestChain,
})
if err != nil {
panic(err)
}
return b
}
// validator the ibc packey // validator the ibc packey
func (ibcp IBCPacket) ValidateBasic() sdk.Error { func (p IBCPacket) ValidateBasic() sdk.Error {
if ibcp.SrcChain == ibcp.DestChain { if p.SrcChain == p.DestChain {
return ErrIdenticalChains(DefaultCodespace).Trace("") return ErrIdenticalChains(DefaultCodespace).Trace("")
} }
if !ibcp.Coins.IsValid() { if !p.Coins.IsValid() {
return sdk.ErrInvalidCoins("") return sdk.ErrInvalidCoins("")
} }
return nil return nil
@ -60,12 +90,7 @@ func (msg IBCTransferMsg) GetSigners() []sdk.Address { return []sdk.Address{msg.
// get the sign bytes for ibc transfer message // get the sign bytes for ibc transfer message
func (msg IBCTransferMsg) GetSignBytes() []byte { func (msg IBCTransferMsg) GetSignBytes() []byte {
cdc := wire.NewCodec() return msg.IBCPacket.GetSignBytes()
bz, err := cdc.MarshalBinary(msg)
if err != nil {
panic(err)
}
return bz
} }
// validate ibc transfer message // validate ibc transfer message
@ -94,10 +119,17 @@ func (msg IBCReceiveMsg) GetSigners() []sdk.Address { return []sdk.Address{msg.R
// get the sign bytes for ibc receive message // get the sign bytes for ibc receive message
func (msg IBCReceiveMsg) GetSignBytes() []byte { func (msg IBCReceiveMsg) GetSignBytes() []byte {
cdc := wire.NewCodec() b, err := msgCdc.MarshalJSON(struct {
bz, err := cdc.MarshalBinary(msg) IBCPacket json.RawMessage
Relayer string
Sequence int64
}{
IBCPacket: json.RawMessage(msg.IBCPacket.GetSignBytes()),
Relayer: sdk.MustBech32ifyAcc(msg.Relayer),
Sequence: msg.Sequence,
})
if err != nil { if err != nil {
panic(err) panic(err)
} }
return bz return b
} }

View File

@ -30,7 +30,11 @@ func (msg MsgUnrevoke) GetSigners() []sdk.Address { return []sdk.Address{msg.Val
// get the bytes for the message signer to sign on // get the bytes for the message signer to sign on
func (msg MsgUnrevoke) GetSignBytes() []byte { func (msg MsgUnrevoke) GetSignBytes() []byte {
b, err := cdc.MarshalJSON(msg) b, err := cdc.MarshalJSON(struct {
ValidatorAddr string `json:"address"`
}{
ValidatorAddr: sdk.MustBech32ifyVal(msg.ValidatorAddr),
})
if err != nil { if err != nil {
panic(err) panic(err)
} }

16
x/slashing/msg_test.go Normal file
View File

@ -0,0 +1,16 @@
package slashing
import (
"testing"
"github.com/stretchr/testify/assert"
sdk "github.com/cosmos/cosmos-sdk/types"
)
func TestMsgUnrevokeGetSignBytes(t *testing.T) {
addr := sdk.Address("abcd")
msg := NewMsgUnrevoke(addr)
bytes := msg.GetSignBytes()
assert.Equal(t, string(bytes), `{"address":"cosmosvaladdr1v93xxeqamr0mv"}`)
}

View File

@ -45,7 +45,20 @@ func (msg MsgCreateValidator) GetSigners() []sdk.Address {
// get the bytes for the message signer to sign on // get the bytes for the message signer to sign on
func (msg MsgCreateValidator) GetSignBytes() []byte { func (msg MsgCreateValidator) GetSignBytes() []byte {
return msgCdc.MustMarshalBinary(msg) b, err := msgCdc.MarshalJSON(struct {
Description
ValidatorAddr string `json:"address"`
PubKey string `json:"pubkey"`
Bond sdk.Coin `json:"bond"`
}{
Description: msg.Description,
ValidatorAddr: sdk.MustBech32ifyVal(msg.ValidatorAddr),
PubKey: sdk.MustBech32ifyValPub(msg.PubKey),
})
if err != nil {
panic(err)
}
return b
} }
// quick validity check // quick validity check
@ -89,7 +102,13 @@ func (msg MsgEditValidator) GetSigners() []sdk.Address {
// get the bytes for the message signer to sign on // get the bytes for the message signer to sign on
func (msg MsgEditValidator) GetSignBytes() []byte { func (msg MsgEditValidator) GetSignBytes() []byte {
b, err := msgCdc.MarshalJSON(msg) b, err := msgCdc.MarshalJSON(struct {
Description
ValidatorAddr string `json:"address"`
}{
Description: msg.Description,
ValidatorAddr: sdk.MustBech32ifyVal(msg.ValidatorAddr),
})
if err != nil { if err != nil {
panic(err) panic(err)
} }
@ -133,7 +152,15 @@ func (msg MsgDelegate) GetSigners() []sdk.Address {
// get the bytes for the message signer to sign on // get the bytes for the message signer to sign on
func (msg MsgDelegate) GetSignBytes() []byte { func (msg MsgDelegate) GetSignBytes() []byte {
b, err := msgCdc.MarshalJSON(msg) b, err := msgCdc.MarshalJSON(struct {
DelegatorAddr string `json:"delegator_addr"`
ValidatorAddr string `json:"validator_addr"`
Bond sdk.Coin `json:"bond"`
}{
DelegatorAddr: sdk.MustBech32ifyAcc(msg.DelegatorAddr),
ValidatorAddr: sdk.MustBech32ifyVal(msg.ValidatorAddr),
Bond: msg.Bond,
})
if err != nil { if err != nil {
panic(err) panic(err)
} }
@ -180,7 +207,15 @@ func (msg MsgUnbond) GetSigners() []sdk.Address { return []sdk.Address{msg.Deleg
// get the bytes for the message signer to sign on // get the bytes for the message signer to sign on
func (msg MsgUnbond) GetSignBytes() []byte { func (msg MsgUnbond) GetSignBytes() []byte {
b, err := msgCdc.MarshalJSON(msg) b, err := msgCdc.MarshalJSON(struct {
DelegatorAddr string `json:"delegator_addr"`
ValidatorAddr string `json:"validator_addr"`
Shares string `json:"shares"`
}{
DelegatorAddr: sdk.MustBech32ifyAcc(msg.DelegatorAddr),
ValidatorAddr: sdk.MustBech32ifyVal(msg.ValidatorAddr),
Shares: msg.Shares,
})
if err != nil { if err != nil {
panic(err) panic(err)
} }