2013-12-24 01:07:43 -05:00
|
|
|
package common
|
2013-12-23 17:38:54 -05:00
|
|
|
|
|
|
|
import (
|
2018-01-22 19:03:49 -05:00
|
|
|
"context"
|
2013-12-23 17:38:54 -05:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"testing"
|
|
|
|
|
2018-01-19 19:18:44 -05:00
|
|
|
"github.com/hashicorp/packer/helper/multistep"
|
2020-09-10 22:43:13 -04:00
|
|
|
"github.com/stretchr/testify/assert"
|
2013-12-23 17:38:54 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
func testVMXFile(t *testing.T) string {
|
|
|
|
tf, err := ioutil.TempFile("", "packer")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2018-07-02 08:44:41 -04:00
|
|
|
|
|
|
|
// displayName must always be set
|
|
|
|
err = WriteVMX(tf.Name(), map[string]string{"displayName": "PackerBuild"})
|
2019-09-27 19:05:12 -04:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("error writing .vmx file: %v", err)
|
|
|
|
}
|
2013-12-23 17:38:54 -05:00
|
|
|
tf.Close()
|
|
|
|
|
|
|
|
return tf.Name()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStepConfigureVMX_impl(t *testing.T) {
|
|
|
|
var _ multistep.Step = new(StepConfigureVMX)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStepConfigureVMX(t *testing.T) {
|
|
|
|
state := testState(t)
|
|
|
|
step := new(StepConfigureVMX)
|
|
|
|
step.CustomData = map[string]string{
|
|
|
|
"foo": "bar",
|
|
|
|
}
|
|
|
|
|
|
|
|
vmxPath := testVMXFile(t)
|
|
|
|
defer os.Remove(vmxPath)
|
|
|
|
state.Put("vmx_path", vmxPath)
|
|
|
|
|
|
|
|
// Test the run
|
2018-01-22 19:03:49 -05:00
|
|
|
if action := step.Run(context.Background(), state); action != multistep.ActionContinue {
|
2013-12-23 17:38:54 -05:00
|
|
|
t.Fatalf("bad action: %#v", action)
|
|
|
|
}
|
|
|
|
if _, ok := state.GetOk("error"); ok {
|
|
|
|
t.Fatal("should NOT have error")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test the resulting data
|
|
|
|
vmxContents, err := ioutil.ReadFile(vmxPath)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
vmxData := ParseVMX(string(vmxContents))
|
|
|
|
|
|
|
|
cases := []struct {
|
|
|
|
Key string
|
|
|
|
Value string
|
|
|
|
}{
|
|
|
|
// Stuff we set
|
|
|
|
{"msg.autoanswer", "true"},
|
|
|
|
{"uuid.action", "create"},
|
|
|
|
|
|
|
|
// Custom data
|
|
|
|
{"foo", "bar"},
|
|
|
|
|
|
|
|
// Stuff that should NOT exist
|
|
|
|
{"floppy0.present", ""},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range cases {
|
|
|
|
if tc.Value == "" {
|
|
|
|
if _, ok := vmxData[tc.Key]; ok {
|
|
|
|
t.Fatalf("should not have key: %s", tc.Key)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if vmxData[tc.Key] != tc.Value {
|
|
|
|
t.Fatalf("bad: %s %#v", tc.Key, vmxData[tc.Key])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStepConfigureVMX_floppyPath(t *testing.T) {
|
|
|
|
state := testState(t)
|
|
|
|
step := new(StepConfigureVMX)
|
|
|
|
|
|
|
|
vmxPath := testVMXFile(t)
|
|
|
|
defer os.Remove(vmxPath)
|
|
|
|
|
|
|
|
state.Put("floppy_path", "foo")
|
|
|
|
state.Put("vmx_path", vmxPath)
|
|
|
|
|
|
|
|
// Test the run
|
2018-01-22 19:03:49 -05:00
|
|
|
if action := step.Run(context.Background(), state); action != multistep.ActionContinue {
|
2013-12-23 17:38:54 -05:00
|
|
|
t.Fatalf("bad action: %#v", action)
|
|
|
|
}
|
|
|
|
if _, ok := state.GetOk("error"); ok {
|
|
|
|
t.Fatal("should NOT have error")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test the resulting data
|
|
|
|
vmxContents, err := ioutil.ReadFile(vmxPath)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
vmxData := ParseVMX(string(vmxContents))
|
|
|
|
|
|
|
|
cases := []struct {
|
|
|
|
Key string
|
|
|
|
Value string
|
|
|
|
}{
|
|
|
|
{"floppy0.present", "TRUE"},
|
|
|
|
{"floppy0.filetype", "file"},
|
|
|
|
{"floppy0.filename", "foo"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range cases {
|
|
|
|
if tc.Value == "" {
|
|
|
|
if _, ok := vmxData[tc.Key]; ok {
|
|
|
|
t.Fatalf("should not have key: %s", tc.Key)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if vmxData[tc.Key] != tc.Value {
|
|
|
|
t.Fatalf("bad: %s %#v", tc.Key, vmxData[tc.Key])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStepConfigureVMX_generatedAddresses(t *testing.T) {
|
|
|
|
state := testState(t)
|
|
|
|
step := new(StepConfigureVMX)
|
|
|
|
|
|
|
|
vmxPath := testVMXFile(t)
|
|
|
|
defer os.Remove(vmxPath)
|
|
|
|
|
2018-07-02 08:44:41 -04:00
|
|
|
additionalTestVmxData := []struct {
|
|
|
|
Key string
|
|
|
|
Value string
|
|
|
|
}{
|
|
|
|
{"foo", "bar"},
|
|
|
|
{"ethernet0.generatedaddress", "foo"},
|
|
|
|
{"ethernet1.generatedaddress", "foo"},
|
|
|
|
{"ethernet1.generatedaddressoffset", "foo"},
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get any existing VMX data from the VMX file
|
|
|
|
vmxData, err := ReadVMX(vmxPath)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the additional key/value pairs we need for this test to the existing VMX data
|
|
|
|
for _, data := range additionalTestVmxData {
|
|
|
|
vmxData[data.Key] = data.Value
|
|
|
|
}
|
|
|
|
|
|
|
|
// Recreate the VMX file so it includes all the data needed for this test
|
|
|
|
err = WriteVMX(vmxPath, vmxData)
|
2013-12-23 17:38:54 -05:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
state.Put("vmx_path", vmxPath)
|
|
|
|
|
|
|
|
// Test the run
|
2018-01-22 19:03:49 -05:00
|
|
|
if action := step.Run(context.Background(), state); action != multistep.ActionContinue {
|
2013-12-23 17:38:54 -05:00
|
|
|
t.Fatalf("bad action: %#v", action)
|
|
|
|
}
|
|
|
|
if _, ok := state.GetOk("error"); ok {
|
|
|
|
t.Fatal("should NOT have error")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test the resulting data
|
|
|
|
vmxContents, err := ioutil.ReadFile(vmxPath)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2018-07-02 08:44:41 -04:00
|
|
|
vmxData = ParseVMX(string(vmxContents))
|
2013-12-23 17:38:54 -05:00
|
|
|
|
|
|
|
cases := []struct {
|
|
|
|
Key string
|
|
|
|
Value string
|
|
|
|
}{
|
|
|
|
{"foo", "bar"},
|
|
|
|
{"ethernet0.generatedaddress", ""},
|
|
|
|
{"ethernet1.generatedaddress", ""},
|
|
|
|
{"ethernet1.generatedaddressoffset", ""},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range cases {
|
|
|
|
if tc.Value == "" {
|
|
|
|
if _, ok := vmxData[tc.Key]; ok {
|
|
|
|
t.Fatalf("should not have key: %s", tc.Key)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if vmxData[tc.Key] != tc.Value {
|
|
|
|
t.Fatalf("bad: %s %#v", tc.Key, vmxData[tc.Key])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-07-02 09:33:04 -04:00
|
|
|
|
|
|
|
// Should fail if the displayName key is not found in the VMX
|
|
|
|
func TestStepConfigureVMX_displayNameMissing(t *testing.T) {
|
|
|
|
state := testState(t)
|
|
|
|
step := new(StepConfigureVMX)
|
|
|
|
|
|
|
|
// testVMXFile adds displayName key/value pair to the VMX
|
|
|
|
vmxPath := testVMXFile(t)
|
|
|
|
defer os.Remove(vmxPath)
|
|
|
|
|
|
|
|
// Bad: Delete displayName from the VMX/Create an empty VMX file
|
|
|
|
err := WriteVMX(vmxPath, map[string]string{})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
state.Put("vmx_path", vmxPath)
|
|
|
|
|
|
|
|
// Test the run
|
|
|
|
if action := step.Run(context.Background(), state); action != multistep.ActionHalt {
|
|
|
|
t.Fatalf("bad action: %#v. Should halt when displayName key is missing from VMX", action)
|
|
|
|
}
|
|
|
|
if _, ok := state.GetOk("error"); !ok {
|
|
|
|
t.Fatal("should store error in state when displayName key is missing from VMX")
|
|
|
|
}
|
|
|
|
}
|
2018-07-02 09:44:55 -04:00
|
|
|
|
|
|
|
// Should store the value of displayName in the statebag
|
|
|
|
func TestStepConfigureVMX_displayNameStore(t *testing.T) {
|
|
|
|
state := testState(t)
|
|
|
|
step := new(StepConfigureVMX)
|
|
|
|
|
|
|
|
// testVMXFile adds displayName key/value pair to the VMX
|
|
|
|
vmxPath := testVMXFile(t)
|
|
|
|
defer os.Remove(vmxPath)
|
|
|
|
|
|
|
|
state.Put("vmx_path", vmxPath)
|
|
|
|
|
|
|
|
// Test the run
|
|
|
|
if action := step.Run(context.Background(), state); action != multistep.ActionContinue {
|
|
|
|
t.Fatalf("bad action: %#v", action)
|
|
|
|
}
|
|
|
|
if _, ok := state.GetOk("error"); ok {
|
|
|
|
t.Fatal("should NOT have error")
|
|
|
|
}
|
|
|
|
|
|
|
|
// The value of displayName must be stored in the statebag
|
|
|
|
if _, ok := state.GetOk("display_name"); !ok {
|
|
|
|
t.Fatalf("displayName should be stored in the statebag as 'display_name'")
|
|
|
|
}
|
|
|
|
}
|
2018-07-02 10:04:21 -04:00
|
|
|
|
|
|
|
func TestStepConfigureVMX_vmxPathBad(t *testing.T) {
|
|
|
|
state := testState(t)
|
|
|
|
step := new(StepConfigureVMX)
|
|
|
|
|
|
|
|
state.Put("vmx_path", "some_bad_path")
|
|
|
|
|
|
|
|
// Test the run
|
|
|
|
if action := step.Run(context.Background(), state); action != multistep.ActionHalt {
|
|
|
|
t.Fatalf("bad action: %#v. Should halt when vmxPath is bad", action)
|
|
|
|
}
|
|
|
|
if _, ok := state.GetOk("error"); !ok {
|
|
|
|
t.Fatal("should store error in state when vmxPath is bad")
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2020-09-10 22:43:13 -04:00
|
|
|
|
|
|
|
func TestStepConfigureVMX_DefaultDiskAndCDROMTypes(t *testing.T) {
|
|
|
|
type testCase struct {
|
|
|
|
inDiskAdapter string
|
|
|
|
inCDromAdapter string
|
|
|
|
expectedOut DiskAndCDConfigData
|
|
|
|
reason string
|
|
|
|
}
|
|
|
|
testcases := []testCase{
|
|
|
|
{
|
|
|
|
inDiskAdapter: "",
|
|
|
|
inCDromAdapter: "",
|
|
|
|
expectedOut: DiskAndCDConfigData{
|
|
|
|
SCSI_Present: "TRUE",
|
|
|
|
SCSI_diskAdapterType: "",
|
|
|
|
SATA_Present: "FALSE",
|
|
|
|
NVME_Present: "FALSE",
|
|
|
|
|
|
|
|
DiskType: "scsi",
|
|
|
|
CDROMType: "ide",
|
|
|
|
CDROMType_PrimarySecondary: "0",
|
|
|
|
},
|
|
|
|
reason: "Test that default creases scsi disk with ide cd",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
inDiskAdapter: "ide",
|
|
|
|
inCDromAdapter: "",
|
|
|
|
expectedOut: DiskAndCDConfigData{
|
|
|
|
SCSI_Present: "FALSE",
|
|
|
|
SCSI_diskAdapterType: "lsilogic",
|
|
|
|
SATA_Present: "FALSE",
|
|
|
|
NVME_Present: "FALSE",
|
|
|
|
|
|
|
|
DiskType: "ide",
|
|
|
|
CDROMType: "ide",
|
|
|
|
CDROMType_PrimarySecondary: "1",
|
|
|
|
},
|
|
|
|
reason: "ide disk adapter should pass through and not get defaulted to scsi",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
inDiskAdapter: "sata",
|
|
|
|
inCDromAdapter: "",
|
|
|
|
expectedOut: DiskAndCDConfigData{
|
|
|
|
SCSI_Present: "FALSE",
|
|
|
|
SCSI_diskAdapterType: "lsilogic",
|
|
|
|
SATA_Present: "TRUE",
|
|
|
|
NVME_Present: "FALSE",
|
|
|
|
|
|
|
|
DiskType: "sata",
|
|
|
|
CDROMType: "sata",
|
|
|
|
CDROMType_PrimarySecondary: "1",
|
|
|
|
},
|
|
|
|
reason: "when disk is set to sata, cdromtype should also default to sata",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
inDiskAdapter: "nvme",
|
|
|
|
inCDromAdapter: "",
|
|
|
|
expectedOut: DiskAndCDConfigData{
|
|
|
|
SCSI_Present: "FALSE",
|
|
|
|
SCSI_diskAdapterType: "lsilogic",
|
|
|
|
SATA_Present: "TRUE",
|
|
|
|
NVME_Present: "TRUE",
|
|
|
|
|
|
|
|
DiskType: "nvme",
|
|
|
|
CDROMType: "sata",
|
|
|
|
CDROMType_PrimarySecondary: "0",
|
|
|
|
},
|
|
|
|
reason: "when disk is set to nvme, cdromtype should default to sata",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
inDiskAdapter: "scsi",
|
|
|
|
inCDromAdapter: "",
|
|
|
|
expectedOut: DiskAndCDConfigData{
|
|
|
|
SCSI_Present: "TRUE",
|
|
|
|
SCSI_diskAdapterType: "lsilogic",
|
|
|
|
SATA_Present: "FALSE",
|
|
|
|
NVME_Present: "FALSE",
|
|
|
|
|
|
|
|
DiskType: "scsi",
|
|
|
|
CDROMType: "ide",
|
|
|
|
CDROMType_PrimarySecondary: "0",
|
|
|
|
},
|
|
|
|
reason: "when disk is set to scsi, adapter should default back to lsilogic",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
inDiskAdapter: "scsi",
|
|
|
|
inCDromAdapter: "scsi",
|
|
|
|
expectedOut: DiskAndCDConfigData{
|
|
|
|
SCSI_Present: "TRUE",
|
|
|
|
SCSI_diskAdapterType: "lsilogic",
|
|
|
|
SATA_Present: "FALSE",
|
|
|
|
NVME_Present: "FALSE",
|
|
|
|
|
|
|
|
DiskType: "scsi",
|
|
|
|
CDROMType: "scsi",
|
|
|
|
CDROMType_PrimarySecondary: "0",
|
|
|
|
},
|
|
|
|
reason: "when cdrom adapter is set, it should override the default",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, tc := range testcases {
|
|
|
|
diskConfigData := DefaultDiskAndCDROMTypes(tc.inDiskAdapter, tc.inCDromAdapter)
|
|
|
|
assert.Equal(t, diskConfigData, tc.expectedOut, tc.reason)
|
|
|
|
}
|
|
|
|
}
|