2020-11-23 11:09:45 +00:00
|
|
|
package testchain
|
|
|
|
|
|
|
|
import (
|
2020-12-13 15:26:35 +00:00
|
|
|
"encoding/json"
|
2020-11-23 11:09:45 +00:00
|
|
|
gio "io"
|
|
|
|
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/compiler"
|
2020-11-18 20:10:48 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/config"
|
2020-11-23 11:09:45 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/config/netmode"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/core/blockchainer"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/core/fee"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/core/native"
|
2020-11-18 20:10:48 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
2020-11-23 11:09:45 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/core/transaction"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/io"
|
2020-11-18 20:10:48 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/nef"
|
2020-11-23 11:09:45 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/emit"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/vm/opcode"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
ownerHash = MultisigScriptHash()
|
|
|
|
ownerScript = MultisigVerificationScript()
|
|
|
|
)
|
|
|
|
|
|
|
|
// NewTransferFromOwner returns transaction transfering funds from NEO and GAS owner.
|
|
|
|
func NewTransferFromOwner(bc blockchainer.Blockchainer, contractHash, to util.Uint160, amount int64,
|
|
|
|
nonce, validUntil uint32) (*transaction.Transaction, error) {
|
|
|
|
w := io.NewBufBinWriter()
|
|
|
|
emit.AppCallWithOperationAndArgs(w.BinWriter, contractHash, "transfer", ownerHash, to, amount, nil)
|
|
|
|
emit.Opcodes(w.BinWriter, opcode.ASSERT)
|
|
|
|
if w.Err != nil {
|
|
|
|
return nil, w.Err
|
|
|
|
}
|
|
|
|
|
|
|
|
script := w.Bytes()
|
|
|
|
tx := transaction.New(netmode.UnitTestNet, script, 10000000)
|
|
|
|
tx.ValidUntilBlock = validUntil
|
|
|
|
tx.Nonce = nonce
|
|
|
|
tx.Signers = []transaction.Signer{{
|
|
|
|
Account: ownerHash,
|
|
|
|
Scopes: transaction.CalledByEntry,
|
|
|
|
AllowedContracts: nil,
|
|
|
|
AllowedGroups: nil,
|
|
|
|
}}
|
|
|
|
return tx, SignTx(bc, tx)
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewDeployTx returns new deployment tx for contract with name with Go code read from r.
|
2020-12-13 15:26:35 +00:00
|
|
|
func NewDeployTx(bc blockchainer.Blockchainer, name string, sender util.Uint160, r gio.Reader) (*transaction.Transaction, util.Uint160, error) {
|
2020-11-18 20:10:48 +00:00
|
|
|
// nef.NewFile() cares about version a lot.
|
|
|
|
config.Version = "0.90.0-test"
|
|
|
|
|
2020-11-23 11:09:45 +00:00
|
|
|
avm, di, err := compiler.CompileWithDebugInfo(name, r)
|
|
|
|
if err != nil {
|
2020-11-18 20:10:48 +00:00
|
|
|
return nil, util.Uint160{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ne, err := nef.NewFile(avm)
|
|
|
|
if err != nil {
|
|
|
|
return nil, util.Uint160{}, err
|
|
|
|
}
|
2020-11-23 11:09:45 +00:00
|
|
|
|
2020-12-10 14:42:12 +00:00
|
|
|
m, err := di.ConvertToManifest(&compiler.Options{Name: name})
|
2020-11-23 11:09:45 +00:00
|
|
|
if err != nil {
|
2020-11-18 20:10:48 +00:00
|
|
|
return nil, util.Uint160{}, err
|
2020-11-23 11:09:45 +00:00
|
|
|
}
|
2020-11-27 18:53:39 +00:00
|
|
|
|
2020-12-13 15:26:35 +00:00
|
|
|
rawManifest, err := json.Marshal(m)
|
2020-11-23 11:09:45 +00:00
|
|
|
if err != nil {
|
2020-11-18 20:10:48 +00:00
|
|
|
return nil, util.Uint160{}, err
|
2020-11-23 11:09:45 +00:00
|
|
|
}
|
2020-12-13 15:26:35 +00:00
|
|
|
neb, err := ne.Bytes()
|
|
|
|
if err != nil {
|
|
|
|
return nil, util.Uint160{}, err
|
|
|
|
}
|
|
|
|
buf := io.NewBufBinWriter()
|
|
|
|
emit.AppCallWithOperationAndArgs(buf.BinWriter, bc.ManagementContractHash(), "deploy", neb, rawManifest)
|
|
|
|
if buf.Err != nil {
|
|
|
|
return nil, util.Uint160{}, buf.Err
|
|
|
|
}
|
2020-11-18 20:10:48 +00:00
|
|
|
|
2020-12-13 15:26:35 +00:00
|
|
|
tx := transaction.New(Network(), buf.Bytes(), 100*native.GASFactor)
|
2020-11-18 20:10:48 +00:00
|
|
|
tx.Signers = []transaction.Signer{{Account: sender}}
|
|
|
|
h := state.CreateContractHash(tx.Sender(), avm)
|
|
|
|
|
|
|
|
return tx, h, nil
|
2020-11-23 11:09:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// SignTx signs provided transactions with validator keys.
|
|
|
|
func SignTx(bc blockchainer.Blockchainer, txs ...*transaction.Transaction) error {
|
|
|
|
for _, tx := range txs {
|
|
|
|
size := io.GetVarSize(tx)
|
|
|
|
netFee, sizeDelta := fee.Calculate(ownerScript)
|
|
|
|
tx.NetworkFee += netFee
|
|
|
|
size += sizeDelta
|
|
|
|
tx.NetworkFee += int64(size) * bc.FeePerByte()
|
|
|
|
data := tx.GetSignedPart()
|
|
|
|
tx.Scripts = []transaction.Witness{{
|
|
|
|
InvocationScript: Sign(data),
|
|
|
|
VerificationScript: ownerScript,
|
|
|
|
}}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|