I have a fuzzy idea of what the git index contains as one does git-adds and git-commits, but I don't have a clue of what happens to these contents when one does a git-merge. I'm particularly interested in learning what the index holds when a merge fails (e.g. due to some conflict).
How do contents of git index evolve during a merge (and what's in the index after a failed merge)?
1.3k views Asked by kjo At
1
There are 1 answers
Related Questions in GIT
- problem to push files on a repository git
- diff3 output in git conflict style, including mergeable hunks
- Git Not In Sync with Local Branch
- Setting up the version control of .dotfiles while the .config is connected to a forked repo
- How to fix overriding the main branch in Git?
- I can't add text to "Message" in VS Code when committing to Git
- How can i redirect pull request from main branch to another branch
- Xcode commits (possibly outside of any branch) disappeared, how to get them back?
- Git/TortoiseGit : how to apply ONLY the changes from ONE commit from branch A, to branch B?
- How can I reintroduce username an password on git using fedora?
- GIT SKIP EMPTY DIRECTORIES
- Git smudge run once per checkout or per commit?
- I can't find ~/.profile or ~/.bashrc in C:/Users/<user>/.ssh folder
- Set environment variable during push for GitHub Actions
- Android WebRTC compile
Related Questions in GIT-MERGE
- Git merge strategies vs. merge drivers vs. mergetools
- How to convert GitHub squash and merge commits to merge commits?
- Git refusing to commit
- How to fix "this branch is 1 commit behind of master branch"
- Visual Studio does not see all merge differences. How can I add them?
- Octopus merge: keep the first parent to the target branch
- Git log of a specific branch already merged
- Revert merged PR from git master branch
- git thinks branch missing commits merged earlier
- Proper way of merging a feature branch into another feature branch
- Git cant create pull request | different commit histories | bfg --delete-files | git merge --allow-unrelated-histories
- Applying Git commits from one branch to another completely unrelated branch
- IntelliJ Merge Revision does not highlights differences and neither let me modify the files
- Calculate the ratio of merged forks to all forks of a repository to analyze if it has any impact on the success of an open-source project
- Why is the BASE file generated by TortoiseGit in merge conflict resolution 0 bytes?
Related Questions in GIT-INDEX
- Pop conflicting Git stash while keeping stash index
- Removing a submodule manually
- Git: Could not reset submodule index when checking out
- For a given Git repository, how do I find the path to its index file?
- How can I recover from "invalid sha1 pointer in resolve-undo" with git fsck?
- Weird behavior of git switch/checkout if index has same content as target commit
- git error: read error while indexing: Invalid argument
- How can navigate one level back at the git interactive staging (add) command
- Is it possible to apply single lines from stash?
- How can I directly and quickly change all the uid/gid of all files in the git index of a git repository
- Why does git checkout file behave like reset staged file followed by checkout unstaged file?
- Can I mark a file for resolution without adding it to the index with one command?
- Why does `git update-index` not affect files not in the current directory?
- Cloning a git repo that will be the upstream parent but I want to make changes to the README for my new project and not ever push that back
- How does Git restore index and wokring directory internally (upon reset/checkout)?
Popular Questions
- How do I undo the most recent local commits in Git?
- How can I remove a specific item from an array in JavaScript?
- How do I delete a Git branch locally and remotely?
- Find all files containing a specific text (string) on Linux?
- How do I revert a Git repository to a previous commit?
- How do I create an HTML button that acts like a link?
- How do I check out a remote Git branch?
- How do I force "git pull" to overwrite local files?
- How do I list all files of a directory?
- How to check whether a string contains a substring in JavaScript?
- How do I redirect to another webpage?
- How can I iterate over rows in a Pandas DataFrame?
- How do I convert a String to an int in Java?
- Does Python have a string 'contains' substring method?
- How do I check if a string contains a specific word?
Trending Questions
- UIImageView Frame Doesn't Reflect Constraints
- Is it possible to use adb commands to click on a view by finding its ID?
- How to create a new web character symbol recognizable by html/javascript?
- Why isn't my CSS3 animation smooth in Google Chrome (but very smooth on other browsers)?
- Heap Gives Page Fault
- Connect ffmpeg to Visual Studio 2008
- Both Object- and ValueAnimator jumps when Duration is set above API LvL 24
- How to avoid default initialization of objects in std::vector?
- second argument of the command line arguments in a format other than char** argv or char* argv[]
- How to improve efficiency of algorithm which generates next lexicographic permutation?
- Navigating to the another actvity app getting crash in android
- How to read the particular message format in android and store in sqlite database?
- Resetting inventory status after order is cancelled
- Efficiently compute powers of X in SSE/AVX
- Insert into an external database using ajax and php : POST 500 (Internal Server Error)
For any given path, there are up to four "version numbers" in the index, numbered 0 (zero) through 3. I'll call them "slots" as if they were actually there for every entry, and then easily indexed (this makes them easier to think about), although actually extra versions are introduced dynamically only when needed. These "virtual slots" can be "empty", meaning the file does not exist.
(Actually, once an entry is created in the index, it's marked with a flag bit,
CE_REMOVED, if needed. This gets hairy because a whole directory full of files can be marked "removed" and then a file can be created with the name of the previous directory and marked "added". Let's just pretend we have fixed slots, there-but-empty, instead. :-) )Slot #0 is the "normal", un-conflicted, all-is-well entry. It contains a bunch of cache data, the path name, and the blob-ID (the SHA-1) for the file stored in the repository.
When a merge succeeds, it's all "business as usual", so the only special case is a conflicted merge. A merge is "conflicted" when slots 1, 2, and/or 3 are non-empty. Skipping over most of the mechanics, what happens is this. The merge uses the "newest" name for all the slots, and:
HEAD, unless you're manually invoking some of the underlying merge machinery) version. If the file was removed inHEAD/ target-of-merge, this slot is empty instead.Once you resolve the conflict and "git add", the #0 slot gets filled in with whatever you "add", wiping out the entries in #1 through #3—or, if you "git rm" the conflicted file, the other stage entries are still removed, but now the #0 slot remains empty, which also resolves the conflict.
More concretely, then, suppose you have a common ancestor that has (among others) these two files:
You're on branch
cleanupand you've renamedgronktobreem, and edited both that andflibby. You decide togit merge work, where they modifiedgronkbut did not rename it, and removedflibby. Some other file(s) merged cleanly.The index will contain three versions of
bleemand two versions offlibby:You can see the original (base) version of
bleemwithgit show :1:bleem. That was calledgronkin the base version (and inworkas well, in this case), but now it's calledbleembecause git believes you renamedgronktobleemincleanup. (Git finds the renames between the merge-base andHEADand then applies the same renaming toworkif necessary, as in this case.)Likewise, you can see the
workversion withgit show :3:bleemorgit show work:gronk, and theHEADversion with any of:git show HEAD:bleem,git show cleanup:bleem, orgit show :2:bleem(slot 2 contains theHEADakacleanupversion, and is named according to the name inHEAD).For
flibby, though, since it was removed inwork, there is no "theirs" (slot 3) version.To resolve the conflicts, you need only tell
git addorgit rmto update the slot-zero entry and remove the 1-through-3 entries. Of course, withgit add, what goes into slot 0 is whatever is in the work directory now, so you generally have to edit the files first.Incidentally, I labeled slots 2 and 3 "ours" and "theirs" above. This is how
git checkouttreats them as well (git checkout --oursandgit checkout --theirslet you write version 2 or 3 into slot 0; such a checkout, like most checkouts, "erases" the other slots too, thus resolving the conflict). However, in a rebase, theHEADbranch is actually the branch being rebased-on-to, and the "theirs" version is your branch-being-rebased. So the ours/theirs terminology is not really that great, in my opinion: it's too easy to get it backwards during a rebase.I should also note that
git checkout -mwill "re-create" a merge conflict, if you're in the middle of a conflicted merge, by erasing slot 0 and "resurrecting" the versions in slots 1-3 as needed (and writing the conflicted merge file to the working directory, obeying any change in yourmerge.conflictstylesetting as well).