mirror of
https://github.com/nspcc-dev/neo-go.git
synced 2025-01-10 05:54:04 +00:00
b2bd8e4a0a
Close #3451 Signed-off-by: Ekaterina Pavlova <ekt@morphbits.io>
142 lines
4.6 KiB
Go
142 lines
4.6 KiB
Go
package standard
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/smartcontract/manifest"
|
|
)
|
|
|
|
// Various validation errors.
|
|
var (
|
|
ErrMethodMissing = errors.New("method missing")
|
|
ErrEventMissing = errors.New("event missing")
|
|
ErrInvalidReturnType = errors.New("invalid return type")
|
|
ErrInvalidParameterCount = errors.New("invalid parameter count")
|
|
ErrInvalidParameterName = errors.New("invalid parameter name")
|
|
ErrInvalidParameterType = errors.New("invalid parameter type")
|
|
ErrSafeMethodMismatch = errors.New("method has wrong safe flag")
|
|
)
|
|
|
|
var checks = map[string][]*Standard{
|
|
manifest.NEP11StandardName: {Nep11NonDivisible, Nep11Divisible},
|
|
manifest.NEP17StandardName: {Nep17},
|
|
manifest.NEP11Payable: {Nep11Payable},
|
|
manifest.NEP17Payable: {Nep17Payable},
|
|
manifest.NEP24StandardName: {Nep24},
|
|
manifest.NEP24Payable: {Nep24Payable},
|
|
}
|
|
|
|
// Check checks if the manifest complies with all provided standards.
|
|
// Currently, only NEP-17 is supported.
|
|
func Check(m *manifest.Manifest, standards ...string) error {
|
|
return check(m, true, standards...)
|
|
}
|
|
|
|
// CheckABI is similar to Check but doesn't check parameter names.
|
|
func CheckABI(m *manifest.Manifest, standards ...string) error {
|
|
return check(m, false, standards...)
|
|
}
|
|
|
|
func check(m *manifest.Manifest, checkNames bool, standards ...string) error {
|
|
for i := range standards {
|
|
ss, ok := checks[standards[i]]
|
|
if ok {
|
|
var err error
|
|
for i := range ss {
|
|
if err = comply(m, checkNames, ss[i]); err == nil {
|
|
break
|
|
}
|
|
}
|
|
if err != nil {
|
|
return fmt.Errorf("manifest is not compliant with '%s': %w", standards[i], err)
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Comply if m has all methods and event from st manifest and they have the same signature.
|
|
// Parameter names are checked to exactly match the ones in the given standard.
|
|
func Comply(m *manifest.Manifest, st *Standard) error {
|
|
return comply(m, true, st)
|
|
}
|
|
|
|
// ComplyABI is similar to Comply but doesn't check parameter names.
|
|
func ComplyABI(m *manifest.Manifest, st *Standard) error {
|
|
return comply(m, false, st)
|
|
}
|
|
|
|
func comply(m *manifest.Manifest, checkNames bool, st *Standard) error {
|
|
if len(st.Required) > 0 {
|
|
if err := check(m, checkNames, st.Required...); err != nil {
|
|
return fmt.Errorf("required standard '%s' is not supported: %w", st.Name, err)
|
|
}
|
|
}
|
|
if st.Base != nil {
|
|
if err := comply(m, checkNames, st.Base); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
for _, stm := range st.ABI.Methods {
|
|
if err := checkMethod(m, &stm, false, checkNames); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
for _, ste := range st.ABI.Events {
|
|
name := ste.Name
|
|
ed := m.ABI.GetEvent(name)
|
|
if ed == nil {
|
|
return fmt.Errorf("%w: event '%s'", ErrEventMissing, name)
|
|
} else if len(ste.Parameters) != len(ed.Parameters) {
|
|
return fmt.Errorf("%w: event '%s' (expected %d, got %d)", ErrInvalidParameterCount,
|
|
name, len(ste.Parameters), len(ed.Parameters))
|
|
}
|
|
for i := range ste.Parameters {
|
|
if checkNames && ste.Parameters[i].Name != ed.Parameters[i].Name {
|
|
return fmt.Errorf("%w: event '%s'[%d] (expected %s, got %s)", ErrInvalidParameterName,
|
|
name, i, ste.Parameters[i].Name, ed.Parameters[i].Name)
|
|
}
|
|
if ste.Parameters[i].Type != ed.Parameters[i].Type {
|
|
return fmt.Errorf("%w: event '%s' (expected %s, got %s)", ErrInvalidParameterType,
|
|
name, ste.Parameters[i].Type, ed.Parameters[i].Type)
|
|
}
|
|
}
|
|
}
|
|
for _, stm := range st.Optional {
|
|
if err := checkMethod(m, &stm, true, checkNames); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func checkMethod(m *manifest.Manifest, expected *manifest.Method,
|
|
allowMissing bool, checkNames bool) error {
|
|
actual := m.ABI.GetMethod(expected.Name, len(expected.Parameters))
|
|
if actual == nil {
|
|
if allowMissing {
|
|
return nil
|
|
}
|
|
return fmt.Errorf("%w: '%s' with %d parameters", ErrMethodMissing,
|
|
expected.Name, len(expected.Parameters))
|
|
}
|
|
if expected.ReturnType != actual.ReturnType {
|
|
return fmt.Errorf("%w: '%s' (expected %s, got %s)", ErrInvalidReturnType,
|
|
expected.Name, expected.ReturnType, actual.ReturnType)
|
|
}
|
|
for i := range expected.Parameters {
|
|
if checkNames && expected.Parameters[i].Name != actual.Parameters[i].Name {
|
|
return fmt.Errorf("%w: '%s'[%d] (expected %s, got %s)", ErrInvalidParameterName,
|
|
expected.Name, i, expected.Parameters[i].Name, actual.Parameters[i].Name)
|
|
}
|
|
if expected.Parameters[i].Type != actual.Parameters[i].Type {
|
|
return fmt.Errorf("%w: '%s'[%d] (expected %s, got %s)", ErrInvalidParameterType,
|
|
expected.Name, i, expected.Parameters[i].Type, actual.Parameters[i].Type)
|
|
}
|
|
}
|
|
if expected.Safe != actual.Safe {
|
|
return fmt.Errorf("'%s' %w: expected %t", expected.Name, ErrSafeMethodMismatch, expected.Safe)
|
|
}
|
|
return nil
|
|
}
|