2020-03-26 22:37:59 +00:00
|
|
|
package cursors
|
|
|
|
|
2020-07-08 23:46:01 +00:00
|
|
|
import (
|
|
|
|
"github.com/influxdata/influxdb/v2/models"
|
|
|
|
"github.com/influxdata/influxql"
|
|
|
|
)
|
2020-04-14 23:33:54 +00:00
|
|
|
|
2020-03-26 22:37:59 +00:00
|
|
|
// FieldType represents the primitive field data types available in tsm.
|
|
|
|
type FieldType int
|
|
|
|
|
|
|
|
const (
|
2020-04-08 23:15:34 +00:00
|
|
|
Float FieldType = iota // means the data type is a float
|
|
|
|
Integer // means the data type is an integer
|
|
|
|
Unsigned // means the data type is an unsigned integer
|
|
|
|
String // means the data type is a string of text
|
2020-04-14 23:33:54 +00:00
|
|
|
Boolean // means the data type is a boolean
|
2020-04-08 23:15:34 +00:00
|
|
|
Undefined // means the data type in unknown or undefined
|
2020-03-26 22:37:59 +00:00
|
|
|
)
|
|
|
|
|
2020-04-14 23:33:54 +00:00
|
|
|
var (
|
|
|
|
fieldTypeToDataTypeMapping = [8]influxql.DataType{
|
|
|
|
Float: influxql.Float,
|
|
|
|
Integer: influxql.Integer,
|
|
|
|
Unsigned: influxql.Unsigned,
|
|
|
|
String: influxql.String,
|
|
|
|
Boolean: influxql.Boolean,
|
|
|
|
Undefined: influxql.Unknown,
|
|
|
|
6: influxql.Unknown,
|
|
|
|
7: influxql.Unknown,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2020-04-16 22:19:28 +00:00
|
|
|
// FieldTypeToDataType returns the equivalent influxql DataType for the field type ft.
|
|
|
|
// If ft is an invalid FieldType, the results are undefined.
|
2020-04-14 23:33:54 +00:00
|
|
|
func FieldTypeToDataType(ft FieldType) influxql.DataType {
|
|
|
|
return fieldTypeToDataTypeMapping[ft&7]
|
|
|
|
}
|
|
|
|
|
2020-04-16 22:19:28 +00:00
|
|
|
// IsLower returns true if the other FieldType has greater precedence than the
|
|
|
|
// current value. Undefined has the lowest precedence.
|
|
|
|
func (ft FieldType) IsLower(other FieldType) bool { return other < ft }
|
|
|
|
|
2020-07-08 23:46:01 +00:00
|
|
|
var (
|
|
|
|
modelsFieldTypeToFieldTypeMapping = [8]FieldType{
|
|
|
|
models.Integer: Integer,
|
|
|
|
models.Float: Float,
|
|
|
|
models.Boolean: Boolean,
|
|
|
|
models.String: String,
|
|
|
|
models.Empty: Undefined,
|
|
|
|
models.Unsigned: Unsigned,
|
|
|
|
6: Undefined,
|
|
|
|
7: Undefined,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
// ModelsFieldTypeToFieldType returns the equivalent FieldType for ft.
|
|
|
|
// If ft is an invalid FieldType, the results are undefined.
|
|
|
|
func ModelsFieldTypeToFieldType(ft models.FieldType) FieldType {
|
|
|
|
return modelsFieldTypeToFieldTypeMapping[ft&7]
|
|
|
|
}
|
|
|
|
|
2020-03-26 22:37:59 +00:00
|
|
|
type MeasurementField struct {
|
2020-04-16 22:19:28 +00:00
|
|
|
Key string // Key is the name of the field
|
|
|
|
Type FieldType // Type is field type
|
|
|
|
Timestamp int64 // Timestamp refers to the maximum timestamp observed for the given field
|
2020-03-26 22:37:59 +00:00
|
|
|
}
|
|
|
|
|
2020-04-14 23:33:54 +00:00
|
|
|
// MeasurementFieldSlice implements sort.Interface and sorts
|
|
|
|
// the slice from lowest to highest precedence. Use sort.Reverse
|
|
|
|
// to sort from highest to lowest.
|
|
|
|
type MeasurementFieldSlice []MeasurementField
|
|
|
|
|
|
|
|
func (m MeasurementFieldSlice) Len() int {
|
|
|
|
return len(m)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m MeasurementFieldSlice) Less(i, j int) bool {
|
|
|
|
ii, jj := &m[i], &m[j]
|
|
|
|
return ii.Key < jj.Key ||
|
|
|
|
(ii.Key == jj.Key &&
|
|
|
|
(ii.Timestamp < jj.Timestamp ||
|
|
|
|
(ii.Timestamp == jj.Timestamp && ii.Type.IsLower(jj.Type))))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m MeasurementFieldSlice) Swap(i, j int) {
|
|
|
|
m[i], m[j] = m[j], m[i]
|
|
|
|
}
|
|
|
|
|
|
|
|
// UniqueByKey performs an in-place update of m, removing duplicate elements
|
|
|
|
// by Key, keeping the first occurrence of each. If the slice is not sorted,
|
|
|
|
// the behavior of UniqueByKey is undefined.
|
|
|
|
func (m *MeasurementFieldSlice) UniqueByKey() {
|
|
|
|
mm := *m
|
|
|
|
if len(mm) < 2 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
j := 0
|
|
|
|
for i := 1; i < len(mm); i++ {
|
|
|
|
if mm[j].Key != mm[i].Key {
|
|
|
|
j++
|
|
|
|
if j != i {
|
|
|
|
// optimization: skip copy if j == i
|
|
|
|
mm[j] = mm[i]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*m = mm[:j+1]
|
|
|
|
}
|
|
|
|
|
2020-03-26 22:37:59 +00:00
|
|
|
type MeasurementFields struct {
|
|
|
|
Fields []MeasurementField
|
|
|
|
}
|
|
|
|
|
2020-04-08 23:15:34 +00:00
|
|
|
type MeasurementFieldsIterator interface {
|
|
|
|
// Next advances the iterator to the next value. It returns false
|
2020-03-26 22:37:59 +00:00
|
|
|
// when there are no more values.
|
|
|
|
Next() bool
|
|
|
|
|
|
|
|
// Value returns the current value.
|
|
|
|
Value() MeasurementFields
|
|
|
|
|
|
|
|
Stats() CursorStats
|
|
|
|
}
|
2020-04-08 23:15:34 +00:00
|
|
|
|
|
|
|
// EmptyMeasurementFieldsIterator is an implementation of MeasurementFieldsIterator that returns
|
|
|
|
// no values.
|
|
|
|
var EmptyMeasurementFieldsIterator = &measurementFieldsIterator{}
|
|
|
|
|
|
|
|
type measurementFieldsIterator struct{}
|
|
|
|
|
|
|
|
func (m *measurementFieldsIterator) Next() bool { return false }
|
|
|
|
func (m *measurementFieldsIterator) Value() MeasurementFields { return MeasurementFields{} }
|
|
|
|
func (m *measurementFieldsIterator) Stats() CursorStats { return CursorStats{} }
|
|
|
|
|
|
|
|
type MeasurementFieldsSliceIterator struct {
|
|
|
|
f []MeasurementFields
|
|
|
|
v MeasurementFields
|
|
|
|
i int
|
|
|
|
stats CursorStats
|
|
|
|
}
|
|
|
|
|
2020-07-08 23:46:01 +00:00
|
|
|
func NewMeasurementFieldsSliceIterator(f []MeasurementFields) *MeasurementFieldsSliceIterator {
|
|
|
|
return &MeasurementFieldsSliceIterator{f: f}
|
|
|
|
}
|
|
|
|
|
2020-04-08 23:15:34 +00:00
|
|
|
func NewMeasurementFieldsSliceIteratorWithStats(f []MeasurementFields, stats CursorStats) *MeasurementFieldsSliceIterator {
|
|
|
|
return &MeasurementFieldsSliceIterator{f: f, stats: stats}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MeasurementFieldsSliceIterator) Next() bool {
|
|
|
|
if s.i < len(s.f) {
|
|
|
|
s.v = s.f[s.i]
|
|
|
|
s.i++
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
s.v = MeasurementFields{}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MeasurementFieldsSliceIterator) Value() MeasurementFields {
|
|
|
|
return s.v
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MeasurementFieldsSliceIterator) Stats() CursorStats {
|
|
|
|
return s.stats
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *MeasurementFieldsSliceIterator) toSlice() []MeasurementFields {
|
|
|
|
if s.i < len(s.f) {
|
|
|
|
return s.f[s.i:]
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// MeasurementFieldsIteratorFlatMap reads the remainder of i, flattening the results
|
|
|
|
// to a single slice.
|
|
|
|
func MeasurementFieldsIteratorFlatMap(i MeasurementFieldsIterator) []MeasurementField {
|
|
|
|
if i == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var res []MeasurementField
|
|
|
|
if si, ok := i.(*MeasurementFieldsSliceIterator); ok {
|
|
|
|
s := si.toSlice()
|
|
|
|
sz := 0
|
|
|
|
for i := range s {
|
|
|
|
sz += len(s[i].Fields)
|
|
|
|
}
|
|
|
|
res = make([]MeasurementField, 0, sz)
|
|
|
|
for i := range s {
|
|
|
|
res = append(res, s[i].Fields...)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for i.Next() {
|
|
|
|
res = append(res, i.Value().Fields...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res
|
|
|
|
}
|