eggs/mercurial-1.7.3-py2.6-linux-x86_64.egg/mercurial/sshrepo.py
changeset 69 c6bca38c1cbf
equal deleted inserted replaced
68:5ff1fc726848 69:c6bca38c1cbf
       
     1 # sshrepo.py - ssh repository proxy class for mercurial
       
     2 #
       
     3 # Copyright 2005, 2006 Matt Mackall <mpm@selenic.com>
       
     4 #
       
     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.
       
     7 
       
     8 from i18n import _
       
     9 import util, error, wireproto
       
    10 import re
       
    11 
       
    12 class remotelock(object):
       
    13     def __init__(self, repo):
       
    14         self.repo = repo
       
    15     def release(self):
       
    16         self.repo.unlock()
       
    17         self.repo = None
       
    18     def __del__(self):
       
    19         if self.repo:
       
    20             self.release()
       
    21 
       
    22 class sshrepository(wireproto.wirerepository):
       
    23     def __init__(self, ui, path, create=0):
       
    24         self._url = path
       
    25         self.ui = ui
       
    26 
       
    27         m = re.match(r'^ssh://(([^@]+)@)?([^:/]+)(:(\d+))?(/(.*))?$', path)
       
    28         if not m:
       
    29             self._abort(error.RepoError(_("couldn't parse location %s") % path))
       
    30 
       
    31         self.user = m.group(2)
       
    32         self.host = m.group(3)
       
    33         self.port = m.group(5)
       
    34         self.path = m.group(7) or "."
       
    35 
       
    36         sshcmd = self.ui.config("ui", "ssh", "ssh")
       
    37         remotecmd = self.ui.config("ui", "remotecmd", "hg")
       
    38 
       
    39         args = util.sshargs(sshcmd, self.host, self.user, self.port)
       
    40 
       
    41         if create:
       
    42             cmd = '%s %s "%s init %s"'
       
    43             cmd = cmd % (sshcmd, args, remotecmd, self.path)
       
    44 
       
    45             ui.note(_('running %s\n') % cmd)
       
    46             res = util.system(cmd)
       
    47             if res != 0:
       
    48                 self._abort(error.RepoError(_("could not create remote repo")))
       
    49 
       
    50         self.validate_repo(ui, sshcmd, args, remotecmd)
       
    51 
       
    52     def url(self):
       
    53         return self._url
       
    54 
       
    55     def validate_repo(self, ui, sshcmd, args, remotecmd):
       
    56         # cleanup up previous run
       
    57         self.cleanup()
       
    58 
       
    59         cmd = '%s %s "%s -R %s serve --stdio"'
       
    60         cmd = cmd % (sshcmd, args, remotecmd, self.path)
       
    61 
       
    62         cmd = util.quotecommand(cmd)
       
    63         ui.note(_('running %s\n') % cmd)
       
    64         self.pipeo, self.pipei, self.pipee = util.popen3(cmd)
       
    65 
       
    66         # skip any noise generated by remote shell
       
    67         self._callstream("hello")
       
    68         r = self._callstream("between", pairs=("%s-%s" % ("0"*40, "0"*40)))
       
    69         lines = ["", "dummy"]
       
    70         max_noise = 500
       
    71         while lines[-1] and max_noise:
       
    72             l = r.readline()
       
    73             self.readerr()
       
    74             if lines[-1] == "1\n" and l == "\n":
       
    75                 break
       
    76             if l:
       
    77                 ui.debug("remote: ", l)
       
    78             lines.append(l)
       
    79             max_noise -= 1
       
    80         else:
       
    81             self._abort(error.RepoError(_("no suitable response from remote hg")))
       
    82 
       
    83         self.capabilities = set()
       
    84         for l in reversed(lines):
       
    85             if l.startswith("capabilities:"):
       
    86                 self.capabilities.update(l[:-1].split(":")[1].split())
       
    87                 break
       
    88 
       
    89     def readerr(self):
       
    90         while 1:
       
    91             size = util.fstat(self.pipee).st_size
       
    92             if size == 0:
       
    93                 break
       
    94             l = self.pipee.readline()
       
    95             if not l:
       
    96                 break
       
    97             self.ui.status(_("remote: "), l)
       
    98 
       
    99     def _abort(self, exception):
       
   100         self.cleanup()
       
   101         raise exception
       
   102 
       
   103     def cleanup(self):
       
   104         try:
       
   105             self.pipeo.close()
       
   106             self.pipei.close()
       
   107             # read the error descriptor until EOF
       
   108             for l in self.pipee:
       
   109                 self.ui.status(_("remote: "), l)
       
   110             self.pipee.close()
       
   111         except:
       
   112             pass
       
   113 
       
   114     __del__ = cleanup
       
   115 
       
   116     def _callstream(self, cmd, **args):
       
   117         self.ui.debug("sending %s command\n" % cmd)
       
   118         self.pipeo.write("%s\n" % cmd)
       
   119         for k, v in sorted(args.iteritems()):
       
   120             self.pipeo.write("%s %d\n" % (k, len(v)))
       
   121             self.pipeo.write(v)
       
   122         self.pipeo.flush()
       
   123 
       
   124         return self.pipei
       
   125 
       
   126     def _call(self, cmd, **args):
       
   127         self._callstream(cmd, **args)
       
   128         return self._recv()
       
   129 
       
   130     def _callpush(self, cmd, fp, **args):
       
   131         r = self._call(cmd, **args)
       
   132         if r:
       
   133             return '', r
       
   134         while 1:
       
   135             d = fp.read(4096)
       
   136             if not d:
       
   137                 break
       
   138             self._send(d)
       
   139         self._send("", flush=True)
       
   140         r = self._recv()
       
   141         if r:
       
   142             return '', r
       
   143         return self._recv(), ''
       
   144 
       
   145     def _decompress(self, stream):
       
   146         return stream
       
   147 
       
   148     def _recv(self):
       
   149         l = self.pipei.readline()
       
   150         self.readerr()
       
   151         try:
       
   152             l = int(l)
       
   153         except:
       
   154             self._abort(error.ResponseError(_("unexpected response:"), l))
       
   155         return self.pipei.read(l)
       
   156 
       
   157     def _send(self, data, flush=False):
       
   158         self.pipeo.write("%d\n" % len(data))
       
   159         if data:
       
   160             self.pipeo.write(data)
       
   161         if flush:
       
   162             self.pipeo.flush()
       
   163         self.readerr()
       
   164 
       
   165     def lock(self):
       
   166         self._call("lock")
       
   167         return remotelock(self)
       
   168 
       
   169     def unlock(self):
       
   170         self._call("unlock")
       
   171 
       
   172     def addchangegroup(self, cg, source, url):
       
   173         '''Send a changegroup to the remote server.  Return an integer
       
   174         similar to unbundle(). DEPRECATED, since it requires locking the
       
   175         remote.'''
       
   176         d = self._call("addchangegroup")
       
   177         if d:
       
   178             self._abort(error.RepoError(_("push refused: %s") % d))
       
   179         while 1:
       
   180             d = cg.read(4096)
       
   181             if not d:
       
   182                 break
       
   183             self.pipeo.write(d)
       
   184             self.readerr()
       
   185 
       
   186         self.pipeo.flush()
       
   187 
       
   188         self.readerr()
       
   189         r = self._recv()
       
   190         if not r:
       
   191             return 1
       
   192         try:
       
   193             return int(r)
       
   194         except:
       
   195             self._abort(error.ResponseError(_("unexpected response:"), r))
       
   196 
       
   197 instance = sshrepository