glances_cpu.py 10.2 KB
Newer Older
A
Alessio Sergi 已提交
1 2
# -*- coding: utf-8 -*-
#
3
# This file is part of Glances.
A
Alessio Sergi 已提交
4
#
5
# Copyright (C) 2015 Nicolargo <nicolas@nicolargo.com>
A
Alessio Sergi 已提交
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

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

22 23
from glances.compat import iterkeys
from glances.cpu_percent import cpu_percent
A
flake8  
Alessio Sergi 已提交
24 25 26
from glances.plugins.glances_plugin import GlancesPlugin

import psutil
27

N
Nicolargo 已提交
28 29 30 31
# 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 已提交
32 33
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 已提交
34
                        'idle': '1.3.6.1.4.1.2021.11.11.0'},
N
Nicolargo 已提交
35
            'windows': {'percent': '1.3.6.1.2.1.25.3.3.1.2'},
N
Nicolargo 已提交
36
            'esxi': {'percent': '1.3.6.1.2.1.25.3.3.1.2'},
N
Nicolargo 已提交
37 38 39
            'netapp': {'system': '1.3.6.1.4.1.789.1.2.1.3.0',
                       'idle': '1.3.6.1.4.1.789.1.2.1.5.0',
                       'nb_log_core': '1.3.6.1.4.1.789.1.2.1.6.0'}}
A
Alessio Sergi 已提交
40

41
# Define the history items list
42 43
# - 'name' define the stat identifier
# - 'color' define the graph color in #RGB format
N
Nicolargo 已提交
44
# - 'y_unit' define the Y label
45
# All items in this list will be historised if the --enable-history tag is set
N
Nicolargo 已提交
46 47
items_history_list = [{'name': 'user', 'color': '#00FF00', 'y_unit': '%'},
                      {'name': 'system', 'color': '#FF0000', 'y_unit': '%'}]
48

A
PEP 257  
Alessio Sergi 已提交
49

N
Nicolargo 已提交
50
class Plugin(GlancesPlugin):
A
Alessio Sergi 已提交
51

A
Alessio Sergi 已提交
52 53 54 55
    """Glances CPU plugin.

    'stats' is a dictionary that contains the system-wide CPU utilization as a
    percentage.
A
Alessio Sergi 已提交
56 57
    """

58
    def __init__(self, args=None):
A
PEP 257  
Alessio Sergi 已提交
59
        """Init the CPU plugin."""
A
Alessio Sergi 已提交
60
        super(Plugin, self).__init__(args=args, items_history_list=items_history_list)
A
Alessio Sergi 已提交
61 62 63 64

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

65
        # Init stats
A
Alessio Sergi 已提交
66
        self.reset()
N
Nicolargo 已提交
67 68

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

72
    @GlancesPlugin._log_result_decorator
A
Alessio Sergi 已提交
73
    def update(self):
A
PEP 257  
Alessio Sergi 已提交
74
        """Update CPU stats using the input method."""
N
Nicolargo 已提交
75 76 77
        # Reset stats
        self.reset()

N
Nicolargo 已提交
78 79
        # Grab CPU stats using psutil's cpu_percent and cpu_times_percent
        # methods
80
        if self.input_method == 'local':
A
Alessio Sergi 已提交
81 82
            # Get all possible values for CPU stats: user, system, idle,
            # nice (UNIX), iowait (Linux), irq (Linux, FreeBSD), steal (Linux 2.6.11+)
N
Nicolargo 已提交
83
            # The following stats are returned by the API but not displayed in the UI:
A
Alessio Sergi 已提交
84
            # softirq (Linux), guest (Linux 2.6.24+), guest_nice (Linux 3.2.0+)
N
Nicolargo 已提交
85
            self.stats['total'] = cpu_percent.get()
A
Alessio Sergi 已提交
86 87 88 89 90
            cpu_times_percent = psutil.cpu_times_percent(interval=0.0)
            for stat in ['user', 'system', 'idle', 'nice', 'iowait',
                         'irq', 'softirq', 'steal', 'guest', 'guest_nice']:
                if hasattr(cpu_times_percent, stat):
                    self.stats[stat] = getattr(cpu_times_percent, stat)
91
        elif self.input_method == 'snmp':
N
Nicolargo 已提交
92
            # Update stats using SNMP
93
            if self.short_system_name in ('windows', 'esxi'):
N
Nicolargo 已提交
94
                # Windows or VMWare ESXi
N
Nicolargo 已提交
95 96 97
                # 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)
                try:
98
                    cpu_stats = self.get_stats_snmp(snmp_oid=snmp_oid[self.short_system_name],
99
                                                    bulk=True)
N
Nicolargo 已提交
100 101 102
                except KeyError:
                    self.reset()

103 104 105 106 107 108 109 110
                # 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:
N
Nicolargo 已提交
111 112
                    self.stats['idle'] = self.stats[
                        'idle'] / self.stats['nb_log_core']
113
                self.stats['idle'] = 100 - self.stats['idle']
114
                self.stats['total'] = 100 - self.stats['idle']
N
Nicolargo 已提交
115 116 117 118

            else:
                # Default behavor
                try:
119
                    self.stats = self.get_stats_snmp(
120
                        snmp_oid=snmp_oid[self.short_system_name])
N
Nicolargo 已提交
121
                except KeyError:
122
                    self.stats = self.get_stats_snmp(
N
Nicolargo 已提交
123
                        snmp_oid=snmp_oid['default'])
124

N
Nicolargo 已提交
125 126 127 128
                if self.stats['idle'] == '':
                    self.reset()
                    return self.stats

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

134 135 136
        # Update the history list
        self.update_stats_history()

137 138 139
        # Update the view
        self.update_views()

A
Alessio Sergi 已提交
140 141
        return self.stats

