Skip to content

Latest commit

 

History

History
209 lines (158 loc) · 6.14 KB

README.md

File metadata and controls

209 lines (158 loc) · 6.14 KB

Ansible tutorial: templates

We'll use the haproxy as loadbalancer. Of course, install is just like we did for apache. But now configuration is a bit more tricky since we need to list all web servers in haproxy's configuration. How can we do that?

HAProxy configuration template

Ansible uses Jinja2, a templating engine for Python. When you write Jinja2 templates, you can use any variable defined by Ansible.

For instance, if you want to output the inventory_name of the host the template is currently built for, you just can write {{ inventory_hostname }} in the Jinja template.

Or if you need the IP of the first ethernet interface (which ansible knows thanks to the setup module), you just write: {{ ansible_default_ipv4.address}} (which is equivalent to {{ ansible_default_ipv4['address'] }}). in your template.

Jinja2 templates also support conditionals, for-loops, etc...

Let's make a templates/ directory and create a Jinja template inside. We'll call it haproxy.cfg.j2. We use the .j2 extension by convention, to make it obvious that this is a Jinja2 template, but this is not necessary.

global
    daemon
    maxconn 256

defaults
    mode http
    timeout connect 5000ms
    timeout client 50000ms
    timeout server 50000ms

listen cluster
    bind {{ ansible_all_ipv4_addresses.1 }}:80
    mode http
    stats enable
    balance roundrobin
{% for backend in groups['web'] %}
    server {{ hostvars[backend]['ansible_hostname'] }} {{ hostvars[backend].ansible_all_ipv4_addresses.1 }} check port 80
{% endfor %}
    option httpchk HEAD /index.php HTTP/1.0

We have many new things going on here.

First, {{ ansible_all_ipv4_addresses.1 }} will be replaced by the 2nd IP of the server, which happens to be 192.168.33.10.

Then, we have a loop. This loop is used to build the backend servers list. It will loop over every host listed in the [web] group (and put this host in the backend variable). For each of the hosts it will render a line using host's facts. All hosts' facts are exposed in the hostvars variable, so it's easy to access another host variables (like its hostname or in this case IP).

We could have written the host list by hand, since we have only 2 of them. But we're hoping that the server will be very successful, and that we'll need a hundred of them. Thus, adding servers to the configuration or swapping some out boils down to adding or removing hosts from the [web] group.

HAProxy playbook

We've done the most difficult part of the job. Writing a playbook to install and configure HAproxy is a breeze:

- hosts: web
  gather_facts: true

- hosts: haproxy
  tasks:
    - name: Installs haproxy load balancer
      apt:
        pkg: haproxy
        state: present
        update_cache: yes

    - name: Pushes configuration
      template:
        src: templates/haproxy.cfg.j2
        dest: /etc/haproxy/haproxy.cfg
        mode: 0640
        owner: root
        group: root
      notify:
        - restart haproxy

    - name: Sets default starting flag to 1
      lineinfile:
        dest: /etc/default/haproxy
        regexp: "^ENABLED"
        line: "ENABLED=1"
      notify:
        - restart haproxy

  handlers:
    - name: restart haproxy
      service:
        name: haproxy
        state: restarted

Looks familiar, isn't it? The only new module here is template, which has the same arguments as copy. We also restrict this playbook to the group haproxy.

And now... let's try this out. Since our inventory contains only hosts necessary for the cluster, we don't need to limit the host list and can even run both playbooks. Well, to tell the truth, we must run both of them at the same time, since the haproxy playbook requires facts from the two webservers. In step-11 we'll show how to avoid this.

$ ansible-playbook -i step-10/hosts step-10/apache.yml step-10/haproxy.yml

PLAY [web] *********************

GATHERING FACTS *********************
ok: [host1]
ok: [host2]

TASK: [Updates apt cache] *********************
ok: [host1]
ok: [host2]

TASK: [Installs necessary packages] *********************
ok: [host1] => (item=apache2,libapache2-mod-php,git)
ok: [host2] => (item=apache2,libapache2-mod-php,git)

TASK: [Push future default virtual host configuration] *********************
ok: [host2]
ok: [host1]

TASK: [Activates our virtualhost] *********************
changed: [host1]
changed: [host2]

TASK: [Check that our config is valid] *********************
changed: [host1]
changed: [host2]

TASK: [Rolling back - Restoring old default virtualhost] *********************
skipping: [host1]
skipping: [host2]

TASK: [Rolling back - Removing out virtualhost] *********************
skipping: [host1]
skipping: [host2]

TASK: [Rolling back - Ending playbook] *********************
skipping: [host1]
skipping: [host2]

TASK: [Deploy our awesome application] *********************
ok: [host2]
ok: [host1]

TASK: [Deactivates the default virtualhost] *********************
changed: [host1]
changed: [host2]

TASK: [Deactivates the default ssl virtualhost] *********************
changed: [host2]
changed: [host1]

NOTIFIED: [restart apache] *********************
changed: [host2]
changed: [host1]

PLAY RECAP *********************
host1              : ok=10   changed=5    unreachable=0    failed=0
host2              : ok=10   changed=5    unreachable=0    failed=0



PLAY [haproxy] *********************

GATHERING FACTS *********************
ok: [host0]

TASK: [Installs haproxy load balancer] *********************
changed: [host0]

TASK: [Pushes configuration] *********************
changed: [host0]

TASK: [Sets default starting flag to 1] *********************
changed: [host0]

NOTIFIED: [restart haproxy] *********************
changed: [host0]

PLAY RECAP *********************
host0              : ok=5    changed=4    unreachable=0    failed=0

Looks good. Now head to http://192.168.33.10/ and see the result. Your cluster is deployed!

you can even peek at HAProxy's statistics at http://192.168.33.10/haproxy?stats.

Now on to the next chapter about "Variables again", in step-11.