/plugins/hg4idea/testData/bin/hgext/convert/darcs.py

https://bitbucket.org/nbargnesi/idea · Python · 167 lines · 134 code · 22 blank · 11 comment · 31 complexity · be1fcf73a0934bd38b38263c74a9722d MD5 · raw file

  1. # darcs.py - darcs support for the convert extension
  2. #
  3. # Copyright 2007-2009 Matt Mackall <mpm@selenic.com> and others
  4. #
  5. # This software may be used and distributed according to the terms of the
  6. # GNU General Public License version 2 or any later version.
  7. from common import NoRepo, checktool, commandline, commit, converter_source
  8. from mercurial.i18n import _
  9. from mercurial import util
  10. import os, shutil, tempfile
  11. # The naming drift of ElementTree is fun!
  12. try:
  13. from xml.etree.cElementTree import ElementTree
  14. except ImportError:
  15. try:
  16. from xml.etree.ElementTree import ElementTree
  17. except ImportError:
  18. try:
  19. from elementtree.cElementTree import ElementTree
  20. except ImportError:
  21. try:
  22. from elementtree.ElementTree import ElementTree
  23. except ImportError:
  24. ElementTree = None
  25. class darcs_source(converter_source, commandline):
  26. def __init__(self, ui, path, rev=None):
  27. converter_source.__init__(self, ui, path, rev=rev)
  28. commandline.__init__(self, ui, 'darcs')
  29. # check for _darcs, ElementTree, _darcs/inventory so that we can
  30. # easily skip test-convert-darcs if ElementTree is not around
  31. if not os.path.exists(os.path.join(path, '_darcs', 'inventories')):
  32. raise NoRepo(_("%s does not look like a darcs repository") % path)
  33. if not os.path.exists(os.path.join(path, '_darcs')):
  34. raise NoRepo(_("%s does not look like a darcs repository") % path)
  35. checktool('darcs')
  36. version = self.run0('--version').splitlines()[0].strip()
  37. if version < '2.1':
  38. raise util.Abort(_('darcs version 2.1 or newer needed (found %r)') %
  39. version)
  40. if ElementTree is None:
  41. raise util.Abort(_("Python ElementTree module is not available"))
  42. self.path = os.path.realpath(path)
  43. self.lastrev = None
  44. self.changes = {}
  45. self.parents = {}
  46. self.tags = {}
  47. def before(self):
  48. self.tmppath = tempfile.mkdtemp(
  49. prefix='convert-' + os.path.basename(self.path) + '-')
  50. output, status = self.run('init', repodir=self.tmppath)
  51. self.checkexit(status)
  52. tree = self.xml('changes', xml_output=True, summary=True,
  53. repodir=self.path)
  54. tagname = None
  55. child = None
  56. for elt in tree.findall('patch'):
  57. node = elt.get('hash')
  58. name = elt.findtext('name', '')
  59. if name.startswith('TAG '):
  60. tagname = name[4:].strip()
  61. elif tagname is not None:
  62. self.tags[tagname] = node
  63. tagname = None
  64. self.changes[node] = elt
  65. self.parents[child] = [node]
  66. child = node
  67. self.parents[child] = []
  68. def after(self):
  69. self.ui.debug('cleaning up %s\n' % self.tmppath)
  70. shutil.rmtree(self.tmppath, ignore_errors=True)
  71. def xml(self, cmd, **kwargs):
  72. etree = ElementTree()
  73. fp = self._run(cmd, **kwargs)
  74. etree.parse(fp)
  75. self.checkexit(fp.close())
  76. return etree.getroot()
  77. def manifest(self):
  78. man = []
  79. output, status = self.run('show', 'files', no_directories=True,
  80. repodir=self.tmppath)
  81. self.checkexit(status)
  82. for line in output.split('\n'):
  83. path = line[2:]
  84. if path:
  85. man.append(path)
  86. return man
  87. def getheads(self):
  88. return self.parents[None]
  89. def getcommit(self, rev):
  90. elt = self.changes[rev]
  91. date = util.strdate(elt.get('local_date'), '%a %b %d %H:%M:%S %Z %Y')
  92. desc = elt.findtext('name') + '\n' + elt.findtext('comment', '')
  93. return commit(author=elt.get('author'), date=util.datestr(date),
  94. desc=desc.strip(), parents=self.parents[rev])
  95. def pull(self, rev):
  96. output, status = self.run('pull', self.path, all=True,
  97. match='hash %s' % rev,
  98. no_test=True, no_posthook=True,
  99. external_merge='/bin/false',
  100. repodir=self.tmppath)
  101. if status:
  102. if output.find('We have conflicts in') == -1:
  103. self.checkexit(status, output)
  104. output, status = self.run('revert', all=True, repodir=self.tmppath)
  105. self.checkexit(status, output)
  106. def getchanges(self, rev):
  107. copies = {}
  108. changes = []
  109. man = None
  110. for elt in self.changes[rev].find('summary').getchildren():
  111. if elt.tag in ('add_directory', 'remove_directory'):
  112. continue
  113. if elt.tag == 'move':
  114. if man is None:
  115. man = self.manifest()
  116. source, dest = elt.get('from'), elt.get('to')
  117. if source in man:
  118. # File move
  119. changes.append((source, rev))
  120. changes.append((dest, rev))
  121. copies[dest] = source
  122. else:
  123. # Directory move, deduce file moves from manifest
  124. source = source + '/'
  125. for f in man:
  126. if not f.startswith(source):
  127. continue
  128. fdest = dest + '/' + f[len(source):]
  129. changes.append((f, rev))
  130. changes.append((fdest, rev))
  131. copies[fdest] = f
  132. else:
  133. changes.append((elt.text.strip(), rev))
  134. self.pull(rev)
  135. self.lastrev = rev
  136. return sorted(changes), copies
  137. def getfile(self, name, rev):
  138. if rev != self.lastrev:
  139. raise util.Abort(_('internal calling inconsistency'))
  140. return open(os.path.join(self.tmppath, name), 'rb').read()
  141. def getmode(self, name, rev):
  142. mode = os.lstat(os.path.join(self.tmppath, name)).st_mode
  143. return (mode & 0111) and 'x' or ''
  144. def gettags(self):
  145. return self.tags