Netmiko Tutorial for beginners :Next step to Network Automation

What is Netmiko

Netmiko is a multi-vendor SSH Python library which makes connecting to network devices via SSH really easy and straightforward. The library simplifies SSH management to network devices is based on the Paramiko SSH library.

As per the documentation, the purposes of this library are the following:

  • Successfully establish an SSH connection to the device
  • Simplify the execution of show commands and the retrieval of output data
  • Simplify execution of configuration commands including possibly commit actions
  • Do the above across a broad set of networking vendors and platforms

Why Netmiko

The question is: if we have Paramiko, why would be need something like Netmiko. As it turns out, Paramiko can sometimes be quite hard to work with. Netmiko is hiding many details of common devive communication. It uses Paramiko for the underlying SSH connectivity, but provides a higher abstraction for the communication with networking devices.

From the creator of Netmiko Kirk Byers:

At the time, I had observed that many individuals encountered similar issues with Python-SSH and network devices.

In this post, we have largely used similar use cases as the ones we described in the Paramiko related posts.

Installing Netmiko


(venv) labdevice:Netmiko_Introduction wauterw$ pip3 install netmiko
Collecting netmiko
***Truncated***
Installing collected packages: pycparser, cffi, pynacl, bcrypt, cryptography, paramiko, scp, pyserial, future, textfsm, netmiko
  Running setup.py install for future ... done
Successfully installed bcrypt-3.1.7 cffi-1.14.0 cryptography-2.8 future-0.18.2 netmiko-3.1.0 paramiko-2.7.1 pycparser-2.20 pynacl-1.3.0 pyserial-3.4 scp-0.13.2 textfsm-1.1.0

Note about equipment

For all the examples, we will use a Cisco sandbox environment delivered by Cisco Devnet. Go check out Devnet, really brilliant. To get a list of all sandboxes, check out this link. For this tutorial, I’m using the IOS XE sandbox (see here) and the IOS XR sandbox (see here).

Disclaimer: this is not production-grade code obviously. One should never store the username and password in the clear, not in the source code itself. The examples in the post are merely conceptual and for informational purposes.

Use case: get device prompt

Netmiko allows us to see the device prompt.

from netmiko import Netmiko

device = {
    "device_type": "cisco_xe",
    "ip": "ios-xe-mgmt-latest.cisco.com",
    "username": "developer",
    "password": "C1sco12345",
    "port": "8181",
}

net_connect = Netmiko(**device)
print(net_connect.find_prompt())

Let’s execute this very basic example:

(base) WAUTERW-M-65P7:test wauterw$ python3 get_prompt.py
csr1000v-1#

You’ll notice that we are in enable mode. The following example will show how to issue the disable command while printing out the prompt. Next, we will issue the enable command again and print out the prompt again.

from netmiko import Netmiko

device = {
    "device_type": "cisco_xe",
    "ip": "ios-xe-mgmt-latest.cisco.com",
    "username": "developer",
    "password": "C1sco12345",
    "secret": "C1sco12345",
    "port": "8181",
}

net_connect = Netmiko(**device)
print(f"Default prompt: {net_connect.find_prompt()}")

net_connect.send_command_timing("disable")
print(f"Disable command: {net_connect.find_prompt()}")

net_connect.enable()
print(f"Enable command: {net_connect.find_prompt()}")

Note that in order for the enable() method to work, you need to pass the secret key in the device information. If not, you will receive an error.

(base) WAUTERW-M-65P7:test wauterw$ python3 get_prompt.py
Default prompt: csr1000v-1#
Disable command: csr1000v-1>
Enable command: csr1000v-1#

Use Case: finding all supported devices

In the above example, we were using a "device_type": "cisco_xe". How do we know what device type to use and how do we know which devices are supported. Probably the easiest (albeit a little strange) method is to specify a wrong device type. In the example above, replace the following snippet:

device = {
    "device_type": "cisco_xe",
    ...}

with

device = {
    "device_type": "foobar",
    ...}

Executing this script will return a list of supported device types:

➜  Netmiko_Introduction git:(master) ✗ python3 get_prompt.py
Traceback (most recent call last):
  File "get_prompt.py", line 11, in <module>
    net_connect = Netmiko(**device)
  File "/usr/local/lib/python3.7/site-packages/netmiko/ssh_dispatcher.py", line 243, in ConnectHandler
    "currently supported platforms are: {}".format(platforms_str)
ValueError: Unsupported device_type: currently supported platforms are: 
a10
accedian
alcatel_aos
alcatel_sros
apresia_aeos
arista_eos
aruba_os
avaya_ers
avaya_vsp
brocade_fastiron
brocade_netiron
brocade_nos
brocade_vdx
brocade_vyos
calix_b6
checkpoint_gaia
ciena_saos
cisco_asa
cisco_ios
cisco_nxos
cisco_s300
cisco_tp
cisco_wlc
cisco_xe
cisco_xr
cloudgenix_ion
...
...

