aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--sos/plugins/pcp.py162
1 files changed, 162 insertions, 0 deletions
diff --git a/sos/plugins/pcp.py b/sos/plugins/pcp.py
new file mode 100644
index 00000000..a0de9152
--- /dev/null
+++ b/sos/plugins/pcp.py
@@ -0,0 +1,162 @@
+# Copyright (C) 2014 Michele Baldessari <michele at acksyn.org>
+
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+
+from sos.plugins import Plugin, RedHatPlugin, DebianPlugin
+from sos.utilities import sos_get_command_output
+import os
+import os.path
+
+
+class Pcp(Plugin, RedHatPlugin, DebianPlugin):
+ """ Collect Performance Co-Pilot data
+ """
+
+ plugin_name = 'pcp'
+ packages = ('pcp',)
+
+ pcp_conffile = '/etc/pcp.conf'
+ option_list = [("all_pcplogs", "gather all logged archive files", "",
+ False)]
+
+ # size-limit total PCP log data collected by default (MB)
+ pcplog_totalsize = 100
+
+ pcp_sysconf_dir = None
+ pcp_var_dir = None
+ pcp_log_dir = None
+
+ pcp_hostname = ''
+
+ def get_size(self, path):
+ total_size = 0
+ for dirpath, dirnames, filenames in os.walk(path):
+ for f in filenames:
+ fp = os.path.join(dirpath, f)
+ total_size += os.path.getsize(fp)
+ return total_size
+
+ def pcp_parse_conffile(self):
+ try:
+ pcpconf = open(self.pcp_conffile, "r")
+ lines = pcpconf.readlines()
+ pcpconf.close()
+ except:
+ return False
+ env_vars = {}
+ for line in lines:
+ if line.startswith('#'):
+ continue
+ try:
+ (key, value) = line.strip().split('=')
+ env_vars[key] = value
+ except:
+ pass
+
+ try:
+ self.pcp_sysconf_dir = env_vars['PCP_SYSCONF_DIR']
+ self.pcp_var_dir = env_vars['PCP_VAR_DIR']
+ self.pcp_log_dir = env_vars['PCP_LOG_DIR']
+ except:
+ # Fail if all three env variables are not found
+ return False
+
+ return True
+
+ def setup(self):
+ if self.get_option("all_pcplogs"):
+ self.pcplog_totalsize = 0
+
+ if not self.pcp_parse_conffile():
+ self.log_warn("could not parse %s" % self.pcp_conffile)
+ return
+
+ # Add PCP_SYSCONF_DIR (/etc/pcp) and PCP_VAR_DIR (/var/lib/pcp/config)
+ # unconditionally. Obviously if someone messes up their /etc/pcp.conf
+ # in a ridiculous way (i.e. setting PCP_SYSCONF_DIR to '/') this will
+ # break badly.
+ self.add_copy_spec(self.pcp_sysconf_dir)
+ var_conf_dir = os.path.join(self.pcp_var_dir, 'config')
+ self.add_copy_spec(var_conf_dir)
+
+ # We explicitely avoid /var/lib/pcp/config/{pmchart,pmlogconf,pmieconf,
+ # pmlogrewrite} as in 99% of the cases they are just copies from the
+ # rpms. It does not make up for a lot of size but it contains many
+ # files
+ self.add_forbidden_path(os.path.join(var_conf_dir, 'pmchart'))
+ self.add_forbidden_path(os.path.join(var_conf_dir, 'pmlogconf'))
+ self.add_forbidden_path(os.path.join(var_conf_dir, 'pmieconf'))
+ self.add_forbidden_path(os.path.join(var_conf_dir, 'pmlogrewrite'))
+
+ # The *default* directory structure for pmlogger is the following:
+ # Dir: PCP_LOG_DIR/pmlogger/HOST/ (we only collect the HOST data
+ # itself)
+ # - YYYYMMDD.HH.MM.{N,N.index,N.meta} N in [0,1,...]
+ # - Latest
+ # - pmlogger.{log,log.prior}
+ #
+ # Can be changed via configuration in PCP_SYSCONF_DIR/pmlogger/control
+ # As a default strategy, collect PCP_LOG_DIR/pmlogger/* only if the
+ # total size is moderately small: < 100MB. Override is possible via
+ # the 'all_pcplogs' option.
+ # FIXME: Doing a recursive size check because add_copy_spec_limit
+ # won't work for directory trees. I.e. we can't say fetch /foo/bar/
+ # only if it is < 100MB. To be killed once the Plugin base class will
+ # add a method for this use case via issue #281
+ ret = sos_get_command_output('hostname')
+ if ret['status'] == 0:
+ # Make sure that if output is not a string we do not barf
+ try:
+ self.pcp_hostname = ret['output'].strip()
+ except:
+ pass
+
+ # Make sure we only add PCP_LOG_DIR/pmlogger/`hostname` if hostname
+ # is set, otherwise we'd collect everything
+ if self.pcp_hostname != '':
+ path = os.path.join(self.pcp_log_dir, 'pmlogger',
+ self.pcp_hostname)
+ dirsize = self.get_size(path)
+ max_mb_size = self.pcplog_totalsize * 1024 * 1024
+ # If explicitely asked collect all logs, otherwise only if < 100MB
+ # in total
+ if self.pcplog_totalsize == 0 or dirsize < max_mb_size:
+ if os.path.isdir(path):
+ self.add_copy_spec(path)
+ else:
+ self.log_warn("%s not found" % path)
+ else:
+ self.log_warn("skipped %s. Size %d bigger than %d" % (path,
+ dirsize, max_mb_size))
+ else:
+ self.log_warn("pcp_hostname was not set. Skipping.")
+
+ # Collect PCP_LOG_DIR/pmcd and PCP_LOG_DIR/NOTICES
+ self.add_copy_spec(os.path.join(self.pcp_log_dir, 'pmcd'))
+ self.add_copy_spec(os.path.join(self.pcp_log_dir, 'NOTICES*'))
+
+ # Collect PCP_VAR_DIR/pmns
+ self.add_copy_spec(os.path.join(self.pcp_var_dir, 'pmns'))
+
+ # Also collect any other log and config files (as suggested by fche)
+ self.add_copy_spec(os.path.join(self.pcp_log_dir, '*/*.log*'))
+ self.add_copy_spec(os.path.join(self.pcp_log_dir, '*/*/*.log*'))
+ self.add_copy_spec(os.path.join(self.pcp_log_dir, '*/*/config*'))
+
+ # Need to get the current status of the PCP infrastructure
+ self.add_cmd_output("pcp")
+
+
+# vim: et ts=4 sw=4