190 lines
4.9 KiB
Go
190 lines
4.9 KiB
Go
package types
|
|
|
|
import (
|
|
"encoding/json"
|
|
)
|
|
|
|
// Transactions messages must fulfill the Msg
|
|
type Msg interface {
|
|
|
|
// Return the message type.
|
|
// Must be alphanumeric or empty.
|
|
Type() string
|
|
|
|
// Get some property of the Msg.
|
|
Get(key interface{}) (value interface{})
|
|
|
|
// Get the canonical byte representation of the Msg.
|
|
GetSignBytes() []byte
|
|
|
|
// ValidateBasic does a simple validation check that
|
|
// doesn't require access to any other information.
|
|
ValidateBasic() Error
|
|
|
|
// Signers returns the addrs of signers that must sign.
|
|
// CONTRACT: All signatures must be present to be valid.
|
|
// CONTRACT: Returns addrs in some deterministic order.
|
|
GetSigners() []Address
|
|
}
|
|
|
|
//__________________________________________________________
|
|
|
|
// Transactions objects must fulfill the Tx
|
|
type Tx interface {
|
|
|
|
// Gets the Msg.
|
|
GetMsg() Msg
|
|
|
|
// Signatures returns the signature of signers who signed the Msg.
|
|
// CONTRACT: Length returned is same as length of
|
|
// pubkeys returned from MsgKeySigners, and the order
|
|
// matches.
|
|
// CONTRACT: If the signature is missing (ie the Msg is
|
|
// invalid), then the corresponding signature is
|
|
// .Empty().
|
|
GetSignatures() []StdSignature
|
|
}
|
|
|
|
var _ Tx = (*StdTx)(nil)
|
|
|
|
// StdTx is a standard way to wrap a Msg with Fee and Signatures.
|
|
// NOTE: the first signature is the FeePayer (Signatures must not be nil).
|
|
type StdTx struct {
|
|
Msg `json:"msg"`
|
|
Fee StdFee `json:"fee"`
|
|
Signatures []StdSignature `json:"signatures"`
|
|
}
|
|
|
|
func NewStdTx(msg Msg, fee StdFee, sigs []StdSignature) StdTx {
|
|
return StdTx{
|
|
Msg: msg,
|
|
Fee: fee,
|
|
Signatures: sigs,
|
|
}
|
|
}
|
|
|
|
//nolint
|
|
func (tx StdTx) GetMsg() Msg { return tx.Msg }
|
|
func (tx StdTx) GetSignatures() []StdSignature { return tx.Signatures }
|
|
|
|
// FeePayer returns the address responsible for paying the fees
|
|
// for the transactions. It's the first address returned by msg.GetSigners().
|
|
// If GetSigners() is empty, this panics.
|
|
func FeePayer(tx Tx) Address {
|
|
return tx.GetMsg().GetSigners()[0]
|
|
}
|
|
|
|
//__________________________________________________________
|
|
|
|
// StdFee includes the amount of coins paid in fees and the maximum
|
|
// gas to be used by the transaction. The ratio yields an effective "gasprice",
|
|
// which must be above some miminum to be accepted into the mempool.
|
|
type StdFee struct {
|
|
Amount Coins `json"amount"`
|
|
Gas int64 `json"gas"`
|
|
}
|
|
|
|
func NewStdFee(gas int64, amount ...Coin) StdFee {
|
|
return StdFee{
|
|
Amount: amount,
|
|
Gas: gas,
|
|
}
|
|
}
|
|
|
|
// fee bytes for signing later
|
|
func (fee StdFee) Bytes() []byte {
|
|
// normalize. XXX
|
|
// this is a sign of something ugly
|
|
// (in the lcd_test, client side its null,
|
|
// server side its [])
|
|
if len(fee.Amount) == 0 {
|
|
fee.Amount = Coins{}
|
|
}
|
|
bz, err := json.Marshal(fee) // TODO
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return bz
|
|
}
|
|
|
|
//__________________________________________________________
|
|
|
|
// StdSignDoc is replay-prevention structure.
|
|
// It includes the result of msg.GetSignBytes(),
|
|
// as well as the ChainID (prevent cross chain replay)
|
|
// and the Sequence numbers for each signature (prevent
|
|
// inchain replay and enforce tx ordering per account).
|
|
type StdSignDoc struct {
|
|
ChainID string `json:"chain_id"`
|
|
Sequences []int64 `json:"sequences"`
|
|
FeeBytes []byte `json:"fee_bytes"`
|
|
MsgBytes []byte `json:"msg_bytes"`
|
|
AltBytes []byte `json:"alt_bytes"`
|
|
}
|
|
|
|
// StdSignBytes returns the bytes to sign for a transaction.
|
|
// TODO: change the API to just take a chainID and StdTx ?
|
|
func StdSignBytes(chainID string, sequences []int64, fee StdFee, msg Msg) []byte {
|
|
bz, err := json.Marshal(StdSignDoc{
|
|
ChainID: chainID,
|
|
Sequences: sequences,
|
|
FeeBytes: fee.Bytes(),
|
|
MsgBytes: msg.GetSignBytes(),
|
|
})
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return bz
|
|
}
|
|
|
|
// StdSignMsg is a convenience structure for passing along
|
|
// a Msg with the other requirements for a StdSignDoc before
|
|
// it is signed. For use in the CLI.
|
|
type StdSignMsg struct {
|
|
ChainID string
|
|
Sequences []int64
|
|
Fee StdFee
|
|
Msg Msg
|
|
// XXX: Alt
|
|
}
|
|
|
|
// get message bytes
|
|
func (msg StdSignMsg) Bytes() []byte {
|
|
return StdSignBytes(msg.ChainID, msg.Sequences, msg.Fee, msg.Msg)
|
|
}
|
|
|
|
//__________________________________________________________
|
|
|
|
// TxDeocder unmarshals transaction bytes
|
|
type TxDecoder func(txBytes []byte) (Tx, Error)
|
|
|
|
//__________________________________________________________
|
|
|
|
var _ Msg = (*TestMsg)(nil)
|
|
|
|
// msg type for testing
|
|
type TestMsg struct {
|
|
signers []Address
|
|
}
|
|
|
|
func NewTestMsg(addrs ...Address) *TestMsg {
|
|
return &TestMsg{
|
|
signers: addrs,
|
|
}
|
|
}
|
|
|
|
//nolint
|
|
func (msg *TestMsg) Type() string { return "TestMsg" }
|
|
func (msg *TestMsg) Get(key interface{}) (value interface{}) { return nil }
|
|
func (msg *TestMsg) GetSignBytes() []byte {
|
|
bz, err := json.Marshal(msg.signers)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return bz
|
|
}
|
|
func (msg *TestMsg) ValidateBasic() Error { return nil }
|
|
func (msg *TestMsg) GetSigners() []Address {
|
|
return msg.signers
|
|
}
|