diff options
author | 2024-08-13 17:16:33 -0700 | |
---|---|---|
committer | 2024-08-27 11:15:25 -0700 | |
commit | b9c54644a213c36cd175f575545e80f7972bc25a (patch) | |
tree | 09a28c93663e8dbe581af05d3e252cd7f2aeadd2 /ci/optimized_targets_test.py | |
parent | b2c2fe8856de5a458f2756727bf4aa32268470a4 (diff) |
Implement get_build_targets_impl in GeneralTestsOptimizer
Implement functionality in GeneralTestsOptimizer to find what targets to
build.
This logic is fairly complex and involves checking for test
configs that download general-tests.zip. Then the configs are checked to
see if they're proper test mapping tests (if they use the
'test-mapping-test-group' option). If they are, then TEST_MAPPING
modules are scanned to see if the list of changed files would cause any
test mapping modules to run. The tests are then further filtered by
test-mapping-test-groups used in the test configs.
In case that a test uses general-tests.zip but does not specify
'test-mapping-test-group' then all bets are off and general-tests.zip
is built in its entirety.
package_outputs is still unimplemented so this will need to be
implemented before the optimization can be enabled.
Test: atest build_test_suites_test && atest optimized_targets_test
Bug: 358215235
Change-Id: I6a7eebfd1b06b380799292eb2019ac17c9af5367
Diffstat (limited to 'ci/optimized_targets_test.py')
-rw-r--r-- | ci/optimized_targets_test.py | 206 |
1 files changed, 206 insertions, 0 deletions
diff --git a/ci/optimized_targets_test.py b/ci/optimized_targets_test.py new file mode 100644 index 0000000000..919c193955 --- /dev/null +++ b/ci/optimized_targets_test.py @@ -0,0 +1,206 @@ +# Copyright 2024, 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. + +"""Tests for optimized_targets.py""" + +import json +import logging +import os +import pathlib +import re +import unittest +from unittest import mock +import optimized_targets +from build_context import BuildContext +from pyfakefs import fake_filesystem_unittest + + +class GeneralTestsOptimizerTest(fake_filesystem_unittest.TestCase): + + def setUp(self): + self.setUpPyfakefs() + + os_environ_patcher = mock.patch.dict('os.environ', {}) + self.addCleanup(os_environ_patcher.stop) + self.mock_os_environ = os_environ_patcher.start() + + self._setup_working_build_env() + self._write_change_info_file() + test_mapping_dir = pathlib.Path('/project/path/file/path') + test_mapping_dir.mkdir(parents=True) + self._write_test_mapping_file() + + def _setup_working_build_env(self): + self.change_info_file = pathlib.Path('/tmp/change_info') + + self.mock_os_environ.update({ + 'CHANGE_INFO': str(self.change_info_file), + }) + + def test_general_tests_optimized(self): + optimizer = self._create_general_tests_optimizer() + + build_targets = optimizer.get_build_targets() + + expected_build_targets = set( + optimized_targets.GeneralTestsOptimizer._REQUIRED_MODULES + ) + expected_build_targets.add('test_mapping_module') + + self.assertSetEqual(build_targets, expected_build_targets) + + def test_no_change_info_no_optimization(self): + del os.environ['CHANGE_INFO'] + + optimizer = self._create_general_tests_optimizer() + + build_targets = optimizer.get_build_targets() + + self.assertSetEqual(build_targets, {'general-tests'}) + + def test_mapping_groups_unused_module_not_built(self): + test_context = self._create_test_context() + test_context['testInfos'][0]['extraOptions'] = [ + { + 'key': 'additional-files-filter', + 'values': ['general-tests.zip'], + }, + { + 'key': 'test-mapping-test-group', + 'values': ['unused-test-mapping-group'], + }, + ] + optimizer = self._create_general_tests_optimizer( + build_context=self._create_build_context(test_context=test_context) + ) + + build_targets = optimizer.get_build_targets() + + expected_build_targets = set( + optimized_targets.GeneralTestsOptimizer._REQUIRED_MODULES + ) + self.assertSetEqual(build_targets, expected_build_targets) + + def test_general_tests_used_by_non_test_mapping_test_no_optimization(self): + test_context = self._create_test_context() + test_context['testInfos'][0]['extraOptions'] = [{ + 'key': 'additional-files-filter', + 'values': ['general-tests.zip'], + }] + optimizer = self._create_general_tests_optimizer( + build_context=self._create_build_context(test_context=test_context) + ) + + build_targets = optimizer.get_build_targets() + + self.assertSetEqual(build_targets, {'general-tests'}) + + def test_malformed_change_info_raises(self): + with open(self.change_info_file, 'w') as f: + f.write('not change info') + + optimizer = self._create_general_tests_optimizer() + + with self.assertRaises(json.decoder.JSONDecodeError): + build_targets = optimizer.get_build_targets() + + def test_malformed_test_mapping_raises(self): + with open('/project/path/file/path/TEST_MAPPING', 'w') as f: + f.write('not test mapping') + + optimizer = self._create_general_tests_optimizer() + + with self.assertRaises(json.decoder.JSONDecodeError): + build_targets = optimizer.get_build_targets() + + def _write_change_info_file(self): + change_info_contents = { + 'changes': [{ + 'projectPath': '/project/path', + 'revisions': [{ + 'fileInfos': [{ + 'path': 'file/path/file_name', + }], + }], + }] + } + + with open(self.change_info_file, 'w') as f: + json.dump(change_info_contents, f) + + def _write_test_mapping_file(self): + test_mapping_contents = { + 'test-mapping-group': [ + { + 'name': 'test_mapping_module', + }, + ], + } + + with open('/project/path/file/path/TEST_MAPPING', 'w') as f: + json.dump(test_mapping_contents, f) + + def _create_general_tests_optimizer( + self, build_context: BuildContext = None + ): + if not build_context: + build_context = self._create_build_context() + return optimized_targets.GeneralTestsOptimizer( + 'general-tests', build_context, None + ) + + def _create_build_context( + self, + general_tests_optimized: bool = True, + test_context: dict[str, any] = None, + ) -> BuildContext: + if not test_context: + test_context = self._create_test_context() + build_context_dict = {} + build_context_dict['enabledBuildFeatures'] = [{'name': 'optimized_build'}] + if general_tests_optimized: + build_context_dict['enabledBuildFeatures'].append({'name': 'general_tests_optimized'}) + build_context_dict['testContext'] = test_context + return BuildContext(build_context_dict) + + def _create_test_context(self): + return { + 'testInfos': [ + { + 'name': 'atp_test', + 'target': 'test_target', + 'branch': 'branch', + 'extraOptions': [ + { + 'key': 'additional-files-filter', + 'values': ['general-tests.zip'], + }, + { + 'key': 'test-mapping-test-group', + 'values': ['test-mapping-group'], + }, + ], + 'command': '/tf/command', + 'extraBuildTargets': [ + 'extra_build_target', + ], + }, + ], + } + + +if __name__ == '__main__': + # Setup logging to be silent so unit tests can pass through TF. + logging.disable(logging.ERROR) + unittest.main() |