Use case: retrieve device uptimes

Let’s continue with another easy example: we will write a Python script that interacts with an IOS XE device and returns the uptime of the router.

We store both devices (IOS XE and IOS XR) in a list of dictionaries (key:value pairs). It allows up to loop over the devices list and execute the command for each device.

Within the for-loop, we first make a connection via Netmiko’s connection method (called Netmiko). This is also the reason why we import the Netmiko library at the top of the script. See here an example from Kirk Byers.

Executing commands is really easy. All we need to do is pass the command we want to execute to the send_command method and capture the output.

from netmiko import Netmiko

devices = [{
    "device_type": "cisco_xr",
    "ip": "sbx-iosxr-mgmt.cisco.com",
    "username": "***",
    "password": "***",
    "port": "8181",
}, {
    "device_type": "cisco_xe",
    "ip": "ios-xe-mgmt-latest.cisco.com",
    "username": "***",
    "password": "***",
    "port": "8181",
}]

for device in devices:
    net_connect = Netmiko(**device)
    output = net_connect.send_command("show version")
    net_connect.disconnect()
    result = output.find('uptime is')
    begin = int(result)
    end = begin + 38
    print(device['ip'] + " => " + output[int(begin):int(end)])

In the Github repo, the above script can be found here. Another simple and straightforward example can be found here. It’s a small variant on the above script and retrieves an overview of all interfaces on both devices.

Use case: Get interface description

In this example, we will retrieve the desciption of the various interfaces on both a Cisco XR and XE device. We will be using a slightly different method compared to above example. In below example, we will not create a list of devices, but we just define each device seperately. I’m just doing this to show the differences but I do prefer the list approach from previous example.

from netmiko import ConnectHandler

cisco_xr = {
    "device_type": "cisco_xr",
    "ip": "sbx-iosxr-mgmt.cisco.com",
    "username": "admin",
    "password": "C1sco12345",
    "port": "8181",
}

cisco_xe = {
    "device_type": "cisco_xe",
    "ip": "ios-xe-mgmt-latest.cisco.com",
    "username": "developer",
    "password": "C1sco12345",
    "port": "8181",
}

for device in (cisco_xr, cisco_xe):
   net_connect = ConnectHandler(**device)
   
   output = net_connect.send_command("show interface description")
   net_connect.disconnect()
   print("-"*100)
   print(output)
   print("-"*100)

Executing this script, will - as expected- show the list of interfaces with their description:

(base) WAUTERW-M-65P7:test wauterw$ python3 show_interface_description.py 
----------------------------------------------------------------------------------------------------

Mon Jun  8 19:39:04.125 UTC

Interface          Status      Protocol    Description
--------------------------------------------------------------------------------
Lo40               up          up          
Lo41               up          up          
Lo42               up          up          
Lo43               up          up          
Lo44               up          up          
Lo100              up          up          ***MERGE LOOPBACK 100****
Lo200              up          up          ***MERGE LOOPBACK 200****
Lo300              up          up          
Nu0                up          up          
Mg0/RP0/CPU0/0     up          up          
Gi0/0/0/0          admin-down  admin-down  
Gi0/0/0/1          admin-down  admin-down  
Gi0/0/0/2          admin-down  admin-down  
Gi0/0/0/3          admin-down  admin-down  
Gi0/0/0/4          admin-down  admin-down  
Gi0/0/0/5          admin-down  admin-down  
Gi0/0/0/6          admin-down  admin-down  

----------------------------------------------------------------------------------------------------
----------------------------------------------------------------------------------------------------
Interface                      Status         Protocol Description
Gi1                            up             up       Changed from Netconf
Gi2                            up             up       Changed from Netconf
Gi3                            admin down     down     Network Interface
Lo901                          up             up       - config by ansible
Lo1012                         up             up       
Lo1013                         up             up       
Lo1019                         up             up       
-----------------------------------------------------------------

Important note: You will see that in this example I’m using the ConnectHandler class instead of the Netmiko class. This is no mistake. In fact, they are the same classes. This means that we can also use the ConnectHandler. Differences are:

  • the net_connect = Netmiko(**device) would become net_connect = ConnectHandler(**device)
  • from netmiko import Netmiko would become from netmiko import ConnectHandler.

In most examples you will see me use the Netmiko class though, I just wanted to point out one could also use the ConnectHandler class.

Use case: parsing with TextFSM

In this example, we will use textFSM to retrieve the output in a form that makes it easier to parse.

