transplant.py
930 lines
| 30.0 KiB
| text/x-python
|
PythonLexer
/ hgext / transplant.py
Brendan Cully
|
r3714 | # Patch transplanting extension for Mercurial | ||
# | ||||
Thomas Arendsen Hein
|
r4635 | # Copyright 2006, 2007 Brendan Cully <brendan@kublai.com> | ||
Brendan Cully
|
r3714 | # | ||
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. | ||
Brendan Cully
|
r3714 | |||
Dirkjan Ochtman
|
r8934 | '''command to transplant changesets from another branch | ||
Brendan Cully
|
r3714 | |||
Mads Kiilerich
|
r19028 | This extension allows you to transplant changes to another parent revision, | ||
possibly in another repository. The transplant is done using 'diff' patches. | ||||
Brendan Cully
|
r3714 | |||
Martin Geisler
|
r8000 | Transplanted patches are recorded in .hg/transplant/transplants, as a | ||
map from a changeset hash to its hash in the source repository. | ||||
Brendan Cully
|
r3714 | ''' | ||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
timeless
|
r28481 | import os | ||
Yuya Nishihara
|
r38182 | |||
Dirkjan Ochtman
|
r7629 | from mercurial.i18n import _ | ||
Joerg Sonnenberger
|
r46729 | from mercurial.node import ( | ||
bin, | ||||
hex, | ||||
short, | ||||
) | ||||
timeless
|
r28481 | from mercurial import ( | ||
bundlerepo, | ||||
cmdutil, | ||||
error, | ||||
exchange, | ||||
hg, | ||||
Yuya Nishihara
|
r35906 | logcmdutil, | ||
timeless
|
r28481 | match, | ||
merge, | ||||
patch, | ||||
Pulkit Goyal
|
r30925 | pycompat, | ||
timeless
|
r28481 | registrar, | ||
revset, | ||||
Yuya Nishihara
|
r31023 | smartset, | ||
Taapas Agrawal
|
r42729 | state as statemod, | ||
timeless
|
r28481 | util, | ||
Pierre-Yves David
|
r31245 | vfs as vfsmod, | ||
timeless
|
r28481 | ) | ||
Yuya Nishihara
|
r37102 | from mercurial.utils import ( | ||
Yuya Nishihara
|
r37138 | procutil, | ||
Yuya Nishihara
|
r37102 | stringutil, | ||
r47706 | urlutil, | |||
Yuya Nishihara
|
r37102 | ) | ||
Dirkjan Ochtman
|
r7629 | |||
Augie Fackler
|
r43346 | |||
Patrick Mezard
|
r16507 | class TransplantError(error.Abort): | ||
pass | ||||
Augie Fackler
|
r43346 | |||
Adrian Buehlmann
|
r14308 | cmdtable = {} | ||
Yuya Nishihara
|
r32337 | command = registrar.command(cmdtable) | ||
Augie Fackler
|
r29841 | # Note for extension authors: ONLY specify testedwith = 'ships-with-hg-core' for | ||
Augie Fackler
|
r25186 | # 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. | ||||
Augie Fackler
|
r43347 | testedwith = b'ships-with-hg-core' | ||
Adrian Buehlmann
|
r14308 | |||
Boris Feld
|
r34470 | configtable = {} | ||
configitem = registrar.configitem(configtable) | ||||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'transplant', | ||
b'filter', | ||||
default=None, | ||||
Boris Feld
|
r34470 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'transplant', | ||
b'log', | ||||
default=None, | ||||
Boris Feld
|
r34471 | ) | ||
Boris Feld
|
r34470 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class transplantentry: | ||
Brendan Cully
|
r3714 | def __init__(self, lnode, rnode): | ||
self.lnode = lnode | ||||
self.rnode = rnode | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class transplants: | ||
Brendan Cully
|
r3714 | def __init__(self, path=None, transplantfile=None, opener=None): | ||
self.path = path | ||||
self.transplantfile = transplantfile | ||||
self.opener = opener | ||||
if not opener: | ||||
Pierre-Yves David
|
r31245 | self.opener = vfsmod.vfs(self.path) | ||
Peter Arrenbrecht
|
r12313 | self.transplants = {} | ||
Brendan Cully
|
r3714 | self.dirty = False | ||
self.read() | ||||
def read(self): | ||||
abspath = os.path.join(self.path, self.transplantfile) | ||||
if self.transplantfile and os.path.exists(abspath): | ||||
Dan Villiom Podlaski Christiansen
|
r14168 | for line in self.opener.read(self.transplantfile).splitlines(): | ||
Joerg Sonnenberger
|
r46729 | lnode, rnode = map(bin, line.split(b':')) | ||
Peter Arrenbrecht
|
r12313 | list = self.transplants.setdefault(rnode, []) | ||
list.append(transplantentry(lnode, rnode)) | ||||
Brendan Cully
|
r3714 | |||
def write(self): | ||||
if self.dirty and self.transplantfile: | ||||
if not os.path.isdir(self.path): | ||||
os.mkdir(self.path) | ||||
Augie Fackler
|
r43347 | fp = self.opener(self.transplantfile, b'w') | ||
Gregory Szorc
|
r49790 | for list in self.transplants.values(): | ||
Peter Arrenbrecht
|
r12349 | for t in list: | ||
Joerg Sonnenberger
|
r46729 | l, r = map(hex, (t.lnode, t.rnode)) | ||
Augie Fackler
|
r43347 | fp.write(l + b':' + r + b'\n') | ||
Brendan Cully
|
r3714 | fp.close() | ||
self.dirty = False | ||||
def get(self, rnode): | ||||
Peter Arrenbrecht
|
r12313 | return self.transplants.get(rnode) or [] | ||
Brendan Cully
|
r3714 | |||
def set(self, lnode, rnode): | ||||
Peter Arrenbrecht
|
r12313 | list = self.transplants.setdefault(rnode, []) | ||
list.append(transplantentry(lnode, rnode)) | ||||
Brendan Cully
|
r3714 | self.dirty = True | ||
def remove(self, transplant): | ||||
Peter Arrenbrecht
|
r12313 | list = self.transplants.get(transplant.rnode) | ||
if list: | ||||
del list[list.index(transplant)] | ||||
self.dirty = True | ||||
Brendan Cully
|
r3714 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class transplanter: | ||
FUJIWARA Katsunori
|
r21411 | def __init__(self, ui, repo, opts): | ||
Brendan Cully
|
r3714 | self.ui = ui | ||
Joerg Sonnenberger
|
r47771 | self.repo = repo | ||
Augie Fackler
|
r43347 | self.path = repo.vfs.join(b'transplant') | ||
Pierre-Yves David
|
r31245 | self.opener = vfsmod.vfs(self.path) | ||
Augie Fackler
|
r43346 | self.transplants = transplants( | ||
Augie Fackler
|
r43347 | self.path, b'transplants', opener=self.opener | ||
Augie Fackler
|
r43346 | ) | ||
FUJIWARA Katsunori
|
r22252 | def getcommiteditor(): | ||
Augie Fackler
|
r43347 | editform = cmdutil.mergeeditform(repo[None], b'transplant') | ||
Augie Fackler
|
r43346 | return cmdutil.getcommiteditor( | ||
editform=editform, **pycompat.strkwargs(opts) | ||||
) | ||||
FUJIWARA Katsunori
|
r22252 | self.getcommiteditor = getcommiteditor | ||
Brendan Cully
|
r3714 | |||
def applied(self, repo, node, parent): | ||||
Augie Fackler
|
r46554 | """returns True if a node is already an ancestor of parent | ||
or is parent or has already been transplanted""" | ||||
Joshua Redstone
|
r17010 | if hasnode(repo, parent): | ||
parentrev = repo.changelog.rev(parent) | ||||
Brendan Cully
|
r3714 | if hasnode(repo, node): | ||
Joshua Redstone
|
r17010 | rev = repo.changelog.rev(node) | ||
Augie Fackler
|
r43346 | reachable = repo.changelog.ancestors( | ||
[parentrev], rev, inclusive=True | ||||
) | ||||
Joshua Redstone
|
r17010 | if rev in reachable: | ||
Brendan Cully
|
r3714 | return True | ||
for t in self.transplants.get(node): | ||||
# it might have been stripped | ||||
if not hasnode(repo, t.lnode): | ||||
self.transplants.remove(t) | ||||
return False | ||||
Joshua Redstone
|
r17010 | lnoderev = repo.changelog.rev(t.lnode) | ||
Augie Fackler
|
r43346 | if lnoderev in repo.changelog.ancestors( | ||
[parentrev], lnoderev, inclusive=True | ||||
): | ||||
Brendan Cully
|
r3714 | return True | ||
return False | ||||
Pierre-Yves David
|
r26346 | def apply(self, repo, source, revmap, merges, opts=None): | ||
Brendan Cully
|
r3714 | '''apply the revisions in revmap one by one in revision order''' | ||
Pierre-Yves David
|
r26346 | if opts is None: | ||
opts = {} | ||||
Matt Mackall
|
r8209 | revs = sorted(revmap) | ||
Martin von Zweigbergk
|
r41444 | p1 = repo.dirstate.p1() | ||
Brendan Cully
|
r3714 | pulls = [] | ||
Siddharth Agarwal
|
r23452 | diffopts = patch.difffeatureopts(self.ui, opts) | ||
Brendan Cully
|
r3714 | diffopts.git = True | ||
FUJIWARA Katsunori
|
r27289 | lock = tr = None | ||
Brendan Cully
|
r3714 | try: | ||
Matt Mackall
|
r4915 | lock = repo.lock() | ||
Augie Fackler
|
r43347 | tr = repo.transaction(b'transplant') | ||
Brendan Cully
|
r3714 | for rev in revs: | ||
node = revmap[rev] | ||||
Joerg Sonnenberger
|
r46729 | revstr = b'%d:%s' % (rev, short(node)) | ||
Brendan Cully
|
r3714 | |||
if self.applied(repo, node, p1): | ||||
Augie Fackler
|
r43346 | self.ui.warn( | ||
Augie Fackler
|
r43347 | _(b'skipping already applied revision %s\n') % revstr | ||
Augie Fackler
|
r43346 | ) | ||
Brendan Cully
|
r3714 | continue | ||
parents = source.changelog.parents(node) | ||||
Augie Fackler
|
r43347 | if not (opts.get(b'filter') or opts.get(b'log')): | ||
Martin Geisler
|
r7744 | # If the changeset parent is the same as the | ||
# wdir's parent, just pull it. | ||||
Brendan Cully
|
r3714 | if parents[0] == p1: | ||
pulls.append(node) | ||||
p1 = node | ||||
continue | ||||
if pulls: | ||||
if source != repo: | ||||
Pierre-Yves David
|
r22699 | exchange.pull(repo, source.peer(), heads=pulls) | ||
Martin von Zweigbergk
|
r46150 | merge.update(repo[pulls[-1]]) | ||
Martin von Zweigbergk
|
r41444 | p1 = repo.dirstate.p1() | ||
Brendan Cully
|
r3714 | pulls = [] | ||
domerge = False | ||||
if node in merges: | ||||
Martin Geisler
|
r7744 | # pulling all the merge revs at once would mean we | ||
# couldn't transplant after the latest even if | ||||
# transplants before them fail. | ||||
Brendan Cully
|
r3714 | domerge = True | ||
if not hasnode(repo, node): | ||||
Pierre-Yves David
|
r22699 | exchange.pull(repo, source.peer(), heads=[node]) | ||
Brendan Cully
|
r3714 | |||
Steven Stallion
|
r16400 | skipmerge = False | ||
Joerg Sonnenberger
|
r47771 | if parents[1] != repo.nullid: | ||
Augie Fackler
|
r43347 | if not opts.get(b'parent'): | ||
Augie Fackler
|
r43346 | self.ui.note( | ||
Augie Fackler
|
r43347 | _(b'skipping merge changeset %d:%s\n') | ||
Joerg Sonnenberger
|
r46729 | % (rev, short(node)) | ||
Augie Fackler
|
r43346 | ) | ||
Steven Stallion
|
r16400 | skipmerge = True | ||
else: | ||||
Augie Fackler
|
r43347 | parent = source.lookup(opts[b'parent']) | ||
Steven Stallion
|
r16400 | if parent not in parents: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | _(b'%s is not a parent of %s') | ||
Joerg Sonnenberger
|
r46729 | % (short(parent), short(node)) | ||
Augie Fackler
|
r43346 | ) | ||
Steven Stallion
|
r16400 | else: | ||
parent = parents[0] | ||||
if skipmerge: | ||||
Brendan Cully
|
r3714 | patchfile = None | ||
else: | ||||
Augie Fackler
|
r43347 | fd, patchfile = pycompat.mkstemp(prefix=b'hg-transplant-') | ||
Augie Fackler
|
r43906 | fp = os.fdopen(fd, 'wb') | ||
Steven Stallion
|
r16400 | gen = patch.diff(source, parent, node, opts=diffopts) | ||
Dirkjan Ochtman
|
r7308 | for chunk in gen: | ||
fp.write(chunk) | ||||
Brendan Cully
|
r3714 | fp.close() | ||
del revmap[rev] | ||||
if patchfile or domerge: | ||||
try: | ||||
Patrick Mezard
|
r16507 | try: | ||
Augie Fackler
|
r43346 | n = self.applyone( | ||
repo, | ||||
node, | ||||
source.changelog.read(node), | ||||
patchfile, | ||||
merge=domerge, | ||||
Augie Fackler
|
r43347 | log=opts.get(b'log'), | ||
filter=opts.get(b'filter'), | ||||
Augie Fackler
|
r43346 | ) | ||
Patrick Mezard
|
r16507 | except TransplantError: | ||
# Do not rollback, it is up to the user to | ||||
# fix the merge or cancel everything | ||||
tr.close() | ||||
raise | ||||
Brendan Cully
|
r4251 | if n and domerge: | ||
Augie Fackler
|
r43346 | self.ui.status( | ||
Joerg Sonnenberger
|
r46729 | _(b'%s merged at %s\n') % (revstr, short(n)) | ||
Augie Fackler
|
r43346 | ) | ||
Brendan Cully
|
r4251 | elif n: | ||
Augie Fackler
|
r43346 | self.ui.status( | ||
Augie Fackler
|
r43347 | _(b'%s transplanted to %s\n') | ||
Joerg Sonnenberger
|
r46729 | % (short(node), short(n)) | ||
Augie Fackler
|
r43346 | ) | ||
Brendan Cully
|
r3714 | finally: | ||
if patchfile: | ||||
os.unlink(patchfile) | ||||
Greg Ward
|
r15204 | tr.close() | ||
Brendan Cully
|
r3714 | if pulls: | ||
Pierre-Yves David
|
r22699 | exchange.pull(repo, source.peer(), heads=pulls) | ||
Martin von Zweigbergk
|
r46150 | merge.update(repo[pulls[-1]]) | ||
Brendan Cully
|
r3714 | finally: | ||
self.saveseries(revmap, merges) | ||||
self.transplants.write() | ||||
Greg Ward
|
r15204 | if tr: | ||
tr.release() | ||||
FUJIWARA Katsunori
|
r25879 | if lock: | ||
lock.release() | ||||
Brendan Cully
|
r3714 | |||
Luke Plant
|
r13579 | def filter(self, filter, node, changelog, patchfile): | ||
Brendan Cully
|
r3714 | '''arbitrarily rewrite changeset before applying it''' | ||
Augie Fackler
|
r43347 | self.ui.status(_(b'filtering %s\n') % patchfile) | ||
Brendan Cully
|
r3759 | user, date, msg = (changelog[1], changelog[2], changelog[4]) | ||
Augie Fackler
|
r43347 | fd, headerfile = pycompat.mkstemp(prefix=b'hg-transplant-') | ||
Augie Fackler
|
r43906 | fp = os.fdopen(fd, 'wb') | ||
Augie Fackler
|
r43347 | fp.write(b"# HG changeset patch\n") | ||
fp.write(b"# User %s\n" % user) | ||||
fp.write(b"# Date %d %d\n" % date) | ||||
fp.write(msg + b'\n') | ||||
Brendan Cully
|
r3759 | fp.close() | ||
try: | ||||
Augie Fackler
|
r43346 | self.ui.system( | ||
Augie Fackler
|
r43347 | b'%s %s %s' | ||
Augie Fackler
|
r43346 | % ( | ||
filter, | ||||
procutil.shellquote(headerfile), | ||||
procutil.shellquote(patchfile), | ||||
), | ||||
environ={ | ||||
Augie Fackler
|
r43347 | b'HGUSER': changelog[1], | ||
Joerg Sonnenberger
|
r46729 | b'HGREVISION': hex(node), | ||
Augie Fackler
|
r43346 | }, | ||
onerr=error.Abort, | ||||
Augie Fackler
|
r43347 | errprefix=_(b'filter failed'), | ||
blockedtag=b'transplant_filter', | ||||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r53279 | user, date, msg = self.parselog(open(headerfile, 'rb'))[1:4] | ||
Brendan Cully
|
r3759 | finally: | ||
os.unlink(headerfile) | ||||
return (user, date, msg) | ||||
Brendan Cully
|
r3714 | |||
Augie Fackler
|
r43346 | def applyone( | ||
self, repo, node, cl, patchfile, merge=False, log=False, filter=None | ||||
): | ||||
Brendan Cully
|
r3714 | '''apply the patch in patchfile to the repository as a transplant''' | ||
(manifest, user, (time, timezone), files, message) = cl[:5] | ||||
Augie Fackler
|
r43347 | date = b"%d %d" % (time, timezone) | ||
extra = {b'transplant_source': node} | ||||
Brendan Cully
|
r3714 | if filter: | ||
Luke Plant
|
r13579 | (user, date, message) = self.filter(filter, node, cl, patchfile) | ||
Brendan Cully
|
r3714 | |||
if log: | ||||
Martin Geisler
|
r9183 | # we don't translate messages inserted into commits | ||
Joerg Sonnenberger
|
r46729 | message += b'\n(transplanted from %s)' % hex(node) | ||
Brendan Cully
|
r3714 | |||
Joerg Sonnenberger
|
r46729 | self.ui.status(_(b'applying %s\n') % short(node)) | ||
Augie Fackler
|
r43347 | self.ui.note(b'%s %s\n%s\n' % (user, date, message)) | ||
Brendan Cully
|
r3714 | |||
if not patchfile and not merge: | ||||
Augie Fackler
|
r43347 | raise error.Abort(_(b'can only omit patchfile if merging')) | ||
Brendan Cully
|
r3714 | if patchfile: | ||
try: | ||||
Patrick Mezard
|
r14564 | files = set() | ||
Patrick Mezard
|
r14382 | patch.patch(self.ui, repo, patchfile, files=files, eolmode=None) | ||
Patrick Mezard
|
r14260 | files = list(files) | ||
Gregory Szorc
|
r25660 | except Exception as inst: | ||
Augie Fackler
|
r43347 | seriespath = os.path.join(self.path, b'series') | ||
Brendan Cully
|
r3757 | if os.path.exists(seriespath): | ||
os.unlink(seriespath) | ||||
Matt Mackall
|
r13878 | p1 = repo.dirstate.p1() | ||
Brendan Cully
|
r3714 | p2 = node | ||
Brendan Cully
|
r3725 | self.log(user, date, message, p1, p2, merge=merge) | ||
Augie Fackler
|
r43347 | self.ui.write(stringutil.forcebytestr(inst) + b'\n') | ||
Augie Fackler
|
r43346 | raise TransplantError( | ||
_( | ||||
Augie Fackler
|
r43347 | b'fix up the working directory and run ' | ||
b'hg transplant --continue' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Brendan Cully
|
r3714 | else: | ||
files = None | ||||
if merge: | ||||
Martin von Zweigbergk
|
r41444 | p1 = repo.dirstate.p1() | ||
Patrick Mezard
|
r16551 | repo.setparents(p1, node) | ||
Martin von Zweigbergk
|
r41825 | m = match.always() | ||
Matt Mackall
|
r8703 | else: | ||
Martin von Zweigbergk
|
r41825 | m = match.exact(files) | ||
Brendan Cully
|
r3714 | |||
Augie Fackler
|
r43346 | n = repo.commit( | ||
message, | ||||
user, | ||||
date, | ||||
extra=extra, | ||||
match=m, | ||||
editor=self.getcommiteditor(), | ||||
) | ||||
Greg Ward
|
r11638 | if not n: | ||
Joerg Sonnenberger
|
r46729 | self.ui.warn(_(b'skipping emptied changeset %s\n') % short(node)) | ||
Patrick Mezard
|
r17319 | return None | ||
Brendan Cully
|
r3714 | if not merge: | ||
self.transplants.set(n, node) | ||||
return n | ||||
timeless
|
r27677 | def canresume(self): | ||
Augie Fackler
|
r43347 | return os.path.exists(os.path.join(self.path, b'journal')) | ||
timeless
|
r27677 | |||
Bryan O'Sullivan
|
r18919 | def resume(self, repo, source, opts): | ||
Brendan Cully
|
r3714 | '''recover last transaction and apply remaining changesets''' | ||
Augie Fackler
|
r43347 | if os.path.exists(os.path.join(self.path, b'journal')): | ||
Bryan O'Sullivan
|
r18926 | n, node = self.recover(repo, source, opts) | ||
Pierre-Yves David
|
r23781 | if n: | ||
Augie Fackler
|
r43346 | self.ui.status( | ||
Joerg Sonnenberger
|
r46729 | _(b'%s transplanted as %s\n') % (short(node), short(n)) | ||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r23781 | else: | ||
Augie Fackler
|
r43346 | self.ui.status( | ||
Joerg Sonnenberger
|
r46729 | _(b'%s skipped due to empty diff\n') % (short(node),) | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | seriespath = os.path.join(self.path, b'series') | ||
Brendan Cully
|
r3714 | if not os.path.exists(seriespath): | ||
Brendan Cully
|
r3758 | self.transplants.write() | ||
Brendan Cully
|
r3714 | return | ||
nodes, merges = self.readseries() | ||||
revmap = {} | ||||
for n in nodes: | ||||
revmap[source.changelog.rev(n)] = n | ||||
os.unlink(seriespath) | ||||
self.apply(repo, source, revmap, merges, opts) | ||||
Bryan O'Sullivan
|
r18926 | def recover(self, repo, source, opts): | ||
Brendan Cully
|
r3714 | '''commit working directory using journal metadata''' | ||
node, user, date, message, parents = self.readlog() | ||||
Steven Stallion
|
r16400 | merge = False | ||
Brendan Cully
|
r3714 | |||
if not user or not date or not message or not parents[0]: | ||||
Augie Fackler
|
r43347 | raise error.Abort(_(b'transplant log file is corrupt')) | ||
Brendan Cully
|
r3714 | |||
Steven Stallion
|
r16400 | parent = parents[0] | ||
if len(parents) > 1: | ||||
Augie Fackler
|
r43347 | if opts.get(b'parent'): | ||
parent = source.lookup(opts[b'parent']) | ||||
Steven Stallion
|
r16400 | if parent not in parents: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | _(b'%s is not a parent of %s') | ||
Joerg Sonnenberger
|
r46729 | % (short(parent), short(node)) | ||
Augie Fackler
|
r43346 | ) | ||
Steven Stallion
|
r16400 | else: | ||
merge = True | ||||
Augie Fackler
|
r43347 | extra = {b'transplant_source': node} | ||
Matt Mackall
|
r4915 | try: | ||
Martin von Zweigbergk
|
r41444 | p1 = repo.dirstate.p1() | ||
Steven Stallion
|
r16400 | if p1 != parent: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Martin von Zweigbergk
|
r43387 | _(b'working directory not at transplant parent %s') | ||
Joerg Sonnenberger
|
r46729 | % hex(parent) | ||
Augie Fackler
|
r43346 | ) | ||
Matt Mackall
|
r4915 | if merge: | ||
Patrick Mezard
|
r16551 | repo.setparents(p1, parents[1]) | ||
Augie Fackler
|
r44041 | st = repo.status() | ||
modified, added, removed, deleted = ( | ||||
st.modified, | ||||
st.added, | ||||
st.removed, | ||||
st.deleted, | ||||
) | ||||
Pierre-Yves David
|
r23781 | if merge or modified or added or removed or deleted: | ||
Augie Fackler
|
r43346 | n = repo.commit( | ||
message, | ||||
user, | ||||
date, | ||||
extra=extra, | ||||
editor=self.getcommiteditor(), | ||||
) | ||||
Pierre-Yves David
|
r23781 | if not n: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b'commit failed')) | ||
Pierre-Yves David
|
r23781 | if not merge: | ||
self.transplants.set(n, node) | ||||
else: | ||||
n = None | ||||
Matt Mackall
|
r4915 | self.unlog() | ||
Brendan Cully
|
r3714 | |||
Matt Mackall
|
r4915 | return n, node | ||
finally: | ||||
FUJIWARA Katsunori
|
r27289 | # TODO: get rid of this meaningless try/finally enclosing. | ||
# this is kept only to reduce changes in a patch. | ||||
pass | ||||
Brendan Cully
|
r3714 | |||
Taapas Agrawal
|
r43030 | def stop(self, ui, repo): | ||
"""logic to stop an interrupted transplant""" | ||||
if self.canresume(): | ||||
Augie Fackler
|
r43347 | startctx = repo[b'.'] | ||
Martin von Zweigbergk
|
r46133 | merge.clean_update(startctx) | ||
Augie Fackler
|
r43347 | ui.status(_(b"stopped the interrupted transplant\n")) | ||
Augie Fackler
|
r43346 | ui.status( | ||
Augie Fackler
|
r43347 | _(b"working directory is now at %s\n") % startctx.hex()[:12] | ||
Augie Fackler
|
r43346 | ) | ||
Taapas Agrawal
|
r43030 | self.unlog() | ||
return 0 | ||||
Brendan Cully
|
r3714 | def readseries(self): | ||
nodes = [] | ||||
merges = [] | ||||
cur = nodes | ||||
Augie Fackler
|
r43347 | for line in self.opener.read(b'series').splitlines(): | ||
if line.startswith(b'# Merges'): | ||||
Brendan Cully
|
r3714 | cur = merges | ||
continue | ||||
Joerg Sonnenberger
|
r46729 | cur.append(bin(line)) | ||
Brendan Cully
|
r3714 | |||
return (nodes, merges) | ||||
def saveseries(self, revmap, merges): | ||||
if not revmap: | ||||
return | ||||
if not os.path.isdir(self.path): | ||||
os.mkdir(self.path) | ||||
Augie Fackler
|
r43347 | series = self.opener(b'series', b'w') | ||
Matt Mackall
|
r8209 | for rev in sorted(revmap): | ||
Joerg Sonnenberger
|
r46729 | series.write(hex(revmap[rev]) + b'\n') | ||
Brendan Cully
|
r3714 | if merges: | ||
Augie Fackler
|
r43347 | series.write(b'# Merges\n') | ||
Brendan Cully
|
r3714 | for m in merges: | ||
Joerg Sonnenberger
|
r46729 | series.write(hex(m) + b'\n') | ||
Brendan Cully
|
r3714 | series.close() | ||
Brendan Cully
|
r3759 | def parselog(self, fp): | ||
parents = [] | ||||
message = [] | ||||
Joerg Sonnenberger
|
r47771 | node = self.repo.nullid | ||
Brendan Cully
|
r3759 | inmsg = False | ||
Luke Plant
|
r13789 | user = None | ||
date = None | ||||
Brendan Cully
|
r3759 | for line in fp.read().splitlines(): | ||
if inmsg: | ||||
message.append(line) | ||||
Augie Fackler
|
r43347 | elif line.startswith(b'# User '): | ||
Brendan Cully
|
r3759 | user = line[7:] | ||
Augie Fackler
|
r43347 | elif line.startswith(b'# Date '): | ||
Brendan Cully
|
r3759 | date = line[7:] | ||
Augie Fackler
|
r43347 | elif line.startswith(b'# Node ID '): | ||
Joerg Sonnenberger
|
r46729 | node = bin(line[10:]) | ||
Augie Fackler
|
r43347 | elif line.startswith(b'# Parent '): | ||
Joerg Sonnenberger
|
r46729 | parents.append(bin(line[9:])) | ||
Augie Fackler
|
r43347 | elif not line.startswith(b'# '): | ||
Brendan Cully
|
r3759 | inmsg = True | ||
message.append(line) | ||||
Luke Plant
|
r13789 | if None in (user, date): | ||
Augie Fackler
|
r43347 | raise error.Abort( | ||
_(b"filter corrupted changeset (no user or date)") | ||||
) | ||||
return (node, user, date, b'\n'.join(message), parents) | ||||
Thomas Arendsen Hein
|
r4516 | |||
Brendan Cully
|
r3725 | def log(self, user, date, message, p1, p2, merge=False): | ||
Brendan Cully
|
r3714 | '''journal changelog metadata for later recover''' | ||
if not os.path.isdir(self.path): | ||||
os.mkdir(self.path) | ||||
Augie Fackler
|
r43347 | fp = self.opener(b'journal', b'w') | ||
fp.write(b'# User %s\n' % user) | ||||
fp.write(b'# Date %s\n' % date) | ||||
Joerg Sonnenberger
|
r46729 | fp.write(b'# Node ID %s\n' % hex(p2)) | ||
fp.write(b'# Parent ' + hex(p1) + b'\n') | ||||
Brendan Cully
|
r3714 | if merge: | ||
Joerg Sonnenberger
|
r46729 | fp.write(b'# Parent ' + hex(p2) + b'\n') | ||
Augie Fackler
|
r43347 | fp.write(message.rstrip() + b'\n') | ||
Brendan Cully
|
r3714 | fp.close() | ||
def readlog(self): | ||||
Augie Fackler
|
r43347 | return self.parselog(self.opener(b'journal')) | ||
Brendan Cully
|
r3714 | |||
def unlog(self): | ||||
'''remove changelog journal''' | ||||
Augie Fackler
|
r43347 | absdst = os.path.join(self.path, b'journal') | ||
Brendan Cully
|
r3714 | if os.path.exists(absdst): | ||
os.unlink(absdst) | ||||
def transplantfilter(self, repo, source, root): | ||||
def matchfn(node): | ||||
if self.applied(repo, node, root): | ||||
return False | ||||
Joerg Sonnenberger
|
r47771 | if source.changelog.parents(node)[1] != repo.nullid: | ||
Brendan Cully
|
r3714 | return False | ||
extra = source.changelog.read(node)[5] | ||||
Augie Fackler
|
r43347 | cnode = extra.get(b'transplant_source') | ||
Brendan Cully
|
r3714 | if cnode and self.applied(repo, cnode, root): | ||
return False | ||||
return True | ||||
return matchfn | ||||
Augie Fackler
|
r43346 | |||
Brendan Cully
|
r3714 | def hasnode(repo, node): | ||
try: | ||||
Martin Geisler
|
r13031 | return repo.changelog.rev(node) is not None | ||
Gregory Szorc
|
r39813 | except error.StorageError: | ||
Brendan Cully
|
r3714 | return False | ||
Augie Fackler
|
r43346 | |||
Brendan Cully
|
r3714 | def browserevs(ui, repo, nodes, opts): | ||
'''interactively transplant changesets''' | ||||
Yuya Nishihara
|
r35906 | displayer = logcmdutil.changesetdisplayer(ui, repo, opts) | ||
Brendan Cully
|
r3714 | transplants = [] | ||
merges = [] | ||||
Augie Fackler
|
r43346 | prompt = _( | ||
Augie Fackler
|
r43347 | b'apply changeset? [ynmpcq?]:' | ||
b'$$ &yes, transplant this changeset' | ||||
b'$$ &no, skip this changeset' | ||||
b'$$ &merge at this changeset' | ||||
b'$$ show &patch' | ||||
b'$$ &commit selected changesets' | ||||
b'$$ &quit and cancel transplant' | ||||
b'$$ &? (show this help)' | ||||
Augie Fackler
|
r43346 | ) | ||
Brendan Cully
|
r3714 | for node in nodes: | ||
Dirkjan Ochtman
|
r7369 | displayer.show(repo[node]) | ||
Brendan Cully
|
r3714 | action = None | ||
while not action: | ||||
Pulkit Goyal
|
r38389 | choice = ui.promptchoice(prompt) | ||
Augie Fackler
|
r43347 | action = b'ynmpcq?'[choice : choice + 1] | ||
if action == b'?': | ||||
FUJIWARA Katsunori
|
r20269 | for c, t in ui.extractchoices(prompt)[1]: | ||
Augie Fackler
|
r43347 | ui.write(b'%s: %s\n' % (c, t)) | ||
Brendan Cully
|
r3714 | action = None | ||
Augie Fackler
|
r43347 | elif action == b'p': | ||
Brendan Cully
|
r3714 | parent = repo.changelog.parents(node)[0] | ||
Dirkjan Ochtman
|
r7308 | for chunk in patch.diff(repo, parent, node): | ||
Martin Geisler
|
r8615 | ui.write(chunk) | ||
Brendan Cully
|
r3714 | action = None | ||
Augie Fackler
|
r43347 | if action == b'y': | ||
Brendan Cully
|
r3714 | transplants.append(node) | ||
Augie Fackler
|
r43347 | elif action == b'm': | ||
Brendan Cully
|
r3714 | merges.append(node) | ||
Augie Fackler
|
r43347 | elif action == b'c': | ||
Brendan Cully
|
r3714 | break | ||
Augie Fackler
|
r43347 | elif action == b'q': | ||
Brendan Cully
|
r3714 | transplants = () | ||
merges = () | ||||
break | ||||
Robert Bachmann
|
r10152 | displayer.close() | ||
Brendan Cully
|
r3714 | return (transplants, merges) | ||
Augie Fackler
|
r43346 | |||
@command( | ||||
Augie Fackler
|
r43347 | b'transplant', | ||
Augie Fackler
|
r43346 | [ | ||
Augie Fackler
|
r43347 | ( | ||
b's', | ||||
b'source', | ||||
b'', | ||||
_(b'transplant changesets from REPO'), | ||||
_(b'REPO'), | ||||
), | ||||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'b', | ||
b'branch', | ||||
[], | ||||
_(b'use this source changeset as head'), | ||||
_(b'REV'), | ||||
), | ||||
( | ||||
b'a', | ||||
b'all', | ||||
Augie Fackler
|
r43346 | None, | ||
Augie Fackler
|
r43347 | _(b'pull all changesets up to the --branch revisions'), | ||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | (b'p', b'prune', [], _(b'skip over REV'), _(b'REV')), | ||
(b'm', b'merge', [], _(b'merge at REV'), _(b'REV')), | ||||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'', | ||
b'parent', | ||||
b'', | ||||
_(b'parent to choose when transplanting merge'), | ||||
_(b'REV'), | ||||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | (b'e', b'edit', False, _(b'invoke editor on commit messages')), | ||
(b'', b'log', None, _(b'append transplant info to log message')), | ||||
(b'', b'stop', False, _(b'stop interrupted transplant')), | ||||
( | ||||
b'c', | ||||
b'continue', | ||||
None, | ||||
Martin von Zweigbergk
|
r43387 | _(b'continue last transplant session after fixing conflicts'), | ||
Augie Fackler
|
r43347 | ), | ||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'', | ||
b'filter', | ||||
b'', | ||||
_(b'filter changesets through command'), | ||||
_(b'CMD'), | ||||
Augie Fackler
|
r43346 | ), | ||
], | ||||
Augie Fackler
|
r43347 | _( | ||
b'hg transplant [-s REPO] [-b BRANCH [-a]] [-p REV] ' | ||||
b'[-m REV] [REV]...' | ||||
), | ||||
Augie Fackler
|
r43346 | helpcategory=command.CATEGORY_CHANGE_MANAGEMENT, | ||
) | ||||
Brendan Cully
|
r3714 | def transplant(ui, repo, *revs, **opts): | ||
Augie Fackler
|
r46554 | """transplant changesets from another branch | ||
Brendan Cully
|
r3714 | |||
Selected changesets will be applied on top of the current working | ||||
Martin Geisler
|
r13605 | directory with the log of the original changeset. The changesets | ||
Mads Kiilerich
|
r19028 | are copied and will thus appear twice in the history with different | ||
identities. | ||||
Consider using the graft command if everything is inside the same | ||||
repository - it will use merges and will usually give a better result. | ||||
Use the rebase extension if the changesets are unpublished and you want | ||||
to move them instead of copying them. | ||||
Martin Geisler
|
r13605 | |||
If --log is specified, log messages will have a comment appended | ||||
of the form:: | ||||
Brendan Cully
|
r3714 | |||
Martin Geisler
|
r9200 | (transplanted from CHANGESETHASH) | ||
Brendan Cully
|
r3714 | |||
You can rewrite the changelog message with the --filter option. | ||||
Martin Geisler
|
r8000 | Its argument will be invoked with the current changelog message as | ||
$1 and the patch as $2. | ||||
Brendan Cully
|
r3714 | |||
Mads Kiilerich
|
r19028 | --source/-s specifies another repository to use for selecting changesets, | ||
just as if it temporarily had been pulled. | ||||
Mads Kiilerich
|
r19027 | If --branch/-b is specified, these revisions will be used as | ||
Mads Kiilerich
|
r19951 | heads when deciding which changesets to transplant, just as if only | ||
Mads Kiilerich
|
r19027 | these revisions had been pulled. | ||
If --all/-a is specified, all the revisions up to the heads specified | ||||
with --branch will be transplanted. | ||||
Brendan Cully
|
r3714 | |||
Mads Kiilerich
|
r19027 | Example: | ||
- transplant all changes up to REV on top of your current revision:: | ||||
hg transplant --branch REV --all | ||||
Brendan Cully
|
r3714 | |||
Martin Geisler
|
r8000 | You can optionally mark selected transplanted changesets as merge | ||
changesets. You will not be prompted to transplant any ancestors | ||||
of a merged transplant, and you can merge descendants of them | ||||
normally instead of transplanting them. | ||||
Brendan Cully
|
r3714 | |||
Steven Stallion
|
r16400 | Merge changesets may be transplanted directly by specifying the | ||
Steven Stallion
|
r16457 | proper parent changeset by calling :hg:`transplant --parent`. | ||
Steven Stallion
|
r16400 | |||
Martin Geisler
|
r11193 | If no merges or revisions are provided, :hg:`transplant` will | ||
start an interactive changeset browser. | ||||
Brendan Cully
|
r3714 | |||
Martin Geisler
|
r8000 | If a changeset application fails, you can fix the merge by hand | ||
Martin Geisler
|
r11193 | and then resume where you left off by calling :hg:`transplant | ||
--continue/-c`. | ||||
Augie Fackler
|
r46554 | """ | ||
Bryan O'Sullivan
|
r27840 | with repo.wlock(): | ||
FUJIWARA Katsunori
|
r27289 | return _dotransplant(ui, repo, *revs, **opts) | ||
Augie Fackler
|
r43346 | |||
FUJIWARA Katsunori
|
r27289 | def _dotransplant(ui, repo, *revs, **opts): | ||
Peter Arrenbrecht
|
r14161 | def incwalk(repo, csets, match=util.always): | ||
for node in csets: | ||||
Brendan Cully
|
r3714 | if match(node): | ||
yield node | ||||
Mads Kiilerich
|
r19027 | def transplantwalk(repo, dest, heads, match=util.always): | ||
Augie Fackler
|
r46554 | """Yield all nodes that are ancestors of a head but not ancestors | ||
Mads Kiilerich
|
r19027 | of dest. | ||
Augie Fackler
|
r46554 | If no heads are specified, the heads of repo will be used.""" | ||
Mads Kiilerich
|
r19027 | if not heads: | ||
heads = repo.heads() | ||||
Brendan Cully
|
r3714 | ancestors = [] | ||
Mads Kiilerich
|
r20988 | ctx = repo[dest] | ||
Mads Kiilerich
|
r19027 | for head in heads: | ||
Mads Kiilerich
|
r20988 | ancestors.append(ctx.ancestor(repo[head]).node()) | ||
Mads Kiilerich
|
r19027 | for node in repo.changelog.nodesbetween(ancestors, heads)[0]: | ||
Brendan Cully
|
r3714 | if match(node): | ||
yield node | ||||
def checkopts(opts, revs): | ||||
Augie Fackler
|
r43347 | if opts.get(b'continue'): | ||
Martin von Zweigbergk
|
r44355 | cmdutil.check_incompatible_arguments( | ||
Martin von Zweigbergk
|
r44655 | opts, b'continue', [b'branch', b'all', b'merge'] | ||
Martin von Zweigbergk
|
r44355 | ) | ||
Brendan Cully
|
r3714 | return | ||
Augie Fackler
|
r43347 | if opts.get(b'stop'): | ||
Martin von Zweigbergk
|
r44355 | cmdutil.check_incompatible_arguments( | ||
Martin von Zweigbergk
|
r44655 | opts, b'stop', [b'branch', b'all', b'merge'] | ||
Martin von Zweigbergk
|
r44355 | ) | ||
Taapas Agrawal
|
r43030 | return | ||
Augie Fackler
|
r43346 | if not ( | ||
Augie Fackler
|
r43347 | opts.get(b'source') | ||
Augie Fackler
|
r43346 | or revs | ||
Augie Fackler
|
r43347 | or opts.get(b'merge') | ||
or opts.get(b'branch') | ||||
Augie Fackler
|
r43346 | ): | ||
raise error.Abort( | ||||
_( | ||||
Augie Fackler
|
r43347 | b'no source URL, branch revision, or revision ' | ||
b'list provided' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Augie Fackler
|
r43347 | if opts.get(b'all'): | ||
if not opts.get(b'branch'): | ||||
raise error.Abort(_(b'--all requires a branch revision')) | ||||
Brendan Cully
|
r3714 | if revs: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Martin von Zweigbergk
|
r43387 | _(b'--all is incompatible with a revision list') | ||
Augie Fackler
|
r43346 | ) | ||
Brendan Cully
|
r3714 | |||
Pulkit Goyal
|
r36208 | opts = pycompat.byteskwargs(opts) | ||
Brendan Cully
|
r3714 | checkopts(opts, revs) | ||
Augie Fackler
|
r43347 | if not opts.get(b'log'): | ||
Matt Mackall
|
r25828 | # deprecated config: transplant.log | ||
Augie Fackler
|
r43347 | opts[b'log'] = ui.config(b'transplant', b'log') | ||
if not opts.get(b'filter'): | ||||
Matt Mackall
|
r25828 | # deprecated config: transplant.filter | ||
Augie Fackler
|
r43347 | opts[b'filter'] = ui.config(b'transplant', b'filter') | ||
Brendan Cully
|
r3714 | |||
FUJIWARA Katsunori
|
r21411 | tp = transplanter(ui, repo, opts) | ||
Brendan Cully
|
r3714 | |||
Martin von Zweigbergk
|
r41444 | p1 = repo.dirstate.p1() | ||
Joerg Sonnenberger
|
r47771 | if len(repo) > 0 and p1 == repo.nullid: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b'no revision checked out')) | ||
if opts.get(b'continue'): | ||||
timeless
|
r27677 | if not tp.canresume(): | ||
Martin von Zweigbergk
|
r46444 | raise error.StateError(_(b'no transplant to continue')) | ||
Augie Fackler
|
r43347 | elif opts.get(b'stop'): | ||
Yuya Nishihara
|
r43033 | if not tp.canresume(): | ||
Martin von Zweigbergk
|
r46444 | raise error.StateError(_(b'no interrupted transplant found')) | ||
Yuya Nishihara
|
r43033 | return tp.stop(ui, repo) | ||
timeless
|
r27677 | else: | ||
cmdutil.checkunfinished(repo) | ||||
Martin von Zweigbergk
|
r41416 | cmdutil.bailifchanged(repo) | ||
Brendan Cully
|
r3714 | |||
Augie Fackler
|
r43347 | sourcerepo = opts.get(b'source') | ||
Peter Arrenbrecht
|
r14161 | if sourcerepo: | ||
r50624 | path = urlutil.get_unique_pull_path_obj(b'transplant', ui, sourcerepo) | |||
peer = hg.peer(repo, opts, path) | ||||
Augie Fackler
|
r43347 | heads = pycompat.maplist(peer.lookup, opts.get(b'branch', ())) | ||
Pierre-Yves David
|
r25679 | target = set(heads) | ||
for r in revs: | ||||
try: | ||||
target.add(peer.lookup(r)) | ||||
except error.RepoError: | ||||
pass | ||||
Augie Fackler
|
r43346 | source, csets, cleanupfn = bundlerepo.getremotechanges( | ||
ui, repo, peer, onlyheads=sorted(target), force=True | ||||
) | ||||
Brendan Cully
|
r3714 | else: | ||
source = repo | ||||
Augie Fackler
|
r43347 | heads = pycompat.maplist(source.lookup, opts.get(b'branch', ())) | ||
Peter Arrenbrecht
|
r14161 | cleanupfn = None | ||
Brendan Cully
|
r3714 | |||
try: | ||||
Augie Fackler
|
r43347 | if opts.get(b'continue'): | ||
Brendan Cully
|
r3724 | tp.resume(repo, source, opts) | ||
Brendan Cully
|
r3714 | return | ||
Benoit Boissinot
|
r10394 | tf = tp.transplantfilter(repo, source, p1) | ||
Augie Fackler
|
r43347 | if opts.get(b'prune'): | ||
Augie Fackler
|
r44937 | prune = { | ||
Augie Fackler
|
r43346 | source[r].node() | ||
Martin von Zweigbergk
|
r48928 | for r in logcmdutil.revrange(source, opts.get(b'prune')) | ||
Augie Fackler
|
r44937 | } | ||
Brendan Cully
|
r3714 | matchfn = lambda x: tf(x) and x not in prune | ||
else: | ||||
matchfn = tf | ||||
Augie Fackler
|
r43347 | merges = pycompat.maplist(source.lookup, opts.get(b'merge', ())) | ||
Brendan Cully
|
r3714 | revmap = {} | ||
if revs: | ||||
Martin von Zweigbergk
|
r48928 | for r in logcmdutil.revrange(source, revs): | ||
Martin von Zweigbergk
|
r37327 | revmap[int(r)] = source[r].node() | ||
Augie Fackler
|
r43347 | elif opts.get(b'all') or not merges: | ||
Brendan Cully
|
r3714 | if source != repo: | ||
Peter Arrenbrecht
|
r14161 | alltransplants = incwalk(source, csets, match=matchfn) | ||
Brendan Cully
|
r3714 | else: | ||
Augie Fackler
|
r43346 | alltransplants = transplantwalk( | ||
source, p1, heads, match=matchfn | ||||
) | ||||
Augie Fackler
|
r43347 | if opts.get(b'all'): | ||
Brendan Cully
|
r3714 | revs = alltransplants | ||
else: | ||||
revs, newmerges = browserevs(ui, source, alltransplants, opts) | ||||
merges.extend(newmerges) | ||||
for r in revs: | ||||
revmap[source.changelog.rev(r)] = r | ||||
for r in merges: | ||||
revmap[source.changelog.rev(r)] = r | ||||
tp.apply(repo, source, revmap, merges, opts) | ||||
finally: | ||||
Peter Arrenbrecht
|
r14161 | if cleanupfn: | ||
cleanupfn() | ||||
Brendan Cully
|
r3714 | |||
Augie Fackler
|
r43346 | |||
Taapas Agrawal
|
r42943 | def continuecmd(ui, repo): | ||
"""logic to resume an interrupted transplant using | ||||
'hg continue'""" | ||||
with repo.wlock(): | ||||
tp = transplanter(ui, repo, {}) | ||||
return tp.resume(repo, repo, {}) | ||||
Augie Fackler
|
r43346 | |||
FUJIWARA Katsunori
|
r28394 | revsetpredicate = registrar.revsetpredicate() | ||
FUJIWARA Katsunori
|
r27586 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | @revsetpredicate(b'transplanted([set])') | ||
Juan Pablo Aroztegi
|
r12581 | def revsettransplanted(repo, subset, x): | ||
Augie Fackler
|
r46554 | """Transplanted changesets in set, or all transplanted changesets.""" | ||
Juan Pablo Aroztegi
|
r12581 | if x: | ||
Mads Kiilerich
|
r17299 | s = revset.getset(repo, subset, x) | ||
Juan Pablo Aroztegi
|
r12581 | else: | ||
Mads Kiilerich
|
r17299 | s = subset | ||
Augie Fackler
|
r43346 | return smartset.baseset( | ||
Augie Fackler
|
r43347 | [r for r in s if repo[r].extra().get(b'transplant_source')] | ||
Augie Fackler
|
r43346 | ) | ||
Juan Pablo Aroztegi
|
r12581 | |||
FUJIWARA Katsunori
|
r28540 | templatekeyword = registrar.templatekeyword() | ||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | @templatekeyword(b'transplanted', requires={b'ctx'}) | ||
Yuya Nishihara
|
r36531 | def kwtransplanted(context, mapping): | ||
FUJIWARA Katsunori
|
r28540 | """String. The node identifier of the transplanted | ||
Patrick Mezard
|
r13689 | changeset if any.""" | ||
Augie Fackler
|
r43347 | ctx = context.resource(mapping, b'ctx') | ||
n = ctx.extra().get(b'transplant_source') | ||||
Joerg Sonnenberger
|
r46729 | return n and hex(n) or b'' | ||
Juan Pablo Aroztegi
|
r12581 | |||
Augie Fackler
|
r43346 | |||
Patrick Mezard
|
r12822 | def extsetup(ui): | ||
Augie Fackler
|
r43346 | statemod.addunfinished( | ||
Augie Fackler
|
r43347 | b'transplant', | ||
fname=b'transplant/journal', | ||||
Augie Fackler
|
r43346 | clearable=True, | ||
Taapas Agrawal
|
r42943 | continuefunc=continuecmd, | ||
Augie Fackler
|
r43346 | statushint=_( | ||
Augie Fackler
|
r43347 | b'To continue: hg transplant --continue\n' | ||
b'To stop: hg transplant --stop' | ||||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | cmdhint=_(b"use 'hg transplant --continue' or 'hg transplant --stop'"), | ||
Taapas Agrawal
|
r42730 | ) | ||
Juan Pablo Aroztegi
|
r12581 | |||
Augie Fackler
|
r43346 | |||
Patrick Mezard
|
r12823 | # tell hggettext to extract docstrings from these functions: | ||
Patrick Mezard
|
r13698 | i18nfunctions = [revsettransplanted, kwtransplanted] | ||