Skip to content

Latest commit

 

History

History
192 lines (143 loc) · 11.6 KB

COOKBOOK.md

File metadata and controls

192 lines (143 loc) · 11.6 KB

Putting the Ansible Suitcase to Work in your Project: A Cookbook

Preramble

Proficiency in modern bash shell scripting is recommended. It is assumed that you know

Here are some resources to touch up on your skills:

Starter Kit

💡 One does not simply fork or copy the suitcase source into one's own Git project.

The recommended way to use the suitcase is through a foosible shell wrapper that downloads the suitcase (once) and executes it (every time). Here is an annotated example:

#!/bin/bash

set -e  # ①
cd "$(cd "$(dirname "$0")"; pwd)"  #  ②

help () {  # ③
    fatal <<HELP_MSG
Usage:

  $0 [ -t sometag ] [ ... ]
HELP_MSG
}

ensure_suitcase () {    # ④
    if ! test -f ansible-deps-cache/.versions 2>/dev/null; then  # ⑤
        curl https://raw.githubusercontent.com/epfl-si/ansible.suitcase/master/install.sh | \
            SUITCASE_DIR=$PWD/ansible-deps-cache \
            SUITCASE_ANSIBLE_VERSION=9.3.0 \
            bash -x
    fi
    . ansible-deps-cache/lib.sh  #  ⑥
    ensure_ansible_runtime  # ⑦
}

ensure_suitcase
[ "$1" == "--help" ] && help
# ⑧
ansible-playbook -i inventory.yml playbook.yml "$@"

① : This tells bash to apply modern error management.

② : The first thing the wrapper script does is to cd into the directory it lives in, so that lines such as ⑥ below work without further ado. This somewhat contrived incantation is space- and control-character-resistant, as well as symlink-resistant (without relying on the realpath command, which Mac OS X lacks); yet, it preserves the caller's working directory in $OLDPWD, should you need to look there for any reason.

③ : This is where you put the minimum amount of info for the intern to figure out how to use the foosible script. Although that is a matter of taste, keeping the help() function as close as possible to the top of the script will also help people who don't run unknown commands to see what they do, even with --help. If you wonder where the fatal function comes from, see ⑥.

④ : The ensure_suitcase() function, as the name implies, ensures that the Ansible suitcase is ready in ./ansible-deps-cache. There is an install-time (once) part to that job (⑤, see below) and a run-time (everytime) part (⑦).
💡 Feel free to search-n'replace the ansible-deps-cache path to suit your preferences e.g. tmp/ansible; but do not attempt to use e.g. $HOME/.suitcase, as the Ansible suitcase doesn't support sharing (even parts of) its runtime between projects.

⑤ : This if ... fi block is the idempotent operation that installs the suitcase by running it as a shell script downloaded from GitHub on-the-spot. That is, unless ansible-deps-cache/.versions already exists, which is the stamp file that the suitcase creates upon successful installation. One can configure a variety of install-time parameters through environment variables, as documented in the comments at the top of the suitcase's install.sh.

⑥ : The suitcase comes with a small run-time shell library with a few handy functions such as fatal (but also warn, read_interactive and more), which your wrapper script may want to call.

⑦ : In order to use the installed suitcase, a couple of environment variables must be set, most notably $PATH so that the wrapper script picks up the right ansible, ansible-playbook etc. executables. This is what the ensure_ansible_runtime function takes care of.

⑧ : The script transfers control to ansible-playbook at the end and... That's pretty much it.

Adding a --prod flag

Maybe you still don't trust Ansible enough to let the intern mess with production through it — That is, until you, and you only, decide to reveal the secret flag to them! If so, here's something you could do:

  1. Teach your foosible wrapper to parse the command line, for instance:
    declare -a ansible_args
    inventory_mode="test"
    while [ "$#" -gt 0 ]; do
      case "$1" in
        --help)
          help ;;
        --prod)
          inventory_mode="test-and-prod"
          shift ;;
        *)
          ansible_args+=("$1")
          shift ;;
      esac
    done
  2. Change -i inventory.yml on the last line into a function invocation (and "$@" into "${ansible_args[@]}"), e.g.
    ansible-playbook $(inventories) playbook.yml "${ansible_args[@]}"
  3. Write inventories as a shell function e.g. like this:
    inventories () {
      case "$inventory_mode" in
        test)          echo "-i inventory/test.yml" ;;
        test-and-prod) echo "-i inventory/test.yml -i inventory/prod.yml" ;;
      esac
    }
  4. Rearrange your Ansible inventory by splitting it into inventory/test.yml and inventory/prod.yml

Accepting the remote ssh keys, once

Upon encountering a remote ssh host to which you never logged in before, Ansible will do nothing special, allowing the ssh command to prompt you to accept the remote host's public key. So far, so good.

