2021-10-23 11:28:03 +00:00
|
|
|
package neotest
|
|
|
|
|
|
|
|
import (
|
2024-12-20 13:03:03 +00:00
|
|
|
"encoding/json"
|
2021-10-23 11:28:03 +00:00
|
|
|
"io"
|
2024-12-20 13:03:03 +00:00
|
|
|
"os"
|
2021-10-23 11:28:03 +00:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/nspcc-dev/neo-go/cli/smartcontract"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/compiler"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/config"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/nef"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Contract contains contract info for deployment.
|
|
|
|
type Contract struct {
|
2024-08-19 11:39:18 +00:00
|
|
|
Hash util.Uint160
|
|
|
|
NEF *nef.File
|
|
|
|
Manifest *manifest.Manifest
|
|
|
|
DebugInfo *compiler.DebugInfo
|
2021-10-23 11:28:03 +00:00
|
|
|
}
|
|
|
|
|
2024-11-28 15:02:18 +00:00
|
|
|
// contracts caches the compiled contracts from FS across multiple tests. The key is a
|
|
|
|
// concatenation of the source file path and the config file path split by | symbol.
|
2021-10-23 11:28:03 +00:00
|
|
|
var contracts = make(map[string]*Contract)
|
|
|
|
|
2022-04-20 18:30:09 +00:00
|
|
|
// CompileSource compiles a contract from the reader and returns its NEF, manifest and hash.
|
2024-10-09 12:52:25 +00:00
|
|
|
// Compiled contract will have "contract.go" used for its file name and coverage
|
|
|
|
// data collection can give wrong results for it, so it's recommended to disable
|
|
|
|
// coverage ([*Executor.DisableCoverage]) when you're deploying contracts
|
|
|
|
// compiled via this function.
|
2022-03-18 15:11:24 +00:00
|
|
|
func CompileSource(t testing.TB, sender util.Uint160, src io.Reader, opts *compiler.Options) *Contract {
|
2021-10-23 11:28:03 +00:00
|
|
|
// nef.NewFile() cares about version a lot.
|
|
|
|
config.Version = "neotest"
|
|
|
|
|
2021-12-02 14:44:53 +00:00
|
|
|
ne, di, err := compiler.CompileWithOptions("contract.go", src, opts)
|
2021-10-23 11:28:03 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
m, err := compiler.CreateManifest(di, opts)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
return &Contract{
|
2024-08-19 11:39:18 +00:00
|
|
|
Hash: state.CreateContractHash(sender, ne.Checksum, m.Name),
|
|
|
|
NEF: ne,
|
|
|
|
Manifest: m,
|
|
|
|
DebugInfo: di,
|
2021-10-23 11:28:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-20 18:30:09 +00:00
|
|
|
// CompileFile compiles a contract from the file and returns its NEF, manifest and hash.
|
2024-11-28 15:02:18 +00:00
|
|
|
// It uses contracts cashes.
|
2022-03-18 15:11:24 +00:00
|
|
|
func CompileFile(t testing.TB, sender util.Uint160, srcPath string, configPath string) *Contract {
|
2024-11-28 15:02:18 +00:00
|
|
|
cacheKey := srcPath + "|" + configPath
|
|
|
|
if c, ok := contracts[cacheKey]; ok {
|
2021-10-23 11:28:03 +00:00
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
// nef.NewFile() cares about version a lot.
|
|
|
|
config.Version = "neotest"
|
|
|
|
|
2021-07-26 12:34:07 +00:00
|
|
|
ne, di, err := compiler.CompileWithOptions(srcPath, nil, nil)
|
2021-10-23 11:28:03 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
conf, err := smartcontract.ParseContractConfig(configPath)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
o := &compiler.Options{}
|
|
|
|
o.Name = conf.Name
|
|
|
|
o.ContractEvents = conf.Events
|
2023-05-08 18:58:28 +00:00
|
|
|
o.DeclaredNamedTypes = conf.NamedTypes
|
2021-10-23 11:28:03 +00:00
|
|
|
o.ContractSupportedStandards = conf.SupportedStandards
|
|
|
|
o.Permissions = make([]manifest.Permission, len(conf.Permissions))
|
|
|
|
for i := range conf.Permissions {
|
|
|
|
o.Permissions[i] = manifest.Permission(conf.Permissions[i])
|
|
|
|
}
|
|
|
|
o.SafeMethods = conf.SafeMethods
|
2022-03-09 10:34:39 +00:00
|
|
|
o.Overloads = conf.Overloads
|
|
|
|
o.SourceURL = conf.SourceURL
|
2021-10-23 11:28:03 +00:00
|
|
|
m, err := compiler.CreateManifest(di, o)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
c := &Contract{
|
2024-08-19 11:39:18 +00:00
|
|
|
Hash: state.CreateContractHash(sender, ne.Checksum, m.Name),
|
|
|
|
NEF: ne,
|
|
|
|
Manifest: m,
|
|
|
|
DebugInfo: di,
|
2021-10-23 11:28:03 +00:00
|
|
|
}
|
2024-11-28 15:02:18 +00:00
|
|
|
contracts[cacheKey] = c
|
2021-10-23 11:28:03 +00:00
|
|
|
return c
|
|
|
|
}
|
2024-12-20 13:03:03 +00:00
|
|
|
|
|
|
|
// ReadNEF loads a contract from the specified NEF and manifest files.
|
|
|
|
func ReadNEF(t testing.TB, sender util.Uint160, nefPath, manifestPath string) *Contract {
|
|
|
|
cacheKey := sender.StringLE() + "|" + nefPath + "|" + manifestPath
|
|
|
|
if c, ok := contracts[cacheKey]; ok {
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
nefBytes, err := os.ReadFile(nefPath)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
ne, err := nef.FileFromBytes(nefBytes)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
manifestBytes, err := os.ReadFile(manifestPath)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
m := new(manifest.Manifest)
|
|
|
|
err = json.Unmarshal(manifestBytes, m)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
hash := state.CreateContractHash(sender, ne.Checksum, m.Name)
|
|
|
|
err = m.IsValid(hash, true)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
c := &Contract{
|
|
|
|
Hash: hash,
|
|
|
|
NEF: &ne,
|
|
|
|
Manifest: m,
|
|
|
|
}
|
|
|
|
|
|
|
|
contracts[cacheKey] = c
|
|
|
|
return c
|
|
|
|
}
|