2022-05-23 09:30:13 -04:00
|
|
|
# SPDX-FileCopyrightText: 2022 Espressif Systems (Shanghai) CO LTD
|
|
|
|
# SPDX-License-Identifier: Apache-2.0
|
2020-09-03 05:16:42 -04:00
|
|
|
import os
|
|
|
|
import re
|
|
|
|
import sys
|
2024-08-19 12:59:47 -04:00
|
|
|
from shutil import copy
|
|
|
|
from shutil import copytree
|
2022-06-03 08:46:56 -04:00
|
|
|
from typing import Dict
|
2020-09-03 05:16:42 -04:00
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
import click
|
|
|
|
from idf_py_actions.tools import PropertyDict
|
2020-09-03 05:16:42 -04:00
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
|
|
|
|
def get_type(action: str) -> str:
|
2021-01-25 21:49:01 -05:00
|
|
|
return action.split('-')[1]
|
2020-09-03 05:16:42 -04:00
|
|
|
|
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def replace_in_file(filename: str, pattern: str, replacement: str) -> None:
|
2020-09-03 05:16:42 -04:00
|
|
|
with open(filename, 'r+') as f:
|
|
|
|
content = f.read()
|
|
|
|
overwritten_content = re.sub(pattern, replacement, content, flags=re.M)
|
|
|
|
f.seek(0)
|
|
|
|
f.write(overwritten_content)
|
|
|
|
f.truncate()
|
|
|
|
|
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def is_empty_and_create(path: str, action: str) -> None:
|
2020-09-03 05:16:42 -04:00
|
|
|
abspath = os.path.abspath(path)
|
|
|
|
if not os.path.exists(abspath):
|
|
|
|
os.makedirs(abspath)
|
|
|
|
elif not os.path.isdir(abspath):
|
2021-01-25 21:49:01 -05:00
|
|
|
print('Your target path is not a directory. Please remove the', os.path.abspath(abspath),
|
|
|
|
'or use different target path.')
|
2020-09-03 05:16:42 -04:00
|
|
|
sys.exit(4)
|
|
|
|
elif len(os.listdir(path)) > 0:
|
2021-01-25 21:49:01 -05:00
|
|
|
print('The directory', abspath, 'is not empty. To create a', get_type(action),
|
|
|
|
'you must empty the directory or choose a different path.')
|
2020-09-03 05:16:42 -04:00
|
|
|
sys.exit(3)
|
|
|
|
|
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def create_project(target_path: str, name: str) -> None:
|
2024-08-19 12:59:47 -04:00
|
|
|
copytree(
|
2024-09-13 04:18:20 -04:00
|
|
|
os.path.join(os.environ['IDF_PATH'], 'tools', 'templates', 'sample_project'),
|
2023-07-08 14:54:42 -04:00
|
|
|
target_path,
|
2024-08-19 12:59:47 -04:00
|
|
|
copy_function=copy,
|
|
|
|
dirs_exist_ok=True,
|
2023-07-08 14:54:42 -04:00
|
|
|
)
|
2021-01-25 21:49:01 -05:00
|
|
|
main_folder = os.path.join(target_path, 'main')
|
|
|
|
os.rename(os.path.join(main_folder, 'main.c'), os.path.join(main_folder, '.'.join((name, 'c'))))
|
|
|
|
replace_in_file(os.path.join(main_folder, 'CMakeLists.txt'), 'main', name)
|
|
|
|
replace_in_file(os.path.join(target_path, 'CMakeLists.txt'), 'main', name)
|
2020-09-03 05:16:42 -04:00
|
|
|
|
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def create_component(target_path: str, name: str) -> None:
|
2024-08-19 12:59:47 -04:00
|
|
|
copytree(
|
2023-07-08 14:54:42 -04:00
|
|
|
os.path.join(os.environ['IDF_PATH'], 'tools', 'templates', 'sample_component'),
|
|
|
|
target_path,
|
2024-08-19 12:59:47 -04:00
|
|
|
copy_function=copy,
|
|
|
|
dirs_exist_ok=True,
|
2023-07-08 14:54:42 -04:00
|
|
|
)
|
2021-01-25 21:49:01 -05:00
|
|
|
os.rename(os.path.join(target_path, 'main.c'), os.path.join(target_path, '.'.join((name, 'c'))))
|
|
|
|
os.rename(os.path.join(target_path, 'include', 'main.h'),
|
|
|
|
os.path.join(target_path, 'include', '.'.join((name, 'h'))))
|
2020-09-03 05:16:42 -04:00
|
|
|
|
2021-01-25 21:49:01 -05:00
|
|
|
replace_in_file(os.path.join(target_path, '.'.join((name, 'c'))), 'main', name)
|
|
|
|
replace_in_file(os.path.join(target_path, 'CMakeLists.txt'), 'main', name)
|
2020-09-03 05:16:42 -04:00
|
|
|
|
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def action_extensions(base_actions: Dict, project_path: str) -> Dict:
|
|
|
|
def create_new(action: str, ctx: click.core.Context, global_args: PropertyDict, **action_args: str) -> Dict:
|
2020-09-03 05:16:42 -04:00
|
|
|
target_path = action_args.get('path') or os.path.join(project_path, action_args['name'])
|
|
|
|
|
|
|
|
is_empty_and_create(target_path, action)
|
|
|
|
|
2021-01-25 21:49:01 -05:00
|
|
|
func_action_map = {'create-project': create_project, 'create-component': create_component}
|
2020-09-03 05:16:42 -04:00
|
|
|
func_action_map[action](target_path, action_args['name'])
|
|
|
|
|
2021-01-25 21:49:01 -05:00
|
|
|
print('The', get_type(action), 'was created in', os.path.abspath(target_path))
|
2020-09-03 05:16:42 -04:00
|
|
|
|
|
|
|
# after the command execution, no other commands are accepted and idf.py terminates
|
|
|
|
sys.exit(0)
|
|
|
|
|
|
|
|
return {
|
2021-01-25 21:49:01 -05:00
|
|
|
'actions': {
|
|
|
|
'create-project': {
|
|
|
|
'callback': create_new,
|
|
|
|
'short_help': 'Create a new project.',
|
|
|
|
'help': ('Create a new project with the name NAME specified as argument. '
|
|
|
|
'For example: '
|
|
|
|
'`idf.py create-project new_proj` '
|
|
|
|
'will create a new project in subdirectory called `new_proj` '
|
|
|
|
'of the current working directory. '
|
2020-09-03 05:16:42 -04:00
|
|
|
"For specifying the new project's path, use either the option --path for specifying the "
|
2021-01-25 21:49:01 -05:00
|
|
|
'destination directory, or the global option -C if the project should be created as a '
|
|
|
|
'subdirectory of the specified directory. '
|
|
|
|
'If the target path does not exist it will be created. If the target folder is not empty '
|
|
|
|
'then the operation will fail with return code 3. '
|
|
|
|
'If the target path is not a folder, the script will fail with return code 4. '
|
|
|
|
'After the execution idf.py terminates '
|
|
|
|
'so this operation should be used alone.'),
|
|
|
|
'arguments': [{'names': ['name']}],
|
|
|
|
'options': [
|
2020-09-03 05:16:42 -04:00
|
|
|
{
|
2021-01-25 21:49:01 -05:00
|
|
|
'names': ['-p', '--path'],
|
|
|
|
'help': ('Set the path for the new project. The project '
|
|
|
|
'will be created directly in the given folder if it does not contain anything'),
|
2020-09-03 05:16:42 -04:00
|
|
|
},
|
|
|
|
],
|
|
|
|
|
|
|
|
},
|
2021-01-25 21:49:01 -05:00
|
|
|
'create-component': {
|
|
|
|
'callback': create_new,
|
|
|
|
'short_help': 'Create a new component.',
|
|
|
|
'help': ('Create a new component with the name NAME specified as argument. '
|
|
|
|
'For example: '
|
|
|
|
'`idf.py create-component new_comp` '
|
|
|
|
'will create a new component in subdirectory called `new_comp` '
|
|
|
|
'of the current working directory. '
|
2020-09-03 05:16:42 -04:00
|
|
|
"For specifying the new component's path use the option -C. "
|
2021-01-25 21:49:01 -05:00
|
|
|
'If the target path does not exist then it will be created. '
|
|
|
|
'If the target folder is not empty '
|
|
|
|
'then the operation will fail with return code 3. '
|
|
|
|
'If the target path is not a folder, the script will fail with return code 4. '
|
|
|
|
'After the execution idf.py terminates '
|
|
|
|
'so this operation should be used alone.'),
|
|
|
|
'arguments': [{'names': ['name']}],
|
2020-09-03 05:16:42 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|