First though, to stress the value of TextFSM later on, we will simply print out the type of the output (of the “show ip interface brief” command)

from netmiko import Netmiko

cisco_xr = {
    "device_type": "cisco_xr",
    "ip": "sbx-iosxr-mgmt.cisco.com",
    "username": "admin",
    "password": "C1sco12345",
    "port": "8181",
}

net_connect = Netmiko(**cisco_xr)
   
output = net_connect.send_command("show ip interface brief")
net_connect.disconnect()
print(type(output))

You will see we get back the following:

(base) WAUTERW-M-65P7:test wauterw$ python3 textFSM.py
<class 'str'>

The output is of type string. This is nice, but it’s rather unpractical to parse this output. What if we could receive the output as a data structure, such as a list? This is possible with . TextFSM is a module that implements a template based state machine for parsing semi-formatted text (see here). A large collection of TextFSM templates for quite some network vendors can be found here.

Therefore, in your current directory you need to clone this repository and also set the NET_TEXTFSM environment variable.

(base) WAUTERW-M-65P7:test wauterw$ git clone https://github.com/networktocode/ntc-templates
Cloning into 'ntc-templates'...
<Truncated>
(base) WAUTERW-M-65P7:test wauterw$ export NET_TEXTFSM=<current_foler>/Netmiko_Introduction/ntc-templates/templates

Next, you will need to replace the following snippet:

net_connect = Netmiko(**cisco_xr)
   
output = net_connect.send_command("show ip interface brief")
net_connect.disconnect()
print(type(output))

with the following code:

net_connect = Netmiko(**cisco_xr)
   
output = net_connect.send_command("show ip interface brief", use_textfsm=True)
net_connect.disconnect()
print(type(output))

for interface in output:
    print(interface['intf'])

Let’s now run this code:

 (base) WAUTERW-M-65P7:test wauterw$ python3 textFSM.py
 <class 'list'>
Loopback40
Loopback41
Loopback42
Loopback43
Loopback44
Loopback100
Loopback200
Loopback300
MgmtEth0/RP0/CPU0/0
GigabitEthernet0/0/0/0
GigabitEthernet0/0/0/1
GigabitEthernet0/0/0/2
GigabitEthernet0/0/0/3
GigabitEthernet0/0/0/4
GigabitEthernet0/0/0/5
GigabitEthernet0/0/0/6

As you will see, we get back a list now, which we can easily iterate through to print for example the interface names. Pretty powerful if you ask me.

This particular example can be found here. If you decide to follow along, please note I did not upload the templats so you will need to do the git clone from above again on your own environment.

Use case: Set interface description

In the previous use case, we have shown an example to execute some simple “show” commands, in other words to read information from the device. In this use case, we want also to give an example on how to write information to the devices.

We will essentially change the interface description of a particular port. It follows the same approach as the previous example. We store the devices in a list of dictionaries and we loop over them in a for-loop. Inside the for-loop, we use the send_config_set method (see documentation here). At a minimum, we need to pass the config_commands, which is an iterable contain all of the configuration commands. Hence, we have created a list called description_config which contains a list of all the commands we want to execute (in order).

Important to understand is that NetMiko will already bring you in the conf t state. Next, we will go into ‘interface GigabitEthernet3’ and then set the description.

from netmiko import Netmiko

devices = [{
   "device_type": "cisco_xe",
   "ip": "ios-xe-mgmt-latest.cisco.com",
   "username": "***",
   "password": "***",
   "port": "8181",
}]

description = 'Description set with Netmiko'

description_config = [
    "interface GigabitEthernet3",
    f"description {description}",
]

for device in devices:
   net_connect = Netmiko(**device)
   output = net_connect.send_config_set(description_config)
   print(output)
   net_connect.disconnect()

Before executing this script, let’s check the interface descriptions (pay attention to ‘Gi3’) by just logging into the device using a terminal.

interface

Let’s now execute our Python script.

interface

And then let’s check again by logging into the device via our terminal. You will notice that indeed the interface description has been changed to the string we used in our little script.

interface

This particular example can be found here.

Use case: Variant to ‘Set interface description’

In the previous use case, we have read the various commands from a list we declared in our Python script. Netmiko also supports a method (send_config_from_file method) to read these commands from a file. It’s pretty similar to the script we used in previous use case.

from netmiko import Netmiko
import logging

devices = [{
   "device_type": "cisco_xe",
   "ip": "ios-xe-mgmt-latest.cisco.com",
   "username": "***",
   "password": "***",
   "port": "8181",
}]

for device in devices:
   net_connect = Netmiko(**device)
   output = net_connect.send_config_from_file('changes.txt')
   print(output)
   net_connect.disconnect()

