2022-10-07 09:38:43 +02:00
/ *
Copyright ( c ) Edgeless Systems GmbH
SPDX - License - Identifier : AGPL - 3.0 - only
* /
package cmd
import (
"context"
"errors"
"fmt"
2023-10-31 12:46:40 +01:00
"os"
"path/filepath"
2023-10-24 15:39:18 +02:00
"time"
2022-10-07 09:38:43 +02:00
"github.com/edgelesssys/constellation/v2/cli/internal/cloudcmd"
2023-06-28 10:28:48 +02:00
"github.com/edgelesssys/constellation/v2/cli/internal/featureset"
2022-10-07 09:38:43 +02:00
"github.com/edgelesssys/constellation/v2/cli/internal/libvirt"
2023-10-09 13:04:29 +02:00
"github.com/edgelesssys/constellation/v2/cli/internal/state"
2023-06-07 16:16:32 +02:00
"github.com/edgelesssys/constellation/v2/internal/api/attestationconfigapi"
2022-10-07 09:38:43 +02:00
"github.com/edgelesssys/constellation/v2/internal/cloud/cloudprovider"
"github.com/edgelesssys/constellation/v2/internal/config"
"github.com/edgelesssys/constellation/v2/internal/constants"
"github.com/edgelesssys/constellation/v2/internal/file"
"github.com/spf13/afero"
"github.com/spf13/cobra"
)
func newMiniUpCmd ( ) * cobra . Command {
cmd := & cobra . Command {
Use : "up" ,
2022-10-14 10:48:20 +02:00
Short : "Create and initialize a new MiniConstellation cluster" ,
2023-01-17 14:01:56 +01:00
Long : "Create and initialize a new MiniConstellation cluster.\n\n" +
"A mini cluster consists of a single control-plane and worker node, hosted using QEMU/KVM." ,
2022-10-07 09:38:43 +02:00
Args : cobra . ExactArgs ( 0 ) ,
RunE : runUp ,
}
2023-03-08 15:48:36 +01:00
cmd . Flags ( ) . Bool ( "merge-kubeconfig" , true , "merge Constellation kubeconfig file with default kubeconfig file in $HOME/.kube/config" )
2022-10-07 09:38:43 +02:00
return cmd
}
2023-01-04 09:46:29 +00:00
type miniUpCmd struct {
2023-06-01 13:55:46 +02:00
log debugLog
2023-06-07 16:16:32 +02:00
configFetcher attestationconfigapi . Fetcher
2023-10-16 15:05:29 +02:00
fileHandler file . Handler
flags rootFlags
2023-01-04 09:46:29 +00:00
}
2023-03-20 11:03:36 +01:00
func runUp ( cmd * cobra . Command , _ [ ] string ) error {
2023-01-04 09:46:29 +00:00
log , err := newCLILogger ( cmd )
if err != nil {
return fmt . Errorf ( "creating logger: %w" , err )
}
defer log . Sync ( )
2023-01-18 13:10:24 +01:00
spinner , err := newSpinnerOrStderr ( cmd )
if err != nil {
return err
}
2022-10-07 19:35:07 +02:00
defer spinner . Stop ( )
2023-10-16 15:05:29 +02:00
m := & miniUpCmd {
log : log ,
configFetcher : attestationconfigapi . NewFetcher ( ) ,
fileHandler : file . NewHandler ( afero . NewOsFs ( ) ) ,
}
if err := m . flags . parse ( cmd . Flags ( ) ) ; err != nil {
return err
}
2023-10-31 12:46:40 +01:00
creator , cleanUp , err := cloudcmd . NewApplier (
cmd . Context ( ) ,
spinner ,
constants . TerraformWorkingDir ,
filepath . Join ( constants . UpgradeDir , "create" ) , // Not used by create
m . flags . tfLogLevel ,
m . fileHandler ,
)
if err != nil {
return err
}
defer cleanUp ( )
2023-01-04 09:46:29 +00:00
return m . up ( cmd , creator , spinner )
2022-10-07 19:35:07 +02:00
}
2023-10-31 12:46:40 +01:00
func ( m * miniUpCmd ) up ( cmd * cobra . Command , creator cloudApplier , spinner spinnerInterf ) error {
2023-01-04 09:46:29 +00:00
if err := m . checkSystemRequirements ( cmd . ErrOrStderr ( ) ) ; err != nil {
2022-10-07 09:38:43 +02:00
return fmt . Errorf ( "system requirements not met: %w" , err )
}
2023-10-31 12:46:40 +01:00
if clean , err := m . fileHandler . IsEmpty ( constants . TerraformWorkingDir ) ; err != nil && ! errors . Is ( err , os . ErrNotExist ) {
return fmt . Errorf ( "checking if terraform working directory is empty: %w" , err )
} else if err == nil && ! clean {
return fmt . Errorf (
"directory %q already exists and is not empty, run 'constellation mini down' before creating a new cluster" ,
m . flags . pathPrefixer . PrefixPrintablePath ( constants . TerraformWorkingDir ) ,
)
}
// create config if not present in directory and set default values
2023-10-16 15:05:29 +02:00
config , err := m . prepareConfig ( cmd )
2022-10-07 09:38:43 +02:00
if err != nil {
return fmt . Errorf ( "preparing config: %w" , err )
}
// create cluster
2022-10-07 19:35:07 +02:00
spinner . Start ( "Creating cluster in QEMU " , false )
2023-10-16 15:05:29 +02:00
err = m . createMiniCluster ( cmd . Context ( ) , creator , config )
2022-10-07 09:38:43 +02:00
spinner . Stop ( )
if err != nil {
return fmt . Errorf ( "creating cluster: %w" , err )
}
cmd . Println ( "Cluster successfully created." )
connectURI := config . Provider . QEMU . LibvirtURI
2023-01-04 09:46:29 +00:00
m . log . Debugf ( "Using connect URI %s" , connectURI )
2022-10-07 09:38:43 +02:00
if connectURI == "" {
connectURI = libvirt . LibvirtTCPConnectURI
}
cmd . Println ( "Connect to the VMs by executing:" )
cmd . Printf ( "\tvirsh -c %s\n\n" , connectURI )
// initialize cluster
2023-10-24 15:39:18 +02:00
if err := m . initializeMiniCluster ( cmd ) ; err != nil {
2022-10-07 09:38:43 +02:00
return fmt . Errorf ( "initializing cluster: %w" , err )
}
2023-01-04 09:46:29 +00:00
m . log . Debugf ( "Initialized cluster" )
2022-10-07 09:38:43 +02:00
return nil
}
// prepareConfig reads a given config, or creates a new minimal QEMU config.
2023-10-16 15:05:29 +02:00
func ( m * miniUpCmd ) prepareConfig ( cmd * cobra . Command ) ( * config . Config , error ) {
_ , err := m . fileHandler . Stat ( constants . ConfigFilename )
2023-08-04 13:53:51 +02:00
if err == nil {
// config already exists, prompt user if they want to use this file
cmd . PrintErrln ( "A config file already exists in the configured workspace." )
ok , err := askToConfirm ( cmd , "Do you want to create the Constellation using that config?" )
2022-10-07 09:38:43 +02:00
if err != nil {
2023-02-07 12:56:25 +01:00
return nil , err
2022-10-07 09:38:43 +02:00
}
2023-08-04 13:53:51 +02:00
if ok {
2023-10-16 15:05:29 +02:00
return m . prepareExistingConfig ( cmd )
2022-10-07 09:38:43 +02:00
}
2023-08-04 13:53:51 +02:00
// user declined to reuse config file, prompt if they want to overwrite it
ok , err = askToConfirm ( cmd , "Do you want to overwrite it and create a new config?" )
2022-10-20 13:36:20 +02:00
if err != nil {
return nil , err
}
if ! ok {
return nil , errors . New ( "not overwriting existing config" )
}
}
2023-08-04 13:53:51 +02:00
2023-06-28 10:28:48 +02:00
if ! featureset . CanUseEmbeddedMeasurmentsAndImage {
cmd . PrintErrln ( "Generating a valid default config is not supported in the OSS build of the Constellation CLI. Consult the documentation for instructions on where to download the enterprise version." )
return nil , errors . New ( "cannot create a mini cluster without a config file in the OSS build" )
}
config , err := config . MiniDefault ( )
if err != nil {
return nil , fmt . Errorf ( "mini default config is invalid: %v" , err )
2023-05-12 17:14:32 +02:00
}
2023-01-18 13:10:24 +01:00
m . log . Debugf ( "Prepared configuration" )
2022-10-07 09:38:43 +02:00
2023-10-16 15:05:29 +02:00
return config , m . fileHandler . WriteYAML ( constants . ConfigFilename , config , file . OptOverwrite )
2022-10-07 09:38:43 +02:00
}
2023-10-16 15:05:29 +02:00
func ( m * miniUpCmd ) prepareExistingConfig ( cmd * cobra . Command ) ( * config . Config , error ) {
conf , err := config . New ( m . fileHandler , constants . ConfigFilename , m . configFetcher , m . flags . force )
2023-08-04 13:53:51 +02:00
var configValidationErr * config . ValidationError
if errors . As ( err , & configValidationErr ) {
cmd . PrintErrln ( configValidationErr . LongMessage ( ) )
}
if err != nil {
return nil , err
}
if conf . GetProvider ( ) != cloudprovider . QEMU {
return nil , errors . New ( "invalid provider for MiniConstellation cluster" )
}
return conf , nil
}
2022-10-07 09:38:43 +02:00
// createMiniCluster creates a new cluster using the given config.
2023-10-31 12:46:40 +01:00
func ( m * miniUpCmd ) createMiniCluster ( ctx context . Context , creator cloudApplier , config * config . Config ) error {
2023-01-04 09:46:29 +00:00
m . log . Debugf ( "Creating mini cluster" )
2023-10-31 12:46:40 +01:00
if _ , err := creator . Plan ( ctx , config ) ; err != nil {
return err
2023-04-14 14:15:07 +02:00
}
2023-10-31 12:46:40 +01:00
infraState , err := creator . Apply ( ctx , config . GetProvider ( ) , cloudcmd . WithoutRollbackOnError )
2022-10-07 09:38:43 +02:00
if err != nil {
return err
}
2023-10-09 13:04:29 +02:00
infraState . UID = constants . MiniConstellationUID // use UID "mini" to identify MiniConstellation clusters.
stateFile := state . New ( ) .
SetInfrastructure ( infraState )
m . log . Debugf ( "Cluster state file contains %v" , stateFile )
2023-10-16 15:05:29 +02:00
return stateFile . WriteToFile ( m . fileHandler , constants . StateFilename )
2022-10-07 09:38:43 +02:00
}
// initializeMiniCluster initializes a QEMU cluster.
2023-10-24 15:39:18 +02:00
func ( m * miniUpCmd ) initializeMiniCluster ( cmd * cobra . Command ) ( retErr error ) {
2023-01-04 09:46:29 +00:00
m . log . Debugf ( "Initializing mini cluster" )
2022-10-07 09:38:43 +02:00
// clean up cluster resources if initialization fails
defer func ( ) {
if retErr != nil {
2022-11-10 10:27:24 +01:00
cmd . PrintErrf ( "An error occurred: %s\n" , retErr )
cmd . PrintErrln ( "Attempting to roll back." )
2022-10-07 09:38:43 +02:00
_ = runDown ( cmd , [ ] string { } )
2022-11-10 10:27:24 +01:00
cmd . PrintErrf ( "Rollback succeeded.\n\n" )
2022-10-07 09:38:43 +02:00
}
} ( )
2023-10-16 15:05:29 +02:00
2023-10-24 15:39:18 +02:00
// Define flags for apply backend that are not set by mini up
cmd . Flags ( ) . StringSlice (
"skip-phases" ,
[ ] string { string ( skipInfrastructurePhase ) , string ( skipK8sPhase ) , string ( skipImagePhase ) } ,
"" ,
)
cmd . Flags ( ) . Bool ( "yes" , false , "" )
cmd . Flags ( ) . Bool ( "skip-helm-wait" , false , "" )
cmd . Flags ( ) . Bool ( "conformance" , false , "" )
cmd . Flags ( ) . Duration ( "timeout" , time . Hour , "" )
2023-10-16 15:05:29 +02:00
2023-10-24 15:39:18 +02:00
if err := runApply ( cmd , nil ) ; err != nil {
2022-10-07 09:38:43 +02:00
return err
}
2023-01-04 09:46:29 +00:00
m . log . Debugf ( "Initialized mini cluster" )
2022-10-07 09:38:43 +02:00
return nil
}