Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Make Chef CI use existing zzz_generated and complete Chef CD for existing platforms #19478

Merged
merged 28 commits into from
Jun 23, 2022
Merged
Show file tree
Hide file tree
Changes from 9 commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
27512d6
Squashed commit of the following:
aBozowski Jun 10, 2022
4a1a8c7
Merge remote-tracking branch 'upstream/master' into bozowski/chef-cd-…
aBozowski Jun 16, 2022
f38c514
Bundle .matter, metadata & ensure .matter commit
aBozowski Jun 16, 2022
32505f4
Add partial check for .matter regen
aBozowski Jun 16, 2022
ac28149
Restyle and spelling
aBozowski Jun 16, 2022
1c668b6
Address comments
aBozowski Jun 17, 2022
fc09041
Remove reflection
aBozowski Jun 17, 2022
c44ab73
Merge remote-tracking branch 'upstream/master' into bozowski/chef-cd-…
aBozowski Jun 20, 2022
29e7815
Generate zzz, .MATTERMD5 should go away in future
aBozowski Jun 20, 2022
fe2fe4e
Merge branch 'project-chip:master' into bozowski/chef-cd-merge
aBozowski Jun 21, 2022
dca1b39
Temp del workflows
aBozowski Jun 21, 2022
6a5f81c
Remove chef exclusion from regen
aBozowski Jun 21, 2022
6d488ef
Make regen all put chef exs in individual dirs
aBozowski Jun 21, 2022
88b80cd
Remove validate zzz
aBozowski Jun 21, 2022
b6dfa21
Remove unused code from chef, change use_zzz
aBozowski Jun 21, 2022
b8b416b
Skip util test files in gen all
aBozowski Jun 21, 2022
f2eeeb8
Commit chef in root zzz
aBozowski Jun 21, 2022
415c7f9
Delete chef zzz
aBozowski Jun 21, 2022
789506a
Clean chef .matter files
aBozowski Jun 21, 2022
6167b9a
Add post build for chef
aBozowski Jun 21, 2022
d504929
Change .matter bundle rc to oot zzz
aBozowski Jun 21, 2022
80e2d0c
Generated files
aBozowski Jun 21, 2022
75ece18
Restore workflows
aBozowski Jun 21, 2022
9c9d5f2
Change open to with
aBozowski Jun 21, 2022
2746118
Merge pull request #13 from aBozowski/bozowski/chef-cd-merge+remove-c…
aBozowski Jun 21, 2022
cf7b182
Merge remote-tracking branch 'upstream/master' into bozowski/chef-cd-…
aBozowski Jun 21, 2022
cba902d
Restyle
aBozowski Jun 21, 2022
a503442
Merge remote-tracking branch 'upstream/master' into bozowski/chef-cd-…
aBozowski Jun 22, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
24 changes: 18 additions & 6 deletions examples/chef/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,17 @@ These additional files will be used by the CI jobs to validate whether the cache
must be regenerated i.e. regeneration is needed when ZAP or the input ZAP files
change.

`--generate_zzz` also creates `{device_name}.matter` and
`{device_name}.MATTERMD5` in `/devices`.

`--validate_zzz` will ensure that the `.matter` file exists, and ensure that the
`.MATTERMD5` file contains a hash matching the dynamically calculated hash of
the respective device's ZAP file.

Note that this check only requests regeneration of `.matter` files when a ZAP
file changes. Changes to the code that generate the `.matter` files will not
cause the validation job to fail.

### Workflow

All CI jobs for chef can be found in `.github/workflows/chef.yaml`.
Expand All @@ -103,10 +114,11 @@ All CI jobs for chef can be found in `.github/workflows/chef.yaml`.
The workflow begins by calling chef with `--validate_zzz`.

`--validate_zzz` will recalculate the current ZAP commit and the md5 of all
example ZAP files and compare with what is committed to `zzz_generated`.
example ZAP files and compare with what is committed to `/zzz_generated` and
`/devices`.

If the validation job fails, it will provide instructions to repair
`zzz_generated` and no builds will run.
If the validation job fails, it will provide instructions to repair the repo and
no builds will run.

