Getting started with Ansible Molecule
Intro
This article is intended as a guide to set up Ansible Molecule for testing Ansible roles by running them against virtual machines. These virtual machines will be controlled by Vagrant using VirtualBox as provider. The code in this guide was developed and tested on AlmaLinux9 and Ubuntu22.04 for the software versions mentioned in Requirements.
All static files used throughout this guide can be found here.
Requirements
System
Since we will use VirtualBox virtual Machines in this guide it's required for your system to have virtualization enabled in your mainboard's BIOS or UEFI. Check this article for further details
This guide is intended to be followed on a Linux system. This article assumes you got a basic understanding of Ansible and how to operate within the Linux terminal. To follow this guide on a Windows system you will need to use the Windows Subsystem for Linux (WSL) since Ansible is not supported on Windows. It does however support remote controlling Windows hosts.
Python
You will need python >= 3.10 to install the latest versions of all required python packages.
Additional the python-venv
and python-pip
packages will be required.
Here just the example install command for Ubuntu22.04
sudo apt-get install python3.12 python3.12-venv python3-pip
Creating a python virtual environment for Ansible first is highly recommended.
Create virtual environment | |
---|---|
1 2 |
|
Next we need a bunch of python packages like Ansible, Molecule and its Vagrant plugin.
Create a project directory and cd
into it.
At the time of writing there seems to be a bug with the latest version of molecule-plugins when used with the latest version of molecule (v25.3.1).
That's why we go for the versions listed here since they seem to work fine together.
Create a requirements.txt
file containing these lines:
requirements.txt | |
---|---|
1 2 3 4 |
|
docker python packages
In case you are using a version < v23.6.0 you might need to install the docker python package due to a bug #32540 in molecule plugins.
Now you can run upgrade pip
and install the requirements.
Install requirements | |
---|---|
1 2 |
|
Tools
As the title suggests you also need Virtualbox and Vagrant installed to follow along.
Vagrant is a virtual machine management tool which allows molecule to create, start and remove virtual machines in an automated way. VirtualBox on the other hand is the virtualization provider and handles all the heavy lifting when it comes to virtualizing your hardware. See the following table for download pages and version used for the following examples.
Tool | Download Page | Version used here |
---|---|---|
Virtualbox | Installers | 7.1.6 |
Vagrant | Install commands | 2.4.3 |
Install vagrant on debian-based systems | |
---|---|
1 2 3 4 |
|
The provided key file command for RPM-based Linux distributions on the VirtualBox website didn't work for me - So I changed it to the one below to make it work.
Install on Debian-based systems | |
---|---|
1 2 3 4 |
|
Install on RedHat-based systems | |
---|---|
1 2 3 4 |
|
Verify the successful installation of both tools by checking their version.
1 2 |
|
Prepare development environment
Initially I came across many guides mentioning the command molecule role init
.
This one doesn't exist anymore since version 6.0.0 - it was removed intentional to get rid of the Ansible-Galaxy dependency.
By now you simply use the ansible-galaxy role init
command to initialize an Ansible role and initialize a molecule scenario from within the role afterwards.
Setup role and molecule scenario | |
---|---|
1 2 3 |
|
For now, we'll just go with the default scenario to keep it simple. Now you got a "molecule" directory inside the role containing a bunch of default .yml files.
📦sample_role
┣ 📂defaults
┃ ┗ 📜main.yml
┣ 📂files
┣ 📂handlers
┃ ┗ 📜main.yml
┣ 📂meta
┃ ┗ 📜main.yml
┣ 📂molecule
┃ ┗ 📂default
┃ ┣ 📜converge.yml
┃ ┣ 📜create.yml
┃ ┣ 📜destroy.yml
┃ ┗ 📜molecule.yml
┣ 📂tasks
┃ ┗ 📜main.yml
┣ 📂templates
┣ 📂tests
┃ ┣ 📜inventory
┃ ┗ 📜test.yml
┣ 📂vars
┃ ┗ 📜main.yml
┗ 📜README.md
For details about how each file and directory inside this role structure is supposed to be used see the Ansible documentation
Instance Creation
Default Instance
Creating a molecule instance is done by running molecule create
if you do that right away from the roles root directory you will most likely encounter the following error:
ERROR Computed fully qualified role name of sample does not follow current galaxy requirements.
Please edit meta/main.yml and assure we can correctly determine full role name:
galaxy_info:
role_name: my_name # if absent directory name hosting role is used instead
namespace: my_galaxy_namespace # if absent, author is used instead
This happens due to molecule running a role name-check by default.
As stated in the documentation you can either disable the check or just add the role_name
and namespace
to the meta/main.yml
file.
Running molecule create
after adding these should, while throwing a bunch of warnings, already work.
Running molecule list
should now show a table similar to this one.
Instance Name | Driver Name | Provisioner Name | Scenario Name | Created | Converged |
---|---|---|---|---|---|
instance | default | ansible | default | true | false |
This will create a default instance using the delegated driver, which is just called "default".
As the title suggests we will use Vagrant as driver with VirtualBox as a provider in this example.
So run molecule destroy
to remove that default instance again.
If you run molecule drivers
you should see a list of installed drivers including vagrant
.
In case vagrant is missing, please check again if you installed all requirements, including the Vagrant plugin.
Take a look at the molecule-plugins repository for additional information
Cleaning up
Molecule stores all instance-related data in a so called ephermal directory and removes it when running molecule reset
.
It's placed at ~/.cache/molecule/<role-name>/<scenario-name>
by default.
Running molecule reset
might result in a python-traceback in some versions of molecule-plugins before version v23.6.0, which is related to docker on RHEL-systems but will still work and remove the directory as expected.
Python traceback explanation for molecule reset
Indicates docker and or the python module isn't installed on your system, see #166 Happens e.g. on Almalinux 9 due to podman being the default container service instead of docker and molecule doesn't seem to like this.
Vagrant Instance
molecule.yml | |
---|---|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 |
|
You can find some explanation of all these settings in the Ansible molecule docs
VirtualBox Network Setup
Assigning a network-interface using a 192.168.56.X
address is crucial here.
VirtualBox sets up two virtual networks by default.
- vboxnet0 - which is Host-only using 192.168.56.1
- NatNetwork - using 10.0.2.X
NatNetwork will be used by default but requires port forwarding from the host to the VM to make it accessible from e.g. a browser on the host To get around this we just assign a static address from the host-only network.
Initialize vagrant scenario | |
---|---|
1 2 3 4 5 6 |
|
Here we begin by initializing a new molecule scenario using flags to explicitly set the driver and provisioner names.
Next we copy the default create.yml
and destroy.yml
files since the default ones will cause connection issues on start-up.
Now replace the molecule config file molecule.yml
as well as the converge.yml
and verify.yml
with the provided ones which use AlmaLinux9. - Get other vagrant boxes on vagrant cloud
Running molecule create
and molecule list
when it's done should now display a vagrant instance.
Access Vagrant Instance
Accessing an instance is supposed to be done by running molecule login --host <hostname>
, this might not work correctly for molecule versions before v23.6.0 due to a bug.
If you are using a version >= v23.6.0 this should work right away.
If you encounter this issue you can run vagrant global-status
to get the vagrant instance IDs and vagrant ssh <id>
to log into one of the VMs displayed. Afterwards just type exit
to drop out of the instance again.
Provision a service
After setting up this vagrant instance successfully it is now time to make it do something using Ansible as its provisioner. We will use these tasks so set up an Apache web-server. This is just a very basic example for demonstration.
tasks.yml | |
---|---|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 |
|
Now replace the content of tasks/main.yml
with these yaml tasks.
Next run molecule converge
to run these tasks against the VirtualBox VM.
After this ran successfully you should be able to just copy the IP address displayed by the debug task e.g. 192.168.56.10
to your browser and see the default Apache web-server page right away.
Even tho this is nice, testing the functionality of this web-server manually isn't quite a scalable approach. It's time to set up automated testing for this role.
Test Vagrant Instance
We will use Ansible for testing as well to stay with the default and to keep it simple. Another popular option for molecule testing is testinfra Take a look now at these test tasks which should be self-explanatory due to their names.
tests.yml | |
---|---|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 |
|
Place these tasks into a file called tests.yml
in the tasks directory to make them easily accessible.
Now you should be able to run molecule verify
to have these tests run against the virtual machine.
Wrap-Up
Well done, at this point you should have a basic setup to implement an Ansible role and test it in an automated and easy to use way against VirtualBox virtual machines. This kind of setup is also quite extensible with additional logic and convenience features as I'll show you in the following articles of this series.
Thanks for reading and stay tuned.