Introduction
A good engineer can meticulously design a great infrastructure. However, what if you ask him to design the same thing multiple times. Well, it's a hell lot of boring & tedious task. Also, it will make the next iteration error prone. Now, here comes, smart engineering by automating the infrastructure. Let's learn to do it using chef and vagrant. I will try not to make this post too much theoretical as there are lots of information out there on web. So, let's start cooking !!!
Brief Intro of Chef and Vagrant
Chef is a open source infrastructure automation framework. Using this framework you can build your whole infrastructure. It can represent your infrastructure as code. Now, what is vargant, imagine you created some code using chef to build your infrastructure. To test your code, you will either need a machine or a VM. Suppose unfortunately your code fails. You will again need a new machine/VM or you will format the older one and you will try again. Seems uninteresting right ? Here comes vagrant to rescue you from these issues. Vagrant is a tool to create, configure and manage virtual environments easily. With the vagrant boxes you can test your chef code easily. Now, enough of theory, let's do some practical stuff.
Installing chef and vagrant
1. Install virtualbox to install VMs. You can download it from the site https://www.virtualbox.org/wiki/Downloads
2. Install vagrant. You can download it from the site http://www.vagrantup.com/downloads.html
3. Install git.
4. Install chef from http://www.getchef.com/chef/install/
Key Components of chef framework :
Some key components of chef which you should know before proceeding further. There are many important components but will focus only on two.
Cookbook : Cookbook is a collection of recipes which tells chef how to install something. Cookbooks to install apache, tomcat, mysql are already there. These cookbooks are very well written which also take care of underlying OS while installing. There are lots of other cookbooks, you can get those from http://community.opscode.com/cookbooks
You can also create you own cookbooks.
Recipe : It's just a ruby file containing bunch of ruby functions to do certain tasks (installation, configuration)
Now, let's try to install Apache for our demo app using chef. First let us set up the environment for this process.
Setting up environment :
1. Download chef-repo from git.
git clone https://github.com/opscode/chef-repo
2. Go into chef-repo directory and create a new directory .chef there. Inside .chef directory create a file called knife.rb. Add following line in knife.rb :
cookbook_path ['../cookbooks']
Above line will configure knife with chef framework and will tell knife where cookbooks are stored.
You will have to provide the path of cookbooks which in this case inside chef-repo directory.
3. Run command :
knife cookbook create myapp
knife cookbook create myapp
This will create you own cookbook with name myapp. This will also have all the directories to define other chef components.
.
|-- attributes
|-- CHANGELOG.md
|-- definitions
|-- files
| `-- default
|-- libraries
|-- metadata.rb
|-- providers
|-- README.md
|-- recipes
| `-- default.rb
|-- resources
`-- templates
`-- default
myapp directory structure will look like :
.
|-- attributes
|-- CHANGELOG.md
|-- definitions
|-- files
| `-- default
|-- libraries
|-- metadata.rb
|-- providers
|-- README.md
|-- recipes
| `-- default.rb
|-- resources
`-- templates
`-- default
Adding Apache cookbook and configuring with chef :
1. Go to 'myapp' directory and run following command :
knife cookbook site install apache2
knife cookbook site install apache2
This will download already created cookbook apache2. This will also download dependencies for apache, if any.
2. Open default.rb file inside myapp/recipes directory. This is the first file which will get executed by chef.
3. Add following snippet of code in default.rb file:
include_recipe "apache2"
apache_site "default" do
apache_site "default" do
enable true
end
As we discussed, cookbook can have various recipes. We are including desired ones in our own cookbook. So, at first line we are including apache2 recipe which comes with apache2 cookbook. On the next line, we are setting value of apache_site attribute to true. This will add a demo html file in apache document root directory. There could be various attributes which you can customize like server timeout, listening_port. You can check more configuration options on site https://github.com/onehealth-cookbooks/apache2.
Configuring vagrant to test your chef project :
1. Let's add vagrant base box to test our chef code. This box will act as a separate new machine for us. We don't have to go through any OS installation steps. You can think of it as a ready to use machine.
vagrant add MyCentOSBox http://www.lyricalsoftware.com/downloads/centos65.box
Above command will add a vagrant box with Cent-OS from given url and vagrant will identify it with name MyCentOSBox. You can also choose any other type of box from http://www.vagrantbox.es/.
Above command will also place a Vagrantfile in the current directory. This file will have following code which tells vagrant which is the default box.
Vagrant.configure("2") do |config|
config.vm.box = "MyCentOSBox"
config.vm.box_url = "http://www.lyricalsoftware.com/downloads/centos65.box"
end
2. To integrate chef with Vagrant for provisioning, open Vagrantfile and add following code.
Vagrant.configure("2") do |config|
config.vm.box = "MyCentOSBox"
config.vm.box_url = "http://www.lyricalsoftware.com/downloads/centos65.box"
config.vm.provision :chef_solo do |chef|
chef.cookbooks_path = "~/chef-repo/cookbooks/"
chef.add_recipe "myapp"
end
end
Above code will configured chef solo with vagrant. We are also providing cookbook path and recipe name which we want to execute by vagrant.
Running chef scripts through vagrant
1. Add port forwarding :
Apache will be install inside virtual machine, so we need some way to test our site through browser. Vagrant allows you to map a port on the your machine i.e. host machine to a port on the virtual machine, by forwarding all the traffic. So, let’s map port 8888 on localhost to port 80 on the virtual machine. You will just have to add following one line inside config code in Vagrantfile.
Vagrant.configure("2") do |config|
config.vm.box = "MyCentOSBox"
config.vm.box_url = "http://www.lyricalsoftware.com/downloads/centos65.box
config.vm.network :forwarded_port, guest: 80, host: 8888
config.vm.provision :chef_solo do |chef|
chef.cookbooks_path = "~/chef-repo/cookbooks/"
chef.add_recipe "myapp"
end
end
2. Now, we are all set to run our chef script through vagrant. Just run the following command. This will boot the vm and also do the provisioning.Vagrant.configure("2") do |config|
config.vm.box = "MyCentOSBox"
config.vm.box_url = "http://www.lyricalsoftware.com/downloads/centos65.box
config.vm.network :forwarded_port, guest: 80, host: 8888
config.vm.provision :chef_solo do |chef|
chef.cookbooks_path = "~/chef-repo/cookbooks/"
chef.add_recipe "myapp"
end
end
vagrant up
If you have already booted vm and just want to re-provision chef. Just run the following command.
vagrant provision
Now you can open http://localhost:8888/ on your machine's browser and you will see apache's default page.
3. To shutdown vm just run command
vagrant halt
4. To try again from scratch :
Just run following command :
vagrant destroy
and followed by command :
vagrant up
You can also login to vm using :
vagrant ssh
Thanks for reading. I hope, using this blog, you will be able to automate your project infrastructure. I would appreciate your comments and suggestions, if any.