60 lines
1.6 KiB
Go
60 lines
1.6 KiB
Go
package main
|
|
|
|
import (
|
|
"github.com/pkg/errors"
|
|
|
|
crypto "github.com/tendermint/go-crypto"
|
|
keys "github.com/tendermint/go-crypto/keys"
|
|
wire "github.com/tendermint/go-wire"
|
|
|
|
bc "github.com/tendermint/basecoin/types"
|
|
)
|
|
|
|
type AppTx struct {
|
|
chainID string
|
|
signers []crypto.PubKey
|
|
Tx *bc.AppTx
|
|
}
|
|
|
|
var _ keys.Signable = &AppTx{}
|
|
|
|
// SignBytes returned the unsigned bytes, needing a signature
|
|
func (s *AppTx) SignBytes() []byte {
|
|
return s.Tx.SignBytes(s.chainID)
|
|
}
|
|
|
|
// Sign will add a signature and pubkey.
|
|
//
|
|
// Depending on the Signable, one may be able to call this multiple times for multisig
|
|
// Returns error if called with invalid data or too many times
|
|
func (s *AppTx) Sign(pubkey crypto.PubKey, sig crypto.Signature) error {
|
|
if len(s.signers) > 0 {
|
|
return errors.New("AppTx already signed")
|
|
}
|
|
s.Tx.SetSignature(sig)
|
|
s.signers = []crypto.PubKey{pubkey}
|
|
return nil
|
|
}
|
|
|
|
// Signers will return the public key(s) that signed if the signature
|
|
// is valid, or an error if there is any issue with the signature,
|
|
// including if there are no signatures
|
|
func (s *AppTx) Signers() ([]crypto.PubKey, error) {
|
|
if len(s.signers) == 0 {
|
|
return nil, errors.New("No signatures on AppTx")
|
|
}
|
|
return s.signers, nil
|
|
}
|
|
|
|
// TxBytes returns the transaction data as well as all signatures
|
|
// It should return an error if Sign was never called
|
|
func (s *AppTx) TxBytes() ([]byte, error) {
|
|
// TODO: verify it is signed
|
|
|
|
// Code and comment from: basecoin/cmd/commands/tx.go
|
|
// Don't you hate having to do this?
|
|
// How many times have I lost an hour over this trick?!
|
|
txBytes := wire.BinaryBytes(bc.TxS{s.Tx})
|
|
return txBytes, nil
|
|
}
|