statichttprepo.py
166 lines
| 5.3 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 | |||
Matt Mackall
|
r3891 | from i18n import _ | ||
Sean Farley
|
r23558 | import changelog, byterange, url, error, namespaces | ||
Dan Villiom Podlaski Christiansen
|
r14091 | import localrepo, manifest, util, scmutil, store | ||
Bryan O'Sullivan
|
r18915 | import urllib, urllib2, errno, os | ||
Bryan O'Sullivan
|
r1325 | |||
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 | ||
Benoit Boissinot
|
r7274 | def seek(self, pos): | ||
self.pos = pos | ||||
def read(self, bytes=None): | ||||
req = urllib2.Request(self.url) | ||||
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
|
r14962 | # Python 2.6+ defines a getcode() function, and 2.4 and | ||
# 2.5 appear to always have an undocumented code attribute | ||||
# set. If we can't read either of those, fall back to 206 | ||||
# and hope for the best. | ||||
code = getattr(f, 'getcode', lambda : getattr(f, 'code', 206))() | ||||
Thomas Arendsen Hein
|
r1821 | except urllib2.HTTPError, inst: | ||
Dirkjan Ochtman
|
r6028 | num = inst.code == 404 and errno.ENOENT or None | ||
raise IOError(num, inst) | ||||
Bryan O'Sullivan
|
r1325 | except urllib2.URLError, 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 | |||
def build_opener(ui, authinfo): | ||||
# urllib cannot handle URLs with embedded user or passwd | ||||
urlopener = url.opener(ui, authinfo) | ||||
urlopener.add_handler(byterange.HTTPRangeHandler()) | ||||
FUJIWARA Katsunori
|
r17649 | class statichttpvfs(scmutil.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') | ||
Dan Villiom Podlaski Christiansen
|
r14091 | f = "/".join((self.base, urllib.quote(path))) | ||
Benoit Boissinot
|
r7274 | return httprangereader(f, urlopener) | ||
FUJIWARA Katsunori
|
r17725 | def join(self, path): | ||
if path: | ||||
return os.path.join(self.base, path) | ||||
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 | |||
opener = build_opener(ui, authinfo) | ||||
mpm@selenic.com
|
r1101 | self.opener = opener(self.path) | ||
FUJIWARA Katsunori
|
r17156 | self.vfs = self.opener | ||
Pierre-Yves David
|
r15922 | self._phasedefaults = [] | ||
Dirkjan Ochtman
|
r6028 | |||
Ryan McElroy
|
r23561 | self.names = namespaces.namespaces() | ||
Sean Farley
|
r23558 | |||
Benoit Boissinot
|
r3851 | try: | ||
Angel Ezquerra
|
r23877 | requirements = scmutil.readrequires(self.vfs, self.supported) | ||
Dirkjan Ochtman
|
r6028 | except IOError, 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() | ||||
Thomas Arendsen Hein
|
r7178 | except IOError, inst: | ||
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 | ||||
Adrian Buehlmann
|
r13426 | self.store = store.store(requirements, self.path, opener) | ||
Matt Mackall
|
r6897 | self.spath = self.store.path | ||
Angel Ezquerra
|
r23878 | self.svfs = self.store.opener | ||
self.sopener = self.svfs | ||||
Matt Mackall
|
r6897 | self.sjoin = self.store.join | ||
Idan Kamara
|
r16115 | self._filecache = {} | ||
Peter Arrenbrecht
|
r17192 | self.requirements = requirements | ||
Benoit Boissinot
|
r3851 | |||
Angel Ezquerra
|
r23878 | self.manifest = manifest.manifest(self.svfs) | ||
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 = {} | ||
Benoit Boissinot
|
r1598 | self.encodepats = None | ||
self.decodepats = 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) | ||||
Martin Geisler
|
r7005 | def lock(self, wait=True): | ||
raise util.Abort(_('cannot lock static-http repository')) | ||||
Vadim Gelfer
|
r2740 | def instance(ui, path, create): | ||
if create: | ||||
raise util.Abort(_('cannot create new static-http repository')) | ||||
Thomas Arendsen Hein
|
r4853 | return statichttprepository(ui, path[7:]) | ||