create: deprecate --control-plane-nodes and --worker-nodes flags

Also print and configure node groups
This commit is contained in:
Malte Poll 2023-08-02 10:36:55 +02:00 committed by Malte Poll
parent c0177c565f
commit 3047cb2798
4 changed files with 54 additions and 86 deletions

View File

@ -57,13 +57,10 @@ func NewCreator(out io.Writer) *Creator {
// CreateOptions are the options for creating a Constellation cluster. // CreateOptions are the options for creating a Constellation cluster.
type CreateOptions struct { type CreateOptions struct {
Provider cloudprovider.Provider Provider cloudprovider.Provider
Config *config.Config Config *config.Config
InsType string image string
ControlPlaneCount int TFLogLevel terraform.LogLevel
WorkerCount int
image string
TFLogLevel terraform.LogLevel
} }
// Create creates the handed amount of instances and all the needed resources. // Create creates the handed amount of instances and all the needed resources.
@ -129,7 +126,7 @@ func (c *Creator) Create(ctx context.Context, opts CreateOptions) (clusterid.Fil
} }
func (c *Creator) createAWS(ctx context.Context, cl tfResourceClient, opts CreateOptions) (tfOutput terraform.ApplyOutput, retErr error) { func (c *Creator) createAWS(ctx context.Context, cl tfResourceClient, opts CreateOptions) (tfOutput terraform.ApplyOutput, retErr error) {
vars := awsTerraformVars(opts.Config, opts.image, &opts.ControlPlaneCount, &opts.WorkerCount) vars := awsTerraformVars(opts.Config, opts.image)
tfOutput, err := runTerraformCreate(ctx, cl, cloudprovider.AWS, vars, c.out, opts.TFLogLevel) tfOutput, err := runTerraformCreate(ctx, cl, cloudprovider.AWS, vars, c.out, opts.TFLogLevel)
if err != nil { if err != nil {
@ -140,7 +137,7 @@ func (c *Creator) createAWS(ctx context.Context, cl tfResourceClient, opts Creat
} }
func (c *Creator) createGCP(ctx context.Context, cl tfResourceClient, opts CreateOptions) (tfOutput terraform.ApplyOutput, retErr error) { func (c *Creator) createGCP(ctx context.Context, cl tfResourceClient, opts CreateOptions) (tfOutput terraform.ApplyOutput, retErr error) {
vars := gcpTerraformVars(opts.Config, opts.image, &opts.ControlPlaneCount, &opts.WorkerCount) vars := gcpTerraformVars(opts.Config, opts.image)
tfOutput, err := runTerraformCreate(ctx, cl, cloudprovider.GCP, vars, c.out, opts.TFLogLevel) tfOutput, err := runTerraformCreate(ctx, cl, cloudprovider.GCP, vars, c.out, opts.TFLogLevel)
if err != nil { if err != nil {
@ -151,7 +148,7 @@ func (c *Creator) createGCP(ctx context.Context, cl tfResourceClient, opts Creat
} }
func (c *Creator) createAzure(ctx context.Context, cl tfResourceClient, opts CreateOptions) (tfOutput terraform.ApplyOutput, retErr error) { func (c *Creator) createAzure(ctx context.Context, cl tfResourceClient, opts CreateOptions) (tfOutput terraform.ApplyOutput, retErr error) {
vars := azureTerraformVars(opts.Config, opts.image, &opts.ControlPlaneCount, &opts.WorkerCount) vars := azureTerraformVars(opts.Config, opts.image)
tfOutput, err := runTerraformCreate(ctx, cl, cloudprovider.Azure, vars, c.out, opts.TFLogLevel) tfOutput, err := runTerraformCreate(ctx, cl, cloudprovider.Azure, vars, c.out, opts.TFLogLevel)
if err != nil { if err != nil {
@ -216,7 +213,7 @@ func (c *Creator) createOpenStack(ctx context.Context, cl tfResourceClient, opts
) )
} }
vars := openStackTerraformVars(opts.Config, opts.image, &opts.ControlPlaneCount, &opts.WorkerCount) vars := openStackTerraformVars(opts.Config, opts.image)
tfOutput, err := runTerraformCreate(ctx, cl, cloudprovider.OpenStack, vars, c.out, opts.TFLogLevel) tfOutput, err := runTerraformCreate(ctx, cl, cloudprovider.OpenStack, vars, c.out, opts.TFLogLevel)
if err != nil { if err != nil {
@ -292,7 +289,7 @@ func (c *Creator) createQEMU(ctx context.Context, cl tfResourceClient, lv libvir
metadataLibvirtURI = "qemu:///system" metadataLibvirtURI = "qemu:///system"
} }
vars := qemuTerraformVars(opts.Config, imagePath, &opts.ControlPlaneCount, &opts.WorkerCount, libvirtURI, libvirtSocketPath, metadataLibvirtURI) vars := qemuTerraformVars(opts.Config, imagePath, libvirtURI, libvirtSocketPath, metadataLibvirtURI)
if opts.Config.Provider.QEMU.Firmware != "" { if opts.Config.Provider.QEMU.Firmware != "" {
vars.Firmware = toPtr(opts.Config.Provider.QEMU.Firmware) vars.Firmware = toPtr(opts.Config.Provider.QEMU.Firmware)

View File

@ -218,12 +218,9 @@ func TestCreator(t *testing.T) {
} }
opts := CreateOptions{ opts := CreateOptions{
Provider: tc.provider, Provider: tc.provider,
Config: tc.config, Config: tc.config,
InsType: "type", TFLogLevel: terraform.LogLevelNone,
ControlPlaneCount: 2,
WorkerCount: 3,
TFLogLevel: terraform.LogLevelNone,
} }
idFile, err := creator.Create(context.Background(), opts) idFile, err := creator.Create(context.Background(), opts)

View File

@ -16,7 +16,6 @@ import (
"github.com/edgelesssys/constellation/v2/internal/api/attestationconfigapi" "github.com/edgelesssys/constellation/v2/internal/api/attestationconfigapi"
"github.com/edgelesssys/constellation/v2/internal/api/versionsapi" "github.com/edgelesssys/constellation/v2/internal/api/versionsapi"
"github.com/edgelesssys/constellation/v2/internal/attestation/variant" "github.com/edgelesssys/constellation/v2/internal/attestation/variant"
"github.com/edgelesssys/constellation/v2/internal/cloud/cloudprovider"
"github.com/edgelesssys/constellation/v2/internal/config" "github.com/edgelesssys/constellation/v2/internal/config"
"github.com/edgelesssys/constellation/v2/internal/constants" "github.com/edgelesssys/constellation/v2/internal/constants"
"github.com/edgelesssys/constellation/v2/internal/file" "github.com/edgelesssys/constellation/v2/internal/file"
@ -35,10 +34,11 @@ func NewCreateCmd() *cobra.Command {
RunE: runCreate, RunE: runCreate,
} }
cmd.Flags().BoolP("yes", "y", false, "create the cluster without further confirmation") cmd.Flags().BoolP("yes", "y", false, "create the cluster without further confirmation")
cmd.Flags().IntP("control-plane-nodes", "c", 0, "number of control-plane nodes (required)") // TODO(malt3): remove deprecated flags in v2.11+
must(cobra.MarkFlagRequired(cmd.Flags(), "control-plane-nodes")) cmd.Flags().IntP("control-plane-nodes", "c", 0, "number of control-plane nodes")
cmd.Flags().IntP("worker-nodes", "w", 0, "number of worker nodes (required)") cmd.Flags().IntP("worker-nodes", "w", 0, "number of worker nodes")
must(cobra.MarkFlagRequired(cmd.Flags(), "worker-nodes")) must(cmd.Flags().MarkDeprecated("control-plane-nodes", "configure the number of control-plane nodes in the configuration file"))
must(cmd.Flags().MarkDeprecated("worker-nodes", "configure the number of worker nodes in the configuration file"))
return cmd return cmd
} }
@ -120,32 +120,34 @@ func (c *createCmd) create(cmd *cobra.Command, creator cloudCreator, fileHandler
} }
provider := conf.GetProvider() provider := conf.GetProvider()
var instanceType string
switch provider { controlPlaneGroup, ok := conf.NodeGroups[constants.DefaultControlPlaneGroupName]
case cloudprovider.AWS: if !ok {
c.log.Debugf("Configuring instance type for AWS") return fmt.Errorf("default control-plane node group %q not found in configuration", constants.DefaultControlPlaneGroupName)
instanceType = conf.Provider.AWS.InstanceType }
case cloudprovider.Azure: workerGroup, ok := conf.NodeGroups[constants.DefaultWorkerGroupName]
c.log.Debugf("Configuring instance type for Azure") if !ok {
instanceType = conf.Provider.Azure.InstanceType return fmt.Errorf("default worker node group %q not found in configuration", constants.DefaultWorkerGroupName)
case cloudprovider.GCP: }
c.log.Debugf("Configuring instance type for GCP") otherGroupNames := make([]string, 0, len(conf.NodeGroups)-2)
instanceType = conf.Provider.GCP.InstanceType for groupName := range conf.NodeGroups {
case cloudprovider.OpenStack: if groupName != constants.DefaultControlPlaneGroupName && groupName != constants.DefaultWorkerGroupName {
c.log.Debugf("Configuring instance type for OpenStack") otherGroupNames = append(otherGroupNames, groupName)
instanceType = conf.Provider.OpenStack.FlavorID }
case cloudprovider.QEMU: }
c.log.Debugf("Configuring instance type for QEMU") if len(otherGroupNames) > 0 {
cpus := conf.Provider.QEMU.VCPUs c.log.Debugf("Creating %d additional node groups: %v", len(otherGroupNames), otherGroupNames)
instanceType = fmt.Sprintf("%d-vCPU", cpus)
} }
c.log.Debugf("Configured with instance type %q", instanceType)
if !flags.yes { if !flags.yes {
// Ask user to confirm action. // Ask user to confirm action.
cmd.Printf("The following Constellation cluster will be created:\n") cmd.Printf("The following Constellation cluster will be created:\n")
cmd.Printf("%d control-plane node%s of type %s will be created.\n", flags.controllerCount, isPlural(flags.controllerCount), instanceType) cmd.Printf(" %d control-plane node%s of type %s will be created.\n", controlPlaneGroup.InitialCount, isPlural(controlPlaneGroup.InitialCount), controlPlaneGroup.InstanceType)
cmd.Printf("%d worker node%s of type %s will be created.\n", flags.workerCount, isPlural(flags.workerCount), instanceType) cmd.Printf(" %d worker node%s of type %s will be created.\n", workerGroup.InitialCount, isPlural(workerGroup.InitialCount), workerGroup.InstanceType)
for _, groupName := range otherGroupNames {
group := conf.NodeGroups[groupName]
cmd.Printf(" group %s with %d node%s of type %s will be created.\n", groupName, group.InitialCount, isPlural(group.InitialCount), group.InstanceType)
}
ok, err := askToConfirm(cmd, "Do you want to create this cluster?") ok, err := askToConfirm(cmd, "Do you want to create this cluster?")
if err != nil { if err != nil {
return err return err
@ -158,12 +160,9 @@ func (c *createCmd) create(cmd *cobra.Command, creator cloudCreator, fileHandler
spinner.Start("Creating", false) spinner.Start("Creating", false)
opts := cloudcmd.CreateOptions{ opts := cloudcmd.CreateOptions{
Provider: provider, Provider: provider,
Config: conf, Config: conf,
InsType: instanceType, TFLogLevel: flags.tfLogLevel,
ControlPlaneCount: flags.controllerCount,
WorkerCount: flags.workerCount,
TFLogLevel: flags.tfLogLevel,
} }
idFile, err := creator.Create(cmd.Context(), opts) idFile, err := creator.Create(cmd.Context(), opts)
spinner.Stop() spinner.Stop()
@ -182,24 +181,6 @@ func (c *createCmd) create(cmd *cobra.Command, creator cloudCreator, fileHandler
// parseCreateFlags parses the flags of the create command. // parseCreateFlags parses the flags of the create command.
func (c *createCmd) parseCreateFlags(cmd *cobra.Command) (createFlags, error) { func (c *createCmd) parseCreateFlags(cmd *cobra.Command) (createFlags, error) {
controllerCount, err := cmd.Flags().GetInt("control-plane-nodes")
if err != nil {
return createFlags{}, fmt.Errorf("parsing number of control-plane nodes: %w", err)
}
c.log.Debugf("Control-plane nodes flag is %d", controllerCount)
if controllerCount < constants.MinControllerCount {
return createFlags{}, fmt.Errorf("number of control-plane nodes must be at least %d", constants.MinControllerCount)
}
workerCount, err := cmd.Flags().GetInt("worker-nodes")
if err != nil {
return createFlags{}, fmt.Errorf("parsing number of worker nodes: %w", err)
}
c.log.Debugf("Worker nodes flag is %d", workerCount)
if workerCount < constants.MinWorkerCount {
return createFlags{}, fmt.Errorf("number of worker nodes must be at least %d", constants.MinWorkerCount)
}
yes, err := cmd.Flags().GetBool("yes") yes, err := cmd.Flags().GetBool("yes")
if err != nil { if err != nil {
return createFlags{}, fmt.Errorf("parsing yes bool: %w", err) return createFlags{}, fmt.Errorf("parsing yes bool: %w", err)
@ -229,23 +210,19 @@ func (c *createCmd) parseCreateFlags(cmd *cobra.Command) (createFlags, error) {
c.log.Debugf("Terraform logs will be written into %s at level %s", constants.TerraformLogFile, logLevel.String()) c.log.Debugf("Terraform logs will be written into %s at level %s", constants.TerraformLogFile, logLevel.String())
return createFlags{ return createFlags{
controllerCount: controllerCount, configPath: configPath,
workerCount: workerCount, tfLogLevel: logLevel,
configPath: configPath, force: force,
tfLogLevel: logLevel, yes: yes,
force: force,
yes: yes,
}, nil }, nil
} }
// createFlags contains the parsed flags of the create command. // createFlags contains the parsed flags of the create command.
type createFlags struct { type createFlags struct {
controllerCount int configPath string
workerCount int tfLogLevel terraform.LogLevel
configPath string force bool
tfLogLevel terraform.LogLevel yes bool
force bool
yes bool
} }
// checkDirClean checks if files of a previous Constellation are left in the current working dir. // checkDirClean checks if files of a previous Constellation are left in the current working dir.

View File

@ -161,12 +161,9 @@ func (m *miniUpCmd) prepareConfig(cmd *cobra.Command, fileHandler file.Handler,
func (m *miniUpCmd) createMiniCluster(ctx context.Context, fileHandler file.Handler, creator cloudCreator, config *config.Config, tfLogLevel terraform.LogLevel) error { func (m *miniUpCmd) createMiniCluster(ctx context.Context, fileHandler file.Handler, creator cloudCreator, config *config.Config, tfLogLevel terraform.LogLevel) error {
m.log.Debugf("Creating mini cluster") m.log.Debugf("Creating mini cluster")
opts := cloudcmd.CreateOptions{ opts := cloudcmd.CreateOptions{
Provider: cloudprovider.QEMU, Provider: cloudprovider.QEMU,
Config: config, Config: config,
InsType: "", TFLogLevel: tfLogLevel,
ControlPlaneCount: 1,
WorkerCount: 1,
TFLogLevel: tfLogLevel,
} }
idFile, err := creator.Create(ctx, opts) idFile, err := creator.Create(ctx, opts)
if err != nil { if err != nil {