Trying out Ansible

It’s been a week since I last tried doing anything with my new homelab rack and it’s still in a mess. Since my previous approch wasn’t working, I’ve decided to try a completely different set of tools, and I figured why not pick up a new skill while I’m at it. I’ve heard a lot about Ansible, and it seems like now is a good time to try it out.

What is Ansible?

Ansible is an automation tool that lets you use YAML files to describe what you want your servers to look like, and then it takes care of making sure that they are in that state. The main advantage of Ansible is that it’s agentless, which means that you don’t need to install anything on the servers that you want to manage. You just need to have SSH access to them. In my case, I’m using my Pi3B+ (which I’ll refer to from now on as ControlPi) as the control node, and the rest of the rack as the managed nodes.

Setting up Ansible

Installing Ansible on ControlPi was pretty straightforward. I opted to use pipx since it was the first installation method on the official documentation. I just had to run the following commands:

sudo apt update
sudo apt install pipx
pipx ensurepath

After that, I installed Ansible with pipx:

pipx install --include-deps ansible

This took a few minutes since the Pi3B+ isn’t the fastest machine, but it eventually finished without any issues. I ran pipx ensurepath again the update the system PATH, source ~/.bashrc to reload the PATH, and then ran ansible --version to check that the installation was successful.

Following the quickstart guide

I followed the quickstart guide on the Ansible documentation to get a feel for how things work. I created a new folder in the home directory called ansible, and then another folder called quickstart inside that. I then created a new file called inventory.ini with the following contents:

[myhosts]
192.168.1.206
192.168.1.207

This file lists the IP addresses of the managed nodes. At this point, it’s just the the two VMs that run docker containers, but I’ll be adding the rest of the rack to this file as I go along.

I then ran the following command to verify the inventory:

ansible-inventory -i inventory.ini --list

It printed out what looked like a json file, so I guess it worked. So continuing with the guide, I tried pinging the managed nodes:

ansible myhosts -m ping -i inventory.ini

But this time it threw an error at me, and got stuck there for a while until I used Ctrl+C to stop it.

192.168.1.207 | UNREACHABLE! => {
    "changed": false,
    "msg": "Failed to connect to the host via ssh: Warning: Permanently added '192.168.1.207' (ED25519) to the list of known hosts.\r\n[email protected]: Permission denied (publickey,password).",
    "unreachable": true
}

I could manually ping the VMs, so I know it’s not a network issue. From what I can tell, it looks like ansible uses the same username as the one running the command to connect to the managed nodes. Since I was running the command as admin, it was trying to connect to the managed nodes as admin. But I don’t have a user on my VMs named admin, so it can’t SSH in.

A quick google search later, and I found that I could specify the username to use with the ansible_user variable. Varaibles looked easier to manage in YAML, so I changed the inventory file to inventory.yaml and updated it to look like this:

docker-vms:
  hosts:
      docker_internal:
        ansible_host: 192.168.1.206
        ansible_user: richard
      docker_external:
        ansible_host: 192.168.1.207
        ansible_user: richard

I then ran the following command to verify the inventory:

ansible-inventory -i inventory.yaml --list

It gave me a similar json as before, but it also gave a warning about invalid characters in a group name.

[WARNING]: Invalid characters were found in group names but not replaced, use -vvvv to see details

I ran the command again with the -vvvv flag to see what the issue was, and it turns out that the group name docker-vms was invalid. I changed it to docker_vms and ran the command again. This time it worked without any issues.

I then tried pinging the managed nodes again:

ansible docker_vms -m ping -i inventory.yaml

But nope, still getting an error.

{ % highlight bash %} docker_external | UNREACHABLE! => { “changed”: false, “msg”: “Failed to connect to the host via ssh: [email protected]: Permission denied (publickey,password).”, “unreachable”: true } { % endhighlight %}

It occurred to me at this point that it’s not pinging the VMs, but rather trying to SSH into them. I hadn’t set up my SSH keys, so I couldn’t connect to them.

Setting up SSH keys

I generated a new SSH key on ControlPi with the following command, then copied it to the managed nodes:

ssh-keygen -t ed25519 -C "admin@pidns"
ssh-copy-id -i ~/.ssh/id_ed25519 [email protected]
ssh-copy-id -i ~/.ssh/id_ed25519 [email protected]

It prompted me for my password on the two managed nodes, and after that I tried the “ping” command again.

ansible docker_vms -m ping -i inventory.yaml

This time, everything worked out fine, but I still got a warning about the python interpreter on the managed nodes.

[WARNING]: Platform linux on host docker_internal is using the discovered Python interpreter at /usr/bin/python3.12, but future installation of another Python interpreter could change the meaning of that path. See https://docs.ansible.com/ansible-core/2.18/reference_appendices/interpreter_discovery.html for more information.
docker_internal | SUCCESS => {
    "ansible_facts": {
        "discovered_interpreter_python": "/usr/bin/python3.12"
    },
    "changed": false,
    "ping": "pong"
}

I’ll have to look into this later, but for now, I’m just happy that I got the ping command to work and I can continue with the quickstart guide.

Quickstart playbook

I created a new file called playbook.yaml with the following contents, then ran the playbook:

- name: My first play
  hosts: docker_vms
  tasks:
   - name: Ping my hosts
     ansible.builtin.ping:

   - name: Print message
     ansible.builtin.debug:
       msg: Hello world
ansible-playbook -i inventory.yaml playbook.yaml

Ansible returned the following output, which seems fine to me other than the python interpreter warnings.

PLAY [My first play] **********************************************************************************************************

TASK [Gathering Facts] ********************************************************************************************************
[WARNING]: Platform linux on host docker_internal is using the discovered Python interpreter at /usr/bin/python3.12, but
future installation of another Python interpreter could change the meaning of that path. See https://docs.ansible.com/ansible-
core/2.18/reference_appendices/interpreter_discovery.html for more information.
ok: [docker_internal]
[WARNING]: Platform linux on host docker_external is using the discovered Python interpreter at /usr/bin/python3.12, but
future installation of another Python interpreter could change the meaning of that path. See https://docs.ansible.com/ansible-
core/2.18/reference_appendices/interpreter_discovery.html for more information.
ok: [docker_external]

TASK [Ping my hosts] **********************************************************************************************************
ok: [docker_external]
ok: [docker_internal]

TASK [Print message] **********************************************************************************************************
ok: [docker_internal] => {
    "msg": "Hello world"
}
ok: [docker_external] => {
    "msg": "Hello world"
}

PLAY RECAP ********************************************************************************************************************
docker_external            : ok=3    changed=0    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0
docker_internal            : ok=3    changed=0    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0

Fixing the python warning

Based on the interpreter discovery documentation, it looks like I can set the python interpreter to use in the inventory file, or I can just silence the warning. I’ll go with the latter for now since I might mess with python interpreters later on and I’d rather not have to update the inventory file every time. To silence the warning, I added the ansible_python_interpreter: auto_silent variable to the inventory file:

docker_vms:
  hosts:
      docker_internal:
        ansible_host: 192.168.1.206
        ansible_user: richard
        ansible_python_interpreter: auto_silent
      docker_external:
        ansible_host: 192.168.1.207
        ansible_user: richard
        ansible_python_interpreter: auto_silent

I then ran the playbook again, and this time I didn’t get any warnings about the python interpreter.

Conclusion

I’ll leave things here for now. I’ve managed to get Ansible installed, set up an inventory file, and run a basic playbook. Moving forward, I’ll likely be setting up some new VMs or LXCs, and I’ll create a new playbook in parallel to automate the process.