|
|
# convcmd - convert extension commands definition
|
|
|
#
|
|
|
# Copyright 2005-2007 Olivia Mackall <olivia@selenic.com>
|
|
|
#
|
|
|
# This software may be used and distributed according to the terms of the
|
|
|
# GNU General Public License version 2 or any later version.
|
|
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
import collections
|
|
|
import heapq
|
|
|
import os
|
|
|
import shutil
|
|
|
import typing
|
|
|
|
|
|
from typing import (
|
|
|
AnyStr,
|
|
|
Dict,
|
|
|
List,
|
|
|
Mapping,
|
|
|
Optional,
|
|
|
Union,
|
|
|
)
|
|
|
|
|
|
from mercurial.i18n import _
|
|
|
from mercurial.pycompat import open
|
|
|
from mercurial import (
|
|
|
encoding,
|
|
|
error,
|
|
|
hg,
|
|
|
pycompat,
|
|
|
scmutil,
|
|
|
util,
|
|
|
)
|
|
|
from mercurial.utils import dateutil
|
|
|
|
|
|
from . import (
|
|
|
bzr,
|
|
|
common,
|
|
|
cvs,
|
|
|
darcs,
|
|
|
filemap,
|
|
|
git,
|
|
|
gnuarch,
|
|
|
hg as hgconvert,
|
|
|
monotone,
|
|
|
p4,
|
|
|
subversion,
|
|
|
)
|
|
|
|
|
|
if typing.TYPE_CHECKING:
|
|
|
from mercurial import (
|
|
|
ui as uimod,
|
|
|
)
|
|
|
|
|
|
mapfile = common.mapfile
|
|
|
MissingTool = common.MissingTool
|
|
|
NoRepo = common.NoRepo
|
|
|
SKIPREV = common.SKIPREV
|
|
|
|
|
|
bzr_source = bzr.bzr_source
|
|
|
convert_cvs = cvs.convert_cvs
|
|
|
convert_git = git.convert_git
|
|
|
darcs_source = darcs.darcs_source
|
|
|
gnuarch_source = gnuarch.gnuarch_source
|
|
|
mercurial_sink = hgconvert.mercurial_sink
|
|
|
mercurial_source = hgconvert.mercurial_source
|
|
|
monotone_source = monotone.monotone_source
|
|
|
p4_source = p4.p4_source
|
|
|
svn_sink = subversion.svn_sink
|
|
|
svn_source = subversion.svn_source
|
|
|
|
|
|
orig_encoding: bytes = b'ascii'
|
|
|
|
|
|
|
|
|
def readauthormap(ui: "uimod.ui", authorfile, authors=None):
|
|
|
if authors is None:
|
|
|
authors = {}
|
|
|
with open(authorfile, b'rb') as afile:
|
|
|
for line in afile:
|
|
|
line = line.strip()
|
|
|
if not line or line.startswith(b'#'):
|
|
|
continue
|
|
|
|
|
|
try:
|
|
|
srcauthor, dstauthor = line.split(b'=', 1)
|
|
|
except ValueError:
|
|
|
msg = _(b'ignoring bad line in author map file %s: %s\n')
|
|
|
ui.warn(msg % (authorfile, line.rstrip()))
|
|
|
continue
|
|
|
|
|
|
srcauthor = srcauthor.strip()
|
|
|
dstauthor = dstauthor.strip()
|
|
|
if authors.get(srcauthor) in (None, dstauthor):
|
|
|
msg = _(b'mapping author %s to %s\n')
|
|
|
ui.debug(msg % (srcauthor, dstauthor))
|
|
|
authors[srcauthor] = dstauthor
|
|
|
continue
|
|
|
|
|
|
m = _(b'overriding mapping for author %s, was %s, will be %s\n')
|
|
|
ui.status(m % (srcauthor, authors[srcauthor], dstauthor))
|
|
|
return authors
|
|
|
|
|
|
|
|
|
def recode(s: AnyStr) -> bytes:
|
|
|
if isinstance(s, str):
|
|
|
return s.encode(pycompat.sysstr(orig_encoding), 'replace')
|
|
|
else:
|
|
|
return s.decode('utf-8').encode(
|
|
|
pycompat.sysstr(orig_encoding), 'replace'
|
|
|
)
|
|
|
|
|
|
|
|
|
def mapbranch(branch: bytes, branchmap: Mapping[bytes, bytes]) -> bytes:
|
|
|
"""
|
|
|
>>> bmap = {b'default': b'branch1'}
|
|
|
>>> for i in [b'', None]:
|
|
|
... mapbranch(i, bmap)
|
|
|
'branch1'
|
|
|
'branch1'
|
|
|
>>> bmap = {b'None': b'branch2'}
|
|
|
>>> for i in [b'', None]:
|
|
|
... mapbranch(i, bmap)
|
|
|
'branch2'
|
|
|
'branch2'
|
|
|
>>> bmap = {b'None': b'branch3', b'default': b'branch4'}
|
|
|
>>> for i in [b'None', b'', None, b'default', b'branch5']:
|
|
|
... mapbranch(i, bmap)
|
|
|
'branch3'
|
|
|
'branch4'
|
|
|
'branch4'
|
|
|
'branch4'
|
|
|
'branch5'
|
|
|
"""
|
|
|
# If branch is None or empty, this commit is coming from the source
|
|
|
# repository's default branch and destined for the default branch in the
|
|
|
# destination repository. For such commits, using a literal "default"
|
|
|
# in branchmap below allows the user to map "default" to an alternate
|
|
|
# default branch in the destination repository.
|
|
|
branch = branchmap.get(branch or b'default', branch)
|
|
|
# At some point we used "None" literal to denote the default branch,
|
|
|
# attempt to use that for backward compatibility.
|
|
|
if not branch:
|
|
|
branch = branchmap.get(b'None', branch)
|
|
|
return branch
|
|
|
|
|
|
|
|
|
source_converters = [
|
|
|
(b'cvs', convert_cvs, b'branchsort'),
|
|
|
(b'git', convert_git, b'branchsort'),
|
|
|
(b'svn', svn_source, b'branchsort'),
|
|
|
(b'hg', mercurial_source, b'sourcesort'),
|
|
|
(b'darcs', darcs_source, b'branchsort'),
|
|
|
(b'mtn', monotone_source, b'branchsort'),
|
|
|
(b'gnuarch', gnuarch_source, b'branchsort'),
|
|
|
(b'bzr', bzr_source, b'branchsort'),
|
|
|
(b'p4', p4_source, b'branchsort'),
|
|
|
]
|
|
|
|
|
|
sink_converters = [
|
|
|
(b'hg', mercurial_sink),
|
|
|
(b'svn', svn_sink),
|
|
|
]
|
|
|
|
|
|
|
|
|
def convertsource(ui: "uimod.ui", path: bytes, type: bytes, revs):
|
|
|
exceptions = []
|
|
|
if type and type not in [s[0] for s in source_converters]:
|
|
|
raise error.Abort(_(b'%s: invalid source repository type') % type)
|
|
|
for name, source, sortmode in source_converters:
|
|
|
try:
|
|
|
if not type or name == type:
|
|
|
return source(ui, name, path, revs), sortmode
|
|
|
except (NoRepo, MissingTool) as inst:
|
|
|
exceptions.append(inst)
|
|
|
if not ui.quiet:
|
|
|
for inst in exceptions:
|
|
|
ui.write(b"%s\n" % pycompat.bytestr(inst.args[0]))
|
|
|
raise error.Abort(_(b'%s: missing or unsupported repository') % path)
|
|
|
|
|
|
|
|
|
def convertsink(
|
|
|
ui: "uimod.ui", path: bytes, type: bytes
|
|
|
) -> Union[hgconvert.mercurial_sink, subversion.svn_sink]:
|
|
|
if type and type not in [s[0] for s in sink_converters]:
|
|
|
raise error.Abort(_(b'%s: invalid destination repository type') % type)
|
|
|
for name, sink in sink_converters:
|
|
|
try:
|
|
|
if not type or name == type:
|
|
|
return sink(ui, name, path)
|
|
|
except NoRepo as inst:
|
|
|
ui.note(_(b"convert: %s\n") % inst)
|
|
|
except MissingTool as inst:
|
|
|
raise error.Abort(b'%s\n' % inst)
|
|
|
raise error.Abort(_(b'%s: unknown repository type') % path)
|
|
|
|
|
|
|
|
|
class progresssource:
|
|
|
def __init__(
|
|
|
self, ui: "uimod.ui", source, filecount: Optional[int]
|
|
|
) -> None:
|
|
|
self.ui = ui
|
|
|
self.source = source
|
|
|
self.progress = ui.makeprogress(
|
|
|
_(b'getting files'), unit=_(b'files'), total=filecount
|
|
|
)
|
|
|
|
|
|
def getfile(self, file, rev):
|
|
|
self.progress.increment(item=file)
|
|
|
return self.source.getfile(file, rev)
|
|
|
|
|
|
def targetfilebelongstosource(self, targetfilename):
|
|
|
return self.source.targetfilebelongstosource(targetfilename)
|
|
|
|
|
|
def lookuprev(self, rev):
|
|
|
return self.source.lookuprev(rev)
|
|
|
|
|
|
def close(self):
|
|
|
self.progress.complete()
|
|
|
|
|
|
|
|
|
# Sorters are used by the `toposort` function to maintain a set of revisions
|
|
|
# which can be converted immediately and pick one
|
|
|
class branchsorter:
|
|
|
"""If the previously converted revision has a child in the
|
|
|
eligible revisions list, pick it. Return the list head
|
|
|
otherwise. Branch sort attempts to minimize branch
|
|
|
switching, which is harmful for Mercurial backend
|
|
|
compression.
|
|
|
"""
|
|
|
|
|
|
def __init__(self, parents):
|
|
|
self.nodes = []
|
|
|
self.parents = parents
|
|
|
self.prev = None
|
|
|
|
|
|
def picknext(self):
|
|
|
next = self.nodes[0]
|
|
|
for n in self.nodes:
|
|
|
if self.prev in self.parents[n]:
|
|
|
next = n
|
|
|
break
|
|
|
self.prev = next
|
|
|
self.nodes.remove(next)
|
|
|
return next
|
|
|
|
|
|
def insert(self, node):
|
|
|
self.nodes.insert(0, node)
|
|
|
|
|
|
def __len__(self):
|
|
|
return self.nodes.__len__()
|
|
|
|
|
|
|
|
|
class keysorter:
|
|
|
"""Key-based sort, ties broken by insertion order"""
|
|
|
|
|
|
def __init__(self, keyfn):
|
|
|
self.heap = []
|
|
|
self.keyfn = keyfn
|
|
|
self.counter = 0
|
|
|
|
|
|
def picknext(self):
|
|
|
return heapq.heappop(self.heap)[2]
|
|
|
|
|
|
def insert(self, node):
|
|
|
counter = self.counter
|
|
|
self.counter = counter + 1
|
|
|
key = self.keyfn(node)
|
|
|
heapq.heappush(self.heap, (key, counter, node))
|
|
|
|
|
|
def __len__(self):
|
|
|
return self.heap.__len__()
|
|
|
|
|
|
|
|
|
class converter:
|
|
|
def __init__(self, ui: "uimod.ui", source, dest, revmapfile, opts) -> None:
|
|
|
self.source = source
|
|
|
self.dest = dest
|
|
|
self.ui = ui
|
|
|
self.opts = opts
|
|
|
self.commitcache = {}
|
|
|
self.authors = {}
|
|
|
self.authorfile = None
|
|
|
|
|
|
# Record converted revisions persistently: maps source revision
|
|
|
# ID to target revision ID (both strings). (This is how
|
|
|
# incremental conversions work.)
|
|
|
self.map = mapfile(ui, revmapfile)
|
|
|
|
|
|
# Read first the dst author map if any
|
|
|
authorfile = self.dest.authorfile()
|
|
|
if authorfile and os.path.exists(authorfile):
|
|
|
self.readauthormap(authorfile)
|
|
|
# Extend/Override with new author map if necessary
|
|
|
if opts.get(b'authormap'):
|
|
|
self.readauthormap(opts.get(b'authormap'))
|
|
|
self.authorfile = self.dest.authorfile()
|
|
|
|
|
|
self.splicemap = self.parsesplicemap(opts.get(b'splicemap'))
|
|
|
self.branchmap = mapfile(ui, opts.get(b'branchmap'))
|
|
|
|
|
|
def parsesplicemap(self, path: bytes) -> Dict[bytes, List[bytes]]:
|
|
|
"""check and validate the splicemap format and
|
|
|
return a child/parents dictionary.
|
|
|
Format checking has two parts.
|
|
|
1. generic format which is same across all source types
|
|
|
2. specific format checking which may be different for
|
|
|
different source type. This logic is implemented in
|
|
|
checkrevformat function in source files like
|
|
|
hg.py, subversion.py etc.
|
|
|
"""
|
|
|
|
|
|
if not path:
|
|
|
return {}
|
|
|
m = {}
|
|
|
try:
|
|
|
with open(path, b'rb') as fp:
|
|
|
for i, line in enumerate(fp):
|
|
|
line = line.splitlines()[0].rstrip()
|
|
|
if not line:
|
|
|
# Ignore blank lines
|
|
|
continue
|
|
|
# split line
|
|
|
lex = common.shlexer(data=line, whitespace=b',')
|
|
|
line = list(lex)
|
|
|
# check number of parents
|
|
|
if not (2 <= len(line) <= 3):
|
|
|
raise error.Abort(
|
|
|
_(
|
|
|
b'syntax error in %s(%d): child parent1'
|
|
|
b'[,parent2] expected'
|
|
|
)
|
|
|
% (path, i + 1)
|
|
|
)
|
|
|
for part in line:
|
|
|
self.source.checkrevformat(part)
|
|
|
child, p1, p2 = line[0], line[1:2], line[2:]
|
|
|
if p1 == p2:
|
|
|
m[child] = p1
|
|
|
else:
|
|
|
m[child] = p1 + p2
|
|
|
# if file does not exist or error reading, exit
|
|
|
except IOError:
|
|
|
raise error.Abort(
|
|
|
_(b'splicemap file not found or error reading %s:') % path
|
|
|
)
|
|
|
return m
|
|
|
|
|
|
def walktree(self, heads):
|
|
|
"""Return a mapping that identifies the uncommitted parents of every
|
|
|
uncommitted changeset."""
|
|
|
visit = list(heads)
|
|
|
known = set()
|
|
|
parents = {}
|
|
|
numcommits = self.source.numcommits()
|
|
|
progress = self.ui.makeprogress(
|
|
|
_(b'scanning'), unit=_(b'revisions'), total=numcommits
|
|
|
)
|
|
|
while visit:
|
|
|
n = visit.pop(0)
|
|
|
if n in known:
|
|
|
continue
|
|
|
if n in self.map:
|
|
|
m = self.map[n]
|
|
|
if m == SKIPREV or self.dest.hascommitfrommap(m):
|
|
|
continue
|
|
|
known.add(n)
|
|
|
progress.update(len(known))
|
|
|
commit = self.cachecommit(n)
|
|
|
parents[n] = []
|
|
|
for p in commit.parents:
|
|
|
parents[n].append(p)
|
|
|
visit.append(p)
|
|
|
progress.complete()
|
|
|
|
|
|
return parents
|
|
|
|
|
|
def mergesplicemap(self, parents, splicemap) -> None:
|
|
|
"""A splicemap redefines child/parent relationships. Check the
|
|
|
map contains valid revision identifiers and merge the new
|
|
|
links in the source graph.
|
|
|
"""
|
|
|
for c in sorted(splicemap):
|
|
|
if c not in parents:
|
|
|
if not self.dest.hascommitforsplicemap(self.map.get(c, c)):
|
|
|
# Could be in source but not converted during this run
|
|
|
self.ui.warn(
|
|
|
_(
|
|
|
b'splice map revision %s is not being '
|
|
|
b'converted, ignoring\n'
|
|
|
)
|
|
|
% c
|
|
|
)
|
|
|
continue
|
|
|
pc = []
|
|
|
for p in splicemap[c]:
|
|
|
# We do not have to wait for nodes already in dest.
|
|
|
if self.dest.hascommitforsplicemap(self.map.get(p, p)):
|
|
|
continue
|
|
|
# Parent is not in dest and not being converted, not good
|
|
|
if p not in parents:
|
|
|
raise error.Abort(_(b'unknown splice map parent: %s') % p)
|
|
|
pc.append(p)
|
|
|
parents[c] = pc
|
|
|
|
|
|
def toposort(self, parents, sortmode):
|
|
|
"""Return an ordering such that every uncommitted changeset is
|
|
|
preceded by all its uncommitted ancestors."""
|
|
|
|
|
|
def mapchildren(parents):
|
|
|
"""Return a (children, roots) tuple where 'children' maps parent
|
|
|
revision identifiers to children ones, and 'roots' is the list of
|
|
|
revisions without parents. 'parents' must be a mapping of revision
|
|
|
identifier to its parents ones.
|
|
|
"""
|
|
|
visit = collections.deque(sorted(parents))
|
|
|
seen = set()
|
|
|
children = {}
|
|
|
roots = []
|
|
|
|
|
|
while visit:
|
|
|
n = visit.popleft()
|
|
|
if n in seen:
|
|
|
continue
|
|
|
seen.add(n)
|
|
|
# Ensure that nodes without parents are present in the
|
|
|
# 'children' mapping.
|
|
|
children.setdefault(n, [])
|
|
|
hasparent = False
|
|
|
for p in parents[n]:
|
|
|
if p not in self.map:
|
|
|
visit.append(p)
|
|
|
hasparent = True
|
|
|
children.setdefault(p, []).append(n)
|
|
|
if not hasparent:
|
|
|
roots.append(n)
|
|
|
|
|
|
return children, roots
|
|
|
|
|
|
def makesourcesorter():
|
|
|
"""Source specific sort."""
|
|
|
keyfn = lambda n: self.commitcache[n].sortkey
|
|
|
return keysorter(keyfn)
|
|
|
|
|
|
def makeclosesorter():
|
|
|
"""Close order sort."""
|
|
|
keyfn = lambda n: (
|
|
|
b'close' not in self.commitcache[n].extra,
|
|
|
self.commitcache[n].sortkey,
|
|
|
)
|
|
|
return keysorter(keyfn)
|
|
|
|
|
|
def makedatesorter():
|
|
|
"""Sort revisions by date."""
|
|
|
|
|
|
def getdate(n):
|
|
|
commit = self.commitcache[n]
|
|
|
# The other entries are here as tie breaker for stability
|
|
|
return (
|
|
|
dateutil.parsedate(commit.date),
|
|
|
commit.rev,
|
|
|
commit.branch,
|
|
|
)
|
|
|
|
|
|
return keysorter(getdate)
|
|
|
|
|
|
if sortmode == b'branchsort':
|
|
|
sorter = branchsorter(parents)
|
|
|
elif sortmode == b'datesort':
|
|
|
sorter = makedatesorter()
|
|
|
elif sortmode == b'sourcesort':
|
|
|
sorter = makesourcesorter()
|
|
|
elif sortmode == b'closesort':
|
|
|
sorter = makeclosesorter()
|
|
|
else:
|
|
|
raise error.Abort(_(b'unknown sort mode: %s') % sortmode)
|
|
|
|
|
|
children, roots = mapchildren(parents)
|
|
|
|
|
|
for node in roots:
|
|
|
sorter.insert(node)
|
|
|
|
|
|
s = []
|
|
|
pendings = {}
|
|
|
while sorter:
|
|
|
n = sorter.picknext()
|
|
|
s.append(n)
|
|
|
|
|
|
# Update dependents list
|
|
|
for c in children.get(n, []):
|
|
|
if c not in pendings:
|
|
|
pendings[c] = [p for p in parents[c] if p not in self.map]
|
|
|
try:
|
|
|
pendings[c].remove(n)
|
|
|
except ValueError:
|
|
|
raise error.Abort(
|
|
|
_(b'cycle detected between %s and %s')
|
|
|
% (recode(c), recode(n))
|
|
|
)
|
|
|
if not pendings[c]:
|
|
|
# Parents are converted, node is eligible
|
|
|
sorter.insert(c)
|
|
|
pendings[c] = None
|
|
|
|
|
|
if len(s) != len(parents):
|
|
|
raise error.Abort(_(b"not all revisions were sorted"))
|
|
|
|
|
|
return s
|
|
|
|
|
|
def writeauthormap(self) -> None:
|
|
|
authorfile = self.authorfile
|
|
|
if authorfile:
|
|
|
self.ui.status(_(b'writing author map file %s\n') % authorfile)
|
|
|
with open(authorfile, b'wb+') as ofile:
|
|
|
for author in self.authors:
|
|
|
ofile.write(
|
|
|
util.tonativeeol(
|
|
|
b"%s=%s\n" % (author, self.authors[author])
|
|
|
)
|
|
|
)
|
|
|
|
|
|
def readauthormap(self, authorfile) -> None:
|
|
|
self.authors = readauthormap(self.ui, authorfile, self.authors)
|
|
|
|
|
|
def cachecommit(self, rev):
|
|
|
commit = self.source.getcommit(rev)
|
|
|
commit.author = self.authors.get(commit.author, commit.author)
|
|
|
commit.branch = mapbranch(commit.branch, self.branchmap)
|
|
|
self.commitcache[rev] = commit
|
|
|
return commit
|
|
|
|
|
|
def copy(self, rev) -> None:
|
|
|
commit = self.commitcache[rev]
|
|
|
full = self.opts.get(b'full')
|
|
|
changes = self.source.getchanges(rev, full)
|
|
|
if isinstance(changes, bytes):
|
|
|
if changes == SKIPREV:
|
|
|
dest = SKIPREV
|
|
|
else:
|
|
|
dest = self.map[changes]
|
|
|
self.map[rev] = dest
|
|
|
return
|
|
|
files, copies, cleanp2 = changes
|
|
|
pbranches = []
|
|
|
if commit.parents:
|
|
|
for prev in commit.parents:
|
|
|
if prev not in self.commitcache:
|
|
|
self.cachecommit(prev)
|
|
|
pbranches.append(
|
|
|
(self.map[prev], self.commitcache[prev].branch)
|
|
|
)
|
|
|
self.dest.setbranch(commit.branch, pbranches)
|
|
|
try:
|
|
|
parents = self.splicemap[rev]
|
|
|
self.ui.status(
|
|
|
_(b'spliced in %s as parents of %s\n')
|
|
|
% (_(b' and ').join(parents), rev)
|
|
|
)
|
|
|
parents = [self.map.get(p, p) for p in parents]
|
|
|
except KeyError:
|
|
|
parents = [b[0] for b in pbranches]
|
|
|
parents.extend(
|
|
|
self.map[x] for x in commit.optparents if x in self.map
|
|
|
)
|
|
|
if len(pbranches) != 2:
|
|
|
cleanp2 = set()
|
|
|
if len(parents) < 3:
|
|
|
source = progresssource(self.ui, self.source, len(files))
|
|
|
else:
|
|
|
# For an octopus merge, we end up traversing the list of
|
|
|
# changed files N-1 times. This tweak to the number of
|
|
|
# files makes it so the progress bar doesn't overflow
|
|
|
# itself.
|
|
|
source = progresssource(
|
|
|
self.ui, self.source, len(files) * (len(parents) - 1)
|
|
|
)
|
|
|
newnode = self.dest.putcommit(
|
|
|
files, copies, parents, commit, source, self.map, full, cleanp2
|
|
|
)
|
|
|
source.close()
|
|
|
self.source.converted(rev, newnode)
|
|
|
self.map[rev] = newnode
|
|
|
|
|
|
def convert(self, sortmode) -> None:
|
|
|
try:
|
|
|
self.source.before()
|
|
|
self.dest.before()
|
|
|
self.source.setrevmap(self.map)
|
|
|
self.ui.status(_(b"scanning source...\n"))
|
|
|
heads = self.source.getheads()
|
|
|
parents = self.walktree(heads)
|
|
|
self.mergesplicemap(parents, self.splicemap)
|
|
|
self.ui.status(_(b"sorting...\n"))
|
|
|
t = self.toposort(parents, sortmode)
|
|
|
num = len(t)
|
|
|
c = None
|
|
|
|
|
|
self.ui.status(_(b"converting...\n"))
|
|
|
progress = self.ui.makeprogress(
|
|
|
_(b'converting'), unit=_(b'revisions'), total=len(t)
|
|
|
)
|
|
|
for i, c in enumerate(t):
|
|
|
num -= 1
|
|
|
desc = self.commitcache[c].desc
|
|
|
if b"\n" in desc:
|
|
|
desc = desc.splitlines()[0]
|
|
|
# convert log message to local encoding without using
|
|
|
# tolocal() because the encoding.encoding convert()
|
|
|
# uses is 'utf-8'
|
|
|
self.ui.status(b"%d %s\n" % (num, recode(desc)))
|
|
|
self.ui.note(_(b"source: %s\n") % recode(c))
|
|
|
progress.update(i)
|
|
|
self.copy(c)
|
|
|
progress.complete()
|
|
|
|
|
|
if not self.ui.configbool(b'convert', b'skiptags'):
|
|
|
tags = self.source.gettags()
|
|
|
ctags = {}
|
|
|
for k in tags:
|
|
|
v = tags[k]
|
|
|
if self.map.get(v, SKIPREV) != SKIPREV:
|
|
|
ctags[k] = self.map[v]
|
|
|
|
|
|
if c and ctags:
|
|
|
nrev, tagsparent = self.dest.puttags(ctags)
|
|
|
if nrev and tagsparent:
|
|
|
# write another hash correspondence to override the
|
|
|
# previous one so we don't end up with extra tag heads
|
|
|
tagsparents = [
|
|
|
e for e in self.map.items() if e[1] == tagsparent
|
|
|
]
|
|
|
if tagsparents:
|
|
|
self.map[tagsparents[0][0]] = nrev
|
|
|
|
|
|
bookmarks = self.source.getbookmarks()
|
|
|
cbookmarks = {}
|
|
|
for k in bookmarks:
|
|
|
v = bookmarks[k]
|
|
|
if self.map.get(v, SKIPREV) != SKIPREV:
|
|
|
cbookmarks[k] = self.map[v]
|
|
|
|
|
|
if c and cbookmarks:
|
|
|
self.dest.putbookmarks(cbookmarks)
|
|
|
|
|
|
self.writeauthormap()
|
|
|
finally:
|
|
|
self.cleanup()
|
|
|
|
|
|
def cleanup(self) -> None:
|
|
|
try:
|
|
|
self.dest.after()
|
|
|
finally:
|
|
|
self.source.after()
|
|
|
self.map.close()
|
|
|
|
|
|
|
|
|
def convert(
|
|
|
ui: "uimod.ui", src, dest: Optional[bytes] = None, revmapfile=None, **opts
|
|
|
) -> None:
|
|
|
opts = pycompat.byteskwargs(opts)
|
|
|
global orig_encoding
|
|
|
orig_encoding = encoding.encoding
|
|
|
encoding.encoding = b'UTF-8'
|
|
|
|
|
|
# support --authors as an alias for --authormap
|
|
|
if not opts.get(b'authormap'):
|
|
|
opts[b'authormap'] = opts.get(b'authors')
|
|
|
|
|
|
if not dest:
|
|
|
dest = hg.defaultdest(src) + b"-hg"
|
|
|
ui.status(_(b"assuming destination %s\n") % dest)
|
|
|
|
|
|
destc = convertsink(ui, dest, opts.get(b'dest_type'))
|
|
|
destc = scmutil.wrapconvertsink(destc)
|
|
|
|
|
|
try:
|
|
|
srcc, defaultsort = convertsource(
|
|
|
ui, src, opts.get(b'source_type'), opts.get(b'rev')
|
|
|
)
|
|
|
except Exception:
|
|
|
for path in destc.created:
|
|
|
shutil.rmtree(path, True)
|
|
|
raise
|
|
|
|
|
|
sortmodes = (b'branchsort', b'datesort', b'sourcesort', b'closesort')
|
|
|
sortmode = [m for m in sortmodes if opts.get(m)]
|
|
|
if len(sortmode) > 1:
|
|
|
raise error.Abort(_(b'more than one sort mode specified'))
|
|
|
if sortmode:
|
|
|
sortmode = sortmode[0]
|
|
|
else:
|
|
|
sortmode = defaultsort
|
|
|
|
|
|
if sortmode == b'sourcesort' and not srcc.hasnativeorder():
|
|
|
raise error.Abort(
|
|
|
_(b'--sourcesort is not supported by this data source')
|
|
|
)
|
|
|
if sortmode == b'closesort' and not srcc.hasnativeclose():
|
|
|
raise error.Abort(
|
|
|
_(b'--closesort is not supported by this data source')
|
|
|
)
|
|
|
|
|
|
fmap = opts.get(b'filemap')
|
|
|
if fmap:
|
|
|
srcc = filemap.filemap_source(ui, srcc, fmap)
|
|
|
destc.setfilemapmode(True)
|
|
|
|
|
|
if not revmapfile:
|
|
|
revmapfile = destc.revmapfile()
|
|
|
|
|
|
c = converter(ui, srcc, destc, revmapfile, opts)
|
|
|
c.convert(sortmode)
|
|
|
|