| Lorenz Brun | 6adf884 | 2021-10-05 13:39:11 +0200 | [diff] [blame] | 1 | package main |
| 2 | |
| 3 | import ( |
| Serge Bazanski | 7eeef0f | 2024-02-05 14:40:15 +0100 | [diff] [blame] | 4 | "context" |
| 5 | "crypto/x509" |
| Serge Bazanski | ca8d951 | 2024-09-12 14:20:57 +0200 | [diff] [blame] | 6 | "os" |
| Mateusz Zalega | 8234c16 | 2022-07-08 17:05:50 +0200 | [diff] [blame] | 7 | "path/filepath" |
| 8 | |
| 9 | "github.com/adrg/xdg" |
| Lorenz Brun | 6adf884 | 2021-10-05 13:39:11 +0200 | [diff] [blame] | 10 | "github.com/spf13/cobra" |
| Serge Bazanski | 1f8cad7 | 2023-03-20 16:58:10 +0100 | [diff] [blame] | 11 | |
| Serge Bazanski | ca8d951 | 2024-09-12 14:20:57 +0200 | [diff] [blame] | 12 | "source.monogon.dev/go/logging" |
| Serge Bazanski | 1f8cad7 | 2023-03-20 16:58:10 +0100 | [diff] [blame] | 13 | "source.monogon.dev/metropolis/cli/metroctl/core" |
| Lorenz Brun | 6adf884 | 2021-10-05 13:39:11 +0200 | [diff] [blame] | 14 | ) |
| 15 | |
| 16 | // rootCmd represents the base command when called without any subcommands |
| 17 | var rootCmd = &cobra.Command{ |
| 18 | Use: "metroctl", |
| 19 | Short: "metroctl controls Metropolis nodes and clusters.", |
| 20 | } |
| 21 | |
| Mateusz Zalega | d5f2f7a | 2022-07-05 18:48:56 +0200 | [diff] [blame] | 22 | type metroctlFlags struct { |
| 23 | // clusterEndpoints is a list of the targeted cluster's endpoints, used by |
| 24 | // commands that perform RPC on it. |
| 25 | clusterEndpoints []string |
| Mateusz Zalega | f777496 | 2022-07-08 12:26:55 +0200 | [diff] [blame] | 26 | // proxyAddr is a SOCKS5 proxy address the cluster will be accessed through. |
| 27 | proxyAddr string |
| Mateusz Zalega | 8234c16 | 2022-07-08 17:05:50 +0200 | [diff] [blame] | 28 | // configPath overrides the default XDG config path |
| 29 | configPath string |
| Mateusz Zalega | b2cac08 | 2022-07-14 14:55:43 +0200 | [diff] [blame] | 30 | // verbose, if set, will make this utility log additional runtime |
| 31 | // information. |
| 32 | verbose bool |
| Mateusz Zalega | db75e21 | 2022-08-04 17:31:34 +0200 | [diff] [blame] | 33 | // format refers to how the output data, except logs, is formatted. |
| 34 | format string |
| 35 | // filter specifies a CEL filter used to narrow down the set of output |
| 36 | // objects. |
| 37 | filter string |
| 38 | // output is an optional output file path the resulting data will be saved |
| 39 | // at. If unspecified, the data will be written to stdout. |
| 40 | output string |
| Serge Bazanski | 7eeef0f | 2024-02-05 14:40:15 +0100 | [diff] [blame] | 41 | // acceptAnyCA will persist the first encountered (while connecting) CA |
| 42 | // certificate of the cluster as the trusted CA certificate for this cluster. |
| 43 | // This is unsafe and should only be used for testing. |
| 44 | acceptAnyCA bool |
| Serge Bazanski | 9884034 | 2024-05-22 13:03:55 +0200 | [diff] [blame] | 45 | // columns is a comma-separated list of column names which selects which columns |
| 46 | // will be output to the user. An empty string means all columns will be |
| 47 | // displayed. |
| 48 | columns string |
| Mateusz Zalega | d5f2f7a | 2022-07-05 18:48:56 +0200 | [diff] [blame] | 49 | } |
| 50 | |
| 51 | var flags metroctlFlags |
| 52 | |
| 53 | func init() { |
| Serge Bazanski | cf33f68 | 2023-03-17 00:16:16 +0100 | [diff] [blame] | 54 | rootCmd.PersistentFlags().StringSliceVar(&flags.clusterEndpoints, "endpoints", nil, "A list of the target cluster's endpoints.") |
| Mateusz Zalega | f777496 | 2022-07-08 12:26:55 +0200 | [diff] [blame] | 55 | rootCmd.PersistentFlags().StringVar(&flags.proxyAddr, "proxy", "", "SOCKS5 proxy address") |
| Mateusz Zalega | 8234c16 | 2022-07-08 17:05:50 +0200 | [diff] [blame] | 56 | rootCmd.PersistentFlags().StringVar(&flags.configPath, "config", filepath.Join(xdg.ConfigHome, "metroctl"), "An alternative cluster config path") |
| Mateusz Zalega | b2cac08 | 2022-07-14 14:55:43 +0200 | [diff] [blame] | 57 | rootCmd.PersistentFlags().BoolVar(&flags.verbose, "verbose", false, "Log additional runtime information") |
| Serge Bazanski | e012b72 | 2023-03-29 17:49:04 +0200 | [diff] [blame] | 58 | rootCmd.PersistentFlags().StringVar(&flags.format, "format", "plaintext", "Data output format") |
| Mateusz Zalega | db75e21 | 2022-08-04 17:31:34 +0200 | [diff] [blame] | 59 | rootCmd.PersistentFlags().StringVar(&flags.filter, "filter", "", "The object filter applied to the output data") |
| Serge Bazanski | 9884034 | 2024-05-22 13:03:55 +0200 | [diff] [blame] | 60 | rootCmd.PersistentFlags().StringVar(&flags.columns, "columns", "", "Comma-separated list of column names to show. If not set, all columns will be shown") |
| Mateusz Zalega | db75e21 | 2022-08-04 17:31:34 +0200 | [diff] [blame] | 61 | rootCmd.PersistentFlags().StringVarP(&flags.output, "output", "o", "", "Redirects output to the specified file") |
| Serge Bazanski | 7eeef0f | 2024-02-05 14:40:15 +0100 | [diff] [blame] | 62 | rootCmd.PersistentFlags().BoolVar(&flags.acceptAnyCA, "insecure-accept-and-persist-first-encountered-ca", false, "Accept the first encountered CA while connecting as the trusted CA for future metroctl connections with this config path. This is very insecure and should only be used for testing.") |
| Mateusz Zalega | d5f2f7a | 2022-07-05 18:48:56 +0200 | [diff] [blame] | 63 | } |
| 64 | |
| Lorenz Brun | 6adf884 | 2021-10-05 13:39:11 +0200 | [diff] [blame] | 65 | func main() { |
| 66 | cobra.CheckErr(rootCmd.Execute()) |
| 67 | } |
| Serge Bazanski | 1f8cad7 | 2023-03-20 16:58:10 +0100 | [diff] [blame] | 68 | |
| Serge Bazanski | 7eeef0f | 2024-02-05 14:40:15 +0100 | [diff] [blame] | 69 | type acceptall struct{} |
| 70 | |
| 71 | func (a *acceptall) Ask(ctx context.Context, _ *core.ConnectOptions, _ *x509.Certificate) (bool, error) { |
| 72 | return true, nil |
| 73 | } |
| 74 | |
| Serge Bazanski | 1f8cad7 | 2023-03-20 16:58:10 +0100 | [diff] [blame] | 75 | // connectOptions returns core.ConnectOptions as defined by the metroctl flags |
| 76 | // currently set. |
| 77 | func connectOptions() *core.ConnectOptions { |
| Serge Bazanski | 7eeef0f | 2024-02-05 14:40:15 +0100 | [diff] [blame] | 78 | var tofu core.CertificateTOFU |
| 79 | if flags.acceptAnyCA { |
| 80 | tofu = &acceptall{} |
| 81 | } |
| Serge Bazanski | ca8d951 | 2024-09-12 14:20:57 +0200 | [diff] [blame] | 82 | logger := logging.NewWriterBackend(os.Stderr) |
| 83 | if !flags.verbose { |
| 84 | logger.MinimumSeverity = logging.WARNING |
| 85 | } |
| Serge Bazanski | 1f8cad7 | 2023-03-20 16:58:10 +0100 | [diff] [blame] | 86 | return &core.ConnectOptions{ |
| Serge Bazanski | 925ec3d | 2024-02-05 14:38:20 +0100 | [diff] [blame] | 87 | ConfigPath: flags.configPath, |
| 88 | ProxyServer: flags.proxyAddr, |
| 89 | Endpoints: flags.clusterEndpoints, |
| Serge Bazanski | ca8d951 | 2024-09-12 14:20:57 +0200 | [diff] [blame] | 90 | ResolverLogger: logger, |
| Serge Bazanski | 7eeef0f | 2024-02-05 14:40:15 +0100 | [diff] [blame] | 91 | TOFU: tofu, |
| Serge Bazanski | 1f8cad7 | 2023-03-20 16:58:10 +0100 | [diff] [blame] | 92 | } |
| 93 | } |