Как предотвратить коммит в 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