| # Copyright (c) 2018 Red Hat, Inc. |
| # |
| # All Rights Reserved. |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); you may |
| # not use this file except in compliance with the License. You may obtain |
| # a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| # License for the specific language governing permissions and limitations |
| # under the License. |
| |
| import collections |
| import subprocess |
| import sys |
| |
| from oslo_log import log |
| from tempest.lib import exceptions as lib_exc |
| |
| from neutron_tempest_plugin.common import ssh |
| from neutron_tempest_plugin import config |
| from neutron_tempest_plugin import exceptions |
| |
| |
| LOG = log.getLogger(__name__) |
| |
| CONF = config.CONF |
| |
| if ssh.Client.proxy_jump_host: |
| # Perform all SSH connections passing through configured SSH server |
| SSH_PROXY_CLIENT = ssh.Client.create_proxy_client() |
| else: |
| SSH_PROXY_CLIENT = None |
| |
| |
| def execute(command, ssh_client=None, timeout=None, check=True): |
| """Execute command inside a remote or local shell |
| |
| :param command: command string to be executed |
| |
| :param ssh_client: SSH client instance used for remote shell execution |
| |
| :param timeout: command execution timeout in seconds |
| |
| :param check: when False it doesn't raises ShellCommandError when |
| exit status is not zero. True by default |
| |
| :returns: STDOUT text when command execution terminates with zero exit |
| status. |
| |
| :raises ShellTimeoutExpired: when timeout expires before command execution |
| terminates. In such case it kills the process, then it eventually would |
| try to read STDOUT and STDERR buffers (not fully implemented) before |
| raising the exception. |
| |
| :raises ShellCommandError: when command execution terminates with non-zero |
| exit status. |
| """ |
| ssh_client = ssh_client or SSH_PROXY_CLIENT |
| if timeout: |
| timeout = float(timeout) |
| |
| if ssh_client: |
| result = execute_remote_command(command=command, timeout=timeout, |
| ssh_client=ssh_client) |
| else: |
| result = execute_local_command(command=command, timeout=timeout) |
| |
| if result.exit_status == 0: |
| LOG.debug("Command %r succeeded:\n" |
| "stderr:\n%s\n" |
| "stdout:\n%s\n", |
| command, result.stderr, result.stdout) |
| elif result.exit_status is None: |
| LOG.debug("Command %r timeout expired (timeout=%s):\n" |
| "stderr:\n%s\n" |
| "stdout:\n%s\n", |
| command, timeout, result.stderr, result.stdout) |
| else: |
| LOG.debug("Command %r failed (exit_status=%s):\n" |
| "stderr:\n%s\n" |
| "stdout:\n%s\n", |
| command, result.exit_status, result.stderr, result.stdout) |
| if check: |
| result.check() |
| |
| return result |
| |
| |
| def execute_remote_command(command, ssh_client, timeout=None): |
| """Execute command on a remote host using SSH client""" |
| LOG.debug("Executing command %r on remote host %r (timeout=%r)...", |
| command, ssh_client.host, timeout) |
| |
| stdout = stderr = exit_status = None |
| |
| try: |
| # TODO(fressi): re-implement to capture stderr |
| stdout = ssh_client.exec_command(command, timeout=timeout) |
| exit_status = 0 |
| |
| except lib_exc.TimeoutException: |
| # TODO(fressi): re-implement to capture STDOUT and STDERR and make |
| # sure process is killed |
| pass |
| |
| except lib_exc.SSHExecCommandFailed as ex: |
| # Please note class SSHExecCommandFailed has been re-based on |
| # top of ShellCommandError |
| stdout = ex.stdout |
| stderr = ex.stderr |
| exit_status = ex.exit_status |
| |
| return ShellExecuteResult(command=command, timeout=timeout, |
| exit_status=exit_status, |
| stdout=stdout, stderr=stderr) |
| |
| |
| def execute_local_command(command, timeout=None): |
| """Execute command on local host using local shell""" |
| |
| LOG.debug("Executing command %r on local host (timeout=%r)...", |
| command, timeout) |
| |
| process = subprocess.Popen(command, shell=True, |
| universal_newlines=True, |
| stdout=subprocess.PIPE, |
| stderr=subprocess.PIPE) |
| |
| if timeout and sys.version_info < (3, 3): |
| # TODO(fressi): re-implement to timeout support on older Pythons |
| LOG.warning("Popen.communicate method doens't support for timeout " |
| "on Python %r", sys.version) |
| timeout = None |
| |
| # Wait for process execution while reading STDERR and STDOUT streams |
| if timeout: |
| try: |
| stdout, stderr = process.communicate(timeout=timeout) |
| except subprocess.TimeoutExpired: |
| # At this state I expect the process to be still running |
| # therefore it has to be kill later after calling poll() |
| LOG.exception("Command %r timeout expired.", command) |
| stdout = stderr = None |
| else: |
| stdout, stderr = process.communicate() |
| |
| # Check process termination status |
| exit_status = process.poll() |
| if exit_status is None: |
| # The process is still running after calling communicate(): |
| # let kill it and then read buffers again |
| process.kill() |
| stdout, stderr = process.communicate() |
| |
| return ShellExecuteResult(command=command, timeout=timeout, |
| stdout=stdout, stderr=stderr, |
| exit_status=exit_status) |
| |
| |
| class ShellExecuteResult(collections.namedtuple( |
| 'ShellExecuteResult', ['command', 'timeout', 'exit_status', 'stdout', |
| 'stderr'])): |
| |
| def check(self): |
| if self.exit_status is None: |
| raise exceptions.ShellTimeoutExpired(command=self.command, |
| timeout=self.timeout, |
| stderr=self.stderr, |
| stdout=self.stdout) |
| |
| elif self.exit_status != 0: |
| raise exceptions.ShellCommandError(command=self.command, |
| exit_status=self.exit_status, |
| stderr=self.stderr, |
| stdout=self.stdout) |