hg.py
284 lines
| 9.6 KiB
| text/x-python
|
PythonLexer
Brendan Cully
|
r4536 | # hg backend for convert extension | ||
Bryan O'Sullivan
|
r5556 | # Notes for hg->hg conversion: | ||
# | ||||
# * Old versions of Mercurial didn't trim the whitespace from the ends | ||||
# of commit messages, but new versions do. Changesets created by | ||||
# those older versions, then converted, may thus have different | ||||
# hashes for changesets that are otherwise identical. | ||||
# | ||||
# * By default, the source revision is stored in the converted | ||||
# revision. This will cause the converted revision to have a | ||||
# different identity than the source. To avoid this, use the | ||||
# following option: "--config convert.hg.saverev=false" | ||||
Bryan O'Sullivan
|
r5013 | |||
Brendan Cully
|
r4536 | import os, time | ||
Bryan O'Sullivan
|
r5014 | from mercurial.i18n import _ | ||
Bryan O'Sullivan
|
r5013 | from mercurial.node import * | ||
Bryan O'Sullivan
|
r5014 | from mercurial import hg, lock, revlog, util | ||
Brendan Cully
|
r4536 | |||
Bryan O'Sullivan
|
r5013 | from common import NoRepo, commit, converter_source, converter_sink | ||
Brendan Cully
|
r4536 | |||
Bryan O'Sullivan
|
r5013 | class mercurial_sink(converter_sink): | ||
Brendan Cully
|
r4763 | def __init__(self, ui, path): | ||
Bryan O'Sullivan
|
r5440 | converter_sink.__init__(self, ui, path) | ||
Brendan Cully
|
r5173 | self.branchnames = ui.configbool('convert', 'hg.usebranchnames', True) | ||
self.clonebranches = ui.configbool('convert', 'hg.clonebranches', False) | ||||
Brendan Cully
|
r5260 | self.tagsbranch = ui.config('convert', 'hg.tagsbranch', 'default') | ||
Brendan Cully
|
r5173 | self.lastbranch = None | ||
Bryan O'Sullivan
|
r5441 | if os.path.isdir(path) and len(os.listdir(path)) > 0: | ||
try: | ||||
self.repo = hg.repository(self.ui, path) | ||||
except hg.RepoError, err: | ||||
ui.print_exc() | ||||
raise NoRepo(err.args[0]) | ||||
else: | ||||
try: | ||||
ui.status(_('initializing destination %s repository\n') % path) | ||||
self.repo = hg.repository(self.ui, path, create=True) | ||||
self.created.append(path) | ||||
except hg.RepoError, err: | ||||
ui.print_exc() | ||||
raise NoRepo("could not create hg repo %s as sink" % path) | ||||
Bryan O'Sullivan
|
r5014 | self.lock = None | ||
self.wlock = None | ||||
Alexis S. L. Carvalho
|
r5378 | self.filemapmode = False | ||
Bryan O'Sullivan
|
r5014 | |||
def before(self): | ||||
Patrick Mezard
|
r5805 | self.ui.debug(_('run hg sink pre-conversion action\n')) | ||
Alexis S. L. Carvalho
|
r5052 | self.wlock = self.repo.wlock() | ||
Bryan O'Sullivan
|
r5014 | self.lock = self.repo.lock() | ||
Alexis S. L. Carvalho
|
r5279 | self.repo.dirstate.clear() | ||
Bryan O'Sullivan
|
r5014 | |||
def after(self): | ||||
Patrick Mezard
|
r5805 | self.ui.debug(_('run hg sink post-conversion action\n')) | ||
Alexis S. L. Carvalho
|
r5279 | self.repo.dirstate.invalidate() | ||
Bryan O'Sullivan
|
r5014 | self.lock = None | ||
self.wlock = None | ||||
Brendan Cully
|
r4536 | |||
Bryan O'Sullivan
|
r5011 | def revmapfile(self): | ||
Brendan Cully
|
r4536 | return os.path.join(self.path, ".hg", "shamap") | ||
Edouard Gomez
|
r4589 | def authorfile(self): | ||
return os.path.join(self.path, ".hg", "authormap") | ||||
Brendan Cully
|
r4536 | def getheads(self): | ||
h = self.repo.changelog.heads() | ||||
Bryan O'Sullivan
|
r5017 | return [ hex(x) for x in h ] | ||
Brendan Cully
|
r4536 | |||
def putfile(self, f, e, data): | ||||
self.repo.wwrite(f, data, e) | ||||
Matt Mackall
|
r4906 | if f not in self.repo.dirstate: | ||
Alexis S. L. Carvalho
|
r5278 | self.repo.dirstate.normallookup(f) | ||
Brendan Cully
|
r4536 | |||
Daniel Holth
|
r4765 | def copyfile(self, source, dest): | ||
self.repo.copy(source, dest) | ||||
Brendan Cully
|
r4536 | def delfile(self, f): | ||
try: | ||||
Bryan O'Sullivan
|
r5343 | util.unlink(self.repo.wjoin(f)) | ||
Brendan Cully
|
r4536 | #self.repo.remove([f]) | ||
Patrick Mezard
|
r5344 | except OSError: | ||
Brendan Cully
|
r4536 | pass | ||
Brendan Cully
|
r5173 | def setbranch(self, branch, pbranch, parents): | ||
if (not self.clonebranches) or (branch == self.lastbranch): | ||||
return | ||||
self.lastbranch = branch | ||||
self.after() | ||||
if not branch: | ||||
branch = 'default' | ||||
if not pbranch: | ||||
pbranch = 'default' | ||||
branchpath = os.path.join(self.path, branch) | ||||
try: | ||||
self.repo = hg.repository(self.ui, branchpath) | ||||
except: | ||||
if not parents: | ||||
self.repo = hg.repository(self.ui, branchpath, create=True) | ||||
else: | ||||
self.ui.note(_('cloning branch %s to %s\n') % (pbranch, branch)) | ||||
hg.clone(self.ui, os.path.join(self.path, pbranch), | ||||
branchpath, rev=parents, update=False, | ||||
stream=True) | ||||
self.repo = hg.repository(self.ui, branchpath) | ||||
Alexis S. L. Carvalho
|
r5402 | self.before() | ||
Brendan Cully
|
r5173 | |||
Brendan Cully
|
r4536 | def putcommit(self, files, parents, commit): | ||
Alexis S. L. Carvalho
|
r5195 | seen = {} | ||
Brendan Cully
|
r4536 | pl = [] | ||
for p in parents: | ||||
if p not in seen: | ||||
pl.append(p) | ||||
seen[p] = 1 | ||||
parents = pl | ||||
Alexis S. L. Carvalho
|
r5378 | nparents = len(parents) | ||
if self.filemapmode and nparents == 1: | ||||
m1node = self.repo.changelog.read(bin(parents[0]))[0] | ||||
parent = parents[0] | ||||
Brendan Cully
|
r4536 | |||
if len(parents) < 2: parents.append("0" * 40) | ||||
if len(parents) < 2: parents.append("0" * 40) | ||||
p2 = parents.pop(0) | ||||
text = commit.desc | ||||
Bryan O'Sullivan
|
r5439 | extra = commit.extra.copy() | ||
Bryan O'Sullivan
|
r5038 | if self.branchnames and commit.branch: | ||
Brendan Cully
|
r4873 | extra['branch'] = commit.branch | ||
if commit.rev: | ||||
extra['convert_revision'] = commit.rev | ||||
Thomas Arendsen Hein
|
r4957 | |||
Brendan Cully
|
r4536 | while parents: | ||
p1 = p2 | ||||
p2 = parents.pop(0) | ||||
a = self.repo.rawcommit(files, text, commit.author, commit.date, | ||||
Bryan O'Sullivan
|
r5017 | bin(p1), bin(p2), extra=extra) | ||
Alexis S. L. Carvalho
|
r5278 | self.repo.dirstate.clear() | ||
Brendan Cully
|
r4536 | text = "(octopus merge fixup)\n" | ||
p2 = hg.hex(self.repo.changelog.tip()) | ||||
Alexis S. L. Carvalho
|
r5378 | if self.filemapmode and nparents == 1: | ||
man = self.repo.manifest | ||||
mnode = self.repo.changelog.read(bin(p2))[0] | ||||
if not man.cmp(m1node, man.revision(mnode)): | ||||
self.repo.rollback() | ||||
self.repo.dirstate.clear() | ||||
return parent | ||||
Brendan Cully
|
r4536 | return p2 | ||
def puttags(self, tags): | ||||
try: | ||||
old = self.repo.wfile(".hgtags").read() | ||||
oldlines = old.splitlines(1) | ||||
oldlines.sort() | ||||
except: | ||||
oldlines = [] | ||||
k = tags.keys() | ||||
k.sort() | ||||
newlines = [] | ||||
for tag in k: | ||||
newlines.append("%s %s\n" % (tags[tag], tag)) | ||||
newlines.sort() | ||||
if newlines != oldlines: | ||||
self.ui.status("updating tags\n") | ||||
f = self.repo.wfile(".hgtags", "w") | ||||
f.write("".join(newlines)) | ||||
f.close() | ||||
if not oldlines: self.repo.add([".hgtags"]) | ||||
date = "%s 0" % int(time.mktime(time.gmtime())) | ||||
Brendan Cully
|
r5260 | extra = {} | ||
if self.tagsbranch != 'default': | ||||
extra['branch'] = self.tagsbranch | ||||
try: | ||||
tagparent = self.repo.changectx(self.tagsbranch).node() | ||||
except hg.RepoError, inst: | ||||
tagparent = nullid | ||||
Brendan Cully
|
r4536 | self.repo.rawcommit([".hgtags"], "update tags", "convert-repo", | ||
Edouard Gomez
|
r5838 | date, tagparent, nullid, extra=extra) | ||
Bryan O'Sullivan
|
r5017 | return hex(self.repo.changelog.tip()) | ||
Bryan O'Sullivan
|
r5013 | |||
Alexis S. L. Carvalho
|
r5378 | def setfilemapmode(self, active): | ||
self.filemapmode = active | ||||
Bryan O'Sullivan
|
r5013 | class mercurial_source(converter_source): | ||
def __init__(self, ui, path, rev=None): | ||||
converter_source.__init__(self, ui, path, rev) | ||||
Bryan O'Sullivan
|
r5556 | self.saverev = ui.configbool('convert', 'hg.saverev', True) | ||
Bryan O'Sullivan
|
r5358 | try: | ||
self.repo = hg.repository(self.ui, path) | ||||
Bryan O'Sullivan
|
r5437 | # try to provoke an exception if this isn't really a hg | ||
# repo, but some other bogus compatible-looking url | ||||
Alexis S. L. Carvalho
|
r5522 | if not self.repo.local(): | ||
raise hg.RepoError() | ||||
Bryan O'Sullivan
|
r5437 | except hg.RepoError: | ||
ui.print_exc() | ||||
Alexis S. L. Carvalho
|
r5522 | raise NoRepo("%s is not a local Mercurial repo" % path) | ||
Bryan O'Sullivan
|
r5013 | self.lastrev = None | ||
self.lastctx = None | ||||
Alexis S. L. Carvalho
|
r5379 | self._changescache = None | ||
Bryan O'Sullivan
|
r5554 | self.convertfp = None | ||
Bryan O'Sullivan
|
r5013 | |||
def changectx(self, rev): | ||||
if self.lastrev != rev: | ||||
self.lastctx = self.repo.changectx(rev) | ||||
self.lastrev = rev | ||||
return self.lastctx | ||||
def getheads(self): | ||||
Bryan O'Sullivan
|
r5131 | if self.rev: | ||
return [hex(self.repo.changectx(self.rev).node())] | ||||
else: | ||||
return [hex(node) for node in self.repo.heads()] | ||||
Bryan O'Sullivan
|
r5013 | |||
def getfile(self, name, rev): | ||||
try: | ||||
return self.changectx(rev).filectx(name).data() | ||||
except revlog.LookupError, err: | ||||
raise IOError(err) | ||||
def getmode(self, name, rev): | ||||
m = self.changectx(rev).manifest() | ||||
return (m.execf(name) and 'x' or '') + (m.linkf(name) and 'l' or '') | ||||
def getchanges(self, rev): | ||||
ctx = self.changectx(rev) | ||||
Alexis S. L. Carvalho
|
r5379 | if self._changescache and self._changescache[0] == rev: | ||
m, a, r = self._changescache[1] | ||||
else: | ||||
m, a, r = self.repo.status(ctx.parents()[0].node(), ctx.node())[:3] | ||||
Bryan O'Sullivan
|
r5013 | changes = [(name, rev) for name in m + a + r] | ||
changes.sort() | ||||
Alexis S. L. Carvalho
|
r5280 | return (changes, self.getcopies(ctx, m + a)) | ||
Bryan O'Sullivan
|
r5013 | |||
Alexis S. L. Carvalho
|
r5280 | def getcopies(self, ctx, files): | ||
Bryan O'Sullivan
|
r5013 | copies = {} | ||
Alexis S. L. Carvalho
|
r5280 | for name in files: | ||
Bryan O'Sullivan
|
r5013 | try: | ||
copies[name] = ctx.filectx(name).renamed()[0] | ||||
except TypeError: | ||||
pass | ||||
return copies | ||||
Thomas Arendsen Hein
|
r5143 | |||
Bryan O'Sullivan
|
r5013 | def getcommit(self, rev): | ||
ctx = self.changectx(rev) | ||||
parents = [hex(p.node()) for p in ctx.parents() if p.node() != nullid] | ||||
Bryan O'Sullivan
|
r5556 | if self.saverev: | ||
crev = rev | ||||
else: | ||||
crev = None | ||||
Bryan O'Sullivan
|
r5013 | return commit(author=ctx.user(), date=util.datestr(ctx.date()), | ||
Bryan O'Sullivan
|
r5556 | desc=ctx.description(), rev=crev, parents=parents, | ||
Bryan O'Sullivan
|
r5439 | branch=ctx.branch(), extra=ctx.extra()) | ||
Bryan O'Sullivan
|
r5013 | |||
def gettags(self): | ||||
tags = [t for t in self.repo.tagslist() if t[0] != 'tip'] | ||||
return dict([(name, hex(node)) for name, node in tags]) | ||||
Alexis S. L. Carvalho
|
r5379 | |||
def getchangedfiles(self, rev, i): | ||||
ctx = self.changectx(rev) | ||||
i = i or 0 | ||||
changes = self.repo.status(ctx.parents()[i].node(), ctx.node())[:3] | ||||
if i == 0: | ||||
self._changescache = (rev, changes) | ||||
return changes[0] + changes[1] + changes[2] | ||||
Bryan O'Sullivan
|
r5554 | def converted(self, rev, destrev): | ||
if self.convertfp is None: | ||||
self.convertfp = open(os.path.join(self.path, '.hg', 'shamap'), | ||||
'a') | ||||
self.convertfp.write('%s %s\n' % (destrev, rev)) | ||||
self.convertfp.flush() | ||||
Patrick Mezard
|
r5805 | |||
def before(self): | ||||
self.ui.debug(_('run hg source pre-conversion action\n')) | ||||
def after(self): | ||||
self.ui.debug(_('run hg source post-conversion action\n')) | ||||