2018-09-18 22:42:38 +02:00
|
|
|
package operators
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"github.com/MontFerret/ferret/pkg/runtime/core"
|
|
|
|
"github.com/MontFerret/ferret/pkg/runtime/values"
|
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
|
|
|
LogicalOperatorType int
|
|
|
|
LogicalOperator struct {
|
|
|
|
*baseOperator
|
|
|
|
value LogicalOperatorType
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
AndType LogicalOperatorType = 0
|
|
|
|
OrType LogicalOperatorType = 1
|
|
|
|
NotType LogicalOperatorType = 2
|
|
|
|
)
|
|
|
|
|
|
|
|
var logicalOperators = map[string]LogicalOperatorType{
|
|
|
|
"&&": AndType,
|
|
|
|
"AND": AndType,
|
|
|
|
"||": OrType,
|
|
|
|
"OR": OrType,
|
|
|
|
"NOT": NotType,
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewLogicalOperator(
|
|
|
|
src core.SourceMap,
|
|
|
|
left core.Expression,
|
|
|
|
right core.Expression,
|
|
|
|
operator string,
|
|
|
|
) (*LogicalOperator, error) {
|
|
|
|
op, exists := logicalOperators[operator]
|
|
|
|
|
|
|
|
if !exists {
|
|
|
|
return nil, core.Error(core.ErrInvalidArgument, "value")
|
|
|
|
}
|
|
|
|
|
|
|
|
return &LogicalOperator{
|
|
|
|
&baseOperator{
|
|
|
|
src,
|
|
|
|
left,
|
|
|
|
right,
|
|
|
|
},
|
|
|
|
op,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (operator *LogicalOperator) Exec(ctx context.Context, scope *core.Scope) (core.Value, error) {
|
2018-09-23 03:06:19 +02:00
|
|
|
if operator.value == NotType {
|
|
|
|
val, err := operator.right.Exec(ctx, scope)
|
2018-09-18 22:42:38 +02:00
|
|
|
|
2018-09-23 03:06:19 +02:00
|
|
|
if err != nil {
|
|
|
|
return values.None, core.SourceError(operator.src, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return Not(val, values.None), nil
|
2018-09-18 22:42:38 +02:00
|
|
|
}
|
|
|
|
|
2018-09-23 03:06:19 +02:00
|
|
|
left, err := operator.left.Exec(ctx, scope)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return values.None, core.SourceError(operator.src, err)
|
2018-09-18 22:42:38 +02:00
|
|
|
}
|
|
|
|
|
2018-09-23 02:28:33 +02:00
|
|
|
leftBool := values.ToBoolean(left)
|
|
|
|
|
|
|
|
if operator.value == AndType && leftBool == values.False {
|
|
|
|
if left.Type() == core.BooleanType {
|
|
|
|
return values.False, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return left, nil
|
2018-09-18 22:42:38 +02:00
|
|
|
}
|
|
|
|
|
2018-09-23 02:28:33 +02:00
|
|
|
if operator.value == OrType && leftBool == values.True {
|
|
|
|
return left, nil
|
2018-09-18 22:42:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
right, err := operator.right.Exec(ctx, scope)
|
|
|
|
|
|
|
|
if err != nil {
|
2018-09-23 03:06:19 +02:00
|
|
|
return values.None, core.SourceError(operator.src, err)
|
2018-09-18 22:42:38 +02:00
|
|
|
}
|
|
|
|
|
2018-09-23 02:28:33 +02:00
|
|
|
return right, nil
|
2018-09-18 22:42:38 +02:00
|
|
|
}
|