* [Bug] git stash generates a different diff then other commands (diff, add, etc) resulting in merge conflicts!
@ 2013-08-08 7:07 Luke San Antonio
2013-08-08 20:54 ` Phil Hord
0 siblings, 1 reply; 5+ messages in thread
From: Luke San Antonio @ 2013-08-08 7:07 UTC (permalink / raw)
To: git
Hi, my name's Luke!
Today, I had a problem merging a stash after immediately creating it.
This is exactly what I did!
git stash save --keep-index
git stash pop
And BAM! Merge conflict! This was especially weird because my file had
this in it (taken directly from my code!)
<<<<<<< Updated upstream
*
* It should be used and passed along to member objects by GameStates!
*
=======
*
* It should be used and passed along to member objects by GameStates!
*
>>>>>>> Stashed changes
They are exactly the same!
Oh, by the way, I should mention that I did not edit any hunks to get
the index the way I wanted it, I have read that doing that causes
merge conflicts similar to this!
Then I got a hunch! I realized that git will refrain from applying a
hunk if it finds it already was applied exactly (that's correct
right?)... So I thought, maybe the patches are similar (represent the
same changes) but aren't *exactly* the same.
I was right!
After saving the stash I take a look at the diff: (git stash show -p)
/*!
- * \brief The default font renderer, global to all who have a pointer to
- * the Game class.
+ * \brief The font renderer implementation, obtained from the config file.
+ *
+ * It should be used and passed along to member objects by GameStates!
*
- * It need not be used at all!
+ * \note It can be cached, but not between GameStates, meaning it should be
+ * cached again every time a new GameState is constructed!
*/
After that, I take a look at the diff in my index: (git diff --staged)
/*!
- * \brief The default font renderer, global to all who have a pointer to
- * the Game class.
+ * \brief The font renderer implementation, obtained from the config file.
*
- * It need not be used at all!
+ * It should be used and passed along to member objects by GameStates!
+ *
+ * \note It can be cached, but not between GameStates, meaning it should be
+ * cached again every time a new GameState is constructed!
*/
Aha! A difference, a difference so tiny it went unnoticed by me, but not by git!
Now the housekeeping:
What I wanted to do:
Apply a stash on top of a 'kept' index.
What I did:
git stash save --keep-index
git stash pop
What I saw happen:
A merge conflict between the same changes (see above).
What I expected to see:
No merge conflict.
How are these different:
The conflict, which shouldn't happen since the changes introduced were the same!
-------------------------------------------------
It seems to me like the stash command is using a slightly different
diff algorithm...
Can anyone explain to me what's going on under the hood so I can
understand this subtle difference? Does anyone know?
Thanks in advance, I'm sure you all will be very helpful!
- Luke
^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [Bug] git stash generates a different diff then other commands (diff, add, etc) resulting in merge conflicts! 2013-08-08 7:07 [Bug] git stash generates a different diff then other commands (diff, add, etc) resulting in merge conflicts! Luke San Antonio @ 2013-08-08 20:54 ` Phil Hord 2013-08-12 5:29 ` Luke San Antonio 0 siblings, 1 reply; 5+ messages in thread From: Phil Hord @ 2013-08-08 20:54 UTC (permalink / raw) To: Luke San Antonio; +Cc: git@vger.kernel.org On Thu, Aug 8, 2013 at 3:07 AM, Luke San Antonio <lukesanantonio@gmail.com> wrote: > > Hi, my name's Luke! > > Today, I had a problem merging a stash after immediately creating it. > This is exactly what I did! > > git stash save --keep-index > git stash pop > > And BAM! Merge conflict! This was especially weird because my file had > this in it (taken directly from my code!) > Luke, I think the issue is that your working directory receives your cached file when you say 'git stash --keep-index'. When you restore the stash, your previous working directory now conflicts with your new working directory, but neither is the same as HEAD. Here's a test script to demonstrate the issue, I think. Did I get this right, Luke? # cd /tmp && rm -rf foo git init foo && cd foo echo "foo" > bar && git add bar && git commit -mfoo echo "bar" > bar && git add bar echo "baz" > bar echo "Before stash bar: $(cat bar)" git stash --keep-index echo "After stash bar: $(cat bar)" git stash apply The output looks like this: $ git init foo && cd foo Initialized empty Git repository in /tmp/foo/.git/ $ git commit --allow-empty -mInitialCommit [master (root-commit) b5ecc7e] InitialCommit $ echo "Bar" > bar && git add bar && git commit -mBar [master 16d708b] Bar 1 file changed, 1 insertion(+) create mode 100644 bar $ echo "bar" > bar && git add bar $ echo "baz" > bar $ echo "Before stash bar: $(cat bar)" Before stash bar: baz $ git stash --keep-index Saved working directory and index state WIP on master: 16d708b Bar HEAD is now at 16d708b Bar $ echo "After stash bar: $(cat bar)" After stash bar: bar $ git stash apply Auto-merging bar CONFLICT (content): Merge conflict in bar Recorded preimage for 'bar' $ cat bar <<<<<<< Updated upstream bar ======= baz >>>>>>> Stashed changes Phil ^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [Bug] git stash generates a different diff then other commands (diff, add, etc) resulting in merge conflicts! 2013-08-08 20:54 ` Phil Hord @ 2013-08-12 5:29 ` Luke San Antonio [not found] ` <CABURp0rWMAs9vT791vp4BEYS-Y9Jmzjmt4bbuB+po8=vkiqUWQ@mail.gmail.com> 0 siblings, 1 reply; 5+ messages in thread From: Luke San Antonio @ 2013-08-12 5:29 UTC (permalink / raw) To: Phil Hord; +Cc: git@vger.kernel.org On 08/08/20130 04:54 PM, Phil Hord wrote: > Luke, > > I think the issue is that your working directory receives your cached > file when you say 'git stash --keep-index'. When you restore the > stash, your previous working directory now conflicts with your new > working directory, but neither is the same as HEAD. > > Here's a test script to demonstrate the issue, I think. Did I get > this right, Luke? > > # cd /tmp && rm -rf foo > git init foo && cd foo > echo "foo" > bar && git add bar && git commit -mfoo > echo "bar" > bar && git add bar > echo "baz" > bar > echo "Before stash bar: $(cat bar)" > git stash --keep-index > echo "After stash bar: $(cat bar)" > git stash apply Actually no, in your script, the bar file has a modification in the working tree which is in the same hunk as a change applied to the index. In my project the changes that were added to the index are not modified further in theworking tree. -------- Not only that, but I found out why git was generated different patches! I realized that when I removed a hunk appearing before the merge conflict from the working tree and index, the merge conflict disappeared! Turns out, we can forget about stashing for a minute! First the hunk in my working tree: @@ -56,12 +56,14 @@ bool running_ = true; /*! - * \brief The default font renderer, global to all who have a pointer to - * the Game class. + * \brief The font renderer implementation, obtained from the config file. * - * It need not be used at all! + * It should be used and passed along to member objects by GameStates! + * + * \note It can be cached, but not between GameStates, meaning it should be + * cached again every time a new GameState is constructed! */ - std::unique_ptr<FontRenderer> font_renderer_ = nullptr; + FontRenderer* font_renderer_ = nullptr; int run(int argc, char* argv[]); Most of this is unimportant, but notice the line number spec:@@ -56,12 +56,14 @@ The line number of this hunk doesn't change! Then I addeda few lines *above* this hunk, (around line 30 I think). Here is the diff again: @@ -56,12 +58,14 @@ bool running_ = true; /*! - * \brief The default font renderer, global to all who have a pointer to - * the Game class. + * \brief The font renderer implementation, obtained from the config file. + * + * It should be used and passed along to member objects by GameStates! * - * It need not be used at all! + * \note It can be cached, but not between GameStates, meaning it should be + * cached again every time a new GameState is constructed! */ - std::unique_ptr<FontRenderer> font_renderer_ = nullptr; + FontRenderer* font_renderer_ = nullptr; int run(int argc, char* argv[]); Notice the new line number spec:@@ -56,12 +58,14 @@ It moves two lines down, because I added those two lines before it, makes sense! But also notice that the patches are different, just because of the two lines above it! I thought I might be able to fix this problem by changing the new diff.algorithm config option to 'patience', but it seems to only affect how patches look, not how they are stored internally... Same problem! Also, I'm wondering why that line was picked up by git if the patches don't match, shouldn't git give me a conflict with the whole hunk, or is the system smarter than that? What if merging suppressed the conflict if both possibilities are the same? Isn't that reasonable, or is there some 1% where this could cause (silent but deadly) data loss. Sorry, I'm just rambling... Anyway, thanks for your help Phil! - Luke ^ permalink raw reply [flat|nested] 5+ messages in thread
[parent not found: <CABURp0rWMAs9vT791vp4BEYS-Y9Jmzjmt4bbuB+po8=vkiqUWQ@mail.gmail.com>]
* Re: [Bug] git stash generates a different diff then other commands (diff, add, etc) resulting in merge conflicts! [not found] ` <CABURp0rWMAs9vT791vp4BEYS-Y9Jmzjmt4bbuB+po8=vkiqUWQ@mail.gmail.com> @ 2013-08-13 5:31 ` Luke San Antonio 2013-08-16 0:05 ` Phil Hord 0 siblings, 1 reply; 5+ messages in thread From: Luke San Antonio @ 2013-08-13 5:31 UTC (permalink / raw) To: Phil Hord; +Cc: git@vger.kernel.org On 08/12/2013 12:05 PM, Phil Hord wrote: > On Mon, Aug 12, 2013 at 1:29 AM, Luke San Antonio > <lukesanantonio@gmail.com> wrote: >> On 08/08/20130 04:54 PM, Phil Hord wrote: >>> Luke, >>> >>> I think the issue is that your working directory receives your cached >>> file when you say 'git stash --keep-index'. When you restore the >>> stash, your previous working directory now conflicts with your new >>> working directory, but neither is the same as HEAD. >>> >>> Here's a test script to demonstrate the issue, I think. Did I get >>> this right, Luke? >>> >>> # cd /tmp && rm -rf foo >>> git init foo && cd foo >>> echo "foo" > bar && git add bar && git commit -mfoo >>> echo "bar" > bar && git add bar >>> echo "baz" > bar >>> echo "Before stash bar: $(cat bar)" >>> git stash --keep-index >>> echo "After stash bar: $(cat bar)" >>> git stash apply >> Actually no, in your script, the bar file has a modification in the working >> tree which is in the same hunk as a change applied to the index. In my >> project the changes that were added to the index are not modified further >> in theworking tree. >> >> -------- >> >> Not only that, but I found out why git was generated different patches! >> I realized that when I removed a hunk appearing before the merge conflict >> from the working tree and index, the merge conflict disappeared! Turns >> out, we can forget about stashing for a minute! >> First the hunk in my working tree: >> >> @@ -56,12 +56,14 @@ >> bool running_ = true; >> >> >> /*! >> - * \brief The default font renderer, global to all who have a pointer >> to >> - * the Game class. >> + * \brief The font renderer implementation, obtained from the config >> file. >> * >> - * It need not be used at all! >> + * It should be used and passed along to member objects by GameStates! >> + * >> + * \note It can be cached, but not between GameStates, meaning it >> should be >> + * cached again every time a new GameState is constructed! >> */ >> - std::unique_ptr<FontRenderer> font_renderer_ = nullptr; >> + FontRenderer* font_renderer_ = nullptr; >> >> int run(int argc, char* argv[]); >> >> Most of this is unimportant, but notice the line number spec:@@ -56,12 >> +56,14 @@ >> The line number of this hunk doesn't change! Then I addeda few lines *above* >> this hunk, (around line 30 I think). Here is the diff again: >> >> @@ -56,12 +58,14 @@ >> bool running_ = true; >> >> >> /*! >> - * \brief The default font renderer, global to all who have a pointer >> to >> - * the Game class. >> + * \brief The font renderer implementation, obtained from the config >> file. >> + * >> + * It should be used and passed along to member objects by GameStates! >> * >> - * It need not be used at all! >> + * \note It can be cached, but not between GameStates, meaning it >> should be >> + * cached again every time a new GameState is constructed! >> */ >> - std::unique_ptr<FontRenderer> font_renderer_ = nullptr; >> + FontRenderer* font_renderer_ = nullptr; >> >> int run(int argc, char* argv[]); >> >> Notice the new line number spec:@@ -56,12 +58,14 @@ >> >> It moves two lines down, because I added those two lines before it, makes >> sense! >> But also notice that the patches are different, just because of the two >> lines >> above it! >> >> I thought I might be able to fix this problem by changing the new >> diff.algorithm >> config option to 'patience', but it seems to only affect how patches look, >> not >> how they are stored internally... Same problem! >> >> Also, I'm wondering why that line was picked up by git if the patches don't >> match, >> shouldn't git give me a conflict with the whole hunk, or is the system >> smarter >> than that? > Git does not store patches. Git stores the entire file. I do not > think the diff algorithm you choose will have any effect on the > results of the merge. But I am pretty clueless about the merge > engine, so I could be off-base on this last part. > >> What if merging suppressed the conflict if both possibilities are the same? >> Isn't >> that reasonable, or is there some 1% where this could cause (silent but >> deadly) >> data loss. > I think that is what Git is meant to do. But I am confused now about > where the failure is occurring for you. Can you demonstrate the > problem by modifying my test script? > > Is this more like it? > > cd /tmp && rm -rf foo > git init foo && cd foo > printf "1\n2 foo\n3\n4\n5\n6\n7\n8 foo\n9\n10\n" > bar && git add bar > git commit -mfoo > printf "1\n2 XXX\n3\n4\n5\n6\n7\n8 foo\n9\n10\n" > bar && git add bar > printf "1\n2 XXX\n3\n4\n5\n6\n7\n8 XXX\n9\n10\n" > bar > echo "Before stash bar: $(cat bar)" > git stash --keep-index > echo "After stash bar: $(cat bar)" > git stash apply > > Phil So I found an isolated case, it's very strange... Here's a script! cd /tmp && rm -rf foo; git init foo && cd foo; git config --local diff.algorithm myers printf "\n\n-----------------\n\n\n /*"'!'"\ \n * ---------\n * ^^^^^^^^^\n *\n \ * =========\n */\n |-----------|\n" > foo; git add foo && git commit -m "foo"; printf "\n-----------------\n\n\n /*"'!'"\ \n * #########\n *\n * !!!!!!!!!\n \ *\n * @@@@@@@@@\n * &&&&&&&&&\n */\n \ |===========|\n" > foo printf "s\nn\ny\ny\ny\n" | git add foo --patch > /dev/null git stash save --keep-index Let me start off by apologizing for that! =D ... Copy and paste that into a terminal and you should have a recreated version of my repository there! Now that the file is partly stashed and partly in the index, check out the difference in diffs: try: git diff --staged then try: git stash show -p You see the difference? Then pop the stash and you'll see a very obfuscated and verbose sample of what I am talking about! Also, sorry about the typos in my last message, I guess I looked past them... Thanks Phil, for you know, helping me out! - Luke ^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [Bug] git stash generates a different diff then other commands (diff, add, etc) resulting in merge conflicts! 2013-08-13 5:31 ` Luke San Antonio @ 2013-08-16 0:05 ` Phil Hord 0 siblings, 0 replies; 5+ messages in thread From: Phil Hord @ 2013-08-16 0:05 UTC (permalink / raw) To: Luke San Antonio Cc: git@vger.kernel.org, Junio C Hamano, Johannes Schindelin, Jonathan Nieder On Tue, Aug 13, 2013 at 1:31 AM, Luke San Antonio <lukesanantonio@gmail.com> wrote: > > So I found an isolated case, it's very strange... > > Here's a script! > > <deleted> Thanks for that. It was hard to read, but it demonstrates the problem well. > ... Copy and paste that into a terminal and you should have a recreated > version of my repository there! Now that the file is partly stashed and > partly in the index, check out the difference in diffs: > > try: > git diff --staged > then try: > git stash show -p This one is pretty sneaky. It is not limited to git-stash. I can demonstrate the problem now using 'git merge-file'. But I can only make this problem show itself when: 1. The collisions are separated by just one line of common code, and 2. One of the lines of common code is duplicated in one of the collisions, and 3. The first two lines of the file are duplicated, and 4. One of the first two lines is deleted on one side but not the other. I have managed to boil the test down to this script: #----------------------------- cat >base <<base 1 duplicate 1 duplicate 3 unchanged 4 will change 5 gets deleted 7 duplicated 8 will change base cat >left <<left 1 duplicate 1 duplicate 3 unchanged 4 changed 7 duplicated 6 new line 7 duplicated 8 changed left sed -e 1d left > right git merge-file -p left base right #----------------------------- The result looks like this, showing the duplicate "collision": 1 duplicate 3 unchanged 4 changed <<<<<<< left 7 duplicated 6 new line 7 duplicated ======= 7 duplicated 6 new line 7 duplicated >>>>>>> right 8 changed But it should look like this instead: 1 duplicate 3 unchanged 4 changed 7 duplicated 6 new line 7 duplicated 8 changed A similar (but different) stupidity shows up if you remove line "3" from all three files. I tested this in 1.6.5 and the same thing occurs there, so this is NOT recent regression. Phil ^ permalink raw reply [flat|nested] 5+ messages in thread
end of thread, other threads:[~2013-08-16 0:05 UTC | newest] Thread overview: 5+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2013-08-08 7:07 [Bug] git stash generates a different diff then other commands (diff, add, etc) resulting in merge conflicts! Luke San Antonio 2013-08-08 20:54 ` Phil Hord 2013-08-12 5:29 ` Luke San Antonio [not found] ` <CABURp0rWMAs9vT791vp4BEYS-Y9Jmzjmt4bbuB+po8=vkiqUWQ@mail.gmail.com> 2013-08-13 5:31 ` Luke San Antonio 2013-08-16 0:05 ` Phil Hord
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox; as well as URLs for NNTP newsgroup(s).