From d3a4cf1f6c4dd633cfe9b07dfc26331238d79f37 Mon Sep 17 00:00:00 2001 From: ganglv <88995770+ganglyu@users.noreply.github.com> Date: Fri, 18 Nov 2022 12:18:34 +0800 Subject: [PATCH] [sonic-host-services]: Support GCU and reload (#1) Signed-off-by: Gang Lv ganglv@microsoft.com Why I did it GNMI needs to use host service to invoke generic_config_updater and config reload. How I did it Add host_modules for generic_config_updater and config reload. Add unit test for host modules. How to verify it Run unit test for sonic-host-services. --- host_modules/config_engine.py | 52 ++++++++++ host_modules/gcu.py | 75 ++++++++++++++ host_modules/showtech.py | 2 +- pytest.ini | 2 +- tests/host_modules/__init__.py | 0 tests/host_modules/config_test.py | 77 ++++++++++++++ tests/host_modules/gcu_test.py | 153 ++++++++++++++++++++++++++++ tests/host_modules/showtech_test.py | 29 ++++++ 8 files changed, 388 insertions(+), 2 deletions(-) create mode 100644 host_modules/config_engine.py create mode 100644 host_modules/gcu.py create mode 100644 tests/host_modules/__init__.py create mode 100644 tests/host_modules/config_test.py create mode 100644 tests/host_modules/gcu_test.py create mode 100644 tests/host_modules/showtech_test.py diff --git a/host_modules/config_engine.py b/host_modules/config_engine.py new file mode 100644 index 00000000..295ba1f6 --- /dev/null +++ b/host_modules/config_engine.py @@ -0,0 +1,52 @@ +"""Config command handler""" + +from host_modules import host_service +import subprocess + +MOD_NAME = 'config' +DEFAULT_CONFIG = '/etc/sonic/config_db.json' + +class Config(host_service.HostModule): + """ + DBus endpoint that executes the config command + """ + @host_service.method(host_service.bus_name(MOD_NAME), in_signature='s', out_signature='is') + def reload(self, config_db_json): + + cmd = ['/usr/local/bin/config', 'reload', '-y'] + if config_db_json and len(config_db_json.strip()): + cmd.append('/dev/stdin') + input_bytes = (config_db_json + '\n').encode('utf-8') + result = subprocess.run(cmd, input=input_bytes, shell=False, stdout=subprocess.PIPE, stderr=subprocess.PIPE) + else: + result = subprocess.run(cmd, shell=False, stdout=subprocess.PIPE, stderr=subprocess.PIPE) + msg = '' + if result.returncode: + lines = result.stderr.decode().split('\n') + for line in lines: + if 'Error' in line: + msg = line + break + return result.returncode, msg + + @host_service.method(host_service.bus_name(MOD_NAME), in_signature='s', out_signature='is') + def save(self, config_file): + + cmd = ['/usr/local/bin/config', 'save', '-y'] + if config_file and config_file != DEFAULT_CONFIG: + cmd.append(config_file) + + result = subprocess.run(cmd, shell=False, stdout=subprocess.PIPE, stderr=subprocess.PIPE) + msg = '' + if result.returncode: + lines = result.stderr.decode().split('\n') + for line in lines: + if 'Error' in line: + msg = line + break + return result.returncode, msg + +def register(): + """Return class and module name""" + return Config, MOD_NAME + diff --git a/host_modules/gcu.py b/host_modules/gcu.py new file mode 100644 index 00000000..8eb73076 --- /dev/null +++ b/host_modules/gcu.py @@ -0,0 +1,75 @@ +"""Generic config updater command handler""" + +from host_modules import host_service +import subprocess + +MOD_NAME = 'gcu' + +class GCU(host_service.HostModule): + """ + DBus endpoint that executes the generic config updater command + """ + @host_service.method(host_service.bus_name(MOD_NAME), in_signature='s', out_signature='is') + def apply_patch_db(self, patch_text): + input_bytes = (patch_text + '\n').encode('utf-8') + cmd = ['/usr/local/bin/config', 'apply-patch', '-f', 'CONFIGDB', '/dev/stdin'] + + result = subprocess.run(cmd, input=input_bytes, shell=False, stdout=subprocess.PIPE, stderr=subprocess.PIPE) + msg = '' + if result.returncode: + lines = result.stderr.decode().split('\n') + for line in lines: + if 'Error' in line: + msg = line + break + return result.returncode, msg + + @host_service.method(host_service.bus_name(MOD_NAME), in_signature='s', out_signature='is') + def apply_patch_yang(self, patch_text): + input_bytes = (patch_text + '\n').encode('utf-8') + cmd = ['/usr/local/bin/config', 'apply-patch', '-f', 'SONICYANG', '/dev/stdin'] + + result = subprocess.run(cmd, input=input_bytes, shell=False, stdout=subprocess.PIPE, stderr=subprocess.PIPE) + msg = '' + if result.returncode: + lines = result.stderr.decode().split('\n') + for line in lines: + if 'Error' in line: + msg = line + break + return result.returncode, msg + + @host_service.method(host_service.bus_name(MOD_NAME), in_signature='s', out_signature='is') + def create_checkpoint(self, checkpoint_file): + + cmd = ['/usr/local/bin/config', 'checkpoint', checkpoint_file] + + result = subprocess.run(cmd, shell=False, stdout=subprocess.PIPE, stderr=subprocess.PIPE) + msg = '' + if result.returncode: + lines = result.stderr.decode().split('\n') + for line in lines: + if 'Error' in line: + msg = line + break + return result.returncode, msg + + @host_service.method(host_service.bus_name(MOD_NAME), in_signature='s', out_signature='is') + def delete_checkpoint(self, checkpoint_file): + + cmd = ['/usr/local/bin/config', 'delete-checkpoint', checkpoint_file] + + result = subprocess.run(cmd, shell=False, stdout=subprocess.PIPE, stderr=subprocess.PIPE) + msg = '' + if result.returncode: + lines = result.stderr.decode().split('\n') + for line in lines: + if 'Error' in line: + msg = line + break + return result.returncode, msg + +def register(): + """Return class and module name""" + return GCU, MOD_NAME + diff --git a/host_modules/showtech.py b/host_modules/showtech.py index 2b603d4f..b9608fab 100644 --- a/host_modules/showtech.py +++ b/host_modules/showtech.py @@ -1,6 +1,6 @@ """Show techsupport command handler""" -import host_service +from host_modules import host_service import subprocess import re diff --git a/pytest.ini b/pytest.ini index 548dc83b..9302db8d 100644 --- a/pytest.ini +++ b/pytest.ini @@ -1,2 +1,2 @@ [pytest] -addopts = --cov=scripts --cov-report html --cov-report term --cov-report xml --ignore=tests/*/test*_vectors.py --junitxml=test-results.xml -vv +addopts = --cov=scripts --cov=host_modules --cov-report html --cov-report term --cov-report xml --ignore=tests/*/test*_vectors.py --junitxml=test-results.xml -vv diff --git a/tests/host_modules/__init__.py b/tests/host_modules/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/tests/host_modules/config_test.py b/tests/host_modules/config_test.py new file mode 100644 index 00000000..8047972c --- /dev/null +++ b/tests/host_modules/config_test.py @@ -0,0 +1,77 @@ +import sys +import os +import pytest +from unittest import mock +from host_modules import config_engine + +class TestConfigEngine(object): + @mock.patch("dbus.SystemBus") + @mock.patch("dbus.service.BusName") + @mock.patch("dbus.service.Object.__init__") + def test_reload(self, MockInit, MockBusName, MockSystemBus): + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 0 + test_msg = b"Error: this is the test message\nHello world\n" + attrs = {"returncode": test_ret, "stderr": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + config_db_json = "{}" + config_stub = config_engine.Config(config_engine.MOD_NAME) + ret, msg = config_stub.reload(config_db_json) + call_args = mock_run.call_args[0][0] + assert "reload" in call_args + assert "/dev/stdin" in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == "", "Return message is wrong" + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 1 + test_msg = b"Error: this is the test message\nHello world\n" + attrs = {"returncode": test_ret, "stderr": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + config_db_json = "{}" + config_stub = config_engine.Config(config_engine.MOD_NAME) + ret, msg = config_stub.reload(config_db_json) + call_args = mock_run.call_args[0][0] + assert "reload" in call_args + assert "/dev/stdin" in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == "Error: this is the test message", "Return message is wrong" + + @mock.patch("dbus.SystemBus") + @mock.patch("dbus.service.BusName") + @mock.patch("dbus.service.Object.__init__") + def test_save(self, MockInit, MockBusName, MockSystemBus): + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 0 + test_msg = b"Error: this is the test message\nHello world\n" + attrs = {"returncode": test_ret, "stderr": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + config_file = "test.patch" + config_stub = config_engine.Config(config_engine.MOD_NAME) + ret, msg = config_stub.save(config_file) + call_args = mock_run.call_args[0][0] + assert "save" in call_args + assert config_file in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == "", "Return message is wrong" + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 1 + test_msg = b"Error: this is the test message\nHello world\n" + attrs = {"returncode": test_ret, "stderr": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + config_file = "test.patch" + config_stub = config_engine.Config(config_engine.MOD_NAME) + ret, msg = config_stub.save(config_file) + call_args = mock_run.call_args[0][0] + assert "save" in call_args + assert config_file in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == "Error: this is the test message", "Return message is wrong" + diff --git a/tests/host_modules/gcu_test.py b/tests/host_modules/gcu_test.py new file mode 100644 index 00000000..7797f983 --- /dev/null +++ b/tests/host_modules/gcu_test.py @@ -0,0 +1,153 @@ +import sys +import os +import pytest +from unittest import mock +from host_modules import gcu + +class TestGCU(object): + @mock.patch("dbus.SystemBus") + @mock.patch("dbus.service.BusName") + @mock.patch("dbus.service.Object.__init__") + def test_apply_patch_db(self, MockInit, MockBusName, MockSystemBus): + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 0 + test_msg = b"Error: this is the test message\nHello world\n" + attrs = {"returncode": test_ret, "stderr": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + patch_text = "{}" + gcu_stub = gcu.GCU(gcu.MOD_NAME) + ret, msg = gcu_stub.apply_patch_db(patch_text) + call_args = mock_run.call_args[0][0] + assert "apply-patch" in call_args + assert "CONFIGDB" in call_args + assert '/dev/stdin' in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == "", "Return message is wrong" + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 1 + test_msg = b"Error: this is the test message\nHello world\n" + attrs = {"returncode": test_ret, "stderr": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + patch_text = "{}" + gcu_stub = gcu.GCU(gcu.MOD_NAME) + ret, msg = gcu_stub.apply_patch_db(patch_text) + call_args = mock_run.call_args[0][0] + assert "apply-patch" in call_args + assert "CONFIGDB" in call_args + assert '/dev/stdin' in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == "Error: this is the test message", "Return message is wrong" + + @mock.patch("dbus.SystemBus") + @mock.patch("dbus.service.BusName") + @mock.patch("dbus.service.Object.__init__") + def test_apply_patch_yang(self, MockInit, MockBusName, MockSystemBus): + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 0 + test_msg = b"Error: this is the test message\nHello world\n" + attrs = {"returncode": test_ret, "stderr": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + patch_text = "{}" + gcu_stub = gcu.GCU(gcu.MOD_NAME) + ret, msg = gcu_stub.apply_patch_yang(patch_text) + call_args = mock_run.call_args[0][0] + assert "apply-patch" in call_args + assert "SONICYANG" in call_args + assert '/dev/stdin' in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == "", "Return message is wrong" + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 1 + test_msg = b"Error: this is the test message\nHello world\n" + attrs = {"returncode": test_ret, "stderr": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + patch_text = "{}" + gcu_stub = gcu.GCU(gcu.MOD_NAME) + ret, msg = gcu_stub.apply_patch_yang(patch_text) + call_args = mock_run.call_args[0][0] + assert "apply-patch" in call_args + assert "SONICYANG" in call_args + assert '/dev/stdin' in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == "Error: this is the test message", "Return message is wrong" + + @mock.patch("dbus.SystemBus") + @mock.patch("dbus.service.BusName") + @mock.patch("dbus.service.Object.__init__") + def test_create_checkpoint(self, MockInit, MockBusName, MockSystemBus): + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 0 + test_msg = b"Error: this is the test message\nHello world\n" + attrs = {"returncode": test_ret, "stderr": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + cp_name = "test_name" + gcu_stub = gcu.GCU(gcu.MOD_NAME) + ret, msg = gcu_stub.create_checkpoint(cp_name) + call_args = mock_run.call_args[0][0] + assert "checkpoint" in call_args + assert "delete-checkpoint" not in call_args + assert cp_name in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == "", "Return message is wrong" + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 1 + test_msg = b"Error: this is the test message\nHello world\n" + attrs = {"returncode": test_ret, "stderr": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + cp_name = "test_name" + gcu_stub = gcu.GCU(gcu.MOD_NAME) + ret, msg = gcu_stub.create_checkpoint(cp_name) + call_args = mock_run.call_args[0][0] + assert "checkpoint" in call_args + assert "delete-checkpoint" not in call_args + assert cp_name in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == "Error: this is the test message", "Return message is wrong" + + @mock.patch("dbus.SystemBus") + @mock.patch("dbus.service.BusName") + @mock.patch("dbus.service.Object.__init__") + def test_delete_checkpoint(self, MockInit, MockBusName, MockSystemBus): + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 0 + test_msg = b"Error: this is the test message\nHello world\n" + attrs = {"returncode": test_ret, "stderr": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + cp_name = "test_name" + gcu_stub = gcu.GCU(gcu.MOD_NAME) + ret, msg = gcu_stub.delete_checkpoint(cp_name) + call_args = mock_run.call_args[0][0] + assert "delete-checkpoint" in call_args + assert cp_name in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == "", "Return message is wrong" + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 1 + test_msg = b"Error: this is the test message\nHello world\n" + attrs = {"returncode": test_ret, "stderr": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + cp_name = "test_name" + gcu_stub = gcu.GCU(gcu.MOD_NAME) + ret, msg = gcu_stub.delete_checkpoint(cp_name) + call_args = mock_run.call_args[0][0] + assert "delete-checkpoint" in call_args + assert cp_name in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == "Error: this is the test message", "Return message is wrong" + \ No newline at end of file diff --git a/tests/host_modules/showtech_test.py b/tests/host_modules/showtech_test.py new file mode 100644 index 00000000..ae376863 --- /dev/null +++ b/tests/host_modules/showtech_test.py @@ -0,0 +1,29 @@ +import sys +import os +import pytest +from unittest import mock +from host_modules import showtech + +class TestShowtech(object): + @mock.patch("dbus.SystemBus") + @mock.patch("dbus.service.BusName") + @mock.patch("dbus.service.Object.__init__") + def test_info(self, MockInit, MockBusName, MockSystemBus): + with mock.patch("subprocess.run") as mock_run: + res_mock = mock.Mock() + test_ret = 0 + exp_msg = "/var/abcdumpdef.gz" + test_msg = "######" + exp_msg + "-------" + date_msg = "yesterday once more" + attrs = {"returncode": test_ret, "stdout": test_msg} + res_mock.configure_mock(**attrs) + mock_run.return_value = res_mock + patch_file = "test.patch" + showtech_stub = showtech.Showtech(showtech.MOD_NAME) + ret, msg = showtech_stub.info(date_msg) + call_args = mock_run.call_args[0][0] + assert "/usr/local/bin/generate_dump" in call_args + assert date_msg in call_args + assert ret == test_ret, "Return value is wrong" + assert msg == exp_msg, "Return message is wrong" + \ No newline at end of file