mirror of
https://github.com/ceph/ceph
synced 2025-01-08 20:21:33 +00:00
263 lines
7.5 KiB
Python
263 lines
7.5 KiB
Python
from cStringIO import StringIO
|
|
|
|
import gevent
|
|
import gevent.event
|
|
import pipes
|
|
import logging
|
|
import shutil
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
class RemoteProcess(object):
|
|
__slots__ = [
|
|
'command', 'stdin', 'stdout', 'stderr', 'exitstatus',
|
|
# for orchestra.remote.Remote to place a backreference
|
|
'remote',
|
|
]
|
|
def __init__(self, command, stdin, stdout, stderr, exitstatus):
|
|
self.command = command
|
|
self.stdin = stdin
|
|
self.stdout = stdout
|
|
self.stderr = stderr
|
|
self.exitstatus = exitstatus
|
|
|
|
class Raw(object):
|
|
def __init__(self, value):
|
|
self.value = value
|
|
|
|
def __repr__(self):
|
|
return '{cls}({value!r})'.format(
|
|
cls=self.__class__.__name__,
|
|
value=self.value,
|
|
)
|
|
|
|
def quote(args):
|
|
def _quote(args):
|
|
for a in args:
|
|
if isinstance(a, Raw):
|
|
yield a.value
|
|
else:
|
|
yield pipes.quote(a)
|
|
return ' '.join(_quote(args))
|
|
|
|
def execute(client, args):
|
|
"""
|
|
Execute a command remotely.
|
|
|
|
Caller needs to handle stdin etc.
|
|
|
|
:param client: SSHConnection to run the command with
|
|
:param args: command to run
|
|
:type args: list of string
|
|
|
|
Returns a RemoteProcess, where exitstatus is a callable that will
|
|
block until the exit status is available.
|
|
"""
|
|
cmd = quote(args)
|
|
log.debug('Running: {cmd!r}'.format(cmd=cmd))
|
|
(in_, out, err) = client.exec_command(cmd)
|
|
|
|
def get_exitstatus():
|
|
status = out.channel.recv_exit_status()
|
|
# -1 on connection loss *and* signals; map to more pythonic None
|
|
if status == -1:
|
|
status = None
|
|
return status
|
|
|
|
r = RemoteProcess(
|
|
command=cmd,
|
|
stdin=in_,
|
|
stdout=out,
|
|
stderr=err,
|
|
# this is a callable that will block until the status is
|
|
# available
|
|
exitstatus=get_exitstatus,
|
|
)
|
|
return r
|
|
|
|
def copy_to_log(f, logger, loglevel=logging.INFO):
|
|
# i can't seem to get fudge to fake an iterable, so using this old
|
|
# api for now
|
|
for line in f.xreadlines():
|
|
line = line.rstrip()
|
|
logger.log(loglevel, line)
|
|
|
|
def copy_and_close(src, fdst):
|
|
if src is not None:
|
|
if isinstance(src, basestring):
|
|
src = StringIO(src)
|
|
shutil.copyfileobj(src, fdst)
|
|
fdst.close()
|
|
|
|
def copy_file_to(f, dst):
|
|
if hasattr(dst, 'log'):
|
|
# looks like a Logger to me; not using isinstance to make life
|
|
# easier for unit tests
|
|
handler = copy_to_log
|
|
else:
|
|
handler = shutil.copyfileobj
|
|
return handler(f, dst)
|
|
|
|
|
|
class CommandFailedError(Exception):
|
|
def __init__(self, command, exitstatus):
|
|
self.command = command
|
|
self.exitstatus = exitstatus
|
|
|
|
def __str__(self):
|
|
return "Command failed with status {status}: {command!r}".format(
|
|
status=self.exitstatus,
|
|
command=self.command,
|
|
)
|
|
|
|
|
|
class CommandCrashedError(Exception):
|
|
def __init__(self, command):
|
|
self.command = command
|
|
|
|
def __str__(self):
|
|
return "Command crashed: {command!r}".format(
|
|
command=self.command,
|
|
)
|
|
|
|
|
|
class ConnectionLostError(Exception):
|
|
def __init__(self, command):
|
|
self.command = command
|
|
|
|
def __str__(self):
|
|
return "SSH connection was lost: {command!r}".format(
|
|
command=self.command,
|
|
)
|
|
|
|
def spawn_asyncresult(fn, *args, **kwargs):
|
|
"""
|
|
Spawn a Greenlet and pass it's results to an AsyncResult.
|
|
|
|
This function is useful to shuffle data from a Greenlet to
|
|
AsyncResult, which then again is useful because any Greenlets that
|
|
raise exceptions will cause tracebacks to be shown on stderr by
|
|
gevent, even when ``.link_exception`` has been called. Using an
|
|
AsyncResult avoids this.
|
|
"""
|
|
r = gevent.event.AsyncResult()
|
|
def wrapper():
|
|
try:
|
|
value = fn(*args, **kwargs)
|
|
except Exception as e:
|
|
r.set_exception(e)
|
|
else:
|
|
r.set(value)
|
|
gevent.spawn(wrapper)
|
|
|
|
return r
|
|
|
|
class Sentinel(object):
|
|
def __init__(self, name):
|
|
self.name = name
|
|
|
|
def __str__(self):
|
|
return self.name
|
|
|
|
PIPE = Sentinel('PIPE')
|
|
|
|
class KludgeFile(object):
|
|
"""
|
|
Wrap Paramiko's ChannelFile in a way that lets ``f.close()``
|
|
actually cause an EOF for the remote command.
|
|
"""
|
|
def __init__(self, wrapped):
|
|
self._wrapped = wrapped
|
|
|
|
def __getattr__(self, name):
|
|
return getattr(self._wrapped, name)
|
|
|
|
def close(self):
|
|
self._wrapped.close()
|
|
self._wrapped.channel.shutdown_write()
|
|
|
|
def run(
|
|
client, args,
|
|
stdin=None, stdout=None, stderr=None,
|
|
logger=None,
|
|
check_status=True,
|
|
wait=True,
|
|
):
|
|
"""
|
|
Run a command remotely.
|
|
|
|
:param client: SSHConnection to run the command with
|
|
:param args: command to run
|
|
:type args: list of string
|
|
:param stdin: Standard input to send; either a string, a file-like object, None, or `PIPE`. `PIPE` means caller is responsible for closing stdin, or command may never exit.
|
|
:param stdout: What to do with standard output. Either a file-like object, a `logging.Logger`, or `None` for copying to default log.
|
|
:param stderr: What to do with standard error. See `stdout`.
|
|
:param logger: If logging, write stdout/stderr to "out" and "err" children of this logger. Defaults to logger named after this module.
|
|
:param check_status: Whether to raise CalledProcessError on non-zero exit status, and . Defaults to True. All signals and connection loss are made to look like SIGHUP.
|
|
:param wait: Whether to wait for process to exit. If False, returned ``r.exitstatus`` s a `gevent.event.AsyncResult`, and the actual status is available via ``.get()``.
|
|
"""
|
|
r = execute(client, args)
|
|
|
|
r.stdin = KludgeFile(wrapped=r.stdin)
|
|
|
|
g_in = None
|
|
if stdin is not PIPE:
|
|
g_in = gevent.spawn(copy_and_close, stdin, r.stdin)
|
|
r.stdin = None
|
|
else:
|
|
assert not wait, "Using PIPE for stdin without wait=False would deadlock."
|
|
|
|
if logger is None:
|
|
logger = log
|
|
|
|
if stderr is None:
|
|
stderr = logger.getChild('err')
|
|
g_err = gevent.spawn(copy_file_to, r.stderr, stderr)
|
|
r.stderr = stderr
|
|
|
|
if stdout is None:
|
|
stdout = logger.getChild('out')
|
|
g_out = gevent.spawn(copy_file_to, r.stdout, stdout)
|
|
r.stdout = stdout
|
|
|
|
def _check_status(status):
|
|
g_err.get()
|
|
g_out.get()
|
|
if g_in is not None:
|
|
g_in.get()
|
|
|
|
status = status()
|
|
if check_status:
|
|
if status is None:
|
|
# command either died due to a signal, or the connection
|
|
# was lost
|
|
transport = client.get_transport()
|
|
if not transport.is_active():
|
|
# look like we lost the connection
|
|
raise ConnectionLostError(command=r.command)
|
|
|
|
# connection seems healthy still, assuming it was a
|
|
# signal; sadly SSH does not tell us which signal
|
|
raise CommandCrashedError(command=r.command)
|
|
if status != 0:
|
|
raise CommandFailedError(command=r.command, exitstatus=status)
|
|
return status
|
|
|
|
if wait:
|
|
r.exitstatus = _check_status(r.exitstatus)
|
|
else:
|
|
r.exitstatus = spawn_asyncresult(_check_status, r.exitstatus)
|
|
|
|
return r
|
|
|
|
|
|
def wait(processes):
|
|
"""
|
|
Wait for all given processes to exit.
|
|
|
|
Raise if any one of them fails.
|
|
"""
|
|
for proc in processes:
|
|
assert isinstance(proc.exitstatus, gevent.event.AsyncResult)
|
|
proc.exitstatus.get()
|