Vagrant is a personal tool to automatically provision virtual machine environments. It comes for free along with VirtualBox plugin, although there are commercial plugins for other virtualization engines such as VMWare. It is a very handy tool, especially for developers: it enables them to quickly mock-up or destroy even complex virtual machine environments by executing just one command. Within this post we'll see how easily it can be installed, and how easy it is operating with it.
A tool for quickly mock-up environments
Vagrant is "a tool for building and managing virtual machine environments in a single workflow": this means that you can define a whole environment made of several virtual machines within a single manifest file called vagrantfile - the syntax of the manifest follows Ruby coding.
Vagrant processes this manifest file and executes the directives stated into the file by connecting to hypervisors using a component called provider: Virtualbox , QEMU and KVM are free providers, but there are also commercial ones like VMWare provider.
Vagrant can be exploited by:
- system engineers to mock-up virtual environments to test installation of new software or to perform proof of concepts
- developers to easily create virtual environments with all the components necessary to perform the integration tests of the software they are developing, without interfering with anybody since the environment is isolated and local to his/hers workstation
If necessary, the vagrantfile can be shared and versioned using the company SCM and even integrated in the toolchain.
Installing Vagrant
Install vagrant is really very simple: just go to https://www.vagrantup.com/downloads.html and download and install HashiCorp's Vagrant for the platform you are using.
Vagrant requires access to the Internet to download Virtual Machine images – these are called Vagrant boxes in Vagrant's terms. In addition to that, the operating system installed into the Virtual Machines requires Internet access to download software packages to be installed (such as RPM from online yum/dnf repositories).
If you are working without a direct connection to the Internet you should ask the administration team of your corporate proxy to grant requests to https://vagrantcloud.com/ and to the online repositories of the operating system of the Virtual Machines you want to install.
After that you should also have to install the Vagrant proxy configuration plugin – issue the following command - in this example the corporate proxy URL is http://proxy.carcano.local:8080
export http_proxy=http://proxy.carcano.local:8080
export https_proxy=http://proxy.carcano.local:8080
vagrant plugin install vagrant-proxyconf
Once installed, each time you want to run vagrant, you must export the following variables right after opening the command prompt (the shell) before running vagrant up:
- VAGRANT_HTTP_PROXY – for example http://proxy.carcano.local:8080
- VAGRANT_HTTPS_PROXY – for example https://proxy.carcano.local:8080
- VAGRANT_FTP_PROXY – for example http://proxy.carcano.local:8080
- VAGRANT_NO_PROXY - localhost,127.0.0.1,.carcano.local
Upgrading Vagrant
Upgrading Vagrant requires only downloading the new version and installing it, overwriting the old installation. If you have installed any plugin, right after upgrading Vagrant you should also upgrade them as well. You can upgrade them as a whole as follows:
vagrant plugin update
Virtualbox
To keep on the free of charge side, and to have an easy to setup solution suitable to every platform, we install Vagrant with the Oracle Virtualbox provider: just go to https://www.virtualbox.org/wiki/Downloads and download and install Oracle's Virtualbox for the platform you are using.
Since Vagrant's default provider is the Virtualbox one, in this case it is not necessary to install a provider.
The only thing that you must really take care of manually doing is to add VirtualBox's path to the PATH system wide environment variable.
For example, on Windows it must be set as shown by the box on the right: by doing so you are able to issue commands such as VBoxManage without having to specify the full path of the folder containing it.
QEMU
Despite, to stay on the free side, Virtualbox is certainly the preferred choice, also QEMU can be an interesting option: M1 processors are quickly gaining market shares, and it may happen that M1 users need to run x86_64 VM. A way for doing that is running vagrant with the QEMU provider - QEMU indeed can easily emulate the x86_64 architecture, enabling the run x86_64 VM - the price to pay are of course performances, so this is a good solution only when dealing with labs or not performance sensitive disposable development environments.
QEMU can be installed as follows:
brew install qemu
and Vagrant's QEMU provider
vagrant plugin install vagrant-vbguest
vagrant plugin install vagrant-qemu
Vagrant Boxes
Downloading Vagrant Boxes
Once both VirtualBox and Vagrant are installed, and proxy has been configured if necessary, you can test your installation by downloading a vagrant box.
You can get a list of public available VagrantBoxes on Hashicorps's VagrantCloud at https://app.vagrantup.com/boxes/search?provider=virtualbox
For example, we can easily install the official centos/8 VagrantBox to our local repository as follows:
vagrant box add --provider virtualbox centos/8
This downloads the CentOS 8 official VagrantBox: it will take some time (of course it mostly depends on the speed of your connection to the Internet), so this can be the right moment to relax with a coffee or a cup of tea.
Updates of the boxes are periodically released. Updating a box is really easy: for example, to update the centos/8 box simply issue
vagrant box update --box centos/8
Although this should be straightforward, it is worth to say that the upgrade is focused only to the box itself: the VMs you already provisioned using that box do not get updated - You should use the updating commands of the guest itself to keep things up to date.
You can of course copy Vagrant boxes from another workstation (for example if you do not have a fast Internet connection, or you are paying for the amount of data you download): boxes are stored beneath the following path:
- C:\Users\%USERNAME%\.vagrant.d\boxes on Windows
- ~/.vagrant.d/boxes on Mac and Linux
Creating Vagrant Boxes
We can of course create our own Vagrant Boxes: despite there exist two different ways of creating Vagrant boxes, the procedure is quite simple.
Extending An Existing Vagrant Box
This is of course the most straightforward way: just light-up the existing Vagrant Box you prefer. You just need to run the "vagrant init" statement followed by the Vagrant box you want to use as by the following example:
vagrant init centos/8
the output is as follows:
A `Vagrantfile` has been placed in this directory. You are now
ready to `vagrant up` your first virtual environment! Please read
the comments in the Vagrantfile as well as documentation on
`vagrantup.com` for more information on using Vagrant.
as you see, the outcome is a Vagrantfile generated on the fly that can be used to start a CentOS 8 VM.
Let's provision and start the VM as follows:
vagrant up
and eventually login to the VM using the following command:
vagrant ssh
we are now connected to the CentOS 8 VM: skip to the "Preparing The Vagrant Box" paragraph.
Creating A Vagrant Box From Scratch
You may prefer to start from scratch, creating a Virtual Machine as you wish: simply create a VM using your favorite Hypervisor - the only mandatory condition is that a Vagrant provider for the Hypervisor you want to use.
The most popular Hypervisor used with Vagrant is certainly VirtualBox, so I will show you how to deal with it, but since (at least at the time of writing this post) it does not support AARM architecture, I show you also how to deal with Parallels.
Create a Virtual Machine and set up it as you wish - for example using a STIG compliant partition layout.
Once the Virtual Machine is up and running, log in to it and go on as described in the "Preparing The Vagrant Box" paragraph.
Preparing The Vagrant Box
Login to the VM's console as root user.
If you haven't already created, create the Vagrant user:
adduser vagrant
and of course assign it a password:
passwd vagrant
then create the "/etc/sudoers.d/vagrant" sudo include file with the following contents:
vagrant ALL=(ALL) NOPASSWD:ALL
become the vagrant user:
sudo -u vagrant
and verify the sudo configuration:
sudo -l
since vagrant has administrative rights, we can use it to run the statement to update the RPM packages installed into the VM - if it is a dnf based Red Hat derived distro type:
sudo dnf update -y
otherwise, if it is an old yum based Red Hat derived distro, simply replace with "yum" the word "dnf" in the previous and in every next statement.
As soon as the update process completes, shutdown the Virtual Machine:
sudo shutdown -h now
After restart, SSH connect to the VM as the vagrant user.
Once logged in, authorize SSH access using the insecure vagrant public key:
sudo dnf install -y wget
mkdir -m 0700 .ssh
wget –no-check-certificate https://raw.github.com/mitchellh/vagrant/master/keys/vagrant.pub -O .ssh/authorized_keys
chmod 0600 .ssh/authorized_keys
Dont' worry for the insecure key - it is automatically changed the first time you instantiate the box.
We need now to attach the ISO file of the DVD containing the source files to build the VirtualBox Guest Additions: attaching it using the GUI is very simple, so I will show you how to do the same using the command line.
First we need to get the list of the configured VMs, so to guess the name or UUID of the VM Vagrant has just deployed:
VBoxManage.exe list vms
on my computer the output is as follows:
"ol92_default_1623083202261_27402" {c385608d-76f3-402b-ac11-cf57c49c8ec2}
if you are instead using Parallels, just type:
prlctl list -a
on my computer the output is as follows:
{d0515d56-c34f-4829-a6e8-3fa3c5411f23} running - ol92
we can now attach the ISO containing the guest additions DVD - the example command below is for VirtualBox running on Windows - please adjust the path to the ISO image to reflect the actual path using the syntax specific to your virtualization host.
VBoxManage.exe storageattach "ol92_default_1623083202261_27402" --storagectl IDE --port 0 --device 1 --type dvddrive --medium "C:\Program Files\Oracle\VirtualBox\VBoxGuestAdditions.iso"
of course, also Parallels have a tools DVD - the statement to attach it is as follows:
prlctl set ol92 --device-set cdrom0 --image /Applications/Parallels\ Desktop.app/Contents/Resources/Tools/prl-tools-lin-arm.iso
in the VM, install the EPEL repository as follows:
sudo dnf install epel-release
we must now install all of the packages required to compile the VirtualBox Guest additions (or the Parallels tools) Kernel modules:
sudo dnf install -y gcc make perl kernel-devel kernel-headers bzip2 dkms
Now we must mount the CD-ROM onto the "/mnt" directory as follows:
sudo mount /dev/cdrom /mnt
cd /mnt
We are now ready to compile and install the kernel module - if you are using VirtualBox just type:
sudo ./VBoxLinuxAdditions.run
if instead you are using Parallels, type the following command:
sudo ./install
the output with VirtualBox is as follows:
Verifying archive integrity... All good.
Uncompressing VirtualBox 6.1.22 Guest Additions for Linux........
VirtualBox Guest Additions installer
Copying additional installer modules ...
Installing additional modules ...
VirtualBox Guest Additions: Starting.
VirtualBox Guest Additions: Building the VirtualBox Guest Additions kernel
modules. This may take a while.
VirtualBox Guest Additions: To build modules for other installed kernels, run
VirtualBox Guest Additions: /sbin/rcvboxadd quicksetup
VirtualBox Guest Additions: or
VirtualBox Guest Additions: /sbin/rcvboxadd quicksetup all
VirtualBox Guest Additions: Building the modules for kernel
4.18.0-305.3.1.el8.x86_64.
wait until it completes.
Now we must cleanup the VM as follows:
sudo rm -rf /var/cache/dnf /etc/sshd/*key*
sudo truncate -s0 ~/.bash_history /var/log/messages /var/log/secure /var/log/cron /var/log/dnf.*
truncate -s0 ~/.bash_history
we have completed the procedure. Let's shutdown the VM:
sudo shutdown -h now
Now we must package the VM into a VagrantBox: with VirtualBox is really easy - in this example we package it within the "centos8.box" file:
vagrant package --output centos8.box
With Parallels is a little bit more cumbersome, since you cannot get the Vagrant Box automatically created.
First shrink the size of the disk as much as possible:
prl_disk_tool compact --hdd /Users/mcarcano/Parallels/ol92.pvm/ol92-0.hdd
In this example the name of the VM is "ol92".
It is then necessary to create a staging directory on the virtualisation host:
mkdir ~/staging
cd ~/staging
In this directory, create the "metadata.json" file with the following contents:
{ "provider": "parallels" }
And create the default Vagrant that will be run generated when typing "vagrant init": the contents must be something as follows:
# -*- mode: ruby -*-
# vi: set ft=ruby :
Vagrant.require_version ">= 1.6.2"
Vagrant.configure("2") do |config|
config.vm.define "vagrant-ol92"
config.vm.box = "grimoire/ol92"
config.vm.communicator = "ssh"
config.ssh.username = "vagrant"
config.vm.guest = :redhat
end
EOF
copy the VM directory into the staging directory:
cp -pr ~/Parallels/ol92.pvm .
so in the end of the day the "staging" directory must contain:
- the "ol92" directory with the VM
- the "metadata.json" file
- the "Vagrant" file
We can now package everything into a VagrantBox for the Parallels provider:
tar cvzf ../ol92.box ol92.pvm Vagrant metadata.json
This completes the procedure to create the Vagrant Box for the Parallels provider.
You can remove the staging directory:
cd ..
rm -rf staging
We can now add this box to the list of installed boxes as follows:
vagrant box add grimoire/ol92 ol92.box
Vagrantfile
Here are some example Vagrantfiles.
X86_64 VM on a M1
As we saw, running X86_64 VM on M1 requires both QEMU as hypervisor along with the "vagrant-qemu" provider.
Here the minimal Vagrantfile for spinning up an Ubuntu 18.04 x86_64 VM to run on an ARM processor:
Vagrant.configure("2") do |config|
config.vm.box = "generic/ubuntu1804"
config.vm.provider "qemu" do |qe|
qe.arch = "x86_64"
qe.machine = "q35"
qe.cpu = "max"
qe.net_device = "virtio-net-pci"
qe.memory = "8G"
qe.smp = "cpus=10,sockets=1,cores=10,threads=1"
end
config.vm.synced_folder '.', '/vagrant', disabled: true
end
More recent images are provided with UEFI boot: this is a problem with QEMU, since it uses BIOS (SeaBIOS).
If you are dealing with such an image the VM won't boot and will remain stuck on the boot screen, since it is unable to find a bootable device.
Luckily we can use the UEFI boot provided by the OVMF project: the Open Virtual Machine Firmware, OVMF, is an EDK II based project to enable UEFI support for Virtual Machines: it contains sample UEFI firmware compatible with both QEMU and KVM. If you want to learn more this is the link to the OVMF - FAQ.
Despite starting from QEMU version 1.6 (December 2013) the OVMF project is pre-installed in QEMU, package managers do not provide it - this means we must manually install it.
Anyway, even when using Mac OS, we can easily lend some files from Debian's Open Virtual Machine's Firmware (OVMF) package.
Once found the DEB package using the above link, download it as follows:
curl -O http://ftp.us.debian.org/debian/pool/main/e/edk2/ovmf_2022.11-6+deb12u1_all.deb
once downloaded, unpack the deb package using the "ar" command line utility:
ar -x ovmf_2022.11-6+deb12u1_all.deb
and extract the contents of the "usr/share/OVMF" directory:
tar xJf data.tar.xz usr/share/OVMF
lastly move it to the "~/Vagrant" directory
mv usr/share/OVMF ~/Vagrant
now that we have the boot code, just pass it to QEMU by adding the "qe.extra_quem_args" directive with the "-bios" option followed by the path to the "OVMF_CODE.fd" file.
For example:
vm.extra_qemu_args = %w(-bios /Users/mcarcano/Vagrant/OVMF/OVMF_CODE.fd)
Vagrantfile For Multiple VM
Mind it is not necessary to have a Vagrantfile for each single VM you want to deploy.
Below you find a quite complex Vagrantfile that deliver the VM described in the "host_vms" array:
# -*- mode: ruby -*-
# vi: set ft=ruby :
$wipe_network_interface = <<-'SCRIPT'
DEV=$(ip -4 addr | grep -v "altname" | grep ${1} -B 1 | head -n 1 | awk '{print $2}' |tr -d :)
[[ -z "${DEV}" ]] && exit 0
CONN_NAME=$(nmcli -t -f NAME,DEVICE con | grep "${DEV}" | grep -v "br-" | cut -d : -f 1)
nmcli conn modify "${CONN_NAME}" ipv4.method auto
nmcli conn modify "${CONN_NAME}" ipv4.address ''
nmcli conn modify "${CONN_NAME}" ipv4.method disabled
nmcli conn modify "${CONN_NAME}" autoconnect no
SCRIPT
$setup_services_interface = <<-'SCRIPT'
DEV=$(ip -4 addr | grep -v "altname" | grep ${1} -B 1 | head -n 1 | awk '{print $2}' |tr -d :)
VLAN=$2
CONN_UUID=$(nmcli -t -f UUID,DEVICE con | grep "${DEV}" | grep -v "\.${VLAN}" | cut -d : -f 1)
nmcli conn delete "${CONN_UUID}"
CONN_UUID=$(nmcli -t -f UUID,DEVICE con | grep "${DEV}" | grep -v "\.${VLAN}" | cut -d : -f 1)
[[ -n "${CONN_UUID}" ]] && nmcli conn delete "${CONN_UUID}"
[[ $(nmcli -t -f NAME conn | grep ${DEV}-vlan.${VLAN}) ]] && exit 0
IP=$3
MASK=$4
ROUTES=$5
nmcli con add type vlan con-name ${DEV}-vlan.${VLAN} ifname ${DEV}.${VLAN} dev ${DEV} id ${VLAN} ip4 ${IP}/${MASK}
nmcli con mod "${DEV}-vlan.${VLAN}" +ipv4.routes "${ROUTES}"
nmcli con up "${DEV}-vlan.${VLAN}"
[[ "$(systemctl is-enabled firewalld)" == "enabled" ]] || systemctl enable firewalld
[[ "$(systemctl is-active firewalld)" == "active" ]] || systemctl start firewalld
firewall-cmd --permanent --new-zone=services1
firewall-cmd --permanent --zone=services1 --add-interface=${DEV}.${VLAN}
firewall-cmd --reload
SCRIPT
$install_postgresql_client = <<-'SCRIPT'
ARCH=$(uname -i)
dnf install -y https://download.postgresql.org/pub/repos/yum/reporpms/EL-9-${ARCH}/pgdg-redhat-repo-latest.noarch.rpm
dnf install -y net-tools postgresql16
SCRIPT
$install_postgresql_server = <<-'SCRIPT'
ARCH=$(uname -i)
dnf install -y https://download.postgresql.org/pub/repos/yum/reporpms/EL-9-${ARCH}/pgdg-redhat-repo-latest.noarch.rpm
dnf install -y net-tools postgresql16 postgresql16-server
firewall-cmd --permanent --zone=services1 --add-service postgresql
firewall-cmd --reload
/usr/pgsql-16/bin/postgresql-16-setup initdb
echo "listen_addresses = '*'" >> /var/lib/pgsql/16/data/postgresql.conf
echo "host all all 192.168.0.0/24 scram-sha-256" >> /var/lib/pgsql/16/data/pg_hba.conf
systemctl enable --now postgresql-16
sudo -u postgres psql -c "alter user postgres with password 'grimoire'"
SCRIPT
$reboot = <<-'SCRIPT'
sudo shutdown -r now
SCRIPT
VAGRANTFILE_API_VERSION = "2"
Vagrant.require_version ">= 1.5"
host_vms=[
{
:hostname => "gw-ca-ut1a001",
:domain => "netdevs.carcano.local",
:infra_net_ip => "172.16.0.11",
:core_net_temporary_ip => "192.168.253.253",
:box => "grimoire/ol92",
:ram => 2048,
:cpu => 2,
:service_class => "netdev"
},
{
:hostname => "gw-ca-up1a001",
:domain => "netdevs.carcano.local",
:infra_net_ip => "172.16.0.12",
:core_net_temporary_ip => "192.168.254.254",
:box => "grimoire/ol92",
:ram => 2048,
:cpu => 2,
:service_class => "netdev"
},
{
:hostname => "as-ca-ut1a001",
:domain => "netdevs.carcano.local",
:core_net_temporary_ip => "192.168.253.11",
:services_net_ip => "192.168.0.10",
:services_net_mask => "24",
:services_net_vlan => "100",
:summary_route => "192.168.0.0/16 192.168.0.254",
:box => "grimoire/ol92",
:ram => 2048,
:cpu => 2,
:service_class => "ws"
},
{
:hostname => "pg-ca-ut1a001",
:domain => "netdevs.carcano.local",
:core_net_temporary_ip => "192.168.253.12",
:services_net_ip => "192.168.6.10",
:services_net_mask => "24",
:services_net_vlan => "101",
:summary_route => "192.168.0.0/16 192.168.6.254",
:box => "grimoire/ol92",
:ram => 2048,
:cpu => 2,
:service_class => "postgresql"
}
]
Vagrant.configure(VAGRANTFILE_API_VERSION) do |config|
host_vms.each do |machine|
config.vm.define machine[:hostname] do |node |
node.vm.box = machine[:box]
node.vm.hostname = "#{machine[:hostname]}.#{machine[:domain]}"
node.vm.network "private_network", ip: machine[:core_net_temporary_ip]
if machine[:service_class] == "netdev"
node.vm.network "private_network", ip: machine[:infra_net_ip]
end
node.vm.provider :virtualbox do |vm|
vm.name = "grimoire_#{machine[:hostname]}"
vm.cpus = machine[:cpu]
vm.customize [ "modifyvm", :id, "--memory", machine[:ram] ]
if machine[:service_class] == "netdev"
vm.customize [ "modifyvm", :id, "--nicpromisc2", "allow-vms" ]
end
end
node.vm.provider :parallels do |vm|
vm.name = "grimoire_#{machine[:hostname]}"
vm.memory = machine[:ram]
vm.cpus = machine[:cpu]
vm.update_guest_tools = false
vm.optimize_power_consumption = false
end
if machine[:service_class] == "netdev"
node.vm.provision :shell, :args => machine[:core_net_temporary_ip], inline: $wipe_network_interface, run: "always"
node.vm.provision :shell, inline: $reboot, run: "always"
end
if machine[:service_class] == "ws"
node.vm.provision :shell, :args => [ machine[:core_net_temporary_ip], machine[:services_net_vlan], machine[:services_net_ip], machine[:services_net_mask], machine[:summary_route] ], inline: $setup_services_interface, run: "always"
node.vm.provision :shell, inline: $install_postgresql_client
end
if machine[:service_class] == "postgresql"
node.vm.provision :shell, :args => [ machine[:core_net_temporary_ip], machine[:services_net_vlan], machine[:services_net_ip], machine[:services_net_mask], machine[:summary_route] ], inline: $setup_services_interface, run: "always"
node.vm.provision :shell, inline: $install_postgresql_server
end
end
end
end
Operating Vagrant
Vagrant let you provision Virtual Machines in a very convenient and easy way: these are the very basic Vagrant commands you need to know to operate it:
install a given vagrant plugin. For example
vagrant plugin install vagrant-scp
update an already installed vagrant plugin. For example:
vagrant plugin update vagrant-scp
removes an already installed vagrant plugin. For example:
vagrant plugin uninstall vagrant-scp
download a box – remember that boxes are specific for a given provider, so you should specify it too. For example:
vagrant box add --provider virtualbox centos/8
delete a box – For example:
vagrant box remove centos/6
update an already downloaded box. For example:
vagrant box update --box centos/8
If no VM name is specified as argument the action is performed for every VM defined in the vagrantfile. For example, to limit the command to jump-ci-upa002 VM only:
vagrant up jump-ci-upa002
shut down VM. If no VM name is specified as argument the action is performed for every VM defined in the vagrantfile. For example to limit the shut down to jump-ci-upa002 VM only:
vagrant halt jump-ci-upa002
Suspend VM. If no VM name is specified as argument the action is performed for every VM defined in the vagrantfile. For example to limit the suspend to jump-ci-upa002 VM only:
vagrant suspend jump-ci-upa002
resume VM. If no VM name is specified as argument the action is performed for every VM defined in the vagrantfile. For example to limit the resume to jump-ci-upa002 VM only:
vagrant resume jump-ci-upa002
Destroy (delete) VM. If no VM name is specified as argument the action is performed for every VM defined in the vagrantfile. For example to limit the destroy to jump-ci-upa002 VM only:
vagrant destroy jump-ci-upa002
list available snapshots for a given VM. For example
vagrant snapshot list jump-ci-up3a001
create a snapshot for a given VM. For example
vagrant snapshot push jump-ci-up3a001
Revert a VM to the specified snapshot. For example:
vagrant snapshot restore jump-ci-up3a001 push_1607852325_7783
Delete the specified snapshot for the given VM:
snapshot delete jump-ci-up3a001 push_1607852325_7783
Troubleshooting Vagrant
Of course it may happen to stumble into errors of various kind while running Vagrant: this if course is most likely to happen when writing Vagrantfiles, but since Vagrant providers rely on the command line of the Hypervisor you are using, weird things may happen also an operating system update (or Hypervisor update).
The most straightforward workaround of errors after an Hypervisor upgrade is upgrading Vagrant too, since it is very likely the most current Vagrant version is compatible with the latest version of your Hypervisor.
If errors appear after an operating system upgrade, or you are developing a custom Vagrantfile and run into errors, troubleshooting with the very few messages printed by Vagrant is certainly very difficult.
Luckily you can very easily change the severity of the messages by setting the "VAGRANT_LOG" variable while launching the "vagrant" statement.
For example:
VAGRANT_LOG=debug vagrant up jump-ci-up3a001
Footnotes
Vagrant is certainly a great tool that every professional should be skilled on, since it is dramatically useful to quickly mock-up things. This post is just an operating cheat-sheet: to really exploit the power of Vagrant you should be able to develop vagrantfiles, but this will be the topic of another post.
1 thought on “Vagrant – Installing and operating”