sshrepo.py
276 lines
| 8.4 KiB
| text/x-python
|
PythonLexer
/ mercurial / sshrepo.py
mpm@selenic.com
|
r1096 | # sshrepo.py - ssh repository proxy class for mercurial | ||
mpm@selenic.com
|
r1089 | # | ||
Vadim Gelfer
|
r2859 | # Copyright 2005, 2006 Matt Mackall <mpm@selenic.com> | ||
mpm@selenic.com
|
r1089 | # | ||
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. | ||
mpm@selenic.com
|
r1089 | |||
Joel Rosdahl
|
r6211 | from node import bin, hex | ||
Matt Mackall
|
r3891 | from i18n import _ | ||
Henrik Stuart
|
r9861 | import repo, util, error, encoding | ||
Henrik Stuart
|
r8563 | import re, urllib | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r6313 | class remotelock(object): | ||
def __init__(self, repo): | ||||
self.repo = repo | ||||
def release(self): | ||||
self.repo.unlock() | ||||
self.repo = None | ||||
def __del__(self): | ||||
if self.repo: | ||||
self.release() | ||||
class sshrepository(repo.repository): | ||||
Sean Meiners
|
r2549 | def __init__(self, ui, path, create=0): | ||
Vadim Gelfer
|
r2673 | self._url = path | ||
mpm@selenic.com
|
r1089 | self.ui = ui | ||
Benoit Boissinot
|
r3599 | m = re.match(r'^ssh://(([^@]+)@)?([^:/]+)(:(\d+))?(/(.*))?$', path) | ||
mpm@selenic.com
|
r1089 | if not m: | ||
Matt Mackall
|
r7642 | self.abort(error.RepoError(_("couldn't parse location %s") % path)) | ||
mpm@selenic.com
|
r1089 | |||
self.user = m.group(2) | ||||
self.host = m.group(3) | ||||
self.port = m.group(5) | ||||
self.path = m.group(7) or "." | ||||
sshcmd = self.ui.config("ui", "ssh", "ssh") | ||||
remotecmd = self.ui.config("ui", "remotecmd", "hg") | ||||
Sean Meiners
|
r2549 | |||
Steve Borho
|
r5644 | args = util.sshargs(sshcmd, self.host, self.user, self.port) | ||
Sean Meiners
|
r2549 | if create: | ||
cmd = '%s %s "%s init %s"' | ||||
cmd = cmd % (sshcmd, args, remotecmd, self.path) | ||||
Martin Geisler
|
r6953 | ui.note(_('running %s\n') % cmd) | ||
Alexis S. L. Carvalho
|
r5292 | res = util.system(cmd) | ||
Sean Meiners
|
r2549 | if res != 0: | ||
Matt Mackall
|
r7642 | self.abort(error.RepoError(_("could not create remote repo"))) | ||
Sean Meiners
|
r2549 | |||
self.validate_repo(ui, sshcmd, args, remotecmd) | ||||
Vadim Gelfer
|
r2673 | def url(self): | ||
return self._url | ||||
Sean Meiners
|
r2549 | def validate_repo(self, ui, sshcmd, args, remotecmd): | ||
Benoit Boissinot
|
r3034 | # cleanup up previous run | ||
self.cleanup() | ||||
Bryan O'Sullivan
|
r1330 | cmd = '%s %s "%s -R %s serve --stdio"' | ||
mpm@selenic.com
|
r1089 | cmd = cmd % (sshcmd, args, remotecmd, self.path) | ||
Alexis S. L. Carvalho
|
r5292 | cmd = util.quotecommand(cmd) | ||
Martin Geisler
|
r6953 | ui.note(_('running %s\n') % cmd) | ||
Martin Geisler
|
r8339 | self.pipeo, self.pipei, self.pipee = util.popen3(cmd) | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r2028 | # skip any noise generated by remote shell | ||
Matt Mackall
|
r2421 | self.do_cmd("hello") | ||
Matt Mackall
|
r2028 | r = self.do_cmd("between", pairs=("%s-%s" % ("0"*40, "0"*40))) | ||
Matt Mackall
|
r2420 | lines = ["", "dummy"] | ||
Thomas Arendsen Hein
|
r2046 | max_noise = 500 | ||
Matt Mackall
|
r2420 | while lines[-1] and max_noise: | ||
l = r.readline() | ||||
Matt Mackall
|
r2028 | self.readerr() | ||
Matt Mackall
|
r2420 | if lines[-1] == "1\n" and l == "\n": | ||
Matt Mackall
|
r2028 | break | ||
Matt Mackall
|
r2420 | if l: | ||
Martin Geisler
|
r9467 | ui.debug("remote: ", l) | ||
Matt Mackall
|
r2420 | lines.append(l) | ||
Thomas Arendsen Hein
|
r2040 | max_noise -= 1 | ||
else: | ||||
Matt Mackall
|
r7642 | self.abort(error.RepoError(_("no suitable response from remote hg"))) | ||
Matt Mackall
|
r2028 | |||
Martin Geisler
|
r8150 | self.capabilities = set() | ||
Matt Mackall
|
r8210 | for l in reversed(lines): | ||
Matt Mackall
|
r2421 | if l.startswith("capabilities:"): | ||
Bryan O'Sullivan
|
r5258 | self.capabilities.update(l[:-1].split(":")[1].split()) | ||
Matt Mackall
|
r2421 | break | ||
mpm@selenic.com
|
r1089 | def readerr(self): | ||
while 1: | ||||
Vadim Gelfer
|
r2176 | size = util.fstat(self.pipee).st_size | ||
Matt Mackall
|
r10282 | if size == 0: | ||
break | ||||
mpm@selenic.com
|
r1089 | l = self.pipee.readline() | ||
Matt Mackall
|
r10282 | if not l: | ||
break | ||||
Benoit Boissinot
|
r1402 | self.ui.status(_("remote: "), l) | ||
mpm@selenic.com
|
r1089 | |||
Matt Mackall
|
r7642 | def abort(self, exception): | ||
Alexis S. L. Carvalho
|
r3380 | self.cleanup() | ||
Thomas Arendsen Hein
|
r3765 | raise exception | ||
Alexis S. L. Carvalho
|
r3380 | |||
Benoit Boissinot
|
r3034 | def cleanup(self): | ||
mpm@selenic.com
|
r1089 | try: | ||
self.pipeo.close() | ||||
self.pipei.close() | ||||
Matt Mackall
|
r1358 | # read the error descriptor until EOF | ||
for l in self.pipee: | ||||
Benoit Boissinot
|
r1402 | self.ui.status(_("remote: "), l) | ||
mpm@selenic.com
|
r1089 | self.pipee.close() | ||
except: | ||||
pass | ||||
Benoit Boissinot
|
r3034 | __del__ = cleanup | ||
mpm@selenic.com
|
r1089 | def do_cmd(self, cmd, **args): | ||
Martin Geisler
|
r9467 | self.ui.debug("sending %s command\n" % cmd) | ||
mpm@selenic.com
|
r1089 | self.pipeo.write("%s\n" % cmd) | ||
Dirkjan Ochtman
|
r7622 | for k, v in args.iteritems(): | ||
mpm@selenic.com
|
r1089 | self.pipeo.write("%s %d\n" % (k, len(v))) | ||
self.pipeo.write(v) | ||||
self.pipeo.flush() | ||||
return self.pipei | ||||
def call(self, cmd, **args): | ||||
Alexis S. L. Carvalho
|
r5978 | self.do_cmd(cmd, **args) | ||
return self._recv() | ||||
def _recv(self): | ||||
l = self.pipei.readline() | ||||
mpm@selenic.com
|
r1089 | self.readerr() | ||
try: | ||||
l = int(l) | ||||
except: | ||||
Matt Mackall
|
r7642 | self.abort(error.ResponseError(_("unexpected response:"), l)) | ||
Alexis S. L. Carvalho
|
r5978 | return self.pipei.read(l) | ||
def _send(self, data, flush=False): | ||||
self.pipeo.write("%d\n" % len(data)) | ||||
if data: | ||||
self.pipeo.write(data) | ||||
if flush: | ||||
self.pipeo.flush() | ||||
self.readerr() | ||||
mpm@selenic.com
|
r1089 | |||
def lock(self): | ||||
self.call("lock") | ||||
return remotelock(self) | ||||
def unlock(self): | ||||
self.call("unlock") | ||||
Eric Hopper
|
r3446 | def lookup(self, key): | ||
Bryan O'Sullivan
|
r5259 | self.requirecap('lookup', _('look up remote revision')) | ||
Eric Hopper
|
r3446 | d = self.call("lookup", key=key) | ||
Eric Hopper
|
r3447 | success, data = d[:-1].split(" ", 1) | ||
Thomas Arendsen Hein
|
r3764 | if int(success): | ||
return bin(data) | ||||
else: | ||||
Matt Mackall
|
r7642 | self.abort(error.RepoError(data)) | ||
Eric Hopper
|
r3446 | |||
mpm@selenic.com
|
r1089 | def heads(self): | ||
d = self.call("heads") | ||||
try: | ||||
return map(bin, d[:-1].split(" ")) | ||||
except: | ||||
Matt Mackall
|
r7642 | self.abort(error.ResponseError(_("unexpected response:"), d)) | ||
mpm@selenic.com
|
r1089 | |||
Henrik Stuart
|
r8563 | def branchmap(self): | ||
d = self.call("branchmap") | ||||
try: | ||||
branchmap = {} | ||||
for branchpart in d.splitlines(): | ||||
branchheads = branchpart.split(' ') | ||||
branchname = urllib.unquote(branchheads[0]) | ||||
Sune Foldager
|
r9878 | # Earlier servers (1.3.x) send branch names in (their) local | ||
# charset. The best we can do is assume it's identical to our | ||||
# own local charset, in case it's not utf-8. | ||||
Henrik Stuart
|
r9861 | try: | ||
Sune Foldager
|
r9878 | branchname.decode('utf-8') | ||
Henrik Stuart
|
r9861 | except UnicodeDecodeError: | ||
Sune Foldager
|
r9878 | branchname = encoding.fromlocal(branchname) | ||
Henrik Stuart
|
r8563 | branchheads = [bin(x) for x in branchheads[1:]] | ||
branchmap[branchname] = branchheads | ||||
return branchmap | ||||
except: | ||||
raise error.ResponseError(_("unexpected response:"), d) | ||||
mpm@selenic.com
|
r1089 | def branches(self, nodes): | ||
n = " ".join(map(hex, nodes)) | ||||
d = self.call("branches", nodes=n) | ||||
try: | ||||
Matt Mackall
|
r10282 | br = [tuple(map(bin, b.split(" "))) for b in d.splitlines()] | ||
mpm@selenic.com
|
r1089 | return br | ||
except: | ||||
Matt Mackall
|
r7642 | self.abort(error.ResponseError(_("unexpected response:"), d)) | ||
mpm@selenic.com
|
r1089 | |||
def between(self, pairs): | ||||
Benoit Boissinot
|
r7207 | n = " ".join(["-".join(map(hex, p)) for p in pairs]) | ||
mpm@selenic.com
|
r1089 | d = self.call("between", pairs=n) | ||
try: | ||||
Matt Mackall
|
r10282 | p = [l and map(bin, l.split(" ")) or [] for l in d.splitlines()] | ||
mpm@selenic.com
|
r1089 | return p | ||
except: | ||||
Matt Mackall
|
r7642 | self.abort(error.ResponseError(_("unexpected response:"), d)) | ||
mpm@selenic.com
|
r1089 | |||
Vadim Gelfer
|
r1736 | def changegroup(self, nodes, kind): | ||
mpm@selenic.com
|
r1089 | n = " ".join(map(hex, nodes)) | ||
Benoit Boissinot
|
r2449 | return self.do_cmd("changegroup", roots=n) | ||
mpm@selenic.com
|
r1089 | |||
Eric Hopper
|
r3446 | def changegroupsubset(self, bases, heads, kind): | ||
Bryan O'Sullivan
|
r5259 | self.requirecap('changegroupsubset', _('look up remote changes')) | ||
Eric Hopper
|
r3446 | bases = " ".join(map(hex, bases)) | ||
heads = " ".join(map(hex, heads)) | ||||
return self.do_cmd("changegroupsubset", bases=bases, heads=heads) | ||||
Vadim Gelfer
|
r2439 | def unbundle(self, cg, heads, source): | ||
Greg Ward
|
r11153 | '''Send cg (a readable file-like object representing the | ||
changegroup to push, typically a chunkbuffer object) to the | ||||
remote server as a bundle. Return an integer indicating the | ||||
result of the push (see localrepository.addchangegroup()).''' | ||||
Vadim Gelfer
|
r2439 | d = self.call("unbundle", heads=' '.join(map(hex, heads))) | ||
if d: | ||||
Alexis S. L. Carvalho
|
r5190 | # remote may send "unsynced changes" | ||
Matt Mackall
|
r7642 | self.abort(error.RepoError(_("push refused: %s") % d)) | ||
Vadim Gelfer
|
r2439 | |||
while 1: | ||||
d = cg.read(4096) | ||||
Alexis S. L. Carvalho
|
r5978 | if not d: | ||
break | ||||
self._send(d) | ||||
Vadim Gelfer
|
r2439 | |||
Alexis S. L. Carvalho
|
r5978 | self._send("", flush=True) | ||
Vadim Gelfer
|
r2439 | |||
Alexis S. L. Carvalho
|
r5978 | r = self._recv() | ||
Alexis S. L. Carvalho
|
r5190 | if r: | ||
# remote may send "unsynced changes" | ||||
Matt Mackall
|
r7642 | self.abort(error.RepoError(_("push failed: %s") % r)) | ||
Alexis S. L. Carvalho
|
r5190 | |||
Alexis S. L. Carvalho
|
r5978 | r = self._recv() | ||
try: | ||||
return int(r) | ||||
except: | ||||
Matt Mackall
|
r7642 | self.abort(error.ResponseError(_("unexpected response:"), r)) | ||
Vadim Gelfer
|
r2439 | |||
Vadim Gelfer
|
r2673 | def addchangegroup(self, cg, source, url): | ||
Greg Ward
|
r11153 | '''Send a changegroup to the remote server. Return an integer | ||
similar to unbundle(). DEPRECATED, since it requires locking the | ||||
remote.''' | ||||
mpm@selenic.com
|
r1089 | d = self.call("addchangegroup") | ||
if d: | ||||
Matt Mackall
|
r7642 | self.abort(error.RepoError(_("push refused: %s") % d)) | ||
mpm@selenic.com
|
r1089 | while 1: | ||
d = cg.read(4096) | ||||
Alexis S. L. Carvalho
|
r5978 | if not d: | ||
break | ||||
mpm@selenic.com
|
r1089 | self.pipeo.write(d) | ||
self.readerr() | ||||
self.pipeo.flush() | ||||
self.readerr() | ||||
Alexis S. L. Carvalho
|
r5978 | r = self._recv() | ||
Vadim Gelfer
|
r2019 | if not r: | ||
return 1 | ||||
Alexis S. L. Carvalho
|
r5978 | try: | ||
return int(r) | ||||
except: | ||||
Matt Mackall
|
r7642 | self.abort(error.ResponseError(_("unexpected response:"), r)) | ||
Vadim Gelfer
|
r2612 | |||
def stream_out(self): | ||||
return self.do_cmd('stream_out') | ||||
Vadim Gelfer
|
r2740 | |||
instance = sshrepository | ||||