123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260 |
- // Copyright 2020-2021 InfluxData, Inc. All rights reserved.
- // Use of this source code is governed by MIT
- // license that can be found in the LICENSE file.
- // Package query defined types for representing flux query result
- package query
- import (
- "fmt"
- "sort"
- "strings"
- "time"
- )
- // FluxTableMetadata holds flux query result table information represented by collection of columns.
- // Each new table is introduced by annotations
- type FluxTableMetadata struct {
- position int
- columns []*FluxColumn
- }
- // FluxColumn holds flux query table column properties
- type FluxColumn struct {
- index int
- name string
- dataType string
- group bool
- defaultValue string
- }
- // FluxRecord represents row in the flux query result table
- type FluxRecord struct {
- table int
- values map[string]interface{}
- }
- // NewFluxTableMetadata creates FluxTableMetadata for the table on position
- func NewFluxTableMetadata(position int) *FluxTableMetadata {
- return NewFluxTableMetadataFull(position, make([]*FluxColumn, 0, 10))
- }
- // NewFluxTableMetadataFull creates FluxTableMetadata
- func NewFluxTableMetadataFull(position int, columns []*FluxColumn) *FluxTableMetadata {
- return &FluxTableMetadata{position: position, columns: columns}
- }
- // Position returns position of the table in the flux query result
- func (f *FluxTableMetadata) Position() int {
- return f.position
- }
- // Columns returns slice of flux query result table
- func (f *FluxTableMetadata) Columns() []*FluxColumn {
- return f.columns
- }
- // AddColumn adds column definition to table metadata
- func (f *FluxTableMetadata) AddColumn(column *FluxColumn) *FluxTableMetadata {
- f.columns = append(f.columns, column)
- return f
- }
- // Column returns flux table column by index.
- // Returns nil if index is out of the bounds.
- func (f *FluxTableMetadata) Column(index int) *FluxColumn {
- if len(f.columns) == 0 || index < 0 || index >= len(f.columns) {
- return nil
- }
- return f.columns[index]
- }
- // String returns FluxTableMetadata string dump
- func (f *FluxTableMetadata) String() string {
- var buffer strings.Builder
- for i, c := range f.columns {
- if i > 0 {
- buffer.WriteString(",")
- }
- buffer.WriteString("col")
- buffer.WriteString(c.String())
- }
- return buffer.String()
- }
- // NewFluxColumn creates FluxColumn for position
- func NewFluxColumn(index int) *FluxColumn {
- return &FluxColumn{index: index}
- }
- // NewFluxColumnFull creates FluxColumn
- func NewFluxColumnFull(dataType string, defaultValue string, name string, group bool, index int) *FluxColumn {
- return &FluxColumn{index: index, name: name, dataType: dataType, group: group, defaultValue: defaultValue}
- }
- // SetDefaultValue sets default value for the column
- func (f *FluxColumn) SetDefaultValue(defaultValue string) {
- f.defaultValue = defaultValue
- }
- // SetGroup set group flag for the column
- func (f *FluxColumn) SetGroup(group bool) {
- f.group = group
- }
- // SetDataType sets data type for the column
- func (f *FluxColumn) SetDataType(dataType string) {
- f.dataType = dataType
- }
- // SetName sets name of the column
- func (f *FluxColumn) SetName(name string) {
- f.name = name
- }
- // DefaultValue returns default value of the column
- func (f *FluxColumn) DefaultValue() string {
- return f.defaultValue
- }
- // IsGroup return true if the column is grouping column
- func (f *FluxColumn) IsGroup() bool {
- return f.group
- }
- // DataType returns data type of the column
- func (f *FluxColumn) DataType() string {
- return f.dataType
- }
- // Name returns name of the column
- func (f *FluxColumn) Name() string {
- return f.name
- }
- // Index returns index of the column
- func (f *FluxColumn) Index() int {
- return f.index
- }
- // String returns FluxColumn string dump
- func (f *FluxColumn) String() string {
- return fmt.Sprintf("{%d: name: %s, datatype: %s, defaultValue: %s, group: %v}", f.index, f.name, f.dataType, f.defaultValue, f.group)
- }
- // NewFluxRecord returns new record for the table with values
- func NewFluxRecord(table int, values map[string]interface{}) *FluxRecord {
- return &FluxRecord{table: table, values: values}
- }
- // Table returns value of the table column
- // It returns zero if the table column is not found
- func (r *FluxRecord) Table() int {
- return int(intValue(r.values, "table"))
- }
- // Start returns the inclusive lower time bound of all records in the current table.
- // Returns empty time.Time if there is no column "_start".
- func (r *FluxRecord) Start() time.Time {
- return timeValue(r.values, "_start")
- }
- // Stop returns the exclusive upper time bound of all records in the current table.
- // Returns empty time.Time if there is no column "_stop".
- func (r *FluxRecord) Stop() time.Time {
- return timeValue(r.values, "_stop")
- }
- // Time returns the time of the record.
- // Returns empty time.Time if there is no column "_time".
- func (r *FluxRecord) Time() time.Time {
- return timeValue(r.values, "_time")
- }
- // Value returns the default _value column value or nil if not present
- func (r *FluxRecord) Value() interface{} {
- return r.ValueByKey("_value")
- }
- // Field returns the field name.
- // Returns empty string if there is no column "_field".
- func (r *FluxRecord) Field() string {
- return stringValue(r.values, "_field")
- }
- // Result returns the value of the _result column, which represents result name.
- // Returns empty string if there is no column "result".
- func (r *FluxRecord) Result() string {
- return stringValue(r.values, "result")
- }
- // Measurement returns the measurement name of the record
- // Returns empty string if there is no column "_measurement".
- func (r *FluxRecord) Measurement() string {
- return stringValue(r.values, "_measurement")
- }
- // Values returns map of the values where key is the column name
- func (r *FluxRecord) Values() map[string]interface{} {
- return r.values
- }
- // ValueByKey returns value for given column key for the record or nil of result has no value the column key
- func (r *FluxRecord) ValueByKey(key string) interface{} {
- return r.values[key]
- }
- // String returns FluxRecord string dump
- func (r *FluxRecord) String() string {
- if len(r.values) == 0 {
- return ""
- }
- i := 0
- keys := make([]string, len(r.values))
- for k := range r.values {
- keys[i] = k
- i++
- }
- sort.Strings(keys)
- var buffer strings.Builder
- buffer.WriteString(fmt.Sprintf("%s:%v", keys[0], r.values[keys[0]]))
- for _, k := range keys[1:] {
- buffer.WriteString(",")
- buffer.WriteString(fmt.Sprintf("%s:%v", k, r.values[k]))
- }
- return buffer.String()
- }
- // timeValue returns time.Time value from values map according to the key
- // Empty time.Time value is returned if key is not found
- func timeValue(values map[string]interface{}, key string) time.Time {
- if val, ok := values[key]; ok {
- if t, ok := val.(time.Time); ok {
- return t
- }
- }
- return time.Time{}
- }
- // stringValue returns string value from values map according to the key
- // Empty string is returned if key is not found
- func stringValue(values map[string]interface{}, key string) string {
- if val, ok := values[key]; ok {
- if s, ok := val.(string); ok {
- return s
- }
- }
- return ""
- }
- // intValue returns int64 value from values map according to the key
- // Zero value is returned if key is not found
- func intValue(values map[string]interface{}, key string) int64 {
- if val, ok := values[key]; ok {
- if i, ok := val.(int64); ok {
- return i
- }
- }
- return 0
- }
|