2020-01-07 19:59:31 -05:00
//go:generate struct-markdown
2020-02-21 10:47:46 -05:00
//go:generate mapstructure-to-hcl2 -type NIC,CreateConfig,DiskConfig
2020-01-07 19:59:31 -05:00
2018-01-24 09:56:14 -05:00
package iso
import (
2018-10-31 17:42:24 -04:00
"context"
2018-01-24 09:56:14 -05:00
"fmt"
2020-04-23 08:07:07 -04:00
"path"
2020-01-08 18:28:26 -05:00
2020-01-07 19:59:31 -05:00
"github.com/hashicorp/packer/builder/vsphere/common"
"github.com/hashicorp/packer/builder/vsphere/driver"
2018-10-31 17:42:24 -04:00
"github.com/hashicorp/packer/helper/multistep"
2018-01-24 09:56:14 -05:00
"github.com/hashicorp/packer/packer"
)
2020-04-13 14:28:50 -04:00
// Defines a Network Adapter
//
// Example that creates two network adapters:
//
2020-08-05 13:23:52 -04:00
// In JSON:
2020-04-13 14:28:50 -04:00
// ```json
// "network_adapters": [
// {
// "network": "VM Network",
// "network_card": "vmxnet3"
// },
// {
// "network": "OtherNetwork",
// "network_card": "vmxnet3"
// }
// ],
// ```
2020-08-05 13:23:52 -04:00
// In HCL2:
// ```hcl
// network_adapters {
// network = "VM Network"
// network_card = "vmxnet3"
// }
// network_adapters {
// network = "OtherNetwork"
// network_card = "vmxnet3"
// }
// ```
2020-02-14 11:51:57 -05:00
type NIC struct {
2020-08-13 11:26:40 -04:00
// Set the network in which the VM will be connected to. If no network is
// specified, `host` must be specified to allow Packer to look for the
// available network. If the network is inside a network folder in vCenter,
// you need to provide the full path to the network.
2020-02-14 11:51:57 -05:00
Network string ` mapstructure:"network" `
// Set VM network card type. Example `vmxnet3`.
NetworkCard string ` mapstructure:"network_card" required:"true" `
// Set network card MAC address
MacAddress string ` mapstructure:"mac_address" `
// Enable DirectPath I/O passthrough
Passthrough * bool ` mapstructure:"passthrough" `
}
2020-04-13 14:28:50 -04:00
// Defines the disk storage for a VM.
//
// Example that will create a 15GB and a 20GB disk on the VM. The second disk will be thin provisioned:
//
2020-08-05 13:23:52 -04:00
// In JSON:
2020-04-13 14:28:50 -04:00
// ```json
// "storage": [
// {
2020-07-10 06:25:46 -04:00
// "disk_size": 15000
2020-04-13 14:28:50 -04:00
// },
// {
// "disk_size": 20000,
// "disk_thin_provisioned": true
// }
// ],
// ```
2020-08-05 13:23:52 -04:00
// In HCL2:
// ```hcl
// storage {
// disk_size = 15000
// }
// storage {
// disk_size = 20000
// disk_thin_provisioned = true
// }
// ```
2020-07-10 06:25:46 -04:00
//
// Example that creates 2 pvscsi controllers and adds 2 disks to each one:
2020-08-05 13:23:52 -04:00
//
// In JSON:
2020-07-10 06:25:46 -04:00
// ```json
// "disk_controller_type": ["pvscsi", "pvscsi"],
// "storage": [
// {
// "disk_size": 15000,
// "disk_controller_index": 0
// },
// {
// "disk_size": 15000,
// "disk_controller_index": 0
// },
// {
// "disk_size": 15000,
// "disk_controller_index": 1
// },
// {
// "disk_size": 15000,
// "disk_controller_index": 1
// }
// ],
// ```
2020-08-05 13:23:52 -04:00
//
// In HCL2:
// ```hcl
// disk_controller_type = ["pvscsi", "pvscsi"]
// storage {
// disk_size = 15000,
// disk_controller_index = 0
// }
// storage {
// disk_size = 15000
// disk_controller_index = 0
// }
// storage {
// disk_size = 15000
// disk_controller_index = 1
// }
// storage {
// disk_size = 15000
// disk_controller_index = 1
// }
// ```
2020-02-21 10:47:46 -05:00
type DiskConfig struct {
2020-04-13 14:28:50 -04:00
// The size of the disk in MB.
2020-02-21 10:47:46 -05:00
DiskSize int64 ` mapstructure:"disk_size" required:"true" `
// Enable VMDK thin provisioning for VM. Defaults to `false`.
DiskThinProvisioned bool ` mapstructure:"disk_thin_provisioned" `
// Enable VMDK eager scrubbing for VM. Defaults to `false`.
DiskEagerlyScrub bool ` mapstructure:"disk_eagerly_scrub" `
2020-07-10 06:25:46 -04:00
// The assigned disk controller. Defaults to the first one (0)
DiskControllerIndex int ` mapstructure:"disk_controller_index" `
2020-02-21 10:47:46 -05:00
}
2018-01-24 09:56:14 -05:00
type CreateConfig struct {
2020-01-07 19:59:31 -05:00
// Set VM hardware version. Defaults to the most current VM hardware
// version supported by vCenter. See
// [VMWare article 1003746](https://kb.vmware.com/s/article/1003746) for
// the full list of supported VM hardware versions.
Version uint ` mapstructure:"vm_version" `
// Set VM OS type. Defaults to `otherGuest`. See [
2020-03-06 09:30:11 -05:00
// here](https://code.vmware.com/apis/358/vsphere/doc/vim.vm.GuestOsDescriptor.GuestOsIdentifier.html)
2020-01-07 19:59:31 -05:00
// for a full list of possible values.
2018-05-06 17:26:04 -04:00
GuestOSType string ` mapstructure:"guest_os_type" `
2020-09-03 05:14:58 -04:00
// Set VM disk controller type. Example `lsilogic`, `pvscsi`, `nvme`, or `scsi`. Use a list to define additional controllers.
// Defaults to `lsilogic`. See
// [SCSI, SATA, and NVMe Storage Controller Conditions, Limitations, and Compatibility](https://docs.vmware.com/en/VMware-vSphere/7.0/com.vmware.vsphere.vm_admin.doc/GUID-5872D173-A076-42FE-8D0B-9DB0EB0E7362.html#GUID-5872D173-A076-42FE-8D0B-9DB0EB0E7362)
// for additional details.
2020-07-10 06:25:46 -04:00
DiskControllerType [ ] string ` mapstructure:"disk_controller_type" `
2020-03-03 11:53:16 -05:00
// A collection of one or more disks to be provisioned along with the VM.
2020-02-21 10:47:46 -05:00
Storage [ ] DiskConfig ` mapstructure:"storage" `
2020-02-14 11:51:57 -05:00
// Network adapters
NICs [ ] NIC ` mapstructure:"network_adapters" `
2020-07-13 20:25:56 -04:00
// Create USB controllers for the virtual machine. "usb" for a usb 2.0 controller. "xhci" for a usb 3.0 controller. There can only be at most one of each.
USBController [ ] string ` mapstructure:"usb_controller" `
2020-01-07 19:59:31 -05:00
// VM notes.
2018-11-08 11:50:52 -05:00
Notes string ` mapstructure:"notes" `
2018-01-24 09:56:14 -05:00
}
func ( c * CreateConfig ) Prepare ( ) [ ] error {
var errs [ ] error
2020-07-10 06:25:46 -04:00
// there should be at least one
if len ( c . DiskControllerType ) == 0 {
c . DiskControllerType = append ( c . DiskControllerType , "" )
}
2020-03-31 11:47:09 -04:00
if len ( c . Storage ) > 0 {
for i , storage := range c . Storage {
if storage . DiskSize == 0 {
errs = append ( errs , fmt . Errorf ( "storage[%d].'disk_size' is required" , i ) )
}
2020-07-10 06:25:46 -04:00
if storage . DiskControllerIndex >= len ( c . DiskControllerType ) {
errs = append ( errs , fmt . Errorf ( "storage[%d].'disk_controller_index' references an unknown disk controller" , i ) )
}
2020-03-31 11:47:09 -04:00
}
2018-03-20 05:09:36 -04:00
}
2018-05-06 17:26:04 -04:00
if c . GuestOSType == "" {
c . GuestOSType = "otherGuest"
2018-05-05 17:41:14 -04:00
}
2020-07-13 20:25:56 -04:00
usbCount := 0
xhciCount := 0
for i , s := range c . USBController {
switch s {
// 1 and true for backwards compatibility
case "usb" , "1" , "true" :
usbCount ++
case "xhci" :
xhciCount ++
2020-07-14 12:11:24 -04:00
// 0 and false for backwards compatibility
case "false" , "0" :
continue
2020-07-13 20:25:56 -04:00
default :
errs = append ( errs , fmt . Errorf ( "usb_controller[%d] references an unknown usb controller" , i ) )
}
}
if usbCount > 1 || xhciCount > 1 {
errs = append ( errs , fmt . Errorf ( "there can only be one usb controller and one xhci controller" ) )
}
2018-05-06 17:26:04 -04:00
return errs
2018-01-24 09:56:14 -05:00
}
type StepCreateVM struct {
2018-05-06 17:26:04 -04:00
Config * CreateConfig
Location * common . LocationConfig
2018-11-16 10:37:54 -05:00
Force bool
2018-01-24 09:56:14 -05:00
}
2018-04-25 07:22:38 -04:00
func ( s * StepCreateVM ) Run ( _ context . Context , state multistep . StateBag ) multistep . StepAction {
2018-01-24 09:56:14 -05:00
ui := state . Get ( "ui" ) . ( packer . Ui )
2020-08-31 04:34:41 -04:00
d := state . Get ( "driver" ) . ( driver . Driver )
2020-04-23 08:07:07 -04:00
vmPath := path . Join ( s . Location . Folder , s . Location . VMName )
2018-01-24 09:56:14 -05:00
2020-04-23 08:07:07 -04:00
err := d . PreCleanVM ( ui , vmPath , s . Force )
if err != nil {
state . Put ( "error" , err )
2018-11-16 10:37:54 -05:00
return multistep . ActionHalt
}
2018-01-24 09:56:14 -05:00
ui . Say ( "Creating VM..." )
2020-02-14 11:51:57 -05:00
// add network/network card an the first nic for backwards compatibility in the type is defined
var networkCards [ ] driver . NIC
for _ , nic := range s . Config . NICs {
networkCards = append ( networkCards , driver . NIC {
Network : nic . Network ,
NetworkCard : nic . NetworkCard ,
MacAddress : nic . MacAddress ,
Passthrough : nic . Passthrough ,
} )
}
2020-02-21 10:47:46 -05:00
// add disk as the first drive for backwards compatibility if the type is defined
var disks [ ] driver . Disk
for _ , disk := range s . Config . Storage {
disks = append ( disks , driver . Disk {
DiskSize : disk . DiskSize ,
DiskEagerlyScrub : disk . DiskEagerlyScrub ,
DiskThinProvisioned : disk . DiskThinProvisioned ,
2020-07-10 06:25:46 -04:00
ControllerIndex : disk . DiskControllerIndex ,
2020-02-21 10:47:46 -05:00
} )
}
2020-04-23 08:07:07 -04:00
vm , err := d . CreateVM ( & driver . CreateConfig {
2020-02-21 10:47:46 -05:00
DiskControllerType : s . Config . DiskControllerType ,
Storage : disks ,
Annotation : s . Config . Notes ,
Name : s . Location . VMName ,
Folder : s . Location . Folder ,
Cluster : s . Location . Cluster ,
Host : s . Location . Host ,
ResourcePool : s . Location . ResourcePool ,
Datastore : s . Location . Datastore ,
GuestOS : s . Config . GuestOSType ,
NICs : networkCards ,
USBController : s . Config . USBController ,
Version : s . Config . Version ,
2018-01-24 09:56:14 -05:00
} )
if err != nil {
2018-02-01 06:47:09 -05:00
state . Put ( "error" , fmt . Errorf ( "error creating vm: %v" , err ) )
2018-01-24 09:56:14 -05:00
return multistep . ActionHalt
}
state . Put ( "vm" , vm )
2018-05-06 17:26:04 -04:00
2018-01-24 09:56:14 -05:00
return multistep . ActionContinue
}
func ( s * StepCreateVM ) Cleanup ( state multistep . StateBag ) {
2020-10-27 09:07:08 -04:00
common . CleanupVM ( state )
2018-01-24 09:56:14 -05:00
}