| # 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. |
| """Tool functions to deal with files.""" |
| |
| import datetime |
| from functools import cache |
| import os |
| from pathlib import Path |
| import textwrap |
| |
| # pylint: disable=import-error |
| from google.protobuf import text_format # type: ignore |
| |
| # pylint: disable=import-error |
| import metadata_pb2 # type: ignore |
| |
| |
| METADATA_FILENAME = 'METADATA' |
| |
| |
| @cache |
| def external_path() -> Path: |
| """Returns the path to //external. |
| |
| We cannot use the relative path from this file to find the top of the tree because |
| this will often be run in a "compiled" form from an arbitrary location in the out |
| directory. We can't fully rely on ANDROID_BUILD_TOP because not all contexts will |
| have run envsetup/lunch either. We use ANDROID_BUILD_TOP whenever it is set, but if |
| it is not set we instead rely on the convention that the CWD is the root of the tree |
| (updater.sh will cd there before executing). |
| |
| There is one other context where this function cannot succeed: CI. Tests run in CI |
| do not have a source tree to find, so calling this function in that context will |
| fail. |
| """ |
| android_top = Path(os.environ.get("ANDROID_BUILD_TOP", os.getcwd())) |
| top = android_top / 'external' |
| |
| if not top.exists(): |
| raise RuntimeError( |
| f"{top} does not exist. This program must be run from the " |
| f"root of an Android tree (CWD is {os.getcwd()})." |
| ) |
| return top |
| |
| |
| def get_absolute_project_path(proj_path: Path) -> Path: |
| """Gets absolute path of a project. |
| |
| Path resolution starts from external/. |
| """ |
| return external_path() / proj_path |
| |
| |
| def get_metadata_path(proj_path: Path) -> Path: |
| """Gets the absolute path of METADATA for a project.""" |
| return get_absolute_project_path(proj_path) / METADATA_FILENAME |
| |
| |
| def get_relative_project_path(proj_path: Path) -> Path: |
| """Gets the relative path of a project starting from external/.""" |
| return get_absolute_project_path(proj_path).relative_to(external_path()) |
| |
| |
| def canonicalize_project_path(proj_path: Path) -> Path: |
| """Returns the canonical representation of the project path. |
| |
| For paths that are in the same tree as external_updater (the common case), the |
| canonical path is the path of the project relative to //external. |
| |
| For paths that are in a different tree (an uncommon case used for updating projects |
| in other builds such as the NDK), the canonical path is the absolute path. |
| """ |
| try: |
| return get_relative_project_path(proj_path) |
| except ValueError: |
| # A less common use case, but the path might be to a non-local tree, in which case |
| # the path will not be relative to our tree. This happens when using |
| # external_updater in another project like the NDK or rr. |
| if proj_path.is_absolute(): |
| return proj_path |
| |
| # Not relative to //external, and not an absolute path. This case hasn't existed |
| # before, so it has no canonical form. |
| raise ValueError( |
| f"{proj_path} must be either an absolute path or relative to {external_path()}" |
| ) |
| |
| |
| def read_metadata(proj_path: Path) -> metadata_pb2.MetaData: |
| """Reads and parses METADATA file for a project. |
| |
| Args: |
| proj_path: Path to the project. |
| |
| Returns: |
| Parsed MetaData proto. |
| |
| Raises: |
| text_format.ParseError: Occurred when the METADATA file is invalid. |
| FileNotFoundError: Occurred when METADATA file is not found. |
| """ |
| |
| with get_metadata_path(proj_path).open('r') as metadata_file: |
| metadata = metadata_file.read() |
| return text_format.Parse(metadata, metadata_pb2.MetaData()) |
| |
| |
| def write_metadata(proj_path: Path, metadata: metadata_pb2.MetaData, keep_date: bool) -> None: |
| """Writes updated METADATA file for a project. |
| |
| This function updates last_upgrade_date in metadata and write to the project |
| directory. |
| |
| Args: |
| proj_path: Path to the project. |
| metadata: The MetaData proto to write. |
| keep_date: Do not change date. |
| """ |
| |
| if not keep_date: |
| date = metadata.third_party.last_upgrade_date |
| now = datetime.datetime.now() |
| date.year = now.year |
| date.month = now.month |
| date.day = now.day |
| try: |
| rel_proj_path = str(get_relative_project_path(proj_path)) |
| except ValueError: |
| # Absolute paths to other trees will not be relative to our tree. There are |
| # not portable instructions for upgrading that project, since the path will |
| # differ between machines (or checkouts). |
| rel_proj_path = "<absolute path to project>" |
| usage_hint = textwrap.dedent(f"""\ |
| # This project was upgraded with external_updater. |
| # Usage: tools/external_updater/updater.sh update {rel_proj_path} |
| # For more info, check https://cs.android.com/android/platform/superproject/+/master:tools/external_updater/README.md |
| |
| """) |
| text_metadata = usage_hint + text_format.MessageToString(metadata) |
| with get_metadata_path(proj_path).open('w') as metadata_file: |
| if metadata.third_party.license_type == metadata_pb2.LicenseType.BY_EXCEPTION_ONLY: |
| metadata_file.write(textwrap.dedent("""\ |
| # THIS PACKAGE HAS SPECIAL LICENSING CONDITIONS. PLEASE |
| # CONSULT THE OWNERS AND [email protected] BEFORE |
| # DEPENDING ON IT IN YOUR PROJECT. |
| |
| """)) |
| metadata_file.write(text_metadata) |