-
Notifications
You must be signed in to change notification settings - Fork 576
/
Copy pathbuild.py
133 lines (111 loc) · 7.04 KB
/
build.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
#!/usr/bin/env python3
#
# Container build system for managing package configurations and multi-stage build chains, with automated testing and dependency tracking.
#
# A "package" is composed of a Dockerfile, configs, and test scripts. These are found under the jetson-containers/packages directory.
# There are also "meta-packages" that have no Dockerfiles themselves, but specify a set of packages to include (e.g. l4t-pytorch)
#
# Configuration metadata (such as the package's dependencies) can be inline YAML in the Dockerfile header.
# It can also be a config.py script that sets build arguments dynamically (i.e. based on the L4T version)
# Subpackages can be dynamically created in the config files (i.e. the various permutations of the ROS containers)
#
# Some example build scenarios:
#
# $ jetson-containers/build.sh --name=xyz pytorch jupyterlab # build container with PyTorch and JupyterLab server
# $ jetson-containers/build.sh --multiple pytorch tensorflow # build separate containers for PyTorch and
# $ jetson-containers/build.sh --multiple ros:humble* # build all ROS Humble containers (can use wildcards)
# $ jetson-containers/build.sh ros:humble-desktop pytorch # build ROS Humble with PyTorch on top
# $ jetson-containers/build.sh --base=xyz:latest pytorch # add PyTorch to an existing container
#
# Typically the jetson-containers/build.sh wrapper script is used to launch this underlying Python module. jetson-containers can also
# build external out-of-tree projects that have their own Dockerfile. And you can add your own package search dirs for other packages.
#
import os
import re
import sys
import pprint
import argparse
import traceback
from jetson_containers import (
build_container, build_containers, find_packages, package_search_dirs,
cprint, to_bool, log_config, log_error, log_status, log_versions, LogConfig
)
parser = argparse.ArgumentParser()
parser.add_argument('packages', type=str, nargs='*', default=[], help='packages or containers to build (filterable by wildcards)')
parser.add_argument('--name', type=str, default='', help="the name of the output container to build")
parser.add_argument('--base', type=str, default='', help="the base container to use at the beginning of the build chain (default: l4t-jetpack)")
parser.add_argument('--multiple', action='store_true', help="the specified packages should be built independently as opposed to chained together")
parser.add_argument('--build-flags', type=str, default='', help="extra flags to pass to 'docker build' commands")
parser.add_argument('--build-args', type=str, default='', help="container build arguments (--build-arg) as a string of comma separated key:value pairs")
parser.add_argument('--use-proxy', action='store_true', help="use the host's proxy envvars for the container build")
parser.add_argument('--package-dirs', type=str, default='', help="additional package search directories (comma or colon-separated)")
parser.add_argument('--list-packages', action='store_true', help="show the list of packages that were found under the search directories")
parser.add_argument('--show-packages', action='store_true', help="show info about one or more packages (if none are specified, all will be listed")
parser.add_argument('--skip-packages', type=str, default='', help="disable certain packages/containers (filterable by wildcards, comma/colon-separated)")
parser.add_argument('--skip-errors', action='store_true', help="continue building when errors occur (only relevant when --multiple is in use)")
parser.add_argument('--skip-tests', type=str, default='', help="comma-separated list of package tests to disable ('intermediate' to disable build-stage tests, 'all' to disable all)")
parser.add_argument('--test-only', type=str, default='', help="only test the specified packages (comma/colon-separated list)")
parser.add_argument('--simulate', action='store_true', help="print out the build commands without actually building the containers")
parser.add_argument('--push', type=str, default='', help="repo or user to push built container image to (no push by default)")
parser.add_argument('--no-github-api', action='store_true', help="disalbe Github API use to force rebuild on new git commits")
parser.add_argument('--log-dir', '--logs', type=str, default=None, help="sets the directory to save container build logs to (default: jetson-containers/logs)")
parser.add_argument('--log-level', type=str, default=None, choices=LogConfig.levels, help="sets the logging verbosity level")
parser.add_argument('--log-colors', type=to_bool, default=None, help=f"enable/disable terminal colors and formatting (defaults to true)")
parser.add_argument('--log-status', type=to_bool, default=None, help=f"enable status bar at bottom of terminal (defaults to true)")
parser.add_argument('--debug', action='store_true', help="enable debug logging")
parser.add_argument('--verbose', action='store_true', help="enable verbose logging")
parser.add_argument('--version', action='store_true', help="print platform version info and exit")
args = parser.parse_args()
# configure logging
log_config(**vars(args))
# validate args
if args.skip_errors and not args.multiple:
raise ValueError("--skip-errors can only be used with --multiple flag")
# split multi-value keyword arguments
args.package_dirs = re.split(',|;|:', args.package_dirs)
args.skip_packages = re.split(',|;|:', args.skip_packages)
args.skip_tests = re.split(',|;|:', args.skip_tests)
args.test_only = re.split(',|;|:', args.test_only)
print(f'\n{args}\n')
log_versions()
cprint(f"\n$ jetson-containers {' '.join(sys.argv[1:])}\n", attrs='bold')
if args.version:
sys.exit()
# cast build args into dictionary
if args.build_args:
try:
key_value_pairs = args.build_args.split(',')
args.build_args = {pair.split(':')[0]: pair.split(':', maxsplit=1)[1] for pair in key_value_pairs}
except(ValueError, IndexError):
raise argparse.ArgumentTypeError("Invalid dictionary format. Use key1:value1, key2:value2 ...")
else:
args.build_args = {}
# add proxy to build args if flag is set
if args.use_proxy:
proxy_vars = ['http_proxy', 'https_proxy', 'no_proxy', 'HTTP_PROXY', 'HTTPS_PROXY', 'NO_PROXY']
for var in proxy_vars:
if var in os.environ:
args.build_args[var] = os.environ[var]
# add package directories
if args.package_dirs:
package_search_dirs(args.package_dirs)
# list/show package info
if args.list_packages or args.show_packages:
packages = find_packages(args.packages, skip=args.skip_packages)
if args.list_packages:
for package in sorted(packages.keys()):
print(package)
if args.show_packages:
pprint.pprint(packages)
sys.exit(0)
try:
# build one multi-stage container from chain of packages
# or launch multiple independent container builds
if not args.multiple:
build_container(**vars(args))
else:
build_containers(**vars(args))
except Exception as error:
log_error(f"Failed building: {', '.join(args.packages)}\n\n{traceback.format_exc()}")
finally:
log_status(done=True)