1
0
mirror of https://github.com/MontFerret/ferret.git synced 2025-07-01 00:45:13 +02:00
Files
ferret/pkg/runtime/expressions/member.go

91 lines
1.9 KiB
Go
Raw Normal View History

2018-09-18 16:42:38 -04:00
package expressions
import (
"context"
"github.com/pkg/errors"
2018-09-18 16:42:38 -04:00
"github.com/MontFerret/ferret/pkg/runtime/core"
"github.com/MontFerret/ferret/pkg/runtime/values"
)
type MemberExpression struct {
src core.SourceMap
source core.Expression
path []*MemberPathSegment
preCompiledPath []core.Value
2018-09-18 16:42:38 -04:00
}
func NewMemberExpression(src core.SourceMap, source core.Expression, path []*MemberPathSegment, preCompiledPath []core.Value) (*MemberExpression, error) {
if source == nil {
return nil, core.Error(core.ErrMissedArgument, "source")
2018-09-18 16:42:38 -04:00
}
if len(path) == 0 {
2018-10-28 01:45:26 -04:00
return nil, core.Error(core.ErrMissedArgument, "path expressions")
2018-09-18 16:42:38 -04:00
}
return &MemberExpression{src, source, path, preCompiledPath}, nil
2018-09-18 16:42:38 -04:00
}
func (e *MemberExpression) Exec(ctx context.Context, scope *core.Scope) (core.Value, error) {
member, err := e.source.Exec(ctx, scope)
2018-09-18 16:42:38 -04:00
if err != nil {
if e.path[0].optional {
return values.None, nil
}
2018-09-18 16:42:38 -04:00
return values.None, core.SourceError(
e.src,
err,
)
}
var segments = e.preCompiledPath
2018-09-18 16:42:38 -04:00
if e.preCompiledPath == nil {
segments = make([]core.Value, len(e.path))
2018-09-18 16:42:38 -04:00
// unfold the path
for i, seg := range e.path {
segment, err := seg.exp.Exec(ctx, scope)
if err != nil {
return values.None, err
}
segments[i] = segment
2018-09-18 16:42:38 -04:00
}
}
2018-09-18 16:42:38 -04:00
var pathErr core.PathError
var out core.Value = values.None
2018-09-18 16:42:38 -04:00
getter, ok := member.(core.Getter)
if ok {
out, pathErr = getter.GetIn(ctx, segments)
} else {
out, pathErr = values.GetIn(ctx, member, segments)
}
if pathErr != nil {
segmentIdx := pathErr.Segment()
// if invalid index is returned, we ignore the optionality check
// and return the pathErr
if segmentIdx >= len(e.path) {
return values.None, errors.New(pathErr.Format(segments))
}
segment := e.path[segmentIdx]
if !segment.optional {
return values.None, errors.New(pathErr.Format(segments))
2019-09-07 01:47:58 -04:00
}
2018-09-18 16:42:38 -04:00
return values.None, nil
2018-09-18 16:42:38 -04:00
}
return out, nil
}