protocol.py
124 lines
| 3.8 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 | ||||
import zlib | ||||
from .common import ( | ||||
HTTP_OK, | ||||
) | ||||
from .. import ( | ||||
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' | ||
Andrew Pritchard
|
r15017 | HGERRTYPE = 'application/hg-error' | ||
Matt Mackall
|
r11595 | |||
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 | ||
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
|
r28530 | postlen = int(self.req.env.get('HTTP_X_HGARGS_POST', 0)) | ||
if postlen: | ||||
args.update(cgi.parse_qs( | ||||
self.req.read(postlen), keep_blank_values=True)) | ||||
return args | ||||
Steven Brown
|
r14093 | chunks = [] | ||
Matt Mackall
|
r14094 | i = 1 | ||
Martin Geisler
|
r14494 | while True: | ||
Matt Mackall
|
r14094 | h = self.req.env.get('HTTP_X_HGARG_' + str(i)) | ||
Steven Brown
|
r14093 | if h is None: | ||
break | ||||
chunks += [h] | ||||
Matt Mackall
|
r14094 | i += 1 | ||
Steven Brown
|
r14093 | args.update(cgi.parse_qs(''.join(chunks), keep_blank_values=True)) | ||
return args | ||||
Dirkjan Ochtman
|
r11621 | def getfile(self, fp): | ||
length = int(self.req.env['CONTENT_LENGTH']) | ||||
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 | ||||
Dirkjan Ochtman
|
r11623 | def groupchunks(self, cg): | ||
Gregory Szorc
|
r29788 | # 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. | ||||
Gregory Szorc
|
r29748 | z = zlib.compressobj(self.ui.configint('server', 'zliblevel', -1)) | ||
Martin Geisler
|
r14494 | while True: | ||
Gregory Szorc
|
r29792 | chunk = cg.read(32768) | ||
Matt Mackall
|
r11595 | if not chunk: | ||
break | ||||
Gregory Szorc
|
r29792 | data = z.compress(chunk) | ||
# Not all calls to compress() emit data. It is cheaper to inspect | ||||
# that here than to send it via the generator. | ||||
if data: | ||||
yield data | ||||
Dirkjan Ochtman
|
r11623 | yield z.flush() | ||
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 | |||
def iscmd(cmd): | ||||
return cmd in wireproto.commands | ||||
def call(repo, req, cmd): | ||||
Idan Kamara
|
r14614 | p = webproto(req, repo.ui) | ||
Dirkjan Ochtman
|
r11625 | rsp = wireproto.dispatch(repo, p, cmd) | ||
Dirkjan Ochtman
|
r11626 | if isinstance(rsp, str): | ||
Mads Kiilerich
|
r18352 | req.respond(HTTP_OK, HGTYPE, body=rsp) | ||
return [] | ||||
Dirkjan Ochtman
|
r11626 | elif isinstance(rsp, wireproto.streamres): | ||
req.respond(HTTP_OK, HGTYPE) | ||||
return rsp.gen | ||||
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 [] | ||||