|
|
ba407d |
From 90b6b709e9f4002376b656b155d00d85382f1828 Mon Sep 17 00:00:00 2001
|
|
|
ba407d |
From: Pavel Moravec <pmoravec@redhat.com>
|
|
|
ba407d |
Date: Mon, 29 Mar 2021 16:23:01 +0200
|
|
|
ba407d |
Subject: [PATCH] [report] add --cmd-timeout option
|
|
|
ba407d |
|
|
|
ba407d |
Add --cmd-timeout option to configure command timeout. Plugin-specific
|
|
|
ba407d |
option of the same name (i.e. -k logs.cmd-timeout=60) can control the
|
|
|
ba407d |
timeout per plugin.
|
|
|
ba407d |
|
|
|
ba407d |
Option defaults and global/plugin-specific option preference follows the
|
|
|
ba407d |
--plugin-timeout rules.
|
|
|
ba407d |
|
|
|
ba407d |
Resolves: #2466
|
|
|
ba407d |
|
|
|
ba407d |
Signed-off-by: Pavel Moravec <pmoravec@redhat.com>
|
|
|
ba407d |
Signed-off-by: Jake Hunsaker <jhunsake@redhat.com>
|
|
|
ba407d |
---
|
|
|
ba407d |
man/en/sos-report.1 | 18 +++++++++-
|
|
|
ba407d |
sos/collector/__init__.py | 3 ++
|
|
|
ba407d |
sos/collector/sosnode.py | 5 +++
|
|
|
ba407d |
sos/options.py | 3 +-
|
|
|
ba407d |
sos/report/__init__.py | 5 ++-
|
|
|
ba407d |
sos/report/plugins/__init__.py | 63 ++++++++++++++++++++++++----------
|
|
|
ba407d |
6 files changed, 76 insertions(+), 21 deletions(-)
|
|
|
ba407d |
|
|
|
ba407d |
diff --git a/man/en/sos-report.1 b/man/en/sos-report.1
|
|
|
ba407d |
index 81005959..51cf3436 100644
|
|
|
ba407d |
--- a/man/en/sos-report.1
|
|
|
ba407d |
+++ b/man/en/sos-report.1
|
|
|
ba407d |
@@ -17,6 +17,7 @@ sosreport \- Collect and package diagnostic and support data
|
|
|
ba407d |
[--label label] [--case-id id]\fR
|
|
|
ba407d |
[--threads threads]\fR
|
|
|
ba407d |
[--plugin-timeout TIMEOUT]\fR
|
|
|
ba407d |
+ [--cmd-timeout TIMEOUT]\fR
|
|
|
ba407d |
[-s|--sysroot SYSROOT]\fR
|
|
|
ba407d |
[-c|--chroot {auto|always|never}\fR
|
|
|
ba407d |
[--tmp-dir directory]\fR
|
|
|
ba407d |
@@ -247,7 +248,7 @@ Specify a timeout in seconds to allow each plugin to run for. A value of 0
|
|
|
ba407d |
means no timeout will be set. A value of -1 is used to indicate the default
|
|
|
ba407d |
timeout of 300 seconds.
|
|
|
ba407d |
|
|
|
ba407d |
-Note that this options sets the timeout for all plugins. If you want to set
|
|
|
ba407d |
+Note that this option sets the timeout for all plugins. If you want to set
|
|
|
ba407d |
a timeout for a specific plugin, use the 'timeout' plugin option available to
|
|
|
ba407d |
all plugins - e.g. '-k logs.timeout=600'.
|
|
|
ba407d |
|
|
|
ba407d |
@@ -255,6 +256,21 @@ The plugin-specific timeout option will override this option. For example, using
|
|
|
ba407d |
\'--plugin-timeout=60 -k logs.timeout=600\' will set a timeout of 600 seconds for
|
|
|
ba407d |
the logs plugin and 60 seconds for all other enabled plugins.
|
|
|
ba407d |
.TP
|
|
|
ba407d |
+.B \--cmd-timeout TIMEOUT
|
|
|
ba407d |
+Specify a timeout limit in seconds for a command execution. Same defaults logic
|
|
|
ba407d |
+from --plugin-timeout applies here.
|
|
|
ba407d |
+
|
|
|
ba407d |
+This option sets the command timeout for all plugins. If you want to set a cmd
|
|
|
ba407d |
+timeout for a specific plugin, use the 'cmd-timeout' plugin option available to
|
|
|
ba407d |
+all plugins - e.g. '-k logs.cmd-timeout=600'.
|
|
|
ba407d |
+
|
|
|
ba407d |
+Again, the same plugin/global precedence logic as for --plugin-timeout applies
|
|
|
ba407d |
+here.
|
|
|
ba407d |
+
|
|
|
ba407d |
+Note that setting --cmd-timeout (or -k logs.cmd-timeout) high should be followed
|
|
|
ba407d |
+by increasing the --plugin-timeout equivalent, otherwise the plugin can easily
|
|
|
ba407d |
+timeout on slow commands execution.
|
|
|
ba407d |
+.TP
|
|
|
ba407d |
.B \--case-id NUMBER
|
|
|
ba407d |
Specify a case identifier to associate with the archive.
|
|
|
ba407d |
Identifiers may include alphanumeric characters, commas and periods ('.').
|
|
|
ba407d |
diff --git a/sos/collector/__init__.py b/sos/collector/__init__.py
|
|
|
ba407d |
index 406c8f35..1ae73508 100644
|
|
|
ba407d |
--- a/sos/collector/__init__.py
|
|
|
ba407d |
+++ b/sos/collector/__init__.py
|
|
|
ba407d |
@@ -82,6 +82,7 @@ class SoSCollector(SoSComponent):
|
|
|
ba407d |
'password_per_node': False,
|
|
|
ba407d |
'plugin_options': [],
|
|
|
ba407d |
'plugin_timeout': None,
|
|
|
ba407d |
+ 'cmd_timeout': None,
|
|
|
ba407d |
'preset': '',
|
|
|
ba407d |
'save_group': '',
|
|
|
ba407d |
'since': '',
|
|
|
ba407d |
@@ -276,6 +277,8 @@ class SoSCollector(SoSComponent):
|
|
|
ba407d |
help='Do not collect env vars in sosreports')
|
|
|
ba407d |
sos_grp.add_argument('--plugin-timeout', type=int, default=None,
|
|
|
ba407d |
help='Set the global plugin timeout value')
|
|
|
ba407d |
+ sos_grp.add_argument('--cmd-timeout', type=int, default=None,
|
|
|
ba407d |
+ help='Set the global command timeout value')
|
|
|
ba407d |
sos_grp.add_argument('--since', default=None,
|
|
|
ba407d |
help=('Escapes archived files older than date. '
|
|
|
ba407d |
'This will also affect --all-logs. '
|
|
|
ba407d |
diff --git a/sos/collector/sosnode.py b/sos/collector/sosnode.py
|
|
|
ba407d |
index a1679655..dbbee12e 100644
|
|
|
ba407d |
--- a/sos/collector/sosnode.py
|
|
|
ba407d |
+++ b/sos/collector/sosnode.py
|
|
|
ba407d |
@@ -664,6 +664,11 @@ class SosNode():
|
|
|
ba407d |
'--skip-files=%s' % (quote(self.opts.skip_files))
|
|
|
ba407d |
)
|
|
|
ba407d |
|
|
|
ba407d |
+ if self.check_sos_version('4.2'):
|
|
|
ba407d |
+ if self.opts.cmd_timeout:
|
|
|
ba407d |
+ sos_opts.append('--cmd-timeout=%s'
|
|
|
ba407d |
+ % quote(str(self.opts.cmd_timeout)))
|
|
|
ba407d |
+
|
|
|
ba407d |
sos_cmd = sos_cmd.replace(
|
|
|
ba407d |
'sosreport',
|
|
|
ba407d |
os.path.join(self.host.sos_bin_path, self.sos_bin)
|
|
|
ba407d |
diff --git a/sos/options.py b/sos/options.py
|
|
|
ba407d |
index b82a7d36..1eda55d6 100644
|
|
|
ba407d |
--- a/sos/options.py
|
|
|
ba407d |
+++ b/sos/options.py
|
|
|
ba407d |
@@ -283,7 +283,8 @@ class SoSOptions():
|
|
|
ba407d |
if name in ("add_preset", "del_preset", "desc", "note"):
|
|
|
ba407d |
return False
|
|
|
ba407d |
# Exception list for options that still need to be reported when 0
|
|
|
ba407d |
- if name in ['log_size', 'plugin_timeout'] and value == 0:
|
|
|
ba407d |
+ if name in ['log_size', 'plugin_timeout', 'cmd_timeout'] \
|
|
|
ba407d |
+ and value == 0:
|
|
|
ba407d |
return True
|
|
|
ba407d |
return has_value(name, value)
|
|
|
ba407d |
|
|
|
ba407d |
diff --git a/sos/report/__init__.py b/sos/report/__init__.py
|
|
|
ba407d |
index 25478ba7..945d0fc1 100644
|
|
|
ba407d |
--- a/sos/report/__init__.py
|
|
|
ba407d |
+++ b/sos/report/__init__.py
|
|
|
ba407d |
@@ -107,6 +107,7 @@ class SoSReport(SoSComponent):
|
|
|
ba407d |
'only_plugins': [],
|
|
|
ba407d |
'preset': 'auto',
|
|
|
ba407d |
'plugin_timeout': 300,
|
|
|
ba407d |
+ 'cmd_timeout': 300,
|
|
|
ba407d |
'profiles': [],
|
|
|
ba407d |
'since': None,
|
|
|
ba407d |
'verify': False,
|
|
|
ba407d |
@@ -266,6 +267,8 @@ class SoSReport(SoSComponent):
|
|
|
ba407d |
help="A preset identifier", default="auto")
|
|
|
ba407d |
report_grp.add_argument("--plugin-timeout", default=None,
|
|
|
ba407d |
help="set a timeout for all plugins")
|
|
|
ba407d |
+ report_grp.add_argument("--cmd-timeout", default=None,
|
|
|
ba407d |
+ help="set a command timeout for all plugins")
|
|
|
ba407d |
report_grp.add_argument("-p", "--profile", "--profiles",
|
|
|
ba407d |
action="extend", dest="profiles", type=str,
|
|
|
ba407d |
default=[],
|
|
|
ba407d |
@@ -709,7 +712,7 @@ class SoSReport(SoSComponent):
|
|
|
ba407d |
|
|
|
ba407d |
self.ui_log.info(_("The following plugin options are available:"))
|
|
|
ba407d |
for (plug, plugname, optname, optparm) in self.all_options:
|
|
|
ba407d |
- if optname in ('timeout', 'postproc'):
|
|
|
ba407d |
+ if optname in ('timeout', 'postproc', 'cmd-timeout'):
|
|
|
ba407d |
continue
|
|
|
ba407d |
# format option value based on its type (int or bool)
|
|
|
ba407d |
if type(optparm["enabled"]) == bool:
|
|
|
ba407d |
diff --git a/sos/report/plugins/__init__.py b/sos/report/plugins/__init__.py
|
|
|
ba407d |
index 02625eb1..779119af 100644
|
|
|
ba407d |
--- a/sos/report/plugins/__init__.py
|
|
|
ba407d |
+++ b/sos/report/plugins/__init__.py
|
|
|
ba407d |
@@ -472,6 +472,9 @@ class Plugin(object):
|
|
|
ba407d |
_default_plug_opts = [
|
|
|
ba407d |
('timeout', 'Timeout in seconds for plugin. The default value (-1) ' +
|
|
|
ba407d |
'defers to the general plugin timeout, 300 seconds', 'fast', -1),
|
|
|
ba407d |
+ ('cmd-timeout', 'Timeout in seconds for a command execution. The ' +
|
|
|
ba407d |
+ 'default value (-1) defers to the general cmd timeout, 300 ' +
|
|
|
ba407d |
+ 'seconds', 'fast', -1),
|
|
|
ba407d |
('postproc', 'Enable post-processing collected plugin data', 'fast',
|
|
|
ba407d |
True)
|
|
|
ba407d |
]
|
|
|
ba407d |
@@ -532,16 +535,15 @@ class Plugin(object):
|
|
|
ba407d |
self.manifest.add_list('commands', [])
|
|
|
ba407d |
self.manifest.add_list('files', [])
|
|
|
ba407d |
|
|
|
ba407d |
- @property
|
|
|
ba407d |
- def timeout(self):
|
|
|
ba407d |
- """Returns either the default plugin timeout value, the value as
|
|
|
ba407d |
- provided on the commandline via -k plugin.timeout=value, or the value
|
|
|
ba407d |
- of the global --plugin-timeout option.
|
|
|
ba407d |
+ def timeout_from_options(self, optname, plugoptname, default_timeout):
|
|
|
ba407d |
+ """Returns either the default [plugin|cmd] timeout value, the value as
|
|
|
ba407d |
+ provided on the commandline via -k plugin.[|cmd-]timeout=value, or the
|
|
|
ba407d |
+ value of the global --[plugin|cmd]-timeout option.
|
|
|
ba407d |
"""
|
|
|
ba407d |
_timeout = None
|
|
|
ba407d |
try:
|
|
|
ba407d |
- opt_timeout = self.get_option('plugin_timeout')
|
|
|
ba407d |
- own_timeout = int(self.get_option('timeout'))
|
|
|
ba407d |
+ opt_timeout = self.get_option(optname)
|
|
|
ba407d |
+ own_timeout = int(self.get_option(plugoptname))
|
|
|
ba407d |
if opt_timeout is None:
|
|
|
ba407d |
_timeout = own_timeout
|
|
|
ba407d |
elif opt_timeout is not None and own_timeout == -1:
|
|
|
ba407d |
@@ -551,10 +553,30 @@ class Plugin(object):
|
|
|
ba407d |
else:
|
|
|
ba407d |
return None
|
|
|
ba407d |
except ValueError:
|
|
|
ba407d |
- return self.plugin_timeout # Default to known safe value
|
|
|
ba407d |
+ return default_timeout # Default to known safe value
|
|
|
ba407d |
if _timeout is not None and _timeout > -1:
|
|
|
ba407d |
return _timeout
|
|
|
ba407d |
- return self.plugin_timeout
|
|
|
ba407d |
+ return default_timeout
|
|
|
ba407d |
+
|
|
|
ba407d |
+ @property
|
|
|
ba407d |
+ def timeout(self):
|
|
|
ba407d |
+ """Returns either the default plugin timeout value, the value as
|
|
|
ba407d |
+ provided on the commandline via -k plugin.timeout=value, or the value
|
|
|
ba407d |
+ of the global --plugin-timeout option.
|
|
|
ba407d |
+ """
|
|
|
ba407d |
+ _timeout = self.timeout_from_options('plugin_timeout', 'timeout',
|
|
|
ba407d |
+ self.plugin_timeout)
|
|
|
ba407d |
+ return _timeout
|
|
|
ba407d |
+
|
|
|
ba407d |
+ @property
|
|
|
ba407d |
+ def cmdtimeout(self):
|
|
|
ba407d |
+ """Returns either the default command timeout value, the value as
|
|
|
ba407d |
+ provided on the commandline via -k plugin.cmd-timeout=value, or the
|
|
|
ba407d |
+ value of the global --cmd-timeout option.
|
|
|
ba407d |
+ """
|
|
|
ba407d |
+ _cmdtimeout = self.timeout_from_options('cmd_timeout', 'cmd-timeout',
|
|
|
ba407d |
+ self.cmd_timeout)
|
|
|
ba407d |
+ return _cmdtimeout
|
|
|
ba407d |
|
|
|
ba407d |
def set_timeout_hit(self):
|
|
|
ba407d |
self._timeout_hit = True
|
|
|
ba407d |
@@ -1235,8 +1257,8 @@ class Plugin(object):
|
|
|
ba407d |
"""
|
|
|
ba407d |
|
|
|
ba407d |
global_options = (
|
|
|
ba407d |
- 'all_logs', 'allow_system_changes', 'log_size', 'plugin_timeout',
|
|
|
ba407d |
- 'since', 'verify'
|
|
|
ba407d |
+ 'all_logs', 'allow_system_changes', 'cmd_timeout', 'log_size',
|
|
|
ba407d |
+ 'plugin_timeout', 'since', 'verify'
|
|
|
ba407d |
)
|
|
|
ba407d |
|
|
|
ba407d |
if optionname in global_options:
|
|
|
ba407d |
@@ -1505,7 +1527,7 @@ class Plugin(object):
|
|
|
ba407d |
'tags': _spec_tags
|
|
|
ba407d |
})
|
|
|
ba407d |
|
|
|
ba407d |
- def add_blockdev_cmd(self, cmds, devices='block', timeout=300,
|
|
|
ba407d |
+ def add_blockdev_cmd(self, cmds, devices='block', timeout=None,
|
|
|
ba407d |
sizelimit=None, chroot=True, runat=None, env=None,
|
|
|
ba407d |
binary=False, prepend_path=None, whitelist=[],
|
|
|
ba407d |
blacklist=[], tags=[]):
|
|
|
ba407d |
@@ -1569,7 +1591,7 @@ class Plugin(object):
|
|
|
ba407d |
whitelist=whitelist, blacklist=blacklist,
|
|
|
ba407d |
tags=_dev_tags)
|
|
|
ba407d |
|
|
|
ba407d |
- def _add_device_cmd(self, cmds, devices, timeout=300, sizelimit=None,
|
|
|
ba407d |
+ def _add_device_cmd(self, cmds, devices, timeout=None, sizelimit=None,
|
|
|
ba407d |
chroot=True, runat=None, env=None, binary=False,
|
|
|
ba407d |
prepend_path=None, whitelist=[], blacklist=[],
|
|
|
ba407d |
tags=[]):
|
|
|
ba407d |
@@ -1627,7 +1649,7 @@ class Plugin(object):
|
|
|
ba407d |
changes=soscmd.changes)
|
|
|
ba407d |
|
|
|
ba407d |
def add_cmd_output(self, cmds, suggest_filename=None,
|
|
|
ba407d |
- root_symlink=None, timeout=cmd_timeout, stderr=True,
|
|
|
ba407d |
+ root_symlink=None, timeout=None, stderr=True,
|
|
|
ba407d |
chroot=True, runat=None, env=None, binary=False,
|
|
|
ba407d |
sizelimit=None, pred=None, subdir=None,
|
|
|
ba407d |
changes=False, foreground=False, tags=[]):
|
|
|
ba407d |
@@ -1849,7 +1871,7 @@ class Plugin(object):
|
|
|
ba407d |
self._log_debug("added string ...'%s' as '%s'" % (summary, filename))
|
|
|
ba407d |
|
|
|
ba407d |
def _collect_cmd_output(self, cmd, suggest_filename=None,
|
|
|
ba407d |
- root_symlink=False, timeout=cmd_timeout,
|
|
|
ba407d |
+ root_symlink=False, timeout=None,
|
|
|
ba407d |
stderr=True, chroot=True, runat=None, env=None,
|
|
|
ba407d |
binary=False, sizelimit=None, subdir=None,
|
|
|
ba407d |
changes=False, foreground=False, tags=[]):
|
|
|
ba407d |
@@ -1883,6 +1905,8 @@ class Plugin(object):
|
|
|
ba407d |
if self._timeout_hit:
|
|
|
ba407d |
return
|
|
|
ba407d |
|
|
|
ba407d |
+ if timeout is None:
|
|
|
ba407d |
+ timeout = self.cmdtimeout
|
|
|
ba407d |
_tags = []
|
|
|
ba407d |
|
|
|
ba407d |
if isinstance(tags, str):
|
|
|
ba407d |
@@ -1975,7 +1999,7 @@ class Plugin(object):
|
|
|
ba407d |
return result
|
|
|
ba407d |
|
|
|
ba407d |
def collect_cmd_output(self, cmd, suggest_filename=None,
|
|
|
ba407d |
- root_symlink=False, timeout=cmd_timeout,
|
|
|
ba407d |
+ root_symlink=False, timeout=None,
|
|
|
ba407d |
stderr=True, chroot=True, runat=None, env=None,
|
|
|
ba407d |
binary=False, sizelimit=None, pred=None,
|
|
|
ba407d |
subdir=None, tags=[]):
|
|
|
ba407d |
@@ -2044,7 +2068,7 @@ class Plugin(object):
|
|
|
ba407d |
tags=tags
|
|
|
ba407d |
)
|
|
|
ba407d |
|
|
|
ba407d |
- def exec_cmd(self, cmd, timeout=cmd_timeout, stderr=True, chroot=True,
|
|
|
ba407d |
+ def exec_cmd(self, cmd, timeout=None, stderr=True, chroot=True,
|
|
|
ba407d |
runat=None, env=None, binary=False, pred=None,
|
|
|
ba407d |
foreground=False, container=False, quotecmd=False):
|
|
|
ba407d |
"""Execute a command right now and return the output and status, but
|
|
|
ba407d |
@@ -2095,6 +2119,9 @@ class Plugin(object):
|
|
|
ba407d |
if not self.test_predicate(cmd=True, pred=pred):
|
|
|
ba407d |
return _default
|
|
|
ba407d |
|
|
|
ba407d |
+ if timeout is None:
|
|
|
ba407d |
+ timeout = self.cmdtimeout
|
|
|
ba407d |
+
|
|
|
ba407d |
if chroot or self.commons['cmdlineopts'].chroot == 'always':
|
|
|
ba407d |
root = self.sysroot
|
|
|
ba407d |
else:
|
|
|
ba407d |
@@ -2331,7 +2358,7 @@ class Plugin(object):
|
|
|
ba407d |
|
|
|
ba407d |
def add_journal(self, units=None, boot=None, since=None, until=None,
|
|
|
ba407d |
lines=None, allfields=False, output=None,
|
|
|
ba407d |
- timeout=cmd_timeout, identifier=None, catalog=None,
|
|
|
ba407d |
+ timeout=None, identifier=None, catalog=None,
|
|
|
ba407d |
sizelimit=None, pred=None, tags=[]):
|
|
|
ba407d |
"""Collect journald logs from one of more units.
|
|
|
ba407d |
|
|
|
ba407d |
--
|
|
|
ba407d |
2.26.3
|
|
|
ba407d |
|