GIT: как заставить слияние зафиксировать предка

В GIT у меня есть две ветки и две коммиты:

 A(master)---B(branch "topic")
  • ГОЛОВКА ветки "мастер" - это фиксация A
  • ГЛАВА раздела "Тема" - это фиксация B
  • commit A - это родительский элемент commit B

Я хотел бы создать слияние commit C в ветке "topic" (у него были бы A и B как родители). (Я знаю, что это кажется странным, и что слияние будет пустым.)

 A(master)---B---C (branch "topic")
  \-------------/

Мне удалось создать это объединение слияния слишком сложным способом (см. Ниже). Есть ли более простой способ создания этого объединения?

Спасибо за ваши ответы!


Начальное состояние:

$ git init plop
Initialized empty Git repository in /tmp/plop/.git/
$ cd plop/
$ git commit -m "Initial commit (commit A)"  --allow-empty
[master (root-commit) a687d4e] Initial commit (commit A)
$ git checkout -b topic
Switched to a new branch 'topic'
$ git commit -m "Some work on my topic branch (commit B)" --allow-empty
[topic d4d1c71] Some work on my topic branch (commit B)
$ #OK, we now reached the initial state

Некоторые попытки:

$ git merge master #Does not work
Already up-to-date.
$ git merge --no-ff -s ours master #Does not work
Already up-to-date.

Есть ли более простой способ добиться следующего?

$ #Let try another way (too complex!)
$ git checkout master
Switched to branch 'master'
$ git merge --no-ff topic
Already up-to-date!
Merge made by recursive.
$ git checkout topic
Switched to branch 'topic'
$ git merge master
Updating d4d1c71..641e7ae
Fast-forward
$ git checkout master
Switched to branch 'master'
$ git reset --hard HEAD^1
HEAD is now at a687d4e Initial commit
$ git checkout topic
Switched to branch 'topic'
$ git log #This is what I wanted to reach
commit 641e7aeb614d9b49796e8f11abd3a0290ac08b40
Merge: a687d4e d4d1c71
Author: xxx <yyy.zzz>
Date:   Sat Jul 23 12:52:41 2011 +0200

    Merge branch 'topic'

commit d4d1c71c87b94335c8852ab7675cbb663965ef7d
Author: xxx <yyy.zzz>
Date:   Sat Jul 23 12:50:11 2011 +0200

    Some work on my topic branch (commit B)

commit a687d4eb88b9f6d661122a5766dd632dd462fbaa
Author: xxx <yyy.zzz>
Date:   Sat Jul 23 12:49:52 2011 +0200

    Initial commit (commit A)

Ответ 1

UPD: более чистый способ сделать то же самое, не вступая напрямую в sha1:

$ echo "merge commit" | git commit-tree topic^{tree} -p master -p topic
4201b6abae6bb06f929ea00fbc35019679d55535

$ git merge 4201b6abae6bb06f929ea00fbc35019679d55535
Updating b826a8e..4201b6a
Fast-forward

Или даже однострочную команду:

$ git merge $(echo "merge commit" | git commit-tree topic^{tree} -p master -p topic)

Подробнее о том, что это делает - прочитайте весь ответ :)


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

Прежде всего, вы должны знать, что sha1 вашего родителя совершает. Я полагаю, что у B есть комментарий "перейти от темы", A имеет комментарий "change from master", и мы сейчас находимся в B (ветка темы).

$ git log --format=oneline -2
b826a8e93ac8da0de5bfb5b70d5f4e7c352a01fa change from topic
8b7653a529fb3ce964fda79bfd57e645441ad893 change from master

Затем вы должны знать sha1 конкретного дерева объекта фиксации B:

$ git cat-file -p topic
tree 867f31c455a371756ec353b54d755f51d98d62c4
parent 8b7653a529fb3ce964fda79bfd57e645441ad893
author ivan-danilov <[email protected]> 1311518908 +0300
committer ivan-danilov <[email protected]> 1311518908 +0300

change from topic

Так что это 867f31c455a371756ec353b54d755f51d98d62c4. И, наконец, вы должны выполнить команду git-commit-tree:

$ echo "merge commit" | git commit-tree 867f31c455a371756ec353b54d755f51d98d62c4 -p b826a8e93ac8da0de5bfb5b70d5f4e7c352a01fa -p 8b7653a529fb3ce964fda79bfd57e645441ad893
4201b6abae6bb06f929ea00fbc35019679d55535

Обратите внимание, что я использовал перенаправление конвейера, поскольку git-commit-tree принимает коммит от stdin потока. Первым параметром является дерево sha1, которое мы получили с git cat-file а два других - это транзакции sha1, которые мы получили с git log.

Вывод из команды - это sha1 вновь созданного слияния. Теперь вы хотите переадресовать ветку темы:

$ git merge 4201b6abae6bb06f929ea00fbc35019679d55535
Updating b826a8e..4201b6a
Fast-forward

Все это. У вас есть то, что вы хотели.

Ответ 2

Имеет смысл только совершить слияние, если есть разница между master и topic - если они расходятся. В вашем случае нет слияния - в topic уже есть все master коммиты, поэтому git не позволит вам создать слияние, которое ничего не делает.

Он отлично работает, если у вас есть фиксация в master, а не в topic:

$ git init plop
Initialized empty Git repository in C:/Temp/plop/.git/
$ cd plop
$ git commit -m "Initial commit (commit A)" --allow-empty
[master (root-commit) b6e2e91] Initial commit (commit A)
$ git commit -m "master-only commit (commit C)" --allow-empty
[master 67b491e] master-only commit (commit C)
$ git checkout HEAD~ -b topic
Switched to a new branch 'topic'
$ git commit -m "Some work on my topic branch (commit B)" --allow-empty
[topic 2251f13] Some work on my topic branch (commit B)
$ git merge master
Already up-to-date!
Merge made by recursive.

В результате чего...

*   592ad46 Merge branch 'master' into topic
|\
| * 67b491e master-only commit (commit C)
* | 2251f13 Some work on my topic branch (commit B)
|/
* b6e2e91 Initial commit (commit A)

Ответ 3

Слияние другого пути должно работать:

git branch tmp master    # tmp points to A
git checkout tmp
git merge --no-ff -m 'odd merge' topic    # merge B+A ==> C
git checkout topic
git reset --hard tmp     # topic now points to C
git branch -d tmp

Ответ 4

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

Если вы проверите мастер и затем слейте тему в мастер, мастер должен перемотать вперед и обновить его HEAD.

Ответ 5

Официального решения этой проблемы нет, есть временное решение. Проверьте фиксацию, что master включен, так что вы находитесь в состоянии автономного состояния. Затем выполните пустую фиксацию. Затем проверьте свою ветку topic и слейте ее в пустой транзакции.

$ git checkout 9123456 # (the latest commit on master)
Note: checking out '9123456'.
You are in 'detached HEAD' state...
$ git commit --allow-empty -m 'empty commit'
[detached HEAD 9123457] empty commit
$ git checkout topic
Warning: you are leaving 1 commit behind ... 9123457 empty commit
Switched to branch 'topic'
$ git merge 9123457