(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 1 | # Copyright (C) 2018 The Android Open Source Project |
| 2 | # |
| 3 | # Licensed under the Apache License, Version 2.0 (the "License"); |
| 4 | # you may not use this file except in compliance with the License. |
| 5 | # You may obtain a copy of the License at |
| 6 | # |
| 7 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 8 | # |
| 9 | # Unless required by applicable law or agreed to in writing, software |
| 10 | # distributed under the License is distributed on an "AS IS" BASIS, |
| 11 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 12 | # See the License for the specific language governing permissions and |
| 13 | # limitations under the License. |
| 14 | """Module to update packages from GitHub archive.""" |
| 15 | |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 16 | import json |
| 17 | import re |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 18 | import urllib.request |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 19 | import urllib.error |
| 20 | from typing import List, Optional, Tuple |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 21 | |
| 22 | import archive_utils |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 23 | from base_updater import Updater |
Haibo Huang | 8845e1e | 2018-09-06 17:02:45 -0700 | [diff] [blame] | 24 | import git_utils |
Thiébaud Weksteen | 4ac289b | 2020-09-28 15:23:29 +0200 | [diff] [blame] | 25 | # pylint: disable=import-error |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 26 | import updater_utils |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 27 | GITHUB_URL_PATTERN: str = (r'^https:\/\/github.com\/([-\w]+)\/([-\w]+)\/' + |
| 28 | r'(releases\/download\/|archive\/)') |
| 29 | GITHUB_URL_RE: re.Pattern = re.compile(GITHUB_URL_PATTERN) |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 30 | |
| 31 | |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 32 | def _edit_distance(str1: str, str2: str) -> int: |
Haibo Huang | 9dcade4 | 2018-08-03 11:52:25 -0700 | [diff] [blame] | 33 | prev = list(range(0, len(str2) + 1)) |
| 34 | for i, chr1 in enumerate(str1): |
| 35 | cur = [i + 1] |
| 36 | for j, chr2 in enumerate(str2): |
| 37 | if chr1 == chr2: |
| 38 | cur.append(prev[j]) |
| 39 | else: |
| 40 | cur.append(min(prev[j + 1], prev[j], cur[j]) + 1) |
| 41 | prev = cur |
| 42 | return prev[len(str2)] |
| 43 | |
| 44 | |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 45 | def choose_best_url(urls: List[str], previous_url: str) -> str: |
Haibo Huang | 9dcade4 | 2018-08-03 11:52:25 -0700 | [diff] [blame] | 46 | """Returns the best url to download from a list of candidate urls. |
| 47 | |
| 48 | This function calculates similarity between previous url and each of new |
| 49 | urls. And returns the one best matches previous url. |
| 50 | |
| 51 | Similarity is measured by editing distance. |
| 52 | |
| 53 | Args: |
| 54 | urls: Array of candidate urls. |
| 55 | previous_url: String of the url used previously. |
| 56 | |
| 57 | Returns: |
| 58 | One url from `urls`. |
| 59 | """ |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 60 | return min(urls, |
| 61 | default="", |
| 62 | key=lambda url: _edit_distance(url, previous_url)) |
Haibo Huang | 9dcade4 | 2018-08-03 11:52:25 -0700 | [diff] [blame] | 63 | |
| 64 | |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 65 | class GithubArchiveUpdater(Updater): |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 66 | """Updater for archives from GitHub. |
| 67 | |
| 68 | This updater supports release archives in GitHub. Version is determined by |
| 69 | release name in GitHub. |
| 70 | """ |
| 71 | |
Sadaf Ebrahimi | 9afd310 | 2023-04-28 17:33:44 +0000 | [diff] [blame] | 72 | UPSTREAM_REMOTE_NAME: str = "update_origin" |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 73 | VERSION_FIELD: str = 'tag_name' |
| 74 | owner: str |
| 75 | repo: str |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 76 | |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 77 | def is_supported_url(self) -> bool: |
Stephen Hines | 5dd2a49 | 2023-11-22 16:40:34 -0800 | [diff] [blame] | 78 | if self._old_identifier.type.lower() != 'archive': |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 79 | return False |
Stephen Hines | 5dd2a49 | 2023-11-22 16:40:34 -0800 | [diff] [blame] | 80 | match = GITHUB_URL_RE.match(self._old_identifier.value) |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 81 | if match is None: |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 82 | return False |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 83 | try: |
| 84 | self.owner, self.repo = match.group(1, 2) |
| 85 | except IndexError: |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 86 | return False |
| 87 | return True |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 88 | |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 89 | def _fetch_latest_release(self) -> Optional[Tuple[str, List[str]]]: |
Thiébaud Weksteen | 4ac289b | 2020-09-28 15:23:29 +0200 | [diff] [blame] | 90 | # pylint: disable=line-too-long |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 91 | url = f'https://api.github.com/repos/{self.owner}/{self.repo}/releases/latest' |
Haibo Huang | 065d1ba | 2020-04-23 18:22:58 -0700 | [diff] [blame] | 92 | try: |
| 93 | with urllib.request.urlopen(url) as request: |
| 94 | data = json.loads(request.read().decode()) |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 95 | except urllib.error.HTTPError as err: |
| 96 | if err.code == 404: |
Haibo Huang | 065d1ba | 2020-04-23 18:22:58 -0700 | [diff] [blame] | 97 | return None |
| 98 | raise |
Haibo Huang | 8845e1e | 2018-09-06 17:02:45 -0700 | [diff] [blame] | 99 | supported_assets = [ |
| 100 | a['browser_download_url'] for a in data['assets'] |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 101 | if archive_utils.is_supported_archive(a['browser_download_url']) |
| 102 | ] |
Sadaf Ebrahimi | 28f76a1 | 2024-02-05 20:39:32 +0000 | [diff] [blame] | 103 | return data[self.VERSION_FIELD], supported_assets |
Haibo Huang | 065d1ba | 2020-04-23 18:22:58 -0700 | [diff] [blame] | 104 | |
Sadaf Ebrahimi | 602dd32 | 2023-06-23 19:30:07 +0000 | [diff] [blame] | 105 | def setup_remote(self) -> None: |
Sadaf Ebrahimi | 9afd310 | 2023-04-28 17:33:44 +0000 | [diff] [blame] | 106 | homepage = f'https://github.com/{self.owner}/{self.repo}' |
| 107 | remotes = git_utils.list_remotes(self._proj_path) |
| 108 | current_remote_url = None |
| 109 | for name, url in remotes.items(): |
| 110 | if name == self.UPSTREAM_REMOTE_NAME: |
| 111 | current_remote_url = url |
| 112 | |
| 113 | if current_remote_url is not None and current_remote_url != homepage: |
| 114 | git_utils.remove_remote(self._proj_path, self.UPSTREAM_REMOTE_NAME) |
| 115 | current_remote_url = None |
| 116 | |
| 117 | if current_remote_url is None: |
| 118 | git_utils.add_remote(self._proj_path, self.UPSTREAM_REMOTE_NAME, homepage) |
| 119 | |
Sadaf Ebrahimi | 2b3546c | 2024-02-28 21:37:12 +0000 | [diff] [blame] | 120 | git_utils.fetch(self._proj_path, self.UPSTREAM_REMOTE_NAME) |
Sadaf Ebrahimi | 9afd310 | 2023-04-28 17:33:44 +0000 | [diff] [blame] | 121 | |
Haibo Huang | a08fb60 | 2020-05-29 16:24:13 -0700 | [diff] [blame] | 122 | def _fetch_latest_tag(self) -> Tuple[str, List[str]]: |
Sadaf Ebrahimi | 9afd310 | 2023-04-28 17:33:44 +0000 | [diff] [blame] | 123 | """We want to avoid hitting GitHub API rate limit by using alternative solutions.""" |
Sadaf Ebrahimi | 2b3546c | 2024-02-28 21:37:12 +0000 | [diff] [blame] | 124 | tags = git_utils.list_remote_tags(self._proj_path, self.UPSTREAM_REMOTE_NAME) |
| 125 | parsed_tags = [updater_utils.parse_remote_tag(tag) for tag in tags] |
Sadaf Ebrahimi | d0ae405 | 2024-03-06 20:41:37 +0000 | [diff] [blame] | 126 | tag = updater_utils.get_latest_stable_release_tag(self._old_identifier.version, parsed_tags) |
Sadaf Ebrahimi | 9afd310 | 2023-04-28 17:33:44 +0000 | [diff] [blame] | 127 | return tag, [] |
Haibo Huang | 065d1ba | 2020-04-23 18:22:58 -0700 | [diff] [blame] | 128 | |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 129 | def _fetch_latest_version(self) -> None: |
Haibo Huang | 065d1ba | 2020-04-23 18:22:58 -0700 | [diff] [blame] | 130 | """Checks upstream and gets the latest release tag.""" |
Stephen Hines | 5dd2a49 | 2023-11-22 16:40:34 -0800 | [diff] [blame] | 131 | self._new_identifier.version, urls = (self._fetch_latest_release() |
Haibo Huang | a08fb60 | 2020-05-29 16:24:13 -0700 | [diff] [blame] | 132 | or self._fetch_latest_tag()) |
Haibo Huang | 8845e1e | 2018-09-06 17:02:45 -0700 | [diff] [blame] | 133 | |
| 134 | # Adds source code urls. |
Sadaf Ebrahimi | 28f76a1 | 2024-02-05 20:39:32 +0000 | [diff] [blame] | 135 | urls.append(f'https://github.com/{self.owner}/{self.repo}/archive/' |
| 136 | f'{self._new_identifier.version}.tar.gz') |
| 137 | urls.append(f'https://github.com/{self.owner}/{self.repo}/archive/' |
| 138 | f'{self._new_identifier.version}.zip') |
Haibo Huang | 8845e1e | 2018-09-06 17:02:45 -0700 | [diff] [blame] | 139 | |
Stephen Hines | 5dd2a49 | 2023-11-22 16:40:34 -0800 | [diff] [blame] | 140 | self._new_identifier.value = choose_best_url(urls, self._old_identifier.value) |
Haibo Huang | 8845e1e | 2018-09-06 17:02:45 -0700 | [diff] [blame] | 141 | |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 142 | def _fetch_latest_commit(self) -> None: |
Sadaf Ebrahimi | 8fe4c12 | 2023-07-11 20:55:42 +0000 | [diff] [blame] | 143 | """Checks upstream and gets the latest commit to default branch.""" |
Haibo Huang | 8845e1e | 2018-09-06 17:02:45 -0700 | [diff] [blame] | 144 | |
Thiébaud Weksteen | 4ac289b | 2020-09-28 15:23:29 +0200 | [diff] [blame] | 145 | # pylint: disable=line-too-long |
Sadaf Ebrahimi | 9afd310 | 2023-04-28 17:33:44 +0000 | [diff] [blame] | 146 | branch = git_utils.detect_default_branch(self._proj_path, |
| 147 | self.UPSTREAM_REMOTE_NAME) |
Stephen Hines | 5dd2a49 | 2023-11-22 16:40:34 -0800 | [diff] [blame] | 148 | self._new_identifier.version = git_utils.get_sha_for_branch( |
Sadaf Ebrahimi | 9afd310 | 2023-04-28 17:33:44 +0000 | [diff] [blame] | 149 | self._proj_path, self.UPSTREAM_REMOTE_NAME + '/' + branch) |
Stephen Hines | 5dd2a49 | 2023-11-22 16:40:34 -0800 | [diff] [blame] | 150 | self._new_identifier.value = ( |
Thiébaud Weksteen | 4ac289b | 2020-09-28 15:23:29 +0200 | [diff] [blame] | 151 | # pylint: disable=line-too-long |
Stephen Hines | 5dd2a49 | 2023-11-22 16:40:34 -0800 | [diff] [blame] | 152 | f'https://github.com/{self.owner}/{self.repo}/archive/{self._new_identifier.version}.zip' |
Haibo Huang | a08fb60 | 2020-05-29 16:24:13 -0700 | [diff] [blame] | 153 | ) |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 154 | |
Haibo Huang | a08fb60 | 2020-05-29 16:24:13 -0700 | [diff] [blame] | 155 | def check(self) -> None: |
Haibo Huang | 24950e7 | 2018-06-29 14:53:39 -0700 | [diff] [blame] | 156 | """Checks update for package. |
| 157 | |
| 158 | Returns True if a new version is available. |
| 159 | """ |
Sadaf Ebrahimi | 602dd32 | 2023-06-23 19:30:07 +0000 | [diff] [blame] | 160 | self.setup_remote() |
Stephen Hines | 5dd2a49 | 2023-11-22 16:40:34 -0800 | [diff] [blame] | 161 | if git_utils.is_commit(self._old_identifier.version): |
Haibo Huang | 39aaab6 | 2019-01-25 12:23:03 -0800 | [diff] [blame] | 162 | self._fetch_latest_commit() |
Haibo Huang | 8845e1e | 2018-09-06 17:02:45 -0700 | [diff] [blame] | 163 | else: |
Haibo Huang | 39aaab6 | 2019-01-25 12:23:03 -0800 | [diff] [blame] | 164 | self._fetch_latest_version() |
Haibo Huang | 24950e7 | 2018-06-29 14:53:39 -0700 | [diff] [blame] | 165 | |
Sadaf Ebrahimi | 7d0bab7 | 2023-10-23 19:45:03 +0000 | [diff] [blame] | 166 | def update(self) -> None: |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 167 | """Updates the package. |
| 168 | |
Haibo Huang | 24950e7 | 2018-06-29 14:53:39 -0700 | [diff] [blame] | 169 | Has to call check() before this function. |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 170 | """ |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 171 | temporary_dir = None |
| 172 | try: |
Haibo Huang | a08fb60 | 2020-05-29 16:24:13 -0700 | [diff] [blame] | 173 | temporary_dir = archive_utils.download_and_extract( |
Stephen Hines | 5dd2a49 | 2023-11-22 16:40:34 -0800 | [diff] [blame] | 174 | self._new_identifier.value) |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 175 | package_dir = archive_utils.find_archive_root(temporary_dir) |
Haibo Huang | 329e681 | 2020-05-29 14:12:20 -0700 | [diff] [blame] | 176 | updater_utils.replace_package(package_dir, self._proj_path) |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 177 | finally: |
Elliott Hughes | 1c3a4f4 | 2018-08-03 15:35:48 -0700 | [diff] [blame] | 178 | # Don't remove the temporary directory, or it'll be impossible |
| 179 | # to debug the failure... |
| 180 | # shutil.rmtree(temporary_dir, ignore_errors=True) |
(raulenrique) | dfdda47 | 2018-06-04 12:02:29 -0700 | [diff] [blame] | 181 | urllib.request.urlcleanup() |