protocol.py
210 lines
| 7.0 KiB
| text/x-python
|
PythonLexer
Dirkjan Ochtman
|
r5598 | # | ||
# Copyright 21 May 2005 - (c) 2005 Jake Edge <jake@edge2.net> | ||||
# Copyright 2005-2007 Matt Mackall <mpm@selenic.com> | ||||
# | ||||
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. | ||
Dirkjan Ochtman
|
r5598 | |||
Yuya Nishihara
|
r27046 | from __future__ import absolute_import | ||
import cgi | ||||
Gregory Szorc
|
r30764 | import struct | ||
Yuya Nishihara
|
r27046 | |||
from .common import ( | ||||
HTTP_OK, | ||||
) | ||||
from .. import ( | ||||
Augie Fackler
|
r34510 | error, | ||
Augie Fackler
|
r34743 | pycompat, | ||
Yuya Nishihara
|
r27046 | util, | ||
wireproto, | ||||
) | ||||
timeless
|
r28861 | stringio = util.stringio | ||
Dirkjan Ochtman
|
r5963 | |||
timeless
|
r28883 | urlerr = util.urlerr | ||
urlreq = util.urlreq | ||||
Dirkjan Ochtman
|
r5993 | HGTYPE = 'application/mercurial-0.1' | ||
Gregory Szorc
|
r30764 | HGTYPE2 = 'application/mercurial-0.2' | ||
Andrew Pritchard
|
r15017 | HGERRTYPE = 'application/hg-error' | ||
Matt Mackall
|
r11595 | |||
Gregory Szorc
|
r30759 | def decodevaluefromheaders(req, headerprefix): | ||
Augie Fackler
|
r34745 | """Decode a long value from multiple HTTP request headers. | ||
Returns the value as a bytes, not a str. | ||||
""" | ||||
Gregory Szorc
|
r30759 | chunks = [] | ||
i = 1 | ||||
Augie Fackler
|
r34745 | prefix = headerprefix.upper().replace(r'-', r'_') | ||
Gregory Szorc
|
r30759 | while True: | ||
Augie Fackler
|
r34745 | v = req.env.get(r'HTTP_%s_%d' % (prefix, i)) | ||
Gregory Szorc
|
r30759 | if v is None: | ||
break | ||||
Augie Fackler
|
r34745 | chunks.append(pycompat.bytesurl(v)) | ||
Gregory Szorc
|
r30759 | i += 1 | ||
return ''.join(chunks) | ||||
Pierre-Yves David
|
r20903 | class webproto(wireproto.abstractserverproto): | ||
Idan Kamara
|
r14614 | def __init__(self, req, ui): | ||
Matt Mackall
|
r11595 | self.req = req | ||
self.response = '' | ||||
Idan Kamara
|
r14614 | self.ui = ui | ||
Gregory Szorc
|
r30562 | self.name = 'http' | ||
Matt Mackall
|
r11595 | def getargs(self, args): | ||
Steven Brown
|
r14093 | knownargs = self._args() | ||
Matt Mackall
|
r11595 | data = {} | ||
keys = args.split() | ||||
for k in keys: | ||||
if k == '*': | ||||
star = {} | ||||
Steven Brown
|
r14093 | for key in knownargs.keys(): | ||
Peter Arrenbrecht
|
r13721 | if key != 'cmd' and key not in keys: | ||
Steven Brown
|
r14093 | star[key] = knownargs[key][0] | ||
Matt Mackall
|
r11595 | data['*'] = star | ||
else: | ||||
Steven Brown
|
r14093 | data[k] = knownargs[k][0] | ||
Matt Mackall
|
r11595 | return [data[k] for k in keys] | ||
Steven Brown
|
r14093 | def _args(self): | ||
args = self.req.form.copy() | ||||
Augie Fackler
|
r34743 | if pycompat.ispy3: | ||
args = {k.encode('ascii'): [v.encode('ascii') for v in vs] | ||||
for k, vs in args.items()} | ||||
Augie Fackler
|
r34744 | postlen = int(self.req.env.get(r'HTTP_X_HGARGS_POST', 0)) | ||
Augie Fackler
|
r28530 | if postlen: | ||
args.update(cgi.parse_qs( | ||||
self.req.read(postlen), keep_blank_values=True)) | ||||
return args | ||||
Gregory Szorc
|
r30759 | |||
Augie Fackler
|
r34744 | argvalue = decodevaluefromheaders(self.req, r'X-HgArg') | ||
Gregory Szorc
|
r30759 | args.update(cgi.parse_qs(argvalue, keep_blank_values=True)) | ||
Steven Brown
|
r14093 | return args | ||
Dirkjan Ochtman
|
r11621 | def getfile(self, fp): | ||
Augie Fackler
|
r34741 | length = int(self.req.env[r'CONTENT_LENGTH']) | ||
Augie Fackler
|
r33820 | # If httppostargs is used, we need to read Content-Length | ||
# minus the amount that was consumed by args. | ||||
Augie Fackler
|
r34741 | length -= int(self.req.env.get(r'HTTP_X_HGARGS_POST', 0)) | ||
Dirkjan Ochtman
|
r11621 | for s in util.filechunkiter(self.req, limit=length): | ||
fp.write(s) | ||||
def redirect(self): | ||||
Idan Kamara
|
r14614 | self.oldio = self.ui.fout, self.ui.ferr | ||
timeless
|
r28861 | self.ui.ferr = self.ui.fout = stringio() | ||
Idan Kamara
|
r14614 | def restore(self): | ||
val = self.ui.fout.getvalue() | ||||
self.ui.ferr, self.ui.fout = self.oldio | ||||
return val | ||||
Gregory Szorc
|
r30206 | |||
Matt Mackall
|
r11595 | def _client(self): | ||
return 'remote:%s:%s:%s' % ( | ||||
self.req.env.get('wsgi.url_scheme') or 'http', | ||||
timeless
|
r28883 | urlreq.quote(self.req.env.get('REMOTE_HOST', '')), | ||
urlreq.quote(self.req.env.get('REMOTE_USER', ''))) | ||||
Matt Mackall
|
r11595 | |||
Gregory Szorc
|
r30764 | def responsetype(self, v1compressible=False): | ||
"""Determine the appropriate response type and compression settings. | ||||
The ``v1compressible`` argument states whether the response with | ||||
application/mercurial-0.1 media types should be zlib compressed. | ||||
Returns a tuple of (mediatype, compengine, engineopts). | ||||
""" | ||||
# For now, if it isn't compressible in the old world, it's never | ||||
# compressible. We can change this to send uncompressed 0.2 payloads | ||||
# later. | ||||
if not v1compressible: | ||||
return HGTYPE, None, None | ||||
# Determine the response media type and compression engine based | ||||
# on the request parameters. | ||||
Augie Fackler
|
r34744 | protocaps = decodevaluefromheaders(self.req, r'X-HgProto').split(' ') | ||
Gregory Szorc
|
r30764 | |||
if '0.2' in protocaps: | ||||
# Default as defined by wire protocol spec. | ||||
compformats = ['zlib', 'none'] | ||||
for cap in protocaps: | ||||
if cap.startswith('comp='): | ||||
compformats = cap[5:].split(',') | ||||
break | ||||
# Now find an agreed upon compression format. | ||||
for engine in wireproto.supportedcompengines(self.ui, self, | ||||
util.SERVERROLE): | ||||
if engine.wireprotosupport().name in compformats: | ||||
opts = {} | ||||
level = self.ui.configint('server', | ||||
'%slevel' % engine.name()) | ||||
if level is not None: | ||||
opts['level'] = level | ||||
return HGTYPE2, engine, opts | ||||
# No mutually supported compression format. Fall back to the | ||||
# legacy protocol. | ||||
# Don't allow untrusted settings because disabling compression or | ||||
# setting a very high compression level could lead to flooding | ||||
# the server's network or CPU. | ||||
r33225 | opts = {'level': self.ui.configint('server', 'zliblevel')} | |||
Gregory Szorc
|
r30764 | return HGTYPE, util.compengines['zlib'], opts | ||
Matt Mackall
|
r11595 | def iscmd(cmd): | ||
return cmd in wireproto.commands | ||||
def call(repo, req, cmd): | ||||
Idan Kamara
|
r14614 | p = webproto(req, repo.ui) | ||
Gregory Szorc
|
r30764 | |||
def genversion2(gen, compress, engine, engineopts): | ||||
# application/mercurial-0.2 always sends a payload header | ||||
# identifying the compression engine. | ||||
name = engine.wireprotosupport().name | ||||
assert 0 < len(name) < 256 | ||||
yield struct.pack('B', len(name)) | ||||
yield name | ||||
if compress: | ||||
for chunk in engine.compressstream(gen, opts=engineopts): | ||||
yield chunk | ||||
else: | ||||
for chunk in gen: | ||||
yield chunk | ||||
Dirkjan Ochtman
|
r11625 | rsp = wireproto.dispatch(repo, p, cmd) | ||
Augie Fackler
|
r34511 | if isinstance(rsp, bytes): | ||
Mads Kiilerich
|
r18352 | req.respond(HTTP_OK, HGTYPE, body=rsp) | ||
return [] | ||||
Dirkjan Ochtman
|
r11626 | elif isinstance(rsp, wireproto.streamres): | ||
Gregory Szorc
|
r30466 | if rsp.reader: | ||
gen = iter(lambda: rsp.reader.read(32768), '') | ||||
else: | ||||
gen = rsp.gen | ||||
Gregory Szorc
|
r30764 | # This code for compression should not be streamres specific. It | ||
# is here because we only compress streamres at the moment. | ||||
mediatype, engine, engineopts = p.responsetype(rsp.v1compressible) | ||||
Gregory Szorc
|
r30466 | |||
Gregory Szorc
|
r30764 | if mediatype == HGTYPE and rsp.v1compressible: | ||
gen = engine.compressstream(gen, engineopts) | ||||
elif mediatype == HGTYPE2: | ||||
gen = genversion2(gen, rsp.v1compressible, engine, engineopts) | ||||
req.respond(HTTP_OK, mediatype) | ||||
Gregory Szorc
|
r30466 | return gen | ||
Dirkjan Ochtman
|
r11626 | elif isinstance(rsp, wireproto.pushres): | ||
Idan Kamara
|
r14614 | val = p.restore() | ||
Mads Kiilerich
|
r18346 | rsp = '%d\n%s' % (rsp.res, val) | ||
Mads Kiilerich
|
r18352 | req.respond(HTTP_OK, HGTYPE, body=rsp) | ||
return [] | ||||
Benoit Boissinot
|
r12703 | elif isinstance(rsp, wireproto.pusherr): | ||
Benoit Boissinot
|
r12704 | # drain the incoming bundle | ||
req.drain() | ||||
Idan Kamara
|
r14614 | p.restore() | ||
Benoit Boissinot
|
r12703 | rsp = '0\n%s\n' % rsp.res | ||
Mads Kiilerich
|
r18352 | req.respond(HTTP_OK, HGTYPE, body=rsp) | ||
return [] | ||||
Andrew Pritchard
|
r15017 | elif isinstance(rsp, wireproto.ooberror): | ||
rsp = rsp.message | ||||
Mads Kiilerich
|
r18352 | req.respond(HTTP_OK, HGERRTYPE, body=rsp) | ||
return [] | ||||
Augie Fackler
|
r34510 | raise error.ProgrammingError('hgweb.protocol internal failure', rsp) | ||