1
0
mirror of https://github.com/MontFerret/ferret.git synced 2024-12-29 01:44:23 +02:00
ferret/pkg/runtime/expressions/clauses/limit.go

86 lines
1.7 KiB
Go
Raw Normal View History

2018-09-18 22:42:38 +02:00
package clauses
import (
"context"
2018-09-18 22:42:38 +02:00
"github.com/MontFerret/ferret/pkg/runtime/collections"
"github.com/MontFerret/ferret/pkg/runtime/core"
"github.com/MontFerret/ferret/pkg/runtime/values/types"
2018-09-18 22:42:38 +02:00
)
type LimitClause struct {
src core.SourceMap
dataSource collections.Iterable
count core.Expression
offset core.Expression
2018-09-18 22:42:38 +02:00
}
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
}
2018-09-18 22:42:38 +02:00
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)
2018-09-18 22:42:38 +02:00
}
return iterator, nil
2018-09-18 22:42:38 +02:00
}
func (clause *LimitClause) parseValue(val core.Value) (int, error) {
if val.Type() == types.Int {
return val.Unwrap().(int), nil
}
if val.Type() == types.Float {
return int(val.Unwrap().(float64)), nil
}
return -1, core.TypeError(val.Type(), types.Int, types.Float)
}