tobiko/tobiko/shell/sh/_ps.py

223 lines
6.6 KiB
Python

# Copyright (c) 2020 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.
from __future__ import absolute_import
import re
import time
import typing
from oslo_log import log
import tobiko
from tobiko.shell.sh import _cmdline
from tobiko.shell.sh import _command
from tobiko.shell.sh import _execute
from tobiko.shell.sh import _hostname
from tobiko.shell import ssh
LOG = log.getLogger(__name__)
class PsError(tobiko.TobikoException):
message = "Unable to list processes from host: {error}"
class PsWaitTimeout(PsError):
message = ("Process(es) still running on host {hostname!r} after "
"{timeout} seconds:\n{processes!s}")
IS_KERNEL_RE = re.compile('^\\[.*\\]$')
_NOT_FOUND = object()
class PsProcessBase:
command: str
pid: int
ssh_client: ssh.SSHClientType
@property
def is_kernel(self) -> bool:
return IS_KERNEL_RE.match(self.command) is not None
@property
def command_line(self) -> typing.Optional[_command.ShellCommand]:
try:
return _cmdline.get_command_line(command=self.command,
pid=self.pid,
ssh_client=self.ssh_client,
_cache_id=id(self))
except _cmdline.GetCommandLineError as ex:
LOG.error(str(ex))
return None
def kill(self, signal: int = None, **execute_params):
command_line = _command.shell_command("kill")
if signal is not None:
command_line += f"-s {signal}"
command_line += str(self.pid)
_execute.execute(command_line,
ssh_client=self.ssh_client,
**execute_params)
class PsProcessTuple(typing.NamedTuple):
"""Process listed by ps command
"""
command: str
pid: int
ssh_client: ssh.SSHClientType
class PsProcess(PsProcessTuple, PsProcessBase):
pass
P = typing.TypeVar('P', bound=PsProcessBase)
def select_processes(
processes: typing.Iterable[PsProcessBase],
command: str = None,
pid: int = None,
is_kernel: typing.Optional[bool] = False,
command_line: _command.ShellCommandType = None) \
-> tobiko.Selection[P]:
selection = tobiko.Selection[P](processes)
if selection and pid is not None:
# filter files by PID
selection = selection.with_attributes(pid=pid)
if selection and command_line is not None:
if command is None:
command = _command.shell_command(command_line)[0]
if selection and command is not None:
# filter processes by command
pattern = re.compile(command)
selection = selection.select(
lambda process: bool(pattern.match(str(process.command))))
if selection and is_kernel is not None:
# filter kernel processes
selection = selection.with_attributes(is_kernel=bool(is_kernel))
if selection and command_line is not None:
pattern = re.compile(str(command_line))
selection = selection.select(
lambda process: bool(pattern.match(str(process.command_line))))
return selection
def list_kernel_processes(**list_params):
return list_processes(is_kernel=True, **list_params)
def list_all_processes(**list_params):
return list_processes(is_kernel=None, **list_params)
def list_processes(
pid: int = None,
command: str = None,
is_kernel: typing.Optional[bool] = False,
ssh_client: ssh.SSHClientType = None,
command_line: _command.ShellCommandType = None,
**execute_params) -> tobiko.Selection[PsProcess]:
"""Returns list of running process
"""
result = _execute.execute('ps -A', expect_exit_status=None,
ssh_client=ssh_client, **execute_params)
output = result.stdout and result.stdout.strip()
if result.exit_status or not output:
raise PsError(error=result.stderr)
# Extract a list of PsProcess instances from table body
processes = tobiko.Selection[PsProcess]()
for process_data in parse_table(lines=output.splitlines(),
schema=PS_TABLE_SCHEMA):
processes.append(PsProcess(ssh_client=ssh_client, **process_data))
return select_processes(processes,
pid=pid,
command=command,
is_kernel=is_kernel,
command_line=command_line)
def wait_for_processes(timeout=float('inf'), sleep_interval=5.,
ssh_client=None, **list_params):
start_time = time.time()
time_left = timeout
while True:
processes = list_processes(timeout=time_left,
ssh_client=ssh_client,
**list_params)
if not processes:
break
time_left = timeout - (time.time() - start_time)
if time_left < sleep_interval:
hostname = _hostname.get_hostname(ssh_client=ssh_client)
process_lines = [
' {pid} {command}'.format(pid=process.pid,
command=process.command)
for process in processes]
raise PsWaitTimeout(timeout=timeout, hostname=hostname,
processes='\n'.join(process_lines))
time.sleep(sleep_interval)
def parse_pid(value):
return 'pid', int(value)
def parse_command(value):
return 'command', str(value)
PS_TABLE_SCHEMA = {
'pid': parse_pid,
'cmd': parse_command,
'command': parse_command,
}
def parse_table(lines, schema, header_line=None):
lines = iter(lines)
while not header_line:
header_line = next(lines)
getters = []
column_names = header_line.strip().lower().split()
for position, name in enumerate(column_names):
getter = schema.get(name)
if getter:
getters.append((position, getter))
for line in lines:
row = line.strip().split()
if row:
yield dict(getter(row[position])
for position, getter in getters)