##// END OF EJS Templates
hgweb: tweak zlib chunking behavior...
Gregory Szorc -
r29792:58467204 default
parent child Browse files
Show More
@@ -1,120 +1,124
1 #
1 #
2 # Copyright 21 May 2005 - (c) 2005 Jake Edge <jake@edge2.net>
2 # Copyright 21 May 2005 - (c) 2005 Jake Edge <jake@edge2.net>
3 # Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
3 # Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
4 #
4 #
5 # This software may be used and distributed according to the terms of the
5 # This software may be used and distributed according to the terms of the
6 # GNU General Public License version 2 or any later version.
6 # GNU General Public License version 2 or any later version.
7
7
8 from __future__ import absolute_import
8 from __future__ import absolute_import
9
9
10 import cgi
10 import cgi
11 import zlib
11 import zlib
12
12
13 from .common import (
13 from .common import (
14 HTTP_OK,
14 HTTP_OK,
15 )
15 )
16
16
17 from .. import (
17 from .. import (
18 util,
18 util,
19 wireproto,
19 wireproto,
20 )
20 )
21 stringio = util.stringio
21 stringio = util.stringio
22
22
23 urlerr = util.urlerr
23 urlerr = util.urlerr
24 urlreq = util.urlreq
24 urlreq = util.urlreq
25
25
26 HGTYPE = 'application/mercurial-0.1'
26 HGTYPE = 'application/mercurial-0.1'
27 HGERRTYPE = 'application/hg-error'
27 HGERRTYPE = 'application/hg-error'
28
28
29 class webproto(wireproto.abstractserverproto):
29 class webproto(wireproto.abstractserverproto):
30 def __init__(self, req, ui):
30 def __init__(self, req, ui):
31 self.req = req
31 self.req = req
32 self.response = ''
32 self.response = ''
33 self.ui = ui
33 self.ui = ui
34 def getargs(self, args):
34 def getargs(self, args):
35 knownargs = self._args()
35 knownargs = self._args()
36 data = {}
36 data = {}
37 keys = args.split()
37 keys = args.split()
38 for k in keys:
38 for k in keys:
39 if k == '*':
39 if k == '*':
40 star = {}
40 star = {}
41 for key in knownargs.keys():
41 for key in knownargs.keys():
42 if key != 'cmd' and key not in keys:
42 if key != 'cmd' and key not in keys:
43 star[key] = knownargs[key][0]
43 star[key] = knownargs[key][0]
44 data['*'] = star
44 data['*'] = star
45 else:
45 else:
46 data[k] = knownargs[k][0]
46 data[k] = knownargs[k][0]
47 return [data[k] for k in keys]
47 return [data[k] for k in keys]
48 def _args(self):
48 def _args(self):
49 args = self.req.form.copy()
49 args = self.req.form.copy()
50 postlen = int(self.req.env.get('HTTP_X_HGARGS_POST', 0))
50 postlen = int(self.req.env.get('HTTP_X_HGARGS_POST', 0))
51 if postlen:
51 if postlen:
52 args.update(cgi.parse_qs(
52 args.update(cgi.parse_qs(
53 self.req.read(postlen), keep_blank_values=True))
53 self.req.read(postlen), keep_blank_values=True))
54 return args
54 return args
55 chunks = []
55 chunks = []
56 i = 1
56 i = 1
57 while True:
57 while True:
58 h = self.req.env.get('HTTP_X_HGARG_' + str(i))
58 h = self.req.env.get('HTTP_X_HGARG_' + str(i))
59 if h is None:
59 if h is None:
60 break
60 break
61 chunks += [h]
61 chunks += [h]
62 i += 1
62 i += 1
63 args.update(cgi.parse_qs(''.join(chunks), keep_blank_values=True))
63 args.update(cgi.parse_qs(''.join(chunks), keep_blank_values=True))
64 return args
64 return args
65 def getfile(self, fp):
65 def getfile(self, fp):
66 length = int(self.req.env['CONTENT_LENGTH'])
66 length = int(self.req.env['CONTENT_LENGTH'])
67 for s in util.filechunkiter(self.req, limit=length):
67 for s in util.filechunkiter(self.req, limit=length):
68 fp.write(s)
68 fp.write(s)
69 def redirect(self):
69 def redirect(self):
70 self.oldio = self.ui.fout, self.ui.ferr
70 self.oldio = self.ui.fout, self.ui.ferr
71 self.ui.ferr = self.ui.fout = stringio()
71 self.ui.ferr = self.ui.fout = stringio()
72 def restore(self):
72 def restore(self):
73 val = self.ui.fout.getvalue()
73 val = self.ui.fout.getvalue()
74 self.ui.ferr, self.ui.fout = self.oldio
74 self.ui.ferr, self.ui.fout = self.oldio
75 return val
75 return val
76 def groupchunks(self, cg):
76 def groupchunks(self, cg):
77 # Don't allow untrusted settings because disabling compression or
77 # Don't allow untrusted settings because disabling compression or
78 # setting a very high compression level could lead to flooding
78 # setting a very high compression level could lead to flooding
79 # the server's network or CPU.
79 # the server's network or CPU.
80 z = zlib.compressobj(self.ui.configint('server', 'zliblevel', -1))
80 z = zlib.compressobj(self.ui.configint('server', 'zliblevel', -1))
81 while True:
81 while True:
82 chunk = cg.read(4096)
82 chunk = cg.read(32768)
83 if not chunk:
83 if not chunk:
84 break
84 break
85 yield z.compress(chunk)
85 data = z.compress(chunk)
86 # Not all calls to compress() emit data. It is cheaper to inspect
87 # that here than to send it via the generator.
88 if data:
89 yield data
86 yield z.flush()
90 yield z.flush()
87 def _client(self):
91 def _client(self):
88 return 'remote:%s:%s:%s' % (
92 return 'remote:%s:%s:%s' % (
89 self.req.env.get('wsgi.url_scheme') or 'http',
93 self.req.env.get('wsgi.url_scheme') or 'http',
90 urlreq.quote(self.req.env.get('REMOTE_HOST', '')),
94 urlreq.quote(self.req.env.get('REMOTE_HOST', '')),
91 urlreq.quote(self.req.env.get('REMOTE_USER', '')))
95 urlreq.quote(self.req.env.get('REMOTE_USER', '')))
92
96
93 def iscmd(cmd):
97 def iscmd(cmd):
94 return cmd in wireproto.commands
98 return cmd in wireproto.commands
95
99
96 def call(repo, req, cmd):
100 def call(repo, req, cmd):
97 p = webproto(req, repo.ui)
101 p = webproto(req, repo.ui)
98 rsp = wireproto.dispatch(repo, p, cmd)
102 rsp = wireproto.dispatch(repo, p, cmd)
99 if isinstance(rsp, str):
103 if isinstance(rsp, str):
100 req.respond(HTTP_OK, HGTYPE, body=rsp)
104 req.respond(HTTP_OK, HGTYPE, body=rsp)
101 return []
105 return []
102 elif isinstance(rsp, wireproto.streamres):
106 elif isinstance(rsp, wireproto.streamres):
103 req.respond(HTTP_OK, HGTYPE)
107 req.respond(HTTP_OK, HGTYPE)
104 return rsp.gen
108 return rsp.gen
105 elif isinstance(rsp, wireproto.pushres):
109 elif isinstance(rsp, wireproto.pushres):
106 val = p.restore()
110 val = p.restore()
107 rsp = '%d\n%s' % (rsp.res, val)
111 rsp = '%d\n%s' % (rsp.res, val)
108 req.respond(HTTP_OK, HGTYPE, body=rsp)
112 req.respond(HTTP_OK, HGTYPE, body=rsp)
109 return []
113 return []
110 elif isinstance(rsp, wireproto.pusherr):
114 elif isinstance(rsp, wireproto.pusherr):
111 # drain the incoming bundle
115 # drain the incoming bundle
112 req.drain()
116 req.drain()
113 p.restore()
117 p.restore()
114 rsp = '0\n%s\n' % rsp.res
118 rsp = '0\n%s\n' % rsp.res
115 req.respond(HTTP_OK, HGTYPE, body=rsp)
119 req.respond(HTTP_OK, HGTYPE, body=rsp)
116 return []
120 return []
117 elif isinstance(rsp, wireproto.ooberror):
121 elif isinstance(rsp, wireproto.ooberror):
118 rsp = rsp.message
122 rsp = rsp.message
119 req.respond(HTTP_OK, HGERRTYPE, body=rsp)
123 req.respond(HTTP_OK, HGERRTYPE, body=rsp)
120 return []
124 return []
General Comments 0
You need to be logged in to leave comments. Login now