142
    def update_views(self):
A
PEP 257  
Alessio Sergi 已提交
143
        """Update stats views."""
144
        # Call the father's method
A
Alessio Sergi 已提交
145
        super(Plugin, self).update_views()
146 147 148 149 150 151 152

        # Add specifics informations
        # Alert and log
        for key in ['user', 'system', 'iowait']:
            if key in self.stats:
                self.views[key]['decoration'] = self.get_alert_log(self.stats[key], header=key)
        # Alert only
153
        for key in ['steal', 'total']:
154 155 156 157 158 159 160
            if key in self.stats:
                self.views[key]['decoration'] = self.get_alert(self.stats[key], header=key)
        # Optional
        for key in ['nice', 'irq', 'iowait', 'steal']:
            if key in self.stats:
                self.views[key]['optional'] = True

A
Alessio Sergi 已提交
161
    def msg_curse(self, args=None):
A
PEP 257  
Alessio Sergi 已提交
162
        """Return the list to display in the UI."""
A
Alessio Sergi 已提交
163 164 165
        # Init the return message
        ret = []

166 167
        # Only process if stats exist and plugin not disable
        if not self.stats or args.disable_cpu:
N
Nicolas Hennion 已提交
168 169
            return ret

A
Alessio Sergi 已提交
170
        # Build the string message
N
Nicolargo 已提交
171 172
        # If user stat is not here, display only idle / total CPU usage (for
        # exemple on Windows OS)
173
        idle_tag = 'user' not in self.stats
A
Alessio Sergi 已提交
174
        # Header
A
Alessio Sergi 已提交
175
        msg = '{0:8}'.format('CPU')
A
Alessio Sergi 已提交
176 177
        ret.append(self.curse_add_line(msg, "TITLE"))
        # Total CPU usage
A
Alessio Sergi 已提交
178
        msg = '{0:>5}%'.format(self.stats['total'])
179
        if idle_tag:
N
Nicolargo 已提交
180
            ret.append(self.curse_add_line(
181
                msg, self.get_views(key='total', option='decoration')))
182 183
        else:
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
184 185
        # Nice CPU
        if 'nice' in self.stats:
A
Alessio Sergi 已提交
186
            msg = '  {0:8}'.format('nice:')
187
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='nice', option='optional')))
A
Alessio Sergi 已提交
188
            msg = '{0:>5}%'.format(self.stats['nice'])
189
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='nice', option='optional')))
A
Alessio Sergi 已提交
190 191 192
        # New line
        ret.append(self.curse_new_line())
        # User CPU
193
        if 'user' in self.stats:
A
Alessio Sergi 已提交
194
            msg = '{0:8}'.format('user:')
A
Alessio Sergi 已提交
195
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
196
            msg = '{0:>5}%'.format(self.stats['user'])
N
Nicolargo 已提交
197
            ret.append(self.curse_add_line(
198
                msg, self.get_views(key='user', option='decoration')))
199
        elif 'idle' in self.stats:
A
Alessio Sergi 已提交
200
            msg = '{0:8}'.format('idle:')
201
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
202
            msg = '{0:>5}%'.format(self.stats['idle'])
203
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
204 205
        # IRQ CPU
        if 'irq' in self.stats:
A
Alessio Sergi 已提交
206
            msg = '  {0:8}'.format('irq:')
207
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='irq', option='optional')))
A
Alessio Sergi 已提交
208
            msg = '{0:>5}%'.format(self.stats['irq'])
209
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='irq', option='optional')))
A
Alessio Sergi 已提交
210 211 212
        # New line
        ret.append(self.curse_new_line())
        # System CPU
213
        if 'system' in self.stats and not idle_tag:
A
Alessio Sergi 已提交
214
            msg = '{0:8}'.format('system:')
A
Alessio Sergi 已提交
215
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
216
            msg = '{0:>5}%'.format(self.stats['system'])
N
Nicolargo 已提交
217
            ret.append(self.curse_add_line(
218
                msg, self.get_views(key='system', option='decoration')))
219
        else:
A
Alessio Sergi 已提交
220
            msg = '{0:8}'.format('core:')
221 222 223
            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 已提交
224 225
        # IOWait CPU
        if 'iowait' in self.stats:
A
Alessio Sergi 已提交
226
            msg = '  {0:8}'.format('iowait:')
227
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='iowait', option='optional')))
A
Alessio Sergi 已提交
228
            msg = '{0:>5}%'.format(self.stats['iowait'])
N
Nicolargo 已提交
229
            ret.append(self.curse_add_line(
230 231
                msg, self.get_views(key='iowait', option='decoration'),
                optional=self.get_views(key='iowait', option='optional')))
A
Alessio Sergi 已提交
232 233
        # New line
        ret.append(self.curse_new_line())
A
Alessio Sergi 已提交
234
        # Idle CPU
235
        if 'idle' in self.stats and not idle_tag:
A
Alessio Sergi 已提交
236
            msg = '{0:8}'.format('idle:')
237
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
238
            msg = '{0:>5}%'.format(self.stats['idle'])
A
Alessio Sergi 已提交
239 240 241
            ret.append(self.curse_add_line(msg))
        # Steal CPU usage
        if 'steal' in self.stats:
A
Alessio Sergi 已提交
242
            msg = '  {0:8}'.format('steal:')
243
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='steal', option='optional')))
A
Alessio Sergi 已提交
244
            msg = '{0:>5}%'.format(self.stats['steal'])
N
Nicolargo 已提交
245
            ret.append(self.curse_add_line(
246 247
                msg, self.get_views(key='steal', option='decoration'),
                optional=self.get_views(key='steal', option='optional')))
A
Alessio Sergi 已提交
248 249 250

        # Return the message with decoration
        return ret