mirror of
https://github.com/MontFerret/ferret.git
synced 2025-07-15 01:25:00 +02:00
Improved path lookup
This commit is contained in:
@ -120,6 +120,66 @@ func TestMember(t *testing.T) {
|
|||||||
So(string(out), ShouldEqual, `"wsx"`)
|
So(string(out), ShouldEqual, `"wsx"`)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
Convey("Deep path", func() {
|
||||||
|
c := compiler.New()
|
||||||
|
|
||||||
|
p, err := c.Compile(`
|
||||||
|
LET obj = {
|
||||||
|
first: {
|
||||||
|
second: {
|
||||||
|
third: {
|
||||||
|
fourth: {
|
||||||
|
fifth: {
|
||||||
|
bottom: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
RETURN obj.first.second.third.fourth.fifth.bottom
|
||||||
|
`)
|
||||||
|
|
||||||
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
|
out, err := p.Run(context.Background())
|
||||||
|
|
||||||
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
|
So(string(out), ShouldEqual, `true`)
|
||||||
|
})
|
||||||
|
|
||||||
|
Convey("Deep computed path", func() {
|
||||||
|
c := compiler.New()
|
||||||
|
|
||||||
|
p, err := c.Compile(`
|
||||||
|
LET obj = {
|
||||||
|
first: {
|
||||||
|
second: {
|
||||||
|
third: {
|
||||||
|
fourth: {
|
||||||
|
fifth: {
|
||||||
|
bottom: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
RETURN obj["first"]["second"]["third"]["fourth"]["fifth"].bottom
|
||||||
|
`)
|
||||||
|
|
||||||
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
|
out, err := p.Run(context.Background())
|
||||||
|
|
||||||
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
|
So(string(out), ShouldEqual, `true`)
|
||||||
|
})
|
||||||
|
|
||||||
Convey("Prop after a func call", func() {
|
Convey("Prop after a func call", func() {
|
||||||
c := compiler.New()
|
c := compiler.New()
|
||||||
|
|
||||||
@ -160,9 +220,9 @@ func TestMember(t *testing.T) {
|
|||||||
|
|
||||||
func BenchmarkMemberArray(b *testing.B) {
|
func BenchmarkMemberArray(b *testing.B) {
|
||||||
p := compiler.New().MustCompile(`
|
p := compiler.New().MustCompile(`
|
||||||
LET arr = [1]
|
LET arr = [[[[1]]]]
|
||||||
|
|
||||||
RETURN arr[0]
|
RETURN arr[0][0][0][0]
|
||||||
`)
|
`)
|
||||||
|
|
||||||
for n := 0; n < b.N; n++ {
|
for n := 0; n < b.N; n++ {
|
||||||
@ -172,9 +232,21 @@ func BenchmarkMemberArray(b *testing.B) {
|
|||||||
|
|
||||||
func BenchmarkMemberObject(b *testing.B) {
|
func BenchmarkMemberObject(b *testing.B) {
|
||||||
p := compiler.New().MustCompile(`
|
p := compiler.New().MustCompile(`
|
||||||
LET obj = { "foo": "bar"}
|
LET obj = {
|
||||||
|
first: {
|
||||||
|
second: {
|
||||||
|
third: {
|
||||||
|
fourth: {
|
||||||
|
fifth: {
|
||||||
|
bottom: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
RETURN obj.foo
|
RETURN obj.first.second.third.fourth.fifth.bottom
|
||||||
`)
|
`)
|
||||||
|
|
||||||
for n := 0; n < b.N; n++ {
|
for n := 0; n < b.N; n++ {
|
||||||
|
@ -80,4 +80,19 @@ func TestParam(t *testing.T) {
|
|||||||
So(string(out), ShouldEqual, `"foobar"`)
|
So(string(out), ShouldEqual, `"foobar"`)
|
||||||
|
|
||||||
})
|
})
|
||||||
|
|
||||||
|
Convey("Should be possible to use in member expression as a computed property", t, func() {
|
||||||
|
prog := compiler.New().
|
||||||
|
MustCompile(`
|
||||||
|
LET obj = { foo: "bar" }
|
||||||
|
RETURN obj[@param]
|
||||||
|
`)
|
||||||
|
|
||||||
|
out := prog.MustRun(
|
||||||
|
context.Background(),
|
||||||
|
runtime.WithParam("param", "foo"),
|
||||||
|
)
|
||||||
|
|
||||||
|
So(string(out), ShouldEqual, `"bar"`)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
@ -35,22 +35,24 @@ func (e *MemberExpression) Exec(ctx context.Context, scope *core.Scope) (core.Va
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
strPath := make([]core.Value, len(e.path))
|
out := val
|
||||||
|
path := make([]core.Value, 1, 1)
|
||||||
|
|
||||||
for idx, exp := range e.path {
|
for _, exp := range e.path {
|
||||||
segment, err := exp.Exec(ctx, scope)
|
segment, err := exp.Exec(ctx, scope)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return values.None, err
|
return values.None, err
|
||||||
}
|
}
|
||||||
|
|
||||||
strPath[idx] = segment
|
path[0] = segment
|
||||||
}
|
c, err := values.GetIn(ctx, out, path)
|
||||||
|
|
||||||
out, err := values.GetIn(ctx, val, strPath)
|
if err != nil {
|
||||||
|
return values.None, core.SourceError(e.src, err)
|
||||||
|
}
|
||||||
|
|
||||||
if err != nil {
|
out = c
|
||||||
return values.None, core.SourceError(e.src, err)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return out, nil
|
return out, nil
|
||||||
|
Reference in New Issue
Block a user