mirror of
https://github.com/empayre/fleet.git
synced 2024-11-06 17:05:18 +00:00
018e10ea66
``` $ fleetctl config set address https://localhost:8080 [+] Set the "address" config key to "https://localhost:8080" in the "default" context $ fleetctl config set ignore_tls true [+] Set the "ignore_tls" config key to "true" in the "default" context $ fleetctl setup --email mike@arpaia.co --password "abc123" [+] Fleet setup successful and context configured! $ cat ~/.fleet/config contexts: default: address: https://localhost:8080 email: mike@arpaia.co ignore_tls: true token: eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzZXNzaW9uX2tleSI6IlUvdm05Vk9wSG0xUlA4SUtjQnBhb2ovWlo1TXppSEVXcFRCNFNPb2tHQnNLUFpDQXFieVpWWnpJb0UvczQzcWkyd1pHZXJOa29SNFVIQ2hNZUc0K09RPT0ifQ.rHawSN8JvD4jjWAPTYX2Ep9ZpMt3u4mSIQcu920C-_s $ fleetctl logout [+] Fleet logout successful and local token cleared! $ cat ~/.fleet/config contexts: default: address: https://localhost:8080 email: mike@arpaia.co ignore_tls: true token: "" ```
293 lines
6.5 KiB
Go
293 lines
6.5 KiB
Go
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"strconv"
|
|
|
|
"github.com/ghodss/yaml"
|
|
"github.com/kolide/kit/env"
|
|
"github.com/pkg/errors"
|
|
"github.com/urfave/cli"
|
|
)
|
|
|
|
type configFile struct {
|
|
Contexts map[string]Context `json:"contexts"`
|
|
}
|
|
|
|
type Context struct {
|
|
Address string `json:"address"`
|
|
Email string `json:"email"`
|
|
Token string `json:"token"`
|
|
IgnoreTLS bool `json:"ignore_tls"`
|
|
}
|
|
|
|
func configFlag() cli.Flag {
|
|
return cli.StringFlag{
|
|
Name: "config",
|
|
Value: fmt.Sprintf("%s/.fleet/config", env.String("HOME", "~/")),
|
|
EnvVar: "CONFIG",
|
|
Usage: "The path to the Fleet config file",
|
|
}
|
|
}
|
|
|
|
func contextFlag() cli.Flag {
|
|
return cli.StringFlag{
|
|
Name: "context",
|
|
Value: "default",
|
|
EnvVar: "CONTEXT",
|
|
Usage: "The Fleet config context",
|
|
}
|
|
}
|
|
|
|
func makeConfigIfNotExists(fp string) error {
|
|
if _, err := os.Stat(filepath.Dir(fp)); os.IsNotExist(err) {
|
|
if err := os.Mkdir(filepath.Dir(fp), 0700); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
_, err := os.OpenFile(fp, os.O_RDONLY|os.O_CREATE, 0600)
|
|
return err
|
|
}
|
|
|
|
func readConfig(fp string) (c configFile, err error) {
|
|
b, err := ioutil.ReadFile(fp)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
err = yaml.Unmarshal(b, &c)
|
|
|
|
if c.Contexts == nil {
|
|
c.Contexts = map[string]Context{
|
|
"default": Context{},
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func writeConfig(fp string, c configFile) error {
|
|
b, err := yaml.Marshal(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return ioutil.WriteFile(fp, b, 0400)
|
|
}
|
|
|
|
func getConfigValue(c *cli.Context, key string) (interface{}, error) {
|
|
var (
|
|
flContext string
|
|
flConfig string
|
|
)
|
|
|
|
flConfig = c.String("config")
|
|
flContext = c.String("context")
|
|
|
|
if err := makeConfigIfNotExists(flConfig); err != nil {
|
|
return nil, errors.Wrapf(err, "error verifying that config exists at %s", flConfig)
|
|
}
|
|
|
|
config, err := readConfig(flConfig)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "error reading config at %s", flConfig)
|
|
}
|
|
|
|
currentContext, ok := config.Contexts[flContext]
|
|
if !ok {
|
|
fmt.Printf("[+] Context %q not found, creating it with default values\n", flContext)
|
|
currentContext = Context{}
|
|
}
|
|
|
|
switch key {
|
|
case "address":
|
|
return currentContext.Address, nil
|
|
case "email":
|
|
return currentContext.Email, nil
|
|
case "token":
|
|
return currentContext.Token, nil
|
|
case "ignore_tls":
|
|
if currentContext.IgnoreTLS {
|
|
return true, nil
|
|
} else {
|
|
return false, nil
|
|
}
|
|
default:
|
|
return nil, fmt.Errorf("%q is an invalid key", key)
|
|
}
|
|
}
|
|
|
|
func setConfigValue(c *cli.Context, key, value string) error {
|
|
var (
|
|
flContext string
|
|
flConfig string
|
|
)
|
|
|
|
flConfig = c.String("config")
|
|
flContext = c.String("context")
|
|
|
|
if err := makeConfigIfNotExists(flConfig); err != nil {
|
|
return errors.Wrapf(err, "error verifying that config exists at %s", flConfig)
|
|
}
|
|
|
|
config, err := readConfig(flConfig)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "error reading config at %s", flConfig)
|
|
}
|
|
|
|
currentContext, ok := config.Contexts[flContext]
|
|
if !ok {
|
|
fmt.Printf("[+] Context %q not found, creating it with default values\n", flContext)
|
|
currentContext = Context{}
|
|
}
|
|
|
|
switch key {
|
|
case "address":
|
|
currentContext.Address = value
|
|
case "email":
|
|
currentContext.Email = value
|
|
case "token":
|
|
currentContext.Token = value
|
|
case "ignore_tls":
|
|
boolValue, err := strconv.ParseBool(value)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "error parsing %q as bool", value)
|
|
}
|
|
currentContext.IgnoreTLS = boolValue
|
|
default:
|
|
return fmt.Errorf("%q is an invalid option", key)
|
|
}
|
|
|
|
config.Contexts[flContext] = currentContext
|
|
|
|
if err := writeConfig(flConfig, config); err != nil {
|
|
return errors.Wrap(err, "error saving config file")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func configSetCommand() cli.Command {
|
|
var (
|
|
flAddress string
|
|
flEmail string
|
|
flToken string
|
|
flIgnoreTLS bool
|
|
)
|
|
return cli.Command{
|
|
Name: "set",
|
|
Usage: "Set config options",
|
|
UsageText: `fleetctl config set [options]`,
|
|
Flags: []cli.Flag{
|
|
configFlag(),
|
|
contextFlag(),
|
|
cli.StringFlag{
|
|
Name: "address",
|
|
EnvVar: "ADDRESS",
|
|
Value: "",
|
|
Destination: &flAddress,
|
|
Usage: "The address of the Fleet API",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "email",
|
|
EnvVar: "EMAIL",
|
|
Value: "",
|
|
Destination: &flEmail,
|
|
Usage: "The email to use when connecting to the Fleet API",
|
|
},
|
|
cli.StringFlag{
|
|
Name: "token",
|
|
EnvVar: "TOKEN",
|
|
Value: "",
|
|
Destination: &flToken,
|
|
Usage: "The Fleet API token",
|
|
},
|
|
cli.BoolFlag{
|
|
Name: "ignore-tls",
|
|
EnvVar: "IGNORE_TLS",
|
|
Destination: &flIgnoreTLS,
|
|
Usage: "Whether or not to ignore the validity of the Fleet TLS certificate",
|
|
},
|
|
},
|
|
Action: func(c *cli.Context) error {
|
|
set := false
|
|
|
|
if flAddress != "" {
|
|
set = true
|
|
if err := setConfigValue(c, "address", flAddress); err != nil {
|
|
return errors.Wrap(err, "error setting address")
|
|
}
|
|
fmt.Printf("[+] Set the address config key to %q in the %q context\n", flAddress, c.String("context"))
|
|
}
|
|
|
|
if flEmail != "" {
|
|
set = true
|
|
if err := setConfigValue(c, "email", flEmail); err != nil {
|
|
return errors.Wrap(err, "error setting email")
|
|
}
|
|
fmt.Printf("[+] Set the email config key to %q in the %q context\n", flEmail, c.String("context"))
|
|
}
|
|
|
|
if flToken != "" {
|
|
set = true
|
|
if err := setConfigValue(c, "token", flToken); err != nil {
|
|
return errors.Wrap(err, "error setting token")
|
|
}
|
|
fmt.Printf("[+] Set the token config key to %q in the %q context\n", flToken, c.String("context"))
|
|
}
|
|
|
|
if flIgnoreTLS {
|
|
set = true
|
|
if err := setConfigValue(c, "ignore_tls", "true"); err != nil {
|
|
return errors.Wrap(err, "error setting ignore_tls")
|
|
}
|
|
fmt.Printf("[+] Set the ignore_tls config key to \"true\" in the %q context\n", c.String("context"))
|
|
}
|
|
|
|
if !set {
|
|
return cli.ShowCommandHelp(c, "set")
|
|
}
|
|
|
|
return nil
|
|
},
|
|
}
|
|
}
|
|
|
|
func configGetCommand() cli.Command {
|
|
return cli.Command{
|
|
Name: "get",
|
|
Usage: "Get a config option",
|
|
UsageText: `fleetctl config get [options]`,
|
|
Flags: []cli.Flag{
|
|
configFlag(),
|
|
contextFlag(),
|
|
},
|
|
Action: func(c *cli.Context) error {
|
|
if len(c.Args()) != 1 {
|
|
return cli.ShowCommandHelp(c, "get")
|
|
}
|
|
|
|
key := c.Args()[0]
|
|
|
|
// validate key
|
|
switch key {
|
|
case "address", "email", "token", "ignore_tls":
|
|
default:
|
|
return cli.ShowCommandHelp(c, "get")
|
|
}
|
|
|
|
value, err := getConfigValue(c, key)
|
|
if err != nil {
|
|
return errors.Wrap(err, "error getting config value")
|
|
}
|
|
|
|
fmt.Printf(" %s.%s => %s\n", c.String("context"), key, value)
|
|
|
|
return nil
|
|
},
|
|
}
|
|
}
|