1
0
mirror of https://github.com/MontFerret/ferret.git synced 2025-01-18 03:22:02 +02:00
ferret/pkg/runtime/expressions/clauses/limit.go
Tim Voronov 291d07cbef
Feature/custom iterator (#173)
* Added CollectionIterator interface

* Added PAGINATION function

* Fixed LIMIT clause

* Fixed linting issues
2018-11-12 19:58:12 -05:00

84 lines
1.7 KiB
Go

package clauses
import (
"context"
"github.com/MontFerret/ferret/pkg/runtime/collections"
"github.com/MontFerret/ferret/pkg/runtime/core"
)
type LimitClause struct {
src core.SourceMap
dataSource collections.Iterable
count core.Expression
offset core.Expression
}
func NewLimitClause(
src core.SourceMap,
dataSource collections.Iterable,
count core.Expression,
offset core.Expression,
) (collections.Iterable, error) {
if dataSource == nil {
return nil, core.Error(core.ErrMissedArgument, "dataSource source")
}
return &LimitClause{src, dataSource, count, offset}, nil
}
func (clause *LimitClause) Iterate(ctx context.Context, scope *core.Scope) (collections.Iterator, error) {
src, err := clause.dataSource.Iterate(ctx, scope)
if err != nil {
return nil, core.SourceError(clause.src, err)
}
count, err := clause.count.Exec(ctx, scope)
if err != nil {
return nil, core.SourceError(clause.src, err)
}
offset, err := clause.offset.Exec(ctx, scope)
if err != nil {
return nil, core.SourceError(clause.src, err)
}
countInt, err := clause.parseValue(count)
if err != nil {
return nil, err
}
offsetInt, err := clause.parseValue(offset)
if err != nil {
return nil, err
}
iterator, err := collections.NewLimitIterator(
src,
int(countInt),
int(offsetInt),
)
if err != nil {
return nil, core.SourceError(clause.src, err)
}
return iterator, nil
}
func (clause *LimitClause) parseValue(val core.Value) (int, error) {
if val.Type() == core.IntType {
return val.Unwrap().(int), nil
}
if val.Type() == core.FloatType {
return int(val.Unwrap().(float64)), nil
}
return -1, core.TypeError(val.Type(), core.IntType, core.FloatType)
}