1084 lines
32 KiB
Go
1084 lines
32 KiB
Go
package tsdb
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
"sort"
|
|
"time"
|
|
|
|
"github.com/influxdb/influxdb/influxql"
|
|
"github.com/influxdb/influxdb/models"
|
|
"github.com/influxdb/influxdb/services/meta"
|
|
)
|
|
|
|
// QueryExecutor executes every statement in an influxdb Query. It is responsible for
|
|
// coordinating between the local influxql.Store, the meta.Store, and the other nodes in
|
|
// the cluster to run the query against their local tsdb.Stores. There should be one executor
|
|
// in a running process
|
|
type QueryExecutor struct {
|
|
// Local data store.
|
|
Store interface {
|
|
DatabaseIndex(name string) *DatabaseIndex
|
|
Shards(ids []uint64) []*Shard
|
|
ExpandSources(sources influxql.Sources) (influxql.Sources, error)
|
|
DeleteDatabase(name string, shardIDs []uint64) error
|
|
DeleteMeasurement(database, name string) error
|
|
DeleteSeries(database string, seriesKeys []string) error
|
|
}
|
|
|
|
// The meta store for accessing and updating cluster and schema data.
|
|
MetaClient interface {
|
|
Database(name string) (*meta.DatabaseInfo, error)
|
|
Databases() ([]meta.DatabaseInfo, error)
|
|
User(name string) (*meta.UserInfo, error)
|
|
AdminUserExists() bool
|
|
Authenticate(username, password string) (*meta.UserInfo, error)
|
|
RetentionPolicy(database, name string) (rpi *meta.RetentionPolicyInfo, err error)
|
|
UserCount() int
|
|
ShardIDsByTimeRange(sources influxql.Sources, tmin, tmax time.Time) (a []uint64, err error)
|
|
ExecuteStatement(stmt influxql.Statement) *influxql.Result
|
|
}
|
|
|
|
// Execute statements relating to statistics and diagnostics.
|
|
MonitorStatementExecutor interface {
|
|
ExecuteStatement(stmt influxql.Statement) *influxql.Result
|
|
}
|
|
|
|
IntoWriter interface {
|
|
WritePointsInto(p *IntoWriteRequest) error
|
|
}
|
|
|
|
Logger *log.Logger
|
|
QueryLogEnabled bool
|
|
}
|
|
|
|
// partial copy of cluster.WriteRequest
|
|
type IntoWriteRequest struct {
|
|
Database string
|
|
RetentionPolicy string
|
|
Points []models.Point
|
|
}
|
|
|
|
// NewQueryExecutor returns a new instance of QueryExecutor.
|
|
func NewQueryExecutor() *QueryExecutor {
|
|
return &QueryExecutor{
|
|
Logger: log.New(os.Stderr, "[query] ", log.LstdFlags),
|
|
}
|
|
}
|
|
|
|
// SetLogger sets the internal logger to the logger passed in.
|
|
func (q *QueryExecutor) SetLogger(l *log.Logger) {
|
|
q.Logger = l
|
|
}
|
|
|
|
// Authorize user u to execute query q on database.
|
|
// database can be "" for queries that do not require a database.
|
|
// If no user is provided it will return an error unless the query's first statement is to create
|
|
// a root user.
|
|
func (q *QueryExecutor) Authorize(u *meta.UserInfo, query *influxql.Query, database string) error {
|
|
// Special case if no users exist.
|
|
if n := q.MetaClient.UserCount(); n == 0 {
|
|
// Ensure there is at least one statement.
|
|
if len(query.Statements) > 0 {
|
|
// First statement in the query must create a user with admin privilege.
|
|
cu, ok := query.Statements[0].(*influxql.CreateUserStatement)
|
|
if ok && cu.Admin == true {
|
|
return nil
|
|
}
|
|
}
|
|
return NewErrAuthorize(q, query, "", database, "create admin user first or disable authentication")
|
|
}
|
|
|
|
if u == nil {
|
|
return NewErrAuthorize(q, query, "", database, "no user provided")
|
|
}
|
|
|
|
// Admin privilege allows the user to execute all statements.
|
|
if u.Admin {
|
|
return nil
|
|
}
|
|
|
|
// Check each statement in the query.
|
|
for _, stmt := range query.Statements {
|
|
// Get the privileges required to execute the statement.
|
|
privs := stmt.RequiredPrivileges()
|
|
|
|
// Make sure the user has the privileges required to execute
|
|
// each statement.
|
|
for _, p := range privs {
|
|
if p.Admin {
|
|
// Admin privilege already checked so statement requiring admin
|
|
// privilege cannot be run.
|
|
msg := fmt.Sprintf("statement '%s', requires admin privilege", stmt)
|
|
return NewErrAuthorize(q, query, u.Name, database, msg)
|
|
}
|
|
|
|
// Use the db name specified by the statement or the db
|
|
// name passed by the caller if one wasn't specified by
|
|
// the statement.
|
|
db := p.Name
|
|
if db == "" {
|
|
db = database
|
|
}
|
|
if !u.Authorize(p.Privilege, db) {
|
|
msg := fmt.Sprintf("statement '%s', requires %s on %s", stmt, p.Privilege.String(), db)
|
|
return NewErrAuthorize(q, query, u.Name, database, msg)
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ExecuteQuery executes an InfluxQL query against the server.
|
|
// It sends results down the passed in chan and closes it when done. It will close the chan
|
|
// on the first statement that throws an error.
|
|
func (q *QueryExecutor) ExecuteQuery(query *influxql.Query, database string, chunkSize int, closing chan struct{}) (<-chan *influxql.Result, error) {
|
|
// Execute each statement. Keep the iterator external so we can
|
|
// track how many of the statements were executed
|
|
results := make(chan *influxql.Result)
|
|
|
|
go func() {
|
|
defer close(results)
|
|
|
|
var i int
|
|
var stmt influxql.Statement
|
|
for i, stmt = range query.Statements {
|
|
// If a default database wasn't passed in by the caller, check the statement.
|
|
// Some types of statements have an associated default database, even if it
|
|
// is not explicitly included.
|
|
defaultDB := database
|
|
if defaultDB == "" {
|
|
if s, ok := stmt.(influxql.HasDefaultDatabase); ok {
|
|
defaultDB = s.DefaultDatabase()
|
|
}
|
|
}
|
|
|
|
// Normalize each statement.
|
|
if err := q.normalizeStatement(stmt, defaultDB); err != nil {
|
|
results <- &influxql.Result{Err: err}
|
|
break
|
|
}
|
|
|
|
// Log each normalized statement.
|
|
if q.QueryLogEnabled {
|
|
q.Logger.Println(stmt.String())
|
|
}
|
|
|
|
var res *influxql.Result
|
|
switch stmt := stmt.(type) {
|
|
case *influxql.SelectStatement:
|
|
if err := q.executeStatement(i, stmt, database, results, chunkSize, closing); err != nil {
|
|
results <- &influxql.Result{Err: err}
|
|
break
|
|
}
|
|
case *influxql.DropSeriesStatement:
|
|
// TODO: handle this in a cluster
|
|
res = q.executeDropSeriesStatement(stmt, database)
|
|
case *influxql.ShowSeriesStatement:
|
|
res = q.executeShowSeriesStatement(stmt, database)
|
|
case *influxql.DropMeasurementStatement:
|
|
// TODO: handle this in a cluster
|
|
res = q.executeDropMeasurementStatement(stmt, database)
|
|
case *influxql.ShowMeasurementsStatement:
|
|
if err := q.executeStatement(i, stmt, database, results, chunkSize, closing); err != nil {
|
|
results <- &influxql.Result{Err: err}
|
|
break
|
|
}
|
|
case *influxql.ShowTagKeysStatement:
|
|
if err := q.executeStatement(i, stmt, database, results, chunkSize, closing); err != nil {
|
|
results <- &influxql.Result{Err: err}
|
|
break
|
|
}
|
|
case *influxql.ShowTagValuesStatement:
|
|
res = q.executeShowTagValuesStatement(stmt, database)
|
|
case *influxql.ShowFieldKeysStatement:
|
|
res = q.executeShowFieldKeysStatement(stmt, database)
|
|
case *influxql.DeleteStatement:
|
|
res = &influxql.Result{Err: ErrInvalidQuery}
|
|
case *influxql.DropDatabaseStatement:
|
|
// TODO: handle this in a cluster
|
|
res = q.executeDropDatabaseStatement(stmt)
|
|
case *influxql.ShowStatsStatement, *influxql.ShowDiagnosticsStatement:
|
|
// Send monitor-related queries to the monitor service.
|
|
res = q.MonitorStatementExecutor.ExecuteStatement(stmt)
|
|
default:
|
|
// Delegate all other meta statements to a separate executor. They don't hit tsdb storage.
|
|
res = q.MetaClient.ExecuteStatement(stmt)
|
|
}
|
|
|
|
if res != nil {
|
|
// set the StatementID for the handler on the other side to combine results
|
|
res.StatementID = i
|
|
|
|
// If an error occurs then stop processing remaining statements.
|
|
results <- res
|
|
if res.Err != nil {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
// if there was an error send results that the remaining statements weren't executed
|
|
for ; i < len(query.Statements)-1; i++ {
|
|
results <- &influxql.Result{Err: ErrNotExecuted}
|
|
}
|
|
}()
|
|
|
|
return results, nil
|
|
}
|
|
|
|
// PlanSelect creates an execution plan for the given SelectStatement and returns an Executor.
|
|
func (q *QueryExecutor) PlanSelect(stmt *influxql.SelectStatement, chunkSize int) (Executor, error) {
|
|
// It is important to "stamp" this time so that everywhere we evaluate `now()` in the statement is EXACTLY the same `now`
|
|
now := time.Now().UTC()
|
|
|
|
// Replace instances of "now()" with the current time, and check the resultant times.
|
|
stmt.Condition = influxql.Reduce(stmt.Condition, &influxql.NowValuer{Now: now})
|
|
tmin, tmax := influxql.TimeRange(stmt.Condition)
|
|
if tmax.IsZero() {
|
|
tmax = now
|
|
}
|
|
if tmin.IsZero() {
|
|
tmin = time.Unix(0, 0)
|
|
}
|
|
|
|
// Expand regex sources to their actual source names.
|
|
sources, err := q.Store.ExpandSources(stmt.Sources)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
stmt.Sources = sources
|
|
|
|
// Convert DISTINCT into a call.
|
|
stmt.RewriteDistinct()
|
|
|
|
// Remove "time" from fields list.
|
|
stmt.RewriteTimeFields()
|
|
|
|
// Filter only shards that contain date range.
|
|
shardIDs, err := q.MetaClient.ShardIDsByTimeRange(stmt.Sources, tmin, tmax)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
shards := q.Store.Shards(shardIDs)
|
|
|
|
// Rewrite wildcards, if any exist.
|
|
tmp, err := stmt.RewriteWildcards(Shards(shards))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
stmt = tmp
|
|
|
|
// Create a set of iterators from a selection.
|
|
itrs, err := influxql.Select(stmt, Shards(shards))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Generate a row emitter from the iterator set.
|
|
em := influxql.NewEmitter(itrs, stmt.TimeAscending())
|
|
em.Columns = stmt.ColumnNames()
|
|
em.OmitTime = stmt.OmitTime
|
|
|
|
// Wrap emitter in an adapter to conform to the Executor interface.
|
|
return (*emitterExecutor)(em), nil
|
|
}
|
|
|
|
// executeDropDatabaseStatement closes all local shards for the database and removes the directory. It then calls to the metastore to remove the database from there.
|
|
// TODO: make this work in a cluster/distributed
|
|
func (q *QueryExecutor) executeDropDatabaseStatement(stmt *influxql.DropDatabaseStatement) *influxql.Result {
|
|
dbi, err := q.MetaClient.Database(stmt.Name)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
} else if dbi == nil {
|
|
if stmt.IfExists {
|
|
return &influxql.Result{}
|
|
}
|
|
return &influxql.Result{Err: ErrDatabaseNotFound(stmt.Name)}
|
|
}
|
|
|
|
var shardIDs []uint64
|
|
for _, rp := range dbi.RetentionPolicies {
|
|
for _, sg := range rp.ShardGroups {
|
|
for _, s := range sg.Shards {
|
|
shardIDs = append(shardIDs, s.ID)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Remove database from meta-store first so that in-flight writes can complete without error, but new ones will
|
|
// be rejected.
|
|
res := q.MetaClient.ExecuteStatement(stmt)
|
|
|
|
// Remove the database from the local store
|
|
err = q.Store.DeleteDatabase(stmt.Name, shardIDs)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
|
|
return res
|
|
}
|
|
|
|
// executeDropMeasurementStatement removes the measurement and all series data from the local store for the given measurement
|
|
func (q *QueryExecutor) executeDropMeasurementStatement(stmt *influxql.DropMeasurementStatement, database string) *influxql.Result {
|
|
if err := q.Store.DeleteMeasurement(database, stmt.Name); err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
return &influxql.Result{}
|
|
}
|
|
|
|
// executeDropSeriesStatement removes all series from the local store that match the drop query
|
|
func (q *QueryExecutor) executeDropSeriesStatement(stmt *influxql.DropSeriesStatement, database string) *influxql.Result {
|
|
// Check for time in WHERE clause (not supported).
|
|
if influxql.HasTimeExpr(stmt.Condition) {
|
|
return &influxql.Result{Err: errors.New("DROP SERIES doesn't support time in WHERE clause")}
|
|
}
|
|
|
|
// Find the database.
|
|
db := q.Store.DatabaseIndex(database)
|
|
if db == nil {
|
|
return &influxql.Result{}
|
|
}
|
|
|
|
// Expand regex expressions in the FROM clause.
|
|
sources, err := q.Store.ExpandSources(stmt.Sources)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
} else if stmt.Sources != nil && len(stmt.Sources) != 0 && len(sources) == 0 {
|
|
return &influxql.Result{}
|
|
}
|
|
|
|
measurements, err := measurementsFromSourcesOrDB(db, sources...)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
|
|
var seriesKeys []string
|
|
for _, m := range measurements {
|
|
var ids SeriesIDs
|
|
var filters FilterExprs
|
|
if stmt.Condition != nil {
|
|
// Get series IDs that match the WHERE clause.
|
|
ids, filters, err = m.walkWhereForSeriesIds(stmt.Condition)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
|
|
// Delete boolean literal true filter expressions.
|
|
// These are returned for `WHERE tagKey = 'tagVal'` type expressions and are okay.
|
|
filters.DeleteBoolLiteralTrues()
|
|
|
|
// Check for unsupported field filters.
|
|
// Any remaining filters means there were fields (e.g., `WHERE value = 1.2`).
|
|
if filters.Len() > 0 {
|
|
return &influxql.Result{Err: errors.New("DROP SERIES doesn't support fields in WHERE clause")}
|
|
}
|
|
} else {
|
|
// No WHERE clause so get all series IDs for this measurement.
|
|
ids = m.seriesIDs
|
|
}
|
|
|
|
for _, id := range ids {
|
|
seriesKeys = append(seriesKeys, m.seriesByID[id].Key)
|
|
}
|
|
}
|
|
|
|
// delete the raw series data
|
|
if err := q.Store.DeleteSeries(database, seriesKeys); err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
// remove them from the index
|
|
db.DropSeries(seriesKeys)
|
|
|
|
return &influxql.Result{}
|
|
}
|
|
|
|
func (q *QueryExecutor) executeShowSeriesStatement(stmt *influxql.ShowSeriesStatement, database string) *influxql.Result {
|
|
// Check for time in WHERE clause (not supported).
|
|
if influxql.HasTimeExpr(stmt.Condition) {
|
|
return &influxql.Result{Err: errors.New("SHOW SERIES doesn't support time in WHERE clause")}
|
|
}
|
|
|
|
// Find the database.
|
|
db := q.Store.DatabaseIndex(database)
|
|
if db == nil {
|
|
return &influxql.Result{}
|
|
}
|
|
|
|
// Expand regex expressions in the FROM clause.
|
|
sources, err := q.Store.ExpandSources(stmt.Sources)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
|
|
// Get the list of measurements we're interested in.
|
|
measurements, err := measurementsFromSourcesOrDB(db, sources...)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
|
|
// Create result struct that will be populated and returned.
|
|
result := &influxql.Result{
|
|
Series: make(models.Rows, 0, len(measurements)),
|
|
}
|
|
|
|
// Loop through measurements to build result. One result row / measurement.
|
|
for _, m := range measurements {
|
|
var ids SeriesIDs
|
|
var filters FilterExprs
|
|
|
|
if stmt.Condition != nil {
|
|
// Get series IDs that match the WHERE clause.
|
|
ids, filters, err = m.walkWhereForSeriesIds(stmt.Condition)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
|
|
// Delete boolean literal true filter expressions.
|
|
filters.DeleteBoolLiteralTrues()
|
|
|
|
// Check for unsupported field filters.
|
|
if filters.Len() > 0 {
|
|
return &influxql.Result{Err: errors.New("SHOW SERIES doesn't support fields in WHERE clause")}
|
|
}
|
|
|
|
// If no series matched, then go to the next measurement.
|
|
if len(ids) == 0 {
|
|
continue
|
|
}
|
|
} else {
|
|
// No WHERE clause so get all series IDs for this measurement.
|
|
ids = m.seriesIDs
|
|
}
|
|
|
|
// Make a new row for this measurement.
|
|
r := &models.Row{
|
|
Name: m.Name,
|
|
Columns: m.TagKeys(),
|
|
}
|
|
|
|
// Loop through series IDs getting matching tag sets.
|
|
for _, id := range ids {
|
|
if s, ok := m.seriesByID[id]; ok {
|
|
values := make([]interface{}, 0, len(r.Columns))
|
|
|
|
// make the series key the first value
|
|
values = append(values, s.Key)
|
|
|
|
for _, column := range r.Columns {
|
|
values = append(values, s.Tags[column])
|
|
}
|
|
|
|
// Add the tag values to the row.
|
|
r.Values = append(r.Values, values)
|
|
}
|
|
}
|
|
// make the id the first column
|
|
r.Columns = append([]string{"_key"}, r.Columns...)
|
|
|
|
// Append the row to the result.
|
|
result.Series = append(result.Series, r)
|
|
}
|
|
|
|
if stmt.Limit > 0 || stmt.Offset > 0 {
|
|
result.Series = q.filterShowSeriesResult(stmt.Limit, stmt.Offset, result.Series)
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
// filterShowSeriesResult will limit the number of series returned based on the limit and the offset.
|
|
// Unlike limit and offset on SELECT statements, the limit and offset don't apply to the number of Rows, but
|
|
// to the number of total Values returned, since each Value represents a unique series.
|
|
func (q *QueryExecutor) filterShowSeriesResult(limit, offset int, rows models.Rows) models.Rows {
|
|
var filteredSeries models.Rows
|
|
seriesCount := 0
|
|
for _, r := range rows {
|
|
var currentSeries [][]interface{}
|
|
|
|
// filter the values
|
|
for _, v := range r.Values {
|
|
if seriesCount >= offset && seriesCount-offset < limit {
|
|
currentSeries = append(currentSeries, v)
|
|
}
|
|
seriesCount++
|
|
}
|
|
|
|
// only add the row back in if there are some values in it
|
|
if len(currentSeries) > 0 {
|
|
r.Values = currentSeries
|
|
filteredSeries = append(filteredSeries, r)
|
|
if seriesCount > limit+offset {
|
|
return filteredSeries
|
|
}
|
|
}
|
|
}
|
|
return filteredSeries
|
|
}
|
|
|
|
func (q *QueryExecutor) planStatement(stmt influxql.Statement, database string, chunkSize int) (Executor, error) {
|
|
switch stmt := stmt.(type) {
|
|
case *influxql.SelectStatement:
|
|
return q.PlanSelect(stmt, chunkSize)
|
|
case *influxql.ShowMeasurementsStatement:
|
|
return q.planShowMeasurements(stmt, database, chunkSize)
|
|
case *influxql.ShowTagKeysStatement:
|
|
return q.planShowTagKeys(stmt, database, chunkSize)
|
|
default:
|
|
return nil, fmt.Errorf("can't plan statement type: %v", stmt)
|
|
}
|
|
}
|
|
|
|
// planShowMeasurements converts the statement to a SELECT and executes it.
|
|
func (q *QueryExecutor) planShowMeasurements(stmt *influxql.ShowMeasurementsStatement, database string, chunkSize int) (Executor, error) {
|
|
// Check for time in WHERE clause (not supported).
|
|
if influxql.HasTimeExpr(stmt.Condition) {
|
|
return nil, errors.New("SHOW MEASUREMENTS doesn't support time in WHERE clause")
|
|
}
|
|
|
|
condition := stmt.Condition
|
|
if source, ok := stmt.Source.(*influxql.Measurement); ok {
|
|
var expr influxql.Expr
|
|
if source.Regex != nil {
|
|
expr = &influxql.BinaryExpr{
|
|
Op: influxql.EQREGEX,
|
|
LHS: &influxql.VarRef{Val: "name"},
|
|
RHS: &influxql.RegexLiteral{Val: source.Regex.Val},
|
|
}
|
|
} else if source.Name != "" {
|
|
expr = &influxql.BinaryExpr{
|
|
Op: influxql.EQ,
|
|
LHS: &influxql.VarRef{Val: "name"},
|
|
RHS: &influxql.StringLiteral{Val: source.Name},
|
|
}
|
|
}
|
|
|
|
// Set condition or "AND" together.
|
|
if condition == nil {
|
|
condition = expr
|
|
} else {
|
|
condition = &influxql.BinaryExpr{Op: influxql.AND, LHS: expr, RHS: condition}
|
|
}
|
|
}
|
|
|
|
ss := &influxql.SelectStatement{
|
|
Fields: influxql.Fields{
|
|
{Expr: &influxql.VarRef{Val: "name"}},
|
|
},
|
|
Sources: influxql.Sources{
|
|
&influxql.Measurement{Name: "_measurements"},
|
|
},
|
|
Condition: condition,
|
|
Offset: stmt.Offset,
|
|
Limit: stmt.Limit,
|
|
SortFields: stmt.SortFields,
|
|
OmitTime: true,
|
|
Dedupe: true,
|
|
}
|
|
|
|
// Normalize the statement.
|
|
if err := q.normalizeStatement(ss, database); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return q.PlanSelect(ss, chunkSize)
|
|
}
|
|
|
|
// planShowTagKeys creates an execution plan for a SHOW MEASUREMENTS statement and returns an Executor.
|
|
func (q *QueryExecutor) planShowTagKeys(stmt *influxql.ShowTagKeysStatement, database string, chunkSize int) (Executor, error) {
|
|
// Check for time in WHERE clause (not supported).
|
|
if influxql.HasTimeExpr(stmt.Condition) {
|
|
return nil, errors.New("SHOW TAG KEYS doesn't support time in WHERE clause")
|
|
}
|
|
|
|
condition := stmt.Condition
|
|
if len(stmt.Sources) > 0 {
|
|
if source, ok := stmt.Sources[0].(*influxql.Measurement); ok {
|
|
var expr influxql.Expr
|
|
if source.Regex != nil {
|
|
expr = &influxql.BinaryExpr{
|
|
Op: influxql.EQREGEX,
|
|
LHS: &influxql.VarRef{Val: "name"},
|
|
RHS: &influxql.RegexLiteral{Val: source.Regex.Val},
|
|
}
|
|
} else if source.Name != "" {
|
|
expr = &influxql.BinaryExpr{
|
|
Op: influxql.EQ,
|
|
LHS: &influxql.VarRef{Val: "name"},
|
|
RHS: &influxql.StringLiteral{Val: source.Name},
|
|
}
|
|
}
|
|
|
|
// Set condition or "AND" together.
|
|
if condition == nil {
|
|
condition = expr
|
|
} else {
|
|
condition = &influxql.BinaryExpr{Op: influxql.AND, LHS: expr, RHS: condition}
|
|
}
|
|
}
|
|
}
|
|
|
|
ss := &influxql.SelectStatement{
|
|
Fields: influxql.Fields{
|
|
{Expr: &influxql.VarRef{Val: "tagKey"}},
|
|
},
|
|
Sources: influxql.Sources{
|
|
&influxql.Measurement{Name: "_tagKeys"},
|
|
},
|
|
Condition: condition,
|
|
Offset: stmt.Offset,
|
|
Limit: stmt.Limit,
|
|
SortFields: stmt.SortFields,
|
|
OmitTime: true,
|
|
Dedupe: true,
|
|
}
|
|
|
|
// Normalize the statement.
|
|
if err := q.normalizeStatement(ss, database); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return q.PlanSelect(ss, chunkSize)
|
|
}
|
|
|
|
func (q *QueryExecutor) executeStatement(statementID int, stmt influxql.Statement, database string, results chan *influxql.Result, chunkSize int, closing chan struct{}) error {
|
|
// Plan statement execution.
|
|
e, err := q.planStatement(stmt, database, chunkSize)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Execute plan.
|
|
ch := e.Execute(closing)
|
|
var writeerr error
|
|
var intoNum int64
|
|
var isinto bool
|
|
// Stream results from the channel. We should send an empty result if nothing comes through.
|
|
resultSent := false
|
|
for row := range ch {
|
|
// We had a write error. Continue draining results from the channel
|
|
// so we don't hang the goroutine in the executor.
|
|
if writeerr != nil {
|
|
continue
|
|
}
|
|
if row.Err != nil {
|
|
return row.Err
|
|
}
|
|
selectstmt, ok := stmt.(*influxql.SelectStatement)
|
|
if ok && selectstmt.Target != nil {
|
|
isinto = true
|
|
// this is a into query. Write results back to database
|
|
writeerr = q.writeInto(row, selectstmt)
|
|
intoNum += int64(len(row.Values))
|
|
} else {
|
|
resultSent = true
|
|
results <- &influxql.Result{StatementID: statementID, Series: []*models.Row{row}}
|
|
}
|
|
}
|
|
if writeerr != nil {
|
|
return writeerr
|
|
} else if isinto {
|
|
results <- &influxql.Result{
|
|
StatementID: statementID,
|
|
Series: []*models.Row{{
|
|
Name: "result",
|
|
// it seems weird to give a time here, but so much stuff breaks if you don't
|
|
Columns: []string{"time", "written"},
|
|
Values: [][]interface{}{{
|
|
time.Unix(0, 0).UTC(),
|
|
intoNum,
|
|
}},
|
|
}},
|
|
}
|
|
return nil
|
|
}
|
|
|
|
if !resultSent {
|
|
results <- &influxql.Result{StatementID: statementID, Series: make([]*models.Row, 0)}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (q *QueryExecutor) writeInto(row *models.Row, selectstmt *influxql.SelectStatement) error {
|
|
// It might seem a bit weird that this is where we do this, since we will have to
|
|
// convert rows back to points. The Executors (both aggregate and raw) are complex
|
|
// enough that changing them to write back to the DB is going to be clumsy
|
|
//
|
|
// it might seem weird to have the write be in the QueryExecutor, but the interweaving of
|
|
// limitedRowWriter and ExecuteAggregate/Raw makes it ridiculously hard to make sure that the
|
|
// results will be the same as when queried normally.
|
|
measurement := intoMeasurement(selectstmt)
|
|
if measurement == "" {
|
|
measurement = row.Name
|
|
}
|
|
intodb, err := intoDB(selectstmt)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
rp := intoRP(selectstmt)
|
|
points, err := convertRowToPoints(measurement, row)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
req := &IntoWriteRequest{
|
|
Database: intodb,
|
|
RetentionPolicy: rp,
|
|
Points: points,
|
|
}
|
|
err = q.IntoWriter.WritePointsInto(req)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (q *QueryExecutor) executeShowTagValuesStatement(stmt *influxql.ShowTagValuesStatement, database string) *influxql.Result {
|
|
// Check for time in WHERE clause (not supported).
|
|
if influxql.HasTimeExpr(stmt.Condition) {
|
|
return &influxql.Result{Err: errors.New("SHOW TAG VALUES doesn't support time in WHERE clause")}
|
|
}
|
|
|
|
// Find the database.
|
|
db := q.Store.DatabaseIndex(database)
|
|
if db == nil {
|
|
return &influxql.Result{}
|
|
}
|
|
|
|
// Expand regex expressions in the FROM clause.
|
|
sources, err := q.Store.ExpandSources(stmt.Sources)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
|
|
// Get the list of measurements we're interested in.
|
|
measurements, err := measurementsFromSourcesOrDB(db, sources...)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
|
|
// Make result.
|
|
result := &influxql.Result{
|
|
Series: make(models.Rows, 0),
|
|
}
|
|
|
|
tagValues := make(map[string]stringSet)
|
|
for _, m := range measurements {
|
|
var ids SeriesIDs
|
|
|
|
if stmt.Condition != nil {
|
|
// Get series IDs that match the WHERE clause.
|
|
ids, _, err = m.walkWhereForSeriesIds(stmt.Condition)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
|
|
// If no series matched, then go to the next measurement.
|
|
if len(ids) == 0 {
|
|
continue
|
|
}
|
|
|
|
// TODO: check return of walkWhereForSeriesIds for fields
|
|
} else {
|
|
// No WHERE clause so get all series IDs for this measurement.
|
|
ids = m.seriesIDs
|
|
}
|
|
|
|
for k, v := range m.tagValuesByKeyAndSeriesID(stmt.TagKeys, ids) {
|
|
_, ok := tagValues[k]
|
|
if !ok {
|
|
tagValues[k] = v
|
|
}
|
|
tagValues[k] = tagValues[k].union(v)
|
|
}
|
|
}
|
|
|
|
for k, v := range tagValues {
|
|
r := &models.Row{
|
|
Name: k + "TagValues",
|
|
Columns: []string{k},
|
|
}
|
|
|
|
vals := v.list()
|
|
sort.Strings(vals)
|
|
|
|
for _, val := range vals {
|
|
v := interface{}(val)
|
|
r.Values = append(r.Values, []interface{}{v})
|
|
}
|
|
|
|
result.Series = append(result.Series, r)
|
|
}
|
|
|
|
sort.Sort(result.Series)
|
|
return result
|
|
}
|
|
|
|
func (q *QueryExecutor) executeShowFieldKeysStatement(stmt *influxql.ShowFieldKeysStatement, database string) *influxql.Result {
|
|
var err error
|
|
|
|
// Find the database.
|
|
db := q.Store.DatabaseIndex(database)
|
|
if db == nil {
|
|
return &influxql.Result{}
|
|
}
|
|
|
|
// Expand regex expressions in the FROM clause.
|
|
sources, err := q.Store.ExpandSources(stmt.Sources)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
|
|
measurements, err := measurementsFromSourcesOrDB(db, sources...)
|
|
if err != nil {
|
|
return &influxql.Result{Err: err}
|
|
}
|
|
|
|
// Make result.
|
|
result := &influxql.Result{
|
|
Series: make(models.Rows, 0, len(measurements)),
|
|
}
|
|
|
|
// Loop through measurements, adding a result row for each.
|
|
for _, m := range measurements {
|
|
// Create a new row.
|
|
r := &models.Row{
|
|
Name: m.Name,
|
|
Columns: []string{"fieldKey"},
|
|
}
|
|
|
|
// Get a list of field names from the measurement then sort them.
|
|
names := m.FieldNames()
|
|
sort.Strings(names)
|
|
|
|
// Add the field names to the result row values.
|
|
for _, n := range names {
|
|
v := interface{}(n)
|
|
r.Values = append(r.Values, []interface{}{v})
|
|
}
|
|
|
|
// Append the row to the result.
|
|
result.Series = append(result.Series, r)
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
// measurementsFromSourcesOrDB returns a list of measurements from the
|
|
// sources passed in or, if sources is empty, a list of all
|
|
// measurement names from the database passed in.
|
|
func measurementsFromSourcesOrDB(db *DatabaseIndex, sources ...influxql.Source) (Measurements, error) {
|
|
var measurements Measurements
|
|
if len(sources) > 0 {
|
|
for _, source := range sources {
|
|
if m, ok := source.(*influxql.Measurement); ok {
|
|
measurement := db.measurements[m.Name]
|
|
if measurement == nil {
|
|
continue
|
|
}
|
|
|
|
measurements = append(measurements, measurement)
|
|
} else {
|
|
return nil, errors.New("identifiers in FROM clause must be measurement names")
|
|
}
|
|
}
|
|
} else {
|
|
// No measurements specified in FROM clause so get all measurements that have series.
|
|
for _, m := range db.Measurements() {
|
|
if m.HasSeries() {
|
|
measurements = append(measurements, m)
|
|
}
|
|
}
|
|
}
|
|
sort.Sort(measurements)
|
|
|
|
return measurements, nil
|
|
}
|
|
|
|
// normalizeStatement adds a default database and policy to the measurements in statement.
|
|
func (q *QueryExecutor) normalizeStatement(stmt influxql.Statement, defaultDatabase string) (err error) {
|
|
// Track prefixes for replacing field names.
|
|
prefixes := make(map[string]string)
|
|
|
|
// Qualify all measurements.
|
|
influxql.WalkFunc(stmt, func(n influxql.Node) {
|
|
if err != nil {
|
|
return
|
|
}
|
|
switch n := n.(type) {
|
|
case *influxql.Measurement:
|
|
e := q.normalizeMeasurement(n, defaultDatabase)
|
|
if e != nil {
|
|
err = e
|
|
return
|
|
}
|
|
prefixes[n.Name] = n.Name
|
|
}
|
|
})
|
|
return
|
|
}
|
|
|
|
// normalizeMeasurement inserts the default database or policy into all measurement names,
|
|
// if required.
|
|
func (q *QueryExecutor) normalizeMeasurement(m *influxql.Measurement, defaultDatabase string) error {
|
|
// Targets (measurements in an INTO clause) can have blank names, which means it will be
|
|
// the same as the measurement name it came from in the FROM clause.
|
|
if !m.IsTarget && m.Name == "" && m.Regex == nil {
|
|
return errors.New("invalid measurement")
|
|
}
|
|
|
|
// Measurement does not have an explicit database? Insert default.
|
|
if m.Database == "" {
|
|
m.Database = defaultDatabase
|
|
}
|
|
|
|
// The database must now be specified by this point.
|
|
if m.Database == "" {
|
|
return errors.New("database name required")
|
|
}
|
|
|
|
// Find database.
|
|
di, err := q.MetaClient.Database(m.Database)
|
|
if err != nil {
|
|
return err
|
|
} else if di == nil {
|
|
return ErrDatabaseNotFound(m.Database)
|
|
}
|
|
|
|
// If no retention policy was specified, use the default.
|
|
if m.RetentionPolicy == "" {
|
|
if di.DefaultRetentionPolicy == "" {
|
|
return fmt.Errorf("default retention policy not set for: %s", di.Name)
|
|
}
|
|
m.RetentionPolicy = di.DefaultRetentionPolicy
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// ErrAuthorize represents an authorization error.
|
|
type ErrAuthorize struct {
|
|
q *QueryExecutor
|
|
query *influxql.Query
|
|
user string
|
|
database string
|
|
message string
|
|
}
|
|
|
|
const authErrLogFmt string = "unauthorized request | user: %q | query: %q | database %q\n"
|
|
|
|
// newAuthorizationError returns a new instance of AuthorizationError.
|
|
func NewErrAuthorize(qe *QueryExecutor, q *influxql.Query, u, db, m string) *ErrAuthorize {
|
|
return &ErrAuthorize{q: qe, query: q, user: u, database: db, message: m}
|
|
}
|
|
|
|
// Error returns the text of the error.
|
|
func (e ErrAuthorize) Error() string {
|
|
e.q.Logger.Printf(authErrLogFmt, e.user, e.query.String(), e.database)
|
|
if e.user == "" {
|
|
return fmt.Sprint(e.message)
|
|
}
|
|
return fmt.Sprintf("%s not authorized to execute %s", e.user, e.message)
|
|
}
|
|
|
|
var (
|
|
// ErrInvalidQuery is returned when executing an unknown query type.
|
|
ErrInvalidQuery = errors.New("invalid query")
|
|
|
|
// ErrNotExecuted is returned when a statement is not executed in a query.
|
|
// This can occur when a previous statement in the same query has errored.
|
|
ErrNotExecuted = errors.New("not executed")
|
|
)
|
|
|
|
func ErrDatabaseNotFound(name string) error { return fmt.Errorf("database not found: %s", name) }
|
|
|
|
func ErrMeasurementNotFound(name string) error { return fmt.Errorf("measurement not found: %s", name) }
|
|
|
|
type uint64Slice []uint64
|
|
|
|
func (a uint64Slice) Len() int { return len(a) }
|
|
func (a uint64Slice) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
|
func (a uint64Slice) Less(i, j int) bool { return a[i] < a[j] }
|
|
|
|
// convertRowToPoints will convert a query result Row into Points that can be written back in.
|
|
// Used for INTO queries
|
|
func convertRowToPoints(measurementName string, row *models.Row) ([]models.Point, error) {
|
|
// figure out which parts of the result are the time and which are the fields
|
|
timeIndex := -1
|
|
fieldIndexes := make(map[string]int)
|
|
for i, c := range row.Columns {
|
|
if c == "time" {
|
|
timeIndex = i
|
|
} else {
|
|
fieldIndexes[c] = i
|
|
}
|
|
}
|
|
|
|
if timeIndex == -1 {
|
|
return nil, errors.New("error finding time index in result")
|
|
}
|
|
|
|
points := make([]models.Point, 0, len(row.Values))
|
|
for _, v := range row.Values {
|
|
vals := make(map[string]interface{})
|
|
for fieldName, fieldIndex := range fieldIndexes {
|
|
val := v[fieldIndex]
|
|
if val != nil {
|
|
vals[fieldName] = v[fieldIndex]
|
|
}
|
|
}
|
|
|
|
p, err := models.NewPoint(measurementName, row.Tags, vals, v[timeIndex].(time.Time))
|
|
if err != nil {
|
|
// Drop points that can't be stored
|
|
continue
|
|
}
|
|
|
|
points = append(points, p)
|
|
}
|
|
|
|
return points, nil
|
|
}
|
|
|
|
func intoDB(stmt *influxql.SelectStatement) (string, error) {
|
|
if stmt.Target.Measurement.Database != "" {
|
|
return stmt.Target.Measurement.Database, nil
|
|
}
|
|
return "", errNoDatabaseInTarget
|
|
}
|
|
|
|
var errNoDatabaseInTarget = errors.New("no database in target")
|
|
|
|
func intoRP(stmt *influxql.SelectStatement) string { return stmt.Target.Measurement.RetentionPolicy }
|
|
func intoMeasurement(stmt *influxql.SelectStatement) string { return stmt.Target.Measurement.Name }
|
|
|
|
// emitterExecutor represents an adapter for emitters to implement Executor.
|
|
type emitterExecutor influxql.Emitter
|
|
|
|
func (e *emitterExecutor) Execute(closing <-chan struct{}) <-chan *models.Row {
|
|
out := make(chan *models.Row, 0)
|
|
go e.execute(out, closing)
|
|
return out
|
|
}
|
|
|
|
func (e *emitterExecutor) execute(out chan *models.Row, closing <-chan struct{}) {
|
|
defer close(out)
|
|
|
|
// Continually read rows from emitter until no more are available.
|
|
em := (*influxql.Emitter)(e)
|
|
for {
|
|
row := em.Emit()
|
|
if row == nil {
|
|
break
|
|
}
|
|
|
|
select {
|
|
case <-closing:
|
|
break
|
|
case out <- row:
|
|
}
|
|
}
|
|
}
|