common.py
459 lines
| 14.5 KiB
| text/x-python
|
PythonLexer
Martin Geisler
|
r8250 | # common.py - common code for the convert extension | ||
# | ||||
# Copyright 2005-2009 Matt Mackall <mpm@selenic.com> 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 | |||
Ben Goswami
|
r19120 | import base64, errno, subprocess, os, datetime, re | ||
Patrick Mezard
|
r5127 | import cPickle as pickle | ||
Martin Geisler
|
r8656 | from mercurial import util | ||
Bryan O'Sullivan
|
r5513 | from mercurial.i18n import _ | ||
Patrick Mezard
|
r5127 | |||
Patrick Mezard
|
r15606 | propertycache = util.propertycache | ||
Patrick Mezard
|
r5127 | def encodeargs(args): | ||
def encodearg(s): | ||||
lines = base64.encodestring(s) | ||||
lines = [l.splitlines()[0] for l in lines] | ||||
return ''.join(lines) | ||||
Thomas Arendsen Hein
|
r5143 | |||
Patrick Mezard
|
r5127 | s = pickle.dumps(args) | ||
return encodearg(s) | ||||
def decodeargs(s): | ||||
s = base64.decodestring(s) | ||||
return pickle.loads(s) | ||||
Brendan Cully
|
r4536 | |||
Matt Mackall
|
r10282 | class MissingTool(Exception): | ||
pass | ||||
Patrick Mezard
|
r6332 | |||
def checktool(exe, name=None, abort=True): | ||||
Patrick Mezard
|
r5497 | name = name or exe | ||
Adrian Buehlmann
|
r14271 | if not util.findexe(exe): | ||
Patrick Mezard
|
r6332 | exc = abort and util.Abort or MissingTool | ||
raise exc(_('cannot find required "%s" tool') % name) | ||||
Patrick Mezard
|
r5497 | |||
Matt Mackall
|
r10282 | class NoRepo(Exception): | ||
pass | ||||
Brendan Cully
|
r4536 | |||
Alexis S. L. Carvalho
|
r5400 | SKIPREV = 'SKIP' | ||
Alexis S. L. Carvalho
|
r5374 | |||
Brendan Cully
|
r4536 | class commit(object): | ||
Bryan O'Sullivan
|
r5439 | def __init__(self, author, date, desc, parents, branch=None, rev=None, | ||
Patrick Mezard
|
r8690 | extra={}, sortkey=None): | ||
Alexis S. L. Carvalho
|
r5984 | self.author = author or 'unknown' | ||
self.date = date or '0 0' | ||||
Bryan O'Sullivan
|
r5024 | self.desc = desc | ||
Bryan O'Sullivan
|
r5012 | self.parents = parents | ||
self.branch = branch | ||||
self.rev = rev | ||||
Bryan O'Sullivan
|
r5439 | self.extra = extra | ||
Patrick Mezard
|
r8690 | self.sortkey = sortkey | ||
Brendan Cully
|
r4536 | |||
class converter_source(object): | ||||
"""Conversion source interface""" | ||||
Bryan O'Sullivan
|
r5556 | def __init__(self, ui, path=None, rev=None): | ||
Brendan Cully
|
r4536 | """Initialize conversion source (or raise NoRepo("message") | ||
exception if path is not a valid repository)""" | ||||
Brendan Cully
|
r4810 | self.ui = ui | ||
self.path = path | ||||
self.rev = rev | ||||
self.encoding = 'utf-8' | ||||
Brendan Cully
|
r4812 | |||
Sean Farley
|
r20373 | def checkhexformat(self, revstr, mapname='splicemap'): | ||
Ben Goswami
|
r19120 | """ fails if revstr is not a 40 byte hex. mercurial and git both uses | ||
such format for their revision numbering | ||||
""" | ||||
Ben Goswami
|
r19122 | if not re.match(r'[0-9a-fA-F]{40,40}$', revstr): | ||
Sean Farley
|
r20373 | raise util.Abort(_('%s entry %s is not a valid revision' | ||
' identifier') % (mapname, revstr)) | ||||
Ben Goswami
|
r19120 | |||
Bryan O'Sullivan
|
r5356 | def before(self): | ||
pass | ||||
def after(self): | ||||
pass | ||||
Bryan O'Sullivan
|
r5510 | def setrevmap(self, revmap): | ||
"""set the map of already-converted revisions""" | ||||
Brendan Cully
|
r4813 | pass | ||
Brendan Cully
|
r4536 | |||
def getheads(self): | ||||
"""Return a list of this repository's heads""" | ||||
Brodie Rao
|
r16687 | raise NotImplementedError | ||
Brendan Cully
|
r4536 | |||
def getfile(self, name, rev): | ||||
Patrick Mezard
|
r11134 | """Return a pair (data, mode) where data is the file content | ||
as a string and mode one of '', 'x' or 'l'. rev is the | ||||
Mads Kiilerich
|
r22296 | identifier returned by a previous call to getchanges(). | ||
Data is None if file is missing/deleted in rev. | ||||
Patrick Mezard
|
r7055 | """ | ||
Brodie Rao
|
r16687 | raise NotImplementedError | ||
Brendan Cully
|
r4536 | |||
Mads Kiilerich
|
r22300 | def getchanges(self, version, full): | ||
Dirkjan Ochtman
|
r7186 | """Returns a tuple of (files, copies). | ||
Patrick Mezard
|
r7055 | |||
files is a sorted list of (filename, id) tuples for all files | ||||
Greg Ward
|
r8444 | changed between version and its first parent returned by | ||
Mads Kiilerich
|
r22300 | getcommit(). If full, all files in that revision is returned. | ||
id is the source revision id of the file. | ||||
Brendan Cully
|
r4536 | |||
Brendan Cully
|
r5121 | copies is a dictionary of dest: source | ||
""" | ||||
Brodie Rao
|
r16687 | raise NotImplementedError | ||
Brendan Cully
|
r4536 | |||
def getcommit(self, version): | ||||
"""Return the commit object for version""" | ||||
Brodie Rao
|
r16687 | raise NotImplementedError | ||
Brendan Cully
|
r4536 | |||
Augie Fackler
|
r22411 | def numcommits(self): | ||
"""Return the number of commits in this source. | ||||
If unknown, return None. | ||||
""" | ||||
return None | ||||
Brendan Cully
|
r4536 | def gettags(self): | ||
Patrick Mezard
|
r8887 | """Return the tags as a dictionary of name: revision | ||
Tag names must be UTF-8 strings. | ||||
""" | ||||
Brodie Rao
|
r16687 | raise NotImplementedError | ||
Brendan Cully
|
r4536 | |||
Brendan Cully
|
r4759 | def recode(self, s, encoding=None): | ||
if not encoding: | ||||
Brendan Cully
|
r4810 | encoding = self.encoding or 'utf-8' | ||
Thomas Arendsen Hein
|
r4957 | |||
Thomas Arendsen Hein
|
r5287 | if isinstance(s, unicode): | ||
return s.encode("utf-8") | ||||
Brendan Cully
|
r4759 | try: | ||
return s.decode(encoding).encode("utf-8") | ||||
Brodie Rao
|
r16688 | except UnicodeError: | ||
Brendan Cully
|
r4759 | try: | ||
return s.decode("latin-1").encode("utf-8") | ||||
Brodie Rao
|
r16688 | except UnicodeError: | ||
Brendan Cully
|
r4759 | return s.decode(encoding, "replace").encode("utf-8") | ||
Alexis S. L. Carvalho
|
r5377 | def getchangedfiles(self, rev, i): | ||
"""Return the files changed by rev compared to parent[i]. | ||||
Thomas Arendsen Hein
|
r5760 | |||
Alexis S. L. Carvalho
|
r5377 | i is an index selecting one of the parents of rev. The return | ||
value should be the list of files that are different in rev and | ||||
this parent. | ||||
If rev has no parents, i is None. | ||||
Thomas Arendsen Hein
|
r5760 | |||
Alexis S. L. Carvalho
|
r5377 | This function is only needed to support --filemap | ||
""" | ||||
Brodie Rao
|
r16687 | raise NotImplementedError | ||
Alexis S. L. Carvalho
|
r5377 | |||
Bryan O'Sullivan
|
r5554 | def converted(self, rev, sinkrev): | ||
'''Notify the source that a revision has been converted.''' | ||||
pass | ||||
Patrick Mezard
|
r8691 | def hasnativeorder(self): | ||
"""Return true if this source has a meaningful, native revision | ||||
order. For instance, Mercurial revisions are store sequentially | ||||
while there is no such global ordering with Darcs. | ||||
""" | ||||
return False | ||||
Constantine Linnick
|
r18819 | def hasnativeclose(self): | ||
"""Return true if this source has ability to close branch. | ||||
""" | ||||
return False | ||||
Patrick Mezard
|
r8693 | def lookuprev(self, rev): | ||
"""If rev is a meaningful revision reference in source, return | ||||
the referenced identifier in the same format used by getcommit(). | ||||
return None otherwise. | ||||
""" | ||||
return None | ||||
Bryan O'Sullivan
|
r5554 | |||
Edouard Gomez
|
r13744 | def getbookmarks(self): | ||
"""Return the bookmarks as a dictionary of name: revision | ||||
Bookmark names are to be UTF-8 strings. | ||||
""" | ||||
return {} | ||||
Sean Farley
|
r20373 | def checkrevformat(self, revstr, mapname='splicemap'): | ||
Ben Goswami
|
r19120 | """revstr is a string that describes a revision in the given | ||
source control system. Return true if revstr has correct | ||||
format. | ||||
""" | ||||
return True | ||||
Brendan Cully
|
r4536 | class converter_sink(object): | ||
"""Conversion sink (target) interface""" | ||||
def __init__(self, ui, path): | ||||
"""Initialize conversion sink (or raise NoRepo("message") | ||||
Bryan O'Sullivan
|
r5441 | exception if path is not a valid repository) | ||
created is a list of paths to remove if a fatal error occurs | ||||
later""" | ||||
self.ui = ui | ||||
Bryan O'Sullivan
|
r5440 | self.path = path | ||
Bryan O'Sullivan
|
r5441 | self.created = [] | ||
Brendan Cully
|
r4536 | |||
Bryan O'Sullivan
|
r5011 | def revmapfile(self): | ||
Brendan Cully
|
r4536 | """Path to a file that will contain lines | ||
source_rev_id sink_rev_id | ||||
mapping equivalent revision identifiers for each system.""" | ||||
Brodie Rao
|
r16687 | raise NotImplementedError | ||
Brendan Cully
|
r4536 | |||
Edouard Gomez
|
r4589 | def authorfile(self): | ||
"""Path to a file that will contain lines | ||||
srcauthor=dstauthor | ||||
mapping equivalent authors identifiers for each system.""" | ||||
Brendan Cully
|
r4590 | return None | ||
Edouard Gomez
|
r4589 | |||
Mads Kiilerich
|
r22300 | def putcommit(self, files, copies, parents, commit, source, revmap, full): | ||
Brendan Cully
|
r4536 | """Create a revision with all changed files listed in 'files' | ||
Patrick Mezard
|
r8693 | and having listed parents. 'commit' is a commit object | ||
containing at a minimum the author, date, and message for this | ||||
changeset. 'files' is a list of (path, version) tuples, | ||||
'copies' is a dictionary mapping destinations to sources, | ||||
'source' is the source repository, and 'revmap' is a mapfile | ||||
Patrick Mezard
|
r11134 | of source revisions to converted revisions. Only getfile() and | ||
Mads Kiilerich
|
r22300 | lookuprev() should be called on 'source'. 'full' means that 'files' | ||
is complete and all other files should be removed. | ||||
Patrick Mezard
|
r6716 | |||
Note that the sink repository is not told to update itself to | ||||
a particular revision (or even what that revision would be) | ||||
before it receives the file data. | ||||
""" | ||||
Brodie Rao
|
r16687 | raise NotImplementedError | ||
Brendan Cully
|
r4536 | |||
def puttags(self, tags): | ||||
"""Put tags into sink. | ||||
Patrick Mezard
|
r8887 | |||
tags: {tagname: sink_rev_id, ...} where tagname is an UTF-8 string. | ||||
Patrick Mezard
|
r9431 | Return a pair (tag_revision, tag_parent_revision), or (None, None) | ||
if nothing was changed. | ||||
Patrick Mezard
|
r8887 | """ | ||
Brodie Rao
|
r16687 | raise NotImplementedError | ||
Patrick Mezard
|
r5127 | |||
Patrick Mezard
|
r5934 | def setbranch(self, branch, pbranches): | ||
Patrick Mezard
|
r6716 | """Set the current branch name. Called before the first putcommit | ||
Brendan Cully
|
r5173 | on the branch. | ||
branch: branch name for subsequent commits | ||||
Patrick Mezard
|
r5934 | pbranches: (converted parent revision, parent branch) tuples""" | ||
Brendan Cully
|
r5173 | pass | ||
Alexis S. L. Carvalho
|
r5378 | |||
def setfilemapmode(self, active): | ||||
"""Tell the destination that we're using a filemap | ||||
Some converter_sources (svn in particular) can claim that a file | ||||
was changed in a revision, even if there was no change. This method | ||||
tells the destination that we're using a filemap and that it should | ||||
filter empty revisions. | ||||
""" | ||||
pass | ||||
Bryan O'Sullivan
|
r5510 | |||
Bryan O'Sullivan
|
r5512 | def before(self): | ||
pass | ||||
def after(self): | ||||
pass | ||||
Edouard Gomez
|
r13744 | def putbookmarks(self, bookmarks): | ||
"""Put bookmarks into sink. | ||||
bookmarks: {bookmarkname: sink_rev_id, ...} | ||||
where bookmarkname is an UTF-8 string. | ||||
""" | ||||
pass | ||||
Bryan O'Sullivan
|
r5512 | |||
Mads Kiilerich
|
r21635 | def hascommitfrommap(self, rev): | ||
"""Return False if a rev mentioned in a filemap is known to not be | ||||
present.""" | ||||
raise NotImplementedError | ||||
Mads Kiilerich
|
r21634 | def hascommitforsplicemap(self, rev): | ||
"""This method is for the special needs for splicemap handling and not | ||||
for general use. Returns True if the sink contains rev, aborts on some | ||||
special cases.""" | ||||
Brodie Rao
|
r16687 | raise NotImplementedError | ||
Patrick Mezard
|
r16106 | |||
Bryan O'Sullivan
|
r5512 | class commandline(object): | ||
def __init__(self, ui, command): | ||||
self.ui = ui | ||||
self.command = command | ||||
def prerun(self): | ||||
pass | ||||
def postrun(self): | ||||
pass | ||||
Patrick Mezard
|
r17413 | def _cmdline(self, cmd, *args, **kwargs): | ||
Bryan O'Sullivan
|
r5512 | cmdline = [self.command, cmd] + list(args) | ||
for k, v in kwargs.iteritems(): | ||||
if len(k) == 1: | ||||
cmdline.append('-' + k) | ||||
else: | ||||
cmdline.append('--' + k.replace('_', '-')) | ||||
try: | ||||
if len(k) == 1: | ||||
cmdline.append('' + v) | ||||
else: | ||||
cmdline[-1] += '=' + v | ||||
except TypeError: | ||||
pass | ||||
cmdline = [util.shellquote(arg) for arg in cmdline] | ||||
Patrick Mezard
|
r7611 | if not self.ui.debugflag: | ||
Ross Lagerwall
|
r17391 | cmdline += ['2>', os.devnull] | ||
Patrick Mezard
|
r5529 | cmdline = ' '.join(cmdline) | ||
Maxim Dounin
|
r5832 | return cmdline | ||
Bryan O'Sullivan
|
r5512 | |||
Maxim Dounin
|
r5832 | def _run(self, cmd, *args, **kwargs): | ||
Patrick Mezard
|
r17413 | def popen(cmdline): | ||
p = subprocess.Popen(cmdline, shell=True, bufsize=-1, | ||||
close_fds=util.closefds, | ||||
stdout=subprocess.PIPE) | ||||
return p | ||||
return self._dorun(popen, cmd, *args, **kwargs) | ||||
Daniel Atallah
|
r13759 | |||
def _run2(self, cmd, *args, **kwargs): | ||||
Patrick Mezard
|
r17413 | return self._dorun(util.popen2, cmd, *args, **kwargs) | ||
Daniel Atallah
|
r13759 | |||
Patrick Mezard
|
r17413 | def _dorun(self, openfunc, cmd, *args, **kwargs): | ||
cmdline = self._cmdline(cmd, *args, **kwargs) | ||||
Martin Geisler
|
r9467 | self.ui.debug('running: %s\n' % (cmdline,)) | ||
Bryan O'Sullivan
|
r5512 | self.prerun() | ||
try: | ||||
Daniel Atallah
|
r13759 | return openfunc(cmdline) | ||
Bryan O'Sullivan
|
r5512 | finally: | ||
self.postrun() | ||||
def run(self, cmd, *args, **kwargs): | ||||
Patrick Mezard
|
r17413 | p = self._run(cmd, *args, **kwargs) | ||
output = p.communicate()[0] | ||||
Bryan O'Sullivan
|
r5512 | self.ui.debug(output) | ||
Patrick Mezard
|
r17413 | return output, p.returncode | ||
Bryan O'Sullivan
|
r5512 | |||
Aleix Conchillo Flaque
|
r6035 | def runlines(self, cmd, *args, **kwargs): | ||
Patrick Mezard
|
r17413 | p = self._run(cmd, *args, **kwargs) | ||
output = p.stdout.readlines() | ||||
p.wait() | ||||
Aleix Conchillo Flaque
|
r6049 | self.ui.debug(''.join(output)) | ||
Patrick Mezard
|
r17413 | return output, p.returncode | ||
Aleix Conchillo Flaque
|
r6035 | |||
Bryan O'Sullivan
|
r5512 | def checkexit(self, status, output=''): | ||
if status: | ||||
if output: | ||||
self.ui.warn(_('%s error:\n') % self.command) | ||||
self.ui.warn(output) | ||||
Adrian Buehlmann
|
r14234 | msg = util.explainexit(status)[0] | ||
Martin Geisler
|
r8970 | raise util.Abort('%s %s' % (self.command, msg)) | ||
Bryan O'Sullivan
|
r5512 | |||
def run0(self, cmd, *args, **kwargs): | ||||
output, status = self.run(cmd, *args, **kwargs) | ||||
self.checkexit(status, output) | ||||
return output | ||||
Aleix Conchillo Flaque
|
r6035 | def runlines0(self, cmd, *args, **kwargs): | ||
output, status = self.runlines(cmd, *args, **kwargs) | ||||
Aleix Conchillo Flaque
|
r6049 | self.checkexit(status, ''.join(output)) | ||
Aleix Conchillo Flaque
|
r6035 | return output | ||
Patrick Mezard
|
r15606 | @propertycache | ||
def argmax(self): | ||||
Maxim Dounin
|
r5832 | # POSIX requires at least 4096 bytes for ARG_MAX | ||
Patrick Mezard
|
r15606 | argmax = 4096 | ||
Maxim Dounin
|
r5832 | try: | ||
Patrick Mezard
|
r15606 | argmax = os.sysconf("SC_ARG_MAX") | ||
Brodie Rao
|
r16688 | except (AttributeError, ValueError): | ||
Maxim Dounin
|
r5832 | pass | ||
# Windows shells impose their own limits on command line length, | ||||
# down to 2047 bytes for cmd.exe under Windows NT/2k and 2500 bytes | ||||
# for older 4nt.exe. See http://support.microsoft.com/kb/830473 for | ||||
# details about cmd.exe limitations. | ||||
# Since ARG_MAX is for command line _and_ environment, lower our limit | ||||
# (and make happy Windows shells while doing this). | ||||
Martin Geisler
|
r15791 | return argmax // 2 - 1 | ||
Maxim Dounin
|
r5832 | |||
Patrick Mezard
|
r17412 | def _limit_arglist(self, arglist, cmd, *args, **kwargs): | ||
cmdlen = len(self._cmdline(cmd, *args, **kwargs)) | ||||
Patrick Mezard
|
r15606 | limit = self.argmax - cmdlen | ||
Maxim Dounin
|
r5832 | bytes = 0 | ||
fl = [] | ||||
for fn in arglist: | ||||
b = len(fn) + 3 | ||||
if bytes + b < limit or len(fl) == 0: | ||||
fl.append(fn) | ||||
bytes += b | ||||
else: | ||||
yield fl | ||||
fl = [fn] | ||||
bytes = b | ||||
if fl: | ||||
yield fl | ||||
def xargs(self, arglist, cmd, *args, **kwargs): | ||||
Patrick Mezard
|
r17412 | for l in self._limit_arglist(arglist, cmd, *args, **kwargs): | ||
Maxim Dounin
|
r5832 | self.run0(cmd, *(list(args) + l), **kwargs) | ||
Bryan O'Sullivan
|
r5510 | |||
class mapfile(dict): | ||||
def __init__(self, ui, path): | ||||
super(mapfile, self).__init__() | ||||
self.ui = ui | ||||
self.path = path | ||||
self.fp = None | ||||
self.order = [] | ||||
self._read() | ||||
def _read(self): | ||||
Stefan Rusek
|
r7774 | if not self.path: | ||
Bryan O'Sullivan
|
r5996 | return | ||
Bryan O'Sullivan
|
r5510 | try: | ||
fp = open(self.path, 'r') | ||||
except IOError, err: | ||||
if err.errno != errno.ENOENT: | ||||
raise | ||||
return | ||||
Patrick Mezard
|
r8047 | for i, line in enumerate(fp): | ||
Patrick Mezard
|
r16190 | line = line.splitlines()[0].rstrip() | ||
if not line: | ||||
# Ignore blank lines | ||||
continue | ||||
Patrick Mezard
|
r8047 | try: | ||
Patrick Mezard
|
r16190 | key, value = line.rsplit(' ', 1) | ||
Patrick Mezard
|
r8047 | except ValueError: | ||
Matt Mackall
|
r10282 | raise util.Abort( | ||
_('syntax error in %s(%d): key/value pair expected') | ||||
% (self.path, i + 1)) | ||||
Bryan O'Sullivan
|
r5510 | if key not in self: | ||
self.order.append(key) | ||||
super(mapfile, self).__setitem__(key, value) | ||||
fp.close() | ||||
Thomas Arendsen Hein
|
r5760 | |||
Bryan O'Sullivan
|
r5510 | def __setitem__(self, key, value): | ||
if self.fp is None: | ||||
try: | ||||
self.fp = open(self.path, 'a') | ||||
except IOError, err: | ||||
raise util.Abort(_('could not open map file %r: %s') % | ||||
(self.path, err.strerror)) | ||||
self.fp.write('%s %s\n' % (key, value)) | ||||
self.fp.flush() | ||||
super(mapfile, self).__setitem__(key, value) | ||||
def close(self): | ||||
Bryan O'Sullivan
|
r5512 | if self.fp: | ||
self.fp.close() | ||||
self.fp = None | ||||
Patrick Mezard
|
r16105 | |||
Julian Cowley
|
r17974 | def makedatetimestamp(t): | ||
"""Like util.makedate() but for time t instead of current time""" | ||||
delta = (datetime.datetime.utcfromtimestamp(t) - | ||||
datetime.datetime.fromtimestamp(t)) | ||||
tz = delta.days * 86400 + delta.seconds | ||||
return t, tz | ||||