1
0
Fork 0
telegraf/plugins/outputs/opentsdb/opentsdb_http.go
Daniel Baumann 4978089aab
Adding upstream version 1.34.4.
Signed-off-by: Daniel Baumann <daniel@debian.org>
2025-05-24 07:26:29 +02:00

187 lines
3.5 KiB
Go

package opentsdb
import (
"bytes"
"compress/gzip"
"encoding/json"
"fmt"
"io"
"net/http"
"net/http/httputil"
"net/url"
"github.com/influxdata/telegraf"
)
type HTTPMetric struct {
Metric string `json:"metric"`
Timestamp int64 `json:"timestamp"`
Value interface{} `json:"value"`
Tags map[string]string `json:"tags"`
}
type openTSDBHttp struct {
Host string
Port int
Scheme string
User *url.Userinfo
BatchSize int
Path string
Debug bool
log telegraf.Logger
metricCounter int
body requestBody
}
type requestBody struct {
b bytes.Buffer
g *gzip.Writer
dbgB bytes.Buffer
w io.Writer
enc *json.Encoder
empty bool
}
func (r *requestBody) reset(debug bool) {
r.b.Reset()
r.dbgB.Reset()
if r.g == nil {
r.g = gzip.NewWriter(&r.b)
} else {
r.g.Reset(&r.b)
}
if debug {
r.w = io.MultiWriter(r.g, &r.dbgB)
} else {
r.w = r.g
}
r.enc = json.NewEncoder(r.w)
//nolint:errcheck // unable to propagate error
io.WriteString(r.w, "[")
r.empty = true
}
func (r *requestBody) addMetric(metric *HTTPMetric) error {
if !r.empty {
if _, err := io.WriteString(r.w, ","); err != nil {
return err
}
}
if err := r.enc.Encode(metric); err != nil {
return fmt.Errorf("metric serialization error %w", err)
}
r.empty = false
return nil
}
func (r *requestBody) close() error {
if _, err := io.WriteString(r.w, "]"); err != nil {
return err
}
if err := r.g.Close(); err != nil {
return fmt.Errorf("error when closing gzip writer: %w", err)
}
return nil
}
func (o *openTSDBHttp) sendDataPoint(metric *HTTPMetric) error {
if o.metricCounter == 0 {
o.body.reset(o.Debug)
}
if err := o.body.addMetric(metric); err != nil {
return err
}
o.metricCounter++
if o.metricCounter == o.BatchSize {
if err := o.flush(); err != nil {
return err
}
o.metricCounter = 0
}
return nil
}
func (o *openTSDBHttp) flush() error {
if o.metricCounter == 0 {
return nil
}
if err := o.body.close(); err != nil {
return err
}
u := url.URL{
Scheme: o.Scheme,
User: o.User,
Host: fmt.Sprintf("%s:%d", o.Host, o.Port),
Path: o.Path,
}
if o.Debug {
u.RawQuery = "details"
}
req, err := http.NewRequest("POST", u.String(), &o.body.b)
if err != nil {
return fmt.Errorf("error when building request: %w", err)
}
req.Header.Set("Content-Type", "application/json")
req.Header.Set("Content-Encoding", "gzip")
if o.Debug {
dump, err := httputil.DumpRequestOut(req, false)
if err != nil {
return fmt.Errorf("error when dumping request: %w", err)
}
fmt.Printf("Sending metrics:\n%s", dump)
fmt.Printf("Body:\n%s\n\n", o.body.dbgB.String())
}
resp, err := http.DefaultClient.Do(req)
if err != nil {
return fmt.Errorf("error when sending metrics: %w", err)
}
defer resp.Body.Close()
if o.Debug {
dump, err := httputil.DumpResponse(resp, true)
if err != nil {
return fmt.Errorf("error when dumping response: %w", err)
}
fmt.Printf("Received response\n%s\n\n", dump)
} else {
// Important so http client reuse connection for next request if need be.
//nolint:errcheck // cannot fail with io.Discard
io.Copy(io.Discard, resp.Body)
}
if resp.StatusCode < 200 || resp.StatusCode > 299 {
if resp.StatusCode < 400 || resp.StatusCode > 499 {
return fmt.Errorf("error sending metrics (status %d)", resp.StatusCode)
}
o.log.Errorf("Received %d status code. Dropping metrics to avoid overflowing buffer.", resp.StatusCode)
}
return nil
}