mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-12 13:32:25 +01:00
2a5b9ff782
* app/vmctl: change api for getting metric names * app/vmctl: fix tests * app/vmctl: add flag to enable backoff policy, fix test, performance improvements * app/vmctl: use one http client * app/vmctl: made linter happy * app/vmctl: updated documentation and CHANGELOG.md * app/vmctl: cleanup * app/vmctl: rename flag * app/vmctl: cleanup * app/vmctl: fix comments * app/vmctl: fix metrics parser problem, improve tests
172 lines
4.8 KiB
Go
172 lines
4.8 KiB
Go
package native
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmctl/auth"
|
|
)
|
|
|
|
const (
|
|
nativeTenantsAddr = "admin/tenants"
|
|
nativeMetricNamesAddr = "api/v1/label/__name__/values"
|
|
)
|
|
|
|
// Client is an HTTP client for exporting and importing
|
|
// time series via native protocol.
|
|
type Client struct {
|
|
AuthCfg *auth.Config
|
|
Addr string
|
|
ExtraLabels []string
|
|
HTTPClient *http.Client
|
|
}
|
|
|
|
// LabelValues represents series from api/v1/series response
|
|
type LabelValues map[string]string
|
|
|
|
// Response represents response from api/v1/label/__name__/values
|
|
type Response struct {
|
|
Status string `json:"status"`
|
|
MetricNames []string `json:"data"`
|
|
}
|
|
|
|
// Explore finds metric names by provided filter from api/v1/label/__name__/values
|
|
func (c *Client) Explore(ctx context.Context, f Filter, tenantID string) ([]string, error) {
|
|
url := fmt.Sprintf("%s/%s", c.Addr, nativeMetricNamesAddr)
|
|
if tenantID != "" {
|
|
url = fmt.Sprintf("%s/select/%s/prometheus/%s", c.Addr, tenantID, nativeMetricNamesAddr)
|
|
}
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("cannot create request to %q: %s", url, err)
|
|
}
|
|
|
|
params := req.URL.Query()
|
|
if f.TimeStart != "" {
|
|
params.Set("start", f.TimeStart)
|
|
}
|
|
if f.TimeEnd != "" {
|
|
params.Set("end", f.TimeEnd)
|
|
}
|
|
params.Set("match[]", f.Match)
|
|
req.URL.RawQuery = params.Encode()
|
|
|
|
resp, err := c.do(req, http.StatusOK)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("series request failed: %s", err)
|
|
}
|
|
|
|
var response Response
|
|
if err := json.NewDecoder(resp.Body).Decode(&response); err != nil {
|
|
return nil, fmt.Errorf("cannot decode series response: %s", err)
|
|
}
|
|
|
|
if err := resp.Body.Close(); err != nil {
|
|
return nil, fmt.Errorf("cannot close series response body: %s", err)
|
|
}
|
|
return response.MetricNames, nil
|
|
}
|
|
|
|
// ImportPipe uses pipe reader in request to process data
|
|
func (c *Client) ImportPipe(ctx context.Context, dstURL string, pr *io.PipeReader) error {
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodPost, dstURL, pr)
|
|
if err != nil {
|
|
return fmt.Errorf("cannot create import request to %q: %s", c.Addr, err)
|
|
}
|
|
|
|
importResp, err := c.do(req, http.StatusNoContent)
|
|
if err != nil {
|
|
return fmt.Errorf("import request failed: %s", err)
|
|
}
|
|
if err := importResp.Body.Close(); err != nil {
|
|
return fmt.Errorf("cannot close import response body: %s", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ExportPipe makes request by provided filter and return io.ReadCloser which can be used to get data
|
|
func (c *Client) ExportPipe(ctx context.Context, url string, f Filter) (io.ReadCloser, error) {
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("cannot create request to %q: %s", c.Addr, err)
|
|
}
|
|
|
|
params := req.URL.Query()
|
|
params.Set("match[]", f.Match)
|
|
if f.TimeStart != "" {
|
|
params.Set("start", f.TimeStart)
|
|
}
|
|
if f.TimeEnd != "" {
|
|
params.Set("end", f.TimeEnd)
|
|
}
|
|
req.URL.RawQuery = params.Encode()
|
|
|
|
// disable compression since it is meaningless for native format
|
|
req.Header.Set("Accept-Encoding", "identity")
|
|
|
|
resp, err := c.do(req, http.StatusOK)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("export request failed: %w", err)
|
|
}
|
|
return resp.Body, nil
|
|
}
|
|
|
|
// GetSourceTenants discovers tenants by provided filter
|
|
func (c *Client) GetSourceTenants(ctx context.Context, f Filter) ([]string, error) {
|
|
u := fmt.Sprintf("%s/%s", c.Addr, nativeTenantsAddr)
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, u, nil)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("cannot create request to %q: %s", u, err)
|
|
}
|
|
|
|
params := req.URL.Query()
|
|
if f.TimeStart != "" {
|
|
params.Set("start", f.TimeStart)
|
|
}
|
|
if f.TimeEnd != "" {
|
|
params.Set("end", f.TimeEnd)
|
|
}
|
|
req.URL.RawQuery = params.Encode()
|
|
|
|
resp, err := c.do(req, http.StatusOK)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("tenants request failed: %s", err)
|
|
}
|
|
|
|
var r struct {
|
|
Tenants []string `json:"data"`
|
|
}
|
|
if err := json.NewDecoder(resp.Body).Decode(&r); err != nil {
|
|
return nil, fmt.Errorf("cannot decode tenants response: %s", err)
|
|
}
|
|
|
|
if err := resp.Body.Close(); err != nil {
|
|
return nil, fmt.Errorf("cannot close tenants response body: %s", err)
|
|
}
|
|
|
|
return r.Tenants, nil
|
|
}
|
|
|
|
func (c *Client) do(req *http.Request, expSC int) (*http.Response, error) {
|
|
if c.AuthCfg != nil {
|
|
c.AuthCfg.SetHeaders(req, true)
|
|
}
|
|
|
|
resp, err := c.HTTPClient.Do(req)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unexpected error when performing request: %w", err)
|
|
}
|
|
|
|
if resp.StatusCode != expSC {
|
|
body, err := io.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to read response body for status code %d: %s", resp.StatusCode, err)
|
|
}
|
|
return nil, fmt.Errorf("unexpected response code %d: %s", resp.StatusCode, string(body))
|
|
}
|
|
return resp, err
|
|
}
|