Add static formatting checks for GitHub Actions
This commit is contained in:
parent
db1eb909fe
commit
79a2e4c1ab
31
.github/workflows/static_checks.yml
vendored
Normal file
31
.github/workflows/static_checks.yml
vendored
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
name: Static checks
|
||||||
|
on: [push, pull_request]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-20.04
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: |
|
||||||
|
sudo apt-get update -qq
|
||||||
|
sudo apt-get install -qq dos2unix recode clang-format
|
||||||
|
pip3 install --user black pygments
|
||||||
|
|
||||||
|
- name: File formatting checks (file_format.sh)
|
||||||
|
run: |
|
||||||
|
bash ./misc/scripts/file_format.sh
|
||||||
|
|
||||||
|
- name: Style checks via clang-format (clang_format.sh)
|
||||||
|
run: |
|
||||||
|
bash ./misc/scripts/clang_format.sh
|
||||||
|
|
||||||
|
- name: Python style checks via black (black_format.sh)
|
||||||
|
run: |
|
||||||
|
bash ./misc/scripts/black_format.sh
|
||||||
|
|
||||||
|
- name: Documentation checks
|
||||||
|
run: |
|
||||||
|
doc/tools/makerst.py --dry-run doc/classes modules
|
35
misc/scripts/black_format.sh
Executable file
35
misc/scripts/black_format.sh
Executable file
@ -0,0 +1,35 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
# This script runs black on all Python files in the repo.
|
||||||
|
|
||||||
|
set -uo pipefail
|
||||||
|
|
||||||
|
# Apply black.
|
||||||
|
echo -e "Formatting Python files..."
|
||||||
|
PY_FILES=$(find \( -path "./.git" \
|
||||||
|
-o -path "./thirdparty" \
|
||||||
|
\) -prune \
|
||||||
|
-o \( -name "SConstruct" \
|
||||||
|
-o -name "SCsub" \
|
||||||
|
-o -name "*.py" \
|
||||||
|
\) -print)
|
||||||
|
black -l 120 $PY_FILES
|
||||||
|
|
||||||
|
git diff > patch.patch
|
||||||
|
FILESIZE="$(stat -c%s patch.patch)"
|
||||||
|
MAXSIZE=5
|
||||||
|
|
||||||
|
# If no patch has been generated all is OK, clean up, and exit.
|
||||||
|
if (( FILESIZE < MAXSIZE )); then
|
||||||
|
printf "Files in this commit comply with the black style rules.\n"
|
||||||
|
rm -f patch.patch
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# A patch has been created, notify the user, clean up, and exit.
|
||||||
|
printf "\n*** The following differences were found between the code "
|
||||||
|
printf "and the formatting rules:\n\n"
|
||||||
|
cat patch.patch
|
||||||
|
printf "\n*** Aborting, please fix your commit(s) with 'git commit --amend' or 'git rebase -i <hash>'\n"
|
||||||
|
rm -f patch.patch
|
||||||
|
exit 1
|
58
misc/scripts/clang_format.sh
Executable file
58
misc/scripts/clang_format.sh
Executable file
@ -0,0 +1,58 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
# This script runs clang-format and fixes copyright headers on all relevant files in the repo.
|
||||||
|
# This is the primary script responsible for fixing style violations.
|
||||||
|
|
||||||
|
set -uo pipefail
|
||||||
|
IFS=$'\n\t'
|
||||||
|
|
||||||
|
CLANG_FORMAT_FILE_EXTS=(".c" ".h" ".cpp" ".hpp" ".cc" ".hh" ".cxx" ".m" ".mm" ".inc" ".java" ".glsl")
|
||||||
|
|
||||||
|
# Loops through all text files tracked by Git.
|
||||||
|
git grep -zIl '' |
|
||||||
|
while IFS= read -rd '' f; do
|
||||||
|
# Exclude some files.
|
||||||
|
if [[ "$f" == "thirdparty"* ]]; then
|
||||||
|
continue
|
||||||
|
elif [[ "$f" == "platform/android/java/lib/src/com/google"* ]]; then
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
|
||||||
|
for extension in ${CLANG_FORMAT_FILE_EXTS[@]}; do
|
||||||
|
if [[ "$f" == *"$extension" ]]; then
|
||||||
|
# Run clang-format.
|
||||||
|
clang-format -i "$f"
|
||||||
|
# Fix copyright headers, but not all files get them.
|
||||||
|
if [[ "$f" == *"inc" ]]; then
|
||||||
|
continue 2
|
||||||
|
elif [[ "$f" == *"glsl" ]]; then
|
||||||
|
continue 2
|
||||||
|
elif [[ "$f" == *"theme_data.h" ]]; then
|
||||||
|
continue 2
|
||||||
|
elif [[ "$f" == "platform/android/java/lib/src/org/godotengine/godot/input/InputManager"* ]]; then
|
||||||
|
continue 2
|
||||||
|
fi
|
||||||
|
python misc/scripts/copyright_headers.py "$f"
|
||||||
|
continue 2
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
done
|
||||||
|
|
||||||
|
git diff > patch.patch
|
||||||
|
FILESIZE="$(stat -c%s patch.patch)"
|
||||||
|
MAXSIZE=5
|
||||||
|
|
||||||
|
# If no patch has been generated all is OK, clean up, and exit.
|
||||||
|
if (( FILESIZE < MAXSIZE )); then
|
||||||
|
printf "Files in this commit comply with the clang-format style rules.\n"
|
||||||
|
rm -f patch.patch
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# A patch has been created, notify the user, clean up, and exit.
|
||||||
|
printf "\n*** The following differences were found between the code "
|
||||||
|
printf "and the formatting rules:\n\n"
|
||||||
|
cat patch.patch
|
||||||
|
printf "\n*** Aborting, please fix your commit(s) with 'git commit --amend' or 'git rebase -i <hash>'\n"
|
||||||
|
rm -f patch.patch
|
||||||
|
exit 1
|
95
misc/scripts/copyright_headers.py
Executable file
95
misc/scripts/copyright_headers.py
Executable file
@ -0,0 +1,95 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
import sys
|
||||||
|
|
||||||
|
header = """\
|
||||||
|
/*************************************************************************/
|
||||||
|
/* $filename */
|
||||||
|
/*************************************************************************/
|
||||||
|
/* This file is part of: */
|
||||||
|
/* GODOT ENGINE */
|
||||||
|
/* https://godotengine.org */
|
||||||
|
/*************************************************************************/
|
||||||
|
/* Copyright (c) 2007-2020 Juan Linietsky, Ariel Manzur. */
|
||||||
|
/* Copyright (c) 2014-2020 Godot Engine contributors (cf. AUTHORS.md). */
|
||||||
|
/* */
|
||||||
|
/* Permission is hereby granted, free of charge, to any person obtaining */
|
||||||
|
/* a copy of this software and associated documentation files (the */
|
||||||
|
/* "Software"), to deal in the Software without restriction, including */
|
||||||
|
/* without limitation the rights to use, copy, modify, merge, publish, */
|
||||||
|
/* distribute, sublicense, and/or sell copies of the Software, and to */
|
||||||
|
/* permit persons to whom the Software is furnished to do so, subject to */
|
||||||
|
/* the following conditions: */
|
||||||
|
/* */
|
||||||
|
/* The above copyright notice and this permission notice shall be */
|
||||||
|
/* included in all copies or substantial portions of the Software. */
|
||||||
|
/* */
|
||||||
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
||||||
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
||||||
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
|
||||||
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
||||||
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
||||||
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
||||||
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
/*************************************************************************/
|
||||||
|
"""
|
||||||
|
|
||||||
|
fname = sys.argv[1]
|
||||||
|
|
||||||
|
# Handle replacing $filename with actual filename and keep alignment
|
||||||
|
fsingle = fname.strip()
|
||||||
|
if fsingle.find("/") != -1:
|
||||||
|
fsingle = fsingle[fsingle.rfind("/") + 1 :]
|
||||||
|
rep_fl = "$filename"
|
||||||
|
rep_fi = fsingle
|
||||||
|
len_fl = len(rep_fl)
|
||||||
|
len_fi = len(rep_fi)
|
||||||
|
# Pad with spaces to keep alignment
|
||||||
|
if len_fi < len_fl:
|
||||||
|
for x in range(len_fl - len_fi):
|
||||||
|
rep_fi += " "
|
||||||
|
elif len_fl < len_fi:
|
||||||
|
for x in range(len_fi - len_fl):
|
||||||
|
rep_fl += " "
|
||||||
|
if header.find(rep_fl) != -1:
|
||||||
|
text = header.replace(rep_fl, rep_fi)
|
||||||
|
else:
|
||||||
|
text = header.replace("$filename", fsingle)
|
||||||
|
text += "\n"
|
||||||
|
|
||||||
|
# We now have the proper header, so we want to ignore the one in the original file
|
||||||
|
# and potentially empty lines and badly formatted lines, while keeping comments that
|
||||||
|
# come after the header, and then keep everything non-header unchanged.
|
||||||
|
# To do so, we skip empty lines that may be at the top in a first pass.
|
||||||
|
# In a second pass, we skip all consecutive comment lines starting with "/*",
|
||||||
|
# then we can append the rest (step 2).
|
||||||
|
|
||||||
|
fileread = open(fname.strip(), "r")
|
||||||
|
line = fileread.readline()
|
||||||
|
header_done = False
|
||||||
|
|
||||||
|
while line.strip() == "": # Skip empty lines at the top
|
||||||
|
line = fileread.readline()
|
||||||
|
|
||||||
|
if line.find("/**********") == -1: # Godot header starts this way
|
||||||
|
# Maybe starting with a non-Godot comment, abort header magic
|
||||||
|
header_done = True
|
||||||
|
|
||||||
|
while not header_done: # Handle header now
|
||||||
|
if line.find("/*") != 0: # No more starting with a comment
|
||||||
|
header_done = True
|
||||||
|
if line.strip() != "":
|
||||||
|
text += line
|
||||||
|
line = fileread.readline()
|
||||||
|
|
||||||
|
while line != "": # Dump everything until EOF
|
||||||
|
text += line
|
||||||
|
line = fileread.readline()
|
||||||
|
|
||||||
|
fileread.close()
|
||||||
|
|
||||||
|
# Write
|
||||||
|
filewrite = open(fname.strip(), "w")
|
||||||
|
filewrite.write(text)
|
||||||
|
filewrite.close()
|
59
misc/scripts/file_format.sh
Executable file
59
misc/scripts/file_format.sh
Executable file
@ -0,0 +1,59 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
# This script ensures proper POSIX text file formatting and a few other things.
|
||||||
|
# This is supplementary to clang-black-format.sh, but should be run before it.
|
||||||
|
|
||||||
|
set -uo pipefail
|
||||||
|
IFS=$'\n\t'
|
||||||
|
|
||||||
|
# Loops through all text files tracked by Git.
|
||||||
|
git grep -zIl '' |
|
||||||
|
while IFS= read -rd '' f; do
|
||||||
|
# Exclude some types of files.
|
||||||
|
if [[ "$f" == *"csproj" ]]; then
|
||||||
|
continue
|
||||||
|
elif [[ "$f" == *"sln" ]]; then
|
||||||
|
continue
|
||||||
|
elif [[ "$f" == *"patch" ]]; then
|
||||||
|
continue
|
||||||
|
elif [[ "$f" == *"pot" ]]; then
|
||||||
|
continue
|
||||||
|
elif [[ "$f" == *"po" ]]; then
|
||||||
|
continue
|
||||||
|
elif [[ "$f" == "thirdparty"* ]]; then
|
||||||
|
continue
|
||||||
|
elif [[ "$f" == "platform/android/java/lib/src/com/google"* ]]; then
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
# Ensures that files are UTF-8 formatted.
|
||||||
|
recode UTF-8 "$f" 2> /dev/null
|
||||||
|
# Ensures that files have LF line endings.
|
||||||
|
dos2unix "$f" 2> /dev/null
|
||||||
|
# Ensures that files do not contain a BOM.
|
||||||
|
sed -i '1s/^\xEF\xBB\xBF//' "$f"
|
||||||
|
# Ensures that files end with newline characters.
|
||||||
|
tail -c1 < "$f" | read -r _ || echo >> "$f";
|
||||||
|
# Remove trailing space characters.
|
||||||
|
sed -z -i 's/\x20\x0A/\x0A/g' "$f"
|
||||||
|
# Remove the character sequence "== true" if it has a leading space.
|
||||||
|
sed -z -i 's/\x20== true//g' "$f"
|
||||||
|
done
|
||||||
|
|
||||||
|
git diff > patch.patch
|
||||||
|
FILESIZE="$(stat -c%s patch.patch)"
|
||||||
|
MAXSIZE=5
|
||||||
|
|
||||||
|
# If no patch has been generated all is OK, clean up, and exit.
|
||||||
|
if (( FILESIZE < MAXSIZE )); then
|
||||||
|
printf "Files in this commit comply with the formatting rules.\n"
|
||||||
|
rm -f patch.patch
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
# A patch has been created, notify the user, clean up, and exit.
|
||||||
|
printf "\n*** The following differences were found between the code "
|
||||||
|
printf "and the formatting rules:\n\n"
|
||||||
|
cat patch.patch
|
||||||
|
printf "\n*** Aborting, please fix your commit(s) with 'git commit --amend' or 'git rebase -i <hash>'\n"
|
||||||
|
rm -f patch.patch
|
||||||
|
exit 1
|
Loading…
x
Reference in New Issue
Block a user