148 lines
3.4 KiB
Go
148 lines
3.4 KiB
Go
package storage
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
|
|
"github.com/influxdata/influxdb/v2"
|
|
"github.com/influxdata/influxdb/v2/models"
|
|
"github.com/influxdata/influxdb/v2/pkg/lifecycle"
|
|
"github.com/influxdata/influxdb/v2/tsdb"
|
|
"github.com/influxdata/influxdb/v2/tsdb/seriesfile"
|
|
"github.com/influxdata/influxdb/v2/tsdb/tsi1"
|
|
"github.com/influxdata/influxql"
|
|
)
|
|
|
|
var (
|
|
errUnexpectedOrg = errors.New("seriesCursor: unexpected org")
|
|
errUnexpectedTagComparisonOperator = errors.New("seriesCursor: unexpected tag comparison operator")
|
|
)
|
|
|
|
type SeriesCursor interface {
|
|
Close()
|
|
Next() (*SeriesCursorRow, error)
|
|
}
|
|
|
|
// seriesCursor is an implementation of SeriesCursor over an tsi1.Index.
|
|
type seriesCursor struct {
|
|
index *tsi1.Index
|
|
indexref *lifecycle.Reference
|
|
sfile *seriesfile.SeriesFile
|
|
sfileref *lifecycle.Reference
|
|
orgID influxdb.ID
|
|
encodedOrgID []byte
|
|
bucketID influxdb.ID
|
|
keys [][]byte
|
|
ofs int
|
|
row SeriesCursorRow
|
|
cond influxql.Expr
|
|
init bool
|
|
}
|
|
|
|
type SeriesCursorRow struct {
|
|
Name []byte
|
|
Tags models.Tags
|
|
}
|
|
|
|
// newSeriesCursor returns a new instance of SeriesCursor.
|
|
func newSeriesCursor(orgID, bucketID influxdb.ID, index *tsi1.Index, sfile *seriesfile.SeriesFile, cond influxql.Expr) (SeriesCursor, error) {
|
|
if cond != nil {
|
|
var err error
|
|
influxql.WalkFunc(cond, func(node influxql.Node) {
|
|
switch n := node.(type) {
|
|
case *influxql.BinaryExpr:
|
|
switch n.Op {
|
|
case influxql.EQ, influxql.NEQ, influxql.EQREGEX, influxql.NEQREGEX, influxql.OR, influxql.AND:
|
|
default:
|
|
err = errUnexpectedTagComparisonOperator
|
|
}
|
|
}
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
indexref, err := index.Acquire()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
sfileref, err := sfile.Acquire()
|
|
if err != nil {
|
|
indexref.Release()
|
|
return nil, err
|
|
}
|
|
|
|
encodedOrgID := tsdb.EncodeOrgName(orgID)
|
|
return &seriesCursor{
|
|
index: index,
|
|
indexref: indexref,
|
|
sfile: sfile,
|
|
sfileref: sfileref,
|
|
orgID: orgID,
|
|
encodedOrgID: encodedOrgID[:],
|
|
bucketID: bucketID,
|
|
cond: cond,
|
|
}, nil
|
|
}
|
|
|
|
// Close closes the iterator. Safe to call multiple times.
|
|
func (cur *seriesCursor) Close() {
|
|
cur.sfileref.Release()
|
|
cur.indexref.Release()
|
|
}
|
|
|
|
// Next emits the next point in the iterator.
|
|
func (cur *seriesCursor) Next() (*SeriesCursorRow, error) {
|
|
if !cur.init {
|
|
if err := cur.readSeriesKeys(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Release before Close(), to hold the resources for as little time as possible.
|
|
cur.sfileref.Release()
|
|
cur.indexref.Release()
|
|
|
|
cur.init = true
|
|
}
|
|
|
|
if cur.ofs < len(cur.keys) {
|
|
cur.row.Name, cur.row.Tags = seriesfile.ParseSeriesKeyInto(cur.keys[cur.ofs], cur.row.Tags)
|
|
if !bytes.HasPrefix(cur.row.Name, cur.encodedOrgID) {
|
|
return nil, errUnexpectedOrg
|
|
}
|
|
cur.ofs++
|
|
return &cur.row, nil
|
|
}
|
|
|
|
return nil, nil
|
|
}
|
|
|
|
func (cur *seriesCursor) readSeriesKeys() error {
|
|
name := tsdb.EncodeName(cur.orgID, cur.bucketID)
|
|
sitr, err := cur.index.MeasurementSeriesByExprIterator(name[:], cur.cond)
|
|
if err != nil {
|
|
return err
|
|
} else if sitr == nil {
|
|
return nil
|
|
}
|
|
defer sitr.Close()
|
|
|
|
for {
|
|
elem, err := sitr.Next()
|
|
if err != nil {
|
|
return err
|
|
} else if elem.SeriesID.IsZero() {
|
|
break
|
|
}
|
|
|
|
key := cur.sfile.SeriesKey(elem.SeriesID)
|
|
if len(key) == 0 {
|
|
continue
|
|
}
|
|
cur.keys = append(cur.keys, key)
|
|
}
|
|
|
|
return nil
|
|
}
|