Create Ubuntu development environment on macOSX using Vagrant and Ansible

To learn about compilers hands-on, I built a Linux development environment with configuration management.

Problems discussed in this article

  • how to install software on guest VM
  • share directory between host and guest

The GitHub repository that manages the configurations explained in this article: https://github.com/Rindrics/envs

Prerequisite

Environment of running examples described in this article:

  • macOS 12.3.1 (MacBook Pro 2020, Intel Core)
  • VirtualBox 6.1
  • Vagrant 2.2.19
    • vagrant-vbguest
  • Ansible 2.13.2
  • Python 3.10.6

Need for Linux development environment

Recently I’ve been planning to learn about low-level technology, especially compilers. I already found an excellent guide for learning how to develop compilers, but the problem is that my development environment—macOSX is incompatible with the examples in that guide. So I need a Linux virtual development environment on my machine.

I wanted the provisioning procedure simple, but I also wanted to manage the VM configuration so that it is reproducible. Thus I decided to provision a Vagrant VM configured using Ansible.

Provision Ubuntu VM with minimum configuration

It’s easy to provision VM using Vagrant. Here’s the minimum Vagrantfile to provision Ubuntu22.04:

VAGRANTFILE_API_VERSION = "2"

Vagrant.configure(VAGRANTFILE_API_VERSION) do |config|
  config.vm.box = "ubuntu/jammy64"
end

To provision the VM, we just need to execute vagrant up:

❯ vagrant up
Bringing machine 'default' up with 'virtualbox' provider...
==> default: Importing base box 'ubuntu/jammy64'...

...(Omitted)

Restarting VM to apply changes...
==> default: Attempting graceful shutdown of VM...
==> default: Booting VM...
==> default: Waiting for machine to boot. This may take a few minutes...
==> default: Machine booted and ready!
==> default: Checking for guest additions in VM...
==> default: Mounting shared folders...
    default: /vagrant => /Users/rindrics/envs/ubuntu22.04

How easy it is!

Configure VM using Ansible

Minimum configuration

Before starting to install specific software on Vagrant VM, I learned how to configure Vagrant VM using Ansible by reading this page.

To start managing the VM using Ansible, I fixed the IP address of the VM:

 VAGRANTFILE_API_VERSION = "2"

 Vagrant.configure(VAGRANTFILE_API_VERSION) do |config|
   config.vm.box = "ubuntu/jammy64"

+  config.vm.network "private_network", ip: "192.168.33.10"
 end

I need to specify that fixed IP in Ansible “inventory file” under a directory with an arbitrary name. In this example, I created provisioning/hosts as follows:

[devenv]
192.168.33.10

Then I told Ansible the location of the inventory file:

 VAGRANTFILE_API_VERSION = "2"

 Vagrant.configure(VAGRANTFILE_API_VERSION) do |config|
   config.vm.box = "ubuntu/jammy64"

   config.vm.network "private_network", ip: "192.168.33.10"
+
+  config.vm.provision "ansible" do |ansible|
+    ansible.inventory_path = "provisioning/hosts"
+  end
 end

Next, I wrote a minimum Ansible “playbook” that does nothing on the guest VM:

---
- name: Setup development environment
  hosts: devenv
  become: yes
  user: vagrant

I saved the configuration above as provisioning/playbook.yml and told Ansible the location:

 VAGRANTFILE_API_VERSION = "2"

 Vagrant.configure(VAGRANTFILE_API_VERSION) do |config|
   config.vm.box = "ubuntu/jammy64"

   config.vm.network "private_network", ip: "192.168.33.10"

   config.vm.provision "ansible" do |ansible|
+    ansible.playbook = "provisioning/playbook.yml"
     ansible.inventory_path = "provisioning/hosts"
   end
 end

All set, I updated the VM configuration as follows:

❯ vagrant provision
==> default: Running provisioner: ansible...
    default: Running ansible-playbook...
 ______________________________________
< PLAY [Setup development environment] >
 --------------------------------------
        \   ^__^
         \  (oo)\_______
            (__)\       )\/\
                ||----w |
                ||     ||

 ________________________
< TASK [Gathering Facts] >
 ------------------------
        \   ^__^
         \  (oo)\_______
            (__)\       )\/\
                ||----w |
                ||     ||

ok: [192.168.33.10]
 ____________
< PLAY RECAP >
 ------------
        \   ^__^
         \  (oo)\_______
            (__)\       )\/\
                ||----w |
                ||     ||

192.168.33.10              : ok=1    changed=0    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0

Cows in Vagrant output told me I managed to call Ansible via Vagrant!

Install a software on VM

To learn how to configure VM using Ansible, I looked for software that is small and easy to install. I chose Starship, my favorite software that enables easy customization of shell prompt.

As explained in “Installation”, we just need to run the following two commands to use Starship on bash:

curl -sS https://starship.rs/install.sh | sh
echo 'eval "$(starship init bash)"' >> ~/.bashrc

I did this via Ansible by updating playbook as follows:

 ---
 - name: Setup development environment
   hosts: devenv
   become: yes
   user: vagrant
+
+  tasks:
+    - name: Install Starship
+      ansible.builtin.shell:
+        cmd: curl -sSf https://starship.rs/install.sh | sh -s -- -y

Then applied the configuration:

