How to get rid of some changeset in Hg?

前端 未结 5 1222
青春惊慌失措
青春惊慌失措 2020-12-29 14:50

Hey, I want to revert some changesets in Hg but I\'m having a hard time doing so.

I commited some changes by accident, so I wanted to undo that. After trying a littl

相关标签:
5条回答
  • 2020-12-29 15:16

    Not certain about the git things, but if you haven't pushed, you can use hg rollback. If you have pushed, there's not really anything decent you can do about it.

    As for the current revision, see hg id.

    0 讨论(0)
  • 2020-12-29 15:19

    See hg strip from the mq extension.

    0 讨论(0)
  • 2020-12-29 15:23

    Mercurial intentionally doesn't make this easy. Mercurial's mindset is that code committed should continue to exist -- that the record of what didn't work out is almost as important as a record of what did. It's better to commit the negation of your work (which is what backout does) and still have the record than it is to actually discard it.

    If, however, you just can't leave that change around (good reasons: it contains a password you can't change, bad reason: It just didn't work out) you have a few options:

    • strip: I don't like strip it modifies the repository and doesn't put the stuff back in the working dir. Strip is disabled by default for a reason.
    • clone: Just do a hg clone -r LASTCHANGESETYOULIKE oldrepo newrepo and now newrepo has everything up to and including LASTCHANGESETYOULIKE. Archive oldrepo and rename newrepo to oldrepo.
    • rollback: This is the one-level undo of the mercurial world. It would have undone your commit right after you did it, but it doesn't work if you've done any other pulls or commits, which you have.
    0 讨论(0)
  • 2020-12-29 15:28

    Here is how to do git reset --soft. Let us assume you have a graph like this:

    ... --- [A] --- [B] --- [X] --- [-X]
    

    where X is the bad commit and -X is the backout you made. You now want to get rid of X and -X while leaving the working copy looking like it did when you committed X. You do

    $ hg update B
    $ hg revert --all --rev X # working copy now looks just like in X
    $ hg strip --force X      # get rid of X and -X
    

    If you want, you can make an alias for this with Mercurial 1.7:

    [alias]
    softreset = !hg update 'p1($1)' &&
                 hg revert --all --rev $1 &&
                 hg strip --force $1
    

    The dirty working copy after hg revert makes it necessary to use hg strip --force. You use this new command as

    $ hg softreset 10
    

    which will remove 10 and any descendents while leaving the changes in 10 in the working copy. You can of course take this further and implement a hg reset command:

    [alias]
    reset = !test "$1" = "--hard" && hg strip $2 || hg softreset $2
    

    The biggest problem with these aliases is the poor error handling. An extension written in Python would be much more robust and maintainable -- perhaps you could make such a "reset" extension and publish it on Bitbucket :)

    0 讨论(0)
  • 2020-12-29 15:29

    You've got some good answers already, but a couple additional notes:

    • If you've got uncommitted local changes, shelve will set them aside for you for later retrieval
    • If your changeset has been pushed to another repository, then short of editing both repositories, a backout is your only option
    • A backout allows you to make a new changeset which is the inverse of the original changeset, getting rid of the undesired changes without changing the history of the repository
    • A rollback is normally your best option if you haven't already pushed to a remote repo since it obliterates the changeset while leaving your working copy intact
    • Strip will get rid of a whole branch of changes, but like rollbacks it will only stay gone if you haven't pushed the changes anywhere else
    • The same goes for cloning a repository up to a prior revision - the other tools are easier anyway
    0 讨论(0)
提交回复
热议问题