You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
186 lines
7.0 KiB
186 lines
7.0 KiB
#!/usr/bin/env python3
|
|
#
|
|
# Copyright 2019 - 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.
|
|
|
|
"""Project config class."""
|
|
|
|
import os
|
|
|
|
from aidegen import constant
|
|
from aidegen.lib import common_util
|
|
from aidegen.lib import errors
|
|
|
|
SKIP_BUILD_INFO = ('If you are sure the related modules and dependencies have '
|
|
'been already built, please try to use command {} to skip '
|
|
'the building process.')
|
|
_SKIP_BUILD_CMD = 'aidegen {} -s'
|
|
_SKIP_BUILD_WARN = (
|
|
'You choose "--skip-build". Skip building jar and module might increase '
|
|
'the risk of the absence of some jar or R/AIDL/logtags java files and '
|
|
'cause the red lines to appear in IDE tool.')
|
|
_INSTANCE_NOT_EXIST_ERROR = ('The instance of {} does not exist. Please '
|
|
'initialize it before using.')
|
|
|
|
|
|
class ProjectConfig():
|
|
"""A singleton class manages AIDEGen's configurations.
|
|
|
|
ProjectConfig is a singleton class that can be accessed in other modules.
|
|
|
|
Usage:
|
|
1. Main module should do it once by instantiating a ProjectConfig with
|
|
users' input arguments and calling init_environment().
|
|
args = aidegen_main.main(sys.argv[1:])
|
|
project_config.ProjectConfig(args).init_environment()
|
|
2. All others can get the ProjectConfig instance by calling
|
|
get_instance().
|
|
project_config.ProjectConfig.get_instance()
|
|
|
|
Class attributes:
|
|
_instance: A singleton instance of ProjectConfig.
|
|
|
|
Attributes:
|
|
ide_name: The IDE name which users prefer to launch.
|
|
is_launch_ide: A boolean for launching IDE in the end of AIDEGen.
|
|
depth: The depth of module referenced by source.
|
|
full_repo: A boolean decides import whole Android source repo.
|
|
is_skip_build: A boolean decides skipping building jars or modules.
|
|
targets: A string list with Android module names or paths.
|
|
verbose: A boolean. If true, display DEBUG level logs.
|
|
ide_installed_path: A string of IDE installed path.
|
|
config_reset: A boolean if true to reset all saved configurations.
|
|
atest_module_info: A ModuleInfo instance.
|
|
language: The programming language users prefer to deal with.
|
|
"""
|
|
|
|
_instance = None
|
|
|
|
def __init__(self, args):
|
|
"""ProjectConfig initialize.
|
|
|
|
Args:
|
|
An argparse.Namespace object holds parsed args.
|
|
"""
|
|
self.language = constant.LANGUAGE_NAME_DICT[args.language[0]]
|
|
self.ide_name = constant.IDE_NAME_DICT[args.ide[0]]
|
|
self.is_launch_ide = not args.no_launch
|
|
self.depth = args.depth
|
|
self.full_repo = args.android_tree
|
|
self.is_skip_build = args.skip_build
|
|
self.targets = args.targets.copy()
|
|
self.verbose = args.verbose
|
|
self.ide_installed_path = args.ide_installed_path
|
|
self.config_reset = args.config_reset
|
|
self.exclude_paths = args.exclude_paths
|
|
self.atest_module_info = None
|
|
ProjectConfig._instance = self
|
|
|
|
def init_environment(self):
|
|
"""Initialize the environment settings for the whole project."""
|
|
self._show_skip_build_msg()
|
|
# TODO(b/159078170): Avoid CLion IDE case for now, we should avoid
|
|
# Android Studio's native project's case in the future.
|
|
targets = self.targets if self.language == constant.JAVA else None
|
|
self.atest_module_info = common_util.get_atest_module_info(targets)
|
|
self.exclude_paths = _transform_exclusive_paths(
|
|
self.atest_module_info, self.exclude_paths)
|
|
self.targets = _check_whole_android_tree(self.targets, self.full_repo)
|
|
self.full_repo = (self.targets[0] == constant.WHOLE_ANDROID_TREE_TARGET)
|
|
|
|
def _show_skip_build_msg(self):
|
|
"""Display different messages if users skip building targets or not."""
|
|
if self.is_skip_build:
|
|
print('\n{} {}\n'.format(
|
|
common_util.COLORED_INFO('Warning:'), _SKIP_BUILD_WARN))
|
|
else:
|
|
msg = SKIP_BUILD_INFO.format(
|
|
common_util.COLORED_INFO(
|
|
_SKIP_BUILD_CMD.format(' '.join(self.targets))))
|
|
print('\n{} {}\n'.format(common_util.COLORED_INFO('INFO:'), msg))
|
|
|
|
@classmethod
|
|
def get_instance(cls):
|
|
"""Get a singleton's instance.
|
|
|
|
Returns:
|
|
A singleton instance of ProjectConfig.
|
|
|
|
Raises:
|
|
An exception of errors.InstanceNotExistError if users didn't
|
|
instantiate a ProjectConfig object before calling this method.
|
|
"""
|
|
if not cls._instance:
|
|
raise errors.InstanceNotExistError(
|
|
_INSTANCE_NOT_EXIST_ERROR.format(str(cls)))
|
|
return cls._instance
|
|
|
|
|
|
def _check_whole_android_tree(targets, android_tree):
|
|
"""Check if it's a building project file for the whole Android tree.
|
|
|
|
The rules:
|
|
1. If users command aidegen under Android root, e.g.,
|
|
root$ aidegen
|
|
that implies users would like to launch the whole Android tree, AIDEGen
|
|
should set the flag android_tree True.
|
|
2. If android_tree is True, add whole Android tree to the project.
|
|
|
|
Args:
|
|
targets: A list of targets to be imported.
|
|
android_tree: A boolean, True if it's a whole Android tree case,
|
|
otherwise False.
|
|
|
|
Returns:
|
|
A list of targets to be built.
|
|
"""
|
|
if common_util.is_android_root(os.getcwd()) and targets == ['']:
|
|
return [constant.WHOLE_ANDROID_TREE_TARGET]
|
|
new_targets = targets.copy()
|
|
if android_tree:
|
|
new_targets.insert(0, constant.WHOLE_ANDROID_TREE_TARGET)
|
|
return new_targets
|
|
|
|
|
|
def is_whole_android_tree(targets, android_tree):
|
|
"""Checks is AIDEGen going to process whole android tree.
|
|
|
|
Args:
|
|
targets: A list of targets to be imported.
|
|
android_tree: A boolean, True if it's a whole Android tree case,
|
|
otherwise False.
|
|
Returns:
|
|
A boolean, True when user is going to import whole Android tree.
|
|
"""
|
|
return (android_tree or
|
|
(common_util.is_android_root(os.getcwd()) and targets == ['']))
|
|
|
|
|
|
def _transform_exclusive_paths(atest_module_info, exclude_paths):
|
|
"""Transforms exclusive paths to relative paths.
|
|
|
|
Args:
|
|
exclude_paths: A list of strings of exclusive paths.
|
|
|
|
Returns:
|
|
A list of relative paths.
|
|
"""
|
|
if not exclude_paths:
|
|
return None
|
|
excludes = []
|
|
for path in exclude_paths:
|
|
exclude_path, _ = common_util.get_related_paths(atest_module_info, path)
|
|
excludes.append(exclude_path)
|
|
return excludes
|