2023-01-17 20:51:46 +01:00
|
|
|
# SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD
|
2022-05-23 15:30:13 +02:00
|
|
|
# SPDX-License-Identifier: Apache-2.0
|
2020-05-25 14:17:07 +02:00
|
|
|
import fnmatch
|
2022-06-30 12:10:41 +02:00
|
|
|
import json
|
2021-06-03 17:14:38 +02:00
|
|
|
import locale
|
2019-10-03 18:26:44 +02:00
|
|
|
import os
|
2021-06-03 17:14:38 +02:00
|
|
|
import re
|
2019-10-03 18:26:44 +02:00
|
|
|
import shutil
|
2019-11-08 16:46:02 +01:00
|
|
|
import subprocess
|
2019-10-03 18:26:44 +02:00
|
|
|
import sys
|
2022-06-03 14:46:56 +02:00
|
|
|
from typing import Any, Dict, List, Optional
|
2021-06-03 17:14:38 +02:00
|
|
|
from urllib.error import URLError
|
|
|
|
from urllib.request import Request, urlopen
|
|
|
|
from webbrowser import open_new_tab
|
2019-10-03 18:26:44 +02:00
|
|
|
|
|
|
|
import click
|
2022-06-03 14:46:56 +02:00
|
|
|
from click.core import Context
|
2021-06-03 17:14:38 +02:00
|
|
|
from idf_py_actions.constants import GENERATORS, PREVIEW_TARGETS, SUPPORTED_TARGETS, URL_TO_DOC
|
2019-10-03 18:26:44 +02:00
|
|
|
from idf_py_actions.errors import FatalError
|
|
|
|
from idf_py_actions.global_options import global_options
|
2022-08-09 15:39:23 +02:00
|
|
|
from idf_py_actions.tools import (PropertyDict, TargetChoice, ensure_build_directory, generate_hints, get_target,
|
feat(tools): esp_idf_size.ng integration
This integrates esp_idf_size.ng, refactored esp-idf-size version, into
esp-idf and enables it by default. The esp_idf_size.ng may be enabled
by using the --ng option, but also via ESP_IDF_SIZE_NG env. variable,
which is used in this integration.
New -l/--legacy option is added, which enforces usage of the old version.
This option can be also set via "ESP_IDF_SIZE_LEGACY" env. variable.
This should allow to easily switch back to old version if there is any
problem.
The new version is used by default for all formats, except for the "json".
Examples:
$ idf.py size # uses refactored version
$ idf.py size --legacy # uses legacy version
$ idf.py size --l # uses legacy version
$ idf.py size --format json # uses legacy version
$ idf.py size --format json2 # uses refactored version
$ export ESP_IDF_SIZE_LEGACY="1" # use legacy version only from now on
ESP_IDF_SIZE_FORCE_TERMINAL, which forces terminal control codes(colors), is also set
when running from idf.py, so the colors are propagated even if stdout
for esp_idf_size.ng is not attached to tty.
The same changes are applied also to the idf_size.py wrapper.
There is an import check if esp_idf_size.ng is available. If not,
we switch into the legacy mode. This should cover situation when the
esp-idf has support for refactored version, but it's not installed.
This should also allow users to bind to a legacy version(<1.0.0) and the
idf.py size and idf_size.py should still work. This also allow us to
restring the version in constraints file if we need to switch back to
legacy version globally.
Signed-off-by: Frantisek Hrbata <frantisek.hrbata@espressif.com>
2023-11-28 12:53:27 +01:00
|
|
|
idf_version, merge_action_lists, print_warning, run_target, yellow_print)
|
2019-10-03 18:26:44 +02:00
|
|
|
|
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def action_extensions(base_actions: Dict, project_path: str) -> Any:
|
|
|
|
def build_target(target_name: str, ctx: Context, args: PropertyDict) -> None:
|
2019-10-03 18:26:44 +02:00
|
|
|
"""
|
|
|
|
Execute the target build system to build target 'target_name'
|
|
|
|
|
|
|
|
Calls ensure_build_directory() which will run cmake to generate a build
|
|
|
|
directory (with the specified generator) as needed.
|
|
|
|
"""
|
|
|
|
ensure_build_directory(args, ctx.info_name)
|
2022-08-12 14:01:30 +02:00
|
|
|
run_target(target_name, args, force_progression=GENERATORS[args.generator].get('force_progression', False))
|
2019-10-03 18:26:44 +02:00
|
|
|
|
feat(tools): esp_idf_size.ng integration
This integrates esp_idf_size.ng, refactored esp-idf-size version, into
esp-idf and enables it by default. The esp_idf_size.ng may be enabled
by using the --ng option, but also via ESP_IDF_SIZE_NG env. variable,
which is used in this integration.
New -l/--legacy option is added, which enforces usage of the old version.
This option can be also set via "ESP_IDF_SIZE_LEGACY" env. variable.
This should allow to easily switch back to old version if there is any
problem.
The new version is used by default for all formats, except for the "json".
Examples:
$ idf.py size # uses refactored version
$ idf.py size --legacy # uses legacy version
$ idf.py size --l # uses legacy version
$ idf.py size --format json # uses legacy version
$ idf.py size --format json2 # uses refactored version
$ export ESP_IDF_SIZE_LEGACY="1" # use legacy version only from now on
ESP_IDF_SIZE_FORCE_TERMINAL, which forces terminal control codes(colors), is also set
when running from idf.py, so the colors are propagated even if stdout
for esp_idf_size.ng is not attached to tty.
The same changes are applied also to the idf_size.py wrapper.
There is an import check if esp_idf_size.ng is available. If not,
we switch into the legacy mode. This should cover situation when the
esp-idf has support for refactored version, but it's not installed.
This should also allow users to bind to a legacy version(<1.0.0) and the
idf.py size and idf_size.py should still work. This also allow us to
restring the version in constraints file if we need to switch back to
legacy version globally.
Signed-off-by: Frantisek Hrbata <frantisek.hrbata@espressif.com>
2023-11-28 12:53:27 +01:00
|
|
|
def size_target(target_name: str, ctx: Context, args: PropertyDict, output_format: str,
|
|
|
|
output_file: str, legacy: bool) -> None:
|
2021-08-18 15:14:47 +02:00
|
|
|
"""
|
|
|
|
Builds the app and then executes a size-related target passed in 'target_name'.
|
|
|
|
`tool_error_handler` handler is used to suppress errors during the build,
|
|
|
|
so size action can run even in case of overflow.
|
|
|
|
"""
|
|
|
|
|
2021-12-08 18:29:14 +01:00
|
|
|
def tool_error_handler(e: int, stdout: str, stderr: str) -> None:
|
2022-08-09 15:39:23 +02:00
|
|
|
for hint in generate_hints(stdout, stderr):
|
|
|
|
yellow_print(hint)
|
2021-08-18 15:14:47 +02:00
|
|
|
|
feat(tools): esp_idf_size.ng integration
This integrates esp_idf_size.ng, refactored esp-idf-size version, into
esp-idf and enables it by default. The esp_idf_size.ng may be enabled
by using the --ng option, but also via ESP_IDF_SIZE_NG env. variable,
which is used in this integration.
New -l/--legacy option is added, which enforces usage of the old version.
This option can be also set via "ESP_IDF_SIZE_LEGACY" env. variable.
This should allow to easily switch back to old version if there is any
problem.
The new version is used by default for all formats, except for the "json".
Examples:
$ idf.py size # uses refactored version
$ idf.py size --legacy # uses legacy version
$ idf.py size --l # uses legacy version
$ idf.py size --format json # uses legacy version
$ idf.py size --format json2 # uses refactored version
$ export ESP_IDF_SIZE_LEGACY="1" # use legacy version only from now on
ESP_IDF_SIZE_FORCE_TERMINAL, which forces terminal control codes(colors), is also set
when running from idf.py, so the colors are propagated even if stdout
for esp_idf_size.ng is not attached to tty.
The same changes are applied also to the idf_size.py wrapper.
There is an import check if esp_idf_size.ng is available. If not,
we switch into the legacy mode. This should cover situation when the
esp-idf has support for refactored version, but it's not installed.
This should also allow users to bind to a legacy version(<1.0.0) and the
idf.py size and idf_size.py should still work. This also allow us to
restring the version in constraints file if we need to switch back to
legacy version globally.
Signed-off-by: Frantisek Hrbata <frantisek.hrbata@espressif.com>
2023-11-28 12:53:27 +01:00
|
|
|
if not legacy and output_format != 'json':
|
|
|
|
try:
|
|
|
|
import esp_idf_size.ng # noqa: F401
|
|
|
|
except ImportError:
|
|
|
|
print_warning('WARNING: refactored esp-idf-size not installed, using legacy mode')
|
|
|
|
legacy = True
|
|
|
|
else:
|
|
|
|
# Legacy mode is used only when explicitly requested with --legacy option
|
|
|
|
# or when "--format json" option is specified. Here we enable the
|
|
|
|
# esp-idf-size refactored version with ESP_IDF_SIZE_NG env. variable.
|
|
|
|
os.environ['ESP_IDF_SIZE_NG'] = '1'
|
|
|
|
# ESP_IDF_SIZE_FORCE_TERMINAL is set to force terminal control codes even
|
|
|
|
# if stdout is not attached to terminal. This is set to pass color codes
|
|
|
|
# from esp-idf-size to idf.py.
|
|
|
|
os.environ['ESP_IDF_SIZE_FORCE_TERMINAL'] = '1'
|
|
|
|
|
|
|
|
if legacy and output_format in ['json2', 'raw', 'tree']:
|
|
|
|
# These formats are supported in new version only.
|
|
|
|
# We would get error from the esp-idf-size anyway, so print error early.
|
|
|
|
raise FatalError(f'Legacy esp-idf-size does not support {output_format} format')
|
|
|
|
|
2022-09-09 16:52:47 +02:00
|
|
|
os.environ['SIZE_OUTPUT_FORMAT'] = output_format
|
2022-09-26 20:06:00 +02:00
|
|
|
if output_file:
|
|
|
|
os.environ['SIZE_OUTPUT_FILE'] = os.path.abspath(output_file)
|
|
|
|
|
2021-08-18 15:14:47 +02:00
|
|
|
ensure_build_directory(args, ctx.info_name)
|
2021-12-08 18:29:14 +01:00
|
|
|
run_target('all', args, force_progression=GENERATORS[args.generator].get('force_progression', False),
|
2022-08-12 14:01:30 +02:00
|
|
|
custom_error_handler=tool_error_handler)
|
2021-08-18 15:14:47 +02:00
|
|
|
run_target(target_name, args)
|
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def list_build_system_targets(target_name: str, ctx: Context, args: PropertyDict) -> None:
|
2020-07-15 16:43:17 +02:00
|
|
|
"""Shows list of targets known to build sytem (make/ninja)"""
|
|
|
|
build_target('help', ctx, args)
|
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def menuconfig(target_name: str, ctx: Context, args: PropertyDict, style: str) -> None:
|
2019-11-22 15:05:46 +01:00
|
|
|
"""
|
|
|
|
Menuconfig target is build_target extended with the style argument for setting the value for the environment
|
|
|
|
variable.
|
|
|
|
"""
|
2023-06-28 10:44:27 +02:00
|
|
|
if sys.platform != 'win32':
|
|
|
|
try:
|
|
|
|
import curses # noqa: F401
|
|
|
|
except ImportError:
|
|
|
|
raise FatalError('\n'.join(
|
|
|
|
['', "menuconfig failed to import the standard Python 'curses' library.",
|
|
|
|
'Please re-run the install script which might be able to fix the issue.']))
|
2019-12-11 14:17:41 +01:00
|
|
|
if sys.version_info[0] < 3:
|
|
|
|
# The subprocess lib cannot accept environment variables as "unicode".
|
|
|
|
# This encoding step is required only in Python 2.
|
|
|
|
style = style.encode(sys.getfilesystemencoding() or 'utf-8')
|
|
|
|
os.environ['MENUCONFIG_STYLE'] = style
|
2021-12-08 18:29:14 +01:00
|
|
|
args.no_hints = True
|
2019-11-22 15:05:46 +01:00
|
|
|
build_target(target_name, ctx, args)
|
|
|
|
|
2023-10-21 12:23:07 +08:00
|
|
|
def save_defconfig(target_name: str, ctx: Context, args: PropertyDict, add_menu_labels: bool) -> None:
|
|
|
|
if add_menu_labels:
|
2023-10-04 14:56:24 +02:00
|
|
|
os.environ['ESP_IDF_KCONFIG_MIN_LABELS'] = '1'
|
|
|
|
else:
|
|
|
|
# unset variable
|
|
|
|
os.environ.pop('ESP_IDF_KCONFIG_MIN_LABELS', None)
|
|
|
|
build_target(target_name, ctx, args)
|
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def fallback_target(target_name: str, ctx: Context, args: PropertyDict) -> None:
|
2019-11-08 16:46:02 +01:00
|
|
|
"""
|
|
|
|
Execute targets that are not explicitly known to idf.py
|
|
|
|
"""
|
|
|
|
ensure_build_directory(args, ctx.info_name)
|
2019-10-03 18:26:44 +02:00
|
|
|
|
2019-11-08 16:46:02 +01:00
|
|
|
try:
|
2021-01-26 10:49:01 +08:00
|
|
|
subprocess.check_output(GENERATORS[args.generator]['dry_run'] + [target_name], cwd=args.build_dir)
|
2019-11-08 16:46:02 +01:00
|
|
|
|
|
|
|
except Exception:
|
2021-11-09 15:30:09 +08:00
|
|
|
if target_name in ['clang-check', 'clang-html-report']:
|
|
|
|
raise FatalError('command "{}" requires an additional plugin "pyclang". '
|
|
|
|
'Please install it via "pip install --upgrade pyclang"'.format(target_name))
|
|
|
|
|
2019-11-08 16:46:02 +01:00
|
|
|
raise FatalError(
|
2019-11-25 15:41:34 +01:00
|
|
|
'command "%s" is not known to idf.py and is not a %s target' % (target_name, args.generator))
|
2019-11-08 16:46:02 +01:00
|
|
|
|
|
|
|
run_target(target_name, args)
|
2019-10-03 18:26:44 +02:00
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def verbose_callback(ctx: Context, param: List, value: str) -> Optional[str]:
|
2019-10-03 18:26:44 +02:00
|
|
|
if not value or ctx.resilient_parsing:
|
2022-06-03 14:46:56 +02:00
|
|
|
return None
|
2019-10-03 18:26:44 +02:00
|
|
|
|
|
|
|
for line in ctx.command.verbose_output:
|
|
|
|
print(line)
|
|
|
|
|
2019-11-25 11:47:55 +01:00
|
|
|
return value
|
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def clean(action: str, ctx: Context, args: PropertyDict) -> None:
|
2019-10-03 18:26:44 +02:00
|
|
|
if not os.path.isdir(args.build_dir):
|
|
|
|
print("Build directory '%s' not found. Nothing to clean." % args.build_dir)
|
|
|
|
return
|
2021-01-26 10:49:01 +08:00
|
|
|
build_target('clean', ctx, args)
|
2019-10-03 18:26:44 +02:00
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def fullclean(action: str, ctx: Context, args: PropertyDict) -> None:
|
2019-10-03 18:26:44 +02:00
|
|
|
build_dir = args.build_dir
|
|
|
|
if not os.path.isdir(build_dir):
|
|
|
|
print("Build directory '%s' not found. Nothing to clean." % build_dir)
|
|
|
|
return
|
|
|
|
if len(os.listdir(build_dir)) == 0:
|
|
|
|
print("Build directory '%s' is empty. Nothing to clean." % build_dir)
|
|
|
|
return
|
|
|
|
|
2021-01-26 10:49:01 +08:00
|
|
|
if not os.path.exists(os.path.join(build_dir, 'CMakeCache.txt')):
|
2019-11-08 16:46:02 +01:00
|
|
|
raise FatalError(
|
|
|
|
"Directory '%s' doesn't seem to be a CMake build directory. Refusing to automatically "
|
|
|
|
"delete files in this directory. Delete the directory manually to 'clean' it." % build_dir)
|
2021-01-26 10:49:01 +08:00
|
|
|
red_flags = ['CMakeLists.txt', '.git', '.svn']
|
2019-10-03 18:26:44 +02:00
|
|
|
for red in red_flags:
|
|
|
|
red = os.path.join(build_dir, red)
|
|
|
|
if os.path.exists(red):
|
|
|
|
raise FatalError(
|
|
|
|
"Refusing to automatically delete files in directory containing '%s'. Delete files manually if you're sure."
|
|
|
|
% red)
|
2022-06-03 14:46:56 +02:00
|
|
|
if args.verbose and len(build_dir) > 1:
|
|
|
|
print('The following symlinks were identified and removed:\n%s' % '\n'.join(build_dir))
|
2019-10-03 18:26:44 +02:00
|
|
|
for f in os.listdir(build_dir): # TODO: once we are Python 3 only, this can be os.scandir()
|
|
|
|
f = os.path.join(build_dir, f)
|
|
|
|
if args.verbose:
|
2021-01-26 10:49:01 +08:00
|
|
|
print('Removing: %s' % f)
|
2019-10-03 18:26:44 +02:00
|
|
|
if os.path.isdir(f):
|
|
|
|
shutil.rmtree(f)
|
|
|
|
else:
|
|
|
|
os.remove(f)
|
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def python_clean(action: str, ctx: Context, args: PropertyDict) -> None:
|
2021-01-26 10:49:01 +08:00
|
|
|
for root, dirnames, filenames in os.walk(os.environ['IDF_PATH']):
|
2020-05-25 14:17:07 +02:00
|
|
|
for d in dirnames:
|
2021-01-26 10:49:01 +08:00
|
|
|
if d == '__pycache__':
|
2020-05-25 14:17:07 +02:00
|
|
|
dir_to_delete = os.path.join(root, d)
|
|
|
|
if args.verbose:
|
2021-01-26 10:49:01 +08:00
|
|
|
print('Removing: %s' % dir_to_delete)
|
2020-05-25 14:17:07 +02:00
|
|
|
shutil.rmtree(dir_to_delete)
|
|
|
|
for filename in fnmatch.filter(filenames, '*.py[co]'):
|
|
|
|
file_to_delete = os.path.join(root, filename)
|
|
|
|
if args.verbose:
|
2021-01-26 10:49:01 +08:00
|
|
|
print('Removing: %s' % file_to_delete)
|
2020-05-25 14:17:07 +02:00
|
|
|
os.remove(file_to_delete)
|
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def set_target(action: str, ctx: Context, args: PropertyDict, idf_target: str) -> None:
|
2021-01-26 10:49:01 +08:00
|
|
|
if (not args['preview'] and idf_target in PREVIEW_TARGETS):
|
2020-07-15 16:43:17 +02:00
|
|
|
raise FatalError(
|
|
|
|
"%s is still in preview. You have to append '--preview' option after idf.py to use any preview feature."
|
|
|
|
% idf_target)
|
2021-01-26 10:49:01 +08:00
|
|
|
args.define_cache_entry.append('IDF_TARGET=' + idf_target)
|
2023-03-06 13:52:29 +01:00
|
|
|
print(f'Set Target to: {idf_target}, new sdkconfig will be created.')
|
|
|
|
env = {'_IDF_PY_SET_TARGET_ACTION': '1'}
|
|
|
|
ensure_build_directory(args, ctx.info_name, True, env)
|
2019-10-03 18:26:44 +02:00
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def reconfigure(action: str, ctx: Context, args: PropertyDict) -> None:
|
2019-10-03 18:26:44 +02:00
|
|
|
ensure_build_directory(args, ctx.info_name, True)
|
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def validate_root_options(ctx: Context, args: PropertyDict, tasks: List) -> None:
|
2023-01-17 20:51:46 +01:00
|
|
|
args.project_dir = os.path.realpath(args.project_dir)
|
|
|
|
if args.build_dir is not None and args.project_dir == os.path.realpath(args.build_dir):
|
2019-11-08 16:46:02 +01:00
|
|
|
raise FatalError(
|
2021-01-26 10:49:01 +08:00
|
|
|
'Setting the build directory to the project directory is not supported. Suggest dropping '
|
2019-11-08 16:46:02 +01:00
|
|
|
"--build-dir option, the default is a 'build' subdirectory inside the project directory.")
|
2019-10-03 18:26:44 +02:00
|
|
|
if args.build_dir is None:
|
2021-01-26 10:49:01 +08:00
|
|
|
args.build_dir = os.path.join(args.project_dir, 'build')
|
2023-01-17 20:51:46 +01:00
|
|
|
args.build_dir = os.path.realpath(args.build_dir)
|
2019-10-03 18:26:44 +02:00
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def idf_version_callback(ctx: Context, param: str, value: str) -> None:
|
2019-10-03 18:26:44 +02:00
|
|
|
if not value or ctx.resilient_parsing:
|
|
|
|
return
|
|
|
|
|
|
|
|
version = idf_version()
|
|
|
|
|
|
|
|
if not version:
|
2021-01-26 10:49:01 +08:00
|
|
|
raise FatalError('ESP-IDF version cannot be determined')
|
2019-10-03 18:26:44 +02:00
|
|
|
|
2021-01-26 10:49:01 +08:00
|
|
|
print('ESP-IDF %s' % version)
|
2019-10-03 18:26:44 +02:00
|
|
|
sys.exit(0)
|
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def list_targets_callback(ctx: Context, param: List, value: int) -> None:
|
2019-12-18 15:58:19 +01:00
|
|
|
if not value or ctx.resilient_parsing:
|
|
|
|
return
|
|
|
|
|
|
|
|
for target in SUPPORTED_TARGETS:
|
|
|
|
print(target)
|
|
|
|
|
2022-06-30 12:10:41 +02:00
|
|
|
if ctx.params.get('preview'):
|
2020-01-15 21:50:19 +08:00
|
|
|
for target in PREVIEW_TARGETS:
|
|
|
|
print(target)
|
|
|
|
|
2019-12-18 15:58:19 +01:00
|
|
|
sys.exit(0)
|
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def show_docs(action: str, ctx: Context, args: PropertyDict, no_browser: bool, language: str, starting_page: str, version: str, target: str) -> None:
|
2021-06-03 17:14:38 +02:00
|
|
|
if language == 'cn':
|
|
|
|
language = 'zh_CN'
|
|
|
|
if not version:
|
|
|
|
# '0.0-dev' here because if 'dev' in version it will transform in to 'latest'
|
2022-06-03 14:46:56 +02:00
|
|
|
version_search = re.search(r'v\d+\.\d+\.?\d*(-dev|-beta\d|-rc)?', idf_version() or '0.0-dev')
|
|
|
|
version = version_search.group() if version_search else 'latest'
|
2021-06-03 17:14:38 +02:00
|
|
|
if 'dev' in version:
|
|
|
|
version = 'latest'
|
|
|
|
elif version[0] != 'v':
|
|
|
|
version = 'v' + version
|
|
|
|
target = target or get_target(args.project_dir) or 'esp32'
|
|
|
|
link = '/'.join([URL_TO_DOC, language, version, target, starting_page or ''])
|
|
|
|
redirect_link = False
|
|
|
|
try:
|
|
|
|
req = Request(link)
|
|
|
|
webpage = urlopen(req)
|
|
|
|
redirect_link = webpage.geturl().endswith('404.html')
|
|
|
|
except URLError:
|
|
|
|
print("We can't check the link's functionality because you don't have an internet connection")
|
|
|
|
if redirect_link:
|
|
|
|
print('Target', target, 'doesn\'t exist for version', version)
|
|
|
|
link = '/'.join([URL_TO_DOC, language, version, starting_page or ''])
|
|
|
|
if not no_browser:
|
|
|
|
print('Opening documentation in the default browser:')
|
|
|
|
print(link)
|
|
|
|
open_new_tab(link)
|
|
|
|
else:
|
|
|
|
print('Please open the documentation link in the browser:')
|
|
|
|
print(link)
|
|
|
|
sys.exit(0)
|
|
|
|
|
2022-06-03 14:46:56 +02:00
|
|
|
def get_default_language() -> str:
|
2021-06-03 17:14:38 +02:00
|
|
|
try:
|
|
|
|
language = 'zh_CN' if locale.getdefaultlocale()[0] == 'zh_CN' else 'en'
|
|
|
|
except ValueError:
|
|
|
|
language = 'en'
|
|
|
|
return language
|
|
|
|
|
2022-06-30 12:10:41 +02:00
|
|
|
def help_and_exit(action: str, ctx: Context, param: List, json_option: bool, add_options: bool) -> None:
|
|
|
|
if json_option:
|
|
|
|
output_dict = {}
|
|
|
|
output_dict['target'] = get_target(param.project_dir) # type: ignore
|
|
|
|
output_dict['actions'] = []
|
|
|
|
actions = ctx.to_info_dict().get('command').get('commands')
|
|
|
|
for a in actions:
|
|
|
|
action_info = {}
|
|
|
|
action_info['name'] = a
|
|
|
|
action_info['description'] = actions[a].get('help')
|
|
|
|
if add_options:
|
|
|
|
action_info['options'] = actions[a].get('params')
|
|
|
|
output_dict['actions'].append(action_info)
|
|
|
|
print(json.dumps(output_dict, sort_keys=True, indent=4))
|
|
|
|
else:
|
|
|
|
print(ctx.get_help())
|
|
|
|
ctx.exit()
|
|
|
|
|
2019-10-03 18:26:44 +02:00
|
|
|
root_options = {
|
2021-01-26 10:49:01 +08:00
|
|
|
'global_options': [
|
2019-10-03 18:26:44 +02:00
|
|
|
{
|
2021-01-26 10:49:01 +08:00
|
|
|
'names': ['--version'],
|
|
|
|
'help': 'Show IDF version and exit.',
|
|
|
|
'is_flag': True,
|
|
|
|
'expose_value': False,
|
|
|
|
'callback': idf_version_callback,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2019-12-18 15:58:19 +01:00
|
|
|
{
|
2021-01-26 10:49:01 +08:00
|
|
|
'names': ['--list-targets'],
|
|
|
|
'help': 'Print list of supported targets and exit.',
|
|
|
|
'is_flag': True,
|
|
|
|
'expose_value': False,
|
|
|
|
'callback': list_targets_callback,
|
2019-12-18 15:58:19 +01:00
|
|
|
},
|
2019-10-03 18:26:44 +02:00
|
|
|
{
|
2021-01-26 10:49:01 +08:00
|
|
|
'names': ['-C', '--project-dir'],
|
|
|
|
'scope': 'shared',
|
|
|
|
'help': 'Project directory.',
|
|
|
|
'type': click.Path(),
|
|
|
|
'default': os.getcwd(),
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
|
|
|
{
|
2021-01-26 10:49:01 +08:00
|
|
|
'names': ['-B', '--build-dir'],
|
|
|
|
'help': 'Build directory.',
|
|
|
|
'type': click.Path(),
|
|
|
|
'default': None,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
|
|
|
{
|
2021-01-26 10:49:01 +08:00
|
|
|
'names': ['-w/-n', '--cmake-warn-uninitialized/--no-warnings'],
|
|
|
|
'help': ('Enable CMake uninitialized variable warnings for CMake files inside the project directory. '
|
2020-08-25 17:04:18 +10:00
|
|
|
"(--no-warnings is now the default, and doesn't need to be specified.)"),
|
2021-01-26 10:49:01 +08:00
|
|
|
'envvar': 'IDF_CMAKE_WARN_UNINITIALIZED',
|
|
|
|
'is_flag': True,
|
|
|
|
'default': False,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
|
|
|
{
|
2021-01-26 10:49:01 +08:00
|
|
|
'names': ['-v', '--verbose'],
|
|
|
|
'help': 'Verbose build output.',
|
|
|
|
'is_flag': True,
|
|
|
|
'is_eager': True,
|
|
|
|
'default': False,
|
|
|
|
'callback': verbose_callback,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2020-01-15 21:50:19 +08:00
|
|
|
{
|
2021-01-26 10:49:01 +08:00
|
|
|
'names': ['--preview'],
|
|
|
|
'help': 'Enable IDF features that are still in preview.',
|
2022-06-30 12:10:41 +02:00
|
|
|
'is_eager': True,
|
2021-01-26 10:49:01 +08:00
|
|
|
'is_flag': True,
|
|
|
|
'default': False,
|
2020-01-15 21:50:19 +08:00
|
|
|
},
|
2019-10-03 18:26:44 +02:00
|
|
|
{
|
2021-01-26 10:49:01 +08:00
|
|
|
'names': ['--ccache/--no-ccache'],
|
|
|
|
'help': 'Use ccache in build. Disabled by default.',
|
|
|
|
'is_flag': True,
|
|
|
|
'envvar': 'IDF_CCACHE_ENABLE',
|
|
|
|
'default': False,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
|
|
|
{
|
2021-01-26 10:49:01 +08:00
|
|
|
'names': ['-G', '--generator'],
|
|
|
|
'help': 'CMake generator.',
|
|
|
|
'type': click.Choice(GENERATORS.keys()),
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
|
|
|
{
|
2021-01-26 10:49:01 +08:00
|
|
|
'names': ['--dry-run'],
|
|
|
|
'help': "Only process arguments, but don't execute actions.",
|
|
|
|
'is_flag': True,
|
|
|
|
'hidden': True,
|
|
|
|
'default': False,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-12-08 18:29:14 +01:00
|
|
|
{
|
|
|
|
'names': ['--no-hints'],
|
|
|
|
'help': 'Disable hints on how to resolve errors and logging.',
|
|
|
|
'is_flag': True,
|
|
|
|
'default': False
|
|
|
|
}
|
2019-10-03 18:26:44 +02:00
|
|
|
],
|
2021-01-26 10:49:01 +08:00
|
|
|
'global_action_callbacks': [validate_root_options],
|
2019-10-03 18:26:44 +02:00
|
|
|
}
|
|
|
|
|
2022-09-09 16:52:47 +02:00
|
|
|
# 'default' is introduced instead of simply setting 'text' as the default so that we know
|
|
|
|
# if the user explicitly specified the format or not. If the format is not specified, then
|
|
|
|
# the legacy OUTPUT_JSON CMake variable will be taken into account.
|
2022-07-14 13:39:53 +02:00
|
|
|
size_options = [{'names': ['--format', 'output_format'],
|
feat(tools): esp_idf_size.ng integration
This integrates esp_idf_size.ng, refactored esp-idf-size version, into
esp-idf and enables it by default. The esp_idf_size.ng may be enabled
by using the --ng option, but also via ESP_IDF_SIZE_NG env. variable,
which is used in this integration.
New -l/--legacy option is added, which enforces usage of the old version.
This option can be also set via "ESP_IDF_SIZE_LEGACY" env. variable.
This should allow to easily switch back to old version if there is any
problem.
The new version is used by default for all formats, except for the "json".
Examples:
$ idf.py size # uses refactored version
$ idf.py size --legacy # uses legacy version
$ idf.py size --l # uses legacy version
$ idf.py size --format json # uses legacy version
$ idf.py size --format json2 # uses refactored version
$ export ESP_IDF_SIZE_LEGACY="1" # use legacy version only from now on
ESP_IDF_SIZE_FORCE_TERMINAL, which forces terminal control codes(colors), is also set
when running from idf.py, so the colors are propagated even if stdout
for esp_idf_size.ng is not attached to tty.
The same changes are applied also to the idf_size.py wrapper.
There is an import check if esp_idf_size.ng is available. If not,
we switch into the legacy mode. This should cover situation when the
esp-idf has support for refactored version, but it's not installed.
This should also allow users to bind to a legacy version(<1.0.0) and the
idf.py size and idf_size.py should still work. This also allow us to
restring the version in constraints file if we need to switch back to
legacy version globally.
Signed-off-by: Frantisek Hrbata <frantisek.hrbata@espressif.com>
2023-11-28 12:53:27 +01:00
|
|
|
'type': click.Choice(['default', 'text', 'csv', 'json', 'json2', 'tree', 'raw']),
|
|
|
|
'help': 'Specify output format: text (same as "default"), csv, json, json2, tree or raw.',
|
2022-09-26 20:06:00 +02:00
|
|
|
'default': 'default'},
|
feat(tools): esp_idf_size.ng integration
This integrates esp_idf_size.ng, refactored esp-idf-size version, into
esp-idf and enables it by default. The esp_idf_size.ng may be enabled
by using the --ng option, but also via ESP_IDF_SIZE_NG env. variable,
which is used in this integration.
New -l/--legacy option is added, which enforces usage of the old version.
This option can be also set via "ESP_IDF_SIZE_LEGACY" env. variable.
This should allow to easily switch back to old version if there is any
problem.
The new version is used by default for all formats, except for the "json".
Examples:
$ idf.py size # uses refactored version
$ idf.py size --legacy # uses legacy version
$ idf.py size --l # uses legacy version
$ idf.py size --format json # uses legacy version
$ idf.py size --format json2 # uses refactored version
$ export ESP_IDF_SIZE_LEGACY="1" # use legacy version only from now on
ESP_IDF_SIZE_FORCE_TERMINAL, which forces terminal control codes(colors), is also set
when running from idf.py, so the colors are propagated even if stdout
for esp_idf_size.ng is not attached to tty.
The same changes are applied also to the idf_size.py wrapper.
There is an import check if esp_idf_size.ng is available. If not,
we switch into the legacy mode. This should cover situation when the
esp-idf has support for refactored version, but it's not installed.
This should also allow users to bind to a legacy version(<1.0.0) and the
idf.py size and idf_size.py should still work. This also allow us to
restring the version in constraints file if we need to switch back to
legacy version globally.
Signed-off-by: Frantisek Hrbata <frantisek.hrbata@espressif.com>
2023-11-28 12:53:27 +01:00
|
|
|
{'names': ['--legacy', '-l'],
|
|
|
|
'is_flag': True,
|
|
|
|
'default': os.environ.get('ESP_IDF_SIZE_LEGACY', '0') == '1',
|
|
|
|
'help': 'Use legacy esp-idf-size version'},
|
2022-09-26 20:06:00 +02:00
|
|
|
{'names': ['--output-file', 'output_file'],
|
|
|
|
'help': 'Print output to the specified file instead of to the standard output'}]
|
2022-07-14 13:39:53 +02:00
|
|
|
|
2019-10-03 18:26:44 +02:00
|
|
|
build_actions = {
|
2021-01-26 10:49:01 +08:00
|
|
|
'actions': {
|
|
|
|
'all': {
|
|
|
|
'aliases': ['build'],
|
|
|
|
'callback': build_target,
|
|
|
|
'short_help': 'Build the project.',
|
|
|
|
'help': (
|
|
|
|
'Build the project. This can involve multiple steps:\n\n'
|
|
|
|
'1. Create the build directory if needed. '
|
2019-11-08 16:46:02 +01:00
|
|
|
"The sub-directory 'build' is used to hold build output, "
|
2021-01-26 10:49:01 +08:00
|
|
|
'although this can be changed with the -B option.\n\n'
|
|
|
|
'2. Run CMake as necessary to configure the project '
|
|
|
|
'and generate build files for the main build tool.\n\n'
|
|
|
|
'3. Run the main build tool (Ninja or GNU Make). '
|
|
|
|
'By default, the build tool is automatically detected '
|
|
|
|
'but it can be explicitly set by passing the -G option to idf.py.\n\n'),
|
|
|
|
'options': global_options,
|
|
|
|
'order_dependencies': [
|
|
|
|
'reconfigure',
|
|
|
|
'menuconfig',
|
|
|
|
'clean',
|
|
|
|
'fullclean',
|
2019-10-03 18:26:44 +02:00
|
|
|
],
|
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'menuconfig': {
|
|
|
|
'callback': menuconfig,
|
|
|
|
'help': 'Run "menuconfig" project configuration tool.',
|
|
|
|
'options': global_options + [
|
2019-11-22 15:05:46 +01:00
|
|
|
{
|
2021-01-26 10:49:01 +08:00
|
|
|
'names': ['--style', '--color-scheme', 'style'],
|
|
|
|
'help': (
|
|
|
|
'Menuconfig style.\n'
|
|
|
|
'The built-in styles include:\n\n'
|
|
|
|
'- default - a yellowish theme,\n\n'
|
|
|
|
'- monochrome - a black and white theme, or\n\n'
|
|
|
|
'- aquatic - a blue theme.\n\n'
|
|
|
|
'It is possible to customize these themes further'
|
|
|
|
' as it is described in the Color schemes section of the kconfiglib documentation.\n'
|
2020-06-26 17:18:25 +02:00
|
|
|
'The default value is \"aquatic\".'),
|
2021-01-26 10:49:01 +08:00
|
|
|
'envvar': 'MENUCONFIG_STYLE',
|
|
|
|
'default': 'aquatic',
|
2019-12-18 15:58:19 +01:00
|
|
|
}
|
|
|
|
],
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'confserver': {
|
|
|
|
'callback': build_target,
|
|
|
|
'help': 'Run JSON configuration server.',
|
|
|
|
'options': global_options,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'size': {
|
2021-08-18 15:14:47 +02:00
|
|
|
'callback': size_target,
|
2021-01-26 10:49:01 +08:00
|
|
|
'help': 'Print basic size information about the app.',
|
2022-07-14 13:39:53 +02:00
|
|
|
'options': global_options + size_options,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'size-components': {
|
2021-08-18 15:14:47 +02:00
|
|
|
'callback': size_target,
|
2021-01-26 10:49:01 +08:00
|
|
|
'help': 'Print per-component size information.',
|
2022-07-14 13:39:53 +02:00
|
|
|
'options': global_options + size_options,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'size-files': {
|
2021-08-18 15:14:47 +02:00
|
|
|
'callback': size_target,
|
2021-01-26 10:49:01 +08:00
|
|
|
'help': 'Print per-source-file size information.',
|
2022-07-14 13:39:53 +02:00
|
|
|
'options': global_options + size_options,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'bootloader': {
|
|
|
|
'callback': build_target,
|
|
|
|
'help': 'Build only bootloader.',
|
|
|
|
'options': global_options,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'app': {
|
|
|
|
'callback': build_target,
|
|
|
|
'help': 'Build only the app.',
|
|
|
|
'order_dependencies': ['clean', 'fullclean', 'reconfigure'],
|
|
|
|
'options': global_options,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-08-11 15:24:17 +02:00
|
|
|
'efuse-common-table': {
|
|
|
|
'callback': build_target,
|
2022-03-18 12:09:14 +01:00
|
|
|
'help': 'Generate C-source for IDF\'s eFuse fields.',
|
2021-01-26 10:49:01 +08:00
|
|
|
'order_dependencies': ['reconfigure'],
|
|
|
|
'options': global_options,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-08-11 15:24:17 +02:00
|
|
|
'efuse-custom-table': {
|
|
|
|
'callback': build_target,
|
|
|
|
'help': 'Generate C-source for user\'s eFuse fields.',
|
|
|
|
'order_dependencies': ['reconfigure'],
|
|
|
|
'options': global_options,
|
|
|
|
},
|
|
|
|
'show-efuse-table': {
|
|
|
|
'callback': build_target,
|
2021-01-26 10:49:01 +08:00
|
|
|
'help': 'Print eFuse table.',
|
|
|
|
'order_dependencies': ['reconfigure'],
|
|
|
|
'options': global_options,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-08-11 15:24:17 +02:00
|
|
|
'partition-table': {
|
|
|
|
'callback': build_target,
|
2021-01-26 10:49:01 +08:00
|
|
|
'help': 'Build only partition table.',
|
|
|
|
'order_dependencies': ['reconfigure'],
|
|
|
|
'options': global_options,
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'build-system-targets': {
|
|
|
|
'callback': list_build_system_targets,
|
|
|
|
'help': 'Print list of build system targets.',
|
2020-07-15 16:43:17 +02:00
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'fallback': {
|
|
|
|
'callback': fallback_target,
|
|
|
|
'help': 'Handle for targets not known for idf.py.',
|
|
|
|
'hidden': True,
|
2021-06-03 17:14:38 +02:00
|
|
|
},
|
|
|
|
'docs': {
|
|
|
|
'callback': show_docs,
|
|
|
|
'help': 'Open web browser with documentation for ESP-IDF',
|
|
|
|
'options': [
|
|
|
|
{
|
|
|
|
'names': ['--no-browser', '-nb'],
|
|
|
|
'is_flag': True,
|
|
|
|
'help': 'Don\'t open browser.'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
'names': ['--language', '-l'],
|
|
|
|
'default': get_default_language(),
|
|
|
|
'type': click.Choice(['en', 'zh_CN', 'cn']),
|
|
|
|
'help': 'Documentation language. Your system language by default (en or cn)'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
'names': ['--starting-page', '-sp'],
|
|
|
|
'help': 'Documentation page (get-started, api-reference etc).'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
'names': ['--version', '-v'],
|
|
|
|
'help': 'Version of ESP-IDF.'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
'names': ['--target', '-t'],
|
|
|
|
'type': TargetChoice(SUPPORTED_TARGETS + PREVIEW_TARGETS + ['']),
|
|
|
|
'help': 'Chip target.'
|
|
|
|
}
|
|
|
|
]
|
2021-12-14 15:31:08 +01:00
|
|
|
},
|
|
|
|
'save-defconfig': {
|
2023-10-04 14:56:24 +02:00
|
|
|
'callback': save_defconfig,
|
2021-12-14 15:31:08 +01:00
|
|
|
'help': 'Generate a sdkconfig.defaults with options different from the default ones',
|
2023-10-04 14:56:24 +02:00
|
|
|
'options': global_options + [{
|
|
|
|
'names': ['--add-menu-labels'],
|
|
|
|
'is_flag': True,
|
|
|
|
'help': 'Add menu labels to minimal config.',
|
|
|
|
}]
|
2019-11-08 16:46:02 +01:00
|
|
|
}
|
2019-10-03 18:26:44 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
clean_actions = {
|
2021-01-26 10:49:01 +08:00
|
|
|
'actions': {
|
|
|
|
'reconfigure': {
|
|
|
|
'callback': reconfigure,
|
|
|
|
'short_help': 'Re-run CMake.',
|
|
|
|
'help': (
|
2019-11-08 16:46:02 +01:00
|
|
|
"Re-run CMake even if it doesn't seem to need re-running. "
|
|
|
|
"This isn't necessary during normal usage, "
|
2021-01-26 10:49:01 +08:00
|
|
|
'but can be useful after adding/removing files from the source tree, '
|
|
|
|
'or when modifying CMake cache variables. '
|
2019-11-08 16:46:02 +01:00
|
|
|
"For example, \"idf.py -DNAME='VALUE' reconfigure\" "
|
|
|
|
'can be used to set variable "NAME" in CMake cache to value "VALUE".'),
|
2021-01-26 10:49:01 +08:00
|
|
|
'options': global_options,
|
|
|
|
'order_dependencies': ['menuconfig', 'fullclean'],
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'set-target': {
|
|
|
|
'callback': set_target,
|
|
|
|
'short_help': 'Set the chip target to build.',
|
|
|
|
'help': (
|
|
|
|
'Set the chip target to build. This will remove the '
|
|
|
|
'existing sdkconfig file and corresponding CMakeCache and '
|
|
|
|
'create new ones according to the new target.\nFor example, '
|
2019-11-08 16:46:02 +01:00
|
|
|
"\"idf.py set-target esp32\" will select esp32 as the new chip "
|
2021-01-26 10:49:01 +08:00
|
|
|
'target.'),
|
|
|
|
'arguments': [
|
2019-10-03 18:26:44 +02:00
|
|
|
{
|
2021-01-26 10:49:01 +08:00
|
|
|
'names': ['idf-target'],
|
|
|
|
'nargs': 1,
|
|
|
|
'type': TargetChoice(SUPPORTED_TARGETS + PREVIEW_TARGETS),
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
|
|
|
],
|
2021-01-26 10:49:01 +08:00
|
|
|
'dependencies': ['fullclean'],
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'clean': {
|
|
|
|
'callback': clean,
|
|
|
|
'short_help': 'Delete build output files from the build directory.',
|
|
|
|
'help': (
|
|
|
|
'Delete build output files from the build directory, '
|
2019-11-08 16:46:02 +01:00
|
|
|
"forcing a 'full rebuild' the next time "
|
|
|
|
"the project is built. Cleaning doesn't delete "
|
2021-01-26 10:49:01 +08:00
|
|
|
'CMake configuration output and some other files'),
|
|
|
|
'order_dependencies': ['fullclean'],
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'fullclean': {
|
|
|
|
'callback': fullclean,
|
|
|
|
'short_help': 'Delete the entire build directory contents.',
|
|
|
|
'help': (
|
|
|
|
'Delete the entire build directory contents. '
|
|
|
|
'This includes all CMake configuration output.'
|
|
|
|
'The next time the project is built, '
|
|
|
|
'CMake will configure it from scratch. '
|
|
|
|
'Note that this option recursively deletes all files '
|
|
|
|
'in the build directory, so use with care.'
|
|
|
|
'Project configuration is not deleted.')
|
2019-10-03 18:26:44 +02:00
|
|
|
},
|
2021-01-26 10:49:01 +08:00
|
|
|
'python-clean': {
|
|
|
|
'callback': python_clean,
|
|
|
|
'short_help': 'Delete generated Python byte code from the IDF directory',
|
|
|
|
'help': (
|
|
|
|
'Delete generated Python byte code from the IDF directory '
|
|
|
|
'which may cause issues when switching between IDF and Python versions. '
|
|
|
|
'It is advised to run this target after switching versions.')
|
2020-05-25 14:17:07 +02:00
|
|
|
},
|
2019-10-03 18:26:44 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-30 12:10:41 +02:00
|
|
|
help_action = {
|
|
|
|
'actions': {
|
|
|
|
'help': {
|
|
|
|
'callback': help_and_exit,
|
|
|
|
'help': 'Show help message and exit.',
|
|
|
|
'hidden': True,
|
|
|
|
'options': [
|
|
|
|
{
|
|
|
|
'names': ['--json', 'json_option'],
|
|
|
|
'is_flag': True,
|
|
|
|
'help': 'Print out actions in machine-readable format for selected target.'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
'names': ['--add-options'],
|
|
|
|
'is_flag': True,
|
|
|
|
'help': 'Add options about actions to machine-readable format.'
|
|
|
|
}
|
|
|
|
],
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return merge_action_lists(root_options, build_actions, clean_actions, help_action)
|