diff --git a/config/main.py b/config/main.py index 983c1c394b..b6550ee7fd 100644 --- a/config/main.py +++ b/config/main.py @@ -1325,16 +1325,6 @@ def reload(db, filename, yes, load_sysinfo, disable_validation, verbose, no_serv click.echo("Input {} config file(s) separated by comma for multiple files ".format(num_cfg_file)) return - if load_sysinfo: - command = "{} -j {} -v DEVICE_METADATA.localhost.hwsku".format(SONIC_CFGGEN_PATH, filename) - proc = subprocess.Popen(command, shell=True, text=True, stdout=subprocess.PIPE) - cfg_hwsku, err = proc.communicate() - if err: - click.echo("Could not get the HWSKU from config file, exiting") - sys.exit(1) - else: - cfg_hwsku = cfg_hwsku.strip() - # For dual ToR devices, cache ARP and FDB info localhost_metadata = db.cfgdb.get_table('DEVICE_METADATA')['localhost'] cache_arp_table = not disable_arp_cache and 'subtype' in localhost_metadata and localhost_metadata['subtype'].lower() == 'dualtor' @@ -1378,6 +1368,25 @@ def reload(db, filename, yes, load_sysinfo, disable_validation, verbose, no_serv click.echo("The config_db file {} doesn't exist".format(file)) continue + if load_sysinfo: + try: + command = "{} -j {} -v DEVICE_METADATA.localhost.hwsku".format(SONIC_CFGGEN_PATH, file) + proc = subprocess.Popen(command, shell=True, text=True, stdout=subprocess.PIPE) + output, err = proc.communicate() + + except FileNotFoundError as e: + click.echo("{}".format(str(e)), err=True) + raise click.Abort() + except Exception as e: + click.echo("{}\n{}".format(type(e), str(e)), err=True) + raise click.Abort() + + if not output: + click.secho("Could not get the HWSKU from config file, Exiting!!!", fg='magenta') + sys.exit(1) + + cfg_hwsku = output.strip() + if namespace is None: config_db = ConfigDBConnector() else: diff --git a/tests/config_reload_input/config_db.json b/tests/config_reload_input/config_db.json new file mode 100644 index 0000000000..4b6a0ad405 --- /dev/null +++ b/tests/config_reload_input/config_db.json @@ -0,0 +1,62 @@ +{ + "DEVICE_METADATA": { + "localhost": { + "docker_routing_config_mode": "split", + "hostname": "sonic", + "hwsku": "Seastone-DX010-25-50", + "mac": "00:e0:ec:89:6e:48", + "platform": "x86_64-cel_seastone-r0", + "type": "ToRRouter" + } + }, + "VLAN_MEMBER": { + "Vlan1000|Ethernet0": { + "tagging_mode": "untagged" + }, + "Vlan1000|Ethernet4": { + "tagging_mode": "untagged" + }, + "Vlan1000|Ethernet8": { + "tagging_mode": "untagged" + } + }, + "VLAN": { + "Vlan1000": { + "vlanid": "1000", + "dhcp_servers": [ + "192.0.0.1", + "192.0.0.2", + "192.0.0.3", + "192.0.0.4" + ] + } + }, + "PORT": { + "Ethernet0": { + "alias": "Eth1", + "lanes": "65, 66, 67, 68", + "description": "Ethernet0 100G link", + "speed": "100000" + }, + "Ethernet4": { + "admin_status": "up", + "alias": "fortyGigE0/4", + "description": "Servers0:eth0", + "index": "1", + "lanes": "29,30,31,32", + "mtu": "9100", + "pfc_asym": "off", + "speed": "40000" + }, + "Ethernet8": { + "admin_status": "up", + "alias": "fortyGigE0/8", + "description": "Servers1:eth0", + "index": "2", + "lanes": "33,34,35,36", + "mtu": "9100", + "pfc_asym": "off", + "speed": "40000" + } + } +} diff --git a/tests/config_reload_input/init_cfg.json b/tests/config_reload_input/init_cfg.json new file mode 100644 index 0000000000..968db5b8f8 --- /dev/null +++ b/tests/config_reload_input/init_cfg.json @@ -0,0 +1,68 @@ +{ + "DEVICE_METADATA": { + "localhost": { + "buffer_model": "traditional", + "default_bgp_status": "up", + "default_pfcwd_status": "disable" + } + }, + "CRM": { + "Config": { + "polling_interval": "300", + "ipv4_route_threshold_type": "percentage", + "ipv4_route_low_threshold": "70", + "ipv4_route_high_threshold": "85", + "ipv6_route_threshold_type": "percentage", + "ipv6_route_low_threshold": "70", + "ipv6_route_high_threshold": "85", + "ipv4_nexthop_threshold_type": "percentage", + "ipv4_nexthop_low_threshold": "70", + "ipv4_nexthop_high_threshold": "85", + "ipv6_nexthop_threshold_type": "percentage", + "ipv6_nexthop_low_threshold": "70", + "ipv6_nexthop_high_threshold": "85", + "ipv4_neighbor_threshold_type": "percentage", + "ipv4_neighbor_low_threshold": "70", + "ipv4_neighbor_high_threshold": "85", + "ipv6_neighbor_threshold_type": "percentage", + "ipv6_neighbor_low_threshold": "70", + "ipv6_neighbor_high_threshold": "85", + "nexthop_group_member_threshold_type": "percentage", + "nexthop_group_member_low_threshold": "70", + "nexthop_group_member_high_threshold": "85", + "nexthop_group_threshold_type": "percentage", + "nexthop_group_low_threshold": "70", + "nexthop_group_high_threshold": "85", + "acl_table_threshold_type": "percentage", + "acl_table_low_threshold": "70", + "acl_table_high_threshold": "85", + "acl_group_threshold_type": "percentage", + "acl_group_low_threshold": "70", + "acl_group_high_threshold": "85", + "acl_entry_threshold_type": "percentage", + "acl_entry_low_threshold": "70", + "acl_entry_high_threshold": "85", + "acl_counter_threshold_type": "percentage", + "acl_counter_low_threshold": "70", + "acl_counter_high_threshold": "85", + "fdb_entry_threshold_type": "percentage", + "fdb_entry_low_threshold": "70", + "fdb_entry_high_threshold": "85", + "snat_entry_threshold_type": "percentage", + "snat_entry_low_threshold": "70", + "snat_entry_high_threshold": "85", + "dnat_entry_threshold_type": "percentage", + "dnat_entry_low_threshold": "70", + "dnat_entry_high_threshold": "85", + "ipmc_entry_threshold_type": "percentage", + "ipmc_entry_low_threshold": "70", + "ipmc_entry_high_threshold": "85", + "mpls_inseg_threshold_type": "percentage", + "mpls_inseg_low_threshold": "70", + "mpls_inseg_high_threshold": "85", + "mpls_nexthop_threshold_type": "percentage", + "mpls_nexthop_low_threshold": "70", + "mpls_nexthop_high_threshold": "85" + } + } +} diff --git a/tests/config_test.py b/tests/config_test.py index 826184664e..dfd1a552c4 100644 --- a/tests/config_test.py +++ b/tests/config_test.py @@ -1,16 +1,33 @@ import filecmp import imp +import importlib import os +import sys import traceback import json -from unittest import mock - import click + +from imp import load_source +from unittest import mock from click.testing import CliRunner from sonic_py_common import device_info from utilities_common.db import Db +import config.main as config + +# Add Test, module and script path. +test_path = os.path.dirname(os.path.abspath(__file__)) +modules_path = os.path.dirname(test_path) +scripts_path = os.path.join(modules_path, "scripts") +sys.path.insert(0, test_path) +sys.path.insert(0, modules_path) +sys.path.insert(0, scripts_path) +os.environ["PATH"] += os.pathsep + scripts_path + +# Config Reload input Path +mock_db_path = os.path.join(test_path, "config_reload_input") + load_minigraph_command_output="""\ Stopping SONiC target ... Running command: /usr/local/bin/sonic-cfggen -H -m --write-to-db @@ -21,6 +38,11 @@ Please note setting loaded from minigraph will be lost after system reboot. To preserve setting, run `config save`. """ +reload_config_with_sys_info_command_output= [ +'Running command: rm -rf /tmp/dropstat-*', +'Running command: /usr/local/bin/sonic-cfggen -H -k Seastone-DX010-25-50 --write-to-db', +'Running command: /usr/local/bin/sonic-cfggen -j /sonic/src/sonic-utilities/tests/config_reload_input/init_cfg.json -j /sonic/src/sonic-utilities/tests/config_reload_input/config_db.json --write-to-db' ] + def mock_run_command_side_effect(*args, **kwargs): command = args[0] @@ -36,6 +58,61 @@ def mock_run_command_side_effect(*args, **kwargs): return '' +# Load sonic-cfggen from source since /usr/local/bin/sonic-cfggen does not have .py extension. +load_source('sonic_cfggen', '/usr/local/bin/sonic-cfggen') + + +class TestConfigReload(object): + @classmethod + def setup_class(cls): + os.environ['UTILITIES_UNIT_TESTING'] = "1" + print("SETUP") + + from .mock_tables import mock_single_asic + importlib.reload(mock_single_asic) + + import config.main + importlib.reload(config.main) + + def test_config_reload(self, get_cmd_module, setup_single_broadcom_asic): + with mock.patch("utilities_common.cli.run_command", mock.MagicMock(side_effect=mock_run_command_side_effect)) as mock_run_command: + (config, show) = get_cmd_module + + jsonfile_config = os.path.join(mock_db_path, "config_db.json") + jsonfile_init_cfg = os.path.join(mock_db_path, "init_cfg.json") + + # create object + config.INIT_CFG_FILE = jsonfile_init_cfg + config.DEFAULT_CONFIG_DB_FILE = jsonfile_config + + db = Db() + runner = CliRunner() + obj = {'config_db': db.cfgdb} + + # simulate 'config reload' to provoke load_sys_info option + result = runner.invoke(config.config.commands["reload"], ["-l", "-n", "-y"], obj=obj) + + print(result.exit_code) + print(result.output) + traceback.print_tb(result.exc_info[2]) + + assert result.exit_code == 0 + + for line in reload_config_with_sys_info_command_output: + assert line in result.output + + @classmethod + def teardown_class(cls): + print("TEARDOWN") + os.environ['UTILITIES_UNIT_TESTING'] = "0" + + # change back to single asic config + from .mock_tables import dbconnector + from .mock_tables import mock_single_asic + importlib.reload(mock_single_asic) + dbconnector.load_namespace_config() + + class TestLoadMinigraph(object): @classmethod def setup_class(cls):