2022-09-26 15:52:31 +02:00
|
|
|
/*
|
|
|
|
Copyright (c) Edgeless Systems GmbH
|
|
|
|
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
|
|
|
package terraform
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"io/fs"
|
2022-11-14 18:18:58 +01:00
|
|
|
"path/filepath"
|
2022-09-26 15:52:31 +02:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/edgelesssys/constellation/v2/internal/cloud/cloudprovider"
|
2022-11-14 18:18:58 +01:00
|
|
|
"github.com/edgelesssys/constellation/v2/internal/constants"
|
2022-09-26 15:52:31 +02:00
|
|
|
"github.com/edgelesssys/constellation/v2/internal/file"
|
|
|
|
"github.com/hashicorp/terraform-exec/tfexec"
|
|
|
|
tfjson "github.com/hashicorp/terraform-json"
|
|
|
|
"github.com/spf13/afero"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"go.uber.org/multierr"
|
|
|
|
)
|
|
|
|
|
2022-11-15 14:00:44 +01:00
|
|
|
func TestPrepareCluster(t *testing.T) {
|
|
|
|
qemuVars := &QEMUVariables{
|
|
|
|
CommonVariables: CommonVariables{
|
|
|
|
Name: "name",
|
|
|
|
CountControlPlanes: 1,
|
|
|
|
CountWorkers: 2,
|
|
|
|
StateDiskSizeGB: 11,
|
|
|
|
},
|
|
|
|
CPUCount: 1,
|
|
|
|
MemorySizeMiB: 1024,
|
|
|
|
ImagePath: "path",
|
|
|
|
ImageFormat: "format",
|
|
|
|
MetadataAPIImage: "api",
|
|
|
|
}
|
|
|
|
|
|
|
|
testCases := map[string]struct {
|
|
|
|
provider cloudprovider.Provider
|
|
|
|
vars Variables
|
|
|
|
fs afero.Fs
|
|
|
|
partiallyExtracted bool
|
|
|
|
wantErr bool
|
|
|
|
}{
|
|
|
|
"qemu": {
|
|
|
|
provider: cloudprovider.QEMU,
|
|
|
|
vars: qemuVars,
|
|
|
|
fs: afero.NewMemMapFs(),
|
|
|
|
wantErr: false,
|
|
|
|
},
|
|
|
|
"no vars": {
|
|
|
|
provider: cloudprovider.QEMU,
|
|
|
|
fs: afero.NewMemMapFs(),
|
|
|
|
wantErr: true,
|
|
|
|
},
|
|
|
|
"continue on partially extracted": {
|
|
|
|
provider: cloudprovider.QEMU,
|
|
|
|
vars: qemuVars,
|
|
|
|
fs: afero.NewMemMapFs(),
|
|
|
|
partiallyExtracted: true,
|
|
|
|
wantErr: false,
|
|
|
|
},
|
|
|
|
"prepare workspace fails": {
|
|
|
|
provider: cloudprovider.QEMU,
|
|
|
|
vars: qemuVars,
|
|
|
|
fs: afero.NewReadOnlyFs(afero.NewMemMapFs()),
|
|
|
|
wantErr: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for name, tc := range testCases {
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
assert := assert.New(t)
|
|
|
|
require := require.New(t)
|
|
|
|
|
|
|
|
c := &Client{
|
|
|
|
tf: &stubTerraform{},
|
|
|
|
file: file.NewHandler(tc.fs),
|
|
|
|
workingDir: constants.TerraformWorkingDir,
|
|
|
|
}
|
|
|
|
|
|
|
|
err := c.PrepareWorkspace(tc.provider, tc.vars)
|
|
|
|
|
|
|
|
// Test case: Check if we can continue to create on an incomplete workspace.
|
|
|
|
if tc.partiallyExtracted {
|
|
|
|
require.NoError(c.file.Remove(filepath.Join(c.workingDir, "main.tf")))
|
|
|
|
err = c.PrepareWorkspace(tc.provider, tc.vars)
|
|
|
|
}
|
|
|
|
|
|
|
|
if tc.wantErr {
|
|
|
|
assert.Error(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
assert.NoError(err)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-27 09:22:29 +02:00
|
|
|
func TestCreateCluster(t *testing.T) {
|
|
|
|
someErr := errors.New("failed")
|
|
|
|
newTestState := func() *tfjson.State {
|
2022-09-26 15:52:31 +02:00
|
|
|
workingState := tfjson.State{
|
|
|
|
Values: &tfjson.StateValues{
|
|
|
|
Outputs: map[string]*tfjson.StateOutput{
|
|
|
|
"ip": {
|
|
|
|
Value: "192.0.2.100",
|
|
|
|
},
|
2022-11-26 19:44:34 +01:00
|
|
|
"initSecret": {
|
|
|
|
Value: "initSecret",
|
|
|
|
},
|
2022-09-26 15:52:31 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
return &workingState
|
|
|
|
}
|
2022-09-27 09:22:29 +02:00
|
|
|
qemuVars := &QEMUVariables{
|
|
|
|
CommonVariables: CommonVariables{
|
|
|
|
Name: "name",
|
|
|
|
CountControlPlanes: 1,
|
|
|
|
CountWorkers: 2,
|
|
|
|
StateDiskSizeGB: 11,
|
|
|
|
},
|
|
|
|
CPUCount: 1,
|
|
|
|
MemorySizeMiB: 1024,
|
|
|
|
ImagePath: "path",
|
|
|
|
ImageFormat: "format",
|
|
|
|
MetadataAPIImage: "api",
|
|
|
|
}
|
2022-09-26 15:52:31 +02:00
|
|
|
|
|
|
|
testCases := map[string]struct {
|
|
|
|
provider cloudprovider.Provider
|
2022-09-27 09:22:29 +02:00
|
|
|
vars Variables
|
2022-09-26 15:52:31 +02:00
|
|
|
tf *stubTerraform
|
|
|
|
fs afero.Fs
|
|
|
|
wantErr bool
|
|
|
|
}{
|
|
|
|
"works": {
|
|
|
|
provider: cloudprovider.QEMU,
|
2022-09-27 09:22:29 +02:00
|
|
|
vars: qemuVars,
|
|
|
|
tf: &stubTerraform{showState: newTestState()},
|
|
|
|
fs: afero.NewMemMapFs(),
|
2022-09-26 15:52:31 +02:00
|
|
|
},
|
|
|
|
"init fails": {
|
|
|
|
provider: cloudprovider.QEMU,
|
2022-11-15 14:00:44 +01:00
|
|
|
vars: qemuVars,
|
2022-09-27 09:22:29 +02:00
|
|
|
tf: &stubTerraform{initErr: someErr},
|
|
|
|
fs: afero.NewMemMapFs(),
|
|
|
|
wantErr: true,
|
2022-09-26 15:52:31 +02:00
|
|
|
},
|
|
|
|
"apply fails": {
|
|
|
|
provider: cloudprovider.QEMU,
|
2022-09-27 09:22:29 +02:00
|
|
|
vars: qemuVars,
|
|
|
|
tf: &stubTerraform{applyErr: someErr},
|
|
|
|
fs: afero.NewMemMapFs(),
|
|
|
|
wantErr: true,
|
2022-09-26 15:52:31 +02:00
|
|
|
},
|
|
|
|
"show fails": {
|
|
|
|
provider: cloudprovider.QEMU,
|
2022-09-27 09:22:29 +02:00
|
|
|
vars: qemuVars,
|
|
|
|
tf: &stubTerraform{showErr: someErr},
|
|
|
|
fs: afero.NewMemMapFs(),
|
|
|
|
wantErr: true,
|
2022-09-26 15:52:31 +02:00
|
|
|
},
|
|
|
|
"no ip": {
|
|
|
|
provider: cloudprovider.QEMU,
|
2022-09-27 09:22:29 +02:00
|
|
|
vars: qemuVars,
|
2022-09-26 15:52:31 +02:00
|
|
|
tf: &stubTerraform{
|
|
|
|
showState: &tfjson.State{
|
|
|
|
Values: &tfjson.StateValues{
|
|
|
|
Outputs: map[string]*tfjson.StateOutput{},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
fs: afero.NewMemMapFs(),
|
|
|
|
wantErr: true,
|
|
|
|
},
|
2022-09-27 09:22:29 +02:00
|
|
|
"ip has wrong type": {
|
2022-09-26 15:52:31 +02:00
|
|
|
provider: cloudprovider.QEMU,
|
2022-09-27 09:22:29 +02:00
|
|
|
vars: qemuVars,
|
2022-09-26 15:52:31 +02:00
|
|
|
tf: &stubTerraform{
|
2022-09-27 09:22:29 +02:00
|
|
|
showState: &tfjson.State{
|
|
|
|
Values: &tfjson.StateValues{
|
|
|
|
Outputs: map[string]*tfjson.StateOutput{"ip": {Value: 42}},
|
|
|
|
},
|
|
|
|
},
|
2022-09-26 15:52:31 +02:00
|
|
|
},
|
2022-09-27 09:22:29 +02:00
|
|
|
fs: afero.NewMemMapFs(),
|
2022-09-26 15:52:31 +02:00
|
|
|
wantErr: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for name, tc := range testCases {
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
assert := assert.New(t)
|
2022-11-15 14:00:44 +01:00
|
|
|
require := require.New(t)
|
2022-09-26 15:52:31 +02:00
|
|
|
|
|
|
|
c := &Client{
|
2022-11-14 18:18:58 +01:00
|
|
|
tf: tc.tf,
|
|
|
|
file: file.NewHandler(tc.fs),
|
|
|
|
workingDir: constants.TerraformWorkingDir,
|
2022-09-26 15:52:31 +02:00
|
|
|
}
|
|
|
|
|
2022-11-15 14:00:44 +01:00
|
|
|
require.NoError(c.PrepareWorkspace(tc.provider, tc.vars))
|
2022-11-26 19:44:34 +01:00
|
|
|
ip, initSecret, err := c.CreateCluster(context.Background())
|
2022-09-27 09:22:29 +02:00
|
|
|
|
2022-09-26 15:52:31 +02:00
|
|
|
if tc.wantErr {
|
|
|
|
assert.Error(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
assert.NoError(err)
|
2022-10-11 12:24:33 +02:00
|
|
|
assert.Equal("192.0.2.100", ip)
|
2022-11-26 19:44:34 +01:00
|
|
|
assert.Equal("initSecret", initSecret)
|
2022-09-26 15:52:31 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestDestroyInstances(t *testing.T) {
|
|
|
|
testCases := map[string]struct {
|
|
|
|
tf *stubTerraform
|
|
|
|
wantErr bool
|
|
|
|
}{
|
|
|
|
"works": {
|
|
|
|
tf: &stubTerraform{},
|
|
|
|
},
|
|
|
|
"destroy fails": {
|
|
|
|
tf: &stubTerraform{
|
|
|
|
destroyErr: errors.New("error"),
|
|
|
|
},
|
|
|
|
wantErr: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for name, tc := range testCases {
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
assert := assert.New(t)
|
|
|
|
|
|
|
|
c := &Client{
|
2022-10-26 15:57:00 +02:00
|
|
|
tf: tc.tf,
|
2022-09-26 15:52:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
err := c.DestroyCluster(context.Background())
|
|
|
|
if tc.wantErr {
|
|
|
|
assert.Error(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
assert.NoError(err)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCleanupWorkspace(t *testing.T) {
|
|
|
|
someContent := []byte("some content")
|
|
|
|
|
|
|
|
testCases := map[string]struct {
|
|
|
|
provider cloudprovider.Provider
|
|
|
|
prepareFS func(file.Handler) error
|
|
|
|
wantErr bool
|
|
|
|
}{
|
|
|
|
"files are cleaned up": {
|
|
|
|
provider: cloudprovider.QEMU,
|
|
|
|
prepareFS: func(f file.Handler) error {
|
|
|
|
var err error
|
|
|
|
err = multierr.Append(err, f.Write("terraform.tfvars", someContent))
|
|
|
|
err = multierr.Append(err, f.Write("terraform.tfstate", someContent))
|
|
|
|
return multierr.Append(err, f.Write("terraform.tfstate.backup", someContent))
|
|
|
|
},
|
|
|
|
},
|
|
|
|
"no error if files do not exist": {
|
|
|
|
provider: cloudprovider.QEMU,
|
|
|
|
prepareFS: func(f file.Handler) error { return nil },
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for name, tc := range testCases {
|
|
|
|
t.Run(name, func(t *testing.T) {
|
|
|
|
assert := assert.New(t)
|
|
|
|
require := require.New(t)
|
|
|
|
|
|
|
|
file := file.NewHandler(afero.NewMemMapFs())
|
|
|
|
require.NoError(tc.prepareFS(file))
|
|
|
|
|
|
|
|
c := &Client{
|
2022-11-14 18:18:58 +01:00
|
|
|
file: file,
|
|
|
|
tf: &stubTerraform{},
|
|
|
|
workingDir: constants.TerraformWorkingDir,
|
2022-09-26 15:52:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
err := c.CleanUpWorkspace()
|
|
|
|
if tc.wantErr {
|
|
|
|
assert.Error(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
assert.NoError(err)
|
2022-11-14 18:18:58 +01:00
|
|
|
_, err = file.Stat(filepath.Join(c.workingDir, "terraform.tfvars"))
|
2022-09-26 15:52:31 +02:00
|
|
|
assert.ErrorIs(err, fs.ErrNotExist)
|
2022-11-14 18:18:58 +01:00
|
|
|
_, err = file.Stat(filepath.Join(c.workingDir, "terraform.tfstate"))
|
2022-09-26 15:52:31 +02:00
|
|
|
assert.ErrorIs(err, fs.ErrNotExist)
|
2022-11-14 18:18:58 +01:00
|
|
|
_, err = file.Stat(filepath.Join(c.workingDir, "terraform.tfstate.backup"))
|
2022-09-26 15:52:31 +02:00
|
|
|
assert.ErrorIs(err, fs.ErrNotExist)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type stubTerraform struct {
|
|
|
|
applyErr error
|
|
|
|
destroyErr error
|
|
|
|
initErr error
|
|
|
|
showErr error
|
|
|
|
showState *tfjson.State
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *stubTerraform) Apply(context.Context, ...tfexec.ApplyOption) error {
|
|
|
|
return s.applyErr
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *stubTerraform) Destroy(context.Context, ...tfexec.DestroyOption) error {
|
|
|
|
return s.destroyErr
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *stubTerraform) Init(context.Context, ...tfexec.InitOption) error {
|
|
|
|
return s.initErr
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *stubTerraform) Show(context.Context, ...tfexec.ShowOption) (*tfjson.State, error) {
|
|
|
|
return s.showState, s.showErr
|
|
|
|
}
|