mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
Merge branch 'ci/add_pre_commit_for_some_pre_check_jobs' into 'master'
CI: Add pre-commit for IDF project See merge request espressif/esp-idf!10682
This commit is contained in:
commit
256333ca13
@ -48,6 +48,7 @@
|
||||
|
||||
/.* @esp-idf-codeowners/tools
|
||||
/.gitlab-ci.yml @esp-idf-codeowners/ci
|
||||
/.pre-commit-config.yaml @esp-idf-codeowners/ci
|
||||
/.readthedocs.yml @esp-idf-codeowners/docs
|
||||
/CMakeLists.txt @esp-idf-codeowners/build-config
|
||||
/Kconfig @esp-idf-codeowners/build-config
|
||||
|
58
.pre-commit-config.yaml
Normal file
58
.pre-commit-config.yaml
Normal file
@ -0,0 +1,58 @@
|
||||
# See https://pre-commit.com for more information
|
||||
# See https://pre-commit.com/hooks.html for more hooks
|
||||
|
||||
repos:
|
||||
- repo: https://github.com/pre-commit/pre-commit-hooks
|
||||
rev: v3.3.0
|
||||
hooks:
|
||||
- id: trailing-whitespace
|
||||
exclude: '.+\.(md|rst)'
|
||||
- id: end-of-file-fixer
|
||||
- id: check-executables-have-shebangs
|
||||
- id: file-contents-sorter
|
||||
files: 'tools/ci/executable-list.txt'
|
||||
- id: mixed-line-ending
|
||||
args: ['-f=lf']
|
||||
- repo: https://gitlab.com/pycqa/flake8
|
||||
rev: 3.8.4
|
||||
hooks:
|
||||
- id: flake8
|
||||
args: ['--config=.flake8', '--tee', '--benchmark']
|
||||
- repo: local
|
||||
hooks:
|
||||
- id: check-executables
|
||||
name: Check File Permissions
|
||||
entry: tools/ci/check_executables.py --action executables
|
||||
language: python
|
||||
types: [executable]
|
||||
exclude: '\.pre-commit/.+'
|
||||
- id: check-executable-list
|
||||
name: Validate executable-list.txt
|
||||
entry: tools/ci/check_executables.py --action list
|
||||
language: python
|
||||
pass_filenames: false
|
||||
always_run: true
|
||||
- id: check-kconfigs
|
||||
name: Validate Kconfig files
|
||||
entry: tools/ci/check_kconfigs.py
|
||||
language: python
|
||||
files: '^Kconfig$|Kconfig.*$'
|
||||
- id: check-deprecated-kconfigs-options
|
||||
name: Check if any Kconfig Options Deprecated
|
||||
entry: tools/ci/check_deprecated_kconfigs.py
|
||||
language: python
|
||||
files: 'sdkconfig\.ci$|sdkconfig\.rename$|sdkconfig.*$'
|
||||
- id: cmake-lint
|
||||
name: Check CMake Files Format
|
||||
entry: cmakelint --linelength=120 --spaces=4
|
||||
language: python
|
||||
additional_dependencies:
|
||||
- https://github.com/richq/cmake-lint/archive/058c6c0ed2536.zip
|
||||
files: 'CMakeLists.txt$|\.cmake$'
|
||||
exclude: '\/third_party\/'
|
||||
- id: check-codeowners
|
||||
name: Validate Codeowner File
|
||||
entry: tools/ci/check_codeowners.py ci-check
|
||||
language: python
|
||||
files: '\.gitlab/CODEOWNERS'
|
||||
pass_filenames: false
|
@ -17,6 +17,8 @@ Before sending us a Pull Request, please consider this list of points:
|
||||
|
||||
* Does any new code conform to the esp-idf :doc:`Style Guide <style-guide>`?
|
||||
|
||||
* Have you installed the :doc:`pre-commit hook <install-pre-commit-hook>` for esp-idf project?
|
||||
|
||||
* Does the code documentation follow requirements in :doc:`documenting-code`?
|
||||
|
||||
* Is the code adequately commented for people to understand how it is structured?
|
||||
@ -52,6 +54,7 @@ Related Documents
|
||||
:maxdepth: 1
|
||||
|
||||
style-guide
|
||||
install-pre-commit-hook
|
||||
documenting-code
|
||||
add-ons-reference
|
||||
creating-examples
|
||||
|
41
docs/en/contribute/install-pre-commit-hook.rst
Normal file
41
docs/en/contribute/install-pre-commit-hook.rst
Normal file
@ -0,0 +1,41 @@
|
||||
Install pre-commit Hook for ESP-IDF Project
|
||||
===========================================
|
||||
|
||||
Required Dependency
|
||||
-------------------
|
||||
|
||||
Python 3.6.1 or above. This is our recommendation python version for IDF developers.
|
||||
|
||||
If you still have versions not compatible, please do not install pre-commit hook and update your python versions.
|
||||
|
||||
Install pre-commit
|
||||
------------------
|
||||
|
||||
Run ``pip install pre-commit``
|
||||
|
||||
Install pre-commit hook
|
||||
-----------------------
|
||||
|
||||
1. Go to the IDF Project Directory
|
||||
|
||||
2. Run ``pre-commit install --allow-missing-config``. Install hook by this approach will let you commit successfully even in branches without the ``.pre-commit-config.yaml``
|
||||
|
||||
3. pre-commit hook will run automatically when you're running ``git commit`` command
|
||||
|
||||
What's More?
|
||||
------------
|
||||
|
||||
For detailed usage, Please refer to the documentation of pre-commit_.
|
||||
|
||||
.. _pre-commit: http://www.pre-commit.com/
|
||||
|
||||
Common Problems For Windows Users
|
||||
---------------------------------
|
||||
|
||||
1. ``/usr/bin/env: python: Permission denied.``
|
||||
|
||||
If you're in Git Bash or MSYS terminal, please check the python executable location by run ``which python``.
|
||||
|
||||
If the executable is under ``~/AppData/Local/Microsoft/WindowsApps/``, then it's a link to Windows AppStore, not a real one.
|
||||
|
||||
Please install python manually and update this in your ``PATH`` environment variable.
|
1
docs/zh_CN/contribute/install-pre-commit-hook.rst
Normal file
1
docs/zh_CN/contribute/install-pre-commit-hook.rst
Normal file
@ -0,0 +1 @@
|
||||
.. include:: ../../en/contribute/install-pre-commit-hook.rst
|
@ -1,66 +0,0 @@
|
||||
#!/usr/bin/env bash
|
||||
# This script finds executable files in the repository, excluding some directories,
|
||||
# then prints the list of all files which are not in executable-list.txt.
|
||||
# Returns with error if this list is non-empty.
|
||||
# Also checks if executable-list.txt is sorted and has no duplicates.
|
||||
|
||||
set -o errexit # Exit if command failed.
|
||||
set -o pipefail # Exit if pipe failed.
|
||||
set -o nounset # Exit if variable not set.
|
||||
|
||||
|
||||
cd $IDF_PATH
|
||||
|
||||
in_list=tools/ci/executable-list.txt
|
||||
tmp_list=$(mktemp)
|
||||
out_list=$(mktemp)
|
||||
|
||||
# build exclude pattern like '-o -path ./components/component/submodule' for each submodule
|
||||
submodule_excludes=$(git config --file .gitmodules --get-regexp path | awk '{ print "-o -path ./" $2 }')
|
||||
|
||||
# figure out which flag to use when searching for executable files
|
||||
if [ "$(uname -s)" == "Darwin" ]; then
|
||||
perm_flag="-perm +111"
|
||||
else
|
||||
perm_flag="-executable"
|
||||
fi
|
||||
|
||||
find . -type d \( \
|
||||
-path ./.git \
|
||||
-o -name build \
|
||||
-o -name builds \
|
||||
$submodule_excludes \
|
||||
\) -prune -o -type f $perm_flag -print \
|
||||
| sed "s|^\./||" > $tmp_list
|
||||
|
||||
# this looks for lines present in tmp_list but not in executable-list.txt
|
||||
comm -13 <(cat $in_list | sed -n "/^#/!p" | sort) <(sort $tmp_list) > $out_list
|
||||
|
||||
ret=0
|
||||
if [ -s $out_list ]; then
|
||||
ret=1
|
||||
echo "Error: the following file(s) have executable flag set:"
|
||||
echo ""
|
||||
cat $out_list
|
||||
echo ""
|
||||
echo "If any files need to be executable (usually, scripts), add them to tools/ci/executable-list.txt"
|
||||
echo "Make the rest of the files non-executable using 'chmod -x <filename>'."
|
||||
echo "On Windows, use 'git update-index --chmod=-x filename' instead."
|
||||
echo ""
|
||||
fi
|
||||
|
||||
if ! diff <(cat $in_list | sed -n "/^#/!p" | sort | uniq) $in_list; then
|
||||
echo "$in_list is not sorted or has duplicate entries"
|
||||
ret=2
|
||||
fi
|
||||
|
||||
for filename in $(cat $in_list | sed -n "/^#/!p"); do
|
||||
if [ ! -f "$filename" ]; then
|
||||
echo "Warning: file '$filename' is present in '$in_list', but does not exist"
|
||||
fi
|
||||
done
|
||||
|
||||
rm $tmp_list
|
||||
rm $out_list
|
||||
|
||||
exit $ret
|
@ -1,14 +0,0 @@
|
||||
#!/bin/sh
|
||||
|
||||
if ! [ -z "$1" ]; then
|
||||
cd "$1"
|
||||
fi
|
||||
|
||||
echo "Checking for Windows line endings in `pwd`"
|
||||
|
||||
if git ls-tree --name-only -r HEAD | xargs file -N | grep CRLF; then
|
||||
echo "Some files have CRLF (Windows-style) line endings. Please convert to LF (Unix-style). git can be configured to do this automatically."
|
||||
exit 1
|
||||
fi
|
||||
|
||||
exit 0
|
@ -22,8 +22,9 @@ import re
|
||||
import subprocess
|
||||
import sys
|
||||
|
||||
from idf_ci_utils import IDF_PATH
|
||||
|
||||
CODEOWNERS_PATH = os.path.join(os.path.dirname(__file__), "..", ".gitlab", "CODEOWNERS")
|
||||
CODEOWNERS_PATH = os.path.join(IDF_PATH, ".gitlab", "CODEOWNERS")
|
||||
CODEOWNER_GROUP_PREFIX = "@esp-idf-codeowners/"
|
||||
|
||||
|
||||
@ -31,9 +32,8 @@ def get_all_files():
|
||||
"""
|
||||
Get list of all file paths in the repository.
|
||||
"""
|
||||
idf_root = os.path.join(os.path.dirname(__file__), "..")
|
||||
# only split on newlines, since file names may contain spaces
|
||||
return subprocess.check_output(["git", "ls-files"], cwd=idf_root).decode("utf-8").strip().split('\n')
|
||||
return subprocess.check_output(["git", "ls-files"], cwd=IDF_PATH).decode("utf-8").strip().split('\n')
|
||||
|
||||
|
||||
def pattern_to_regex(pattern):
|
||||
@ -121,7 +121,7 @@ def action_ci_check(args):
|
||||
errors = []
|
||||
|
||||
def add_error(msg):
|
||||
errors.append("Error at CODEOWNERS:{}: {}".format(line_no, msg))
|
||||
errors.append("{}:{}: {}".format(CODEOWNERS_PATH, line_no, msg))
|
||||
|
||||
all_files = get_all_files()
|
||||
prev_path_pattern = ""
|
@ -16,11 +16,15 @@
|
||||
|
||||
from __future__ import print_function
|
||||
from __future__ import unicode_literals
|
||||
|
||||
import argparse
|
||||
import os
|
||||
import sys
|
||||
from io import open
|
||||
|
||||
from check_kconfigs import valid_directory
|
||||
from idf_ci_utils import get_submodule_dirs
|
||||
|
||||
# FILES_TO_CHECK used as "startswith" pattern to match sdkconfig.defaults variants
|
||||
FILES_TO_CHECK = ('sdkconfig.ci', 'sdkconfig.defaults')
|
||||
|
||||
@ -48,52 +52,67 @@ def _valid_directory(path):
|
||||
|
||||
|
||||
def main():
|
||||
default_path = os.getenv('IDF_PATH', None)
|
||||
|
||||
parser = argparse.ArgumentParser(description='Kconfig options checker')
|
||||
parser.add_argument('--directory', '-d', help='Path to directory to check recursively '
|
||||
'(for example $IDF_PATH)',
|
||||
type=_valid_directory,
|
||||
required=default_path is None,
|
||||
default=default_path)
|
||||
parser.add_argument('files', nargs='*',
|
||||
help='Kconfig files')
|
||||
parser.add_argument('--includes', '-d', nargs='*',
|
||||
help='Extra paths for recursively searching Kconfig files. (for example $IDF_PATH)',
|
||||
type=valid_directory)
|
||||
parser.add_argument('--exclude-submodules', action='store_true',
|
||||
help='Exclude submodules')
|
||||
args = parser.parse_args()
|
||||
|
||||
# IGNORE_DIRS makes sense when the required directory is IDF_PATH
|
||||
check_ignore_dirs = default_path is not None and os.path.abspath(args.directory) == os.path.abspath(default_path)
|
||||
success_counter = 0
|
||||
failure_counter = 0
|
||||
ignore_counter = 0
|
||||
|
||||
ignores = 0
|
||||
files_to_check = []
|
||||
deprecated_options = set()
|
||||
errors = []
|
||||
|
||||
for root, dirnames, filenames in os.walk(args.directory):
|
||||
for filename in filenames:
|
||||
full_path = os.path.join(root, filename)
|
||||
path_in_idf = os.path.relpath(full_path, args.directory)
|
||||
ignore_dirs = IGNORE_DIRS
|
||||
if args.exclude_submodules:
|
||||
for submodule in get_submodule_dirs(full_path=True):
|
||||
ignore_dirs = ignore_dirs + tuple(submodule)
|
||||
|
||||
if filename.startswith(FILES_TO_CHECK):
|
||||
if check_ignore_dirs and path_in_idf.startswith(IGNORE_DIRS):
|
||||
print('{}: Ignored'.format(path_in_idf))
|
||||
ignores += 1
|
||||
continue
|
||||
files_to_check.append(full_path)
|
||||
elif filename == 'sdkconfig.rename':
|
||||
deprecated_options |= _parse_path(full_path)
|
||||
files = [os.path.abspath(file_path) for file_path in args.files]
|
||||
|
||||
for path in files_to_check:
|
||||
used_options = _parse_path(path, '=')
|
||||
if args.includes:
|
||||
for directory in args.includes:
|
||||
for root, dirnames, filenames in os.walk(directory):
|
||||
for filename in filenames:
|
||||
full_path = os.path.join(root, filename)
|
||||
if filename.startswith(FILES_TO_CHECK):
|
||||
files.append(full_path)
|
||||
elif filename == 'sdkconfig.rename':
|
||||
deprecated_options |= _parse_path(full_path)
|
||||
|
||||
for full_path in files:
|
||||
if full_path.startswith(ignore_dirs):
|
||||
print('{}: Ignored'.format(full_path))
|
||||
ignore_counter += 1
|
||||
continue
|
||||
used_options = _parse_path(full_path, '=')
|
||||
used_deprecated_options = deprecated_options & used_options
|
||||
if len(used_deprecated_options) > 0:
|
||||
errors.append('{}: The following options are deprecated: {}'.format(path,
|
||||
', '.join(used_deprecated_options)))
|
||||
print('{}: The following options are deprecated: {}'
|
||||
.format(full_path, ', '.join(used_deprecated_options)))
|
||||
failure_counter += 1
|
||||
else:
|
||||
print('{}: OK'.format(full_path))
|
||||
success_counter += 1
|
||||
|
||||
if ignores > 0:
|
||||
print('{} files have been ignored.'.format(ignores))
|
||||
if ignore_counter > 0:
|
||||
print('{} files have been ignored.'.format(ignore_counter))
|
||||
if success_counter > 0:
|
||||
print('{} files have been successfully checked.'.format(success_counter))
|
||||
if failure_counter > 0:
|
||||
print('{} files have errors. Please take a look at the log.'.format(failure_counter))
|
||||
return 1
|
||||
|
||||
if len(errors) > 0:
|
||||
print('\n\n'.join(errors))
|
||||
sys.exit(1)
|
||||
if not files:
|
||||
print('WARNING: no files specified. Please specify files or use '
|
||||
'"--includes" to search Kconfig files recursively')
|
||||
return 0
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
||||
sys.exit(main())
|
||||
|
72
tools/ci/check_executables.py
Executable file
72
tools/ci/check_executables.py
Executable file
@ -0,0 +1,72 @@
|
||||
#!/usr/bin/env python
|
||||
#
|
||||
# Copyright 2020 Espressif Systems (Shanghai) PTE LTD
|
||||
#
|
||||
# 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 argparse
|
||||
import os
|
||||
from sys import exit
|
||||
|
||||
|
||||
def _strip_each_item(iterable):
|
||||
res = []
|
||||
for item in iterable:
|
||||
if item:
|
||||
res.append(item.strip())
|
||||
return res
|
||||
|
||||
|
||||
IDF_PATH = os.getenv('IDF_PATH', os.getcwd())
|
||||
EXECUTABLE_LIST_FN = os.path.join(IDF_PATH, 'tools/ci/executable-list.txt')
|
||||
known_executables = _strip_each_item(open(EXECUTABLE_LIST_FN).readlines())
|
||||
|
||||
|
||||
def check_executable_list():
|
||||
ret = 0
|
||||
for index, fn in enumerate(known_executables):
|
||||
if not os.path.exists(os.path.join(IDF_PATH, fn)):
|
||||
print('{}:{} {} not exists. Please remove it manually'.format(EXECUTABLE_LIST_FN, index + 1, fn))
|
||||
ret = 1
|
||||
return ret
|
||||
|
||||
|
||||
def check_executables(files):
|
||||
ret = 0
|
||||
for fn in files:
|
||||
if fn not in known_executables:
|
||||
print('"{}" is not in {}'.format(fn, EXECUTABLE_LIST_FN))
|
||||
ret = 1
|
||||
return ret
|
||||
|
||||
|
||||
def main():
|
||||
parser = argparse.ArgumentParser()
|
||||
parser.add_argument('--action', choices=['executables', 'list'], required=True,
|
||||
help='if "executables", pass all your executables to see if it\'s in the list.'
|
||||
'if "list", check if all items on your list exist')
|
||||
parser.add_argument('filenames', nargs='*', help='Filenames to check.')
|
||||
args = parser.parse_args()
|
||||
|
||||
if args.action == 'executables':
|
||||
ret = check_executables(args.filenames)
|
||||
elif args.action == 'list':
|
||||
ret = check_executable_list()
|
||||
else:
|
||||
raise ValueError
|
||||
|
||||
return ret
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
exit(main())
|
@ -16,12 +16,15 @@
|
||||
|
||||
from __future__ import print_function
|
||||
from __future__ import unicode_literals
|
||||
import os
|
||||
import sys
|
||||
import re
|
||||
|
||||
import argparse
|
||||
import os
|
||||
import re
|
||||
import sys
|
||||
from io import open
|
||||
|
||||
from idf_ci_utils import get_submodule_dirs, IDF_PATH
|
||||
|
||||
# regular expression for matching Kconfig files
|
||||
RE_KCONFIG = r'^Kconfig(\.projbuild)?(\.in)?$'
|
||||
|
||||
@ -33,10 +36,10 @@ OUTPUT_SUFFIX = '.new'
|
||||
# accepts tuples but no lists.
|
||||
IGNORE_DIRS = (
|
||||
# Kconfigs from submodules need to be ignored:
|
||||
os.path.join('components', 'mqtt', 'esp-mqtt'),
|
||||
os.path.join(IDF_PATH, 'components', 'mqtt', 'esp-mqtt'),
|
||||
# Test Kconfigs are also ignored
|
||||
os.path.join('tools', 'ldgen', 'test', 'data'),
|
||||
os.path.join('tools', 'kconfig_new', 'test'),
|
||||
os.path.join(IDF_PATH, 'tools', 'ldgen', 'test', 'data'),
|
||||
os.path.join(IDF_PATH, 'tools', 'kconfig_new', 'test'),
|
||||
)
|
||||
|
||||
SPACES_PER_INDENT = 4
|
||||
@ -282,8 +285,8 @@ class IndentAndNameChecker(BaseChecker):
|
||||
common_prefix_len = len(common_prefix)
|
||||
if common_prefix_len < self.min_prefix_length:
|
||||
raise InputError(self.path_in_idf, line_number,
|
||||
'The common prefix for the config names of the menu ending at this line is "{}". '
|
||||
'All config names in this menu should start with the same prefix of {} characters '
|
||||
'The common prefix for the config names of the menu ending at this line is "{}".\n'
|
||||
'\tAll config names in this menu should start with the same prefix of {} characters '
|
||||
'or more.'.format(common_prefix, self.min_prefix_length),
|
||||
line) # no suggested correction for this
|
||||
if len(self.prefix_stack) > 0:
|
||||
@ -370,80 +373,107 @@ def valid_directory(path):
|
||||
return path
|
||||
|
||||
|
||||
def main():
|
||||
default_path = os.getenv('IDF_PATH', None)
|
||||
def validate_kconfig_file(kconfig_full_path, verbose=False): # type: (str, bool) -> bool
|
||||
suggestions_full_path = kconfig_full_path + OUTPUT_SUFFIX
|
||||
fail = False
|
||||
|
||||
with open(kconfig_full_path, 'r', encoding='utf-8') as f, \
|
||||
open(suggestions_full_path, 'w', encoding='utf-8', newline='\n') as f_o, \
|
||||
LineRuleChecker(kconfig_full_path) as line_checker, \
|
||||
SourceChecker(kconfig_full_path) as source_checker, \
|
||||
IndentAndNameChecker(kconfig_full_path, debug=verbose) as indent_and_name_checker:
|
||||
try:
|
||||
for line_number, line in enumerate(f, start=1):
|
||||
try:
|
||||
for checker in [line_checker, indent_and_name_checker, source_checker]:
|
||||
checker.process_line(line, line_number)
|
||||
# The line is correct therefore we echo it to the output file
|
||||
f_o.write(line)
|
||||
except InputError as e:
|
||||
print(e)
|
||||
fail = True
|
||||
f_o.write(e.suggested_line)
|
||||
except UnicodeDecodeError:
|
||||
raise ValueError("The encoding of {} is not Unicode.".format(kconfig_full_path))
|
||||
|
||||
if fail:
|
||||
print('\t{} has been saved with suggestions for resolving the issues.\n'
|
||||
'\tPlease note that the suggestions can be wrong and '
|
||||
'you might need to re-run the checker several times '
|
||||
'for solving all issues'.format(suggestions_full_path))
|
||||
print('\tPlease fix the errors and run {} for checking the correctness of '
|
||||
'Kconfig files.'.format(os.path.abspath(__file__)))
|
||||
return False
|
||||
else:
|
||||
print('{}: OK'.format(kconfig_full_path))
|
||||
try:
|
||||
os.remove(suggestions_full_path)
|
||||
except Exception:
|
||||
# not a serious error is when the file cannot be deleted
|
||||
print('{} cannot be deleted!'.format(suggestions_full_path))
|
||||
finally:
|
||||
return True
|
||||
|
||||
|
||||
def main():
|
||||
parser = argparse.ArgumentParser(description='Kconfig style checker')
|
||||
parser.add_argument('--verbose', '-v', help='Print more information (useful for debugging)',
|
||||
action='store_true', default=False)
|
||||
parser.add_argument('--directory', '-d', help='Path to directory where Kconfigs should be recursively checked '
|
||||
'(for example $IDF_PATH)',
|
||||
type=valid_directory,
|
||||
required=default_path is None,
|
||||
default=default_path)
|
||||
parser.add_argument('files', nargs='*',
|
||||
help='Kconfig files')
|
||||
parser.add_argument('--verbose', '-v', action='store_true',
|
||||
help='Print more information (useful for debugging)')
|
||||
parser.add_argument('--includes', '-d', nargs='*',
|
||||
help='Extra paths for recursively searching Kconfig files. (for example $IDF_PATH)',
|
||||
type=valid_directory)
|
||||
parser.add_argument('--exclude-submodules', action='store_true',
|
||||
help='Exclude submodules')
|
||||
args = parser.parse_args()
|
||||
|
||||
success_couter = 0
|
||||
success_counter = 0
|
||||
failure_counter = 0
|
||||
ignore_counter = 0
|
||||
failure = False
|
||||
|
||||
# IGNORE_DIRS makes sense when the required directory is IDF_PATH
|
||||
check_ignore_dirs = default_path is not None and os.path.abspath(args.directory) == os.path.abspath(default_path)
|
||||
ignore_dirs = IGNORE_DIRS
|
||||
if args.exclude_submodules:
|
||||
ignore_dirs = ignore_dirs + tuple(get_submodule_dirs(full_path=True))
|
||||
|
||||
for root, dirnames, filenames in os.walk(args.directory):
|
||||
for filename in filenames:
|
||||
full_path = os.path.join(root, filename)
|
||||
path_in_idf = os.path.relpath(full_path, args.directory)
|
||||
if re.search(RE_KCONFIG, filename):
|
||||
if check_ignore_dirs and path_in_idf.startswith(IGNORE_DIRS):
|
||||
print('{}: Ignored'.format(path_in_idf))
|
||||
ignore_counter += 1
|
||||
continue
|
||||
suggestions_full_path = full_path + OUTPUT_SUFFIX
|
||||
with open(full_path, 'r', encoding='utf-8') as f, \
|
||||
open(suggestions_full_path, 'w', encoding='utf-8', newline='\n') as f_o, \
|
||||
LineRuleChecker(path_in_idf) as line_checker, \
|
||||
SourceChecker(path_in_idf) as source_checker, \
|
||||
IndentAndNameChecker(path_in_idf, debug=args.verbose) as indent_and_name_checker:
|
||||
try:
|
||||
for line_number, line in enumerate(f, start=1):
|
||||
try:
|
||||
for checker in [line_checker, indent_and_name_checker, source_checker]:
|
||||
checker.process_line(line, line_number)
|
||||
# The line is correct therefore we echo it to the output file
|
||||
f_o.write(line)
|
||||
except InputError as e:
|
||||
print(e)
|
||||
failure = True
|
||||
f_o.write(e.suggested_line)
|
||||
except UnicodeDecodeError:
|
||||
raise ValueError("The encoding of {} is not Unicode.".format(path_in_idf))
|
||||
files = [os.path.abspath(file_path) for file_path in args.files]
|
||||
|
||||
if failure:
|
||||
print('{} has been saved with suggestions for resolving the issues. Please note that the '
|
||||
'suggestions can be wrong and you might need to re-run the checker several times '
|
||||
'for solving all issues'.format(path_in_idf + OUTPUT_SUFFIX))
|
||||
print('Please fix the errors and run {} for checking the correctness of '
|
||||
'Kconfigs.'.format(os.path.relpath(os.path.abspath(__file__), args.directory)))
|
||||
sys.exit(1)
|
||||
else:
|
||||
success_couter += 1
|
||||
print('{}: OK'.format(path_in_idf))
|
||||
try:
|
||||
os.remove(suggestions_full_path)
|
||||
except Exception:
|
||||
# not a serious error is when the file cannot be deleted
|
||||
print('{} cannot be deleted!'.format(suggestions_full_path))
|
||||
elif re.search(RE_KCONFIG, filename, re.IGNORECASE):
|
||||
# On Windows Kconfig files are working with different cases!
|
||||
raise ValueError('Incorrect filename of {}. The case should be "Kconfig"!'.format(path_in_idf))
|
||||
if args.includes:
|
||||
for directory in args.includes:
|
||||
for root, dirnames, filenames in os.walk(directory):
|
||||
for filename in filenames:
|
||||
full_path = os.path.join(root, filename)
|
||||
if re.search(RE_KCONFIG, filename):
|
||||
files.append(full_path)
|
||||
elif re.search(RE_KCONFIG, filename, re.IGNORECASE):
|
||||
# On Windows Kconfig files are working with different cases!
|
||||
print('{}: Incorrect filename. The case should be "Kconfig"!'.format(full_path))
|
||||
failure_counter += 1
|
||||
|
||||
for full_path in files:
|
||||
if full_path.startswith(ignore_dirs):
|
||||
print('{}: Ignored'.format(full_path))
|
||||
ignore_counter += 1
|
||||
continue
|
||||
is_valid = validate_kconfig_file(full_path, args.verbose)
|
||||
if is_valid:
|
||||
success_counter += 1
|
||||
else:
|
||||
failure_counter += 1
|
||||
|
||||
if ignore_counter > 0:
|
||||
print('{} files have been ignored.'.format(ignore_counter))
|
||||
if success_counter > 0:
|
||||
print('{} files have been successfully checked.'.format(success_counter))
|
||||
if failure_counter > 0:
|
||||
print('{} files have errors. Please take a look at the log.'.format(failure_counter))
|
||||
return 1
|
||||
|
||||
if success_couter > 0:
|
||||
print('{} files have been successfully checked.'.format(success_couter))
|
||||
if not files:
|
||||
print('WARNING: no files specified. Please specify files or use '
|
||||
'"--includes" to search Kconfig files recursively')
|
||||
return 0
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
||||
sys.exit(main())
|
@ -1,31 +0,0 @@
|
||||
#!/usr/bin/env python
|
||||
#
|
||||
# internal use only for CI
|
||||
# get latest MR IID by source branch
|
||||
|
||||
import argparse
|
||||
import os
|
||||
|
||||
from gitlab_api import Gitlab
|
||||
|
||||
|
||||
def get_MR_IID_by_source_branch(source_branch):
|
||||
if not source_branch:
|
||||
return ''
|
||||
gl = Gitlab(os.getenv('CI_PROJECT_ID'))
|
||||
if not gl.project:
|
||||
return ''
|
||||
mrs = gl.project.mergerequests.list(state='opened', source_branch=source_branch)
|
||||
if mrs:
|
||||
mr = mrs[0] # one source branch can only have one opened MR at one moment
|
||||
return mr.iid
|
||||
return ''
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
parser = argparse.ArgumentParser(description='Get the latest MR IID by source branch, if not found, return empty string')
|
||||
parser.add_argument('source_branch', nargs='?', help='source_branch') # won't fail if it's empty
|
||||
|
||||
args = parser.parse_args()
|
||||
|
||||
print(get_MR_IID_by_source_branch(args.source_branch))
|
85
tools/ci/ci_get_mr_info.py
Normal file
85
tools/ci/ci_get_mr_info.py
Normal file
@ -0,0 +1,85 @@
|
||||
#!/usr/bin/env python
|
||||
#
|
||||
# internal use only for CI
|
||||
# get latest MR information by source branch
|
||||
#
|
||||
# Copyright 2020 Espressif Systems (Shanghai) PTE LTD
|
||||
#
|
||||
# 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 argparse
|
||||
import os
|
||||
import subprocess
|
||||
|
||||
from gitlab_api import Gitlab
|
||||
|
||||
|
||||
def _get_mr_obj(source_branch):
|
||||
if not source_branch:
|
||||
return None
|
||||
gl = Gitlab(os.getenv('CI_PROJECT_ID', 'espressif/esp-idf'))
|
||||
if not gl.project:
|
||||
return None
|
||||
mrs = gl.project.mergerequests.list(state='opened', source_branch=source_branch)
|
||||
if mrs:
|
||||
return mrs[0] # one source branch can only have one opened MR at one moment
|
||||
else:
|
||||
return None
|
||||
|
||||
|
||||
def get_mr_iid(source_branch): # type: (str) -> str
|
||||
mr = _get_mr_obj(source_branch)
|
||||
if not mr:
|
||||
return ''
|
||||
else:
|
||||
return str(mr.iid)
|
||||
|
||||
|
||||
def get_mr_changed_files(source_branch):
|
||||
mr = _get_mr_obj(source_branch)
|
||||
if not mr:
|
||||
return ''
|
||||
|
||||
return subprocess.check_output(['git', 'diff', '--name-only',
|
||||
'origin/{}...origin/{}'.format(mr.target_branch, source_branch)]).decode('utf8')
|
||||
|
||||
|
||||
def get_mr_commits(source_branch):
|
||||
mr = _get_mr_obj(source_branch)
|
||||
if not mr:
|
||||
return ''
|
||||
return '\n'.join([commit.id for commit in mr.commits()])
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
parser = argparse.ArgumentParser(description='Get the latest merge request info by pipeline')
|
||||
actions = parser.add_subparsers(dest='action', help='info type')
|
||||
|
||||
common_args = argparse.ArgumentParser(add_help=False)
|
||||
common_args.add_argument('src_branch', nargs='?', help='source branch')
|
||||
|
||||
actions.add_parser('id', parents=[common_args])
|
||||
actions.add_parser('files', parents=[common_args])
|
||||
actions.add_parser('commits', parents=[common_args])
|
||||
|
||||
args = parser.parse_args()
|
||||
|
||||
if args.action == 'id':
|
||||
print(get_mr_iid(args.src_branch))
|
||||
elif args.action == 'files':
|
||||
print(get_mr_changed_files(args.src_branch))
|
||||
elif args.action == 'commits':
|
||||
print(get_mr_commits(args.src_branch))
|
||||
else:
|
||||
raise NotImplementedError('not possible to get here')
|
@ -432,9 +432,11 @@ code_quality_check:
|
||||
- .rules:trigger
|
||||
allow_failure: true
|
||||
script:
|
||||
- export CI_MERGE_REQUEST_IID=`python ${CI_PROJECT_DIR}/tools/ci/ci_get_latest_mr_iid.py ${CI_COMMIT_BRANCH} | xargs`
|
||||
- export CI_MR_IID=$(python ${CI_PROJECT_DIR}/tools/ci/ci_get_mr_info.py id ${CI_COMMIT_BRANCH})
|
||||
- export CI_MR_COMMITS=$(python ${CI_PROJECT_DIR}/tools/ci/ci_get_mr_info.py commits ${CI_COMMIT_BRANCH} | tr '\n' ',')
|
||||
# test if this branch have merge request, if not, exit 0
|
||||
- test -n "$CI_MERGE_REQUEST_IID" || exit 0
|
||||
- test -n "$CI_MR_COMMITS" || exit 0
|
||||
- sonar-scanner
|
||||
-Dsonar.analysis.mode=preview
|
||||
-Dsonar.host.url=$SONAR_HOST_URL
|
||||
@ -445,12 +447,12 @@ code_quality_check:
|
||||
-Dsonar.projectBaseDir=$CI_PROJECT_DIR
|
||||
-Dsonar.exclusions=$EXCLUSIONS
|
||||
-Dsonar.gitlab.project_id=$CI_PROJECT_ID
|
||||
-Dsonar.gitlab.commit_sha=$(git log --pretty=format:%H origin/master..origin/$CI_COMMIT_REF_NAME | tr '\n' ',')
|
||||
-Dsonar.gitlab.commit_sha=CI_MR_COMMITS
|
||||
-Dsonar.gitlab.ref_name=$CI_COMMIT_REF_NAME
|
||||
-Dsonar.cxx.clangtidy.reportPath=$REPORT_DIR/clang_tidy_report.txt
|
||||
-Dsonar.cxx.includeDirectories=components,/usr/include
|
||||
-Dsonar.python.pylint_config=.pylintrc
|
||||
-Dsonar.gitlab.ci_merge_request_iid=$CI_MERGE_REQUEST_IID
|
||||
-Dsonar.gitlab.ci_merge_request_iid=$CI_MR_IID
|
||||
-Dsonar.gitlab.merge_request_discussion=true
|
||||
-Dsonar.branch.name=$CI_COMMIT_REF_NAME
|
||||
|
||||
|
@ -15,15 +15,27 @@
|
||||
- .pre_check_base_template
|
||||
- .before_script_lesser
|
||||
|
||||
check_line_endings:
|
||||
.check_pre_commit_template:
|
||||
extends: .pre_check_job_template
|
||||
script:
|
||||
- tools/ci/check-line-endings.sh ${IDF_PATH}
|
||||
stage: pre_check
|
||||
image: "$CI_DOCKER_REGISTRY/esp-idf-pre-commit:1"
|
||||
before_script:
|
||||
- source tools/ci/utils.sh
|
||||
- export PYTHONPATH="$CI_PROJECT_DIR/tools:$CI_PROJECT_DIR/tools/ci/python_packages:$PYTHONPATH"
|
||||
|
||||
check_permissions:
|
||||
extends: .pre_check_job_template
|
||||
check_pre_commit_master_release:
|
||||
extends:
|
||||
- .check_pre_commit_template
|
||||
- .rules:protected
|
||||
script:
|
||||
- tools/ci/check-executable.sh
|
||||
- git diff-tree --no-commit-id --name-only -r $CI_COMMIT_SHA | xargs pre-commit run --files
|
||||
|
||||
check_pre_commit_MR:
|
||||
extends:
|
||||
- .check_pre_commit_template
|
||||
- .rules:dev
|
||||
script:
|
||||
- python ${CI_PROJECT_DIR}/tools/ci/ci_get_mr_info.py files ${CI_COMMIT_BRANCH} | xargs pre-commit run --files
|
||||
|
||||
check_docs_lang_sync:
|
||||
extends: .pre_check_job_template
|
||||
@ -79,18 +91,8 @@ check_kconfigs:
|
||||
- tools/*/*/*/Kconfig*.new
|
||||
expire_in: 1 week
|
||||
script:
|
||||
- ${IDF_PATH}/tools/ci/multirun_with_pyenv.sh ${IDF_PATH}/tools/test_check_kconfigs.py
|
||||
- ${IDF_PATH}/tools/check_kconfigs.py
|
||||
|
||||
check_deprecated_kconfig_options:
|
||||
extends: .pre_check_job_template_with_filter
|
||||
script:
|
||||
- ${IDF_PATH}/tools/ci/check_deprecated_kconfigs.py
|
||||
|
||||
check_cmake_style:
|
||||
extends: .pre_check_job_template
|
||||
script:
|
||||
tools/cmake/run_cmake_lint.sh
|
||||
- ${IDF_PATH}/tools/ci/multirun_with_pyenv.sh ${IDF_PATH}/tools/ci/test_check_kconfigs.py
|
||||
- ${IDF_PATH}/tools/ci/check_kconfigs.py
|
||||
|
||||
check_wifi_lib_md5:
|
||||
extends: .pre_check_base_template
|
||||
@ -183,11 +185,6 @@ clang_tidy_check_all:
|
||||
BOT_NEEDS_TRIGGER_BY_NAME: 1
|
||||
BOT_LABEL_STATIC_ANALYSIS_ALL: 1
|
||||
|
||||
check_codeowners:
|
||||
extends: .pre_check_job_template
|
||||
script:
|
||||
- tools/codeowners.py ci-check
|
||||
|
||||
# For release tag pipelines only, make sure the tag was created with 'git tag -a' so it will update
|
||||
# the version returned by 'git describe'
|
||||
check_version_tag:
|
||||
|
@ -32,18 +32,18 @@ examples/system/ota/otatool/otatool_example.sh
|
||||
install.fish
|
||||
install.sh
|
||||
tools/build_apps.py
|
||||
tools/check_kconfigs.py
|
||||
tools/check_python_dependencies.py
|
||||
tools/ci/apply_bot_filter.py
|
||||
tools/ci/build_template_app.sh
|
||||
tools/ci/check-executable.sh
|
||||
tools/ci/check-line-endings.sh
|
||||
tools/ci/check_build_warnings.py
|
||||
tools/ci/check_callgraph.py
|
||||
tools/ci/check_codeowners.py
|
||||
tools/ci/check_deprecated_kconfigs.py
|
||||
tools/ci/check_examples_cmake_make.py
|
||||
tools/ci/check_examples_rom_header.sh
|
||||
tools/ci/check_executables.py
|
||||
tools/ci/check_idf_version.sh
|
||||
tools/ci/check_kconfigs.py
|
||||
tools/ci/check_readme_links.py
|
||||
tools/ci/check_rom_apis.sh
|
||||
tools/ci/check_ut_cmake_make.sh
|
||||
@ -61,11 +61,10 @@ tools/ci/push_to_github.sh
|
||||
tools/ci/test_autocomplete.py
|
||||
tools/ci/test_build_system.sh
|
||||
tools/ci/test_build_system_cmake.sh
|
||||
tools/ci/test_check_kconfigs.py
|
||||
tools/ci/test_configure_ci_environment.sh
|
||||
tools/ci/utils.sh
|
||||
tools/cmake/convert_to_cmake.py
|
||||
tools/cmake/run_cmake_lint.sh
|
||||
tools/codeowners.py
|
||||
tools/docker/entrypoint.sh
|
||||
tools/docker/hooks/build
|
||||
tools/esp_app_trace/logtrace_proc.py
|
||||
@ -94,7 +93,6 @@ tools/ldgen/test/test_generation.py
|
||||
tools/mass_mfg/mfg_gen.py
|
||||
tools/mkdfu.py
|
||||
tools/set-submodules-to-github.sh
|
||||
tools/test_check_kconfigs.py
|
||||
tools/test_idf_monitor/run_test_idf_monitor.py
|
||||
tools/test_idf_py/test_idf_py.py
|
||||
tools/test_idf_size/test.sh
|
||||
|
46
tools/ci/idf_ci_utils.py
Normal file
46
tools/ci/idf_ci_utils.py
Normal file
@ -0,0 +1,46 @@
|
||||
# internal use only for CI
|
||||
# some CI related util functions
|
||||
#
|
||||
# Copyright 2020 Espressif Systems (Shanghai) PTE LTD
|
||||
#
|
||||
# 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 logging
|
||||
import os
|
||||
import subprocess
|
||||
|
||||
IDF_PATH = os.getenv('IDF_PATH', os.path.join(os.path.dirname(__file__), '..', '..'))
|
||||
|
||||
|
||||
def get_submodule_dirs(full_path=False): # type: (bool) -> list
|
||||
"""
|
||||
To avoid issue could be introduced by multi-os or additional dependency,
|
||||
we use python and git to get this output
|
||||
:return: List of submodule dirs
|
||||
"""
|
||||
dirs = []
|
||||
try:
|
||||
lines = subprocess.check_output(
|
||||
['git', 'config', '--file', os.path.realpath(os.path.join(IDF_PATH, '.gitmodules')),
|
||||
'--get-regexp', 'path']).decode('utf8').strip().split('\n')
|
||||
for line in lines:
|
||||
_, path = line.split(' ')
|
||||
if full_path:
|
||||
dirs.append(os.path.join(IDF_PATH, path))
|
||||
else:
|
||||
dirs.append(path)
|
||||
except Exception as e:
|
||||
logging.warning(str(e))
|
||||
|
||||
return dirs
|
@ -1,25 +0,0 @@
|
||||
#!/usr/bin/env bash
|
||||
#
|
||||
# Run cmakelint on all cmake files in IDF_PATH (except third party)
|
||||
#
|
||||
# cmakelint: https://github.com/richq/cmake-lint
|
||||
#
|
||||
# NOTE: This script makes use of features in (currently unreleased)
|
||||
# cmakelint >1.4. Install directly from github as follows:
|
||||
#
|
||||
# pip install https://github.com/richq/cmake-lint/archive/058c6c0ed2536.zip
|
||||
#
|
||||
|
||||
if [ -z "${IDF_PATH}" ]; then
|
||||
echo "IDF_PATH variable needs to be set"
|
||||
exit 3
|
||||
fi
|
||||
|
||||
cd "$IDF_PATH"
|
||||
|
||||
# Only list the "main" IDF repo, don't check any files in submodules (which may contain
|
||||
# third party CMakeLists.txt)
|
||||
git ls-tree --full-tree --name-only -r HEAD | grep -v "/third_party/" | grep "^CMakeLists.txt$\|\.cmake$" \
|
||||
| xargs cmakelint --linelength=120 --spaces=4
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user