2020-02-21 02:01:09 -05:00
|
|
|
package common
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"regexp"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
2020-12-17 16:29:25 -05:00
|
|
|
"github.com/hashicorp/packer-plugin-sdk/multistep"
|
|
|
|
packersdk "github.com/hashicorp/packer-plugin-sdk/packer"
|
2020-02-21 02:01:09 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
type StepSetFirstBootDevice struct {
|
|
|
|
Generation uint
|
|
|
|
FirstBootDevice string
|
|
|
|
}
|
|
|
|
|
|
|
|
func ParseBootDeviceIdentifier(deviceIdentifier string, generation uint) (string, uint, uint, error) {
|
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
// all input strings are forced to upperCase for comparison, I believe this is
|
|
|
|
// safe as all of our values are 7bit ASCII clean.
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
lookupDeviceIdentifier := strings.ToUpper(deviceIdentifier)
|
|
|
|
|
2020-02-22 03:03:42 -05:00
|
|
|
if generation == 1 {
|
2020-02-22 01:29:05 -05:00
|
|
|
|
|
|
|
// Gen1 values are a simple set of if/then/else values, which we coalesce into a map
|
|
|
|
// here for simplicity
|
|
|
|
|
2020-02-22 03:03:42 -05:00
|
|
|
lookupTable := map[string]string{
|
2020-02-22 01:29:05 -05:00
|
|
|
"FLOPPY": "FLOPPY",
|
2020-02-22 03:03:42 -05:00
|
|
|
"IDE": "IDE",
|
|
|
|
"NET": "NET",
|
|
|
|
"CD": "CD",
|
|
|
|
"DVD": "CD",
|
2020-02-22 01:29:05 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
controllerType, isDefined := lookupTable[lookupDeviceIdentifier]
|
2020-02-22 03:03:42 -05:00
|
|
|
if !isDefined {
|
2020-02-22 01:29:05 -05:00
|
|
|
|
|
|
|
return "", 0, 0, fmt.Errorf("The value %q is not a properly formatted device group identifier.", deviceIdentifier)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// success
|
|
|
|
return controllerType, 0, 0, nil
|
2020-02-21 02:01:09 -05:00
|
|
|
}
|
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
// everything else is treated as generation 2... the first set of lookups covers
|
|
|
|
// the simple options..
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 03:03:42 -05:00
|
|
|
lookupTable := map[string]string{
|
|
|
|
"CD": "CD",
|
2020-02-22 01:29:05 -05:00
|
|
|
"DVD": "CD",
|
|
|
|
"NET": "NET",
|
2020-02-21 02:01:09 -05:00
|
|
|
}
|
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
controllerType, isDefined := lookupTable[lookupDeviceIdentifier]
|
2020-02-22 03:03:42 -05:00
|
|
|
if isDefined {
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
// these types do not require controllerNumber or controllerLocation
|
|
|
|
return controllerType, 0, 0, nil
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
}
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
// not a simple option, check for a controllerType:controllerNumber:controllerLocation formatted
|
|
|
|
// device..
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 03:40:41 -05:00
|
|
|
r, err := regexp.Compile(`^(IDE|SCSI):(\d+):(\d+)$`)
|
2020-02-22 01:29:05 -05:00
|
|
|
if err != nil {
|
|
|
|
return "", 0, 0, err
|
|
|
|
}
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
controllerMatch := r.FindStringSubmatch(lookupDeviceIdentifier)
|
|
|
|
if controllerMatch != nil {
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
var controllerLocation int64
|
|
|
|
var controllerNumber int64
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
// NOTE: controllerNumber and controllerLocation cannot be negative, the regex expression
|
|
|
|
// would not have matched if either number was signed
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
controllerNumber, err = strconv.ParseInt(controllerMatch[2], 10, 8)
|
|
|
|
if err == nil {
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
controllerLocation, err = strconv.ParseInt(controllerMatch[3], 10, 8)
|
|
|
|
if err == nil {
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
return controllerMatch[1], uint(controllerNumber), uint(controllerLocation), nil
|
2020-02-21 02:01:09 -05:00
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
}
|
2020-02-21 02:01:09 -05:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
return "", 0, 0, err
|
2020-02-21 02:01:09 -05:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-02-22 01:29:05 -05:00
|
|
|
return "", 0, 0, fmt.Errorf("The value %q is not a properly formatted device identifier.", deviceIdentifier)
|
2020-02-21 02:01:09 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *StepSetFirstBootDevice) Run(ctx context.Context, state multistep.StateBag) multistep.StepAction {
|
|
|
|
|
|
|
|
driver := state.Get("driver").(Driver)
|
2020-11-19 14:54:31 -05:00
|
|
|
ui := state.Get("ui").(packersdk.Ui)
|
2020-02-21 02:01:09 -05:00
|
|
|
vmName := state.Get("vmName").(string)
|
|
|
|
|
|
|
|
if s.FirstBootDevice != "" {
|
|
|
|
|
|
|
|
controllerType, controllerNumber, controllerLocation, err := ParseBootDeviceIdentifier(s.FirstBootDevice, s.Generation)
|
|
|
|
if err == nil {
|
|
|
|
|
|
|
|
switch {
|
|
|
|
|
|
|
|
case controllerType == "CD":
|
|
|
|
{
|
|
|
|
// the "DVD" controller is special, we only apply the setting if we actually mounted
|
|
|
|
// an ISO and only if that was mounted as the "IsoUrl" not a secondary ISO.
|
|
|
|
|
|
|
|
dvdControllerState := state.Get("os.dvd.properties")
|
|
|
|
if dvdControllerState == nil {
|
|
|
|
|
|
|
|
ui.Say("First Boot Device is DVD, but no primary ISO mounted. Ignoring.")
|
|
|
|
return multistep.ActionContinue
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
ui.Say(fmt.Sprintf("Setting boot device to %q", s.FirstBootDevice))
|
|
|
|
dvdController := dvdControllerState.(DvdControllerProperties)
|
|
|
|
err = driver.SetFirstBootDevice(vmName, controllerType, dvdController.ControllerNumber, dvdController.ControllerLocation, s.Generation)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
// anything else, we just pass as is..
|
|
|
|
ui.Say(fmt.Sprintf("Setting boot device to %q", s.FirstBootDevice))
|
|
|
|
err = driver.SetFirstBootDevice(vmName, controllerType, controllerNumber, controllerLocation, s.Generation)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
err := fmt.Errorf("Error setting first boot device: %s", err)
|
|
|
|
state.Put("error", err)
|
|
|
|
ui.Error(err.Error())
|
|
|
|
return multistep.ActionHalt
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return multistep.ActionContinue
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *StepSetFirstBootDevice) Cleanup(state multistep.StateBag) {
|
|
|
|
// do nothing
|
|
|
|
}
|