sshserver.py
242 lines
| 7.2 KiB
| text/x-python
|
PythonLexer
/ mercurial / sshserver.py
Vadim Gelfer
|
r2399 | # sshserver.py - ssh protocol server support for mercurial | ||
Vadim Gelfer
|
r2396 | # | ||
Thomas Arendsen Hein
|
r4635 | # Copyright 2005-2007 Matt Mackall <mpm@selenic.com> | ||
Vadim Gelfer
|
r2859 | # Copyright 2006 Vadim Gelfer <vadim.gelfer@gmail.com> | ||
Vadim Gelfer
|
r2396 | # | ||
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. | ||
Vadim Gelfer
|
r2396 | |||
Matt Mackall
|
r3891 | from i18n import _ | ||
Joel Rosdahl
|
r6211 | from node import bin, hex | ||
Matt Mackall
|
r11369 | import streamclone, util, hook, pushkey | ||
Dirkjan Ochtman
|
r9713 | import os, sys, tempfile, urllib, copy | ||
Vadim Gelfer
|
r2396 | |||
class sshserver(object): | ||||
Dirkjan Ochtman
|
r9713 | |||
Matt Mackall
|
r11369 | caps = 'unbundle lookup changegroupsubset branchmap pushkey'.split() | ||
Dirkjan Ochtman
|
r9713 | |||
Vadim Gelfer
|
r2396 | def __init__(self, ui, repo): | ||
self.ui = ui | ||||
self.repo = repo | ||||
self.lock = None | ||||
self.fin = sys.stdin | ||||
self.fout = sys.stdout | ||||
Matt Mackall
|
r5833 | hook.redirect(True) | ||
Vadim Gelfer
|
r2396 | sys.stdout = sys.stderr | ||
# Prevent insertion/deletion of CRs | ||||
util.set_binary(self.fin) | ||||
util.set_binary(self.fout) | ||||
def getarg(self): | ||||
argline = self.fin.readline()[:-1] | ||||
arg, l = argline.split() | ||||
val = self.fin.read(int(l)) | ||||
return arg, val | ||||
def respond(self, v): | ||||
self.fout.write("%d\n" % len(v)) | ||||
self.fout.write(v) | ||||
self.fout.flush() | ||||
def serve_forever(self): | ||||
Ronny Pfannschmidt
|
r8109 | try: | ||
Matt Mackall
|
r10282 | while self.serve_one(): | ||
pass | ||||
Ronny Pfannschmidt
|
r8109 | finally: | ||
if self.lock is not None: | ||||
self.lock.release() | ||||
Vadim Gelfer
|
r2396 | sys.exit(0) | ||
def serve_one(self): | ||||
cmd = self.fin.readline()[:-1] | ||||
if cmd: | ||||
impl = getattr(self, 'do_' + cmd, None) | ||||
Matt Mackall
|
r10282 | if impl: | ||
impl() | ||||
Vadim Gelfer
|
r2397 | else: self.respond("") | ||
Vadim Gelfer
|
r2396 | return cmd != '' | ||
Eric Hopper
|
r3446 | def do_lookup(self): | ||
arg, key = self.getarg() | ||||
assert arg == 'key' | ||||
Eric Hopper
|
r3447 | try: | ||
r = hex(self.repo.lookup(key)) | ||||
success = 1 | ||||
Martin Geisler
|
r9198 | except Exception, inst: | ||
Eric Hopper
|
r3447 | r = str(inst) | ||
success = 0 | ||||
self.respond("%s %s\n" % (success, r)) | ||||
Eric Hopper
|
r3446 | |||
Henrik Stuart
|
r8562 | def do_branchmap(self): | ||
branchmap = self.repo.branchmap() | ||||
heads = [] | ||||
for branch, nodes in branchmap.iteritems(): | ||||
branchname = urllib.quote(branch) | ||||
branchnodes = [hex(node) for node in nodes] | ||||
heads.append('%s %s' % (branchname, ' '.join(branchnodes))) | ||||
self.respond('\n'.join(heads)) | ||||
Vadim Gelfer
|
r2396 | def do_heads(self): | ||
h = self.repo.heads() | ||||
self.respond(" ".join(map(hex, h)) + "\n") | ||||
Matt Mackall
|
r2419 | def do_hello(self): | ||
'''the hello command returns a set of lines describing various | ||||
interesting things about the server, in an RFC822-like format. | ||||
Currently the only one defined is "capabilities", which | ||||
consists of a line in the form: | ||||
capabilities: space separated list of tokens | ||||
''' | ||||
Dirkjan Ochtman
|
r9713 | caps = copy.copy(self.caps) | ||
Matt Mackall
|
r10377 | if streamclone.allowed(self.repo.ui): | ||
Matt Mackall
|
r4258 | caps.append('stream=%d' % self.repo.changelog.version) | ||
Vadim Gelfer
|
r2621 | self.respond("capabilities: %s\n" % (' '.join(caps),)) | ||
Matt Mackall
|
r2419 | |||
Vadim Gelfer
|
r2396 | def do_lock(self): | ||
Vadim Gelfer
|
r2439 | '''DEPRECATED - allowing remote client to lock repo is not safe''' | ||
Vadim Gelfer
|
r2396 | self.lock = self.repo.lock() | ||
self.respond("") | ||||
def do_unlock(self): | ||||
Vadim Gelfer
|
r2439 | '''DEPRECATED''' | ||
Vadim Gelfer
|
r2396 | if self.lock: | ||
self.lock.release() | ||||
self.lock = None | ||||
self.respond("") | ||||
def do_branches(self): | ||||
arg, nodes = self.getarg() | ||||
nodes = map(bin, nodes.split(" ")) | ||||
r = [] | ||||
for b in self.repo.branches(nodes): | ||||
r.append(" ".join(map(hex, b)) + "\n") | ||||
self.respond("".join(r)) | ||||
def do_between(self): | ||||
arg, pairs = self.getarg() | ||||
pairs = [map(bin, p.split("-")) for p in pairs.split(" ")] | ||||
r = [] | ||||
for b in self.repo.between(pairs): | ||||
r.append(" ".join(map(hex, b)) + "\n") | ||||
self.respond("".join(r)) | ||||
def do_changegroup(self): | ||||
nodes = [] | ||||
arg, roots = self.getarg() | ||||
nodes = map(bin, roots.split(" ")) | ||||
cg = self.repo.changegroup(nodes, 'serve') | ||||
while True: | ||||
d = cg.read(4096) | ||||
if not d: | ||||
break | ||||
self.fout.write(d) | ||||
self.fout.flush() | ||||
Eric Hopper
|
r3446 | def do_changegroupsubset(self): | ||
argmap = dict([self.getarg(), self.getarg()]) | ||||
bases = [bin(n) for n in argmap['bases'].split(' ')] | ||||
heads = [bin(n) for n in argmap['heads'].split(' ')] | ||||
cg = self.repo.changegroupsubset(bases, heads, 'serve') | ||||
while True: | ||||
d = cg.read(4096) | ||||
if not d: | ||||
break | ||||
self.fout.write(d) | ||||
self.fout.flush() | ||||
Vadim Gelfer
|
r2396 | def do_addchangegroup(self): | ||
Vadim Gelfer
|
r2439 | '''DEPRECATED''' | ||
Vadim Gelfer
|
r2396 | if not self.lock: | ||
self.respond("not locked") | ||||
return | ||||
self.respond("") | ||||
Matt Mackall
|
r11442 | r = self.repo.addchangegroup(self.fin, 'serve', self.client_url(), | ||
lock=self.lock) | ||||
Vadim Gelfer
|
r2396 | self.respond(str(r)) | ||
Vadim Gelfer
|
r2439 | |||
Vadim Gelfer
|
r2673 | def client_url(self): | ||
client = os.environ.get('SSH_CLIENT', '').split(' ', 1)[0] | ||||
return 'remote:ssh:' + client | ||||
Thomas Arendsen Hein
|
r3223 | |||
Vadim Gelfer
|
r2439 | def do_unbundle(self): | ||
their_heads = self.getarg()[1].split() | ||||
def check_heads(): | ||||
heads = map(hex, self.repo.heads()) | ||||
return their_heads == [hex('force')] or their_heads == heads | ||||
# fail early if possible | ||||
if not check_heads(): | ||||
self.respond(_('unsynced changes')) | ||||
return | ||||
self.respond('') | ||||
# write bundle data to temporary file because it can be big | ||||
Benoit Boissinot
|
r9742 | fd, tempname = tempfile.mkstemp(prefix='hg-unbundle-') | ||
fp = os.fdopen(fd, 'wb+') | ||||
Vadim Gelfer
|
r2439 | try: | ||
count = int(self.fin.readline()) | ||||
while count: | ||||
fp.write(self.fin.read(count)) | ||||
count = int(self.fin.readline()) | ||||
was_locked = self.lock is not None | ||||
if not was_locked: | ||||
self.lock = self.repo.lock() | ||||
try: | ||||
if not check_heads(): | ||||
# someone else committed/pushed/unbundled while we | ||||
# were transferring data | ||||
self.respond(_('unsynced changes')) | ||||
return | ||||
self.respond('') | ||||
# push can proceed | ||||
fp.seek(0) | ||||
Matt Mackall
|
r11442 | r = self.repo.addchangegroup(fp, 'serve', self.client_url(), | ||
lock=self.lock) | ||||
Vadim Gelfer
|
r2439 | self.respond(str(r)) | ||
finally: | ||||
if not was_locked: | ||||
self.lock.release() | ||||
self.lock = None | ||||
finally: | ||||
Benoit Boissinot
|
r9742 | fp.close() | ||
os.unlink(tempname) | ||||
Vadim Gelfer
|
r2439 | |||
Vadim Gelfer
|
r2612 | def do_stream_out(self): | ||
Dirkjan Ochtman
|
r6925 | try: | ||
for chunk in streamclone.stream_out(self.repo): | ||||
self.fout.write(chunk) | ||||
self.fout.flush() | ||||
except streamclone.StreamException, inst: | ||||
self.fout.write(str(inst)) | ||||
self.fout.flush() | ||||
Matt Mackall
|
r11369 | |||
def do_pushkey(self): | ||||
arg, key = self.getarg() | ||||
arg, namespace = self.getarg() | ||||
arg, new = self.getarg() | ||||
arg, old = self.getarg() | ||||
r = pushkey.push(self.repo, namespace, key, old, new) | ||||
self.respond('%s\n' % int(r)) | ||||
def do_listkeys(self): | ||||
arg, namespace = self.getarg() | ||||
d = pushkey.list(self.repo, namespace).items() | ||||
t = '\n'.join(['%s\t%s' % (k.encode('string-escape'), | ||||
v.encode('string-escape')) for k, v in d]) | ||||
self.respond(t) | ||||