Home > esxi_macos_vm_creation, Mac OS X, macOS, Scripting, VMware, VMware ESXi > Building VMs on ESXi using esxi_macos_vm_creation.sh

Building VMs on ESXi using esxi_macos_vm_creation.sh

As part of my testing workflow, I’ve been using VMs running on a ESXi server running ESXi 6.5. To help me quickly build those VMs, I have been using a script named esxi_macos_vm_creation.sh for building VMs. This script is forked from Tamas Piros’s auto-create script for standing up Linux VMs on free ESXi:

https://github.com/tpiros/auto-create

My fork of the auto-create script is designed to create and configure virtual machines with Apple operating systems as the guest OS, hosted on a VMware ESXi server running on Apple hardware. The script assumes that the virtual machines are built using copied VMDK disk files, where the VMDK files are generated by AutoDMG and vfuse. For more details, see below the jump.

Building VMDK files

In the process I’m following, the workflow to generate the VMDK files works like this:

1. Create disk image using AutoDMG. The disk image includes a firstboot package, which includes tools to configure the VM on its first startup.

Screen Shot 2017 04 09 at 11 21 58 AM

2. Once the disk image is generated, use vfuse to create VMDK files using a command similar to the one shown below:

Screen Shot 2017 04 09 at 12 08 22 PM

3. vfuse will create a VM for VMware Fusion. To access the VMDK files, right-click on the VM and select Show Package Contents.

Screen Shot 2017 04 09 at 12 10 26 PM

 

4. The VMDK files will be inside. The conversion process will create two files:

  • filename.vmdk
  • filename-flat.vmdk
Screen Shot 2017 04 09 at 12 10 54 PM


Screen Shot 2017 04 09 at 12 10 40 PM

This is expected behavior and both of these files will be needed.

Uploading VMDK files

Once the VMDK files are created and available, they need to be uploaded to a convenient location on an ESXi datastore. For information on how to upload these files, see the link below from the VMware documentation for vSphere:

Upload Files to Datastores:
https://pubs.vmware.com/vsphere-65/index.jsp?topic=%2Fcom.vmware.vsphere.storage.doc%2FGUID-58D77EA5-50D9-4A8E-A15A-D7B3ABA11B87.html

The esxi_macos_vm_creation script will need to be stored on the ESXi server, so also upload it to a convenient location on an ESXi datastore.

 

Running the script

This script is designed to create and configure virtual machines running Apple operating systems, hosted on a VMware ESXi server running on Apple hardware. The script assumes that the virtual machines are built using copied VMDK disk files. In this case, the VMDK files will be the disk images created by AutoDMG and converted to VMDK files using vfuse.

Note: When creating the VM, you will need to provide the location of the filename.vmdk file . As long as filename.vmdk and filename-flat.vmdk are both available in the same directory on the datastore, ESXi will automatically also reference and include filename-flat.vmdk when creating the VM.

The script is designed to be stored on an ESXi datastore and run from the ESXi server’s command line interface, which is why it needed to be uploaded to an ESXi datastore in the previous step.

Usage:

/path/to/esxi_macos_vm_creation.sh -n -d -c -h -i -o -r -s -v -p

Options:

  • -n: Name of VM (required)
  • -d: Location of a VMDK disk file (required). Location must be in this format – /vmfs/volumes/datastore_number_here/path/to/filename_here.vmdk
  • -c: Number of virtual CPUs
  • -h: VMware Hardware Version
  • -i: Location of an ISO image. Location must be in this format – /vmfs/volumes/datastore_number_here/path/to/iso_file.iso
  • -o: Apple OS version
  • -r: RAM size in MB
  • -s: Disk size in GB
  • -v: VNC port between 5900 and 5909
  • -p: VNC password. Maximum password length is eight characters.

The script has several default variables set up:

ESXi datastore location for new VMs: /vmfs/volumes/datastore1
Number of processors for new VMs: 2
Amount of RAM for new VMs: 4096 MBs
Hard drive size for new VMs: 40 GB
Guest OS: darwin14-64 (this is the OS setting for OS X Yosemite.)
Hardware Version: 11 (this is the maximum Hardware Version available for ESXi 6.0.x)

