If you create Virtual machines often and manually install the same tools repeatedly, this post is for you. Using ansible, I have written a simple playbook to create the virtual machines within seconds. The playbook would automatically install any desired software you want. The playbook would also run any command you would like to execute just after the Virtual machine creation, like user creation.
The great thing is all this is in one command. You need ansible and a Linux host to start.
Prerequisite :
- Your host machine must be a Linux machine. For example, Ubuntu. This procedure would not work for windows host machines.
- You have ansible installed on your host machine.
- Any other dependencies would be auto-installed by the playbook, so end user does not need to bother too much.
Procedure:
Step-1: Cloning the git repo with the playbook
git clone https://github.com/technekey/libvert-ansible-vm-create.git
Step-2: Install ansible dependencies
ansible-galaxy install -r requirments.yml --force
Step-3: Setup the SSH Keys and customization via the host_vars/localhost/defaults.yml
You can do the sizing of the VM, like:
- Number of cpu, , Eg 2
- RAM size, Eg, 4G
- Disk size, Eg, 120G
- user name, Eg, technekey
- password, Eg, technekey
- commands you would like to execute after VM creation. Eg, touch /foo/bar
- Packages you would like to install after VM creation. E.g., vim, openssh-server
- SSH keys, Eg, RSA, DSA, EDSA public key for passwordless access.
- Image location(this could be HTTPS URL or a local path to the img file)
- OS variant name
I strongly recommend setting up an SSH public key in this file for passwordless SSH to the VM.
#######################################################
# This file contains, some of the default values
# user should change as per their need
#######################################################
######################################################################################
# The default action of the playbook is to create the VM, if you want to delete vm
# set this playbook_action=delete via extra var
######################################################################################
playbook_action: create
######################################################################################
# STRING: location at which ISO will be downloaded if URL is provided as image_source
#######################################################################################
download_location: ~/vm_images
#######################################################################################
# STRING: location at which VM Virtual disk will be kept, location should have enough
# space and any loss to the files in this dir, will cause VM corruption.
########################################################################################
vm_disk_location: ~/vm_disks
########################################################################################
# STRING: default user and pass for the VM
########################################################################################
allow_ssh_pass: True # If this is set to false, VM will not be accessible via user/pass
default_username: technekey
default_password: technekey
########################################################################################
# default sizing of VM, either change here or using extra-vars
# INT: vcpus and memory_mb
# STRING: disk_size, Must use suffix of unit. Eg: 120G
########################################################################################
memory_mb: 2048
vcpus: 2
disk_size: 40G
########################################################################################
# STRING: Image location, could be a URL(https://.......) or a local path(/foo/bar/.....)
########################################################################################
image_source: "https://cloud-images.ubuntu.com/jammy/current/jammy-server-cloudimg-amd64.img"
#########################################################################################
# STRING: value can be obtained by 'virt-install --osinfo list'
#
#########################################################################################
os_variant: ubuntufocal
###########################################################################################
# STRING: SSH public key to login to the VM
###########################################################################################
# FILL THIS BEFORE USING THE PLAYBOOK
ssh_key: ""
###########################################################################################
# LIST: Tools you would want to install automatically after VM creation
###########################################################################################
package_list:
- vim
- build-essential
- python3-pip
##############################################################################################
# List: Boot commands, the commands to be executed just after boot before anything else
#
##############################################################################################
boot_commands:
- date
- touch /tmp/start
##############################################################################################
# LIST: Run commands, the commands to be executed just before the cloud-init finishes
#
##############################################################################################
run_commands:
- date
- touch /tmp/end
step-4: Ready to create VM
The playbook expects, two mandatory extra-vars, 1st is the name of the VM(vm_name) and the action that we would want to perform(to create playbook_action=create
, similarly for VM deletion playbook_action=delete
)
ansible-playbook vm_manage.yml -e vm_name='demo-vm' -e playbook_action=create
Example of the execution:
ansible-playbook vm_manage.yml -e vm_name='demo-vm' -e playbook_action=create
[WARNING]: No inventory was parsed, only implicit localhost is available
[WARNING]: provided hosts list is empty, only localhost is available. Note that the implicit localhost does not match 'all'
PLAY [Deploy VM using Cloud-init] *********************************************************************************************************************************
TASK [Gathering Facts] ********************************************************************************************************************************************
[Tue Dec 27 15:40:36 2022]
ok: [localhost]
TASK [Starting validation of mandatory variables and dependencies] ************************************************************************************************
[Tue Dec 27 15:40:37 2022]
TASK [input_validation_and_dependency_installation : Display warnings] ********************************************************************************************
[Tue Dec 27 15:40:37 2022]
[WARNING]: Consider checking the host_vars/localhost/defaults.yml file to customize the VM size and login options
ok: [localhost] => {}
TASK [input_validation_and_dependency_installation : Validate the minimum required hostvars configuration] ********************************************************
[Tue Dec 27 15:40:37 2022]
skipping: [localhost] => (item=vm_name)
skipping: [localhost] => (item=download_location)
skipping: [localhost] => (item=vm_disk_location)
skipping: [localhost] => (item=allow_ssh_pass)
skipping: [localhost] => (item=default_username)
skipping: [localhost] => (item=default_password)
skipping: [localhost] => (item=memory_mb)
skipping: [localhost] => (item=vcpus)
skipping: [localhost] => (item=disk_size)
skipping: [localhost] => (item=image_source)
skipping: [localhost] => (item=os_variant)
skipping: [localhost] => (item=ssh_key)
skipping: [localhost] => (item=required_directories)
TASK [input_validation_and_dependency_installation : Create the required directory] *******************************************************************************
[Tue Dec 27 15:40:37 2022]
ok: [localhost] => (item=/home/technekey/vm_images)
ok: [localhost] => (item=/home/technekey/vm_disks)
changed: [localhost] => (item=/home/technekey/vm_disks/demo-vm)
TASK [input_validation_and_dependency_installation : Install the dependencies on the host] ************************************************************************
[Tue Dec 27 15:40:37 2022]
ok: [localhost] => (item=python3-libvirt)
ok: [localhost] => (item=libvirt-clients)
ok: [localhost] => (item=virtinst)
ok: [localhost] => (item=guestfs-tools)
ok: [localhost] => (item=qemu-utils)
ok: [localhost] => (item=qemu-kvm)
ok: [localhost] => (item=cloud-image-utils)
TASK [input_validation_and_dependency_installation : Loading the path of all required binaries in the ansible host] ***********************************************
[Tue Dec 27 15:40:42 2022]
changed: [localhost] => (item=virt-install)
changed: [localhost] => (item=virsh)
changed: [localhost] => (item=virt-ls)
changed: [localhost] => (item=virt-cat)
changed: [localhost] => (item=qemu-img)
changed: [localhost] => (item=cloud-localds)
TASK [input_validation_and_dependency_installation : List the Existing KVM present on the host machine] ***********************************************************
[Tue Dec 27 15:40:42 2022]
ok: [localhost]
TASK [input_validation_and_dependency_installation : Check if the user requested VM is already present in the Host] ***********************************************
[Tue Dec 27 15:40:43 2022]
ok: [localhost]
TASK [input_validation_and_dependency_installation : Display the presence of the vm(demo-vm)in the Host] **********************************************************
[Tue Dec 27 15:40:43 2022]
ok: [localhost] => {}
MSG:
" demo-vm is not present in the host. Continuing the VM creation.
"
TASK [input_validation_and_dependency_installation : meta] ********************************************************************************************************
[Tue Dec 27 15:40:43 2022]
skipping: [localhost]
TASK [input_validation_and_dependency_installation : Determine the valid list of OS-Variants] *********************************************************************
[Tue Dec 27 15:40:43 2022]
changed: [localhost]
TASK [input_validation_and_dependency_installation : assert the os-variant] ***************************************************************************************
[Tue Dec 27 15:40:43 2022]
ok: [localhost] => {
"changed": false
}
MSG:
OS Variant assertion passed
TASK [Create a VM] ************************************************************************************************************************************************
[Tue Dec 27 15:40:43 2022]
TASK [Download Image and resize the image] ************************************************************************************************************************
[Tue Dec 27 15:40:43 2022]
TASK [image-download : Download the image from remote location] ***************************************************************************************************
[Tue Dec 27 15:40:43 2022]
ok: [localhost]
TASK [image-download : Set Image file name] ***********************************************************************************************************************
[Tue Dec 27 15:40:44 2022]
ok: [localhost]
TASK [image-download : Get the format of the downloaded image] ****************************************************************************************************
[Tue Dec 27 15:40:44 2022]
changed: [localhost]
TASK [image-download : Get the format of the downloaded image] ****************************************************************************************************
[Tue Dec 27 15:40:44 2022]
ok: [localhost]
TASK [image-download : Set the qcow2 file name] *******************************************************************************************************************
[Tue Dec 27 15:40:44 2022]
ok: [localhost]
TASK [image-download : Convert the image to qcow2] ****************************************************************************************************************
[Tue Dec 27 15:40:44 2022]
changed: [localhost]
TASK [image-download : Resize the disk /home/technekey/vm_images/jammy-server-cloudimg-amd64.img] ************************************************************************
[Tue Dec 27 15:40:46 2022]
changed: [localhost]
TASK [Starting cloud-config image buildup] ************************************************************************************************************************
[Tue Dec 27 15:40:47 2022]
TASK [cloud-config-creation : Create a temp cloud-init file] ******************************************************************************************************
[Tue Dec 27 15:40:47 2022]
changed: [localhost]
TASK [cloud-config-creation : Create a template to add the VM-NAME(demo-vm) to the temp cloud-init file] **********************************************************
[Tue Dec 27 15:40:47 2022]
changed: [localhost]
TASK [cloud-config-creation : Set cloud-config file name] *********************************************************************************************************
[Tue Dec 27 15:40:47 2022]
ok: [localhost]
TASK [cloud-config-creation : Create a seed image] ****************************************************************************************************************
[Tue Dec 27 15:40:47 2022]
changed: [localhost]
TASK [cloud-config-creation : Cleanup the Default cloud-init file] ************************************************************************************************
[Tue Dec 27 15:40:47 2022]
changed: [localhost]
TASK [Starting the virt-install] **********************************************************************************************************************************
[Tue Dec 27 15:40:47 2022]
TASK [virt-install : Do the install for Default network] **********************************************************************************************************
[Tue Dec 27 15:40:48 2022]
changed: [localhost]
TASK [Check for cloud-init completion for demo-vm] ****************************************************************************************************************
[Tue Dec 27 15:40:49 2022]
TASK [cloud_init_check : Check the VM status] *********************************************************************************************************************
[Tue Dec 27 15:40:49 2022]
changed: [localhost]
TASK [cloud_init_check : Wait for cloud-init result file to get created] ******************************************************************************************
[Tue Dec 27 15:40:50 2022]
FAILED - RETRYING: [localhost]: Wait for cloud-init result file to get created (90 retries left).
FAILED - RETRYING: [localhost]: Wait for cloud-init result file to get created (89 retries left).
FAILED - RETRYING: [localhost]: Wait for cloud-init result file to get created (88 retries left).
FAILED - RETRYING: [localhost]: Wait for cloud-init result file to get created (87 retries left).
FAILED - RETRYING: [localhost]: Wait for cloud-init result file to get created (86 retries left).
FAILED - RETRYING: [localhost]: Wait for cloud-init result file to get created (85 retries left).
changed: [localhost]
TASK [cloud_init_check : Waiting for Cloud init to complete] ******************************************************************************************************
[Tue Dec 27 15:42:05 2022]
changed: [localhost]
TASK [cloud_init_check : Validate the cloud-init execution results] ***********************************************************************************************
[Tue Dec 27 15:42:07 2022]
ok: [localhost] => {
"changed": false
}
MSG:
Cloud-init exeution passed
TASK [cloud_init_check : Capture the Current Status of the VM] ****************************************************************************************************
[Tue Dec 27 15:42:08 2022]
ok: [localhost]
TASK [cloud_init_check : Grab the IP address of the VM Created] ***************************************************************************************************
[Tue Dec 27 15:42:08 2022]
changed: [localhost]
TASK [cloud_init_check : Assert that IP address is available for the vm(demo-vm)] *********************************************************************************
[Tue Dec 27 15:42:08 2022]
ok: [localhost] => {
"changed": false
}
MSG:
IP Address is available for the VM=demo-vm
TASK [cloud_init_check : Remove the old host entry] ***************************************************************************************************************
[Tue Dec 27 15:42:08 2022]
ok: [localhost]
TASK [cloud_init_check : Add IP address of all hosts to all hosts] ************************************************************************************************
[Tue Dec 27 15:42:08 2022]
changed: [localhost]
TASK [cloud_init_check : Remove the old host entry] ***************************************************************************************************************
[Tue Dec 27 15:42:08 2022]
changed: [localhost]
TASK [cloud_init_check : Add IP address of all hosts to all hosts] ************************************************************************************************
[Tue Dec 27 15:42:08 2022]
changed: [localhost]
TASK [cloud_init_check : Show VM info] ****************************************************************************************************************************
[Tue Dec 27 15:42:09 2022]
ok: [localhost] => {}
MSG:
[{'state': 'running', 'maxMem': '2097152', 'memory': '2097152', 'nrVirtCpu': 2, 'cpuTime': '31340000000', 'autostart': 0}, '192.168.122.20']
TASK [Delete the VM (demo-vm)] ************************************************************************************************************************************
[Tue Dec 27 15:42:09 2022]
skipping: [localhost]
PLAY RECAP ********************************************************************************************************************************************************
localhost : ok=35 changed=18 unreachable=0 failed=0 skipped=2 rescued=0 ignored=0
[localhost]
15:40:36 Gathering Facts Passed 1.06s
15:40:37 Starting validation of mandatory variables and dependencies Passed 0.00s
15:40:37 input_validation_and_dependency_installation : Display warnings Passed 0.02s
15:40:37 input_validation_and_dependency_installation : Validate the minimum re... Skipped 0.04s
15:40:37 input_validation_and_dependency_installation : Create the required dir... Passed 0.43s
15:40:37 input_validation_and_dependency_installation : Install the dependencie... Passed 4.20s
15:40:42 input_validation_and_dependency_installation : Loading the path of all... Passed 0.82s
15:40:42 input_validation_and_dependency_installation : List the Existing KVM p... Passed 0.20s
15:40:43 input_validation_and_dependency_installation : Check if the user reque... Passed 0.01s
15:40:43 input_validation_and_dependency_installation : Display the presence of... Passed 0.01s
15:40:43 input_validation_and_dependency_installation : meta Skipped 0.02s
15:40:43 input_validation_and_dependency_installation : Determine the valid lis... Passed 0.54s
15:40:43 input_validation_and_dependency_installation : assert the os-variant Passed 0.01s
15:40:43 Create a VM Passed 0.01s
15:40:43 Download Image and resize the image Passed 0.00s
15:40:43 image-download : Download the image from remote location Passed 0.60s
15:40:44 image-download : Set Image file name Passed 0.01s
15:40:44 image-download : Get the format of the downloaded image Passed 0.13s
15:40:44 image-download : Get the format of the downloaded image Passed 0.01s
15:40:44 image-download : Set the qcow2 file name Passed 0.01s
15:40:44 image-download : Convert the image to qcow2 Passed 1.57s
15:40:46 image-download : Resize the disk /home/technekey/vm_images/jammy-server-cloud... Passed 0.71s
15:40:47 Starting cloud-config image buildup Passed 0.00s
15:40:47 cloud-config-creation : Create a temp cloud-init file Passed 0.20s
15:40:47 cloud-config-creation : Create a template to add the VM-NAME(demo-vm) ... Passed 0.43s
15:40:47 cloud-config-creation : Set cloud-config file name Passed 0.01s
15:40:47 cloud-config-creation : Create a seed image Passed 0.13s
15:40:47 cloud-config-creation : Cleanup the Default cloud-init file Passed 0.13s
15:40:47 Starting the virt-install Passed 0.00s
15:40:48 virt-install : Do the install for Default network Passed 1.85s
15:40:49 Check for cloud-init completion for demo-vm Passed 0.01s
15:40:49 cloud_init_check : Check the VM status Passed 0.17s
15:40:50 cloud_init_check : Wait for cloud-init result file to get created Passed 75.28s
15:42:05 cloud_init_check : Waiting for Cloud init to complete Passed 2.56s
15:42:07 cloud_init_check : Validate the cloud-init execution results Passed 0.05s
15:42:08 cloud_init_check : Capture the Current Status of the VM Passed 0.15s
15:42:08 cloud_init_check : Grab the IP address of the VM Created Passed 0.16s
15:42:08 cloud_init_check : Assert that IP address is available for the vm(demo... Passed 0.01s
15:42:08 cloud_init_check : Remove the old host entry Passed 0.21s
15:42:08 cloud_init_check : Add IP address of all hosts to all hosts Passed 0.12s
15:42:08 cloud_init_check : Remove the old host entry Passed 0.12s
15:42:08 cloud_init_check : Add IP address of all hosts to all hosts Passed 0.13s
15:42:09 cloud_init_check : Show VM info Passed 0.01s
15:42:09 Delete the VM (demo-vm) Skipped 0.01s
Total Playbook execution time: 92.82563495635986 sec
[localhost]
Step-5: Validate VM is UP
virsh list
Cleanup/VM deletion:
ansible-playbook vm_manage.yml -e vm_name='demo-vm' -e playbook_action=delete
Summary:
I have tested this playbook on the Ubuntu host machine with a variety of cloud images of different distros.
I hope this playbook would help you save time. I would really appreciate if you leave a comment if this page help you. Also, if you have any questions, please leave them here.