##// END OF EJS Templates
protocol: use generators instead of req.write() for hgweb stream responses
Dirkjan Ochtman -
r11626:2f8adc60 default
parent child Browse files
Show More
@@ -1,78 +1,68
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 import cStringIO, zlib, sys, urllib
8 import cStringIO, zlib, sys, urllib
9 from mercurial import util, wireproto
9 from mercurial import util, wireproto
10 from common import HTTP_OK
10 from common import HTTP_OK
11
11
12 HGTYPE = 'application/mercurial-0.1'
12 HGTYPE = 'application/mercurial-0.1'
13
13
14 class webproto(object):
14 class webproto(object):
15 def __init__(self, req):
15 def __init__(self, req):
16 self.req = req
16 self.req = req
17 self.response = ''
17 self.response = ''
18 def getargs(self, args):
18 def getargs(self, args):
19 data = {}
19 data = {}
20 keys = args.split()
20 keys = args.split()
21 for k in keys:
21 for k in keys:
22 if k == '*':
22 if k == '*':
23 star = {}
23 star = {}
24 for key in self.req.form.keys():
24 for key in self.req.form.keys():
25 if key not in keys:
25 if key not in keys:
26 star[key] = self.req.form[key][0]
26 star[key] = self.req.form[key][0]
27 data['*'] = star
27 data['*'] = star
28 else:
28 else:
29 data[k] = self.req.form[k][0]
29 data[k] = self.req.form[k][0]
30 return [data[k] for k in keys]
30 return [data[k] for k in keys]
31 def getfile(self, fp):
31 def getfile(self, fp):
32 length = int(self.req.env['CONTENT_LENGTH'])
32 length = int(self.req.env['CONTENT_LENGTH'])
33 for s in util.filechunkiter(self.req, limit=length):
33 for s in util.filechunkiter(self.req, limit=length):
34 fp.write(s)
34 fp.write(s)
35 def redirect(self):
35 def redirect(self):
36 self.oldio = sys.stdout, sys.stderr
36 self.oldio = sys.stdout, sys.stderr
37 sys.stderr = sys.stdout = cStringIO.StringIO()
37 sys.stderr = sys.stdout = cStringIO.StringIO()
38 def groupchunks(self, cg):
38 def groupchunks(self, cg):
39 z = zlib.compressobj()
39 z = zlib.compressobj()
40 while 1:
40 while 1:
41 chunk = cg.read(4096)
41 chunk = cg.read(4096)
42 if not chunk:
42 if not chunk:
43 break
43 break
44 yield z.compress(chunk)
44 yield z.compress(chunk)
45 yield z.flush()
45 yield z.flush()
46 def sendresponse(self, s):
47 self.req.respond(HTTP_OK, HGTYPE, length=len(s))
48 self.response = s
49 def sendstream(self, source):
50 self.req.respond(HTTP_OK, HGTYPE)
51 for chunk in source.gen:
52 self.req.write(chunk)
53 def sendpushresponse(self, rsp):
54 val = sys.stdout.getvalue()
55 sys.stdout, sys.stderr = self.oldio
56 self.req.respond(HTTP_OK, HGTYPE)
57 self.response = '%d\n%s' % (rsp.res, val)
58
59 handlers = {
60 str: sendresponse,
61 wireproto.streamres: sendstream,
62 wireproto.pushres: sendpushresponse,
63 }
64
65 def _client(self):
46 def _client(self):
66 return 'remote:%s:%s:%s' % (
47 return 'remote:%s:%s:%s' % (
67 self.req.env.get('wsgi.url_scheme') or 'http',
48 self.req.env.get('wsgi.url_scheme') or 'http',
68 urllib.quote(self.req.env.get('REMOTE_HOST', '')),
49 urllib.quote(self.req.env.get('REMOTE_HOST', '')),
69 urllib.quote(self.req.env.get('REMOTE_USER', '')))
50 urllib.quote(self.req.env.get('REMOTE_USER', '')))
70
51
71 def iscmd(cmd):
52 def iscmd(cmd):
72 return cmd in wireproto.commands
53 return cmd in wireproto.commands
73
54
74 def call(repo, req, cmd):
55 def call(repo, req, cmd):
75 p = webproto(req)
56 p = webproto(req)
76 rsp = wireproto.dispatch(repo, p, cmd)
57 rsp = wireproto.dispatch(repo, p, cmd)
77 webproto.handlers[rsp.__class__](p, rsp)
58 if isinstance(rsp, str):
78 return [p.response]
59 req.respond(HTTP_OK, HGTYPE, length=len(rsp))
60 return [rsp]
61 elif isinstance(rsp, wireproto.streamres):
62 req.respond(HTTP_OK, HGTYPE)
63 return rsp.gen
64 elif isinstance(rsp, wireproto.pushres):
65 val = sys.stdout.getvalue()
66 sys.stdout, sys.stderr = p.oldio
67 req.respond(HTTP_OK, HGTYPE)
68 return ['%d\n%s' % (rsp.res, val)]
General Comments 0
You need to be logged in to leave comments. Login now