summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--nova/process.py90
-rw-r--r--nova/tests/process_unittest.py2
-rw-r--r--nova/virt/libvirt_conn.py2
-rw-r--r--tools/install_venv.py15
4 files changed, 44 insertions, 65 deletions
diff --git a/nova/process.py b/nova/process.py
index 2dc56372f..24ea3eb7f 100644
--- a/nova/process.py
+++ b/nova/process.py
@@ -54,19 +54,20 @@ class UnexpectedErrorOutput(IOError):
IOError.__init__(self, "got stdout: %r\nstderr: %r" % (stdout, stderr))
-# NOTE(termie): this too
-class _BackRelay(protocol.ProcessProtocol):
+# This is based on _BackRelay from twister.internal.utils, but modified to capture
+# both stdout and stderr without odd stderr handling, and also to handle stdin
+class BackRelayWithInput(protocol.ProcessProtocol):
"""
Trivial protocol for communicating with a process and turning its output
into the result of a L{Deferred}.
@ivar deferred: A L{Deferred} which will be called back with all of stdout
- and, if C{errortoo} is true, all of stderr as well (mixed together in
- one string). If C{errortoo} is false and any bytes are received over
- stderr, this will fire with an L{_UnexpectedErrorOutput} instance and
- the attribute will be set to C{None}.
+ and all of stderr as well (as a tuple). C{terminate_on_stderr} is true
+ and any bytes are received over stderr, this will fire with an
+ L{_UnexpectedErrorOutput} instance and the attribute will be set to
+ C{None}.
- @ivar onProcessEnded: If C{errortoo} is false and bytes are received over
+ @ivar onProcessEnded: If C{terminate_on_stderr} is false and bytes are received over
stderr, this attribute will refer to a L{Deferred} which will be called
back when the process ends. This C{Deferred} is also associated with
the L{_UnexpectedErrorOutput} which C{deferred} fires with earlier in
@@ -74,52 +75,43 @@ class _BackRelay(protocol.ProcessProtocol):
ended, in addition to knowing when bytes have been received via stderr.
"""
- def __init__(self, deferred, errortoo=0):
+ def __init__(self, deferred, startedDeferred=None, terminate_on_stderr=False,
+ check_exit_code=True, input=None):
self.deferred = deferred
- self.s = StringIO.StringIO()
- if errortoo:
- self.errReceived = self.errReceivedIsGood
- else:
- self.errReceived = self.errReceivedIsBad
-
- def errReceivedIsBad(self, text):
- if self.deferred is not None:
+ self.stdout = StringIO.StringIO()
+ self.stderr = StringIO.StringIO()
+ self.startedDeferred = startedDeferred
+ self.terminate_on_stderr = terminate_on_stderr
+ self.check_exit_code = check_exit_code
+ self.input = input
+
+ def errReceived(self, text):
+ self.sterr.write(text)
+ if self.terminate_on_stderr and (self.deferred is not None):
self.onProcessEnded = defer.Deferred()
- err = UnexpectedErrorOutput(text, self.onProcessEnded)
- self.deferred.errback(failure.Failure(err))
+ self.deferred.errback(UnexpectedErrorOutput(stdout=self.stdout.getvalue(), stderr=self.stderr.getvalue()))
self.deferred = None
self.transport.loseConnection()
- def errReceivedIsGood(self, text):
- self.s.write(text)
+ def errReceived(self, text):
+ self.stderr.write(text)
def outReceived(self, text):
- self.s.write(text)
+ self.stdout.write(text)
def processEnded(self, reason):
if self.deferred is not None:
- self.deferred.callback(self.s.getvalue())
+ stdout, stderr = self.stdout.getvalue(), self.stderr.getvalue()
+ try:
+ if self.check_exit_code:
+ reason.trap(error.ProcessDone)
+ self.deferred.callback((stdout, stderr))
+ except:
+ self.deferred.errback(UnexpectedErrorOutput(stdout, stderr))
elif self.onProcessEnded is not None:
self.onProcessEnded.errback(reason)
-class BackRelayWithInput(_BackRelay):
- def __init__(self, deferred, startedDeferred=None, error_ok=0,
- input=None):
- # Twisted doesn't use new-style classes in most places :(
- _BackRelay.__init__(self, deferred, errortoo=error_ok)
- self.error_ok = error_ok
- self.input = input
- self.stderr = StringIO.StringIO()
- self.startedDeferred = startedDeferred
-
- def errReceivedIsBad(self, text):
- self.stderr.write(text)
- self.transport.loseConnection()
-
- def errReceivedIsGood(self, text):
- self.stderr.write(text)
-
def connectionMade(self):
if self.startedDeferred:
self.startedDeferred.callback(self)
@@ -127,31 +119,15 @@ class BackRelayWithInput(_BackRelay):
self.transport.write(self.input)
self.transport.closeStdin()
- def processEnded(self, reason):
- if self.deferred is not None:
- stdout, stderr = self.s.getvalue(), self.stderr.getvalue()
- try:
- # NOTE(termie): current behavior means if error_ok is True
- # we won't throw an error even if the process
- # exited with a non-0 status, so you can't be
- # okay with stderr output and not with bad exit
- # codes.
- if not self.error_ok:
- reason.trap(error.ProcessDone)
- self.deferred.callback((stdout, stderr))
- except:
- self.deferred.errback(UnexpectedErrorOutput(stdout, stderr))
-
-
def getProcessOutput(executable, args=None, env=None, path=None, reactor=None,
- error_ok=0, input=None, startedDeferred=None):
+ check_exit_code=True, input=None, startedDeferred=None):
if reactor is None:
from twisted.internet import reactor
args = args and args or ()
env = env and env and {}
d = defer.Deferred()
p = BackRelayWithInput(
- d, startedDeferred=startedDeferred, error_ok=error_ok, input=input)
+ d, startedDeferred=startedDeferred, check_exit_code=check_exit_code, input=input)
# NOTE(vish): commands come in as unicode, but self.executes needs
# strings or process.spawn raises a deprecation warning
executable = str(executable)
diff --git a/nova/tests/process_unittest.py b/nova/tests/process_unittest.py
index 75187e1fc..25c60c616 100644
--- a/nova/tests/process_unittest.py
+++ b/nova/tests/process_unittest.py
@@ -48,7 +48,7 @@ class ProcessTestCase(test.TrialTestCase):
def test_execute_stderr(self):
pool = process.ProcessPool(2)
- d = pool.simple_execute('cat BAD_FILE', error_ok=1)
+ d = pool.simple_execute('cat BAD_FILE', check_exit_code=False)
def _check(rv):
self.assertEqual(rv[0], '')
self.assert_('No such file' in rv[1])
diff --git a/nova/virt/libvirt_conn.py b/nova/virt/libvirt_conn.py
index c545e4190..6cb9acb29 100644
--- a/nova/virt/libvirt_conn.py
+++ b/nova/virt/libvirt_conn.py
@@ -197,7 +197,7 @@ class LibvirtConnection(object):
execute = lambda cmd, input=None: \
process.simple_execute(cmd=cmd,
input=input,
- error_ok=1)
+ check_exit_code=True)
key = data['key_data']
net = None
diff --git a/tools/install_venv.py b/tools/install_venv.py
index 0b35fc8e9..b9eac70e6 100644
--- a/tools/install_venv.py
+++ b/tools/install_venv.py
@@ -18,7 +18,7 @@ def die(message, *args):
sys.exit(1)
-def run_command(cmd, redirect_output=True, error_ok=False):
+def run_command(cmd, redirect_output=True, check_exit_code=True):
# Useful for debugging:
#print >>sys.stderr, ' '.join(cmd)
if redirect_output:
@@ -28,23 +28,26 @@ def run_command(cmd, redirect_output=True, error_ok=False):
proc = subprocess.Popen(cmd, stdout=stdout)
output = proc.communicate()[0]
- if not error_ok and proc.returncode != 0:
+ if check_exit_code and proc.returncode != 0:
die('Command "%s" failed.\n%s', ' '.join(cmd), output)
return output
def check_dependencies():
"""Make sure pip and virtualenv are on the path."""
+ # Perl also has a pip program. Hopefully the user has installed the right one!
print 'Checking for pip...',
- if not run_command(['which', 'pip']).strip():
+ if not run_command(['which', 'pip'], check_exit_code=False).strip():
die('ERROR: pip not found.\n\nNova development requires pip,'
- ' please install it using your favorite package management tool')
+ ' please install it using your favorite package management tool '
+ ' (e.g. "sudo apt-get install python-pip")')
print 'done.'
print 'Checking for virtualenv...',
- if not run_command(['which', 'virtualenv']).strip():
+ if not run_command(['which', 'virtualenv'], check_exit_code=False).strip():
die('ERROR: virtualenv not found.\n\nNova development requires virtualenv,'
- ' please install it using your favorite package management tool')
+ ' please install it using your favorite package management tool '
+ ' (e.g. "sudo easy_install virtualenv")')
print 'done.'