bzr.py
337 lines
| 12.0 KiB
| text/x-python
|
PythonLexer
Martin Geisler
|
r8250 | # bzr.py - bzr support for the convert extension | ||
# | ||||
# Copyright 2008, 2009 Marek Kubica <marek@xivilization.net> and others | ||||
# | ||||
# 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. | ||
Martin Geisler
|
r8250 | |||
Raphaël Gomès
|
r48168 | # This module is for handling Breezy imports or `brz`, but it's also compatible | ||
# with Bazaar or `bzr`, that was formerly known as Bazaar-NG; | ||||
# it cannot access `bar` repositories, but they were never used very much. | ||||
Marek Kubica
|
r7053 | |||
import os | ||||
Yuya Nishihara
|
r29205 | |||
from mercurial.i18n import _ | ||||
Gregory Szorc
|
r43375 | from mercurial import ( | ||
demandimport, | ||||
error, | ||||
r48432 | util, | |||
Gregory Szorc
|
r43375 | ) | ||
timeless
|
r28411 | from . import common | ||
Raphaël Gomès
|
r48168 | |||
Marek Kubica
|
r7053 | # these do not work with demandimport, blacklist | ||
Augie Fackler
|
r43346 | demandimport.IGNORES.update( | ||
Augie Fackler
|
r46554 | [ | ||
Matt Harbison
|
r50445 | 'breezy.transactions', | ||
'breezy.urlutils', | ||||
'ElementPath', | ||||
Augie Fackler
|
r46554 | ] | ||
Augie Fackler
|
r43346 | ) | ||
Marek Kubica
|
r7053 | |||
try: | ||||
# bazaar imports | ||||
Raphaël Gomès
|
r48168 | import breezy.bzr.bzrdir | ||
import breezy.errors | ||||
import breezy.revision | ||||
import breezy.revisionspec | ||||
Augie Fackler
|
r43346 | |||
Raphaël Gomès
|
r48168 | bzrdir = breezy.bzr.bzrdir | ||
errors = breezy.errors | ||||
revision = breezy.revision | ||||
revisionspec = breezy.revisionspec | ||||
Saurabh Singh
|
r34489 | revisionspec.RevisionSpec | ||
Marek Kubica
|
r7053 | except ImportError: | ||
pass | ||||
Raphaël Gomès
|
r48168 | supportedkinds = ('file', 'symlink') | ||
Patrick Mezard
|
r8045 | |||
Augie Fackler
|
r43346 | |||
timeless
|
r28411 | class bzr_source(common.converter_source): | ||
Marek Kubica
|
r7053 | """Reads Bazaar repositories by using the Bazaar Python libraries""" | ||
Matt Harbison
|
r35168 | def __init__(self, ui, repotype, path, revs=None): | ||
super(bzr_source, self).__init__(ui, repotype, path, revs=revs) | ||||
Marek Kubica
|
r7053 | |||
Augie Fackler
|
r43347 | if not os.path.exists(os.path.join(path, b'.bzr')): | ||
Augie Fackler
|
r43346 | raise common.NoRepo( | ||
Augie Fackler
|
r43347 | _(b'%s does not look like a Bazaar repository') % path | ||
Augie Fackler
|
r43346 | ) | ||
Matt Mackall
|
r7973 | |||
Marek Kubica
|
r7053 | try: | ||
Raphaël Gomès
|
r48168 | # access breezy stuff | ||
Patrick Mezard
|
r16060 | bzrdir | ||
Marek Kubica
|
r7053 | except NameError: | ||
Augie Fackler
|
r43347 | raise common.NoRepo(_(b'Bazaar modules could not be loaded')) | ||
Marek Kubica
|
r7053 | |||
r48432 | path = util.abspath(path) | |||
Patrick Mezard
|
r8470 | self._checkrepotype(path) | ||
Patrick Mezard
|
r16060 | try: | ||
Raphaël Gomès
|
r48168 | bzr_dir = bzrdir.BzrDir.open(path.decode()) | ||
self.sourcerepo = bzr_dir.open_repository() | ||||
Patrick Mezard
|
r16060 | except errors.NoRepositoryPresent: | ||
Augie Fackler
|
r43346 | raise common.NoRepo( | ||
Augie Fackler
|
r43347 | _(b'%s does not look like a Bazaar repository') % path | ||
Augie Fackler
|
r43346 | ) | ||
Marek Kubica
|
r7053 | self._parentids = {} | ||
Augie Fackler
|
r43347 | self._saverev = ui.configbool(b'convert', b'bzr.saverev') | ||
Marek Kubica
|
r7053 | |||
Patrick Mezard
|
r8470 | def _checkrepotype(self, path): | ||
# Lightweight checkouts detection is informational but probably | ||||
# fragile at API level. It should not terminate the conversion. | ||||
try: | ||||
Raphaël Gomès
|
r48168 | dir = bzrdir.BzrDir.open_containing(path.decode())[0] | ||
Patrick Mezard
|
r8470 | try: | ||
tree = dir.open_workingtree(recommend_upgrade=False) | ||||
branch = tree.branch | ||||
Brodie Rao
|
r12063 | except (errors.NoWorkingTree, errors.NotLocalUrl): | ||
Patrick Mezard
|
r8470 | tree = None | ||
branch = dir.open_branch() | ||||
Augie Fackler
|
r43346 | if ( | ||
tree is not None | ||||
Raphaël Gomès
|
r48168 | and tree.controldir.root_transport.base | ||
!= branch.controldir.root_transport.base | ||||
Augie Fackler
|
r43346 | ): | ||
self.ui.warn( | ||||
_( | ||||
Augie Fackler
|
r43347 | b'warning: lightweight checkouts may cause ' | ||
b'conversion failures, try with a regular ' | ||||
b'branch instead.\n' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Brodie Rao
|
r16689 | except Exception: | ||
Augie Fackler
|
r43347 | self.ui.note(_(b'bzr source type could not be determined\n')) | ||
Patrick Mezard
|
r8470 | |||
Marek Kubica
|
r7053 | def before(self): | ||
"""Before the conversion begins, acquire a read lock | ||||
for all the operations that might need it. Fortunately | ||||
read locks don't block other reads or writes to the | ||||
repository, so this shouldn't have any impact on the usage of | ||||
the source repository. | ||||
The alternative would be locking on every operation that | ||||
needs locks (there are currently two: getting the file and | ||||
getting the parent map) and releasing immediately after, | ||||
but this approach can take even 40% longer.""" | ||||
self.sourcerepo.lock_read() | ||||
def after(self): | ||||
self.sourcerepo.unlock() | ||||
Patrick Mezard
|
r16099 | def _bzrbranches(self): | ||
return self.sourcerepo.find_branches(using=True) | ||||
Marek Kubica
|
r7053 | def getheads(self): | ||
Durham Goode
|
r25748 | if not self.revs: | ||
Patrick Mezard
|
r16099 | # Set using=True to avoid nested repositories (see issue3254) | ||
heads = sorted([b.last_revision() for b in self._bzrbranches()]) | ||||
Patrick Mezard
|
r16060 | else: | ||
revid = None | ||||
Patrick Mezard
|
r16099 | for branch in self._bzrbranches(): | ||
Patrick Mezard
|
r16060 | try: | ||
Raphaël Gomès
|
r48168 | revspec = self.revs[0].decode() | ||
r = revisionspec.RevisionSpec.from_string(revspec) | ||||
Patrick Mezard
|
r16060 | info = r.in_history(branch) | ||
except errors.BzrError: | ||||
pass | ||||
revid = info.rev_id | ||||
if revid is None: | ||||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | _(b'%s is not a valid revision') % self.revs[0] | ||
Augie Fackler
|
r43346 | ) | ||
Patrick Mezard
|
r16060 | heads = [revid] | ||
Patrick Mezard
|
r16061 | # Empty repositories return 'null:', which cannot be retrieved | ||
Augie Fackler
|
r43347 | heads = [h for h in heads if h != b'null:'] | ||
Patrick Mezard
|
r16060 | return heads | ||
Marek Kubica
|
r7053 | |||
def getfile(self, name, rev): | ||||
Raphaël Gomès
|
r48168 | name = name.decode() | ||
Marek Kubica
|
r7053 | revtree = self.sourcerepo.revision_tree(rev) | ||
Raphaël Gomès
|
r48168 | |||
try: | ||||
kind = revtree.kind(name) | ||||
except breezy.errors.NoSuchFile: | ||||
return None, None | ||||
Patrick Mezard
|
r8423 | if kind not in supportedkinds: | ||
Marek Kubica
|
r7053 | # the file is not available anymore - was deleted | ||
Mads Kiilerich
|
r22296 | return None, None | ||
Raphaël Gomès
|
r48168 | mode = self._modecache[(name.encode(), rev)] | ||
if kind == 'symlink': | ||||
target = revtree.get_symlink_target(name) | ||||
Patrick Mezard
|
r8423 | if target is None: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | _(b'%s.%s symlink has no target') % (name, rev) | ||
Augie Fackler
|
r43346 | ) | ||
Raphaël Gomès
|
r48168 | return target.encode(), mode | ||
Patrick Mezard
|
r8423 | else: | ||
Raphaël Gomès
|
r48168 | sio = revtree.get_file(name) | ||
Patrick Mezard
|
r11134 | return sio.read(), mode | ||
Marek Kubica
|
r7053 | |||
Mads Kiilerich
|
r22300 | def getchanges(self, version, full): | ||
if full: | ||||
Augie Fackler
|
r43347 | raise error.Abort(_(b"convert from cvs does not support --full")) | ||
Marek Kubica
|
r7053 | self._modecache = {} | ||
self._revtree = self.sourcerepo.revision_tree(version) | ||||
# get the parentids from the cache | ||||
parentids = self._parentids.pop(version) | ||||
# only diff against first parent id | ||||
prevtree = self.sourcerepo.revision_tree(parentids[0]) | ||||
Mads Kiilerich
|
r24395 | files, changes = self._gettreechanges(self._revtree, prevtree) | ||
return files, changes, set() | ||||
Marek Kubica
|
r7053 | |||
def getcommit(self, version): | ||||
rev = self.sourcerepo.get_revision(version) | ||||
# populate parent id cache | ||||
if not rev.parent_ids: | ||||
parents = [] | ||||
self._parentids[version] = (revision.NULL_REVISION,) | ||||
else: | ||||
parents = self._filterghosts(rev.parent_ids) | ||||
self._parentids[version] = parents | ||||
Raphaël Gomès
|
r48168 | branch = rev.properties.get('branch-nick', 'default') | ||
if branch == 'trunk': | ||||
branch = 'default' | ||||
Augie Fackler
|
r43346 | return common.commit( | ||
parents=parents, | ||||
Augie Fackler
|
r43347 | date=b'%d %d' % (rev.timestamp, -rev.timezone), | ||
Augie Fackler
|
r43346 | author=self.recode(rev.committer), | ||
desc=self.recode(rev.message), | ||||
Raphaël Gomès
|
r48168 | branch=branch.encode('utf8'), | ||
Augie Fackler
|
r43346 | rev=version, | ||
saverev=self._saverev, | ||||
) | ||||
Marek Kubica
|
r7053 | |||
def gettags(self): | ||||
bytetags = {} | ||||
Patrick Mezard
|
r16099 | for branch in self._bzrbranches(): | ||
Patrick Mezard
|
r16060 | if not branch.supports_tags(): | ||
return {} | ||||
tagdict = branch.tags.get_tag_dict() | ||||
Gregory Szorc
|
r49768 | for name, rev in tagdict.items(): | ||
Patrick Mezard
|
r16060 | bytetags[self.recode(name)] = rev | ||
Marek Kubica
|
r7053 | return bytetags | ||
def getchangedfiles(self, rev, i): | ||||
self._modecache = {} | ||||
curtree = self.sourcerepo.revision_tree(rev) | ||||
if i is not None: | ||||
Patrick Mezard
|
r8165 | parentid = self._parentids[rev][i] | ||
Marek Kubica
|
r7053 | else: | ||
# no parent id, get the empty revision | ||||
parentid = revision.NULL_REVISION | ||||
prevtree = self.sourcerepo.revision_tree(parentid) | ||||
changes = [e[0] for e in self._gettreechanges(curtree, prevtree)[0]] | ||||
return changes | ||||
def _gettreechanges(self, current, origin): | ||||
Benoit Boissinot
|
r10394 | revid = current._revision_id | ||
Marek Kubica
|
r7053 | changes = [] | ||
renames = {} | ||||
Patrick Mezard
|
r15461 | seen = set() | ||
Matt Harbison
|
r35199 | |||
# Fall back to the deprecated attribute for legacy installations. | ||||
try: | ||||
inventory = origin.root_inventory | ||||
except AttributeError: | ||||
inventory = origin.inventory | ||||
Patrick Mezard
|
r15461 | # Process the entries by reverse lexicographic name order to | ||
# handle nested renames correctly, most specific first. | ||||
Raphaël Gomès
|
r48168 | |||
def key(c): | ||||
return c.path[0] or c.path[1] or "" | ||||
Augie Fackler
|
r43346 | curchanges = sorted( | ||
current.iter_changes(origin), | ||||
Raphaël Gomès
|
r48168 | key=key, | ||
Augie Fackler
|
r43346 | reverse=True, | ||
) | ||||
Raphaël Gomès
|
r48168 | for change in curchanges: | ||
paths = change.path | ||||
kind = change.kind | ||||
executable = change.executable | ||||
Marek Kubica
|
r7053 | if paths[0] == u'' or paths[1] == u'': | ||
# ignore changes to tree root | ||||
continue | ||||
# bazaar tracks directories, mercurial does not, so | ||||
# we have to rename the directory contents | ||||
Raphaël Gomès
|
r48168 | if kind[1] == 'directory': | ||
if kind[0] not in (None, 'directory'): | ||||
Patrick Mezard
|
r8126 | # Replacing 'something' with a directory, record it | ||
# so it can be removed. | ||||
changes.append((self.recode(paths[0]), revid)) | ||||
Raphaël Gomès
|
r48168 | if kind[0] == 'directory' and None not in paths: | ||
Patrick Mezard
|
r15461 | renaming = paths[0] != paths[1] | ||
Marek Kubica
|
r7053 | # neither an add nor an delete - a move | ||
# rename all directory contents manually | ||||
Matt Harbison
|
r35199 | subdir = inventory.path2id(paths[0]) | ||
Marek Kubica
|
r7053 | # get all child-entries of the directory | ||
Matt Harbison
|
r35199 | for name, entry in inventory.iter_entries(subdir): | ||
Marek Kubica
|
r7053 | # hg does not track directory renames | ||
Raphaël Gomès
|
r48168 | if entry.kind == 'directory': | ||
Marek Kubica
|
r7053 | continue | ||
Raphaël Gomès
|
r48168 | frompath = self.recode(paths[0] + '/' + name) | ||
Patrick Mezard
|
r15461 | if frompath in seen: | ||
# Already handled by a more specific change entry | ||||
# This is important when you have: | ||||
# a => b | ||||
# a/c => a/c | ||||
# Here a/c must not be renamed into b/c | ||||
continue | ||||
seen.add(frompath) | ||||
if not renaming: | ||||
continue | ||||
Raphaël Gomès
|
r48168 | topath = self.recode(paths[1] + '/' + name) | ||
Marek Kubica
|
r7053 | # register the files as changed | ||
changes.append((frompath, revid)) | ||||
changes.append((topath, revid)) | ||||
# add to mode cache | ||||
Augie Fackler
|
r43346 | mode = ( | ||
Augie Fackler
|
r43347 | (entry.executable and b'x') | ||
Raphaël Gomès
|
r48168 | or (entry.kind == 'symlink' and b's') | ||
Augie Fackler
|
r43347 | or b'' | ||
Augie Fackler
|
r43346 | ) | ||
Marek Kubica
|
r7053 | self._modecache[(topath, revid)] = mode | ||
# register the change as move | ||||
renames[topath] = frompath | ||||
Mads Kiilerich
|
r17424 | # no further changes, go to the next change | ||
Marek Kubica
|
r7053 | continue | ||
# we got unicode paths, need to convert them | ||||
Patrick Mezard
|
r16059 | path, topath = paths | ||
if path is not None: | ||||
path = self.recode(path) | ||||
if topath is not None: | ||||
topath = self.recode(topath) | ||||
Patrick Mezard
|
r15461 | seen.add(path or topath) | ||
Marek Kubica
|
r7053 | |||
if topath is None: | ||||
# file deleted | ||||
changes.append((path, revid)) | ||||
continue | ||||
# renamed | ||||
if path and path != topath: | ||||
renames[topath] = path | ||||
Xavier ALT
|
r8035 | changes.append((path, revid)) | ||
Marek Kubica
|
r7053 | |||
# populate the mode cache | ||||
kind, executable = [e[1] for e in (kind, executable)] | ||||
Raphaël Gomès
|
r48168 | mode = (executable and b'x') or (kind == 'symlink' and b'l') or b'' | ||
Marek Kubica
|
r7053 | self._modecache[(topath, revid)] = mode | ||
changes.append((topath, revid)) | ||||
return changes, renames | ||||
def _filterghosts(self, ids): | ||||
"""Filters out ghost revisions which hg does not support, see | ||||
<http://bazaar-vcs.org/GhostRevision> | ||||
""" | ||||
parentmap = self.sourcerepo.get_parent_map(ids) | ||||
Dirkjan Ochtman
|
r7060 | parents = tuple([parent for parent in ids if parent in parentmap]) | ||
Marek Kubica
|
r7053 | return parents | ||