bundlerepo.py
759 lines
| 24.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. | ||||
""" | ||||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
Gregory Szorc
|
r25920 | |||
r51905 | import contextlib | |||
Gregory Szorc
|
r25920 | import os | ||
import shutil | ||||
from .i18n import _ | ||||
Joerg Sonnenberger
|
r46729 | from .node import ( | ||
hex, | ||||
nullrev, | ||||
) | ||||
Gregory Szorc
|
r25920 | |||
from . import ( | ||||
bundle2, | ||||
changegroup, | ||||
changelog, | ||||
cmdutil, | ||||
discovery, | ||||
Matt Harbison
|
r39843 | encoding, | ||
Gregory Szorc
|
r25920 | error, | ||
exchange, | ||||
filelog, | ||||
localrepo, | ||||
manifest, | ||||
mdiff, | ||||
pathutil, | ||||
phases, | ||||
Pulkit Goyal
|
r30519 | pycompat, | ||
Gregory Szorc
|
r25920 | revlog, | ||
r48186 | revlogutils, | |||
Gregory Szorc
|
r25920 | util, | ||
Pierre-Yves David
|
r31240 | vfs as vfsmod, | ||
Gregory Szorc
|
r25920 | ) | ||
r47669 | from .utils import ( | |||
urlutil, | ||||
) | ||||
Benoit Boissinot
|
r1942 | |||
r47838 | from .revlogutils import ( | |||
constants as revlog_constants, | ||||
) | ||||
Augie Fackler
|
r43346 | |||
Benoit Boissinot
|
r1946 | class bundlerevlog(revlog.revlog): | ||
r47921 | def __init__(self, opener, target, radix, cgunpacker, 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
|
r31240 | opener = vfsmod.readonlyvfs(opener) | ||
r47921 | revlog.revlog.__init__(self, opener, target=target, radix=radix) | |||
Gregory Szorc
|
r35075 | self.bundle = cgunpacker | ||
Matt Mackall
|
r6750 | n = len(self) | ||
Mads Kiilerich
|
r18643 | self.repotiprev = n - 1 | ||
Augie Fackler
|
r43346 | self.bundlerevs = set() # used by 'bundle()' revset expression | ||
Gregory Szorc
|
r35075 | for deltadata in cgunpacker.deltaiter(): | ||
Raphaël Gomès
|
r47445 | node, p1, p2, cs, deltabase, delta, flags, sidedata = deltadata | ||
Benoit Boissinot
|
r14142 | |||
size = len(delta) | ||||
Gregory Szorc
|
r35075 | start = cgunpacker.tell() - size | ||
Benoit Boissinot
|
r14142 | |||
r43942 | if self.index.has_node(node): | |||
Benoit Boissinot
|
r14142 | # this can happen if two branches make the same change | ||
r43964 | self.bundlerevs.add(self.index.rev(node)) | |||
Benoit Boissinot
|
r1942 | continue | ||
Joerg Sonnenberger
|
r46418 | if cs == node: | ||
linkrev = nullrev | ||||
else: | ||||
linkrev = linkmapper(cs) | ||||
Benoit Boissinot
|
r14142 | |||
Benoit Boissinot
|
r1942 | for p in (p1, p2): | ||
r43942 | if not self.index.has_node(p): | |||
Augie Fackler
|
r43346 | raise error.LookupError( | ||
r47926 | p, self.display_id, _(b"unknown parent") | |||
Augie Fackler
|
r43346 | ) | ||
Mads Kiilerich
|
r18416 | |||
r43942 | if not self.index.has_node(deltabase): | |||
Matt Harbison
|
r50870 | raise error.LookupError( | ||
r47926 | deltabase, self.display_id, _(b'unknown delta base') | |||
Augie Fackler
|
r43346 | ) | ||
Mads Kiilerich
|
r18416 | |||
baserev = self.rev(deltabase) | ||||
r47914 | # start, size, full unc. size, base (unused), link, p1, p2, node, sidedata_offset (unused), sidedata_size (unused) | |||
r48190 | e = revlogutils.entry( | |||
flags=flags, | ||||
data_offset=start, | ||||
data_compressed_length=size, | ||||
data_delta_base=baserev, | ||||
link_rev=linkrev, | ||||
parent_rev_1=self.rev(p1), | ||||
parent_rev_2=self.rev(p2), | ||||
node_id=node, | ||||
Augie Fackler
|
r43346 | ) | ||
Martin von Zweigbergk
|
r38886 | self.index.append(e) | ||
Mads Kiilerich
|
r18411 | self.bundlerevs.add(n) | ||
Benoit Boissinot
|
r1942 | n += 1 | ||
r51905 | @contextlib.contextmanager | |||
def reading(self): | ||||
if self.repotiprev < 0: | ||||
yield | ||||
else: | ||||
with super().reading() as x: | ||||
yield x | ||||
r51919 | 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 | |||
Augie Fackler
|
r43346 | return mdiff.textdiff(self.rawdata(rev1), self.rawdata(rev2)) | ||
Benoit Boissinot
|
r1942 | |||
r51919 | def _rawtext(self, node, rev): | |||
r43089 | if rev is None: | |||
Matt Mackall
|
r16375 | rev = self.rev(node) | ||
r43089 | validated = False | |||
Jun Wu
|
r31836 | rawtext = None | ||
Benoit Boissinot
|
r1942 | 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: | ||
r51989 | if ( | |||
self._inner._revisioncache | ||||
and self._inner._revisioncache[1] == iterrev | ||||
): | ||||
rawtext = self._inner._revisioncache[2] | ||||
Benoit Boissinot
|
r1942 | break | ||
Mads Kiilerich
|
r18415 | chain.append(iterrev) | ||
Mads Kiilerich
|
r18643 | iterrev = self.index[iterrev][3] | ||
r43089 | if iterrev == nullrev: | |||
Augie Fackler
|
r43347 | rawtext = b'' | ||
r43089 | elif rawtext is None: | |||
Augie Fackler
|
r43346 | r = super(bundlerevlog, self)._rawtext( | ||
r51919 | self.node(iterrev), | |||
iterrev, | ||||
Augie Fackler
|
r43346 | ) | ||
r43089 | __, rawtext, validated = r | |||
if chain: | ||||
validated = False | ||||
Benoit Boissinot
|
r1942 | while chain: | ||
Benoit Boissinot
|
r9676 | delta = self._chunk(chain.pop()) | ||
Jun Wu
|
r31836 | rawtext = mdiff.patches(rawtext, [delta]) | ||
r43089 | return rev, rawtext, validated | |||
Benoit Boissinot
|
r1942 | |||
Gregory Szorc
|
r35038 | def addrevision(self, *args, **kwargs): | ||
raise NotImplementedError | ||||
def addgroup(self, *args, **kwargs): | ||||
Benoit Boissinot
|
r1942 | raise NotImplementedError | ||
Gregory Szorc
|
r35038 | |||
def strip(self, *args, **kwargs): | ||||
Benoit Boissinot
|
r1942 | raise NotImplementedError | ||
Gregory Szorc
|
r35038 | |||
Benoit Boissinot
|
r1942 | def checksize(self): | ||
raise NotImplementedError | ||||
Augie Fackler
|
r43346 | |||
Benoit Boissinot
|
r1946 | class bundlechangelog(bundlerevlog, changelog.changelog): | ||
Gregory Szorc
|
r35075 | def __init__(self, opener, cgunpacker): | ||
Benoit Boissinot
|
r1946 | changelog.changelog.__init__(self, opener) | ||
Benoit Boissinot
|
r14142 | linkmapper = lambda x: x | ||
Augie Fackler
|
r43346 | bundlerevlog.__init__( | ||
r47838 | self, | |||
opener, | ||||
(revlog_constants.KIND_CHANGELOG, None), | ||||
r47921 | self.radix, | |||
r47838 | cgunpacker, | |||
linkmapper, | ||||
Augie Fackler
|
r43346 | ) | ||
Benoit Boissinot
|
r1942 | |||
Durham Goode
|
r30373 | class bundlemanifest(bundlerevlog, manifest.manifestrevlog): | ||
Augie Fackler
|
r43346 | def __init__( | ||
Joerg Sonnenberger
|
r47538 | self, | ||
nodeconstants, | ||||
opener, | ||||
cgunpacker, | ||||
linkmapper, | ||||
dirlogstarts=None, | ||||
dir=b'', | ||||
Augie Fackler
|
r43346 | ): | ||
r51921 | # XXX manifestrevlog is not actually a revlog , so mixing it with | |||
# bundlerevlog is not a good idea. | ||||
Joerg Sonnenberger
|
r47538 | manifest.manifestrevlog.__init__(self, nodeconstants, opener, tree=dir) | ||
Augie Fackler
|
r43346 | bundlerevlog.__init__( | ||
r47838 | self, | |||
opener, | ||||
(revlog_constants.KIND_MANIFESTLOG, dir), | ||||
r47921 | self._revlog.radix, | |||
r47838 | cgunpacker, | |||
linkmapper, | ||||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r29715 | if dirlogstarts is None: | ||
dirlogstarts = {} | ||||
Augie Fackler
|
r43347 | if self.bundle.version == b"03": | ||
Augie Fackler
|
r29715 | dirlogstarts = _getfilestarts(self.bundle) | ||
self._dirlogstarts = dirlogstarts | ||||
self._linkmapper = linkmapper | ||||
Benoit Boissinot
|
r1942 | |||
Augie Fackler
|
r29715 | def dirlog(self, d): | ||
if d in self._dirlogstarts: | ||||
self.bundle.seek(self._dirlogstarts[d]) | ||||
return bundlemanifest( | ||||
Joerg Sonnenberger
|
r47538 | self.nodeconstants, | ||
Augie Fackler
|
r43346 | self.opener, | ||
self.bundle, | ||||
self._linkmapper, | ||||
self._dirlogstarts, | ||||
dir=d, | ||||
) | ||||
Augie Fackler
|
r29715 | return super(bundlemanifest, self).dirlog(d) | ||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r37515 | class bundlefilelog(filelog.filelog): | ||
Gregory Szorc
|
r35075 | def __init__(self, opener, path, cgunpacker, linkmapper): | ||
Benoit Boissinot
|
r1946 | filelog.filelog.__init__(self, opener, path) | ||
Augie Fackler
|
r43346 | self._revlog = bundlerevlog( | ||
r47838 | opener, | |||
# XXX should use the unencoded path | ||||
target=(revlog_constants.KIND_FILELOG, path), | ||||
r47921 | radix=self._revlog.radix, | |||
r47838 | cgunpacker=cgunpacker, | |||
linkmapper=linkmapper, | ||||
Augie Fackler
|
r43346 | ) | ||
Sune Foldager
|
r14287 | |||
Sune Foldager
|
r17193 | class bundlepeer(localrepo.localpeer): | ||
def canpush(self): | ||||
return False | ||||
Augie Fackler
|
r43346 | |||
Eric Sumner
|
r23631 | class bundlephasecache(phases.phasecache): | ||
def __init__(self, *args, **kwargs): | ||||
super(bundlephasecache, self).__init__(*args, **kwargs) | ||||
r51821 | if hasattr(self, 'opener'): | |||
Pierre-Yves David
|
r31240 | self.opener = vfsmod.readonlyvfs(self.opener) | ||
Eric Sumner
|
r23631 | |||
def write(self): | ||||
raise NotImplementedError | ||||
def _write(self, fp): | ||||
raise NotImplementedError | ||||
r52313 | def _updateroots(self, repo, phase, newroots, tr, invalidate=True): | |||
r52296 | self._phaseroots[phase] = newroots | |||
r52313 | if invalidate: | |||
self.invalidate() | ||||
Eric Sumner
|
r23631 | self.dirty = True | ||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r35075 | def _getfilestarts(cgunpacker): | ||
Gregory Szorc
|
r35076 | filespos = {} | ||
Gregory Szorc
|
r35075 | for chunkdata in iter(cgunpacker.filelogheader, {}): | ||
Augie Fackler
|
r43347 | fname = chunkdata[b'filename'] | ||
Gregory Szorc
|
r35076 | filespos[fname] = cgunpacker.tell() | ||
Gregory Szorc
|
r35075 | for chunk in iter(lambda: cgunpacker.deltachunk(None), {}): | ||
Augie Fackler
|
r29712 | pass | ||
Gregory Szorc
|
r35076 | return filespos | ||
Augie Fackler
|
r29712 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class bundlerepository: | ||
Gregory Szorc
|
r35041 | """A repository instance that is a union of a local repo and a bundle. | ||
Instances represent a read-only repository composed of a local repository | ||||
with the contents of a bundle file applied. The repository instance is | ||||
conceptually similar to the state of a repository after an | ||||
``hg unbundle`` operation. However, the contents of the bundle are never | ||||
applied to the actual base repository. | ||||
Gregory Szorc
|
r39640 | |||
Instances constructed directly are not usable as repository objects. | ||||
Use instance() or makebundlerepository() to create instances. | ||||
Gregory Szorc
|
r35041 | """ | ||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r39640 | def __init__(self, bundlepath, url, tempparent): | ||
self._tempparent = tempparent | ||||
self._url = url | ||||
Augie Fackler
|
r43347 | self.ui.setconfig(b'phases', b'publish', False, b'bundlerepo') | ||
Vadim Gelfer
|
r2673 | |||
r51093 | # dict with the mapping 'filename' -> position in the changegroup. | |||
self._cgfilespos = {} | ||||
self._bundlefile = None | ||||
self._cgunpacker = None | ||||
Benoit Boissinot
|
r2273 | self.tempfile = None | ||
Augie Fackler
|
r43347 | f = util.posixfile(bundlepath, b"rb") | ||
Gregory Szorc
|
r39640 | bundle = exchange.readbundle(self.ui, f, bundlepath) | ||
Matt Mackall
|
r12044 | |||
Gregory Szorc
|
r35044 | if isinstance(bundle, bundle2.unbundle20): | ||
self._bundlefile = bundle | ||||
Gregory Szorc
|
r35112 | cgpart = None | ||
Gregory Szorc
|
r35113 | for part in bundle.iterparts(seekable=True): | ||
Matt Harbison , Pierre-Yves David pierre-yves.david@octobus.net
|
r51095 | if part.type == b'phase-heads': | ||
self._handle_bundle2_phase_part(bundle, part) | ||||
elif part.type == b'changegroup': | ||||
Gregory Szorc
|
r35112 | if cgpart: | ||
Augie Fackler
|
r43346 | raise NotImplementedError( | ||
Martin von Zweigbergk
|
r43387 | b"can't process multiple changegroups" | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r35112 | cgpart = part | ||
r51091 | self._handle_bundle2_cg_part(bundle, part) | |||
Eric Sumner
|
r24073 | |||
Gregory Szorc
|
r35112 | if not cgpart: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b"No changegroups found")) | ||
Gregory Szorc
|
r35112 | |||
# This is required to placate a later consumer, which expects | ||||
# the payload offset to be at the beginning of the changegroup. | ||||
# We need to do this after the iterparts() generator advances | ||||
# because iterparts() will seek to end of payload after the | ||||
# generator returns control to iterparts(). | ||||
cgpart.seek(0, os.SEEK_SET) | ||||
Gregory Szorc
|
r35044 | elif isinstance(bundle, changegroup.cg1unpacker): | ||
r51092 | self._handle_bundle1(bundle, bundlepath) | |||
Gregory Szorc
|
r35042 | else: | ||
Augie Fackler
|
r43347 | raise error.Abort( | ||
r51096 | _(b'bundle type %r cannot be read') % type(bundle) | |||
Augie Fackler
|
r43347 | ) | ||
Pierre-Yves David
|
r26801 | |||
r51094 | def _handle_bundle1(self, bundle, bundlepath): | |||
if bundle.compressed(): | ||||
f = self._writetempbundle(bundle.read, b'.hg10un', header=b'HG10UN') | ||||
bundle = exchange.readbundle(self.ui, f, bundlepath, self.vfs) | ||||
self._bundlefile = bundle | ||||
self._cgunpacker = bundle | ||||
Eric Sumner
|
r23632 | self.firstnewrev = self.changelog.repotiprev + 1 | ||
Augie Fackler
|
r43346 | phases.retractboundary( | ||
self, | ||||
None, | ||||
phases.draft, | ||||
[ctx.node() for ctx in self[self.firstnewrev :]], | ||||
) | ||||
Eric Sumner
|
r23632 | |||
r51091 | def _handle_bundle2_cg_part(self, bundle, part): | |||
assert part.type == b'changegroup' | ||||
Gregory Szorc
|
r35074 | cgstream = part | ||
Matt Harbison , Pierre-Yves David pierre-yves.david@octobus.net
|
r51095 | targetphase = part.params.get(b'targetphase') | ||
try: | ||||
targetphase = int(targetphase) | ||||
except TypeError: | ||||
pass | ||||
if targetphase is None: | ||||
targetphase = phases.draft | ||||
if targetphase not in phases.allphases: | ||||
m = _(b'unsupported targetphase: %d') | ||||
m %= targetphase | ||||
raise error.Abort(m) | ||||
Augie Fackler
|
r43347 | version = part.params.get(b'version', b'01') | ||
Gregory Szorc
|
r35074 | legalcgvers = changegroup.supportedincomingversions(self) | ||
if version not in legalcgvers: | ||||
Augie Fackler
|
r43347 | msg = _(b'Unsupported changegroup version: %s') | ||
Gregory Szorc
|
r35074 | raise error.Abort(msg % version) | ||
if bundle.compressed(): | ||||
Augie Fackler
|
r43347 | cgstream = self._writetempbundle(part.read, b'.cg%sun' % version) | ||
Gregory Szorc
|
r35074 | |||
Augie Fackler
|
r43347 | self._cgunpacker = changegroup.getunbundler(version, cgstream, b'UN') | ||
Durham Goode
|
r33890 | |||
r51094 | self.firstnewrev = self.changelog.repotiprev + 1 | |||
phases.retractboundary( | ||||
self, | ||||
None, | ||||
Matt Harbison , Pierre-Yves David pierre-yves.david@octobus.net
|
r51095 | targetphase, | ||
r51094 | [ctx.node() for ctx in self[self.firstnewrev :]], | |||
) | ||||
Matt Harbison , Pierre-Yves David pierre-yves.david@octobus.net
|
r51095 | def _handle_bundle2_phase_part(self, bundle, part): | ||
assert part.type == b'phase-heads' | ||||
unfi = self.unfiltered() | ||||
headsbyphase = phases.binarydecode(part) | ||||
phases.updatephases(unfi, lambda: None, headsbyphase) | ||||
Augie Fackler
|
r43347 | def _writetempbundle(self, readfn, suffix, header=b''): | ||
Augie Fackler
|
r46554 | """Write a temporary file to disk""" | ||
Augie Fackler
|
r43347 | fdtemp, temp = self.vfs.mkstemp(prefix=b"hg-bundle-", suffix=suffix) | ||
Durham Goode
|
r33888 | self.tempfile = temp | ||
Augie Fackler
|
r43906 | with os.fdopen(fdtemp, 'wb') as fptemp: | ||
Durham Goode
|
r33888 | fptemp.write(header) | ||
while True: | ||||
Raphaël Gomès
|
r52596 | chunk = readfn(2**18) | ||
Durham Goode
|
r33888 | if not chunk: | ||
break | ||||
fptemp.write(chunk) | ||||
Augie Fackler
|
r43347 | return self.vfs.open(self.tempfile, mode=b"rb") | ||
Durham Goode
|
r33888 | |||
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 | ||
Gregory Szorc
|
r35045 | self._cgunpacker.changelogheader() | ||
c = bundlechangelog(self.svfs, self._cgunpacker) | ||||
self.manstart = self._cgunpacker.tell() | ||||
Matt Mackall
|
r8260 | return c | ||
r42710 | def _refreshchangelog(self): | |||
# changelog for bundle repo are not filecache, this method is not | ||||
# applicable. | ||||
pass | ||||
Gregory Szorc
|
r39799 | @localrepo.unfilteredpropertycache | ||
def manifestlog(self): | ||||
Gregory Szorc
|
r35045 | self._cgunpacker.seek(self.manstart) | ||
Benoit Boissinot
|
r14144 | # consume the header if it exists | ||
Gregory Szorc
|
r35045 | self._cgunpacker.manifestheader() | ||
Pierre-Yves David
|
r28221 | linkmapper = self.unfiltered().changelog.rev | ||
Joerg Sonnenberger
|
r47538 | rootstore = bundlemanifest( | ||
self.nodeconstants, self.svfs, self._cgunpacker, linkmapper | ||||
) | ||||
Gregory Szorc
|
r35045 | self.filestart = self._cgunpacker.tell() | ||
Gregory Szorc
|
r39799 | |||
Augie Fackler
|
r43346 | return manifest.manifestlog( | ||
self.svfs, self, rootstore, self.narrowmatch() | ||||
) | ||||
Matt Mackall
|
r8260 | |||
Durham Goode
|
r35012 | def _consumemanifest(self): | ||
"""Consumes the manifest portion of the bundle, setting filestart so the | ||||
file portion can be read.""" | ||||
Gregory Szorc
|
r35045 | self._cgunpacker.seek(self.manstart) | ||
self._cgunpacker.manifestheader() | ||||
for delta in self._cgunpacker.deltaiter(): | ||||
Durham Goode
|
r35012 | pass | ||
Gregory Szorc
|
r35045 | self.filestart = self._cgunpacker.tell() | ||
Durham Goode
|
r35012 | |||
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 | ||
Durham Goode
|
r35012 | |||
# If filestart was not set by self.manifestlog, that means the | ||||
# manifestlog implementation did not consume the manifests from the | ||||
# changegroup (ex: it might be consuming trees from a separate bundle2 | ||||
# part instead). So we need to manually consume it. | ||||
Augie Fackler
|
r43906 | if 'filestart' not in self.__dict__: | ||
Durham Goode
|
r35012 | self._consumemanifest() | ||
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): | ||
Gregory Szorc
|
r35076 | if not self._cgfilespos: | ||
Gregory Szorc
|
r35045 | self._cgunpacker.seek(self.filestart) | ||
Gregory Szorc
|
r35076 | self._cgfilespos = _getfilestarts(self._cgunpacker) | ||
Brendan Cully
|
r5262 | |||
Gregory Szorc
|
r35076 | if f in self._cgfilespos: | ||
self._cgunpacker.seek(self._cgfilespos[f]) | ||||
Pierre-Yves David
|
r28186 | linkmapper = self.unfiltered().changelog.rev | ||
Gregory Szorc
|
r35045 | return bundlefilelog(self.svfs, f, self._cgunpacker, linkmapper) | ||
Benoit Boissinot
|
r1942 | else: | ||
Gregory Szorc
|
r37347 | return super(bundlerepository, self).file(f) | ||
Benoit Boissinot
|
r1942 | |||
Matt Mackall
|
r12347 | def close(self): | ||
"""Close assigned bundle file immediately.""" | ||||
Gregory Szorc
|
r35043 | 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 | ||||
Manuel Jacob
|
r51309 | def peer(self, path=None, remotehidden=False): | ||
return bundlepeer(self, path=path, remotehidden=remotehidden) | ||||
Sune Foldager
|
r17193 | |||
Dirkjan Ochtman
|
r7435 | def getcwd(self): | ||
Augie Fackler
|
r43346 | return encoding.getcwd() # always outside the repo | ||
Dirkjan Ochtman
|
r7435 | |||
liscju
|
r28714 | # Check if parents exist in localrepo before setting | ||
Joerg Sonnenberger
|
r47771 | def setparents(self, p1, p2=None): | ||
if p2 is None: | ||||
p2 = self.nullid | ||||
liscju
|
r28714 | p1rev = self.changelog.rev(p1) | ||
p2rev = self.changelog.rev(p2) | ||||
Augie Fackler
|
r43347 | msg = _(b"setting parent to node %s that only exists in the bundle\n") | ||
liscju
|
r28714 | if self.changelog.repotiprev < p1rev: | ||
Joerg Sonnenberger
|
r46729 | self.ui.warn(msg % hex(p1)) | ||
liscju
|
r28714 | if self.changelog.repotiprev < p2rev: | ||
Joerg Sonnenberger
|
r46729 | self.ui.warn(msg % hex(p2)) | ||
liscju
|
r28714 | return super(bundlerepository, self).setparents(p1, p2) | ||
Sune Foldager
|
r15597 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r39585 | def instance(ui, path, create, intents=None, createopts=None): | ||
Vadim Gelfer
|
r2740 | if create: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b'cannot create new bundle repository')) | ||
Matt Mackall
|
r25830 | # internal config: bundle.mainreporoot | ||
Augie Fackler
|
r43347 | parentpath = ui.config(b"bundle", b"mainreporoot") | ||
Matt Mackall
|
r16042 | if not parentpath: | ||
# try to find the correct path to the working directory repo | ||||
Matt Harbison
|
r39843 | parentpath = cmdutil.findrepo(encoding.getcwd()) | ||
Matt Mackall
|
r16042 | if parentpath is None: | ||
Augie Fackler
|
r43347 | parentpath = b'' | ||
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. | ||||
Matt Harbison
|
r39843 | cwd = encoding.getcwd() | ||
Peter Arrenbrecht
|
r5664 | if parentpath == cwd: | ||
Augie Fackler
|
r43347 | parentpath = b'' | ||
Peter Arrenbrecht
|
r5664 | else: | ||
FUJIWARA Katsunori
|
r24834 | cwd = pathutil.normasprefix(cwd) | ||
Peter Arrenbrecht
|
r5664 | if parentpath.startswith(cwd): | ||
Augie Fackler
|
r43346 | parentpath = parentpath[len(cwd) :] | ||
r47669 | u = urlutil.url(path) | |||
Brodie Rao
|
r13826 | path = u.localpath() | ||
Augie Fackler
|
r43347 | if u.scheme == b'bundle': | ||
s = path.split(b"+", 1) | ||||
Vadim Gelfer
|
r2740 | 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 | ||
Gregory Szorc
|
r39639 | |||
return makebundlerepository(ui, repopath, bundlename) | ||||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r39639 | def makebundlerepository(ui, repopath, bundlepath): | ||
"""Make a bundle repository object based on repo and bundle paths.""" | ||||
Gregory Szorc
|
r39640 | if repopath: | ||
Augie Fackler
|
r43347 | url = b'bundle:%s+%s' % (util.expandpath(repopath), bundlepath) | ||
Gregory Szorc
|
r39640 | else: | ||
Augie Fackler
|
r43347 | url = b'bundle:%s' % bundlepath | ||
Gregory Szorc
|
r39640 | |||
# Because we can't make any guarantees about the type of the base | ||||
# repository, we can't have a static class representing the bundle | ||||
# repository. We also can't make any guarantees about how to even | ||||
# call the base repository's constructor! | ||||
# | ||||
# So, our strategy is to go through ``localrepo.instance()`` to construct | ||||
# a repo instance. Then, we dynamically create a new type derived from | ||||
# both it and our ``bundlerepository`` class which overrides some | ||||
# functionality. We then change the type of the constructed repository | ||||
# to this new type and initialize the bundle-specific bits of it. | ||||
try: | ||||
Martin von Zweigbergk
|
r39980 | repo = localrepo.instance(ui, repopath, create=False) | ||
Gregory Szorc
|
r39640 | tempparent = None | ||
Matt Harbison
|
r50864 | except error.RequirementError: | ||
raise # no fallback if the backing repo is unsupported | ||||
Gregory Szorc
|
r39640 | except error.RepoError: | ||
tempparent = pycompat.mkdtemp() | ||||
try: | ||||
Martin von Zweigbergk
|
r39980 | repo = localrepo.instance(ui, tempparent, create=True) | ||
Gregory Szorc
|
r39640 | except Exception: | ||
shutil.rmtree(tempparent) | ||||
raise | ||||
Martin von Zweigbergk
|
r39980 | class derivedbundlerepository(bundlerepository, repo.__class__): | ||
Gregory Szorc
|
r39640 | pass | ||
repo.__class__ = derivedbundlerepository | ||||
bundlerepository.__init__(repo, bundlepath, url, tempparent) | ||||
return repo | ||||
Nicolas Dumazet
|
r12734 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class bundletransactionmanager: | ||
Eric Sumner
|
r23633 | def transaction(self): | ||
return None | ||||
def close(self): | ||||
raise NotImplementedError | ||||
def release(self): | ||||
raise NotImplementedError | ||||
Augie Fackler
|
r43346 | |||
def getremotechanges( | ||||
ui, repo, peer, onlyheads=None, bundlename=None, force=False | ||||
): | ||||
Augie Fackler
|
r46554 | """obtains a bundle of changes incoming from peer | ||
Peter Arrenbrecht
|
r14161 | |||
"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 | ||||
Gregory Szorc
|
r37660 | original "peer" 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 | ||
Gregory Szorc
|
r37660 | the changes; it closes both the original "peer" and the one returned | ||
Brodie Rao
|
r16683 | here. | ||
Augie Fackler
|
r46554 | """ | ||
Augie Fackler
|
r43346 | tmp = discovery.findcommonincoming(repo, peer, 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 | ||
Gregory Szorc
|
r37660 | return repo, [], peer.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 | ||
Gregory Szorc
|
r37660 | localrepo = peer.local() | ||
Sune Foldager
|
r17191 | if bundlename or not localrepo: | ||
Gregory Szorc
|
r37660 | # create a bundle (uncompressed if peer repo is not local) | ||
Nicolas Dumazet
|
r12734 | |||
Pierre-Yves David
|
r29684 | # developer config: devel.legacy.exchange | ||
Augie Fackler
|
r43347 | legexc = ui.configlist(b'devel', b'legacy.exchange') | ||
forcebundle1 = b'bundle2' not in legexc and b'bundle1' in legexc | ||||
Augie Fackler
|
r43346 | canbundle2 = ( | ||
not forcebundle1 | ||||
Augie Fackler
|
r43347 | and peer.capable(b'getbundle') | ||
and peer.capable(b'bundle2') | ||||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r26544 | if canbundle2: | ||
Gregory Szorc
|
r37661 | with peer.commandexecutor() as e: | ||
Augie Fackler
|
r43346 | b2 = e.callcommand( | ||
Augie Fackler
|
r43347 | b'getbundle', | ||
Augie Fackler
|
r43346 | { | ||
Augie Fackler
|
r43347 | b'source': b'incoming', | ||
b'common': common, | ||||
b'heads': rheads, | ||||
b'bundlecaps': exchange.caps20to10( | ||||
repo, role=b'client' | ||||
), | ||||
b'cg': True, | ||||
Augie Fackler
|
r43346 | }, | ||
).result() | ||||
Gregory Szorc
|
r37661 | |||
Augie Fackler
|
r43346 | fname = bundle = changegroup.writechunks( | ||
ui, b2._forwardchunks(), bundlename | ||||
) | ||||
Pierre-Yves David
|
r26544 | else: | ||
Augie Fackler
|
r43347 | if peer.capable(b'getbundle'): | ||
Gregory Szorc
|
r37661 | with peer.commandexecutor() as e: | ||
Augie Fackler
|
r43346 | cg = e.callcommand( | ||
Augie Fackler
|
r43347 | b'getbundle', | ||
Augie Fackler
|
r43346 | { | ||
Augie Fackler
|
r43347 | b'source': b'incoming', | ||
b'common': common, | ||||
b'heads': rheads, | ||||
Augie Fackler
|
r43346 | }, | ||
).result() | ||||
Augie Fackler
|
r43347 | elif onlyheads is None and not peer.capable(b'changegroupsubset'): | ||
Pierre-Yves David
|
r26543 | # compat with older servers when pulling all remote heads | ||
Gregory Szorc
|
r37653 | |||
Gregory Szorc
|
r37660 | with peer.commandexecutor() as e: | ||
Augie Fackler
|
r43346 | cg = e.callcommand( | ||
Augie Fackler
|
r43347 | b'changegroup', | ||
Augie Fackler
|
r46554 | { | ||
b'nodes': incoming, | ||||
b'source': b'incoming', | ||||
}, | ||||
Augie Fackler
|
r43346 | ).result() | ||
Gregory Szorc
|
r37653 | |||
Pierre-Yves David
|
r26543 | rheads = None | ||
else: | ||||
Gregory Szorc
|
r37660 | with peer.commandexecutor() as e: | ||
Augie Fackler
|
r43346 | cg = e.callcommand( | ||
Augie Fackler
|
r43347 | b'changegroupsubset', | ||
Augie Fackler
|
r43346 | { | ||
Augie Fackler
|
r43347 | b'bases': incoming, | ||
b'heads': rheads, | ||||
b'source': b'incoming', | ||||
Augie Fackler
|
r43346 | }, | ||
).result() | ||||
Gregory Szorc
|
r37653 | |||
Pierre-Yves David
|
r26543 | if localrepo: | ||
Augie Fackler
|
r43347 | bundletype = b"HG10BZ" | ||
Pierre-Yves David
|
r26543 | else: | ||
Augie Fackler
|
r43347 | bundletype = b"HG10UN" | ||
Augie Fackler
|
r43346 | fname = bundle = bundle2.writebundle(ui, cg, bundlename, 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 | ||
Augie Fackler
|
r43346 | localrepo = bundlerepo = makebundlerepository( | ||
repo.baseui, repo.root, fname | ||||
) | ||||
Gregory Szorc
|
r39639 | |||
Gregory Szorc
|
r37660 | # this repo contains local and peer now, so filter out local again | ||
Peter Arrenbrecht
|
r14161 | 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: | ||
Augie Fackler
|
r43346 | reponodes = [ctx.node() for ctx in bundlerepo[bundlerepo.firstnewrev :]] | ||
Gregory Szorc
|
r37661 | |||
with peer.commandexecutor() as e: | ||||
Augie Fackler
|
r43346 | remotephases = e.callcommand( | ||
Augie Fackler
|
r46554 | b'listkeys', | ||
{ | ||||
b'namespace': b'phases', | ||||
}, | ||||
Augie Fackler
|
r43346 | ).result() | ||
Eric Sumner
|
r23633 | |||
r49055 | pullop = exchange.pulloperation( | |||
bundlerepo, peer, path=None, heads=reponodes | ||||
) | ||||
Eric Sumner
|
r23633 | pullop.trmanager = bundletransactionmanager() | ||
exchange._pullapplyphases(pullop, remotephases) | ||||
Peter Arrenbrecht
|
r14161 | def cleanup(): | ||
if bundlerepo: | ||||
bundlerepo.close() | ||||
if bundle: | ||||
Matt Mackall
|
r21694 | os.unlink(bundle) | ||
Gregory Szorc
|
r37660 | peer.close() | ||
Peter Arrenbrecht
|
r14161 | |||
return (localrepo, csets, cleanup) | ||||