2
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

pythonを使ってCisco IOS機器のバックアップを取得してみた

Last updated at Posted at 2022-01-08

はじめに

お勉強のためにpythonの以下ライブラリを使ってCisco IOS機器を操作してみました。
機器情報の取得やコンフィグ投入等、多様な操作ができることがわかりましたが、今回はそれぞれのライブラリを使ってバックアップを取得する(show runの結果を取得してファイルに保存する)スクリプトを作成したので参考までに共有します。

使用するライブラリ

前提

  • 実行環境(Cisco IOSに接続するクライアント):ubuntu 20.04
  • pythonバージョン:3.8.10
  • 実行環境上で各種ライブラリをインストールします。
pip install paramiko netmiko napalm
  • 以下図のUbuntuからIOUルータ3台のバックアップを取得します。
    image.png

paramiko

paramikoの汎用モジュール(paramiko_mod.py)と、
バックアップを実行するメインファイル(paramiko_ios_backup_executer.py)の2部構成です。

①paramikoの汎用モジュール

paramiko_mod.py
import paramiko
import time

from paramiko.client import AutoAddPolicy

def connect(server_ip: str, server_port: str, user: str, password: str): 
    '''SSHサーバに接続'''
    ssh_client = paramiko.SSHClient()
    ssh_client.set_missing_host_key_policy(AutoAddPolicy())
    print(f'Connecting to {server_ip}')
    ssh_client.connect(
        hostname=server_ip, 
        port=server_port,
        username=user,
        password=password,
        look_for_keys=False,
        allow_agent=False
    )
    return ssh_client

def get_shell(ssh_client):
    '''対話型シェルを返す'''
    shell = ssh_client.invoke_shell()
    return shell

def send_command(shell, command: str, timeout=1):
    '''コマンドを実行'''
    print(f'Sending command: {command}')
    shell.send(command + '\n')
    time.sleep(timeout)

def show(shell, n=10000):
    '''結果をデコードして返す'''
    # 結果をバイト列で取得
    output = shell.recv(n)
    return output.decode()

def close(ssh_client):
    '''SSH接続を閉じる'''
    if ssh_client.get_transport().is_active() == True:
        print('Closing the connection')
        ssh_client.close()
  • SSH認証方式はユーザ認証としています。公開鍵認証、SSHエージェントも利用できるようです。
  • send_command(shell, command: str, timeout=1):
    • コマンド結果が指定秒以内に返って来ない場合があるため、timeout値は呼び出し時に変更できるようにしています。

②バックアップを実行するメインファイル(paramiko)

paramiko_ios_backup_executer.py
import os
import datetime
import threading
import paramiko_mod
from datetime import datetime

def backup(router):
    # sshクライアントを生成
    client = paramiko_mod.connect(**router)

    # 対話型シェルを呼び出す
    shell = paramiko_mod.get_shell(client)
    
    # ssh接続先で実行するコマンドをシェルに渡す
    paramiko_mod.send_command(shell, 'enable')
    paramiko_mod.send_command(shell, '######') # enableパスワード
    paramiko_mod.send_command(shell, 'terminal length 0')
    
    # sh runの結果が取得できない場合はtimeout値を調整
    paramiko_mod.send_command(shell, 'sh run', timeout=3)     

    # 結果を取得&リストに整形
    output = paramiko_mod.show(shell)
    
    # [show run]コマンド 実行結果の不要な箇所を削除するため、一度リストに変換する
    output_list = output.splitlines()
    output_list = output_list[11:-1]

    # リストを文字列に戻す 
    output = '\n'.join(output_list)
    print(output)

    # バックアップ先のディレクトリがなければ作成
    backup_dir = f'backup/{router["server_ip"]}'
    if not os.path.exists(backup_dir):
        os.makedirs(backup_dir)

    # 結果をバックアップファイルに書き出す
    bk_file_name = f'{router["server_ip"]}_{now.year}{now.month}{now.day}'
    with open(f'{backup_dir}/{bk_file_name}', 'w') as f:
        f.write(output)

    paramiko_mod.close(client)

router1 = {
    'server_ip': '192.168.15.40',
    'server_port': '22',
    'user': 'XXXXXXX',
    'password': 'XXXXXXX'
}
router2 = {
    'server_ip': '192.168.15.41',
    'server_port': '22',
    'user': 'XXXXXXX',
    'password': 'XXXXXXX'
}
router3 = {
    'server_ip': '192.168.15.42',
    'server_port': '22',
    'user': 'XXXXXXX',
    'password': 'XXXXXXX'
}

routers = [router1, router2, router3]

if __name__ == '__main__':

    now = datetime.now()
    
    # バックアップ関数をスレッドリストに詰める
    threads = [threading.Thread(target=backup, args=(router,)) for router in routers]

    # マルチスレッドでバックアップ開始
    for th in threads:
        th.start()

    # バックアップがすべて終わるまで待機
    for th in threads:
        th.join()

  • 本スクリプトでは、3台のCiscoルータのバックアップ処理をマルチスレッドで実行しています。
  • バックアップファイルはカレントディレクトのbackup/{router["server_ip"]}配下に保存される仕様です。
  • パスワードやユーザ情報はハードコーディングしていますが、環境変数や別の設定ファイル(ini, yml等)に分ける等の実装がよいと思います。(対象機器の台数が多くなった場合の設定管理の意味も含めて)

