Skip to content

Commit

Permalink
update vm to raise wrapped runtime errors (#247)
Browse files Browse the repository at this point in the history
* update vm to raise wrapped runtime errors

special errors returned from custom builtin modules can be identified easily by wrapping and using errors package's Is() and As() method.

* added test for runtime errors
  • Loading branch information
Ozan HACIBEKİROĞLU authored Feb 16, 2020
1 parent e01d7f4 commit 0854675
Show file tree
Hide file tree
Showing 2 changed files with 146 additions and 3 deletions.
6 changes: 3 additions & 3 deletions vm.go
Original file line number Diff line number Diff line change
Expand Up @@ -80,14 +80,14 @@ func (v *VM) Run() (err error) {
if err != nil {
filePos := v.fileSet.Position(
v.curFrame.fn.SourcePos(v.ip - 1))
err = fmt.Errorf("Runtime Error: %s\n\tat %s",
err.Error(), filePos)
err = fmt.Errorf("Runtime Error: %w\n\tat %s",
err, filePos)
for v.framesIndex > 1 {
v.framesIndex--
v.curFrame = &v.frames[v.framesIndex-1]
filePos = v.fileSet.Position(
v.curFrame.fn.SourcePos(v.curFrame.ip - 1))
err = fmt.Errorf("%s\n\tat %s", err.Error(), filePos)
err = fmt.Errorf("%w\n\tat %s", err, filePos)
}
return err
}
Expand Down
143 changes: 143 additions & 0 deletions vm_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,19 @@ func (o *testopts) Skip2ndPass() *testopts {
return c
}

type customError struct {
err error
str string
}

func (c *customError) Error() string {
return c.str
}

func (c *customError) Unwrap() error {
return c.err
}

func TestArray(t *testing.T) {
expectRun(t, `out = [1, 2 * 2, 3 + 3]`, nil, ARR{1, 4, 6})

Expand Down Expand Up @@ -912,6 +925,82 @@ export func() {
}`), "Runtime Error: invalid operation: int + string\n\tat mod2:4:9")
}

func TestVMErrorUnwrap(t *testing.T) {
userErr := errors.New("user runtime error")
userFunc := func(err error) *tengo.UserFunction {
return &tengo.UserFunction{Name: "user_func", Value: func(args ...tengo.Object) (tengo.Object, error) {
return nil, err
}}
}
userModule := func(err error) *tengo.BuiltinModule {
return &tengo.BuiltinModule{
Attrs: map[string]tengo.Object{
"afunction": &tengo.UserFunction{
Name: "afunction",
Value: func(a ...tengo.Object) (tengo.Object, error) {
return nil, err
},
},
},
}
}

expectError(t, `user_func()`,
Opts().Symbol("user_func", userFunc(userErr)),
"Runtime Error: "+userErr.Error(),
)
expectErrorIs(t, `user_func()`,
Opts().Symbol("user_func", userFunc(userErr)),
userErr,
)

wrapUserErr := &customError{err: userErr, str: "custom error"}

expectErrorIs(t, `user_func()`,
Opts().Symbol("user_func", userFunc(wrapUserErr)),
wrapUserErr,
)
expectErrorIs(t, `user_func()`,
Opts().Symbol("user_func", userFunc(wrapUserErr)),
userErr,
)
var asErr1 *customError
expectErrorAs(t, `user_func()`,
Opts().Symbol("user_func", userFunc(wrapUserErr)),
&asErr1,
)
require.True(t, asErr1.Error() == wrapUserErr.Error(),
"expected error as:%v, got:%v", wrapUserErr, asErr1)

expectError(t, `import("mod1").afunction()`,
Opts().Module("mod1", userModule(userErr)),
"Runtime Error: "+userErr.Error(),
)
expectErrorIs(t, `import("mod1").afunction()`,
Opts().Module("mod1", userModule(userErr)),
userErr,
)
expectError(t, `import("mod1").afunction()`,
Opts().Module("mod1", userModule(wrapUserErr)),
"Runtime Error: "+wrapUserErr.Error(),
)
expectErrorIs(t, `import("mod1").afunction()`,
Opts().Module("mod1", userModule(wrapUserErr)),
wrapUserErr,
)
expectErrorIs(t, `import("mod1").afunction()`,
Opts().Module("mod1", userModule(wrapUserErr)),
userErr,
)
var asErr2 *customError
expectErrorAs(t, `import("mod1").afunction()`,
Opts().Module("mod1", userModule(wrapUserErr)),
&asErr2,
)
require.True(t, asErr2.Error() == wrapUserErr.Error(),
"expected error as:%v, got:%v", wrapUserErr, asErr2)
}

func TestError(t *testing.T) {
expectRun(t, `out = error(1)`, nil, errorObject(1))
expectRun(t, `out = error(1).value`, nil, 1)
Expand Down Expand Up @@ -3266,6 +3355,60 @@ func expectError(
expected, err.Error(), strings.Join(trace, "\n"))
}

func expectErrorIs(
t *testing.T,
input string,
opts *testopts,
expected error,
) {
if opts == nil {
opts = Opts()
}
symbols := opts.symbols
modules := opts.modules
maxAllocs := opts.maxAllocs

// parse
program := parse(t, input)
if program == nil {
return
}

// compiler/VM
_, trace, err := traceCompileRun(program, symbols, modules, maxAllocs)
require.Error(t, err, "\n"+strings.Join(trace, "\n"))
require.True(t, errors.Is(err, expected),
"expected error is: %s, got: %s\n%s",
expected.Error(), err.Error(), strings.Join(trace, "\n"))
}

func expectErrorAs(
t *testing.T,
input string,
opts *testopts,
expected interface{},
) {
if opts == nil {
opts = Opts()
}
symbols := opts.symbols
modules := opts.modules
maxAllocs := opts.maxAllocs

// parse
program := parse(t, input)
if program == nil {
return
}

// compiler/VM
_, trace, err := traceCompileRun(program, symbols, modules, maxAllocs)
require.Error(t, err, "\n"+strings.Join(trace, "\n"))
require.True(t, errors.As(err, expected),
"expected error as: %v, got: %v\n%s",
expected, err, strings.Join(trace, "\n"))
}

type vmTracer struct {
Out []string
}
Expand Down

0 comments on commit 0854675

Please sign in to comment.