Show More
statichttprepo.py
221 lines
| 6.6 KiB
| text/x-python
|
PythonLexer
/ mercurial / statichttprepo.py
mpm@selenic.com
|
r1101 | # statichttprepo.py - simple http repository class for mercurial | ||
# | ||||
# This provides read-only repo access to repositories exported via static http | ||||
# | ||||
Thomas Arendsen Hein
|
r4635 | # Copyright 2005-2007 Matt Mackall <mpm@selenic.com> | ||
mpm@selenic.com
|
r1101 | # | ||
Martin Geisler
|
r8225 | # 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. | ||
mpm@selenic.com
|
r1101 | |||
Gregory Szorc
|
r25978 | from __future__ import absolute_import | ||
import errno | ||||
from .i18n import _ | ||||
from . import ( | ||||
changelog, | ||||
error, | ||||
localrepo, | ||||
manifest, | ||||
namespaces, | ||||
Yuya Nishihara
|
r34944 | pathutil, | ||
Gregory Szorc
|
r25978 | scmutil, | ||
store, | ||||
url, | ||||
util, | ||||
Pierre-Yves David
|
r31241 | vfs as vfsmod, | ||
Gregory Szorc
|
r25978 | ) | ||
Bryan O'Sullivan
|
r1325 | |||
timeless
|
r28883 | urlerr = util.urlerr | ||
urlreq = util.urlreq | ||||
Benoit Boissinot
|
r7274 | class httprangereader(object): | ||
def __init__(self, url, opener): | ||||
# we assume opener has HTTPRangeHandler | ||||
self.url = url | ||||
self.pos = 0 | ||||
self.opener = opener | ||||
Nicolas Dumazet
|
r11066 | self.name = url | ||
Gregory Szorc
|
r27705 | |||
def __enter__(self): | ||||
return self | ||||
def __exit__(self, exc_type, exc_value, traceback): | ||||
self.close() | ||||
Benoit Boissinot
|
r7274 | def seek(self, pos): | ||
self.pos = pos | ||||
def read(self, bytes=None): | ||||
timeless
|
r28883 | req = urlreq.request(self.url) | ||
Benoit Boissinot
|
r7274 | end = '' | ||
if bytes: | ||||
end = self.pos + bytes - 1 | ||||
Alexander Boyd
|
r16882 | if self.pos or end: | ||
req.add_header('Range', 'bytes=%d-%s' % (self.pos, end)) | ||||
Benoit Boissinot
|
r7274 | |||
Bryan O'Sullivan
|
r1325 | try: | ||
Benoit Boissinot
|
r7274 | f = self.opener.open(req) | ||
data = f.read() | ||||
Augie Fackler
|
r25196 | code = f.code | ||
timeless
|
r28883 | except urlerr.httperror as inst: | ||
Dirkjan Ochtman
|
r6028 | num = inst.code == 404 and errno.ENOENT or None | ||
raise IOError(num, inst) | ||||
timeless
|
r28883 | except urlerr.urlerror as inst: | ||
Thomas Arendsen Hein
|
r1821 | raise IOError(None, inst.reason[1]) | ||
mpm@selenic.com
|
r1101 | |||
Patrick Mezard
|
r8612 | if code == 200: | ||
# HTTPRangeHandler does nothing if remote does not support | ||||
# Range headers and returns the full entity. Let's slice it. | ||||
if bytes: | ||||
data = data[self.pos:self.pos + bytes] | ||||
else: | ||||
data = data[self.pos:] | ||||
elif bytes: | ||||
Benoit Boissinot
|
r7274 | data = data[:bytes] | ||
Patrick Mezard
|
r8612 | self.pos += len(data) | ||
Benoit Boissinot
|
r7274 | return data | ||
Siddharth Agarwal
|
r20055 | def readlines(self): | ||
return self.read().splitlines(True) | ||||
Nicolas Dumazet
|
r11066 | def __iter__(self): | ||
Siddharth Agarwal
|
r20055 | return iter(self.readlines()) | ||
Nicolas Dumazet
|
r11066 | def close(self): | ||
pass | ||||
Benoit Boissinot
|
r7274 | |||
Augie Fackler
|
r36443 | # _RangeError and _HTTPRangeHandler were originally in byterange.py, | ||
# which was itself extracted from urlgrabber. See the last version of | ||||
# byterange.py from history if you need more information. | ||||
class _RangeError(IOError): | ||||
"""Error raised when an unsatisfiable range is requested.""" | ||||
class _HTTPRangeHandler(urlreq.basehandler): | ||||
"""Handler that enables HTTP Range headers. | ||||
This was extremely simple. The Range header is a HTTP feature to | ||||
begin with so all this class does is tell urllib2 that the | ||||
"206 Partial Content" response from the HTTP server is what we | ||||
expected. | ||||
""" | ||||
def http_error_206(self, req, fp, code, msg, hdrs): | ||||
# 206 Partial Content Response | ||||
r = urlreq.addinfourl(fp, hdrs, req.get_full_url()) | ||||
r.code = code | ||||
r.msg = msg | ||||
return r | ||||
def http_error_416(self, req, fp, code, msg, hdrs): | ||||
# HTTP's Range Not Satisfiable error | ||||
raise _RangeError('Requested Range Not Satisfiable') | ||||
Benoit Boissinot
|
r7274 | def build_opener(ui, authinfo): | ||
# urllib cannot handle URLs with embedded user or passwd | ||||
urlopener = url.opener(ui, authinfo) | ||||
Augie Fackler
|
r36443 | urlopener.add_handler(_HTTPRangeHandler()) | ||
Benoit Boissinot
|
r7274 | |||
Pierre-Yves David
|
r31241 | class statichttpvfs(vfsmod.abstractvfs): | ||
Dan Villiom Podlaski Christiansen
|
r14091 | def __init__(self, base): | ||
self.base = base | ||||
Mads Kiilerich
|
r23552 | def __call__(self, path, mode='r', *args, **kw): | ||
Adrian Buehlmann
|
r13533 | if mode not in ('r', 'rb'): | ||
Nicolas Dumazet
|
r11066 | raise IOError('Permission denied') | ||
timeless
|
r28883 | f = "/".join((self.base, urlreq.quote(path))) | ||
Benoit Boissinot
|
r7274 | return httprangereader(f, urlopener) | ||
FUJIWARA Katsunori
|
r17725 | def join(self, path): | ||
if path: | ||||
Yuya Nishihara
|
r34944 | return pathutil.join(self.base, path) | ||
FUJIWARA Katsunori
|
r17725 | else: | ||
return self.base | ||||
FUJIWARA Katsunori
|
r17649 | return statichttpvfs | ||
mpm@selenic.com
|
r1101 | |||
Peter Arrenbrecht
|
r17192 | class statichttppeer(localrepo.localpeer): | ||
def local(self): | ||||
return None | ||||
Sune Foldager
|
r17193 | def canpush(self): | ||
return False | ||||
Peter Arrenbrecht
|
r17192 | |||
mpm@selenic.com
|
r1101 | class statichttprepository(localrepo.localrepository): | ||
FUJIWARA Katsunori
|
r19778 | supported = localrepo.localrepository._basesupported | ||
mpm@selenic.com
|
r1101 | def __init__(self, ui, path): | ||
Vadim Gelfer
|
r2673 | self._url = path | ||
mpm@selenic.com
|
r1101 | self.ui = ui | ||
Benoit Boissinot
|
r3853 | |||
Nicolas Dumazet
|
r11066 | self.root = path | ||
Brodie Rao
|
r14076 | u = util.url(path.rstrip('/') + "/.hg") | ||
Brodie Rao
|
r13819 | self.path, authinfo = u.authinfo() | ||
Benoit Boissinot
|
r7274 | |||
Pierre-Yves David
|
r31147 | vfsclass = build_opener(ui, authinfo) | ||
self.vfs = vfsclass(self.path) | ||||
Boris Feld
|
r33533 | self.cachevfs = vfsclass(self.vfs.join('cache')) | ||
Pierre-Yves David
|
r15922 | self._phasedefaults = [] | ||
Dirkjan Ochtman
|
r6028 | |||
Ryan McElroy
|
r23561 | self.names = namespaces.namespaces() | ||
Gregory Szorc
|
r32730 | self.filtername = None | ||
Sean Farley
|
r23558 | |||
Benoit Boissinot
|
r3851 | try: | ||
Angel Ezquerra
|
r23877 | requirements = scmutil.readrequires(self.vfs, self.supported) | ||
Gregory Szorc
|
r25660 | except IOError as inst: | ||
Thomas Arendsen Hein
|
r7178 | if inst.errno != errno.ENOENT: | ||
raise | ||||
Adrian Buehlmann
|
r14482 | requirements = set() | ||
Thomas Arendsen Hein
|
r7178 | # check if it is a non-empty old-style repository | ||
try: | ||||
Angel Ezquerra
|
r23877 | fp = self.vfs("00changelog.i") | ||
Dan Villiom Podlaski Christiansen
|
r13400 | fp.read(1) | ||
fp.close() | ||||
Gregory Szorc
|
r25660 | except IOError as inst: | ||
Thomas Arendsen Hein
|
r7178 | if inst.errno != errno.ENOENT: | ||
raise | ||||
# we do not care about empty old-style repositories here | ||||
Dirkjan Ochtman
|
r6028 | msg = _("'%s' does not appear to be an hg repository") % path | ||
Matt Mackall
|
r7637 | raise error.RepoError(msg) | ||
Benoit Boissinot
|
r3851 | |||
# setup store | ||||
Pierre-Yves David
|
r31147 | self.store = store.store(requirements, self.path, vfsclass) | ||
Matt Mackall
|
r6897 | self.spath = self.store.path | ||
Angel Ezquerra
|
r23878 | self.svfs = self.store.opener | ||
Matt Mackall
|
r6897 | self.sjoin = self.store.join | ||
Idan Kamara
|
r16115 | self._filecache = {} | ||
Peter Arrenbrecht
|
r17192 | self.requirements = requirements | ||
Benoit Boissinot
|
r3851 | |||
Durham Goode
|
r30219 | self.manifestlog = manifest.manifestlog(self.svfs, self) | ||
Angel Ezquerra
|
r23878 | self.changelog = changelog.changelog(self.svfs) | ||
Greg Ward
|
r9146 | self._tags = None | ||
mpm@selenic.com
|
r1101 | self.nodetagscache = None | ||
Pierre-Yves David
|
r18189 | self._branchcaches = {} | ||
Durham Goode
|
r24373 | self._revbranchcache = None | ||
Benoit Boissinot
|
r1598 | self.encodepats = None | ||
self.decodepats = None | ||||
Durham Goode
|
r24377 | self._transref = None | ||
Peter Arrenbrecht
|
r17192 | |||
def _restrictcapabilities(self, caps): | ||||
Pierre-Yves David
|
r20962 | caps = super(statichttprepository, self)._restrictcapabilities(caps) | ||
Peter Arrenbrecht
|
r17192 | return caps.difference(["pushkey"]) | ||
mpm@selenic.com
|
r1101 | |||
Vadim Gelfer
|
r2673 | def url(self): | ||
Matt Mackall
|
r7211 | return self._url | ||
Vadim Gelfer
|
r2673 | |||
mpm@selenic.com
|
r1101 | def local(self): | ||
return False | ||||
Vadim Gelfer
|
r2740 | |||
Peter Arrenbrecht
|
r17192 | def peer(self): | ||
return statichttppeer(self) | ||||
Gregory Szorc
|
r33605 | def wlock(self, wait=True): | ||
Yuya Nishihara
|
r36675 | raise error.LockUnavailable(0, _('lock not available'), 'lock', | ||
Gregory Szorc
|
r33605 | _('cannot lock static-http repository')) | ||
Martin Geisler
|
r7005 | def lock(self, wait=True): | ||
Pierre-Yves David
|
r26587 | raise error.Abort(_('cannot lock static-http repository')) | ||
Martin Geisler
|
r7005 | |||
Pierre-Yves David
|
r29738 | def _writecaches(self): | ||
pass # statichttprepository are read only | ||||
Gregory Szorc
|
r39585 | def instance(ui, path, create, intents=None, createopts=None): | ||
Vadim Gelfer
|
r2740 | if create: | ||
Pierre-Yves David
|
r26587 | raise error.Abort(_('cannot create new static-http repository')) | ||
Thomas Arendsen Hein
|
r4853 | return statichttprepository(ui, path[7:]) | ||