#### Build

Expand All @@ -116,6 +128,6 @@ which run in parallel.
These jobs use a platform-specific image with base `chip-build`.

The build jobs call chef with the options `--ci -t <PLATFORM>`. The `--ci`
option will execute builds for all devices specified in `_CI_ALLOW_LIST` defined
in `chef.py` (so long as these devices are also in `/devices`) on the specified
platform.
option will execute builds for all devices specified in
`cicd_config["ci_allow_list"]` defined in `chef.py` (so long as these devices
are also in `/devices`) on the specified platform.
223 changes: 190 additions & 33 deletions examples/chef/chef.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@

import constants
import stateful_shell
from sample_app_util import zap_file_parser

TermColors = constants.TermColors

Expand All @@ -40,8 +41,9 @@
_CHEF_ZZZ_ROOT = os.path.join(_CHEF_SCRIPT_PATH, "zzz_generated")
_CI_DEVICE_MANIFEST_NAME = "INPUTMD5.txt"
_CI_ZAP_MANIFEST_NAME = "ZAPSHA.txt"
_CICD_CONFIG_FILE_NAME = os.path.join(_CHEF_SCRIPT_PATH, "cicd_meta.json")
_CI_ALLOW_LIST = ["rootnode_dimmablelight_gY80DaqEUL"]
_CI_MATTER_MD5_EXT = ".MATTERMD5"
_CICD_CONFIG_FILE_NAME = os.path.join(_CHEF_SCRIPT_PATH, "cicd_config.json")
_CD_STAGING_DIR = os.path.join(_CHEF_SCRIPT_PATH, "staging")

gen_dir = "" # Filled in after sample app type is read from args.

Expand Down Expand Up @@ -116,11 +118,12 @@ def check_zap() -> str:


def generate_device_manifest(
write_manifest_file: bool = False) -> Dict[str, Any]:
"""Produces dictionary containing md5 of device dir zap files.
write_manifest_files: bool = False) -> Dict[str, Any]:
"""Produces dictionary containing md5 of device dir zap files
and zap commit.

Args:
write_manifest_file: Serialize manifest in tree.
write_manifest_files: Serialize digests in tree.
Returns:
Dict containing MD5 of device dir zap files.
"""
Expand All @@ -135,26 +138,24 @@ def generate_device_manifest(
device_file_md5 = hashlib.md5(device_file_data).hexdigest()
devices_manifest[device_name] = device_file_md5
flush_print(f"Current digest for {device_name} : {device_file_md5}")
if write_manifest_file:
if write_manifest_files:
device_zzz_dir = os.path.join(_CHEF_ZZZ_ROOT, device_name)
device_zzz_md5_file = os.path.join(device_zzz_dir, _CI_DEVICE_MANIFEST_NAME)
with open(device_zzz_md5_file, "w+") as md5_file:
md5_file.write(device_file_md5)
device_zzz_zap_sha_file = os.path.join(device_zzz_dir, _CI_ZAP_MANIFEST_NAME)
with open(device_zzz_zap_sha_file, "w+") as zap_sha_file:
zap_sha_file.write(zap_sha)
device_matter_md5_file_path = os.path.join(_DEVICE_FOLDER,
device_name+_CI_MATTER_MD5_EXT)
with open(device_matter_md5_file_path, "w+") as matter_md5_file:
matter_md5_file.write(device_file_md5)
return ci_manifest


def load_cicd_config() -> Dict[str, Any]:
with open(_CICD_CONFIG_FILE_NAME) as config_file:
config = json.loads(config_file.read())
for platform_name, platform_config in config.items():
has_build_dir = "build_dir" in platform_config
has_plat_label = "platform_label" in platform_config
if not has_build_dir or not has_plat_label:
flush_print(f"{platform_name} CICD config missing build_dir or platform_label")
exit(1)
return config


Expand All @@ -168,11 +169,130 @@ def flush_print(
with_border: Add boarder above and below to_print.
"""
if with_border:
border = ('-' * 64) + '\n'
border = ('-' * len(to_print)) + '\n'
to_print = f"{border}{to_print}\n{border}"
print(to_print, flush=True)