The ESXi datastore value can be changed by editing the DATASTORE variable in the script. The other default values are overridden if alternate values are enabled by the script’s available options.

Examples

To set up a VM specifying only the VM name and VMDK location:

/path/to/esxi_macos_vm_creation.sh -n VM_Name_Goes_Here -d /vmfs/volumes/datastore_number_here/path/to/filename_here.vmdk

The output should appear similar to that shown below:

computername:~ username$ ssh root@esxi.demo.com
Password: 
The time and date of this login have been sent to the system logs.

VMware offers supported, powerful system administration tools.  Please
see www.vmware.com/go/sysadmintools for details.

The ESXi Shell can be disabled by an administrative user. See the
vSphere Security documentation for more information.
[root@esxi:~] /path/to/esxi_macos_vm_creation.sh -n macOSVM -d /vmfs/volumes/datastore1/template/filename.vmdk
Destination disk format: VMFS thin-provisioned
Cloning disk '/vmfs/volumes/datastore1/template/filename.vmdk'...
Clone: 100% done.
Powering on VM:
The Virtual Machine is now configured and the VM has been started up. The VM is set to use the following configuration:
Name: macOSVM
CPU: 2
RAM: 4096
Guest OS: darwin14-64
Hardware Version: 11
Hard drive size: 40
No ISO added.
VNC not enabled.
[root@esxi:~]

To set up a VM using a name with spaces and/or special characters, add quotation marks to the VM name:

/path/to/esxi_macos_vm_creation.sh -n "VM's Name Goes Here!" -d /vmfs/volumes/datastore_number_here/path/to/filename_here.vmdk

Other flags can be added as needed:

To set up a VM and add more CPUs:

/path/to/esxi_macos_vm_creation.sh -n VM_Name_Goes_Here -c 4 -d /vmfs/volumes/datastore_number_here/path/to/filename_here.vmdk

To set up a VM and enable VNC on port 5901 with the password set to the word password:

/path/to/esxi_macos_vm_creation.sh -n VM_Name_Goes_Here -d /vmfs/volumes/datastore_number_here/path/to/filename_here.vmdk -v 5901 -p password

To set up a VM named MacOS VM 10.12 using a VDMK stored on /vmfs/volumes/datastore1/template and named macos-vm.vmdk with 4 CPUs, 8 GBs of RAM, a 52 GB hard drive, set to HW Version 13, guest OS set to macOS Sierra and VNC enabled on port 5902 with the VNC password set to the word password:

/path/to/esxi_macos_vm_creation.sh -n "MacOS VM 10.12" -d /vmfs/volumes/datastore1/template/macos-vm.vmdk -c 4 -r 8192 -s 52 -h 13 -o darwin16-64 -v 5902 -p password

The output should appear similar to what’s shown below:

computername:~ username$ ssh root@esxi.demo.com
Password: 
The time and date of this login have been sent to the system logs.

VMware offers supported, powerful system administration tools.  Please
see www.vmware.com/go/sysadmintools for details.

The ESXi Shell can be disabled by an administrative user. See the
vSphere Security documentation for more information.
[root@esxi:~] /path/to/esxi_macos_vm_creation.sh -n "MacOS VM 10.12" -d /vmfs/volumes/datastore1/template/macos-vm.vmdk -c 4 -r 8192 -s 52 -h 13 -o darwin16-64 -v 5902 -p password
Destination disk format: VMFS thin-provisioned
Cloning disk '/vmfs/volumes/datastore1/template/macos-vm.vmdk'...
Clone: 100% done.
Grow: 100% done.
Powering on VM:
The Virtual Machine is now configured and the VM has been started up. The VM is set to use the following configuration:
Name: MacOS VM 10.12
CPU: 4
RAM: 8192
Guest OS: darwin16-64
Hardware Version: 13
Hard drive size: 52
No ISO added.
VNC Port: 5902
VNC Password: password
[root@esxi:~]

The script is available below. It is also available from GitHub at the following address:

