Configures and starts N LXC containers to use in the Travis CI environment for simpler testing of Ansible roles across different distributions.
So you want to test your Ansible roles on Travis CI, but you don't want to use Docker because it doesn't mimic a full OS? LXC is what you want to use. This role will hopefully abstract much of the boilerplate you might otherwise use.
To get started, a minimal .travis.yml
that thoroughly tests that your role is
valid, idempotent, and functional may look like this:
---
language: python
sudo: required
dist: bionic
install:
- pip install ansible
- ansible-galaxy install lae.travis-lxc,v0.9.0
- ansible-playbook tests/install.yml -i tests/inventory
before_script: cd tests/
script:
# Validates your deployment playbook's syntax, which should contain your role
- ansible-playbook -i inventory deploy.yml --syntax-check
# Runs the deployment playbook
- ansible-playbook -i inventory deploy.yml
# Runs the deployment playbook again, saving output to a file called play.log,
# then checks that there are no changed/failed tasks and fails if there are.
- 'ANSIBLE_STDOUT_CALLBACK=debug ANSIBLE_DISPLAY_SKIPPED_HOSTS=no ANSIBLE_DISPLAY_OK_HOSTS=no
unbuffer ansible-playbook -vvi inventory deploy.yml &>play.log; printf "Idempotence: ";
grep -A1 "PLAY RECAP" play.log | grep -qP "changed=0 .*failed=0 .*"
&& (echo "PASS"; exit 0) || (echo "FAIL"; cat play.log; exit 1)'
# Integration tests and what not
- ANSIBLE_STDOUT_CALLBACK=debug ansible-playbook -i inventory -v test.yml
You'll note that four files are referenced. You can decide how to define your build process, but the following is what typically serves most purposes:
- tests/install.yml: executes
lae.travis-lxc
and other pre-install steps - tests/deploy.yml: executes the role you're testing
- tests/test.yml: executes validation tests against your deployment
- tests/inventory: contains a list of LXC container hostnames
install.yml
may look like this:
---
- hosts: localhost
connection: local
roles:
- lae.travis-lxc
vars:
test_profiles:
- profile: debian-buster
- profile: ubuntu-focal
- profile: centos-7
- profile: alpine-v3.11
# Add any other setup tasks you might need but don't necessarily need in your role
- hosts: all
tasks: []
The first play brings up three containers of three different distributions. The
second play could be used to either run other roles or pre-installation tasks
that you would expect your role not to do (for example, install epel-release
or create a device node for FUSE (because LXC doesn't do that for you)).
deploy.yml
may look like this:
---
- hosts: all
become: true
any_errors_fatal: true
roles:
- ansible-role-strawberry-milk
vars:
number_of_cartons: 15
This is basically a rendition of what ansible-galaxy init
would spit out in
test.yml
. This would have everything you need to execute your role properly.
For more complex roles, it makes sense to split variables out into the
tests/group_vars
folder and configure your inventory appropriately.
test.yml
should contain your tests, if you wanted to run any:
---
- hosts: all
tasks:
- name: Ensure that the Strawberry Milk HTTP service is running
uri:
url: "http://{{ inventory_hostname }}:1515"
- block:
- name: Print out Strawberry Milk configuration
shell: cat /etc/strawberry_milk.conf
changed_when: false
- name: Print out system logs
shell: "cat /var/log/messages || cat /var/log/syslog || journalctl -xb"
ignore_errors: yes
This can be useful to ensure that a service is running, that a cluster is in a
healthy state, that certain files are being created...you get the idea. The
block
I have here is an area where I run diagnostic-like tasks to help me
debug issues, which includes printing out logs and the sort. It's wrapped with
ignore_errors
so that tasks here don't affect the build (one major
contendant that errors is the log printing task when testing multiple distros).
And finally, the inventory:
debian-buster-01
ubuntu-focal-01
centos-7-01
alpine-v3-11-01
Hostnames are generated from two parts, a prefix and suffix. By default, these
are generated from the profile
key in test_profiles
in the format of
{{ profile }}-{{ suffix }}
, where suffix by default is 01
.
Once you have those files written, you're ready to test your role in Travis CI. However, you probably want more out of it, so let's go over some other topics.
It's likely you'll want to test your role against the development branch as well
as all currently supported Ansible releases. This is something you'd want to
configure in .travis.yml
and there are various ways to go about it:
env:
- ANSIBLE_GIT_VERSION='devel'
- ANSIBLE_VERSION='~=2.9.0'
- ANSIBLE_VERSION='~=2.9.0'
- ANSIBLE_VERSION='~=2.7.0'
install:
- if [ "$ANSIBLE_GIT_VERSION" ]; then pip install "https://github.com/ansible/ansible/archive/${ANSIBLE_GIT_VERSION}.tar.gz";
else pip install "ansible${ANSIBLE_VERSION}"; fi
- ansible --version
Here, we've added an install task that will either take ANSIBLE_GIT_VERSION
,
as a valid reference in the Ansible git repository, or ANSIBLE_VERSION
, a
valid version string that can be passed to pip during installation.
You can drop pretty much anything in tests/ansible.cfg
.
[defaults]
callback_whitelist=profile_tasks
forks=20
internal_poll_interval = 0.001
This runs the profile_tasks
callback on your playbook, which helps to identify
which tasks take the longest to complete. You could use this to identify any
performance regressions, for example. If you're bringing up and running your
playbook against multiple containers, specify forks
. internal_poll_interval
is a good general setting to have when you have multiple tasks/loops.
LXC images can be cached to save on bootstrapping time, especially when you're
testing against several profiles. Drop the following in your .travis.yml
and
this role will take care of the rest.
cache:
directories:
- "$HOME/lxc"
pip: true
(pip: true
doesn't mean anything for this role, but it's included here since
you might want to cache your Ansible installation as well.)
To specify what distributions to test against, use test_profiles
. Supported
profiles include (feel free to request/contribute new ones):
test_profiles:
- profile: alpine-v3.11
- profile: alpine-v3.10
- profile: alpine-v3.9
- profile: centos-7
- profile: debian-buster
- profile: debian-stretch
- profile: ubuntu-focal
- profile: ubuntu-bionic
- profile: ubuntu-xenial
The following profiles have definitions, but are not necessarily actively supported as targets by this role (i.e. tests are no longer ran against these), either because they're officially EOL upstream or are relatively old. No guarantees are made that they are still functional (but they probably are).
test_profiles:
- profile: alpine-v3.8
- profile: alpine-v3.7
- profile: alpine-v3.6
- profile: centos-6
- profile: debian-jessie
- profile: debian-wheezy
- profile: fedora-28
- profile: fedora-27
- profile: fedora-26
- profile: fedora-25
- profile: ubuntu-trusty
You can look at vars/main.yml
for more information about those profiles.
A test container, if no prefix is specified, is given a hostname of
{{ profile }}-{{ suffix }}
, where profile
is sanitized for usage in a DNS
name. Default prefixes are defined in vars/main.yml
, so refer to it if you
are unsure what a particular profile's prefix is. If test_host_suffixes
is
not defined, suffix
here becomes a zero-padded double digit integer starting
from 1 (up to the requested number of hosts specified by
test_hosts_per_profile
).
For example, the following creates debian01
, debian02
, and debian03
:
test_profiles:
- profile: debian-buster
prefix: debian
test_hosts_per_profile: 3
The following creates ubuntu-app-python2
and ubuntu-app-python3
:
test_profiles:
- profile: ubuntu-focal
prefix: ubuntu-
test_host_suffixes:
- app-python2
- app-python3
You can also override the container configuration used, if necessary (for e.g. mounting a shared folder):
container_config:
- "lxc.aa_profile=unconfined"
- "lxc.mount.auto=proc:rw sys:rw cgroup-full:rw"
- "lxc.cgroup.devices.allow=a *:* rmw"
On the off-chance you need to ("missing" packages should be installed by default within this role, so open an issue), you can install extra packages inside the test containers as well:
additional_packages:
- make
If caching is identified to be enabled in .travis.yml
, you can selectively
cache a subset of your test profiles by specifying them in lxc_cache_profiles
.
These must be valid profiles and present in test_profiles
.
To cache to directory different from $HOME/lxc
, modify lxc_cache_directory
.
If you need to disable the usage of OverlayFS in the LXC containers (e.g. if
you're attempting to use OverlayFS inside of the LXC container), set
lxc_use_overlayfs
to no
(or any False
variant)
Musee Ullah (@lae, [email protected])
Wilmar den Ouden (@wilmardo)
This role is currently still pre-1.0 and thus is not guaranteed to be stable. If you run into an issue using this role, please open an issue with a brief description and any appropriate logs so that it can be fixed and we can be one step closer to our first stable release.
Please make sure you are pinning to a specific version (pinning to minor may be fine) when using this role. Failure to do so may result in your tests beginning to fail due to breaking changes in a minor version release before a 1.0 release.