def unwrap_cmd(cmd: str) -> str:
"""Dedent and replace new line with space."""
return textwrap.dedent(cmd).replace("\n", " ")


def bundle(platform: str, device_name: str) -> None:
"""Filters files from the build output folder for CD.
Clears _CD_STAGING_DIR.
Calls bundle_{platform}(device_name).
exit(1) for missing bundle_{platform}.
Adds .matter files into _CD_STAGING_DIR.
Generates metadata for device in _CD_STAGING_DIR.

Args:
platform: The platform to bundle.
device_name: The example to bundle.
"""
matter_file = f"{device_name}.matter"
zap_file = os.path.join(_DEVICE_FOLDER, f"{device_name}.zap")
flush_print(f"Bundling {platform}", with_border=True)
flush_print(f"Cleaning {_CD_STAGING_DIR}")
shutil.rmtree(_CD_STAGING_DIR, ignore_errors=True)
os.mkdir(_CD_STAGING_DIR)
if platform == "linux":
bundle_linux(device_name)
elif platform == "nrfconnect":
bundle_nrfconnect(device_name)
elif platform == "esp32":
bundle_esp32(device_name)
else:
flush_print(f"No bundle function for {platform}!")
exit(1)
flush_print(f"Copying {matter_file}")
src_item = os.path.join(_DEVICE_FOLDER, matter_file)
dest_item = os.path.join(_CD_STAGING_DIR, matter_file)
shutil.copy(src_item, dest_item)
flush_print(f"Generating metadata for {device_name}")
metadata_file = zap_file_parser.generate_hash_metadata_file(zap_file)
metadata_dest = os.path.join(_CD_STAGING_DIR,
os.path.basename(metadata_file))
shutil.copy(metadata_file, metadata_dest)


#
# Per-platform bundle functions
#


def bundle_linux(device_name: str) -> None:
linux_root = os.path.join(_CHEF_SCRIPT_PATH,
"linux",
"out")
map_file_name = f"{device_name}.map"
src_item = os.path.join(linux_root, device_name)
dest_item = os.path.join(_CD_STAGING_DIR, device_name)
shutil.copy(src_item, dest_item)
src_item = os.path.join(linux_root, map_file_name)
dest_item = os.path.join(_CD_STAGING_DIR, map_file_name)
shutil.copy(src_item, dest_item)


def bundle_nrfconnect(device_name: str) -> None:
zephyr_exts = ["elf", "map", "hex"]
script_files = ["firmware_utils.py",
"nrfconnect_firmware_utils.py"]
nrf_root = os.path.join(_CHEF_SCRIPT_PATH,
"nrfconnect",
"build",
"zephyr")
scripts_root = os.path.join(_REPO_BASE_PATH,
"scripts",
"flashing")
gen_script_path = os.path.join(scripts_root,
"gen_flashing_script.py")
sub_dir = os.path.join(_CD_STAGING_DIR, device_name)
os.mkdir(sub_dir)
for zephyr_ext in zephyr_exts:
input_base = f"zephyr.{zephyr_ext}"
output_base = f"{device_name}.{zephyr_ext}"
src_item = os.path.join(nrf_root, input_base)
if zephyr_ext == "hex":
dest_item = os.path.join(sub_dir, output_base)
else:
dest_item = os.path.join(_CD_STAGING_DIR, output_base)
shutil.copy(src_item, dest_item)
for script_file in script_files:
src_item = os.path.join(scripts_root, script_file)
dest_item = os.path.join(sub_dir, script_file)
shutil.copy(src_item, dest_item)
shell.run_cmd(f"cd {sub_dir}")
command = f"""\
python3 {gen_script_path} nrfconnect
--output {device_name}.flash.py
--application {device_name}.hex"""
shell.run_cmd(unwrap_cmd(command))


