2018-03-04 13:56:49 +00:00
|
|
|
package transaction
|
|
|
|
|
|
|
|
import (
|
2020-03-23 14:31:28 +00:00
|
|
|
"encoding/hex"
|
|
|
|
"encoding/json"
|
2020-01-29 13:38:58 +00:00
|
|
|
"errors"
|
2019-11-27 10:12:04 +00:00
|
|
|
"fmt"
|
|
|
|
|
2020-03-03 14:21:42 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/crypto/hash"
|
2020-03-23 14:31:28 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/encoding/address"
|
2020-03-03 14:21:42 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/io"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
2018-03-04 13:56:49 +00:00
|
|
|
)
|
|
|
|
|
Implement rpc server method: sendrawtransaction (#174)
* Added new config attributes: 'SecondsPerBlock','LowPriorityThreshold'
* Added new files:
* Added new method: CompareTo
* Fixed empty Slice case
* Added new methods: LessThan, GreaterThan, Equal, CompareTo
* Added new method: InputIntersection
* Added MaxTransactionSize, GroupOutputByAssetID
* Added ned method: ScriptHash
* Added new method: IsDoubleSpend
* Refactor blockchainer, Added Feer interface, Verify and GetMemPool method
* 1) Added MemPool
2) Added new methods to satisfy the blockchainer interface: IsLowPriority, Verify, GetMemPool
* Added new methods: RelayTxn, RelayDirectly
* Fixed tests
* Implemented RPC server method sendrawtransaction
* Refactor getrawtransaction, sendrawtransaction in separate methods
* Moved 'secondsPerBlock' to config file
* Implemented Kim suggestions:
1) Fixed data race issues
2) refactor Verify method
3) Get rid of unused InputIntersection method due to refactoring Verify method
4) Fixed bug in https://github.com/CityOfZion/neo-go/pull/174#discussion_r264108135
5) minor simplications of the code
* Fixed minor issues related to
1) space
2) getter methods do not need pointer on the receiver
3) error message
4) refactoring CompareTo method in uint256.go
* Fixed small issues
* Use sync.RWMutex instead of sync.Mutex
* Refined (R)Lock/(R)Unlock
* return error instead of bool in Verify methods
2019-03-20 12:30:05 +00:00
|
|
|
const (
|
|
|
|
// MaxTransactionSize is the upper limit size in bytes that a transaction can reach. It is
|
|
|
|
// set to be 102400.
|
|
|
|
MaxTransactionSize = 102400
|
2020-04-15 06:50:13 +00:00
|
|
|
// MaxValidUntilBlockIncrement is the upper increment size of blockhain height in blocs after
|
|
|
|
// exceeding that a transaction should fail validation. It is set to be 2102400.
|
|
|
|
MaxValidUntilBlockIncrement = 2102400
|
2020-04-30 13:00:33 +00:00
|
|
|
// MaxCosigners is maximum number of cosigners that can be contained within a transaction.
|
|
|
|
// It is set to be 16.
|
|
|
|
MaxCosigners = 16
|
Implement rpc server method: sendrawtransaction (#174)
* Added new config attributes: 'SecondsPerBlock','LowPriorityThreshold'
* Added new files:
* Added new method: CompareTo
* Fixed empty Slice case
* Added new methods: LessThan, GreaterThan, Equal, CompareTo
* Added new method: InputIntersection
* Added MaxTransactionSize, GroupOutputByAssetID
* Added ned method: ScriptHash
* Added new method: IsDoubleSpend
* Refactor blockchainer, Added Feer interface, Verify and GetMemPool method
* 1) Added MemPool
2) Added new methods to satisfy the blockchainer interface: IsLowPriority, Verify, GetMemPool
* Added new methods: RelayTxn, RelayDirectly
* Fixed tests
* Implemented RPC server method sendrawtransaction
* Refactor getrawtransaction, sendrawtransaction in separate methods
* Moved 'secondsPerBlock' to config file
* Implemented Kim suggestions:
1) Fixed data race issues
2) refactor Verify method
3) Get rid of unused InputIntersection method due to refactoring Verify method
4) Fixed bug in https://github.com/CityOfZion/neo-go/pull/174#discussion_r264108135
5) minor simplications of the code
* Fixed minor issues related to
1) space
2) getter methods do not need pointer on the receiver
3) error message
4) refactoring CompareTo method in uint256.go
* Fixed small issues
* Use sync.RWMutex instead of sync.Mutex
* Refined (R)Lock/(R)Unlock
* return error instead of bool in Verify methods
2019-03-20 12:30:05 +00:00
|
|
|
)
|
|
|
|
|
2018-03-04 13:56:49 +00:00
|
|
|
// Transaction is a process recorded in the NEO blockchain.
|
|
|
|
type Transaction struct {
|
|
|
|
// The type of the transaction.
|
2020-03-23 14:31:28 +00:00
|
|
|
Type TXType
|
2018-03-04 13:56:49 +00:00
|
|
|
|
|
|
|
// The trading version which is currently 0.
|
2020-03-23 14:31:28 +00:00
|
|
|
Version uint8
|
2018-03-04 13:56:49 +00:00
|
|
|
|
2020-04-10 10:41:49 +00:00
|
|
|
// Random number to avoid hash collision.
|
|
|
|
Nonce uint32
|
|
|
|
|
2020-04-16 14:10:42 +00:00
|
|
|
// Address signed the transaction.
|
|
|
|
Sender util.Uint160
|
|
|
|
|
2020-05-08 17:54:24 +00:00
|
|
|
// Fee to be burned.
|
|
|
|
SystemFee util.Fixed8
|
|
|
|
|
|
|
|
// Fee to be distributed to consensus nodes.
|
|
|
|
NetworkFee util.Fixed8
|
|
|
|
|
2020-04-15 06:50:13 +00:00
|
|
|
// Maximum blockchain height exceeding which
|
|
|
|
// transaction should fail verification.
|
|
|
|
ValidUntilBlock uint32
|
|
|
|
|
2018-03-04 13:56:49 +00:00
|
|
|
// Data specific to the type of the transaction.
|
|
|
|
// This is always a pointer to a <Type>Transaction.
|
2020-03-23 14:31:28 +00:00
|
|
|
Data TXer
|
2018-03-04 13:56:49 +00:00
|
|
|
|
|
|
|
// Transaction attributes.
|
2020-03-23 14:31:28 +00:00
|
|
|
Attributes []Attribute
|
2018-03-04 13:56:49 +00:00
|
|
|
|
2020-04-30 13:00:33 +00:00
|
|
|
// Transaction cosigners (not include Sender).
|
|
|
|
Cosigners []Cosigner
|
|
|
|
|
2018-03-04 13:56:49 +00:00
|
|
|
// The inputs of the transaction.
|
2020-03-23 14:31:28 +00:00
|
|
|
Inputs []Input
|
2018-03-04 13:56:49 +00:00
|
|
|
|
|
|
|
// The outputs of the transaction.
|
2020-03-23 14:31:28 +00:00
|
|
|
Outputs []Output
|
2018-03-04 13:56:49 +00:00
|
|
|
|
|
|
|
// The scripts that comes with this transaction.
|
|
|
|
// Scripts exist out of the verification script
|
|
|
|
// and invocation script.
|
2020-03-23 14:31:28 +00:00
|
|
|
Scripts []Witness
|
2018-03-14 09:36:59 +00:00
|
|
|
|
2019-09-23 17:13:44 +00:00
|
|
|
// Hash of the transaction (double SHA256).
|
2018-03-14 09:36:59 +00:00
|
|
|
hash util.Uint256
|
2018-03-17 11:53:21 +00:00
|
|
|
|
2019-09-23 17:13:44 +00:00
|
|
|
// Hash of the transaction used to verify it (single SHA256).
|
|
|
|
verificationHash util.Uint256
|
|
|
|
|
2018-03-17 11:53:21 +00:00
|
|
|
// Trimmed indicates this is a transaction from trimmed
|
|
|
|
// data.
|
2020-03-23 14:31:28 +00:00
|
|
|
Trimmed bool
|
2018-03-17 11:53:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewTrimmedTX returns a trimmed transaction with only its hash
|
|
|
|
// and Trimmed to true.
|
|
|
|
func NewTrimmedTX(hash util.Uint256) *Transaction {
|
|
|
|
return &Transaction{
|
|
|
|
hash: hash,
|
|
|
|
Trimmed: true,
|
|
|
|
}
|
2018-03-14 09:36:59 +00:00
|
|
|
}
|
|
|
|
|
2019-10-22 14:56:03 +00:00
|
|
|
// Hash returns the hash of the transaction.
|
2018-03-14 09:36:59 +00:00
|
|
|
func (t *Transaction) Hash() util.Uint256 {
|
2018-03-25 10:45:54 +00:00
|
|
|
if t.hash.Equals(util.Uint256{}) {
|
2019-09-23 17:26:53 +00:00
|
|
|
if t.createHash() != nil {
|
|
|
|
panic("failed to compute hash!")
|
|
|
|
}
|
2018-03-25 10:45:54 +00:00
|
|
|
}
|
2018-03-14 09:36:59 +00:00
|
|
|
return t.hash
|
2018-03-04 13:56:49 +00:00
|
|
|
}
|
|
|
|
|
2019-09-23 17:13:44 +00:00
|
|
|
// VerificationHash returns the hash of the transaction used to verify it.
|
|
|
|
func (t *Transaction) VerificationHash() util.Uint256 {
|
|
|
|
if t.verificationHash.Equals(util.Uint256{}) {
|
2019-09-23 17:26:53 +00:00
|
|
|
if t.createHash() != nil {
|
|
|
|
panic("failed to compute hash!")
|
|
|
|
}
|
2019-09-23 17:13:44 +00:00
|
|
|
}
|
|
|
|
return t.verificationHash
|
|
|
|
}
|
|
|
|
|
2018-03-04 13:56:49 +00:00
|
|
|
// AddOutput adds the given output to the transaction outputs.
|
|
|
|
func (t *Transaction) AddOutput(out *Output) {
|
2019-12-09 14:14:10 +00:00
|
|
|
t.Outputs = append(t.Outputs, *out)
|
2018-03-04 13:56:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// AddInput adds the given input to the transaction inputs.
|
|
|
|
func (t *Transaction) AddInput(in *Input) {
|
2019-12-09 14:14:10 +00:00
|
|
|
t.Inputs = append(t.Inputs, *in)
|
2018-03-04 13:56:49 +00:00
|
|
|
}
|
|
|
|
|
2019-09-16 16:31:49 +00:00
|
|
|
// DecodeBinary implements Serializable interface.
|
|
|
|
func (t *Transaction) DecodeBinary(br *io.BinReader) {
|
2019-12-12 17:17:50 +00:00
|
|
|
t.Type = TXType(br.ReadB())
|
|
|
|
t.Version = uint8(br.ReadB())
|
2020-04-10 10:41:49 +00:00
|
|
|
t.Nonce = br.ReadU32LE()
|
2020-05-04 07:35:56 +00:00
|
|
|
t.Sender.DecodeBinary(br)
|
2020-05-08 17:54:24 +00:00
|
|
|
t.SystemFee.DecodeBinary(br)
|
|
|
|
if t.SystemFee < 0 {
|
|
|
|
br.Err = errors.New("negative system fee")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
t.NetworkFee.DecodeBinary(br)
|
|
|
|
if t.NetworkFee < 0 {
|
|
|
|
br.Err = errors.New("negative network fee")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if t.NetworkFee+t.SystemFee < t.SystemFee {
|
|
|
|
br.Err = errors.New("too big fees: int 64 overflow")
|
|
|
|
return
|
|
|
|
}
|
2020-04-15 06:50:13 +00:00
|
|
|
t.ValidUntilBlock = br.ReadU32LE()
|
2019-09-16 16:31:49 +00:00
|
|
|
t.decodeData(br)
|
2018-03-04 13:56:49 +00:00
|
|
|
|
2019-11-14 07:50:03 +00:00
|
|
|
br.ReadArray(&t.Attributes)
|
2020-04-30 13:00:33 +00:00
|
|
|
|
|
|
|
br.ReadArray(&t.Cosigners, MaxCosigners)
|
|
|
|
for i := 0; i < len(t.Cosigners); i++ {
|
|
|
|
for j := i + 1; j < len(t.Cosigners); j++ {
|
|
|
|
if t.Cosigners[i].Account.Equals(t.Cosigners[j].Account) {
|
|
|
|
br.Err = errors.New("transaction cosigners should be unique")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-14 07:50:03 +00:00
|
|
|
br.ReadArray(&t.Inputs)
|
|
|
|
br.ReadArray(&t.Outputs)
|
2020-02-28 13:53:26 +00:00
|
|
|
for i := range t.Outputs {
|
|
|
|
if t.Outputs[i].Amount.LessThan(0) {
|
|
|
|
br.Err = errors.New("negative output")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
2019-11-14 07:50:03 +00:00
|
|
|
br.ReadArray(&t.Scripts)
|
2018-03-04 13:56:49 +00:00
|
|
|
|
2018-03-14 09:36:59 +00:00
|
|
|
// Create the hash of the transaction at decode, so we dont need
|
|
|
|
// to do it anymore.
|
2019-09-16 16:31:49 +00:00
|
|
|
if br.Err == nil {
|
|
|
|
br.Err = t.createHash()
|
|
|
|
}
|
2018-03-04 13:56:49 +00:00
|
|
|
}
|
|
|
|
|
2019-09-16 16:31:49 +00:00
|
|
|
func (t *Transaction) decodeData(r *io.BinReader) {
|
2018-03-04 13:56:49 +00:00
|
|
|
switch t.Type {
|
|
|
|
case InvocationType:
|
2020-05-27 20:32:19 +00:00
|
|
|
t.Data = &InvocationTX{}
|
2019-09-16 16:31:49 +00:00
|
|
|
t.Data.(*InvocationTX).DecodeBinary(r)
|
2018-03-04 13:56:49 +00:00
|
|
|
case ClaimType:
|
|
|
|
t.Data = &ClaimTX{}
|
2019-09-16 16:31:49 +00:00
|
|
|
t.Data.(*ClaimTX).DecodeBinary(r)
|
2018-03-14 09:36:59 +00:00
|
|
|
case ContractType:
|
|
|
|
t.Data = &ContractTX{}
|
2019-09-16 16:31:49 +00:00
|
|
|
t.Data.(*ContractTX).DecodeBinary(r)
|
2018-03-17 11:53:21 +00:00
|
|
|
case RegisterType:
|
|
|
|
t.Data = &RegisterTX{}
|
2019-09-16 16:31:49 +00:00
|
|
|
t.Data.(*RegisterTX).DecodeBinary(r)
|
2018-03-17 11:53:21 +00:00
|
|
|
case IssueType:
|
|
|
|
t.Data = &IssueTX{}
|
2019-09-16 16:31:49 +00:00
|
|
|
t.Data.(*IssueTX).DecodeBinary(r)
|
2018-03-17 11:53:21 +00:00
|
|
|
default:
|
2019-11-27 10:12:04 +00:00
|
|
|
r.Err = fmt.Errorf("invalid TX type %x", t.Type)
|
2018-03-04 13:56:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-16 16:31:49 +00:00
|
|
|
// EncodeBinary implements Serializable interface.
|
|
|
|
func (t *Transaction) EncodeBinary(bw *io.BinWriter) {
|
|
|
|
t.encodeHashableFields(bw)
|
2019-11-13 07:36:29 +00:00
|
|
|
bw.WriteArray(t.Scripts)
|
2018-03-14 09:36:59 +00:00
|
|
|
}
|
|
|
|
|
2019-10-22 14:56:03 +00:00
|
|
|
// encodeHashableFields encodes the fields that are not used for
|
2018-03-14 09:36:59 +00:00
|
|
|
// signing the transaction, which are all fields except the scripts.
|
2019-09-16 16:31:49 +00:00
|
|
|
func (t *Transaction) encodeHashableFields(bw *io.BinWriter) {
|
2020-03-03 18:20:58 +00:00
|
|
|
noData := t.Type == ContractType
|
|
|
|
if t.Data == nil && !noData {
|
2020-01-29 13:38:58 +00:00
|
|
|
bw.Err = errors.New("transaction has no data")
|
|
|
|
return
|
|
|
|
}
|
2019-12-12 17:17:50 +00:00
|
|
|
bw.WriteB(byte(t.Type))
|
|
|
|
bw.WriteB(byte(t.Version))
|
2020-04-10 10:41:49 +00:00
|
|
|
bw.WriteU32LE(t.Nonce)
|
2020-05-04 07:35:56 +00:00
|
|
|
t.Sender.EncodeBinary(bw)
|
2020-05-08 17:54:24 +00:00
|
|
|
t.SystemFee.EncodeBinary(bw)
|
|
|
|
t.NetworkFee.EncodeBinary(bw)
|
2020-04-15 06:50:13 +00:00
|
|
|
bw.WriteU32LE(t.ValidUntilBlock)
|
2018-03-14 09:36:59 +00:00
|
|
|
|
|
|
|
// Underlying TXer.
|
2020-03-03 18:20:58 +00:00
|
|
|
if !noData {
|
|
|
|
t.Data.EncodeBinary(bw)
|
|
|
|
}
|
2018-03-04 13:56:49 +00:00
|
|
|
|
|
|
|
// Attributes
|
2019-11-13 07:36:29 +00:00
|
|
|
bw.WriteArray(t.Attributes)
|
2018-03-04 13:56:49 +00:00
|
|
|
|
2020-04-30 13:00:33 +00:00
|
|
|
// Cosigners
|
|
|
|
bw.WriteArray(t.Cosigners)
|
|
|
|
|
2018-03-04 13:56:49 +00:00
|
|
|
// Inputs
|
2019-11-13 07:36:29 +00:00
|
|
|
bw.WriteArray(t.Inputs)
|
2018-03-04 13:56:49 +00:00
|
|
|
|
|
|
|
// Outputs
|
2019-11-13 07:36:29 +00:00
|
|
|
bw.WriteArray(t.Outputs)
|
2018-03-04 13:56:49 +00:00
|
|
|
}
|
|
|
|
|
2018-03-25 10:45:54 +00:00
|
|
|
// createHash creates the hash of the transaction.
|
|
|
|
func (t *Transaction) createHash() error {
|
2019-09-16 09:18:13 +00:00
|
|
|
buf := io.NewBufBinWriter()
|
2019-09-16 16:31:49 +00:00
|
|
|
t.encodeHashableFields(buf.BinWriter)
|
|
|
|
if buf.Err != nil {
|
|
|
|
return buf.Err
|
2018-03-25 10:45:54 +00:00
|
|
|
}
|
|
|
|
|
2019-09-23 17:13:44 +00:00
|
|
|
b := buf.Bytes()
|
|
|
|
t.verificationHash = hash.Sha256(b)
|
2019-12-12 15:01:30 +00:00
|
|
|
t.hash = hash.Sha256(t.verificationHash.BytesBE())
|
2018-03-25 10:45:54 +00:00
|
|
|
|
|
|
|
return nil
|
2018-03-04 13:56:49 +00:00
|
|
|
}
|
2018-03-21 16:11:04 +00:00
|
|
|
|
Implement rpc server method: sendrawtransaction (#174)
* Added new config attributes: 'SecondsPerBlock','LowPriorityThreshold'
* Added new files:
* Added new method: CompareTo
* Fixed empty Slice case
* Added new methods: LessThan, GreaterThan, Equal, CompareTo
* Added new method: InputIntersection
* Added MaxTransactionSize, GroupOutputByAssetID
* Added ned method: ScriptHash
* Added new method: IsDoubleSpend
* Refactor blockchainer, Added Feer interface, Verify and GetMemPool method
* 1) Added MemPool
2) Added new methods to satisfy the blockchainer interface: IsLowPriority, Verify, GetMemPool
* Added new methods: RelayTxn, RelayDirectly
* Fixed tests
* Implemented RPC server method sendrawtransaction
* Refactor getrawtransaction, sendrawtransaction in separate methods
* Moved 'secondsPerBlock' to config file
* Implemented Kim suggestions:
1) Fixed data race issues
2) refactor Verify method
3) Get rid of unused InputIntersection method due to refactoring Verify method
4) Fixed bug in https://github.com/CityOfZion/neo-go/pull/174#discussion_r264108135
5) minor simplications of the code
* Fixed minor issues related to
1) space
2) getter methods do not need pointer on the receiver
3) error message
4) refactoring CompareTo method in uint256.go
* Fixed small issues
* Use sync.RWMutex instead of sync.Mutex
* Refined (R)Lock/(R)Unlock
* return error instead of bool in Verify methods
2019-03-20 12:30:05 +00:00
|
|
|
// GroupOutputByAssetID groups all TX outputs by their assetID.
|
|
|
|
func (t Transaction) GroupOutputByAssetID() map[util.Uint256][]*Output {
|
|
|
|
m := make(map[util.Uint256][]*Output)
|
2019-12-09 14:14:10 +00:00
|
|
|
for i := range t.Outputs {
|
|
|
|
hash := t.Outputs[i].AssetID
|
|
|
|
m[hash] = append(m[hash], &t.Outputs[i])
|
Implement rpc server method: sendrawtransaction (#174)
* Added new config attributes: 'SecondsPerBlock','LowPriorityThreshold'
* Added new files:
* Added new method: CompareTo
* Fixed empty Slice case
* Added new methods: LessThan, GreaterThan, Equal, CompareTo
* Added new method: InputIntersection
* Added MaxTransactionSize, GroupOutputByAssetID
* Added ned method: ScriptHash
* Added new method: IsDoubleSpend
* Refactor blockchainer, Added Feer interface, Verify and GetMemPool method
* 1) Added MemPool
2) Added new methods to satisfy the blockchainer interface: IsLowPriority, Verify, GetMemPool
* Added new methods: RelayTxn, RelayDirectly
* Fixed tests
* Implemented RPC server method sendrawtransaction
* Refactor getrawtransaction, sendrawtransaction in separate methods
* Moved 'secondsPerBlock' to config file
* Implemented Kim suggestions:
1) Fixed data race issues
2) refactor Verify method
3) Get rid of unused InputIntersection method due to refactoring Verify method
4) Fixed bug in https://github.com/CityOfZion/neo-go/pull/174#discussion_r264108135
5) minor simplications of the code
* Fixed minor issues related to
1) space
2) getter methods do not need pointer on the receiver
3) error message
4) refactoring CompareTo method in uint256.go
* Fixed small issues
* Use sync.RWMutex instead of sync.Mutex
* Refined (R)Lock/(R)Unlock
* return error instead of bool in Verify methods
2019-03-20 12:30:05 +00:00
|
|
|
}
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
2020-02-28 07:05:13 +00:00
|
|
|
// GetSignedPart returns a part of the transaction which must be signed.
|
|
|
|
func (t *Transaction) GetSignedPart() []byte {
|
|
|
|
buf := io.NewBufBinWriter()
|
|
|
|
t.encodeHashableFields(buf.BinWriter)
|
|
|
|
if buf.Err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return buf.Bytes()
|
|
|
|
}
|
|
|
|
|
2019-10-22 14:56:03 +00:00
|
|
|
// Bytes converts the transaction to []byte
|
2019-02-20 17:39:32 +00:00
|
|
|
func (t *Transaction) Bytes() []byte {
|
2019-09-16 09:18:13 +00:00
|
|
|
buf := io.NewBufBinWriter()
|
2019-09-16 16:31:49 +00:00
|
|
|
t.EncodeBinary(buf.BinWriter)
|
|
|
|
if buf.Err != nil {
|
2019-02-20 17:39:32 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return buf.Bytes()
|
|
|
|
}
|
2020-03-23 14:31:28 +00:00
|
|
|
|
2020-04-13 09:02:18 +00:00
|
|
|
// NewTransactionFromBytes decodes byte array into *Transaction
|
|
|
|
func NewTransactionFromBytes(b []byte) (*Transaction, error) {
|
|
|
|
tx := &Transaction{}
|
|
|
|
r := io.NewBinReaderFromBuf(b)
|
|
|
|
tx.DecodeBinary(r)
|
|
|
|
if r.Err != nil {
|
|
|
|
return nil, r.Err
|
|
|
|
}
|
|
|
|
return tx, nil
|
|
|
|
}
|
|
|
|
|
2020-05-08 17:54:24 +00:00
|
|
|
// FeePerByte returns NetworkFee of the transaction divided by
|
|
|
|
// its size
|
|
|
|
func (t *Transaction) FeePerByte() util.Fixed8 {
|
|
|
|
return util.Fixed8(int64(t.NetworkFee) / int64(io.GetVarSize(t)))
|
|
|
|
}
|
|
|
|
|
2020-03-23 14:31:28 +00:00
|
|
|
// transactionJSON is a wrapper for Transaction and
|
|
|
|
// used for correct marhalling of transaction.Data
|
|
|
|
type transactionJSON struct {
|
2020-04-15 06:50:13 +00:00
|
|
|
TxID util.Uint256 `json:"txid"`
|
|
|
|
Size int `json:"size"`
|
|
|
|
Type TXType `json:"type"`
|
|
|
|
Version uint8 `json:"version"`
|
|
|
|
Nonce uint32 `json:"nonce"`
|
2020-04-16 14:10:42 +00:00
|
|
|
Sender string `json:"sender"`
|
2020-05-08 17:54:24 +00:00
|
|
|
SystemFee util.Fixed8 `json:"sys_fee"`
|
|
|
|
NetworkFee util.Fixed8 `json:"net_fee"`
|
2020-04-15 06:50:13 +00:00
|
|
|
ValidUntilBlock uint32 `json:"valid_until_block"`
|
|
|
|
Attributes []Attribute `json:"attributes"`
|
2020-04-30 13:00:33 +00:00
|
|
|
Cosigners []Cosigner `json:"cosigners"`
|
2020-04-15 06:50:13 +00:00
|
|
|
Inputs []Input `json:"vin"`
|
|
|
|
Outputs []Output `json:"vout"`
|
|
|
|
Scripts []Witness `json:"scripts"`
|
2020-03-23 14:31:28 +00:00
|
|
|
|
2020-04-26 17:16:39 +00:00
|
|
|
Claims []Input `json:"claims,omitempty"`
|
|
|
|
Script string `json:"script,omitempty"`
|
|
|
|
Asset *registeredAsset `json:"asset,omitempty"`
|
2020-03-23 14:31:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON implements json.Marshaler interface.
|
|
|
|
func (t *Transaction) MarshalJSON() ([]byte, error) {
|
|
|
|
tx := transactionJSON{
|
2020-04-15 06:50:13 +00:00
|
|
|
TxID: t.Hash(),
|
|
|
|
Size: io.GetVarSize(t),
|
|
|
|
Type: t.Type,
|
|
|
|
Version: t.Version,
|
|
|
|
Nonce: t.Nonce,
|
2020-04-16 14:10:42 +00:00
|
|
|
Sender: address.Uint160ToString(t.Sender),
|
2020-04-15 06:50:13 +00:00
|
|
|
ValidUntilBlock: t.ValidUntilBlock,
|
|
|
|
Attributes: t.Attributes,
|
2020-04-30 13:00:33 +00:00
|
|
|
Cosigners: t.Cosigners,
|
2020-04-15 06:50:13 +00:00
|
|
|
Inputs: t.Inputs,
|
|
|
|
Outputs: t.Outputs,
|
|
|
|
Scripts: t.Scripts,
|
2020-05-08 17:54:24 +00:00
|
|
|
SystemFee: t.SystemFee,
|
|
|
|
NetworkFee: t.NetworkFee,
|
2020-03-23 14:31:28 +00:00
|
|
|
}
|
|
|
|
switch t.Type {
|
|
|
|
case ClaimType:
|
|
|
|
tx.Claims = t.Data.(*ClaimTX).Claims
|
|
|
|
case InvocationType:
|
|
|
|
tx.Script = hex.EncodeToString(t.Data.(*InvocationTX).Script)
|
|
|
|
case RegisterType:
|
|
|
|
transaction := *t.Data.(*RegisterTX)
|
|
|
|
tx.Asset = ®isteredAsset{
|
|
|
|
AssetType: transaction.AssetType,
|
|
|
|
Name: json.RawMessage(transaction.Name),
|
|
|
|
Amount: transaction.Amount,
|
|
|
|
Precision: transaction.Precision,
|
|
|
|
Owner: transaction.Owner,
|
|
|
|
Admin: address.Uint160ToString(transaction.Admin),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return json.Marshal(tx)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalJSON implements json.Unmarshaler interface.
|
|
|
|
func (t *Transaction) UnmarshalJSON(data []byte) error {
|
|
|
|
tx := new(transactionJSON)
|
|
|
|
if err := json.Unmarshal(data, tx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
t.Type = tx.Type
|
|
|
|
t.Version = tx.Version
|
2020-04-10 10:41:49 +00:00
|
|
|
t.Nonce = tx.Nonce
|
2020-04-15 06:50:13 +00:00
|
|
|
t.ValidUntilBlock = tx.ValidUntilBlock
|
2020-03-23 14:31:28 +00:00
|
|
|
t.Attributes = tx.Attributes
|
2020-04-30 13:00:33 +00:00
|
|
|
t.Cosigners = tx.Cosigners
|
2020-03-23 14:31:28 +00:00
|
|
|
t.Inputs = tx.Inputs
|
|
|
|
t.Outputs = tx.Outputs
|
|
|
|
t.Scripts = tx.Scripts
|
2020-05-08 17:54:24 +00:00
|
|
|
t.SystemFee = tx.SystemFee
|
|
|
|
t.NetworkFee = tx.NetworkFee
|
2020-04-16 14:10:42 +00:00
|
|
|
sender, err := address.StringToUint160(tx.Sender)
|
|
|
|
if err != nil {
|
|
|
|
return errors.New("cannot unmarshal tx: bad sender")
|
|
|
|
}
|
|
|
|
t.Sender = sender
|
2020-03-23 14:31:28 +00:00
|
|
|
switch tx.Type {
|
|
|
|
case ClaimType:
|
|
|
|
t.Data = &ClaimTX{
|
|
|
|
Claims: tx.Claims,
|
|
|
|
}
|
|
|
|
case InvocationType:
|
|
|
|
bytes, err := hex.DecodeString(tx.Script)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
t.Data = &InvocationTX{
|
2020-05-27 20:32:19 +00:00
|
|
|
Script: bytes,
|
2020-03-23 14:31:28 +00:00
|
|
|
}
|
|
|
|
case RegisterType:
|
|
|
|
admin, err := address.StringToUint160(tx.Asset.Admin)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
t.Data = &RegisterTX{
|
|
|
|
AssetType: tx.Asset.AssetType,
|
|
|
|
Name: string(tx.Asset.Name),
|
|
|
|
Amount: tx.Asset.Amount,
|
|
|
|
Precision: tx.Asset.Precision,
|
|
|
|
Owner: tx.Asset.Owner,
|
|
|
|
Admin: admin,
|
|
|
|
}
|
|
|
|
case ContractType:
|
|
|
|
t.Data = &ContractTX{}
|
|
|
|
case IssueType:
|
|
|
|
t.Data = &IssueTX{}
|
|
|
|
}
|
|
|
|
if t.Hash() != tx.TxID {
|
|
|
|
return errors.New("txid doesn't match transaction hash")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|