influxdb/cmd/influx/main.go

484 lines
14 KiB
Go
Raw Normal View History

package main
import (
"encoding/csv"
"encoding/json"
"flag"
"fmt"
"io"
"net/url"
"os"
"os/user"
"path/filepath"
"strconv"
"strings"
"text/tabwriter"
"github.com/influxdb/influxdb/client"
"github.com/peterh/liner"
)
// These variables are populated via the Go linker.
var (
version string = "0.9"
)
const (
default_host = "localhost"
default_port = 8086
default_format = "column"
)
2015-02-03 17:22:50 +00:00
type CommandLine struct {
Client *client.Client
Line *liner.State
2015-02-03 17:22:50 +00:00
Host string
Port int
Username string
Password string
Database string
Version string
Pretty bool // controls pretty print for json
Format string // controls the output format. Valid values are json, csv, or column
}
func main() {
2015-02-03 17:22:50 +00:00
c := CommandLine{}
fs := flag.NewFlagSet("default", flag.ExitOnError)
2015-02-03 17:22:50 +00:00
fs.StringVar(&c.Host, "host", default_host, "influxdb host to connect to")
fs.IntVar(&c.Port, "port", default_port, "influxdb port to connect to")
fs.StringVar(&c.Username, "username", c.Username, "username to connect to the server.")
fs.StringVar(&c.Password, "password", c.Password, `password to connect to the server. Leaving blank will prompt for password (--password="")`)
2015-02-03 17:22:50 +00:00
fs.StringVar(&c.Database, "database", c.Database, "database to connect to the server.")
fs.StringVar(&c.Format, "output", default_format, "format specifies the format of the server responses: json, csv, or column")
fs.Parse(os.Args[1:])
var promptForPassword bool
// determine if they set the password flag but provided no value
for _, v := range os.Args {
v = strings.ToLower(v)
if (strings.HasPrefix(v, "-password") || strings.HasPrefix(v, "--password")) && c.Password == "" {
promptForPassword = true
break
}
}
2015-01-23 23:34:05 +00:00
// TODO Determine if we are an ineractive shell or running commands
fmt.Println("InfluxDB shell " + version)
c.Line = liner.NewLiner()
defer c.Line.Close()
if promptForPassword {
p, e := c.Line.PasswordPrompt("password: ")
if e != nil {
fmt.Println("Unable to parse password.")
} else {
c.Password = p
}
}
c.connect("")
var historyFile string
usr, err := user.Current()
// Only load history if we can get the user
if err == nil {
2015-01-29 00:22:28 +00:00
historyFile = filepath.Join(usr.HomeDir, ".influx_history")
if f, err := os.Open(historyFile); err == nil {
c.Line.ReadHistory(f)
f.Close()
}
}
for {
l, e := c.Line.Prompt("> ")
if e != nil {
break
}
if c.ParseCommand(l) {
// write out the history
if len(historyFile) > 0 {
c.Line.AppendHistory(l)
if f, err := os.Create(historyFile); err == nil {
c.Line.WriteHistory(f)
f.Close()
}
}
} else {
break // exit main loop
}
}
}
2015-02-03 17:22:50 +00:00
func (c *CommandLine) ParseCommand(cmd string) bool {
lcmd := strings.TrimSpace(strings.ToLower(cmd))
switch {
case strings.HasPrefix(lcmd, "exit"):
// signal the program to exit
return false
case strings.HasPrefix(lcmd, "gopher"):
gopher()
case strings.HasPrefix(lcmd, "connect"):
c.connect(cmd)
case strings.HasPrefix(lcmd, "auth"):
c.SetAuth()
case strings.HasPrefix(lcmd, "help"):
help()
case strings.HasPrefix(lcmd, "format"):
c.SetFormat(cmd)
case strings.HasPrefix(lcmd, "settings"):
c.Settings()
case strings.HasPrefix(lcmd, "pretty"):
2015-02-03 17:22:50 +00:00
c.Pretty = !c.Pretty
if c.Pretty {
fmt.Println("Pretty print enabled")
} else {
fmt.Println("Pretty print disabled")
}
case strings.HasPrefix(lcmd, "use"):
2015-01-30 21:31:53 +00:00
c.use(cmd)
case lcmd == "":
break
default:
c.executeQuery(cmd)
}
return true
}
2015-02-03 17:22:50 +00:00
func (c *CommandLine) connect(cmd string) {
var cl *client.Client
if cmd != "" {
// Remove the "connect" keyword if it exists
cmd = strings.TrimSpace(strings.Replace(cmd, "connect", "", -1))
if cmd == "" {
return
}
if strings.Contains(cmd, ":") {
h := strings.Split(cmd, ":")
if i, e := strconv.Atoi(h[1]); e != nil {
fmt.Printf("Connect error: Invalid port number %q: %s\n", cmd, e)
return
} else {
2015-02-03 17:22:50 +00:00
c.Port = i
}
2015-02-03 16:34:13 +00:00
if h[0] == "" {
2015-02-03 17:22:50 +00:00
c.Host = default_host
2015-02-03 16:34:13 +00:00
} else {
2015-02-03 17:22:50 +00:00
c.Host = h[0]
2015-02-03 16:34:13 +00:00
}
} else {
2015-02-03 17:22:50 +00:00
c.Host = cmd
// If they didn't specify a port, always use the default port
2015-02-03 17:22:50 +00:00
c.Port = default_port
}
}
u := url.URL{
Scheme: "http",
}
2015-02-03 17:22:50 +00:00
if c.Port > 0 {
u.Host = fmt.Sprintf("%s:%d", c.Host, c.Port)
} else {
2015-02-03 17:22:50 +00:00
u.Host = c.Host
}
2015-02-03 17:22:50 +00:00
if c.Username != "" {
u.User = url.UserPassword(c.Username, c.Password)
}
cl, err := client.NewClient(
client.Config{
2015-02-11 20:31:15 +00:00
URL: u,
Username: c.Username,
Password: c.Password,
UserAgent: "InfluxDBShell/" + version,
})
if err != nil {
fmt.Printf("Could not create client %s", err)
2015-01-26 21:12:58 +00:00
return
}
2015-02-03 17:22:50 +00:00
c.Client = cl
if _, v, e := c.Client.Ping(); e != nil {
fmt.Printf("Failed to connect to %s\n", c.Client.Addr())
2015-01-26 21:12:58 +00:00
} else {
2015-02-03 17:22:50 +00:00
c.Version = v
fmt.Printf("Connected to %s version %s\n", c.Client.Addr(), c.Version)
2015-01-26 21:12:58 +00:00
}
}
func (c *CommandLine) SetAuth() {
u, e := c.Line.Prompt("username: ")
if e != nil {
fmt.Printf("Unable to process input: %s", e)
return
}
c.Username = strings.TrimSpace(u)
p, e := c.Line.PasswordPrompt("password: ")
if e != nil {
fmt.Printf("Unable to process input: %s", e)
return
}
c.Password = p
}
2015-02-03 17:22:50 +00:00
func (c *CommandLine) use(cmd string) {
args := strings.Split(strings.TrimSpace(cmd), " ")
2015-01-30 21:31:53 +00:00
if len(args) != 2 {
fmt.Printf("Could not parse database name from %q.\n", cmd)
return
}
d := args[1]
2015-02-03 17:22:50 +00:00
c.Database = d
2015-01-30 21:31:53 +00:00
fmt.Printf("Using database %s\n", d)
}
func (c *CommandLine) SetFormat(cmd string) {
// Remove the "format" keyword if it exists
cmd = strings.TrimSpace(strings.Replace(cmd, "format", "", -1))
// normalize cmd
cmd = strings.ToLower(cmd)
switch cmd {
case "json", "csv", "column":
c.Format = cmd
default:
fmt.Printf("Unknown format %q. Please use json, csv, or column.\n", cmd)
}
}
2015-02-03 17:22:50 +00:00
func (c *CommandLine) executeQuery(query string) {
results, err := c.Client.Query(client.Query{Command: query, Database: c.Database})
if err != nil {
fmt.Printf("ERR: %s\n", err)
return
}
2015-02-10 16:12:18 +00:00
c.FormatResults(results, os.Stdout)
if results.Error() != nil {
fmt.Printf("ERR: %s\n", results.Error())
2015-02-10 16:12:18 +00:00
if c.Database == "" {
fmt.Println("Warning: It is possible this error is due to not setting a database.")
fmt.Println(`Please set a database with the command "use <database>".`)
}
}
}
func (c *CommandLine) FormatResults(results *client.Results, w io.Writer) {
switch c.Format {
case "json":
WriteJSON(results, c.Pretty, w)
case "csv":
WriteCSV(results, w)
case "column":
WriteColumns(results, w)
default:
fmt.Fprintf(w, "Unknown output format %q.\n", c.Format)
}
}
func WriteJSON(results *client.Results, pretty bool, w io.Writer) {
var data []byte
var err error
if pretty {
data, err = json.MarshalIndent(results, "", " ")
} else {
data, err = json.Marshal(results)
}
if err != nil {
fmt.Fprintf(w, "Unable to parse json: %s\n", err)
return
}
fmt.Fprintln(w, string(data))
}
func WriteCSV(results *client.Results, w io.Writer) {
csvw := csv.NewWriter(w)
for _, result := range results.Results {
2015-02-03 22:10:32 +00:00
// Create a tabbed writer for each result as they won't always line up
rows := resultToCSV(result, "\t", false)
for _, r := range rows {
csvw.Write(strings.Split(r, "\t"))
}
csvw.Flush()
}
}
func WriteColumns(results *client.Results, w io.Writer) {
for _, result := range results.Results {
// Create a tabbed writer for each result a they won't always line up
w := new(tabwriter.Writer)
w.Init(os.Stdout, 0, 8, 1, '\t', 0)
2015-03-10 23:31:58 +00:00
csv := resultToCSV(result, "\t", false)
for _, r := range csv {
fmt.Fprintln(w, r)
}
w.Flush()
}
}
2015-02-04 17:03:21 +00:00
func resultToCSV(result client.Result, seperator string, headerLines bool) []string {
rows := []string{}
// Create a tabbed writer for each result a they won't always line up
for i, row := range result.Series {
// gather tags
2015-03-10 23:23:26 +00:00
var hasTags bool
tags := []string{}
for k, v := range row.Tags {
hasTags = true
tags = append(tags, fmt.Sprintf("%s=%s", k, v))
}
2015-03-10 23:23:26 +00:00
columnNames := []string{}
if hasTags {
columnNames = append([]string{"tags"}, columnNames...)
}
2015-03-10 23:23:26 +00:00
if row.Name != "" {
columnNames = append([]string{"name"}, columnNames...)
}
for _, column := range row.Columns {
columnNames = append(columnNames, column)
}
// Output a line seperator if we have more than one set or results
if i > 0 {
rows = append(rows, "")
}
rows = append(rows, strings.Join(columnNames, seperator))
if headerLines {
// create column underscores
lines := []string{}
for _, columnName := range columnNames {
lines = append(lines, strings.Repeat("-", len(columnName)))
}
rows = append(rows, strings.Join(lines, seperator))
}
for _, v := range row.Values {
var values []string
if row.Name != "" {
values = append(values, row.Name)
}
if len(tags) > 0 {
values = append(values, strings.Join(tags, ","))
}
for _, vv := range v {
values = append(values, interfaceToString(vv))
}
rows = append(rows, strings.Join(values, seperator))
}
}
return rows
}
func interfaceToString(v interface{}) string {
switch t := v.(type) {
case nil:
return ""
case bool:
return fmt.Sprintf("%v", v)
case int, int8, int16, int32, int64, uint, uint8, uint16, uint32, uint64, uintptr:
return fmt.Sprintf("%d", t)
case float32, float64:
return fmt.Sprintf("%v", t)
default:
return fmt.Sprintf("%v", t)
}
}
func (c *CommandLine) Settings() {
w := new(tabwriter.Writer)
w.Init(os.Stdout, 0, 8, 1, '\t', 0)
if c.Port > 0 {
fmt.Fprintf(w, "Host\t%s:%d\n", c.Host, c.Port)
} else {
fmt.Fprintf(w, "Host\t%s\n", c.Host)
}
fmt.Fprintf(w, "Username\t%s\n", c.Username)
fmt.Fprintf(w, "Database\t%s\n", c.Database)
fmt.Fprintf(w, "Pretty\t%v\n", c.Pretty)
fmt.Fprintf(w, "Format\t%s\n", c.Format)
fmt.Fprintln(w)
w.Flush()
}
func help() {
fmt.Println(`Usage:
connect <host:port> connect to another node
auth prompt for username and password
pretty toggle pretty print
use <db_name> set current databases
format <format> set the output format: json, csv, or column
settings output the current settings for the shell
exit quit the influx shell
show databases show database names
show series show series information
show measurements show measurement information
show tag keys show tag key information
show tag values show tag value information
a full list of influxql commands can be found at:
http://influxdb.com/docs
`)
}
2015-01-23 23:34:05 +00:00
2015-01-26 16:24:17 +00:00
func gopher() {
2015-01-23 23:34:05 +00:00
fmt.Println(`
2015-01-26 16:24:17 +00:00
.-::-::://:-::- .:/++/'
'://:-''/oo+//++o+/.://o- ./+:
.:-. '++- .o/ '+yydhy' o-
.:/. .h: :osoys .smMN- :/
-/:.' s- /MMMymh. '/y/ s'
-+s:'''' d -mMMms// '-/o:
-/++/++/////:. o: '... s- :s.
:+-+s-' ':/' 's- /+ 'o:
'+-'o: /ydhsh. '//. '-o- o-
.y. o: .MMMdm+y ':+++:::/+:.' s:
.-h/ y- 'sdmds'h -+ydds:::-.' 'h.
.//-.d' o: '.' 'dsNMMMNh:.:++' :y
+y. 'd 's. .s:mddds: ++ o/
'N- odd 'o/. './o-s-' .---+++' o-
'N' yNd .://:/:::::. -s -+/s/./s' 'o/'
so' .h '''' ////s: '+. .s +y'
os/-.y' 's' 'y::+ +d'
'.:o/ -+:-:.' so.---.'
o' 'd-.''/s'
.s' :y.''.y
-s mo:::'
:: yh
// '''' /M'
o+ .s///:/. 'N:
:+ /: -s' ho
's- -/s/:+/.+h' +h
ys' ':' '-. -d
oh .h
/o .s
s. .h
-y .d
m/ -h
+d /o
'N- y:
h: m.
s- -d
o- s+
+- 'm'
s/ oo--.
y- /s ':+'
s' 'od--' .d:
-+ ':o: ':+-/+
y- .:+- '
//o- '.:+/.
.-:+/' ''-/+/.
./:' ''.:o+/-'
.+o:/:/+-' ''.-+ooo/-'
o: -h///++////-.
/: .o/
//+ 'y
./sooy.
2015-01-23 23:34:05 +00:00
`)
}