glances_cpu.py 8.5 KB
Newer Older
A
Alessio Sergi 已提交
1 2
# -*- coding: utf-8 -*-
#
3
# This file is part of Glances.
A
Alessio Sergi 已提交
4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
#
# Copyright (C) 2014 Nicolargo <nicolas@nicolargo.com>
#
# 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

"""CPU plugin."""
A
Alessio Sergi 已提交
21

22
from glances.plugins.glances_plugin import GlancesPlugin
A
Alessio Sergi 已提交
23

A
Alessio Sergi 已提交
24 25
import psutil

N
Nicolargo 已提交
26 27 28 29
# SNMP OID
# percentage of user CPU time: .1.3.6.1.4.1.2021.11.9.0
# percentages of system CPU time: .1.3.6.1.4.1.2021.11.10.0
# percentages of idle CPU time: .1.3.6.1.4.1.2021.11.11.0
N
Nicolargo 已提交
30 31
snmp_oid = {'default': {'user': '1.3.6.1.4.1.2021.11.9.0',
                        'system': '1.3.6.1.4.1.2021.11.10.0',
N
Nicolargo 已提交
32 33
                        'idle': '1.3.6.1.4.1.2021.11.11.0'},
            'windows': {'percent': '1.3.6.1.2.1.25.3.3.1.2'}}
A
Alessio Sergi 已提交
34 35

class Plugin(GlancesPlugin):
A
PEP 257  
Alessio Sergi 已提交
36

A
Alessio Sergi 已提交
37
    """
A
PEP 257  
Alessio Sergi 已提交
38
    Glances' CPU plugin.
A
Alessio Sergi 已提交
39 40 41 42

    stats is a dict
    """

43
    def __init__(self, args=None):
A
PEP 257  
Alessio Sergi 已提交
44
        """Init the CPU plugin."""
45
        GlancesPlugin.__init__(self, args=args)
A
Alessio Sergi 已提交
46 47 48 49 50 51 52 53 54 55

        # We want to display the stat in the curse interface
        self.display_curse = True
        # Set the message position
        # It is NOT the curse position but the Glances column/line
        # Enter -1 to right align
        self.column_curse = 0
        # Enter -1 to diplay bottom
        self.line_curse = 1

56 57
        # Init stats
        self.first_call = True
A
Alessio Sergi 已提交
58
        self.reset()
N
Nicolargo 已提交
59 60

    def reset(self):
A
PEP 257  
Alessio Sergi 已提交
61
        """Reset/init the stats."""
62 63
        self.stats = {}

A
Alessio Sergi 已提交
64
    def update(self):
A
PEP 257  
Alessio Sergi 已提交
65
        """Update CPU stats using the input method."""
N
Nicolargo 已提交
66 67 68
        # Reset stats
        self.reset()

69
        if self.get_input() == 'local':
N
Nicolargo 已提交
70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91
            # Update stats using the standard system lib

            # Grab CPU using the PSUtil cpu_times_percent method
            cputimespercent = psutil.cpu_times_percent(interval=0.0, percpu=False)

            # Get all possible value for CPU stats
            # user
            # system
            # idle
            # nice (UNIX)
            # iowait (Linux)
            # irq (Linux, FreeBSD)
            # softirq (Linux)
            # steal (Linux >= 2.6.11)
            # The following stats are returned by the API but not displayed in the UI:
            # guest (Linux >= 2.6.24)
            # guest_nice (Linux >= 3.2.0)
            for cpu in ['user', 'system', 'idle', 'nice',
                        'iowait', 'irq', 'softirq', 'steal',
                        'guest', 'guest_nice']:
                if hasattr(cputimespercent, cpu):
                    self.stats[cpu] = getattr(cputimespercent, cpu)
92
        elif self.get_input() == 'snmp':
N
Nicolargo 已提交
93
            # Update stats using SNMP
94

N
Nicolargo 已提交
95 96 97 98 99 100
            if self.get_short_system_name() == 'windows':
                # Windows
                # You can find the CPU utilization of windows system by querying the oid
                # Give also the number of core (number of element in the table)
                # print snmp_oid[self.get_short_system_name()]
                try:
101 102
                    cpu_stats = self.set_stats_snmp(snmp_oid=snmp_oid[self.get_short_system_name()], 
                                                    bulk=True)
N
Nicolargo 已提交
103 104 105
                except KeyError:
                    self.reset()

106 107 108 109 110 111 112 113 114 115
                # Iter through CPU and compute the idle CPU stats
                self.stats['nb_log_core'] = 0
                self.stats['idle'] = 0
                for c in cpu_stats:
                    if c.startswith('percent'):
                        self.stats['idle'] += float(cpu_stats['percent.3'])
                        self.stats['nb_log_core'] += 1
                if self.stats['nb_log_core'] > 0:
                    self.stats['idle'] = self.stats['idle'] / self.stats['nb_log_core']
                self.stats['idle'] = 100 - self.stats['idle']
N
Nicolargo 已提交
116 117 118 119 120 121 122 123 124 125 126 127

            else:
                # Default behavor
                try:
                    self.stats = self.set_stats_snmp(snmp_oid=snmp_oid[self.get_short_system_name()])
                except KeyError:
                    self.stats = self.set_stats_snmp(snmp_oid=snmp_oid['default'])
               
                if self.stats['idle'] == '':
                    self.reset()
                    return self.stats

