From 94dfa3515246e2691d99b17734f9a2e770ff50de Mon Sep 17 00:00:00 2001 From: Joseph P Date: Sun, 21 Apr 2024 21:11:35 +0000 Subject: [PATCH] refactor: update type hints, arguments list, container engine list --- .github/workflows/github-ci.yml | 1 + .gitlab-ci.yml | 1 + Dockerfile | 4 +- README.md | 10 +- builder/__main__.py | 70 ++----- builder/clients/github.py | 6 +- builder/clients/pa.py | 1 - builder/clients/rom_api.py | 13 +- builder/commands/assets.py | 3 - builder/commands/bundle.py | 37 ++-- builder/configs/__init__.py | 4 +- builder/configs/{arguments.py => argument.py} | 32 +-- .../configs/{directories.py => directory.py} | 0 builder/core/assets_collector.py | 33 +-- builder/core/kernel_builder.py | 115 +++++----- builder/engines/__init__.py | 3 +- builder/engines/docker_engine.py | 44 ---- ...ntainer_engine.py => generic_container.py} | 116 +++++----- builder/engines/podman_engine.py | 5 - builder/interfaces/__init__.py | 2 +- builder/interfaces/clients.py | 3 +- builder/interfaces/commands.py | 35 ++-- builder/interfaces/engines.py | 30 +-- builder/interfaces/modules.py | 2 +- builder/tools/cleaning.py | 14 +- builder/tools/commands.py | 24 +-- builder/tools/fileoperations.py | 42 ++-- builder/tools/messages.py | 21 +- builder/utils/bridge.py | 4 +- builder/utils/resource_manager.py | 49 +++-- poetry.lock | 198 +++++++++--------- pyproject.toml | 5 + scripts/multi_build.py | 4 +- scripts/run_tests.py | 31 +-- .../unit/wrapper/core/test_kernel_builder.py | 2 +- 35 files changed, 450 insertions(+), 514 deletions(-) rename builder/configs/{arguments.py => argument.py} (63%) rename builder/configs/{directories.py => directory.py} (100%) delete mode 100644 builder/engines/docker_engine.py rename builder/engines/{container_engine.py => generic_container.py} (59%) delete mode 100644 builder/engines/podman_engine.py diff --git a/.github/workflows/github-ci.yml b/.github/workflows/github-ci.yml index 6620a78..c7e1319 100644 --- a/.github/workflows/github-ci.yml +++ b/.github/workflows/github-ci.yml @@ -37,6 +37,7 @@ jobs: python3 -m poetry config virtualenvs.create false python3 -m poetry install --no-root export PYTHONPATH=$(pwd) + python3 scripts/run_tests.py python3 scripts/multi_build.py - name: Publish Artifacts uses: actions/upload-artifact@v3 diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index 791772e..c37c223 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -18,6 +18,7 @@ job-build: - python3 -m poetry config virtualenvs.create false - python3 -m poetry install --no-root - export PYTHONPATH=$(pwd) + - python3 scripts/run_tests.py - python3 scripts/multi_build.py artifacts: paths: diff --git a/Dockerfile b/Dockerfile index b180327..83ace15 100755 --- a/Dockerfile +++ b/Dockerfile @@ -9,13 +9,11 @@ COPY . ${WDIR} WORKDIR ${WDIR} ENV PYTHONPATH ${WDIR} -# install system packages; -# NeoVim is added for debugging sessions. +# install system packages RUN \ apt-get update \ && \ apt-get install -y \ - neovim \ curl \ wget \ git \ diff --git a/README.md b/README.md index 0d13ff7..3f47a72 100755 --- a/README.md +++ b/README.md @@ -82,14 +82,14 @@ The custom build wrapper (aka "builder") consists of 2 core components and 3 pri Components: -- kernel builder; -- assets collector. +- `kernel_builder`; +- `assets_collector`. Commands: -- kernel; -- assets; -- bundle. +- `kernel`; +- `assets`; +- `bundle`. ```help $ python3 builder --help diff --git a/builder/__main__.py b/builder/__main__.py index c32dcca..f58188b 100644 --- a/builder/__main__.py +++ b/builder/__main__.py @@ -4,11 +4,12 @@ import json import argparse -from builder.tools import cleaning as cm, messages as msg +from builder.tools import cleaning as cm, messages as msg, commands as ccmd from builder.configs import ArgumentConfig, DirectoryConfig as dcfg -from builder.engines import DockerEngine, PodmanEngine +from builder.engines import GenericContainerEngine from builder.commands import KernelCommand, AssetsCommand, BundleCommand + def parse_args() -> argparse.Namespace: """Parse the script arguments.""" # show the 'help' message if no arguments supplied @@ -31,14 +32,11 @@ def parse_args() -> argparse.Namespace: help_codename = "select device codename" help_benv = "select build environment" help_clean = "remove Docker/Podman image from the host machine after build" - help_loglvl = "select log level" choices_benv = ("local", "docker", "podman") - choices_loglvl = ("normal", "verbose", "quiet") choices_base = ("los", "pa", "x", "aosp") help_logfile = "save logs to a file" help_ksu = "add KernelSU support" help_lkv = "select Linux Kernel Version" - default_loglvl = "normal" # kernel parser_kernel.add_argument( "--build-env", @@ -75,13 +73,6 @@ def parse_args() -> argparse.Namespace: dest="clean_image", help=help_clean ) - parser_kernel.add_argument( - "--log-level", - dest="loglvl", - choices=choices_loglvl, - default=default_loglvl, - help=help_loglvl - ) parser_kernel.add_argument( "-o", "--output", dest="outlog", @@ -136,13 +127,6 @@ def parse_args() -> argparse.Namespace: action="store_true", help="autoclean 'assets' folder if it exists" ) - parser_assets.add_argument( - "--log-level", - dest="loglvl", - choices=choices_loglvl, - default=default_loglvl, - help=help_loglvl - ) parser_assets.add_argument( "-o", "--output", dest="outlog", @@ -197,13 +181,6 @@ def parse_args() -> argparse.Namespace: dest="clean_image", help=help_clean ) - parser_bundle.add_argument( - "--log-level", - dest="loglvl", - choices=choices_loglvl, - default=default_loglvl, - help=help_loglvl - ) parser_bundle.add_argument( "-o", "--output", dest="outlog", @@ -224,7 +201,6 @@ def main(args: argparse.Namespace) -> None: if args.clean_root: cm.root() sys.exit(0) - os.environ["LOGLEVEL"] = args.loglvl # define env variable with kernel version with open(dcfg.root / "pyproject.toml", encoding="utf-8") as f: os.environ["KVERSION"] = f.read().split("version = \"")[1].split("\"")[0] @@ -238,39 +214,37 @@ def main(args: argparse.Namespace) -> None: if args.outlog in os.listdir(): os.remove(args.outlog) os.environ["OSTREAM"] = args.outlog - msg.outputstream() # determine the build match args.benv: - case "docker": - DockerEngine(**json.loads(acfg.model_dump_json())).run() - case "podman": - PodmanEngine(**json.loads(acfg.model_dump_json())).run() + case "docker" | "podman": + with GenericContainerEngine(**json.loads(acfg.model_dump_json())) as engined_cmd: + ccmd.launch(engined_cmd) case "local": match args.command: case "kernel": KernelCommand( - codename = acfg.codename, - base = acfg.base, - lkv = acfg.lkv, - clean_kernel = acfg.clean_kernel, - ksu = acfg.ksu, + codename = args.codename, + base = args.base, + lkv = args.lkv, + clean_kernel = args.clean_kernel, + ksu = args.ksu, ).run() case "assets": AssetsCommand( - codename = acfg.codename, - base = acfg.base, - chroot = acfg.chroot, - clean_assets = acfg.clean_assets, - rom_only = acfg.rom_only, - ksu = acfg.ksu, + codename = args.codename, + base = args.base, + chroot = args.chroot, + clean_assets = args.clean_assets, + rom_only = args.rom_only, + ksu = args.ksu, ).run() case "bundle": BundleCommand( - codename = acfg.codename, - base = acfg.base, - lkv = acfg.lkv, - package_type = acfg.package_type, - ksu = acfg.ksu, + codename = args.codename, + base = args.base, + lkv = args.lkv, + package_type = args.package_type, + ksu = args.ksu, ).run() diff --git a/builder/clients/github.py b/builder/clients/github.py index bbee91b..529b18c 100644 --- a/builder/clients/github.py +++ b/builder/clients/github.py @@ -12,10 +12,8 @@ class GitHubApi(BaseModel): """Limited interaction with GitHub API. - :param _endpoint: API endpoint to interact with. - :param _direct_url: Direct URL for the specified repo. - :param project: GitHub project name (owner/repo). - :param file_filter: A filter to select specific files from project's artifacts. + :param str project: GitHub project name (owner/repo). + :param Optional[str]=None file_filter: A filter to select specific files from project's artifacts. """ _endpoint = "https://api.github.com/repos/{}/releases/latest" diff --git a/builder/clients/pa.py b/builder/clients/pa.py index 72d45dc..c389075 100644 --- a/builder/clients/pa.py +++ b/builder/clients/pa.py @@ -11,7 +11,6 @@ class ParanoidAndroidApi(RomApi): rom_name: str = "PA" @override - @property def codename_mapper(self) -> str: # specific rules for PA's API specials = { diff --git a/builder/clients/rom_api.py b/builder/clients/rom_api.py index 712f4d4..0a4b1e0 100644 --- a/builder/clients/rom_api.py +++ b/builder/clients/rom_api.py @@ -8,10 +8,10 @@ class RomApi(BaseModel, IRomApi): """A generic class for interacting with ROMs' APIs. - :param endpoint: API endpoint to interact with. - :param json_key: A JSON key to look for in the response. - :param rom_name: ROM project's name. - :param rom_only: Flag indicating ROM-only asset collection. + :param str endpoint: API endpoint to interact with. + :param str json_key: A JSON key to look for in the response. + :param str rom_name: ROM project's name. + :param bool rom_only: Flag indicating ROM-only asset collection. """ endpoint: str @@ -22,9 +22,8 @@ class RomApi(BaseModel, IRomApi): def __init__(self, **kwargs) -> None: super().__init__(**kwargs) - self.endpoint = self.endpoint.format(self.codename_mapper) + self.endpoint = self.endpoint.format(self.codename_mapper()) - @property def codename_mapper(self) -> str: # by default, codename is devicename return self.codename @@ -39,4 +38,4 @@ def run(self) -> str: f"Could not connect to {self.rom_name} API, HTTP status code: {data.status_code}", dont_exit=exit_flag ) - return data + return str(data) diff --git a/builder/commands/assets.py b/builder/commands/assets.py index 527b4dc..b857b28 100644 --- a/builder/commands/assets.py +++ b/builder/commands/assets.py @@ -3,6 +3,3 @@ class AssetsCommand(AssetsCollector): """A command responsible for launching the 'assets_collector' core module directly.""" - - def __init__(self, **kwargs) -> None: - super().__init__(**kwargs) diff --git a/builder/commands/bundle.py b/builder/commands/bundle.py index 3160989..f617eed 100644 --- a/builder/commands/bundle.py +++ b/builder/commands/bundle.py @@ -2,6 +2,7 @@ import json import shutil import itertools +from typing import Literal from pydantic import BaseModel from builder.core import KernelBuilder, AssetsCollector @@ -14,10 +15,10 @@ class BundleCommand(BaseModel, IBundleCommand): """A command that packages the artifacts produced both by 'kernel_builder' and 'assets_collector' core modules. - :param base: Kernel source base. - :param lkv: Linux kernel version. - :param package_type: Package type. - :param ksu: Flag indicating KernelSU support. + :param str base: Kernel source base. + :param str lkv: Linux kernel version. + :param str package_type: Package type. + :param bool ksu: Flag indicating KernelSU support. """ codename: str @@ -26,7 +27,7 @@ class BundleCommand(BaseModel, IBundleCommand): package_type: str ksu: bool - def _build_kernel(self, rom_name: str, clean_only: bool = False) -> None: + def build_kernel(self, rom_name: str, clean_only: bool = False) -> None: if not dcfg.kernel.is_dir() or clean_only is True: KernelBuilder( codename = self.codename, @@ -40,7 +41,7 @@ def _build_kernel(self, rom_name: str, clean_only: bool = False) -> None: def _rom_only_flag(self) -> bool: return True if "full" not in self.package_type else False - def _collect_assets(self, rom_name: str, chroot: str) -> None: + def collect_assets(self, rom_name: str, chroot: Literal["full", "minimal"]) -> None: AssetsCollector( codename = self.codename, base = rom_name, @@ -50,7 +51,7 @@ def _collect_assets(self, rom_name: str, chroot: str) -> None: ksu = self.ksu, ).run() - def _conan_sources(self) -> None: + def conan_sources(self) -> None: sourcedir = dcfg.root / "source" print("\n", end="") msg.note("Copying sources for Conan packaging..") @@ -71,12 +72,12 @@ def _conan_sources(self) -> None: msg.done("Done!") @staticmethod - def _conan_options(json_file: str) -> dict: + def conan_options(json_file: str) -> dict: with open(json_file, encoding="utf-8") as f: json_data = json.load(f) return json_data - def _conan_package(self, options: list[str], reference: str) -> None: + def conan_package(self, options: tuple[str, ...], reference: str) -> None: cmd = f"conan export-pkg . {reference}" for option_value in options: # not the best solution, but will work temporarily for 'rom' and 'chroot' options @@ -87,7 +88,7 @@ def _conan_package(self, options: list[str], reference: str) -> None: ccmd.launch(cmd) @staticmethod - def _conan_upload(reference: str) -> None: + def conan_upload(reference: str) -> None: # configure Conan client and upload packages url = "https://gitlab.com/api/v4/projects/40803264/packages/conan" alias = "zero-kernel-conan" @@ -101,9 +102,9 @@ def run(self) -> None: # determine the bundle type and process it match self.package_type: case "slim" | "full": - self._build_kernel(self.base) + self.build_kernel(self.base) # "full" chroot is hardcoded here - self._collect_assets(self.base, "full") + self.collect_assets(self.base, "full") # clean up if dcfg.bundle.is_dir(): contents = dcfg.bundle.glob("*") @@ -134,13 +135,13 @@ def run(self) -> None: option_sets = list(itertools.product([self.base], chroot)) # build and upload Conan packages for opset in option_sets: - self._build_kernel(opset[0]) - self._build_kernel(opset[0], True) - self._conan_sources() - self._collect_assets(opset[0], opset[1]) - self._conan_package(opset, reference) + self.build_kernel(opset[0]) + self.build_kernel(opset[0], True) + self.conan_sources() + self.collect_assets(opset[0], opset[1]) + self.conan_package(opset, reference) # upload packages if os.getenv("CONAN_UPLOAD_CUSTOM") == "1": - self._conan_upload(reference) + self.conan_upload(reference) # navigate back to root directory os.chdir(dcfg.root) diff --git a/builder/configs/__init__.py b/builder/configs/__init__.py index 04ba9a8..9611d5d 100644 --- a/builder/configs/__init__.py +++ b/builder/configs/__init__.py @@ -1,2 +1,2 @@ -from .arguments import ArgumentConfig -from .directories import DirectoryConfig +from .argument import ArgumentConfig +from .directory import DirectoryConfig diff --git a/builder/configs/arguments.py b/builder/configs/argument.py similarity index 63% rename from builder/configs/arguments.py rename to builder/configs/argument.py index d9ec23c..5684631 100644 --- a/builder/configs/arguments.py +++ b/builder/configs/argument.py @@ -1,8 +1,8 @@ import json import platform from pathlib import Path -from typing import Optional from pydantic import BaseModel +from typing import Optional, Literal from builder.tools import commands as ccmd, messages as msg @@ -10,23 +10,23 @@ class ArgumentConfig(BaseModel): """A variable storage to use across the application. - :param benv: Build environment. - :param command: Builder command to be launched. - :param codename: Device codename. - :param base: Kernel source base. - :param lkv: Linux kernel version. - :param chroot: Chroot type. - :param package_type: Package type. - :param clean_kernel: Flag to clean folder with kernel sources. - :param clean_assets: Flag to clean folder for assets storage. - :param clean_image: Flag to clean a Docker/Podman image from local cache. - :param rom_only: Flag indicating ROM-only asset collection. - :param conan_upload: Flag to enable Conan upload. - :param ksu: Flag indicating KernelSU support. + :param Literal["docker","podman"] benv: Build environment. + :param Literal["kernel","assets","bundle"] command: Builder command to be launched. + :param str codename: Device codename. + :param str base: Kernel source base. + :param str lkv: Linux kernel version. + :param Optional[str]=None chroot: Chroot type. + :param Optional[str]=None package_type: Package type. + :param Optional[bool]=False clean_kernel: Flag to clean folder with kernel sources. + :param Optional[bool]=False clean_assets: Flag to clean folder for assets storage. + :param Optional[bool]=False clean_image: Flag to clean a Docker/Podman image from local cache. + :param Optional[bool]=False rom_only: Flag indicating ROM-only asset collection. + :param Optional[bool]=False conan_upload: Flag to enable Conan upload. + :param Optional[bool]=False ksu: Flag indicating KernelSU support. """ - benv: str - command: str + benv: Literal["docker", "podman"] + command: Literal["kernel", "assets", "bundle"] codename: str base: str lkv: Optional[str] = None diff --git a/builder/configs/directories.py b/builder/configs/directory.py similarity index 100% rename from builder/configs/directories.py rename to builder/configs/directory.py diff --git a/builder/core/assets_collector.py b/builder/core/assets_collector.py index 81b4528..353561b 100644 --- a/builder/core/assets_collector.py +++ b/builder/core/assets_collector.py @@ -1,4 +1,5 @@ import os +from typing import Literal from pydantic import BaseModel from builder.tools import cleaning as cm, fileoperations as fo, messages as msg @@ -10,34 +11,32 @@ class AssetsCollector(BaseModel, IAssetsCollector): """Assets collector. - :param codename: Device codename. - :param base: Kernel source base. - :param chroot: Chroot type. - :param rom_only: Flag indicating ROM-only asset collection. - :param ksu: Flag indicating KernelSU support. + :param str codename: Device codename. + :param str base: Kernel source base. + :param Literal["full","minimal"] chroot: Chroot type. + :param bool rom_only: Flag indicating ROM-only asset collection. + :param bool ksu: Flag indicating KernelSU support. """ codename: str base: str - chroot: str + chroot: Literal["full", "minimal"] clean_assets: bool rom_only: bool ksu: bool @property def rom_collector_dto(self) -> LineageOsApi | ParanoidAndroidApi | None: - rom_collector_dto = "" match self.base: case "los": - rom_collector_dto = LineageOsApi(codename=self.codename, rom_only=self.rom_only) + return LineageOsApi(codename=self.codename, rom_only=self.rom_only) case "pa": - rom_collector_dto = ParanoidAndroidApi(codename=self.codename, rom_only=self.rom_only) + return ParanoidAndroidApi(codename=self.codename, rom_only=self.rom_only) case "x" | "aosp": msg.note("Selected kernel base is ROM-universal, no specific ROM image will be collected") - return rom_collector_dto @property - def assets(self) -> tuple[str | None] | list[str | None]: + def assets(self) -> tuple[str, str | None] | list[str] | None: # define dm-verity and forceencrypt disabler (DFD) and SU manager dfd = GitHubApi(project="seppzer0/Disable_Dm-Verity_ForceEncrypt").run() su_manager = "tiann/KernelSU" if self.ksu else "topjohnwu/Magisk" @@ -47,9 +46,9 @@ def assets(self) -> tuple[str | None] | list[str | None]: msg.cancel("Cancelling ROM-only asset collection") else: # add DFD alongside the ROM - assets = (self.rom_collector_dto.run(), dfd) print("\n", end="") msg.done("ROM-only asset collection complete!") + return (self.rom_collector_dto.run(), dfd) # process the non-"RON-only" download else: assets = [ @@ -93,7 +92,8 @@ def assets(self) -> tuple[str | None] | list[str | None]: # finally, add ROM (if kernel base is not universal) and DFD into assets list if self.rom_collector_dto: assets.append(self.rom_collector_dto.run()) - return assets + return assets + return None def _check(self) -> None: os.chdir(dcfg.root) @@ -122,9 +122,10 @@ def run(self) -> None: msg.banner("zero asset collector") self._check() os.chdir(dcfg.assets) - for e in self.assets: - if e is not None: - fo.download(e) + if isinstance(self.assets, list): + for e in self.assets: + if e is not None: + fo.download(e) print("\n", end="") msg.done("Assets collected!") os.chdir(dcfg.root) diff --git a/builder/core/kernel_builder.py b/builder/core/kernel_builder.py index 281677a..08fdeb5 100644 --- a/builder/core/kernel_builder.py +++ b/builder/core/kernel_builder.py @@ -13,11 +13,11 @@ class KernelBuilder(BaseModel, IKernelBuilder): """Kernel builder. - :param codename: Device codename. - :param base: Kernel source base. - :param lkv: Linux kernel version. - :param clean_kernel: Flag to clean folder with kernel sources. - :param ksu: Flag indicating KernelSU support. + :param str codename: Device codename. + :param str base: Kernel source base. + :param str lkv: Linux kernel version. + :param bool clean_kernel: Flag to clean folder with kernel sources. + :param bool ksu: Flag indicating KernelSU support. """ codename: str @@ -52,9 +52,9 @@ def _defconfig(self) -> Path: def _clean_build(self) -> None: print("\n", end="") msg.note("Cleaning the build environment..") - cm.git(self._rcs.paths[self.codename]["path"]) - cm.git(self._rcs.paths["AnyKernel3"]["path"]) - cm.git(self._rcs.paths["KernelSU"]["path"]) + cm.git(self._rcs.paths[self.codename]) + cm.git(self._rcs.paths["AnyKernel3"]) + cm.git(self._rcs.paths["KernelSU"]) for fn in os.listdir(): if fn == "localversion" or fn.endswith(".zip"): cm.remove(fn) @@ -63,49 +63,49 @@ def _clean_build(self) -> None: def _patch_strict_prototypes(self) -> None: msg.note("Patching sources for Clang 15+ compatibility..") data = { - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "char" /\ "diag" /\ "diagchar_core.c": ("void diag_ws_init()", "void diag_ws_on_notify()", "void diag_ws_release()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "char" /\ "diag" /\ "diag_mux.c": ("int diag_mux_init()", "void diag_mux_exit()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "char" /\ "diag" /\ "diag_memorydevice.c": ("void diag_md_open_all()", "void diag_md_close_all()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "char" /\ "diag" /\ "diag_dci.c": ("void diag_dci_wakeup_clients()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "char" /\ "diag" /\ "diagfwd_bridge.c": ("void diagfwd_bridge_exit()", "uint16_t diag_get_remote_device_mask()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "char" /\ "diag" /\ "diagfwd_mhi.c": ("int diag_mhi_init()", "void diag_mhi_exit()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "media" /\ "platform" /\ @@ -115,7 +115,7 @@ def _patch_strict_prototypes(self) -> None: "msm_camera_tz_util.c": ("struct qseecom_handle *msm_camera_tz_get_ta_handle()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "media" /\ "platform" /\ @@ -124,7 +124,7 @@ def _patch_strict_prototypes(self) -> None: "msm_vidc_common.c": ("void msm_comm_handle_thermal_event()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "soc" /\ "qcom" /\ @@ -132,7 +132,7 @@ def _patch_strict_prototypes(self) -> None: "msm_bus_rpm_smd.c": ("static int voice_svc_dummy_reg()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "soc" /\ "qcom" /\ @@ -140,7 +140,7 @@ def _patch_strict_prototypes(self) -> None: "msm_bus_rpm_smd.c": ("void msm_bus_rpm_set_mt_mask()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "staging" /\ "qca-wifi-host-cmn" /\ @@ -150,7 +150,7 @@ def _patch_strict_prototypes(self) -> None: "ce_service.c": ("struct ce_ops *ce_services_legacy()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "staging" /\ "qcacld-3.0" /\ @@ -160,7 +160,7 @@ def _patch_strict_prototypes(self) -> None: "wlan_hdd_main.c": ("hdd_adapter_t *hdd_get_first_valid_adapter()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "video" /\ "fbdev" /\ @@ -168,7 +168,7 @@ def _patch_strict_prototypes(self) -> None: "mdss_mdp.c": ("struct irq_info *mdss_intr_line()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "video" /\ "fbdev" /\ @@ -179,7 +179,7 @@ def _patch_strict_prototypes(self) -> None: # the following files are not present in 4.14 if self._linux_kernel_version != "4.14": extra_non_414 = { - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "soc" /\ "qcom" /\ @@ -187,7 +187,7 @@ def _patch_strict_prototypes(self) -> None: "voice_svc.c": ("void msm_bus_rpm_set_mt_mask()", "static int voice_svc_dummy_reg()"), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "thermal" /\ "msm_thermal-dev.c": @@ -197,7 +197,7 @@ def _patch_strict_prototypes(self) -> None: # PA needs this, LineageOS does not if self.base == "pa": extra_pa = { - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "staging" /\ "qca-wifi-host-cmn" /\ @@ -207,7 +207,7 @@ def _patch_strict_prototypes(self) -> None: "target_if_main.c": ("struct target_if_ctx *target_if_get_ctx()",), - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "staging" /\ "qca-wifi-host-cmn" /\ @@ -229,15 +229,15 @@ def _patch_strict_prototypes(self) -> None: msg.done("Done!") def _patch_anykernel3(self) -> None: - cm.remove(self._rcs.paths["AnyKernel3"]["path"] / "ramdisk") - cm.remove(self._rcs.paths["AnyKernel3"]["path"] / "models") + cm.remove(self._rcs.paths["AnyKernel3"] / "ramdisk") + cm.remove(self._rcs.paths["AnyKernel3"] / "models") fo.ucopy( dcfg.root / "builder" / "modifications" / self._ucodename / "anykernel3" / "ramdisk", - self._rcs.paths["AnyKernel3"]["path"] / "ramdisk" + self._rcs.paths["AnyKernel3"] / "ramdisk" ) fo.ucopy( dcfg.root / "builder" / "modifications" / self._ucodename / "anykernel3" / "anykernel.sh", - self._rcs.paths["AnyKernel3"]["path"] / "anykernel.sh" + self._rcs.paths["AnyKernel3"] / "anykernel.sh" ) def _patch_rtl8812au_source_mod_v5642(self) -> None: @@ -286,8 +286,8 @@ def _patch_rtl8812au(self) -> None: # copy RTL8812AU sources into kernel sources msg.note("Adding RTL8812AU drivers into the kernel..") fo.ucopy( - self._rcs.paths["rtl8812au"]["path"], - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths["rtl8812au"], + self._rcs.paths[self.codename] /\ "drivers" /\ "net" /\ "wireless" /\ @@ -296,7 +296,7 @@ def _patch_rtl8812au(self) -> None: ) # modify sources depending on driver version os.chdir( - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "drivers" /\ "net" /\ "wireless" /\ @@ -307,18 +307,18 @@ def _patch_rtl8812au(self) -> None: cm.remove(".git*") os.chdir(dcfg.root) # include the driver into build process - makefile = self._rcs.paths[self.codename]["path"] /\ + makefile = self._rcs.paths[self.codename] /\ "drivers" /\ "net" /\ "wireless" /\ "realtek" /\ "Makefile" - kconfig = self._rcs.paths[self.codename]["path"] /\ + kconfig = self._rcs.paths[self.codename] /\ "drivers" /\ "net" /\ "wireless" /\ "Kconfig" - defconfig = self._rcs.paths[self.codename]["path"] /\ + defconfig = self._rcs.paths[self.codename] /\ "arch" /\ "arm64" /\ "configs" /\ @@ -349,22 +349,22 @@ def _patch_ksu(self) -> None: msg.note("Adding KernelSU into the kernel..") # extract KSU_GIT_VERSION environment variable manually goback = Path.cwd() - os.chdir(self._rcs.paths["KernelSU"]["path"]) + os.chdir(self._rcs.paths["KernelSU"]) os.environ["KSU_GIT_VERSION"] = str( # official formula documented in KernelSU's Makefile - 10000 + int(ccmd.launch("git rev-list --count HEAD", get_output=True)) + 200 + 10000 + int(ccmd.launch("git rev-list --count HEAD", get_output=True)) + 200 # type: ignore ) os.chdir(goback) - makefile = self._rcs.paths[self.codename]["path"] /\ + makefile = self._rcs.paths[self.codename] /\ "drivers" /\ "Makefile" - kconfig = self._rcs.paths[self.codename]["path"] /\ + kconfig = self._rcs.paths[self.codename] /\ "drivers" /\ "Kconfig" # include into the build process via symlink os.symlink( - self._rcs.paths["KernelSU"]["path"] / "kernel", - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths["KernelSU"] / "kernel", + self._rcs.paths[self.codename] /\ "drivers" /\ "kernelsu" ) @@ -376,7 +376,7 @@ def _patch_ksu(self) -> None: "source \"drivers/kernelsu/Kconfig\"" ) # either patch kernel or KernelSU sources, depending on Linux kernel version - target_dir = dcfg.root / "KernelSU" if self._linux_kernel_version == "4.14" else self._rcs.paths[self.codename]["path"] + target_dir = dcfg.root / "KernelSU" if self._linux_kernel_version == "4.14" else self._rcs.paths[self.codename] fo.ucopy( dcfg.root / "builder" / "modifications" / self._ucodename / self._linux_kernel_version / "kernelsu-compat.patch", target_dir @@ -385,7 +385,7 @@ def _patch_ksu(self) -> None: fo.apply_patch("kernelsu-compat.patch") os.chdir(goback) # add configs into defconfig - defconfig = self._rcs.paths[self.codename]["path"] /\ + defconfig = self._rcs.paths[self.codename] /\ "arch" /\ "arm64" /\ "configs" /\ @@ -408,14 +408,14 @@ def _patch_qcacld(self) -> None: goback = Path.cwd() fo.ucopy( dcfg.root / "builder" / "modifications" / self._ucodename / self._linux_kernel_version / "qcacld_pa.patch", - self._rcs.paths[self.codename]["path"] + self._rcs.paths[self.codename] ) - os.chdir(self._rcs.paths[self.codename]["path"]) + os.chdir(self._rcs.paths[self.codename]) fo.apply_patch("qcacld_pa.patch") os.chdir(goback) def _patch_ioctl(self) -> None: - ioctl = self._rcs.paths[self.codename]["path"] /\ + ioctl = self._rcs.paths[self.codename] /\ "drivers" /\ "platform" /\ "msm" /\ @@ -430,17 +430,17 @@ def _patch_ioctl(self) -> None: def _patch_kernel(self) -> None: # -Wstrict-prototypes patch to build with Clang 15+ - clang_cmd = f'{self._rcs.paths["clang"]["path"] / "bin" / "clang"} --version' - clang_ver = ccmd.launch(clang_cmd, get_output=True).split("clang version ")[1].split(".")[0] + clang_cmd = f'{self._rcs.paths["clang"] / "bin" / "clang"} --version' + clang_ver = str(ccmd.launch(clang_cmd, get_output=True)).split("clang version ")[1].split(".")[0] if int(clang_ver) >= 15: self._patch_strict_prototypes() # apply .patch files fo.ucopy( dcfg.root / "builder" / "modifications" / self._ucodename / self._linux_kernel_version, - self._rcs.paths[self.codename]["path"], + self._rcs.paths[self.codename], ("kernelsu-compat.patch", "qcacld_pa.patch") ) - os.chdir(self._rcs.paths[self.codename]["path"]) + os.chdir(self._rcs.paths[self.codename]) for pf in Path.cwd().glob("*.patch"): fo.apply_patch(pf) # add support for CONFIG_MAC80211 kernel option @@ -470,7 +470,7 @@ def _patch_all(self) -> None: def _build(self) -> None: print("\n", end="") msg.note("Launching the build..") - os.chdir(self._rcs.paths[self.codename]["path"]) + os.chdir(self._rcs.paths[self.codename]) # launch "make" punits = ccmd.launch("nproc --all", get_output=True) cmd1 = "make -j{} O=out {} "\ @@ -511,7 +511,7 @@ def _build(self) -> None: def _linux_kernel_version(self) -> str: data = "" version = [] - with open(self._rcs.paths[self.codename]["path"] / "Makefile", encoding="utf-8") as f: + with open(self._rcs.paths[self.codename] / "Makefile", encoding="utf-8") as f: data = f.read() params = ("VERSION", "PATCHLEVEL") # find the required lines in a single data run-through @@ -528,13 +528,13 @@ def _create_zip(self) -> None: print("\n", end="") msg.note("Forming final ZIP file..") fo.ucopy( - self._rcs.paths[self.codename]["path"] /\ + self._rcs.paths[self.codename] /\ "out" /\ "arch" /\ "arm64" /\ "boot" /\ "Image.gz-dtb", - self._rcs.paths["AnyKernel3"]["path"] / "Image.gz-dtb" + self._rcs.paths["AnyKernel3"] / "Image.gz-dtb" ) # define kernel versions: Linux and internal verbase = self._linux_kernel_version @@ -545,7 +545,7 @@ def _create_zip(self) -> None: kdir = dcfg.root / dcfg.kernel if not kdir.is_dir(): os.mkdir(kdir) - os.chdir(self._rcs.paths["AnyKernel3"]["path"]) + os.chdir(self._rcs.paths["AnyKernel3"]) # this is not the best solution, but is the easiest cmd = f"zip -r9 {kdir / name_full}.zip . -x *.git* *README* *LICENSE* *placeholder" ccmd.launch(cmd) @@ -556,7 +556,8 @@ def run(self) -> None: os.chdir(dcfg.root) msg.banner("zero kernel builder") msg.note("Setting up tools and links..") - self._rcs.path_gen() + self._rcs.read_data() + self._rcs.generate_paths() self._rcs.download() self._rcs.export_path() self._clean_build() diff --git a/builder/engines/__init__.py b/builder/engines/__init__.py index 6caf1fe..d0b69f5 100644 --- a/builder/engines/__init__.py +++ b/builder/engines/__init__.py @@ -1,2 +1 @@ -from .docker_engine import DockerEngine -from .podman_engine import PodmanEngine +from .generic_container import GenericContainerEngine diff --git a/builder/engines/docker_engine.py b/builder/engines/docker_engine.py deleted file mode 100644 index 0d8b6df..0000000 --- a/builder/engines/docker_engine.py +++ /dev/null @@ -1,44 +0,0 @@ -import os -from typing import override - -from builder.tools import commands as ccmd, messages as msg -from builder.interfaces import IDockerEngine - -from builder.engines.container_engine import ContainerEngine - - -class DockerEngine(ContainerEngine, IDockerEngine): - """Docker engine.""" - - @staticmethod - def _force_buildkit() -> None: - os.environ["DOCKER_BUILDKIT"] = "1" - - def _check_cache(self) -> bool: - img_cache_cmd = f'{self.benv} images --format {"{{.Repository}}"}' - img_cache = ccmd.launch(img_cache_cmd, get_output=True) - check = True if self.name_image in img_cache else False - return check - - @override - def run(self) -> None: - self._force_buildkit() - # build image only if it is not present in local cache - if not self._check_cache(): - self.build() - else: - msg.note("\nDocker image already in local cache, skipping it's build..\n") - # form the final command to create container - cmd = '{} run {} {} /bin/bash -c "{}"'.format( - self.benv, - " ".join(self.container_options), - self.name_image, - self.builder_cmd - ) - # prepare directories - self.create_dirs() - ccmd.launch(cmd) - # navigate to root directory and clean image from host machine - os.chdir(self.wdir_local) - if self.clean_image: - ccmd.launch(f"{self.benv} rmi {self.name_image}") diff --git a/builder/engines/container_engine.py b/builder/engines/generic_container.py similarity index 59% rename from builder/engines/container_engine.py rename to builder/engines/generic_container.py index 5fbc776..19f48ab 100644 --- a/builder/engines/container_engine.py +++ b/builder/engines/generic_container.py @@ -1,16 +1,16 @@ import os import shutil from pathlib import Path -from typing import Optional from pydantic import BaseModel +from typing import Optional, Literal from subprocess import CompletedProcess from builder.tools import commands as ccmd, messages as msg from builder.configs import DirectoryConfig as dcfg -from builder.interfaces import IContainerEngine +from builder.interfaces import IGenericContainerEngine -class ContainerEngine(BaseModel, IContainerEngine): +class GenericContainerEngine(BaseModel, IGenericContainerEngine): """A generic container engine for containerized builds. Note that here paths from DirectoryConfig are not used @@ -19,36 +19,32 @@ class ContainerEngine(BaseModel, IContainerEngine): directory structure. We only need to specify directory names and not full paths. - :param name_image: Name of the Docker/Podman image. - :param name_container: Name of the Docker/Podman container. - :param wdir_container: Working directory in the container. - :param wdir_local: Working directory from the local environment (aka root of the repo). - :param benv: Build environment. - :param command: Builder command to be launched. - :param codename: Device codename. - :param base: Kernel source base. - :param lkv: Linux kernel version. - :param chroot: Chroot type. - :param package_type: Package type. - :param clean_kernel: Flag to clean folder for kernel storage. - :param clean_assets: Flag to clean folder for assets storage. - :param clean_image: Flag to clean a Docker/Podman image from local cache. - :param rom_only: Flag indicating ROM-only asset collection. - :param conan_upload: Flag to enable Conan upload. - :param ksu: Flag to add KernelSU support into the kernel. + :param Literal["docker","podman"] benv: Build environment. + :param Literal["kernel","assets","bundle"] command: Builder command to be launched. + :param str codename: Device codename. + :param str base: Kernel source base. + :param str lkv: Linux kernel version. + :param Optional[Literal["full","minimal"]]=None chroot: Chroot type. + :param Optional[bool]=False package_type: Package type. + :param Optional[bool]=False clean_kernel: Flag to clean folder for kernel storage. + :param Optional[bool]=False clean_assets: Flag to clean folder for assets storage. + :param Optional[bool]=False clean_image: Flag to clean a Docker/Podman image from local cache. + :param Optional[bool]=False rom_only: Flag indicating ROM-only asset collection. + :param Optional[bool]=False conan_upload: Flag to enable Conan upload. + :param Optional[bool]=False ksu: Flag to add KernelSU support into the kernel. """ - name_image: str = "zero-kernel-image" - name_container: str = "zero-kernel-container" - wdir_container: Path = Path("/", "zero_build") - wdir_local: Path = dcfg.root + _name_image: str = "zero-kernel-image" + _name_container: str = "zero-kernel-container" + _wdir_container: Path = Path("/", "zero_build") + _wdir_local: Path = dcfg.root - benv: str - command: str + benv: Literal["docker", "podman"] + command: Literal["kernel", "assets", "bundle"] codename: str base: str lkv: Optional[str] = None - chroot: Optional[str] = None + chroot: Optional[Literal["full", "minimal"]] = None package_type: Optional[str] = None clean_kernel: Optional[bool] = False clean_assets: Optional[bool] = False @@ -57,14 +53,22 @@ class ContainerEngine(BaseModel, IContainerEngine): conan_upload: Optional[bool] = False ksu: Optional[bool] = False + @staticmethod + def _force_buildkit() -> None: + """Force enable Docker BuildKit.""" + os.environ["DOCKER_BUILDKIT"] = "1" + @property def dir_bundle_conan(self) -> Path: - res = "" if os.getenv("CONAN_USER_HOME"): - res = Path(os.getenv("CONAN_USER_HOME")) + return Path(os.getenv("CONAN_USER_HOME")) # type: ignore else: - res = Path(os.getenv("HOME"), ".conan") - return res + return Path(os.getenv("HOME"), ".conan") # type: ignore + + def check_cache(self) -> bool: + img_cache_cmd = f'{self.benv} images --format {"{{.Repository}}"}' + img_cache = str(ccmd.launch(img_cache_cmd, get_output=True)) + return True if self._name_image in img_cache else False @property def builder_cmd(self) -> str: @@ -93,36 +97,36 @@ def builder_cmd(self) -> str: # extend the command with the selected packaging option if self.command == "bundle": if self.package_type in ("slim", "full"): - cmd += f" && chmod 777 -R {self.wdir_container / dcfg.bundle.name}" + cmd += f" && chmod 777 -R {self._wdir_container / dcfg.bundle.name}" else: cmd += " && chmod 777 -R /root/.conan" return cmd @property def container_options(self) -> list[str]: - # declare the base + # declare the base of options options = [ "-i", "--rm", "-e KVERSION={}".format(os.getenv("KVERSION")), "-e LOGLEVEL={}".format(os.getenv("LOGLEVEL")), - "-w {}".format(self.wdir_container), + "-w {}".format(self._wdir_container), ] # define volume mounting template v_template = "-v {}:{}/{}" # mount directories match self.command: case "kernel": - options.append(v_template.format(dcfg.kernel, self.wdir_container, dcfg.kernel.name)) + options.append(v_template.format(dcfg.kernel, self._wdir_container, dcfg.kernel.name)) case "assets": - options.append(v_template.format(dcfg.assets, self.wdir_container, dcfg.assets.name)) + options.append(v_template.format(dcfg.assets, self._wdir_container, dcfg.assets.name)) case "bundle": match self.package_type: case "slim" | "full": - options.append(v_template.format(dcfg.bundle, self.wdir_container, dcfg.bundle.name)) + options.append(v_template.format(dcfg.bundle, self._wdir_container, dcfg.bundle.name)) case "conan": if self.conan_upload: - options.append('-e CONAN_UPLOAD_CUSTOM=1') + options.append("-e CONAN_UPLOAD_CUSTOM=1") # determine the path to local Conan cache and check if it exists if self.dir_bundle_conan.is_dir(): options.append(f'-v {self.dir_bundle_conan}:/"/root/.conan"') @@ -144,35 +148,45 @@ def create_dirs(self) -> None: shutil.rmtree(dcfg.bundle, ignore_errors=True) os.mkdir(dcfg.bundle) - def build(self) -> CompletedProcess: + def build_image(self) -> str | None | CompletedProcess: print("\n") alias = self.benv.capitalize() msg.note(f"Building the {alias} image..") - os.chdir(self.wdir_local) + os.chdir(self._wdir_local) # NOTE: this will crash in GitLab CI/CD (Docker-in-Docker), requires a workaround cmd = "{} build . -f {} -t {} --load".format( self.benv, - self.wdir_local / 'Dockerfile', - self.name_image + self._wdir_local / 'Dockerfile', + self._name_image ) res = ccmd.launch(cmd) msg.done("Done!") print("\n") return res - def run(self) -> None: - self.build() - # form the final command to create container - cmd = '{} run {} {} /bin/bash -c "{}"'.format( + @property + def run_cmd(self) -> str: + return '{} run {} {} /bin/bash -c "{}"'.format( self.benv, " ".join(self.container_options), - self.name_image, + self._name_image, self.builder_cmd ) - # prepare directories + + def __enter__(self) -> str: + # prepare Docker if selected + if self.benv == "docker": + self._force_buildkit() + # build the image and prepare directories + if not self.check_cache(): + self.build_image() + else: + msg.note(f"\n{self.benv.capitalize()} image already in local cache, skipping it's build..\n") self.create_dirs() - ccmd.launch(cmd) + return self.run_cmd + + def __exit__(self, exc_type, exc_value, traceback) -> None: # navigate to root directory and clean image from host machine - os.chdir(self.wdir_local) + os.chdir(self._wdir_local) if self.clean_image: - ccmd.launch(f"{self.benv} rmi {self.name_image}") + ccmd.launch(f"{self.benv} rmi {self._name_image}") diff --git a/builder/engines/podman_engine.py b/builder/engines/podman_engine.py deleted file mode 100644 index e49dfd7..0000000 --- a/builder/engines/podman_engine.py +++ /dev/null @@ -1,5 +0,0 @@ -from builder.engines.container_engine import ContainerEngine - - -class PodmanEngine(ContainerEngine): - """Podman engine.""" diff --git a/builder/interfaces/__init__.py b/builder/interfaces/__init__.py index adf955f..a9db3b0 100644 --- a/builder/interfaces/__init__.py +++ b/builder/interfaces/__init__.py @@ -1,4 +1,4 @@ from .clients import IRomApi from .modules import IKernelBuilder, IAssetsCollector -from .engines import IContainerEngine, IDockerEngine +from .engines import IGenericContainerEngine from .commands import IBundleCommand diff --git a/builder/interfaces/clients.py b/builder/interfaces/clients.py index 2bb20e3..4820b37 100644 --- a/builder/interfaces/clients.py +++ b/builder/interfaces/clients.py @@ -4,13 +4,12 @@ class IRomApi(ABC): """An interface for interacting with ROM projects' APIs.""" - @property @abstractmethod def codename_mapper(self) -> str: """Codename-to-devicename mapper.""" raise NotImplementedError() @abstractmethod - def run(self) -> None: + def run(self) -> str: """Execute the API interaction logic.""" raise NotImplementedError() diff --git a/builder/interfaces/commands.py b/builder/interfaces/commands.py index 2327103..fee7127 100644 --- a/builder/interfaces/commands.py +++ b/builder/interfaces/commands.py @@ -1,3 +1,4 @@ +from typing import Literal from abc import ABC, abstractmethod @@ -5,58 +6,52 @@ class IBundleCommand(ABC): """An interface for the bundle creator.""" @abstractmethod - def _build_kernel(self, rom_name: str, clean_only: bool = False) -> None: + def build_kernel(self, rom_name: str, clean_only: bool = False) -> None: """Build the kernel. - :param rom_name: Name of the ROM. - :param clean_only: Append an argument to just clean the kernel directory. + :param str rom_name: Name of the ROM. + :param bool clean_only: Append an argument to just clean the kernel directory. """ raise NotImplementedError() - @property @abstractmethod - def _rom_only_flag(self) -> bool: - """Determine the value of the --rom-only flag.""" - raise NotImplementedError() - - @abstractmethod - def _collect_assets(self, rom_name: str, chroot: str) -> None: + def collect_assets(self, rom_name: str, chroot: Literal["full", "minimal"]) -> None: """Collect assets. - :param rom_name: Name of the ROM. - :param chroot: Type of chroot. + :param str rom_name: Name of the ROM. + :param bool chroot: Type of chroot. """ raise NotImplementedError() @abstractmethod - def _conan_sources(self) -> None: + def conan_sources(self) -> None: """Prepare sources for rebuildable Conan packages.""" raise NotImplementedError() @staticmethod @abstractmethod - def _conan_options(json_file: str) -> dict: + def conan_options(json_file: str) -> dict: """Read Conan options from a JSON file. - :param json_file: Name of the JSON file to read data from. + :param str json_file: Name of the JSON file to read data from. """ raise NotImplementedError() @abstractmethod - def _conan_package(self, options: list[str], reference: str) -> None: + def conan_package(self, options: tuple[str, ...], reference: str) -> None: """Create the Conan package. - :param options: Conan options. - :param reference: Conan reference. + :param tuple[str,...] options: Conan options. + :param str reference: Conan reference. """ raise NotImplementedError() @staticmethod @abstractmethod - def _conan_upload(reference: str) -> None: + def conan_upload(reference: str) -> None: """Upload Conan component to the remote. - :param reference: Conan reference. + :param str reference: Conan reference. """ raise NotImplementedError() diff --git a/builder/interfaces/engines.py b/builder/interfaces/engines.py index 874fd78..bd0ff12 100644 --- a/builder/interfaces/engines.py +++ b/builder/interfaces/engines.py @@ -1,9 +1,10 @@ from pathlib import Path from abc import ABC, abstractmethod +from subprocess import CompletedProcess -class IContainerEngine(ABC): - """An interface for ContainerEngine.""" +class IGenericContainerEngine(ABC): + """An interface for GenericContainerEngine.""" @property @abstractmethod @@ -11,6 +12,11 @@ def dir_bundle_conan(self) -> Path: """Determine path to Conan's local cache.""" raise NotImplementedError() + @abstractmethod + def check_cache(self) -> bool: + """Check local cache for target image presence.""" + raise NotImplementedError() + @property def builder_cmd(self) -> str: """Return the launch command for the builder.""" @@ -28,26 +34,22 @@ def create_dirs(self) -> None: raise NotImplementedError() @abstractmethod - def build(self) -> None: + def build_image(self) -> str | None | CompletedProcess: """Build the image.""" raise NotImplementedError() + @property @abstractmethod - def run(self) -> None: - """Execute the containerized build logic.""" + def run_cmd(self) -> str: + """Form command for container launch.""" raise NotImplementedError() - -class IDockerEngine(ABC): - """An interface for Docker-specific methods.""" - - @staticmethod @abstractmethod - def _force_buildkit() -> None: - """Force enable Docker BuildKit.""" + def __enter__(self) -> str: + """Magic method for preparing the containerized build.""" raise NotImplementedError() @abstractmethod - def _check_cache(self) -> bool: - """Check local Docker cache for the specified image.""" + def __exit__(self, exc_type, exc_value, traceback) -> None: + """Magic method for post-build operations for the container engine.""" raise NotImplementedError() diff --git a/builder/interfaces/modules.py b/builder/interfaces/modules.py index 223d93c..22161d8 100644 --- a/builder/interfaces/modules.py +++ b/builder/interfaces/modules.py @@ -135,7 +135,7 @@ def rom_collector_dto(self) -> LineageOsApi | ParanoidAndroidApi | None: @property @abstractmethod - def assets(self) -> list[str | LineageOsApi | ParanoidAndroidApi]: + def assets(self) -> tuple[str, str | None] | list[str] | None: """Form the full list of assets for collections.""" raise NotImplementedError() diff --git a/builder/tools/cleaning.py b/builder/tools/cleaning.py index 08ce767..d6d376d 100644 --- a/builder/tools/cleaning.py +++ b/builder/tools/cleaning.py @@ -9,13 +9,13 @@ from builder.configs import DirectoryConfig as dcfg -def remove(elements: str | Path | list[Path]) -> None: +def remove(elements: str | Path | list[Path | str]) -> None: """An ultimate Pythonic alternative to 'rm -rf'. Here, all Path() objects will have to be converted into str. Because of such specific as directories starting with a "." (e.g., .github). - :param elements: Files and/or directories to remove. + :param str/Path/list[Path] elements: Files and/or directories to remove. """ # if a given argument is a string --> convert it into a one-element list if isinstance(elements, str) or isinstance(elements, Path): @@ -35,21 +35,21 @@ def remove(elements: str | Path | list[Path]) -> None: remove(fn) -def on_rm_error(func, path: Path, exc_info): +def on_rm_error(func, path: str, exc_info): """For Windows system to remove a .git folder. :param func: Function to be used along with. - :param path: Path that is being removed. + :param Path path: Path that is being removed. :exc_info param: Misc info. """ os.chmod(path, stat.S_IWRITE) os.unlink(path) -def git(directory: Path) -> None: +def git(directory: Path | str) -> None: """Clean up a git directory. - :param directory: Path to the directory. + :param Path/str directory: Path to the directory. """ goback = Path.cwd() os.chdir(directory) @@ -61,7 +61,7 @@ def git(directory: Path) -> None: def root(extra: Optional[list[str]] = []) -> None: """Fully clean the root directory. - :param extra: Extra elements to be removed. + :param Optional[list[str]]=[] extra: Extra elements to be removed. """ trsh = [ dcfg.kernel, diff --git a/builder/tools/commands.py b/builder/tools/commands.py index f99ab59..1aa2687 100644 --- a/builder/tools/commands.py +++ b/builder/tools/commands.py @@ -1,6 +1,6 @@ import os import subprocess -from typing import Optional +from typing import Optional, Literal from subprocess import CompletedProcess from builder.tools import messages as msg @@ -9,13 +9,13 @@ def launch( cmd: str, get_output: Optional[bool] = False, - loglvl: str = os.getenv("LOGLEVEL", "normal") + loglvl: Optional[Literal["normal", "quiet"]] = "normal" ) -> str | CompletedProcess | None: """A custom subprocess wrapper to launch commands. - :param cmd: A command to launch. - :param get_output: A switch to get the piped output of the command. - :param loglvl: Log level. + :param str cmd: A command to launch. + :param Optional[bool]=False get_output: A switch to get the piped output of the command. + :param str loglvl: Log level. """ # determine stdout and check some of the cases cstdout = subprocess.DEVNULL if loglvl == "quiet" else os.getenv("OSTREAM", None) @@ -26,16 +26,12 @@ def launch( cstdout = open(cstdout, "a") if loglvl == "quiet" and os.getenv("OSTREAM"): msg.error("Cannot run 'quiet' build with file logging") - elif loglvl == "verbose": - print(f"[cmd] {cmd}") try: result = subprocess.run(cmd, shell=True, check=True, stdout=cstdout, stderr=subprocess.STDOUT) # return only output if required if get_output is True: - result = result.stdout.decode("utf-8").rstrip() - except Exception: - msg.error(f"Error executing command: {cmd}") - # if output stream is a file -- close it - if isinstance(cstdout, str): - cstdout.close() - return result + return result.stdout.decode("utf-8").rstrip() + else: + return result + except Exception as e: + msg.error(f"Error executing command: {cmd} -> {e}") diff --git a/builder/tools/fileoperations.py b/builder/tools/fileoperations.py index 7ed97ac..a9d9f80 100644 --- a/builder/tools/fileoperations.py +++ b/builder/tools/fileoperations.py @@ -1,18 +1,18 @@ import os import shutil import requests -from typing import Tuple from pathlib import Path +from typing import Optional from builder.tools import commands as ccmd, messages as msg -def ucopy(src: Path, dst: Path, exceptions: Tuple[str] = ()) -> None: +def ucopy(src: Path, dst: Path, exceptions: Optional[tuple[str | Path, ...]] = ()) -> None: """A universal method to copy files into desired destinations. - :param src: Source path. - :param dst: Destination path. - :param exceptions: Elements that will not be removed. + :param Path src: Source path. + :param Path dst: Destination path. + :param Optional[tuple[str/Path,...]]=() exceptions: Elements that will not be removed. """ # for a directory (it's contents) if src.is_dir(): @@ -21,7 +21,7 @@ def ucopy(src: Path, dst: Path, exceptions: Tuple[str] = ()) -> None: contents = os.listdir(src) for e in contents: # do not copy restricted files - if e not in exceptions and e != src: + if e not in exceptions and e != src: # type: ignore src_e = src / e dst_e = dst / e if src_e.is_dir(): @@ -36,7 +36,7 @@ def ucopy(src: Path, dst: Path, exceptions: Tuple[str] = ()) -> None: def download(url: str) -> None: """A simple file downloader. - :param url: URL to the file. + :param str url: URL to the file. """ fn = url.split("/")[-1] msg.note(f"Downloading {fn} ..") @@ -57,12 +57,12 @@ def download(url: str) -> None: msg.done("Done!") -def replace_lines(filename: Path, og_lines: Tuple[str], nw_lines: Tuple[str]) -> None: +def replace_lines(filename: Path, og_lines: tuple[str, ...], nw_lines: tuple[str, ...]) -> None: """Replace lines in the specified file. - :param filename: Path to the filename. - :param og_lines: Original lines to be replaced. - :param nw_lines: New lines in place of original lines. + :param Path filename: Path to the filename. + :param tuple[str,...] og_lines: Original lines to be replaced. + :param tuple[str,...] nw_lines: New lines in place of original lines. """ filename_new = Path(str(filename) + "_new") with open(filename, encoding="utf-8") as data: @@ -79,10 +79,10 @@ def replace_lines(filename: Path, og_lines: Tuple[str], nw_lines: Tuple[str]) -> def replace_nth(filename: Path, og_string: str, nw_string: str, occurence: int) -> None: """Replace the n-th occurence of subtring in specified file. - :param filename: Path to the filename. - :param og_string: Original string to be replaced. - :param nw_string: New string used to replace the original one. - :param occurence: The index of occurence to replace. + :param Path filename: Path to the filename. + :param str og_string: Original string to be replaced. + :param str nw_string: New string used to replace the original one. + :param int occurence: The index of occurence to replace. """ filename_new = Path(str(filename) + "_new") with open(filename, encoding="utf-8") as data: @@ -98,12 +98,12 @@ def replace_nth(filename: Path, og_string: str, nw_string: str, occurence: int) os.replace(filename_new, filename) -def insert_before_line(filename: str, pointer_line: str, new_line: str) -> None: +def insert_before_line(filename: str | Path, pointer_line: str, new_line: str) -> None: """Insert new line before the specified one. - :param filename: Name of the file. - :param pointer_line: The line before which new line will be inserted. - :param new_line: The line being inserted. + :param str/Path filename: Name of the file. + :param str pointer_line: The line before which new line will be inserted. + :param str new_line: The line being inserted. """ with open(filename, "r+", encoding="utf-8") as f: a = [x.rstrip() for x in f] @@ -119,10 +119,10 @@ def insert_before_line(filename: str, pointer_line: str, new_line: str) -> None: f.write(line + "\n") -def apply_patch(filename: str) -> None: +def apply_patch(filename: str | Path) -> None: """Apply .patch file. - :param filename: Name of the .patch file. + :param str/Path filename: Name of the .patch file. """ msg.note(f"Applying patch: {filename}") ccmd.launch(f"patch -p1 -s --no-backup-if-mismatch -i {filename}") diff --git a/builder/tools/messages.py b/builder/tools/messages.py index 8bae3fb..8454b5c 100644 --- a/builder/tools/messages.py +++ b/builder/tools/messages.py @@ -6,7 +6,7 @@ def banner(text: str) -> None: """A custom simple banner. - :param text: Text to wrap. + :param str text: Text to wrap. """ banner_len = len(text) + 6 print("\n" + "*" * banner_len) @@ -19,7 +19,7 @@ def banner(text: str) -> None: def note(text: str) -> None: """A "note" text wrapper. - :param text: Text to wrap. + :param str text: Text to wrap. """ print(f"[ * ] {text}") @@ -29,7 +29,7 @@ def error(text: str, dont_exit: bool = False) -> None: Includes system exit with an error code. - :param text: Text to wrap. + :param str text: Text to wrap. """ print(f"[ ! ] {text}", file=sys.stderr) if not dont_exit: @@ -39,7 +39,7 @@ def error(text: str, dont_exit: bool = False) -> None: def cancel(text: str) -> None: """A "cancel" text wrapper. - :param text: Text to wrap. + :param str text: Text to wrap. """ print(f"[ ~ ] {text}") sys.exit(0) @@ -48,7 +48,7 @@ def cancel(text: str) -> None: def done(text: str) -> None: """A "done" text wrapper. - :param text: Text to wrap. + :param str text: Text to wrap. """ print(f"[ \u2713 ] {text}") @@ -58,15 +58,6 @@ def debug(text: str) -> None: Intended for debugging sessions. - :param text: Text to wrap. + :param str text: Text to wrap. """ print(f"[ DEBUG ] {text}") - - -def outputstream() -> None: - """Stream output messages.""" - stream = os.getenv("OSTREAM", "screen") - if stream != "screen": - sys.stdout = open(stream, "a") - else: - sys.stdout = io.TextIOWrapper(open(sys.stdout.fileno(), 'wb', 0), write_through=True) diff --git a/builder/utils/bridge.py b/builder/utils/bridge.py index 56cbfca..a5c9e64 100644 --- a/builder/utils/bridge.py +++ b/builder/utils/bridge.py @@ -109,7 +109,8 @@ def main(args: argparse.Namespace) -> None: # if no command was selected, then shared tools are (supposed to be) installed if args.shared: rm = ResourceManager() - rm.path_gen() + rm.read_data() + rm.generate_paths() rm.download() else: # technically this part of code cannot be reached and is just an extra precaution @@ -117,5 +118,4 @@ def main(args: argparse.Namespace) -> None: if __name__ == "__main__": - msg.outputstream() main(parse_args()) diff --git a/builder/utils/resource_manager.py b/builder/utils/resource_manager.py index 3b398bf..ef41df0 100644 --- a/builder/utils/resource_manager.py +++ b/builder/utils/resource_manager.py @@ -1,6 +1,7 @@ import os import json import tarfile +from pathlib import Path from typing import Optional from builder.tools import cleaning as cm, commands as ccmd, fileoperations as fo, messages as msg @@ -10,20 +11,25 @@ class ResourceManager: """An entity for managing build resources.""" - paths: list[str] = [] + _data: dict[str, dict[str, str]] = {} + paths: dict[str, Path] = {} def __init__( - self, - codename: Optional[str] = None, - lkv: Optional[str] = None, - base: Optional[str] = None - ) -> None: + self, + codename: Optional[str] = None, + lkv: Optional[str] = None, + base: Optional[str] = None + ) -> None: self._codename = codename self._lkv = lkv self._base = base - def path_gen(self) -> dict[str]: - """Generate paths from JSON data.""" + def __getitem__(self, arg: Path) -> slice: + """A custom getitem implementation for accessing data via Path-type indexes.""" + return slice(*[{True: lambda n: None, False: int}[x == ""](x) for x in (str(arg).split(":") + ["", "", ""])[:3]]) + + def read_data(self) -> None: + """Read data from all of the JSON files.""" os.chdir(dcfg.root) # define paths tools = "" @@ -42,22 +48,25 @@ def path_gen(self) -> dict[str]: msg.error("Arguments were specified for an unsupported build, exiting..") device = {self._codename: data[self._codename][self._lkv][self._base]} # join tools and devices manifests - self.paths = {**tools, **device} + self._data = {**tools, **device} else: - self.paths = tools + self._data = tools msg.note("Only shared tools are installed.") - for e in self.paths: + + def generate_paths(self) -> None: + """Generate paths with Path objects.""" + for e in self._data: # convert path into it's absolute form - self.paths[e]["path"] = dcfg.root / self.paths[e]["path"] + self.paths[e] = dcfg.root / self._data[e]["path"] def download(self) -> None: """Download files from URLs.""" - for e in self.paths: + for e in self._data: # break data into individual required vars - path = self.paths[e]["path"] - url = self.paths[e]["url"] + path = Path(self._data[e]["path"]) # type: ignore + url = self._data[e]["url"] # type: ignore # break further processing into "generic" and "git" groups - ftype = self.paths[e]["type"] + ftype = self._data[e]["type"] # type: ignore match ftype: case "generic": # download and unpack @@ -76,8 +85,8 @@ def download(self) -> None: msg.note(f"Found an existing path: {path}") case "git": # break data into individual vars - branch = self.paths[e]["branch"] - commit = self.paths[e]["commit"] + branch = self._data[e]["branch"] # type: ignore + commit = self._data[e]["commit"] # type: ignore cmd = f"git clone -b {branch} --depth 1 --remote-submodules --recurse-submodules --shallow-submodules {url} {path}" # full commit history is required in two instances: # - for KernelSU -- to define it's version based on *full* commit history; @@ -100,6 +109,6 @@ def download(self) -> None: def export_path(self) -> None: """Add path to PATH.""" for elem in self.paths: - p = self.paths[elem]["path"] + p = self.paths[elem] pathenv = str(f"{p}/bin/") - os.environ["PATH"] = pathenv + os.pathsep + os.getenv("PATH") + os.environ["PATH"] = pathenv + os.pathsep + os.getenv("PATH") # type: ignore diff --git a/poetry.lock b/poetry.lock index aa1cfc7..90660ee 100644 --- a/poetry.lock +++ b/poetry.lock @@ -298,13 +298,13 @@ files = [ [[package]] name = "idna" -version = "3.6" +version = "3.7" description = "Internationalized Domain Names in Applications (IDNA)" optional = false python-versions = ">=3.5" files = [ - {file = "idna-3.6-py3-none-any.whl", hash = "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f"}, - {file = "idna-3.6.tar.gz", hash = "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca"}, + {file = "idna-3.7-py3-none-any.whl", hash = "sha256:82fee1fc78add43492d3a1898bfa6d8a904cc97d8427f683ed8e798d07761aa0"}, + {file = "idna-3.7.tar.gz", hash = "sha256:028ff3aadf0609c1fd278d8ea3089299412a7a8b9bd005dd08b9f8285bcb5cfc"}, ] [[package]] @@ -512,13 +512,13 @@ files = [ [[package]] name = "pluggy" -version = "1.4.0" +version = "1.5.0" description = "plugin and hook calling mechanisms for python" optional = false python-versions = ">=3.8" files = [ - {file = "pluggy-1.4.0-py3-none-any.whl", hash = "sha256:7db9f7b503d67d1c5b95f59773ebb58a8c1c288129a88665838012cfb07b8981"}, - {file = "pluggy-1.4.0.tar.gz", hash = "sha256:8c85c2876142a764e5b7548e7d9a0e0ddb46f5185161049a79b7e974454223be"}, + {file = "pluggy-1.5.0-py3-none-any.whl", hash = "sha256:44e1ad92c8ca002de6377e165f3e0f1be63266ab4d554740532335b9d75ea669"}, + {file = "pluggy-1.5.0.tar.gz", hash = "sha256:2cffa88e94fdc978c4c574f15f9e59b7f4201d439195c3715ca9e2486f1d0cf1"}, ] [package.extras] @@ -537,18 +537,18 @@ files = [ [[package]] name = "pydantic" -version = "2.6.4" +version = "2.7.0" description = "Data validation using Python type hints" optional = false python-versions = ">=3.8" files = [ - {file = "pydantic-2.6.4-py3-none-any.whl", hash = "sha256:cc46fce86607580867bdc3361ad462bab9c222ef042d3da86f2fb333e1d916c5"}, - {file = "pydantic-2.6.4.tar.gz", hash = "sha256:b1704e0847db01817624a6b86766967f552dd9dbf3afba4004409f908dcc84e6"}, + {file = "pydantic-2.7.0-py3-none-any.whl", hash = "sha256:9dee74a271705f14f9a1567671d144a851c675b072736f0a7b2608fd9e495352"}, + {file = "pydantic-2.7.0.tar.gz", hash = "sha256:b5ecdd42262ca2462e2624793551e80911a1e989f462910bb81aef974b4bb383"}, ] [package.dependencies] annotated-types = ">=0.4.0" -pydantic-core = "2.16.3" +pydantic-core = "2.18.1" typing-extensions = ">=4.6.1" [package.extras] @@ -556,90 +556,90 @@ email = ["email-validator (>=2.0.0)"] [[package]] name = "pydantic-core" -version = "2.16.3" -description = "" +version = "2.18.1" +description = "Core functionality for Pydantic validation and serialization" optional = false python-versions = ">=3.8" files = [ - {file = "pydantic_core-2.16.3-cp310-cp310-macosx_10_12_x86_64.whl", hash = "sha256:75b81e678d1c1ede0785c7f46690621e4c6e63ccd9192af1f0bd9d504bbb6bf4"}, - {file = "pydantic_core-2.16.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:9c865a7ee6f93783bd5d781af5a4c43dadc37053a5b42f7d18dc019f8c9d2bd1"}, - {file = "pydantic_core-2.16.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:162e498303d2b1c036b957a1278fa0899d02b2842f1ff901b6395104c5554a45"}, - {file = "pydantic_core-2.16.3-cp310-cp310-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:2f583bd01bbfbff4eaee0868e6fc607efdfcc2b03c1c766b06a707abbc856187"}, - {file = "pydantic_core-2.16.3-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b926dd38db1519ed3043a4de50214e0d600d404099c3392f098a7f9d75029ff8"}, - {file = "pydantic_core-2.16.3-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:716b542728d4c742353448765aa7cdaa519a7b82f9564130e2b3f6766018c9ec"}, - {file = "pydantic_core-2.16.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fc4ad7f7ee1a13d9cb49d8198cd7d7e3aa93e425f371a68235f784e99741561f"}, - {file = "pydantic_core-2.16.3-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:bd87f48924f360e5d1c5f770d6155ce0e7d83f7b4e10c2f9ec001c73cf475c99"}, - {file = "pydantic_core-2.16.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:0df446663464884297c793874573549229f9eca73b59360878f382a0fc085979"}, - {file = "pydantic_core-2.16.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:4df8a199d9f6afc5ae9a65f8f95ee52cae389a8c6b20163762bde0426275b7db"}, - {file = "pydantic_core-2.16.3-cp310-none-win32.whl", hash = "sha256:456855f57b413f077dff513a5a28ed838dbbb15082ba00f80750377eed23d132"}, - {file = "pydantic_core-2.16.3-cp310-none-win_amd64.whl", hash = "sha256:732da3243e1b8d3eab8c6ae23ae6a58548849d2e4a4e03a1924c8ddf71a387cb"}, - {file = "pydantic_core-2.16.3-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:519ae0312616026bf4cedc0fe459e982734f3ca82ee8c7246c19b650b60a5ee4"}, - {file = "pydantic_core-2.16.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:b3992a322a5617ded0a9f23fd06dbc1e4bd7cf39bc4ccf344b10f80af58beacd"}, - {file = "pydantic_core-2.16.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8d62da299c6ecb04df729e4b5c52dc0d53f4f8430b4492b93aa8de1f541c4aac"}, - {file = "pydantic_core-2.16.3-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:2acca2be4bb2f2147ada8cac612f8a98fc09f41c89f87add7256ad27332c2fda"}, - {file = "pydantic_core-2.16.3-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1b662180108c55dfbf1280d865b2d116633d436cfc0bba82323554873967b340"}, - {file = "pydantic_core-2.16.3-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e7c6ed0dc9d8e65f24f5824291550139fe6f37fac03788d4580da0d33bc00c97"}, - {file = "pydantic_core-2.16.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a6b1bb0827f56654b4437955555dc3aeeebeddc47c2d7ed575477f082622c49e"}, - {file = "pydantic_core-2.16.3-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:e56f8186d6210ac7ece503193ec84104da7ceb98f68ce18c07282fcc2452e76f"}, - {file = "pydantic_core-2.16.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:936e5db01dd49476fa8f4383c259b8b1303d5dd5fb34c97de194560698cc2c5e"}, - {file = "pydantic_core-2.16.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:33809aebac276089b78db106ee692bdc9044710e26f24a9a2eaa35a0f9fa70ba"}, - {file = "pydantic_core-2.16.3-cp311-none-win32.whl", hash = "sha256:ded1c35f15c9dea16ead9bffcde9bb5c7c031bff076355dc58dcb1cb436c4721"}, - {file = "pydantic_core-2.16.3-cp311-none-win_amd64.whl", hash = "sha256:d89ca19cdd0dd5f31606a9329e309d4fcbb3df860960acec32630297d61820df"}, - {file = "pydantic_core-2.16.3-cp311-none-win_arm64.whl", hash = "sha256:6162f8d2dc27ba21027f261e4fa26f8bcb3cf9784b7f9499466a311ac284b5b9"}, - {file = "pydantic_core-2.16.3-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:0f56ae86b60ea987ae8bcd6654a887238fd53d1384f9b222ac457070b7ac4cff"}, - {file = "pydantic_core-2.16.3-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:c9bd22a2a639e26171068f8ebb5400ce2c1bc7d17959f60a3b753ae13c632975"}, - {file = "pydantic_core-2.16.3-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4204e773b4b408062960e65468d5346bdfe139247ee5f1ca2a378983e11388a2"}, - {file = "pydantic_core-2.16.3-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:f651dd19363c632f4abe3480a7c87a9773be27cfe1341aef06e8759599454120"}, - {file = "pydantic_core-2.16.3-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:aaf09e615a0bf98d406657e0008e4a8701b11481840be7d31755dc9f97c44053"}, - {file = "pydantic_core-2.16.3-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8e47755d8152c1ab5b55928ab422a76e2e7b22b5ed8e90a7d584268dd49e9c6b"}, - {file = "pydantic_core-2.16.3-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:500960cb3a0543a724a81ba859da816e8cf01b0e6aaeedf2c3775d12ee49cade"}, - {file = "pydantic_core-2.16.3-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:cf6204fe865da605285c34cf1172879d0314ff267b1c35ff59de7154f35fdc2e"}, - {file = "pydantic_core-2.16.3-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:d33dd21f572545649f90c38c227cc8631268ba25c460b5569abebdd0ec5974ca"}, - {file = "pydantic_core-2.16.3-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:49d5d58abd4b83fb8ce763be7794d09b2f50f10aa65c0f0c1696c677edeb7cbf"}, - {file = "pydantic_core-2.16.3-cp312-none-win32.whl", hash = "sha256:f53aace168a2a10582e570b7736cc5bef12cae9cf21775e3eafac597e8551fbe"}, - {file = "pydantic_core-2.16.3-cp312-none-win_amd64.whl", hash = "sha256:0d32576b1de5a30d9a97f300cc6a3f4694c428d956adbc7e6e2f9cad279e45ed"}, - {file = "pydantic_core-2.16.3-cp312-none-win_arm64.whl", hash = "sha256:ec08be75bb268473677edb83ba71e7e74b43c008e4a7b1907c6d57e940bf34b6"}, - {file = "pydantic_core-2.16.3-cp38-cp38-macosx_10_12_x86_64.whl", hash = "sha256:b1f6f5938d63c6139860f044e2538baeee6f0b251a1816e7adb6cbce106a1f01"}, - {file = "pydantic_core-2.16.3-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:2a1ef6a36fdbf71538142ed604ad19b82f67b05749512e47f247a6ddd06afdc7"}, - {file = "pydantic_core-2.16.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:704d35ecc7e9c31d48926150afada60401c55efa3b46cd1ded5a01bdffaf1d48"}, - {file = "pydantic_core-2.16.3-cp38-cp38-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:d937653a696465677ed583124b94a4b2d79f5e30b2c46115a68e482c6a591c8a"}, - {file = "pydantic_core-2.16.3-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c9803edf8e29bd825f43481f19c37f50d2b01899448273b3a7758441b512acf8"}, - {file = "pydantic_core-2.16.3-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:72282ad4892a9fb2da25defeac8c2e84352c108705c972db82ab121d15f14e6d"}, - {file = "pydantic_core-2.16.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7f752826b5b8361193df55afcdf8ca6a57d0232653494ba473630a83ba50d8c9"}, - {file = "pydantic_core-2.16.3-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:4384a8f68ddb31a0b0c3deae88765f5868a1b9148939c3f4121233314ad5532c"}, - {file = "pydantic_core-2.16.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:a4b2bf78342c40b3dc830880106f54328928ff03e357935ad26c7128bbd66ce8"}, - {file = "pydantic_core-2.16.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:13dcc4802961b5f843a9385fc821a0b0135e8c07fc3d9949fd49627c1a5e6ae5"}, - {file = "pydantic_core-2.16.3-cp38-none-win32.whl", hash = "sha256:e3e70c94a0c3841e6aa831edab1619ad5c511199be94d0c11ba75fe06efe107a"}, - {file = "pydantic_core-2.16.3-cp38-none-win_amd64.whl", hash = "sha256:ecdf6bf5f578615f2e985a5e1f6572e23aa632c4bd1dc67f8f406d445ac115ed"}, - {file = "pydantic_core-2.16.3-cp39-cp39-macosx_10_12_x86_64.whl", hash = "sha256:bda1ee3e08252b8d41fa5537413ffdddd58fa73107171a126d3b9ff001b9b820"}, - {file = "pydantic_core-2.16.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:21b888c973e4f26b7a96491c0965a8a312e13be108022ee510248fe379a5fa23"}, - {file = "pydantic_core-2.16.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:be0ec334369316fa73448cc8c982c01e5d2a81c95969d58b8f6e272884df0074"}, - {file = "pydantic_core-2.16.3-cp39-cp39-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:b5b6079cc452a7c53dd378c6f881ac528246b3ac9aae0f8eef98498a75657805"}, - {file = "pydantic_core-2.16.3-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:7ee8d5f878dccb6d499ba4d30d757111847b6849ae07acdd1205fffa1fc1253c"}, - {file = "pydantic_core-2.16.3-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:7233d65d9d651242a68801159763d09e9ec96e8a158dbf118dc090cd77a104c9"}, - {file = "pydantic_core-2.16.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c6119dc90483a5cb50a1306adb8d52c66e447da88ea44f323e0ae1a5fcb14256"}, - {file = "pydantic_core-2.16.3-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:578114bc803a4c1ff9946d977c221e4376620a46cf78da267d946397dc9514a8"}, - {file = "pydantic_core-2.16.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:d8f99b147ff3fcf6b3cc60cb0c39ea443884d5559a30b1481e92495f2310ff2b"}, - {file = "pydantic_core-2.16.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:4ac6b4ce1e7283d715c4b729d8f9dab9627586dafce81d9eaa009dd7f25dd972"}, - {file = "pydantic_core-2.16.3-cp39-none-win32.whl", hash = "sha256:e7774b570e61cb998490c5235740d475413a1f6de823169b4cf94e2fe9e9f6b2"}, - {file = "pydantic_core-2.16.3-cp39-none-win_amd64.whl", hash = "sha256:9091632a25b8b87b9a605ec0e61f241c456e9248bfdcf7abdf344fdb169c81cf"}, - {file = "pydantic_core-2.16.3-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:36fa178aacbc277bc6b62a2c3da95226520da4f4e9e206fdf076484363895d2c"}, - {file = "pydantic_core-2.16.3-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:dcca5d2bf65c6fb591fff92da03f94cd4f315972f97c21975398bd4bd046854a"}, - {file = "pydantic_core-2.16.3-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2a72fb9963cba4cd5793854fd12f4cfee731e86df140f59ff52a49b3552db241"}, - {file = "pydantic_core-2.16.3-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b60cc1a081f80a2105a59385b92d82278b15d80ebb3adb200542ae165cd7d183"}, - {file = "pydantic_core-2.16.3-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:cbcc558401de90a746d02ef330c528f2e668c83350f045833543cd57ecead1ad"}, - {file = "pydantic_core-2.16.3-pp310-pypy310_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:fee427241c2d9fb7192b658190f9f5fd6dfe41e02f3c1489d2ec1e6a5ab1e04a"}, - {file = "pydantic_core-2.16.3-pp310-pypy310_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:f4cb85f693044e0f71f394ff76c98ddc1bc0953e48c061725e540396d5c8a2e1"}, - {file = "pydantic_core-2.16.3-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:b29eeb887aa931c2fcef5aa515d9d176d25006794610c264ddc114c053bf96fe"}, - {file = "pydantic_core-2.16.3-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:a425479ee40ff021f8216c9d07a6a3b54b31c8267c6e17aa88b70d7ebd0e5e5b"}, - {file = "pydantic_core-2.16.3-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:5c5cbc703168d1b7a838668998308018a2718c2130595e8e190220238addc96f"}, - {file = "pydantic_core-2.16.3-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:99b6add4c0b39a513d323d3b93bc173dac663c27b99860dd5bf491b240d26137"}, - {file = "pydantic_core-2.16.3-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:75f76ee558751746d6a38f89d60b6228fa174e5172d143886af0f85aa306fd89"}, - {file = "pydantic_core-2.16.3-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:00ee1c97b5364b84cb0bd82e9bbf645d5e2871fb8c58059d158412fee2d33d8a"}, - {file = "pydantic_core-2.16.3-pp39-pypy39_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:287073c66748f624be4cef893ef9174e3eb88fe0b8a78dc22e88eca4bc357ca6"}, - {file = "pydantic_core-2.16.3-pp39-pypy39_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:ed25e1835c00a332cb10c683cd39da96a719ab1dfc08427d476bce41b92531fc"}, - {file = "pydantic_core-2.16.3-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:86b3d0033580bd6bbe07590152007275bd7af95f98eaa5bd36f3da219dcd93da"}, - {file = "pydantic_core-2.16.3.tar.gz", hash = "sha256:1cac689f80a3abab2d3c0048b29eea5751114054f032a941a32de4c852c59cad"}, + {file = "pydantic_core-2.18.1-cp310-cp310-macosx_10_12_x86_64.whl", hash = "sha256:ee9cf33e7fe14243f5ca6977658eb7d1042caaa66847daacbd2117adb258b226"}, + {file = "pydantic_core-2.18.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:6b7bbb97d82659ac8b37450c60ff2e9f97e4eb0f8a8a3645a5568b9334b08b50"}, + {file = "pydantic_core-2.18.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:df4249b579e75094f7e9bb4bd28231acf55e308bf686b952f43100a5a0be394c"}, + {file = "pydantic_core-2.18.1-cp310-cp310-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:d0491006a6ad20507aec2be72e7831a42efc93193d2402018007ff827dc62926"}, + {file = "pydantic_core-2.18.1-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2ae80f72bb7a3e397ab37b53a2b49c62cc5496412e71bc4f1277620a7ce3f52b"}, + {file = "pydantic_core-2.18.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:58aca931bef83217fca7a390e0486ae327c4af9c3e941adb75f8772f8eeb03a1"}, + {file = "pydantic_core-2.18.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1be91ad664fc9245404a789d60cba1e91c26b1454ba136d2a1bf0c2ac0c0505a"}, + {file = "pydantic_core-2.18.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:667880321e916a8920ef49f5d50e7983792cf59f3b6079f3c9dac2b88a311d17"}, + {file = "pydantic_core-2.18.1-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:f7054fdc556f5421f01e39cbb767d5ec5c1139ea98c3e5b350e02e62201740c7"}, + {file = "pydantic_core-2.18.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:030e4f9516f9947f38179249778709a460a3adb516bf39b5eb9066fcfe43d0e6"}, + {file = "pydantic_core-2.18.1-cp310-none-win32.whl", hash = "sha256:2e91711e36e229978d92642bfc3546333a9127ecebb3f2761372e096395fc649"}, + {file = "pydantic_core-2.18.1-cp310-none-win_amd64.whl", hash = "sha256:9a29726f91c6cb390b3c2338f0df5cd3e216ad7a938762d11c994bb37552edb0"}, + {file = "pydantic_core-2.18.1-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:9ece8a49696669d483d206b4474c367852c44815fca23ac4e48b72b339807f80"}, + {file = "pydantic_core-2.18.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:7a5d83efc109ceddb99abd2c1316298ced2adb4570410defe766851a804fcd5b"}, + {file = "pydantic_core-2.18.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5f7973c381283783cd1043a8c8f61ea5ce7a3a58b0369f0ee0ee975eaf2f2a1b"}, + {file = "pydantic_core-2.18.1-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:54c7375c62190a7845091f521add19b0f026bcf6ae674bdb89f296972272e86d"}, + {file = "pydantic_core-2.18.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:dd63cec4e26e790b70544ae5cc48d11b515b09e05fdd5eff12e3195f54b8a586"}, + {file = "pydantic_core-2.18.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:561cf62c8a3498406495cfc49eee086ed2bb186d08bcc65812b75fda42c38294"}, + {file = "pydantic_core-2.18.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:68717c38a68e37af87c4da20e08f3e27d7e4212e99e96c3d875fbf3f4812abfc"}, + {file = "pydantic_core-2.18.1-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2d5728e93d28a3c63ee513d9ffbac9c5989de8c76e049dbcb5bfe4b923a9739d"}, + {file = "pydantic_core-2.18.1-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:f0f17814c505f07806e22b28856c59ac80cee7dd0fbb152aed273e116378f519"}, + {file = "pydantic_core-2.18.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:d816f44a51ba5175394bc6c7879ca0bd2be560b2c9e9f3411ef3a4cbe644c2e9"}, + {file = "pydantic_core-2.18.1-cp311-none-win32.whl", hash = "sha256:09f03dfc0ef8c22622eaa8608caa4a1e189cfb83ce847045eca34f690895eccb"}, + {file = "pydantic_core-2.18.1-cp311-none-win_amd64.whl", hash = "sha256:27f1009dc292f3b7ca77feb3571c537276b9aad5dd4efb471ac88a8bd09024e9"}, + {file = "pydantic_core-2.18.1-cp311-none-win_arm64.whl", hash = "sha256:48dd883db92e92519201f2b01cafa881e5f7125666141a49ffba8b9facc072b0"}, + {file = "pydantic_core-2.18.1-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:b6b0e4912030c6f28bcb72b9ebe4989d6dc2eebcd2a9cdc35fefc38052dd4fe8"}, + {file = "pydantic_core-2.18.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:f3202a429fe825b699c57892d4371c74cc3456d8d71b7f35d6028c96dfecad31"}, + {file = "pydantic_core-2.18.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a3982b0a32d0a88b3907e4b0dc36809fda477f0757c59a505d4e9b455f384b8b"}, + {file = "pydantic_core-2.18.1-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:25595ac311f20e5324d1941909b0d12933f1fd2171075fcff763e90f43e92a0d"}, + {file = "pydantic_core-2.18.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:14fe73881cf8e4cbdaded8ca0aa671635b597e42447fec7060d0868b52d074e6"}, + {file = "pydantic_core-2.18.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ca976884ce34070799e4dfc6fbd68cb1d181db1eefe4a3a94798ddfb34b8867f"}, + {file = "pydantic_core-2.18.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:684d840d2c9ec5de9cb397fcb3f36d5ebb6fa0d94734f9886032dd796c1ead06"}, + {file = "pydantic_core-2.18.1-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:54764c083bbe0264f0f746cefcded6cb08fbbaaf1ad1d78fb8a4c30cff999a90"}, + {file = "pydantic_core-2.18.1-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:201713f2f462e5c015b343e86e68bd8a530a4f76609b33d8f0ec65d2b921712a"}, + {file = "pydantic_core-2.18.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:fd1a9edb9dd9d79fbeac1ea1f9a8dd527a6113b18d2e9bcc0d541d308dae639b"}, + {file = "pydantic_core-2.18.1-cp312-none-win32.whl", hash = "sha256:d5e6b7155b8197b329dc787356cfd2684c9d6a6b1a197f6bbf45f5555a98d411"}, + {file = "pydantic_core-2.18.1-cp312-none-win_amd64.whl", hash = "sha256:9376d83d686ec62e8b19c0ac3bf8d28d8a5981d0df290196fb6ef24d8a26f0d6"}, + {file = "pydantic_core-2.18.1-cp312-none-win_arm64.whl", hash = "sha256:c562b49c96906b4029b5685075fe1ebd3b5cc2601dfa0b9e16c2c09d6cbce048"}, + {file = "pydantic_core-2.18.1-cp38-cp38-macosx_10_12_x86_64.whl", hash = "sha256:3e352f0191d99fe617371096845070dee295444979efb8f27ad941227de6ad09"}, + {file = "pydantic_core-2.18.1-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:c0295d52b012cbe0d3059b1dba99159c3be55e632aae1999ab74ae2bd86a33d7"}, + {file = "pydantic_core-2.18.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:56823a92075780582d1ffd4489a2e61d56fd3ebb4b40b713d63f96dd92d28144"}, + {file = "pydantic_core-2.18.1-cp38-cp38-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:dd3f79e17b56741b5177bcc36307750d50ea0698df6aa82f69c7db32d968c1c2"}, + {file = "pydantic_core-2.18.1-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:38a5024de321d672a132b1834a66eeb7931959c59964b777e8f32dbe9523f6b1"}, + {file = "pydantic_core-2.18.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d2ce426ee691319d4767748c8e0895cfc56593d725594e415f274059bcf3cb76"}, + {file = "pydantic_core-2.18.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2adaeea59849ec0939af5c5d476935f2bab4b7f0335b0110f0f069a41024278e"}, + {file = "pydantic_core-2.18.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:9b6431559676a1079eac0f52d6d0721fb8e3c5ba43c37bc537c8c83724031feb"}, + {file = "pydantic_core-2.18.1-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:85233abb44bc18d16e72dc05bf13848a36f363f83757541f1a97db2f8d58cfd9"}, + {file = "pydantic_core-2.18.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:641a018af4fe48be57a2b3d7a1f0f5dbca07c1d00951d3d7463f0ac9dac66622"}, + {file = "pydantic_core-2.18.1-cp38-none-win32.whl", hash = "sha256:63d7523cd95d2fde0d28dc42968ac731b5bb1e516cc56b93a50ab293f4daeaad"}, + {file = "pydantic_core-2.18.1-cp38-none-win_amd64.whl", hash = "sha256:907a4d7720abfcb1c81619863efd47c8a85d26a257a2dbebdb87c3b847df0278"}, + {file = "pydantic_core-2.18.1-cp39-cp39-macosx_10_12_x86_64.whl", hash = "sha256:aad17e462f42ddbef5984d70c40bfc4146c322a2da79715932cd8976317054de"}, + {file = "pydantic_core-2.18.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:94b9769ba435b598b547c762184bcfc4783d0d4c7771b04a3b45775c3589ca44"}, + {file = "pydantic_core-2.18.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:80e0e57cc704a52fb1b48f16d5b2c8818da087dbee6f98d9bf19546930dc64b5"}, + {file = "pydantic_core-2.18.1-cp39-cp39-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:76b86e24039c35280ceee6dce7e62945eb93a5175d43689ba98360ab31eebc4a"}, + {file = "pydantic_core-2.18.1-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:12a05db5013ec0ca4a32cc6433f53faa2a014ec364031408540ba858c2172bb0"}, + {file = "pydantic_core-2.18.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:250ae39445cb5475e483a36b1061af1bc233de3e9ad0f4f76a71b66231b07f88"}, + {file = "pydantic_core-2.18.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a32204489259786a923e02990249c65b0f17235073149d0033efcebe80095570"}, + {file = "pydantic_core-2.18.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:6395a4435fa26519fd96fdccb77e9d00ddae9dd6c742309bd0b5610609ad7fb2"}, + {file = "pydantic_core-2.18.1-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:2533ad2883f001efa72f3d0e733fb846710c3af6dcdd544fe5bf14fa5fe2d7db"}, + {file = "pydantic_core-2.18.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:b560b72ed4816aee52783c66854d96157fd8175631f01ef58e894cc57c84f0f6"}, + {file = "pydantic_core-2.18.1-cp39-none-win32.whl", hash = "sha256:582cf2cead97c9e382a7f4d3b744cf0ef1a6e815e44d3aa81af3ad98762f5a9b"}, + {file = "pydantic_core-2.18.1-cp39-none-win_amd64.whl", hash = "sha256:ca71d501629d1fa50ea7fa3b08ba884fe10cefc559f5c6c8dfe9036c16e8ae89"}, + {file = "pydantic_core-2.18.1-pp310-pypy310_pp73-macosx_10_12_x86_64.whl", hash = "sha256:e178e5b66a06ec5bf51668ec0d4ac8cfb2bdcb553b2c207d58148340efd00143"}, + {file = "pydantic_core-2.18.1-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:72722ce529a76a4637a60be18bd789d8fb871e84472490ed7ddff62d5fed620d"}, + {file = "pydantic_core-2.18.1-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2fe0c1ce5b129455e43f941f7a46f61f3d3861e571f2905d55cdbb8b5c6f5e2c"}, + {file = "pydantic_core-2.18.1-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d4284c621f06a72ce2cb55f74ea3150113d926a6eb78ab38340c08f770eb9b4d"}, + {file = "pydantic_core-2.18.1-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:1a0c3e718f4e064efde68092d9d974e39572c14e56726ecfaeebbe6544521f47"}, + {file = "pydantic_core-2.18.1-pp310-pypy310_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:2027493cc44c23b598cfaf200936110433d9caa84e2c6cf487a83999638a96ac"}, + {file = "pydantic_core-2.18.1-pp310-pypy310_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:76909849d1a6bffa5a07742294f3fa1d357dc917cb1fe7b470afbc3a7579d539"}, + {file = "pydantic_core-2.18.1-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:ee7ccc7fb7e921d767f853b47814c3048c7de536663e82fbc37f5eb0d532224b"}, + {file = "pydantic_core-2.18.1-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:ee2794111c188548a4547eccc73a6a8527fe2af6cf25e1a4ebda2fd01cdd2e60"}, + {file = "pydantic_core-2.18.1-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:a139fe9f298dc097349fb4f28c8b81cc7a202dbfba66af0e14be5cfca4ef7ce5"}, + {file = "pydantic_core-2.18.1-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d074b07a10c391fc5bbdcb37b2f16f20fcd9e51e10d01652ab298c0d07908ee2"}, + {file = "pydantic_core-2.18.1-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c69567ddbac186e8c0aadc1f324a60a564cfe25e43ef2ce81bcc4b8c3abffbae"}, + {file = "pydantic_core-2.18.1-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:baf1c7b78cddb5af00971ad5294a4583188bda1495b13760d9f03c9483bb6203"}, + {file = "pydantic_core-2.18.1-pp39-pypy39_pp73-musllinux_1_1_aarch64.whl", hash = "sha256:2684a94fdfd1b146ff10689c6e4e815f6a01141781c493b97342cdc5b06f4d5d"}, + {file = "pydantic_core-2.18.1-pp39-pypy39_pp73-musllinux_1_1_x86_64.whl", hash = "sha256:73c1bc8a86a5c9e8721a088df234265317692d0b5cd9e86e975ce3bc3db62a59"}, + {file = "pydantic_core-2.18.1-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:e60defc3c15defb70bb38dd605ff7e0fae5f6c9c7cbfe0ad7868582cb7e844a6"}, + {file = "pydantic_core-2.18.1.tar.gz", hash = "sha256:de9d3e8717560eb05e28739d1b35e4eac2e458553a52a301e51352a7ffc86a35"}, ] [package.dependencies] @@ -679,13 +679,13 @@ tests = ["coverage[toml] (==5.0.4)", "pytest (>=6.0.0,<7.0.0)"] [[package]] name = "pyright" -version = "1.1.357" +version = "1.1.359" description = "Command line wrapper for pyright" optional = false python-versions = ">=3.7" files = [ - {file = "pyright-1.1.357-py3-none-any.whl", hash = "sha256:1cf29ee38e4928131895cd8e90eef37b5b77e2ed72a14e6e8e2405266f5f0aca"}, - {file = "pyright-1.1.357.tar.gz", hash = "sha256:7c66261116c78c5fa9629134fe85c54cc5302ab73e376be4b0a99d89c80a9403"}, + {file = "pyright-1.1.359-py3-none-any.whl", hash = "sha256:5582777be7eab73512277ac7da7b41e15bc0737f488629cb9babd96e0769be61"}, + {file = "pyright-1.1.359.tar.gz", hash = "sha256:f0eab50f3dafce8a7302caeafd6a733f39901a2bf5170bb23d77fd607c8a8dbc"}, ] [package.dependencies] @@ -848,18 +848,18 @@ jupyter = ["ipywidgets (>=7.5.1,<9)"] [[package]] name = "setuptools" -version = "69.2.0" +version = "69.5.1" description = "Easily download, build, install, upgrade, and uninstall Python packages" optional = false python-versions = ">=3.8" files = [ - {file = "setuptools-69.2.0-py3-none-any.whl", hash = "sha256:c21c49fb1042386df081cb5d86759792ab89efca84cf114889191cd09aacc80c"}, - {file = "setuptools-69.2.0.tar.gz", hash = "sha256:0ff4183f8f42cd8fa3acea16c45205521a4ef28f73c6391d8a25e92893134f2e"}, + {file = "setuptools-69.5.1-py3-none-any.whl", hash = "sha256:c636ac361bc47580504644275c9ad802c50415c7522212252c033bd15f301f32"}, + {file = "setuptools-69.5.1.tar.gz", hash = "sha256:6c1fccdac05a97e598fb0ae3bbed5904ccb317337a51139dcd51453611bbb987"}, ] [package.extras] -docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (<7.2.5)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier"] -testing = ["build[virtualenv]", "filelock (>=3.4.0)", "importlib-metadata", "ini2toml[lite] (>=0.9)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "mypy (==1.9)", "packaging (>=23.2)", "pip (>=19.1)", "pytest (>=6)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-home (>=0.5)", "pytest-mypy (>=0.9.1)", "pytest-perf", "pytest-ruff (>=0.2.1)", "pytest-timeout", "pytest-xdist (>=3)", "tomli", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] +docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier"] +testing = ["build[virtualenv]", "filelock (>=3.4.0)", "importlib-metadata", "ini2toml[lite] (>=0.9)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "mypy (==1.9)", "packaging (>=23.2)", "pip (>=19.1)", "pytest (>=6,!=8.1.1)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-home (>=0.5)", "pytest-mypy", "pytest-perf", "pytest-ruff (>=0.2.1)", "pytest-timeout", "pytest-xdist (>=3)", "tomli", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] testing-integration = ["build[virtualenv] (>=1.0.3)", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "packaging (>=23.2)", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"] [[package]] diff --git a/pyproject.toml b/pyproject.toml index 1716d6f..3a80029 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -20,6 +20,7 @@ bandit = "^1.7" pytest = "^8.0" pytest-cov = "^4.1" pyright = "^1.1" +pylint = "^3.1" [tool.coverage.run] source = [ @@ -33,6 +34,10 @@ source = [ [tool.coverage.report] show_missing = true +[tool.pyright] +include=["builder"] +pythonVersion = "3.12" + [build-system] requires = ["poetry-core"] build-backend = "poetry.core.masonry.api" diff --git a/scripts/multi_build.py b/scripts/multi_build.py index 5d56241..d10118a 100644 --- a/scripts/multi_build.py +++ b/scripts/multi_build.py @@ -19,8 +19,8 @@ def parse_args() -> argparse.Namespace: def rmove(src: Path, dst: Path) -> None: """Recursively move files from one directory to another. - :param src: Source path. - :param dst: Destination path. + :param Path src: Source path. + :param Path dst: Destination path. """ # for a directory (it's contents) if src.is_dir(): diff --git a/scripts/run_tests.py b/scripts/run_tests.py index 1bef0ad..d19f1a9 100644 --- a/scripts/run_tests.py +++ b/scripts/run_tests.py @@ -1,42 +1,47 @@ import os +import platform import subprocess from pathlib import Path from subprocess import CompletedProcess ROOTPATH: Path = Path(__file__).absolute().parents[1] +INTEPRETER: str = "python" if platform.system() == "Windows" else "python3" class Tester: """A single class for all types of tests.""" + @staticmethod def _launch_cmd(cmd: str) -> CompletedProcess: """Launch specified command.""" return subprocess.run(cmd, shell=True, check=True) - def pytest_checks(self) -> CompletedProcess: - """Run unit tests with Pytest and coverage checks.""" - os.environ["PYTHONPATH"] = ROOTPATH - return self._launch_cmd("python3 -m pytest -vv tests/ --cov") + def pyright_checks(self) -> None: + """Run typing checks with Pyright.""" + print("\n=== Pyright checks: Start ===") + self._launch_cmd(f"{INTEPRETER} -m pyright") + print("=== Pyright checks: Finish ===") - def pyright_checks(self) -> CompletedProcess: - """Run type (hint) checks with Pyright.""" - return self._launch_cmd("python3 -m pyright builder") + def pytest_checks(self) -> None: + """Run unit tests with Pytest and coverage checks.""" + os.environ["PYTHONPATH"] = str(ROOTPATH) + print("\n=== Pytest checks: Start ===") + self._launch_cmd(f"{INTEPRETER} -m pytest -vv tests/ --cov") + print("=== Pytest checks: Finish ===") - def bandit_checks(self) -> list[CompletedProcess]: + def bandit_checks(self) -> None: """Run SAST with Bandit.""" fmts = ("json", "html") - cps = [] for fmt in fmts: - cps.append(self._launch_cmd(f"python3 -m bandit -r -f {fmt} {ROOTPATH} -o report.{fmt}")) - return cps + self._launch_cmd(f"python3 -m bandit -r -f {fmt} {ROOTPATH} -o report.{fmt}") def main() -> None: os.chdir(ROOTPATH) t = Tester() - t.pytest_checks() t.pyright_checks() - t.bandit_checks() + t.pytest_checks() + #t.bandit_checks() if __name__ == "__main__": diff --git a/tests/unit/wrapper/core/test_kernel_builder.py b/tests/unit/wrapper/core/test_kernel_builder.py index 18204e4..4135180 100644 --- a/tests/unit/wrapper/core/test_kernel_builder.py +++ b/tests/unit/wrapper/core/test_kernel_builder.py @@ -21,7 +21,7 @@ ) ) ) -def test__defconfig__check(config: dict[str], expected_defconfig: Path) -> bool: +def test__defconfig__check(config: dict[str, str], expected_defconfig: Path) -> None: """Test defconfig path definition.""" t = KernelBuilder(**config) res_actual = t._defconfig