# 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 in updaters:
|
try:
|
return updater(url, proj_path, metadata)
|
except ValueError:
|
pass
|
|
raise ValueError('No supported URL.')
|
|
|
def replace_package(source_dir, target_dir):
|
"""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:
|
return match.group('prefix', 'version', 'suffix')
|
except IndexError:
|
raise ValueError('Invalid version.')
|
|
|
def _match_and_get_version(prefix, suffix, version):
|
try:
|
version_prefix, version, version_suffix = _parse_version(version)
|
except ValueError:
|
return []
|
|
right_format = (version_prefix == prefix and version_suffix == suffix)
|
|
return [right_format] + [int(v) for v in VERSION_SPLITTER_RE.split(version)]
|
|
|
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.
|
"""
|
prefix, _, suffix = _parse_version(current_version)
|
|
latest = max(version_list,
|
key=lambda ver: _match_and_get_version(
|
prefix, suffix, ver),
|
default=[])
|
if not latest:
|
raise ValueError('No matching version.')
|
return latest
|