2015-01-22 22:23:55 +00:00
|
|
|
package client
|
|
|
|
|
2015-01-22 23:40:32 +00:00
|
|
|
import (
|
2015-01-23 20:37:53 +00:00
|
|
|
"bytes"
|
2015-01-23 00:18:24 +00:00
|
|
|
"encoding/json"
|
2015-01-29 21:07:43 +00:00
|
|
|
"errors"
|
2015-01-29 18:10:13 +00:00
|
|
|
"fmt"
|
2015-06-01 22:22:12 +00:00
|
|
|
"io/ioutil"
|
2015-07-29 15:09:52 +00:00
|
|
|
"net"
|
2015-01-22 23:40:32 +00:00
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2015-07-29 15:09:52 +00:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
2015-01-22 23:40:32 +00:00
|
|
|
"time"
|
|
|
|
|
2015-01-29 21:07:43 +00:00
|
|
|
"github.com/influxdb/influxdb/influxql"
|
2015-06-01 22:22:12 +00:00
|
|
|
"github.com/influxdb/influxdb/tsdb"
|
2015-01-22 23:40:32 +00:00
|
|
|
)
|
|
|
|
|
2015-07-29 15:09:52 +00:00
|
|
|
const (
|
2015-08-03 18:40:48 +00:00
|
|
|
// DefaultHost is the default host used to connect to an InfluxDB instance
|
|
|
|
DefaultHost = "localhost"
|
|
|
|
|
|
|
|
// DefaultPort is the default port used to connect to an InfluxDB instance
|
|
|
|
DefaultPort = 8086
|
|
|
|
|
|
|
|
// DefaultTimeout is the default connection timeout used to connect to an InfluxDB instance
|
|
|
|
DefaultTimeout = 0
|
2015-07-29 15:09:52 +00:00
|
|
|
)
|
|
|
|
|
2015-03-07 14:36:13 +00:00
|
|
|
// Query is used to send a command to the server. Both Command and Database are required.
|
2015-03-06 17:29:32 +00:00
|
|
|
type Query struct {
|
|
|
|
Command string
|
|
|
|
Database string
|
|
|
|
}
|
|
|
|
|
2015-07-29 15:09:52 +00:00
|
|
|
// ParseConnectionString will parse a string to create a valid connection URL
|
|
|
|
func ParseConnectionString(path string, ssl bool) (url.URL, error) {
|
|
|
|
var host string
|
|
|
|
var port int
|
|
|
|
|
|
|
|
if strings.Contains(path, ":") {
|
|
|
|
h := strings.Split(path, ":")
|
|
|
|
i, e := strconv.Atoi(h[1])
|
|
|
|
if e != nil {
|
|
|
|
return url.URL{}, fmt.Errorf("invalid port number %q: %s\n", path, e)
|
|
|
|
}
|
|
|
|
port = i
|
|
|
|
if h[0] == "" {
|
2015-08-03 18:40:48 +00:00
|
|
|
host = DefaultHost
|
2015-07-29 15:09:52 +00:00
|
|
|
} else {
|
|
|
|
host = h[0]
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
host = path
|
|
|
|
// If they didn't specify a port, always use the default port
|
2015-08-03 18:40:48 +00:00
|
|
|
port = DefaultPort
|
2015-07-29 15:09:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
u := url.URL{
|
|
|
|
Scheme: "http",
|
|
|
|
}
|
|
|
|
if ssl {
|
|
|
|
u.Scheme = "https"
|
|
|
|
}
|
|
|
|
u.Host = net.JoinHostPort(host, strconv.Itoa(port))
|
|
|
|
|
|
|
|
return u, nil
|
|
|
|
}
|
|
|
|
|
2015-03-07 14:36:13 +00:00
|
|
|
// Config is used to specify what server to connect to.
|
|
|
|
// URL: The URL of the server connecting to.
|
|
|
|
// Username/Password are optional. They will be passed via basic auth if provided.
|
2015-03-09 15:51:47 +00:00
|
|
|
// UserAgent: If not provided, will default "InfluxDBClient",
|
2015-06-05 21:23:13 +00:00
|
|
|
// Timeout: If not provided, will default to 0 (no timeout)
|
2015-01-22 22:23:55 +00:00
|
|
|
type Config struct {
|
2015-02-11 20:31:15 +00:00
|
|
|
URL url.URL
|
|
|
|
Username string
|
|
|
|
Password string
|
|
|
|
UserAgent string
|
2015-05-28 19:10:21 +00:00
|
|
|
Timeout time.Duration
|
2015-09-09 20:17:48 +00:00
|
|
|
Precision string
|
2015-01-22 22:23:55 +00:00
|
|
|
}
|
|
|
|
|
2015-07-29 15:09:52 +00:00
|
|
|
// NewConfig will create a config to be used in connecting to the client
|
2015-08-06 16:46:25 +00:00
|
|
|
func NewConfig() Config {
|
|
|
|
return Config{
|
2015-08-03 18:47:48 +00:00
|
|
|
Timeout: DefaultTimeout,
|
2015-07-29 15:09:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-07 14:36:13 +00:00
|
|
|
// Client is used to make calls to the server.
|
2015-01-22 22:23:55 +00:00
|
|
|
type Client struct {
|
2015-01-23 22:49:23 +00:00
|
|
|
url url.URL
|
2015-01-23 18:05:04 +00:00
|
|
|
username string
|
|
|
|
password string
|
2015-01-22 23:40:32 +00:00
|
|
|
httpClient *http.Client
|
2015-02-11 20:31:15 +00:00
|
|
|
userAgent string
|
2015-09-09 20:17:48 +00:00
|
|
|
precision string
|
2015-01-22 22:23:55 +00:00
|
|
|
}
|
|
|
|
|
2015-06-01 22:22:12 +00:00
|
|
|
const (
|
|
|
|
ConsistencyOne = "one"
|
|
|
|
ConsistencyAll = "all"
|
|
|
|
ConsistencyQuorum = "quorum"
|
|
|
|
ConsistencyAny = "any"
|
|
|
|
)
|
|
|
|
|
2015-03-07 14:36:13 +00:00
|
|
|
// NewClient will instantiate and return a connected client to issue commands to the server.
|
2015-08-06 16:46:25 +00:00
|
|
|
func NewClient(c Config) (*Client, error) {
|
2015-01-22 23:40:32 +00:00
|
|
|
client := Client{
|
2015-01-23 22:49:23 +00:00
|
|
|
url: c.URL,
|
2015-01-23 18:05:04 +00:00
|
|
|
username: c.Username,
|
|
|
|
password: c.Password,
|
2015-05-28 19:10:21 +00:00
|
|
|
httpClient: &http.Client{Timeout: c.Timeout},
|
2015-02-11 20:31:15 +00:00
|
|
|
userAgent: c.UserAgent,
|
2015-09-09 20:17:48 +00:00
|
|
|
precision: c.Precision,
|
2015-01-22 23:40:32 +00:00
|
|
|
}
|
2015-03-07 14:31:23 +00:00
|
|
|
if client.userAgent == "" {
|
2015-03-09 15:51:47 +00:00
|
|
|
client.userAgent = "InfluxDBClient"
|
2015-03-07 14:31:23 +00:00
|
|
|
}
|
2015-01-22 23:40:32 +00:00
|
|
|
return &client, nil
|
2015-01-22 22:23:55 +00:00
|
|
|
}
|
|
|
|
|
2015-04-13 17:08:26 +00:00
|
|
|
// SetAuth will update the username and passwords
|
|
|
|
func (c *Client) SetAuth(u, p string) {
|
|
|
|
c.username = u
|
|
|
|
c.password = p
|
|
|
|
}
|
|
|
|
|
2015-09-11 08:04:18 +00:00
|
|
|
// SetPrecision will update the precision
|
|
|
|
func (c *Client) SetPrecision(precision string) {
|
2015-09-15 21:09:58 +00:00
|
|
|
c.precision = precision
|
2015-09-11 08:04:18 +00:00
|
|
|
}
|
|
|
|
|
2015-03-31 16:26:00 +00:00
|
|
|
// Query sends a command to the server and returns the Response
|
|
|
|
func (c *Client) Query(q Query) (*Response, 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-09-09 20:54:57 +00:00
|
|
|
if c.precision != "" {
|
|
|
|
values.Set("epoch", c.precision)
|
|
|
|
}
|
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
|
|
|
|
}
|
2015-03-07 14:31:23 +00:00
|
|
|
req.Header.Set("User-Agent", c.userAgent)
|
2015-04-13 17:08:26 +00:00
|
|
|
if c.username != "" {
|
|
|
|
req.SetBasicAuth(c.username, c.password)
|
|
|
|
}
|
|
|
|
|
2015-02-11 20:31:15 +00:00
|
|
|
resp, err := c.httpClient.Do(req)
|
2015-01-23 00:18:24 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
2015-03-31 16:26:00 +00:00
|
|
|
var response Response
|
2015-02-03 00:20:34 +00:00
|
|
|
dec := json.NewDecoder(resp.Body)
|
|
|
|
dec.UseNumber()
|
2015-04-13 21:40:45 +00:00
|
|
|
decErr := dec.Decode(&response)
|
2015-04-13 21:20:26 +00:00
|
|
|
|
2015-04-13 21:40:45 +00:00
|
|
|
// 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 {
|
|
|
|
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.Error() == nil {
|
|
|
|
return &response, fmt.Errorf("received status code %d from server", resp.StatusCode)
|
|
|
|
}
|
2015-03-31 16:26:00 +00:00
|
|
|
return &response, nil
|
2015-01-22 22:23:55 +00:00
|
|
|
}
|
|
|
|
|
2015-03-07 14:36:13 +00:00
|
|
|
// Write takes BatchPoints and allows for writing of multiple points with defaults
|
2015-03-31 16:26:00 +00:00
|
|
|
// If successful, error is nil and Response is nil
|
|
|
|
// If an error occurs, Response may contain additional information if populated.
|
|
|
|
func (c *Client) Write(bp BatchPoints) (*Response, error) {
|
2015-08-03 18:40:48 +00:00
|
|
|
u := c.url
|
|
|
|
u.Path = "write"
|
2015-01-23 20:37:53 +00:00
|
|
|
|
2015-06-01 22:22:12 +00:00
|
|
|
var b bytes.Buffer
|
|
|
|
for _, p := range bp.Points {
|
2015-06-05 21:51:05 +00:00
|
|
|
if p.Raw != "" {
|
|
|
|
if _, err := b.WriteString(p.Raw); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
} else {
|
2015-06-24 21:20:40 +00:00
|
|
|
for k, v := range bp.Tags {
|
2015-06-29 00:45:47 +00:00
|
|
|
if p.Tags == nil {
|
|
|
|
p.Tags = make(map[string]string, len(bp.Tags))
|
|
|
|
}
|
2015-06-24 21:20:40 +00:00
|
|
|
p.Tags[k] = v
|
|
|
|
}
|
|
|
|
|
2015-06-05 21:51:05 +00:00
|
|
|
if _, err := b.WriteString(p.MarshalString()); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-06-01 22:22:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if err := b.WriteByte('\n'); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-03-06 19:44:17 +00:00
|
|
|
}
|
2015-01-23 20:37:53 +00:00
|
|
|
|
2015-08-03 18:40:48 +00:00
|
|
|
req, err := http.NewRequest("POST", u.String(), &b)
|
2015-02-11 20:31:15 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-06-09 22:45:26 +00:00
|
|
|
req.Header.Set("Content-Type", "")
|
2015-03-07 14:31:23 +00:00
|
|
|
req.Header.Set("User-Agent", c.userAgent)
|
2015-04-23 00:15:39 +00:00
|
|
|
if c.username != "" {
|
|
|
|
req.SetBasicAuth(c.username, c.password)
|
|
|
|
}
|
2015-06-01 22:22:12 +00:00
|
|
|
params := req.URL.Query()
|
2015-08-03 18:40:48 +00:00
|
|
|
params.Set("db", bp.Database)
|
|
|
|
params.Set("rp", bp.RetentionPolicy)
|
|
|
|
params.Set("precision", bp.Precision)
|
|
|
|
params.Set("consistency", bp.WriteConsistency)
|
2015-06-01 22:22:12 +00:00
|
|
|
req.URL.RawQuery = params.Encode()
|
2015-04-23 00:15:39 +00:00
|
|
|
|
2015-02-11 20:31:15 +00:00
|
|
|
resp, err := c.httpClient.Do(req)
|
2015-07-29 15:09:52 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
var response Response
|
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
2015-08-03 18:40:48 +00:00
|
|
|
if err != nil {
|
2015-07-29 15:09:52 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusNoContent && resp.StatusCode != http.StatusOK {
|
|
|
|
var err = fmt.Errorf(string(body))
|
|
|
|
response.Err = err
|
|
|
|
return &response, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// WriteLineProtocol takes a string with line returns to delimit each write
|
|
|
|
// If successful, error is nil and Response is nil
|
|
|
|
// If an error occurs, Response may contain additional information if populated.
|
|
|
|
func (c *Client) WriteLineProtocol(data, database, retentionPolicy, precision, writeConsistency string) (*Response, error) {
|
2015-08-03 18:40:48 +00:00
|
|
|
u := c.url
|
|
|
|
u.Path = "write"
|
2015-07-29 15:09:52 +00:00
|
|
|
|
2015-08-03 18:40:48 +00:00
|
|
|
r := strings.NewReader(data)
|
2015-07-29 15:09:52 +00:00
|
|
|
|
2015-08-03 18:40:48 +00:00
|
|
|
req, err := http.NewRequest("POST", u.String(), r)
|
2015-07-29 15:09:52 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
req.Header.Set("Content-Type", "")
|
|
|
|
req.Header.Set("User-Agent", c.userAgent)
|
|
|
|
if c.username != "" {
|
|
|
|
req.SetBasicAuth(c.username, c.password)
|
|
|
|
}
|
|
|
|
params := req.URL.Query()
|
2015-08-03 18:40:48 +00:00
|
|
|
params.Set("db", database)
|
|
|
|
params.Set("rp", retentionPolicy)
|
|
|
|
params.Set("precision", precision)
|
|
|
|
params.Set("consistency", writeConsistency)
|
2015-07-29 15:09:52 +00:00
|
|
|
req.URL.RawQuery = params.Encode()
|
|
|
|
|
|
|
|
resp, err := c.httpClient.Do(req)
|
2015-01-23 20:37:53 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
2015-03-31 16:26:00 +00:00
|
|
|
var response Response
|
2015-06-01 22:22:12 +00:00
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
2015-08-03 18:40:48 +00:00
|
|
|
if err != nil {
|
2015-03-07 02:34:31 +00:00
|
|
|
return nil, err
|
2015-03-06 23:21:27 +00:00
|
|
|
}
|
2015-02-03 00:20:34 +00:00
|
|
|
|
2015-06-01 22:22:12 +00:00
|
|
|
if resp.StatusCode != http.StatusNoContent && resp.StatusCode != http.StatusOK {
|
2015-08-03 18:40:48 +00:00
|
|
|
err := fmt.Errorf(string(body))
|
2015-06-01 22:22:12 +00:00
|
|
|
response.Err = err
|
|
|
|
return &response, err
|
2015-01-23 20:37:53 +00:00
|
|
|
}
|
2015-03-06 23:21:27 +00:00
|
|
|
|
2015-03-07 14:36:13 +00:00
|
|
|
return nil, nil
|
2015-01-22 22:23:55 +00:00
|
|
|
}
|
2015-01-22 23:40:32 +00:00
|
|
|
|
2015-03-07 14:43:22 +00:00
|
|
|
// Ping will check to see if the server is up
|
2015-06-28 06:54:34 +00:00
|
|
|
// Ping returns how long the request took, the version of the server it connected to, and an error if one occurred.
|
2015-01-26 21:12:58 +00:00
|
|
|
func (c *Client) Ping() (time.Duration, string, error) {
|
2015-01-22 23:40:32 +00:00
|
|
|
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
|
|
|
|
}
|
2015-03-07 14:31:23 +00:00
|
|
|
req.Header.Set("User-Agent", c.userAgent)
|
2015-04-23 00:15:39 +00:00
|
|
|
if c.username != "" {
|
|
|
|
req.SetBasicAuth(c.username, c.password)
|
|
|
|
}
|
|
|
|
|
2015-02-11 20:31:15 +00:00
|
|
|
resp, err := c.httpClient.Do(req)
|
2015-01-22 23:40:32 +00:00
|
|
|
if err != nil {
|
2015-01-26 21:12:58 +00:00
|
|
|
return 0, "", err
|
2015-01-22 23:40:32 +00:00
|
|
|
}
|
2015-05-12 18:15:08 +00:00
|
|
|
defer resp.Body.Close()
|
|
|
|
|
2015-01-26 21:12:58 +00:00
|
|
|
version := resp.Header.Get("X-Influxdb-Version")
|
|
|
|
return time.Since(now), version, nil
|
2015-01-22 23:40:32 +00:00
|
|
|
}
|
|
|
|
|
2015-01-29 21:07:43 +00:00
|
|
|
// Structs
|
|
|
|
|
|
|
|
// Result represents a resultset returned from a single statement.
|
|
|
|
type Result struct {
|
2015-02-23 05:21:49 +00:00
|
|
|
Series []influxql.Row
|
|
|
|
Err error
|
2015-01-29 21:07:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON encodes the result into JSON.
|
|
|
|
func (r *Result) MarshalJSON() ([]byte, error) {
|
|
|
|
// Define a struct that outputs "error" as a string.
|
|
|
|
var o struct {
|
2015-02-23 05:21:49 +00:00
|
|
|
Series []influxql.Row `json:"series,omitempty"`
|
|
|
|
Err string `json:"error,omitempty"`
|
2015-01-29 21:07:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Copy fields to output struct.
|
2015-02-23 05:21:49 +00:00
|
|
|
o.Series = r.Series
|
2015-01-29 21:07:43 +00:00
|
|
|
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 {
|
2015-02-23 05:21:49 +00:00
|
|
|
Series []influxql.Row `json:"series,omitempty"`
|
|
|
|
Err string `json:"error,omitempty"`
|
2015-01-29 21:07:43 +00:00
|
|
|
}
|
|
|
|
|
2015-02-03 00:20:34 +00:00
|
|
|
dec := json.NewDecoder(bytes.NewBuffer(b))
|
|
|
|
dec.UseNumber()
|
|
|
|
err := dec.Decode(&o)
|
2015-01-29 21:07:43 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-02-23 05:21:49 +00:00
|
|
|
r.Series = o.Series
|
2015-01-29 21:07:43 +00:00
|
|
|
if o.Err != "" {
|
|
|
|
r.Err = errors.New(o.Err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-03-31 16:26:00 +00:00
|
|
|
// Response represents a list of statement results.
|
|
|
|
type Response struct {
|
2015-02-03 00:20:34 +00:00
|
|
|
Results []Result
|
2015-01-29 21:07:43 +00:00
|
|
|
Err error
|
|
|
|
}
|
|
|
|
|
2015-03-31 16:26:00 +00:00
|
|
|
// MarshalJSON encodes the response into JSON.
|
|
|
|
func (r *Response) MarshalJSON() ([]byte, error) {
|
2015-01-29 21:07:43 +00:00
|
|
|
// Define a struct that outputs "error" as a string.
|
|
|
|
var o struct {
|
2015-02-03 00:20:34 +00:00
|
|
|
Results []Result `json:"results,omitempty"`
|
|
|
|
Err string `json:"error,omitempty"`
|
2015-01-29 21:07:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Copy fields to output struct.
|
|
|
|
o.Results = r.Results
|
|
|
|
if r.Err != nil {
|
|
|
|
o.Err = r.Err.Error()
|
|
|
|
}
|
|
|
|
|
|
|
|
return json.Marshal(&o)
|
|
|
|
}
|
|
|
|
|
2015-03-31 16:26:00 +00:00
|
|
|
// UnmarshalJSON decodes the data into the Response struct
|
|
|
|
func (r *Response) UnmarshalJSON(b []byte) error {
|
2015-01-29 21:07:43 +00:00
|
|
|
var o struct {
|
2015-02-03 00:20:34 +00:00
|
|
|
Results []Result `json:"results,omitempty"`
|
|
|
|
Err string `json:"error,omitempty"`
|
2015-01-29 21:07:43 +00:00
|
|
|
}
|
|
|
|
|
2015-02-03 00:20:34 +00:00
|
|
|
dec := json.NewDecoder(bytes.NewBuffer(b))
|
|
|
|
dec.UseNumber()
|
|
|
|
err := dec.Decode(&o)
|
2015-01-29 21:07:43 +00:00
|
|
|
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.
|
2015-03-31 16:26:00 +00:00
|
|
|
func (r Response) Error() error {
|
2015-03-22 05:33:04 +00:00
|
|
|
if r.Err != nil {
|
|
|
|
return r.Err
|
2015-02-10 00:58:30 +00:00
|
|
|
}
|
2015-03-22 05:33:04 +00:00
|
|
|
for _, result := range r.Results {
|
|
|
|
if result.Err != nil {
|
|
|
|
return result.Err
|
2015-01-29 21:07:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-02-23 22:37:10 +00:00
|
|
|
// Point defines the fields that will be written to the database
|
2015-05-21 22:59:26 +00:00
|
|
|
// Measurement, Time, and Fields are required
|
2015-04-23 17:44:16 +00:00
|
|
|
// Precision can be specified if the time is in epoch format (integer).
|
2015-03-07 14:43:22 +00:00
|
|
|
// Valid values for Precision are n, u, ms, s, m, and h
|
2015-01-29 21:07:43 +00:00
|
|
|
type Point struct {
|
2015-05-21 22:59:26 +00:00
|
|
|
Measurement string
|
|
|
|
Tags map[string]string
|
|
|
|
Time time.Time
|
|
|
|
Fields map[string]interface{}
|
|
|
|
Precision string
|
2015-06-05 21:51:05 +00:00
|
|
|
Raw string
|
2015-01-29 21:07:43 +00:00
|
|
|
}
|
|
|
|
|
2015-03-06 18:20:30 +00:00
|
|
|
// MarshalJSON will format the time in RFC3339Nano
|
2015-03-06 19:04:26 +00:00
|
|
|
// Precision is also ignored as it is only used for writing, not reading
|
|
|
|
// Or another way to say it is we always send back in nanosecond precision
|
2015-03-06 18:20:30 +00:00
|
|
|
func (p *Point) MarshalJSON() ([]byte, error) {
|
|
|
|
point := struct {
|
2015-05-21 22:59:26 +00:00
|
|
|
Measurement string `json:"measurement,omitempty"`
|
|
|
|
Tags map[string]string `json:"tags,omitempty"`
|
|
|
|
Time string `json:"time,omitempty"`
|
|
|
|
Fields map[string]interface{} `json:"fields,omitempty"`
|
|
|
|
Precision string `json:"precision,omitempty"`
|
2015-03-06 18:20:30 +00:00
|
|
|
}{
|
2015-05-21 22:59:26 +00:00
|
|
|
Measurement: p.Measurement,
|
|
|
|
Tags: p.Tags,
|
|
|
|
Fields: p.Fields,
|
|
|
|
Precision: p.Precision,
|
2015-03-06 19:04:26 +00:00
|
|
|
}
|
|
|
|
// Let it omit empty if it's really zero
|
2015-04-23 17:44:16 +00:00
|
|
|
if !p.Time.IsZero() {
|
|
|
|
point.Time = p.Time.UTC().Format(time.RFC3339Nano)
|
2015-03-06 18:20:30 +00:00
|
|
|
}
|
|
|
|
return json.Marshal(&point)
|
|
|
|
}
|
|
|
|
|
2015-06-01 22:22:12 +00:00
|
|
|
func (p *Point) MarshalString() string {
|
2015-06-02 16:40:52 +00:00
|
|
|
return tsdb.NewPoint(p.Measurement, p.Tags, p.Fields, p.Time).String()
|
2015-06-01 22:22:12 +00:00
|
|
|
}
|
|
|
|
|
2015-01-29 21:07:43 +00:00
|
|
|
// UnmarshalJSON decodes the data into the Point struct
|
|
|
|
func (p *Point) UnmarshalJSON(b []byte) error {
|
|
|
|
var normal struct {
|
2015-05-21 22:59:26 +00:00
|
|
|
Measurement string `json:"measurement"`
|
|
|
|
Tags map[string]string `json:"tags"`
|
|
|
|
Time time.Time `json:"time"`
|
|
|
|
Precision string `json:"precision"`
|
|
|
|
Fields map[string]interface{} `json:"fields"`
|
2015-01-29 21:07:43 +00:00
|
|
|
}
|
|
|
|
var epoch struct {
|
2015-05-21 22:59:26 +00:00
|
|
|
Measurement string `json:"measurement"`
|
|
|
|
Tags map[string]string `json:"tags"`
|
|
|
|
Time *int64 `json:"time"`
|
|
|
|
Precision string `json:"precision"`
|
|
|
|
Fields map[string]interface{} `json:"fields"`
|
2015-01-29 21:07:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if err := func() error {
|
|
|
|
var err error
|
2015-02-03 00:20:34 +00:00
|
|
|
dec := json.NewDecoder(bytes.NewBuffer(b))
|
|
|
|
dec.UseNumber()
|
|
|
|
if err = dec.Decode(&epoch); err != nil {
|
2015-01-29 21:07:43 +00:00
|
|
|
return err
|
|
|
|
}
|
2015-04-23 17:44:16 +00:00
|
|
|
// Convert from epoch to time.Time, but only if Time
|
2015-02-04 02:30:05 +00:00
|
|
|
// was actually set.
|
|
|
|
var ts time.Time
|
2015-04-23 17:44:16 +00:00
|
|
|
if epoch.Time != nil {
|
|
|
|
ts, err = EpochToTime(*epoch.Time, epoch.Precision)
|
2015-02-04 02:30:05 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-01-29 21:07:43 +00:00
|
|
|
}
|
2015-05-21 22:59:26 +00:00
|
|
|
p.Measurement = epoch.Measurement
|
2015-01-29 21:07:43 +00:00
|
|
|
p.Tags = epoch.Tags
|
2015-04-23 17:44:16 +00:00
|
|
|
p.Time = ts
|
2015-01-29 23:44:10 +00:00
|
|
|
p.Precision = epoch.Precision
|
2015-02-23 22:37:10 +00:00
|
|
|
p.Fields = normalizeFields(epoch.Fields)
|
2015-01-29 21:07:43 +00:00
|
|
|
return nil
|
|
|
|
}(); err == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-02-03 00:20:34 +00:00
|
|
|
dec := json.NewDecoder(bytes.NewBuffer(b))
|
|
|
|
dec.UseNumber()
|
|
|
|
if err := dec.Decode(&normal); err != nil {
|
2015-01-29 21:07:43 +00:00
|
|
|
return err
|
|
|
|
}
|
2015-04-23 17:44:16 +00:00
|
|
|
normal.Time = SetPrecision(normal.Time, normal.Precision)
|
2015-05-21 22:59:26 +00:00
|
|
|
p.Measurement = normal.Measurement
|
2015-01-29 21:07:43 +00:00
|
|
|
p.Tags = normal.Tags
|
2015-04-23 17:44:16 +00:00
|
|
|
p.Time = normal.Time
|
2015-01-29 23:44:10 +00:00
|
|
|
p.Precision = normal.Precision
|
2015-02-23 22:37:10 +00:00
|
|
|
p.Fields = normalizeFields(normal.Fields)
|
2015-01-29 21:07:43 +00:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-02-03 00:20:34 +00:00
|
|
|
// 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-03 00:20:34 +00:00
|
|
|
|
2015-02-23 22:37:10 +00:00
|
|
|
for k, v := range fields {
|
2015-02-03 00:20:34 +00:00
|
|
|
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
|
2015-02-03 00:20:34 +00:00
|
|
|
default:
|
2015-02-23 22:37:10 +00:00
|
|
|
newFields[k] = v
|
2015-02-03 00:20:34 +00:00
|
|
|
}
|
|
|
|
}
|
2015-02-23 22:37:10 +00:00
|
|
|
return newFields
|
2015-02-03 00:20:34 +00:00
|
|
|
}
|
|
|
|
|
2015-03-06 18:20:30 +00:00
|
|
|
// BatchPoints is used to send batched data in a single write.
|
2015-03-07 14:43:22 +00:00
|
|
|
// Database and Points are required
|
|
|
|
// If no retention policy is specified, it will use the databases default retention policy.
|
|
|
|
// If tags are specified, they will be "merged" with all points. If a point already has that tag, it is ignored.
|
2015-04-23 17:44:16 +00:00
|
|
|
// If time is specified, it will be applied to any point with an empty time.
|
|
|
|
// Precision can be specified if the time is in epoch format (integer).
|
2015-03-07 14:43:22 +00:00
|
|
|
// Valid values for Precision are n, u, ms, s, m, and h
|
2015-03-06 18:20:30 +00:00
|
|
|
type BatchPoints struct {
|
2015-06-01 22:22:12 +00:00
|
|
|
Points []Point `json:"points,omitempty"`
|
|
|
|
Database string `json:"database,omitempty"`
|
|
|
|
RetentionPolicy string `json:"retentionPolicy,omitempty"`
|
|
|
|
Tags map[string]string `json:"tags,omitempty"`
|
|
|
|
Time time.Time `json:"time,omitempty"`
|
|
|
|
Precision string `json:"precision,omitempty"`
|
|
|
|
WriteConsistency string `json:"-"`
|
2015-03-06 18:20:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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"`
|
2015-04-23 17:44:16 +00:00
|
|
|
Time time.Time `json:"time"`
|
2015-03-06 18:20:30 +00:00
|
|
|
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"`
|
2015-04-23 17:44:16 +00:00
|
|
|
Time *int64 `json:"time"`
|
2015-03-06 18:20:30 +00:00
|
|
|
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
|
2015-04-23 17:44:16 +00:00
|
|
|
if epoch.Time != nil {
|
|
|
|
ts, err = EpochToTime(*epoch.Time, epoch.Precision)
|
2015-03-06 18:20:30 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bp.Points = epoch.Points
|
|
|
|
bp.Database = epoch.Database
|
|
|
|
bp.RetentionPolicy = epoch.RetentionPolicy
|
|
|
|
bp.Tags = epoch.Tags
|
2015-04-23 17:44:16 +00:00
|
|
|
bp.Time = ts
|
2015-03-06 18:20:30 +00:00
|
|
|
bp.Precision = epoch.Precision
|
|
|
|
return nil
|
|
|
|
}(); err == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(b, &normal); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-04-23 17:44:16 +00:00
|
|
|
normal.Time = SetPrecision(normal.Time, normal.Precision)
|
2015-03-06 18:20:30 +00:00
|
|
|
bp.Points = normal.Points
|
|
|
|
bp.Database = normal.Database
|
|
|
|
bp.RetentionPolicy = normal.RetentionPolicy
|
|
|
|
bp.Tags = normal.Tags
|
2015-04-23 17:44:16 +00:00
|
|
|
bp.Time = normal.Time
|
2015-03-06 18:20:30 +00:00
|
|
|
bp.Precision = normal.Precision
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-01-22 23:40:32 +00:00
|
|
|
// utility functions
|
|
|
|
|
2015-03-07 14:43:22 +00:00
|
|
|
// Addr provides the current url as a string of the server the client is connected to.
|
2015-01-22 23:40:32 +00:00
|
|
|
func (c *Client) Addr() string {
|
2015-01-23 22:49:23 +00:00
|
|
|
return c.url.String()
|
2015-01-22 23:40:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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:
|
2015-05-29 17:16:59 +00:00
|
|
|
return time.Time{}, fmt.Errorf("Unknown precision %q", precision)
|
2015-01-29 18:10:13 +00:00
|
|
|
}
|
|
|
|
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
|
|
|
|
}
|