diff --git a/lisa/tools/__init__.py b/lisa/tools/__init__.py index eb865b2a96..95d0f48fda 100644 --- a/lisa/tools/__init__.py +++ b/lisa/tools/__init__.py @@ -16,6 +16,7 @@ ) from .aria import Aria +from .b4 import B4 from .blkid import Blkid from .bzip2 import Bzip2 from .cargo import Cargo @@ -130,6 +131,7 @@ __all__ = [ "AptAddRepository", "Aria", + "B4", "Blkid", "Bzip2", "Cargo", diff --git a/lisa/tools/b4.py b/lisa/tools/b4.py new file mode 100644 index 0000000000..3704766374 --- /dev/null +++ b/lisa/tools/b4.py @@ -0,0 +1,62 @@ +import pathlib +import re +from typing import List, Type + +from lisa.executable import Tool +from lisa.operating_system import Debian +from lisa.tools.git import Git +from lisa.tools.python import Pip +from lisa.util import LisaException, find_group_in_lines + + +class B4(Tool): + # Output log is of the form + # git am /mnt/code/linux/v2_20241029_xxx_offers.mbx + _output_file_pattern = re.compile( + r"^.*git.*/(?P[\w-]+\.mbx).*$", re.MULTILINE + ) + + @property + def command(self) -> str: + return "b4" + + @property + def dependencies(self) -> List[Type[Tool]]: + return [Git] + + @property + def can_install(self) -> bool: + return True + + def _install(self) -> bool: + if isinstance(self.node.os, Debian): + self.node.os.install_packages("b4") + installed = self._check_exists() + if not installed: + pip = self.node.tools[Pip] + pip.install_packages("b4", install_to_user=True) + return self._check_exists() + + def apply( + self, message_id: str, cwd: pathlib.PurePath, sudo: bool = False + ) -> pathlib.PurePath: + """ + Download the patch using the message id and apply it to the git repository. + """ + result = self.run( + f"am -o '{cwd}' '{message_id}'", + force_run=True, + expected_exit_code=0, + sudo=sudo, + ) + filename = find_group_in_lines( + lines=result.stdout, pattern=self._output_file_pattern, single_line=False + ).get("filename") + if not filename: + raise LisaException("Failed to get filename from b4 am output") + filepath = pathlib.PurePath(cwd, filename) + + git = self.node.tools[Git] + git.apply(cwd=cwd, patches=filepath) + + return filepath diff --git a/lisa/transformers/kernel_source_installer.py b/lisa/transformers/kernel_source_installer.py index aa245a2d56..69db736207 100644 --- a/lisa/transformers/kernel_source_installer.py +++ b/lisa/transformers/kernel_source_installer.py @@ -11,7 +11,7 @@ from lisa.base_tools import Mv from lisa.node import Node from lisa.operating_system import CBLMariner, CpuArchitecture, Redhat, Ubuntu -from lisa.tools import Cp, Echo, Git, Make, Sed, Uname +from lisa.tools import B4, Cp, Echo, Git, Make, Sed, Uname from lisa.tools.gcc import Gcc from lisa.tools.lscpu import Lscpu from lisa.util import ( @@ -81,6 +81,12 @@ class PatchModifierSchema(BaseModifierSchema): file_pattern: str = "*.patch" +@dataclass_json() +@dataclass +class B4PatchModifierSchema(BaseModifierSchema): + message_id: str = field(default="", metadata=field_metadata(required=True)) + + @dataclass_json() @dataclass class SourceInstallerSchema(BaseInstallerSchema): @@ -572,3 +578,19 @@ def _get_code_path(path: str, node: Node, default_name: str) -> PurePath: code_path = node.working_path / default_name return code_path + + +class B4PatchModifier(BaseModifier): + @classmethod + def type_name(cls) -> str: + return "b4_patch" + + @classmethod + def type_schema(cls) -> Type[schema.TypedSchema]: + return B4PatchModifierSchema + + def modify(self) -> None: + runbook: B4PatchModifierSchema = self.runbook + b4 = self._node.tools[B4] + message_id = runbook.message_id + b4.apply(message_id=message_id, cwd=self._code_path)