dcrd/blockchain/stake/staketx.go
Aaron Campbell 03678bb754 multi: Correct typos.
Correct typos found by reading code and creative grepping.
2019-08-16 17:37:58 -05:00

1008 lines
34 KiB
Go

// Copyright (c) 2015-2019 The Decred developers
// Use of this source code is governed by an ISC
// license that can be found in the LICENSE file.
//
// Contains a collection of functions that determine what type of stake tx a
// given tx is and does a cursory check for sanity.
package stake
import (
"bytes"
"encoding/binary"
"errors"
"fmt"
"math"
"math/big"
"github.com/decred/dcrd/chaincfg/chainhash"
"github.com/decred/dcrd/dcrec"
"github.com/decred/dcrd/dcrutil/v2"
"github.com/decred/dcrd/txscript/v2"
"github.com/decred/dcrd/wire"
)
// TxType indicates the type of tx (regular or stake type).
type TxType int
// Possible TxTypes.
const (
TxTypeRegular TxType = iota
TxTypeSStx
TxTypeSSGen
TxTypeSSRtx
)
const (
// consensusVersion = txscript.consensusVersion
consensusVersion = 0
// MaxInputsPerSStx is the maximum number of inputs allowed in an SStx.
MaxInputsPerSStx = 64
// MaxOutputsPerSStx is the maximum number of outputs allowed in an SStx;
// you need +1 for the tagged SStx output.
MaxOutputsPerSStx = MaxInputsPerSStx*2 + 1
// NumInputsPerSSGen is the exact number of inputs for an SSGen
// (stakebase) tx. Inputs are a tagged SStx output and a stakebase (null)
// input.
NumInputsPerSSGen = 2 // SStx and stakebase
// MaxOutputsPerSSGen is the maximum number of outputs in an SSGen tx,
// which are all outputs to the addresses specified in the OP_RETURNs of
// the original SStx referenced as input plus reference and vote
// OP_RETURN outputs in the zeroeth and first position.
MaxOutputsPerSSGen = MaxInputsPerSStx + 2
// NumInputsPerSSRtx is the exact number of inputs for an SSRtx (stake
// revocation tx); the only input should be the SStx output.
NumInputsPerSSRtx = 1
// MaxOutputsPerSSRtx is the maximum number of outputs in an SSRtx, which
// are all outputs to the addresses specified in the OP_RETURNs of the
// original SStx referenced as input plus a reference to the block header
// hash of the block in which voting was missed.
MaxOutputsPerSSRtx = MaxInputsPerSStx
// SStxPKHMinOutSize is the minimum size of an OP_RETURN commitment output
// for an SStx tx.
// 20 bytes P2SH/P2PKH + 8 byte amount + 4 byte fee range limits
SStxPKHMinOutSize = 32
// SStxPKHMaxOutSize is the maximum size of an OP_RETURN commitment output
// for an SStx tx.
SStxPKHMaxOutSize = 77
// SSGenBlockReferenceOutSize is the size of a block reference OP_RETURN
// output for an SSGen tx.
SSGenBlockReferenceOutSize = 38
// SSGenVoteBitsOutputMinSize is the minimum size for a VoteBits push
// in an SSGen.
SSGenVoteBitsOutputMinSize = 4
// SSGenVoteBitsOutputMaxSize is the maximum size for a VoteBits push
// in an SSGen.
SSGenVoteBitsOutputMaxSize = 77
// MaxSingleBytePushLength is the largest maximum push for an
// SStx commitment or VoteBits push.
MaxSingleBytePushLength = 75
// SSGenVoteBitsExtendedMaxSize is the maximum size for a VoteBitsExtended
// push in an SSGen.
//
// The final vote transaction includes a single data push for all vote
// bits concatenated. The non-extended vote bits occupy the first 2
// bytes, thus the max number of extended vote bits is the maximum
// allow length for a single byte data push minus the 2 bytes required
// by the non-extended vote bits.
SSGenVoteBitsExtendedMaxSize = MaxSingleBytePushLength - 2
// SStxVoteReturnFractionMask extracts the return fraction from a
// commitment output version.
// If after applying this mask &0x003f is given, the entire amount of
// the output is allowed to be spent as fees if the flag to allow fees
// is set.
SStxVoteReturnFractionMask = 0x003f
// SStxRevReturnFractionMask extracts the return fraction from a
// commitment output version.
// If after applying this mask &0x3f00 is given, the entire amount of
// the output is allowed to be spent as fees if the flag to allow fees
// is set.
SStxRevReturnFractionMask = 0x3f00
// SStxVoteFractionFlag is a bitflag mask specifying whether or not to
// apply a fractional limit to the amount used for fees in a vote.
// 00000000 00000000 = No fees allowed
// 00000000 01000000 = Apply fees rule
SStxVoteFractionFlag = 0x0040
// SStxRevFractionFlag is a bitflag mask specifying whether or not to
// apply a fractional limit to the amount used for fees in a vote.
// 00000000 00000000 = No fees allowed
// 01000000 00000000 = Apply fees rule
SStxRevFractionFlag = 0x4000
// VoteConsensusVersionAbsent is the value of the consensus version
// for a short read of the voteBits.
VoteConsensusVersionAbsent = 0
)
var (
// validSStxAddressOutPrefix is the valid prefix for a 30-byte
// minimum OP_RETURN push for a commitment for an SStx.
// Example SStx address out:
// 0x6a (OP_RETURN)
// 0x1e (OP_DATA_30, push length: 30 bytes)
//
// 0x?? 0x?? 0x?? 0x?? (20 byte public key hash)
// 0x?? 0x?? 0x?? 0x??
// 0x?? 0x?? 0x?? 0x??
// 0x?? 0x?? 0x?? 0x??
// 0x?? 0x??
//
// 0x?? 0x?? 0x?? 0x?? (8 byte amount)
// 0x?? 0x?? 0x?? 0x??
//
// 0x?? 0x?? (2 byte range limits)
validSStxAddressOutMinPrefix = []byte{txscript.OP_RETURN, txscript.OP_DATA_30}
// validSSGenReferenceOutPrefix is the valid prefix for a block
// reference output for an SSGen tx.
// Example SStx address out:
// 0x6a (OP_RETURN)
// 0x24 (OP_DATA_36, push length: 36 bytes)
//
// 0x?? 0x?? 0x?? 0x?? (32 byte block header hash for the block
// 0x?? 0x?? 0x?? 0x?? you wish to vote on)
// 0x?? 0x?? 0x?? 0x??
// 0x?? 0x?? 0x?? 0x??
// 0x?? 0x?? 0x?? 0x??
// 0x?? 0x?? 0x?? 0x??
// 0x?? 0x?? 0x?? 0x??
// 0x?? 0x?? 0x?? 0x??
//
// 0x?? 0x?? 0x?? 0x?? (4 byte uint32 for the height of the block
// that you wish to vote on)
validSSGenReferenceOutPrefix = []byte{txscript.OP_RETURN, txscript.OP_DATA_36}
// validSSGenVoteOutMinPrefix is the valid prefix for a vote output for an
// SSGen tx.
// 0x6a (OP_RETURN)
// 0x02 (OP_DATA_2 to OP_DATA_75, push length: 2-75 bytes)
//
// 0x?? 0x?? (VoteBits) ... 0x??
validSSGenVoteOutMinPrefix = []byte{txscript.OP_RETURN, txscript.OP_DATA_2}
// zeroHash is the zero value for a chainhash.Hash and is defined as
// a package level variable to avoid the need to create a new instance
// every time a check is needed.
zeroHash = &chainhash.Hash{}
)
// VoteBits is a field representing the mandatory 2-byte field of voteBits along
// with the optional 73-byte extended field for votes.
type VoteBits struct {
Bits uint16
ExtendedBits []byte
}
// VoteVersionTuple contains the extracted vote bits and version from votes
// (SSGen).
type VoteVersionTuple struct {
Version uint32
Bits uint16
}
// SpentTicketsInBlock stores the hashes of the spent (both voted and revoked)
// tickets of a given block, along with the vote information.
type SpentTicketsInBlock struct {
VotedTickets []chainhash.Hash
RevokedTickets []chainhash.Hash
Votes []VoteVersionTuple
}
// --------------------------------------------------------------------------------
// Accessory Stake Functions
// --------------------------------------------------------------------------------
// isNullOutpoint determines whether or not a previous transaction output point
// is set.
func isNullOutpoint(tx *wire.MsgTx) bool {
nullInOP := tx.TxIn[0].PreviousOutPoint
if nullInOP.Index == math.MaxUint32 && nullInOP.Hash.IsEqual(zeroHash) &&
nullInOP.Tree == wire.TxTreeRegular {
return true
}
return false
}
// isNullFraudProof determines whether or not a previous transaction fraud proof
// is set.
func isNullFraudProof(tx *wire.MsgTx) bool {
txIn := tx.TxIn[0]
switch {
case txIn.BlockHeight != wire.NullBlockHeight:
return false
case txIn.BlockIndex != wire.NullBlockIndex:
return false
}
return true
}
// IsStakeBase returns whether or not a tx could be considered as having a
// topically valid stake base present.
func IsStakeBase(tx *wire.MsgTx) bool {
// A stake base (SSGen) must only have two transaction inputs.
if len(tx.TxIn) != 2 {
return false
}
// The previous output of a coin base must have a max value index and
// a zero hash, as well as null fraud proofs.
if !isNullOutpoint(tx) {
return false
}
if !isNullFraudProof(tx) {
return false
}
return true
}
// MinimalOutput is a struct encoding a minimally sized output for use in parsing
// stake related information.
type MinimalOutput struct {
PkScript []byte
Value int64
Version uint16
}
// ConvertToMinimalOutputs converts a transaction to its minimal outputs
// derivative.
func ConvertToMinimalOutputs(tx *wire.MsgTx) []*MinimalOutput {
minOuts := make([]*MinimalOutput, len(tx.TxOut))
for i, txOut := range tx.TxOut {
minOuts[i] = &MinimalOutput{
PkScript: txOut.PkScript,
Value: txOut.Value,
Version: txOut.Version,
}
}
return minOuts
}
// SStxStakeOutputInfo takes an SStx as input and scans through its outputs,
// returning the pubkeyhashs and amounts for any NullDataTy's (future
// commitments to stake generation rewards).
func SStxStakeOutputInfo(outs []*MinimalOutput) ([]bool, [][]byte, []int64,
[]int64, [][]bool, [][]uint16) {
expectedInLen := len(outs) / 2
isP2SH := make([]bool, expectedInLen)
addresses := make([][]byte, expectedInLen)
amounts := make([]int64, expectedInLen)
changeAmounts := make([]int64, expectedInLen)
allSpendRules := make([][]bool, expectedInLen)
allSpendLimits := make([][]uint16, expectedInLen)
// Cycle through the inputs and pull the proportional amounts
// and commit to PKHs/SHs.
for idx, out := range outs {
// We only care about the outputs where we get proportional
// amounts and the PKHs/SHs to send rewards to, which is all
// the odd numbered output indexes.
if (idx > 0) && (idx%2 != 0) {
// The MSB (sign), not used ever normally, encodes whether
// or not it is a P2PKH or P2SH for the input.
amtEncoded := make([]byte, 8)
copy(amtEncoded, out.PkScript[22:30])
isP2SH[idx/2] = !(amtEncoded[7]&(1<<7) == 0) // MSB set?
amtEncoded[7] &= ^uint8(1 << 7) // Clear bit
addresses[idx/2] = out.PkScript[2:22]
amounts[idx/2] = int64(binary.LittleEndian.Uint64(amtEncoded))
// Get flags and restrictions for the outputs to be
// make in either a vote or revocation.
spendRules := make([]bool, 2)
spendLimits := make([]uint16, 2)
// This bitflag is true/false.
feeLimitUint16 := binary.LittleEndian.Uint16(out.PkScript[30:32])
spendRules[0] = (feeLimitUint16 & SStxVoteFractionFlag) ==
SStxVoteFractionFlag
spendRules[1] = (feeLimitUint16 & SStxRevFractionFlag) ==
SStxRevFractionFlag
allSpendRules[idx/2] = spendRules
// This is the fraction to use out of 64.
spendLimits[0] = feeLimitUint16 & SStxVoteReturnFractionMask
spendLimits[1] = feeLimitUint16 & SStxRevReturnFractionMask
spendLimits[1] >>= 8
allSpendLimits[idx/2] = spendLimits
}
// Here we only care about the change amounts, so scan
// the change outputs (even indices) and save their
// amounts.
if (idx > 0) && (idx%2 == 0) {
changeAmounts[(idx/2)-1] = out.Value
}
}
return isP2SH, addresses, amounts, changeAmounts, allSpendRules,
allSpendLimits
}
// TxSStxStakeOutputInfo takes an SStx as input and scans through its outputs,
// returning the pubkeyhashs and amounts for any NullDataTy's (future
// commitments to stake generation rewards).
func TxSStxStakeOutputInfo(tx *wire.MsgTx) ([]bool, [][]byte, []int64, []int64,
[][]bool, [][]uint16) {
return SStxStakeOutputInfo(ConvertToMinimalOutputs(tx))
}
// AddrFromSStxPkScrCommitment extracts a P2SH or P2PKH address from a ticket
// commitment pkScript.
func AddrFromSStxPkScrCommitment(pkScript []byte, params dcrutil.AddressParams) (dcrutil.Address, error) {
if len(pkScript) < SStxPKHMinOutSize {
return nil, stakeRuleError(ErrSStxBadCommitAmount, "short read "+
"of sstx commit pkscript")
}
// The MSB of the encoded amount specifies if the output is P2SH. Since
// it is encoded with little endian, the MSB is in final byte in the encoded
// amount.
//
// This is a faster equivalent of:
//
// amtBytes := script[22:30]
// amtEncoded := binary.LittleEndian.Uint64(amtBytes)
// isP2SH := (amtEncoded & uint64(1<<63)) != 0
isP2SH := pkScript[29]&0x80 != 0
// The 20 byte PKH or SH.
hashBytes := pkScript[2:22]
// Return the correct address type.
if isP2SH {
return dcrutil.NewAddressScriptHashFromHash(hashBytes, params)
}
return dcrutil.NewAddressPubKeyHash(hashBytes, params,
dcrec.STEcdsaSecp256k1)
}
// AmountFromSStxPkScrCommitment extracts a commitment amount from a
// ticket commitment pkScript.
func AmountFromSStxPkScrCommitment(pkScript []byte) (dcrutil.Amount, error) {
if len(pkScript) < SStxPKHMinOutSize {
return 0, stakeRuleError(ErrSStxBadCommitAmount, "short read "+
"of sstx commit pkscript")
}
// The MSB (sign), not used ever normally, encodes whether
// or not it is a P2PKH or P2SH for the input.
amtEncoded := make([]byte, 8)
copy(amtEncoded, pkScript[22:30])
amtEncoded[7] &= ^uint8(1 << 7) // Clear bit for P2SH flag
return dcrutil.Amount(binary.LittleEndian.Uint64(amtEncoded)), nil
}
// SSGenBlockVotedOn takes an SSGen tx and returns the block voted on in the
// first OP_RETURN by hash and height.
//
// This function is only safe to be called on a transaction that
// has passed IsSSGen.
func SSGenBlockVotedOn(tx *wire.MsgTx) (chainhash.Hash, uint32) {
// Get the block header hash. Note that the actual number of bytes is
// specified here over using chainhash.HashSize in order to statically
// assert hash sizes have not changed.
var blockHash [32]byte
copy(blockHash[:], tx.TxOut[0].PkScript[2:34])
// Get the block height.
height := binary.LittleEndian.Uint32(tx.TxOut[0].PkScript[34:38])
return chainhash.Hash(blockHash), height
}
// SSGenVoteBits takes an SSGen tx as input and scans through its
// outputs, returning the VoteBits of the index 1 output.
//
// This function is only safe to be called on a transaction that
// has passed IsSSGen.
func SSGenVoteBits(tx *wire.MsgTx) uint16 {
return binary.LittleEndian.Uint16(tx.TxOut[1].PkScript[2:4])
}
// SSGenVersion takes an SSGen tx as input and returns the network
// consensus version from the VoteBits output. If there is a short
// read, the network consensus version is considered 0 or "unset".
//
// This function is only safe to be called on a transaction that
// has passed IsSSGen.
func SSGenVersion(tx *wire.MsgTx) uint32 {
if len(tx.TxOut[1].PkScript) < 8 {
return VoteConsensusVersionAbsent
}
return binary.LittleEndian.Uint32(tx.TxOut[1].PkScript[4:8])
}
// SStxNullOutputAmounts takes an array of input amounts, change amounts, and a
// ticket purchase amount, calculates the adjusted proportion from the purchase
// amount, stores it in an array, then returns the array. That is, for any given
// SStx, this function calculates the proportional outputs that any single user
// should receive.
// Returns: (1) Fees (2) Output Amounts (3) Error
func SStxNullOutputAmounts(amounts []int64,
changeAmounts []int64,
amountTicket int64) (int64, []int64, error) {
lengthAmounts := len(amounts)
if lengthAmounts != len(changeAmounts) {
errStr := fmt.Sprintf("amounts was not equal in length " +
"to change amounts!")
return 0, nil, errors.New(errStr)
}
if amountTicket <= 0 {
errStr := fmt.Sprintf("committed amount was too small!")
return 0, nil, stakeRuleError(ErrSStxBadCommitAmount, errStr)
}
contribAmounts := make([]int64, lengthAmounts)
sum := int64(0)
// Now we want to get the adjusted amounts. The algorithm is like this:
// 1 foreach amount
// 2 subtract change from input, store
// 3 add this amount to sum
// 4 check sum against the total committed amount
for i := 0; i < lengthAmounts; i++ {
contribAmounts[i] = amounts[i] - changeAmounts[i]
if contribAmounts[i] < 0 {
errStr := fmt.Sprintf("change at idx %v spent more coins than "+
"allowed (have: %v, spent: %v)", i, amounts[i], changeAmounts[i])
return 0, nil, stakeRuleError(ErrSStxBadChangeAmts, errStr)
}
sum += contribAmounts[i]
}
fees := sum - amountTicket
return fees, contribAmounts, nil
}
// CalculateRewards takes a list of SStx adjusted output amounts, the amount used
// to purchase that ticket, and the reward for an SSGen tx and subsequently
// generates what the outputs should be in the SSGen tx. If used for calculating
// the outputs for an SSRtx, pass 0 for subsidy.
func CalculateRewards(amounts []int64, amountTicket int64,
subsidy int64) []int64 {
outputsAmounts := make([]int64, len(amounts))
// SSGen handling
amountWithStakebase := amountTicket + subsidy
// Get the sum of the amounts contributed between both fees
// and contributions to the ticket.
totalContrib := int64(0)
for _, amount := range amounts {
totalContrib += amount
}
// Now we want to get the adjusted amounts including the reward.
// The algorithm is like this:
// 1 foreach amount
// 2 amount *= 2^32
// 3 amount /= amountTicket
// 4 amount *= amountWithStakebase
// 5 amount /= 2^32
amountWithStakebaseBig := big.NewInt(amountWithStakebase)
totalContribBig := big.NewInt(totalContrib)
for idx, amount := range amounts {
amountBig := big.NewInt(amount) // We need > 64 bits
// mul amountWithStakebase
amountBig.Mul(amountBig, amountWithStakebaseBig)
// mul 2^32
amountBig.Lsh(amountBig, 32)
// div totalContrib
amountBig.Div(amountBig, totalContribBig)
// div 2^32
amountBig.Rsh(amountBig, 32)
// make int64
outputsAmounts[idx] = amountBig.Int64()
}
return outputsAmounts
}
// --------------------------------------------------------------------------------
// Stake Transaction Identification Functions
// --------------------------------------------------------------------------------
// CheckSStx returns an error if a transaction is not a stake submission
// transaction. It does some simple validation steps to make sure the number of
// inputs, number of outputs, and the input/output scripts are valid.
//
// SStx transactions are specified as below.
// Inputs:
// untagged output 1 [index 0]
// untagged output 2 [index 1]
// ...
// untagged output MaxInputsPerSStx [index MaxInputsPerSStx-1]
//
// Outputs:
// OP_SSTX tagged output [index 0]
// OP_RETURN push of input 1's address for reward receiving [index 1]
// OP_SSTXCHANGE tagged output for input 1 [index 2]
// OP_RETURN push of input 2's address for reward receiving [index 3]
// OP_SSTXCHANGE tagged output for input 2 [index 4]
// ...
// OP_RETURN push of input MaxInputsPerSStx's address for reward receiving
// [index (MaxInputsPerSStx*2)-2]
// OP_SSTXCHANGE tagged output [index (MaxInputsPerSStx*2)-1]
//
// The output OP_RETURN pushes should be of size 20 bytes (standard address).
func CheckSStx(tx *wire.MsgTx) error {
// Check to make sure there aren't too many inputs.
// CheckTransactionSanity already makes sure that number of inputs is
// greater than 0, so no need to check that.
if len(tx.TxIn) > MaxInputsPerSStx {
return stakeRuleError(ErrSStxTooManyInputs, "SStx has too many "+
"inputs")
}
// Check to make sure there aren't too many outputs.
if len(tx.TxOut) > MaxOutputsPerSStx {
return stakeRuleError(ErrSStxTooManyOutputs, "SStx has too many "+
"outputs")
}
// Check to make sure there are some outputs.
if len(tx.TxOut) == 0 {
return stakeRuleError(ErrSStxNoOutputs, "SStx has no "+
"outputs")
}
// Check to make sure that all output scripts are the consensus version.
for idx, txOut := range tx.TxOut {
if txOut.Version != consensusVersion {
errStr := fmt.Sprintf("invalid script version found in "+
"txOut idx %v", idx)
return stakeRuleError(ErrSStxInvalidOutputs, errStr)
}
}
// Ensure that the first output is tagged OP_SSTX.
if txscript.GetScriptClass(tx.TxOut[0].Version, tx.TxOut[0].PkScript) !=
txscript.StakeSubmissionTy {
return stakeRuleError(ErrSStxInvalidOutputs, "First SStx output "+
"should have been OP_SSTX tagged, but it was not")
}
// Ensure that the number of outputs is equal to the number of inputs
// + 1.
if (len(tx.TxIn)*2 + 1) != len(tx.TxOut) {
return stakeRuleError(ErrSStxInOutProportions, "The number of "+
"inputs in the SStx tx was not the number of outputs/2 - 1")
}
// Ensure that the rest of the odd outputs are 28-byte OP_RETURN pushes that
// contain putative pubkeyhashes, and that the rest of the odd outputs are
// OP_SSTXCHANGE tagged.
for outTxIndex := 1; outTxIndex < len(tx.TxOut); outTxIndex++ {
scrVersion := tx.TxOut[outTxIndex].Version
rawScript := tx.TxOut[outTxIndex].PkScript
// Check change outputs.
if outTxIndex%2 == 0 {
if txscript.GetScriptClass(scrVersion, rawScript) !=
txscript.StakeSubChangeTy {
str := fmt.Sprintf("SStx output at output index %d was not "+
"an sstx change output", outTxIndex)
return stakeRuleError(ErrSStxInvalidOutputs, str)
}
continue
}
// Else (odd) check commitment outputs. The script should be a
// NullDataTy output.
if txscript.GetScriptClass(scrVersion, rawScript) !=
txscript.NullDataTy {
str := fmt.Sprintf("SStx output at output index %d was not "+
"a NullData (OP_RETURN) push", outTxIndex)
return stakeRuleError(ErrSStxInvalidOutputs, str)
}
// The length of the output script should be between 32 and 77 bytes long.
if len(rawScript) < SStxPKHMinOutSize ||
len(rawScript) > SStxPKHMaxOutSize {
str := fmt.Sprintf("SStx output at output index %d was a "+
"NullData (OP_RETURN) push of the wrong size", outTxIndex)
return stakeRuleError(ErrSStxInvalidOutputs, str)
}
// The OP_RETURN output script prefix should conform to the standard.
outputScriptBuffer := bytes.NewBuffer(rawScript)
outputScriptPrefix := outputScriptBuffer.Next(2)
minPush := validSStxAddressOutMinPrefix[1]
maxPush := validSStxAddressOutMinPrefix[1] +
(MaxSingleBytePushLength - minPush)
pushLen := outputScriptPrefix[1]
pushLengthValid := (pushLen >= minPush) && (pushLen <= maxPush)
// The first byte should be OP_RETURN, while the second byte should be a
// valid push length.
if !(outputScriptPrefix[0] == validSStxAddressOutMinPrefix[0]) ||
!pushLengthValid {
errStr := fmt.Sprintf("sstx commitment at output idx %v had "+
"an invalid prefix", outTxIndex)
return stakeRuleError(ErrSStxInvalidOutputs,
errStr)
}
}
return nil
}
// IsSStx returns whether or not a transaction is a stake submission transaction.
// These are also known as tickets.
func IsSStx(tx *wire.MsgTx) bool {
return CheckSStx(tx) == nil
}
// CheckSSGen returns an error if a transaction is not a stake submission
// generation transaction. It does some simple validation steps to make sure
// the number of inputs, number of outputs, and the input/output scripts are
// valid.
//
// This does NOT check to see if the subsidy is valid or whether or not the
// value of input[0] + subsidy = value of the outputs.
//
// SSGen transactions are specified as below.
// Inputs:
// Stakebase null input [index 0]
// SStx-tagged output [index 1]
//
// Outputs:
// OP_RETURN push of 40 bytes containing: [index 0]
// i. 32-byte block header of block being voted on.
// ii. 8-byte int of this block's height.
// OP_RETURN push of 2 bytes containing votebits [index 1]
// SSGen-tagged output to address from SStx-tagged output's tx index output 1
// [index 2]
// SSGen-tagged output to address from SStx-tagged output's tx index output 2
// [index 3]
// ...
// SSGen-tagged output to address from SStx-tagged output's tx index output
// MaxInputsPerSStx [index MaxOutputsPerSSgen - 1]
func CheckSSGen(tx *wire.MsgTx) error {
// Check to make sure there aren't too many inputs.
// CheckTransactionSanity already makes sure that number of inputs is
// greater than 0, so no need to check that.
if len(tx.TxIn) != NumInputsPerSSGen {
return stakeRuleError(ErrSSGenWrongNumInputs, "SSgen tx has an "+
"invalid number of inputs")
}
// Check to make sure there aren't too many outputs.
if len(tx.TxOut) > MaxOutputsPerSSGen {
return stakeRuleError(ErrSSGenTooManyOutputs, "SSgen tx has too "+
"many outputs")
}
// Check to make sure there are enough outputs.
if len(tx.TxOut) < 2 {
return stakeRuleError(ErrSSGenNoOutputs, "SSgen tx does not "+
"have enough outputs")
}
// Ensure that the first input is a stake base null input.
// Also checks to make sure that there aren't too many or too few inputs.
if !IsStakeBase(tx) {
return stakeRuleError(ErrSSGenNoStakebase, "SSGen tx did not "+
"include a stakebase in the zeroeth input position")
}
// Check to make sure that the output used as input came from TxTreeStake.
for i, txin := range tx.TxIn {
// Skip the stakebase
if i == 0 {
continue
}
if txin.PreviousOutPoint.Index != 0 {
errStr := fmt.Sprintf("SSGen used an invalid input idx (got %v, "+
"want 0)", txin.PreviousOutPoint.Index)
return stakeRuleError(ErrSSGenWrongIndex, errStr)
}
if txin.PreviousOutPoint.Tree != wire.TxTreeStake {
return stakeRuleError(ErrSSGenWrongTxTree, "SSGen used "+
"a non-stake input")
}
}
// Check to make sure that all output scripts are the consensus version.
for _, txOut := range tx.TxOut {
if txOut.Version != consensusVersion {
return stakeRuleError(ErrSSGenBadGenOuts, "invalid "+
"script version found in txOut")
}
}
// Ensure the number of outputs is equal to the number of inputs found in
// the original SStx + 2.
// TODO: Do this in validate, requires DB and valid chain.
// Ensure that the second input is an SStx tagged output.
// TODO: Do this in validate, as we don't want to actually lookup
// old tx here. This function is for more general sorting.
// Ensure that the first output is an OP_RETURN push.
zeroethOutputVersion := tx.TxOut[0].Version
zeroethOutputScript := tx.TxOut[0].PkScript
if txscript.GetScriptClass(zeroethOutputVersion, zeroethOutputScript) !=
txscript.NullDataTy {
return stakeRuleError(ErrSSGenNoReference, "First SSGen output "+
"should have been an OP_RETURN data push, but was not")
}
// Ensure that the first output is the correct size.
if len(zeroethOutputScript) != SSGenBlockReferenceOutSize {
return stakeRuleError(ErrSSGenBadReference, "First SSGen output "+
"should have been 43 bytes long, but was not")
}
// The OP_RETURN output script prefix for block referencing should
// conform to the standard.
zeroethOutputScriptBuffer := bytes.NewBuffer(zeroethOutputScript)
zeroethOutputScriptPrefix := zeroethOutputScriptBuffer.Next(2)
if !bytes.Equal(zeroethOutputScriptPrefix,
validSSGenReferenceOutPrefix) {
return stakeRuleError(ErrSSGenBadReference, "First SSGen output "+
"had an invalid prefix")
}
// Ensure that the block header hash given in the first 32 bytes of the
// OP_RETURN push is a valid block header and found in the main chain.
// TODO: This is validate level stuff, do this there.
// Ensure that the second output is an OP_RETURN push.
firstOutputVersion := tx.TxOut[1].Version
firstOutputScript := tx.TxOut[1].PkScript
if txscript.GetScriptClass(firstOutputVersion, firstOutputScript) !=
txscript.NullDataTy {
return stakeRuleError(ErrSSGenNoVotePush, "Second SSGen output "+
"should have been an OP_RETURN data push, but was not")
}
// The length of the output script should be between 4 and 77 bytes long.
if len(firstOutputScript) < SSGenVoteBitsOutputMinSize ||
len(firstOutputScript) > SSGenVoteBitsOutputMaxSize {
str := fmt.Sprintf("SSGen votebits output at output index 1 was a " +
"NullData (OP_RETURN) push of the wrong size")
return stakeRuleError(ErrSSGenBadVotePush, str)
}
// The OP_RETURN output script prefix for voting should conform to the
// standard.
firstOutputScriptBuffer := bytes.NewBuffer(firstOutputScript)
firstOutputScriptPrefix := firstOutputScriptBuffer.Next(2)
minPush := validSSGenVoteOutMinPrefix[1]
maxPush := validSSGenVoteOutMinPrefix[1] +
(MaxSingleBytePushLength - minPush)
pushLen := firstOutputScriptPrefix[1]
pushLengthValid := (pushLen >= minPush) && (pushLen <= maxPush)
// The first byte should be OP_RETURN, while the second byte should be a
// valid push length.
if !(firstOutputScriptPrefix[0] == validSSGenVoteOutMinPrefix[0]) ||
!pushLengthValid {
return stakeRuleError(ErrSSGenBadVotePush, "Second SSGen output "+
"had an invalid prefix")
}
// Ensure that the tx height given in the last 8 bytes is StakeMaturity
// many blocks ahead of the block in which that SStx appears, otherwise
// this ticket has failed to mature and the SStx must be invalid.
// TODO: This is validate level stuff, do this there.
// Ensure that the remaining outputs are OP_SSGEN tagged.
for outTxIndex := 2; outTxIndex < len(tx.TxOut); outTxIndex++ {
scrVersion := tx.TxOut[outTxIndex].Version
rawScript := tx.TxOut[outTxIndex].PkScript
// The script should be a OP_SSGEN tagged output.
if txscript.GetScriptClass(scrVersion, rawScript) !=
txscript.StakeGenTy {
str := fmt.Sprintf("SSGen tx output at output index %d was not "+
"an OP_SSGEN tagged output", outTxIndex)
return stakeRuleError(ErrSSGenBadGenOuts, str)
}
}
return nil
}
// IsSSGen returns whether or not a transaction is a stake submission generation
// transaction. There are also known as votes.
func IsSSGen(tx *wire.MsgTx) bool {
return CheckSSGen(tx) == nil
}
// CheckSSRtx returns an error if a transaction is not a stake submission
// revocation transaction. It does some simple validation steps to make sure
// the number of inputs, number of outputs, and the input/output scripts are
// valid.
//
// SSRtx transactions are specified as below.
// Inputs:
// SStx-tagged output [index 0]
//
// Outputs:
// SSGen-tagged output to address from SStx-tagged output's tx index output 1
// [index 0]
// SSGen-tagged output to address from SStx-tagged output's tx index output 2
// [index 1]
// ...
// SSGen-tagged output to address from SStx-tagged output's tx index output
// MaxInputsPerSStx [index MaxOutputsPerSSRtx - 1]
func CheckSSRtx(tx *wire.MsgTx) error {
// Check to make sure there is the correct number of inputs.
// CheckTransactionSanity already makes sure that number of inputs is
// greater than 0, so no need to check that.
if len(tx.TxIn) != NumInputsPerSSRtx {
return stakeRuleError(ErrSSRtxWrongNumInputs, "SSRtx has an "+
" invalid number of inputs")
}
// Check to make sure there aren't too many outputs.
if len(tx.TxOut) > MaxOutputsPerSSRtx {
return stakeRuleError(ErrSSRtxTooManyOutputs, "SSRtx has too "+
"many outputs")
}
// Check to make sure there are some outputs.
if len(tx.TxOut) == 0 {
return stakeRuleError(ErrSSRtxNoOutputs, "SSRtx has no "+
"outputs")
}
// Check to make sure that all output scripts are the consensus version.
for _, txOut := range tx.TxOut {
if txOut.Version != consensusVersion {
return stakeRuleError(ErrSSRtxBadOuts, "invalid "+
"script version found in txOut")
}
}
// Check to make sure that the output used as input came from TxTreeStake.
for _, txin := range tx.TxIn {
if txin.PreviousOutPoint.Tree != wire.TxTreeStake {
return stakeRuleError(ErrSSRtxWrongTxTree, "SSRtx used "+
"a non-stake input")
}
}
// Ensure that the first input is an SStx tagged output.
// TODO: Do this in validate, needs a DB and chain.
// Ensure that the tx height given in the last 8 bytes is StakeMaturity
// many blocks ahead of the block in which that SStx appear, otherwise
// this ticket has failed to mature and the SStx must be invalid.
// TODO: Do this in validate, needs a DB and chain.
// Ensure that the outputs are OP_SSRTX tagged.
// Ensure that the tx height given in the last 8 bytes is StakeMaturity
// many blocks ahead of the block in which that SStx appear, otherwise
// this ticket has failed to mature and the SStx must be invalid.
// TODO: This is validate level stuff, do this there.
// Ensure that the outputs are OP_SSRTX tagged.
for outTxIndex := 0; outTxIndex < len(tx.TxOut); outTxIndex++ {
scrVersion := tx.TxOut[outTxIndex].Version
rawScript := tx.TxOut[outTxIndex].PkScript
// The script should be a OP_SSRTX tagged output.
if txscript.GetScriptClass(scrVersion, rawScript) !=
txscript.StakeRevocationTy {
str := fmt.Sprintf("SSRtx output at output index %d was not "+
"an OP_SSRTX tagged output", outTxIndex)
return stakeRuleError(ErrSSRtxBadOuts, str)
}
}
// Ensure the number of outputs is equal to the number of inputs found in
// the original SStx.
// TODO: Do this in validate, needs a DB and chain.
return nil
}
// IsSSRtx returns whether or not a transaction is a stake submission revocation
// transaction. There are also known as revocations.
func IsSSRtx(tx *wire.MsgTx) bool {
return CheckSSRtx(tx) == nil
}
// DetermineTxType determines the type of stake transaction a transaction is; if
// none, it returns that it is an assumed regular tx.
func DetermineTxType(tx *wire.MsgTx) TxType {
if IsSStx(tx) {
return TxTypeSStx
}
if IsSSGen(tx) {
return TxTypeSSGen
}
if IsSSRtx(tx) {
return TxTypeSSRtx
}
return TxTypeRegular
}
// IsStakeSubmissionTxOut indicates whether the txOut identified by the
// given index is a stake submission output. Stake Submission outputs are
// the odd-numbered outputs of an SStx transaction.
//
// This function is only safe to be called on a transaction that
// has passed IsSStx.
func IsStakeSubmissionTxOut(index int) bool {
return (index % 2) != 0
}
// FindSpentTicketsInBlock returns information about tickets spent in a given
// block. This includes voted and revoked tickets, and the vote bits of each
// spent ticket. This is faster than calling the individual functions to
// determine ticket state if all information regarding spent tickets is needed.
//
// Note that the returned hashes are of the originally purchased *tickets* and
// **NOT** of the vote/revoke transaction.
//
// The tickets are determined **only** from the STransactions of the provided
// block and no validation is performed.
//
// This function is only safe to be called with a block that has previously
// had all header commitments validated.
func FindSpentTicketsInBlock(block *wire.MsgBlock) *SpentTicketsInBlock {
votes := make([]VoteVersionTuple, 0, block.Header.Voters)
voters := make([]chainhash.Hash, 0, block.Header.Voters)
revocations := make([]chainhash.Hash, 0, block.Header.Revocations)
for _, stx := range block.STransactions {
if IsSSGen(stx) {
voters = append(voters, stx.TxIn[1].PreviousOutPoint.Hash)
votes = append(votes, VoteVersionTuple{
Version: SSGenVersion(stx),
Bits: SSGenVoteBits(stx),
})
continue
}
if IsSSRtx(stx) {
revocations = append(revocations, stx.TxIn[0].PreviousOutPoint.Hash)
continue
}
}
return &SpentTicketsInBlock{
VotedTickets: voters,
Votes: votes,
RevokedTickets: revocations,
}
}