| # Copyright 2018, 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. |
| |
| """ |
| Classes for test mapping related objects |
| """ |
| |
| |
| import copy |
| import fnmatch |
| import os |
| import re |
| |
| import atest_utils |
| import constants |
| |
| TEST_MAPPING = 'TEST_MAPPING' |
| |
| |
| class TestDetail(object): |
| """Stores the test details set in a TEST_MAPPING file.""" |
| |
| def __init__(self, details): |
| """TestDetail constructor |
| |
| Parse test detail from a dictionary, e.g., |
| { |
| "name": "SettingsUnitTests", |
| "host": true, |
| "options": [ |
| { |
| "instrumentation-arg": |
| "annotation=android.platform.test.annotations.Presubmit" |
| }, |
| "file_patterns": ["(/|^)Window[^/]*\\.java", |
| "(/|^)Activity[^/]*\\.java"] |
| } |
| |
| Args: |
| details: A dictionary of test detail. |
| """ |
| self.name = details['name'] |
| self.options = [] |
| # True if the test should run on host and require no device. |
| self.host = details.get('host', False) |
| assert isinstance(self.host, bool), 'host can only have boolean value.' |
| options = details.get('options', []) |
| for option in options: |
| assert len(option) == 1, 'Each option can only have one key.' |
| self.options.append(copy.deepcopy(option).popitem()) |
| self.options.sort(key=lambda o: o[0]) |
| self.file_patterns = details.get('file_patterns', []) |
| |
| def __str__(self): |
| """String value of the TestDetail object.""" |
| host_info = (', runs on host without device required.' if self.host |
| else '') |
| if not self.options: |
| return self.name + host_info |
| options = '' |
| for option in self.options: |
| options += '%s: %s, ' % option |
| |
| return '%s (%s)%s' % (self.name, options.strip(', '), host_info) |
| |
| def __hash__(self): |
| """Get the hash of TestDetail based on the details""" |
| return hash(str(self)) |
| |
| def __eq__(self, other): |
| return str(self) == str(other) |
| |
| |
| class Import(object): |
| """Store test mapping import details.""" |
| |
| def __init__(self, test_mapping_file, details): |
| """Import constructor |
| |
| Parse import details from a dictionary, e.g., |
| { |
| "path": "..\folder1" |
| } |
| in which, project is the name of the project, by default it's the |
| current project of the containing TEST_MAPPING file. |
| |
| Args: |
| test_mapping_file: Path to the TEST_MAPPING file that contains the |
| import. |
| details: A dictionary of details about importing another |
| TEST_MAPPING file. |
| """ |
| self.test_mapping_file = test_mapping_file |
| self.path = details['path'] |
| |
| def __str__(self): |
| """String value of the Import object.""" |
| return 'Source: %s, path: %s' % (self.test_mapping_file, self.path) |
| |
| def get_path(self): |
| """Get the path to TEST_MAPPING import directory.""" |
| path = os.path.realpath(os.path.join( |
| os.path.dirname(self.test_mapping_file), self.path)) |
| if os.path.exists(path): |
| return path |
| root_dir = os.environ.get(constants.ANDROID_BUILD_TOP, os.sep) |
| path = os.path.realpath(os.path.join(root_dir, self.path)) |
| if os.path.exists(path): |
| return path |
| # The import path can't be located. |
| return None |
| |
| |
| def is_match_file_patterns(test_mapping_file, test_detail): |
| """Check if the changed file names match the regex pattern defined in |
| file_patterns of TEST_MAPPING files. |
| |
| Args: |
| test_mapping_file: Path to a TEST_MAPPING file. |
| test_detail: A TestDetail object. |
| |
| Returns: |
| True if the test's file_patterns setting is not set or contains a |
| pattern matches any of the modified files. |
| """ |
| # Only check if the altered files are located in the same or sub directory |
| # of the TEST_MAPPING file. Extract the relative path of the modified files |
| # which match file patterns. |
| file_patterns = test_detail.get('file_patterns', []) |
| if not file_patterns: |
| return True |
| test_mapping_dir = os.path.dirname(test_mapping_file) |
| modified_files = atest_utils.get_modified_files(test_mapping_dir) |
| if not modified_files: |
| return False |
| modified_files_in_source_dir = [ |
| os.path.relpath(filepath, test_mapping_dir) |
| for filepath in fnmatch.filter(modified_files, |
| os.path.join(test_mapping_dir, '*')) |
| ] |
| for modified_file in modified_files_in_source_dir: |
| # Force to run the test if it's in a TEST_MAPPING file included in the |
| # changesets. |
| if modified_file == constants.TEST_MAPPING: |
| return True |
| for pattern in file_patterns: |
| if re.search(pattern, modified_file): |
| return True |
| return False |