2015-03-27 15:56:00 +00:00
|
|
|
# InfluxDB Client
|
2015-01-22 22:08:30 +00:00
|
|
|
|
2015-03-27 15:56:00 +00:00
|
|
|
## Description
|
|
|
|
|
2015-03-27 20:01:45 +00:00
|
|
|
A Go client library written and maintained by the **InfluxDB** team.
|
2015-03-27 15:56:00 +00:00
|
|
|
This package provides convenience functions to read and write time series data.
|
|
|
|
It uses the HTTP protocol to communicate with your **InfluxDB** cluster.
|
|
|
|
|
|
|
|
|
|
|
|
## Getting Started
|
|
|
|
|
|
|
|
### Connecting To Your Database
|
|
|
|
|
2015-03-27 20:01:45 +00:00
|
|
|
Connecting to an **InfluxDB** database is straightforward. You will need a host
|
|
|
|
name, a port and the cluster user credentials if applicable. The default port is 8086.
|
|
|
|
You can customize these settings to your specific installation via the
|
|
|
|
**InfluxDB** configuration file.
|
|
|
|
|
|
|
|
Thought not necessary for experimentation, you may want to create a new user
|
|
|
|
and authenticate the connection to your database.
|
|
|
|
|
|
|
|
For more information please check out the
|
|
|
|
[Cluster Admin Docs](http://influxdb.com/docs/v0.9/query_language/database_administration.html).
|
|
|
|
|
|
|
|
For the impatients, you can create a new admin user _bubba_ by firing off the
|
|
|
|
[InfluxDB CLI](https://github.com/influxdb/influxdb/blob/master/cmd/influx/main.go).
|
|
|
|
|
|
|
|
```shell
|
|
|
|
influx
|
|
|
|
> create user bubba with password 'bumblebeetuna'
|
|
|
|
> grant all privileges to bubba
|
|
|
|
```
|
|
|
|
|
|
|
|
And now for good measure set the credentials in you shell environment.
|
|
|
|
In the example below we will use $INFLUX_USER and $INFLUX_PWD
|
|
|
|
|
|
|
|
Now with the "admintrivia" out of the way, let's connect to our database.
|
|
|
|
|
|
|
|
NOTE: If you've opt out of creating a user, you can ommit Username/Password out
|
|
|
|
of the configuration below.
|
2015-03-27 15:56:00 +00:00
|
|
|
|
|
|
|
```go
|
|
|
|
package main
|
|
|
|
|
|
|
|
import "github.com/influxdb/influxdb/client"
|
|
|
|
|
|
|
|
const (
|
|
|
|
MyHost = "localhost"
|
|
|
|
MyPort = 8086
|
|
|
|
MyDB = "square_holes"
|
|
|
|
MyMeasurement = "shapes"
|
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
u, err := url.Parse(fmt.Sprintf("http://%s:%d", MyHost, MyPort))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
2015-03-27 20:01:45 +00:00
|
|
|
|
2015-03-27 15:56:00 +00:00
|
|
|
conf := client.Config{
|
|
|
|
URL: *u,
|
2015-03-27 20:01:45 +00:00
|
|
|
Username: os.Getenv("INFLUX_USER"),
|
|
|
|
Password: os.Getenv("INFLUX_PWD"),
|
2015-03-27 15:56:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
con, err := client.NewClient(conf)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
dur, ver, err := con.Ping()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
log.Printf("Happy as a Hippo! %v, %s", dur, ver)
|
|
|
|
}
|
|
|
|
|
|
|
|
```
|
|
|
|
|
|
|
|
### Inserting Data
|
|
|
|
|
|
|
|
Time series data aka *points* are written to the database using batch inserts.
|
|
|
|
The mechanism is to create one or more points and then create a batch aka *batch points*
|
|
|
|
and write these to a given database and series. A series is a combination of a
|
|
|
|
measurement (time/values) and a set of tags.
|
|
|
|
|
|
|
|
In this sample we will create a batch of a 1k points. Each point as a timestamp and
|
|
|
|
a single value as well as 2 tags indicating a shape and color. We write these points
|
|
|
|
to a database called _square_holes_ using a measurement named _shapes_.
|
|
|
|
|
|
|
|
NOTE: In this example, we are specifically assigning time, tags and precision
|
|
|
|
to each point. Alternately, you can specify a timestamp, tags and precision at
|
2015-03-27 20:01:45 +00:00
|
|
|
the batch point level that could be used as defaults if an associated point
|
2015-03-27 15:56:00 +00:00
|
|
|
does not provide these metrics.
|
|
|
|
|
2015-03-27 20:01:45 +00:00
|
|
|
NOTE: You can specify a RetentionPolicy as part of the batch points. If not
|
|
|
|
provided it will use the database _default_ retention policy. If you did not
|
|
|
|
set a retention policy the default is forever (0).
|
2015-03-27 15:56:00 +00:00
|
|
|
|
|
|
|
```go
|
|
|
|
func writePoints(con *client.Client) {
|
|
|
|
var (
|
|
|
|
shapes = []string{"circle", "rectangle", "square", "triangle"}
|
|
|
|
colors = []string{"red", "blue", "green"}
|
|
|
|
sampleSize = 1000
|
|
|
|
pts = make([]client.Point, sampleSize)
|
|
|
|
)
|
|
|
|
|
|
|
|
rand.Seed(42)
|
|
|
|
for i := 0; i < sampleSize; i++ {
|
|
|
|
pts[i] = client.Point{
|
|
|
|
Name: "shapes",
|
|
|
|
Tags: map[string]string{
|
|
|
|
"color": strconv.Itoa(rand.Intn(len(colors))),
|
|
|
|
"shape": strconv.Itoa(rand.Intn(len(shapes))),
|
|
|
|
},
|
|
|
|
Fields: map[string]interface{}{
|
|
|
|
"value": rand.Intn(sampleSize),
|
|
|
|
},
|
|
|
|
Timestamp: time.Now(),
|
|
|
|
Precision: "s",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bps := client.BatchPoints{
|
|
|
|
Points: pts,
|
|
|
|
Database: MyDB,
|
2015-03-27 20:01:45 +00:00
|
|
|
RetentionPolicy: "default",
|
2015-03-27 15:56:00 +00:00
|
|
|
}
|
|
|
|
_, err := con.Write(bps)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
### Querying Data
|
|
|
|
|
|
|
|
One nice advantage of using **InfluxDB** is having the ability to query your data
|
|
|
|
using familiar SQL constructs. In this example we are providing a convenience
|
|
|
|
function to query the database as follows:
|
|
|
|
|
|
|
|
```go
|
|
|
|
// queryDB convenience to query the database
|
|
|
|
func queryDB(con *client.Client, cmd string) (res []client.Result, err error) {
|
|
|
|
q := client.Query{
|
|
|
|
Command: cmd,
|
|
|
|
Database: MyDB,
|
|
|
|
}
|
|
|
|
if results, err := con.Query(q); err == nil {
|
|
|
|
if results.Error() != nil {
|
|
|
|
return res, results.Error()
|
|
|
|
}
|
|
|
|
res = results.Results
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
#### Creating A Database
|
|
|
|
```go
|
|
|
|
_, err := queryDB(con, fmt.Sprintf("create database %s", MyDB))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
#### Count Records
|
|
|
|
```go
|
|
|
|
q := fmt.Sprintf("select count(%s) from %s", "value", MyMeasurement)
|
|
|
|
res, err := queryDB(con, q)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
count := res[0].Series[0].Values[0][1]
|
|
|
|
log.Printf("Found a total of `%v records", count)
|
|
|
|
|
|
|
|
```
|
|
|
|
|
|
|
|
#### Find the last 10 _shapes_ records
|
|
|
|
|
|
|
|
```go
|
|
|
|
q := fmt.Sprintf("select * from %s limit %d", MyMeasurement, 20)
|
|
|
|
res, err = queryDB(con, q)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, row := range res[0].Series[0].Values {
|
|
|
|
t, err := time.Parse(time.RFC3339, row[0].(string))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
val, err := row[1].(json.Number).Int64()
|
|
|
|
log.Printf("[%2d] %s: %03d\n", i, t.Format(time.Stamp), val)
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
## Go Docs
|
|
|
|
|
|
|
|
Please refer to
|
|
|
|
[http://godoc.org/github.com/influxdb/influxdb/client](http://godoc.org/github.com/influxdb/influxdb/client)
|
2015-03-09 19:07:50 +00:00
|
|
|
for documentation.
|
2015-03-07 14:57:30 +00:00
|
|
|
|
2015-03-07 14:43:22 +00:00
|
|
|
|
2015-03-27 15:56:00 +00:00
|
|
|
## See Also
|
|
|
|
|
|
|
|
You can see a use of the client libray in the
|
|
|
|
[InfluxDB CLI](https://github.com/influxdb/influxdb/blob/master/cmd/influx/main.go).
|