import os import sys import logging import json from .tools import to_str from .external import ExternalDataInfo TESTING_OUT_DIR_NAME = "testing_out_stuff" # XXX import from test.const yatest_logger = logging.getLogger("ya.test") class RunMode(object): Run = "run" List = "list" class TestMisconfigurationException(Exception): pass class Ya(object): """ Adds integration with ya, helps in finding dependencies """ def __init__( self, mode=None, source_root=None, build_root=None, dep_roots=None, output_dir=None, test_params=None, context=None, python_path=None, valgrind_path=None, gdb_path=None, data_root=None, ): context_file_path = os.environ.get("YA_TEST_CONTEXT_FILE", None) if context_file_path: with open(context_file_path, 'r') as afile: test_context = json.load(afile) context_runtime = test_context["runtime"] context_internal = test_context.get("internal", {}) context_build = test_context.get("build", {}) else: context_runtime = {} context_internal = {} context_build = {} self._mode = mode self._build_root = to_str(context_runtime.get("build_root", "")) or build_root self._source_root = to_str(context_runtime.get("source_root", "")) or source_root or self._detect_source_root() self._output_dir = to_str(context_runtime.get("output_path", "")) or output_dir or self._detect_output_root() if not self._output_dir: raise Exception("Run ya make -t before running test binary") if not self._source_root: logging.warning("Source root was not set neither determined, use --source-root to set it explicitly") if not self._build_root: if self._source_root: self._build_root = self._source_root else: logging.warning("Build root was not set neither determined, use --build-root to set it explicitly") if data_root: self._data_root = data_root elif self._source_root: self._data_root = os.path.abspath(os.path.join(self._source_root, "..", "arcadia_tests_data")) self._dep_roots = dep_roots self._python_path = to_str(context_runtime.get("python_bin", "")) or python_path self._valgrind_path = valgrind_path self._gdb_path = to_str(context_runtime.get("gdb_bin", "")) or gdb_path self._test_params = {} self._context = {} self._test_item_node_id = None ram_drive_path = to_str(context_runtime.get("ram_drive_path", "")) if ram_drive_path: self._test_params["ram_drive_path"] = ram_drive_path if test_params: self._test_params.update(dict(x.split('=', 1) for x in test_params)) self._test_params.update(context_runtime.get("test_params", {})) self._context["project_path"] = context_runtime.get("project_path") self._context["modulo"] = context_runtime.get("split_count", 1) self._context["modulo_index"] = context_runtime.get("split_index", 0) self._context["work_path"] = context_runtime.get("work_path") self._context["sanitize"] = context_build.get("sanitizer") self._context["ya_trace_path"] = context_internal.get("trace_file") self._env_file = context_internal.get("env_file") if context: self._context.update(context) @property def source_root(self): return self._source_root @property def data_root(self): return self._data_root @property def build_root(self): return self._build_root @property def dep_roots(self): return self._dep_roots @property def output_dir(self): return self._output_dir @property def python_path(self): return self._python_path or sys.executable @property def valgrind_path(self): if not self._valgrind_path: raise ValueError("path to valgrind was not pass correctly, use --valgrind-path to fix it") return self._valgrind_path @property def gdb_path(self): return self._gdb_path @property def env_file(self): return self._env_file def get_binary(self, *path): assert self._build_root, "Build root was not set neither determined, use --build-root to set it explicitly" path = list(path) if os.name == "nt": if not path[-1].endswith(".exe"): path[-1] += ".exe" target_dirs = [self.build_root] # Search for binaries within PATH dirs to be able to get path to the binaries specified by basename for exectests if 'PATH' in os.environ: target_dirs += os.environ['PATH'].split(':') for target_dir in target_dirs: binary_path = os.path.join(target_dir, *path) if os.path.exists(binary_path): yatest_logger.debug("Binary was found by %s", binary_path) return binary_path error_message = "Cannot find binary '{binary}': make sure it was added in the DEPENDS section".format(binary=path) yatest_logger.debug(error_message) if self._mode == RunMode.Run: raise TestMisconfigurationException(error_message) def file(self, path, diff_tool=None, local=False, diff_file_name=None, diff_tool_timeout=None): return ExternalDataInfo.serialize_file(path, diff_tool=diff_tool, local=local, diff_file_name=diff_file_name, diff_tool_timeout=diff_tool_timeout) def get_param(self, key, default=None): return self._test_params.get(key, default) def get_param_dict_copy(self): return dict(self._test_params) def get_context(self, key): return self._context.get(key) def _detect_source_root(self): root = None try: import library.python.find_root # try to determine source root from cwd cwd = os.getcwd() root = library.python.find_root.detect_root(cwd) if not root: # try to determine root pretending we are in the test work dir made from --keep-temps run env_subdir = os.path.join("environment", "arcadia") root = library.python.find_root.detect_root(cwd, detector=lambda p: os.path.exists(os.path.join(p, env_subdir))) except ImportError: logging.warning("Unable to import library.python.find_root") return root def _detect_output_root(self): # if run from kept test working dir if os.path.exists(TESTING_OUT_DIR_NAME): return TESTING_OUT_DIR_NAME # if run from source dir if sys.version_info.major == 3: test_results_dir = "py3test" else: test_results_dir = "pytest" test_results_output_path = os.path.join("test-results", test_results_dir, TESTING_OUT_DIR_NAME) if os.path.exists(test_results_output_path): return test_results_output_path if os.path.exists(os.path.dirname(test_results_output_path)): os.mkdir(test_results_output_path) return test_results_output_path return None def set_test_item_node_id(self, node_id): self._test_item_node_id = node_id def get_test_item_node_id(self): assert self._test_item_node_id return self._test_item_node_id @property def pytest_config(self): if not hasattr(self, "_pytest_config"): import library.python.pytest.plugins.ya as ya_plugin self._pytest_config = ya_plugin.pytest_config return self._pytest_config def set_metric_value(self, name, val): node_id = self.get_test_item_node_id() if node_id not in self.pytest_config.test_metrics: self.pytest_config.test_metrics[node_id] = {} self.pytest_config.test_metrics[node_id][name] = val def get_metric_value(self, name, default=None): res = self.pytest_config.test_metrics.get(self.get_test_item_node_id(), {}).get(name) if res is None: return default return res