Recovering added file after doing git reset
I added a new file F1 and made changes to another file F2 but then did a "git reset --hard HEAD^" and I have lost all the changes to the files.
Is there SOME way, I can get them back.
I did look at a related question here: How can I undo git reset --hard HEAD~1? but, that question assumes that the one has done a git commit.
You can (with some work) recover state of file at the last "git add <file>". You can use
$ git fsck --cache --no-reflogs --lost-found --unreachable HEAD
and then examine files in '.git/lost-found/other' directory.
Please read git fsck manpage.
(I'm assuming that the missing file is not part of any commit. Otherwise, git log --all -g --diff-filter=D --stat
is your friend.)
Get list of unreachable files that git
knows a file name:
git fsck --unreachable --no-reflogs --no-cache HEAD | fgrep " tree "
| cut -d " " -f3 | xargs -r -n1 git ls-tree
| fgrep " blob " | cut -d " " -f 3- | sort -k2 -u
If you see something interesting, git cat-file blob SHA-1-of-interesting-file
will output the file to standard output. (Example: git cat-file blob b8f0bdf56 > recovered-logo.png
)
Unfortunately, if the missing file is not part of the any commit, git does not have a timestamp and as such, you cannot print various versions of files ordered by time.
If the missing file has never been staged ( git stage
or git add
) or stashed ( git stash
), you're pretty much out of luck because as far as git knows, the file never did exist. (You may still try doing a git fsck --no-reflogs --lost-found
and looking around in directory .git/lost-found/other
to see if you have anything worth keeping in case git indeed has a copy of your missing file by some lucky accident. You do not have file names to help you in this case, only file contents.)
In case you just lost some commits (instead of just files), you'll probably want to run something like this:
gitk --all $( git fsck | awk '/dangling commit/ {print $3}'; git log -g --pretty='format:%H' )
That will run gitk
with all the branches, all the reflog and all the dangling commits. You may want to add -n 10000
or some other limit in case your repo has really many commits (say linux kernel). If you do not have gitk
, you may instead run lesser version using only command line like this:
git log --all --decorate --stat --graph --date-order $( git fsck | awk '/dangling commit/ {print $3}'; git log -g --pretty='format:%H' )
or a version with less verbose output
git log --all --decorate --oneline --graph --date-order $( git fsck | awk '/dangling commit/ {print $3}'; git log -g --pretty='format:%H' )
There is a git plugin
that does this out of the box:
https://github.com/pendashteh/git-recover-index
$ cd /path/to/disatered/repo
$ git clone git@github.com:pendashteh/git-recover-index.git $HOME/.git-recover-index
$ $HOME/.git-recover-index/git-recover-index.sh
链接地址: http://www.djcxy.com/p/552.html
上一篇: 撤消git rebase