influxdb/query/influxql/map.go

199 lines
5.8 KiB
Go

package influxql
import (
"fmt"
"time"
"github.com/influxdata/influxql"
"github.com/influxdata/platform/query"
"github.com/influxdata/platform/query/ast"
"github.com/influxdata/platform/query/execute"
"github.com/influxdata/platform/query/functions"
"github.com/influxdata/platform/query/semantic"
)
// mapCursor holds the mapping of expressions to specific fields that happens at the end of
// the transpilation.
// TODO(jsternberg): This abstraction might be useful for subqueries, but we only need the id
// at the moment so just hold that.
type mapCursor struct {
id query.OperationID
}
func (c *mapCursor) ID() query.OperationID {
return c.id
}
func (c *mapCursor) Keys() []influxql.Expr {
panic("unimplemented")
}
func (c *mapCursor) Value(expr influxql.Expr) (string, bool) {
panic("unimplemented")
}
// mapFields will take the list of symbols and maps each of the operations
// using the column names.
func (t *transpilerState) mapFields(in cursor) (cursor, error) {
columns := t.stmt.ColumnNames()
if len(columns) != len(t.stmt.Fields) {
// TODO(jsternberg): This scenario should not be possible. Replace the use of ColumnNames with a more
// statically verifiable list of columns when we process the fields from the select statement instead
// of doing this in the future.
panic("number of columns does not match the number of fields")
}
properties := make([]*semantic.Property, 0, len(t.stmt.Fields)+1)
properties = append(properties, &semantic.Property{
Key: &semantic.Identifier{Name: execute.DefaultTimeColLabel},
Value: &semantic.MemberExpression{
Object: &semantic.IdentifierExpression{
Name: "r",
},
Property: execute.DefaultTimeColLabel,
},
})
for i, f := range t.stmt.Fields {
if ref, ok := f.Expr.(*influxql.VarRef); ok && ref.Val == "time" {
// Skip past any time columns.
continue
}
value, err := t.mapField(f.Expr, in)
if err != nil {
return nil, err
}
properties = append(properties, &semantic.Property{
Key: &semantic.Identifier{Name: columns[i]},
Value: value,
})
}
id := t.op("map", &functions.MapOpSpec{
Fn: &semantic.FunctionExpression{
Params: []*semantic.FunctionParam{{
Key: &semantic.Identifier{Name: "r"},
}},
Body: &semantic.ObjectExpression{
Properties: properties,
},
},
MergeKey: true,
}, in.ID())
return &mapCursor{id: id}, nil
}
func (t *transpilerState) mapField(expr influxql.Expr, in cursor) (semantic.Expression, error) {
if sym, ok := in.Value(expr); ok {
return &semantic.MemberExpression{
Object: &semantic.IdentifierExpression{
Name: "r",
},
Property: sym,
}, nil
}
switch expr := expr.(type) {
case *influxql.Call:
if isMathFunction(expr) {
return nil, fmt.Errorf("unimplemented math function: %q", expr.Name)
}
return nil, fmt.Errorf("missing symbol for %s", expr)
case *influxql.VarRef:
return nil, fmt.Errorf("missing symbol for %s", expr)
case *influxql.BinaryExpr:
return t.evalBinaryExpr(expr, in)
case *influxql.ParenExpr:
return t.mapField(expr.Expr, in)
case *influxql.StringLiteral:
if ts, err := expr.ToTimeLiteral(time.UTC); err == nil {
return &semantic.DateTimeLiteral{Value: ts.Val}, nil
}
return &semantic.StringLiteral{Value: expr.Val}, nil
case *influxql.NumberLiteral:
return &semantic.FloatLiteral{Value: expr.Val}, nil
case *influxql.IntegerLiteral:
return &semantic.IntegerLiteral{Value: expr.Val}, nil
case *influxql.BooleanLiteral:
return &semantic.BooleanLiteral{Value: expr.Val}, nil
case *influxql.DurationLiteral:
return &semantic.DurationLiteral{Value: expr.Val}, nil
case *influxql.TimeLiteral:
return &semantic.DateTimeLiteral{Value: expr.Val}, nil
case *influxql.RegexLiteral:
return &semantic.RegexpLiteral{Value: expr.Val}, nil
default:
// TODO(jsternberg): Handle the other expressions by turning them into
// an equivalent expression.
return nil, fmt.Errorf("unimplemented: %T", expr)
}
}
func (t *transpilerState) evalBinaryExpr(expr *influxql.BinaryExpr, in cursor) (semantic.Expression, error) {
fn := func() func(left, right semantic.Expression) semantic.Expression {
b := evalBuilder{}
switch expr.Op {
case influxql.EQ:
return b.eval(ast.EqualOperator)
case influxql.NEQ:
return b.eval(ast.NotEqualOperator)
case influxql.GT:
return b.eval(ast.GreaterThanOperator)
case influxql.GTE:
return b.eval(ast.GreaterThanEqualOperator)
case influxql.LT:
return b.eval(ast.LessThanOperator)
case influxql.LTE:
return b.eval(ast.LessThanEqualOperator)
case influxql.ADD:
return b.eval(ast.AdditionOperator)
case influxql.SUB:
return b.eval(ast.SubtractionOperator)
case influxql.AND:
return b.logical(ast.AndOperator)
case influxql.OR:
return b.logical(ast.OrOperator)
case influxql.EQREGEX:
return b.eval(ast.RegexpMatchOperator)
case influxql.NEQREGEX:
return b.eval(ast.NotRegexpMatchOperator)
default:
return nil
}
}()
if fn == nil {
return nil, fmt.Errorf("unimplemented binary expression: %s", expr.Op)
}
lhs, err := t.mapField(expr.LHS, in)
if err != nil {
return nil, err
}
rhs, err := t.mapField(expr.RHS, in)
if err != nil {
return nil, err
}
return fn(lhs, rhs), nil
}
// evalBuilder is used for namespacing the logical and eval wrapping functions.
type evalBuilder struct{}
func (evalBuilder) logical(op ast.LogicalOperatorKind) func(left, right semantic.Expression) semantic.Expression {
return func(left, right semantic.Expression) semantic.Expression {
return &semantic.LogicalExpression{
Operator: op,
Left: left,
Right: right,
}
}
}
func (evalBuilder) eval(op ast.OperatorKind) func(left, right semantic.Expression) semantic.Expression {
return func(left, right semantic.Expression) semantic.Expression {
return &semantic.BinaryExpression{
Operator: op,
Left: left,
Right: right,
}
}
}