Ansible is a super powerful system configuration tool that is normally used to manage sets of servers. However, as you might have derived from the title it can also be used to configure a local deployment.

Edit: I had to reinstall Arch, this playbook works, mostly. vars/pacman_packages.yaml was missing some packages that later steps needed. It still saved me hours of work though.

End Goal

Configure a brand new installation of Arch Linux such that one (ansible) command can install and configure the system.

During installation the following dependencies must be manually installed. Either during the pacstrap step or with pacman when chrooted in.

base-devel ansible git

After the system has restarted and the user is logged into a tty as root, and the homed user has been created, run the following command:

ansible-pull -U https://git.kgb33.dev/kgb33/ansible.git

The completed ansible project can be viewed at this repo.

Basics

Ansible has a command ansible-pull that can pull down and run playbooks defined within a remote git repository.

The repository should have the following structure.

Project Structure

 ./
├──  vars/
│  └──  installed-packages
├──  ansible.cfg
├──  inventory.yml
├──  local.yml
└──  README.md

This project has three main files.

  • local.yml - The main playbook.
  • ansible.cfg - Optional, Used to configure ansible.
  • inventory.yml - Optional, suppresses no inventory file warnings.

Note: While both yml and yaml are valid yaml file extensions, ansible will only detect files ending with .yml.

Installing Packages

Ansible already knows how to install packages from the standard distribution’s package managers (apt, dnf, pacman, etc.). Installing AUR packages with paru requires a custom plugin.

To use the distribution package manager create a task as follows.

- hosts: localhost
  tasks:
    - name: Install tmux
      become: yes
      package:
        name: tmux
        state: present

This is fairly tedious to write out a task for each package. [variables][ansible-variables] can be used instead.

- hosts: localhost
  tasks:
    - name: Install Packages
      become: yes
      package:
        name: "{{item}}"
        state: present
      with_items:
        - htop
        - tmux
        - git

Here the name: "{{item}}" acts kinda like a for-each loop. The task will run for each item in with_items.

Variable Files

I have a lot of packages installed, 87 to be exact, I don’t want to have to type out a line for each one. Variables and Variable files can help us with this. Furthermore, these files can be auto-generated with some terminal magic.

Variable files are .yml files and lists of variables are defined using yaml lists.

pacman_packages:
  - alacritty
  - ansible
  - autoconf
  ...

The following script creates vars/pacman_packages.yml and vars/aur_packages.yml. These files contain the currently installed packages on the system from the Arch Linux repositories and from the AUR respectively.

#!/bin/sh
[ -d vars ] || mkdir vars # Create dir if it does not exist

# Generate AUR package list
pacman -Qqm \
        | awk 'BEGIN{print "aur_packages:"}; {printf"  - %s\n", $1};' \
        > vars/aur_packages.yml

# Generate pacman package list
pacman -Qqe \
        | grep -v "$(pacman -Qqm)" \
        | awk 'BEGIN{print "pacman_packages:"}; {printf"  - %s\n", $1};' \
        > vars/pacman_packages.yml

Then to use these files in the Install Packages tasks change the yaml to the following.

- hosts: localhost
  vars_files:
    - vars/pacman_packages.yml
    - vars/aur_packages.yml
  tasks:
    - name: Install Packages
      become: yes
      package:
        name: "{{pacman_packages}}"
        state: present

Note: Here we are passing the list of packages to pacman, rather than one at a time. This speeds up the task because it uses only one call to pacman, which limits the pre/post installation tasks, as well as allowing us to take advantage of parallel downloads.

Updating pacman.conf to Enable Parallel Downloads

By default parallel downloads is disabled in /etc/pacman.conf. To enable it (and change some other settings) create a new task before the “Install Packages” task. Furthermore, running reflector might be a good idea before downloading hundreds of packages.

  tasks:
    - name: Update pacman.conf Settings
      lineinfile:
        path: /etc/pacman.conf
        regexp: "{{ item.regexp }}"
        line: "{{ item.line }}"
      loop:
        - { regexp: '^#Color', line: 'Color' }
        - { regexp: '^#VerbosePkgLists', line: 'VerbosePkgLists' }
        - { regexp: '^#ParallelDownloads', line: 'ParallelDownloads = 7' }

    - name: Install Reflector
      become: yes
      pacman:
        name: reflector
        state: present

    - name: Update Reflector Settings
      lineinfile:
        path: /etc/xdg/reflector/reflector.conf
        regexp: "{{ item.regexp }}"
        line: "{{ item.line }}"
      loop:
        - { regexp: '^# --country France,Germany', line: '--country US,CA ' }
        - { regexp: '^--latest', line: '--latest 50' }

    - name: Enable Reflector.timer
      systemd:
        name: reflector.timer
        state: started
        enabled: yes

    - name: Install Packages
 	...

