pre-commit-black 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  1. #!/usr/bin/env bash
  2. # git pre-commit hook that runs a black stylecheck.
  3. # Based on pre-commit-clang-format.
  4. ##################################################################
  5. # SETTINGS
  6. # Set path to black binary.
  7. BLACK=`which black 2>/dev/null`
  8. BLACK_OPTIONS="-l 120"
  9. # Remove any older patches from previous commits. Set to true or false.
  10. DELETE_OLD_PATCHES=false
  11. # File types to parse.
  12. FILE_NAMES="SConstruct SCsub"
  13. FILE_EXTS=".py"
  14. # Use pygmentize instead of cat to parse diff with highlighting.
  15. # Install it with `pip install pygments` (Linux) or `easy_install Pygments` (Mac)
  16. PYGMENTIZE=`which pygmentize 2>/dev/null`
  17. if [ ! -z "$PYGMENTIZE" ]; then
  18. READER="pygmentize -l diff"
  19. else
  20. READER=cat
  21. fi
  22. # Path to zenity
  23. ZENITY=`which zenity 2>/dev/null`
  24. # Path to xmessage
  25. XMSG=`which xmessage 2>/dev/null`
  26. # Path to powershell (Windows only)
  27. PWSH=`which powershell 2>/dev/null`
  28. # Path to osascript (macOS only)
  29. OSA=`which osascript 2>/dev/null`
  30. ##################################################################
  31. # There should be no need to change anything below this line.
  32. . "$(dirname -- "$0")/canonicalize_filename.sh"
  33. # exit on error
  34. set -e
  35. # check whether the given file matches any of the set extensions
  36. matches_name_or_extension() {
  37. local filename=$(basename "$1")
  38. local extension=".${filename##*.}"
  39. for name in $FILE_NAMES; do [[ "$name" == "$filename" ]] && return 0; done
  40. for ext in $FILE_EXTS; do [[ "$ext" == "$extension" ]] && return 0; done
  41. return 1
  42. }
  43. # necessary check for initial commit
  44. if git rev-parse --verify HEAD >/dev/null 2>&1 ; then
  45. against=HEAD
  46. else
  47. # Initial commit: diff against an empty tree object
  48. against=4b825dc642cb6eb9a060e54bf8d69288fbee4904
  49. fi
  50. if [ ! -x "$BLACK" ] ; then
  51. if [ ! -t 1 ] ; then
  52. if [ -x "$ZENITY" ] ; then
  53. $ZENITY --error --title="Error" --text="Error: black executable not found."
  54. exit 1
  55. elif [ -x "$XMSG" ] ; then
  56. $XMSG -center -title "Error" "Error: black executable not found."
  57. exit 1
  58. elif [ -x "$OSA" ] ; then
  59. asmessage="$(canonicalize_filename "$(dirname -- "$0")/asmessage.applescript")"
  60. $OSA "$asmessage" -center -title "Error" --text "Error: black executable not found."
  61. exit 1
  62. elif [ \( \( "$OSTYPE" = "msys" \) -o \( "$OSTYPE" = "win32" \) \) -a \( -x "$PWSH" \) ]; then
  63. winmessage="$(canonicalize_filename "$(dirname -- "$0")/winmessage.ps1")"
  64. $PWSH -noprofile -executionpolicy bypass -file "$winmessage" -center -title "Error" --text "Error: black executable not found."
  65. exit 1
  66. fi
  67. fi
  68. printf "Error: black executable not found.\n"
  69. printf "Set the correct path in $(canonicalize_filename "$0").\n"
  70. exit 1
  71. fi
  72. # create a random filename to store our generated patch
  73. prefix="pre-commit-black"
  74. suffix="$(date +%s)"
  75. patch="/tmp/$prefix-$suffix.patch"
  76. # clean up any older black patches
  77. $DELETE_OLD_PATCHES && rm -f /tmp/$prefix*.patch
  78. # create one patch containing all changes to the files
  79. git diff-index --cached --diff-filter=ACMR --name-only $against -- | while read file;
  80. do
  81. # ignore thirdparty files
  82. if grep -q "thirdparty" <<< $file; then
  83. continue;
  84. fi
  85. # ignore file if not one of the names or extensions we handle
  86. if ! matches_name_or_extension "$file"; then
  87. continue;
  88. fi
  89. # format our file with black, create a patch with diff and append it to our $patch
  90. # The sed call is necessary to transform the patch from
  91. # --- $file timestamp
  92. # +++ $file timestamp
  93. # to both lines working on the same file and having a/ and b/ prefix.
  94. # Else it can not be applied with 'git apply'.
  95. "$BLACK" "$BLACK_OPTIONS" --diff "$file" | \
  96. sed -e "1s|--- |--- a/|" -e "2s|+++ |+++ b/|" >> "$patch"
  97. done
  98. # if no patch has been generated all is ok, clean up the file stub and exit
  99. if [ ! -s "$patch" ] ; then
  100. printf "Files in this commit comply with the black formatter rules.\n"
  101. rm -f "$patch"
  102. exit 0
  103. fi
  104. # a patch has been created, notify the user and exit
  105. printf "\nThe following differences were found between the code to commit "
  106. printf "and the black formatter rules:\n\n"
  107. if [ -t 1 ] ; then
  108. $READER "$patch"
  109. printf "\n"
  110. # Allows us to read user input below, assigns stdin to keyboard
  111. exec < /dev/tty
  112. terminal="1"
  113. else
  114. cat "$patch"
  115. printf "\n"
  116. # Allows non zero zenity/powershell output
  117. set +e
  118. terminal="0"
  119. fi
  120. while true; do
  121. if [ $terminal = "0" ] ; then
  122. if [ -x "$ZENITY" ] ; then
  123. choice=$($ZENITY --text-info --filename="$patch" --width=800 --height=600 --title="Do you want to apply that patch?" --ok-label="Apply" --cancel-label="Do not apply" --extra-button="Apply and stage")
  124. if [ "$?" = "0" ] ; then
  125. yn="Y"
  126. else
  127. if [ "$choice" = "Apply and stage" ] ; then
  128. yn="S"
  129. else
  130. yn="N"
  131. fi
  132. fi
  133. elif [ -x "$XMSG" ] ; then
  134. $XMSG -file "$patch" -buttons "Apply":100,"Apply and stage":200,"Do not apply":0 -center -default "Do not apply" -geometry 800x600 -title "Do you want to apply that patch?"
  135. choice=$?
  136. if [ "$choice" = "100" ] ; then
  137. yn="Y"
  138. elif [ "$choice" = "200" ] ; then
  139. yn="S"
  140. else
  141. yn="N"
  142. fi
  143. elif [ -x "$OSA" ] ; then
  144. asmessage="$(canonicalize_filename "$(dirname -- "$0")/asmessage.applescript")"
  145. choice=`$OSA "$asmessage" -file "$patch" -buttons "Apply":100,"Apply and stage":200,"Do not apply":0 -center -default "Do not apply" -geometry 800x600 -title "Do you want to apply that patch?"`
  146. if [ "$choice" = "100" ] ; then
  147. yn="Y"
  148. elif [ "$choice" = "200" ] ; then
  149. yn="S"
  150. else
  151. yn="N"
  152. fi
  153. elif [ \( \( "$OSTYPE" = "msys" \) -o \( "$OSTYPE" = "win32" \) \) -a \( -x "$PWSH" \) ]; then
  154. winmessage="$(canonicalize_filename "$(dirname -- "$0")/winmessage.ps1")"
  155. $PWSH -noprofile -executionpolicy bypass -file "$winmessage" -file "$patch" -buttons "Apply":100,"Apply and stage":200,"Do not apply":0 -center -default "Do not apply" -geometry 800x600 -title "Do you want to apply that patch?"
  156. choice=$?
  157. if [ "$choice" = "100" ] ; then
  158. yn="Y"
  159. elif [ "$choice" = "200" ] ; then
  160. yn="S"
  161. else
  162. yn="N"
  163. fi
  164. else
  165. printf "Error: zenity, xmessage, osascript, or powershell executable not found.\n"
  166. exit 1
  167. fi
  168. else
  169. read -p "Do you want to apply that patch (Y - Apply, N - Do not apply, S - Apply and stage files)? [Y/N/S] " yn
  170. fi
  171. case $yn in
  172. [Yy] ) git apply $patch;
  173. printf "The patch was applied. You can now stage the changes and commit again.\n\n";
  174. break
  175. ;;
  176. [Nn] ) printf "\nYou can apply these changes with:\n git apply $patch\n";
  177. printf "(may need to be called from the root directory of your repository)\n";
  178. printf "Aborting commit. Apply changes and commit again or skip checking with";
  179. printf " --no-verify (not recommended).\n\n";
  180. break
  181. ;;
  182. [Ss] ) git apply $patch;
  183. git diff-index --cached --diff-filter=ACMR --name-only $against -- | while read file;
  184. do git add $file;
  185. done
  186. printf "The patch was applied and the changed files staged. You can now commit.\n\n";
  187. break
  188. ;;
  189. * ) echo "Please answer yes or no."
  190. ;;
  191. esac
  192. done
  193. exit 1 # we don't commit in any case