mirror of
				https://github.com/xcat2/confluent.git
				synced 2025-10-25 00:15:48 +00:00 
			
		
		
		
	Pythons default handling of Ctrl-C is not in line with most command line utilities. Wrap the exception and imitate more conventional behavior.
		
			
				
	
	
		
			216 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			216 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #!/usr/bin/env python
 | |
| # vim: tabstop=4 shiftwidth=4 softtabstop=4
 | |
| 
 | |
| # Copyright 2015 Lenovo
 | |
| #
 | |
| # 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 csv
 | |
| import optparse
 | |
| import os
 | |
| import sys
 | |
| import time
 | |
| 
 | |
| path = os.path.dirname(os.path.realpath(__file__))
 | |
| path = os.path.realpath(os.path.join(path, '..', 'lib', 'python'))
 | |
| if path.startswith('/opt'):
 | |
|     sys.path.append(path)
 | |
| 
 | |
| import confluent.client as client
 | |
| 
 | |
| sensorcollections = {
 | |
|     'all': 'sensors/hardware/all/all',
 | |
|     'temperature': 'sensors/hardware/temperature/all',
 | |
|     'temp': 'sensors/hardware/temperature/all',
 | |
|     'power': 'sensors/hardware/power/all',
 | |
|     'fans': 'sensors/hardware/fans/all',
 | |
|     'fanspeed': 'sensors/hardware/fans/all',
 | |
| }
 | |
| 
 | |
| 
 | |
| argparser = optparse.OptionParser(
 | |
|     usage="Usage: %prog [options] noderange [sensor(s)")
 | |
| argparser.add_option('-i', '--interval', type='int',
 | |
|                      help='Interval to do repeated samples over')
 | |
| argparser.add_option('-n', '--numreadings', type='int',
 | |
|                      help='Number of readings to gather')
 | |
| argparser.add_option('-c', '--csv', action='store_true',
 | |
|                      help='Output in CSV format')
 | |
| (options, args) = argparser.parse_args()
 | |
| repeatmode = False
 | |
| if options.interval:
 | |
|     repeatmode = True
 | |
| if options.numreadings:
 | |
|     repeatmode = options.numreadings
 | |
|     if options.interval is None:
 | |
|         options.interval = 1
 | |
| try:
 | |
|     noderange = args[0]
 | |
| except IndexError:
 | |
|     argparser.print_usage()
 | |
|     sys.exit(1)
 | |
| sensors = []
 | |
| for sensorgroup in args[1:]:
 | |
|     for sensor in sensorgroup.split(','):
 | |
|         sensor = sensor.replace(' ', '_').lower()
 | |
|         if '/' not in sensor:
 | |
|             if sensor in sensorcollections:
 | |
|                 sensors.append(sensorcollections[sensor])
 | |
|             else:
 | |
|                 sensors.append('sensors/hardware/all/' + sensor)
 | |
| if not sensors:
 | |
|     sensors = ['sensors/hardware/all/all']
 | |
| 
 | |
| session = client.Command()
 | |
| exitcode = 0
 | |
| sensorheaders = {}
 | |
| 
 | |
| 
 | |
| def sensorpass(showout=True, appendtime=False):
 | |
|     global exitcode
 | |
|     resultdata = {}
 | |
|     for reqsensor in sensors:
 | |
|         for reading in session.read(
 | |
|                 '/noderange/' + noderange + '/' + reqsensor):
 | |
|             if 'error' in reading:
 | |
|                 sys.stderr.write('Error: {0}\n'.format(reading['error']))
 | |
|                 if 'errorcode' in reading:
 | |
|                     exitcode |= exitcode
 | |
|                 else:
 | |
|                     exitcode |= 1
 | |
|             if 'databynode' not in reading:
 | |
|                 continue
 | |
|             reading = reading['databynode']
 | |
|             for node in reading:
 | |
|                 if node not in resultdata:
 | |
|                     resultdata[node] = {}
 | |
|                 if 'error' in reading[node]:
 | |
|                     sys.stderr.write(
 | |
|                         '{0}: Error: {1}\n'.format(node,
 | |
|                                                    reading[node]['error']))
 | |
|                 if 'sensors' not in reading[node]:
 | |
|                     continue
 | |
|                 for sensedata in reading[node]['sensors']:
 | |
|                     for redundant_state in ('Non-Critical', 'Critical'):
 | |
|                         try:
 | |
|                             sensedata['states'].remove(redundant_state)
 | |
|                         except ValueError:
 | |
|                             pass
 | |
|                     resultdata[node][sensedata['name']] = sensedata
 | |
|                     sensorname = sensedata['name']
 | |
|                     sensorheaders[sensorname] = sensorname
 | |