Specialized package managers

Unfortunately pacman cannot install AUR packages, an AUR helper is needed for that. My preferred AUR helper is paru, but any helper should work.

makepkg cannot be run as root, In fact, the Arch Linux wiki recommends creating a user that can run pacman without a password.

Create two new tasks as follows:

 tasks:
    ...
    - name: Create AUR-Builder User
	user: name=aur_builder
    - name: Give aur_builder Pacman privileges
	lineinfile:
        path: /etc/sudoers.d/aur_builder-allow-to-sudo-pacman
		state: present
        line: "aur_builder ALL=(ALL) NOPASSWD: /usr/bin/pacman"
		validate: /usr/sbin/visudo -cf %s
        create: yes

The first task creates a new user, aur_builder, then the second allows it to run pacman without a password. Additionally, because aur_builder does not have a password, only root can su into it.

Install paru

Paru is installed using a series of shell commands. Furthermore, because Paru is written in rust, and rust takes a while to compile, the task checks to see if the binary is present at /usr/bin/paru before running.

tasks:
    ...
    - name: Install paru
	become: yes
    become_user: aur_builder
	args:
        creates: /usr/bin/paru
	shell: |
		git clone https://aur.archlinux.org/paru.git
		cd paru
        yes | makepkg -si
		cd -
        rm -rf paru

Use paru to install packages.

The Ansible pacman plugin allows the executable to be overridden. This allows all the logic built into community.general.pacman to extend to paru.

  tasks:
    ...
    - name: Install AUR Packages
      become: yes
      become_user: aur_builder
      pacman:
        name: "{{aur_packages}}"
        state: present
        executable: paru

pipx

I have several python applications installed. Rather than allow pacman to manage different python libraries, pipx installs each application in its own separate virtual environment.

Ansible has a plugin [community.general.pipx][ansible-pipx] to manage pipx. As a result, the task will closely mirror the pacman and paru tasks.

Update package-list.sh to create a variable file for pipx too.

# Generate pipx package list
pipx list \
        | grep "package" \
        | awk 'BEGIN{print "pipx_packages:"}; {printf"  - %s\n", $2};' \
        > vars/pipx_packages.yml

Then update the vars_files and add a new task.

- hosts: localhost
  vars_files:
    - vars/pacman_packages.yml
    - vars/aur_packages.yml
    - vars/pipx_packages.yml
  tasks:
    ...
    - name: Install pipx Packages
      community.general.pipx:
        name: "{{ item }}"
      with_items:
        - "{{ pipx_packages }}"

pipx can only install one item at a time, hence the for-each style with_items syntax.

Create user

I use systemd-homed for account management, unfortunately it is not currently scriptable and requires a interactive element, as a result the user must be created manually before running the Ansible playbook using the following command.

homectl create --identity=./vars/kgb33.identity

Where the $USER.identity file is the output of homectl inspect $USER -EE with the privileged key removed.

Once the user has been created Ansible can modified it in the following tasks.

Add to sudoers

We already modified the aur_builder’s permissions in a previous step, changing ours is a very similar step.

    - name: Give kgb33 sudo rights
      lineinfile:
        path: /etc/sudoers.d/kgb33-sudo
        state: present
        line: "kgb33 ALL=(ALL) ALL"
        validate: /usr/sbin/visudo -cf %s
        create: yes

Add to docker group

Here we use the group builtin to ensure that the group docker exists, then the user kgb33 is added to the docker group.

    - name: Ensure Docker Group Exists
      group:
        name: docker
        state: present

    - name: Add kgb33 to the docker role
      user:
        name: kgb33
        groups: docker
        append: yes

Pull down dotfiles

Use a script task to clone and checkout dotfiles. See here for a more in-depth tutorial.

    - name: Install dotfiles
      become: yes
      become_user: kgb33
      args:
        creates: /home/kgb33/.dotfiles/
      shell: |
        cd $HOME
        git clone --bare git@github.com:KGB33/.dotfiles.git $HOME/.dotfiles
        /usr/bin/git --git-dir=$HOME/.dotfiles/ --work-tree=$HOME checkout