1
0
mirror of https://github.com/MontFerret/ferret.git synced 2025-01-18 03:22:02 +02:00
ferret/pkg/runtime/expressions/for.go
2019-07-13 13:39:01 -04:00

186 lines
3.4 KiB
Go

package expressions
import (
"context"
"github.com/MontFerret/ferret/pkg/runtime/collections"
"github.com/MontFerret/ferret/pkg/runtime/core"
"github.com/MontFerret/ferret/pkg/runtime/expressions/clauses"
"github.com/MontFerret/ferret/pkg/runtime/values"
)
type ForExpression struct {
src core.SourceMap
dataSource collections.Iterable
predicate core.Expression
distinct bool
spread bool
}
func NewForExpression(
src core.SourceMap,
dataSource collections.Iterable,
predicate core.Expression,
distinct,
spread bool,
) (*ForExpression, error) {
if dataSource == nil {
return nil, core.Error(core.ErrMissedArgument, "missed source expression")
}
if predicate == nil {
return nil, core.Error(core.ErrMissedArgument, "missed return expression")
}
return &ForExpression{
src,
dataSource,
predicate,
distinct,
spread,
}, nil
}
func (e *ForExpression) AddLimit(src core.SourceMap, size, count core.Expression) error {
limit, err := clauses.NewLimitClause(src, e.dataSource, size, count)
if err != nil {
return err
}
e.dataSource = limit
return nil
}
func (e *ForExpression) AddFilter(src core.SourceMap, exp core.Expression) error {
filter, err := clauses.NewFilterClause(src, e.dataSource, exp)
if err != nil {
return err
}
e.dataSource = filter
return nil
}
func (e *ForExpression) AddSort(src core.SourceMap, sorters ...*clauses.SorterExpression) error {
sort, err := clauses.NewSortClause(src, e.dataSource, sorters...)
if err != nil {
return err
}
e.dataSource = sort
return nil
}
func (e *ForExpression) AddCollect(src core.SourceMap, params *clauses.Collect) error {
collect, err := clauses.NewCollectClause(src, e.dataSource, params)
if err != nil {
return err
}
e.dataSource = collect
return nil
}
func (e *ForExpression) AddStatement(stmt core.Expression) error {
tap, ok := e.dataSource.(*BlockExpression)
if !ok {
t, err := NewBlockExpression(e.dataSource)
if err != nil {
return err
}
tap = t
e.dataSource = tap
}
tap.Add(stmt)
return nil
}
func (e *ForExpression) Exec(ctx context.Context, scope *core.Scope) (core.Value, error) {
select {
case <-ctx.Done():
return values.None, core.ErrTerminated
default:
iterator, err := e.dataSource.Iterate(ctx, scope)
if err != nil {
return values.None, err
}
// Hash map for a check for uniqueness
var hashTable map[uint64]bool
if e.distinct {
hashTable = make(map[uint64]bool)
}
res := values.NewArray(10)
for {
nextScope, err := iterator.Next(ctx, scope)
if err != nil {
if core.IsNoMoreData(err) {
break
}
return values.None, core.SourceError(e.src, err)
}
out, err := e.predicate.Exec(ctx, nextScope)
if err != nil {
return values.None, err
}
var add bool
// The result shouldn't be distinct
// Just add the output
if !e.distinct {
add = true
} else {
// We need to check whether the value already exists in the result set
hash := out.Hash()
_, exists := hashTable[hash]
if !exists {
hashTable[hash] = true
add = true
}
}
if add {
if !e.spread {
res.Push(out)
} else {
elements, ok := out.(*values.Array)
if !ok {
return values.None, core.Error(core.ErrInvalidOperation, "spread of non-array value")
}
elements.ForEach(func(i core.Value, _ int) bool {
res.Push(i)
return true
})
}
}
}
return res, nil
}
}