186 lines
5.1 KiB
Go
186 lines
5.1 KiB
Go
package std
|
|
|
|
import (
|
|
"bytes"
|
|
|
|
jsonc "github.com/gibson042/canonicaljson-go"
|
|
"github.com/gogo/protobuf/jsonpb"
|
|
|
|
"github.com/cosmos/cosmos-sdk/client"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
|
|
"github.com/cosmos/cosmos-sdk/x/auth"
|
|
)
|
|
|
|
var (
|
|
_ sdk.Tx = (*Transaction)(nil)
|
|
_ client.ClientTx = (*Transaction)(nil)
|
|
)
|
|
|
|
func NewTransaction(fee auth.StdFee, memo string, sdkMsgs []sdk.Msg) (*Transaction, error) {
|
|
tx := &Transaction{
|
|
StdTxBase: auth.NewStdTxBase(fee, nil, memo),
|
|
}
|
|
|
|
if err := tx.SetMsgs(sdkMsgs...); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return tx, nil
|
|
}
|
|
|
|
// GetMsgs returns all the messages in a Transaction as a slice of sdk.Msg.
|
|
func (tx Transaction) GetMsgs() []sdk.Msg {
|
|
msgs := make([]sdk.Msg, len(tx.Msgs))
|
|
|
|
for i, m := range tx.Msgs {
|
|
msgs[i] = m.GetMsg()
|
|
}
|
|
|
|
return msgs
|
|
}
|
|
|
|
// GetSigners returns the addresses that must sign the transaction. Addresses are
|
|
// returned in a deterministic order. They are accumulated from the GetSigners
|
|
// method for each Msg in the order they appear in tx.GetMsgs(). Duplicate addresses
|
|
// will be omitted.
|
|
func (tx Transaction) GetSigners() []sdk.AccAddress {
|
|
var signers []sdk.AccAddress
|
|
seen := map[string]bool{}
|
|
|
|
for _, msg := range tx.GetMsgs() {
|
|
for _, addr := range msg.GetSigners() {
|
|
if !seen[addr.String()] {
|
|
signers = append(signers, addr)
|
|
seen[addr.String()] = true
|
|
}
|
|
}
|
|
}
|
|
|
|
return signers
|
|
}
|
|
|
|
// ValidateBasic does a simple and lightweight validation check that doesn't
|
|
// require access to any other information.
|
|
func (tx Transaction) ValidateBasic() error {
|
|
stdSigs := tx.GetSignatures()
|
|
|
|
if tx.Fee.Gas > auth.MaxGasWanted {
|
|
return sdkerrors.Wrapf(
|
|
sdkerrors.ErrInvalidRequest, "invalid gas supplied; %d > %d", tx.Fee.Gas, auth.MaxGasWanted,
|
|
)
|
|
}
|
|
if tx.Fee.Amount.IsAnyNegative() {
|
|
return sdkerrors.Wrapf(
|
|
sdkerrors.ErrInsufficientFee, "invalid fee provided: %s", tx.Fee.Amount,
|
|
)
|
|
}
|
|
if len(stdSigs) == 0 {
|
|
return sdkerrors.ErrNoSignatures
|
|
}
|
|
if len(stdSigs) != len(tx.GetSigners()) {
|
|
return sdkerrors.Wrapf(
|
|
sdkerrors.ErrUnauthorized, "wrong number of signers; expected %d, got %d", tx.GetSigners(), len(stdSigs),
|
|
)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// SetMsgs sets the messages for a Transaction. It will overwrite any existing
|
|
// messages set.
|
|
func (tx *Transaction) SetMsgs(sdkMsgs ...sdk.Msg) error {
|
|
msgs := make([]Message, len(sdkMsgs))
|
|
for i, msg := range sdkMsgs {
|
|
m := &Message{}
|
|
if err := m.SetMsg(msg); err != nil {
|
|
return err
|
|
}
|
|
|
|
msgs[i] = *m
|
|
}
|
|
|
|
tx.Msgs = msgs
|
|
return nil
|
|
}
|
|
|
|
// GetSignatures returns all the transaction's signatures.
|
|
func (tx Transaction) GetSignatures() []sdk.Signature {
|
|
sdkSigs := make([]sdk.Signature, len(tx.Signatures))
|
|
for i, sig := range tx.Signatures {
|
|
sdkSigs[i] = sig
|
|
}
|
|
|
|
return sdkSigs
|
|
}
|
|
|
|
// SetSignatures sets the transaction's signatures. It will overwrite any
|
|
// existing signatures set.
|
|
func (tx *Transaction) SetSignatures(sdkSigs ...sdk.Signature) {
|
|
sigs := make([]auth.StdSignature, len(tx.Signatures))
|
|
for i, sig := range sdkSigs {
|
|
sigs[i] = auth.NewStdSignature(sig.GetPubKey(), sig.GetSignature())
|
|
}
|
|
|
|
tx.Signatures = sigs
|
|
}
|
|
|
|
// GetFee returns the transaction's fee.
|
|
func (tx Transaction) GetFee() sdk.Fee {
|
|
return tx.Fee
|
|
}
|
|
|
|
// SetFee sets the transaction's fee. It will overwrite any existing fee set.
|
|
func (tx *Transaction) SetFee(fee sdk.Fee) {
|
|
tx.Fee = auth.NewStdFee(fee.GetGas(), fee.GetAmount())
|
|
}
|
|
|
|
// GetMemo returns the transaction's memo.
|
|
func (tx Transaction) GetMemo() string {
|
|
return tx.Memo
|
|
}
|
|
|
|
// SetMemo sets the transaction's memo. It will overwrite any existing memo set.
|
|
func (tx *Transaction) SetMemo(memo string) {
|
|
tx.Memo = memo
|
|
}
|
|
|
|
// CanonicalSignBytes returns the canonical JSON bytes to sign over for the
|
|
// Transaction given a chain ID, account sequence and account number. The JSON
|
|
// encoding ensures all field names adhere to their proto definition, default
|
|
// values are omitted, and follows the JSON Canonical Form.
|
|
func (tx Transaction) CanonicalSignBytes(cid string, num, seq uint64) ([]byte, error) {
|
|
return NewSignDoc(num, seq, cid, tx.Memo, tx.Fee, tx.Msgs...).CanonicalSignBytes()
|
|
}
|
|
|
|
func NewSignDoc(num, seq uint64, cid, memo string, fee auth.StdFee, msgs ...Message) *SignDoc {
|
|
return &SignDoc{
|
|
StdSignDocBase: auth.NewStdSignDocBase(num, seq, cid, memo, fee),
|
|
Msgs: msgs,
|
|
}
|
|
}
|
|
|
|
// CanonicalSignBytes returns the canonical JSON bytes to sign over, where the
|
|
// SignDoc is derived from a Transaction. The JSON encoding ensures all field
|
|
// names adhere to their proto definition, default values are omitted, and follows
|
|
// the JSON Canonical Form.
|
|
func (sd *SignDoc) CanonicalSignBytes() ([]byte, error) {
|
|
jm := &jsonpb.Marshaler{EmitDefaults: false, OrigName: false}
|
|
buf := new(bytes.Buffer)
|
|
|
|
// first, encode via canonical Protocol Buffer JSON
|
|
if err := jm.Marshal(buf, sd); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
genericJSON := make(map[string]interface{})
|
|
|
|
// decode canonical proto encoding into a generic map
|
|
if err := jsonc.Unmarshal(buf.Bytes(), &genericJSON); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// finally, return the canonical JSON encoding via JSON Canonical Form
|
|
return jsonc.Marshal(genericJSON)
|
|
}
|