Upon running a task that spans several unknown hosts, Ansible will attempt to run them in parallel (as it should). This will result in multiple ssh commands prompting you to type yes at the same time and (worse) competing with each other to read your bytes from the terminal, which makes it very unlikely indeed to transmit a correct and complete y, e, s, carriage-return sequence to any one of the ssh subprocesses. Your intern will be very confused.

Solution: create an ansible.cfg file that reads

[ssh_connection]
ssh_args = -o StrictHostKeyChecking=accept-new

(Credits to baconadmin)

Lore

This chapter contains additional suggestions or quote-unquote “best practices” that cannot just be boiled down into a shell wrapper. As a consequence, you will have to heed them by yourselves in your suitcase-using Ansible code.

Use the suitcase's Python for your local tasks

If one (or more) of your inventory targets is something like a Kubernetes cluster (not a “real” host that Ansible reaches over ssh), you will probably want to use the local connection plugin, which you can do right from the inventory.yml file like this:

all:
  hosts:
    # Arguably not a host, but it plays one on TV
    my-kubernetes:
      ansible_connection: local

In such a case, you will want to exert control on ansible_python_interpreter so that the “remote” Python that Ansible invokes for tasks is the one inside the suitcase, it being conveniently guaranteed to have access to the SUITCASE_PIP_EXTRA packages (like, typically, kubernetes):

all:
  hosts:
    # Arguably not a host, but it plays one on TV
    my-kubernetes:
      ansible_connection: local
      ansible_python_interpreter: "{{ ansible_playbook_python }}"

If you have INI-style inventory, and despite the lack of a text quoting feature in that case, you could make it work by squeezing out all the whitespace from inside the mustaches, like this:

[all]
my-kubernetes      ansible_connection=local  ansible_python_interpreter={{ansible_playbook_python}}

Add -F /dev/null to all your ansible_ssh_common_args

  • Fact #1: most /etc/ssh/ssh_configs out there will have a stanza like
    SendEnv LANG LC_*
  • Fact #2: most remote /etc/ssh/sshd_configs will have a matching AcceptEnv stanza
  • Fact #3: facts #1 and #2 combined can be pretty convenient for interactive ssh sessions. However...
  • Fact #4: a lot of people have no clue how locales work. When using their code as part of some Ansible module, you don't want to have to debug why it works on your workstation (configured with French locale, except without Unicode), but not your intern's (which has German for LC_PAPER and Polish for everything else).
  • Fact #5: you don't really need a site- or user-wide ssh configuration file for Ansible to operate properly; see next § for more on this topic.

Therefore, consider telling Ansible to ignore ~/.ssh/config and /etc/ssh/config entirely:

all:
  vars:
    ansible_ssh_common_args: '-F /dev/null'

💡 The ssh(1) manpage suggests a theoretically less heavy-handed approach, using -o SendEnv='-LC_*' -o SendEnv=-LANG instead. The drawback of that, of course, is that it doesn't work in practice — The hyphen form of SendEnv doesn't appear to have any effect whatsoever on OpenSSH_9.0p1's ssh client (the one that Mac OS X Ventura ships with).

Rely less on homemade fragments in your teammates' ~/.ssh/config

Your Ansible project uses the suitcase, presumably because it strives to be portable and self-contained. Therefore, its proper operation should avoid depending on ad-hoc configuration steps such as your colleagues having to edit their ~/.ssh/config manually, inasmuch as possible. (Note that this doesn't rule out homemade ~/.ssh/config fragments being disseminated as part of the team lore, e.g. for the purpose of manual ssh access outside of Ansible; that is the reason why the users guide brushes up on this topic.)

Consider setting the ansible_ssh_common_args variable in your inventory so as to provide ~/.ssh/config-equivalent functionality to your configuration-as-code — regardless of whether you disabled ~/.ssh/config in the first place, as per the advice in the previous §. In particular, you have to make sure that Ansible is able to reach your fleet over ssh, preferably without clunky and error-prone steps being involved such as typing in ssh passwords1. For instance, suppose you are using Ansible to administer a cluster in which only headnode is reachable directly through ssh. You would then want to configure the ProxyJump ssh option for all nodes except headnode. Combining this task with advice from the previous §, this works out to the following excerpt for your YAML inventory:

all:
  vars:
    ansible_user: root
    ansible_ssh_common_args: "-F /dev/null -o ProxyJump=headnode"
  hosts:
    headnode:
      ansible_ssh_common_args: "-F /dev/null"

Ansible's precedence rules for vars do what you want here, i.e. the setting in headnode for ansible_ssh_common_args shadows the one in all. (Based on the same reference material, ansible_user: root will apply to the whole cluster, including headnode as the vars for the latter don't have an override for it.)

Footnotes

  1. In fact, it turns out that Ansible's primitive support for prompting for secrets makes password-based ssh access either unwieldy, insecure, or downright infeasible, depending on the specifics.