From 95206e2530bffa26605a735837c8d54d4dc813aa Mon Sep 17 00:00:00 2001 From: Judy Joseph Date: Wed, 7 Sep 2022 15:32:00 -0700 Subject: [PATCH 1/5] Update hostcfgd to get the macsec capability from metadata populated from "macsec_enabled" flag in platform_env.conf file. Update the feature state in CONFIG_DB/STATE_DB in namespaces as well. --- scripts/hostcfgd | 23 +++++++++++++++++++++-- 1 file changed, 21 insertions(+), 2 deletions(-) diff --git a/scripts/hostcfgd b/scripts/hostcfgd index d549d560bc83..e701deb00b9e 100755 --- a/scripts/hostcfgd +++ b/scripts/hostcfgd @@ -11,7 +11,7 @@ import signal import re import jinja2 from sonic_py_common import device_info -from swsscommon.swsscommon import ConfigDBConnector, DBConnector, Table +from swsscommon.swsscommon import ConfigDBConnector, DBConnector, Table, SonicDBConfig from swsscommon import swsscommon # FILE @@ -197,6 +197,7 @@ class FeatureHandler(object): self._config_db = config_db self._feature_state_table = feature_state_table self._device_config = device_config + self._device_macsec_support = device_info.get_macsec_support_metadata() self._cached_config = {} self.is_multi_npu = device_info.is_multi_npu() self._device_running_config = device_info.get_device_runtime_metadata() @@ -211,6 +212,7 @@ class FeatureHandler(object): device_config = {} device_config.update(self._device_config) device_config.update(self._device_running_config) + device_config.update(self._device_macsec_support) feature = Feature(feature_name, feature_cfg, device_config) self._cached_config.setdefault(feature_name, Feature(feature_name, {})) @@ -247,7 +249,7 @@ class FeatureHandler(object): device_config = {} device_config.update(self._device_config) device_config.update(self._device_running_config) - + device_config.update(self._device_macsec_support) feature = Feature(feature_name, feature_table[feature_name], device_config) self._cached_config.setdefault(feature_name, feature) @@ -460,9 +462,22 @@ class FeatureHandler(object): def resync_feature_state(self, feature): self._config_db.mod_entry('FEATURE', feature.name, {'state': feature.state}) + # resync the feature state in CONFIG_DB in namespaces in multi-asic platform + namespaces = device_info.get_namespaces() + for namespace in namespaces: + db = ConfigDBConnector(namespace=namespace) + db.connect() + db.mod_entry('FEATURE', feature.name, {'state': feature.state}) + def set_feature_state(self, feature, state): self._feature_state_table.set(feature.name, [('state', state)]) + # Update the feature state in STATE_DB in namespaces in multi-asic platform + namespaces = device_info.get_namespaces() + for namespace in namespaces: + db_conn = DBConnector(STATE_DB, 0, False, namespace); + feature_state_tbl = Table(db_conn, 'FEATURE') + feature_state_tbl.set(feature.name, [('state', state)]) class Iptables(object): def __init__(self): @@ -1470,6 +1485,10 @@ class HostConfigDaemon: self.is_multi_npu = device_info.is_multi_npu() + # Initlaize Global config that loads all database*.json + if self.is_multi_npu: + SonicDBConfig.initializeGlobalConfig() + # Initialize AAACfg self.aaacfg = AaaCfg() From fd3480371872654c048029a87d5aae37d023791e Mon Sep 17 00:00:00 2001 From: Judy Joseph Date: Tue, 27 Sep 2022 09:07:31 -0700 Subject: [PATCH 2/5] Connect to DBs in namespace once in _init_ of FeatureHandler. --- scripts/hostcfgd | 33 +++++++++++++++++++-------------- 1 file changed, 19 insertions(+), 14 deletions(-) diff --git a/scripts/hostcfgd b/scripts/hostcfgd index e701deb00b9e..7215dfbcd3b4 100755 --- a/scripts/hostcfgd +++ b/scripts/hostcfgd @@ -197,10 +197,25 @@ class FeatureHandler(object): self._config_db = config_db self._feature_state_table = feature_state_table self._device_config = device_config - self._device_macsec_support = device_info.get_macsec_support_metadata() self._cached_config = {} self.is_multi_npu = device_info.is_multi_npu() self._device_running_config = device_info.get_device_runtime_metadata() + self._device_macsec_support = device_info.get_macsec_support_metadata() + self.ns_cfg_db = {} + self.ns_feature_state_tbl = {} + + # Initlaize Global config that loads all database*.json + if self.is_multi_npu: + SonicDBConfig.initializeGlobalConfig() + namespaces = device_info.get_namespaces() + for ns in namespaces: + #Connect to ConfigDB in each namespace + self.ns_cfg_db[ns] = ConfigDBConnector(namespace=ns) + self.ns_cfg_db[ns].connect(wait_for_init=True, retry_on=True) + + #Connect to stateDB in each namespace + db_conn = DBConnector(STATE_DB, 0, False, ns); + self.ns_feature_state_tbl[ns] = Table(db_conn, 'FEATURE') def handler(self, feature_name, op, feature_cfg): if not feature_cfg: @@ -463,21 +478,15 @@ class FeatureHandler(object): self._config_db.mod_entry('FEATURE', feature.name, {'state': feature.state}) # resync the feature state in CONFIG_DB in namespaces in multi-asic platform - namespaces = device_info.get_namespaces() - for namespace in namespaces: - db = ConfigDBConnector(namespace=namespace) - db.connect() + for ns, db in self.ns_cfg_db.items(): db.mod_entry('FEATURE', feature.name, {'state': feature.state}) def set_feature_state(self, feature, state): self._feature_state_table.set(feature.name, [('state', state)]) # Update the feature state in STATE_DB in namespaces in multi-asic platform - namespaces = device_info.get_namespaces() - for namespace in namespaces: - db_conn = DBConnector(STATE_DB, 0, False, namespace); - feature_state_tbl = Table(db_conn, 'FEATURE') - feature_state_tbl.set(feature.name, [('state', state)]) + for ns, tbl in self.ns_feature_state_tbl.items(): + tbl.set(feature.name, [('state', state)]) class Iptables(object): def __init__(self): @@ -1485,10 +1494,6 @@ class HostConfigDaemon: self.is_multi_npu = device_info.is_multi_npu() - # Initlaize Global config that loads all database*.json - if self.is_multi_npu: - SonicDBConfig.initializeGlobalConfig() - # Initialize AAACfg self.aaacfg = AaaCfg() From 7c3aca03b8ac8dd48b2fc479ea651578ad355cd5 Mon Sep 17 00:00:00 2001 From: Judy Joseph Date: Thu, 29 Sep 2022 15:41:47 -0700 Subject: [PATCH 3/5] macsec_supported info is part of DEVICE_RUNTIME_METADATA itself. --- scripts/hostcfgd | 3 --- 1 file changed, 3 deletions(-) diff --git a/scripts/hostcfgd b/scripts/hostcfgd index 7215dfbcd3b4..6ae8f18f0b6b 100755 --- a/scripts/hostcfgd +++ b/scripts/hostcfgd @@ -200,7 +200,6 @@ class FeatureHandler(object): self._cached_config = {} self.is_multi_npu = device_info.is_multi_npu() self._device_running_config = device_info.get_device_runtime_metadata() - self._device_macsec_support = device_info.get_macsec_support_metadata() self.ns_cfg_db = {} self.ns_feature_state_tbl = {} @@ -227,7 +226,6 @@ class FeatureHandler(object): device_config = {} device_config.update(self._device_config) device_config.update(self._device_running_config) - device_config.update(self._device_macsec_support) feature = Feature(feature_name, feature_cfg, device_config) self._cached_config.setdefault(feature_name, Feature(feature_name, {})) @@ -264,7 +262,6 @@ class FeatureHandler(object): device_config = {} device_config.update(self._device_config) device_config.update(self._device_running_config) - device_config.update(self._device_macsec_support) feature = Feature(feature_name, feature_table[feature_name], device_config) self._cached_config.setdefault(feature_name, feature) From 185547fc78eb3dff84357cf3d73a9e00b2566b46 Mon Sep 17 00:00:00 2001 From: judyjoseph Date: Fri, 7 Oct 2022 02:53:25 -0400 Subject: [PATCH 4/5] Add UT to improve coverage, for namespace config update --- tests/common/mock_configdb.py | 2 +- tests/hostcfgd/hostcfgd_test.py | 74 +++++++++-------- tests/hostcfgd/test_vectors.py | 135 ++++++++++++++++++++++++++++++++ 3 files changed, 176 insertions(+), 35 deletions(-) diff --git a/tests/common/mock_configdb.py b/tests/common/mock_configdb.py index 1ee279a6b9ae..70b8da6afabe 100644 --- a/tests/common/mock_configdb.py +++ b/tests/common/mock_configdb.py @@ -57,5 +57,5 @@ def listen(self, init_data_handler=None): class MockDBConnector(): - def __init__(self, db, val): + def __init__(self, db, val, tcpFlag=False, name=None): pass diff --git a/tests/hostcfgd/hostcfgd_test.py b/tests/hostcfgd/hostcfgd_test.py index 93ba7b9666a6..bc79aeb164f8 100644 --- a/tests/hostcfgd/hostcfgd_test.py +++ b/tests/hostcfgd/hostcfgd_test.py @@ -28,7 +28,6 @@ hostcfgd.DBConnector = MockDBConnector hostcfgd.Table = mock.Mock() - class TestFeatureHandler(TestCase): """Test methods of `FeatureHandler` class. """ @@ -127,39 +126,46 @@ def test_sync_state_field(self, test_scenario_name, config_data, fs): with mock.patch("sonic_py_common.device_info.get_device_runtime_metadata", return_value=config_data['device_runtime_metadata']): with mock.patch("sonic_py_common.device_info.is_multi_npu", return_value=True if 'num_npu' in config_data else False): with mock.patch("sonic_py_common.device_info.get_num_npus", return_value=config_data['num_npu'] if 'num_npu' in config_data else 1): - popen_mock = mock.Mock() - attrs = config_data['popen_attributes'] - popen_mock.configure_mock(**attrs) - mocked_subprocess.Popen.return_value = popen_mock - - device_config = {} - device_config['DEVICE_METADATA'] = MockConfigDb.CONFIG_DB['DEVICE_METADATA'] - device_config.update(config_data['device_runtime_metadata']) - - feature_handler = hostcfgd.FeatureHandler(MockConfigDb(), feature_state_table_mock, device_config) - - feature_table = MockConfigDb.CONFIG_DB['FEATURE'] - feature_handler.sync_state_field(feature_table) - - feature_systemd_name_map = {} - for feature_name in feature_table.keys(): - feature = hostcfgd.Feature(feature_name, feature_table[feature_name], device_config) - feature_names, _ = feature_handler.get_multiasic_feature_instances(feature) - feature_systemd_name_map[feature_name] = feature_names - - is_any_difference = self.checks_config_table(MockConfigDb.get_config_db()['FEATURE'], - config_data['expected_config_db']['FEATURE']) - assert is_any_difference, "'FEATURE' table in 'CONFIG_DB' is modified unexpectedly!" - - feature_table_state_db_calls = self.get_state_db_set_calls(feature_table) - - self.checks_systemd_config_file(config_data['config_db']['FEATURE'], feature_systemd_name_map) - mocked_subprocess.check_call.assert_has_calls(config_data['enable_feature_subprocess_calls'], - any_order=True) - mocked_subprocess.check_call.assert_has_calls(config_data['daemon_reload_subprocess_call'], - any_order=True) - feature_state_table_mock.set.assert_has_calls(feature_table_state_db_calls) - self.checks_systemd_config_file(config_data['config_db']['FEATURE'], feature_systemd_name_map) + with mock.patch("sonic_py_common.device_info.get_namespaces", return_value=["asic{}".format(a) for a in range(config_data['num_npu'])] if 'num_npu' in config_data else []): + popen_mock = mock.Mock() + attrs = config_data['popen_attributes'] + popen_mock.configure_mock(**attrs) + mocked_subprocess.Popen.return_value = popen_mock + + device_config = {} + device_config['DEVICE_METADATA'] = MockConfigDb.CONFIG_DB['DEVICE_METADATA'] + device_config.update(config_data['device_runtime_metadata']) + + feature_handler = hostcfgd.FeatureHandler(MockConfigDb(), feature_state_table_mock, device_config) + feature_table = MockConfigDb.CONFIG_DB['FEATURE'] + feature_handler.sync_state_field(feature_table) + + feature_systemd_name_map = {} + for feature_name in feature_table.keys(): + feature = hostcfgd.Feature(feature_name, feature_table[feature_name], device_config) + feature_names, _ = feature_handler.get_multiasic_feature_instances(feature) + feature_systemd_name_map[feature_name] = feature_names + + is_any_difference = self.checks_config_table(MockConfigDb.get_config_db()['FEATURE'], + config_data['expected_config_db']['FEATURE']) + assert is_any_difference, "'FEATURE' table in 'CONFIG_DB' is modified unexpectedly!" + + if 'num_npu' in config_data: + for ns in range(config_data['num_npu']): + namespace = "asic{}".format(ns) + is_any_difference = self.checks_config_table(feature_handler.ns_cfg_db[namespace].get_config_db()['FEATURE'], + config_data['expected_config_db']['FEATURE']) + assert is_any_difference, "'FEATURE' table in 'CONFIG_DB' in namespace {} is modified unexpectedly!".format(namespace) + + feature_table_state_db_calls = self.get_state_db_set_calls(feature_table) + + self.checks_systemd_config_file(config_data['config_db']['FEATURE'], feature_systemd_name_map) + mocked_subprocess.check_call.assert_has_calls(config_data['enable_feature_subprocess_calls'], + any_order=True) + mocked_subprocess.check_call.assert_has_calls(config_data['daemon_reload_subprocess_call'], + any_order=True) + feature_state_table_mock.set.assert_has_calls(feature_table_state_db_calls) + self.checks_systemd_config_file(config_data['config_db']['FEATURE'], feature_systemd_name_map) @parameterized.expand(HOSTCFGD_TEST_VECTOR) @patchfs diff --git a/tests/hostcfgd/test_vectors.py b/tests/hostcfgd/test_vectors.py index 4f6dca3056a6..0a39b48053d1 100644 --- a/tests/hostcfgd/test_vectors.py +++ b/tests/hostcfgd/test_vectors.py @@ -1030,6 +1030,141 @@ }, }, ], + [ + "Chassis_LineCard_VOQ_multinpu", + { + "num_npu": 2, + "device_runtime_metadata": { + "DEVICE_RUNTIME_METADATA": { + "CHASSIS_METADATA": { + "module_type": "linecard", + "chassis_type": "voq" + }, + "ETHERNET_PORTS_PRESENT":True, + "MACSEC_SUPPORTED":True + } + }, + "config_db": { + "DEVICE_METADATA": { + "localhost": { + "type": "SpineRouter", + } + }, + "KDUMP": { + "config": { + "enabled": "false", + "num_dumps": "3", + "memory": "0M-2G:256M,2G-4G:320M,4G-8G:384M,8G-:448M" + } + }, + "FEATURE": { + "bgp": { + "state": "{% if not DEVICE_RUNTIME_METADATA['ETHERNET_PORTS_PRESENT'] or ('CHASSIS_METADATA' in DEVICE_RUNTIME_METADATA and DEVICE_RUNTIME_METADATA['CHASSIS_METADATA']['module_type'] in ['supervisor']) %}disabled{% else %}enabled{% endif %}", + "has_timer": "False", + "has_global_scope": "False", + "has_per_asic_scope": "True", + "auto_restart": "enabled", + "high_mem_alert": "disabled" + }, + "teamd": { + "state": "{% if not DEVICE_RUNTIME_METADATA['ETHERNET_PORTS_PRESENT'] %}disabled{% else %}enabled{% endif %}", + "has_timer": "False", + "has_global_scope": "False", + "has_per_asic_scope": "True", + "auto_restart": "enabled", + "high_mem_alert": "disabled" + }, + "lldp": { + "state": "enabled", + "has_timer": "False", + "has_global_scope": "True", + "has_per_asic_scope": "{% if not DEVICE_RUNTIME_METADATA['ETHERNET_PORTS_PRESENT'] or ('CHASSIS_METADATA' in DEVICE_RUNTIME_METADATA and DEVICE_RUNTIME_METADATA['CHASSIS_METADATA']['module_type'] in ['supervisor']) %}False{% else %}True{% endif %}", + "auto_restart": "enabled", + "high_mem_alert": "disabled" + }, + "macsec": { + "state": "{% if 'type' in DEVICE_METADATA['localhost'] and DEVICE_METADATA['localhost']['type'] == 'SpineRouter' and DEVICE_RUNTIME_METADATA['MACSEC_SUPPORTED'] %}enabled{% else %}disabled{% endif %}", + "has_timer": "False", + "has_global_scope": "False", + "has_per_asic_scope": "True", + "auto_restart": "enabled", + "high_mem_alert": "disabled" + } + }, + }, + "expected_config_db": { + "FEATURE": { + "bgp": { + "auto_restart": "enabled", + "has_global_scope": "False", + "has_per_asic_scope": "True", + "has_timer": "False", + "high_mem_alert": "disabled", + "state": "enabled" + }, + "teamd": { + "auto_restart": "enabled", + "has_global_scope": "False", + "has_per_asic_scope": "True", + "has_timer": "False", + "high_mem_alert": "disabled", + "state": "enabled" + }, + "lldp": { + "auto_restart": "enabled", + "has_global_scope": "True", + "has_per_asic_scope": "True", + "has_timer": "False", + "high_mem_alert": "disabled", + "state": "enabled" + }, + "macsec": { + "auto_restart": "enabled", + "has_global_scope": "False", + "has_per_asic_scope": "True", + "has_timer": "False", + "high_mem_alert": "disabled", + "state": "enabled" + } + }, + }, + "enable_feature_subprocess_calls": [ + call('sudo systemctl unmask bgp@0.service', shell=True), + call('sudo systemctl enable bgp@0.service', shell=True), + call('sudo systemctl start bgp@0.service', shell=True), + call('sudo systemctl unmask bgp@1.service', shell=True), + call('sudo systemctl enable bgp@1.service', shell=True), + call('sudo systemctl start bgp@1.service', shell=True), + call('sudo systemctl unmask teamd@0.service', shell=True), + call('sudo systemctl enable teamd@0.service', shell=True), + call('sudo systemctl start teamd@0.service', shell=True), + call('sudo systemctl unmask teamd@1.service', shell=True), + call('sudo systemctl enable teamd@1.service', shell=True), + call('sudo systemctl start teamd@1.service', shell=True), + call('sudo systemctl unmask lldp.service', shell=True), + call('sudo systemctl enable lldp.service', shell=True), + call('sudo systemctl start lldp.service', shell=True), + call('sudo systemctl unmask lldp@0.service', shell=True), + call('sudo systemctl enable lldp@0.service', shell=True), + call('sudo systemctl start lldp@0.service', shell=True), + call('sudo systemctl unmask lldp@1.service', shell=True), + call('sudo systemctl enable lldp@1.service', shell=True), + call('sudo systemctl start lldp@1.service', shell=True), + call('sudo systemctl unmask macsec@0.service', shell=True), + call('sudo systemctl enable macsec@0.service', shell=True), + call('sudo systemctl start macsec@0.service', shell=True), + call('sudo systemctl unmask macsec@1.service', shell=True), + call('sudo systemctl enable macsec@1.service', shell=True), + call('sudo systemctl start macsec@1.service', shell=True) + ], + "daemon_reload_subprocess_call": [ + call("sudo systemctl daemon-reload", shell=True), + ], + "popen_attributes": { + 'communicate.return_value': ('output', 'error') + }, + }, + ] ] From 31c61080b829900b3b825a869fcdda6e1e7320fc Mon Sep 17 00:00:00 2001 From: judyjoseph Date: Wed, 12 Oct 2022 16:18:09 -0400 Subject: [PATCH 5/5] Sync has_per_asic_scope attribute to config_db in all namespaces for multi-asic --- scripts/hostcfgd | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/scripts/hostcfgd b/scripts/hostcfgd index 6ae8f18f0b6b..53cd841df474 100755 --- a/scripts/hostcfgd +++ b/scripts/hostcfgd @@ -348,6 +348,10 @@ class FeatureHandler(object): self.set_feature_state(feature, self.FEATURE_STATE_FAILED) return self._config_db.mod_entry('FEATURE', feature_config.name, {'has_per_asic_scope': str(feature_config.has_per_asic_scope)}) + + # sync has_per_asic_scope to CONFIG_DB in namespaces in multi-asic platform + for ns, db in self.ns_cfg_db.items(): + db.mod_entry('FEATURE', feature_config.name, {'has_per_asic_scope': str(feature_config.has_per_asic_scope)}) def update_systemd_config(self, feature_config): """Updates `Restart=` field in feature's systemd configuration file @@ -474,14 +478,14 @@ class FeatureHandler(object): def resync_feature_state(self, feature): self._config_db.mod_entry('FEATURE', feature.name, {'state': feature.state}) - # resync the feature state in CONFIG_DB in namespaces in multi-asic platform + # resync the feature state to CONFIG_DB in namespaces in multi-asic platform for ns, db in self.ns_cfg_db.items(): db.mod_entry('FEATURE', feature.name, {'state': feature.state}) def set_feature_state(self, feature, state): self._feature_state_table.set(feature.name, [('state', state)]) - # Update the feature state in STATE_DB in namespaces in multi-asic platform + # Update the feature state to STATE_DB in namespaces in multi-asic platform for ns, tbl in self.ns_feature_state_tbl.items(): tbl.set(feature.name, [('state', state)])