netmiko

機器の設定情報を記述したjsonファイル(target_device_info.json)と、
バックアップを実行するメインファイル(netmiko_ios_backup_executer.py)の2部構成です。

①機器の設定情報を記述したjsonファイル

target_device_info.json
[
    {
        "host": "192.168.15.40",
        "port": "22", 
        "username": "XXXXX", 
        "password": "XXXXX", 
        "device_type": "cisco_ios",
        "secret": "XXXXX",
        "verbose": "True"
    },
    {
        "host": "192.168.15.41",
        "port": "22", 
        "username": "XXXXX", 
        "password": "XXXXX", 
        "device_type": "cisco_ios",
        "secret": "XXXXX",
        "verbose": "True"
    },
    {
        "host": "192.168.15.42",
        "port": "22", 
        "username": "XXXXX", 
        "password": "XXXXX", 
        "device_type": "cisco_ios",
        "secret": "XXXXX",
        "verbose": "True"
    }
]

②バックアップを実行するメインファイル(netmiko)

paramiko_ios_backup_executer.py
from datetime import datetime
import os
import netmiko
import json
import threading

def backup(device):
    print(f'Connect {device["host"]}')
    connection = netmiko.ConnectHandler(**device)

    connection.enable()

    output = connection.send_command('show run')

    # [show run]コマンド 実行結果の不要な箇所を削除するため、一度リストに変換する
    output_list = output.splitlines()
    output_list = output_list[11:-1]
    # リストを文字列に戻す
    output = '\n'.join(output_list)

    # バックアップ先のディレクトリがなければ作成
    backup_dir = f'backup/{device["host"]}'
    if not os.path.exists(backup_dir):
        os.makedirs(backup_dir)

    # 結果をバックアップファイルに書き出す
    now = datetime.now()
    bk_file_name = f'{device["host"]}_{now.year}{now.month}{now.day}'
    with open(f'{backup_dir}/{bk_file_name}', 'w') as f:
        f.write(output)

    print(f'Disconnect {device["host"]}')
    connection.disconnect() 

if __name__ == "__main__":

    # 接続機器の設定情報が記述されたjsonを読み込む
    with open(r"target_device_info.json", "r") as f:
        device_info_list = json.load(f)

    threads = list()
    for info in device_info_list:

        # バックアップ関数をスレッドリストに詰める
        thread = threading.Thread(target=backup, args=(info,))
        threads.append(thread)

    # マルチスレッドでバックアップを実行
    for th in threads:
      th.start()

    # すべてのバックアップ処理が完了するまで待機
    for th in threads:
      th.join()
  • バックアップ対象機器の情報を設定ファイルから読み込む仕様です。
  • paramikoのバックアップスクリプトと同様に処理をマルチスレッドで実行しています。
  • バックアップファイルはカレントディレクトのbackup/{device["host"]}配下に保存される仕様です。

napalm

netmikoと同じく、機器の設定情報を記述したjsonファイル(target_device_info.json)と、
バックアップを実行するメインファイル(napalm_ios_backup_executer.py)の2部構成です。

①機器の設定情報を記述したjsonファイル

target_device_info.json
[
    {
        "host": "192.168.15.40",
        "username": "admin1", 
        "password": "cisco1", 
        "device_type": "ios",
        "secret": "cisco1"
    },
    {
        "host": "192.168.15.41",
        "username": "admin1", 
        "password": "cisco1", 
        "device_type": "ios",
        "secret": "cisco1"
    },
    {
        "host": "192.168.15.42",
        "username": "admin1", 
        "password": "cisco1", 
        "device_type": "ios",
        "secret": "cisco1"
    }
]

②バックアップを実行するメインファイル(napalm)

napalm_ios_backup_executer.py
from datetime import datetime
import os
from napalm import get_network_driver
import json
import threading

def backup(device):
    print(f'Connect {device["host"]}')

    # 接続設定を読み込む
    driver = get_network_driver(device["device_type"])
    option_args = {'secret': device["secret"]}
    ios = driver(
        device["host"], device["username"], device["password"], optional_args=option_args)

    # 接続    
    ios.open()

    # sh run の結果を取得    
    output = ios.get_config()['running']
    print(output)

    # バックアップ先のディレクトリがなければ作成
    backup_dir = f'backup/{device["host"]}'
    if not os.path.exists(backup_dir):
        os.makedirs(backup_dir)

    # 結果をバックアップファイルに書き出す
    now = datetime.now()
    bk_file_name = f'{device["host"]}_{now.year}{now.month}{now.day}'
    with open(f'{backup_dir}/{bk_file_name}', 'w') as f:
        f.write(output)

    print(f'Disconnect {device["host"]}')

    ios.close()

if __name__ == "__main__":

    # 接続機器の設定情報が記述されたjsonを読み込む
    with open(r"target_device_info.json", "r") as f:
        device_info_list = json.load(f)

    threads = list()
    for info in device_info_list:

        # バックアップ関数をスレッドリストに詰める
        thread = threading.Thread(target=backup, args=(info,))
        threads.append(thread)

    # マルチスレッドでバックアップを実行
    for th in threads:
        th.start()

    # すべてのバックアップ処理が完了するまで待機
    for th in threads:
        th.join()

2
4
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
2
4

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?