Exercise 3.0 - An introduction to templating with Jinja2

Generally speaking, when one talks about network automation the focus is specifically around configuration management of devices. In this lab you will learn how to use Ansible as a tool to generate living, dynamic documentation.

This allows the ability to generate reports and documents, using the same information and can cater to the needs of a hands-on-keyboard network engineer to a manager who needs to understand the state of the network with a glance of a web-page!

Jinja2 is a powerful templating engine for Python. There is native integration of Jinja2 with Ansible. Jinja2 allows for manipulating variables and implementing logical constructs. In combination with the Ansible template module, the automation engineer has a powerful tool at their disposal to generate live or dynamic reports.

In this lab you will learn how to use the template module to pass collected data from devices to a Jinja2 template. The template module then renders the output as a markdown file.

Step 1

Create a new playbook called router_report.yml and add the following play definition to it:

---
- name: GENERATE OS REPORT FROM ROUTERS
  hosts: cisco
  connection: network_cli
  gather_facts: no

Step 2

Add a task that collects the facts using the ios_facts module. Recollect that we used this module in an earlier lab.

---
- name: GENERATE OS REPORT FROM ROUTERS
  hosts: cisco
  connection: network_cli
  gather_facts: no

  tasks:
    - name: GATHER ROUTER FACTS
      ios_facts:

Recall that the facts modules automatically populate the ansible_net_version and ansible_net_serial_number variables within the play. You can validate this by running the playbook in verbose mode.

Step 3

Rather than using debug or verbose mode to display the output on the screen, go ahead and add a new task using the template module as follows:


---
- name: GENERATE OS REPORT FROM ROUTERS
  hosts: cisco
  connection: network_cli
  gather_facts: no

  tasks:
    - name: GATHER ROUTER FACTS
      ios_facts:

    - name: ENSURE REPORTS FOLDER
      run_once: true
      file:
        name: reports
        state: directory

    - name: RENDER FACTS AS A REPORT
      template:
        src: os_report.j2
        dest: reports/{{ inventory_hostname }}.md

Let’s break this task down a bit. The template module has a src parameter that has a value of os_report.j2. In the next few steps, we will create this file. This will be the Jinja2 template, used to generate the desired report. The dest parameter specifies the destination file name to render the report into.

Step 4

The next step is to create a Jinja2 template. Ansible will look for the template file in the current working directory and within a directory called templates automatically. Convention/best-practice is to create the template file within the templates directory.

Using vi, nano or another text editor, go ahead and create the file called os_report.j2 under the templates directory. Add the following into the template file:



{{ inventory_hostname.upper() }}
---
{{ ansible_net_serialnum }} : {{ ansible_net_version }}



This file simply contains some of the variables we have been using in our playbooks until now.

Note: Python inbuilt methods for datatypes are available natively in Jinja2 making it very easy to manipulate the formatting etc.

Step 5

With this in place, go ahead and run the playbook:

[student1@ansible networking-workshop]$ ansible-playbook -i lab_inventory/hosts router_report.yml

PLAY [GENERATE OS REPORT FROM ROUTERS] ******************************************************************************************************************************************************

TASK [GATHER ROUTER FACTS] ******************************************************************************************************************************************************************
ok: [rtr4]
ok: [rtr3]
ok: [rtr2]
ok: [rtr1]

TASK [ENSURE REPORTS FOLDER] ********************************************************************************
changed: [rtr1]

TASK [RENDER FACTS AS A REPORT] *************************************************************************************************************************************************************
changed: [rtr4]
changed: [rtr2]
changed: [rtr3]
changed: [rtr1]

PLAY RECAP **********************************************************************************************************************************************************************************
rtr1                       : ok=2    changed=1    unreachable=0    failed=0   
rtr2                       : ok=2    changed=1    unreachable=0    failed=0   
rtr3                       : ok=2    changed=1    unreachable=0    failed=0   
rtr4                       : ok=2    changed=1    unreachable=0    failed=0   

[student1@ansible networking-workshop]$

Step 6

After the playbook run, you should see the following files appear in the reports directory:

reports/
├── rtr1.md
├── rtr2.md
├── rtr3.md
└── rtr4.md

0 directories, 4 files

The contents of one of them for example:

[student1@ansible networking-workshop]$ cat reports/rtr4.md


RTR4
---
9TCM27U9TQG : 16.08.01a

[student1@ansible networking-workshop]$

Step 7

While it is nice to have the data, it would be even better to consolidate all these individual router reports into a single document. Let’s add a new task to do that


---
- name: GENERATE OS REPORT FROM ROUTERS
  hosts: cisco
  connection: network_cli
  gather_facts: no

  tasks:
    - name: GATHER ROUTER FACTS
      ios_facts:

    - name: ENSURE REPORTS FOLDER
      run_once: true
      file:
        name: reports
        state: directory

    - name: RENDER FACTS AS A REPORT
      template:
        src: os_report.j2
        dest: reports/{{ inventory_hostname }}.md

    - name: CONSOLIDATE THE IOS DATA
      assemble:
        src: reports/
        dest: network_os_report.md
      delegate_to: localhost
      run_once: yes

Here we are using the assemble module. The src parameter specifies the directory that contain file fragments that need to be consolidated and the dest parameter provides the file to render the fragments into.

Note: The delegate_to can be used to specify tasks that need to be executed locally. The run_once directive will ensure that the given task is executed only once.

Step 8

Go ahead and run the playbook.

[student1@ansible networking-workshop]$ ansible-playbook -i lab_inventory/hosts router_report.yml

PLAY [GENERATE OS REPORT FROM ROUTERS] **********************************************************************

TASK [GATHER ROUTER FACTS] **********************************************************************************
ok: [rtr4]
ok: [rtr3]
ok: [rtr1]
ok: [rtr2]

TASK [ENSURE REPORTS FOLDER] ********************************************************************************
changed: [rtr1]

TASK [RENDER FACTS AS A REPORT] *****************************************************************************
changed: [rtr2]
changed: [rtr1]
changed: [rtr4]
changed: [rtr3]

TASK [CONSOLIDATE THE IOS DATA] *****************************************************************************
changed: [rtr1 -> localhost]

PLAY RECAP **************************************************************************************************
rtr1                       : ok=4    changed=3    unreachable=0    failed=0   
rtr2                       : ok=2    changed=1    unreachable=0    failed=0   
rtr3                       : ok=2    changed=1    unreachable=0    failed=0   
rtr4                       : ok=2    changed=1    unreachable=0    failed=0   

[student1@ansible networking-workshop]$

Step 9

A new file called network_os_report.md will now be available in the playbook root. Use the cat command to view it’s contents:

[student1@ansible networking-workshop]$ cat network_os_report.md


RTR1
---
9YJXS2VD3Q7 : 16.08.01a



RTR2
---
9QHUCH0VZI9 : 16.08.01a



RTR3
---
9ZGJ5B1DL14 : 16.08.01a



RTR4
---
9TCM27U9TQG : 16.08.01a

[student1@ansible networking-workshop]$

Note: Markdown files can be rendered visually as HTML

At this point, with 3 small tasks, you have an OS report on all the IOS devices in your network. This is a simple example but the principle remains as you expand upon the capabilities. For example, you can build status reports and dashboards that rely on the output of device show commands.

Ansible provides the tools and methods to extend network automation beyond configuration management to more robust capabilities, such as, generating documentation and or reports.

Complete

You have completed lab exercise 3.0


Click Here to return to the Ansible Linklight - Networking Workshop