# -*- coding: utf-8 -*- # # This file is part of Glances. # # Copyright (C) 2014 Nicolargo # # Glances is free software; you can redistribute it and/or modify # it under the terms of the GNU Lesser General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # Glances 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 Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public License # along with this program. If not, see . """File system plugin.""" # Glances libs from glances.plugins.glances_plugin import GlancesPlugin # PSutil lib for local grab import psutil # SNMP OID # The snmpd.conf needs to be edited. # Add the following to enable it on all disk # ... # includeAllDisks 10% # ... # The OIDs are as follows (for the first disk) # Path where the disk is mounted: .1.3.6.1.4.1.2021.9.1.2.1 # Path of the device for the partition: .1.3.6.1.4.1.2021.9.1.3.1 # Total size of the disk/partion (kBytes): .1.3.6.1.4.1.2021.9.1.6.1 # Available space on the disk: .1.3.6.1.4.1.2021.9.1.7.1 # Used space on the disk: .1.3.6.1.4.1.2021.9.1.8.1 # Percentage of space used on disk: .1.3.6.1.4.1.2021.9.1.9.1 # Percentage of inodes used on disk: .1.3.6.1.4.1.2021.9.1.10.1 snmp_oid = {'default': {'mnt_point': '1.3.6.1.4.1.2021.9.1.2', 'device_name': '1.3.6.1.4.1.2021.9.1.3', 'size': '1.3.6.1.4.1.2021.9.1.6', 'used': '1.3.6.1.4.1.2021.9.1.8', 'percent': '1.3.6.1.4.1.2021.9.1.9'}, 'windows': {'mnt_point': '1.3.6.1.2.1.25.2.3.1.3', 'alloc_unit': '1.3.6.1.2.1.25.2.3.1.4', 'size': '1.3.6.1.2.1.25.2.3.1.5', 'used': '1.3.6.1.2.1.25.2.3.1.6'}} snmp_oid['esxi'] = snmp_oid['windows'] # Define the history items list # All items in this list will be historised if the --enable-history tag is set # 'color' define the graph color in #RGB format items_history_list = [{'name': 'percent', 'color': '#00FF00'}] class Plugin(GlancesPlugin): """Glances' file system plugin. stats is a list """ def __init__(self, args=None): """Init the plugin.""" GlancesPlugin.__init__(self, args=args, items_history_list=items_history_list) # We want to display the stat in the curse interface self.display_curse = True # Init the stats self.reset() def reset(self): """Reset/init the stats.""" self.stats = [] def update(self): """Update the FS stats using the input method.""" # Reset the list self.reset() if self.get_input() == 'local': # Update stats using the standard system lib # Grab the stats using the PsUtil disk_partitions # If 'all'=False return physical devices only (e.g. hard disks, cd-rom drives, USB keys) # and ignore all others (e.g. memory partitions such as /dev/shm) try: fs_stat = psutil.disk_partitions(all=False) except UnicodeDecodeError: return self.stats # Loop over fs for fs in range(len(fs_stat)): fs_current = {} fs_current['device_name'] = fs_stat[fs].device fs_current['fs_type'] = fs_stat[fs].fstype fs_current['mnt_point'] = fs_stat[fs].mountpoint # Grab the disk usage try: fs_usage = psutil.disk_usage(fs_current['mnt_point']) except OSError: # Correct issue #346 # Disk is ejected during the command continue fs_current['size'] = fs_usage.total fs_current['used'] = fs_usage.used # fs_current['avail'] = fs_usage.free fs_current['percent'] = fs_usage.percent self.stats.append(fs_current) elif self.get_input() == 'snmp': # Update stats using SNMP # SNMP bulk command to get all file system in one shot try: fs_stat = self.set_stats_snmp(snmp_oid=snmp_oid[self.get_short_system_name()], bulk=True) except KeyError: fs_stat = self.set_stats_snmp(snmp_oid=snmp_oid['default'], bulk=True) # Loop over fs if self.get_short_system_name() in ('windows', 'esxi'): # Windows or ESXi tips for fs in fs_stat: # Memory stats are grabed in the same OID table (ignore it) if fs == 'Virtual Memory' or fs == 'Physical Memory' or fs == 'Real Memory': continue fs_current = {} fs_current['device_name'] = '' fs_current['mnt_point'] = fs.partition(' ')[0] fs_current['size'] = int(fs_stat[fs]['size']) * int(fs_stat[fs]['alloc_unit']) fs_current['used'] = int(fs_stat[fs]['used']) * int(fs_stat[fs]['alloc_unit']) fs_current['percent'] = float(fs_current['used'] * 100 / fs_current['size']) self.stats.append(fs_current) else: # Default behavor for fs in fs_stat: fs_current = {} fs_current['device_name'] = fs_stat[fs]['device_name'] fs_current['mnt_point'] = fs fs_current['size'] = int(fs_stat[fs]['size']) * 1024 fs_current['used'] = int(fs_stat[fs]['used']) * 1024 fs_current['percent'] = float(fs_stat[fs]['percent']) self.stats.append(fs_current) # Update the history list self.update_stats_history('mnt_point') return self.stats def msg_curse(self, args=None, max_width=None): """Return the dict to display in the curse interface.""" # Init the return message ret = [] # Only process if stats exist and display plugin enable... if self.stats == [] or args.disable_fs: return ret # Max size for the fsname name if max_width is not None and max_width >= 23: # Interface size name = max_width - space for interfaces bitrate fsname_max_width = max_width - 14 else: fsname_max_width = 9 # Build the string message # Header msg = '{0:{width}}'.format(_("FILE SYS"), width=fsname_max_width) ret.append(self.curse_add_line(msg, "TITLE")) msg = '{0:>7}'.format(_("Used")) ret.append(self.curse_add_line(msg)) msg = '{0:>7}'.format(_("Total")) ret.append(self.curse_add_line(msg)) # Disk list (sorted by name) for i in sorted(self.stats, key=lambda fs: fs['mnt_point']): # New line ret.append(self.curse_new_line()) if i['device_name'] == '' or i['device_name'] == 'none': mnt_point = i['mnt_point'][-fsname_max_width+1:] elif len(i['mnt_point']) + len(i['device_name'].split('/')[-1]) <= fsname_max_width - 3: # If possible concatenate mode info... Glances touch inside :) mnt_point = i['mnt_point'] + ' (' + i['device_name'].split('/')[-1] + ')' elif len(i['mnt_point']) > fsname_max_width: # Cut mount point name if it is too long mnt_point = '_' + i['mnt_point'][-fsname_max_width+1:] else: mnt_point = i['mnt_point'] msg = '{0:{width}}'.format(mnt_point, width=fsname_max_width) ret.append(self.curse_add_line(msg)) msg = '{0:>7}'.format(self.auto_unit(i['used'])) ret.append(self.curse_add_line(msg, self.get_alert(i['used'], max=i['size']))) msg = '{0:>7}'.format(self.auto_unit(i['size'])) ret.append(self.curse_add_line(msg)) return ret