def bundle_esp32(device_name: str) -> None:
"""Reference example for bundle_{platform}
functions, which should copy/move files from a build
output dir into _CD_STAGING_DIR to be archived.

Args:
device_name: The device to bundle.
"""
esp_root = os.path.join(_CHEF_SCRIPT_PATH,
"esp32",
"build")
manifest_file = os.path.join(esp_root,
"chip-shell.flashbundle.txt")
with open(manifest_file) as manifest:
for item in manifest:
item = item.strip()
src_item = os.path.join(esp_root, item)
dest_item = os.path.join(_CD_STAGING_DIR, item)
os.makedirs(os.path.dirname(dest_item), exist_ok=True)
shutil.copy(src_item, dest_item)


def main(argv: Sequence[str]) -> None:
check_python_version()
config = load_config()
Expand Down Expand Up @@ -258,7 +378,7 @@ def main(argv: Sequence[str]) -> None:
parser.add_option("", "--build_all", help="For use in CD only. Builds and bundles all chef examples for the specified platform. Uses --use_zzz. Chef exits after completion.",
dest="build_all", action="store_true")
parser.add_option(
"", "--ci", help="Builds Chef examples defined in _CI_ALLOW_LIST. Uses --use_zzz. Uses specified target from -t. Chef exits after completion.", dest="ci", action="store_true")
"", "--ci", help="Builds Chef examples defined in cicd_config. Uses --use_zzz. Uses specified target from -t. Chef exits after completion.", dest="ci", action="store_true")

options, _ = parser.parse_args(argv)

Expand All @@ -281,13 +401,18 @@ def main(argv: Sequence[str]) -> None:
cd ../../..
./examples/chef/chef.py --generate_zzz
git add examples/chef/zzz_generated
Ensure you are running with the latest version of ZAP from master!""")
git add examples/chef/devices
Ensure you are running with the latest version of ZAP from master!
""")
ci_manifest = generate_device_manifest()
current_zap = ci_manifest["zap_commit"]
for device, device_md5 in ci_manifest["devices"].items():
zzz_dir = os.path.join(_CHEF_ZZZ_ROOT, device)
device_zap_sha_file = os.path.join(zzz_dir, _CI_ZAP_MANIFEST_NAME)
device_md5_file = os.path.join(zzz_dir, _CI_DEVICE_MANIFEST_NAME)
matter_file = os.path.join(_DEVICE_FOLDER, f"{device}.matter")
device_matter_md5_file = os.path.join(_DEVICE_FOLDER,
device+_CI_MATTER_MD5_EXT)
help_msg = f"{device}: {fix_instructions}"
if not os.path.exists(device_zap_sha_file):
flush_print(f"ZAP VERSION MISSING {help_msg}")
Expand All @@ -307,6 +432,18 @@ def main(argv: Sequence[str]) -> None:
if output_cached_md5 != device_md5:
flush_print(f"INPUT MD5 MISMATCH {help_msg}")
exit(1)
if not os.path.exists(matter_file):
flush_print(f"MISSING MATTER FILE {help_msg}")
exit(1)
if not os.path.exists(device_matter_md5_file):
flush_print(f"MISSING MATTER MD5 {help_msg}")
exit(1)
else:
with open(device_matter_md5_file) as matter_md5:
output_cached_md5 = matter_md5.read()
if output_cached_md5 != device_md5:
flush_print(f"MATTER MD5 MISMATCH {help_msg}")
exit(1)
flush_print("Cached ZAP output is up to date!")
exit(0)

