mirror of
https://github.com/grafana/grafana.git
synced 2025-01-10 08:03:58 -06:00
3fb6319d1b
* Move files to prometheus-library * refactor core prometheus to use prometheus-library * modify client transport options * mock * have a type * import aliases * rename * call the right method * remove unrelated test from the library * update codeowners * go work sync * update go.work.sum * make swagger-clean && make openapi3-gen * add promlib to makefile * remove clilogger * Export the function * update unit test * add prometheus_test.go * fix mock type * use mapUtil from grafana-plugin-sdk-go
173 lines
4.8 KiB
Go
173 lines
4.8 KiB
Go
package client
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"io"
|
|
"net/http"
|
|
"net/url"
|
|
"path"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/grafana/grafana-plugin-sdk-go/backend"
|
|
|
|
"github.com/grafana/grafana/pkg/promlib/models"
|
|
)
|
|
|
|
type doer interface {
|
|
Do(req *http.Request) (*http.Response, error)
|
|
}
|
|
|
|
// Client is a custom Prometheus client. Reason for this is that Prom Go client serializes response into its own
|
|
// objects, we have to go through them and then serialize again into DataFrame which isn't very efficient. Using custom
|
|
// client we can parse response directly into DataFrame.
|
|
type Client struct {
|
|
doer doer
|
|
method string
|
|
baseUrl string
|
|
}
|
|
|
|
func NewClient(d doer, method, baseUrl string) *Client {
|
|
return &Client{doer: d, method: method, baseUrl: baseUrl}
|
|
}
|
|
|
|
func (c *Client) QueryRange(ctx context.Context, q *models.Query) (*http.Response, error) {
|
|
tr := q.TimeRange()
|
|
qv := map[string]string{
|
|
"query": q.Expr,
|
|
"start": formatTime(tr.Start),
|
|
"end": formatTime(tr.End),
|
|
"step": strconv.FormatFloat(tr.Step.Seconds(), 'f', -1, 64),
|
|
}
|
|
|
|
req, err := c.createQueryRequest(ctx, "api/v1/query_range", qv)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return c.doer.Do(req)
|
|
}
|
|
|
|
func (c *Client) QueryInstant(ctx context.Context, q *models.Query) (*http.Response, error) {
|
|
// We do not need a time range here.
|
|
// Instant query evaluates at a single point in time.
|
|
// Using q.TimeRange is aligning the query range to step.
|
|
// Which causes a misleading time point.
|
|
// Instead of aligning we use time point directly.
|
|
// https://prometheus.io/docs/prometheus/latest/querying/api/#instant-queries
|
|
qv := map[string]string{"query": q.Expr, "time": formatTime(q.End)}
|
|
req, err := c.createQueryRequest(ctx, "api/v1/query", qv)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return c.doer.Do(req)
|
|
}
|
|
|
|
func (c *Client) QueryExemplars(ctx context.Context, q *models.Query) (*http.Response, error) {
|
|
tr := q.TimeRange()
|
|
qv := map[string]string{
|
|
"query": q.Expr,
|
|
"start": formatTime(tr.Start),
|
|
"end": formatTime(tr.End),
|
|
}
|
|
|
|
req, err := c.createQueryRequest(ctx, "api/v1/query_exemplars", qv)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return c.doer.Do(req)
|
|
}
|
|
|
|
func (c *Client) QueryResource(ctx context.Context, req *backend.CallResourceRequest) (*http.Response, error) {
|
|
// The way URL is represented in CallResourceRequest and what we need for the fetch function is different
|
|
// so here we have to do a bit of parsing, so we can then compose it with the base url in correct way.
|
|
reqUrlParsed, err := url.Parse(req.URL)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
u, err := c.createUrl(req.Path, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
u.RawQuery = reqUrlParsed.RawQuery
|
|
|
|
// We use method from the request, as for resources front end may do a fallback to GET if POST does not work
|
|
// nad we want to respect that.
|
|
httpRequest, err := createRequest(ctx, req.Method, u, bytes.NewReader(req.Body))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return c.doer.Do(httpRequest)
|
|
}
|
|
|
|
func (c *Client) createQueryRequest(ctx context.Context, endpoint string, qv map[string]string) (*http.Request, error) {
|
|
if strings.ToUpper(c.method) == http.MethodPost {
|
|
u, err := c.createUrl(endpoint, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
v := make(url.Values)
|
|
for key, val := range qv {
|
|
v.Set(key, val)
|
|
}
|
|
|
|
return createRequest(ctx, c.method, u, strings.NewReader(v.Encode()))
|
|
}
|
|
|
|
u, err := c.createUrl(endpoint, qv)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return createRequest(ctx, c.method, u, http.NoBody)
|
|
}
|
|
|
|
func (c *Client) createUrl(endpoint string, qs map[string]string) (*url.URL, error) {
|
|
finalUrl, err := url.ParseRequestURI(c.baseUrl)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
finalUrl.Path = path.Join(finalUrl.Path, endpoint)
|
|
|
|
// don't re-encode the Query if not needed
|
|
if len(qs) != 0 {
|
|
urlQuery := finalUrl.Query()
|
|
|
|
for key, val := range qs {
|
|
urlQuery.Set(key, val)
|
|
}
|
|
|
|
finalUrl.RawQuery = urlQuery.Encode()
|
|
}
|
|
|
|
return finalUrl, nil
|
|
}
|
|
|
|
func createRequest(ctx context.Context, method string, u *url.URL, bodyReader io.Reader) (*http.Request, error) {
|
|
request, err := http.NewRequestWithContext(ctx, method, u.String(), bodyReader)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if strings.ToUpper(method) == http.MethodPost {
|
|
// This may not be true but right now we don't have more information here and seems like we send just this type
|
|
// of encoding right now if it is a POST
|
|
request.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
|
// This allows transport to retry request. See https://github.com/prometheus/client_golang/pull/1022
|
|
// It's set to nil so it is not actually sent over the wire, just used in Go http lib to retry requests.
|
|
request.Header["Idempotency-Key"] = nil
|
|
}
|
|
return request, nil
|
|
}
|
|
|
|
func formatTime(t time.Time) string {
|
|
return strconv.FormatFloat(float64(t.Unix())+float64(t.Nanosecond())/1e9, 'f', -1, 64)
|
|
}
|