blob: 92ddcd66643821165896900ebcb26debdab30508 [file] [log] [blame] [edit]
# 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.
"""Module to check updates from Git upstream."""
import base_updater
import fileutils
import git_utils
# pylint: disable=import-error
from manifest import Manifest
class GitUpdater(base_updater.Updater):
"""Updater for Git upstream."""
UPSTREAM_REMOTE_NAME: str = "update_origin"
def is_supported_url(self) -> bool:
return git_utils.is_valid_url(self._proj_path, self._old_identifier.value)
@staticmethod
def _is_likely_android_remote(url: str) -> bool:
"""Returns True if the URL is likely to be the project's Android remote."""
# There isn't a strict rule for finding the correct remote for
# upstream-master/main, so we have to guess. Be careful to filter out
# things that look almost right but aren't. Here's an example of a
# project that has a lot of false positives:
#
# aosp /usr/local/google/home/danalbert/src/mirrors/android/refs/aosp/toolchain/rr.git (fetch)
# aosp persistent-https://android.git.corp.google.com/toolchain/rr (push)
# origin https://github.com/DanAlbert/rr.git (fetch)
# origin https://github.com/DanAlbert/rr.git (push)
# unmirrored persistent-https://android.git.corp.google.com/toolchain/rr (fetch)
# unmirrored persistent-https://android.git.corp.google.com/toolchain/rr (push)
# update_origin https://github.com/rr-debugger/rr (fetch)
# update_origin https://github.com/rr-debugger/rr (push)
# upstream https://github.com/rr-debugger/rr.git (fetch)
# upstream https://github.com/rr-debugger/rr.git (push)
#
# unmirrored is the correct remote here. It's not a local path,
# and contains either /platform/external/ or /toolchain/ (the two
# common roots for third- party Android imports).
if '://' not in url:
# Skip anything that's likely a local GoB mirror.
return False
if '/platform/external/' in url:
return True
if '/toolchain/' in url:
return True
return False
def setup_remote(self) -> None:
remotes = git_utils.list_remotes(self._proj_path)
current_remote_url = None
android_remote_name: str | None = None
for name, url in remotes.items():
if name == self.UPSTREAM_REMOTE_NAME:
current_remote_url = url
if self._is_likely_android_remote(url):
android_remote_name = name
if android_remote_name is None:
remotes_formatted = "\n".join(f"{k} {v}" for k, v in remotes.items())
raise RuntimeError(
f"Could not determine android remote for {self._proj_path}. Tried:\n"
f"{remotes_formatted}")
if current_remote_url is not None and current_remote_url != self._old_identifier.value:
git_utils.remove_remote(self._proj_path, self.UPSTREAM_REMOTE_NAME)
current_remote_url = None
if current_remote_url is None:
git_utils.add_remote(self._proj_path, self.UPSTREAM_REMOTE_NAME,
self._old_identifier.value)
branch = git_utils.detect_default_branch(self._proj_path,
self.UPSTREAM_REMOTE_NAME)
git_utils.fetch(self._proj_path, self.UPSTREAM_REMOTE_NAME, branch)
git_utils.fetch(self._proj_path, android_remote_name,
self._determine_android_fetch_ref())
def check(self) -> None:
"""Checks upstream and returns whether a new version is available."""
self.setup_remote()
possible_alternative_new_ver: str | None = None
if git_utils.is_commit(self._old_identifier.version):
# Update to remote head.
self._new_identifier.version = self.current_head_of_upstream_default_branch()
# Some libraries don't have a tag. We only populate
# _alternative_new_ver if there is a tag newer than _old_ver.
# Checks if there is a tag newer than AOSP's SHA
if (tag := self.latest_tag_of_upstream_default_branch()) is not None:
possible_alternative_new_ver = tag
else:
# Update to the latest version tag.
tag = self.latest_tag_of_upstream_default_branch()
if tag is None:
project = fileutils.canonicalize_project_path(self.project_path)
raise RuntimeError(
f"{project} is currently tracking upstream tags but no tags were "
"found in the upstream repository"
)
self._new_identifier.version = tag
# Checks if there is a SHA newer than AOSP's tag
possible_alternative_new_ver = self.current_head_of_upstream_default_branch()
if possible_alternative_new_ver is not None and git_utils.is_ancestor(
self._proj_path,
self._old_identifier.version,
possible_alternative_new_ver
):
self._alternative_new_ver = possible_alternative_new_ver
def latest_tag_of_upstream_default_branch(self) -> str | None:
branch = git_utils.detect_default_branch(self._proj_path,
self.UPSTREAM_REMOTE_NAME)
return git_utils.get_most_recent_tag(
self._proj_path, self.UPSTREAM_REMOTE_NAME + '/' + branch)
def current_head_of_upstream_default_branch(self) -> str:
branch = git_utils.detect_default_branch(self._proj_path,
self.UPSTREAM_REMOTE_NAME)
return git_utils.get_sha_for_branch(
self._proj_path, self.UPSTREAM_REMOTE_NAME + '/' + branch)
def update(self) -> None:
"""Updates the package.
Has to call check() before this function.
"""
print(f"Running `git merge {self._new_identifier.version}`...")
git_utils.merge(self._proj_path, self._new_identifier.version)
def _determine_android_fetch_ref(self) -> str:
"""Returns the ref that should be fetched from the android remote."""
# It isn't particularly efficient to reparse the tree for every
# project, but we don't guarantee that all paths passed to updater.sh
# are actually in the same tree so it wouldn't necessarily be correct
# to do this once at the top level. This isn't the slow part anyway,
# so it can be dealt with if that ever changes.
root = fileutils.find_tree_containing(self._proj_path)
manifest = Manifest.for_tree(root)
manifest_path = str(self._proj_path.relative_to(root))
try:
project = manifest.project_with_path(manifest_path)
except KeyError as ex:
raise RuntimeError(
f"Did not find {manifest_path} in {manifest.path} (tree root is {root})"
) from ex
return project.revision