128 129 130
                # Convert SNMP stats to float
                for key in self.stats.iterkeys():
                    self.stats[key] = float(self.stats[key])
A
Alessio Sergi 已提交
131 132 133 134

        return self.stats

    def msg_curse(self, args=None):
A
PEP 257  
Alessio Sergi 已提交
135
        """Return the list to display in the curse interface."""
A
Alessio Sergi 已提交
136 137 138
        # Init the return message
        ret = []

N
Nicolas Hennion 已提交
139
        # Only process if stats exist...
140
        if self.stats == {}:
N
Nicolas Hennion 已提交
141 142
            return ret

A
Alessio Sergi 已提交
143
        # Build the string message
144 145
        # If user stat is not here, display only idle / total CPU usage (for exemple on Windows OS)
        idle_tag = 'user' not in self.stats
A
Alessio Sergi 已提交
146
        # Header
A
Alessio Sergi 已提交
147
        msg = '{0:8}'.format(_("CPU"))
A
Alessio Sergi 已提交
148 149
        ret.append(self.curse_add_line(msg, "TITLE"))
        # Total CPU usage
A
Alessio Sergi 已提交
150
        msg = '{0:>6.1%}'.format((100 - self.stats['idle']) / 100)
151 152 153 154
        if idle_tag:
            ret.append(self.curse_add_line(msg, self.get_alert_log((100 - self.stats['idle']) / 100, header="system")))
        else:
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
155 156
        # Nice CPU
        if 'nice' in self.stats:
A
Alessio Sergi 已提交
157
            msg = '  {0:8}'.format(_("nice:"))
158
            ret.append(self.curse_add_line(msg, optional=True))
A
Alessio Sergi 已提交
159
            msg = '{0:>6.1%}'.format(self.stats['nice'] / 100)
A
Alessio Sergi 已提交
160 161 162 163
            ret.append(self.curse_add_line(msg, optional=True))
        # New line
        ret.append(self.curse_new_line())
        # User CPU
164
        if 'user' in self.stats:
A
Alessio Sergi 已提交
165
            msg = '{0:8}'.format(_("user:"))
A
Alessio Sergi 已提交
166
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
167
            msg = '{0:>6.1%}'.format(self.stats['user'] / 100)
168 169 170 171 172 173
            ret.append(self.curse_add_line(msg, self.get_alert_log(self.stats['user'], header="user")))            
        elif 'idle' in self.stats:
            msg = '{0:8}'.format(_("idle:"))
            ret.append(self.curse_add_line(msg))
            msg = '{0:>6.1%}'.format(self.stats['idle'] / 100)
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
174 175
        # IRQ CPU
        if 'irq' in self.stats:
A
Alessio Sergi 已提交
176
            msg = '  {0:8}'.format(_("irq:"))
A
Alessio Sergi 已提交
177
            ret.append(self.curse_add_line(msg, optional=True))
A
Alessio Sergi 已提交
178
            msg = '{0:>6.1%}'.format(self.stats['irq'] / 100)
A
Alessio Sergi 已提交
179 180 181 182
            ret.append(self.curse_add_line(msg, optional=True))
        # New line
        ret.append(self.curse_new_line())
        # System CPU
183
        if 'system' in self.stats and not idle_tag:
A
Alessio Sergi 已提交
184
            msg = '{0:8}'.format(_("system:"))
A
Alessio Sergi 已提交
185
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
186
            msg = '{0:>6.1%}'.format(self.stats['system'] / 100)
A
Alessio Sergi 已提交
187
            ret.append(self.curse_add_line(msg, self.get_alert_log(self.stats['system'], header="system")))
188 189 190 191 192
        else:
            msg = '{0:8}'.format(_("core:"))
            ret.append(self.curse_add_line(msg))
            msg = '{0:>6}'.format(self.stats['nb_log_core'])
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
193 194
        # IOWait CPU
        if 'iowait' in self.stats:
A
Alessio Sergi 已提交
195
            msg = '  {0:8}'.format(_("iowait:"))
A
Alessio Sergi 已提交
196
            ret.append(self.curse_add_line(msg, optional=True))
A
Alessio Sergi 已提交
197
            msg = '{0:>6.1%}'.format(self.stats['iowait'] / 100)
A
Alessio Sergi 已提交
198
            ret.append(self.curse_add_line(msg, self.get_alert_log(self.stats['iowait'], header="iowait"), optional=True))
A
Alessio Sergi 已提交
199 200
        # New line
        ret.append(self.curse_new_line())
A
Alessio Sergi 已提交
201
        # Idle CPU
202
        if 'idle' in self.stats and not idle_tag:
A
Alessio Sergi 已提交
203
            msg = '{0:8}'.format(_("idle:"))
204
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
205
            msg = '{0:>6.1%}'.format(self.stats['idle'] / 100)
A
Alessio Sergi 已提交
206 207 208
            ret.append(self.curse_add_line(msg))
        # Steal CPU usage
        if 'steal' in self.stats:
A
Alessio Sergi 已提交
209
            msg = '  {0:8}'.format(_("steal:"))
210
            ret.append(self.curse_add_line(msg, optional=True))
A
Alessio Sergi 已提交
211
            msg = '{0:>6.1%}'.format(self.stats['steal'] / 100)
A
Alessio Sergi 已提交
212
            ret.append(self.curse_add_line(msg, self.get_alert(self.stats['steal'], header="steal"), optional=True))
A
Alessio Sergi 已提交
213 214 215

        # Return the message with decoration
        return ret