| # Copyright (C) 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. |
| """Helper functions for updaters.""" |
| |
| import os |
| import re |
| import subprocess |
| import sys |
| |
| |
| def create_updater(metadata, proj_path, updaters): |
| """Creates corresponding updater object for a project. |
| |
| Args: |
| metadata: Parsed proto for METADATA file. |
| proj_path: Absolute path for the project. |
| |
| Returns: |
| An updater object. |
| |
| Raises: |
| ValueError: Occurred when there's no updater for all urls. |
| """ |
| for url in metadata.third_party.url: |
| for updater_cls in updaters: |
| updater = updater_cls(proj_path, url, metadata.third_party.version) |
| if updater.is_supported_url(): |
| return updater |
| |
| raise ValueError('No supported URL.') |
| |
| |
| def replace_package(source_dir, target_dir) -> None: |
| """Invokes a shell script to prepare and update a project. |
| |
| Args: |
| source_dir: Path to the new downloaded and extracted package. |
| target_dir: The path to the project in Android source tree. |
| """ |
| |
| print('Updating {} using {}.'.format(target_dir, source_dir)) |
| script_path = os.path.join(os.path.dirname(sys.argv[0]), |
| 'update_package.sh') |
| subprocess.check_call(['bash', script_path, source_dir, target_dir]) |
| |
| |
| VERSION_SPLITTER_PATTERN = r'[\.\-_]' |
| VERSION_PATTERN = (r'^(?P<prefix>[^\d]*)' + r'(?P<version>\d+(' + |
| VERSION_SPLITTER_PATTERN + r'\d+)*)' + r'(?P<suffix>.*)$') |
| VERSION_RE = re.compile(VERSION_PATTERN) |
| VERSION_SPLITTER_RE = re.compile(VERSION_SPLITTER_PATTERN) |
| |
| |
| def _parse_version(version): |
| match = VERSION_RE.match(version) |
| if match is None: |
| raise ValueError('Invalid version.') |
| try: |
| prefix, version, suffix = match.group('prefix', 'version', 'suffix') |
| version = [int(v) for v in VERSION_SPLITTER_RE.split(version)] |
| return (version, prefix, suffix) |
| except IndexError: |
| raise ValueError('Invalid version.') |
| |
| |
| def _match_and_get_version(old_ver, version): |
| try: |
| new_ver = _parse_version(version) |
| except ValueError: |
| return [] |
| |
| right_format = (new_ver[1:] == old_ver[1:]) |
| right_length = len(new_ver[0]) == len(old_ver[0]) |
| |
| return [right_format, right_length, new_ver[0]] |
| |
| |
| def get_latest_version(current_version, version_list): |
| """Gets the latest version name from a list of versions. |
| |
| The new version must have the same prefix and suffix with old version. |
| If no matched version is newer, current version name will be returned. |
| """ |
| parsed_current_ver = _parse_version(current_version) |
| |
| latest = max( |
| version_list, |
| key=lambda ver: _match_and_get_version(parsed_current_ver, ver), |
| default=[]) |
| if not latest: |
| raise ValueError('No matching version.') |
| return latest |