##// END OF EJS Templates
record: omit meaningless 'qrefresh' suggestion at 'hg qrefresh -i'...
record: omit meaningless 'qrefresh' suggestion at 'hg qrefresh -i' Before this patch, 'hg qrefresh -i' under non-interactive mode suggests 'use qrefresh instead', and it obviously meaningless. To omit meaningless 'qrefresh' suggestion at 'hg qrefresh -i', this patch specifies 'None' for 'cmdsuggest' argument of 'cmdutil.dorecord()'.

File last commit:

r25774:4f8c20fe default
r25798:08f2177b default
Show More
shelve.py
800 lines | 28.7 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
FUJIWARA Katsunori
shelve: remove useless and incorrect code paths for file access...
r19964 raise util.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)
FUJIWARA Katsunori
shelve: add "applybundle()" to invoke "readbundle()" with relative path and vfs...
r20982 changegroup.addchangegroup(self.repo, gen, 'unshelve',
Pierre-Yves David
shelve: use `targetphase` while unbundling...
r22042 '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))
FUJIWARA Katsunori
shelve: add "writebundle()" to invoke "writebundle()" with relative path and vfs
r20983 def writebundle(self, cg):
Eric Sumner
changegroup.writebundle: provide ui...
r23895 changegroup.writebundle(self.ui, cg, self.fname, 'HG10UN', self.vfs)
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:
raise util.Abort(_('this version of shelve is incompatible '
'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
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
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 def publicancestors(ctx):
Mads Kiilerich
shelve: really pass publicancestors to changegroupsubset - not the parents...
r20408 """Compute the public ancestors of a commit.
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:
raise util.Abort(_('cannot shelve while merging'))
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']
David Soria Parra
shelve: copy bookmarks and restore them after a commit...
r19874 wlock = lock = tr = bms = None
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 try:
wlock = repo.wlock()
lock = repo.lock()
David Soria Parra
shelve: copy bookmarks and restore them after a commit...
r19874 bms = repo._bookmarks.copy()
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():
raise util.Abort(_("a shelved change named '%s' already exists")
% name)
else:
for n in gennames():
if not shelvedfile(repo, n, 'hg').exists():
name = n
break
else:
raise util.Abort(_("too many shelved changes named '%s'") %
label)
# ensure we are not creating a subdirectory or a hidden file
if '/' in name or '\\' in name:
raise util.Abort(_('shelved change names may not contain slashes'))
if name.startswith('.'):
raise util.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:
node = cmdutil.dorecord(ui, repo, interactivecommitfunc, 'commit',
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
bases = list(publicancestors(repo[node]))
Pierre-Yves David
localrepo: move the changegroupsubset method in changegroup module...
r20927 cg = changegroup.changegroupsubset(repo, bases, [node], 'shelve')
FUJIWARA Katsunori
shelve: add "writebundle()" to invoke "writebundle()" with relative path and vfs
r20983 shelvedfile(repo, name, 'hg').writebundle(cg)
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())
finally:
David Soria Parra
shelve: copy bookmarks and restore them after a commit...
r19874 if bms:
# restore old bookmarks
repo._bookmarks.update(bms)
repo._bookmarks.write()
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 if tr:
tr.abort()
lockmod.release(lock, wlock)
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:
raise util.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
raise util.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:
raise util.Abort(_("--%s expects a single shelf") % subcommand)
shelfname = pats[0]
if not shelvedfile(repo, shelfname, 'patch').exists():
raise util.Abort(_("cannot find shelf %s") % shelfname)
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():
raise util.Abort(_('working directory parents do not match unshelve '
'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')
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)
finally:
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']:
raise util.Abort(
_("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
defaults to 10 but can be overridden using the shelve.maxbackups
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:
raise util.Abort(_('cannot use both abort and continue'))
if shelved:
raise util.Abort(_('cannot combine abort/continue with '
'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
raise util.Abort(_('no unshelve operation underway'))
if abortf:
return unshelveabort(ui, repo, state, opts)
elif continuef:
return unshelvecontinue(ui, repo, state, opts)
elif len(shelved) > 1:
raise util.Abort(_('can only unshelve one change at a time'))
elif not shelved:
shelved = listshelves(repo)
if not shelved:
raise util.Abort(_('no shelved changes to apply!'))
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():
FUJIWARA Katsunori
shelve: remove unused variable assignment...
r19970 raise util.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)
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:
David Soria Parra
shelve: add a shelve extension to save/restore working changes...
r19854 raise util.Abort(_("options '--%s' and '--%s' may not be "
"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:
raise util.Abort(_("cannot specify names when using '--cleanup'"))
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'")])