record.py
433 lines
| 15.9 KiB
| text/x-python
|
PythonLexer
/ hgext / record.py
Bryan O'Sullivan
|
r5037 | # record.py | ||
# | ||||
# Copyright 2007 Bryan O'Sullivan <bos@serpentine.com> | ||||
# | ||||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Bryan O'Sullivan
|
r5037 | |||
Dirkjan Ochtman
|
r8934 | '''commands to interactively select changes for commit/qrefresh''' | ||
Bryan O'Sullivan
|
r5037 | |||
FUJIWARA Katsunori
|
r20266 | from mercurial.i18n import _ | ||
Idan Kamara
|
r18287 | from mercurial import cmdutil, commands, extensions, hg, patch | ||
Bryan O'Sullivan
|
r5037 | from mercurial import util | ||
Laurent Charignon
|
r24264 | import copy, cStringIO, errno, os, shutil, tempfile | ||
Bryan O'Sullivan
|
r5037 | |||
Idan Kamara
|
r14408 | cmdtable = {} | ||
command = cmdutil.command(cmdtable) | ||||
Augie Fackler
|
r16743 | testedwith = 'internal' | ||
Idan Kamara
|
r14408 | |||
Patrick Mezard
|
r13293 | def filterpatch(ui, headers): | ||
Kirill Smelkov
|
r5826 | """Interactively filter patch chunks into applied-only chunks""" | ||
A. S. Budden
|
r16324 | def prompt(skipfile, skipall, query, chunk): | ||
Kirill Smelkov
|
r5826 | """prompt query, and process base inputs | ||
Thomas Arendsen Hein
|
r6210 | |||
Kirill Smelkov
|
r5826 | - y/n for the rest of file | ||
- y/n for the rest | ||||
- ? (help) | ||||
- q (quit) | ||||
Patrick Mezard
|
r13291 | Return True/False and possibly updated skipfile and skipall. | ||
Kirill Smelkov
|
r5826 | """ | ||
A. S. Budden
|
r16324 | newpatches = None | ||
Patrick Mezard
|
r13291 | if skipall is not None: | ||
A. S. Budden
|
r16324 | return skipall, skipfile, skipall, newpatches | ||
Patrick Mezard
|
r13291 | if skipfile is not None: | ||
A. S. Budden
|
r16324 | return skipfile, skipfile, skipall, newpatches | ||
Bryan O'Sullivan
|
r5154 | while True: | ||
Matt Mackall
|
r19226 | resps = _('[Ynesfdaq?]' | ||
'$$ &Yes, record this change' | ||||
'$$ &No, skip this change' | ||||
FUJIWARA Katsunori
|
r20266 | '$$ &Edit this change manually' | ||
Matt Mackall
|
r19226 | '$$ &Skip remaining changes to this file' | ||
'$$ Record remaining changes to this &file' | ||||
'$$ &Done, skip remaining changes and files' | ||||
'$$ Record &all changes to all remaining files' | ||||
'$$ &Quit, recording no changes' | ||||
FUJIWARA Katsunori
|
r20266 | '$$ &? (display help)') | ||
Matt Mackall
|
r19226 | r = ui.promptchoice("%s %s" % (query, resps)) | ||
Martin Geisler
|
r10694 | ui.write("\n") | ||
A. S. Budden
|
r16324 | if r == 8: # ? | ||
FUJIWARA Katsunori
|
r20266 | for c, t in ui.extractchoices(resps)[1]: | ||
ui.write('%s - %s\n' % (c, t.lower())) | ||||
Bryan O'Sullivan
|
r5154 | continue | ||
Simon Heimberg
|
r9048 | elif r == 0: # yes | ||
Martin Geisler
|
r9837 | ret = True | ||
Simon Heimberg
|
r9048 | elif r == 1: # no | ||
Martin Geisler
|
r9837 | ret = False | ||
A. S. Budden
|
r16324 | elif r == 2: # Edit patch | ||
if chunk is None: | ||||
ui.write(_('cannot edit patch for whole file')) | ||||
ui.write("\n") | ||||
continue | ||||
if chunk.header.binary(): | ||||
ui.write(_('cannot edit patch for binary file')) | ||||
ui.write("\n") | ||||
continue | ||||
# Patch comment based on the Git one (based on comment at end of | ||||
# http://mercurial.selenic.com/wiki/RecordExtension) | ||||
phelp = '---' + _(""" | ||||
To remove '-' lines, make them ' ' lines (context). | ||||
To remove '+' lines, delete them. | ||||
Lines starting with # will be removed from the patch. | ||||
If the patch applies cleanly, the edited hunk will immediately be | ||||
added to the record list. If it does not apply cleanly, a rejects | ||||
file will be generated: you can use that when you try again. If | ||||
all lines of the hunk are removed, then the edit is aborted and | ||||
the hunk is left unchanged. | ||||
""") | ||||
(patchfd, patchfn) = tempfile.mkstemp(prefix="hg-editor-", | ||||
suffix=".diff", text=True) | ||||
Matt Mackall
|
r16328 | ncpatchfp = None | ||
A. S. Budden
|
r16324 | try: | ||
# Write the initial patch | ||||
f = os.fdopen(patchfd, "w") | ||||
chunk.header.write(f) | ||||
chunk.write(f) | ||||
f.write('\n'.join(['# ' + i for i in phelp.splitlines()])) | ||||
f.close() | ||||
# Start the editor and wait for it to complete | ||||
editor = ui.geteditor() | ||||
Yuya Nishihara
|
r23270 | ui.system("%s \"%s\"" % (editor, patchfn), | ||
environ={'HGUSER': ui.username()}, | ||||
onerr=util.Abort, errprefix=_("edit failed")) | ||||
A. S. Budden
|
r16324 | # Remove comment lines | ||
patchfp = open(patchfn) | ||||
ncpatchfp = cStringIO.StringIO() | ||||
for line in patchfp: | ||||
if not line.startswith('#'): | ||||
ncpatchfp.write(line) | ||||
patchfp.close() | ||||
ncpatchfp.seek(0) | ||||
Laurent Charignon
|
r24265 | newpatches = patch.parsepatch(ncpatchfp) | ||
A. S. Budden
|
r16324 | finally: | ||
os.unlink(patchfn) | ||||
del ncpatchfp | ||||
# Signal that the chunk shouldn't be applied as-is, but | ||||
# provide the new patch to be used instead. | ||||
ret = False | ||||
elif r == 3: # Skip | ||||
Patrick Mezard
|
r13291 | ret = skipfile = False | ||
A. S. Budden
|
r16324 | elif r == 4: # file (Record remaining) | ||
Patrick Mezard
|
r13291 | ret = skipfile = True | ||
A. S. Budden
|
r16324 | elif r == 5: # done, skip remaining | ||
Patrick Mezard
|
r13291 | ret = skipall = False | ||
A. S. Budden
|
r16324 | elif r == 6: # all | ||
Patrick Mezard
|
r13291 | ret = skipall = True | ||
A. S. Budden
|
r16324 | elif r == 7: # quit | ||
Bryan O'Sullivan
|
r5154 | raise util.Abort(_('user quit')) | ||
A. S. Budden
|
r16324 | return ret, skipfile, skipall, newpatches | ||
Patrick Mezard
|
r13291 | |||
seen = set() | ||||
applied = {} # 'filename' -> [] of chunks | ||||
skipfile, skipall = None, None | ||||
Patrick Mezard
|
r13295 | pos, total = 1, sum(len(h.hunks) for h in headers) | ||
Patrick Mezard
|
r13293 | for h in headers: | ||
Patrick Mezard
|
r13295 | pos += len(h.hunks) | ||
Patrick Mezard
|
r13293 | skipfile = None | ||
fixoffset = 0 | ||||
hdr = ''.join(h.header) | ||||
if hdr in seen: | ||||
continue | ||||
seen.add(hdr) | ||||
if skipall is None: | ||||
h.pretty(ui) | ||||
msg = (_('examine changes to %s?') % | ||||
Matt Mackall
|
r17081 | _(' and ').join("'%s'" % f for f in h.files())) | ||
A. S. Budden
|
r16324 | r, skipfile, skipall, np = prompt(skipfile, skipall, msg, None) | ||
Patrick Mezard
|
r13293 | if not r: | ||
continue | ||||
applied[h.filename()] = [h] | ||||
if h.allhunks(): | ||||
applied[h.filename()] += h.hunks | ||||
continue | ||||
for i, chunk in enumerate(h.hunks): | ||||
Patrick Mezard
|
r13291 | if skipfile is None and skipall is None: | ||
Bryan O'Sullivan
|
r5037 | chunk.pretty(ui) | ||
Martin Geisler
|
r13773 | if total == 1: | ||
Nikolaj Sjujskij
|
r17566 | msg = _("record this change to '%s'?") % chunk.filename() | ||
Martin Geisler
|
r13773 | else: | ||
idx = pos - len(h.hunks) + i | ||||
Nikolaj Sjujskij
|
r17566 | msg = _("record change %d/%d to '%s'?") % (idx, total, | ||
chunk.filename()) | ||||
A. S. Budden
|
r16324 | r, skipfile, skipall, newpatches = prompt(skipfile, | ||
skipall, msg, chunk) | ||||
Martin Geisler
|
r9837 | if r: | ||
Bryan O'Sullivan
|
r5037 | if fixoffset: | ||
chunk = copy.copy(chunk) | ||||
chunk.toline += fixoffset | ||||
applied[chunk.filename()].append(chunk) | ||||
A. S. Budden
|
r16324 | elif newpatches is not None: | ||
for newpatch in newpatches: | ||||
for newhunk in newpatch.hunks: | ||||
if fixoffset: | ||||
newhunk.toline += fixoffset | ||||
applied[newhunk.filename()].append(newhunk) | ||||
Bryan O'Sullivan
|
r5037 | else: | ||
fixoffset += chunk.removed - chunk.added | ||||
Renato Cunha
|
r11500 | return sum([h for h in applied.itervalues() | ||
if h[0].special() or len(h) > 1], []) | ||||
Bryan O'Sullivan
|
r5037 | |||
Idan Kamara
|
r14408 | @command("record", | ||
Ingo Proetel
|
r14597 | # same options as commit + white space diff options | ||
Jordi GutiƩrrez Hermoso
|
r20300 | commands.table['^commit|ci'][1][:] + commands.diffwsopts, | ||
Idan Kamara
|
r14408 | _('hg record [OPTION]... [FILE]...')) | ||
Bryan O'Sullivan
|
r5037 | def record(ui, repo, *pats, **opts): | ||
Bryan O'Sullivan
|
r5154 | '''interactively select changes to commit | ||
Martin Geisler
|
r10973 | If a list of files is omitted, all changes reported by :hg:`status` | ||
Martin Geisler
|
r9272 | will be candidates for recording. | ||
Bryan O'Sullivan
|
r5154 | |||
Martin Geisler
|
r10973 | See :hg:`help dates` for a list of formats valid for -d/--date. | ||
Thomas Arendsen Hein
|
r6163 | |||
Martin Geisler
|
r9272 | You will be prompted for whether to record changes to each | ||
modified file, and for files with multiple changes, for each | ||||
change to use. For each query, the following responses are | ||||
possible:: | ||||
Bryan O'Sullivan
|
r5154 | |||
Martin Geisler
|
r9157 | y - record this change | ||
n - skip this change | ||||
A. S. Budden
|
r16324 | e - edit this change manually | ||
Bryan O'Sullivan
|
r5154 | |||
Martin Geisler
|
r9157 | s - skip remaining changes to this file | ||
f - record remaining changes to this file | ||||
Bryan O'Sullivan
|
r5154 | |||
Martin Geisler
|
r9157 | d - done, skip remaining changes and files | ||
a - record all changes to all remaining files | ||||
q - quit, recording no changes | ||||
Bryan O'Sullivan
|
r5154 | |||
Nicolas Dumazet
|
r11237 | ? - display help | ||
This command is not available when committing a merge.''' | ||||
Bryan O'Sullivan
|
r5037 | |||
Idan Kamara
|
r14425 | dorecord(ui, repo, commands.commit, 'commit', False, *pats, **opts) | ||
Kirill Smelkov
|
r5830 | |||
Matt Mackall
|
r15184 | def qrefresh(origfn, ui, repo, *pats, **opts): | ||
if not opts['interactive']: | ||||
return origfn(ui, repo, *pats, **opts) | ||||
Idan Kamara
|
r14426 | mq = extensions.find('mq') | ||
def committomq(ui, repo, *pats, **opts): | ||||
# At this point the working copy contains only changes that | ||||
# were accepted. All other changes were reverted. | ||||
# We can't pass *pats here since qrefresh will undo all other | ||||
# changed files in the patch that aren't in pats. | ||||
mq.refresh(ui, repo, **opts) | ||||
# backup all changed files | ||||
dorecord(ui, repo, committomq, 'qrefresh', True, *pats, **opts) | ||||
Kirill Smelkov
|
r5830 | |||
Gregory Szorc
|
r21251 | # This command registration is replaced during uisetup(). | ||
Gregory Szorc
|
r21787 | @command('qrecord', | ||
[], | ||||
_('hg qrecord [OPTION]... PATCH [FILE]...'), | ||||
inferrepo=True) | ||||
Kirill Smelkov
|
r5932 | def qrecord(ui, repo, patch, *pats, **opts): | ||
'''interactively record a new patch | ||||
Kirill Smelkov
|
r5830 | |||
Martin Geisler
|
r10973 | See :hg:`help qnew` & :hg:`help record` for more information and | ||
Martin Geisler
|
r9272 | usage. | ||
Kirill Smelkov
|
r5830 | ''' | ||
try: | ||||
mq = extensions.find('mq') | ||||
except KeyError: | ||||
raise util.Abort(_("'mq' extension not loaded")) | ||||
Idan Kamara
|
r14424 | repo.mq.checkpatchname(patch) | ||
Dan Villiom Podlaski Christiansen
|
r10323 | def committomq(ui, repo, *pats, **opts): | ||
Idan Kamara
|
r14424 | opts['checkname'] = False | ||
Kirill Smelkov
|
r5932 | mq.new(ui, repo, patch, *pats, **opts) | ||
Kirill Smelkov
|
r5830 | |||
Idan Kamara
|
r14425 | dorecord(ui, repo, committomq, 'qnew', False, *pats, **opts) | ||
Kirill Smelkov
|
r5827 | |||
Matt Mackall
|
r15184 | def qnew(origfn, ui, repo, patch, *args, **opts): | ||
if opts['interactive']: | ||||
return qrecord(ui, repo, patch, *args, **opts) | ||||
return origfn(ui, repo, patch, *args, **opts) | ||||
Idan Kamara
|
r14425 | def dorecord(ui, repo, commitfunc, cmdsuggest, backupall, *pats, **opts): | ||
Matt Mackall
|
r8208 | if not ui.interactive(): | ||
Idan Kamara
|
r14407 | raise util.Abort(_('running non-interactively, use %s instead') % | ||
cmdsuggest) | ||||
Bryan O'Sullivan
|
r5037 | |||
Sumeet
|
r17446 | # make sure username is set before going interactive | ||
Prasoon Shukla
|
r20171 | if not opts.get('user'): | ||
ui.username() # raise exception, username not provided | ||||
Sumeet
|
r17446 | |||
Matt Mackall
|
r6600 | def recordfunc(ui, repo, message, match, opts): | ||
Kirill Smelkov
|
r5827 | """This is generic record driver. | ||
Kevin Bullock
|
r13195 | Its job is to interactively filter local changes, and | ||
accordingly prepare working directory into a state in which the | ||||
job can be delegated to a non-interactive commit command such as | ||||
'commit' or 'qrefresh'. | ||||
Kirill Smelkov
|
r5827 | |||
Kevin Bullock
|
r13195 | After the actual job is done by non-interactive command, the | ||
working directory is restored to its original state. | ||||
Kirill Smelkov
|
r5827 | |||
Kevin Bullock
|
r13195 | In the end we'll record interesting changes, and everything else | ||
will be left in place, so the user can continue working. | ||||
Kirill Smelkov
|
r5827 | """ | ||
Dirkjan Ochtman
|
r7754 | |||
Matt Mackall
|
r19497 | cmdutil.checkunfinished(repo, commit=True) | ||
Nicolas Dumazet
|
r11237 | merge = len(repo[None].parents()) > 1 | ||
if merge: | ||||
raise util.Abort(_('cannot partially commit a merge ' | ||||
timeless
|
r13023 | '(use "hg commit" instead)')) | ||
Nicolas Dumazet
|
r11237 | |||
Martin von Zweigbergk
|
r22921 | status = repo.status(match=match) | ||
Siddharth Agarwal
|
r23435 | diffopts = patch.difffeatureopts(ui, opts=opts, whitespace=True) | ||
diffopts.nodates = True | ||||
diffopts.git = True | ||||
Laurent Charignon
|
r24235 | originalchunks = patch.diff(repo, changes=status, opts=diffopts) | ||
Bryan O'Sullivan
|
r5037 | fp = cStringIO.StringIO() | ||
Laurent Charignon
|
r24235 | fp.write(''.join(originalchunks)) | ||
Bryan O'Sullivan
|
r5037 | fp.seek(0) | ||
Kirill Smelkov
|
r5827 | # 1. filter patch, so we have intending-to apply subset of it | ||
Mads Kiilerich
|
r18953 | try: | ||
Laurent Charignon
|
r24265 | chunks = filterpatch(ui, patch.parsepatch(fp)) | ||
Mads Kiilerich
|
r18953 | except patch.PatchError, err: | ||
raise util.Abort(_('error parsing patch: %s') % err) | ||||
Bryan O'Sullivan
|
r5037 | del fp | ||
Martin Geisler
|
r8152 | contenders = set() | ||
Bryan O'Sullivan
|
r5037 | for h in chunks: | ||
Matt Mackall
|
r10282 | try: | ||
contenders.update(set(h.files())) | ||||
except AttributeError: | ||||
pass | ||||
Thomas Arendsen Hein
|
r5143 | |||
Martin von Zweigbergk
|
r22921 | changed = status.modified + status.added + status.removed | ||
Dirkjan Ochtman
|
r7754 | newfiles = [f for f in changed if f in contenders] | ||
Bryan O'Sullivan
|
r5037 | if not newfiles: | ||
ui.status(_('no changes to record\n')) | ||||
return 0 | ||||
Laurent Charignon
|
r24235 | newandmodifiedfiles = set() | ||
for h in chunks: | ||||
Laurent Charignon
|
r24263 | ishunk = isinstance(h, patch.recordhunk) | ||
Laurent Charignon
|
r24235 | isnew = h.filename() in status.added | ||
if ishunk and isnew and not h in originalchunks: | ||||
newandmodifiedfiles.add(h.filename()) | ||||
Martin von Zweigbergk
|
r22921 | modified = set(status.modified) | ||
Bryan O'Sullivan
|
r5037 | |||
Kirill Smelkov
|
r5827 | # 2. backup changed files, so we can restore them in the end | ||
Laurent Charignon
|
r24235 | |||
Idan Kamara
|
r14425 | if backupall: | ||
tobackup = changed | ||||
else: | ||||
Laurent Charignon
|
r24235 | tobackup = [f for f in newfiles | ||
if f in modified or f in newandmodifiedfiles] | ||||
Idan Kamara
|
r14425 | |||
Bryan O'Sullivan
|
r5037 | backups = {} | ||
Idan Kamara
|
r14425 | if tobackup: | ||
backupdir = repo.join('record-backups') | ||||
try: | ||||
os.mkdir(backupdir) | ||||
except OSError, err: | ||||
if err.errno != errno.EEXIST: | ||||
raise | ||||
Bryan O'Sullivan
|
r5037 | try: | ||
Kirill Smelkov
|
r5827 | # backup continues | ||
Idan Kamara
|
r14425 | for f in tobackup: | ||
Bryan O'Sullivan
|
r5037 | fd, tmpname = tempfile.mkstemp(prefix=f.replace('/', '_')+'.', | ||
dir=backupdir) | ||||
os.close(fd) | ||||
Martin Geisler
|
r9467 | ui.debug('backup %r as %r\n' % (f, tmpname)) | ||
Bryan O'Sullivan
|
r5037 | util.copyfile(repo.wjoin(f), tmpname) | ||
Brodie Rao
|
r13099 | shutil.copystat(repo.wjoin(f), tmpname) | ||
Bryan O'Sullivan
|
r5037 | backups[f] = tmpname | ||
fp = cStringIO.StringIO() | ||||
for c in chunks: | ||||
Laurent Charignon
|
r24235 | fname = c.filename() | ||
if fname in backups or fname in newandmodifiedfiles: | ||||
Bryan O'Sullivan
|
r5037 | c.write(fp) | ||
dopatch = fp.tell() | ||||
fp.seek(0) | ||||
Laurent Charignon
|
r24235 | [os.unlink(c) for c in newandmodifiedfiles] | ||
Kirill Smelkov
|
r5827 | # 3a. apply filtered patch to clean repo (clean) | ||
Bryan O'Sullivan
|
r5037 | if backups: | ||
Matt Mackall
|
r13878 | hg.revert(repo, repo.dirstate.p1(), | ||
Renato Cunha
|
r11564 | lambda key: key in backups) | ||
Bryan O'Sullivan
|
r5037 | |||
Kirill Smelkov
|
r5827 | # 3b. (apply) | ||
Bryan O'Sullivan
|
r5037 | if dopatch: | ||
Dirkjan Ochtman
|
r6950 | try: | ||
Martin Geisler
|
r9467 | ui.debug('applying patch\n') | ||
Dirkjan Ochtman
|
r6950 | ui.debug(fp.getvalue()) | ||
Siddharth Agarwal
|
r24268 | patch.internalpatch(ui, repo, fp, 1, eolmode=None) | ||
Dirkjan Ochtman
|
r6950 | except patch.PatchError, err: | ||
Patrick Mezard
|
r12674 | raise util.Abort(str(err)) | ||
Bryan O'Sullivan
|
r5037 | del fp | ||
Kevin Bullock
|
r13195 | # 4. We prepared working directory according to filtered | ||
# patch. Now is the time to delegate the job to | ||||
# commit/qrefresh or the like! | ||||
Kirill Smelkov
|
r5827 | |||
Jordi GutiƩrrez Hermoso
|
r21805 | # Make all of the pathnames absolute. | ||
Pierre-Yves David
|
r20334 | newfiles = [repo.wjoin(nf) for nf in newfiles] | ||
commitfunc(ui, repo, *newfiles, **opts) | ||||
Kirill Smelkov
|
r5827 | |||
Bryan O'Sullivan
|
r5037 | return 0 | ||
finally: | ||||
Kirill Smelkov
|
r5827 | # 5. finally restore backed-up files | ||
Bryan O'Sullivan
|
r5037 | try: | ||
for realname, tmpname in backups.iteritems(): | ||||
Martin Geisler
|
r9467 | ui.debug('restoring %r to %r\n' % (tmpname, realname)) | ||
Bryan O'Sullivan
|
r5128 | util.copyfile(tmpname, repo.wjoin(realname)) | ||
Brodie Rao
|
r13099 | # Our calls to copystat() here and above are a | ||
# hack to trick any editors that have f open that | ||||
# we haven't modified them. | ||||
# | ||||
# Also note that this racy as an editor could | ||||
# notice the file's mtime before we've finished | ||||
# writing it. | ||||
shutil.copystat(tmpname, repo.wjoin(realname)) | ||||
Bryan O'Sullivan
|
r5037 | os.unlink(tmpname) | ||
Idan Kamara
|
r14425 | if tobackup: | ||
os.rmdir(backupdir) | ||||
Bryan O'Sullivan
|
r5037 | except OSError: | ||
pass | ||||
Brodie Rao
|
r10825 | |||
# wrap ui.write so diff output can be labeled/colorized | ||||
def wrapwrite(orig, *args, **kw): | ||||
label = kw.pop('label', '') | ||||
for chunk, l in patch.difflabel(lambda: args): | ||||
orig(chunk, label=label + l) | ||||
oldwrite = ui.write | ||||
extensions.wrapfunction(ui, 'write', wrapwrite) | ||||
try: | ||||
return cmdutil.commit(ui, repo, recordfunc, pats, opts) | ||||
finally: | ||||
ui.write = oldwrite | ||||
Bryan O'Sullivan
|
r5037 | |||
Martin Geisler
|
r9710 | def uisetup(ui): | ||
Kirill Smelkov
|
r5830 | try: | ||
mq = extensions.find('mq') | ||||
except KeyError: | ||||
return | ||||
Idan Kamara
|
r14408 | cmdtable["qrecord"] = \ | ||
Idan Kamara
|
r14427 | (qrecord, | ||
# same options as qnew, but copy them so we don't get | ||||
Ingo Proetel
|
r14597 | # -i/--interactive for qrecord and add white space diff options | ||
Jordi GutiƩrrez Hermoso
|
r20300 | mq.cmdtable['^qnew'][1][:] + commands.diffwsopts, | ||
Idan Kamara
|
r14408 | _('hg qrecord [OPTION]... PATCH [FILE]...')) | ||
Idan Kamara
|
r14426 | |||
Matt Mackall
|
r15184 | _wrapcmd('qnew', mq.cmdtable, qnew, _("interactively record a new patch")) | ||
Idan Kamara
|
r14426 | _wrapcmd('qrefresh', mq.cmdtable, qrefresh, | ||
_("interactively select changes to refresh")) | ||||
def _wrapcmd(cmd, table, wrapfn, msg): | ||||
Matt Mackall
|
r15184 | entry = extensions.wrapcommand(table, cmd, wrapfn) | ||
Idan Kamara
|
r14426 | entry[1].append(('i', 'interactive', None, msg)) | ||