Git прием/обновление крючков и новых ветвей
У меня проблема с крюком обновления. В случае новой ветки она получает 0000000000000000000000000000000000000000 как "oldrev". И я не знаю, как справиться с этим делом.
У нас есть требование, чтобы каждое сообщение о передаче ссылалось на действительную проблему Jira. Поэтому я установил "обновление" в нашем центральном репозитории. Этот крюк получает "oldrev" и "newrev". Затем я передаю их в "git rev-list" следующим образом:
git rev-list $oldrev..$newrev
Это дает мне список всех revs, через которые я могу выполнить итерацию, и делать все, что мне нужно.
Проблема заключается в том, что когда пользователь нажимает новую ветку, крючок получает 0000000000000000000000000000000000000000 как oldrev. И "git rev-list" просто жалуется:
fatal: Invalid revision range 0000000000000000000000000000000000000000..21bac83b2
Итак, как мне получить список всех оборотов, которые находятся на этой новой ветке? Я обыскал сеть уже довольно давно и ничего не нашел. В примере hooks я нашел либо
- не справляются с этой проблемой и не выполняются с указанным выше сообщением об ошибке
- неправильно попытайтесь исправить проблему, установив oldrev в "", который возвращает неверные результаты из rev-list
- просто отказаться, когда они сталкиваются с этим oldrev
Ни один из этих звуков не является особенно захватывающим.
И кто-нибудь знает, как получить правильный ответ в этом случае? Я думал о запросе git для "дайте мне все обороты, которые достижимы из newrev, но не из каких-либо других ветвей (= все ветки, кроме нового)". Но даже это дало бы неправильный ответ, если бы слияние новой ветки с любым из старых было.
Ответы
Ответ 1
В этом случае термин "правильный ответ" является немного неоднозначным. Я на самом деле думаю, что "все обороты, достигнутые от newrev, но нигде больше", совершенно верны. Это справедливо даже в случае слияния - в этом случае вы должны увидеть, что коммиты уникальны для нового ref и commit commit, но не для коммитов, которые были объединены.
Итак, я бы сказал, проверьте, является ли "oldrev" все нулями, и если это так, действуйте соответственно:
if [ "$oldrev" -eq 0 ]; then
# list everything reachable from newrev but not any heads
git rev-list $(git for-each-ref --format='%(refname)' refs/heads/* | sed 's/^/\^/') "$newrev"
else
git rev-list "$oldrev..$newrev"
fi
Ответ 2
Когда $oldrev
- все нули, другая команда git rev-list
делает все, что вам нужно:
git rev-list $newrev --not --branches=*
предоставит вам список исправлений, доступных из $newrev
, но не из любых ветвей.
Обратите внимание, что это определенно не делает то же самое, что и git rev-list $oldrev..$newrev
, когда oldrev не все нули, поэтому вы хотите проверить, в каком случае вы находитесь, и выбрать соответствующую команду для запуска.
Ответ 3
Я просто понял это сам.
git log newref --notherheads
- это ключ для получения всех журналов ветки, которые не находятся ни в одной другой ветки. Ниже мой python script, чтобы проверить правильную максимальную длину строки сообщений фиксации.
import sys
import commands
ref = sys.argv[1]
old = sys.argv[2]
new = sys.argv[3]
x = 0
# only a tag is pushed to server, nothing to check
if ref.find('refs/tags/') >= 0:
if len(ref.strip('refs/tags/')) > 25:
print 'tag name is longer than 25 characters'
exit(1)
else:
exit(0)
# either a new branch is pushed or an empty repo is being pushed
if old == '0000000000000000000000000000000000000000':
heads = commands.getoutput("git for-each-ref --format='%(refname)' 'refs/heads/*'")
heads = heads.replace(ref+'\n','').replace('\n',' ')
hashes = commands.getoutput('git log '+new+' --pretty=%H --not '+heads).split('\n')
else:
hashes = commands.getoutput('git log '+old+'..'+new+' --pretty=%H').split('\n')
for hash in hashes:
subject = commands.getoutput('git show '+hash+' --format=%s --summary').split('\n')
body = commands.getoutput('git show '+hash+' --format=%b --summary').split('\n')
if len(subject[0]) > 75:
print
print 'commit: '+hash
print 'bad commit message(s): header line is too long or second line is not blank (max 75 chars)'
print 'bad line: "%s"' % subject[0]
print 'length of header line: %d' % len(subject[0])
print 'try again with correct message format'
print
x = 1
for line in body:
if len(line) > 75:
print
print 'commit: '+hash
print 'bad commit message(s): description lines are too long (max 75 chars)'
print 'bad line: "%s"' % line
print 'length of line: %d' % len(line)
print 'try again with correct message format'
print
x = 1
if x == 0:
exit(0)
else:
exit(1)
Ответ 4
Я решил это для своего крючка обновления, используя следующее:
if [ "$oldrev" -eq 0 ]; then
git log "$(git show-branch --merge-base)".."$newrev"; else
foo;
fi