Skip to content
This repository has been archived by the owner on Apr 2, 2024. It is now read-only.

Ansible role that prepares a Travis CI environment and creates LXC containers for testing roles.

License

Notifications You must be signed in to change notification settings

lae/ansible-role-travis-lxc

Repository files navigation

lae.travis-lxc

Build Status Ansible Galaxy Role

Configures and starts N LXC containers to use in the Travis CI environment for simpler testing of Ansible roles across different distributions.

Usage

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.

Testing multiple Ansible versions

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.

Ansible performance and profiling

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.

Caching

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.)

Role Variables

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)

Contributors

Musee Ullah (@lae, lae@lae.is)
Wilmar den Ouden (@wilmardo)

Stability

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.