influxdb/client/influxdb.go

462 lines
11 KiB
Go
Raw Normal View History

package client
import (
2015-01-23 20:37:53 +00:00
"bytes"
2015-01-23 00:18:24 +00:00
"encoding/json"
"errors"
2015-01-29 18:10:13 +00:00
"fmt"
"net/http"
"net/url"
"time"
"github.com/influxdb/influxdb/influxql"
)
type Query struct {
Command string
Database string
}
type Config struct {
2015-02-11 20:31:15 +00:00
URL url.URL
Username string
Password string
UserAgent string
}
type Client struct {
url url.URL
2015-01-23 18:05:04 +00:00
username string
password string
httpClient *http.Client
2015-02-11 20:31:15 +00:00
userAgent string
}
func NewClient(c Config) (*Client, error) {
client := Client{
url: c.URL,
2015-01-23 18:05:04 +00:00
username: c.Username,
password: c.Password,
httpClient: &http.Client{},
2015-02-11 20:31:15 +00:00
userAgent: c.UserAgent,
}
return &client, nil
}
// BatchPoints is used to send batched data in a single write.
type BatchPoints struct {
Points []Point `json:"points"`
Database string `json:"database"`
RetentionPolicy string `json:"retentionPolicy"`
Tags map[string]string `json:"tags"`
Timestamp time.Time `json:"timestamp"`
Precision string `json:"precision"`
}
// UnmarshalJSON decodes the data into the BatchPoints struct
func (bp *BatchPoints) UnmarshalJSON(b []byte) error {
var normal struct {
Points []Point `json:"points"`
Database string `json:"database"`
RetentionPolicy string `json:"retentionPolicy"`
Tags map[string]string `json:"tags"`
Timestamp time.Time `json:"timestamp"`
Precision string `json:"precision"`
}
var epoch struct {
Points []Point `json:"points"`
Database string `json:"database"`
RetentionPolicy string `json:"retentionPolicy"`
Tags map[string]string `json:"tags"`
Timestamp *int64 `json:"timestamp"`
Precision string `json:"precision"`
}
if err := func() error {
var err error
if err = json.Unmarshal(b, &epoch); err != nil {
return err
}
// Convert from epoch to time.Time
var ts time.Time
if epoch.Timestamp != nil {
ts, err = EpochToTime(*epoch.Timestamp, epoch.Precision)
if err != nil {
return err
}
}
bp.Points = epoch.Points
bp.Database = epoch.Database
bp.RetentionPolicy = epoch.RetentionPolicy
bp.Tags = epoch.Tags
bp.Timestamp = ts
bp.Precision = epoch.Precision
return nil
}(); err == nil {
return nil
}
if err := json.Unmarshal(b, &normal); err != nil {
return err
}
normal.Timestamp = SetPrecision(normal.Timestamp, normal.Precision)
bp.Points = normal.Points
bp.Database = normal.Database
bp.RetentionPolicy = normal.RetentionPolicy
bp.Tags = normal.Tags
bp.Timestamp = normal.Timestamp
bp.Precision = normal.Precision
return nil
}
func (c *Client) Query(q Query) (*Results, error) {
2015-01-26 21:12:58 +00:00
u := c.url
u.Path = "query"
values := u.Query()
2015-01-23 00:18:24 +00:00
values.Set("q", q.Command)
values.Set("db", q.Database)
2015-01-26 21:12:58 +00:00
u.RawQuery = values.Encode()
2015-01-23 00:18:24 +00:00
2015-02-11 20:31:15 +00:00
req, err := http.NewRequest("GET", u.String(), nil)
if err != nil {
return nil, err
}
if c.userAgent != "" {
req.Header.Set("User-Agent", c.userAgent)
}
resp, err := c.httpClient.Do(req)
2015-01-23 00:18:24 +00:00
if err != nil {
return nil, err
}
defer resp.Body.Close()
var results Results
dec := json.NewDecoder(resp.Body)
dec.UseNumber()
err = dec.Decode(&results)
2015-01-23 00:18:24 +00:00
if err != nil {
return nil, err
}
return &results, nil
}
func (c *Client) Write(bp BatchPoints) (*Results, error) {
c.url.Path = "write"
2015-01-23 20:37:53 +00:00
type data struct {
Points []Point `json:"points"`
Database string `json:"database"`
RetentionPolicy string `json:"retentionPolicy"`
2015-01-23 20:37:53 +00:00
}
b := []byte{}
err := json.Unmarshal(b, &bp)
2015-01-23 20:37:53 +00:00
2015-02-11 20:31:15 +00:00
req, err := http.NewRequest("POST", c.url.String(), bytes.NewBuffer(b))
if err != nil {
return nil, err
}
req.Header.Set("Content-Type", "application/json")
if c.userAgent != "" {
req.Header.Set("User-Agent", c.userAgent)
}
resp, err := c.httpClient.Do(req)
2015-01-23 20:37:53 +00:00
if err != nil {
return nil, err
}
defer resp.Body.Close()
var results Results
dec := json.NewDecoder(resp.Body)
dec.UseNumber()
err = dec.Decode(&results)
2015-01-23 20:37:53 +00:00
if err != nil {
return nil, err
}
return &results, nil
}
2015-01-26 21:12:58 +00:00
func (c *Client) Ping() (time.Duration, string, error) {
now := time.Now()
2015-01-26 21:12:58 +00:00
u := c.url
u.Path = "ping"
2015-02-11 20:31:15 +00:00
req, err := http.NewRequest("GET", u.String(), nil)
if err != nil {
return 0, "", err
}
if c.userAgent != "" {
req.Header.Set("User-Agent", c.userAgent)
}
resp, err := c.httpClient.Do(req)
if err != nil {
2015-01-26 21:12:58 +00:00
return 0, "", err
}
2015-01-26 21:12:58 +00:00
version := resp.Header.Get("X-Influxdb-Version")
return time.Since(now), version, nil
}
// Structs
// Result represents a resultset returned from a single statement.
type Result struct {
Series []influxql.Row
Err error
}
// MarshalJSON encodes the result into JSON.
func (r *Result) MarshalJSON() ([]byte, error) {
// Define a struct that outputs "error" as a string.
var o struct {
Series []influxql.Row `json:"series,omitempty"`
Err string `json:"error,omitempty"`
}
// Copy fields to output struct.
o.Series = r.Series
if r.Err != nil {
o.Err = r.Err.Error()
}
return json.Marshal(&o)
}
// UnmarshalJSON decodes the data into the Result struct
func (r *Result) UnmarshalJSON(b []byte) error {
var o struct {
Series []influxql.Row `json:"series,omitempty"`
Err string `json:"error,omitempty"`
}
dec := json.NewDecoder(bytes.NewBuffer(b))
dec.UseNumber()
err := dec.Decode(&o)
if err != nil {
return err
}
r.Series = o.Series
if o.Err != "" {
r.Err = errors.New(o.Err)
}
return nil
}
// Results represents a list of statement results.
type Results struct {
Results []Result
Err error
}
func (r Results) MarshalJSON() ([]byte, error) {
// Define a struct that outputs "error" as a string.
var o struct {
Results []Result `json:"results,omitempty"`
Err string `json:"error,omitempty"`
}
// Copy fields to output struct.
o.Results = r.Results
if r.Err != nil {
o.Err = r.Err.Error()
}
return json.Marshal(&o)
}
// UnmarshalJSON decodes the data into the Results struct
func (r *Results) UnmarshalJSON(b []byte) error {
var o struct {
Results []Result `json:"results,omitempty"`
Err string `json:"error,omitempty"`
}
dec := json.NewDecoder(bytes.NewBuffer(b))
dec.UseNumber()
err := dec.Decode(&o)
if err != nil {
return err
}
r.Results = o.Results
if o.Err != "" {
r.Err = errors.New(o.Err)
}
return nil
}
// Error returns the first error from any statement.
// Returns nil if no errors occurred on any statements.
func (a Results) Error() error {
2015-02-10 00:58:30 +00:00
if a.Err != nil {
return a.Err
}
for _, r := range a.Results {
if r.Err != nil {
return r.Err
}
}
return nil
}
2015-01-29 23:33:31 +00:00
// Timestamp is a custom type so we marshal JSON properly into UTC nanosecond time
type Timestamp time.Time
2015-01-29 23:33:31 +00:00
// Time returns the time represented by the Timestamp
func (t Timestamp) Time() time.Time {
return time.Time(t)
}
2015-01-29 23:33:31 +00:00
// MarshalJSON returns time in UTC with nanoseconds
func (t Timestamp) MarshalJSON() ([]byte, error) {
// Always send back in UTC with nanoseconds
s := t.Time().UTC().Format(time.RFC3339Nano)
return []byte(`"` + s + `"`), nil
}
2015-02-23 22:37:10 +00:00
// Point defines the fields that will be written to the database
type Point struct {
Name string `json:"name"`
Tags map[string]string `json:"tags"`
Timestamp Timestamp `json:"timestamp"`
2015-02-23 22:37:10 +00:00
Fields map[string]interface{} `json:"fields"`
Precision string `json:"precision"`
}
// UnmarshalJSON decodes the data into the Point struct
func (p *Point) UnmarshalJSON(b []byte) error {
var normal struct {
Name string `json:"name"`
Tags map[string]string `json:"tags"`
Timestamp time.Time `json:"timestamp"`
2015-01-29 23:22:43 +00:00
Precision string `json:"precision"`
2015-02-23 22:37:10 +00:00
Fields map[string]interface{} `json:"fields"`
}
var epoch struct {
Name string `json:"name"`
Tags map[string]string `json:"tags"`
Timestamp *int64 `json:"timestamp"`
Precision string `json:"precision"`
2015-02-23 22:37:10 +00:00
Fields map[string]interface{} `json:"fields"`
}
if err := func() error {
var err error
dec := json.NewDecoder(bytes.NewBuffer(b))
dec.UseNumber()
if err = dec.Decode(&epoch); err != nil {
return err
}
// Convert from epoch to time.Time, but only if Timestamp
// was actually set.
var ts time.Time
if epoch.Timestamp != nil {
ts, err = EpochToTime(*epoch.Timestamp, epoch.Precision)
if err != nil {
return err
}
}
p.Name = epoch.Name
p.Tags = epoch.Tags
p.Timestamp = Timestamp(ts)
p.Precision = epoch.Precision
2015-02-23 22:37:10 +00:00
p.Fields = normalizeFields(epoch.Fields)
return nil
}(); err == nil {
return nil
}
dec := json.NewDecoder(bytes.NewBuffer(b))
dec.UseNumber()
if err := dec.Decode(&normal); err != nil {
return err
}
2015-01-29 23:22:43 +00:00
normal.Timestamp = SetPrecision(normal.Timestamp, normal.Precision)
p.Name = normal.Name
p.Tags = normal.Tags
p.Timestamp = Timestamp(normal.Timestamp)
p.Precision = normal.Precision
2015-02-23 22:37:10 +00:00
p.Fields = normalizeFields(normal.Fields)
return nil
}
// Remove any notion of json.Number
2015-02-23 22:37:10 +00:00
func normalizeFields(fields map[string]interface{}) map[string]interface{} {
newFields := map[string]interface{}{}
2015-02-23 22:37:10 +00:00
for k, v := range fields {
switch v := v.(type) {
case json.Number:
jv, e := v.Float64()
if e != nil {
panic(fmt.Sprintf("unable to convert json.Number to float64: %s", e))
}
2015-02-23 22:37:10 +00:00
newFields[k] = jv
default:
2015-02-23 22:37:10 +00:00
newFields[k] = v
}
}
2015-02-23 22:37:10 +00:00
return newFields
}
// utility functions
func (c *Client) Addr() string {
return c.url.String()
}
// helper functions
2015-01-29 23:33:31 +00:00
// EpochToTime takes a unix epoch time and uses precision to return back a time.Time
2015-01-29 18:10:13 +00:00
func EpochToTime(epoch int64, precision string) (time.Time, error) {
if precision == "" {
precision = "s"
}
var t time.Time
switch precision {
case "h":
t = time.Unix(0, epoch*int64(time.Hour))
case "m":
t = time.Unix(0, epoch*int64(time.Minute))
case "s":
t = time.Unix(0, epoch*int64(time.Second))
case "ms":
t = time.Unix(0, epoch*int64(time.Millisecond))
case "u":
t = time.Unix(0, epoch*int64(time.Microsecond))
case "n":
t = time.Unix(0, epoch)
default:
return time.Time{}, fmt.Errorf("Unknowm precision %q", precision)
}
return t, nil
}
2015-01-29 23:22:43 +00:00
// SetPrecision will round a time to the specified precision
func SetPrecision(t time.Time, precision string) time.Time {
switch precision {
case "n":
case "u":
return t.Round(time.Microsecond)
case "ms":
return t.Round(time.Millisecond)
case "s":
return t.Round(time.Second)
case "m":
return t.Round(time.Minute)
case "h":
return t.Round(time.Hour)
}
return t
}
func detect(values ...string) string {
for _, v := range values {
if v != "" {
return v
}
}
return ""
}