bundlerepo.py
557 lines
| 19.9 KiB
| text/x-python
|
PythonLexer
/ mercurial / bundlerepo.py
Martin Geisler
|
r8226 | # bundlerepo.py - repository class for viewing uncompressed bundles | ||
# | ||||
# Copyright 2006, 2007 Benoit Boissinot <bboissin@gmail.com> | ||||
# | ||||
# 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. | ||
Benoit Boissinot
|
r1942 | |||
Martin Geisler
|
r8227 | """Repository class for viewing uncompressed bundles. | ||
This provides a read-only repository interface to bundles as if they | ||||
were part of the actual repository. | ||||
""" | ||||
Gregory Szorc
|
r25920 | from __future__ import absolute_import | ||
import os | ||||
import shutil | ||||
import tempfile | ||||
from .i18n import _ | ||||
from .node import nullid | ||||
from . import ( | ||||
bundle2, | ||||
changegroup, | ||||
changelog, | ||||
cmdutil, | ||||
discovery, | ||||
error, | ||||
exchange, | ||||
filelog, | ||||
localrepo, | ||||
manifest, | ||||
mdiff, | ||||
liscju
|
r28714 | node as nodemod, | ||
Gregory Szorc
|
r25920 | pathutil, | ||
phases, | ||||
Pulkit Goyal
|
r30519 | pycompat, | ||
Gregory Szorc
|
r25920 | revlog, | ||
scmutil, | ||||
util, | ||||
) | ||||
Benoit Boissinot
|
r1942 | |||
Benoit Boissinot
|
r1946 | class bundlerevlog(revlog.revlog): | ||
Benoit Boissinot
|
r14142 | def __init__(self, opener, indexfile, bundle, linkmapper): | ||
Benoit Boissinot
|
r1942 | # How it works: | ||
Mads Kiilerich
|
r18410 | # To retrieve a revision, we need to know the offset of the revision in | ||
# the bundle (an unbundle object). We store this offset in the index | ||||
Mads Kiilerich
|
r18643 | # (start). The base of the delta is stored in the base field. | ||
Benoit Boissinot
|
r1942 | # | ||
Mads Kiilerich
|
r18410 | # To differentiate a rev in the bundle from a rev in the revlog, we | ||
Mads Kiilerich
|
r18643 | # check revision against repotiprev. | ||
Pierre-Yves David
|
r18215 | opener = scmutil.readonlyvfs(opener) | ||
Matt Mackall
|
r4257 | revlog.revlog.__init__(self, opener, indexfile) | ||
Matt Mackall
|
r12332 | self.bundle = bundle | ||
Matt Mackall
|
r6750 | n = len(self) | ||
Mads Kiilerich
|
r18643 | self.repotiprev = n - 1 | ||
Benoit Boissinot
|
r14142 | chain = None | ||
Mads Kiilerich
|
r18411 | self.bundlerevs = set() # used by 'bundle()' revset expression | ||
Augie Fackler
|
r29711 | getchunk = lambda: bundle.deltachunk(chain) | ||
for chunkdata in iter(getchunk, {}): | ||||
Benoit Boissinot
|
r14142 | node = chunkdata['node'] | ||
p1 = chunkdata['p1'] | ||||
p2 = chunkdata['p2'] | ||||
cs = chunkdata['cs'] | ||||
deltabase = chunkdata['deltabase'] | ||||
delta = chunkdata['delta'] | ||||
size = len(delta) | ||||
start = bundle.tell() - size | ||||
link = linkmapper(cs) | ||||
Benoit Boissinot
|
r1942 | if node in self.nodemap: | ||
Benoit Boissinot
|
r14142 | # this can happen if two branches make the same change | ||
chain = node | ||||
Mads Kiilerich
|
r18411 | self.bundlerevs.add(self.nodemap[node]) | ||
Benoit Boissinot
|
r1942 | continue | ||
Benoit Boissinot
|
r14142 | |||
Benoit Boissinot
|
r1942 | for p in (p1, p2): | ||
Brodie Rao
|
r16686 | if p not in self.nodemap: | ||
Sune Foldager
|
r9650 | raise error.LookupError(p, self.indexfile, | ||
Matt Mackall
|
r7633 | _("unknown parent")) | ||
Mads Kiilerich
|
r18416 | |||
if deltabase not in self.nodemap: | ||||
raise LookupError(deltabase, self.indexfile, | ||||
_('unknown delta base')) | ||||
baserev = self.rev(deltabase) | ||||
Benoit Boissinot
|
r5167 | # start, size, full unc. size, base (unused), link, p1, p2, node | ||
Mads Kiilerich
|
r18643 | e = (revlog.offset_type(start, 0), size, -1, baserev, link, | ||
Matt Mackall
|
r4979 | self.rev(p1), self.rev(p2), node) | ||
self.index.insert(-1, e) | ||||
Benoit Boissinot
|
r1942 | self.nodemap[node] = n | ||
Mads Kiilerich
|
r18411 | self.bundlerevs.add(n) | ||
Benoit Boissinot
|
r14142 | chain = node | ||
Benoit Boissinot
|
r1942 | n += 1 | ||
Benoit Boissinot
|
r9676 | def _chunk(self, rev): | ||
Mads Kiilerich
|
r18643 | # Warning: in case of bundle, the diff is against what we stored as | ||
# delta base, not against rev - 1 | ||||
Benoit Boissinot
|
r1942 | # XXX: could use some caching | ||
Mads Kiilerich
|
r18643 | if rev <= self.repotiprev: | ||
Benoit Boissinot
|
r9676 | return revlog.revlog._chunk(self, rev) | ||
Matt Mackall
|
r12332 | self.bundle.seek(self.start(rev)) | ||
return self.bundle.read(self.length(rev)) | ||||
Benoit Boissinot
|
r1942 | |||
def revdiff(self, rev1, rev2): | ||||
"""return or calculate a delta between two revisions""" | ||||
Mads Kiilerich
|
r18643 | if rev1 > self.repotiprev and rev2 > self.repotiprev: | ||
Benoit Boissinot
|
r1942 | # hot path for bundle | ||
Mads Kiilerich
|
r18643 | revb = self.index[rev2][3] | ||
Benoit Boissinot
|
r1942 | if revb == rev1: | ||
Benoit Boissinot
|
r9676 | return self._chunk(rev2) | ||
Mads Kiilerich
|
r18643 | elif rev1 <= self.repotiprev and rev2 <= self.repotiprev: | ||
Benoit Boissinot
|
r4028 | return revlog.revlog.revdiff(self, rev1, rev2) | ||
Benoit Boissinot
|
r1942 | |||
Matt Mackall
|
r4989 | return mdiff.textdiff(self.revision(self.node(rev1)), | ||
Mads Kiilerich
|
r18413 | self.revision(self.node(rev2))) | ||
Benoit Boissinot
|
r1942 | |||
Remi Chaintron
|
r30743 | def revision(self, nodeorrev, raw=False): | ||
Patrick Mezard
|
r16435 | """return an uncompressed revision of a given node or revision | ||
number. | ||||
""" | ||||
Matt Mackall
|
r16375 | if isinstance(nodeorrev, int): | ||
rev = nodeorrev | ||||
node = self.node(rev) | ||||
else: | ||||
node = nodeorrev | ||||
rev = self.rev(node) | ||||
Matt Mackall
|
r10282 | if node == nullid: | ||
return "" | ||||
Benoit Boissinot
|
r1942 | |||
text = None | ||||
chain = [] | ||||
Mads Kiilerich
|
r18415 | iterrev = rev | ||
Benoit Boissinot
|
r1942 | # reconstruct the revision if it is from a changegroup | ||
Mads Kiilerich
|
r18643 | while iterrev > self.repotiprev: | ||
Mads Kiilerich
|
r18415 | if self._cache and self._cache[1] == iterrev: | ||
Matt Mackall
|
r4984 | text = self._cache[2] | ||
Benoit Boissinot
|
r1942 | break | ||
Mads Kiilerich
|
r18415 | chain.append(iterrev) | ||
Mads Kiilerich
|
r18643 | iterrev = self.index[iterrev][3] | ||
Benoit Boissinot
|
r1942 | if text is None: | ||
Wojciech Lopata
|
r19629 | text = self.baserevision(iterrev) | ||
Benoit Boissinot
|
r1942 | |||
while chain: | ||||
Benoit Boissinot
|
r9676 | delta = self._chunk(chain.pop()) | ||
Matt Mackall
|
r4989 | text = mdiff.patches(text, [delta]) | ||
Benoit Boissinot
|
r1942 | |||
Remi Chaintron
|
r30745 | text, validatehash = self._processflags(text, self.flags(rev), | ||
'read', raw=raw) | ||||
if validatehash: | ||||
self.checkhash(text, node, rev=rev) | ||||
Mads Kiilerich
|
r18415 | self._cache = (node, rev, text) | ||
Benoit Boissinot
|
r1942 | return text | ||
Wojciech Lopata
|
r19629 | def baserevision(self, nodeorrev): | ||
# Revlog subclasses may override 'revision' method to modify format of | ||||
# content retrieved from revlog. To use bundlerevlog with such class one | ||||
# needs to override 'baserevision' and make more specific call here. | ||||
return revlog.revlog.revision(self, nodeorrev) | ||||
Benoit Boissinot
|
r1942 | def addrevision(self, text, transaction, link, p1=None, p2=None, d=None): | ||
raise NotImplementedError | ||||
Peter Arrenbrecht
|
r6647 | def addgroup(self, revs, linkmapper, transaction): | ||
Benoit Boissinot
|
r1942 | raise NotImplementedError | ||
def strip(self, rev, minlink): | ||||
raise NotImplementedError | ||||
def checksize(self): | ||||
raise NotImplementedError | ||||
Benoit Boissinot
|
r1946 | class bundlechangelog(bundlerevlog, changelog.changelog): | ||
Matt Mackall
|
r12332 | def __init__(self, opener, bundle): | ||
Benoit Boissinot
|
r1946 | changelog.changelog.__init__(self, opener) | ||
Benoit Boissinot
|
r14142 | linkmapper = lambda x: x | ||
bundlerevlog.__init__(self, opener, self.indexfile, bundle, | ||||
linkmapper) | ||||
Benoit Boissinot
|
r1942 | |||
Wojciech Lopata
|
r19629 | def baserevision(self, nodeorrev): | ||
# Although changelog doesn't override 'revision' method, some extensions | ||||
# may replace this class with another that does. Same story with | ||||
# manifest and filelog classes. | ||||
Yuya Nishihara
|
r24882 | |||
# This bypasses filtering on changelog.node() and rev() because we need | ||||
# revision text of the bundle base even if it is hidden. | ||||
oldfilter = self.filteredrevs | ||||
try: | ||||
self.filteredrevs = () | ||||
return changelog.changelog.revision(self, nodeorrev) | ||||
finally: | ||||
self.filteredrevs = oldfilter | ||||
Wojciech Lopata
|
r19629 | |||
Durham Goode
|
r30373 | class bundlemanifest(bundlerevlog, manifest.manifestrevlog): | ||
Augie Fackler
|
r29715 | def __init__(self, opener, bundle, linkmapper, dirlogstarts=None, dir=''): | ||
Durham Goode
|
r30373 | manifest.manifestrevlog.__init__(self, opener, dir=dir) | ||
Matt Mackall
|
r12332 | bundlerevlog.__init__(self, opener, self.indexfile, bundle, | ||
Matt Mackall
|
r4257 | linkmapper) | ||
Augie Fackler
|
r29715 | if dirlogstarts is None: | ||
dirlogstarts = {} | ||||
if self.bundle.version == "03": | ||||
dirlogstarts = _getfilestarts(self.bundle) | ||||
self._dirlogstarts = dirlogstarts | ||||
self._linkmapper = linkmapper | ||||
Benoit Boissinot
|
r1942 | |||
Wojciech Lopata
|
r19629 | def baserevision(self, nodeorrev): | ||
Durham Goode
|
r26399 | node = nodeorrev | ||
if isinstance(node, int): | ||||
node = self.node(node) | ||||
Durham Goode
|
r29925 | if node in self.fulltextcache: | ||
result = self.fulltextcache[node].tostring() | ||||
Durham Goode
|
r26399 | else: | ||
Durham Goode
|
r30373 | result = manifest.manifestrevlog.revision(self, nodeorrev) | ||
Durham Goode
|
r26399 | return result | ||
Wojciech Lopata
|
r19629 | |||
Augie Fackler
|
r29715 | def dirlog(self, d): | ||
if d in self._dirlogstarts: | ||||
self.bundle.seek(self._dirlogstarts[d]) | ||||
return bundlemanifest( | ||||
self.opener, self.bundle, self._linkmapper, | ||||
self._dirlogstarts, dir=d) | ||||
return super(bundlemanifest, self).dirlog(d) | ||||
Benoit Boissinot
|
r1946 | class bundlefilelog(bundlerevlog, filelog.filelog): | ||
Martin von Zweigbergk
|
r24921 | def __init__(self, opener, path, bundle, linkmapper): | ||
Benoit Boissinot
|
r1946 | filelog.filelog.__init__(self, opener, path) | ||
Matt Mackall
|
r12332 | bundlerevlog.__init__(self, opener, self.indexfile, bundle, | ||
Matt Mackall
|
r4257 | linkmapper) | ||
Sune Foldager
|
r14287 | |||
Wojciech Lopata
|
r19629 | def baserevision(self, nodeorrev): | ||
return filelog.filelog.revision(self, nodeorrev) | ||||
Sune Foldager
|
r17193 | class bundlepeer(localrepo.localpeer): | ||
def canpush(self): | ||||
return False | ||||
Eric Sumner
|
r23631 | class bundlephasecache(phases.phasecache): | ||
def __init__(self, *args, **kwargs): | ||||
super(bundlephasecache, self).__init__(*args, **kwargs) | ||||
if util.safehasattr(self, 'opener'): | ||||
self.opener = scmutil.readonlyvfs(self.opener) | ||||
def write(self): | ||||
raise NotImplementedError | ||||
def _write(self, fp): | ||||
raise NotImplementedError | ||||
def _updateroots(self, phase, newroots, tr): | ||||
self.phaseroots[phase] = newroots | ||||
self.invalidate() | ||||
self.dirty = True | ||||
Augie Fackler
|
r29712 | def _getfilestarts(bundle): | ||
bundlefilespos = {} | ||||
for chunkdata in iter(bundle.filelogheader, {}): | ||||
fname = chunkdata['filename'] | ||||
bundlefilespos[fname] = bundle.tell() | ||||
for chunk in iter(lambda: bundle.deltachunk(None), {}): | ||||
pass | ||||
return bundlefilespos | ||||
Benoit Boissinot
|
r1946 | class bundlerepository(localrepo.localrepository): | ||
Benoit Boissinot
|
r1942 | def __init__(self, ui, path, bundlename): | ||
Pierre-Yves David
|
r26800 | def _writetempbundle(read, suffix, header=''): | ||
"""Write a temporary file to disk | ||||
This is closure because we need to make sure this tracked by | ||||
self.tempfile for cleanup purposes.""" | ||||
fdtemp, temp = self.vfs.mkstemp(prefix="hg-bundle-", | ||||
suffix=".hg10un") | ||||
self.tempfile = temp | ||||
Pulkit Goyal
|
r30924 | with os.fdopen(fdtemp, pycompat.sysstr('wb')) as fptemp: | ||
Pierre-Yves David
|
r26800 | fptemp.write(header) | ||
while True: | ||||
chunk = read(2**18) | ||||
if not chunk: | ||||
break | ||||
fptemp.write(chunk) | ||||
return self.vfs.open(self.tempfile, mode="rb") | ||||
John Mulligan
|
r6314 | self._tempparent = None | ||
try: | ||||
localrepo.localrepository.__init__(self, ui, path) | ||||
Matt Mackall
|
r7637 | except error.RepoError: | ||
John Mulligan
|
r6314 | self._tempparent = tempfile.mkdtemp() | ||
Martin Geisler
|
r9198 | localrepo.instance(ui, self._tempparent, 1) | ||
John Mulligan
|
r6314 | localrepo.localrepository.__init__(self, ui, self._tempparent) | ||
Mads Kiilerich
|
r20790 | self.ui.setconfig('phases', 'publish', False, 'bundlerepo') | ||
Vadim Gelfer
|
r2673 | |||
Peter Arrenbrecht
|
r6129 | if path: | ||
Alexander Solovyov
|
r11154 | self._url = 'bundle:' + util.expandpath(path) + '+' + bundlename | ||
Peter Arrenbrecht
|
r6129 | else: | ||
self._url = 'bundle:' + bundlename | ||||
Vadim Gelfer
|
r2673 | |||
Benoit Boissinot
|
r2273 | self.tempfile = None | ||
Adrian Buehlmann
|
r13274 | f = util.posixfile(bundlename, "rb") | ||
Eric Sumner
|
r24072 | self.bundlefile = self.bundle = exchange.readbundle(ui, f, bundlename) | ||
Matt Mackall
|
r12044 | |||
Eric Sumner
|
r24073 | if isinstance(self.bundle, bundle2.unbundle20): | ||
Pierre-Yves David
|
r26803 | cgstream = None | ||
for part in self.bundle.iterparts(): | ||||
if part.type == 'changegroup': | ||||
if cgstream is not None: | ||||
raise NotImplementedError("can't process " | ||||
"multiple changegroups") | ||||
cgstream = part | ||||
version = part.params.get('version', '01') | ||||
Augie Fackler
|
r29713 | legalcgvers = changegroup.supportedincomingversions(self) | ||
if version not in legalcgvers: | ||||
Pierre-Yves David
|
r26803 | msg = _('Unsupported changegroup version: %s') | ||
raise error.Abort(msg % version) | ||||
if self.bundle.compressed(): | ||||
cgstream = _writetempbundle(part.read, | ||||
".cg%sun" % version) | ||||
Eric Sumner
|
r24073 | |||
Pierre-Yves David
|
r26803 | if cgstream is None: | ||
liscju
|
r29389 | raise error.Abort(_('No changegroups found')) | ||
Pierre-Yves David
|
r26803 | cgstream.seek(0) | ||
Eric Sumner
|
r24073 | |||
Martin von Zweigbergk
|
r27751 | self.bundle = changegroup.getunbundler(version, cgstream, 'UN') | ||
Eric Sumner
|
r24073 | |||
Pierre-Yves David
|
r26801 | elif self.bundle.compressed(): | ||
f = _writetempbundle(self.bundle.read, '.hg10un', header='HG10UN') | ||||
self.bundlefile = self.bundle = exchange.readbundle(ui, f, | ||||
bundlename, | ||||
self.vfs) | ||||
Benoit Boissinot
|
r1942 | # dict with the mapping 'filename' -> position in the bundle | ||
self.bundlefilespos = {} | ||||
Brendan Cully
|
r5262 | |||
Eric Sumner
|
r23632 | self.firstnewrev = self.changelog.repotiprev + 1 | ||
phases.retractboundary(self, None, phases.draft, | ||||
[ctx.node() for ctx in self[self.firstnewrev:]]) | ||||
Pierre-Yves David
|
r18014 | @localrepo.unfilteredpropertycache | ||
Eric Sumner
|
r23631 | def _phasecache(self): | ||
return bundlephasecache(self, self._phasedefaults) | ||||
@localrepo.unfilteredpropertycache | ||||
Matt Mackall
|
r8260 | def changelog(self): | ||
Benoit Boissinot
|
r14144 | # consume the header if it exists | ||
self.bundle.changelogheader() | ||||
Angel Ezquerra
|
r23878 | c = bundlechangelog(self.svfs, self.bundle) | ||
Matt Mackall
|
r12332 | self.manstart = self.bundle.tell() | ||
Matt Mackall
|
r8260 | return c | ||
Durham Goode
|
r30218 | def _constructmanifest(self): | ||
Matt Mackall
|
r12332 | self.bundle.seek(self.manstart) | ||
Benoit Boissinot
|
r14144 | # consume the header if it exists | ||
self.bundle.manifestheader() | ||||
Pierre-Yves David
|
r28221 | linkmapper = self.unfiltered().changelog.rev | ||
m = bundlemanifest(self.svfs, self.bundle, linkmapper) | ||||
Matt Mackall
|
r12332 | self.filestart = self.bundle.tell() | ||
Matt Mackall
|
r8260 | return m | ||
Pierre-Yves David
|
r18014 | @localrepo.unfilteredpropertycache | ||
Matt Mackall
|
r8260 | def manstart(self): | ||
self.changelog | ||||
return self.manstart | ||||
Pierre-Yves David
|
r18014 | @localrepo.unfilteredpropertycache | ||
Matt Mackall
|
r8260 | def filestart(self): | ||
Durham Goode
|
r30375 | self.manifestlog | ||
Matt Mackall
|
r8260 | return self.filestart | ||
Benoit Boissinot
|
r1942 | |||
Vadim Gelfer
|
r2673 | def url(self): | ||
return self._url | ||||
Benoit Boissinot
|
r1942 | def file(self, f): | ||
Brendan Cully
|
r5262 | if not self.bundlefilespos: | ||
Matt Mackall
|
r12332 | self.bundle.seek(self.filestart) | ||
Augie Fackler
|
r29712 | self.bundlefilespos = _getfilestarts(self.bundle) | ||
Brendan Cully
|
r5262 | |||
Benoit Boissinot
|
r1942 | if f in self.bundlefilespos: | ||
Matt Mackall
|
r12332 | self.bundle.seek(self.bundlefilespos[f]) | ||
Pierre-Yves David
|
r28186 | linkmapper = self.unfiltered().changelog.rev | ||
return bundlefilelog(self.svfs, f, self.bundle, linkmapper) | ||||
Benoit Boissinot
|
r1942 | else: | ||
Angel Ezquerra
|
r23878 | return filelog.filelog(self.svfs, f) | ||
Benoit Boissinot
|
r1942 | |||
Matt Mackall
|
r12347 | def close(self): | ||
"""Close assigned bundle file immediately.""" | ||||
Eric Sumner
|
r24072 | self.bundlefile.close() | ||
Klaus Koch
|
r12962 | if self.tempfile is not None: | ||
FUJIWARA Katsunori
|
r20981 | self.vfs.unlink(self.tempfile) | ||
John Mulligan
|
r6314 | if self._tempparent: | ||
shutil.rmtree(self._tempparent, True) | ||||
Vadim Gelfer
|
r2740 | |||
Matt Mackall
|
r6315 | def cancopy(self): | ||
return False | ||||
Sune Foldager
|
r17193 | def peer(self): | ||
return bundlepeer(self) | ||||
Dirkjan Ochtman
|
r7435 | def getcwd(self): | ||
Pulkit Goyal
|
r30519 | return pycompat.getcwd() # always outside the repo | ||
Dirkjan Ochtman
|
r7435 | |||
liscju
|
r28714 | # Check if parents exist in localrepo before setting | ||
def setparents(self, p1, p2=nullid): | ||||
p1rev = self.changelog.rev(p1) | ||||
p2rev = self.changelog.rev(p2) | ||||
msg = _("setting parent to node %s that only exists in the bundle\n") | ||||
if self.changelog.repotiprev < p1rev: | ||||
self.ui.warn(msg % nodemod.hex(p1)) | ||||
if self.changelog.repotiprev < p2rev: | ||||
self.ui.warn(msg % nodemod.hex(p2)) | ||||
return super(bundlerepository, self).setparents(p1, p2) | ||||
Sune Foldager
|
r15597 | |||
Vadim Gelfer
|
r2740 | def instance(ui, path, create): | ||
if create: | ||||
Pierre-Yves David
|
r26587 | raise error.Abort(_('cannot create new bundle repository')) | ||
Matt Mackall
|
r25830 | # internal config: bundle.mainreporoot | ||
Peter Arrenbrecht
|
r5664 | parentpath = ui.config("bundle", "mainreporoot", "") | ||
Matt Mackall
|
r16042 | if not parentpath: | ||
# try to find the correct path to the working directory repo | ||||
Pulkit Goyal
|
r30519 | parentpath = cmdutil.findrepo(pycompat.getcwd()) | ||
Matt Mackall
|
r16042 | if parentpath is None: | ||
parentpath = '' | ||||
Peter Arrenbrecht
|
r5664 | if parentpath: | ||
# Try to make the full path relative so we get a nice, short URL. | ||||
# In particular, we don't want temp dir names in test outputs. | ||||
Pulkit Goyal
|
r30519 | cwd = pycompat.getcwd() | ||
Peter Arrenbrecht
|
r5664 | if parentpath == cwd: | ||
parentpath = '' | ||||
else: | ||||
FUJIWARA Katsunori
|
r24834 | cwd = pathutil.normasprefix(cwd) | ||
Peter Arrenbrecht
|
r5664 | if parentpath.startswith(cwd): | ||
parentpath = parentpath[len(cwd):] | ||||
Brodie Rao
|
r14076 | u = util.url(path) | ||
Brodie Rao
|
r13826 | path = u.localpath() | ||
if u.scheme == 'bundle': | ||||
Vadim Gelfer
|
r2740 | s = path.split("+", 1) | ||
if len(s) == 1: | ||||
Peter Arrenbrecht
|
r5664 | repopath, bundlename = parentpath, s[0] | ||
Vadim Gelfer
|
r2740 | else: | ||
repopath, bundlename = s | ||||
else: | ||||
Peter Arrenbrecht
|
r5664 | repopath, bundlename = parentpath, path | ||
Vadim Gelfer
|
r2740 | return bundlerepository(ui, repopath, bundlename) | ||
Nicolas Dumazet
|
r12734 | |||
Eric Sumner
|
r23633 | class bundletransactionmanager(object): | ||
def transaction(self): | ||||
return None | ||||
def close(self): | ||||
raise NotImplementedError | ||||
def release(self): | ||||
raise NotImplementedError | ||||
Peter Arrenbrecht
|
r14161 | def getremotechanges(ui, repo, other, onlyheads=None, bundlename=None, | ||
Peter Arrenbrecht
|
r14073 | force=False): | ||
Peter Arrenbrecht
|
r14161 | '''obtains a bundle of changes incoming from other | ||
"onlyheads" restricts the returned changes to those reachable from the | ||||
specified heads. | ||||
"bundlename", if given, stores the bundle to this file path permanently; | ||||
Peter Arrenbrecht
|
r14190 | otherwise it's stored to a temp file and gets deleted again when you call | ||
the returned "cleanupfn". | ||||
Peter Arrenbrecht
|
r14161 | "force" indicates whether to proceed on unrelated repos. | ||
Returns a tuple (local, csets, cleanupfn): | ||||
Brodie Rao
|
r16683 | "local" is a local repo from which to obtain the actual incoming | ||
changesets; it is a bundlerepo for the obtained bundle when the | ||||
original "other" is remote. | ||||
Peter Arrenbrecht
|
r14161 | "csets" lists the incoming changeset node ids. | ||
Brodie Rao
|
r16683 | "cleanupfn" must be called without arguments when you're done processing | ||
the changes; it closes both the original "other" and the one returned | ||||
here. | ||||
Peter Arrenbrecht
|
r14161 | ''' | ||
Brodie Rao
|
r16683 | tmp = discovery.findcommonincoming(repo, other, heads=onlyheads, | ||
force=force) | ||||
Nicolas Dumazet
|
r12734 | common, incoming, rheads = tmp | ||
if not incoming: | ||||
try: | ||||
Sune Foldager
|
r15091 | if bundlename: | ||
Matt Mackall
|
r21694 | os.unlink(bundlename) | ||
Idan Kamara
|
r14004 | except OSError: | ||
Nicolas Dumazet
|
r12734 | pass | ||
Mads Kiilerich
|
r18138 | return repo, [], other.close | ||
Nicolas Dumazet
|
r12734 | |||
Mads Kiilerich
|
r22182 | commonset = set(common) | ||
rheads = [x for x in rheads if x not in commonset] | ||||
Nicolas Dumazet
|
r12734 | bundle = None | ||
Peter Arrenbrecht
|
r14161 | bundlerepo = None | ||
Sune Foldager
|
r17191 | localrepo = other.local() | ||
if bundlename or not localrepo: | ||||
Nicolas Dumazet
|
r12734 | # create a bundle (uncompressed if other repo is not local) | ||
Pierre-Yves David
|
r29684 | # developer config: devel.legacy.exchange | ||
legexc = ui.configlist('devel', 'legacy.exchange') | ||||
Pierre-Yves David
|
r29689 | forcebundle1 = 'bundle2' not in legexc and 'bundle1' in legexc | ||
Pierre-Yves David
|
r29684 | canbundle2 = (not forcebundle1 | ||
Pierre-Yves David
|
r26544 | and other.capable('getbundle') | ||
and other.capable('bundle2')) | ||||
if canbundle2: | ||||
kwargs = {} | ||||
kwargs['common'] = common | ||||
kwargs['heads'] = rheads | ||||
kwargs['bundlecaps'] = exchange.caps20to10(repo) | ||||
kwargs['cg'] = True | ||||
b2 = other.getbundle('incoming', **kwargs) | ||||
fname = bundle = changegroup.writechunks(ui, b2._forwardchunks(), | ||||
bundlename) | ||||
else: | ||||
Pierre-Yves David
|
r26543 | if other.capable('getbundle'): | ||
cg = other.getbundle('incoming', common=common, heads=rheads) | ||||
elif onlyheads is None and not other.capable('changegroupsubset'): | ||||
# compat with older servers when pulling all remote heads | ||||
cg = other.changegroup(incoming, "incoming") | ||||
rheads = None | ||||
else: | ||||
cg = other.changegroupsubset(incoming, rheads, 'incoming') | ||||
if localrepo: | ||||
bundletype = "HG10BZ" | ||||
else: | ||||
bundletype = "HG10UN" | ||||
Martin von Zweigbergk
|
r28666 | fname = bundle = bundle2.writebundle(ui, cg, bundlename, | ||
Pierre-Yves David
|
r26543 | bundletype) | ||
Nicolas Dumazet
|
r12734 | # keep written bundle? | ||
if bundlename: | ||||
bundle = None | ||||
Sune Foldager
|
r17191 | if not localrepo: | ||
Nicolas Dumazet
|
r12734 | # use the created uncompressed bundlerepo | ||
Simon Heimberg
|
r18825 | localrepo = bundlerepo = bundlerepository(repo.baseui, repo.root, | ||
fname) | ||||
Peter Arrenbrecht
|
r14161 | # this repo contains local and other now, so filter out local again | ||
common = repo.heads() | ||||
Pierre-Yves David
|
r18568 | if localrepo: | ||
# Part of common may be remotely filtered | ||||
# So use an unfiltered version | ||||
# The discovery process probably need cleanup to avoid that | ||||
localrepo = localrepo.unfiltered() | ||||
Peter Arrenbrecht
|
r14161 | |||
Peter Arrenbrecht
|
r14412 | csets = localrepo.changelog.findmissing(common, rheads) | ||
Nicolas Dumazet
|
r12734 | |||
Eric Sumner
|
r23633 | if bundlerepo: | ||
reponodes = [ctx.node() for ctx in bundlerepo[bundlerepo.firstnewrev:]] | ||||
remotephases = other.listkeys('phases') | ||||
pullop = exchange.pulloperation(bundlerepo, other, heads=reponodes) | ||||
pullop.trmanager = bundletransactionmanager() | ||||
exchange._pullapplyphases(pullop, remotephases) | ||||
Peter Arrenbrecht
|
r14161 | def cleanup(): | ||
if bundlerepo: | ||||
bundlerepo.close() | ||||
if bundle: | ||||
Matt Mackall
|
r21694 | os.unlink(bundle) | ||
Peter Arrenbrecht
|
r14190 | other.close() | ||
Peter Arrenbrecht
|
r14161 | |||
return (localrepo, csets, cleanup) | ||||