glances_system.py 7.6 KB
Newer Older
N
Nicolas Hennion 已提交
1 2
# -*- coding: utf-8 -*-
#
3
# This file is part of Glances.
N
Nicolas Hennion 已提交
4
#
5
# Copyright (C) 2015 Nicolargo <nicolas@nicolargo.com>
N
Nicolas Hennion 已提交
6 7 8 9 10 11 12 13 14 15 16 17 18
#
# 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 <http://www.gnu.org/licenses/>.
A
PEP 257  
Alessio Sergi 已提交
19 20

"""System plugin."""
N
Nicolas Hennion 已提交
21

N
Nicolas Hennion 已提交
22 23
import os
import platform
N
Nicolargo 已提交
24
import re
A
Alessio Sergi 已提交
25
from io import open
N
Nicolas Hennion 已提交
26

27
from glances.compat import iteritems
28
from glances.plugins.glances_plugin import GlancesPlugin
N
Nicolas Hennion 已提交
29

30
# SNMP OID
N
Nicolargo 已提交
31
snmp_oid = {'default': {'hostname': '1.3.6.1.2.1.1.5.0',
N
Nicolargo 已提交
32
                        'system_name': '1.3.6.1.2.1.1.1.0'},
N
Nicolargo 已提交
33 34 35
            'netapp': {'hostname': '1.3.6.1.2.1.1.5.0',
                       'system_name': '1.3.6.1.2.1.1.1.0',
                       'platform': '1.3.6.1.4.1.789.1.1.5.0'}}
N
Nicolargo 已提交
36 37 38

# SNMP to human read
# Dict (key: OS short name) of dict (reg exp OID to human)
N
Nicolargo 已提交
39 40
# Windows:
# http://msdn.microsoft.com/en-us/library/windows/desktop/ms724832%28v=vs.85%29.aspx
N
Nicolargo 已提交
41 42 43 44 45 46 47
snmp_to_human = {'windows': {'Windows Version 6.3': 'Windows 8.1 or Server 2012R2',
                             'Windows Version 6.2': 'Windows 8 or Server 2012',
                             'Windows Version 6.1': 'Windows 7 or Server 2008R2',
                             'Windows Version 6.0': 'Windows Vista or Server 2008',
                             'Windows Version 5.2': 'Windows XP 64bits or 2003 server',
                             'Windows Version 5.1': 'Windows XP',
                             'Windows Version 5.0': 'Windows 2000'}}
A
Alessio Sergi 已提交
48

A
Alessio Sergi 已提交
49

50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
def _linux_os_release():
    """Try to determine the name of a Linux distribution.

    This function checks for the /etc/os-release file.
    It takes the name from the 'NAME' field and the version from 'VERSION_ID'.
    An empty string is returned if the above values cannot be determined.
    """
    pretty_name = ''
    ashtray = {}
    keys = ['NAME', 'VERSION_ID']
    try:
        with open(os.path.join('/etc', 'os-release')) as f:
            for line in f:
                for key in keys:
                    if line.startswith(key):
                        ashtray[key] = line.strip().split('=')[1][1:-1]
    except (OSError, IOError):
        return pretty_name

    if ashtray:
        if 'NAME' in ashtray:
            pretty_name = ashtray['NAME']
        if 'VERSION_ID' in ashtray:
73
            pretty_name += ' {}'.format(ashtray['VERSION_ID'])
74 75 76 77

    return pretty_name


N
Nicolas Hennion 已提交
78
class Plugin(GlancesPlugin):
A
PEP 257  
Alessio Sergi 已提交
79 80

    """Glances' host/system plugin.
N
Nicolas Hennion 已提交
81

N
Nicolas Hennion 已提交
82
    stats is a dict
N
Nicolas Hennion 已提交
83 84
    """

85
    def __init__(self, args=None):
A
PEP 257  
Alessio Sergi 已提交
86
        """Init the plugin."""
A
Alessio Sergi 已提交
87
        super(Plugin, self).__init__(args=args)
88 89 90 91

        # We want to display the stat in the curse interface
        self.display_curse = True

92
        # Init the stats
A
Alessio Sergi 已提交
93
        self.reset()
94 95

    def reset(self):
A
PEP 257  
Alessio Sergi 已提交
96
        """Reset/init the stats."""
N
Nicolas Hennion 已提交
97
        self.stats = {}
98

99 100
    @GlancesPlugin._check_decorator
    @GlancesPlugin._log_result_decorator
101
    def update(self):
A
PEP 257  
Alessio Sergi 已提交
102 103
        """Update the host/system info using the input method.

104 105 106 107 108
        Return the stats (dict)
        """
        # Reset stats
        self.reset()

109
        if self.input_method == 'local':
