mirror of
https://github.com/davidallendj/magellan.git
synced 2025-12-20 11:37:01 -07:00
163 lines
5.3 KiB
Go
163 lines
5.3 KiB
Go
// The cmd package implements the interface for the magellan CLI. The files
|
|
// contained in this package only contains implementations for handling CLI
|
|
// arguments and passing them to functions within magellan's internal API.
|
|
//
|
|
// Each CLI subcommand will have at least one corresponding internal file
|
|
// with an API routine that implements the command's functionality. The main
|
|
// API routine will usually be the first function defined in the fill.
|
|
//
|
|
// For example:
|
|
//
|
|
// cmd/scan.go --> internal/scan.go ( magellan.ScanForAssets() )
|
|
// cmd/collect.go --> internal/collect.go ( magellan.CollectAll() )
|
|
// cmd/list.go --> none (doesn't have API call since it's simple)
|
|
// cmd/update.go --> internal/update.go ( magellan.UpdateFirmware() )
|
|
package cmd
|
|
|
|
import (
|
|
"fmt"
|
|
"net"
|
|
"os"
|
|
"os/user"
|
|
|
|
magellan "github.com/OpenCHAMI/magellan/internal"
|
|
"github.com/OpenCHAMI/magellan/internal/api/smd"
|
|
"github.com/spf13/cobra"
|
|
"github.com/spf13/viper"
|
|
)
|
|
|
|
var (
|
|
currentUser *user.User
|
|
accessToken string
|
|
format string
|
|
timeout int
|
|
concurrency int
|
|
ports []int
|
|
hosts []string
|
|
protocol string
|
|
cacertPath string
|
|
username string
|
|
password string
|
|
cachePath string
|
|
outputPath string
|
|
configPath string
|
|
verbose bool
|
|
)
|
|
|
|
// The `root` command doesn't do anything on it's own except display
|
|
// a help message and then exits.
|
|
var rootCmd = &cobra.Command{
|
|
Use: "magellan",
|
|
Short: "Tool for BMC discovery",
|
|
Long: "",
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
if len(args) == 0 {
|
|
cmd.Help()
|
|
os.Exit(0)
|
|
}
|
|
},
|
|
}
|
|
|
|
// This Execute() function is called from main to run the CLI.
|
|
func Execute() {
|
|
if err := rootCmd.Execute(); err != nil {
|
|
fmt.Fprintln(os.Stderr, err)
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
// LoadAccessToken() tries to load a JWT string from an environment
|
|
// variable, file, or config in that order. If loading the token
|
|
// fails with one options, it will fallback to the next option until
|
|
// all options are exhausted.
|
|
//
|
|
// Returns a token as a string with no error if successful.
|
|
// Alternatively, returns an empty string with an error if a token is
|
|
// not able to be loaded.
|
|
func LoadAccessToken() (string, error) {
|
|
// try to load token from env var
|
|
testToken := os.Getenv("ACCESS_TOKEN")
|
|
if testToken != "" {
|
|
return testToken, nil
|
|
}
|
|
|
|
// try reading access token from a file
|
|
b, err := os.ReadFile(tokenPath)
|
|
if err == nil {
|
|
return string(b), nil
|
|
}
|
|
|
|
// TODO: try to load token from config
|
|
testToken = viper.GetString("access_token")
|
|
if testToken != "" {
|
|
return testToken, nil
|
|
}
|
|
return "", fmt.Errorf("failed toload token from environment variable, file, or config")
|
|
}
|
|
|
|
func init() {
|
|
currentUser, _ = user.Current()
|
|
cobra.OnInitialize(InitializeConfig)
|
|
rootCmd.PersistentFlags().IntVar(&concurrency, "concurrency", -1, "set the number of concurrent processes")
|
|
rootCmd.PersistentFlags().IntVar(&timeout, "timeout", 30, "set the timeout")
|
|
rootCmd.PersistentFlags().StringVarP(&configPath, "config", "c", "", "set the config file path")
|
|
rootCmd.PersistentFlags().BoolVarP(&verbose, "verbose", "v", false, "set output verbosity")
|
|
rootCmd.PersistentFlags().StringVar(&accessToken, "access-token", "", "set the access token")
|
|
rootCmd.PersistentFlags().StringVar(&cachePath, "cache", fmt.Sprintf("/tmp/%smagellan/magellan.db", currentUser.Username+"/"), "set the scanning result cache path")
|
|
|
|
// bind viper config flags with cobra
|
|
viper.BindPFlag("concurrency", rootCmd.Flags().Lookup("concurrency"))
|
|
viper.BindPFlag("timeout", rootCmd.Flags().Lookup("timeout"))
|
|
viper.BindPFlag("verbose", rootCmd.Flags().Lookup("verbose"))
|
|
viper.BindPFlag("cache", rootCmd.Flags().Lookup("cache"))
|
|
viper.BindPFlags(rootCmd.Flags())
|
|
}
|
|
|
|
// InitializeConfig() initializes a new config object by loading it
|
|
// from a file given a non-empty string.
|
|
//
|
|
// See the 'LoadConfig' function in 'internal/config' for details.
|
|
func InitializeConfig() {
|
|
if configPath != "" {
|
|
magellan.LoadConfig(configPath)
|
|
}
|
|
}
|
|
|
|
// SetDefaults() resets all of the viper properties back to their
|
|
// default values.
|
|
//
|
|
// TODO: This function should probably be moved to 'internal/config.go'
|
|
// instead of in this file.
|
|
func SetDefaults() {
|
|
viper.SetDefault("threads", 1)
|
|
viper.SetDefault("timeout", 30)
|
|
viper.SetDefault("config", "")
|
|
viper.SetDefault("verbose", false)
|
|
viper.SetDefault("cache", "/tmp/magellan/magellan.db")
|
|
viper.SetDefault("scan.hosts", []string{})
|
|
viper.SetDefault("scan.ports", []int{})
|
|
viper.SetDefault("scan.subnets", []string{})
|
|
viper.SetDefault("scan.subnet-masks", []net.IP{})
|
|
viper.SetDefault("scan.disable-probing", false)
|
|
viper.SetDefault("collect.driver", []string{"redfish"})
|
|
viper.SetDefault("collect.host", smd.Host)
|
|
viper.SetDefault("collect.port", smd.Port)
|
|
viper.SetDefault("collect.user", "")
|
|
viper.SetDefault("collect.pass", "")
|
|
viper.SetDefault("collect.protocol", "https")
|
|
viper.SetDefault("collect.output", "/tmp/magellan/data/")
|
|
viper.SetDefault("collect.force-update", false)
|
|
viper.SetDefault("collect.ca-cert", "")
|
|
viper.SetDefault("bmc-host", "")
|
|
viper.SetDefault("bmc-port", 443)
|
|
viper.SetDefault("user", "")
|
|
viper.SetDefault("pass", "")
|
|
viper.SetDefault("transfer-protocol", "HTTP")
|
|
viper.SetDefault("protocol", "https")
|
|
viper.SetDefault("firmware-url", "")
|
|
viper.SetDefault("firmware-version", "")
|
|
viper.SetDefault("component", "")
|
|
viper.SetDefault("secure-tls", false)
|
|
viper.SetDefault("status", false)
|
|
|
|
}
|