Skip to content

Commit e9640ef

Browse files
authored
test(gno.land): add unit tests for sdk/vm.vmHandler (#2459)
```console gnome$ go test -v ./sdk/vm -run TestVmHandler === RUN TestVmHandlerQuery_Eval === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.Echo("hello") === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.PubString === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.ConstString === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.pvString === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.counter === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.GetCounter() === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.Inc() === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.pvEcho("hello") === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.1337 === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.13.37 === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.float64(1337) === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.myStructInst === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.myStructInst.Foo() === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.myStruct === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.Inc === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.fn()("hi") === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.sl === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.sl[1] === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.println(1234) 1234 === RUN TestVmHandlerQuery_Eval/gno.land/r/hello === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.doesnotexist === RUN TestVmHandlerQuery_Eval/gno.land/r/doesnotexist.Foo === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.Panic() === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.panic("bar") === RUN TestVmHandlerQuery_Eval/gno.land/r/hello.sl[6] --- PASS: TestVmHandlerQuery_Eval (0.03s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.Echo("hello") (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.PubString (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.ConstString (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.pvString (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.counter (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.GetCounter() (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.Inc() (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.pvEcho("hello") (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.1337 (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.13.37 (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.float64(1337) (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.myStructInst (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.myStructInst.Foo() (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.myStruct (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.Inc (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.fn()("hi") (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.sl (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.sl[1] (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.println(1234) (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.doesnotexist (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/doesnotexist.Foo (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.Panic() (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.panic("bar") (0.00s) --- PASS: TestVmHandlerQuery_Eval/gno.land/r/hello.sl[6] (0.00s) === RUN TestVmHandlerQuery_Funcs === RUN TestVmHandlerQuery_Funcs/gno.land/r/hello === RUN TestVmHandlerQuery_Funcs/gno.land/r/doesnotexist === RUN TestVmHandlerQuery_Funcs/std === RUN TestVmHandlerQuery_Funcs/strings --- PASS: TestVmHandlerQuery_Funcs (0.00s) --- PASS: TestVmHandlerQuery_Funcs/gno.land/r/hello (0.00s) --- PASS: TestVmHandlerQuery_Funcs/gno.land/r/doesnotexist (0.00s) --- PASS: TestVmHandlerQuery_Funcs/std (0.00s) --- PASS: TestVmHandlerQuery_Funcs/strings (0.00s) === RUN TestVmHandlerQuery_File === RUN TestVmHandlerQuery_File/gno.land/r/hello/hello.gno === RUN TestVmHandlerQuery_File/gno.land/r/hello/README.md === RUN TestVmHandlerQuery_File/gno.land/r/hello/doesnotexist.gno === RUN TestVmHandlerQuery_File/gno.land/r/hello === RUN TestVmHandlerQuery_File/gno.land/r/doesnotexist === RUN TestVmHandlerQuery_File/gno.land/r/doesnotexist/hello.gno --- PASS: TestVmHandlerQuery_File (0.00s) --- PASS: TestVmHandlerQuery_File/gno.land/r/hello/hello.gno (0.00s) --- PASS: TestVmHandlerQuery_File/gno.land/r/hello/README.md (0.00s) --- PASS: TestVmHandlerQuery_File/gno.land/r/hello/doesnotexist.gno (0.00s) --- PASS: TestVmHandlerQuery_File/gno.land/r/hello (0.00s) --- PASS: TestVmHandlerQuery_File/gno.land/r/doesnotexist (0.00s) --- PASS: TestVmHandlerQuery_File/gno.land/r/doesnotexist/hello.gno (0.00s) PASS ok github.com/gnolang/gno/gno.land/pkg/sdk/vm (cached) ``` --------- Signed-off-by: moul <[email protected]>
1 parent 494976d commit e9640ef

File tree

3 files changed

+278
-3
lines changed

3 files changed

+278
-3
lines changed

gno.land/pkg/sdk/vm/common_test.go

+3-1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ type testEnv struct {
2323
vmk *VMKeeper
2424
bank bankm.BankKeeper
2525
acck authm.AccountKeeper
26+
vmh vmHandler
2627
}
2728

2829
func setupTestEnv() testEnv {
@@ -62,6 +63,7 @@ func _setupTestEnv(cacheStdlibs bool) testEnv {
6263
}
6364
vmk.CommitGnoTransactionStore(stdlibCtx)
6465
mcw.MultiWrite()
66+
vmh := NewHandler(vmk)
6567

66-
return testEnv{ctx: ctx, vmk: vmk, bank: bank, acck: acck}
68+
return testEnv{ctx: ctx, vmk: vmk, bank: bank, acck: acck, vmh: vmh}
6769
}

gno.land/pkg/sdk/vm/gas_test.go

+1-2
Original file line numberDiff line numberDiff line change
@@ -143,7 +143,6 @@ func setupAddPkg(success bool) (sdk.Context, sdk.Tx, vmHandler) {
143143
ctx := env.ctx
144144
// conduct base gas meter tests from a non-genesis block since genesis block use infinite gas meter instead.
145145
ctx = ctx.WithBlockHeader(&bft.Header{Height: int64(1)})
146-
vmHandler := NewHandler(env.vmk)
147146
// Create an account with 10M ugnot (10gnot)
148147
addr := crypto.AddressFromPreimage([]byte("test1"))
149148
acc := env.acck.NewAccountWithAddress(ctx, addr)
@@ -183,5 +182,5 @@ func Echo() UnknowType {
183182
fee := std.NewFee(500000, std.MustParseCoin(ugnot.ValueString(1)))
184183
tx := std.NewTx(msgs, fee, []std.Signature{}, "")
185184

186-
return ctx, tx, vmHandler
185+
return ctx, tx, env.vmh
187186
}

gno.land/pkg/sdk/vm/handler_test.go

+274
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,13 @@
11
package vm
22

33
import (
4+
"fmt"
45
"testing"
56

7+
"github.com/gnolang/gno/gnovm"
8+
abci "github.com/gnolang/gno/tm2/pkg/bft/abci/types"
9+
"github.com/gnolang/gno/tm2/pkg/crypto"
10+
"github.com/gnolang/gno/tm2/pkg/std"
611
"github.com/stretchr/testify/assert"
712
)
813

@@ -48,3 +53,272 @@ func Test_parseQueryEval_panic(t *testing.T) {
4853
parseQueryEvalData("gno.land/r/demo/users")
4954
})
5055
}
56+
57+
func TestVmHandlerQuery_Eval(t *testing.T) {
58+
tt := []struct {
59+
input []byte
60+
expectedResult string
61+
expectedResultMatch string
62+
expectedErrorMatch string
63+
expectedPanicMatch string
64+
// XXX: expectedEvents
65+
}{
66+
// valid queries
67+
{input: []byte(`gno.land/r/hello.Echo("hello")`), expectedResult: `("echo:hello" string)`},
68+
{input: []byte(`gno.land/r/hello.caller()`), expectedResult: `("" std.Address)`}, // FIXME?
69+
{input: []byte(`gno.land/r/hello.GetHeight()`), expectedResult: `(0 int64)`},
70+
// {input: []byte(`gno.land/r/hello.time.RFC3339`), expectedResult: `test`}, // not working, but should we care?
71+
{input: []byte(`gno.land/r/hello.PubString`), expectedResult: `("public string" string)`},
72+
{input: []byte(`gno.land/r/hello.ConstString`), expectedResult: `("const string" string)`},
73+
{input: []byte(`gno.land/r/hello.pvString`), expectedResult: `("private string" string)`},
74+
{input: []byte(`gno.land/r/hello.counter`), expectedResult: `(42 int)`},
75+
{input: []byte(`gno.land/r/hello.GetCounter()`), expectedResult: `(42 int)`},
76+
{input: []byte(`gno.land/r/hello.Inc()`), expectedResult: `(43 int)`},
77+
{input: []byte(`gno.land/r/hello.pvEcho("hello")`), expectedResult: `("pvecho:hello" string)`},
78+
{input: []byte(`gno.land/r/hello.1337`), expectedResult: `(1337 int)`},
79+
{input: []byte(`gno.land/r/hello.13.37`), expectedResult: `(13.37 float64)`},
80+
{input: []byte(`gno.land/r/hello.float64(1337)`), expectedResult: `(1337 float64)`},
81+
{input: []byte(`gno.land/r/hello.myStructInst`), expectedResult: `(struct{(1000 int)} gno.land/r/hello.myStruct)`},
82+
{input: []byte(`gno.land/r/hello.myStructInst.Foo()`), expectedResult: `("myStruct.Foo" string)`},
83+
{input: []byte(`gno.land/r/hello.myStruct`), expectedResultMatch: `\(typeval{gno.land/r/hello.myStruct \(0x.*\)} type{}\)`},
84+
{input: []byte(`gno.land/r/hello.Inc`), expectedResult: `(Inc func()( int))`},
85+
{input: []byte(`gno.land/r/hello.fn()("hi")`), expectedResult: `("echo:hi" string)`},
86+
{input: []byte(`gno.land/r/hello.sl`), expectedResultMatch: `(slice[ref(.*)] []int)`}, // XXX: should return the actual value
87+
{input: []byte(`gno.land/r/hello.sl[1]`), expectedResultMatch: `(slice[ref(.*)] []int)`}, // XXX: should return the actual value
88+
{input: []byte(`gno.land/r/hello.println(1234)`), expectedResultMatch: `^$`}, // XXX: compare stdout?
89+
90+
// panics
91+
{input: []byte(`gno.land/r/hello`), expectedPanicMatch: `expected <pkgpath>.<expression> syntax in query input data`},
92+
93+
// errors
94+
{input: []byte(`gno.land/r/hello.doesnotexist`), expectedErrorMatch: `^/:0:0: name doesnotexist not declared:`}, // multiline error
95+
{input: []byte(`gno.land/r/doesnotexist.Foo`), expectedErrorMatch: `^invalid package path$`},
96+
{input: []byte(`gno.land/r/hello.Panic()`), expectedErrorMatch: `^foo$`},
97+
{input: []byte(`gno.land/r/hello.sl[6]`), expectedErrorMatch: `^slice index out of bounds: 6 \(len=5\)$`},
98+
}
99+
100+
for _, tc := range tt {
101+
name := string(tc.input)
102+
t.Run(name, func(t *testing.T) {
103+
env := setupTestEnv()
104+
ctx := env.vmk.MakeGnoTransactionStore(env.ctx)
105+
vmHandler := env.vmh
106+
107+
// Give "addr1" some gnots.
108+
addr := crypto.AddressFromPreimage([]byte("addr1"))
109+
acc := env.acck.NewAccountWithAddress(ctx, addr)
110+
env.acck.SetAccount(ctx, acc)
111+
env.bank.SetCoins(ctx, addr, std.MustParseCoins("10000000ugnot"))
112+
assert.True(t, env.bank.GetCoins(ctx, addr).IsEqual(std.MustParseCoins("10000000ugnot")))
113+
114+
// Create test package.
115+
files := []*gnovm.MemFile{
116+
{"hello.gno", `
117+
package hello
118+
119+
import "std"
120+
import "time"
121+
122+
var _ = time.RFC3339
123+
func caller() std.Address { return std.GetOrigCaller() }
124+
var GetHeight = std.GetHeight
125+
var sl = []int{1,2,3,4,5}
126+
func fn() func(string) string { return Echo }
127+
type myStruct struct{a int}
128+
var myStructInst = myStruct{a: 1000}
129+
func (ms myStruct) Foo() string { return "myStruct.Foo" }
130+
func Panic() { panic("foo") }
131+
var counter int = 42
132+
var pvString = "private string"
133+
var PubString = "public string"
134+
const ConstString = "const string"
135+
func Echo(msg string) string { return "echo:"+msg }
136+
func GetCounter() int { return counter }
137+
func Inc() int { counter += 1; return counter }
138+
func pvEcho(msg string) string { return "pvecho:"+msg }
139+
`},
140+
}
141+
pkgPath := "gno.land/r/hello"
142+
msg1 := NewMsgAddPackage(addr, pkgPath, files)
143+
err := env.vmk.AddPackage(ctx, msg1)
144+
assert.NoError(t, err)
145+
env.vmk.CommitGnoTransactionStore(ctx)
146+
147+
req := abci.RequestQuery{
148+
Path: "vm/qeval",
149+
Data: tc.input,
150+
}
151+
152+
defer func() {
153+
if r := recover(); r != nil {
154+
output := fmt.Sprintf("%v", r)
155+
assert.Regexp(t, tc.expectedPanicMatch, output)
156+
} else {
157+
assert.Equal(t, tc.expectedPanicMatch, "", "should not panic")
158+
}
159+
}()
160+
res := vmHandler.Query(env.ctx, req)
161+
if tc.expectedPanicMatch == "" {
162+
if tc.expectedErrorMatch == "" {
163+
assert.True(t, res.IsOK(), "should not have error")
164+
if tc.expectedResult != "" {
165+
assert.Equal(t, string(res.Data), tc.expectedResult)
166+
}
167+
if tc.expectedResultMatch != "" {
168+
assert.Regexp(t, tc.expectedResultMatch, string(res.Data))
169+
}
170+
} else {
171+
assert.False(t, res.IsOK(), "should have an error")
172+
errmsg := res.Error.Error()
173+
assert.Regexp(t, tc.expectedErrorMatch, errmsg)
174+
}
175+
}
176+
})
177+
}
178+
}
179+
180+
func TestVmHandlerQuery_Funcs(t *testing.T) {
181+
tt := []struct {
182+
input []byte
183+
expectedResult string
184+
expectedErrorMatch string
185+
}{
186+
// valid queries
187+
{input: []byte(`gno.land/r/hello`), expectedResult: `[{"FuncName":"Panic","Params":null,"Results":null},{"FuncName":"Echo","Params":[{"Name":"msg","Type":"string","Value":""}],"Results":[{"Name":"_","Type":"string","Value":""}]},{"FuncName":"GetCounter","Params":null,"Results":[{"Name":"_","Type":"int","Value":""}]},{"FuncName":"Inc","Params":null,"Results":[{"Name":"_","Type":"int","Value":""}]}]`},
188+
{input: []byte(`gno.land/r/doesnotexist`), expectedErrorMatch: `invalid package path`},
189+
{input: []byte(`std`), expectedErrorMatch: `invalid package path`},
190+
{input: []byte(`strings`), expectedErrorMatch: `invalid package path`},
191+
}
192+
193+
for _, tc := range tt {
194+
name := string(tc.input)
195+
t.Run(name, func(t *testing.T) {
196+
env := setupTestEnv()
197+
ctx := env.vmk.MakeGnoTransactionStore(env.ctx)
198+
vmHandler := env.vmh
199+
200+
// Give "addr1" some gnots.
201+
addr := crypto.AddressFromPreimage([]byte("addr1"))
202+
acc := env.acck.NewAccountWithAddress(ctx, addr)
203+
env.acck.SetAccount(ctx, acc)
204+
env.bank.SetCoins(ctx, addr, std.MustParseCoins("10000000ugnot"))
205+
assert.True(t, env.bank.GetCoins(ctx, addr).IsEqual(std.MustParseCoins("10000000ugnot")))
206+
207+
// Create test package.
208+
files := []*gnovm.MemFile{
209+
{"hello.gno", `
210+
package hello
211+
212+
var sl = []int{1,2,3,4,5}
213+
func fn() func(string) string { return Echo }
214+
type myStruct struct{a int}
215+
var myStructInst = myStruct{a: 1000}
216+
func (ms myStruct) Foo() string { return "myStruct.Foo" }
217+
func Panic() { panic("foo") }
218+
var counter int = 42
219+
var pvString = "private string"
220+
var PubString = "public string"
221+
const ConstString = "const string"
222+
func Echo(msg string) string { return "echo:"+msg }
223+
func GetCounter() int { return counter }
224+
func Inc() int { counter += 1; return counter }
225+
func pvEcho(msg string) string { return "pvecho:"+msg }
226+
`},
227+
}
228+
pkgPath := "gno.land/r/hello"
229+
msg1 := NewMsgAddPackage(addr, pkgPath, files)
230+
err := env.vmk.AddPackage(ctx, msg1)
231+
assert.NoError(t, err)
232+
233+
req := abci.RequestQuery{
234+
Path: "vm/qfuncs",
235+
Data: tc.input,
236+
}
237+
238+
res := vmHandler.Query(env.ctx, req)
239+
if tc.expectedErrorMatch == "" {
240+
assert.True(t, res.IsOK(), "should not have error")
241+
if tc.expectedResult != "" {
242+
assert.Equal(t, string(res.Data), tc.expectedResult)
243+
}
244+
} else {
245+
assert.False(t, res.IsOK(), "should have an error")
246+
errmsg := res.Error.Error()
247+
assert.Regexp(t, tc.expectedErrorMatch, errmsg)
248+
}
249+
})
250+
}
251+
}
252+
253+
func TestVmHandlerQuery_File(t *testing.T) {
254+
tt := []struct {
255+
input []byte
256+
expectedResult string
257+
expectedResultMatch string
258+
expectedErrorMatch string
259+
expectedPanicMatch string
260+
// XXX: expectedEvents
261+
}{
262+
// valid queries
263+
{input: []byte(`gno.land/r/hello/hello.gno`), expectedResult: "package hello\n\nfunc Hello() string { return \"hello\" }\n"},
264+
{input: []byte(`gno.land/r/hello/README.md`), expectedResult: "# Hello"},
265+
{input: []byte(`gno.land/r/hello/doesnotexist.gno`), expectedErrorMatch: `file "gno.land/r/hello/doesnotexist.gno" is not available`},
266+
{input: []byte(`gno.land/r/hello`), expectedResult: "README.md\nhello.gno"},
267+
{input: []byte(`gno.land/r/doesnotexist`), expectedErrorMatch: `package "gno.land/r/doesnotexist" is not available`},
268+
{input: []byte(`gno.land/r/doesnotexist/hello.gno`), expectedErrorMatch: `file "gno.land/r/doesnotexist/hello.gno" is not available`},
269+
}
270+
271+
for _, tc := range tt {
272+
name := string(tc.input)
273+
t.Run(name, func(t *testing.T) {
274+
env := setupTestEnv()
275+
ctx := env.vmk.MakeGnoTransactionStore(env.ctx)
276+
vmHandler := env.vmh
277+
278+
// Give "addr1" some gnots.
279+
addr := crypto.AddressFromPreimage([]byte("addr1"))
280+
acc := env.acck.NewAccountWithAddress(ctx, addr)
281+
env.acck.SetAccount(ctx, acc)
282+
env.bank.SetCoins(ctx, addr, std.MustParseCoins("10000000ugnot"))
283+
assert.True(t, env.bank.GetCoins(ctx, addr).IsEqual(std.MustParseCoins("10000000ugnot")))
284+
285+
// Create test package.
286+
files := []*gnovm.MemFile{
287+
{"README.md", "# Hello"},
288+
{"hello.gno", "package hello\n\nfunc Hello() string { return \"hello\" }\n"},
289+
}
290+
pkgPath := "gno.land/r/hello"
291+
msg1 := NewMsgAddPackage(addr, pkgPath, files)
292+
err := env.vmk.AddPackage(ctx, msg1)
293+
assert.NoError(t, err)
294+
295+
req := abci.RequestQuery{
296+
Path: "vm/qfile",
297+
Data: tc.input,
298+
}
299+
300+
defer func() {
301+
if r := recover(); r != nil {
302+
output := fmt.Sprintf("%v", r)
303+
assert.Regexp(t, tc.expectedPanicMatch, output)
304+
} else {
305+
assert.Equal(t, "", tc.expectedPanicMatch, "should not panic")
306+
}
307+
}()
308+
res := vmHandler.Query(env.ctx, req)
309+
if tc.expectedErrorMatch == "" {
310+
assert.True(t, res.IsOK(), "should not have error")
311+
if tc.expectedResult != "" {
312+
assert.Equal(t, string(res.Data), tc.expectedResult)
313+
}
314+
if tc.expectedResultMatch != "" {
315+
assert.Regexp(t, tc.expectedResultMatch, string(res.Data))
316+
}
317+
} else {
318+
assert.False(t, res.IsOK(), "should have an error")
319+
errmsg := res.Error.Error()
320+
assert.Regexp(t, tc.expectedErrorMatch, errmsg)
321+
}
322+
})
323+
}
324+
}

0 commit comments

Comments
 (0)