Ansible for Localhost
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
andyaml
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 stylewith_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