#!/usr/bin/env python3 """Generate burn in tests to run on certain build variants.""" from collections import namedtuple import logging import os import sys from typing import Any, Dict, List import click from git import Repo from shrub.v2 import ShrubProject, BuildVariant, ExistingTask from evergreen.api import RetryingEvergreenApi, EvergreenApi # Get relative imports to work when the package is not installed on the PYTHONPATH. if __name__ == "__main__" and __package__ is None: sys.path.append(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) # pylint: disable=wrong-import-position from buildscripts.evergreen_burn_in_tests import GenerateBurnInExecutor, GenerateConfig, \ EvergreenFileChangeDetector from buildscripts.util.fileops import write_file_to_dir import buildscripts.util.read_config as read_config from buildscripts.ciconfig import evergreen from buildscripts.ciconfig.evergreen import EvergreenProjectConfig, Variant from buildscripts.burn_in_tests import create_tests_by_task, RepeatConfig, DEFAULT_REPO_LOCATIONS # pylint: enable=wrong-import-position EXTERNAL_LOGGERS = { "evergreen", "git", "urllib3", } CONFIG_DIRECTORY = "generated_burn_in_tags_config" CONFIG_FILE = "burn_in_tags_gen.json" EVERGREEN_FILE = "etc/evergreen.yml" EVG_CONFIG_FILE = ".evergreen.yml" COMPILE_TASK = "compile_and_archive_dist_test_TG" # Burn in tags requires running on RHEL80 currently. COMPILE_TASK_DISTRO = "rhel80-large" TASK_ID_EXPANSION = "task_id" ConfigOptions = namedtuple("ConfigOptions", [ "build_variant", "run_build_variant", "base_commit", "max_revisions", "branch", "check_evergreen", "distro", "repeat_tests_secs", "repeat_tests_min", "repeat_tests_max", "project", ]) def _get_config_options(expansions_file_data, build_variant, run_build_variant): """ Get the configuration to use. :param expansions_file_data: Config data file to use. :param build_variant: The buildvariant the current patch should be compared against to figure out which tests have changed. :param run_build_variant: The buildvariant the generated task should be run on. :return: ConfigOptions for the generated task to use. """ base_commit = expansions_file_data["revision"] max_revisions = int(expansions_file_data["max_revisions"]) branch = expansions_file_data["branch_name"] is_patch = expansions_file_data.get("is_patch", False) check_evergreen = is_patch != "true" distro = expansions_file_data["distro_id"] repeat_tests_min = int(expansions_file_data["repeat_tests_min"]) repeat_tests_max = int(expansions_file_data["repeat_tests_max"]) repeat_tests_secs = int(expansions_file_data["repeat_tests_secs"]) project = expansions_file_data["project"] return ConfigOptions(build_variant, run_build_variant, base_commit, max_revisions, branch, check_evergreen, distro, repeat_tests_secs, repeat_tests_min, repeat_tests_max, project) def _create_evg_build_variant_map(expansions_file_data, evergreen_conf): """ Generate relationship of base buildvariant to generated buildvariant. :param expansions_file_data: Config data file to use. :param evergreen_conf: Evergreen configuration. :return: Map of base buildvariants to their generated buildvariants. """ burn_in_tags_gen_variant = expansions_file_data["build_variant"] burn_in_tags_gen_variant_config = evergreen_conf.get_variant(burn_in_tags_gen_variant) burn_in_tag_build_variants = burn_in_tags_gen_variant_config.expansions.get( "burn_in_tag_buildvariants") if burn_in_tag_build_variants: return { base_variant: f"{base_variant}-required" for base_variant in burn_in_tag_build_variants.split(" ") } return {} def _generate_evg_build_variant( source_build_variant: Variant, run_build_variant: str, bypass_build_variant: str, ) -> BuildVariant: """ Generate a shrub build variant for the given run build variant. :param source_build_variant: The build variant to base configuration on. :param run_build_variant: The build variant to generate. :param bypass_build_variant: The build variant to get compile artifacts from. :return: Shrub build variant configuration. """ display_name = f"! {source_build_variant.display_name}" run_on = source_build_variant.run_on modules = source_build_variant.modules expansions = source_build_variant.expansions expansions["burn_in_bypass"] = bypass_build_variant build_variant = BuildVariant(run_build_variant, display_name, expansions=expansions, modules=modules, run_on=run_on) build_variant.add_existing_task(ExistingTask(COMPILE_TASK), distros=[COMPILE_TASK_DISTRO]) return build_variant # pylint: disable=too-many-arguments,too-many-locals def _generate_evg_tasks(evergreen_api: EvergreenApi, shrub_project: ShrubProject, task_expansions: Dict[str, Any], build_variant_map: Dict[str, str], repos: List[Repo], evg_conf: EvergreenProjectConfig, install_dir: str) -> None: """ Generate burn in tests tasks for a given shrub config and group of build variants. :param evergreen_api: Evergreen.py object. :param shrub_project: Shrub config object that the build variants will be built upon. :param task_expansions: Dictionary of expansions for the running task. :param build_variant_map: Map of base buildvariants to their generated buildvariant. :param repos: Git repositories. """ for build_variant, run_build_variant in build_variant_map.items(): config_options = _get_config_options(task_expansions, build_variant, run_build_variant) task_id = task_expansions[TASK_ID_EXPANSION] change_detector = EvergreenFileChangeDetector(task_id, evergreen_api) changed_tests = change_detector.find_changed_tests(repos) tests_by_task = create_tests_by_task(build_variant, evg_conf, changed_tests, install_dir) if tests_by_task: shrub_build_variant = _generate_evg_build_variant( evg_conf.get_variant(build_variant), run_build_variant, task_expansions["build_variant"]) gen_config = GenerateConfig(build_variant, config_options.project, run_build_variant, config_options.distro, include_gen_task=False).validate(evg_conf) repeat_config = RepeatConfig(repeat_tests_min=config_options.repeat_tests_min, repeat_tests_max=config_options.repeat_tests_max, repeat_tests_secs=config_options.repeat_tests_secs) burn_in_generator = GenerateBurnInExecutor(gen_config, repeat_config, evergreen_api) burn_in_generator.add_config_for_build_variant(shrub_build_variant, tests_by_task) shrub_project.add_build_variant(shrub_build_variant) def burn_in(task_expansions: Dict[str, Any], evg_conf: EvergreenProjectConfig, evergreen_api: RetryingEvergreenApi, repos: List[Repo], install_dir: str): """ Execute main program. :param task_expansions: Dictionary of expansions for the running task. :param evg_conf: Evergreen configuration. :param evergreen_api: Evergreen.py object. :param repos: Git repositories. :param install_dir: path to bin directory of a testable installation """ shrub_project = ShrubProject.empty() build_variant_map = _create_evg_build_variant_map(task_expansions, evg_conf) _generate_evg_tasks(evergreen_api, shrub_project, task_expansions, build_variant_map, repos, evg_conf, install_dir) write_file_to_dir(CONFIG_DIRECTORY, CONFIG_FILE, shrub_project.json()) def _configure_logging(verbose: bool): """ Configure logging for the application. :param verbose: If True set log level to DEBUG. """ level = logging.DEBUG if verbose else logging.INFO logging.basicConfig( format="[%(asctime)s - %(name)s - %(levelname)s] %(message)s", level=level, stream=sys.stdout, ) for log_name in EXTERNAL_LOGGERS: logging.getLogger(log_name).setLevel(logging.WARNING) @click.command() @click.option("--expansion-file", "expansion_file", required=True, help="Location of expansions file generated by evergreen.") @click.option("--verbose", is_flag=True) @click.option("--install-dir", "install_dir", required=True, help="Path to bin directory of a testable installation") def main(expansion_file: str, verbose: bool, install_dir: str): """ Run new or changed tests in repeated mode to validate their stability. burn_in_tags detects jstests that are new or changed since the last git command and then runs those tests in a loop to validate their reliability. \f :param expansion_file: The expansion file containing the configuration params. """ _configure_logging(verbose) evg_api = RetryingEvergreenApi.get_api(config_file=EVG_CONFIG_FILE) repos = [Repo(x) for x in DEFAULT_REPO_LOCATIONS if os.path.isdir(x)] expansions_file_data = read_config.read_config_file(expansion_file) evg_conf = evergreen.parse_evergreen_file(EVERGREEN_FILE) burn_in(expansions_file_data, evg_conf, evg_api, repos, install_dir) if __name__ == "__main__": main() # pylint: disable=no-value-for-parameter