# Copyright 2017, The Android Open Source Project # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. #pylint: disable=too-many-lines """ Command Line Translator for atest. """ import json import logging import os import sys import time import atest_error import constants import test_finder_handler import test_mapping TEST_MAPPING = 'TEST_MAPPING' #pylint: disable=no-self-use class CLITranslator(object): """ CLITranslator class contains public method translate() and some private helper methods. The atest tool can call the translate() method with a list of strings, each string referencing a test to run. Translate() will "translate" this list of test strings into a list of build targets and a list of TradeFederation run commands. Translation steps for a test string reference: 1. Narrow down the type of reference the test string could be, i.e. whether it could be referencing a Module, Class, Package, etc. 2. Try to find the test files assuming the test string is one of these types of reference. 3. If test files found, generate Build Targets and the Run Command. """ def __init__(self, module_info=None): """CLITranslator constructor Args: module_info: ModuleInfo class that has cached module-info.json. """ self.mod_info = module_info def _get_test_infos(self, tests, test_mapping_test_details=None): """Return set of TestInfos based on passed in tests. Args: tests: List of strings representing test references. test_mapping_test_details: List of TestDetail for tests configured in TEST_MAPPING files. Returns: Set of TestInfos based on the passed in tests. """ test_infos = set() if not test_mapping_test_details: test_mapping_test_details = [None] * len(tests) for test, tm_test_detail in zip(tests, test_mapping_test_details): test_found = False for finder in test_finder_handler.get_find_methods_for_test( self.mod_info, test): # For tests in TEST_MAPPING, find method is only related to # test name, so the details can be set after test_info object # is created. test_info = finder.find_method(finder.test_finder_instance, test) if test_info: if tm_test_detail: test_info.data[constants.TI_MODULE_ARG] = ( tm_test_detail.options) test_infos.add(test_info) test_found = True break if not test_found: raise atest_error.NoTestFoundError('No test found for: %s' % test) return test_infos def _find_tests_by_test_mapping( self, path='', test_group=constants.TEST_GROUP_PRESUBMIT, file_name=TEST_MAPPING): """Find tests defined in TEST_MAPPING in the given path. Args: path: A string of path in source. Default is set to '', i.e., CWD. test_group: Group of tests to run. Default is set to `presubmit`. file_name: Name of TEST_MAPPING file. Default is set to `TEST_MAPPING`. The argument is added for testing purpose. Returns: A tuple of (tests, all_tests), where, tests is a set of tests (test_mapping.TestDetail) defined in TEST_MAPPING file of the given path, and its parent directories, with matching test_group. all_tests is a dictionary of all tests in TEST_MAPPING files, grouped by test group. """ path = os.path.realpath(path) if path == constants.ANDROID_BUILD_TOP or path == os.sep: return None, None tests = set() all_tests = {} test_mapping_dict = None test_mapping_file = os.path.join(path, file_name) if os.path.exists(test_mapping_file): with open(test_mapping_file) as json_file: test_mapping_dict = json.load(json_file) for test_group_name, test_list in test_mapping_dict.items(): grouped_tests = all_tests.setdefault(test_group_name, set()) grouped_tests.update( [test_mapping.TestDetail(test) for test in test_list]) for test in test_mapping_dict.get(test_group, []): tests.add(test_mapping.TestDetail(test)) parent_dir_tests, parent_dir_all_tests = ( self._find_tests_by_test_mapping( os.path.dirname(path), test_group, file_name)) if parent_dir_tests: tests.update(parent_dir_tests) if parent_dir_all_tests: for test_group_name, test_list in parent_dir_all_tests.items(): grouped_tests = all_tests.setdefault(test_group_name, set()) grouped_tests.update(test_list) if test_group == constants.TEST_GROUP_POSTSUBMIT: tests.update(all_tests.get( constants.TEST_GROUP_PRESUBMIT, set())) return tests, all_tests def _gather_build_targets(self, test_infos): targets = set() for test_info in test_infos: targets |= test_info.build_targets return targets def translate(self, tests): """Translate atest command line into build targets and run commands. Args: tests: A list of strings referencing the tests to run. Returns: A tuple with set of build_target strings and list of TestInfos. """ # Test details from TEST_MAPPING files test_details_list = None if not tests: # Pull out tests from test mapping # TODO(dshi): Support other groups of tests in TEST_MAPPING files, # e.g., postsubmit. test_details, all_test_details = self._find_tests_by_test_mapping() test_details_list = list(test_details) if test_details_list: tests = [detail.name for detail in test_details_list] else: logging.warn( 'No tests of group %s found in TEST_MAPPING at %s or its ' 'parent directories.\nYou might be missing atest arguments,' ' try `atest --help` for more information', constants.TEST_GROUP_PRESUBMIT, os.path.realpath('')) if all_test_details: tests = '' for test_group, test_list in all_test_details.items(): tests += '%s:\n' % test_group for test_detail in sorted(test_list): tests += '\t%s\n' % test_detail logging.warn( 'All available tests in TEST_MAPPING files are:\n%s', tests) sys.exit(constants.EXIT_CODE_TEST_NOT_FOUND) logging.info('Finding tests: %s', tests) if test_details_list: details = '\n'.join([str(detail) for detail in test_details_list]) logging.info('Test details:\n%s', details) start = time.time() test_infos = self._get_test_infos(tests, test_details_list) end = time.time() logging.debug('Found tests in %ss', end - start) for test_info in test_infos: logging.debug('%s\n', test_info) build_targets = self._gather_build_targets(test_infos) return build_targets, test_infos