110 111 112 113 114
            # Update stats using the standard system lib
            self.stats['os_name'] = platform.system()
            self.stats['hostname'] = platform.node()
            self.stats['platform'] = platform.architecture()[0]
            if self.stats['os_name'] == "Linux":
A
Alessio Sergi 已提交
115 116
                linux_distro = platform.linux_distribution()
                if linux_distro[0] == '':
117
                    self.stats['linux_distro'] = _linux_os_release()
118 119 120
                else:
                    self.stats['linux_distro'] = ' '.join(linux_distro[:2])
                self.stats['os_version'] = platform.release()
121
            elif self.stats['os_name'].endswith('BSD'):
122 123 124 125 126 127
                self.stats['os_version'] = platform.release()
            elif self.stats['os_name'] == "Darwin":
                self.stats['os_version'] = platform.mac_ver()[0]
            elif self.stats['os_name'] == "Windows":
                os_version = platform.win32_ver()
                self.stats['os_version'] = ' '.join(os_version[::2])
N
Nicolargo 已提交
128 129
                # if the python version is 32 bit perhaps the windows operating
                # system is 64bit
N
nicolargo 已提交
130 131
                if self.stats['platform'] == '32bit' and 'PROCESSOR_ARCHITEW6432' in os.environ:
                    self.stats['platform'] = '64bit'
N
Nicolas Hennion 已提交
132
            else:
133
                self.stats['os_version'] = ""
134 135 136
            # Add human readable name
            if self.stats['os_name'] == "Linux":
                self.stats['hr_name'] = self.stats['linux_distro']
A
Alessio Sergi 已提交
137
            else:
138
                self.stats['hr_name'] = '{} {}'.format(
N
Nicolargo 已提交
139
                    self.stats['os_name'], self.stats['os_version'])
140
            self.stats['hr_name'] += ' {}'.format(self.stats['platform'])
141

142
        elif self.input_method == 'snmp':
N
Nicolargo 已提交
143
            # Update stats using SNMP
N
Nicolargo 已提交
144
            try:
145
                self.stats = self.get_stats_snmp(
146
                    snmp_oid=snmp_oid[self.short_system_name])
N
Nicolargo 已提交
147
            except KeyError:
148
                self.stats = self.get_stats_snmp(snmp_oid=snmp_oid['default'])
N
Nicolargo 已提交
149 150 151
            # Default behavor: display all the information
            self.stats['os_name'] = self.stats['system_name']
            # Windows OS tips
152
            if self.short_system_name == 'windows':
A
Alessio Sergi 已提交
153
                for r, v in iteritems(snmp_to_human['windows']):
N
Nicolargo 已提交
154
                    if re.search(r, self.stats['system_name']):
155
                        self.stats['os_name'] = v
N
Nicolargo 已提交
156
                        break
157 158
            # Add human readable name
            self.stats['hr_name'] = self.stats['os_name']
159

160
        return self.stats
161

N
Nicolas Hennion 已提交
162
    def msg_curse(self, args=None):
A
PEP 257  
Alessio Sergi 已提交
163
        """Return the string to display in the curse interface."""
164 165 166 167
        # Init the return message
        ret = []

        # Build the string message
168
        if args.client:
169
            # Client mode
170
            if args.cs_status.lower() == "connected":
A
Alessio Sergi 已提交
171
                msg = 'Connected to '
N
Nicolas Hennion 已提交
172
                ret.append(self.curse_add_line(msg, 'OK'))
173
            elif args.cs_status.lower() == "snmp":
A
Alessio Sergi 已提交
174
                msg = 'SNMP from '
175
                ret.append(self.curse_add_line(msg, 'OK'))
176
            elif args.cs_status.lower() == "disconnected":
A
Alessio Sergi 已提交
177
                msg = 'Disconnected from '
N
Nicolas Hennion 已提交
178
                ret.append(self.curse_add_line(msg, 'CRITICAL'))
179 180

        # Hostname is mandatory
181
        msg = self.stats['hostname']
N
Nicolas Hennion 已提交
182 183
        ret.append(self.curse_add_line(msg, "TITLE"))
        # System info
A
Alessio Sergi 已提交
184
        if self.stats['os_name'] == "Linux" and self.stats['linux_distro']:
185 186 187 188
            msg = ' ({} {} / {} {})'.format(self.stats['linux_distro'],
                                            self.stats['platform'],
                                            self.stats['os_name'],
                                            self.stats['os_version'])
189
        else:
190
            try:
191 192 193
                msg = ' ({} {} {})'.format(self.stats['os_name'],
                                           self.stats['os_version'],
                                           self.stats['platform'])
A
Alessio Sergi 已提交
194
            except Exception:
195
                msg = ' ({})'.format(self.stats['os_name'])
N
Nicolas Hennion 已提交
196
        ret.append(self.curse_add_line(msg, optional=True))
A
Alessio Sergi 已提交
197 198

        # Return the message with decoration
199
        return ret