У вас включен AdBlock или иной блокировщик рекламы.

Пожалуйста, отключите его, доход от рекламы помогает развитию сайта и появлению новых статей.

Спасибо за понимание.

В другой раз
DevGang блог о програмировании
Авторизоваться

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