2020-08-20 05:34:14 +00:00
|
|
|
package compiler_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"math/big"
|
|
|
|
"testing"
|
2021-11-12 16:30:50 +00:00
|
|
|
|
|
|
|
"github.com/stretchr/testify/require"
|
2020-08-20 05:34:14 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestDefer(t *testing.T) {
|
|
|
|
t.Run("Simple", func(t *testing.T) {
|
|
|
|
src := `package main
|
|
|
|
var a int
|
|
|
|
func Main() int {
|
|
|
|
return h() + a
|
|
|
|
}
|
|
|
|
func h() int {
|
|
|
|
defer f()
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
func f() { a += 2 }`
|
|
|
|
eval(t, src, big.NewInt(3))
|
|
|
|
})
|
|
|
|
t.Run("ValueUnchanged", func(t *testing.T) {
|
|
|
|
src := `package main
|
|
|
|
var a int
|
|
|
|
func Main() int {
|
|
|
|
defer f()
|
|
|
|
a = 3
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
func f() { a += 2 }`
|
|
|
|
eval(t, src, big.NewInt(3))
|
|
|
|
})
|
|
|
|
t.Run("Function", func(t *testing.T) {
|
|
|
|
src := `package main
|
|
|
|
var a int
|
|
|
|
func Main() int {
|
|
|
|
return h() + a
|
|
|
|
}
|
|
|
|
func h() int {
|
|
|
|
defer f()
|
|
|
|
a = 3
|
|
|
|
return g()
|
|
|
|
}
|
|
|
|
func g() int {
|
|
|
|
a++
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
func f() { a += 2 }`
|
|
|
|
eval(t, src, big.NewInt(10))
|
|
|
|
})
|
|
|
|
t.Run("MultipleDefers", func(t *testing.T) {
|
|
|
|
src := `package main
|
|
|
|
var a int
|
|
|
|
func Main() int {
|
|
|
|
return h() + a
|
|
|
|
}
|
|
|
|
func h() int {
|
|
|
|
defer f()
|
|
|
|
defer g()
|
|
|
|
a = 3
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
func g() { a *= 2 }
|
|
|
|
func f() { a += 2 }`
|
|
|
|
eval(t, src, big.NewInt(11))
|
|
|
|
})
|
|
|
|
t.Run("FunctionLiteral", func(t *testing.T) {
|
|
|
|
src := `package main
|
|
|
|
var a int
|
|
|
|
func Main() int {
|
|
|
|
return h() + a
|
|
|
|
}
|
|
|
|
func h() int {
|
|
|
|
defer func() {
|
|
|
|
a = 10
|
|
|
|
}()
|
|
|
|
a = 3
|
|
|
|
return a
|
|
|
|
}`
|
|
|
|
eval(t, src, big.NewInt(13))
|
|
|
|
})
|
2021-11-12 16:30:50 +00:00
|
|
|
t.Run("NoReturnReturn", func(t *testing.T) {
|
|
|
|
src := `package main
|
|
|
|
var i int
|
|
|
|
func Main() {
|
|
|
|
defer func() {
|
|
|
|
i++
|
|
|
|
}()
|
|
|
|
return
|
|
|
|
}`
|
|
|
|
vm := vmAndCompile(t, src)
|
|
|
|
err := vm.Run()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 0, vm.Estack().Len(), "stack contains unexpected items")
|
|
|
|
})
|
|
|
|
t.Run("NoReturnNoReturn", func(t *testing.T) {
|
|
|
|
src := `package main
|
|
|
|
var i int
|
|
|
|
func Main() {
|
|
|
|
defer func() {
|
|
|
|
i++
|
|
|
|
}()
|
|
|
|
}`
|
|
|
|
vm := vmAndCompile(t, src)
|
|
|
|
err := vm.Run()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 0, vm.Estack().Len(), "stack contains unexpected items")
|
|
|
|
})
|
2020-08-20 05:34:14 +00:00
|
|
|
}
|
2020-08-21 12:37:46 +00:00
|
|
|
|
|
|
|
func TestRecover(t *testing.T) {
|
|
|
|
t.Run("Panic", func(t *testing.T) {
|
|
|
|
src := `package foo
|
|
|
|
var a int
|
|
|
|
func Main() int {
|
|
|
|
return h() + a
|
|
|
|
}
|
|
|
|
func h() int {
|
|
|
|
defer func() {
|
|
|
|
if r := recover(); r != nil {
|
|
|
|
a = 3
|
|
|
|
} else {
|
|
|
|
a = 4
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
a = 1
|
|
|
|
panic("msg")
|
|
|
|
return a
|
|
|
|
}`
|
|
|
|
eval(t, src, big.NewInt(3))
|
|
|
|
})
|
|
|
|
t.Run("NoPanic", func(t *testing.T) {
|
|
|
|
src := `package foo
|
|
|
|
var a int
|
|
|
|
func Main() int {
|
|
|
|
return h() + a
|
|
|
|
}
|
|
|
|
func h() int {
|
|
|
|
defer func() {
|
|
|
|
if r := recover(); r != nil {
|
|
|
|
a = 3
|
|
|
|
} else {
|
|
|
|
a = 4
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
a = 1
|
|
|
|
return a
|
|
|
|
}`
|
|
|
|
eval(t, src, big.NewInt(5))
|
|
|
|
})
|
|
|
|
t.Run("PanicInDefer", func(t *testing.T) {
|
|
|
|
src := `package foo
|
|
|
|
var a int
|
|
|
|
func Main() int {
|
|
|
|
return h() + a
|
|
|
|
}
|
|
|
|
func h() int {
|
2020-09-02 12:20:11 +00:00
|
|
|
defer func() { a += 2; recover() }()
|
|
|
|
defer func() { a *= 3; recover(); panic("again") }()
|
2020-08-21 12:37:46 +00:00
|
|
|
a = 1
|
|
|
|
panic("msg")
|
|
|
|
return a
|
|
|
|
}`
|
|
|
|
eval(t, src, big.NewInt(5))
|
|
|
|
})
|
|
|
|
}
|
2021-04-28 13:10:44 +00:00
|
|
|
|
|
|
|
func TestDeferNoGlobals(t *testing.T) {
|
|
|
|
src := `package foo
|
|
|
|
func Main() int {
|
|
|
|
a := 1
|
|
|
|
defer func() { recover() }()
|
|
|
|
panic("msg")
|
|
|
|
return a
|
|
|
|
}`
|
|
|
|
eval(t, src, big.NewInt(0))
|
|
|
|
}
|