Expand All @@ -317,12 +454,11 @@ def main(argv: Sequence[str]) -> None:
if options.do_bootstrap_zap:
if sys.platform == "linux" or sys.platform == "linux2":
flush_print("Installing ZAP OS package dependencies")
install_deps_cmd = textwrap.dedent("""\
install_deps_cmd = """\
sudo apt-get install node node-yargs npm
libpixman-1-dev libcairo2-dev libpango1.0-dev node-pre-gyp
libjpeg9-dev libgif-dev node-typescript""")
install_deps_cmd = install_deps_cmd.replace("\n", " ")
shell.run_cmd(install_deps_cmd)
libjpeg9-dev libgif-dev node-typescript"""
shell.run_cmd(unwrap_cmd(install_deps_cmd))
if sys.platform == "darwin":
flush_print("Installation of ZAP OS packages not supported on MacOS")
if sys.platform == "win32":
Expand Down Expand Up @@ -353,53 +489,74 @@ def main(argv: Sequence[str]) -> None:
device_name,
"zap-generated")
os.makedirs(device_out_dir)
shell.run_cmd(textwrap.dedent(f"""\
{_REPO_BASE_PATH}/scripts/tools/zap/generate.py \
{_CHEF_SCRIPT_PATH}/devices/{device_name}.zap -o {device_out_dir}"""))
command = f"""\
{_REPO_BASE_PATH}/scripts/tools/zap/generate.py
{_CHEF_SCRIPT_PATH}/devices/{device_name}.zap -o {device_out_dir}"""
shell.run_cmd(unwrap_cmd(command))
shell.run_cmd(f"touch {device_out_dir}/af-gen-event.h")
generate_device_manifest(write_manifest_file=True)
generate_device_manifest(write_manifest_files=True)
exit(0)

#
# CI
#

if options.ci:
for device_name in [d for d in _DEVICE_LIST if d in _CI_ALLOW_LIST]:
for device_name in [d for d in _DEVICE_LIST if d in cicd_config["ci_allow_list"]]:
if options.build_target == "nrfconnect":
shell.run_cmd("export GNUARMEMB_TOOLCHAIN_PATH=\"$PW_ARM_CIPD_INSTALL_DIR\"")
shell.run_cmd(f"cd {_CHEF_SCRIPT_PATH}")
command = f"./chef.py -cbr --use_zzz -d {device_name} -t {options.build_target}"
flush_print(f"Building {command}", with_border=True)
shell.run_cmd(command)
# TODO call per-platform bundle function for extra validation
bundle(options.build_target, device_name)
exit(0)

#
# Build all
# CD
#

if options.build_all:
flush_print("Building all chef examples")
archive_prefix = "/workspace/artifacts/"
archive_suffix = ".tar.gz"
os.makedirs(archive_prefix, exist_ok=True)
failed_builds = []
for device_name in _DEVICE_LIST:
for platform, platform_meta in cicd_config.items():
directory = platform_meta['build_dir']
label = platform_meta['platform_label']
output_dir = os.path.join(_CHEF_SCRIPT_PATH, directory)
for platform, label in cicd_config["cd_platforms"].items():
command = f"./chef.py -cbr --use_zzz -d {device_name} -t {platform}"
flush_print(f"Building {command}", with_border=True)
shell.run_cmd(f"cd {_CHEF_SCRIPT_PATH}")
shell.run_cmd("export GNUARMEMB_TOOLCHAIN_PATH=\"$PW_ARM_CIPD_INSTALL_DIR\"")
shell.run_cmd(command)
# TODO Needs to call per-platform bundle function
try:
shell.run_cmd(command)
except RuntimeError as build_fail_error:
failed_builds.append((device_name, platform, "build"))
flush_print(str(build_fail_error))
break
try:
bundle(platform, device_name)
except FileNotFoundError as bundle_fail_error:
failed_builds.append((device_name, platform, "bundle"))
flush_print(str(bundle_fail_error))
break
archive_name = f"{label}-{device_name}"
archive_full_name = archive_prefix + archive_name + archive_suffix
flush_print(f"Adding build output to archive {archive_full_name}")
if os.path.exists(archive_full_name):
os.remove(archive_full_name)
with tarfile.open(archive_full_name, "w:gz") as tar:
tar.add(output_dir, arcname=".")
tar.add(_CD_STAGING_DIR, arcname=".")
if len(failed_builds) == 0:
flush_print("No build failures", with_border=True)
else:
flush_print("Logging build failures", with_border=True)
for failed_build in failed_builds:
fail_log = f"""\
Device: {failed_build[0]},
Platform: {failed_build[1]},
Phase: {failed_build[2]}"""
flush_print(unwrap_cmd(fail_log))
exit(0)

#
Expand Down
Loading