1
0
Fork 0
frr/tests/topotests/isis_rlfa_topo1/test_isis_rlfa_topo1.py
Daniel Baumann 3124f89aed
Adding upstream version 10.1.1.
Signed-off-by: Daniel Baumann <daniel@debian.org>
2025-02-05 10:03:58 +01:00

644 lines
19 KiB
Python
Executable file

#!/usr/bin/env python
# SPDX-License-Identifier: ISC
#
# test_isis_rlfa_topo1.py
# Part of NetDEF Topology Tests
#
# Copyright (c) 2020 by
# Network Device Education Foundation, Inc. ("NetDEF")
#
"""
test_isis_rlfa_topo1.py:
+---------+ +---------+
| | | |
| RT1 | | RT2 |
| +---------------------+ |
| | | |
+---+-----+ +------+--+
| |
| |
| |
+---+-----+ +------+--+
| | | |
| RT3 | | RT4 |
| | | |
| | | |
+---+-----+ +------+--+
| |
| |
| |
+---+-----+ +------+--+
| | | |
| RT5 | | RT6 |
| | | |
| | | |
+---+-----+ +------+--+
| |
| |
| |
+---+-----+ +------+--+
| | | |
| RT7 | | RT8 |
| +---------------------+ |
| | | |
+---------+ +---------+
"""
import os
import sys
import pytest
import json
import tempfile
from functools import partial
# Save the Current Working Directory to find configuration files.
CWD = os.path.dirname(os.path.realpath(__file__))
sys.path.append(os.path.join(CWD, "../"))
# pylint: disable=C0413
# Import topogen and topotest helpers
from lib import topotest
from lib.topogen import Topogen, TopoRouter, get_topogen
from lib.topolog import logger
# Required to instantiate the topology builder class.
pytestmark = [pytest.mark.isisd, pytest.mark.ldpd]
# Global multi-dimensional dictionary containing all expected outputs
outputs = {}
def build_topo(tgen):
"Build function"
#
# Define FRR Routers
#
for router in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6", "rt7", "rt8"]:
tgen.add_router(router)
#
# Define connections
#
switch = tgen.add_switch("s1")
switch.add_link(tgen.gears["rt1"], nodeif="eth-rt2")
switch.add_link(tgen.gears["rt2"], nodeif="eth-rt1")
switch = tgen.add_switch("s2")
switch.add_link(tgen.gears["rt1"], nodeif="eth-rt3")
switch.add_link(tgen.gears["rt3"], nodeif="eth-rt1")
switch = tgen.add_switch("s3")
switch.add_link(tgen.gears["rt2"], nodeif="eth-rt4")
switch.add_link(tgen.gears["rt4"], nodeif="eth-rt2")
switch = tgen.add_switch("s4")
switch.add_link(tgen.gears["rt3"], nodeif="eth-rt5")
switch.add_link(tgen.gears["rt5"], nodeif="eth-rt3")
switch = tgen.add_switch("s5")
switch.add_link(tgen.gears["rt4"], nodeif="eth-rt6")
switch.add_link(tgen.gears["rt6"], nodeif="eth-rt4")
switch = tgen.add_switch("s6")
switch.add_link(tgen.gears["rt5"], nodeif="eth-rt7")
switch.add_link(tgen.gears["rt7"], nodeif="eth-rt5")
switch = tgen.add_switch("s7")
switch.add_link(tgen.gears["rt6"], nodeif="eth-rt8")
switch.add_link(tgen.gears["rt8"], nodeif="eth-rt6")
switch = tgen.add_switch("s8")
switch.add_link(tgen.gears["rt7"], nodeif="eth-rt8")
switch.add_link(tgen.gears["rt8"], nodeif="eth-rt7")
#
# Populate multi-dimensional dictionary containing all expected outputs
#
files = [
"show_ip_route.ref",
"show_ipv6_route.ref",
"show_yang_interface_isis_adjacencies.ref",
]
for rname in ["rt1"]:
outputs[rname] = {}
for step in range(1, 10 + 1):
outputs[rname][step] = {}
for file in files:
if step == 1:
# Get snapshots relative to the expected initial network convergence
filename = "{}/{}/step{}/{}".format(CWD, rname, step, file)
outputs[rname][step][file] = open(filename).read()
else:
if file == "show_yang_interface_isis_adjacencies.ref":
continue
# Get diff relative to the previous step
filename = "{}/{}/step{}/{}.diff".format(CWD, rname, step, file)
# Create temporary files in order to apply the diff
f_in = tempfile.NamedTemporaryFile(mode="w")
f_in.write(outputs[rname][step - 1][file])
f_in.flush()
f_out = tempfile.NamedTemporaryFile(mode="r")
os.system(
"patch -s -o %s %s %s" % (f_out.name, f_in.name, filename)
)
# Store the updated snapshot and remove the temporary files
outputs[rname][step][file] = open(f_out.name).read()
f_in.close()
f_out.close()
def setup_module(mod):
"Sets up the pytest environment"
tgen = Topogen(build_topo, mod.__name__)
tgen.start_topology()
router_list = tgen.routers()
# For all registered routers, load the zebra configuration file
for rname, router in router_list.items():
router.load_config(
TopoRouter.RD_ZEBRA, os.path.join(CWD, "{}/zebra.conf".format(rname))
)
router.load_config(
TopoRouter.RD_ISIS, os.path.join(CWD, "{}/isisd.conf".format(rname))
)
router.load_config(
TopoRouter.RD_LDP, os.path.join(CWD, "{}/ldpd.conf".format(rname))
)
tgen.start_router()
def teardown_module(mod):
"Teardown the pytest environment"
tgen = get_topogen()
# This function tears down the whole topology.
tgen.stop_topology()
def router_compare_json_output(rname, command, reference):
"Compare router JSON output"
logger.info('Comparing router "%s" "%s" output', rname, command)
tgen = get_topogen()
expected = json.loads(reference)
# Run test function until we get an result. Wait at most 60 seconds.
test_func = partial(topotest.router_json_cmp, tgen.gears[rname], command, expected)
_, diff = topotest.run_and_expect(test_func, None, count=120, wait=0.5)
assertmsg = '"{}" JSON output mismatches the expected result'.format(rname)
assert diff is None, assertmsg
#
# Step 1
#
# Test initial network convergence
#
def test_isis_adjacencies_step1():
logger.info("Test (step 1): check IS-IS adjacencies")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
for rname in ["rt1"]:
router_compare_json_output(
rname,
"show yang operational-data /frr-interface:lib isisd",
outputs[rname][1]["show_yang_interface_isis_adjacencies.ref"],
)
def test_rib_ipv4_step1():
logger.info("Test (step 1): verify IPv4 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ip route isis json", outputs[rname][1]["show_ip_route.ref"]
)
def test_rib_ipv6_step1():
logger.info("Test (step 1): verify IPv6 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ipv6 route isis json", outputs[rname][1]["show_ipv6_route.ref"]
)
#
# Step 2
#
# Action(s):
# -Configure rt8 (rt1's PQ router) to not accept targeted hello messages
#
# Expected changes:
# -All rt1 backup routes should be uninstalled
#
def test_rib_ipv4_step2():
logger.info("Test (step 2): verify IPv4 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
logger.info("Configuring rt8 to not accept targeted hello messages")
tgen.net["rt8"].cmd(
'vtysh -c "conf t" -c "mpls ldp" -c "address-family ipv4" -c "no discovery targeted-hello accept"'
)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ip route isis json", outputs[rname][2]["show_ip_route.ref"]
)
def test_rib_ipv6_step2():
logger.info("Test (step 2): verify IPv6 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ipv6 route isis json", outputs[rname][2]["show_ipv6_route.ref"]
)
#
# Step 3
#
# Action(s):
# -Configure rt8 (rt1's PQ router) to accept targeted hello messages
#
# Expected changes:
# -All rt1 previously uninstalled backup routes should be reinstalled
#
def test_rib_ipv4_step3():
logger.info("Test (step 3): verify IPv4 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
logger.info("Configuring rt8 to accept targeted hello messages")
tgen.net["rt8"].cmd(
'vtysh -c "conf t" -c "mpls ldp" -c "address-family ipv4" -c "discovery targeted-hello accept"'
)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ip route isis json", outputs[rname][3]["show_ip_route.ref"]
)
def test_rib_ipv6_step3():
logger.info("Test (step 3): verify IPv6 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ipv6 route isis json", outputs[rname][3]["show_ipv6_route.ref"]
)
#
# Step 4
#
# Action(s):
# -Disable RLFA on rt1's eth-rt2 interface
#
# Expected changes:
# -All non-ECMP routes whose primary nexthop is eth-rt2 should lose their backup nexthops
#
def test_rib_ipv4_step4():
logger.info("Test (step 4): verify IPv4 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
logger.info("Disabling RLFA on rt1's eth-rt2 interface")
tgen.net["rt1"].cmd(
'vtysh -c "conf t" -c "interface eth-rt2" -c "no isis fast-reroute remote-lfa tunnel mpls-ldp"'
)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ip route isis json", outputs[rname][4]["show_ip_route.ref"]
)
def test_rib_ipv6_step4():
logger.info("Test (step 4): verify IPv6 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ipv6 route isis json", outputs[rname][4]["show_ipv6_route.ref"]
)
#
# Step 5
#
# Action(s):
# -Disable RLFA on rt1's eth-rt3 interface
#
# Expected changes:
# -All non-ECMP routes whose primary nexthop is eth-rt3 should lose their backup nexthops
#
def test_rib_ipv4_step5():
logger.info("Test (step 5): verify IPv4 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
logger.info("Disabling RLFA on rt1's eth-rt3 interface")
tgen.net["rt1"].cmd(
'vtysh -c "conf t" -c "interface eth-rt3" -c "no isis fast-reroute remote-lfa tunnel mpls-ldp"'
)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ip route isis json", outputs[rname][5]["show_ip_route.ref"]
)
def test_rib_ipv6_step5():
logger.info("Test (step 5): verify IPv6 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ipv6 route isis json", outputs[rname][5]["show_ipv6_route.ref"]
)
#
# Step 6
#
# Action(s):
# -Re-enable RLFA on rt1's eth-rt2 and eth-rt3 interfaces
#
# Expected changes:
# -Revert changes from the previous two steps (reinstall all backup routes)
#
def test_rib_ipv4_step6():
logger.info("Test (step 6): verify IPv4 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
logger.info("Re-enabling RLFA on rt1's eth-rt2 and eth-rt3 interfaces")
tgen.net["rt1"].cmd(
'vtysh -c "conf t" -c "interface eth-rt2" -c "isis fast-reroute remote-lfa tunnel mpls-ldp"'
)
tgen.net["rt1"].cmd(
'vtysh -c "conf t" -c "interface eth-rt3" -c "isis fast-reroute remote-lfa tunnel mpls-ldp"'
)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ip route isis json", outputs[rname][6]["show_ip_route.ref"]
)
def test_rib_ipv6_step6():
logger.info("Test (step 6): verify IPv6 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ipv6 route isis json", outputs[rname][6]["show_ipv6_route.ref"]
)
#
# Step 7
#
# Action(s):
# -Configure a PQ node prefix-list filter
#
# Expected changes:
# -All backup routes should be uninstalled
#
def test_rib_ipv4_step7():
logger.info("Test (step 7): verify IPv4 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
logger.info("Configuring a PQ node prefix-list filter")
tgen.net["rt1"].cmd(
'vtysh -c "conf t" -c "router isis 1" -c "fast-reroute remote-lfa prefix-list PLIST"'
)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ip route isis json", outputs[rname][7]["show_ip_route.ref"]
)
def test_rib_ipv6_step7():
logger.info("Test (step 7): verify IPv6 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ipv6 route isis json", outputs[rname][7]["show_ipv6_route.ref"]
)
#
# Step 8
#
# Action(s):
# -Configure a prefix-list allowing rt8 as a PQ node
#
# Expected changes:
# -All backup routes should be installed again
#
def test_rib_ipv4_step8():
logger.info("Test (step 8): verify IPv4 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
logger.info("Configuring a prefix-list allowing rt8 as a PQ node")
tgen.net["rt1"].cmd(
'vtysh -c "conf t" -c "ip prefix-list PLIST seq 5 permit 10.0.255.8/32"'
)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ip route isis json", outputs[rname][8]["show_ip_route.ref"]
)
def test_rib_ipv6_step8():
logger.info("Test (step 8): verify IPv6 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ipv6 route isis json", outputs[rname][8]["show_ipv6_route.ref"]
)
#
# Step 9
#
# Action(s):
# -Change the maximum metric up to the PQ node to 30 on the eth-rt2 interface
#
# Expected changes:
# -All non-ECMP routes whose primary nexthop is eth-rt2 should lose their backup nexthops
#
def test_rib_ipv4_step9():
logger.info("Test (step 9): verify IPv4 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
logger.info(
"Changing the maximum metric up to the PQ node to 30 on the eth-rt2 interface"
)
tgen.net["rt1"].cmd(
'vtysh -c "conf t" -c "interface eth-rt2" -c "isis fast-reroute remote-lfa maximum-metric 30"'
)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ip route isis json", outputs[rname][9]["show_ip_route.ref"]
)
def test_rib_ipv6_step9():
logger.info("Test (step 9): verify IPv6 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ipv6 route isis json", outputs[rname][9]["show_ipv6_route.ref"]
)
#
# Step 10
#
# Action(s):
# -Change the maximum metric up to the PQ node to 40 on the eth-rt2 interface
#
# Expected changes:
# -All non-ECMP routes whose primary nexthop is eth-rt2 should recover their backup nexthops
#
def test_rib_ipv4_step10():
logger.info("Test (step 10): verify IPv4 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
logger.info(
"Changing the maximum metric up to the PQ node to 40 on the eth-rt2 interface"
)
tgen.net["rt1"].cmd(
'vtysh -c "conf t" -c "interface eth-rt2" -c "isis fast-reroute remote-lfa maximum-metric 40"'
)
for rname in ["rt1"]:
router_compare_json_output(
rname, "show ip route isis json", outputs[rname][10]["show_ip_route.ref"]
)
def test_rib_ipv6_step10():
logger.info("Test (step 10): verify IPv6 RIB")
tgen = get_topogen()
# Skip if previous fatal error condition is raised
if tgen.routers_have_failure():
pytest.skip(tgen.errors)
for rname in ["rt1"]:
router_compare_json_output(
rname,
"show ipv6 route isis json",
outputs[rname][10]["show_ipv6_route.ref"],
)
# Memory leak test template
def test_memory_leak():
"Run the memory leak test and report results."
tgen = get_topogen()
if not tgen.is_memleak_enabled():
pytest.skip("Memory leak test/report is disabled")
tgen.report_memory_leaks()
if __name__ == "__main__":
args = ["-s"] + sys.argv[1:]
sys.exit(pytest.main(args))