mirror of
https://github.com/grafana/grafana.git
synced 2024-11-23 01:16:31 -06:00
140 lines
3.2 KiB
Go
140 lines
3.2 KiB
Go
package expr
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/grafana/grafana-plugin-sdk-go/backend"
|
|
|
|
"github.com/grafana/grafana/pkg/apimachinery/identity"
|
|
"github.com/grafana/grafana/pkg/services/datasources"
|
|
)
|
|
|
|
// Request is similar to plugins.DataQuery but with the Time Ranges is per Query.
|
|
type Request struct {
|
|
Headers map[string]string
|
|
Debug bool
|
|
OrgId int64
|
|
Queries []Query
|
|
User identity.Requester
|
|
}
|
|
|
|
// Query is like plugins.DataSubQuery, but with a a time range, and only the UID
|
|
// for the data source. Also interval is a time.Duration.
|
|
type Query struct {
|
|
RefID string
|
|
TimeRange TimeRange
|
|
DataSource *datasources.DataSource `json:"datasource"`
|
|
JSON json.RawMessage
|
|
Interval time.Duration
|
|
QueryType string
|
|
MaxDataPoints int64
|
|
}
|
|
|
|
// TimeRange is a time.Time based TimeRange.
|
|
type TimeRange interface {
|
|
AbsoluteTime(now time.Time) backend.TimeRange
|
|
}
|
|
|
|
type AbsoluteTimeRange struct {
|
|
From time.Time
|
|
To time.Time
|
|
}
|
|
|
|
func (r AbsoluteTimeRange) AbsoluteTime(_ time.Time) backend.TimeRange {
|
|
return backend.TimeRange{
|
|
From: r.From,
|
|
To: r.To,
|
|
}
|
|
}
|
|
|
|
// RelativeTimeRange is a time range relative to some absolute time.
|
|
type RelativeTimeRange struct {
|
|
From time.Duration
|
|
To time.Duration
|
|
}
|
|
|
|
func (r RelativeTimeRange) AbsoluteTime(t time.Time) backend.TimeRange {
|
|
return backend.TimeRange{
|
|
From: t.Add(r.From),
|
|
To: t.Add(r.To),
|
|
}
|
|
}
|
|
|
|
// TransformData takes Queries which are either expressions nodes
|
|
// or are datasource requests.
|
|
func (s *Service) TransformData(ctx context.Context, now time.Time, req *Request) (r *backend.QueryDataResponse, err error) {
|
|
if s.isDisabled() {
|
|
return nil, fmt.Errorf("server side expressions are disabled")
|
|
}
|
|
|
|
start := time.Now()
|
|
ctx, span := s.tracer.Start(ctx, "SSE.TransformData")
|
|
defer func() {
|
|
var respStatus string
|
|
switch {
|
|
case err == nil:
|
|
respStatus = "success"
|
|
default:
|
|
respStatus = "failure"
|
|
}
|
|
duration := float64(time.Since(start).Nanoseconds()) / float64(time.Millisecond)
|
|
s.metrics.expressionsQuerySummary.WithLabelValues(respStatus).Observe(duration)
|
|
|
|
span.End()
|
|
}()
|
|
|
|
// Build the pipeline from the request, checking for ordering issues (e.g. loops)
|
|
// and parsing graph nodes from the queries.
|
|
pipeline, err := s.BuildPipeline(req)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Execute the pipeline
|
|
responses, err := s.ExecutePipeline(ctx, now, pipeline)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Get which queries have the Hide property so they those queries' results
|
|
// can be excluded from the response.
|
|
hidden, err := hiddenRefIDs(req.Queries)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if len(hidden) != 0 {
|
|
filteredRes := backend.NewQueryDataResponse()
|
|
for refID, res := range responses.Responses {
|
|
if _, ok := hidden[refID]; !ok {
|
|
filteredRes.Responses[refID] = res
|
|
}
|
|
}
|
|
responses = filteredRes
|
|
}
|
|
|
|
return responses, nil
|
|
}
|
|
|
|
func hiddenRefIDs(queries []Query) (map[string]struct{}, error) {
|
|
hidden := make(map[string]struct{})
|
|
|
|
for _, query := range queries {
|
|
hide := struct {
|
|
Hide bool `json:"hide"`
|
|
}{}
|
|
|
|
if err := json.Unmarshal(query.JSON, &hide); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if hide.Hide {
|
|
hidden[query.RefID] = struct{}{}
|
|
}
|
|
}
|
|
return hidden, nil
|
|
}
|