mirror of
https://github.com/nspcc-dev/neo-go.git
synced 2025-01-05 09:35:49 +00:00
b2bd8e4a0a
Close #3451 Signed-off-by: Ekaterina Pavlova <ekt@morphbits.io>
1012 lines
32 KiB
Go
1012 lines
32 KiB
Go
package rpcbinding
|
|
|
|
import (
|
|
"cmp"
|
|
"fmt"
|
|
"slices"
|
|
"strings"
|
|
"text/template"
|
|
"unicode"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract"
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/binding"
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest/standard"
|
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
|
)
|
|
|
|
// The set of constants containing parts of RPC binding template. Each block of code
|
|
// including template definition and var/type/method definitions contain new line at the
|
|
// start and ends with a new line. On adding new block of code to the template, please,
|
|
// ensure that this block has new line at the start and in the end of the block.
|
|
const (
|
|
eventDefinition = `{{ define "EVENT" }}
|
|
// {{.Name}} represents "{{.ManifestName}}" event emitted by the contract.
|
|
type {{.Name}} struct {
|
|
{{- range $index, $arg := .Parameters}}
|
|
{{ upperFirst .Name}} {{.Type}}
|
|
{{- end}}
|
|
}
|
|
{{ end }}`
|
|
|
|
safemethodDefinition = `{{ define "SAFEMETHOD" }}
|
|
// {{.Name}} {{.Comment}}
|
|
func (c *ContractReader) {{.Name}}({{range $index, $arg := .Arguments -}}
|
|
{{- if ne $index 0}}, {{end}}
|
|
{{- .Name}} {{.Type}}
|
|
{{- end}}) {{if .ReturnType }}({{ .ReturnType }}, error) {
|
|
return {{if and (not .ItemTo) (eq .Unwrapper "Item")}}func(item stackitem.Item, err error) ({{ .ReturnType }}, error) {
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return {{addIndent (etTypeConverter .ExtendedReturn "item") "\t"}}
|
|
} ( {{- end -}} {{if .ItemTo -}} itemTo{{ .ItemTo }}( {{- end -}}
|
|
unwrap.{{.Unwrapper}}(c.invoker.Call(c.hash, "{{ .NameABI }}"
|
|
{{- range $arg := .Arguments -}}, {{.Name}}{{end -}} )) {{- if or .ItemTo (eq .Unwrapper "Item") -}} ) {{- end}}
|
|
{{- else -}} (*result.Invoke, error) {
|
|
c.invoker.Call(c.hash, "{{ .NameABI }}"
|
|
{{- range $arg := .Arguments -}}, {{.Name}}{{end}})
|
|
{{- end}}
|
|
}
|
|
{{ if eq .Unwrapper "SessionIterator" }}
|
|
// {{.Name}}Expanded is similar to {{.Name}} (uses the same contract
|
|
// method), but can be useful if the server used doesn't support sessions and
|
|
// doesn't expand iterators. It creates a script that will get the specified
|
|
// number of result items from the iterator right in the VM and return them to
|
|
// you. It's only limited by VM stack and GAS available for RPC invocations.
|
|
func (c *ContractReader) {{.Name}}Expanded({{range $index, $arg := .Arguments}}{{.Name}} {{.Type}}, {{end}}_numOfIteratorItems int) ([]stackitem.Item, error) {
|
|
return unwrap.Array(c.invoker.CallAndExpandIterator(c.hash, "{{.NameABI}}", _numOfIteratorItems{{range $arg := .Arguments}}, {{.Name}}{{end}}))
|
|
}
|
|
{{ end }}{{ end }}`
|
|
methodDefinition = `{{ define "METHOD" }}{{ if eq .ReturnType "bool"}}
|
|
func (c *Contract) scriptFor{{.Name}}({{range $index, $arg := .Arguments -}}
|
|
{{- if ne $index 0}}, {{end}}
|
|
{{- .Name}} {{.Type}}
|
|
{{- end}}) ([]byte, error) {
|
|
return smartcontract.CreateCallWithAssertScript(c.hash, "{{ .NameABI }}"{{- range $index, $arg := .Arguments -}}, {{.Name}}{{end}})
|
|
}
|
|
{{ end }}
|
|
// {{.Name}} {{.Comment}}
|
|
// This transaction is signed and immediately sent to the network.
|
|
// The values returned are its hash, ValidUntilBlock value and error if any.
|
|
func (c *Contract) {{.Name}}({{range $index, $arg := .Arguments -}}
|
|
{{- if ne $index 0}}, {{end}}
|
|
{{- .Name}} {{.Type}}
|
|
{{- end}}) (util.Uint256, uint32, error) {
|
|
{{if ne .ReturnType "bool"}}return c.actor.SendCall(c.hash, "{{ .NameABI }}"
|
|
{{- range $index, $arg := .Arguments -}}, {{.Name}}{{end}}){{else}}script, err := c.scriptFor{{.Name}}({{- range $index, $arg := .Arguments -}}{{- if ne $index 0}}, {{end}}{{.Name}}{{end}})
|
|
if err != nil {
|
|
return util.Uint256{}, 0, err
|
|
}
|
|
return c.actor.SendRun(script){{end}}
|
|
}
|
|
|
|
// {{.Name}}Transaction {{.Comment}}
|
|
// This transaction is signed, but not sent to the network, instead it's
|
|
// returned to the caller.
|
|
func (c *Contract) {{.Name}}Transaction({{range $index, $arg := .Arguments -}}
|
|
{{- if ne $index 0}}, {{end}}
|
|
{{- .Name}} {{.Type}}
|
|
{{- end}}) (*transaction.Transaction, error) {
|
|
{{if ne .ReturnType "bool"}}return c.actor.MakeCall(c.hash, "{{ .NameABI }}"
|
|
{{- range $index, $arg := .Arguments -}}, {{.Name}}{{end}}){{else}}script, err := c.scriptFor{{.Name}}({{- range $index, $arg := .Arguments -}}{{- if ne $index 0}}, {{end}}{{.Name}}{{end}})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return c.actor.MakeRun(script){{end}}
|
|
}
|
|
|
|
// {{.Name}}Unsigned {{.Comment}}
|
|
// This transaction is not signed, it's simply returned to the caller.
|
|
// Any fields of it that do not affect fees can be changed (ValidUntilBlock,
|
|
// Nonce), fee values (NetworkFee, SystemFee) can be increased as well.
|
|
func (c *Contract) {{.Name}}Unsigned({{range $index, $arg := .Arguments -}}
|
|
{{- if ne $index 0}}, {{end}}
|
|
{{- .Name}} {{.Type}}
|
|
{{- end}}) (*transaction.Transaction, error) {
|
|
{{if ne .ReturnType "bool"}}return c.actor.MakeUnsignedCall(c.hash, "{{ .NameABI }}", nil
|
|
{{- range $index, $arg := .Arguments -}}, {{.Name}}{{end}}){{else}}script, err := c.scriptFor{{.Name}}({{- range $index, $arg := .Arguments -}}{{- if ne $index 0}}, {{end}}{{.Name}}{{end}})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return c.actor.MakeUnsignedRun(script, nil){{end}}
|
|
}
|
|
{{end}}`
|
|
|
|
bindingDefinition = `// Code generated by neo-go contract generate-rpcwrapper --manifest <file.json> --out <file.go> [--hash <hash>] [--config <config>]; DO NOT EDIT.
|
|
|
|
// Package {{.PackageName}} contains RPC wrappers for {{.ContractName}} contract.
|
|
package {{.PackageName}}
|
|
|
|
import (
|
|
{{range $m := .Imports}} "{{ $m }}"
|
|
{{end}})
|
|
{{if len .Hash}}
|
|
// Hash contains contract hash.
|
|
var Hash = {{ .Hash }}
|
|
{{end -}}
|
|
{{- range $index, $typ := .NamedTypes }}
|
|
// {{toTypeName $typ.Name}} is a contract-specific {{$typ.Name}} type used by its methods.
|
|
type {{toTypeName $typ.Name}} struct {
|
|
{{- range $m := $typ.Fields}}
|
|
{{ upperFirst .Field}} {{etTypeToStr .ExtendedType}}
|
|
{{- end}}
|
|
}
|
|
{{end}}
|
|
{{- range $e := .CustomEvents }}{{template "EVENT" $e }}{{ end -}}
|
|
{{- if .HasReader}}
|
|
// Invoker is used by ContractReader to call various safe methods.
|
|
type Invoker interface {
|
|
{{if or .IsNep11D .IsNep11ND}} nep11.Invoker
|
|
{{else -}}
|
|
{{ if .IsNep17}} nep17.Invoker
|
|
{{else if len .SafeMethods}} Call(contract util.Uint160, operation string, params ...any) (*result.Invoke, error)
|
|
{{end -}}
|
|
{{if .HasIterator}} CallAndExpandIterator(contract util.Uint160, method string, maxItems int, params ...any) (*result.Invoke, error)
|
|
TerminateSession(sessionID uuid.UUID) error
|
|
TraverseIterator(sessionID uuid.UUID, iterator *result.Iterator, num int) ([]stackitem.Item, error)
|
|
{{end -}}
|
|
{{end -}}
|
|
}
|
|
{{end -}}
|
|
{{- if .HasWriter}}
|
|
// Actor is used by Contract to call state-changing methods.
|
|
type Actor interface {
|
|
{{- if .HasReader}}
|
|
Invoker
|
|
{{end}}
|
|
{{- if or .IsNep11D .IsNep11ND}}
|
|
nep11.Actor
|
|
{{else if .IsNep17}}
|
|
nep17.Actor
|
|
{{end}}
|
|
{{- if len .Methods}}
|
|
MakeCall(contract util.Uint160, method string, params ...any) (*transaction.Transaction, error)
|
|
MakeRun(script []byte) (*transaction.Transaction, error)
|
|
MakeUnsignedCall(contract util.Uint160, method string, attrs []transaction.Attribute, params ...any) (*transaction.Transaction, error)
|
|
MakeUnsignedRun(script []byte, attrs []transaction.Attribute) (*transaction.Transaction, error)
|
|
SendCall(contract util.Uint160, method string, params ...any) (util.Uint256, uint32, error)
|
|
SendRun(script []byte) (util.Uint256, uint32, error)
|
|
{{end -}}
|
|
}
|
|
{{end -}}
|
|
{{- if .HasReader}}
|
|
// ContractReader implements safe contract methods.
|
|
type ContractReader struct {
|
|
{{if .IsNep11D}}nep11.DivisibleReader
|
|
{{end -}}
|
|
{{if .IsNep11ND}}nep11.NonDivisibleReader
|
|
{{end -}}
|
|
{{if .IsNep17}}nep17.TokenReader
|
|
{{end -}}
|
|
{{if .IsNep24}}nep24.RoyaltyReader
|
|
{{end -}}
|
|
invoker Invoker
|
|
hash util.Uint160
|
|
}
|
|
{{end -}}
|
|
{{- if .HasWriter}}
|
|
// Contract implements all contract methods.
|
|
type Contract struct {
|
|
{{if .HasReader}}ContractReader
|
|
{{end -}}
|
|
{{if .IsNep11D}}nep11.DivisibleWriter
|
|
{{end -}}
|
|
{{if .IsNep11ND}}nep11.BaseWriter
|
|
{{end -}}
|
|
{{if .IsNep17}}nep17.TokenWriter
|
|
{{end -}}
|
|
actor Actor
|
|
hash util.Uint160
|
|
}
|
|
{{end -}}
|
|
{{- if .HasReader}}
|
|
// NewReader creates an instance of ContractReader using {{if len .Hash -}}Hash{{- else -}}provided contract hash{{- end}} and the given Invoker.
|
|
func NewReader(invoker Invoker{{- if not (len .Hash) -}}, hash util.Uint160{{- end -}}) *ContractReader {
|
|
{{if len .Hash -}}
|
|
var hash = Hash
|
|
{{end -}}
|
|
return &ContractReader{
|
|
{{- if .IsNep11D}}*nep11.NewDivisibleReader(invoker, hash), {{end}}
|
|
{{- if .IsNep11ND}}*nep11.NewNonDivisibleReader(invoker, hash), {{end}}
|
|
{{- if .IsNep17}}*nep17.NewReader(invoker, hash), {{end -}}
|
|
{{- if .IsNep24}}*nep24.NewRoyaltyReader(invoker, hash), {{end -}}
|
|
invoker, hash}
|
|
}
|
|
{{end -}}
|
|
{{- if .HasWriter}}
|
|
// New creates an instance of Contract using {{if len .Hash -}}Hash{{- else -}}provided contract hash{{- end}} and the given Actor.
|
|
func New(actor Actor{{- if not (len .Hash) -}}, hash util.Uint160{{- end -}}) *Contract {
|
|
{{if len .Hash -}}
|
|
var hash = Hash
|
|
{{end -}}
|
|
{{if .IsNep11D}}var nep11dt = nep11.NewDivisible(actor, hash)
|
|
{{end -}}
|
|
{{if .IsNep11ND}}var nep11ndt = nep11.NewNonDivisible(actor, hash)
|
|
{{end -}}
|
|
{{if .IsNep17}}var nep17t = nep17.New(actor, hash)
|
|
{{end -}}
|
|
{{if .IsNep24}}var nep24t = nep24.NewRoyaltyReader(actor, hash)
|
|
{{end -}}
|
|
return &Contract{
|
|
{{- if .HasReader}}ContractReader{
|
|
{{- if .IsNep11D}}nep11dt.DivisibleReader, {{end -}}
|
|
{{- if .IsNep11ND}}nep11ndt.NonDivisibleReader, {{end -}}
|
|
{{- if .IsNep17}}nep17t.TokenReader, {{end -}}
|
|
{{- if .IsNep24}}*nep24t, {{end -}}
|
|
actor, hash}, {{end -}}
|
|
{{- if .IsNep11D}}nep11dt.DivisibleWriter, {{end -}}
|
|
{{- if .IsNep11ND}}nep11ndt.BaseWriter, {{end -}}
|
|
{{- if .IsNep17}}nep17t.TokenWriter, {{end -}}
|
|
actor, hash}
|
|
}
|
|
{{end -}}
|
|
{{- range $m := .SafeMethods }}{{template "SAFEMETHOD" $m }}{{ end -}}
|
|
{{- range $m := .Methods -}}{{template "METHOD" $m }}{{ end -}}
|
|
{{- range $index, $typ := .NamedTypes }}
|
|
// itemTo{{toTypeName $typ.Name}} converts stack item into *{{toTypeName $typ.Name}}.
|
|
// NULL item is returned as nil pointer without error.
|
|
func itemTo{{toTypeName $typ.Name}}(item stackitem.Item, err error) (*{{toTypeName $typ.Name}}, error) {
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
_, null := item.(stackitem.Null)
|
|
if null {
|
|
return nil, nil
|
|
}
|
|
var res = new({{toTypeName $typ.Name}})
|
|
err = res.FromStackItem(item)
|
|
return res, err
|
|
}
|
|
|
|
// FromStackItem retrieves fields of {{toTypeName $typ.Name}} from the given
|
|
// [stackitem.Item] or returns an error if it's not possible to do to so.
|
|
func (res *{{toTypeName $typ.Name}}) FromStackItem(item stackitem.Item) error {
|
|
arr, ok := item.Value().([]stackitem.Item)
|
|
if !ok {
|
|
return errors.New("not an array")
|
|
}
|
|
if len(arr) != {{len $typ.Fields}} {
|
|
return errors.New("wrong number of structure elements")
|
|
}
|
|
{{if len .Fields}}
|
|
var (
|
|
index = -1
|
|
err error
|
|
)
|
|
{{- range $m := $typ.Fields}}
|
|
index++
|
|
res.{{ upperFirst .Field}}, err = {{etTypeConverter .ExtendedType "arr[index]"}}
|
|
if err != nil {
|
|
return fmt.Errorf("field {{ upperFirst .Field}}: %w", err)
|
|
}
|
|
{{end}}
|
|
{{- end}}
|
|
return nil
|
|
}
|
|
{{ end -}}
|
|
{{- range $e := .CustomEvents }}
|
|
// {{$e.Name}}sFromApplicationLog retrieves a set of all emitted events
|
|
// with "{{$e.ManifestName}}" name from the provided [result.ApplicationLog].
|
|
func {{$e.Name}}sFromApplicationLog(log *result.ApplicationLog) ([]*{{$e.Name}}, error) {
|
|
if log == nil {
|
|
return nil, errors.New("nil application log")
|
|
}
|
|
|
|
var res []*{{$e.Name}}
|
|
for i, ex := range log.Executions {
|
|
for j, e := range ex.Events {
|
|
if e.Name != "{{$e.ManifestName}}" {
|
|
continue
|
|
}
|
|
event := new({{$e.Name}})
|
|
err := event.FromStackItem(e.Item)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to deserialize {{$e.Name}} from stackitem (execution #%d, event #%d): %w", i, j, err)
|
|
}
|
|
res = append(res, event)
|
|
}
|
|
}
|
|
|
|
return res, nil
|
|
}
|
|
|
|
// FromStackItem converts provided [stackitem.Array] to {{$e.Name}} or
|
|
// returns an error if it's not possible to do to so.
|
|
func (e *{{$e.Name}}) FromStackItem(item *stackitem.Array) error {
|
|
if item == nil {
|
|
return errors.New("nil item")
|
|
}
|
|
arr, ok := item.Value().([]stackitem.Item)
|
|
if !ok {
|
|
return errors.New("not an array")
|
|
}
|
|
if len(arr) != {{len $e.Parameters}} {
|
|
return errors.New("wrong number of structure elements")
|
|
}
|
|
|
|
{{if len $e.Parameters}}var (
|
|
index = -1
|
|
err error
|
|
)
|
|
{{- range $p := $e.Parameters}}
|
|
index++
|
|
e.{{ upperFirst .Name}}, err = {{etTypeConverter .ExtType "arr[index]"}}
|
|
if err != nil {
|
|
return fmt.Errorf("field {{ upperFirst .Name}}: %w", err)
|
|
}
|
|
{{end}}
|
|
{{- end}}
|
|
return nil
|
|
}
|
|
{{end -}}`
|
|
|
|
srcTmpl = bindingDefinition +
|
|
eventDefinition +
|
|
safemethodDefinition +
|
|
methodDefinition
|
|
)
|
|
|
|
type (
|
|
ContractTmpl struct {
|
|
binding.ContractTmpl
|
|
|
|
SafeMethods []SafeMethodTmpl
|
|
CustomEvents []CustomEventTemplate
|
|
NamedTypes []binding.ExtendedType
|
|
|
|
IsNep11D bool
|
|
IsNep11ND bool
|
|
IsNep17 bool
|
|
IsNep24 bool
|
|
IsNep24Payable bool
|
|
|
|
HasReader bool
|
|
HasWriter bool
|
|
HasIterator bool
|
|
}
|
|
|
|
SafeMethodTmpl struct {
|
|
binding.MethodTmpl
|
|
Unwrapper string
|
|
ItemTo string
|
|
ExtendedReturn binding.ExtendedType
|
|
}
|
|
|
|
CustomEventTemplate struct {
|
|
// Name is the event's name that will be used as the event structure name in
|
|
// the resulting RPC binding. It is a valid go structure name and may differ
|
|
// from ManifestName.
|
|
Name string
|
|
// ManifestName is the event's name declared in the contract manifest.
|
|
// It may contain any UTF8 character.
|
|
ManifestName string
|
|
Parameters []EventParamTmpl
|
|
}
|
|
|
|
EventParamTmpl struct {
|
|
binding.ParamTmpl
|
|
|
|
// ExtType holds the event parameter's type information provided by Manifest,
|
|
// i.e. simple types only.
|
|
ExtType binding.ExtendedType
|
|
}
|
|
)
|
|
|
|
// NewConfig initializes and returns a new config instance.
|
|
func NewConfig() binding.Config {
|
|
return binding.NewConfig()
|
|
}
|
|
|
|
// Generate writes Go file containing smartcontract bindings to the `cfg.Output`.
|
|
// It doesn't check manifest from Config for validity, incorrect manifest can
|
|
// lead to unexpected results.
|
|
func Generate(cfg binding.Config) error {
|
|
// Avoid changing *cfg.Manifest.
|
|
mfst := *cfg.Manifest
|
|
mfst.ABI.Methods = slices.Clone(mfst.ABI.Methods)
|
|
cfg.Manifest = &mfst
|
|
|
|
var imports = make(map[string]struct{})
|
|
var ctr ContractTmpl
|
|
|
|
// Strip standard methods from NEP-XX packages.
|
|
for _, std := range cfg.Manifest.SupportedStandards {
|
|
switch std {
|
|
case manifest.NEP11StandardName:
|
|
imports["github.com/nspcc-dev/neo-go/pkg/rpcclient/nep11"] = struct{}{}
|
|
if standard.ComplyABI(cfg.Manifest, standard.Nep11Divisible) == nil {
|
|
ctr.IsNep11D = true
|
|
} else if standard.ComplyABI(cfg.Manifest, standard.Nep11NonDivisible) == nil {
|
|
ctr.IsNep11ND = true
|
|
}
|
|
case manifest.NEP17StandardName:
|
|
if standard.ComplyABI(cfg.Manifest, standard.Nep17) == nil {
|
|
imports["github.com/nspcc-dev/neo-go/pkg/rpcclient/nep17"] = struct{}{}
|
|
ctr.IsNep17 = true
|
|
}
|
|
case manifest.NEP24StandardName:
|
|
if standard.ComplyABI(cfg.Manifest, standard.Nep24) == nil {
|
|
imports["github.com/nspcc-dev/neo-go/pkg/rpcclient/nep24"] = struct{}{}
|
|
ctr.IsNep24 = true
|
|
}
|
|
case manifest.NEP24Payable:
|
|
if standard.ComplyABI(cfg.Manifest, standard.Nep24Payable) == nil {
|
|
ctr.IsNep24Payable = true
|
|
}
|
|
}
|
|
}
|
|
|
|
if ctr.IsNep11D {
|
|
mfst.ABI.Methods = dropStdMethods(mfst.ABI.Methods, standard.Nep11Divisible)
|
|
}
|
|
if ctr.IsNep11ND {
|
|
mfst.ABI.Methods = dropStdMethods(mfst.ABI.Methods, standard.Nep11NonDivisible)
|
|
}
|
|
if ctr.IsNep11D || ctr.IsNep11ND {
|
|
mfst.ABI.Events = dropStdEvents(mfst.ABI.Events, standard.Nep11Base)
|
|
}
|
|
if ctr.IsNep17 {
|
|
mfst.ABI.Methods = dropStdMethods(mfst.ABI.Methods, standard.Nep17)
|
|
mfst.ABI.Events = dropStdEvents(mfst.ABI.Events, standard.Nep17)
|
|
}
|
|
if ctr.IsNep24 {
|
|
mfst.ABI.Methods = dropStdMethods(mfst.ABI.Methods, standard.Nep24)
|
|
cfg = dropNep24Types(cfg)
|
|
}
|
|
if ctr.IsNep24Payable {
|
|
mfst.ABI.Events = dropStdEvents(mfst.ABI.Events, standard.Nep24Payable)
|
|
}
|
|
|
|
// OnNepXXPayment handlers normally can't be called directly.
|
|
if standard.ComplyABI(cfg.Manifest, standard.Nep11Payable) == nil {
|
|
mfst.ABI.Methods = dropStdMethods(mfst.ABI.Methods, standard.Nep11Payable)
|
|
}
|
|
if standard.ComplyABI(cfg.Manifest, standard.Nep17Payable) == nil {
|
|
mfst.ABI.Methods = dropStdMethods(mfst.ABI.Methods, standard.Nep17Payable)
|
|
}
|
|
|
|
ctr.ContractTmpl = binding.TemplateFromManifest(cfg, scTypeToGo)
|
|
ctr = scTemplateToRPC(cfg, ctr, imports, scTypeToGo)
|
|
ctr.NamedTypes = make([]binding.ExtendedType, 0, len(cfg.NamedTypes))
|
|
for k := range cfg.NamedTypes {
|
|
ctr.NamedTypes = append(ctr.NamedTypes, cfg.NamedTypes[k])
|
|
}
|
|
slices.SortFunc(ctr.NamedTypes, func(a, b binding.ExtendedType) int { return strings.Compare(a.Name, b.Name) })
|
|
|
|
// Check resulting named types and events don't have duplicating field names.
|
|
for _, t := range ctr.NamedTypes {
|
|
fDict := make(map[string]struct{})
|
|
for _, n := range t.Fields {
|
|
name := upperFirst(n.Field)
|
|
if _, ok := fDict[name]; ok {
|
|
return fmt.Errorf("named type `%s` has two fields with identical resulting binding name `%s`", t.Name, name)
|
|
}
|
|
fDict[name] = struct{}{}
|
|
}
|
|
}
|
|
for _, e := range ctr.CustomEvents {
|
|
fDict := make(map[string]struct{})
|
|
for _, n := range e.Parameters {
|
|
name := upperFirst(n.Name)
|
|
if _, ok := fDict[name]; ok {
|
|
return fmt.Errorf("event `%s` has two fields with identical resulting binding name `%s`", e.Name, name)
|
|
}
|
|
fDict[name] = struct{}{}
|
|
}
|
|
}
|
|
|
|
var srcTemplate = template.Must(template.New("generate").Funcs(template.FuncMap{
|
|
"addIndent": addIndent,
|
|
"etTypeConverter": etTypeConverter,
|
|
"etTypeToStr": func(et binding.ExtendedType) string {
|
|
r, _ := extendedTypeToGo(et, cfg.NamedTypes)
|
|
return r
|
|
},
|
|
"toTypeName": toTypeName,
|
|
"cutPointer": cutPointer,
|
|
"upperFirst": upperFirst,
|
|
}).Parse(srcTmpl))
|
|
|
|
return binding.FExecute(srcTemplate, cfg.Output, ctr)
|
|
}
|
|
|
|
func dropManifestMethods(meths []manifest.Method, manifested []manifest.Method) []manifest.Method {
|
|
return slices.DeleteFunc(meths, func(m manifest.Method) bool {
|
|
return slices.ContainsFunc(manifested, func(e manifest.Method) bool {
|
|
return 0 == cmp.Or(
|
|
cmp.Compare(m.Name, e.Name),
|
|
cmp.Compare(len(m.Parameters), len(e.Parameters)),
|
|
)
|
|
})
|
|
})
|
|
}
|
|
|
|
func dropManifestEvents(events []manifest.Event, manifested []manifest.Event) []manifest.Event {
|
|
return slices.DeleteFunc(events, func(e manifest.Event) bool {
|
|
return slices.ContainsFunc(manifested, func(v manifest.Event) bool {
|
|
return 0 == cmp.Or(
|
|
cmp.Compare(e.Name, v.Name),
|
|
cmp.Compare(len(e.Parameters), len(v.Parameters)),
|
|
)
|
|
})
|
|
})
|
|
}
|
|
|
|
func dropStdMethods(meths []manifest.Method, std *standard.Standard) []manifest.Method {
|
|
meths = dropManifestMethods(meths, std.Manifest.ABI.Methods)
|
|
if std.Optional != nil {
|
|
meths = dropManifestMethods(meths, std.Optional)
|
|
}
|
|
if std.Base != nil {
|
|
return dropStdMethods(meths, std.Base)
|
|
}
|
|
return meths
|
|
}
|
|
|
|
func dropStdEvents(events []manifest.Event, std *standard.Standard) []manifest.Event {
|
|
events = dropManifestEvents(events, std.Manifest.ABI.Events)
|
|
if std.Base != nil {
|
|
return dropStdEvents(events, std.Base)
|
|
}
|
|
return events
|
|
}
|
|
|
|
// dropNep24Types removes NamedTypes of NEP-24 from the config if they are used only from the methods of the standard.
|
|
func dropNep24Types(cfg binding.Config) binding.Config {
|
|
var targetTypeName string
|
|
// Find structure returned by standard.MethodRoyaltyInfo method
|
|
// and remove it from binding.Config.NamedTypes as it will be imported from nep24 package.
|
|
if royaltyInfo, ok := cfg.Types[standard.MethodRoyaltyInfo]; ok && royaltyInfo.Value != nil {
|
|
returnType, exists := cfg.NamedTypes[royaltyInfo.Value.Name]
|
|
if !exists || returnType.Fields == nil || len(returnType.Fields) != 2 ||
|
|
returnType.Fields[0].ExtendedType.Base != smartcontract.Hash160Type ||
|
|
returnType.Fields[1].ExtendedType.Base != smartcontract.IntegerType {
|
|
return cfg
|
|
}
|
|
targetTypeName = royaltyInfo.Value.Name
|
|
} else {
|
|
return cfg
|
|
}
|
|
found := false
|
|
for _, typeDef := range cfg.Types {
|
|
if typeDef.Value != nil && typeDef.Value.Name == targetTypeName {
|
|
if found {
|
|
return cfg
|
|
}
|
|
found = true
|
|
}
|
|
}
|
|
|
|
if found {
|
|
delete(cfg.NamedTypes, targetTypeName)
|
|
}
|
|
return cfg
|
|
}
|
|
|
|
func extendedTypeToGo(et binding.ExtendedType, named map[string]binding.ExtendedType) (string, string) {
|
|
switch et.Base {
|
|
case smartcontract.AnyType:
|
|
return "any", ""
|
|
case smartcontract.BoolType:
|
|
return "bool", ""
|
|
case smartcontract.IntegerType:
|
|
return "*big.Int", "math/big"
|
|
case smartcontract.ByteArrayType:
|
|
return "[]byte", ""
|
|
case smartcontract.StringType:
|
|
return "string", ""
|
|
case smartcontract.Hash160Type:
|
|
return "util.Uint160", "github.com/nspcc-dev/neo-go/pkg/util"
|
|
case smartcontract.Hash256Type:
|
|
return "util.Uint256", "github.com/nspcc-dev/neo-go/pkg/util"
|
|
case smartcontract.PublicKeyType:
|
|
return "*keys.PublicKey", "github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
|
case smartcontract.SignatureType:
|
|
return "[]byte", ""
|
|
case smartcontract.ArrayType:
|
|
if len(et.Name) > 0 {
|
|
return "*" + toTypeName(et.Name), "github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
|
} else if et.Value != nil {
|
|
if et.Value.Base == smartcontract.PublicKeyType { // Special array wrapper.
|
|
return "keys.PublicKeys", "github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
|
}
|
|
sub, pkg := extendedTypeToGo(*et.Value, named)
|
|
return "[]" + sub, pkg
|
|
}
|
|
return "[]any", ""
|
|
|
|
case smartcontract.MapType:
|
|
kt, _ := extendedTypeToGo(binding.ExtendedType{Base: et.Key}, named)
|
|
var vt string
|
|
if et.Value != nil {
|
|
vt, _ = extendedTypeToGo(*et.Value, named)
|
|
} else {
|
|
vt = "any"
|
|
}
|
|
return "map[" + kt + "]" + vt, "github.com/nspcc-dev/neo-go/pkg/vm/stackitem"
|
|
case smartcontract.InteropInterfaceType:
|
|
return "any", ""
|
|
case smartcontract.VoidType:
|
|
return "", ""
|
|
default:
|
|
panic("unreachable")
|
|
}
|
|
}
|
|
|
|
func etTypeConverter(et binding.ExtendedType, v string) string {
|
|
switch et.Base {
|
|
case smartcontract.AnyType:
|
|
return v + ".Value(), error(nil)"
|
|
case smartcontract.BoolType:
|
|
return v + ".TryBool()"
|
|
case smartcontract.IntegerType:
|
|
return v + ".TryInteger()"
|
|
case smartcontract.ByteArrayType, smartcontract.SignatureType:
|
|
return v + ".TryBytes()"
|
|
case smartcontract.StringType:
|
|
return `func(item stackitem.Item) (string, error) {
|
|
b, err := item.TryBytes()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
if !utf8.Valid(b) {
|
|
return "", errors.New("not a UTF-8 string")
|
|
}
|
|
return string(b), nil
|
|
}(` + v + `)`
|
|
case smartcontract.Hash160Type:
|
|
return `func(item stackitem.Item) (util.Uint160, error) {
|
|
b, err := item.TryBytes()
|
|
if err != nil {
|
|
return util.Uint160{}, err
|
|
}
|
|
u, err := util.Uint160DecodeBytesBE(b)
|
|
if err != nil {
|
|
return util.Uint160{}, err
|
|
}
|
|
return u, nil
|
|
}(` + v + `)`
|
|
case smartcontract.Hash256Type:
|
|
return `func(item stackitem.Item) (util.Uint256, error) {
|
|
b, err := item.TryBytes()
|
|
if err != nil {
|
|
return util.Uint256{}, err
|
|
}
|
|
u, err := util.Uint256DecodeBytesBE(b)
|
|
if err != nil {
|
|
return util.Uint256{}, err
|
|
}
|
|
return u, nil
|
|
}(` + v + `)`
|
|
case smartcontract.PublicKeyType:
|
|
return `func(item stackitem.Item) (*keys.PublicKey, error) {
|
|
b, err := item.TryBytes()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
k, err := keys.NewPublicKeyFromBytes(b, elliptic.P256())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return k, nil
|
|
}(` + v + `)`
|
|
case smartcontract.ArrayType:
|
|
if len(et.Name) > 0 {
|
|
return "itemTo" + toTypeName(et.Name) + "(" + v + ", nil)"
|
|
} else if et.Value != nil {
|
|
at, _ := extendedTypeToGo(et, nil)
|
|
return `func(item stackitem.Item) (` + at + `, error) {
|
|
arr, ok := item.Value().([]stackitem.Item)
|
|
if !ok {
|
|
return nil, errors.New("not an array")
|
|
}
|
|
res := make(` + at + `, len(arr))
|
|
for i := range res {
|
|
res[i], err = ` + addIndent(etTypeConverter(*et.Value, "arr[i]"), "\t\t") + `
|
|
if err != nil {
|
|
return nil, fmt.Errorf("item %d: %w", i, err)
|
|
}
|
|
}
|
|
return res, nil
|
|
}(` + v + `)`
|
|
}
|
|
return etTypeConverter(binding.ExtendedType{
|
|
Base: smartcontract.ArrayType,
|
|
Value: &binding.ExtendedType{
|
|
Base: smartcontract.AnyType,
|
|
},
|
|
}, v)
|
|
|
|
case smartcontract.MapType:
|
|
if et.Value != nil {
|
|
at, _ := extendedTypeToGo(et, nil)
|
|
return `func(item stackitem.Item) (` + at + `, error) {
|
|
m, ok := item.Value().([]stackitem.MapElement)
|
|
if !ok {
|
|
return nil, fmt.Errorf("%s is not a map", item.Type().String())
|
|
}
|
|
res := make(` + at + `)
|
|
for i := range m {
|
|
k, err := ` + addIndent(etTypeConverter(binding.ExtendedType{Base: et.Key}, "m[i].Key"), "\t\t") + `
|
|
if err != nil {
|
|
return nil, fmt.Errorf("key %d: %w", i, err)
|
|
}
|
|
v, err := ` + addIndent(etTypeConverter(*et.Value, "m[i].Value"), "\t\t") + `
|
|
if err != nil {
|
|
return nil, fmt.Errorf("value %d: %w", i, err)
|
|
}
|
|
res[k] = v
|
|
}
|
|
return res, nil
|
|
}(` + v + `)`
|
|
}
|
|
return etTypeConverter(binding.ExtendedType{
|
|
Base: smartcontract.MapType,
|
|
Key: et.Key,
|
|
Value: &binding.ExtendedType{
|
|
Base: smartcontract.AnyType,
|
|
},
|
|
}, v)
|
|
case smartcontract.InteropInterfaceType:
|
|
return "item.Value(), nil"
|
|
case smartcontract.VoidType:
|
|
return ""
|
|
default:
|
|
panic("unreachable")
|
|
}
|
|
}
|
|
|
|
func scTypeToGo(name string, typ smartcontract.ParamType, cfg *binding.Config) (string, string) {
|
|
et, ok := cfg.Types[name]
|
|
if !ok {
|
|
et = binding.ExtendedType{Base: typ}
|
|
}
|
|
return extendedTypeToGo(et, cfg.NamedTypes)
|
|
}
|
|
|
|
func scTemplateToRPC(cfg binding.Config, ctr ContractTmpl, imports map[string]struct{}, scTypeConverter func(string, smartcontract.ParamType, *binding.Config) (string, string)) ContractTmpl {
|
|
for i := range ctr.Imports {
|
|
imports[ctr.Imports[i]] = struct{}{}
|
|
}
|
|
if !cfg.Hash.Equals(util.Uint160{}) {
|
|
ctr.Hash = fmt.Sprintf("%#v", cfg.Hash)
|
|
}
|
|
for i := 0; i < len(ctr.Methods); i++ {
|
|
abim := cfg.Manifest.ABI.GetMethod(ctr.Methods[i].NameABI, len(ctr.Methods[i].Arguments))
|
|
if abim.Safe {
|
|
ctr.SafeMethods = append(ctr.SafeMethods, SafeMethodTmpl{MethodTmpl: ctr.Methods[i]})
|
|
et, ok := cfg.Types[abim.Name]
|
|
if ok {
|
|
ctr.SafeMethods[len(ctr.SafeMethods)-1].ExtendedReturn = et
|
|
if abim.ReturnType == smartcontract.ArrayType && len(et.Name) > 0 {
|
|
ctr.SafeMethods[len(ctr.SafeMethods)-1].ItemTo = cutPointer(ctr.Methods[i].ReturnType)
|
|
}
|
|
}
|
|
ctr.Methods = slices.Delete(ctr.Methods, i, i+1)
|
|
i--
|
|
} else {
|
|
ctr.Methods[i].Comment = fmt.Sprintf("creates a transaction invoking `%s` method of the contract.", ctr.Methods[i].NameABI)
|
|
if ctr.Methods[i].ReturnType == "bool" {
|
|
imports["github.com/nspcc-dev/neo-go/pkg/smartcontract"] = struct{}{}
|
|
}
|
|
}
|
|
}
|
|
for _, et := range cfg.NamedTypes {
|
|
addETImports(et, cfg.NamedTypes, imports)
|
|
}
|
|
if len(cfg.NamedTypes) > 0 {
|
|
imports["errors"] = struct{}{}
|
|
}
|
|
for _, abiEvent := range cfg.Manifest.ABI.Events {
|
|
eBindingName := ToEventBindingName(abiEvent.Name)
|
|
eTmp := CustomEventTemplate{
|
|
Name: eBindingName,
|
|
ManifestName: abiEvent.Name,
|
|
}
|
|
for i := range abiEvent.Parameters {
|
|
pBindingName := ToParameterBindingName(abiEvent.Parameters[i].Name)
|
|
fullPName := eBindingName + "." + pBindingName
|
|
typeStr, pkg := scTypeConverter(fullPName, abiEvent.Parameters[i].Type, &cfg)
|
|
if pkg != "" {
|
|
imports[pkg] = struct{}{}
|
|
}
|
|
|
|
var (
|
|
extType binding.ExtendedType
|
|
ok bool
|
|
)
|
|
if extType, ok = cfg.Types[fullPName]; !ok {
|
|
extType = binding.ExtendedType{
|
|
Base: abiEvent.Parameters[i].Type,
|
|
}
|
|
addETImports(extType, cfg.NamedTypes, imports)
|
|
}
|
|
eTmp.Parameters = append(eTmp.Parameters, EventParamTmpl{
|
|
ParamTmpl: binding.ParamTmpl{
|
|
Name: pBindingName,
|
|
Type: typeStr,
|
|
},
|
|
ExtType: extType,
|
|
})
|
|
}
|
|
ctr.CustomEvents = append(ctr.CustomEvents, eTmp)
|
|
}
|
|
|
|
if len(ctr.CustomEvents) > 0 {
|
|
imports["github.com/nspcc-dev/neo-go/pkg/neorpc/result"] = struct{}{}
|
|
imports["github.com/nspcc-dev/neo-go/pkg/vm/stackitem"] = struct{}{}
|
|
imports["fmt"] = struct{}{}
|
|
imports["errors"] = struct{}{}
|
|
}
|
|
|
|
for i := range ctr.SafeMethods {
|
|
switch ctr.SafeMethods[i].ReturnType {
|
|
case "any":
|
|
abim := cfg.Manifest.ABI.GetMethod(ctr.SafeMethods[i].NameABI, len(ctr.SafeMethods[i].Arguments))
|
|
if abim.ReturnType == smartcontract.InteropInterfaceType {
|
|
imports["github.com/google/uuid"] = struct{}{}
|
|
imports["github.com/nspcc-dev/neo-go/pkg/vm/stackitem"] = struct{}{}
|
|
imports["github.com/nspcc-dev/neo-go/pkg/neorpc/result"] = struct{}{}
|
|
ctr.SafeMethods[i].ReturnType = "uuid.UUID, result.Iterator"
|
|
ctr.SafeMethods[i].Unwrapper = "SessionIterator"
|
|
ctr.HasIterator = true
|
|
} else {
|
|
imports["github.com/nspcc-dev/neo-go/pkg/vm/stackitem"] = struct{}{}
|
|
ctr.SafeMethods[i].ReturnType = "any"
|
|
ctr.SafeMethods[i].Unwrapper = "Item"
|
|
}
|
|
case "bool":
|
|
ctr.SafeMethods[i].Unwrapper = "Bool"
|
|
case "*big.Int":
|
|
ctr.SafeMethods[i].Unwrapper = "BigInt"
|
|
case "string":
|
|
ctr.SafeMethods[i].Unwrapper = "UTF8String"
|
|
case "util.Uint160":
|
|
ctr.SafeMethods[i].Unwrapper = "Uint160"
|
|
case "util.Uint256":
|
|
ctr.SafeMethods[i].Unwrapper = "Uint256"
|
|
case "*keys.PublicKey":
|
|
ctr.SafeMethods[i].Unwrapper = "PublicKey"
|
|
case "[]byte":
|
|
ctr.SafeMethods[i].Unwrapper = "Bytes"
|
|
case "[]any":
|
|
imports["github.com/nspcc-dev/neo-go/pkg/vm/stackitem"] = struct{}{}
|
|
ctr.SafeMethods[i].ReturnType = "[]stackitem.Item"
|
|
ctr.SafeMethods[i].Unwrapper = "Array"
|
|
case "*stackitem.Map":
|
|
ctr.SafeMethods[i].Unwrapper = "Map"
|
|
case "[]bool":
|
|
ctr.SafeMethods[i].Unwrapper = "ArrayOfBools"
|
|
case "[]*big.Int":
|
|
ctr.SafeMethods[i].Unwrapper = "ArrayOfBigInts"
|
|
case "[][]byte":
|
|
ctr.SafeMethods[i].Unwrapper = "ArrayOfBytes"
|
|
case "[]string":
|
|
ctr.SafeMethods[i].Unwrapper = "ArrayOfUTF8Strings"
|
|
case "[]util.Uint160":
|
|
ctr.SafeMethods[i].Unwrapper = "ArrayOfUint160"
|
|
case "[]util.Uint256":
|
|
ctr.SafeMethods[i].Unwrapper = "ArrayOfUint256"
|
|
case "keys.PublicKeys":
|
|
ctr.SafeMethods[i].Unwrapper = "ArrayOfPublicKeys"
|
|
default:
|
|
addETImports(ctr.SafeMethods[i].ExtendedReturn, cfg.NamedTypes, imports)
|
|
ctr.SafeMethods[i].Unwrapper = "Item"
|
|
}
|
|
}
|
|
|
|
imports["github.com/nspcc-dev/neo-go/pkg/util"] = struct{}{}
|
|
if len(ctr.SafeMethods) > 0 {
|
|
imports["github.com/nspcc-dev/neo-go/pkg/rpcclient/unwrap"] = struct{}{}
|
|
if !(ctr.IsNep17 || ctr.IsNep11D || ctr.IsNep11ND || ctr.IsNep24) {
|
|
imports["github.com/nspcc-dev/neo-go/pkg/neorpc/result"] = struct{}{}
|
|
}
|
|
}
|
|
if len(ctr.Methods) > 0 {
|
|
imports["github.com/nspcc-dev/neo-go/pkg/core/transaction"] = struct{}{}
|
|
}
|
|
if len(ctr.Methods) > 0 || ctr.IsNep17 || ctr.IsNep11D || ctr.IsNep11ND {
|
|
ctr.HasWriter = true
|
|
}
|
|
if len(ctr.SafeMethods) > 0 || ctr.IsNep17 || ctr.IsNep11D || ctr.IsNep11ND || ctr.IsNep24 {
|
|
ctr.HasReader = true
|
|
}
|
|
ctr.Imports = ctr.Imports[:0]
|
|
for imp := range imports {
|
|
ctr.Imports = append(ctr.Imports, imp)
|
|
}
|
|
slices.Sort(ctr.Imports)
|
|
return ctr
|
|
}
|
|
|
|
func addETImports(et binding.ExtendedType, named map[string]binding.ExtendedType, imports map[string]struct{}) {
|
|
_, pkg := extendedTypeToGo(et, named)
|
|
if pkg != "" {
|
|
imports[pkg] = struct{}{}
|
|
}
|
|
// Additional packages used during decoding.
|
|
switch et.Base {
|
|
case smartcontract.StringType:
|
|
imports["unicode/utf8"] = struct{}{}
|
|
imports["errors"] = struct{}{}
|
|
case smartcontract.PublicKeyType:
|
|
imports["crypto/elliptic"] = struct{}{}
|
|
case smartcontract.MapType:
|
|
imports["fmt"] = struct{}{}
|
|
case smartcontract.ArrayType:
|
|
imports["errors"] = struct{}{}
|
|
imports["fmt"] = struct{}{}
|
|
default:
|
|
}
|
|
if et.Value != nil {
|
|
addETImports(*et.Value, named, imports)
|
|
}
|
|
if et.Base == smartcontract.MapType {
|
|
addETImports(binding.ExtendedType{Base: et.Key}, named, imports)
|
|
}
|
|
for i := range et.Fields {
|
|
addETImports(et.Fields[i].ExtendedType, named, imports)
|
|
}
|
|
}
|
|
|
|
func cutPointer(s string) string {
|
|
if s[0] == '*' {
|
|
return s[1:]
|
|
}
|
|
return s
|
|
}
|
|
|
|
func toTypeName(s string) string {
|
|
return strings.Map(func(c rune) rune {
|
|
if c == '.' {
|
|
return -1
|
|
}
|
|
return c
|
|
}, upperFirst(s))
|
|
}
|
|
|
|
func addIndent(str string, ind string) string {
|
|
return strings.ReplaceAll(str, "\n", "\n"+ind)
|
|
}
|
|
|
|
// ToEventBindingName converts event name specified in the contract manifest to
|
|
// a valid go exported event structure name.
|
|
func ToEventBindingName(eventName string) string {
|
|
return toPascalCase(eventName) + "Event"
|
|
}
|
|
|
|
// ToParameterBindingName converts parameter name specified in the contract
|
|
// manifest to a valid go structure's exported field name.
|
|
func ToParameterBindingName(paramName string) string {
|
|
return toPascalCase(paramName)
|
|
}
|
|
|
|
// toPascalCase removes all non-unicode characters from the provided string and
|
|
// converts it to pascal case using space as delimiter.
|
|
func toPascalCase(s string) string {
|
|
var res string
|
|
ss := strings.Split(s, " ")
|
|
for i := range ss { // TODO: use DecodeRuneInString instead.
|
|
var word string
|
|
for _, ch := range ss[i] {
|
|
var ok bool
|
|
if len(res) == 0 && len(word) == 0 {
|
|
ok = unicode.IsLetter(ch)
|
|
} else {
|
|
ok = unicode.IsLetter(ch) || unicode.IsDigit(ch) || ch == '_'
|
|
}
|
|
if ok {
|
|
word += string(ch)
|
|
}
|
|
}
|
|
if len(word) > 0 {
|
|
res += upperFirst(word)
|
|
}
|
|
}
|
|
return res
|
|
}
|
|
|
|
func upperFirst(s string) string {
|
|
return strings.ToUpper(s[0:1]) + s[1:]
|
|
}
|