mirror of
https://github.com/edgelesssys/constellation.git
synced 2024-10-01 01:36:09 -04:00
99b12e4035
Signed-off-by: Daniel Weiße <dw@edgeless.systems>
183 lines
5.8 KiB
Go
183 lines
5.8 KiB
Go
/*
|
|
Copyright (c) Edgeless Systems GmbH
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
*/
|
|
|
|
package cmd
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/edgelesssys/constellation/v2/internal/cloud/cloudprovider"
|
|
"github.com/edgelesssys/constellation/v2/internal/compatibility"
|
|
"github.com/edgelesssys/constellation/v2/internal/config"
|
|
"github.com/edgelesssys/constellation/v2/internal/constants"
|
|
"github.com/edgelesssys/constellation/v2/internal/file"
|
|
"github.com/edgelesssys/constellation/v2/internal/variant"
|
|
"github.com/edgelesssys/constellation/v2/internal/versions"
|
|
"github.com/siderolabs/talos/pkg/machinery/config/encoder"
|
|
"github.com/spf13/afero"
|
|
"github.com/spf13/cobra"
|
|
"golang.org/x/mod/semver"
|
|
)
|
|
|
|
func newConfigGenerateCmd() *cobra.Command {
|
|
cmd := &cobra.Command{
|
|
Use: "generate {aws|azure|gcp|openstack|qemu}",
|
|
Short: "Generate a default configuration file",
|
|
Long: "Generate a default configuration file for your selected cloud provider.",
|
|
Args: cobra.MatchAll(
|
|
cobra.ExactArgs(1),
|
|
isCloudProvider(0),
|
|
),
|
|
ValidArgsFunction: generateCompletion,
|
|
RunE: runConfigGenerate,
|
|
}
|
|
cmd.Flags().StringP("file", "f", constants.ConfigFilename, "path to output file, or '-' for stdout")
|
|
cmd.Flags().StringP("kubernetes", "k", semver.MajorMinor(config.Default().KubernetesVersion), "Kubernetes version to use in format MAJOR.MINOR")
|
|
|
|
return cmd
|
|
}
|
|
|
|
type generateFlags struct {
|
|
file string
|
|
k8sVersion string
|
|
}
|
|
|
|
type configGenerateCmd struct {
|
|
log debugLog
|
|
}
|
|
|
|
func runConfigGenerate(cmd *cobra.Command, args []string) error {
|
|
log, err := newCLILogger(cmd)
|
|
if err != nil {
|
|
return fmt.Errorf("creating logger: %w", err)
|
|
}
|
|
defer log.Sync()
|
|
fileHandler := file.NewHandler(afero.NewOsFs())
|
|
provider := cloudprovider.FromString(args[0])
|
|
cg := &configGenerateCmd{log: log}
|
|
return cg.configGenerate(cmd, fileHandler, provider)
|
|
}
|
|
|
|
func (cg *configGenerateCmd) configGenerate(cmd *cobra.Command, fileHandler file.Handler, provider cloudprovider.Provider) error {
|
|
flags, err := parseGenerateFlags(cmd)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
cg.log.Debugf("Parsed flags as %v", flags)
|
|
cg.log.Debugf("Using cloud provider %s", provider.String())
|
|
conf := createConfig(provider)
|
|
conf.KubernetesVersion = flags.k8sVersion
|
|
if flags.file == "-" {
|
|
content, err := encoder.NewEncoder(conf).Encode()
|
|
if err != nil {
|
|
return fmt.Errorf("encoding config content: %w", err)
|
|
}
|
|
|
|
cg.log.Debugf("Writing YAML data to stdout")
|
|
_, err = cmd.OutOrStdout().Write(content)
|
|
return err
|
|
}
|
|
|
|
cg.log.Debugf("Writing YAML data to configuration file")
|
|
if err := fileHandler.WriteYAML(flags.file, conf, file.OptMkdirAll); err != nil {
|
|
return err
|
|
}
|
|
|
|
cmd.Println("Config file written to", flags.file)
|
|
cmd.Println("Please fill in your CSP-specific configuration before proceeding.")
|
|
cmd.Println("For more information refer to the documentation:")
|
|
cmd.Println("\thttps://docs.edgeless.systems/constellation/getting-started/first-steps")
|
|
|
|
return nil
|
|
}
|
|
|
|
// createConfig creates a config file for the given provider.
|
|
func createConfig(provider cloudprovider.Provider) *config.Config {
|
|
conf := config.Default()
|
|
conf.RemoveProviderExcept(provider)
|
|
|
|
// set a lower default for QEMU's state disk
|
|
if provider == cloudprovider.QEMU {
|
|
conf.StateDiskSizeGB = 10
|
|
}
|
|
|
|
// TODO(AB#2976): Replace hardcoded values with user input
|
|
switch provider {
|
|
case cloudprovider.AWS:
|
|
conf.AttestationVariant = variant.AWSNitroTPM{}.String()
|
|
case cloudprovider.Azure:
|
|
conf.AttestationVariant = variant.AzureSEVSNP{}.String()
|
|
case cloudprovider.GCP:
|
|
conf.AttestationVariant = variant.GCPSEVES{}.String()
|
|
case cloudprovider.QEMU:
|
|
conf.AttestationVariant = variant.QEMUVTPM{}.String()
|
|
}
|
|
|
|
return conf
|
|
}
|
|
|
|
// supportedVersions prints the supported version without v prefix and without patch version.
|
|
// Should only be used when accepting Kubernetes versions from --kubernetes.
|
|
func supportedVersions() string {
|
|
builder := strings.Builder{}
|
|
for i, version := range versions.SupportedK8sVersions() {
|
|
if i > 0 {
|
|
builder.WriteString(" ")
|
|
}
|
|
builder.WriteString(strings.TrimPrefix(semver.MajorMinor(version), "v"))
|
|
}
|
|
return builder.String()
|
|
}
|
|
|
|
func parseGenerateFlags(cmd *cobra.Command) (generateFlags, error) {
|
|
file, err := cmd.Flags().GetString("file")
|
|
if err != nil {
|
|
return generateFlags{}, fmt.Errorf("parsing file flag: %w", err)
|
|
}
|
|
k8sVersion, err := cmd.Flags().GetString("kubernetes")
|
|
if err != nil {
|
|
return generateFlags{}, fmt.Errorf("parsing kuberentes flag: %w", err)
|
|
}
|
|
resolvedVersion, err := resolveK8sVersion(k8sVersion)
|
|
if err != nil {
|
|
return generateFlags{}, fmt.Errorf("resolving kuberentes version from flag: %w", err)
|
|
}
|
|
|
|
return generateFlags{
|
|
file: file,
|
|
k8sVersion: resolvedVersion,
|
|
}, nil
|
|
}
|
|
|
|
// createCompletion handles the completion of the create command. It is frequently called
|
|
// while the user types arguments of the command to suggest completion.
|
|
func generateCompletion(_ *cobra.Command, args []string, _ string) ([]string, cobra.ShellCompDirective) {
|
|
switch len(args) {
|
|
case 0:
|
|
return []string{"aws", "gcp", "azure", "qemu"}, cobra.ShellCompDirectiveNoFileComp
|
|
default:
|
|
return []string{}, cobra.ShellCompDirectiveError
|
|
}
|
|
}
|
|
|
|
// resolveK8sVersion takes the user input from --kubernetes and transforms a MAJOR.MINOR definition into a supported
|
|
// MAJOR.MINOR.PATCH release.
|
|
func resolveK8sVersion(k8sVersion string) (string, error) {
|
|
prefixedVersion := compatibility.EnsurePrefixV(k8sVersion)
|
|
if !semver.IsValid(prefixedVersion) {
|
|
return "", fmt.Errorf("kubernetes flag does not specify a valid semantic version: %s", k8sVersion)
|
|
}
|
|
|
|
extendedVersion := config.K8sVersionFromMajorMinor(prefixedVersion)
|
|
if extendedVersion == "" {
|
|
return "", fmt.Errorf("--kubernetes (%s) does not specify a valid Kubernetes version. Supported versions: %s", strings.TrimPrefix(k8sVersion, "v"), supportedVersions())
|
|
}
|
|
|
|
return extendedVersion, nil
|
|
}
|