As you can see, the commands are read from the changes.txt file. See below the contents of that file:

# Changes.txt
interface GigabitEthernet3
description This is set via Netmiko and text file

Let’s execute the script: interface

And verify if things got changed indeed:

interface

This particular example can be found here.

Use case: Add Loopback interfaces

In this use case, we will use some things we learned in previous posts. We will do the following:

  • read device information from a YML file
  • read variables (containing loopback information) from a YML file
  • Use Jinja2 template
  • Use Netmiko to configure the loopback interfaces

In case some of this is new, I recommend you to have a look at the following blog posts:

  • Parsing YML file: see here
  • Working with Jinja2: see here

Let’s have a look at the YML file first:

interfaces:
  - name: Loopback2001
    description: Description for Loopback 2000
    ipv4_addr: 200.200.200.200
    ipv4_mask: 255.255.255.255
  - name:  Loopback2002
    description: Description for Loopback 2001
    ipv4_addr: 200.200.200.201
    ipv4_mask: 255.255.255.255

As mentioned, this file contains the required information to configure the loopback interfaces. We will also read the host information from a file called hosts.yml.

# Define list of hosts
---
hosts:
  - name: sbx-iosxr-mgmt.cisco.com
    username: ***
    password: ***
    port: 8181
    cmd: "show running-config"
    type: cisco_xr
  - name: ios-xe-mgmt-latest.cisco.com
    username: ***
    password: ***
    port: 8181
    cmd: "show running-config"
    type: cisco_xe

The Jinja2 file is relative straightforward as well. Essentially, we use a for loop to iterate over the loopback interfaces. Each time, we use the data that is passed to this template from the Python file.

{% for interface in data.interfaces %}
interface {{interface.name}}
description {{interface.description}}
 ip address {{interface.ipv4_addr}} {{interface.ipv4_mask}}
{% endfor %}

The Python file is first reading in the host information from the hosts.yml file. The interface information we read from the YML file. The host and interface information is now available to us in a Python dictionary.

Next, we use the Jinja2 library to render the template with the variables. The end result of this is that loopback_config contains the completed (absolute) file with all loopback information.

Next, we simply loop over the devices (hosts) and for each host we create an SSH connection via Netmiko. And finally we use the send_config_set method to which we pass the loopback_config information (which contains nothing more than the commands to configure a loopback interface).

import yaml
from netmiko import Netmiko
from jinja2 import Environment, FileSystemLoader

hosts = yaml.load(open('hosts.yml'), Loader=yaml.SafeLoader)
interfaces = yaml.load(open('loopback.yml'), Loader=yaml.SafeLoader)

env = Environment(loader = FileSystemLoader('.'), trim_blocks=True, autoescape=True)
template = env.get_template('loopback.j2')
loopback_config = template.render(data=interfaces)

for host in hosts["hosts"]:
   net_connect = Netmiko(
      host = host["name"],
      username = host["username"],
      password = host["password"],
      port = host["port"],
      device_type = host["type"]
   )
   print(f"Logged into {host['name']} successfully")
   output = net_connect.send_config_set(loopback_config.split("\n"))

Let’s run this example.

(base) WAUTERW-M-65P7:test wauterw$ python3 script.py 
***Truncated***
Logged into ios-xe-mgmt-latest.cisco.com successfully
config term
Enter configuration commands, one per line.  End with CNTL/Z.
csr1000v-1(config)#interface Loopback3001
csr1000v-1(config-if)#description Description for Loopback 3001
csr1000v-1(config-if)# ip address 200.200.200.230 255.255.255.255
csr1000v-1(config-if)#interface Loopback3002
csr1000v-1(config-if)#description Description for Loopback 3002
csr1000v-1(config-if)# ip address 200.200.200.231 255.255.255.255
csr1000v-1(config-if)#
csr1000v-1(config-if)#end

To see if the interfaces were correctly added, we can use a script we mentioned earlier in this blogpost, e.g this one.

Interface              IP-Address      OK? Method Status                Protocol
GigabitEthernet1       10.10.20.48     YES other  up                    up      
GigabitEthernet2       10.121.23.120   YES other  up                    up      
GigabitEthernet2.4     unassigned      YES manual deleted               down    
***Truncated***     
Loopback3001           200.200.200.230 YES manual up                    up      
Loopback3002           200.200.200.231 YES manual up                    up      
Port-channel1          unassigned      YES unset  down                  down    
VirtualPortGroup0      10.10.20.48     YES unset  up                    up   

Note: Netmiko will go automatically into config mode so therefore you don’t see the conf t command in the list of commands.

That’s it for this post. Hope you liked this little introduction to Netmiko. Please refer this GitHub page .

GibHub Samples