#!/usr/bin/env python # SPDX-License-Identifier: ISC # # test_isis_tilfa_topo1.py # Part of NetDEF Topology Tests # # Copyright (c) 2020 by # Network Device Education Foundation, Inc. ("NetDEF") # """ test_isis_tilfa_topo1.py: +---------+ | | | RT1 | | 1.1.1.1 | | | +---------+ |eth-sw1 | | | +---------+ | +---------+ | | | | | | RT2 |eth-sw1 | eth-sw1| RT3 | | 2.2.2.2 +----------+----------+ 3.3.3.3 | | | 10.0.1.0/24 | | +---------+ +---------+ eth-rt4-1| |eth-rt4-2 eth-rt5-1| |eth-rt5-2 | | | | 10.0.2.0/24| |10.0.3.0/24 10.0.4.0/24| |10.0.5.0/24 | | | | eth-rt2-1| |eth-rt2-2 eth-rt3-1| |eth-rt3-2 +---------+ +---------+ | | | | | RT4 | 10.0.6.0/24 | RT5 | | 4.4.4.4 +---------------------+ 5.5.5.5 | | |eth-rt5 eth-rt4| | +---------+ +---------+ eth-rt6| |eth-rt6 | | 10.0.7.0/24| |10.0.8.0/24 | +---------+ | | | | | | | RT6 | | +----------+ 6.6.6.6 +-----------+ eth-rt4| |eth-rt5 +---------+ """ import os import sys import pytest import json from functools import partial from time import sleep # 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] # 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"]: tgen.add_router(router) # # Define connections # switch = tgen.add_switch("s1") switch.add_link(tgen.gears["rt1"], nodeif="eth-sw1") switch.add_link(tgen.gears["rt2"], nodeif="eth-sw1") switch.add_link(tgen.gears["rt3"], nodeif="eth-sw1") switch = tgen.add_switch("s2") switch.add_link(tgen.gears["rt2"], nodeif="eth-rt4-1") switch.add_link(tgen.gears["rt4"], nodeif="eth-rt2-1") switch = tgen.add_switch("s3") switch.add_link(tgen.gears["rt2"], nodeif="eth-rt4-2") switch.add_link(tgen.gears["rt4"], nodeif="eth-rt2-2") switch = tgen.add_switch("s4") switch.add_link(tgen.gears["rt3"], nodeif="eth-rt5-1") switch.add_link(tgen.gears["rt5"], nodeif="eth-rt3-1") switch = tgen.add_switch("s5") switch.add_link(tgen.gears["rt3"], nodeif="eth-rt5-2") switch.add_link(tgen.gears["rt5"], nodeif="eth-rt3-2") switch = tgen.add_switch("s6") switch.add_link(tgen.gears["rt4"], nodeif="eth-rt5") switch.add_link(tgen.gears["rt5"], nodeif="eth-rt4") switch = tgen.add_switch("s7") switch.add_link(tgen.gears["rt4"], nodeif="eth-rt6") switch.add_link(tgen.gears["rt6"], nodeif="eth-rt4") switch = tgen.add_switch("s8") switch.add_link(tgen.gears["rt5"], nodeif="eth-rt6") switch.add_link(tgen.gears["rt6"], nodeif="eth-rt5") 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_BFD, os.path.join(CWD, "{}/bfdd.conf".format(rname)) ) tgen.start_router() def teardown_module(): "Teardown the pytest environment" tgen = get_topogen() # This function tears down the whole topology. tgen.stop_topology() def filter_json(data, keys_to_keep): """ Filters a dictionary, keeping only the specified keys. """ return {k: v for k, v in data.items() if k in keys_to_keep} def regen_data(rname, command, step, file, wait): """ Regenerates reference data. """ # Sleep enough time to ensure the protocol has converged if rname == "rt1": sleep(wait) if step == 10: sleep(10) # Get and parse JSON output tgen = get_topogen() output = json.loads(tgen.gears[rname].vtysh_cmd(command)) # Default JSON separators separators = (",", ":") # Process JSON output based on the specified file if file == "show_yang_interface_isis_adjacencies.ref": # Filter out the loopback interface output["frr-interface:lib"]["interface"] = [ interface for interface in output["frr-interface:lib"]["interface"] if interface["name"] != "lo" ] # Filter out unwanted fields for interface in output["frr-interface:lib"]["interface"]: keys_to_keep = {"name", "vrf", "state"} filtered_interface = filter_json(interface, keys_to_keep) interface.clear() interface.update(filtered_interface) keys_to_keep = {"frr-isisd:isis"} filtered_state = filter_json(interface["state"], keys_to_keep) interface["state"].clear() interface["state"].update(filtered_state) keys_to_keep = {"adjacencies"} filtered_isis = filter_json( interface["state"]["frr-isisd:isis"], keys_to_keep ) interface["state"]["frr-isisd:isis"].clear() interface["state"]["frr-isisd:isis"].update(filtered_isis) if "adjacencies" in interface["state"]["frr-isisd:isis"]: for adjacency in interface["state"]["frr-isisd:isis"]["adjacencies"][ "adjacency" ]: keys_to_keep = { "neighbor-sys-type", "neighbor-sysid", "hold-timer", "neighbor-priority", "state", } filtered_adjacency = filter_json(adjacency, keys_to_keep) adjacency.clear() adjacency.update(filtered_adjacency) # Adjust separators to match libyang's output. separators = (",", ": ") elif file == "show_ip_route.ref" or file == "show_ipv6_route.ref": # Filter out unwanted fields keys_to_keep_route = { "prefix", "protocol", "selected", "destSelected", "distance", "metric", "installed", "nexthops", "backupNexthops", } keys_to_keep_nh = { "fib", "ip", "afi", "interfaceName", "active", "backupIndex", "labels", } for prefix_key, prefix_value in output.items(): filtered_routes = [] for route in prefix_value: if "nexthops" in route: filtered_nhs = [] for nh in route["nexthops"]: if nh["ip"].startswith("fe80"): del nh["ip"] filtered_nhs.append(filter_json(nh, keys_to_keep_nh)) route["nexthops"] = filtered_nhs if "backupNexthops" in route: filtered_nhs = [] for nh in route["backupNexthops"]: if nh["ip"].startswith("fe80"): del nh["ip"] filtered_nhs.append(filter_json(nh, keys_to_keep_nh)) route["backupNexthops"] = filtered_nhs filtered_routes.append(filter_json(route, keys_to_keep_route)) output[prefix_key] = filtered_routes elif file == "show_mpls_table.ref": # Filter out Adj-SID labels output = {int(key): value for key, value in output.items() if int(key) >= 16000} # Filter out unwanted fields keys_to_keep_label = { "inLabel", "installed", "nexthops", "backupNexthops", } keys_to_keep_nh = { "type", "outLabel", "installed", "interface", "nexthop", "backupIndex", } for label_key, label_value in output.items(): if "nexthops" in label_value: filtered_nhs = [] for nh in label_value["nexthops"]: if nh["nexthop"].startswith("fe80"): del nh["nexthop"] filtered_nhs.append(filter_json(nh, keys_to_keep_nh)) label_value["nexthops"] = filtered_nhs if "backupNexthops" in label_value: filtered_nhs = [] for nh in label_value["backupNexthops"]: if nh["nexthop"].startswith("fe80"): del nh["nexthop"] filtered_nhs.append(filter_json(nh, keys_to_keep_nh)) label_value["backupNexthops"] = filtered_nhs output[label_key] = filter_json(label_value, keys_to_keep_label) elif file.startswith("show_bfd_peer"): keys_to_keep = ["multihop", "peer", "interface", "status"] output = filter_json(output, keys_to_keep) # Save the processed output to a file filename = "{}/{}/step{}/{}".format(CWD, rname, step, file) output = json.dumps(output, separators=separators, indent=2).replace("/", "\\/") with open(filename, "w", encoding="ascii") as file: file.write(output + "\n") def router_compare_json_output(rname, command, step, file, count=120, wait=0.5): "Compare router JSON output" # Regenerate reference data when the REGEN_DATA environment variable is set if os.environ.get("REGEN_DATA") is not None: regen_data(rname, command, step, file, count * wait) return tgen = get_topogen() logger.info('Comparing router "%s" "%s" output', rname, command) reference = open("{}/{}/step{}/{}".format(CWD, rname, step, file)).read() 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=count, wait=wait) 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", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show yang operational-data /frr-interface:lib isisd", 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", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ip route isis json", 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", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ipv6 route isis json", 1, "show_ipv6_route.ref" ) def test_mpls_lib_step1(): logger.info("Test (step 1): verify MPLS LIB") tgen = get_topogen() # Skip if previous fatal error condition is raised if tgen.routers_have_failure(): pytest.skip(tgen.errors) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show mpls table json", 1, "show_mpls_table.ref" ) # # Step 2 # # Action(s): # -Disable TI-LFA link protection on rt2's eth-sw1 interface # # Expected changes: # -rt2 should uninstall the backup nexthops from destinations reachable over eth-sw1. # 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("Disabling TI-LFA link protection on rt2's eth-sw1 interface") tgen.net["rt2"].cmd( 'vtysh -c "conf t" -c "interface eth-sw1" -c "no isis fast-reroute ti-lfa"' ) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ip route isis json", 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", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ipv6 route isis json", 2, "show_ipv6_route.ref" ) def test_mpls_lib_step2(): logger.info("Test (step 2): verify MPLS LIB") tgen = get_topogen() # Skip if previous fatal error condition is raised if tgen.routers_have_failure(): pytest.skip(tgen.errors) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show mpls table json", 2, "show_mpls_table.ref" ) # # Step 3 # # Action(s): # -Enable TI-LFA link protection on rt2's eth-sw1 interface # # Expected changes: # -rt2 should install backup nexthops for destinations reachable over eth-sw1. # 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("Enabling TI-LFA link protection on rt2's eth-sw1 interface") tgen.net["rt2"].cmd( 'vtysh -c "conf t" -c "interface eth-sw1" -c "isis fast-reroute ti-lfa"' ) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ip route isis json", 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", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ipv6 route isis json", 3, "show_ipv6_route.ref" ) def test_mpls_lib_step3(): logger.info("Test (step 3): verify MPLS LIB") tgen = get_topogen() # Skip if previous fatal error condition is raised if tgen.routers_have_failure(): pytest.skip(tgen.errors) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show mpls table json", 3, "show_mpls_table.ref" ) # # Step 4 # # Action(s): # -Disable SR on rt4 # # Expected changes: # -rt4 should uninstall all Prefix-SIDs from the network # -rt4 should uninstall all TI-LFA backup nexthops # -All routers should uninstall rt4's Prefix-SIDs # -All routers should uninstall all SR labels for destinations whose nexthop is rt4 # -All routers should uninstall all TI-LFA backup nexthops that point to rt4 # -All routers should uninstall all TI-LFA backup nexthops that use rt4's Prefix-SIDs # 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 SR on rt4") tgen.net["rt4"].cmd( 'vtysh -c "conf t" -c "router isis 1" -c "no segment-routing on"' ) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ip route isis json", 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", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ipv6 route isis json", 4, "show_ipv6_route.ref" ) def test_mpls_lib_step4(): logger.info("Test (step 4): verify MPLS LIB") tgen = get_topogen() # Skip if previous fatal error condition is raised if tgen.routers_have_failure(): pytest.skip(tgen.errors) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show mpls table json", 4, "show_mpls_table.ref" ) # # Step 5 # # Action(s): # -Enable SR on rt4 # # Expected changes: # -Reverse all changes done on the previous step # 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("Enabling SR on rt4") tgen.net["rt4"].cmd('vtysh -c "conf t" -c "router isis 1" -c "segment-routing on"') for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ip route isis json", 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", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ipv6 route isis json", 5, "show_ipv6_route.ref" ) def test_mpls_lib_step5(): logger.info("Test (step 5): verify MPLS LIB") tgen = get_topogen() # Skip if previous fatal error condition is raised if tgen.routers_have_failure(): pytest.skip(tgen.errors) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show mpls table json", 5, "show_mpls_table.ref" ) # # Step 6 # # Action(s): # -Change rt5's SRGB # # Expected changes: # -All routers should update all SR labels for destinations whose primary or backup nexthop is rt5 # 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("Changing rt5's SRGB") tgen.net["rt5"].cmd( 'vtysh -c "conf t" -c "router isis 1" -c "segment-routing global-block 30000 37999"' ) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ip route isis json", 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", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ipv6 route isis json", 6, "show_ipv6_route.ref" ) def test_mpls_lib_step6(): logger.info("Test (step 6): verify MPLS LIB") tgen = get_topogen() # Skip if previous fatal error condition is raised if tgen.routers_have_failure(): pytest.skip(tgen.errors) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show mpls table json", 6, "show_mpls_table.ref" ) # # Step 7 # # Action(s): # -Delete rt5's Prefix-SIDs # # Expected changes: # -All routers should uninstall rt5's Prefix-SIDs # -All routers should uninstall all TI-LFA backup nexthops that use rt5's Prefix-SIDs # 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("Deleting rt5's Prefix-SIDs") tgen.net["rt5"].cmd( 'vtysh -c "conf t" -c "router isis 1" -c "no segment-routing prefix 5.5.5.5/32 index 50"' ) tgen.net["rt5"].cmd( 'vtysh -c "conf t" -c "router isis 1" -c "no segment-routing prefix 2001:db8:1000::5/128 index 51"' ) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ip route isis json", 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", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ipv6 route isis json", 7, "show_ipv6_route.ref" ) def test_mpls_lib_step7(): logger.info("Test (step 7): verify MPLS LIB") tgen = get_topogen() # Skip if previous fatal error condition is raised if tgen.routers_have_failure(): pytest.skip(tgen.errors) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show mpls table json", 7, "show_mpls_table.ref" ) # # Step 8 # # Action(s): # -Re-add rt5's Prefix-SIDs # # Expected changes: # -Reverse all changes done on the previous step # 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("Re-adding rt5's Prefix-SIDs") tgen.net["rt5"].cmd( 'vtysh -c "conf t" -c "router isis 1" -c "segment-routing prefix 5.5.5.5/32 index 50"' ) tgen.net["rt5"].cmd( 'vtysh -c "conf t" -c "router isis 1" -c "segment-routing prefix 2001:db8:1000::5/128 index 51"' ) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ip route isis json", 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", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ipv6 route isis json", 8, "show_ipv6_route.ref" ) def test_mpls_lib_step8(): logger.info("Test (step 8): verify MPLS LIB") tgen = get_topogen() # Skip if previous fatal error condition is raised if tgen.routers_have_failure(): pytest.skip(tgen.errors) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show mpls table json", 8, "show_mpls_table.ref" ) # # Step 9 # # Action(s): # -Change rt5's Prefix-SIDs # # Expected changes: # -All routers should update rt5's Prefix-SIDs # -All routers should update all TI-LFA backup nexthops that use rt5's Prefix-SIDs # 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("Re-adding rt5's Prefix-SIDs") tgen.net["rt5"].cmd( 'vtysh -c "conf t" -c "router isis 1" -c "segment-routing prefix 5.5.5.5/32 index 500"' ) tgen.net["rt5"].cmd( 'vtysh -c "conf t" -c "router isis 1" -c "segment-routing prefix 2001:db8:1000::5/128 index 501"' ) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ip route isis json", 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", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show ipv6 route isis json", 9, "show_ipv6_route.ref" ) def test_mpls_lib_step9(): logger.info("Test (step 9): verify MPLS LIB") tgen = get_topogen() # Skip if previous fatal error condition is raised if tgen.routers_have_failure(): pytest.skip(tgen.errors) for rname in ["rt1", "rt2", "rt3", "rt4", "rt5", "rt6"]: router_compare_json_output( rname, "show mpls table json", 9, "show_mpls_table.ref" ) # # Step 10 # # Action(s): # - Enable ISIS BFD between rt5 and rt6 # - Verify that the BFD session is up # - Configure an SPF delay-ietf initial delay of 60 seconds on both rt5 and rt6 # - Shut down the eth-rt5 interface on rt6 from the switch side to test fast-reroute # # Expected changes: # - Verify that the BFD session is down # - Routes should switch over to use alternate paths # - On rt5, the switchover should be triggered by the link down event # - On rt6, the switchover should be triggered by the BFD down event, since it has # link-detect disabled on the eth-rt5 interface # 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("Enabling ISIS BFD between rt5 and rt6") tgen.net["rt5"].cmd('vtysh -c "conf t" -c "int eth-rt6" -c "isis bfd"') tgen.net["rt6"].cmd('vtysh -c "conf t" -c "int eth-rt5" -c "isis bfd"') logger.info("Checking if the BFD session is up") expect = '{"multihop":false,"peer":"10.0.8.5","interface":"eth-rt5","status":"up"}' router_compare_json_output( "rt6", "show bfd peer 10.0.8.5 json", 10, "show_bfd_peer_up.ref" ) logger.info("Setting SPF delay-ietf initial delay to 60 seconds") for rname in ["rt5", "rt6"]: tgen.net[rname].cmd( 'vtysh -c "conf t" -c "router isis 1" -c "spf-delay-ietf init-delay 60000 short-delay 0 long-delay 0 holddown 0 time-to-learn 0"' ) logger.info( "Shutting down rt5 interface to rt6 from the switch side to test fast-reroute" ) tgen.net.cmd_raises("ip link set %s down" % tgen.net["s8"].intfs[0]) logger.info("Verifying if the BFD session is down") expect = ( '{"multihop":false,"peer":"10.0.8.5","interface":"eth-rt5","status":"down"}' ) router_compare_json_output( "rt6", "show bfd peer 10.0.8.5 json", 10, "show_bfd_peer_down.ref" ) for rname in ["rt5", "rt6"]: router_compare_json_output( rname, "show ip route isis json", 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 ["rt5", "rt6"]: router_compare_json_output( rname, "show ipv6 route isis json", 10, "show_ipv6_route.ref", ) def test_mpls_lib_step10(): logger.info("Test (step 10): verify MPLS LIB") tgen = get_topogen() # Skip if previous fatal error condition is raised if tgen.routers_have_failure(): pytest.skip(tgen.errors) for rname in ["rt5", "rt6"]: router_compare_json_output( rname, "show mpls table json", 10, "show_mpls_table.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))