https://github.com/rtrouton/esxi_macos_vm_creation


#!/bin/sh
# Parameters:
#
# Virtual machine name (required)
# CPU (number of cores)
# VMDK (location of VMDK file, required)
# RAM (memory size in MB)
# Hard Drive size (in GB)
# Hardware Version
# ISO (Location of ISO image, optional)
# Apple OS version
# VNC port
# VNC password
#
# Default values: CPU: 2, RAM: 4096, Hard drive size: 40 GB, Hardware Version: 11, ISO: 'blank', OS version: darwin14-64
#
# Usage:
# When using the script, command should look something like this:
#
# /vmfs/volumes/datastore_number_here/path/to/esxi_macos_vm_creation.sh -n VM_Name_Goes_Here -d /vmfs/volumes/datastore_number_here/path/to/filename_here.vmdk
#
# To set up a VM using a name with spaces and/or special characters, add quotation marks to the VM name:
#
# /vmfs/volumes/datastore_number_here/path/to/esxi_macos_vm_creation.sh -n "VM's Name Goes Here!" -d /vmfs/volumes/datastore_number_here/path/to/filename_here.vmdk
#
# Other flags can be added as needed:
#
# To set up a VM and add more CPUs:
#
# /vmfs/volumes/datastore_number_here/path/to/esxi_macos_vm_creation.sh -n VM_Name_Goes_Here -c 4 -d /vmfs/volumes/datastore_number_here/path/to/filename_here.vmdk
#
# To set up a VM and enable VNC on port 5901 with the password of "password":
#
# /vmfs/volumes/datastore_number_here/path/to/esxi_macos_vm_creation.sh -n VM_Name_Goes_Here -d /vmfs/volumes/datastore_number_here/path/to/filename_here.vmdk -v 5901 -p password
#
# To set up a VM named "MacOS VM 10.12" (no quotes) using a VDMK stored on /vmfs/volumes/datastore1/template and named "macos-vm.vmdk" (no quotes) with 4 CPUs
# 8 GBs of RAM, a 52 GB hard drive, VMware Hardware Version set to HW Version 13, guest OS set to macOS Sierra, and VNC enabled on port 5902 with the VNC password set to the word "password" (no quotes)
#
# /vmfs/volumes/datastore_number_here/path/to/esxi_macos_vm_creation.sh -n "MacOS VM 10.12" -d /vmfs/volumes/datastore1/template/macos-vm.vmdk -c 4 -r 8192 -s 52 -h 13 -o darwin16-64 -v 5902 -p password
phelp() {
echo ""
echo "Usage: ./esxi_macos_vm_creation.sh options: -n -d -c -h -i -o -r -s -v -p"
echo ""
echo "n: Name of VM (required). If using a name with spaces and/or special characters, add quotation marks to the VM name."
echo "c: Number of virtual CPUs. Default number is two."
echo "d: Location of a VMDK disk file (required). Location must be in this format – /vmfs/volumes/datastore_number_here/path/to/vmdk_file.vmdk"
echo "h: VMware hardware version. ESXi 5.5 supports up to HW 10, ESXi 6.x supports up to HW 11 and ESXi 6.5 supports up to HW 13."
echo "i: Location of an ISO image. Location must be in this format – /vmfs/volumes/datastore_number_here/path/to/iso_file.iso"
echo "o: Mac operating system version. Default is set to darwin14, which reports the guest OS as OS X Yosemite."
echo ""
echo "VMware guest OS values for the following versions of OS X and macOS:"
echo ""
echo "Mac OS 10.7 – darwin11-64"
echo "OS 10.8 – darwin12-64"
echo "OS 10.9 – darwin13-64"
echo "OS 10.10 – darwin14-64"
echo "OS 10.11 – darwin15-64"
echo "macOS 10.12 – darwin16-64"
echo ""
echo "r: RAM size in MB. Default number is 4096, for 4 GBs of memory."
echo "s: Disk size in GB. Default size is 40 GB. You can specify if you want it to be larger."
echo "v: VNC port is between 5900 and 5909 (required if also using the -p option)"
echo "p: VNC password (required if also using the -v option). Maximum password length is eight characters."
echo ""
echo "Default script values: ESXi datastore location: /vmfs/volumes/datastore1, CPU: 2, RAM: 4096MB, Hard drive size: 40GB, Guest OS: darwin14-64, Hardware Version: 11"
echo ""
}
# Set datastore location
DATASTORE="/vmfs/volumes/datastore1"
# Default variables. These values are overriden if alternate values
# are enabled by the script's available options.
CPU=2
RAM=4096
SIZE=40
ISO=""
OSVERS="darwin14-64"
HWVERS="11"
FLAG=true
ERR=false
# Error checking:
#
# The NAME has to be entered (i.e. the $NAME variable cannot be blank.)
# The CPU has to be an integer and it has to be between 1 and 32. Modify the if statement if you want to give more than 32 cores to your Virtual Machine, and also email me pls 🙂
# The VMDK has to have its location provided and we are checking for an actual .vmdk extension
# RAM has to be an integer and has to be greater than 0.
# The hard drive size has to be an integer and has to be greater than 0.
# The hardware version has to be an integer and has to be greater than 0
# If the ISO parameter is added, we are checking for an actual .iso extension
# If the VNC port parameter is used, it has to be an integer and has to be between 5900 and 5909.
# Note: If needed, this port limitation can be changed by editing the script.
# If the VNC password parameter is used, a password must be entered.
while getopts n:c:d:h:i:o:r:s:v:p: option
do
case $option in
n)
NAME=${OPTARG};
FLAG=false;
if [ -z "${NAME}" ]; then
ERR=true
MSG="$MSG | Please make sure to enter a VM name."
fi
;;
c)
CPU=${OPTARG}
if [ `echo "${CPU}" | egrep "^-?[0-9]+$"` ]; then
if [ "${CPU}" -le "0" ] || [ "${CPU}" -ge "32" ]; then
ERR=true
MSG="$MSG | The number of cores has to be between 1 and 32."
fi
else
ERR=true
MSG="$MSG | The CPU core number has to be an integer."
fi
;;
d)
VMDK=${OPTARG}
FLAG=false;
if [ -z "${VMDK}" ]; then
ERR=true
MSG="$MSG | Please provide a path to a VMDK file."
fi
if [ ! `echo "${VMDK}" | egrep "^.*\.(vmdk)$"` ]; then
ERR=true
MSG="$MSG | The extension should be .vmdk"
fi
;;
h)
HWVERS=${OPTARG}
if [ `echo "${HWVERS}" | egrep "^-?[0-9]+$"` ]; then
if [ "${HWVERS}" -eq "${HWVERS}" ] && [ "${HWVERS}" -lt "10" ]; then
ERR=true
MSG="$MSG | Please assign a value of 10 or more for the hardware version. ESXi 5.5 supports up to HW 10, ESXi 6.0.x supports up to HW 11 and ESXi 6.5 supports up to HW 13."
fi
else
ERR=true
MSG="$MSG | The hardware version has to be an integer."
fi
;;
i)
ISO=${OPTARG}
if [ ! `echo "${ISO}" | egrep "^.*\.(iso)$"` ]; then
ERR=true
MSG="$MSG | The extension should be .iso"
fi
;;
o)
OSVERS=${OPTARG};
if [ -z "${OSVERS}" ]; then
ERR=true
MSG="$MSG | Please make sure to enter a valid OS version. darwin14-64 = OS X 10.10 Yosemite, darwin15-64 = OS X 10.11 El Capitan, and darwin16-64 = macOS 10.12 Sierra."
fi
;;
r)
RAM=${OPTARG}
if [ `echo "${RAM}" | egrep "^-?[0-9]+$"` ]; then
if [ "${RAM}" -le "0" ]; then
ERR=true
MSG="$MSG | Please assign more than 1MB memory to the VM."
fi
else
ERR=true
MSG="$MSG | The RAM size has to be an integer."
fi
;;
s)
SIZE=${OPTARG}
if [ `echo "${SIZE}" | egrep "^-?[0-9]+$"` ]; then
if [ "${SIZE}" -le "40" ]; then
ERR=true
MSG="$MSG | Please assign more than 40 GB for the hard drive size."
fi
else
ERR=true
MSG="$MSG | The hard drive size has to be an integer."
fi
;;
v)
VNCPORT=${OPTARG}
if [ `echo "${VNCPORT}" | egrep "^-?[0-9]+$"` ]; then
if [ "${VNCPORT}" -lt "5900" ] || [ "${VNCPORT}" -gt "5909" ]; then
ERR=true
MSG="$MSG | Please assign a port number for VNC between 5900 and 5909."
fi
else
ERR=true
MSG="$MSG | The VNC port has to be an integer."
fi
;;
p)
VNCPASS=${OPTARG}
if [ -z "${VNCPASS}" ]; then
ERR=true
MSG="$MSG | Please make sure to enter a VNC pasword. Maximum password length is eight characters."
fi
;;
\?) echo "Unknown option: -$OPTARG" >&2; phelp; exit 1;;
🙂 echo "Missing option argument for -$OPTARG" >&2; phelp; exit 1;;
*) echo "Unimplemented option: -$OPTARG" >&2; phelp; exit 1;;
esac
done
if [ -z "${NAME}" ]; then
echo ""
echo ">> PROBLEM << : Please specify the name of the machine using the -n option. Displaying script options below."
echo ""
phelp
exit 1
fi
if [ -z "${VMDK}" ]; then
echo ""
echo ">> PROBLEM << : Please specify the location of a VMDK disk file with the -d parameter. Location must be in this format – /vmfs/volumes/datastore_number_here/path/to/vmdk_file.vmdk. Displaying script options below."
echo ""
phelp
exit 1
fi
if [ "${VNCPORT}" != "" ] && [ -z "${VNCPASS}" ]; then
echo ""
echo ">> PROBLEM << : Please specify a password for VNC using the -p option. Displaying script options below."
echo ""
phelp
exit 1
elif [ "${VNCPASS}" != "" ] && [ -z "${VNCPORT}" ]; then
echo ""
echo ">> PROBLEM << : Please specify a port number for VNC using the -v option. Displaying script options below."
echo ""
phelp
exit 1
fi
if [ "${VNCPORT}" != "" ] && [ "${VNCPASS}" != "" ]; then
VNCSTATUS=TRUE
else
VNCSTATUS=FALSE
fi
if $ERR; then
echo $MSG
exit 1
fi
if [ -d "${DATASTORE}"/"${NAME}" ]; then
echo "Directory – ${NAME} already exists, can't recreate it."
exit
fi
#Creating the folder for the Virtual Machine
mkdir -p "${DATASTORE}"/"${NAME}"
#Creating VM disk from vmdk template using vmkfstools
# Link: https://kb.vmware.com/kb/1027876
vmkfstools -i "${VMDK}" "${DATASTORE}"/"${NAME}"/"${NAME}".vmdk -d thin
# If specified size is larger than 40 GBs, resize VMDK file using vmkfstools
# Link: https://kb.vmware.com/kb/1002019
if [ "${SIZE}" -gt "40" ]; then
vmkfstools -X "${SIZE}"g "${DATASTORE}"/"${NAME}"/"${NAME}".vmdk
fi
#Creating the config file
touch "${DATASTORE}"/"${NAME}"/"${NAME}".vmx
#writing information into the configuration file
if [ "${VNCSTATUS}" = "TRUE" ]; then
cat << EOF > "${DATASTORE}"/"${NAME}"/"${NAME}".vmx
config.version = "8"
virtualHW.version = "${HWVERS}"
vmci0.present = "TRUE"
displayName = "${NAME}"
floppy0.present = "FALSE"
numvcpus = "${CPU}"
scsi0.present = "TRUE"
scsi0.sharedBus = "none"
scsi0.virtualDev = "lsilogic"
sata0.present = "TRUE"
memsize = "${RAM}"
scsi0:0.present = "TRUE"
scsi0:0.fileName = "${NAME}.vmdk"
scsi0:0.deviceType = "scsi-hardDisk"
sata0:1.present = "TRUE"
sata0:1.fileName = "${ISO}"
sata0:1.deviceType = "cdrom-image"
pciBridge0.present = "TRUE"
pciBridge4.present = "TRUE"
pciBridge4.virtualDev = "pcieRootPort"
pciBridge4.functions = "8"
pciBridge5.present = "TRUE"
pciBridge5.virtualDev = "pcieRootPort"
pciBridge5.functions = "8"
pciBridge6.present = "TRUE"
pciBridge6.virtualDev = "pcieRootPort"
pciBridge6.functions = "8"
pciBridge7.present = "TRUE"
pciBridge7.virtualDev = "pcieRootPort"
pciBridge7.functions = "8"
ethernet0.virtualDev = "e1000e"
ethernet0.networkName = "VM Network"
ethernet0.addressType = "generated"
ethernet0.present = "TRUE"
usb.present = "TRUE"
guestOS = "${OSVERS}"
RemoteDisplay.vnc.enabled = "${VNCSTATUS}"
RemoteDisplay.vnc.port = "${VNCPORT}"
RemoteDisplay.vnc.key = "${VNCPASS}"
smc.present = "TRUE"
EOF
else
cat << EOF > "${DATASTORE}"/"${NAME}"/"${NAME}".vmx
config.version = "8"
virtualHW.version = "${HWVERS}"
vmci0.present = "TRUE"
displayName = "${NAME}"
floppy0.present = "FALSE"
numvcpus = "${CPU}"
scsi0.present = "TRUE"
scsi0.sharedBus = "none"
scsi0.virtualDev = "lsilogic"
sata0.present = "TRUE"
memsize = "${RAM}"
scsi0:0.present = "TRUE"
scsi0:0.fileName = "${NAME}.vmdk"
scsi0:0.deviceType = "scsi-hardDisk"
sata0:1.present = "TRUE"
sata0:1.fileName = "${ISO}"
sata0:1.deviceType = "cdrom-image"
pciBridge0.present = "TRUE"
pciBridge4.present = "TRUE"
pciBridge4.virtualDev = "pcieRootPort"
pciBridge4.functions = "8"
pciBridge5.present = "TRUE"
pciBridge5.virtualDev = "pcieRootPort"
pciBridge5.functions = "8"
pciBridge6.present = "TRUE"
pciBridge6.virtualDev = "pcieRootPort"
pciBridge6.functions = "8"
pciBridge7.present = "TRUE"
pciBridge7.virtualDev = "pcieRootPort"
pciBridge7.functions = "8"
ethernet0.virtualDev = "e1000e"
ethernet0.networkName = "VM Network"
ethernet0.addressType = "generated"
ethernet0.present = "TRUE"
usb.present = "TRUE"
guestOS = "${OSVERS}"
smc.present = "TRUE"
EOF
fi
#Adding Virtual Machine to VM register
MYVM=`vim-cmd solo/registervm "${DATASTORE}"/"${NAME}"/"${NAME}".vmx`
#Powering up virtual machine:
vim-cmd vmsvc/power.on "${MYVM}"
echo "The Virtual Machine is now configured and the VM has been started up. The VM is set to use the following configuration:"
echo "Name: ${NAME}"
echo "CPU: ${CPU}"
echo "RAM: ${RAM}"
echo "Guest OS: ${OSVERS}"
echo "Hardware Version: ${HWVERS}"
echo "Hard drive size: ${SIZE}"
if [ -n "${ISO}" ]; then
echo "ISO: ${ISO}"
else
echo "No ISO added."
fi
if [ "$VNCSTATUS" = "TRUE" ]; then
echo "VNC Port: ${VNCPORT}"
echo "VNC Password: ${VNCPASS}"
else
echo "VNC not enabled."
fi
exit

  1. magisternavis
    April 12, 2017 at 8:24 pm

    How did you manage (if…) to solve the problem that virtual macs represent themselves with a question mark to other macs in the network (because of unknown hardware on OS side) as well as get iCloud stuff work (same reason)?

  1. No trackbacks yet.

Leave a comment