subrepo.py
580 lines
| 20.1 KiB
| text/x-python
|
PythonLexer
/ mercurial / subrepo.py
Matt Mackall
|
r8812 | # subrepo.py - sub-repository handling for Mercurial | ||
# | ||||
David Soria Parra
|
r10324 | # Copyright 2009-2010 Matt Mackall <mpm@selenic.com> | ||
Matt Mackall
|
r8812 | # | ||
# 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. | ||
Matt Mackall
|
r8812 | |||
Edouard Gomez
|
r11109 | import errno, os, re, xml.dom.minidom, shutil, urlparse, posixpath | ||
Matt Mackall
|
r8814 | from i18n import _ | ||
Martin Geisler
|
r12167 | import config, util, node, error, cmdutil | ||
Abderrahim Kitouni
|
r9092 | hg = None | ||
Matt Mackall
|
r8813 | |||
Augie Fackler
|
r10177 | nullstate = ('', '', 'empty') | ||
Matt Mackall
|
r8812 | |||
Martin Geisler
|
r11775 | def state(ctx, ui): | ||
Mads Kiilerich
|
r11571 | """return a state dict, mapping subrepo paths configured in .hgsub | ||
to tuple: (source from .hgsub, revision from .hgsubstate, kind | ||||
(key in types dict)) | ||||
""" | ||||
Matt Mackall
|
r8812 | p = config.config() | ||
def read(f, sections=None, remap=None): | ||||
if f in ctx: | ||||
Matt Mackall
|
r10174 | p.parse(f, ctx[f].data(), sections, remap, read) | ||
else: | ||||
raise util.Abort(_("subrepo spec file %s not found") % f) | ||||
if '.hgsub' in ctx: | ||||
read('.hgsub') | ||||
Matt Mackall
|
r8812 | |||
Martin Geisler
|
r11775 | for path, src in ui.configitems('subpaths'): | ||
p.set('subpaths', path, src, ui.configsource('subpaths', path)) | ||||
Matt Mackall
|
r8812 | rev = {} | ||
if '.hgsubstate' in ctx: | ||||
try: | ||||
for l in ctx['.hgsubstate'].data().splitlines(): | ||||
Matt Mackall
|
r9752 | revision, path = l.split(" ", 1) | ||
Matt Mackall
|
r8812 | rev[path] = revision | ||
except IOError, err: | ||||
if err.errno != errno.ENOENT: | ||||
raise | ||||
state = {} | ||||
for path, src in p[''].items(): | ||||
Augie Fackler
|
r10177 | kind = 'hg' | ||
if src.startswith('['): | ||||
if ']' not in src: | ||||
raise util.Abort(_('missing ] in subrepo source')) | ||||
kind, src = src.split(']', 1) | ||||
kind = kind[1:] | ||||
Martin Geisler
|
r11775 | |||
for pattern, repl in p.items('subpaths'): | ||||
Martin Geisler
|
r11961 | # Turn r'C:\foo\bar' into r'C:\\foo\\bar' since re.sub | ||
# does a string decode. | ||||
repl = repl.encode('string-escape') | ||||
# However, we still want to allow back references to go | ||||
# through unharmed, so we turn r'\\1' into r'\1'. Again, | ||||
# extra escapes are needed because re.sub string decodes. | ||||
repl = re.sub(r'\\\\([0-9]+)', r'\\\1', repl) | ||||
Martin Geisler
|
r11775 | try: | ||
src = re.sub(pattern, repl, src, 1) | ||||
except re.error, e: | ||||
raise util.Abort(_("bad subrepository pattern in %s: %s") | ||||
% (p.source('subpaths', pattern), e)) | ||||
David Soria Parra
|
r10457 | state[path] = (src.strip(), rev.get(path, ''), kind) | ||
Matt Mackall
|
r8812 | |||
return state | ||||
Matt Mackall
|
r8813 | |||
def writestate(repo, state): | ||||
Mads Kiilerich
|
r11571 | """rewrite .hgsubstate in (outer) repo with these subrepo states""" | ||
Matt Mackall
|
r8813 | repo.wwrite('.hgsubstate', | ||
''.join(['%s %s\n' % (state[s][1], s) | ||||
for s in sorted(state)]), '') | ||||
Matt Mackall
|
r8814 | def submerge(repo, wctx, mctx, actx): | ||
Mads Kiilerich
|
r11571 | """delegated from merge.applyupdates: merging of .hgsubstate file | ||
in working context, merging context and ancestor context""" | ||||
Matt Mackall
|
r8814 | if mctx == actx: # backwards? | ||
actx = wctx.p1() | ||||
s1 = wctx.substate | ||||
s2 = mctx.substate | ||||
sa = actx.substate | ||||
sm = {} | ||||
Matt Mackall
|
r9782 | repo.ui.debug("subrepo merge %s %s %s\n" % (wctx, mctx, actx)) | ||
Matt Mackall
|
r9779 | def debug(s, msg, r=""): | ||
if r: | ||||
Augie Fackler
|
r10177 | r = "%s:%s:%s" % r | ||
Matt Mackall
|
r9782 | repo.ui.debug(" subrepo %s: %s %s\n" % (s, msg, r)) | ||
Matt Mackall
|
r9779 | |||
Matt Mackall
|
r8814 | for s, l in s1.items(): | ||
Matt Mackall
|
r11470 | a = sa.get(s, nullstate) | ||
Matt Mackall
|
r11463 | ld = l # local state with possible dirty flag for compares | ||
Matt Mackall
|
r11470 | if wctx.sub(s).dirty(): | ||
Matt Mackall
|
r11463 | ld = (l[0], l[1] + "+") | ||
Matt Mackall
|
r11470 | if wctx == actx: # overwrite | ||
a = ld | ||||
Matt Mackall
|
r11463 | |||
Matt Mackall
|
r8814 | if s in s2: | ||
r = s2[s] | ||||
Matt Mackall
|
r11463 | if ld == r or r == a: # no change or local is newer | ||
Matt Mackall
|
r8814 | sm[s] = l | ||
continue | ||||
Matt Mackall
|
r11463 | elif ld == a: # other side changed | ||
Matt Mackall
|
r9782 | debug(s, "other changed, get", r) | ||
Matt Mackall
|
r8814 | wctx.sub(s).get(r) | ||
sm[s] = r | ||||
Matt Mackall
|
r11463 | elif ld[0] != r[0]: # sources differ | ||
Simon Heimberg
|
r9048 | if repo.ui.promptchoice( | ||
Matt Mackall
|
r8814 | _(' subrepository sources for %s differ\n' | ||
Dongsheng Song
|
r8908 | 'use (l)ocal source (%s) or (r)emote source (%s)?') | ||
Matt Mackall
|
r8814 | % (s, l[0], r[0]), | ||
Simon Heimberg
|
r9048 | (_('&Local'), _('&Remote')), 0): | ||
Matt Mackall
|
r9782 | debug(s, "prompt changed, get", r) | ||
Matt Mackall
|
r8814 | wctx.sub(s).get(r) | ||
sm[s] = r | ||||
Matt Mackall
|
r11463 | elif ld[1] == a[1]: # local side is unchanged | ||
Matt Mackall
|
r9782 | debug(s, "other side changed, get", r) | ||
Matt Mackall
|
r8814 | wctx.sub(s).get(r) | ||
sm[s] = r | ||||
else: | ||||
Matt Mackall
|
r9782 | debug(s, "both sides changed, merge with", r) | ||
Matt Mackall
|
r8814 | wctx.sub(s).merge(r) | ||
sm[s] = l | ||||
Matt Mackall
|
r11463 | elif ld == a: # remote removed, local unchanged | ||
Matt Mackall
|
r9782 | debug(s, "remote removed, remove") | ||
Matt Mackall
|
r8814 | wctx.sub(s).remove() | ||
else: | ||||
Simon Heimberg
|
r9048 | if repo.ui.promptchoice( | ||
Matt Mackall
|
r8814 | _(' local changed subrepository %s which remote removed\n' | ||
Dongsheng Song
|
r8908 | 'use (c)hanged version or (d)elete?') % s, | ||
Martin Geisler
|
r9049 | (_('&Changed'), _('&Delete')), 0): | ||
Matt Mackall
|
r9782 | debug(s, "prompt remove") | ||
Matt Mackall
|
r8814 | wctx.sub(s).remove() | ||
for s, r in s2.items(): | ||||
if s in s1: | ||||
continue | ||||
elif s not in sa: | ||||
Matt Mackall
|
r9782 | debug(s, "remote added, get", r) | ||
Augie Fackler
|
r10175 | mctx.sub(s).get(r) | ||
Matt Mackall
|
r8814 | sm[s] = r | ||
elif r != sa[s]: | ||||
Simon Heimberg
|
r9048 | if repo.ui.promptchoice( | ||
Matt Mackall
|
r8814 | _(' remote changed subrepository %s which local removed\n' | ||
Dongsheng Song
|
r8908 | 'use (c)hanged version or (d)elete?') % s, | ||
Martin Geisler
|
r9049 | (_('&Changed'), _('&Delete')), 0) == 0: | ||
Matt Mackall
|
r9782 | debug(s, "prompt recreate", r) | ||
Matt Mackall
|
r8814 | wctx.sub(s).get(r) | ||
sm[s] = r | ||||
# record merged .hgsubstate | ||||
writestate(repo, sm) | ||||
Mads Kiilerich
|
r12752 | def reporelpath(repo): | ||
"""return path to this (sub)repo as seen from outermost repo""" | ||||
parent = repo | ||||
while hasattr(parent, '_subparent'): | ||||
parent = parent._subparent | ||||
return repo.root[len(parent.root)+1:] | ||||
def subrelpath(sub): | ||||
Mads Kiilerich
|
r11571 | """return path to this subrepo as seen from outermost repo""" | ||
Edouard Gomez
|
r11112 | if not hasattr(sub, '_repo'): | ||
return sub._path | ||||
Mads Kiilerich
|
r12752 | return reporelpath(sub._repo) | ||
Edouard Gomez
|
r11112 | |||
Mads Kiilerich
|
r12753 | def _abssource(repo, push=False, abort=True): | ||
"""return pull/push path of repo - either based on parent repo .hgsub info | ||||
or on the top repo config. Abort or return None if no source found.""" | ||||
Matt Mackall
|
r8814 | if hasattr(repo, '_subparent'): | ||
source = repo._subsource | ||||
if source.startswith('/') or '://' in source: | ||||
return source | ||||
Mads Kiilerich
|
r12753 | parent = _abssource(repo._subparent, push, abort=False) | ||
if parent: | ||||
if '://' in parent: | ||||
if parent[-1] == '/': | ||||
parent = parent[:-1] | ||||
r = urlparse.urlparse(parent + '/' + source) | ||||
r = urlparse.urlunparse((r[0], r[1], | ||||
posixpath.normpath(r[2]), | ||||
r[3], r[4], r[5])) | ||||
return r | ||||
else: # plain file system path | ||||
return posixpath.normpath(os.path.join(parent, repo._subsource)) | ||||
else: # recursion reached top repo | ||||
if push and repo.ui.config('paths', 'default-push'): | ||||
return repo.ui.config('paths', 'default-push') | ||||
if repo.ui.config('paths', 'default'): | ||||
return repo.ui.config('paths', 'default') | ||||
if abort: | ||||
raise util.Abort(_("default path for subrepository %s not found") % | ||||
reporelpath(repo)) | ||||
Matt Mackall
|
r8814 | |||
Martin Geisler
|
r12176 | def itersubrepos(ctx1, ctx2): | ||
"""find subrepos in ctx1 or ctx2""" | ||||
# Create a (subpath, ctx) mapping where we prefer subpaths from | ||||
# ctx1. The subpaths from ctx2 are important when the .hgsub file | ||||
# has been modified (in ctx2) but not yet committed (in ctx1). | ||||
subpaths = dict.fromkeys(ctx2.substate, ctx2) | ||||
subpaths.update(dict.fromkeys(ctx1.substate, ctx1)) | ||||
for subpath, ctx in sorted(subpaths.iteritems()): | ||||
yield subpath, ctx.sub(subpath) | ||||
Matt Mackall
|
r8813 | def subrepo(ctx, path): | ||
Mads Kiilerich
|
r11571 | """return instance of the right subrepo class for subrepo in path""" | ||
Matt Mackall
|
r8813 | # subrepo inherently violates our import layering rules | ||
# because it wants to make repo objects from deep inside the stack | ||||
# so we manually delay the circular imports to not break | ||||
# scripts that don't use our demand-loading | ||||
Abderrahim Kitouni
|
r9092 | global hg | ||
import hg as h | ||||
Matt Mackall
|
r8814 | hg = h | ||
Matt Mackall
|
r8813 | |||
Matt Mackall
|
r8997 | util.path_auditor(ctx._repo.root)(path) | ||
Matt Mackall
|
r8813 | state = ctx.substate.get(path, nullstate) | ||
Augie Fackler
|
r10177 | if state[2] not in types: | ||
Benoit Boissinot
|
r10299 | raise util.Abort(_('unknown subrepo type %s') % state[2]) | ||
Augie Fackler
|
r10177 | return types[state[2]](ctx, path, state[:2]) | ||
Matt Mackall
|
r8813 | |||
Martin Geisler
|
r11559 | # subrepo classes need to implement the following abstract class: | ||
class abstractsubrepo(object): | ||||
def dirty(self): | ||||
"""returns true if the dirstate of the subrepo does not match | ||||
current stored state | ||||
""" | ||||
raise NotImplementedError | ||||
Brodie Rao
|
r12506 | def checknested(self, path): | ||
Martin Geisler
|
r12162 | """check if path is a subrepository within this repository""" | ||
return False | ||||
Martin Geisler
|
r11559 | def commit(self, text, user, date): | ||
"""commit the current changes to the subrepo with the given | ||||
log message. Use given user and date if possible. Return the | ||||
new state of the subrepo. | ||||
""" | ||||
raise NotImplementedError | ||||
def remove(self): | ||||
"""remove the subrepo | ||||
Matt Mackall
|
r8813 | |||
Martin Geisler
|
r11559 | (should verify the dirstate is not dirty first) | ||
""" | ||||
raise NotImplementedError | ||||
def get(self, state): | ||||
"""run whatever commands are needed to put the subrepo into | ||||
this state | ||||
""" | ||||
raise NotImplementedError | ||||
def merge(self, state): | ||||
"""merge currently-saved state with the new state.""" | ||||
raise NotImplementedError | ||||
def push(self, force): | ||||
Martin Geisler
|
r11572 | """perform whatever action is analogous to 'hg push' | ||
Martin Geisler
|
r11559 | |||
This may be a no-op on some systems. | ||||
""" | ||||
raise NotImplementedError | ||||
Martin Geisler
|
r12270 | def add(self, ui, match, dryrun, prefix): | ||
return [] | ||||
Martin Geisler
|
r11559 | |||
Martin Geisler
|
r12166 | def status(self, rev2, **opts): | ||
return [], [], [], [], [], [], [] | ||||
Martin Geisler
|
r12167 | def diff(self, diffopts, node2, match, prefix, **opts): | ||
pass | ||||
Martin Geisler
|
r12272 | def outgoing(self, ui, dest, opts): | ||
return 1 | ||||
Martin Geisler
|
r12274 | def incoming(self, ui, source, opts): | ||
return 1 | ||||
Martin Geisler
|
r12322 | def files(self): | ||
"""return filename iterator""" | ||||
raise NotImplementedError | ||||
def filedata(self, name): | ||||
"""return file data""" | ||||
raise NotImplementedError | ||||
def fileflags(self, name): | ||||
"""return file flags""" | ||||
return '' | ||||
Martin Geisler
|
r12323 | def archive(self, archiver, prefix): | ||
for name in self.files(): | ||||
flags = self.fileflags(name) | ||||
mode = 'x' in flags and 0755 or 0644 | ||||
symlink = 'l' in flags | ||||
archiver.addfile(os.path.join(prefix, self._path, name), | ||||
mode, symlink, self.filedata(name)) | ||||
Martin Geisler
|
r11559 | class hgsubrepo(abstractsubrepo): | ||
Matt Mackall
|
r8813 | def __init__(self, ctx, path, state): | ||
self._path = path | ||||
self._state = state | ||||
r = ctx._repo | ||||
root = r.wjoin(path) | ||||
Benoit Boissinot
|
r10666 | create = False | ||
if not os.path.exists(os.path.join(root, '.hg')): | ||||
create = True | ||||
Matt Mackall
|
r8814 | util.makedirs(root) | ||
Benoit Boissinot
|
r10666 | self._repo = hg.repository(r.ui, root, create=create) | ||
Matt Mackall
|
r8814 | self._repo._subparent = r | ||
self._repo._subsource = state[0] | ||||
Matt Mackall
|
r8813 | |||
Benoit Boissinot
|
r10666 | if create: | ||
fp = self._repo.opener("hgrc", "w", text=True) | ||||
fp.write('[paths]\n') | ||||
def addpathconfig(key, value): | ||||
Mads Kiilerich
|
r12753 | if value: | ||
fp.write('%s = %s\n' % (key, value)) | ||||
self._repo.ui.setconfig('paths', key, value) | ||||
Benoit Boissinot
|
r10666 | |||
Mads Kiilerich
|
r12753 | defpath = _abssource(self._repo, abort=False) | ||
defpushpath = _abssource(self._repo, True, abort=False) | ||||
Benoit Boissinot
|
r10666 | addpathconfig('default', defpath) | ||
Edouard Gomez
|
r10697 | if defpath != defpushpath: | ||
addpathconfig('default-push', defpushpath) | ||||
Benoit Boissinot
|
r10666 | fp.close() | ||
Martin Geisler
|
r12270 | def add(self, ui, match, dryrun, prefix): | ||
return cmdutil.add(ui, self._repo, match, dryrun, True, | ||||
os.path.join(prefix, self._path)) | ||||
Martin Geisler
|
r12166 | def status(self, rev2, **opts): | ||
try: | ||||
rev1 = self._state[1] | ||||
ctx1 = self._repo[rev1] | ||||
ctx2 = self._repo[rev2] | ||||
return self._repo.status(ctx1, ctx2, **opts) | ||||
except error.RepoLookupError, inst: | ||||
Wagner Bruna
|
r12503 | self._repo.ui.warn(_('warning: error "%s" in subrepository "%s"\n') | ||
Mads Kiilerich
|
r12752 | % (inst, subrelpath(self))) | ||
Martin Geisler
|
r12166 | return [], [], [], [], [], [], [] | ||
Martin Geisler
|
r12167 | def diff(self, diffopts, node2, match, prefix, **opts): | ||
try: | ||||
node1 = node.bin(self._state[1]) | ||||
Patrick Mezard
|
r12209 | # We currently expect node2 to come from substate and be | ||
# in hex format | ||||
Martin Geisler
|
r12210 | if node2 is not None: | ||
node2 = node.bin(node2) | ||||
Martin Geisler
|
r12167 | cmdutil.diffordiffstat(self._repo.ui, self._repo, diffopts, | ||
node1, node2, match, | ||||
prefix=os.path.join(prefix, self._path), | ||||
listsubrepos=True, **opts) | ||||
except error.RepoLookupError, inst: | ||||
Wagner Bruna
|
r12503 | self._repo.ui.warn(_('warning: error "%s" in subrepository "%s"\n') | ||
Mads Kiilerich
|
r12752 | % (inst, subrelpath(self))) | ||
Martin Geisler
|
r12167 | |||
Martin Geisler
|
r12323 | def archive(self, archiver, prefix): | ||
abstractsubrepo.archive(self, archiver, prefix) | ||||
rev = self._state[1] | ||||
ctx = self._repo[rev] | ||||
for subpath in ctx.substate: | ||||
s = subrepo(ctx, subpath) | ||||
s.archive(archiver, os.path.join(prefix, self._path)) | ||||
Matt Mackall
|
r8813 | def dirty(self): | ||
r = self._state[1] | ||||
if r == '': | ||||
return True | ||||
w = self._repo[None] | ||||
Benoit Boissinot
|
r10666 | if w.p1() != self._repo[r]: # version checked out change | ||
Matt Mackall
|
r8813 | return True | ||
return w.dirty() # working directory changed | ||||
Martin Geisler
|
r12162 | def checknested(self, path): | ||
return self._repo._checknested(self._repo.wjoin(path)) | ||||
Matt Mackall
|
r8813 | def commit(self, text, user, date): | ||
Mads Kiilerich
|
r12752 | self._repo.ui.debug("committing subrepo %s\n" % subrelpath(self)) | ||
Matt Mackall
|
r8813 | n = self._repo.commit(text, user, date) | ||
if not n: | ||||
return self._repo['.'].hex() # different version checked out | ||||
return node.hex(n) | ||||
Matt Mackall
|
r8814 | |||
def remove(self): | ||||
# we can't fully delete the repository as it may contain | ||||
# local-only history | ||||
Mads Kiilerich
|
r12752 | self._repo.ui.note(_('removing subrepo %s\n') % subrelpath(self)) | ||
Matt Mackall
|
r8814 | hg.clean(self._repo, node.nullid, False) | ||
Matt Mackall
|
r9507 | def _get(self, state): | ||
Augie Fackler
|
r10177 | source, revision, kind = state | ||
Matt Mackall
|
r8814 | try: | ||
self._repo.lookup(revision) | ||||
except error.RepoError: | ||||
self._repo._subsource = source | ||||
srcurl = _abssource(self._repo) | ||||
Martin Geisler
|
r10671 | self._repo.ui.status(_('pulling subrepo %s from %s\n') | ||
Mads Kiilerich
|
r12752 | % (subrelpath(self), srcurl)) | ||
Matt Mackall
|
r8814 | other = hg.repository(self._repo.ui, srcurl) | ||
self._repo.pull(other) | ||||
Matt Mackall
|
r9507 | def get(self, state): | ||
self._get(state) | ||||
Augie Fackler
|
r10177 | source, revision, kind = state | ||
Matt Mackall
|
r9782 | self._repo.ui.debug("getting subrepo %s\n" % self._path) | ||
Matt Mackall
|
r8814 | hg.clean(self._repo, revision, False) | ||
def merge(self, state): | ||||
Matt Mackall
|
r9507 | self._get(state) | ||
Matt Mackall
|
r9781 | cur = self._repo['.'] | ||
dst = self._repo[state[1]] | ||||
Benoit Boissinot
|
r10251 | anc = dst.ancestor(cur) | ||
if anc == cur: | ||||
Mads Kiilerich
|
r12752 | self._repo.ui.debug("updating subrepo %s\n" % subrelpath(self)) | ||
Matt Mackall
|
r9781 | hg.update(self._repo, state[1]) | ||
Benoit Boissinot
|
r10251 | elif anc == dst: | ||
Mads Kiilerich
|
r12752 | self._repo.ui.debug("skipping subrepo %s\n" % subrelpath(self)) | ||
Matt Mackall
|
r9781 | else: | ||
Mads Kiilerich
|
r12752 | self._repo.ui.debug("merging subrepo %s\n" % subrelpath(self)) | ||
Matt Mackall
|
r9781 | hg.merge(self._repo, state[1], remind=False) | ||
Matt Mackall
|
r8815 | |||
def push(self, force): | ||||
# push subrepos depth-first for coherent ordering | ||||
c = self._repo[''] | ||||
subs = c.substate # only repos that are committed | ||||
for s in sorted(subs): | ||||
Matt Mackall
|
r11067 | if not c.sub(s).push(force): | ||
return False | ||||
Matt Mackall
|
r8815 | |||
dsturl = _abssource(self._repo, True) | ||||
Edouard Gomez
|
r11111 | self._repo.ui.status(_('pushing subrepo %s to %s\n') % | ||
Mads Kiilerich
|
r12752 | (subrelpath(self), dsturl)) | ||
Matt Mackall
|
r8815 | other = hg.repository(self._repo.ui, dsturl) | ||
Matt Mackall
|
r11067 | return self._repo.push(other, force) | ||
Augie Fackler
|
r10177 | |||
Martin Geisler
|
r12272 | def outgoing(self, ui, dest, opts): | ||
return hg.outgoing(ui, self._repo, _abssource(self._repo, True), opts) | ||||
Martin Geisler
|
r12274 | def incoming(self, ui, source, opts): | ||
return hg.incoming(ui, self._repo, _abssource(self._repo, False), opts) | ||||
Martin Geisler
|
r12322 | def files(self): | ||
rev = self._state[1] | ||||
ctx = self._repo[rev] | ||||
return ctx.manifest() | ||||
def filedata(self, name): | ||||
rev = self._state[1] | ||||
return self._repo[rev][name].data() | ||||
def fileflags(self, name): | ||||
rev = self._state[1] | ||||
ctx = self._repo[rev] | ||||
return ctx.flags(name) | ||||
Martin Geisler
|
r11559 | class svnsubrepo(abstractsubrepo): | ||
Augie Fackler
|
r10178 | def __init__(self, ctx, path, state): | ||
self._path = path | ||||
self._state = state | ||||
self._ctx = ctx | ||||
self._ui = ctx._repo.ui | ||||
Martin Geisler
|
r11560 | def _svncommand(self, commands, filename=''): | ||
path = os.path.join(self._ctx._repo.origroot, self._path, filename) | ||||
Brett Cannon
|
r10954 | cmd = ['svn'] + commands + [path] | ||
Augie Fackler
|
r10178 | cmd = [util.shellquote(arg) for arg in cmd] | ||
cmd = util.quotecommand(' '.join(cmd)) | ||||
Patrick Mezard
|
r10199 | env = dict(os.environ) | ||
Patrick Mezard
|
r10271 | # Avoid localized output, preserve current locale for everything else. | ||
env['LC_MESSAGES'] = 'C' | ||||
Patrick Mezard
|
r10199 | write, read, err = util.popen3(cmd, env=env, newlines=True) | ||
Augie Fackler
|
r10178 | retdata = read.read() | ||
err = err.read().strip() | ||||
if err: | ||||
raise util.Abort(err) | ||||
return retdata | ||||
def _wcrev(self): | ||||
Patrick Mezard
|
r10272 | output = self._svncommand(['info', '--xml']) | ||
doc = xml.dom.minidom.parseString(output) | ||||
entries = doc.getElementsByTagName('entry') | ||||
if not entries: | ||||
Augie Fackler
|
r10178 | return 0 | ||
Patrick Mezard
|
r10272 | return int(entries[0].getAttribute('revision') or 0) | ||
Augie Fackler
|
r10178 | |||
Patrick Mezard
|
r10273 | def _wcchanged(self): | ||
"""Return (changes, extchanges) where changes is True | ||||
if the working directory was changed, and extchanges is | ||||
True if any of these changes concern an external entry. | ||||
""" | ||||
Patrick Mezard
|
r10272 | output = self._svncommand(['status', '--xml']) | ||
Patrick Mezard
|
r10273 | externals, changes = [], [] | ||
Patrick Mezard
|
r10272 | doc = xml.dom.minidom.parseString(output) | ||
Patrick Mezard
|
r10273 | for e in doc.getElementsByTagName('entry'): | ||
s = e.getElementsByTagName('wc-status') | ||||
if not s: | ||||
continue | ||||
item = s[0].getAttribute('item') | ||||
props = s[0].getAttribute('props') | ||||
path = e.getAttribute('path') | ||||
if item == 'external': | ||||
externals.append(path) | ||||
if (item not in ('', 'normal', 'unversioned', 'external') | ||||
or props not in ('', 'none')): | ||||
changes.append(path) | ||||
for path in changes: | ||||
for ext in externals: | ||||
if path == ext or path.startswith(ext + os.sep): | ||||
return True, True | ||||
return bool(changes), False | ||||
Augie Fackler
|
r10178 | |||
def dirty(self): | ||||
Patrick Mezard
|
r10273 | if self._wcrev() == self._state[1] and not self._wcchanged()[0]: | ||
Augie Fackler
|
r10178 | return False | ||
return True | ||||
def commit(self, text, user, date): | ||||
# user and date are out of our hands since svn is centralized | ||||
Patrick Mezard
|
r10273 | changed, extchanged = self._wcchanged() | ||
if not changed: | ||||
Augie Fackler
|
r10178 | return self._wcrev() | ||
Patrick Mezard
|
r10273 | if extchanged: | ||
# Do not try to commit externals | ||||
raise util.Abort(_('cannot commit svn externals')) | ||||
Augie Fackler
|
r10178 | commitinfo = self._svncommand(['commit', '-m', text]) | ||
self._ui.status(commitinfo) | ||||
Martin Geisler
|
r12060 | newrev = re.search('Committed revision ([0-9]+).', commitinfo) | ||
Augie Fackler
|
r10178 | if not newrev: | ||
raise util.Abort(commitinfo.splitlines()[-1]) | ||||
newrev = newrev.groups()[0] | ||||
self._ui.status(self._svncommand(['update', '-r', newrev])) | ||||
return newrev | ||||
def remove(self): | ||||
if self.dirty(): | ||||
Benoit Boissinot
|
r10299 | self._ui.warn(_('not removing repo %s because ' | ||
'it has changes.\n' % self._path)) | ||||
Augie Fackler
|
r10178 | return | ||
Benoit Boissinot
|
r10510 | self._ui.note(_('removing subrepo %s\n') % self._path) | ||
Augie Fackler
|
r10178 | shutil.rmtree(self._ctx.repo.join(self._path)) | ||
def get(self, state): | ||||
status = self._svncommand(['checkout', state[0], '--revision', state[1]]) | ||||
Martin Geisler
|
r12060 | if not re.search('Checked out revision [0-9]+.', status): | ||
Augie Fackler
|
r10178 | raise util.Abort(status.splitlines()[-1]) | ||
self._ui.status(status) | ||||
def merge(self, state): | ||||
old = int(self._state[1]) | ||||
new = int(state[1]) | ||||
if new > old: | ||||
self.get(state) | ||||
def push(self, force): | ||||
Matt Mackall
|
r11455 | # push is a no-op for SVN | ||
return True | ||||
Augie Fackler
|
r10178 | |||
Martin Geisler
|
r12322 | def files(self): | ||
output = self._svncommand(['list']) | ||||
# This works because svn forbids \n in filenames. | ||||
return output.splitlines() | ||||
def filedata(self, name): | ||||
return self._svncommand(['cat'], name) | ||||
Augie Fackler
|
r10177 | types = { | ||
'hg': hgsubrepo, | ||||
Augie Fackler
|
r10178 | 'svn': svnsubrepo, | ||
Augie Fackler
|
r10177 | } | ||