##// END OF EJS Templates
merge: make in-memory changes visible to external update hooks...
merge: make in-memory changes visible to external update hooks 51844b8b5017 (while 3.4 code-freeze) made all 'update' hooks run after releasing wlock for visibility of in-memory dirstate changes. But this breaks paired invocation of 'preupdate' and 'update' hooks. For example, 'hg backout --merge' for TARGET revision, which isn't parent of CURRENT, consists of steps below: 1. update from CURRENT to TARGET 2. commit BACKOUT revision, which backs TARGET out 3. update from BACKOUT to CURRENT 4. merge TARGET into CURRENT Then, we expects hooks to run in the order below: - 'preupdate' on CURRENT for (1) - 'update' on TARGET for (1) - 'preupdate' on BACKOUT for (3) - 'update' on CURRENT for (3) - 'preupdate' on TARGET for (4) - 'update' on CURRENT/TARGET for (4) But hooks actually run in the order below: - 'preupdate' on CURRENT for (1) - 'preupdate' on BACKOUT for (3) - 'preupdate' on TARGET for (4) - 'update' on TARGET for (1), but actually on CURRENT/TARGET - 'update' on CURRENT for (3), but actually on CURRENT/TARGET - 'update' on CURRENT for (4), but actually on CURRENT/TARGET Root cause of the issue focused by 51844b8b5017 is that external 'update' hook process can't view in-memory changes (especially, of dirstate), because they aren't written out until the end of transaction (or wlock). Now, hooks can be invoked just after updating, because previous patches made in-memory changes visible to external process. This patch may break backward compatibility from the point of view of "scheduling hook execution", but should be reasonable because 'update' hooks had been executed in this order before 3.4. This patch tests "hg backout" and "hg unshelve", because the former activates the transaction before 'update' hook invocation, but the former doesn't.

File last commit:

