mirror of
https://github.com/prometheus/prometheus.git
synced 2024-11-16 10:34:06 -08:00
8ebeed0b44
The Client type is already exposed, but can't be used without the config for it also being exposed. Using the remote.Client from other programs is useful to do full end-to-end tests of Prometheus's remote protocol against adapter implementations.
259 lines
6.9 KiB
Go
259 lines
6.9 KiB
Go
// Copyright 2016 The Prometheus Authors
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package remote
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"time"
|
|
|
|
"github.com/golang/protobuf/proto"
|
|
"github.com/golang/snappy"
|
|
"golang.org/x/net/context"
|
|
"golang.org/x/net/context/ctxhttp"
|
|
|
|
"github.com/prometheus/common/model"
|
|
"github.com/prometheus/prometheus/config"
|
|
"github.com/prometheus/prometheus/storage/metric"
|
|
"github.com/prometheus/prometheus/util/httputil"
|
|
)
|
|
|
|
const maxErrMsgLen = 256
|
|
|
|
// Client allows reading and writing from/to a remote HTTP endpoint.
|
|
type Client struct {
|
|
index int // Used to differentiate metrics.
|
|
url *config.URL
|
|
client *http.Client
|
|
timeout time.Duration
|
|
}
|
|
|
|
// ClientConfig configures a Client.
|
|
type ClientConfig struct {
|
|
URL *config.URL
|
|
Timeout model.Duration
|
|
HTTPClientConfig config.HTTPClientConfig
|
|
}
|
|
|
|
// NewClient creates a new Client.
|
|
func NewClient(index int, conf *ClientConfig) (*Client, error) {
|
|
// If not specified in config, allow HTTP connections for remote API to use keep-alive
|
|
if conf.HTTPClientConfig.KeepAlive == nil {
|
|
val := true
|
|
conf.HTTPClientConfig.KeepAlive = &val
|
|
}
|
|
httpClient, err := httputil.NewClientFromConfig(conf.HTTPClientConfig)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &Client{
|
|
index: index,
|
|
url: conf.URL,
|
|
client: httpClient,
|
|
timeout: time.Duration(conf.Timeout),
|
|
}, nil
|
|
}
|
|
|
|
type recoverableError struct {
|
|
error
|
|
}
|
|
|
|
// Store sends a batch of samples to the HTTP endpoint.
|
|
func (c *Client) Store(samples model.Samples) error {
|
|
req := &WriteRequest{
|
|
Timeseries: make([]*TimeSeries, 0, len(samples)),
|
|
}
|
|
for _, s := range samples {
|
|
ts := &TimeSeries{
|
|
Labels: make([]*LabelPair, 0, len(s.Metric)),
|
|
}
|
|
for k, v := range s.Metric {
|
|
ts.Labels = append(ts.Labels,
|
|
&LabelPair{
|
|
Name: string(k),
|
|
Value: string(v),
|
|
})
|
|
}
|
|
ts.Samples = []*Sample{
|
|
{
|
|
Value: float64(s.Value),
|
|
TimestampMs: int64(s.Timestamp),
|
|
},
|
|
}
|
|
req.Timeseries = append(req.Timeseries, ts)
|
|
}
|
|
|
|
data, err := proto.Marshal(req)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
compressed := snappy.Encode(nil, data)
|
|
httpReq, err := http.NewRequest("POST", c.url.String(), bytes.NewBuffer(compressed))
|
|
if err != nil {
|
|
// Errors from NewRequest are from unparseable URLs, so are not
|
|
// recoverable.
|
|
return err
|
|
}
|
|
httpReq.Header.Add("Content-Encoding", "snappy")
|
|
httpReq.Header.Set("Content-Type", "application/x-protobuf")
|
|
httpReq.Header.Set("X-Prometheus-Remote-Write-Version", "0.1.0")
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), c.timeout)
|
|
defer cancel()
|
|
|
|
httpResp, err := ctxhttp.Do(ctx, c.client, httpReq)
|
|
if err != nil {
|
|
// Errors from client.Do are from (for example) network errors, so are
|
|
// recoverable.
|
|
return recoverableError{err}
|
|
}
|
|
defer httpResp.Body.Close()
|
|
|
|
if httpResp.StatusCode/100 != 2 {
|
|
scanner := bufio.NewScanner(io.LimitReader(httpResp.Body, maxErrMsgLen))
|
|
line := ""
|
|
if scanner.Scan() {
|
|
line = scanner.Text()
|
|
}
|
|
err = fmt.Errorf("server returned HTTP status %s: %s", httpResp.Status, line)
|
|
}
|
|
if httpResp.StatusCode/100 == 5 {
|
|
return recoverableError{err}
|
|
}
|
|
return err
|
|
}
|
|
|
|
// Name identifies the client.
|
|
func (c Client) Name() string {
|
|
return fmt.Sprintf("%d:%s", c.index, c.url)
|
|
}
|
|
|
|
// Read reads from a remote endpoint.
|
|
func (c *Client) Read(ctx context.Context, from, through model.Time, matchers metric.LabelMatchers) (model.Matrix, error) {
|
|
req := &ReadRequest{
|
|
// TODO: Support batching multiple queries into one read request,
|
|
// as the protobuf interface allows for it.
|
|
Queries: []*Query{{
|
|
StartTimestampMs: int64(from),
|
|
EndTimestampMs: int64(through),
|
|
Matchers: labelMatchersToProto(matchers),
|
|
}},
|
|
}
|
|
|
|
data, err := proto.Marshal(req)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to marshal read request: %v", err)
|
|
}
|
|
|
|
compressed := snappy.Encode(nil, data)
|
|
httpReq, err := http.NewRequest("POST", c.url.String(), bytes.NewBuffer(compressed))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to create request: %v", err)
|
|
}
|
|
httpReq.Header.Add("Content-Encoding", "snappy")
|
|
httpReq.Header.Set("Content-Type", "application/x-protobuf")
|
|
httpReq.Header.Set("X-Prometheus-Remote-Read-Version", "0.1.0")
|
|
|
|
ctx, cancel := context.WithTimeout(ctx, c.timeout)
|
|
defer cancel()
|
|
|
|
httpResp, err := ctxhttp.Do(ctx, c.client, httpReq)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error sending request: %v", err)
|
|
}
|
|
defer httpResp.Body.Close()
|
|
if httpResp.StatusCode/100 != 2 {
|
|
return nil, fmt.Errorf("server returned HTTP status %s", httpResp.Status)
|
|
}
|
|
|
|
compressed, err = ioutil.ReadAll(httpResp.Body)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error reading response: %v", err)
|
|
}
|
|
|
|
uncompressed, err := snappy.Decode(nil, compressed)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error reading response: %v", err)
|
|
}
|
|
|
|
var resp ReadResponse
|
|
err = proto.Unmarshal(uncompressed, &resp)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to unmarshal response body: %v", err)
|
|
}
|
|
|
|
if len(resp.Results) != len(req.Queries) {
|
|
return nil, fmt.Errorf("responses: want %d, got %d", len(req.Queries), len(resp.Results))
|
|
}
|
|
|
|
return matrixFromProto(resp.Results[0].Timeseries), nil
|
|
}
|
|
|
|
func labelMatchersToProto(matchers metric.LabelMatchers) []*LabelMatcher {
|
|
pbMatchers := make([]*LabelMatcher, 0, len(matchers))
|
|
for _, m := range matchers {
|
|
var mType MatchType
|
|
switch m.Type {
|
|
case metric.Equal:
|
|
mType = MatchType_EQUAL
|
|
case metric.NotEqual:
|
|
mType = MatchType_NOT_EQUAL
|
|
case metric.RegexMatch:
|
|
mType = MatchType_REGEX_MATCH
|
|
case metric.RegexNoMatch:
|
|
mType = MatchType_REGEX_NO_MATCH
|
|
default:
|
|
panic("invalid matcher type")
|
|
}
|
|
pbMatchers = append(pbMatchers, &LabelMatcher{
|
|
Type: mType,
|
|
Name: string(m.Name),
|
|
Value: string(m.Value),
|
|
})
|
|
}
|
|
return pbMatchers
|
|
}
|
|
|
|
func matrixFromProto(seriesSet []*TimeSeries) model.Matrix {
|
|
m := make(model.Matrix, 0, len(seriesSet))
|
|
for _, ts := range seriesSet {
|
|
var ss model.SampleStream
|
|
ss.Metric = labelPairsToMetric(ts.Labels)
|
|
ss.Values = make([]model.SamplePair, 0, len(ts.Samples))
|
|
for _, s := range ts.Samples {
|
|
ss.Values = append(ss.Values, model.SamplePair{
|
|
Value: model.SampleValue(s.Value),
|
|
Timestamp: model.Time(s.TimestampMs),
|
|
})
|
|
}
|
|
m = append(m, &ss)
|
|
}
|
|
|
|
return m
|
|
}
|
|
|
|
func labelPairsToMetric(labelPairs []*LabelPair) model.Metric {
|
|
metric := make(model.Metric, len(labelPairs))
|
|
for _, l := range labelPairs {
|
|
metric[model.LabelName(l.Name)] = model.LabelValue(l.Value)
|
|
}
|
|
return metric
|
|
}
|