mirror of
https://github.com/grafana/grafana.git
synced 2025-01-04 13:17:16 -06:00
6d64c603c2
* Expr: fix failure to execute due to OrgID Get orgID from the plugin context, which makes more sense anyways. makes expressions work again after https://github.com/grafana/grafana/pull/29449 changes. * Do not save organisation on its alert query model Co-authored-by: Sofia Papagiannaki <sofia@grafana.com>
184 lines
4.4 KiB
Go
184 lines
4.4 KiB
Go
package expr
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
|
|
"github.com/grafana/grafana-plugin-sdk-go/backend"
|
|
"github.com/grafana/grafana/pkg/expr/mathexp"
|
|
|
|
"gonum.org/v1/gonum/graph"
|
|
"gonum.org/v1/gonum/graph/simple"
|
|
"gonum.org/v1/gonum/graph/topo"
|
|
)
|
|
|
|
// NodeType is the type of a DPNode. Currently either a expression command or datasource query.
|
|
type NodeType int
|
|
|
|
const (
|
|
// TypeCMDNode is a NodeType for expression commands.
|
|
TypeCMDNode NodeType = iota
|
|
// TypeDatasourceNode is a NodeType for datasource queries.
|
|
TypeDatasourceNode
|
|
)
|
|
|
|
// Node is a node in a Data Pipeline. Node is either a expression command or a datasource query.
|
|
type Node interface {
|
|
ID() int64 // ID() allows the gonum graph node interface to be fulfilled
|
|
NodeType() NodeType
|
|
RefID() string
|
|
Execute(c context.Context, vars mathexp.Vars) (mathexp.Results, error)
|
|
String() string
|
|
}
|
|
|
|
// DataPipeline is an ordered set of nodes returned from DPGraph processing.
|
|
type DataPipeline []Node
|
|
|
|
// execute runs all the command/datasource requests in the pipeline return a
|
|
// map of the refId of the of each command
|
|
func (dp *DataPipeline) execute(c context.Context) (mathexp.Vars, error) {
|
|
vars := make(mathexp.Vars)
|
|
for _, node := range *dp {
|
|
res, err := node.Execute(c, vars)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
vars[node.RefID()] = res
|
|
}
|
|
return vars, nil
|
|
}
|
|
|
|
// BuildPipeline builds a graph of the nodes, and returns the nodes in an
|
|
// executable order.
|
|
func buildPipeline(req *backend.QueryDataRequest) (DataPipeline, error) {
|
|
graph, err := buildDependencyGraph(req)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
nodes, err := buildExecutionOrder(graph)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return nodes, nil
|
|
}
|
|
|
|
// buildDependencyGraph returns a dependency graph for a set of queries.
|
|
func buildDependencyGraph(req *backend.QueryDataRequest) (*simple.DirectedGraph, error) {
|
|
graph, err := buildGraph(req)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
registry := buildNodeRegistry(graph)
|
|
|
|
if err := buildGraphEdges(graph, registry); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return graph, nil
|
|
}
|
|
|
|
// buildExecutionOrder returns a sequence of nodes ordered by dependency.
|
|
func buildExecutionOrder(graph *simple.DirectedGraph) ([]Node, error) {
|
|
sortedNodes, err := topo.Sort(graph)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
nodes := make([]Node, len(sortedNodes))
|
|
for i, v := range sortedNodes {
|
|
nodes[i] = v.(Node)
|
|
}
|
|
|
|
return nodes, nil
|
|
}
|
|
|
|
// buildNodeRegistry returns a lookup table for reference IDs to respective node.
|
|
func buildNodeRegistry(g *simple.DirectedGraph) map[string]Node {
|
|
res := make(map[string]Node)
|
|
|
|
nodeIt := g.Nodes()
|
|
|
|
for nodeIt.Next() {
|
|
if dpNode, ok := nodeIt.Node().(Node); ok {
|
|
res[dpNode.RefID()] = dpNode
|
|
}
|
|
}
|
|
|
|
return res
|
|
}
|
|
|
|
// buildGraph creates a new graph populated with nodes for every query.
|
|
func buildGraph(req *backend.QueryDataRequest) (*simple.DirectedGraph, error) {
|
|
dp := simple.NewDirectedGraph()
|
|
|
|
for _, query := range req.Queries {
|
|
rawQueryProp := make(map[string]interface{})
|
|
err := json.Unmarshal(query.JSON, &rawQueryProp)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
rn := &rawNode{
|
|
Query: rawQueryProp,
|
|
RefID: query.RefID,
|
|
TimeRange: query.TimeRange,
|
|
QueryType: query.QueryType,
|
|
}
|
|
|
|
dsName, err := rn.GetDatasourceName()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var node graph.Node
|
|
switch dsName {
|
|
case DatasourceName:
|
|
node, err = buildCMDNode(dp, rn)
|
|
default: // If it's not an expression query, it's a data source query.
|
|
node, err = buildDSNode(dp, rn, req.PluginContext.OrgID)
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
dp.AddNode(node)
|
|
}
|
|
return dp, nil
|
|
}
|
|
|
|
// buildGraphEdges generates graph edges based on each node's dependencies.
|
|
func buildGraphEdges(dp *simple.DirectedGraph, registry map[string]Node) error {
|
|
nodeIt := dp.Nodes()
|
|
|
|
for nodeIt.Next() {
|
|
node := nodeIt.Node().(Node)
|
|
|
|
if node.NodeType() != TypeCMDNode {
|
|
// datasource node, nothing to do for now. Although if we want expression results to be
|
|
// used as datasource query params some day this will need change
|
|
continue
|
|
}
|
|
|
|
cmdNode := node.(*CMDNode)
|
|
|
|
for _, neededVar := range cmdNode.Command.NeedsVars() {
|
|
neededNode, ok := registry[neededVar]
|
|
if !ok {
|
|
return fmt.Errorf("unable to find dependent node '%v'", neededVar)
|
|
}
|
|
|
|
if neededNode.ID() == cmdNode.ID() {
|
|
return fmt.Errorf("can not add self referencing node for var '%v' ", neededVar)
|
|
}
|
|
|
|
edge := dp.NewEdge(neededNode, cmdNode)
|
|
|
|
dp.SetEdge(edge)
|
|
}
|
|
}
|
|
return nil
|
|
}
|