Как предотвратить коммит в master/dev ветку по используя Pre-push Hooks в Git?
Вы когда-нибудь хотели биться головой о стену, когда случайно отправили коммиты в master / development ветки (даже если они были локальные)? Конечно, в таких случаях можно отменить фиксацию `git reset HEAD files`, однако это не очень приятная вещь.
К счастью, в git есть много хуков, а pre-push - это тот, который позволяет вам сделать некоторые проверки до того, как git фактически его применит. Pre-push - это на самом деле bash-скрипт, который запускается перед командой `git push`. Если скрипт возвращает 1, операция отменится. Если скрипт вернет 0, он продолжит операцию push.
Pre-push находится в каждом репозитории в скрытом каталоге .git/hooks. Ниже приведен пример (называется pre-push.sample).
#!/bin/sh
remote="$1"
url="$2"
z40=0000000000000000000000000000000000000000
while read local_ref local_sha remote_ref remote_sha
do
if [ "$local_sha" = $z40 ]
then
# Handle delete
:
else
if [ "$remote_sha" = $z40 ]
then
# New branch, examine all commits
range="$local_sha"
else
# Update to existing branch, examine new commits
range="$remote_sha..$local_sha"
fi
# Check for WIP commit
commit=`git rev-list -n 1 --grep '^WIP' "$range"`
if [ -n "$commit" ]
then
echo >&2 "Found WIP commit in $local_ref, not pushing"
exit 1
fi
fi
done
exit 0Мы можем проверить, находится ли текущая голова на одной из главных веток, например master, origin / development, если да, то завершите работу с кодом 1. Следующий pre-push скрипт должен иметь права chmod с разрешением на выполнение. Все, что он делает, это проверяет, запрещена ли текущая ветка для коммита, если да, возвращаемое значение 1 в противном случае указывает вернется 0 и комит применится.
#!/bin/bash
master_branches=('origin/develop' 'master' 'develop')
current_branch=$(git symbolic-ref HEAD | sed -e 's,.*/\(.*\),\1,')
for branch in "${master_branches[@]}"
do
if [ $branch = $current_branch ]; then
echo "You're about to push $branch and you don't want that"
exit 1
fi
done
exit 0Вот как это выглядит, когда вы пытаетесь применить изменения к мастеру:
$ git branch * master $ git push origin master You're about to push master and you don't want that error: failed to push some refs to 'https://github.com/DoctorLai/ACM.git'
Поскольку мы хотим использовать pre-push в каждом репозитории, мы можете скопировать файл в root (корневую директорию) и использовать следующую команду bash, чтобы найти и скопировать файл во весе каталоги .git/hooks.
find . -type d -name "hooks" | grep ".git/hooks" | xargs echo cp pre-push
Я полагаю, что в Windows pre-push все еще написан на BASH. Однако, если вышеупомянутое не работает в Windows, вы можете попробовать следующий сценарий Windows BATCH - или просто для образовательных целей.
@echo off
REM pre-push.cmd
REM To bypass this hook and force the push, use 'git push --no-verify'setlocal enabledelayedexpansion
set branches=origin^/develop master developfor /f "tokens=2" %%i in ('git branch') do (set current=%%i)
for %%j in (%branches%) do (
if /i "%%j"=="%current%" (
echo You're about to push to %%j, you don't want that
exit /b 1
)
)
endlocal
exit /b 0