protocol.py
206 lines
| 6.7 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 | |||
Dirkjan Ochtman
|
r9713 | import cStringIO, zlib, tempfile, errno, os, sys, urllib, copy | ||
Dirkjan Ochtman
|
r5598 | from mercurial import util, streamclone | ||
Joel Rosdahl
|
r6211 | from mercurial.node import bin, hex | ||
Dirkjan Ochtman
|
r6154 | from mercurial import changegroup as changegroupmod | ||
Benoit Boissinot
|
r7281 | from common import ErrorResponse, HTTP_OK, HTTP_NOT_FOUND, HTTP_SERVER_ERROR | ||
Dirkjan Ochtman
|
r5598 | |||
Dirkjan Ochtman
|
r5963 | # __all__ is populated with the allowed commands. Be sure to add to it if | ||
# you're adding a new command, or the new command won't work. | ||||
__all__ = [ | ||||
'lookup', 'heads', 'branches', 'between', 'changegroup', | ||||
'changegroupsubset', 'capabilities', 'unbundle', 'stream_out', | ||||
Henrik Stuart
|
r8562 | 'branchmap', | ||
Dirkjan Ochtman
|
r5963 | ] | ||
Dirkjan Ochtman
|
r5993 | HGTYPE = 'application/mercurial-0.1' | ||
Dirkjan Ochtman
|
r9713 | basecaps = 'lookup changegroupsubset branchmap'.split() | ||
Dirkjan Ochtman
|
r5993 | |||
Dirkjan Ochtman
|
r6781 | def lookup(repo, req): | ||
Dirkjan Ochtman
|
r5598 | try: | ||
Dirkjan Ochtman
|
r6781 | r = hex(repo.lookup(req.form['key'][0])) | ||
Dirkjan Ochtman
|
r5598 | success = 1 | ||
Martin Geisler
|
r9198 | except Exception, inst: | ||
Dirkjan Ochtman
|
r5598 | r = str(inst) | ||
success = 0 | ||||
resp = "%s %s\n" % (success, r) | ||||
Dirkjan Ochtman
|
r5993 | req.respond(HTTP_OK, HGTYPE, length=len(resp)) | ||
Dirkjan Ochtman
|
r6784 | yield resp | ||
Dirkjan Ochtman
|
r5598 | |||
Dirkjan Ochtman
|
r6781 | def heads(repo, req): | ||
resp = " ".join(map(hex, repo.heads())) + "\n" | ||||
Dirkjan Ochtman
|
r5993 | req.respond(HTTP_OK, HGTYPE, length=len(resp)) | ||
Dirkjan Ochtman
|
r6784 | yield resp | ||
Dirkjan Ochtman
|
r5598 | |||
Henrik Stuart
|
r8562 | def branchmap(repo, req): | ||
branches = repo.branchmap() | ||||
heads = [] | ||||
for branch, nodes in branches.iteritems(): | ||||
branchname = urllib.quote(branch) | ||||
branchnodes = [hex(node) for node in nodes] | ||||
heads.append('%s %s' % (branchname, ' '.join(branchnodes))) | ||||
resp = '\n'.join(heads) | ||||
req.respond(HTTP_OK, HGTYPE, length=len(resp)) | ||||
yield resp | ||||
Dirkjan Ochtman
|
r6781 | def branches(repo, req): | ||
Dirkjan Ochtman
|
r5598 | nodes = [] | ||
Christian Ebert
|
r5915 | if 'nodes' in req.form: | ||
Dirkjan Ochtman
|
r5598 | nodes = map(bin, req.form['nodes'][0].split(" ")) | ||
resp = cStringIO.StringIO() | ||||
Dirkjan Ochtman
|
r6781 | for b in repo.branches(nodes): | ||
Dirkjan Ochtman
|
r5598 | resp.write(" ".join(map(hex, b)) + "\n") | ||
resp = resp.getvalue() | ||||
Dirkjan Ochtman
|
r5993 | req.respond(HTTP_OK, HGTYPE, length=len(resp)) | ||
Dirkjan Ochtman
|
r6784 | yield resp | ||
Dirkjan Ochtman
|
r5598 | |||
Dirkjan Ochtman
|
r6781 | def between(repo, req): | ||
Sune Foldager
|
r10530 | pairs = [map(bin, p.split("-")) | ||
for p in req.form['pairs'][0].split(" ")] | ||||
Sune Foldager
|
r10531 | resp = ''.join(" ".join(map(hex, b)) + "\n" for b in repo.between(pairs)) | ||
Dirkjan Ochtman
|
r5993 | req.respond(HTTP_OK, HGTYPE, length=len(resp)) | ||
Dirkjan Ochtman
|
r6784 | yield resp | ||
Dirkjan Ochtman
|
r5598 | |||
Dirkjan Ochtman
|
r6781 | def changegroup(repo, req): | ||
Dirkjan Ochtman
|
r5993 | req.respond(HTTP_OK, HGTYPE) | ||
Dirkjan Ochtman
|
r5598 | nodes = [] | ||
Christian Ebert
|
r5915 | if 'roots' in req.form: | ||
Dirkjan Ochtman
|
r5598 | nodes = map(bin, req.form['roots'][0].split(" ")) | ||
z = zlib.compressobj() | ||||
Dirkjan Ochtman
|
r6781 | f = repo.changegroup(nodes, 'serve') | ||
Dirkjan Ochtman
|
r5598 | while 1: | ||
chunk = f.read(4096) | ||||
if not chunk: | ||||
break | ||||
Dirkjan Ochtman
|
r6784 | yield z.compress(chunk) | ||
Dirkjan Ochtman
|
r5598 | |||
Dirkjan Ochtman
|
r6784 | yield z.flush() | ||
Dirkjan Ochtman
|
r5598 | |||
Dirkjan Ochtman
|
r6781 | def changegroupsubset(repo, req): | ||
Dirkjan Ochtman
|
r5993 | req.respond(HTTP_OK, HGTYPE) | ||
Dirkjan Ochtman
|
r5598 | bases = [] | ||
heads = [] | ||||
Christian Ebert
|
r5915 | if 'bases' in req.form: | ||
Dirkjan Ochtman
|
r5598 | bases = [bin(x) for x in req.form['bases'][0].split(' ')] | ||
Christian Ebert
|
r5915 | if 'heads' in req.form: | ||
Dirkjan Ochtman
|
r5598 | heads = [bin(x) for x in req.form['heads'][0].split(' ')] | ||
z = zlib.compressobj() | ||||
Dirkjan Ochtman
|
r6781 | f = repo.changegroupsubset(bases, heads, 'serve') | ||
Dirkjan Ochtman
|
r5598 | while 1: | ||
chunk = f.read(4096) | ||||
if not chunk: | ||||
break | ||||
Dirkjan Ochtman
|
r6784 | yield z.compress(chunk) | ||
Dirkjan Ochtman
|
r5598 | |||
Dirkjan Ochtman
|
r6784 | yield z.flush() | ||
Dirkjan Ochtman
|
r5598 | |||
Dirkjan Ochtman
|
r6781 | def capabilities(repo, req): | ||
Dirkjan Ochtman
|
r9713 | caps = copy.copy(basecaps) | ||
Matt Mackall
|
r10377 | if streamclone.allowed(repo.ui): | ||
Dirkjan Ochtman
|
r6781 | caps.append('stream=%d' % repo.changelog.version) | ||
Dirkjan Ochtman
|
r6780 | if changegroupmod.bundlepriority: | ||
caps.append('unbundle=%s' % ','.join(changegroupmod.bundlepriority)) | ||||
rsp = ' '.join(caps) | ||||
req.respond(HTTP_OK, HGTYPE, length=len(rsp)) | ||||
Dirkjan Ochtman
|
r6784 | yield rsp | ||
Dirkjan Ochtman
|
r5598 | |||
Dirkjan Ochtman
|
r6781 | def unbundle(repo, req): | ||
Dirkjan Ochtman
|
r6335 | |||
Dirkjan Ochtman
|
r6779 | proto = req.env.get('wsgi.url_scheme') or 'http' | ||
Dirkjan Ochtman
|
r5598 | their_heads = req.form['heads'][0].split(' ') | ||
def check_heads(): | ||||
Dirkjan Ochtman
|
r6781 | heads = map(hex, repo.heads()) | ||
Dirkjan Ochtman
|
r5598 | return their_heads == [hex('force')] or their_heads == heads | ||
# fail early if possible | ||||
if not check_heads(): | ||||
Dirkjan Ochtman
|
r7180 | req.drain() | ||
Dirkjan Ochtman
|
r6926 | raise ErrorResponse(HTTP_OK, 'unsynced changes') | ||
Dirkjan Ochtman
|
r5598 | |||
# do not lock repo until all changegroup data is | ||||
# streamed. save to temporary file. | ||||
fd, tempname = tempfile.mkstemp(prefix='hg-unbundle-') | ||||
fp = os.fdopen(fd, 'wb+') | ||||
try: | ||||
length = int(req.env['CONTENT_LENGTH']) | ||||
for s in util.filechunkiter(req, limit=length): | ||||
fp.write(s) | ||||
try: | ||||
Dirkjan Ochtman
|
r6781 | lock = repo.lock() | ||
Dirkjan Ochtman
|
r5598 | try: | ||
if not check_heads(): | ||||
Dirkjan Ochtman
|
r6926 | raise ErrorResponse(HTTP_OK, 'unsynced changes') | ||
Dirkjan Ochtman
|
r5598 | |||
fp.seek(0) | ||||
header = fp.read(6) | ||||
Dirkjan Ochtman
|
r6154 | if header.startswith('HG') and not header.startswith('HG10'): | ||
raise ValueError('unknown bundle version') | ||||
elif header not in changegroupmod.bundletypes: | ||||
raise ValueError('unknown bundle compression type') | ||||
gen = changegroupmod.unbundle(header, fp) | ||||
Dirkjan Ochtman
|
r5598 | |||
# send addchangegroup output to client | ||||
Jesse Glick
|
r6265 | oldio = sys.stdout, sys.stderr | ||
sys.stderr = sys.stdout = cStringIO.StringIO() | ||||
Dirkjan Ochtman
|
r5598 | |||
try: | ||||
Henrik Stuart
|
r8846 | url = 'remote:%s:%s:%s' % ( | ||
proto, | ||||
urllib.quote(req.env.get('REMOTE_HOST', '')), | ||||
urllib.quote(req.env.get('REMOTE_USER', ''))) | ||||
Dirkjan Ochtman
|
r5598 | try: | ||
Dirkjan Ochtman
|
r6781 | ret = repo.addchangegroup(gen, 'serve', url) | ||
Dirkjan Ochtman
|
r5598 | except util.Abort, inst: | ||
sys.stdout.write("abort: %s\n" % inst) | ||||
ret = 0 | ||||
finally: | ||||
val = sys.stdout.getvalue() | ||||
Jesse Glick
|
r6265 | sys.stdout, sys.stderr = oldio | ||
Dirkjan Ochtman
|
r6926 | req.respond(HTTP_OK, HGTYPE) | ||
Dirkjan Ochtman
|
r6788 | return '%d\n%s' % (ret, val), | ||
Dirkjan Ochtman
|
r5598 | finally: | ||
Ronny Pfannschmidt
|
r8109 | lock.release() | ||
Dirkjan Ochtman
|
r6154 | except ValueError, inst: | ||
Dirkjan Ochtman
|
r6926 | raise ErrorResponse(HTTP_OK, inst) | ||
Dirkjan Ochtman
|
r5598 | except (OSError, IOError), inst: | ||
error = getattr(inst, 'strerror', 'Unknown error') | ||||
Mark Determann
|
r10951 | if not isinstance(error, str): | ||
error = 'Error: %s' % str(error) | ||||
Dirkjan Ochtman
|
r5598 | if inst.errno == errno.ENOENT: | ||
Dirkjan Ochtman
|
r5993 | code = HTTP_NOT_FOUND | ||
Dirkjan Ochtman
|
r5598 | else: | ||
Dirkjan Ochtman
|
r5993 | code = HTTP_SERVER_ERROR | ||
Sune Foldager
|
r9694 | filename = getattr(inst, 'filename', '') | ||
# Don't send our filesystem layout to the client | ||||
if filename and filename.startswith(repo.root): | ||||
filename = filename[len(repo.root)+1:] | ||||
text = '%s: %s' % (error, filename) | ||||
else: | ||||
text = error.replace(repo.root + os.path.sep, '') | ||||
raise ErrorResponse(code, text) | ||||
Dirkjan Ochtman
|
r5598 | finally: | ||
fp.close() | ||||
os.unlink(tempname) | ||||
Dirkjan Ochtman
|
r6781 | def stream_out(repo, req): | ||
Dirkjan Ochtman
|
r5993 | req.respond(HTTP_OK, HGTYPE) | ||
Dirkjan Ochtman
|
r6925 | try: | ||
Matt Mackall
|
r10377 | for chunk in streamclone.stream_out(repo): | ||
Dirkjan Ochtman
|
r6925 | yield chunk | ||
except streamclone.StreamException, inst: | ||||
yield str(inst) | ||||