# Copyright (c) 2011 The Chromium OS Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
import logging
import os
import pipes
import threading
from autotest_lib.client.common_lib import error
class _HelperThread(threading.Thread):
"""Make a thread to run the command in."""
def __init__(self, host, cmd):
super(_HelperThread, self).__init__()
self._host = host
self._cmd = cmd
self._result = None
self.daemon = True
def run(self):
logging.info('Helper thread running: %s', self._cmd)
# NB: set ignore_status as we're always terminated w/ pkill
self._result = self._host.run(self._cmd, ignore_status=True)
@property
def result(self):
"""
@returns string result of running our command if the command has
finished, and None otherwise.
"""
return self._result
class Command(object):
"""
Encapsulates a command run on a remote machine.
Future work is to have this get the PID (by prepending 'echo $$;
exec' to the command and parsing the output).
"""
def __init__(self, host, cmd, pkill_argument=None):
"""
Run a command on a remote host in the background.
@param host Host object representing the remote machine.
@param cmd String command to run on the remote machine.
@param pkill_argument String argument to pkill to kill the remote
process.
"""
if pkill_argument is None:
# Attempt to guess what a suitable pkill argument would look like.
pkill_argument = os.path.basename(cmd.split()[0])
self._command_name = pipes.quote(pkill_argument)
self._host = host
self._thread = _HelperThread(self._host, cmd)
self._thread.start()
def join(self, signal=None, timeout=5.0):
"""
Kills the remote command and waits until it dies. Takes an optional
signal argument to control which signal to send the process to be
killed.
@param signal Signal string to give to pkill (e.g. SIGNAL_INT).
@param timeout float number of seconds to wait for join to finish.
"""
if signal is None:
signal_arg = ''
else:
# In theory, it should be hard to pass something evil for signal if
# we make sure it's an integer before passing it to pkill.
signal_arg = '-' + str(int(signal))
# Ignore status because the command may have exited already
self._host.run("pkill %s %s" % (signal_arg, self._command_name),
ignore_status=True)
self._thread.join(timeout)
if self._thread.isAlive():
raise error.TestFail('Failed to kill remote command: %s' %
self._command_name)
def __enter__(self):
return self
def __exit__(self, exception, value, traceback):
self.join()
return False
@property
def result(self):
"""
@returns string result of running our command if the command has
finished, and None otherwise.
"""
return self._thread.result