sshpeer.py
710 lines
| 23.0 KiB
| text/x-python
|
PythonLexer
/ mercurial / sshpeer.py
Peter Arrenbrecht
|
r17192 | # sshpeer.py - ssh repository proxy class for mercurial | ||
# | ||||
Raphaël Gomès
|
r47575 | # Copyright 2005, 2006 Olivia Mackall <olivia@selenic.com> | ||
Peter Arrenbrecht
|
r17192 | # | ||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
Gregory Szorc
|
r25975 | |||
Peter Arrenbrecht
|
r17192 | import re | ||
Gregory Szorc
|
r35994 | import uuid | ||
Gregory Szorc
|
r25975 | |||
from .i18n import _ | ||||
from . import ( | ||||
error, | ||||
Pulkit Goyal
|
r33100 | pycompat, | ||
Gregory Szorc
|
r25975 | util, | ||
Gregory Szorc
|
r36553 | wireprototypes, | ||
Gregory Szorc
|
r37632 | wireprotov1peer, | ||
Gregory Szorc
|
r37803 | wireprotov1server, | ||
Gregory Szorc
|
r25975 | ) | ||
Yuya Nishihara
|
r37138 | from .utils import ( | ||
procutil, | ||||
Augie Fackler
|
r38494 | stringutil, | ||
r47669 | urlutil, | |||
Yuya Nishihara
|
r37138 | ) | ||
Peter Arrenbrecht
|
r17192 | |||
Augie Fackler
|
r43346 | |||
Peter Arrenbrecht
|
r17192 | def _serverquote(s): | ||
Yuya Nishihara
|
r35475 | """quote a string for the remote shell ... which we assume is sh""" | ||
Matt Mackall
|
r23671 | if not s: | ||
return s | ||||
Augie Fackler
|
r43347 | if re.match(b'[a-zA-Z0-9@%_+=:,./-]*$', s): | ||
Peter Arrenbrecht
|
r17192 | return s | ||
Augie Fackler
|
r43347 | return b"'%s'" % s.replace(b"'", b"'\\''") | ||
Peter Arrenbrecht
|
r17192 | |||
Augie Fackler
|
r43346 | |||
Valentin Gatien-Baron
|
r45387 | def _forwardoutput(ui, pipe, warn=False): | ||
Pierre-Yves David
|
r25244 | """display all data currently available on pipe as remote output. | ||
This is non blocking.""" | ||||
Valentin Gatien-Baron
|
r47429 | if pipe and not pipe.closed: | ||
Yuya Nishihara
|
r37138 | s = procutil.readpipe(pipe) | ||
Matt Harbison
|
r36851 | if s: | ||
Valentin Gatien-Baron
|
r45387 | display = ui.warn if warn else ui.status | ||
Matt Harbison
|
r36851 | for l in s.splitlines(): | ||
Valentin Gatien-Baron
|
r45387 | display(_(b"remote: "), l, b'\n') | ||
Pierre-Yves David
|
r25244 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class doublepipe: | ||
Pierre-Yves David
|
r25421 | """Operate a side-channel pipe in addition of a main one | ||
The side-channel pipe contains server output to be forwarded to the user | ||||
input. The double pipe will behave as the "main" pipe, but will ensure the | ||||
content of the "side" pipe is properly processed while we wait for blocking | ||||
call on the "main" pipe. | ||||
If large amounts of data are read from "main", the forward will cease after | ||||
the first bytes start to appear. This simplifies the implementation | ||||
without affecting actual output of sshpeer too much as we rarely issue | ||||
large read for data not yet emitted by the server. | ||||
The main pipe is expected to be a 'bufferedinputpipe' from the util module | ||||
Augie Fackler
|
r31953 | that handle all the os specific bits. This class lives in this module | ||
Mads Kiilerich
|
r26781 | because it focus on behavior specific to the ssh protocol.""" | ||
Pierre-Yves David
|
r25421 | |||
def __init__(self, ui, main, side): | ||||
self._ui = ui | ||||
self._main = main | ||||
self._side = side | ||||
def _wait(self): | ||||
"""wait until some data are available on main or side | ||||
return a pair of boolean (ismainready, issideready) | ||||
(This will only wait for data if the setup is supported by `util.poll`) | ||||
""" | ||||
Augie Fackler
|
r43346 | if ( | ||
isinstance(self._main, util.bufferedinputpipe) | ||||
and self._main.hasbuffer | ||||
): | ||||
Gregory Szorc
|
r36387 | # Main has data. Assume side is worth poking at. | ||
return True, True | ||||
Pierre-Yves David
|
r25421 | fds = [self._main.fileno(), self._side.fileno()] | ||
try: | ||||
act = util.poll(fds) | ||||
except NotImplementedError: | ||||
# non supported yet case, assume all have data. | ||||
act = fds | ||||
return (self._main.fileno() in act, self._side.fileno() in act) | ||||
Pierre-Yves David
|
r25456 | def write(self, data): | ||
Augie Fackler
|
r43347 | return self._call(b'write', data) | ||
Pierre-Yves David
|
r25456 | |||
Pierre-Yves David
|
r25421 | def read(self, size): | ||
Augie Fackler
|
r43347 | r = self._call(b'read', size) | ||
Augie Fackler
|
r32062 | if size != 0 and not r: | ||
# We've observed a condition that indicates the | ||||
# stdout closed unexpectedly. Check stderr one | ||||
# more time and snag anything that's there before | ||||
# letting anyone know the main part of the pipe | ||||
# closed prematurely. | ||||
_forwardoutput(self._ui, self._side) | ||||
return r | ||||
Pierre-Yves David
|
r25421 | |||
Joerg Sonnenberger
|
r38735 | def unbufferedread(self, size): | ||
Augie Fackler
|
r43347 | r = self._call(b'unbufferedread', size) | ||
Joerg Sonnenberger
|
r38735 | if size != 0 and not r: | ||
# We've observed a condition that indicates the | ||||
# stdout closed unexpectedly. Check stderr one | ||||
# more time and snag anything that's there before | ||||
# letting anyone know the main part of the pipe | ||||
# closed prematurely. | ||||
_forwardoutput(self._ui, self._side) | ||||
return r | ||||
Pierre-Yves David
|
r25421 | def readline(self): | ||
Augie Fackler
|
r43347 | return self._call(b'readline') | ||
Pierre-Yves David
|
r25421 | |||
Pierre-Yves David
|
r25455 | def _call(self, methname, data=None): | ||
Augie Fackler
|
r46554 | """call <methname> on "main", forward output of "side" while blocking""" | ||
Pierre-Yves David
|
r25455 | # data can be '' or 0 | ||
if (data is not None and not data) or self._main.closed: | ||||
Pierre-Yves David
|
r25421 | _forwardoutput(self._ui, self._side) | ||
Augie Fackler
|
r43347 | return b'' | ||
Pierre-Yves David
|
r25421 | while True: | ||
mainready, sideready = self._wait() | ||||
if sideready: | ||||
_forwardoutput(self._ui, self._side) | ||||
if mainready: | ||||
r51813 | meth = getattr(self._main, pycompat.sysstr(methname)) | |||
Pierre-Yves David
|
r25455 | if data is None: | ||
Pierre-Yves David
|
r25421 | return meth() | ||
else: | ||||
Pierre-Yves David
|
r25455 | return meth(data) | ||
Pierre-Yves David
|
r25421 | |||
def close(self): | ||||
return self._main.close() | ||||
Valentin Gatien-Baron
|
r47416 | @property | ||
def closed(self): | ||||
return self._main.closed | ||||
Pierre-Yves David
|
r25456 | def flush(self): | ||
return self._main.flush() | ||||
Augie Fackler
|
r43346 | |||
Valentin Gatien-Baron
|
r47418 | def _cleanuppipes(ui, pipei, pipeo, pipee, warn): | ||
Gregory Szorc
|
r35951 | """Clean up pipes used by an SSH connection.""" | ||
Valentin Gatien-Baron
|
r47418 | didsomething = False | ||
if pipeo and not pipeo.closed: | ||||
didsomething = True | ||||
Gregory Szorc
|
r35951 | pipeo.close() | ||
Valentin Gatien-Baron
|
r47418 | if pipei and not pipei.closed: | ||
didsomething = True | ||||
Gregory Szorc
|
r35951 | pipei.close() | ||
Valentin Gatien-Baron
|
r47418 | if pipee and not pipee.closed: | ||
didsomething = True | ||||
Gregory Szorc
|
r35951 | # Try to read from the err descriptor until EOF. | ||
try: | ||||
for l in pipee: | ||||
Augie Fackler
|
r43347 | ui.status(_(b'remote: '), l) | ||
Gregory Szorc
|
r35951 | except (IOError, ValueError): | ||
pass | ||||
pipee.close() | ||||
Valentin Gatien-Baron
|
r47418 | if didsomething and warn is not None: | ||
# Encourage explicit close of sshpeers. Closing via __del__ is | ||||
# not very predictable when exceptions are thrown, which has led | ||||
# to deadlocks due to a peer get gc'ed in a fork | ||||
# We add our own stack trace, because the stacktrace when called | ||||
# from __del__ is useless. | ||||
Valentin Gatien-Baron
|
r47419 | ui.develwarn(b'missing close on SSH connection created at:\n%s' % warn) | ||
Valentin Gatien-Baron
|
r47418 | |||
Augie Fackler
|
r43346 | |||
Manuel Jacob
|
r51316 | def _makeconnection( | ||
ui, sshcmd, args, remotecmd, path, sshenv=None, remotehidden=False | ||||
): | ||||
Gregory Szorc
|
r35953 | """Create an SSH connection to a server. | ||
Returns a tuple of (process, stdin, stdout, stderr) for the | ||||
spawned process. | ||||
""" | ||||
Augie Fackler
|
r43347 | cmd = b'%s %s %s' % ( | ||
Gregory Szorc
|
r35953 | sshcmd, | ||
args, | ||||
Augie Fackler
|
r43346 | procutil.shellquote( | ||
Manuel Jacob
|
r51316 | b'%s -R %s serve --stdio%s' | ||
% ( | ||||
_serverquote(remotecmd), | ||||
_serverquote(path), | ||||
b' --hidden' if remotehidden else b'', | ||||
) | ||||
Augie Fackler
|
r43346 | ), | ||
) | ||||
Gregory Szorc
|
r35953 | |||
Augie Fackler
|
r43347 | ui.debug(b'running %s\n' % cmd) | ||
Gregory Szorc
|
r35953 | |||
# no buffer allow the use of 'select' | ||||
# feel free to remove buffering and select usage when we ultimately | ||||
# move to threading. | ||||
Yuya Nishihara
|
r37138 | stdin, stdout, stderr, proc = procutil.popen4(cmd, bufsize=0, env=sshenv) | ||
Gregory Szorc
|
r35953 | |||
return proc, stdin, stdout, stderr | ||||
Augie Fackler
|
r43346 | |||
Joerg Sonnenberger
|
r37411 | def _clientcapabilities(): | ||
"""Return list of capabilities of this client. | ||||
Returns a list of capabilities that are supported by this client. | ||||
""" | ||||
r51595 | protoparams = {b'partial-pull'} | |||
Augie Fackler
|
r43346 | comps = [ | ||
e.wireprotosupport().name | ||||
for e in util.compengines.supportedwireengines(util.CLIENTROLE) | ||||
] | ||||
Augie Fackler
|
r43347 | protoparams.add(b'comp=%s' % b','.join(comps)) | ||
Joerg Sonnenberger
|
r37411 | return protoparams | ||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r35956 | def _performhandshake(ui, stdin, stdout, stderr): | ||
def badresponse(): | ||||
Valentin Gatien-Baron
|
r45387 | # Flush any output on stderr. In general, the stderr contains errors | ||
# from the remote (ssh errors, some hg errors), and status indications | ||||
# (like "adding changes"), with no current way to tell them apart. | ||||
# Here we failed so early that it's almost certainly only errors, so | ||||
# use warn=True so -q doesn't hide them. | ||||
_forwardoutput(ui, stderr, warn=True) | ||||
Gregory Szorc
|
r36388 | |||
Augie Fackler
|
r43347 | msg = _(b'no suitable response from remote hg') | ||
hint = ui.config(b'ui', b'ssherrorhint') | ||||
Gregory Szorc
|
r35956 | raise error.RepoError(msg, hint=hint) | ||
Gregory Szorc
|
r35994 | # The handshake consists of sending wire protocol commands in reverse | ||
# order of protocol implementation and then sniffing for a response | ||||
# to one of them. | ||||
# | ||||
# Those commands (from oldest to newest) are: | ||||
Gregory Szorc
|
r35957 | # | ||
Gregory Szorc
|
r35994 | # ``between`` | ||
# Asks for the set of revisions between a pair of revisions. Command | ||||
# present in all Mercurial server implementations. | ||||
Gregory Szorc
|
r35957 | # | ||
Gregory Szorc
|
r35994 | # ``hello`` | ||
# Instructs the server to advertise its capabilities. Introduced in | ||||
# Mercurial 0.9.1. | ||||
# | ||||
# ``upgrade`` | ||||
# Requests upgrade from default transport protocol version 1 to | ||||
# a newer version. Introduced in Mercurial 4.6 as an experimental | ||||
# feature. | ||||
Gregory Szorc
|
r35957 | # | ||
# The ``between`` command is issued with a request for the null | ||||
# range. If the remote is a Mercurial server, this request will | ||||
# generate a specific response: ``1\n\n``. This represents the | ||||
# wire protocol encoded value for ``\n``. We look for ``1\n\n`` | ||||
# in the output stream and know this is the response to ``between`` | ||||
# and we're at the end of our handshake reply. | ||||
# | ||||
# The response to the ``hello`` command will be a line with the | ||||
# length of the value returned by that command followed by that | ||||
# value. If the server doesn't support ``hello`` (which should be | ||||
# rare), that line will be ``0\n``. Otherwise, the value will contain | ||||
# RFC 822 like lines. Of these, the ``capabilities:`` line contains | ||||
# the capabilities of the server. | ||||
# | ||||
Gregory Szorc
|
r35994 | # The ``upgrade`` command isn't really a command in the traditional | ||
# sense of version 1 of the transport because it isn't using the | ||||
# proper mechanism for formatting insteads: instead, it just encodes | ||||
# arguments on the line, delimited by spaces. | ||||
# | ||||
# The ``upgrade`` line looks like ``upgrade <token> <capabilities>``. | ||||
# If the server doesn't support protocol upgrades, it will reply to | ||||
# this line with ``0\n``. Otherwise, it emits an | ||||
# ``upgraded <token> <protocol>`` line to both stdout and stderr. | ||||
# Content immediately following this line describes additional | ||||
# protocol and server state. | ||||
# | ||||
Gregory Szorc
|
r35957 | # In addition to the responses to our command requests, the server | ||
# may emit "banner" output on stdout. SSH servers are allowed to | ||||
# print messages to stdout on login. Issuing commands on connection | ||||
# allows us to flush this banner output from the server by scanning | ||||
# for output to our well-known ``between`` command. Of course, if | ||||
# the banner contains ``1\n\n``, this will throw off our detection. | ||||
Augie Fackler
|
r43347 | requestlog = ui.configbool(b'devel', b'debug.peer-request') | ||
Gregory Szorc
|
r35956 | |||
Gregory Szorc
|
r35994 | # Generate a random token to help identify responses to version 2 | ||
# upgrade request. | ||||
Gregory Szorc
|
r36060 | token = pycompat.sysbytes(str(uuid.uuid4())) | ||
Gregory Szorc
|
r35994 | |||
Gregory Szorc
|
r35956 | try: | ||
Augie Fackler
|
r43347 | pairsarg = b'%s-%s' % (b'0' * 40, b'0' * 40) | ||
Gregory Szorc
|
r35956 | handshake = [ | ||
Augie Fackler
|
r43347 | b'hello\n', | ||
b'between\n', | ||||
b'pairs %d\n' % len(pairsarg), | ||||
Gregory Szorc
|
r35956 | pairsarg, | ||
] | ||||
if requestlog: | ||||
Augie Fackler
|
r43347 | ui.debug(b'devel-peer-request: hello+between\n') | ||
ui.debug(b'devel-peer-request: pairs: %d bytes\n' % len(pairsarg)) | ||||
ui.debug(b'sending hello command\n') | ||||
ui.debug(b'sending between command\n') | ||||
Gregory Szorc
|
r35956 | |||
Augie Fackler
|
r43347 | stdin.write(b''.join(handshake)) | ||
Gregory Szorc
|
r35956 | stdin.flush() | ||
except IOError: | ||||
badresponse() | ||||
Gregory Szorc
|
r35994 | # Assume version 1 of wire protocol by default. | ||
Gregory Szorc
|
r36553 | protoname = wireprototypes.SSHV1 | ||
Augie Fackler
|
r38494 | reupgraded = re.compile(b'^upgraded %s (.*)$' % stringutil.reescape(token)) | ||
Gregory Szorc
|
r35994 | |||
Augie Fackler
|
r43347 | lines = [b'', b'dummy'] | ||
Gregory Szorc
|
r35956 | max_noise = 500 | ||
while lines[-1] and max_noise: | ||||
try: | ||||
l = stdout.readline() | ||||
Valentin Gatien-Baron
|
r45387 | _forwardoutput(ui, stderr, warn=True) | ||
Gregory Szorc
|
r35994 | |||
# Look for reply to protocol upgrade request. It has a token | ||||
# in it, so there should be no false positives. | ||||
m = reupgraded.match(l) | ||||
if m: | ||||
protoname = m.group(1) | ||||
Augie Fackler
|
r43347 | ui.debug(b'protocol upgraded to %s\n' % protoname) | ||
Gregory Szorc
|
r35994 | # If an upgrade was handled, the ``hello`` and ``between`` | ||
# requests are ignored. The next output belongs to the | ||||
# protocol, so stop scanning lines. | ||||
break | ||||
# Otherwise it could be a banner, ``0\n`` response if server | ||||
# doesn't support upgrade. | ||||
Augie Fackler
|
r43347 | if lines[-1] == b'1\n' and l == b'\n': | ||
Gregory Szorc
|
r35956 | break | ||
if l: | ||||
Augie Fackler
|
r43347 | ui.debug(b'remote: ', l) | ||
Gregory Szorc
|
r35956 | lines.append(l) | ||
max_noise -= 1 | ||||
except IOError: | ||||
badresponse() | ||||
else: | ||||
badresponse() | ||||
caps = set() | ||||
Gregory Szorc
|
r35994 | # For version 1, we should see a ``capabilities`` line in response to the | ||
# ``hello`` command. | ||||
Gregory Szorc
|
r36553 | if protoname == wireprototypes.SSHV1: | ||
Gregory Szorc
|
r35994 | for l in reversed(lines): | ||
# Look for response to ``hello`` command. Scan from the back so | ||||
# we don't misinterpret banner output as the command reply. | ||||
Augie Fackler
|
r43347 | if l.startswith(b'capabilities:'): | ||
caps.update(l[:-1].split(b':')[1].split()) | ||||
Gregory Szorc
|
r35994 | break | ||
# Error if we couldn't find capabilities, this means: | ||||
Gregory Szorc
|
r35958 | # | ||
# 1. Remote isn't a Mercurial server | ||||
# 2. Remote is a <0.9.1 Mercurial server | ||||
# 3. Remote is a future Mercurial server that dropped ``hello`` | ||||
Gregory Szorc
|
r35994 | # and other attempted handshake mechanisms. | ||
Gregory Szorc
|
r35958 | if not caps: | ||
badresponse() | ||||
Gregory Szorc
|
r36388 | # Flush any output on stderr before proceeding. | ||
Valentin Gatien-Baron
|
r45387 | _forwardoutput(ui, stderr, warn=True) | ||
Gregory Szorc
|
r36388 | |||
Gregory Szorc
|
r35996 | return protoname, caps | ||
Gregory Szorc
|
r35956 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r37632 | class sshv1peer(wireprotov1peer.wirepeer): | ||
Augie Fackler
|
r43346 | def __init__( | ||
Manuel Jacob
|
r51309 | self, | ||
ui, | ||||
path, | ||||
proc, | ||||
stdin, | ||||
stdout, | ||||
stderr, | ||||
caps, | ||||
autoreadstderr=True, | ||||
remotehidden=False, | ||||
Augie Fackler
|
r43346 | ): | ||
Gregory Szorc
|
r35954 | """Create a peer from an existing SSH connection. | ||
``proc`` is a handle on the underlying SSH process. | ||||
``stdin``, ``stdout``, and ``stderr`` are handles on the stdio | ||||
pipes for that process. | ||||
Gregory Szorc
|
r35956 | ``caps`` is a set of capabilities supported by the remote. | ||
Gregory Szorc
|
r36550 | ``autoreadstderr`` denotes whether to automatically read from | ||
stderr and to forward its output. | ||||
Gregory Szorc
|
r35954 | """ | ||
Manuel Jacob
|
r51309 | super().__init__(ui, path=path, remotehidden=remotehidden) | ||
Gregory Szorc
|
r35953 | # self._subprocess is unused. Keeping a handle on the process | ||
# holds a reference and prevents it from being garbage collected. | ||||
Gregory Szorc
|
r35954 | self._subprocess = proc | ||
Gregory Szorc
|
r36388 | |||
# And we hook up our "doublepipe" wrapper to allow querying | ||||
# stderr any time we perform I/O. | ||||
Gregory Szorc
|
r36550 | if autoreadstderr: | ||
stdout = doublepipe(ui, util.bufferedinputpipe(stdout), stderr) | ||||
stdin = doublepipe(ui, stdin, stderr) | ||||
Gregory Szorc
|
r36388 | |||
Gregory Szorc
|
r35954 | self._pipeo = stdin | ||
self._pipei = stdout | ||||
self._pipee = stderr | ||||
Gregory Szorc
|
r35956 | self._caps = caps | ||
Gregory Szorc
|
r36626 | self._autoreadstderr = autoreadstderr | ||
Valentin Gatien-Baron
|
r47418 | self._initstack = b''.join(util.getstackframes(1)) | ||
Manuel Jacob
|
r51316 | self._remotehidden = remotehidden | ||
Peter Arrenbrecht
|
r17192 | |||
Gregory Szorc
|
r36385 | # Commands that have a "framed" response where the first line of the | ||
# response contains the length of that response. | ||||
_FRAMED_COMMANDS = { | ||||
Augie Fackler
|
r43347 | b'batch', | ||
Gregory Szorc
|
r36385 | } | ||
Gregory Szorc
|
r37336 | # Begin of ipeerconnection interface. | ||
Gregory Szorc
|
r33803 | |||
Peter Arrenbrecht
|
r17192 | def url(self): | ||
r50656 | return self.path.loc | |||
Peter Arrenbrecht
|
r17192 | |||
Gregory Szorc
|
r33803 | def local(self): | ||
return None | ||||
def canpush(self): | ||||
return True | ||||
def close(self): | ||||
Valentin Gatien-Baron
|
r47415 | self._cleanup() | ||
Gregory Szorc
|
r33803 | |||
Gregory Szorc
|
r37336 | # End of ipeerconnection interface. | ||
Gregory Szorc
|
r33803 | |||
Gregory Szorc
|
r37336 | # Begin of ipeercommands interface. | ||
Gregory Szorc
|
r33803 | |||
def capabilities(self): | ||||
return self._caps | ||||
Gregory Szorc
|
r37336 | # End of ipeercommands interface. | ||
Gregory Szorc
|
r33803 | |||
Gregory Szorc
|
r33763 | def _readerr(self): | ||
_forwardoutput(self.ui, self._pipee) | ||||
Peter Arrenbrecht
|
r17192 | |||
def _abort(self, exception): | ||||
Gregory Szorc
|
r33763 | self._cleanup() | ||
Peter Arrenbrecht
|
r17192 | raise exception | ||
Valentin Gatien-Baron
|
r47418 | def _cleanup(self, warn=None): | ||
_cleanuppipes(self.ui, self._pipei, self._pipeo, self._pipee, warn=warn) | ||||
Peter Arrenbrecht
|
r17192 | |||
Valentin Gatien-Baron
|
r47418 | def __del__(self): | ||
self._cleanup(warn=self._initstack) | ||||
Peter Arrenbrecht
|
r17192 | |||
Gregory Szorc
|
r36385 | def _sendrequest(self, cmd, args, framed=False): | ||
Augie Fackler
|
r43346 | if self.ui.debugflag and self.ui.configbool( | ||
Augie Fackler
|
r43347 | b'devel', b'debug.peer-request' | ||
Augie Fackler
|
r43346 | ): | ||
Boris Feld
|
r35717 | dbg = self.ui.debug | ||
Augie Fackler
|
r43347 | line = b'devel-peer-request: %s\n' | ||
Boris Feld
|
r35717 | dbg(line % cmd) | ||
for key, value in sorted(args.items()): | ||||
if not isinstance(value, dict): | ||||
Augie Fackler
|
r43347 | dbg(line % b' %s: %d bytes' % (key, len(value))) | ||
Boris Feld
|
r35717 | else: | ||
for dk, dv in sorted(value.items()): | ||||
Augie Fackler
|
r43347 | dbg(line % b' %s-%s: %d' % (key, dk, len(dv))) | ||
self.ui.debug(b"sending %s command\n" % cmd) | ||||
self._pipeo.write(b"%s\n" % cmd) | ||||
Gregory Szorc
|
r37803 | _func, names = wireprotov1server.commands[cmd] | ||
Peter Arrenbrecht
|
r17192 | keys = names.split() | ||
wireargs = {} | ||||
for k in keys: | ||||
Augie Fackler
|
r43347 | if k == b'*': | ||
wireargs[b'*'] = args | ||||
Peter Arrenbrecht
|
r17192 | break | ||
else: | ||||
wireargs[k] = args[k] | ||||
del args[k] | ||||
Gregory Szorc
|
r49768 | for k, v in sorted(wireargs.items()): | ||
Augie Fackler
|
r43347 | self._pipeo.write(b"%s %d\n" % (k, len(v))) | ||
Peter Arrenbrecht
|
r17192 | if isinstance(v, dict): | ||
Gregory Szorc
|
r49768 | for dk, dv in v.items(): | ||
Augie Fackler
|
r43347 | self._pipeo.write(b"%s %d\n" % (dk, len(dv))) | ||
Gregory Szorc
|
r33763 | self._pipeo.write(dv) | ||
Peter Arrenbrecht
|
r17192 | else: | ||
Gregory Szorc
|
r33763 | self._pipeo.write(v) | ||
self._pipeo.flush() | ||||
Peter Arrenbrecht
|
r17192 | |||
Gregory Szorc
|
r36385 | # We know exactly how many bytes are in the response. So return a proxy | ||
# around the raw output stream that allows reading exactly this many | ||||
# bytes. Callers then can read() without fear of overrunning the | ||||
# response. | ||||
if framed: | ||||
amount = self._getamount() | ||||
return util.cappedreader(self._pipei, amount) | ||||
Gregory Szorc
|
r33763 | return self._pipei | ||
Peter Arrenbrecht
|
r17192 | |||
Gregory Szorc
|
r36384 | def _callstream(self, cmd, **args): | ||
args = pycompat.byteskwargs(args) | ||||
Gregory Szorc
|
r36385 | return self._sendrequest(cmd, args, framed=cmd in self._FRAMED_COMMANDS) | ||
Gregory Szorc
|
r36384 | |||
Pierre-Yves David
|
r20905 | def _callcompressable(self, cmd, **args): | ||
Gregory Szorc
|
r36384 | args = pycompat.byteskwargs(args) | ||
Gregory Szorc
|
r36385 | return self._sendrequest(cmd, args, framed=cmd in self._FRAMED_COMMANDS) | ||
Pierre-Yves David
|
r20905 | |||
Peter Arrenbrecht
|
r17192 | def _call(self, cmd, **args): | ||
Gregory Szorc
|
r36384 | args = pycompat.byteskwargs(args) | ||
Gregory Szorc
|
r36385 | return self._sendrequest(cmd, args, framed=True).read() | ||
Peter Arrenbrecht
|
r17192 | |||
def _callpush(self, cmd, fp, **args): | ||||
Gregory Szorc
|
r36390 | # The server responds with an empty frame if the client should | ||
# continue submitting the payload. | ||||
Peter Arrenbrecht
|
r17192 | r = self._call(cmd, **args) | ||
if r: | ||||
Augie Fackler
|
r43347 | return b'', r | ||
Gregory Szorc
|
r36390 | |||
# The payload consists of frames with content followed by an empty | ||||
# frame. | ||||
Augie Fackler
|
r43347 | for d in iter(lambda: fp.read(4096), b''): | ||
Gregory Szorc
|
r36383 | self._writeframed(d) | ||
Augie Fackler
|
r43347 | self._writeframed(b"", flush=True) | ||
Gregory Szorc
|
r36390 | |||
# In case of success, there is an empty frame and a frame containing | ||||
# the integer result (as a string). | ||||
# In case of error, there is a non-empty frame containing the error. | ||||
Gregory Szorc
|
r36383 | r = self._readframed() | ||
Peter Arrenbrecht
|
r17192 | if r: | ||
Augie Fackler
|
r43347 | return b'', r | ||
return self._readframed(), b'' | ||||
Peter Arrenbrecht
|
r17192 | |||
Pierre-Yves David
|
r21073 | def _calltwowaystream(self, cmd, fp, **args): | ||
Gregory Szorc
|
r36390 | # The server responds with an empty frame if the client should | ||
# continue submitting the payload. | ||||
Pierre-Yves David
|
r21073 | r = self._call(cmd, **args) | ||
if r: | ||||
# XXX needs to be made better | ||||
Augie Fackler
|
r43347 | raise error.Abort(_(b'unexpected remote reply: %s') % r) | ||
Gregory Szorc
|
r36390 | |||
# The payload consists of frames with content followed by an empty | ||||
# frame. | ||||
Augie Fackler
|
r43347 | for d in iter(lambda: fp.read(4096), b''): | ||
Gregory Szorc
|
r36383 | self._writeframed(d) | ||
Augie Fackler
|
r43347 | self._writeframed(b"", flush=True) | ||
Gregory Szorc
|
r36390 | |||
Gregory Szorc
|
r33763 | return self._pipei | ||
Peter Arrenbrecht
|
r17192 | |||
Augie Fackler
|
r28438 | def _getamount(self): | ||
Gregory Szorc
|
r33763 | l = self._pipei.readline() | ||
Augie Fackler
|
r43347 | if l == b'\n': | ||
Gregory Szorc
|
r36626 | if self._autoreadstderr: | ||
self._readerr() | ||||
Augie Fackler
|
r43347 | msg = _(b'check previous remote output') | ||
Pierre-Yves David
|
r25243 | self._abort(error.OutOfBandError(hint=msg)) | ||
Gregory Szorc
|
r36626 | if self._autoreadstderr: | ||
self._readerr() | ||||
Peter Arrenbrecht
|
r17192 | try: | ||
Augie Fackler
|
r28438 | return int(l) | ||
Peter Arrenbrecht
|
r17192 | except ValueError: | ||
Augie Fackler
|
r43347 | self._abort(error.ResponseError(_(b"unexpected response:"), l)) | ||
Augie Fackler
|
r28438 | |||
Gregory Szorc
|
r36383 | def _readframed(self): | ||
Gregory Szorc
|
r36646 | size = self._getamount() | ||
if not size: | ||||
return b'' | ||||
return self._pipei.read(size) | ||||
Peter Arrenbrecht
|
r17192 | |||
Gregory Szorc
|
r36383 | def _writeframed(self, data, flush=False): | ||
Augie Fackler
|
r43347 | self._pipeo.write(b"%d\n" % len(data)) | ||
Peter Arrenbrecht
|
r17192 | if data: | ||
Gregory Szorc
|
r33763 | self._pipeo.write(data) | ||
Peter Arrenbrecht
|
r17192 | if flush: | ||
Gregory Szorc
|
r33763 | self._pipeo.flush() | ||
Gregory Szorc
|
r36626 | if self._autoreadstderr: | ||
self._readerr() | ||||
Peter Arrenbrecht
|
r17192 | |||
Augie Fackler
|
r43346 | |||
Manuel Jacob
|
r51309 | def _make_peer( | ||
ui, | ||||
path, | ||||
proc, | ||||
stdin, | ||||
stdout, | ||||
stderr, | ||||
autoreadstderr=True, | ||||
remotehidden=False, | ||||
): | ||||
Gregory Szorc
|
r36505 | """Make a peer instance from existing pipes. | ||
``path`` and ``proc`` are stored on the eventual peer instance and may | ||||
not be used for anything meaningful. | ||||
``stdin``, ``stdout``, and ``stderr`` are the pipes connected to the | ||||
SSH server's stdio handles. | ||||
This function is factored out to allow creating peers that don't | ||||
actually spawn a new process. It is useful for starting SSH protocol | ||||
servers and clients via non-standard means, which can be useful for | ||||
testing. | ||||
""" | ||||
try: | ||||
protoname, caps = _performhandshake(ui, stdin, stdout, stderr) | ||||
except Exception: | ||||
Valentin Gatien-Baron
|
r47418 | _cleanuppipes(ui, stdout, stdin, stderr, warn=None) | ||
Gregory Szorc
|
r36505 | raise | ||
Gregory Szorc
|
r36553 | if protoname == wireprototypes.SSHV1: | ||
Augie Fackler
|
r43346 | return sshv1peer( | ||
ui, | ||||
path, | ||||
proc, | ||||
stdin, | ||||
stdout, | ||||
stderr, | ||||
caps, | ||||
autoreadstderr=autoreadstderr, | ||||
Manuel Jacob
|
r51309 | remotehidden=remotehidden, | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r36505 | else: | ||
Valentin Gatien-Baron
|
r47418 | _cleanuppipes(ui, stdout, stdin, stderr, warn=None) | ||
Augie Fackler
|
r43346 | raise error.RepoError( | ||
Augie Fackler
|
r43347 | _(b'unknown version of SSH protocol: %s') % protoname | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r36505 | |||
Manuel Jacob
|
r51309 | def make_peer( | ||
ui, path, create, intents=None, createopts=None, remotehidden=False | ||||
): | ||||
Gregory Szorc
|
r35949 | """Create an SSH peer. | ||
Gregory Szorc
|
r37632 | The returned object conforms to the ``wireprotov1peer.wirepeer`` interface. | ||
Gregory Szorc
|
r35949 | """ | ||
r50656 | u = urlutil.url(path.loc, parsequery=False, parsefragment=False) | |||
Augie Fackler
|
r43347 | if u.scheme != b'ssh' or not u.host or u.path is None: | ||
Felipe Resende
|
r52403 | raise error.RepoError(_(b"couldn't parse location %s") % path.loc) | ||
Gregory Szorc
|
r35949 | |||
r50656 | urlutil.checksafessh(path.loc) | |||
Gregory Szorc
|
r35949 | |||
if u.passwd is not None: | ||||
Augie Fackler
|
r43347 | raise error.RepoError(_(b'password in URL not supported')) | ||
Gregory Szorc
|
r35949 | |||
Augie Fackler
|
r43347 | sshcmd = ui.config(b'ui', b'ssh') | ||
remotecmd = ui.config(b'ui', b'remotecmd') | ||||
sshaddenv = dict(ui.configitems(b'sshenv')) | ||||
Yuya Nishihara
|
r37138 | sshenv = procutil.shellenviron(sshaddenv) | ||
Augie Fackler
|
r43347 | remotepath = u.path or b'.' | ||
Gregory Szorc
|
r35950 | |||
Yuya Nishihara
|
r37138 | args = procutil.sshargs(sshcmd, u.host, u.user, u.port) | ||
Gregory Szorc
|
r35950 | |||
if create: | ||||
Gregory Szorc
|
r39585 | # We /could/ do this, but only if the remote init command knows how to | ||
# handle them. We don't yet make any assumptions about that. And without | ||||
# querying the remote, there's no way of knowing if the remote even | ||||
# supports said requested feature. | ||||
if createopts: | ||||
Augie Fackler
|
r43346 | raise error.RepoError( | ||
Augie Fackler
|
r43347 | _( | ||
b'cannot create remote SSH repositories ' | ||||
b'with extra options' | ||||
) | ||||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r39585 | |||
Augie Fackler
|
r43347 | cmd = b'%s %s %s' % ( | ||
Augie Fackler
|
r43346 | sshcmd, | ||
args, | ||||
procutil.shellquote( | ||||
Augie Fackler
|
r43347 | b'%s init %s' | ||
Augie Fackler
|
r43346 | % (_serverquote(remotecmd), _serverquote(remotepath)) | ||
), | ||||
) | ||||
Augie Fackler
|
r43347 | ui.debug(b'running %s\n' % cmd) | ||
res = ui.system(cmd, blockedtag=b'sshpeer', environ=sshenv) | ||||
Gregory Szorc
|
r35950 | if res != 0: | ||
Augie Fackler
|
r43347 | raise error.RepoError(_(b'could not create remote repo')) | ||
Gregory Szorc
|
r35950 | |||
Augie Fackler
|
r43346 | proc, stdin, stdout, stderr = _makeconnection( | ||
Manuel Jacob
|
r51316 | ui, | ||
sshcmd, | ||||
args, | ||||
remotecmd, | ||||
remotepath, | ||||
sshenv, | ||||
remotehidden=remotehidden, | ||||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r35953 | |||
Manuel Jacob
|
r51309 | peer = _make_peer( | ||
ui, path, proc, stdin, stdout, stderr, remotehidden=remotehidden | ||||
) | ||||
Joerg Sonnenberger
|
r37411 | |||
# Finally, if supported by the server, notify it about our own | ||||
# capabilities. | ||||
Augie Fackler
|
r43347 | if b'protocaps' in peer.capabilities(): | ||
Joerg Sonnenberger
|
r37411 | try: | ||
Augie Fackler
|
r43346 | peer._call( | ||
Augie Fackler
|
r43347 | b"protocaps", caps=b' '.join(sorted(_clientcapabilities())) | ||
Augie Fackler
|
r43346 | ) | ||
Joerg Sonnenberger
|
r37411 | except IOError: | ||
peer._cleanup() | ||||
Augie Fackler
|
r43347 | raise error.RepoError(_(b'capability exchange failed')) | ||
Joerg Sonnenberger
|
r37411 | |||
return peer | ||||