On Mon, Jul 17, 2023 at 9:31 PM Jeremy Spewock <jspew...@iol.unh.edu> wrote:
> > > On Mon, Jul 17, 2023 at 10:50 AM Juraj Linkeš <juraj.lin...@pantheon.tech> > wrote: > >> I found additional things while working with the interactive shell code. >> >> On Thu, Jul 13, 2023 at 6:54 PM <jspew...@iol.unh.edu> wrote: >> > >> > From: Jeremy Spewock <jspew...@iol.unh.edu> >> > >> > Adds a new test suite for running smoke tests that verify general >> > configuration aspects of the system under test. If any of these tests >> > fail, the DTS execution terminates as part of a "fail-fast" model. >> > >> > Signed-off-by: Jeremy Spewock <jspew...@iol.unh.edu> >> > --- >> >> <snip> >> >> > diff --git a/dts/framework/config/conf_yaml_schema.json >> b/dts/framework/config/conf_yaml_schema.json >> > index ca2d4a1ef2..61f52b4365 100644 >> > --- a/dts/framework/config/conf_yaml_schema.json >> > +++ b/dts/framework/config/conf_yaml_schema.json >> <snip> >> > @@ -211,8 +332,27 @@ >> > ] >> > } >> > }, >> > + "skip_smoke_tests": { >> > + "description": "Optional field that allows you to skip >> smoke testing", >> > + "type": "boolean" >> > + }, >> > "system_under_test": { >> > - "$ref": "#/definitions/node_name" >> > + "type":"object", >> > + "properties": { >> > + "node_name": { >> > + "$ref": "#/definitions/node_name" >> > + }, >> > + "vdevs": { >> > + "description": "Opentional list of names of vdevs to >> be used in execution", >> >> Typo in Opentional >> >> > + "type": "array", >> > + "items": { >> > + "type": "string" >> > + } >> > + } >> > + }, >> > + "required": [ >> > + "node_name" >> > + ] >> > } >> > }, >> > "additionalProperties": false, >> > diff --git a/dts/framework/dts.py b/dts/framework/dts.py >> > index 0502284580..7b09d8fba8 100644 >> > --- a/dts/framework/dts.py >> > +++ b/dts/framework/dts.py >> <snip> >> > @@ -118,14 +124,15 @@ def _run_build_target( >> > >> > try: >> > sut_node.set_up_build_target(build_target) >> > - result.dpdk_version = sut_node.dpdk_version >> > + # result.dpdk_version = sut_node.dpdk_version >> > + >> build_target_result.add_build_target_versions(sut_node.get_build_target_info()) >> >> The additions to execution_result and build_target_result are >> inconsistent - one modifies __init__, the other doesn't. The time at >> which we have the info available is different, which is the reason for >> the inconsistency, but I'd rather make all results classes as >> consistent as possible - create from config, add other info after >> that. >> >> > build_target_result.update_setup(Result.PASS) >> > except Exception as e: >> > dts_logger.exception("Build target setup failed.") >> > build_target_result.update_setup(Result.FAIL, e) >> > >> > else: >> > - _run_suites(sut_node, execution, build_target_result) >> > + _run_all_suites(sut_node, execution, build_target_result) >> > >> > finally: >> > try: >> >> <snip> >> >> > diff --git a/dts/framework/remote_session/remote/__init__.py >> b/dts/framework/remote_session/remote/__init__.py >> > index 8a1512210a..03fd309f2b 100644 >> > --- a/dts/framework/remote_session/remote/__init__.py >> > +++ b/dts/framework/remote_session/remote/__init__.py >> > @@ -1,16 +1,26 @@ >> > # SPDX-License-Identifier: BSD-3-Clause >> > # Copyright(c) 2023 PANTHEON.tech s.r.o. >> > +# Copyright(c) 2023 University of New Hampshire >> > >> > # pylama:ignore=W0611 >> > >> > from framework.config import NodeConfiguration >> > from framework.logger import DTSLOG >> > >> > +from .interactive_remote_session import InteractiveRemoteSession >> > +from .interactive_shell import InteractiveShell >> > from .remote_session import CommandResult, RemoteSession >> > from .ssh_session import SSHSession >> > +from .testpmd_shell import TestPmdDevice, TestPmdShell >> > >> > >> > def create_remote_session( >> > node_config: NodeConfiguration, name: str, logger: DTSLOG >> > ) -> RemoteSession: >> > return SSHSession(node_config, name, logger) >> > + >> > + >> > +def create_interactive_session( >> > + node_config: NodeConfiguration, name: str, logger: DTSLOG >> > +) -> InteractiveRemoteSession: >> >> The name parameter is not used, remove it please. >> >> > + return InteractiveRemoteSession(node_config, logger) >> >> <snip> >> >> > diff --git a/dts/framework/remote_session/remote/testpmd_shell.py >> b/dts/framework/remote_session/remote/testpmd_shell.py >> > new file mode 100644 >> > index 0000000000..c0261c00f6 >> > --- /dev/null >> > +++ b/dts/framework/remote_session/remote/testpmd_shell.py >> > @@ -0,0 +1,74 @@ >> > +# SPDX-License-Identifier: BSD-3-Clause >> > +# Copyright(c) 2023 University of New Hampshire >> > + >> > +from pathlib import PurePath >> > + >> > +from paramiko import SSHClient # type: ignore >> > + >> > +from framework.logger import DTSLOG >> > +from framework.settings import SETTINGS >> > + >> > +from .interactive_shell import InteractiveShell >> > + >> > + >> > +class TestPmdDevice(object): >> > + pci_address: str >> > + >> > + def __init__(self, pci_address: str): >> > + self.pci_address = pci_address >> > + >> > + def __str__(self) -> str: >> > + return self.pci_address >> > + >> > + >> > +class TestPmdShell(InteractiveShell): >> > + expected_prompt: str = "testpmd>" >> > + _eal_flags: str >> > + >> > + def __init__( >> > + self, >> > + interactive_session: SSHClient, >> > + logger: DTSLOG, >> > + path_to_testpmd: PurePath, >> > + eal_flags: str, >> > + timeout: float = SETTINGS.timeout, >> > + ) -> None: >> > + """Initializes an interactive testpmd session using specified >> parameters.""" >> > + self._eal_flags = eal_flags >> > + >> > + super(TestPmdShell, self).__init__( >> > + interactive_session, >> > + logger=logger, >> > + path_to_app=path_to_testpmd, >> > + timeout=timeout, >> > + ) >> > + >> > + def _start_application(self) -> None: >> > + """Starts a new interactive testpmd shell using >> _path_to_app.""" >> > + self.send_command( >> > + f"{self._path_to_app} {self._eal_flags} -- -i", >> > + ) >> > + >> > + def send_command(self, command: str, prompt: str = >> expected_prompt) -> str: >> > + """Specific way of handling the command for testpmd >> > + >> > + An extra newline character is consumed in order to force the >> current line into >> > + the stdout buffer. >> > + """ >> > + return self.send_command_get_output(f"{command}\n", prompt) >> > + >> >> We don't really need two methods with different names which basically >> do the same - both send a command and get output. >> We could use super() to modify the method in subclasses. Or we could >> introduce a new class attribute storing the command suffix, similarly >> to expected_prompt. >> >> A note on the class attributes: we should properly document them in >> InteractiveShell so that it's clear what should be considered to be >> overridden in derived classes. >> >> > + def get_devices(self) -> list[TestPmdDevice]: >> > + """Get a list of device names that are known to testpmd >> > + >> > + Uses the device info listed in testpmd and then parses the >> output to >> > + return only the names of the devices. >> > + >> > + Returns: >> > + A list of strings representing device names (e.g. >> 0000:14:00.1) >> > + """ >> > + dev_info: str = self.send_command("show device info all") >> > + dev_list: list[TestPmdDevice] = [] >> > + for line in dev_info.split("\n"): >> > + if "device name:" in line.lower(): >> > + dev_list.append(TestPmdDevice(line.strip().split(": >> ")[1].strip())) >> >> We should pass the full line to TestPmdDevice and process the line in >> the class. This splits the logic properly - the full line is needed to >> get other information about the device. >> >> > + return dev_list >> > diff --git a/dts/framework/test_result.py b/dts/framework/test_result.py >> > index 743919820c..fe1994673e 100644 >> > --- a/dts/framework/test_result.py >> > +++ b/dts/framework/test_result.py >> <snip> >> > @@ -267,14 +288,17 @@ class DTSResult(BaseResult): >> > def __init__(self, logger: DTSLOG): >> > super(DTSResult, self).__init__() >> > self.dpdk_version = None >> > + self.output = None >> > self._logger = logger >> > self._errors = [] >> > self._return_code = ErrorSeverity.NO_ERR >> > self._stats_result = None >> > self._stats_filename = os.path.join(SETTINGS.output_dir, >> "statistics.txt") >> > >> > - def add_execution(self, sut_node: NodeConfiguration) -> >> ExecutionResult: >> > - execution_result = ExecutionResult(sut_node) >> > + def add_execution( >> > + self, sut_node: NodeConfiguration, sut_version_info: NodeInfo >> > + ) -> ExecutionResult: >> > + execution_result = ExecutionResult(sut_node, sut_version_info) >> > self._inner_results.append(execution_result) >> > return execution_result >> > >> > @@ -296,7 +320,8 @@ def process(self) -> None: >> > for error in self._errors: >> > self._logger.debug(repr(error)) >> > >> > - self._stats_result = Statistics(self.dpdk_version) >> > + self._stats_result = Statistics(self.output) >> >> By changing this (and the semi-related line 121 in dts.py) we've >> changed the behavior a bit. We've talked about removing the part which >> modifies self.output (or something similar) and these (and other >> related parts of the code) look like the remnants of that change. >> Let's remove these as well. >> >> > + # add information gathered from the smoke tests to the >> statistics >> > self.add_stats(self._stats_result) >> > with open(self._stats_filename, "w+") as stats_file: >> > stats_file.write(str(self._stats_result)) >> >> <snip> >> >> > diff --git a/dts/tests/TestSuite_smoke_tests.py >> b/dts/tests/TestSuite_smoke_tests.py >> > new file mode 100644 >> > index 0000000000..9cf547205f >> > --- /dev/null >> > +++ b/dts/tests/TestSuite_smoke_tests.py >> > @@ -0,0 +1,113 @@ >> > +# SPDX-License-Identifier: BSD-3-Clause >> > +# Copyright(c) 2023 University of New Hampshire >> > + >> > +import re >> > + >> > +from framework.config import InteractiveApp, PortConfig >> > +from framework.remote_session import TestPmdDevice, TestPmdShell >> > +from framework.settings import SETTINGS >> > +from framework.test_suite import TestSuite >> > +from framework.utils import REGEX_FOR_PCI_ADDRESS >> > + >> > + >> > +class SmokeTests(TestSuite): >> > + is_blocking = True >> > + # dicts in this list are expected to have two keys: >> > + # "pci_address" and "current_driver" >> > + nics_in_node: list[PortConfig] = [] >> > + >> > + def set_up_suite(self) -> None: >> > + """ >> > + Setup: >> > + Set the build directory path and generate a list of NICs >> in the SUT node. >> > + """ >> > + self.dpdk_build_dir_path = self.sut_node.remote_dpdk_build_dir >> > + self.nics_in_node = self.sut_node.config.ports >> > + >> > + def test_unit_tests(self) -> None: >> > + """ >> > + Test: >> > + Run the fast-test unit-test suite through meson. >> > + """ >> > + self.sut_node.main_session.send_command( >> > + f"meson test -C {self.dpdk_build_dir_path} --suite >> fast-tests", >> > + 300, >> > + verify=True, >> > + ) >> >> Just a note: we can just add privileged=True to these calls to make >> then run as root. >> >> > + >> > + def test_driver_tests(self) -> None: >> > + """ >> > + Test: >> > + Run the driver-test unit-test suite through meson. >> > + """ >> > + list_of_vdevs = "" >> >> This looks more like a string of vdev args :-). I'd rename it to >> vdev_args. >> >> > + for dev in self.sut_node._execution_config.vdevs: >> >> The test suite should be working with sut_node objects, not >> configuration. We should create the VirtualDevice objects from >> _execution_config.vdevs in _set_up_execution and then only work with >> those. And maybe reset the list in _tear_down_execution. >> >> > + list_of_vdevs += f"--vdev {dev} " >> > + list_of_vdevs = list_of_vdevs[:-1] >> > + if list_of_vdevs: >> > + self._logger.info( >> > + "Running driver tests with the following virtual " >> > + f"devices: {list_of_vdevs}" >> > + ) >> > + self.sut_node.main_session.send_command( >> > + f"meson test -C {self.dpdk_build_dir_path} --suite >> driver-tests " >> > + f'--test-args "{list_of_vdevs}"', >> > + 300, >> > + verify=True, >> > + ) >> > + else: >> > + self.sut_node.main_session.send_command( >> > + f"meson test -C {self.dpdk_build_dir_path} --suite >> driver-tests", >> > + 300, >> > + verify=True, >> > + ) >> > + >> > + def test_devices_listed_in_testpmd(self) -> None: >> > + """ >> > + Test: >> > + Uses testpmd driver to verify that devices have been found >> by testpmd. >> > + """ >> > + testpmd_driver = >> self.sut_node.create_interactive_shell(InteractiveApp.testpmd) >> > + # We know it should always be a TestPmdShell but mypy doesn't >> > + assert isinstance(testpmd_driver, TestPmdShell) >> > + dev_list: list[TestPmdDevice] = testpmd_driver.get_devices() >> > + for nic in self.nics_in_node: >> > + self.verify( >> > + nic.pci in map(str, dev_list), >> >> map(str, dev_list) gets called for every nic. I'm more inclined toward >> composing the list right when assigning to dev_list. Or you could look >> into returning (from get_devices()) a derived List overloading the >> __contains__ method :-) >> >> > + f"Device {nic.pci} was not listed in testpmd's >> available devices, " >> > + "please check your configuration", >> > + ) >> > + >> > + def test_device_bound_to_driver(self) -> None: >> > + """ >> > + Test: >> > + Ensure that all drivers listed in the config are bound to >> the correct driver. >> > + """ >> > + path_to_devbind = self.sut_node.main_session.join_remote_path( >> > + self.sut_node._remote_dpdk_dir, "usertools", >> "dpdk-devbind.py" >> > + ) >> > + >> > + all_nics_in_dpdk_devbind = >> self.sut_node.main_session.send_command( >> > + f"{path_to_devbind} --status | awk >> '{REGEX_FOR_PCI_ADDRESS}'", >> > + SETTINGS.timeout, >> > + ).stdout >> > + >> > + for nic in self.nics_in_node: >> > + # This regular expression finds the line in the above >> string that starts >> > + # with the address for the nic we are on in the loop and >> then captures the >> > + # name of the driver in a group >> > + devbind_info_for_nic = re.search( >> > + f"{nic.pci}[^\\n]*drv=([\\d\\w]*) [^\\n]*", >> > + all_nics_in_dpdk_devbind, >> > + ) >> > + self.verify( >> > + devbind_info_for_nic is not None, >> > + f"Failed to find configured device ({nic.pci}) using >> dpdk-devbind.py", >> > + ) >> > + # We know this isn't None, but mypy doesn't >> > + assert devbind_info_for_nic is not None >> > + self.verify( >> > + devbind_info_for_nic.group(1) == nic.os_driver, >> > + f"Driver for device {nic.pci} does not match driver >> listed in " >> > + f"configuration (bound to >> {devbind_info_for_nic.group(1)})", >> > + ) >> > -- >> > 2.41.0 >> > >> > > Thanks for the comments, I went through and addressed them. As we talked > about today, I did also add parts from your patch in regards to the > interactive shells and I added the ability for them to use sudo. We had > talked about using a static method for this, but this would not work > because what type of session you are using is an important distinction. I > could not pass the entire OSSession class into the interactive shell, so I > instead added just a reference to the method for accessing elevated > privileges. I had to add a few # type: ignore lines for this to work due to > the issue marked here: https://github.com/python/mypy/issues/2427 . > Please see my new version. > I have some more comments on the new version, the main being that we can actually pass a static method (from the proper object thus from the proper class). I think that makes more sense since the method should've been static from the get go. I've explained a bit in the comments and I'm attaching a diff here.
diff --git a/dts/framework/remote_session/linux_session.py b/dts/framework/remote_session/linux_session.py index f13f399121..f64aa8efb0 100644 --- a/dts/framework/remote_session/linux_session.py +++ b/dts/framework/remote_session/linux_session.py @@ -14,7 +14,8 @@ class LinuxSession(PosixSession): The implementation of non-Posix compliant parts of Linux remote sessions. """ - def _get_privileged_command(self, command: str) -> str: + @staticmethod + def _get_privileged_command(command: str) -> str: return f"sudo -- sh -c '{command}'" def get_remote_cpus(self, use_first_core: bool) -> list[LogicalCore]: diff --git a/dts/framework/remote_session/os_session.py b/dts/framework/remote_session/os_session.py index f03275b21d..dad8d4da1b 100644 --- a/dts/framework/remote_session/os_session.py +++ b/dts/framework/remote_session/os_session.py @@ -100,8 +100,9 @@ def create_interactive_shell( timeout, ) + @staticmethod @abstractmethod - def _get_privileged_command(self, command: str) -> str: + def _get_privileged_command(command: str) -> str: """Modify the command so that it executes with administrative privileges. Args: diff --git a/dts/framework/remote_session/remote/interactive_shell.py b/dts/framework/remote_session/remote/interactive_shell.py index 4d9c7638a5..3512f1f8a9 100644 --- a/dts/framework/remote_session/remote/interactive_shell.py +++ b/dts/framework/remote_session/remote/interactive_shell.py @@ -22,7 +22,6 @@ class InteractiveShell: _app_args: str _default_prompt: str = "" _privileged: bool - _get_privileged_command: Callable[[str], str] # Allows for app specific extra characters to be appended to commands _command_extra_chars: str = "" path: PurePath @@ -34,7 +33,7 @@ def __init__( logger: DTSLOG, startup_command: str, privileged: bool, - _get_privileged_command: Callable[[str], str], + get_privileged_command: Callable[[str], str], app_args: str = "", timeout: float = SETTINGS.timeout, ) -> None: @@ -48,11 +47,10 @@ def __init__( self._timeout = timeout self._startup_command = startup_command self._app_args = app_args - self._get_privileged_command = _get_privileged_command # type: ignore self._privileged = privileged - self._start_application() + self._start_application(get_privileged_command) - def _start_application(self) -> None: + def _start_application(self, get_privileged_command: Callable[[str], str]) -> None: """Starts a new interactive application based on _startup_command. This method is often overridden by subclasses as their process for @@ -60,7 +58,7 @@ def _start_application(self) -> None: """ start_command = f"{self._startup_command} {self._app_args}" if self._privileged: - start_command = self._get_privileged_command(start_command) # type: ignore + start_command = get_privileged_command(start_command) self.send_command(start_command) def send_command(self, command: str, prompt: str | None = None) -> str: diff --git a/dts/framework/remote_session/remote/testpmd_shell.py b/dts/framework/remote_session/remote/testpmd_shell.py index 4abe70d421..af8ac2a057 100644 --- a/dts/framework/remote_session/remote/testpmd_shell.py +++ b/dts/framework/remote_session/remote/testpmd_shell.py @@ -2,6 +2,7 @@ # Copyright(c) 2023 University of New Hampshire from pathlib import PurePath +from typing import Callable from .interactive_shell import InteractiveShell @@ -24,10 +25,10 @@ class TestPmdShell(InteractiveShell): "\n" # We want to append an extra newline to every command ) - def _start_application(self) -> None: + def _start_application(self, get_privileged_command: Callable[[str], str]) -> None: """See "_start_application" in InteractiveShell.""" self._app_args += " -- -i" - super()._start_application() + super()._start_application(get_privileged_command) def get_devices(self) -> list[TestPmdDevice]: """Get a list of device names that are known to testpmd