318 lines
7.6 KiB
Go
318 lines
7.6 KiB
Go
package functions
|
|
|
|
import (
|
|
"fmt"
|
|
"math"
|
|
|
|
"github.com/influxdata/platform/query"
|
|
"github.com/influxdata/platform/query/execute"
|
|
"github.com/influxdata/platform/query/interpreter"
|
|
"github.com/influxdata/platform/query/plan"
|
|
"github.com/influxdata/platform/query/semantic"
|
|
)
|
|
|
|
const DifferenceKind = "difference"
|
|
|
|
type DifferenceOpSpec struct {
|
|
NonNegative bool `json:"non_negative"`
|
|
Columns []string `json:"columns"`
|
|
}
|
|
|
|
var differenceSignature = query.DefaultFunctionSignature()
|
|
|
|
func init() {
|
|
differenceSignature.Params["nonNegative"] = semantic.Bool
|
|
derivativeSignature.Params["columns"] = semantic.NewArrayType(semantic.String)
|
|
|
|
query.RegisterFunction(DifferenceKind, createDifferenceOpSpec, differenceSignature)
|
|
query.RegisterOpSpec(DifferenceKind, newDifferenceOp)
|
|
plan.RegisterProcedureSpec(DifferenceKind, newDifferenceProcedure, DifferenceKind)
|
|
execute.RegisterTransformation(DifferenceKind, createDifferenceTransformation)
|
|
}
|
|
|
|
func createDifferenceOpSpec(args query.Arguments, a *query.Administration) (query.OperationSpec, error) {
|
|
if err := a.AddParentFromArgs(args); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err := a.AddParentFromArgs(args)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
spec := new(DifferenceOpSpec)
|
|
|
|
if nn, ok, err := args.GetBool("nonNegative"); err != nil {
|
|
return nil, err
|
|
} else if ok {
|
|
spec.NonNegative = nn
|
|
}
|
|
|
|
if cols, ok, err := args.GetArray("columns", semantic.String); err != nil {
|
|
return nil, err
|
|
} else if ok {
|
|
columns, err := interpreter.ToStringArray(cols)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
spec.Columns = columns
|
|
} else {
|
|
spec.Columns = []string{execute.DefaultValueColLabel}
|
|
}
|
|
|
|
return spec, nil
|
|
}
|
|
|
|
func newDifferenceOp() query.OperationSpec {
|
|
return new(DifferenceOpSpec)
|
|
}
|
|
|
|
func (s *DifferenceOpSpec) Kind() query.OperationKind {
|
|
return DifferenceKind
|
|
}
|
|
|
|
type DifferenceProcedureSpec struct {
|
|
NonNegative bool `json:"non_negative"`
|
|
Columns []string `json:"columns"`
|
|
}
|
|
|
|
func newDifferenceProcedure(qs query.OperationSpec, pa plan.Administration) (plan.ProcedureSpec, error) {
|
|
spec, ok := qs.(*DifferenceOpSpec)
|
|
if !ok {
|
|
return nil, fmt.Errorf("invalid spec type %T", qs)
|
|
}
|
|
|
|
return &DifferenceProcedureSpec{
|
|
NonNegative: spec.NonNegative,
|
|
Columns: spec.Columns,
|
|
}, nil
|
|
}
|
|
|
|
func (s *DifferenceProcedureSpec) Kind() plan.ProcedureKind {
|
|
return DifferenceKind
|
|
}
|
|
func (s *DifferenceProcedureSpec) Copy() plan.ProcedureSpec {
|
|
ns := new(DifferenceProcedureSpec)
|
|
*ns = *s
|
|
if s.Columns != nil {
|
|
ns.Columns = make([]string, len(s.Columns))
|
|
copy(ns.Columns, s.Columns)
|
|
}
|
|
return ns
|
|
}
|
|
|
|
func createDifferenceTransformation(id execute.DatasetID, mode execute.AccumulationMode, spec plan.ProcedureSpec, a execute.Administration) (execute.Transformation, execute.Dataset, error) {
|
|
s, ok := spec.(*DifferenceProcedureSpec)
|
|
if !ok {
|
|
return nil, nil, fmt.Errorf("invalid spec type %T", spec)
|
|
}
|
|
cache := execute.NewTableBuilderCache(a.Allocator())
|
|
d := execute.NewDataset(id, mode, cache)
|
|
t := NewDifferenceTransformation(d, cache, s)
|
|
return t, d, nil
|
|
}
|
|
|
|
type differenceTransformation struct {
|
|
d execute.Dataset
|
|
cache execute.TableBuilderCache
|
|
|
|
nonNegative bool
|
|
columns []string
|
|
}
|
|
|
|
func NewDifferenceTransformation(d execute.Dataset, cache execute.TableBuilderCache, spec *DifferenceProcedureSpec) *differenceTransformation {
|
|
return &differenceTransformation{
|
|
d: d,
|
|
cache: cache,
|
|
nonNegative: spec.NonNegative,
|
|
columns: spec.Columns,
|
|
}
|
|
}
|
|
|
|
func (t *differenceTransformation) RetractTable(id execute.DatasetID, key query.GroupKey) error {
|
|
return t.d.RetractTable(key)
|
|
}
|
|
|
|
func (t *differenceTransformation) Process(id execute.DatasetID, tbl query.Table) error {
|
|
builder, created := t.cache.TableBuilder(tbl.Key())
|
|
if !created {
|
|
return fmt.Errorf("difference found duplicate table with key: %v", tbl.Key())
|
|
}
|
|
cols := tbl.Cols()
|
|
differences := make([]*difference, len(cols))
|
|
for j, c := range cols {
|
|
found := false
|
|
for _, label := range t.columns {
|
|
if c.Label == label {
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
|
|
if found {
|
|
var typ query.DataType
|
|
switch c.Type {
|
|
case query.TInt, query.TUInt:
|
|
typ = query.TInt
|
|
case query.TFloat:
|
|
typ = query.TFloat
|
|
}
|
|
builder.AddCol(query.ColMeta{
|
|
Label: c.Label,
|
|
Type: typ,
|
|
})
|
|
differences[j] = newDifference(j, t.nonNegative)
|
|
} else {
|
|
builder.AddCol(c)
|
|
}
|
|
}
|
|
|
|
// We need to drop the first row since its derivative is undefined
|
|
firstIdx := 1
|
|
return tbl.Do(func(cr query.ColReader) error {
|
|
l := cr.Len()
|
|
|
|
if l != 0 {
|
|
for j, c := range cols {
|
|
d := differences[j]
|
|
switch c.Type {
|
|
case query.TBool:
|
|
builder.AppendBools(j, cr.Bools(j)[firstIdx:])
|
|
case query.TInt:
|
|
if d != nil {
|
|
for i := 0; i < l; i++ {
|
|
v := d.updateInt(cr.Ints(j)[i])
|
|
if i != 0 || firstIdx == 0 {
|
|
builder.AppendInt(j, v)
|
|
}
|
|
}
|
|
} else {
|
|
builder.AppendInts(j, cr.Ints(j)[firstIdx:])
|
|
}
|
|
case query.TUInt:
|
|
if d != nil {
|
|
for i := 0; i < l; i++ {
|
|
v := d.updateUInt(cr.UInts(j)[i])
|
|
if i != 0 || firstIdx == 0 {
|
|
builder.AppendInt(j, v)
|
|
}
|
|
}
|
|
} else {
|
|
builder.AppendUInts(j, cr.UInts(j)[firstIdx:])
|
|
}
|
|
case query.TFloat:
|
|
if d != nil {
|
|
for i := 0; i < l; i++ {
|
|
v := d.updateFloat(cr.Floats(j)[i])
|
|
if i != 0 || firstIdx == 0 {
|
|
builder.AppendFloat(j, v)
|
|
}
|
|
}
|
|
} else {
|
|
builder.AppendFloats(j, cr.Floats(j)[firstIdx:])
|
|
}
|
|
case query.TString:
|
|
builder.AppendStrings(j, cr.Strings(j)[firstIdx:])
|
|
case query.TTime:
|
|
builder.AppendTimes(j, cr.Times(j)[firstIdx:])
|
|
}
|
|
}
|
|
}
|
|
|
|
// Now that we skipped the first row, start at 0 for the rest of the batches
|
|
firstIdx = 0
|
|
return nil
|
|
})
|
|
}
|
|
|
|
func (t *differenceTransformation) UpdateWatermark(id execute.DatasetID, mark execute.Time) error {
|
|
return t.d.UpdateWatermark(mark)
|
|
}
|
|
func (t *differenceTransformation) UpdateProcessingTime(id execute.DatasetID, pt execute.Time) error {
|
|
return t.d.UpdateProcessingTime(pt)
|
|
}
|
|
func (t *differenceTransformation) Finish(id execute.DatasetID, err error) {
|
|
t.d.Finish(err)
|
|
}
|
|
|
|
func newDifference(col int, nonNegative bool) *difference {
|
|
return &difference{
|
|
col: col,
|
|
first: true,
|
|
nonNegative: nonNegative,
|
|
}
|
|
}
|
|
|
|
type difference struct {
|
|
col int
|
|
first bool
|
|
nonNegative bool
|
|
|
|
pIntValue int64
|
|
pUIntValue uint64
|
|
pFloatValue float64
|
|
}
|
|
|
|
func (d *difference) updateInt(v int64) int64 {
|
|
if d.first {
|
|
d.pIntValue = v
|
|
d.first = false
|
|
return 0
|
|
}
|
|
|
|
diff := v - d.pIntValue
|
|
|
|
d.pIntValue = v
|
|
|
|
if d.nonNegative && diff < 0 {
|
|
//TODO(nathanielc): Return null when we have null support
|
|
// Also see https://github.com/influxdata/platform/query/issues/217
|
|
return v
|
|
}
|
|
|
|
return diff
|
|
}
|
|
func (d *difference) updateUInt(v uint64) int64 {
|
|
if d.first {
|
|
d.pUIntValue = v
|
|
d.first = false
|
|
return 0
|
|
}
|
|
|
|
var diff int64
|
|
if d.pUIntValue > v {
|
|
// Prevent uint64 overflow by applying the negative sign after the conversion to an int64.
|
|
diff = int64(d.pUIntValue-v) * -1
|
|
} else {
|
|
diff = int64(v - d.pUIntValue)
|
|
}
|
|
|
|
d.pUIntValue = v
|
|
|
|
if d.nonNegative && diff < 0 {
|
|
//TODO(nathanielc): Return null when we have null support
|
|
// Also see https://github.com/influxdata/platform/query/issues/217
|
|
return int64(v)
|
|
}
|
|
|
|
return diff
|
|
}
|
|
func (d *difference) updateFloat(v float64) float64 {
|
|
if d.first {
|
|
d.pFloatValue = v
|
|
d.first = false
|
|
return math.NaN()
|
|
}
|
|
|
|
diff := v - d.pFloatValue
|
|
d.pFloatValue = v
|
|
|
|
if d.nonNegative && diff < 0 {
|
|
//TODO(nathanielc): Return null when we have null support
|
|
// Also see https://github.com/influxdata/platform/query/issues/217
|
|
return v
|
|
}
|
|
|
|
return diff
|
|
}
|