How do I revert a Git repository to a previous commit?
청렴결백한 만능 재주꾼2023. 5. 14. 06:07
반응형
How do I revert from my current state to a snapshot made on a certain commit?
If I do git log, then I get the following output:
$ git log
commit a867b4af366350be2e7c21b8de9cc6504678a61b`
Author: Me <me@me.com>
Date: Thu Nov 4 18:59:41 2010 -0400
blah blah blah...
commit 25eee4caef46ae64aa08e8ab3f988bc917ee1ce4
Author: Me <me@me.com>
Date: Thu Nov 4 05:13:39 2010 -0400
more blah blah blah...
commit 0766c053c0ea2035e90f504928f8df3c9363b8bd
Author: Me <me@me.com>
Date: Thu Nov 4 00:55:06 2010 -0400
And yet more blah blah...
commit 0d1d7fc32e5a947fbd92ee598033d85bfc445a50
Author: Me <me@me.com>
Date: Wed Nov 3 23:56:08 2010 -0400
Yep, more blah blah.
How do I revert to the commit from November 3, i.e. commit 0d1d7fc?
This depends a lot on what you mean by "revert".
Temporarily switch to a different commit
If you want to temporarily go back to it, fool around, then come back to where you are, all you have to do is check out the desired commit:
# This will detach your HEAD, that is, leave you with no branch checked out:
git checkout 0d1d7fc32
Or if you want to make commits while you're there, go ahead and make a new branch while you're at it:
git checkout -b old-state 0d1d7fc32
To go back to where you were, just check out the branch you were on again. (If you've made changes, as always when switching branches, you'll have to deal with them as appropriate. You could reset to throw them away; you could stash, checkout, stash pop to take them with you; you could commit them to a branch there if you want a branch there.)
Hard delete unpublished commits
If, on the other hand, you want to really get rid of everything you've done since then, there are two possibilities. One, if you haven't published any of these commits, simply reset:
# This will destroy any local modifications.
# Don't do it if you have uncommitted work you want to keep.
git reset --hard 0d1d7fc32
# Alternatively, if there's work to keep:
git stash
git reset --hard 0d1d7fc32
git stash pop
# This saves the modifications, then reapplies that patch after resetting.
# You could get merge conflicts, if you've modified things which were
# changed since the commit you reset to.
If you mess up, you've already thrown away your local changes, but you can at least get back to where you were before by resetting again.
Undo published commits with new commits
On the other hand, if you've published the work, you probably don't want to reset the branch, since that's effectively rewriting history. In that case, you could indeed revert the commits. With Git, revert has a very specific meaning: create a commit with the reverse patch to cancel it out. This way you don't rewrite any history.
# This will create three separate revert commits:
git revert a867b4af 25eee4ca 0766c053
# It also takes ranges. This will revert the last two commits:
git revert HEAD~2..HEAD
#Similarly, you can revert a range of commits using commit hashes (non inclusive of first hash):
git revert 0d1d7fc..a867b4a
# Reverting a merge commit
git revert -m 1 <merge_commit_sha>
# To get just one, you could use `rebase -i` to squash them afterwards
# Or, you could do it manually (be sure to do this at top level of the repo)
# get your index and work tree into the desired state, without changing HEAD:
git checkout 0d1d7fc32 .
# Then commit. Be sure and write a good message describing what you just did
git commit
If you decide you didn't want to revert after all, you can revert the revert (as described here) or reset back to before the revert (see the previous section).
Lots of complicated and dangerous answers here, but it's actually easy:
git revert --no-commit 0766c053..HEAD
git commit
This will revert everything from the HEAD back to the commit hash, meaning it will recreate that commit state in the working tree as if every commit after 0766c053 had been walked back. You can then commit the current tree, and it will create a brand new commit essentially equivalent to the commit you "reverted" to.
(The --no-commit flag lets git revert all the commits at once- otherwise you'll be prompted for a message for each commit in the range, littering your history with unnecessary new commits.)
This is a safe and easy way to rollback to a previous state. No history is destroyed, so it can be used for commits that have already been made public.
Yarin
Rogue Coder?
Working on your own and just want it to work? Follow these instructions below, they’ve worked reliably for me and many others for years.
Working with others? Git is complicated. Read the comments below this answer before you do something rash.
Reverting Working Copy to Most Recent Commit
To revert to a previous commit, ignoring any changes:
git reset --hard HEAD
where HEAD is the last commit in your current branch
Reverting The Working Copy to an Older Commit
To revert to a commit that's older than the most recent commit:
# Resets index to former commit; replace '56e05fced' with your commit code
git reset 56e05fced
# Moves pointer back to previous HEAD
git reset --soft HEAD@{1}
git commit -m "Revert to 56e05fced"
# Updates working copy to reflect the new commit
git reset --hard
The best option for me and probably others is the Git reset option:
git reset --hard <commidId> && git clean -f
This has been the best option for me! It is simple, fast and effective!
** Note:** As mentioned in comments don't do this if you're sharing your branch with other people who have copies of the old commits
Also from the comments, if you wanted a less 'ballzy' method you could use
git clean -i
Pogrindis
Before answering let's add some background, explaining what this HEAD is.
First of all what is HEAD?
HEAD is simply a reference to the current commit (latest) on the current branch. There can only be a single HEAD at any given time (excluding git worktree).
The content of HEAD is stored inside .git/HEAD, and it contains the 40 bytes SHA-1 of the current commit.
detached HEAD
If you are not on the latest commit - meaning that HEAD is pointing to a prior commit in history it's called detached HEAD.
On the command line it will look like this - SHA-1 instead of the branch name since the HEAD is not pointing to the the tip of the current branch:
A few options on how to recover from a detached HEAD:
git checkout <commit_id>
git checkout -b <new branch> <commit_id>
git checkout HEAD~X // x is the number of commits t go back
This will checkout new branch pointing to the desired commit. This command will checkout to a given commit.
At this point you can create a branch and start to work from this point on:
# Checkout a given commit.
# Doing so will result in a `detached HEAD` which mean that the `HEAD`
# is not pointing to the latest so you will need to checkout branch
# in order to be able to update the code.
git checkout <commit-id>
# Create a new branch forked to the given commit
git checkout -b <branch name>
You can always use the reflog as well. git reflog will display any change which updated the HEAD and checking out the desired reflog entry will set the HEAD back to this commit.
Every time the HEAD is modified there will be a new entry in the reflog
# This will destroy any local modifications.
# Don't do it if you have uncommitted work you want to keep.
git reset --hard 0d1d7fc32
# Alternatively, if there's work to keep:
git stash
git reset --hard 0d1d7fc32
git stash pop
# This saves the modifications, then reapplies that patch after resetting.
# You could get merge conflicts, if you've modified things which were
# changed since the commit you reset to.
Note: (Since Git 2.7) you can also use the git rebase --no-autostash as well.
This schema illustrates which command does what. As you can see there reset && checkout modify the HEAD.
If you want to keep your changes, you can also use:
git reset --soft [previous Commit SHA id here]
Then it will save your changes.
kiran boghra
If you want to "uncommit", erase the last commit message, and put the modified files back in staging, you would use the command:
git reset --soft HEAD~1
--soft indicates that the uncommitted files should be retained as working files opposed to --hard which would discard them.
HEAD~1 is the last commit. If you want to rollback 3 commits you could use HEAD~3. If you want to rollback to a specific revision number, you could also do that using its SHA hash.
This is an extremely useful command in situations where you committed the wrong thing and you want to undo that last commit.
I have tried a lot of ways to revert local changes in Git, and it seems that this works the best if you just want to revert to the latest commit state.
git add . && git checkout master -f
Short description:
It will NOT create any commits as git revert does.
It will NOT detach your HEAD like git checkout <commithashcode> does.
It WILL override all your local changes and DELETE all added files since the last commit in the branch.
It works only with branches names, so you can revert only to latest commit in the branch this way.
I found a much more convenient and simple way to achieve the results above:
git add . && git reset --hard HEAD
where HEAD points to the latest commit at you current branch.
It is the same code code as boulder_ruby suggested, but I have added git add . before git reset --hard HEAD to erase all new files created since the last commit since this is what most people expect I believe when reverting to the latest commit.
Roman Minenok
The best way is:
git reset --hard <commidId> && git push --force
This will reset the branch to the specific commit and then will upload the remote server with the same commits as you have in local.
Be careful with the --force flag as it removes all the subsequent commits after the selected commit without the option to recover them.
david.t_92
OK, going back to a previous commit in Git is quite easy...
Revert back without keeping the changes:
git reset --hard <commit>
Revert back with keeping the changes:
git reset --soft <commit>
Explanation: using git reset, you can reset to a specific state. It's common using it with a commit hash as you see above.
But as you see the difference is using the two flags --soft and --hard, by default git reset using --soft flag, but it's a good practice always using the flag, I explain each flag:
--soft
The default flag as explained, not need to provide it, does not change the working tree, but it adds all changed files ready to commit, so you go back to the commit status which changes to files get unstaged.
--hard
Be careful with this flag. It resets the working tree and all changes to tracked files and all will be gone!
I also created the image below that may happen in a real life working with Git:
Alireza
Assuming you're talking about master and on that respective branch (that said, this could be any working branch you're concerned with):
# Reset local master branch to November 3rd commit ID
git reset --hard 0d1d7fc32e5a947fbd92ee598033d85bfc445a50
# Reset remote master branch to November 3rd commit ID
git push -f origin 0d1d7fc32e5a947fbd92ee598033d85bfc445a50:master
I found the answer from in a blog post (now no longer exists)
Note that this is Resetting and Forcing the change to the remote, so that if others on your team have already git pulled, you will cause problems for them. You are destroying the change history, which is an important reason why people use git in the first place.
Better to use revert (see other answers) than reset.
If you're a one man team then it probably doesn't matter.
markreyes
Say you have the following commits in a text file named ~/commits-to-revert.txt (I used git log --pretty=oneline to get them)
Create a Bash shell script to revert each of them:
#!/bin/bash
cd /path/to/working/copy
for i in `cat ~/commits-to-revert.txt`
do
git revert $i --no-commit
done
This reverts everything back to the previous state, including file and directory creations, and deletions, commit it to your branch and you retain the history, but you have it reverted back to the same file structure. Why Git doesn't have a git revert --to <hash> is beyond me.
Lance Caraccioli
Extra Alternatives to Jefromi's Solutions
Jefromi's solutions are definitely the best ones, and you should definitely use them. However, for the sake of completeness, I also wanted to show these other alternative solutions that can also be used to revert a commit (in the sense that you create a new commit that undoes changes in previous commit, just like what git revert does).
To be clear, these alternatives are not the best way to revert commits, Jefromi's solutions are, but I just want to point out that you can also use these other methods to achieve the same thing as git revert.
# Reset the index to the desired commit
git reset --hard <commit>
# Move the branch pointer back to the previous HEAD
git reset --soft HEAD@{1}
# Commit the changes
git commit -m "Revert to <commit>"
This basically works by using the fact that soft resets will leave the state of the previous commit staged in the index/staging-area, which you can then commit.
Alternative 2: Delete the Current Tree and Replace with the New One
Similarly to alternative #1, this reproduces the state of <commit> in the current working copy. It is necessary to do git rm first because git checkout won't remove files that have been added since <commit>.
Community Wiki
Here is a much simpler way to go back to a previous commit (and have it in an uncommited state, to do with it whatever you like):
git reset HEAD~1
So, no need for commit ids and so on :)
Paul Walczewski
Caution! This command can cause losing commit history, if user put the wrong commit mistakenly. Always have en extra backup of your git some
where else just in case if you do mistakes, than you are a bit safer.
:)
I have had a similar issue and wanted to revert back to an earlier commit. In my case I was not interested to keep the newer commit, hence I used Hard.
This is how I did it:
git reset --hard CommitId && git clean -f
This will revert on the local repository, and here after using git push -f will update the remote repository.
git push -f
For instance, if you want to completely ignore the commit with the name enforce non-group manage policies from the next image
you'd run
git reset --hard dd52eb9 && git clean -f
followed by
git push -f
After, you won't see that commit (enforce non-group manage policies) there
maytham-ɯɐɥʇʎɐɯ
After all the changes, when you push all these commands, you might have to use:
git push -f ...
And not only git push.
sivi
There is a command (not a part of core Git, but it is in the git-extras package) specifically for reverting and staging old commits:
You can complete all these initial steps yourself and push back to the Git repository.
Pull the latest version of your repository from Bitbucket using the git pull --all command.
Run the Git log command with -n 4 from your terminal. The number after the -n determines the number of commits in the log starting from the most recent commit in your local history.
$ git log -n 4
Reset the head of your repository's history using the git reset --hard HEAD~N where N is the number of commits you want to take the head back. In the following example the head would be set back one commit, to the last commit in the repository history:
Push the change to Git repository using git push --force to force push the change.
If you want the Git repository to a previous commit:
till you get the required commit. To make the HEAD point to that, do
git reset --hard HEAD~1
or git reset --hard HEAD~2 or whatever.
tonythomas01
Revert to most recent commit and ignoring all local changes:
git reset --hard HEAD
Mohammed Irfan Tirupattur
If the situation is an urgent one, and you just want to do what the questioner asked in a quick and dirty way, assuming your project is under a directory called, for example, "my project":
QUICK AND DIRTY: depending on the circumstances, quick and dirty may in fact be very GOOD. What my solution here does is NOT replace irreversibly the files you have in your working directory with files hauled up/extracted from the depths of the git repository lurking beneath your .git/ directory using fiendishly clever and diabolically powerful git commands, of which there are many. YOU DO NOT HAVE TO DO SUCH DEEP-SEA DIVING TO RECOVER what may appear to be a disastrous situation, and attempting to do so without sufficient expertise may prove fatal.
Copy the whole directory and call it something else, like "my project - copy". Assuming your git repository ("repo") files are under the "my project" directory (the default place for them, under a directory called ".git"), you will now have copied both your work files and your repo files.
This will return the state of the repo under "my project" to what it was when you made that commit (a "commit" means a snapshot of your working files). All commits since then will be lost forever under "my project", BUT... they will still be present in the repo under "my project - copy" since you copied all those files - including the ones under .../.git/.
You then have two versions on your system... you can examine or copy or modify files of interest, or whatever, from the previous commit. You can completely discard the files under "my project - copy", if you have decided the new work since the restored commit was going nowhere...
The obvious thing if you want to carry on with the state of the project without actually discarding the work since this retrieved commit is to rename your directory again: Delete the project containing the retrieved commit (or give it a temporary name) and rename your "my project - copy" directory back to "my project". Then maybe try to understand some of the other answers here, and probably do another commit fairly soon.
Git is a brilliant creation but absolutely no-one is able to just "pick it up on the fly": also people who try to explain it far too often assume prior knowledge of other VCS [Version Control Systems] and delve far too deep far too soon, and commit other crimes, like using interchangeable terms for "checking out" - in ways which sometimes appear almost calculated to confuse a beginner.
To save yourself much stress, learn from my scars. You have to pretty much have to read a book on Git - I'd recommend "Version Control with Git". Do it sooner rather than later. If you do, bear in mind that much of the complexity of Git comes from branching and then remerging: you can skip those parts in any book. From your question there's no reason why people should be blinding you with science.
Especially if, for example, this is a desperate situation and you're a newbie with Git!
PS: One other thought: It is (now) actually quite simple to keep the Git repo in a directory other than the one with the working files. This would mean you would not have to copy the entire Git repository using the above quick & dirty solution. See the answer by Fryer using --separate-git-dirhere. Be warned, though: If you have a "separate-directory" repository which you don't copy, and you do a hard reset, all versions subsequent to the reset commit will be lost forever, unless you have, as you absolutely should, regularly backed up your repository, preferably to the Cloud (e.g. Google Drive) among other places.
On this subject of "backing up to the Cloud", the next step is to open an account (free of course) with GitHub or (better in my view) GitLab. You can then regularly do a git push command to make your Cloud repo up-to-date "properly". But again, talking about this may be too much too soon.
mike rodent
This is one more way to directly reset to a recent commit
git stash
git stash clear
It directly clears all the changes that you have been making since the last commit.
PS: It has a little problem; it also deletes all you recently stored stash changes. Which I guess in most cases should not matter.
Point Networks
Revert is the command to rollback the commits.
git revert <commit1> <commit2>
Sample:
git revert 2h3h23233
It is capable of taking range from the HEAD like below. Here 1 says "revert last commit."
git revert HEAD~1..HEAD
And then do:
git push
Sireesh Yarlagadda
To completely clean a coder's directory up from some accidental changes, we used:
git add -A .
git reset --hard HEAD
Just git reset --hard HEAD will get rid of modifications, but it won't get rid of "new" files. In their case they'd accidentally dragged an important folder somewhere random, and all those files were being treated as new by Git, so a reset --hard didn't fix it. By running the git add -A . beforehand, it explicitly tracked them all with git, to be wiped out by the reset.
Chris Moschini
I believe some people may come to this question wanting to know how to rollback committed changes they've made in their master - ie throw everything away and go back to origin/master, in which case, do this:
To keep the changes from the previous commit to HEAD and move to the previous commit, do:
git reset <SHA>
If changes are not required from the previous commit to HEAD and just discard all changes, do:
git reset --hard <SHA>
Vishnu Atrai
Try resetting to the desired commit:
git reset <COMMIT_ID>
To check COMMIT_ID use:
git log
This will reset all changed files to un-added state.
Now you can checkout all un-added files by
git checkout .
To verify your changes use:
git log
UPDATE
If you have one and only commit in your repo, try
git update-ref -d HEAD
optimistanoop
Idea: You basically want to replace the current working tree state with the one from a previous commit and then create a commit out of it. Ignored files should best be not changed. Here is how:
Emtpy the working tree *.
git rm -r --cached . && git clean -f -d
Bring the working tree in the state we want **.
git checkout 0d1d7fc3 .
Create the revert commit.
git add --all && git commit -m "revert to 0d1d7fc3"
At first I thought that Yarins answer would be the best, but it doesn't work for merge commits. This solution does.
Additionally it does not delete anything (pushed or upushed) from the history. It produces one clean commit which represents the state we want to revert back to.
* by removing untracked but not ignored files (the ones specified in .gitignore) from working tree. The working tree is empty except for the ignored files which we wanted to keep (if not specifiy -x option for clean)
** When a path is specified (here: .), checkout leaves HEAD alone.
Willi Mentzel
As your commits are pushed remotely, you need to remove them. Let me assume your branch is develop and it is pushed over origin.
You first need to remove develop from origin:
git push origin :develop (note the colon)
Then you need to get develop to the status you want, let me assume the commit hash is EFGHIJK: