from pathlib import Path from loguru import logger from abratest.dir_manager import DirManager from abratest.env_manager import EnvFile, EnvManager from abratest.html_helper import merge_html_files from abratest.runner import Runner from abratest.runner_dict import RUNNER_DICT from abratest.utils import rmtree class Coordinator: def __init__(self, env_paths_list: list[Path], output_dir: Path, session_id: str, recipes_dir: Path) -> None: # logging out_string = "".join([e.name + "\n" for e in env_paths_list]) out_string += f"output_dir = {output_dir}\n" out_string += f"session_id = {session_id}" logger.info(f"initialize Coordinator instance with\nenv_paths_list =\n{out_string}") self.DIR = DirManager(output_dir=output_dir, session_id=session_id, recipes_dir=recipes_dir) self.ENV = EnvManager(env_paths_list) def setup_test(self) -> None: logger.info("calling setup_test()") self.DIR.create_all_dirs() self.ENV.copy_env_files(self.DIR) def run_test(self) -> None: logger.info("calling run_test()") self.runners: list[Runner] = self._load_runners(self.ENV.env_files) for runner in self.runners: runner.run_setups() for runner in self.runners: runner.run_tests() for runner in self.runners: runner.run_cleanups() logger.info("run_test() finished") def _load_runners(self, env_files: list[EnvFile]) -> list[Runner]: """Creates an instance of the correct Runner class for each given env file""" runners: list[Runner] = [] for env_file in env_files: RunnerClass = RUNNER_DICT[env_file.config["TYPE"]] dependency_classes: list[type[Runner]] = [] for dependency in RunnerClass.dependencies: dependency_classes.append(RUNNER_DICT[dependency]) runner_instance = RunnerClass(dotenv_path=env_file.env_path, DIR=self.DIR) runner_instance._dependency_runners = dependency_classes runners.append(runner_instance) return runners def combine_html(self) -> None: """combines all generated pytest html reports into one""" in_path = str(self.DIR.RECORDS / "html") out_path = str(self.DIR.RECORDS / "full-report.html") title = "combined.html" merge_html_files(in_path, out_path, title) def collect_traces(self): """moves all traces into SESSION/RECORDS dir if tests are rerun and generate another trace, the new trace will get a unique name such as tracename-0 tracename-1 ... """ def get_new_path(root_dir: Path, base_name: str, index=0) -> Path: new_name_alt = base_name + f"-{index}" if not (root_dir / new_name_alt).is_dir(): return root_dir / new_name_alt else: index += 1 return get_new_path(root_dir, base_name, index=index) trace_root_dir = self.DIR.RECORDS / "traces" for f in trace_root_dir.rglob("*/trace.zip"): new_path = get_new_path(self.DIR.RECORDS, f.parent.name) f.parent.rename(new_path) rmtree(trace_root_dir)