From 757226155cc403ea6c13cde7d0cda4267b44ac91 Mon Sep 17 00:00:00 2001 From: Andrei Litvin Date: Thu, 26 Jan 2023 10:14:23 -0500 Subject: [PATCH] Add a script to simplify getting relevant zap binary if not manually installed (#24651) * Start writing a zap download script. Not yet functional * Restyle * Started detecting zap version to use * Added some functionality - at least zap release download seems to work * Restyle * Minimal documentation for the release download * Add support for zap development version checkout as well * Restyle * Remove some left over comments * Switch download default to project_root/.zap --- .gitignore | 3 + scripts/tools/zap/zap_download.py | 224 ++++++++++++++++++++++++++++++ 2 files changed, 227 insertions(+) create mode 100755 scripts/tools/zap/zap_download.py diff --git a/.gitignore b/.gitignore index 8d4ea014d762c6..5c9f408a1cb968 100644 --- a/.gitignore +++ b/.gitignore @@ -71,3 +71,6 @@ compile_commands.json # log files *.log examples/thermostat/ameba/build + +# Downloaded zap without a pigweed root (via zap_download.py) +.zap diff --git a/scripts/tools/zap/zap_download.py b/scripts/tools/zap/zap_download.py new file mode 100755 index 00000000000000..951691d65fbd19 --- /dev/null +++ b/scripts/tools/zap/zap_download.py @@ -0,0 +1,224 @@ +#!/usr/bin/env python3 + +# Copyright (c) 2023 Project CHIP Authors +# +# 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. + +import enum +import io +import logging +import os +import re +import shlex +import shutil +import subprocess +import sys +import zipfile +from typing import Optional + +import click +import requests + +try: + import coloredlogs + _has_coloredlogs = True +except: + _has_coloredlogs = False + +# Supported log levels, mapping string values required for argument +# parsing into logging constants +__LOG_LEVELS__ = { + 'debug': logging.DEBUG, + 'info': logging.INFO, + 'warn': logging.WARN, + 'fatal': logging.FATAL, +} + + +class DownloadType(enum.Enum): + RELEASE = enum.auto() # asking for a zip file release download + SOURCE = enum.auto() # asking for a source download (will work on arm64 for example) + + +def _GetDefaultExtractRoot(): + if 'PW_ENVIRONMENT_ROOT' in os.environ: + return os.environ['PW_ENVIRONMENT_ROOT'] + else: + return ".zap" + + +def _LogPipeLines(pipe, prefix): + l = logging.getLogger().getChild(prefix) + for line in iter(pipe.readline, b''): + line = line.strip().decode('utf-8', errors="ignore") + l.info('%s' % line) + + +def _ExecuteProcess(cmd, cwd): + logging.info('Executing %r in %s' % (cmd, cwd)) + + process = subprocess.Popen( + cmd, + stdout=subprocess.PIPE, + stderr=subprocess.STDOUT, + cwd=cwd) + + with process.stdout: + _LogPipeLines(process.stdout, cmd[0]) + + exitcode = process.wait() + if exitcode != 0: + raise Exception("Error executing process: %d" % exitcode) + + +def _SetupSourceZap(install_directory: str, zap_version: str): + if os.path.exists(install_directory): + logging.warning("Completely re-creating %s", install_directory) + shutil.rmtree(install_directory) + + os.makedirs(install_directory, exist_ok=True) + + _ExecuteProcess( + f"git clone --depth 1 --branch {zap_version} https://github.com/project-chip/zap.git .".split(), + install_directory + ) + + _ExecuteProcess(f"npm ci".split(), install_directory) + + +def _SetupReleaseZap(install_directory: str, zap_version: str): + """ + Downloads the given [zap_version] into "[install_directory]/zap-[zap_version]/". + + Will download the given release from github releases. + """ + + if sys.platform == 'linux': + zap_platform = 'linux' + elif sys.platform == 'darwin': + zap_platform = 'mac' + else: + raise Exception('Unknown platform - do not know what zip file to download.') + + url = f"https://github.com/project-chip/zap/releases/download/{zap_version}/zap-{zap_platform}.zip" + + logging.info("Fetching: %s", url) + + r = requests.get(url, stream=True) + z = zipfile.ZipFile(io.BytesIO(r.content)) + + logging.info("Data downloaded, extracting ...") + z.extractall(install_directory) + logging.info("Done extracting.") + + +def _GetZapVersionToUse(project_root): + """ + Heuristic to figure out what zap version should be used. + + Looks at the given project root and tries to figure out the zap tag/version. + """ + + # We have several locations for zap versioning: + # - CI is likely the most reliable as long as we use the "latest build" + # - zap_execution.py is what is currently used, but it is a minimum version + # + # Based on the above, we assume CI is using the latest build (will not be + # out of sync more than a few days) and even if it is not, zap is often + # backwards compatible (new features added, but codegen should not change + # that often for fixed inputs) + # + # This heuristic may be bad at times, however then you can also override the + # version in command line parameters + + match_re = re.compile(r'.*ENV\s+ZAP_VERSION=([^# ]*)') + + docker_path = os.path.join(project_root, "integrations/docker/images/chip-build/Dockerfile") + + with open(docker_path, 'rt') as f: + for l in f.readlines(): + l = l.strip() + m = match_re.match(l) + if not m: + continue + return m.group(1) + + raise Exception(f"Failed to determine version from {docker_path}") + + +@click.command() +@click.option( + '--log-level', + default='INFO', + show_default=True, + type=click.Choice(__LOG_LEVELS__.keys(), case_sensitive=False), + callback=lambda c, p, v: __LOG_LEVELS__[v], + help='Determines the verbosity of script output') +@click.option( + '--sdk-root', + default=os.path.realpath(os.path.join(os.path.dirname(os.path.realpath(__file__)), '..', '..', '..')), + show_default=True, + help='Path to the SDK root (where zap versioning exists).') +@click.option( + '--extract-root', + default=_GetDefaultExtractRoot(), + show_default=True, + help='Directory where too unpack/checkout zap') +@click.option( + '--zap-version', + default=None, + help='Force to checkout this zap version instead of trying to auto-detect') +@click.option( + '--zap', + default='RELEASE', + show_default=True, + type=click.Choice(DownloadType.__members__, case_sensitive=False), + callback=lambda c, p, v: getattr(DownloadType, v), + help='What type of zap download to perform') +def main(log_level: str, sdk_root: str, extract_root: str, zap_version: Optional[str], zap: DownloadType): + if _has_coloredlogs: + coloredlogs.install(level=log_level, fmt='%(asctime)s %(name)s %(levelname)-7s %(message)s') + else: + logging.basicConfig( + level=log_level, + format='%(asctime)s %(name)s %(levelname)-7s %(message)s', + datefmt='%Y-%m-%d %H:%M:%S' + ) + + if extract_root == ".zap": + # Place .zap in the project root + extract_root = os.path.join(sdk_root, extract_root) + + if not zap_version: + zap_version = _GetZapVersionToUse(sdk_root) + logging.info('Found required zap version to be: %s' % zap_version) + + logging.debug('User requested to download a %s zap version %s into %s', zap, zap_version, extract_root) + + install_directory = os.path.join(extract_root, f"zap-{zap_version}") + + if zap == DownloadType.SOURCE: + install_directory = install_directory + "-src" + _SetupSourceZap(install_directory, zap_version) + + # Make sure the results can be used in scripts + print(f"export ZAP_DEVELOPMENT_PATH={shlex.quote(install_directory)}") + else: + _SetupReleaseZap(install_directory, zap_version) + + # Make sure the results can be used in scripts + print(f"export ZAP_INSTALL_PATH={shlex.quote(install_directory)}") + + +if __name__ == '__main__': + main()