influxdb/cmd/influx/main.go

453 lines
9.7 KiB
Go

package main
import (
"context"
"fmt"
"io"
"io/ioutil"
"os"
"path/filepath"
"strings"
"sync"
"github.com/influxdata/influxdb"
"github.com/influxdata/influxdb/bolt"
"github.com/influxdata/influxdb/cmd/influx/config"
"github.com/influxdata/influxdb/cmd/influx/internal"
"github.com/influxdata/influxdb/http"
"github.com/influxdata/influxdb/internal/fs"
"github.com/influxdata/influxdb/kit/cli"
"github.com/influxdata/influxdb/kv"
"github.com/influxdata/influxdb/pkg/httpc"
"github.com/spf13/cobra"
"github.com/spf13/viper"
"go.uber.org/zap"
)
const maxTCPConnections = 128
func main() {
influxCmd := influxCmd()
if err := influxCmd.Execute(); err != nil {
seeHelp(influxCmd, nil)
os.Exit(1)
}
}
var (
httpClient *httpc.Client
)
func newHTTPClient() (*httpc.Client, error) {
if httpClient != nil {
return httpClient, nil
}
c, err := http.NewHTTPClient(flags.Host, flags.Token, flags.skipVerify)
if err != nil {
return nil, err
}
httpClient = c
return httpClient, nil
}
type (
cobraRunEFn func(cmd *cobra.Command, args []string) error
cobraRunEMiddleware func(fn cobraRunEFn) cobraRunEFn
genericCLIOptFn func(*genericCLIOpts)
)
type genericCLIOpts struct {
in io.Reader
w io.Writer
errW io.Writer
runEWrapFn cobraRunEMiddleware
}
func (o genericCLIOpts) newCmd(use string, runE func(*cobra.Command, []string) error) *cobra.Command {
cmd := &cobra.Command{
Args: cobra.NoArgs,
Use: use,
RunE: runE,
}
if runE != nil && o.runEWrapFn != nil {
cmd.RunE = o.runEWrapFn(runE)
}
cmd.SetOut(o.w)
cmd.SetIn(o.in)
cmd.SetErr(o.errW)
return cmd
}
func (o genericCLIOpts) newTabWriter() *internal.TabWriter {
return internal.NewTabWriter(o.w)
}
func in(r io.Reader) genericCLIOptFn {
return func(o *genericCLIOpts) {
o.in = r
}
}
func out(w io.Writer) genericCLIOptFn {
return func(o *genericCLIOpts) {
o.w = w
}
}
func err(w io.Writer) genericCLIOptFn {
return func(o *genericCLIOpts) {
o.errW = w
}
}
func runEMiddlware(mw cobraRunEMiddleware) genericCLIOptFn {
return func(o *genericCLIOpts) {
o.runEWrapFn = mw
}
}
type globalFlags struct {
config.Config
local bool
skipVerify bool
}
var flags globalFlags
type cmdInfluxBuilder struct {
genericCLIOpts
once sync.Once
}
func newInfluxCmdBuilder(optFns ...genericCLIOptFn) *cmdInfluxBuilder {
builder := new(cmdInfluxBuilder)
opt := genericCLIOpts{
in: os.Stdin,
w: os.Stdout,
errW: os.Stderr,
runEWrapFn: checkSetupRunEMiddleware(&flags),
}
for _, optFn := range optFns {
optFn(&opt)
}
builder.genericCLIOpts = opt
return builder
}
func (b *cmdInfluxBuilder) cmd(childCmdFns ...func(f *globalFlags, opt genericCLIOpts) *cobra.Command) *cobra.Command {
b.once.Do(func() {
// enforce that viper options only ever get set once
setViperOptions()
})
cmd := b.newCmd("influx", nil)
cmd.Short = "Influx Client"
cmd.SilenceUsage = true
for _, childCmd := range childCmdFns {
cmd.AddCommand(childCmd(&flags, b.genericCLIOpts))
}
fOpts := flagOpts{
{
DestP: &flags.Token,
Flag: "token",
Short: 't',
Desc: "API token to be used throughout client calls",
Persistent: true,
},
{
DestP: &flags.Host,
Flag: "host",
Default: "http://localhost:9999",
Desc: "HTTP address of Influx",
Persistent: true,
},
}
fOpts.mustRegister(cmd)
if flags.Token == "" {
// migration credential token
migrateOldCredential()
// this is after the flagOpts register b/c we don't want to show the default value
// in the usage display. This will add it as the config, then if a token flag
// is provided too, the flag will take precedence.
flags.Config = getConfigFromDefaultPath()
}
cmd.PersistentFlags().BoolVar(&flags.local, "local", false, "Run commands locally against the filesystem")
cmd.PersistentFlags().BoolVar(&flags.skipVerify, "skip-verify", false, "SkipVerify controls whether a client verifies the server's certificate chain and host name.")
// Update help description for all commands in command tree
walk(cmd, func(c *cobra.Command) {
c.Flags().BoolP("help", "h", false, fmt.Sprintf("Help for the %s command ", c.Name()))
})
// completion command goes last, after the walk, so that all
// commands have every flag listed in the bash|zsh completions.
cmd.AddCommand(completionCmd(cmd))
return cmd
}
func influxCmd(opts ...genericCLIOptFn) *cobra.Command {
builder := newInfluxCmdBuilder(opts...)
return builder.cmd(
cmdAuth,
cmdBackup,
cmdBucket,
cmdDelete,
cmdOrganization,
cmdPing,
cmdPkg,
cmdConfig,
cmdQuery,
cmdTranspile,
cmdREPL,
cmdSecret,
cmdSetup,
cmdTask,
cmdUser,
cmdWrite,
)
}
func fetchSubCommand(parent *cobra.Command, args []string) *cobra.Command {
var err error
var cmd *cobra.Command
// Workaround FAIL with "go test -v" or "cobra.test -test.v", see #155
if args == nil && filepath.Base(os.Args[0]) != "cobra.test" {
args = os.Args[1:]
}
if parent.TraverseChildren {
cmd, _, err = parent.Traverse(args)
} else {
cmd, _, err = parent.Find(args)
}
// return nil if any errs
if err != nil {
return nil
}
return cmd
}
func seeHelp(c *cobra.Command, args []string) {
if c = fetchSubCommand(c, args); c == nil {
return //return here, since cobra already handles the error
}
c.Printf("See '%s -h' for help\n", c.CommandPath())
}
func defaultConfigPath() (string, string, error) {
dir, err := fs.InfluxDir()
if err != nil {
return "", "", err
}
return filepath.Join(dir, http.DefaultConfigsFile), dir, nil
}
func getConfigFromDefaultPath() config.Config {
path, _, err := defaultConfigPath()
if err != nil {
return config.DefaultConfig
}
r, err := os.Open(path)
if err != nil {
return config.DefaultConfig
}
activated, _ := config.ParseActiveConfig(r)
return activated
}
func migrateOldCredential() {
dir, err := fs.InfluxDir()
if err != nil {
return // no need for migration
}
tokB, err := ioutil.ReadFile(filepath.Join(dir, http.DefaultTokenFile))
if err != nil {
return // no need for migration
}
err = writeConfigToPath(strings.TrimSpace(string(tokB)), "", filepath.Join(dir, http.DefaultConfigsFile), dir)
if err != nil {
return
}
// ignore the remove err
_ = os.Remove(filepath.Join(dir, http.DefaultTokenFile))
}
func writeConfigToPath(tok, org, path, dir string) error {
p := &config.DefaultConfig
p.Token = tok
p.Org = org
pp := map[string]config.Config{
"default": *p,
}
return config.LocalConfigsSVC{
Path: path,
Dir: dir,
}.WriteConfigs(pp)
}
func checkSetup(host string, skipVerify bool) error {
s := &http.SetupService{
Addr: host,
InsecureSkipVerify: skipVerify,
}
isOnboarding, err := s.IsOnboarding(context.Background())
if err != nil {
return err
}
if isOnboarding {
return fmt.Errorf("the instance at %q has not been setup. Please run `influx setup` before issuing any additional commands", host)
}
return nil
}
func checkSetupRunEMiddleware(f *globalFlags) cobraRunEMiddleware {
return func(fn cobraRunEFn) cobraRunEFn {
return func(cmd *cobra.Command, args []string) error {
err := fn(cmd, args)
if err == nil {
return nil
}
if setupErr := checkSetup(f.Host, f.skipVerify); setupErr != nil && influxdb.EUnauthorized != influxdb.ErrorCode(setupErr) {
return internal.ErrorFmt(setupErr)
}
return internal.ErrorFmt(err)
}
}
}
// walk calls f for c and all of its children.
func walk(c *cobra.Command, f func(*cobra.Command)) {
f(c)
for _, c := range c.Commands() {
walk(c, f)
}
}
func newLocalKVService() (*kv.Service, error) {
boltFile, err := fs.BoltFile()
if err != nil {
return nil, err
}
store := bolt.NewKVStore(zap.NewNop(), boltFile)
if err := store.Open(context.Background()); err != nil {
return nil, err
}
return kv.NewService(zap.NewNop(), store), nil
}
type organization struct {
id, name string
}
func (o *organization) register(cmd *cobra.Command, persistent bool) {
opts := flagOpts{
{
DestP: &o.id,
Flag: "org-id",
Desc: "The ID of the organization",
Persistent: persistent,
},
{
DestP: &o.name,
Flag: "org",
Short: 'o',
Desc: "The name of the organization",
Persistent: persistent,
},
}
opts.mustRegister(cmd)
}
func (o *organization) getID(orgSVC influxdb.OrganizationService) (influxdb.ID, error) {
if o.id != "" {
influxOrgID, err := influxdb.IDFromString(o.id)
if err != nil {
return 0, fmt.Errorf("invalid org ID provided: %s", err.Error())
}
return *influxOrgID, nil
} else if o.name != "" {
org, err := orgSVC.FindOrganization(context.Background(), influxdb.OrganizationFilter{
Name: &o.name,
})
if err != nil {
return 0, fmt.Errorf("%v", err)
}
return org.ID, nil
}
return 0, fmt.Errorf("failed to locate an organization id")
}
func (o *organization) validOrgFlags(f *globalFlags) error {
if o.id == "" && o.name == "" && f != nil {
o.name = f.Org
}
if o.id == "" && o.name == "" {
return fmt.Errorf("must specify org-id, or org name")
} else if o.id != "" && o.name != "" {
return fmt.Errorf("must specify org-id, or org name not both")
}
return nil
}
type flagOpts []cli.Opt
func (f flagOpts) mustRegister(cmd *cobra.Command) {
for i := range f {
envVar := f[i].Flag
if e := f[i].EnvVar; e != "" {
envVar = e
}
f[i].Desc = fmt.Sprintf(
"%s; Maps to env var $INFLUX_%s",
f[i].Desc,
strings.ToUpper(strings.Replace(envVar, "-", "_", -1)),
)
}
cli.BindOptions(cmd, f)
}
func setViperOptions() {
viper.SetEnvPrefix("INFLUX")
viper.AutomaticEnv()
viper.SetEnvKeyReplacer(strings.NewReplacer("-", "_"))
}
func newBucketService() (influxdb.BucketService, error) {
if flags.local {
return newLocalKVService()
}
client, err := newHTTPClient()
if err != nil {
return nil, err
}
return &http.BucketService{
Client: client,
}, nil
}