370 lines
8.8 KiB
Go
370 lines
8.8 KiB
Go
package influx
|
|
|
|
import (
|
|
"context"
|
|
"crypto/tls"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"net/url"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/influxdata/chronograf"
|
|
)
|
|
|
|
var _ chronograf.TimeSeries = &Client{}
|
|
var _ chronograf.TSDBStatus = &Client{}
|
|
var _ chronograf.Databases = &Client{}
|
|
|
|
// Shared transports for all clients to prevent leaking connections
|
|
var (
|
|
skipVerifyTransport = &http.Transport{
|
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: true},
|
|
}
|
|
defaultTransport = &http.Transport{}
|
|
)
|
|
|
|
// Client is a device for retrieving time series data from an InfluxDB instance
|
|
type Client struct {
|
|
URL *url.URL
|
|
Authorizer Authorizer
|
|
InsecureSkipVerify bool
|
|
Logger chronograf.Logger
|
|
}
|
|
|
|
// Response is a partial JSON decoded InfluxQL response used
|
|
// to check for some errors
|
|
type Response struct {
|
|
Results json.RawMessage
|
|
Err string `json:"error,omitempty"`
|
|
}
|
|
|
|
// MarshalJSON returns the raw results bytes from the response
|
|
func (r Response) MarshalJSON() ([]byte, error) {
|
|
return r.Results, nil
|
|
}
|
|
|
|
func (c *Client) query(u *url.URL, q chronograf.Query) (chronograf.Response, error) {
|
|
u.Path = "query"
|
|
req, err := http.NewRequest("POST", u.String(), nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
req.Header.Set("Content-Type", "application/json")
|
|
command := q.Command
|
|
// TODO(timraymond): move this upper Query() function
|
|
if len(q.TemplateVars) > 0 {
|
|
command, err = TemplateReplace(q.Command, q.TemplateVars, time.Now())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
logs := c.Logger.
|
|
WithField("component", "proxy").
|
|
WithField("host", req.Host).
|
|
WithField("command", command).
|
|
WithField("db", q.DB).
|
|
WithField("rp", q.RP)
|
|
logs.Debug("query")
|
|
|
|
params := req.URL.Query()
|
|
params.Set("q", command)
|
|
params.Set("db", q.DB)
|
|
params.Set("rp", q.RP)
|
|
params.Set("epoch", "ms")
|
|
if q.Epoch != "" {
|
|
params.Set("epoch", q.Epoch)
|
|
}
|
|
req.URL.RawQuery = params.Encode()
|
|
|
|
if c.Authorizer != nil {
|
|
if err := c.Authorizer.Set(req); err != nil {
|
|
logs.Error("Error setting authorization header ", err)
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
hc := &http.Client{}
|
|
if c.InsecureSkipVerify {
|
|
hc.Transport = skipVerifyTransport
|
|
} else {
|
|
hc.Transport = defaultTransport
|
|
}
|
|
resp, err := hc.Do(req)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
var response Response
|
|
dec := json.NewDecoder(resp.Body)
|
|
decErr := dec.Decode(&response)
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
return nil, fmt.Errorf("received status code %d from server: err: %s", resp.StatusCode, response.Err)
|
|
}
|
|
|
|
// ignore this error if we got an invalid status code
|
|
if decErr != nil && decErr.Error() == "EOF" && resp.StatusCode != http.StatusOK {
|
|
decErr = nil
|
|
}
|
|
|
|
// If we got a valid decode error, send that back
|
|
if decErr != nil {
|
|
logs.WithField("influx_status", resp.StatusCode).
|
|
Error("Error parsing results from influxdb: err:", decErr)
|
|
return nil, decErr
|
|
}
|
|
|
|
// If we don't have an error in our json response, and didn't get statusOK
|
|
// then send back an error
|
|
if resp.StatusCode != http.StatusOK && response.Err != "" {
|
|
logs.
|
|
WithField("influx_status", resp.StatusCode).
|
|
Error("Received non-200 response from influxdb")
|
|
|
|
return &response, fmt.Errorf("received status code %d from server",
|
|
resp.StatusCode)
|
|
}
|
|
return &response, nil
|
|
}
|
|
|
|
type result struct {
|
|
Response chronograf.Response
|
|
Err error
|
|
}
|
|
|
|
// Query issues a request to a configured InfluxDB instance for time series
|
|
// information specified by query. Queries must be "fully-qualified," and
|
|
// include both the database and retention policy. In-flight requests can be
|
|
// cancelled using the provided context.
|
|
func (c *Client) Query(ctx context.Context, q chronograf.Query) (chronograf.Response, error) {
|
|
resps := make(chan (result))
|
|
go func() {
|
|
resp, err := c.query(c.URL, q)
|
|
resps <- result{resp, err}
|
|
}()
|
|
|
|
select {
|
|
case resp := <-resps:
|
|
return resp.Response, resp.Err
|
|
case <-ctx.Done():
|
|
return nil, chronograf.ErrUpstreamTimeout
|
|
}
|
|
}
|
|
|
|
// Connect caches the URL and optional Bearer Authorization for the data source
|
|
func (c *Client) Connect(ctx context.Context, src *chronograf.Source) error {
|
|
u, err := url.Parse(src.URL)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
c.Authorizer = DefaultAuthorization(src)
|
|
// Only allow acceptance of all certs if the scheme is https AND the user opted into to the setting.
|
|
if u.Scheme == "https" && src.InsecureSkipVerify {
|
|
c.InsecureSkipVerify = src.InsecureSkipVerify
|
|
}
|
|
|
|
c.URL = u
|
|
return nil
|
|
}
|
|
|
|
// Users transforms InfluxDB into a user store
|
|
func (c *Client) Users(ctx context.Context) chronograf.UsersStore {
|
|
return c
|
|
}
|
|
|
|
// Roles aren't support in OSS
|
|
func (c *Client) Roles(ctx context.Context) (chronograf.RolesStore, error) {
|
|
return nil, fmt.Errorf("Roles not support in open-source InfluxDB. Roles are support in Influx Enterprise")
|
|
}
|
|
|
|
// Ping hits the influxdb ping endpoint and returns the type of influx
|
|
func (c *Client) Ping(ctx context.Context) error {
|
|
_, _, err := c.pingTimeout(ctx)
|
|
return err
|
|
}
|
|
|
|
// Version hits the influxdb ping endpoint and returns the version of influx
|
|
func (c *Client) Version(ctx context.Context) (string, error) {
|
|
version, _, err := c.pingTimeout(ctx)
|
|
return version, err
|
|
}
|
|
|
|
// Type hits the influxdb ping endpoint and returns the type of influx running
|
|
func (c *Client) Type(ctx context.Context) (string, error) {
|
|
_, tsdbType, err := c.pingTimeout(ctx)
|
|
return tsdbType, err
|
|
}
|
|
|
|
func (c *Client) pingTimeout(ctx context.Context) (string, string, error) {
|
|
resps := make(chan (pingResult))
|
|
go func() {
|
|
version, tsdbType, err := c.ping(c.URL)
|
|
resps <- pingResult{version, tsdbType, err}
|
|
}()
|
|
|
|
select {
|
|
case resp := <-resps:
|
|
return resp.Version, resp.Type, resp.Err
|
|
case <-ctx.Done():
|
|
return "", "", chronograf.ErrUpstreamTimeout
|
|
}
|
|
}
|
|
|
|
type pingResult struct {
|
|
Version string
|
|
Type string
|
|
Err error
|
|
}
|
|
|
|
func (c *Client) ping(u *url.URL) (string, string, error) {
|
|
u.Path = "ping"
|
|
|
|
req, err := http.NewRequest("GET", u.String(), nil)
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
|
|
hc := &http.Client{}
|
|
if c.InsecureSkipVerify {
|
|
hc.Transport = skipVerifyTransport
|
|
} else {
|
|
hc.Transport = defaultTransport
|
|
}
|
|
|
|
resp, err := hc.Do(req)
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return "", "", err
|
|
}
|
|
|
|
if resp.StatusCode != http.StatusNoContent {
|
|
var err = fmt.Errorf(string(body))
|
|
return "", "", err
|
|
}
|
|
|
|
version := resp.Header.Get("X-Influxdb-Build")
|
|
if version == "ENT" {
|
|
return version, chronograf.InfluxEnterprise, nil
|
|
}
|
|
version = resp.Header.Get("X-Influxdb-Version")
|
|
if strings.Contains(version, "-c") {
|
|
return version, chronograf.InfluxEnterprise, nil
|
|
} else if strings.Contains(version, "relay") {
|
|
return version, chronograf.InfluxRelay, nil
|
|
}
|
|
|
|
return version, chronograf.InfluxDB, nil
|
|
}
|
|
|
|
// Write POSTs line protocol to a database and retention policy
|
|
func (c *Client) Write(ctx context.Context, points []chronograf.Point) error {
|
|
for _, point := range points {
|
|
if err := c.writePoint(ctx, &point); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *Client) writePoint(ctx context.Context, point *chronograf.Point) error {
|
|
lp, err := toLineProtocol(point)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = c.write(ctx, c.URL, point.Database, point.RetentionPolicy, lp)
|
|
if err == nil {
|
|
return nil
|
|
}
|
|
|
|
// Some influxdb errors should not be treated as errors
|
|
if strings.Contains(err.Error(), "hinted handoff queue not empty") {
|
|
// This is an informational message
|
|
return nil
|
|
}
|
|
|
|
// If the database was not found, try to recreate it:
|
|
if strings.Contains(err.Error(), "database not found") {
|
|
_, err = c.CreateDB(ctx, &chronograf.Database{
|
|
Name: point.Database,
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// retry the write
|
|
return c.write(ctx, c.URL, point.Database, point.RetentionPolicy, lp)
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
func (c *Client) write(ctx context.Context, u *url.URL, db, rp, lp string) error {
|
|
u.Path = "write"
|
|
req, err := http.NewRequest("POST", u.String(), strings.NewReader(lp))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
req.Header.Set("Content-Type", "text/plain; charset=utf-8")
|
|
if c.Authorizer != nil {
|
|
if err := c.Authorizer.Set(req); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
params := req.URL.Query()
|
|
params.Set("db", db)
|
|
params.Set("rp", rp)
|
|
req.URL.RawQuery = params.Encode()
|
|
|
|
hc := &http.Client{}
|
|
if c.InsecureSkipVerify {
|
|
hc.Transport = skipVerifyTransport
|
|
} else {
|
|
hc.Transport = defaultTransport
|
|
}
|
|
|
|
errChan := make(chan (error))
|
|
go func() {
|
|
resp, err := hc.Do(req)
|
|
if err != nil {
|
|
errChan <- err
|
|
return
|
|
}
|
|
|
|
if resp.StatusCode == http.StatusNoContent {
|
|
errChan <- nil
|
|
return
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
var response Response
|
|
dec := json.NewDecoder(resp.Body)
|
|
err = dec.Decode(&response)
|
|
if err != nil && err.Error() != "EOF" {
|
|
errChan <- err
|
|
return
|
|
}
|
|
|
|
errChan <- errors.New(response.Err)
|
|
return
|
|
}()
|
|
|
|
select {
|
|
case err := <-errChan:
|
|
return err
|
|
case <-ctx.Done():
|
|
return chronograf.ErrUpstreamTimeout
|
|
}
|
|
}
|