r26697:008761af default
r26752:949e8c62 default
Show More
shelve.py
833 lines | 29.9 KiB | text/x-python | PythonLexer
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 # shelve.py - save/restore working directory state
#
# Copyright 2013 Facebook, Inc.
#
# This software may be used and distributed according to the terms of the
# GNU General Public License version 2 or any later version.
"""save and restore changes to the working directory
The "hg shelve" command saves changes made to the working directory
and reverts those changes, resetting the working directory to a clean
state.
Later on, the "hg unshelve" command restores the changes saved by "hg
shelve". Changes can be restored even after updating to a different
parent, in which case Mercurial's merge machinery will resolve any
conflicts if necessary.
You can have more than one shelved change outstanding at a time; each
shelved change has a distinct name. For details, see the help for "hg
shelve".
"""
Martin von Zweigbergk
util: drop alias for collections.deque...
r25113 import collections
Colin Chan
shelve: always backup shelves instead of deleting them...
r25712 import itertools
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 from mercurial.i18n import _
Mads Kiilerich
shelve: publicancestors do not have to visit nullrev...
r20407 from mercurial.node import nullid, nullrev, bin, hex
Mads Kiilerich
shelve: mention walk options in help
r20409 from mercurial import changegroup, cmdutil, scmutil, phases, commands
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 from mercurial import error, hg, mdiff, merge, patch, repair, util
Matt Mackall
shelve: add a bundlerepo method
r22898 from mercurial import templatefilters, exchange, bundlerepo
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 from mercurial import lock as lockmod
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 from hgext import rebase
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 import errno
cmdtable = {}
command = cmdutil.command(cmdtable)
Augie Fackler
extensions: document that `testedwith = 'internal'` is special...
r25186 # Note for extension authors: ONLY specify testedwith = 'internal' for
# extensions which SHIP WITH MERCURIAL. Non-mainline extensions should
# be specifying the version(s) of Mercurial they are tested with, or
# leave the attribute unspecified.
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 testedwith = 'internal'
Colin Chan
shelve: only keep the latest N shelve backups...
r25713 backupdir = 'shelve-backup'
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 class shelvedfile(object):
Pierre-Yves David
shelve: add minimal documentation to all functions...
r19909 """Helper for the file storing a single shelve
Martin von Zweigbergk
shelve: avoid writing file that is never read from...
r22581 Handles common functions on shelve files (.hg/.patch) using
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 the vfs layer"""
def __init__(self, repo, name, filetype=None):
self.repo = repo
self.name = name
self.vfs = scmutil.vfs(repo.join('shelved'))
Colin Chan
shelve: only keep the latest N shelve backups...
r25713 self.backupvfs = scmutil.vfs(repo.join(backupdir))
Eric Sumner
changegroup.writebundle: provide ui...
r23895 self.ui = self.repo.ui
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if filetype:
self.fname = name + '.' + filetype
else:
self.fname = name
def exists(self):
return self.vfs.exists(self.fname)
def filename(self):
return self.vfs.join(self.fname)
Colin Chan
shelve: always backup shelves instead of deleting them...
r25712 def backupfilename(self):
def gennames(base):
yield base
base, ext = base.rsplit('.', 1)
for i in itertools.count(1):
yield '%s-%d.%s' % (base, i, ext)
name = self.backupvfs.join(self.fname)
for n in gennames(name):
if not self.backupvfs.exists(n):
return n
def movetobackup(self):
if not self.backupvfs.isdir():
self.backupvfs.makedir()
util.rename(self.filename(), self.backupfilename())
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
def stat(self):
return self.vfs.stat(self.fname)
def opener(self, mode='rb'):
try:
return self.vfs(self.fname, mode)
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except IOError as err:
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if err.errno != errno.ENOENT:
raise
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("shelved change '%s' not found") % self.name)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
FUJIWARA Katsunori
shelve: add "applybundle()" to invoke "readbundle()" with relative path and vfs...
r20982 def applybundle(self):
fp = self.opener()
try:
Pierre-Yves David
bundle2: add a ui argument to readbundle...
r21064 gen = exchange.readbundle(self.repo.ui, fp, self.fname, self.vfs)
Augie Fackler
shelve: use cg?unpacker.apply() instead of changegroup.addchangegroup()
r26697 gen.apply(self.repo, 'unshelve',
'bundle:' + self.vfs.join(self.fname),
targetphase=phases.secret)
FUJIWARA Katsunori
shelve: add "applybundle()" to invoke "readbundle()" with relative path and vfs...
r20982 finally:
fp.close()
Matt Mackall
shelve: add a bundlerepo method
r22898 def bundlerepo(self):
return bundlerepo.bundlerepository(self.repo.baseui, self.repo.root,
self.vfs.join(self.fname))
Pierre-Yves David
shelve: move changegroup generation inside writebundle...
r26506 def writebundle(self, bases, node):
Pierre-Yves David
shelve: bundle using bundle2 if repository is general delta (issue4862)...
r26507 btype = 'HG10BZ'
cgversion = '01'
compression = None
if 'generaldelta' in self.repo.requirements:
btype = 'HG20'
cgversion = '02'
compression = 'BZ'
cg = changegroup.changegroupsubset(self.repo, bases, [node], 'shelve',
version=cgversion)
changegroup.writebundle(self.ui, cg, self.fname, btype, self.vfs,
compression=compression)
FUJIWARA Katsunori
shelve: add "writebundle()" to invoke "writebundle()" with relative path and vfs
r20983
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 class shelvedstate(object):
Augie Fackler
shelve: some docstring cleanups
r19911 """Handle persistence during unshelving operations.
Pierre-Yves David
shelve: add minimal documentation to all functions...
r19909
Handles saving and restoring a shelved state. Ensures that different
Augie Fackler
shelve: some docstring cleanups
r19911 versions of a shelved state are possible and handles them appropriately.
"""
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 _version = 1
_filename = 'shelvedstate'
@classmethod
def load(cls, repo):
Angel Ezquerra
localrepo: remove all external users of localrepo.opener...
r23877 fp = repo.vfs(cls._filename)
Pierre-Yves David
shelve: drop pickle usage...
r19904 try:
version = int(fp.readline().strip())
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
Pierre-Yves David
shelve: drop pickle usage...
r19904 if version != cls._version:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('this version of shelve is incompatible '
Pierre-Yves David
shelve: drop pickle usage...
r19904 'with the version used in this repo'))
name = fp.readline().strip()
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 wctx = fp.readline().strip()
pendingctx = fp.readline().strip()
Pierre-Yves David
shelve: drop pickle usage...
r19904 parents = [bin(h) for h in fp.readline().split()]
stripnodes = [bin(h) for h in fp.readline().split()]
finally:
fp.close()
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
obj = cls()
obj.name = name
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 obj.wctx = repo[bin(wctx)]
obj.pendingctx = repo[bin(pendingctx)]
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 obj.parents = parents
obj.stripnodes = stripnodes
return obj
@classmethod
Durham Goode
unshelve: don't commit unknown files during unshelve (issue4113)...
r20149 def save(cls, repo, name, originalwctx, pendingctx, stripnodes):
Angel Ezquerra
localrepo: remove all external users of localrepo.opener...
r23877 fp = repo.vfs(cls._filename, 'wb')
Pierre-Yves David
shelve: drop pickle usage...
r19904 fp.write('%i\n' % cls._version)
fp.write('%s\n' % name)
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 fp.write('%s\n' % hex(originalwctx.node()))
fp.write('%s\n' % hex(pendingctx.node()))
Pierre-Yves David
shelve: drop pickle usage...
r19904 fp.write('%s\n' % ' '.join([hex(p) for p in repo.dirstate.parents()]))
fp.write('%s\n' % ' '.join([hex(n) for n in stripnodes]))
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 fp.close()
Pierre-Yves David
shelve: use the class constant in the clear method...
r19908 @classmethod
def clear(cls, repo):
util.unlinkpath(repo.join(cls._filename), ignoremissing=True)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
Colin Chan
shelve: only keep the latest N shelve backups...
r25713 def cleanupoldbackups(repo):
vfs = scmutil.vfs(repo.join(backupdir))
maxbackups = repo.ui.configint('shelve', 'maxbackups', 10)
hgfiles = [f for f in vfs.listdir() if f.endswith('.hg')]
hgfiles = sorted([(vfs.stat(f).st_mtime, f) for f in hgfiles])
FUJIWARA Katsunori
shelve: keep old backups if timestamp can't decide exact order of them...
r25774 if 0 < maxbackups and maxbackups < len(hgfiles):
bordermtime = hgfiles[-maxbackups][0]
else:
bordermtime = None
Colin Chan
shelve: only keep the latest N shelve backups...
r25713 for mtime, f in hgfiles[:len(hgfiles) - maxbackups]:
FUJIWARA Katsunori
shelve: keep old backups if timestamp can't decide exact order of them...
r25774 if mtime == bordermtime:
# keep it, because timestamp can't decide exact order of backups
continue
Colin Chan
shelve: only keep the latest N shelve backups...
r25713 base = f[:-3]
for ext in 'hg patch'.split():
try:
vfs.unlink(base + '.' + ext)
except OSError as err:
if err.errno != errno.ENOENT:
raise
FUJIWARA Katsunori
shelve: add utility to abort current transaction but keep dirstate...
r26522 def _aborttransaction(repo):
'''Abort current transaction for shelve/unshelve, but keep dirstate
'''
backupname = 'dirstate.shelve'
dirstatebackup = None
try:
# create backup of (un)shelved dirstate, because aborting transaction
# should restore dirstate to one at the beginning of the
# transaction, which doesn't include the result of (un)shelving
fp = repo.vfs.open(backupname, "w")
dirstatebackup = backupname
# clearing _dirty/_dirtypl of dirstate by _writedirstate below
# is unintentional. but it doesn't cause problem in this case,
# because no code path refers them until transaction is aborted.
repo.dirstate._writedirstate(fp) # write in-memory changes forcibly
tr = repo.currenttransaction()
tr.abort()
# restore to backuped dirstate
repo.vfs.rename(dirstatebackup, 'dirstate')
dirstatebackup = None
finally:
if dirstatebackup:
repo.vfs.unlink(dirstatebackup)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 def createcmd(ui, repo, pats, opts):
Augie Fackler
shelve: some docstring cleanups
r19911 """subcommand that creates a new shelve"""
Pierre-Yves David
shelve: add minimal documentation to all functions...
r19909
Pierre-Yves David
shelve: rename 'publicancestors' to something accurate (issue4737)...
r26602 def mutableancestors(ctx):
"""return all mutable ancestors for ctx (included)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
Mads Kiilerich
shelve: really pass publicancestors to changegroupsubset - not the parents...
r20408 Much faster than the revset ancestors(ctx) & draft()"""
Mads Kiilerich
shelve: publicancestors do not have to visit nullrev...
r20407 seen = set([nullrev])
Martin von Zweigbergk
util: drop alias for collections.deque...
r25113 visit = collections.deque()
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 visit.append(ctx)
while visit:
ctx = visit.popleft()
Mads Kiilerich
shelve: really pass publicancestors to changegroupsubset - not the parents...
r20408 yield ctx.node()
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 for parent in ctx.parents():
rev = parent.rev()
if rev not in seen:
seen.add(rev)
if parent.mutable():
visit.append(parent)
wctx = repo[None]
parents = wctx.parents()
if len(parents) > 1:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('cannot shelve while merging'))
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 parent = parents[0]
# we never need the user, so we use a generic user for all shelve operations
user = 'shelve@localhost'
Ryan McElroy
bookmarks: rename bookmarkcurrent to activebookmark (API)...
r24947 label = repo._activebookmark or parent.branch() or 'default'
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
# slashes aren't allowed in filenames, therefore we rename it
Sean Farley
shelve: remove unused variable caught by pyflakes
r20937 label = label.replace('/', '_')
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
def gennames():
yield label
for i in xrange(1, 100):
yield '%s-%02d' % (label, i)
def commitfunc(ui, repo, message, match, opts):
Sean Farley
shelve: only save mq state if enabled...
r19885 hasmq = util.safehasattr(repo, 'mq')
if hasmq:
saved, repo.mq.checkapplied = repo.mq.checkapplied, False
Pierre-Yves David
shelve: do not retract phase boundary by hand...
r22040 backup = repo.ui.backupconfig('phases', 'new-commit')
David Soria Parra
shelve: allow shelving of a change with an mq patch applied...
r19856 try:
Pierre-Yves David
shelve: do not retract phase boundary by hand...
r22040 repo.ui. setconfig('phases', 'new-commit', phases.secret)
FUJIWARA Katsunori
shelve: pass 'editform' argument to 'cmdutil.getcommiteditor'...
r22005 editor = cmdutil.getcommiteditor(editform='shelve.shelve', **opts)
FUJIWARA Katsunori
shelve: accept '--edit' like other commands creating new changeset...
r21852 return repo.commit(message, user, opts.get('date'), match,
FUJIWARA Katsunori
shelve: pass 'editform' argument to 'cmdutil.getcommiteditor'...
r22005 editor=editor)
David Soria Parra
shelve: allow shelving of a change with an mq patch applied...
r19856 finally:
Pierre-Yves David
shelve: do not retract phase boundary by hand...
r22040 repo.ui.restoreconfig(backup)
Sean Farley
shelve: only save mq state if enabled...
r19885 if hasmq:
repo.mq.checkapplied = saved
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
if parent.node() != nullid:
Mads Kiilerich
shelve: add 'changes to' prefix to default shelve message...
r20411 desc = "changes to '%s'" % parent.description().split('\n', 1)[0]
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 else:
Mads Kiilerich
shelve: add 'changes to' prefix to default shelve message...
r20411 desc = '(changes in empty repository)'
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
if not opts['message']:
opts['message'] = desc
name = opts['name']
FUJIWARA Katsunori
bookmarks: use recordchange instead of writing if transaction is active...
r26520 wlock = lock = tr = None
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 try:
wlock = repo.wlock()
lock = repo.lock()
Mads Kiilerich
spelling: random spell checker fixes
r19951 # use an uncommitted transaction to generate the bundle to avoid
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 # pull races. ensure we don't print the abort message to stderr.
tr = repo.transaction('commit', report=lambda x: None)
if name:
if shelvedfile(repo, name, 'hg').exists():
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("a shelved change named '%s' already exists"
) % name)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 else:
for n in gennames():
if not shelvedfile(repo, n, 'hg').exists():
name = n
break
else:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("too many shelved changes named '%s'") %
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 label)
# ensure we are not creating a subdirectory or a hidden file
if '/' in name or '\\' in name:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('shelved change names may not contain slashes'))
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if name.startswith('.'):
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("shelved change names may not start with '.'"))
Laurent Charignon
shelve: add interactive mode command line option
r24477 interactive = opts.get('interactive', False)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
Laurent Charignon
shelve: add interactive mode...
r24478 def interactivecommitfunc(ui, repo, *pats, **opts):
match = scmutil.match(repo['.'], pats, {})
message = opts['message']
return commitfunc(ui, repo, message, match, opts)
if not interactive:
node = cmdutil.commit(ui, repo, commitfunc, pats, opts)
else:
FUJIWARA Katsunori
shelve: omit incorrect 'commit' suggestion at 'hg shelve -i'...
r25799 node = cmdutil.dorecord(ui, repo, interactivecommitfunc, None,
Laurent Charignon
shelve: add interactive mode...
r24478 False, cmdutil.recordfilter, *pats, **opts)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if not node:
stat = repo.status(match=scmutil.match(repo[None], pats, opts))
Martin von Zweigbergk
shelve: access status fields by name rather than index
r22922 if stat.deleted:
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 ui.status(_("nothing changed (%d missing files, see "
Martin von Zweigbergk
shelve: access status fields by name rather than index
r22922 "'hg status')\n") % len(stat.deleted))
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 else:
ui.status(_("nothing changed\n"))
return 1
Pierre-Yves David
shelve: rename 'publicancestors' to something accurate (issue4737)...
r26602 bases = list(mutableancestors(repo[node]))
Pierre-Yves David
shelve: move changegroup generation inside writebundle...
r26506 shelvedfile(repo, name, 'hg').writebundle(bases, node)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 cmdutil.export(repo, [node],
fp=shelvedfile(repo, name, 'patch').opener('wb'),
opts=mdiff.diffopts(git=True))
David Soria Parra
shelve: copy bookmarks and restore them after a commit...
r19874
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if ui.formatted():
desc = util.ellipsis(desc, ui.termwidth())
ui.status(_('shelved as %s\n') % name)
hg.update(repo, parent.node())
FUJIWARA Katsunori
shelve: restore shelved dirstate explicitly after aborting transaction...
r26523
_aborttransaction(repo)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 finally:
FUJIWARA Katsunori
shelve: restore shelved dirstate explicitly after aborting transaction...
r26523 lockmod.release(tr, lock, wlock)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
def cleanupcmd(ui, repo):
Augie Fackler
shelve: some docstring cleanups
r19911 """subcommand that deletes all shelves"""
Pierre-Yves David
shelve: add minimal documentation to all functions...
r19909
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 wlock = None
try:
wlock = repo.wlock()
Mads Kiilerich
cleanup: avoid _ for local unused tmp variables - that is reserved for i18n...
r22199 for (name, _type) in repo.vfs.readdir('shelved'):
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 suffix = name.rsplit('.', 1)[-1]
Martin von Zweigbergk
shelve: avoid writing file that is never read from...
r22581 if suffix in ('hg', 'patch'):
Colin Chan
shelve: always backup shelves instead of deleting them...
r25712 shelvedfile(repo, name).movetobackup()
Colin Chan
shelve: only keep the latest N shelve backups...
r25713 cleanupoldbackups(repo)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 finally:
lockmod.release(wlock)
def deletecmd(ui, repo, pats):
Augie Fackler
shelve: some docstring cleanups
r19911 """subcommand that deletes a specific shelve"""
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if not pats:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('no shelved changes specified!'))
Matt Mackall
shelve: use try/except/finally
r25080 wlock = repo.wlock()
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 try:
Matt Mackall
shelve: use try/except/finally
r25080 for name in pats:
for suffix in 'hg patch'.split():
Colin Chan
shelve: always backup shelves instead of deleting them...
r25712 shelvedfile(repo, name, suffix).movetobackup()
Colin Chan
shelve: only keep the latest N shelve backups...
r25713 cleanupoldbackups(repo)
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except OSError as err:
Matt Mackall
shelve: use try/except/finally
r25080 if err.errno != errno.ENOENT:
raise
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("shelved change '%s' not found") % name)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 finally:
lockmod.release(wlock)
def listshelves(repo):
Augie Fackler
shelve: some docstring cleanups
r19911 """return all shelves in repo as list of (time, filename)"""
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 try:
names = repo.vfs.readdir('shelved')
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except OSError as err:
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if err.errno != errno.ENOENT:
raise
return []
info = []
Mads Kiilerich
cleanup: avoid _ for local unused tmp variables - that is reserved for i18n...
r22199 for (name, _type) in names:
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 pfx, sfx = name.rsplit('.', 1)
if not pfx or sfx != 'patch':
continue
st = shelvedfile(repo, name).stat()
info.append((st.st_mtime, shelvedfile(repo, pfx).filename()))
return sorted(info, reverse=True)
def listcmd(ui, repo, pats, opts):
Augie Fackler
shelve: some docstring cleanups
r19911 """subcommand that displays the list of shelves"""
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 pats = set(pats)
width = 80
if not ui.plain():
width = ui.termwidth()
namelabel = 'shelve.newest'
for mtime, name in listshelves(repo):
sname = util.split(name)[1]
if pats and sname not in pats:
continue
ui.write(sname, label=namelabel)
namelabel = 'shelve.name'
if ui.quiet:
ui.write('\n')
continue
ui.write(' ' * (16 - len(sname)))
used = 16
David Soria Parra
shelve: new output format for shelve listings...
r19855 age = '(%s)' % templatefilters.age(util.makedate(mtime), abbrev=True)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 ui.write(age, label='shelve.age')
David Soria Parra
shelve: new output format for shelve listings...
r19855 ui.write(' ' * (12 - len(age)))
used += 12
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 fp = open(name + '.patch', 'rb')
try:
while True:
line = fp.readline()
if not line:
break
if not line.startswith('#'):
desc = line.rstrip()
if ui.formatted():
desc = util.ellipsis(desc, width - used)
ui.write(desc)
break
ui.write('\n')
if not (opts['patch'] or opts['stat']):
continue
difflines = fp.readlines()
if opts['patch']:
for chunk, label in patch.difflabel(iter, difflines):
ui.write(chunk, label=label)
if opts['stat']:
for chunk, label in patch.diffstatui(difflines, width=width,
git=True):
ui.write(chunk, label=label)
finally:
fp.close()
Tony Tung
shelve: allow --patch and --stat without --list for a single shelf...
r25104 def singlepatchcmds(ui, repo, pats, opts, subcommand):
"""subcommand that displays a single shelf"""
if len(pats) != 1:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("--%s expects a single shelf") % subcommand)
Tony Tung
shelve: allow --patch and --stat without --list for a single shelf...
r25104 shelfname = pats[0]
if not shelvedfile(repo, shelfname, 'patch').exists():
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("cannot find shelf %s") % shelfname)
Tony Tung
shelve: allow --patch and --stat without --list for a single shelf...
r25104
listcmd(ui, repo, pats, opts)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 def checkparents(repo, state):
Pierre-Yves David
shelve: add minimal documentation to all functions...
r19909 """check parent while resuming an unshelve"""
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if state.parents != repo.dirstate.parents():
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('working directory parents do not match unshelve '
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 'state'))
Takumi IINO
shelve: make unshelve work even if it don't run in repository root...
r19943 def pathtofiles(repo, files):
cwd = repo.getcwd()
return [repo.pathto(f, cwd) for f in files]
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 def unshelveabort(ui, repo, state, opts):
Pierre-Yves David
shelve: add minimal documentation to all functions...
r19909 """subcommand that abort an in-progress unshelve"""
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 wlock = repo.wlock()
lock = None
try:
checkparents(repo, state)
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961
util.rename(repo.join('unshelverebasestate'),
repo.join('rebasestate'))
try:
rebase.rebase(ui, repo, **{
'abort' : True
})
except Exception:
util.rename(repo.join('rebasestate'),
repo.join('unshelverebasestate'))
raise
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 lock = repo.lock()
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961
Durham Goode
unshelve: don't commit unknown files during unshelve (issue4113)...
r20149 mergefiles(ui, repo, state.wctx, state.pendingctx)
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961
Jordi Gutiérrez Hermoso
strip: remove -b/--backup codepaths...
r22057 repair.strip(ui, repo, state.stripnodes, backup=False, topic='shelve')
Christian Delahousse
shelve: delete shelve statefile on any exception during abort...
r26681 finally:
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 shelvedstate.clear(repo)
ui.warn(_("unshelve of '%s' aborted\n") % state.name)
lockmod.release(lock, wlock)
Durham Goode
unshelve: don't commit unknown files during unshelve (issue4113)...
r20149 def mergefiles(ui, repo, wctx, shelvectx):
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 """updates to wctx and merges the changes from shelvectx into the
Durham Goode
unshelve: don't commit unknown files during unshelve (issue4113)...
r20149 dirstate."""
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 oldquiet = ui.quiet
try:
ui.quiet = True
hg.update(repo, wctx.node())
files = []
files.extend(shelvectx.files())
files.extend(shelvectx.parents()[0].files())
Durham Goode
unshelve: don't commit unknown files during unshelve (issue4113)...
r20149
# revert will overwrite unknown files, so move them out of the way
Martin von Zweigbergk
shelve: access status fields by name rather than index
r22922 for file in repo.status(unknown=True).unknown:
Durham Goode
unshelve: don't commit unknown files during unshelve (issue4113)...
r20149 if file in files:
util.rename(file, file + ".orig")
Matt Mackall
unshelve: silence internal revert...
r22184 ui.pushbuffer(True)
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 cmdutil.revert(ui, repo, shelvectx, repo.dirstate.parents(),
*pathtofiles(repo, files),
**{'no_backup': True})
Matt Mackall
unshelve: silence internal revert...
r22184 ui.popbuffer()
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 finally:
ui.quiet = oldquiet
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 def unshelvecleanup(ui, repo, name, opts):
Augie Fackler
shelve: some docstring cleanups
r19911 """remove related files after an unshelve"""
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if not opts['keep']:
Martin von Zweigbergk
shelve: avoid writing file that is never read from...
r22581 for filetype in 'hg patch'.split():
Colin Chan
shelve: always backup shelves instead of deleting them...
r25712 shelvedfile(repo, name, filetype).movetobackup()
Colin Chan
shelve: only keep the latest N shelve backups...
r25713 cleanupoldbackups(repo)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
def unshelvecontinue(ui, repo, state, opts):
Pierre-Yves David
shelve: add minimal documentation to all functions...
r19909 """subcommand to continue an in-progress unshelve"""
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 # We're finishing off a merge. First parent is our original
# parent, second is the temporary "fake" commit we're unshelving.
wlock = repo.wlock()
lock = None
try:
checkparents(repo, state)
ms = merge.mergestate(repo)
if [f for f in ms if ms[f] == 'u']:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 _("unresolved conflicts, can't continue"),
hint=_("see 'hg resolve', then 'hg unshelve --continue'"))
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 lock = repo.lock()
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961
util.rename(repo.join('unshelverebasestate'),
repo.join('rebasestate'))
try:
rebase.rebase(ui, repo, **{
'continue' : True
})
except Exception:
util.rename(repo.join('rebasestate'),
repo.join('unshelverebasestate'))
raise
shelvectx = repo['tip']
if not shelvectx in state.pendingctx.children():
# rebase was a no-op, so it produced no child commit
shelvectx = state.pendingctx
Jordi Gutiérrez Hermoso
shelve: don't delete "." when rebase is a no-op (issue4398)...
r22842 else:
# only strip the shelvectx if the rebase produced it
state.stripnodes.append(shelvectx.node())
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961
Durham Goode
unshelve: don't commit unknown files during unshelve (issue4113)...
r20149 mergefiles(ui, repo, state.wctx, shelvectx)
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961
Jordi Gutiérrez Hermoso
strip: remove -b/--backup codepaths...
r22057 repair.strip(ui, repo, state.stripnodes, backup=False, topic='shelve')
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 shelvedstate.clear(repo)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 unshelvecleanup(ui, repo, state.name, opts)
ui.status(_("unshelve of '%s' complete\n") % state.name)
finally:
lockmod.release(lock, wlock)
@command('unshelve',
[('a', 'abort', None,
_('abort an incomplete unshelve operation')),
('c', 'continue', None,
_('continue an incomplete unshelve operation')),
('', 'keep', None,
Mads Kiilerich
shelve: introduce secret option for using fixed date for temporary commit...
r20960 _('keep shelve after unshelving')),
('', 'date', '',
_('set date for temporary commits (DEPRECATED)'), _('DATE'))],
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 _('hg unshelve [SHELVED]'))
def unshelve(ui, repo, *shelved, **opts):
"""restore a shelved change to the working directory
This command accepts an optional name of a shelved change to
restore. If none is given, the most recent shelved change is used.
If a shelved change is applied successfully, the bundle that
Colin Chan
shelve: always backup shelves instead of deleting them...
r25712 contains the shelved changes is moved to a backup location
(.hg/shelve-backup).
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
Since you can restore a shelved change on top of an arbitrary
commit, it is possible that unshelving will result in a conflict
between your changes and the commits you are unshelving onto. If
this occurs, you must resolve the conflict, then use
``--continue`` to complete the unshelve operation. (The bundle
Colin Chan
shelve: always backup shelves instead of deleting them...
r25712 will not be moved until you successfully complete the unshelve.)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
(Alternatively, you can use ``--abort`` to abandon an unshelve
that causes a conflict. This reverts the unshelved changes, and
Colin Chan
shelve: always backup shelves instead of deleting them...
r25712 leaves the bundle in place.)
Colin Chan
shelve: only keep the latest N shelve backups...
r25713
After a successful unshelve, the shelved changes are stored in a
backup directory. Only the N most recent backups are kept. N
Matt Mackall
shelve: make maxbackup doc check-config friendly
r25852 defaults to 10 but can be overridden using the ``shelve.maxbackups``
Colin Chan
shelve: only keep the latest N shelve backups...
r25713 configuration option.
FUJIWARA Katsunori
shelve: keep old backups if timestamp can't decide exact order of them...
r25774
.. container:: verbose
Timestamp in seconds is used to decide order of backups. More
than ``maxbackups`` backups are kept, if same timestamp
prevents from deciding exact order of them, for safety.
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 """
abortf = opts['abort']
continuef = opts['continue']
if not abortf and not continuef:
cmdutil.checkunfinished(repo)
if abortf or continuef:
if abortf and continuef:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('cannot use both abort and continue'))
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if shelved:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('cannot combine abort/continue with '
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 'naming a shelved change'))
try:
state = shelvedstate.load(repo)
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except IOError as err:
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if err.errno != errno.ENOENT:
raise
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('no unshelve operation underway'))
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
if abortf:
return unshelveabort(ui, repo, state, opts)
elif continuef:
return unshelvecontinue(ui, repo, state, opts)
elif len(shelved) > 1:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('can only unshelve one change at a time'))
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 elif not shelved:
shelved = listshelves(repo)
if not shelved:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_('no shelved changes to apply!'))
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 basename = util.split(shelved[0][1])[1]
ui.status(_("unshelving change '%s'\n") % basename)
else:
basename = shelved[0]
Martin von Zweigbergk
shelve: avoid writing file that is never read from...
r22581 if not shelvedfile(repo, basename, 'patch').exists():
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("shelved change '%s' not found") % basename)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
Mads Kiilerich
shelve: be quiet when unshelve pulls from the shelve bundle...
r20412 oldquiet = ui.quiet
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 wlock = lock = tr = None
try:
Pierre-Yves David
shelve: acquire lock in the right order...
r24704 wlock = repo.wlock()
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 lock = repo.lock()
tr = repo.transaction('unshelve', report=lambda x: None)
oldtiprev = len(repo)
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961
Mads Kiilerich
shelve: repo['.'] is not a wctx but a pctx...
r20958 pctx = repo['.']
tmpwctx = pctx
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 # The goal is to have a commit structure like so:
Mads Kiilerich
shelve: repo['.'] is not a wctx but a pctx...
r20958 # ...-> pctx -> tmpwctx -> shelvectx
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 # where tmpwctx is an optional commit with the user's pending changes
# and shelvectx is the unshelved changes. Then we merge it all down
Mads Kiilerich
shelve: repo['.'] is not a wctx but a pctx...
r20958 # to the original pctx.
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961
# Store pending changes in a commit
Martin von Zweigbergk
shelve: access status fields by name rather than index
r22922 s = repo.status()
if s.modified or s.added or s.removed or s.deleted:
Mads Kiilerich
shelve: status messages from unshelve...
r20413 ui.status(_("temporarily committing pending changes "
"(restore with 'hg unshelve --abort')\n"))
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 def commitfunc(ui, repo, message, match, opts):
hasmq = util.safehasattr(repo, 'mq')
if hasmq:
saved, repo.mq.checkapplied = repo.mq.checkapplied, False
Pierre-Yves David
shelve: do not retract phase boundary by hand...
r22040 backup = repo.ui.backupconfig('phases', 'new-commit')
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 try:
Pierre-Yves David
shelve: do not retract phase boundary by hand...
r22040 repo.ui. setconfig('phases', 'new-commit', phases.secret)
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 return repo.commit(message, 'shelve@localhost',
opts.get('date'), match)
finally:
Pierre-Yves David
shelve: do not retract phase boundary by hand...
r22040 repo.ui.restoreconfig(backup)
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 if hasmq:
repo.mq.checkapplied = saved
tempopts = {}
tempopts['message'] = "pending changes temporary commit"
Mads Kiilerich
shelve: introduce secret option for using fixed date for temporary commit...
r20960 tempopts['date'] = opts.get('date')
Mads Kiilerich
shelve: be quiet when unshelve pulls from the shelve bundle...
r20412 ui.quiet = True
node = cmdutil.commit(ui, repo, commitfunc, [], tempopts)
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 tmpwctx = repo[node]
FUJIWARA Katsunori
shelve: add "applybundle()" to invoke "readbundle()" with relative path and vfs...
r20982 ui.quiet = True
shelvedfile(repo, basename, 'hg').applybundle()
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
Mads Kiilerich
shelve: be quiet when unshelve pulls from the shelve bundle...
r20412 ui.quiet = oldquiet
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 shelvectx = repo['tip']
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 # If the shelve is not immediately on top of the commit
# we'll be merging with, rebase it to be on top.
if tmpwctx.node() != shelvectx.parents()[0].node():
Mads Kiilerich
shelve: status messages from unshelve...
r20413 ui.status(_('rebasing shelved changes\n'))
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 try:
rebase.rebase(ui, repo, **{
'rev' : [shelvectx.rev()],
'dest' : str(tmpwctx.rev()),
'keep' : True,
})
except error.InterventionRequired:
tr.close()
stripnodes = [repo.changelog.node(rev)
for rev in xrange(oldtiprev, len(repo))]
Mads Kiilerich
shelve: repo['.'] is not a wctx but a pctx...
r20958 shelvedstate.save(repo, basename, pctx, tmpwctx, stripnodes)
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961
util.rename(repo.join('rebasestate'),
repo.join('unshelverebasestate'))
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 raise error.InterventionRequired(
_("unresolved conflicts (see 'hg resolve', then "
"'hg unshelve --continue')"))
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961
# refresh ctx after rebase completes
shelvectx = repo['tip']
if not shelvectx in tmpwctx.children():
# rebase was a no-op, so it produced no child commit
shelvectx = tmpwctx
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
Mads Kiilerich
shelve: repo['.'] is not a wctx but a pctx...
r20958 mergefiles(ui, repo, pctx, shelvectx)
Durham Goode
shelve: use rebase instead of merge (issue4068)...
r19961 shelvedstate.clear(repo)
# The transaction aborting will strip all the commits for us,
# but it doesn't update the inmemory structures, so addchangegroup
# hooks still fire and try to operate on the missing commits.
# Clean up manually to prevent this.
David Soria Parra
shelve: unshelve using an unfiltered repository...
r20064 repo.unfiltered().changelog.strip(oldtiprev, tr)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854
unshelvecleanup(ui, repo, basename, opts)
FUJIWARA Katsunori
shelve: restore unshelved dirstate explicitly after aborting transaction...
r26524
_aborttransaction(repo)
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 finally:
Mads Kiilerich
shelve: be quiet when unshelve pulls from the shelve bundle...
r20412 ui.quiet = oldquiet
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if tr:
tr.release()
lockmod.release(lock, wlock)
@command('shelve',
[('A', 'addremove', None,
_('mark new/missing files as added/removed before shelving')),
('', 'cleanup', None,
_('delete all shelved changes')),
('', 'date', '',
_('shelve with the specified commit date'), _('DATE')),
('d', 'delete', None,
_('delete the named shelved change(s)')),
FUJIWARA Katsunori
shelve: accept '--edit' like other commands creating new changeset...
r21852 ('e', 'edit', False,
_('invoke editor on commit messages')),
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 ('l', 'list', None,
_('list current shelves')),
('m', 'message', '',
_('use text as shelve message'), _('TEXT')),
('n', 'name', '',
_('use the given name for the shelved commit'), _('NAME')),
('p', 'patch', None,
_('show patch')),
Laurent Charignon
shelve: add interactive mode command line option
r24477 ('i', 'interactive', None,
Laurent Charignon
selve: make 'shelve --interactive' not experimental...
r25260 _('interactive mode, only works while creating a shelve')),
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 ('', 'stat', None,
Mads Kiilerich
shelve: mention walk options in help
r20409 _('output diffstat-style summary of changes'))] + commands.walkopts,
Mads Kiilerich
shelve: mention FILE options in help...
r20410 _('hg shelve [OPTION]... [FILE]...'))
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 def shelvecmd(ui, repo, *pats, **opts):
'''save and set aside changes from the working directory
Shelving takes files that "hg status" reports as not clean, saves
the modifications to a bundle (a shelved change), and reverts the
files so that their state in the working directory becomes clean.
To restore these changes to the working directory, using "hg
unshelve"; this will work even if you switch to a different
commit.
When no files are specified, "hg shelve" saves all not-clean
files. If specific files or directories are named, only changes to
those files are shelved.
Each shelved change has a name that makes it easier to find later.
The name of a shelved change defaults to being based on the active
bookmark, or if there is no active bookmark, the current named
branch. To specify a different name, use ``--name``.
To see a list of existing shelved changes, use the ``--list``
option. For each shelved change, this will print its name, age,
and description; use ``--patch`` or ``--stat`` for more details.
To delete specific shelved changes, use ``--delete``. To delete
all shelved changes, use ``--cleanup``.
'''
cmdutil.checkunfinished(repo)
FUJIWARA Katsunori
shelve: refactor option combination check to easily add new ones...
r21851 allowables = [
Tony Tung
shelve: refactor allowables to specify sets of valid operations...
r25103 ('addremove', set(['create'])), # 'create' is pseudo action
('cleanup', set(['cleanup'])),
# ('date', set(['create'])), # ignored for passing '--date "0 0"' in tests
('delete', set(['delete'])),
('edit', set(['create'])),
('list', set(['list'])),
('message', set(['create'])),
('name', set(['create'])),
Tony Tung
shelve: allow --patch and --stat without --list for a single shelf...
r25104 ('patch', set(['patch', 'list'])),
('stat', set(['stat', 'list'])),
FUJIWARA Katsunori
shelve: refactor option combination check to easily add new ones...
r21851 ]
def checkopt(opt):
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if opts[opt]:
FUJIWARA Katsunori
shelve: refactor option combination check to easily add new ones...
r21851 for i, allowable in allowables:
Tony Tung
shelve: refactor allowables to specify sets of valid operations...
r25103 if opts[i] and opt not in allowable:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("options '--%s' and '--%s' may not be "
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 "used together") % (opt, i))
return True
FUJIWARA Katsunori
shelve: refactor option combination check to easily add new ones...
r21851 if checkopt('cleanup'):
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if pats:
Pierre-Yves David
error: get Abort from 'error' instead of 'util'...
r26587 raise error.Abort(_("cannot specify names when using '--cleanup'"))
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 return cleanupcmd(ui, repo)
FUJIWARA Katsunori
shelve: refactor option combination check to easily add new ones...
r21851 elif checkopt('delete'):
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 return deletecmd(ui, repo, pats)
FUJIWARA Katsunori
shelve: refactor option combination check to easily add new ones...
r21851 elif checkopt('list'):
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 return listcmd(ui, repo, pats, opts)
Tony Tung
shelve: allow --patch and --stat without --list for a single shelf...
r25104 elif checkopt('patch'):
return singlepatchcmds(ui, repo, pats, opts, subcommand='patch')
elif checkopt('stat'):
return singlepatchcmds(ui, repo, pats, opts, subcommand='stat')
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 else:
return createcmd(ui, repo, pats, opts)
def extsetup(ui):
cmdutil.unfinishedstates.append(
FUJIWARA Katsunori
shelve: disallow commit while unshelve is in progress...
r19963 [shelvedstate._filename, False, False,
_('unshelve already in progress'),
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 _("use 'hg unshelve --continue' or 'hg unshelve --abort'")])