Setting up a VM on TrueNAS Scale using cloud-init
How to use cloud-init under TrueNAS Scale to spin up preconfigured virtual machines.
1-Download a cloud-init image
https://cloud-images.ubuntu.com/releases/22.04/
data:image/s3,"s3://crabby-images/91dba/91dba2d363ad0c42445f0a8eae774b1490ed6ed8" alt=""
Download the .img version of the latest cloud image.
2-Prepare the VM disk
Create the Zvol for the VM
data:image/s3,"s3://crabby-images/360c0/360c027d22e524d11f09b316f3f6323e0928279a" alt=""
Or from the command line:
zfs create -V 20G vms/cloud-init-test
Copy the cloud image into the Zvol
qemu-img convert -O raw /downloads/ISOs/ubuntu-22.04-server-cloudimg-amd64.img /dev/zvol/vms/cloud-init-test
3-Create the cloud-init seed image
Use a different system (or the same TrueNAS system if the cloud-image-utils package is installed).
Create the user-data.yaml file
#cloud-config
chpasswd:
list: |
ubuntu:12345
expire: False
hostname: cloud-init-test
ssh_authorized_keys:
- ssh-rsa AAAAB3Nza...
- ssh-rsa AAAAB3Nza...
Create cloud-init seed image
cloud-localds --verbose --vendor-data cloud-init-test-seed.qcow2 user-data.yaml
Copy the cloud-init seed image to TrueNAS
scp cloud-init-test-seed.qcow2 root@truenas:/mnt/vms/
4-Create the VM
data:image/s3,"s3://crabby-images/68024/680249ed81a5429998026215ce58759e38a959c6" alt=""
data:image/s3,"s3://crabby-images/2f413/2f41301d18ac147aba6f57fcc9f8c4c3d5ce767d" alt=""
data:image/s3,"s3://crabby-images/7a1fc/7a1fcf48a20b0d8d9dd65f804c30536922d55976" alt=""
data:image/s3,"s3://crabby-images/2a6ca/2a6caf0af7b0b636d20ed61b7f03c54202568fd9" alt=""
data:image/s3,"s3://crabby-images/cbec1/cbec1736c17ced7544e011bf171ba3d3a4e8836a" alt=""
data:image/s3,"s3://crabby-images/e4151/e4151a63ba9a5fdf65707fabf233c7d161e5997a" alt=""
data:image/s3,"s3://crabby-images/3e75c/3e75cf44ea0ce15ef53f8dba615bb731f115b170" alt=""
Add cloud-init seed image as a CDROM
Set as order 1004 to avoid attempting to boot from the CDROM.
data:image/s3,"s3://crabby-images/838b2/838b211301a34c949bde0aed4b7f236d120e072e" alt=""
data:image/s3,"s3://crabby-images/24869/248696b86661b4a06d86d12e937430402f9b3c1d" alt=""
data:image/s3,"s3://crabby-images/3642d/3642d80f6143bc49bfb28c765252d1ad68df8d41" alt=""
The same process of creating the VM and attaching the devices using the command line:
#!/bin/sh
# Create the VM
RESULT=`midclt call vm.create '{"name": "cloud_init_test", "cpu_mode": "HOST-MODEL", "bootloader": "UEFI_CSM", "threads": 2, "memory": 1024}'`
VM_ID=`echo ${RESULT} | jq '.id'`
# Add the display
midclt call vm.device.create '{"vm": '${VM_ID}', "dtype": "DISPLAY", "order": 1002, "attributes": {"web": true, "type": "VNC", "bind": "0.0.0.0", "wait": false}}'
# Obtain a random MAC address
MAC_ADDRESS=`midclt call vm.random_mac`
# Add the NIC
midclt call vm.device.create '{"vm": '${VM_ID}', "dtype": "NIC", "order": 1003, "attributes": {"type": "VIRTIO", "nic_attach": "br0", "mac": "'${MAC_ADDRESS}'"}}'
# Add the disk
midclt call vm.device.create '{"vm": '${VM_ID}', "dtype": "DISK", "order": 1001, "attributes": {"path": "/dev/zvol/vms/cloud-init-test","type": "VIRTIO"}}'
# Add the CDROM
midclt call vm.device.create '{"vm": '${VM_ID}', "dtype": "CDROM", "order": 1004, "attributes": {"path":"/mnt/vms/cloud-init-test-seed.qcow2"}}'
5-Start VM
data:image/s3,"s3://crabby-images/63466/634663f6873f60bbdb8b943ec20c875d333f308a" alt=""
Or using the command line:
midclt call vm.start '{"vm": '${VM_ID}'}'