grafana/pkg/expr/ml.go
2023-09-25 13:10:47 +03:00

157 lines
5.2 KiB
Go

package expr
import (
"context"
"errors"
"fmt"
"net/http"
"strings"
"time"
"github.com/grafana/grafana-plugin-sdk-go/backend"
jsoniter "github.com/json-iterator/go"
"gonum.org/v1/gonum/graph/simple"
"github.com/grafana/grafana/pkg/api/response"
"github.com/grafana/grafana/pkg/expr/mathexp"
"github.com/grafana/grafana/pkg/expr/ml"
"github.com/grafana/grafana/pkg/plugins"
"github.com/grafana/grafana/pkg/plugins/httpresponsesender"
)
var (
errMLPluginDoesNotExist = fmt.Errorf("expression type Machine Learning cannot be executed. Plugin '%s' must be installed and initialized", mlPluginID)
)
const (
// mlDatasourceID is similar to a fake ID for CMDNode. There is no specific reason for the selection of this value.
mlDatasourceID = -200
// DatasourceUID is the string constant used as the datasource name in requests
// to identify it as an expression command when use in Datasource.UID.
MLDatasourceUID = "__ml__"
// mlPluginID is a known constant and used in other places of the code
mlPluginID = "grafana-ml-app"
)
// MLNode is a node of expression tree that evaluates the expression by sending the payload to Machine Learning back-end.
// See ml.UnmarshalCommand for supported commands.
type MLNode struct {
baseNode
command ml.Command
TimeRange TimeRange
request *Request
}
// NodeType returns the data pipeline node type.
func (m *MLNode) NodeType() NodeType {
return TypeMLNode
}
// NodeType returns the data pipeline node type.
func (m *MLNode) NeedsVars() []string {
return []string{}
}
// Execute initializes plugin API client, executes a ml.Command and then converts the result of the execution.
// Returns non-empty mathexp.Results if evaluation was successful. Returns QueryError if command execution failed
func (m *MLNode) Execute(ctx context.Context, now time.Time, _ mathexp.Vars, s *Service) (r mathexp.Results, e error) {
logger := logger.FromContext(ctx).New("datasourceType", mlPluginID, "queryRefId", m.refID)
var result mathexp.Results
timeRange := m.TimeRange.AbsoluteTime(now)
// get the plugin configuration that will be used by client (auth, host, etc)
pCtx, err := s.pCtxProvider.Get(ctx, mlPluginID, m.request.User, m.request.OrgId)
if err != nil {
if errors.Is(err, plugins.ErrPluginNotRegistered) {
return result, errMLPluginDoesNotExist
}
return result, fmt.Errorf("failed to get plugin settings: %w", err)
}
// Plugin must be initialized by the admin first. That will create service account, and update plugin settings so all requests can use it.
// Fail if it is not initialized.
if pCtx.AppInstanceSettings == nil || !jsoniter.Get(pCtx.AppInstanceSettings.JSONData, "initialized").ToBool() {
return mathexp.Results{}, errMLPluginDoesNotExist
}
// responseType and respStatus will be updated below. Use defer to ensure that debug log message is always emitted
responseType := "unknown"
respStatus := "success"
defer func() {
if e != nil {
responseType = "error"
respStatus = "failure"
}
logger.Debug("Data source queried", "responseType", responseType)
useDataplane := strings.HasPrefix("dataplane-", responseType)
s.metrics.dsRequests.WithLabelValues(respStatus, fmt.Sprintf("%t", useDataplane), mlPluginID).Inc()
}()
// Execute the command and provide callback function for sending a request via plugin API.
// This lets us make commands abstracted from peculiarities of the transfer protocol.
data, err := m.command.Execute(timeRange.From, timeRange.To, func(method string, path string, payload []byte) (response.Response, error) {
crReq := &backend.CallResourceRequest{
PluginContext: pCtx,
Path: path,
Method: method,
URL: path,
Headers: make(map[string][]string, len(m.request.Headers)),
Body: payload,
}
// copy headers from the request to evaluate the expression pipeline. Usually this contains information from upstream, e.g. FromAlert
for key, val := range m.request.Headers {
crReq.SetHTTPHeader(key, val)
}
resp := response.CreateNormalResponse(make(http.Header), nil, 0)
httpSender := httpresponsesender.New(resp)
err = s.pluginsClient.CallResource(ctx, crReq, httpSender)
if err != nil {
return nil, err
}
return resp, nil
})
if err != nil {
return result, MakeQueryError(m.refID, "ml", err)
}
// data is not guaranteed to be specified. In this case simulate NoData scenario
if data == nil {
data = &backend.QueryDataResponse{Responses: map[string]backend.DataResponse{}}
}
dataFrames, err := getResponseFrame(data, m.refID)
if err != nil {
return mathexp.Results{}, MakeQueryError(m.refID, "ml", err)
}
// process the response the same way DSNode does. Use plugin ID as data source type. Semantically, they are the same.
responseType, result, err = convertDataFramesToResults(ctx, dataFrames, mlPluginID, s, logger)
return result, err
}
func (s *Service) buildMLNode(dp *simple.DirectedGraph, rn *rawNode, req *Request) (Node, error) {
if rn.TimeRange == nil {
return nil, errors.New("time range must be specified")
}
cmd, err := ml.UnmarshalCommand(rn.QueryRaw, s.cfg.AppURL)
if err != nil {
return nil, err
}
return &MLNode{
baseNode: baseNode{
id: rn.idx,
refID: rn.RefID,
},
TimeRange: rn.TimeRange,
command: cmd,
request: req,
}, nil
}