|                     if sensedata['units'] not in (None, u''):
 | |
|                         sensorheaders[sensorname] += u' ({0})'.format(
 | |
|                             sensedata['units'])
 | |
|                     if showout:
 | |
|                         if sensedata['value'] is None:
 | |
|                             showval = ''
 | |
|                         else:
 | |
|                             showval = u' {0} '.format(sensedata['value'])
 | |
|                         if sensedata['units'] not in (None, u''):
 | |
|                             showval += sensedata['units']
 | |
|                         if sensedata['health'] != 'ok':
 | |
|                             datadescription = [sensedata['health']]
 | |
|                         else:
 | |
|                             datadescription = []
 | |
|                         datadescription.extend(sensedata['states'])
 | |
|                         if datadescription:
 | |
|                             if showval == '':
 | |
|                                 showval += u' {0}'.format(
 | |
|                                     ','.join(datadescription))
 | |
|                             else:
 | |
|                                 showval += u' ({0})'.format(
 | |
|                                     ','.join(datadescription))
 | |
|                         if appendtime:
 | |
|                             showval += ' @' + time.strftime(
 | |
|                                 '%Y-%m-%dT%H:%M:%S')
 | |
|                         print(u'{0}: {1}:{2}'.format(
 | |
|                             node, sensedata['name'], showval).encode('utf-8'))
 | |
|     return resultdata
 | |
| 
 | |
| 
 | |
| def format_csv(csvwriter, orderedsensors, resdata, showtime=True):
 | |
|     for nodekey in resdata:
 | |
|         if showtime:
 | |
|             rowdata = [time.strftime('%Y-%m-%dT%H:%M:%S'), nodekey]
 | |
|         else:
 | |
|             rowdata = [nodekey]
 | |
|         for sensorkey in orderedsensors:
 | |
|             try:
 | |
|                 datum = resdata[nodekey][sensorkey]['value']
 | |
|                 if datum is None:
 | |
|                     if resdata[nodekey][sensorkey]['health'] != 'ok':
 | |
|                         datum = resdata[nodekey][sensorkey]['health']
 | |
|                     else:
 | |
|                         datum = ''
 | |
|                     if resdata[nodekey][sensorkey]['states']:
 | |
|                         healthstates = ','.join(
 | |
|                             resdata[nodekey][sensorkey]['states'])
 | |
|                         if datum != '':
 | |
|                             datum = ','.join([datum, healthstates])
 | |
|                         else:
 | |
|                             datum = healthstates
 | |
|                 rowdata.append(datum)
 | |
|             except KeyError:
 | |
|                 rowdata.append('N/A')
 | |
|         csvwriter.writerow(rowdata)
 | |
| 
 | |
| 
 | |
| def main():
 | |
|     linebyline = True
 | |
|     headernames = []
 | |
|     orderedsensors = []
 | |
|     csvwriter = None
 | |
|     if options.interval or options.csv:
 | |
|         resdata = sensorpass(False)
 | |
|         for name in sensorheaders:
 | |
|             orderedsensors.append(name)
 | |
|         orderedsensors.sort()
 | |
|         for name in orderedsensors:
 | |
|             headernames.append(sensorheaders[name].encode('utf-8'))
 | |
|     if options.csv:
 | |
|         linebyline = False
 | |
|         csvwriter = csv.writer(sys.stdout)
 | |
|         if options.interval:
 | |
|             csvwriter.writerow(['time', 'node'] + headernames)
 | |
|         else:
 | |
|             csvwriter.writerow(['node'] + headernames)
 | |
|     if options.interval:
 | |
|         # first do a pass to swallow up probable causes of uneven timing
 | |
|         # for example if some have sdrs fetch and others not, this should
 | |
|         # get a common baseline going
 | |
|         while True:
 | |
|             nextstart = os.times()[4] + options.interval
 | |
|             resdata = sensorpass(linebyline, True)
 | |
|             if options.csv:
 | |
|                 format_csv(csvwriter, orderedsensors, resdata)
 | |
|             if options.numreadings:
 | |
|                 options.numreadings -= 1
 | |
|                 if options.numreadings <= 0:
 | |
|                     sys.exit(exitcode)
 | |
|             sleeptime = nextstart - os.times()[4]
 | |
|             if sleeptime > 0:
 | |
|                 time.sleep(nextstart - os.times()[4])
 | |
|     else:
 | |
|         if options.csv:
 | |
|             format_csv(csvwriter, orderedsensors, resdata, showtime=False)
 | |
|         else:
 | |
|             sensorpass(True)
 | |
| 
 | |
| 
 | |
| try:
 | |
|     main()
 | |
| except KeyboardInterrupt:
 | |
|     print('')
 | |
|     sys.exit(0)
 |