Skip to content

Commit

Permalink
Add a script to simplify getting relevant zap binary if not manually …
Browse files Browse the repository at this point in the history
…installed (project-chip#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
  • Loading branch information
andy31415 authored and David Lechner committed Mar 22, 2023
1 parent 0abd23a commit 7572261
Show file tree
Hide file tree
Showing 2 changed files with 227 additions and 0 deletions.
3 changes: 3 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -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
224 changes: 224 additions & 0 deletions scripts/tools/zap/zap_download.py
Original file line number Diff line number Diff line change
@@ -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()

0 comments on commit 7572261

Please sign in to comment.