❯ vagrant provision
==> default: Running provisioner: ansible...
    default: Running ansible-playbook...
 ______________________________________
< PLAY [Setup development environment] >
 --------------------------------------
        \   ^__^
         \  (oo)\_______
            (__)\       )\/\
                ||----w |
                ||     ||

 ________________________
< TASK [Gathering Facts] >
 ------------------------
        \   ^__^
         \  (oo)\_______
            (__)\       )\/\
                ||----w |
                ||     ||

ok: [192.168.33.10]
 _________________________
< TASK [Install Starship] >
 -------------------------
        \   ^__^
         \  (oo)\_______
            (__)\       )\/\
                ||----w |
                ||     ||

changed: [192.168.33.10]
 ____________
< PLAY RECAP >
 ------------
        \   ^__^
         \  (oo)\_______
            (__)\       )\/\
                ||----w |
                ||     ||

192.168.33.10              : ok=2    changed=1    unreachable=0    failed=0    skipped=0    rescued=0    ignored=0

Starship seems to be installed on the VM.

Running Ansible commands in verbose mode is useful to know what is happening in the VM:

 VAGRANTFILE_API_VERSION = "2"

 Vagrant.configure(VAGRANTFILE_API_VERSION) do |config|
   config.vm.box = "ubuntu/jammy64"

   config.vm.network "private_network", ip: "192.168.33.10"

   config.vm.provision "ansible" do |ansible|
     ansible.playbook = "provisioning/playbook.yml"
     ansible.inventory_path = "provisioning/hosts"
     ansible.limit = 'devenv'
+    ansible.verbose = true
   end
 end

To use Starship, we have to update ~/.bashrc on VM. I added a task to the playbook:

 ---
 - name: Setup development environment
   hosts: devenv
   become: yes
   user: vagrant

   tasks:
     - name: Install Starship
       ansible.builtin.shell:
         cmd: curl -sSf https://starship.rs/install.sh | sh -s -- -y
+
+    - name: Add eval statement to .bashrc
+      lineinfile:
+        dest: /home/vagrant/.bashrc
+        line: "eval \"$(starship init bash)\""
+        owner: vagrant

and applied it:

 ______________________________________
< TASK [Add eval statement to .bashrc] >
 --------------------------------------
        \   ^__^
         \  (oo)\_______
            (__)\       )\/\
                ||----w |
                ||     ||

changed: [192.168.33.10]

Done.

Finally, I logged in to the VM by vagrant ssh:

❯ vagrant ssh
Welcome to Ubuntu 22.04 LTS (GNU/Linux 5.15.0-41-generic x86_64)

 * Documentation:  https://help.ubuntu.com
 * Management:     https://landscape.canonical.com
 * Support:        https://ubuntu.com/advantage

  System information as of Sun Nov  6 04:58:58 UTC 2022

  System load:  0.0               Processes:               107
  Usage of /:   4.8% of 38.70GB   Users logged in:         0
  Memory usage: 21%               IPv4 address for enp0s3: 10.0.2.15
  Swap usage:   0%                IPv4 address for enp0s8: 192.168.33.10


108 updates can be applied immediately.
57 of these updates are standard security updates.
To see these additional updates run: apt list --upgradable


Last login: Sun Nov  6 04:59:03 2022 from 192.168.33.1

vagrant in 🌐 ubuntu-jammy in ~

A pretty prompt tells the installation of Starship succeeded.

Share directory between host and guest

During development, it is useful to have a host–guest shared folder.

So I also configured synced folder as follows:

 VAGRANTFILE_API_VERSION = "2"

 Vagrant.configure(VAGRANTFILE_API_VERSION) do |config|
   config.vm.box = "ubuntu/jammy64"

   config.vm.network "private_network", ip: "192.168.33.10"
+  config.vm.synced_folder "./shared", "/home/vagrant/shared", :mount_options => [ "dmode=777", "fmode=777"]

   config.vm.provision "ansible" do |ansible|
     ansible.playbook = "provisioning/playbook.yml"
     ansible.inventory_path = "provisioning/hosts"
     ansible.limit = 'devenv'
   end
 end

On host macOS, I created the synced directory and blank file under it:

❯ mkdir shared
❯ touch shared/foo

To see whether the shared/foo exists on the VM, I applied the configuration and logged in to the VM:

❯ vagrant reload
...(Omitted)

❯ vagrant ssh
Welcome to Ubuntu 22.04 LTS (GNU/Linux 5.15.0-41-generic x86_64)

 * Documentation:  https://help.ubuntu.com
 * Management:     https://landscape.canonical.com
 * Support:        https://ubuntu.com/advantage

  System information as of Sun Nov  6 05:08:41 UTC 2022

  System load:  0.01025390625     Processes:               111
  Usage of /:   4.8% of 38.70GB   Users logged in:         0
  Memory usage: 21%               IPv4 address for enp0s3: 10.0.2.15
  Swap usage:   0%                IPv4 address for enp0s8: 192.168.33.10


108 updates can be applied immediately.
57 of these updates are standard security updates.
To see these additional updates run: apt list --upgradable


Last login: Sun Nov  6 05:06:41 2022 from 10.0.2.2

vagrant in 🌐 ubuntu-jammy in ~
❯ ls shared
foo

Yes, there it is!

I’ve successfully obtained a virtual development environment to start learning about compilers.


comments powered by Disqus