glances_cpu.py 14.9 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) 2021 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
from glances.logger import logger
23
from glances.timer import getTimeSinceLastUpdate
24 25
from glances.compat import iterkeys
from glances.cpu_percent import cpu_percent
26 27
from glances.globals import LINUX
from glances.plugins.glances_core import Plugin as CorePlugin
A
flake8  
Alessio Sergi 已提交
28 29 30
from glances.plugins.glances_plugin import GlancesPlugin

import psutil
31

N
Nicolargo 已提交
32 33 34 35
# 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 已提交
36 37
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 已提交
38
                        'idle': '1.3.6.1.4.1.2021.11.11.0'},
N
Nicolargo 已提交
39
            'windows': {'percent': '1.3.6.1.2.1.25.3.3.1.2'},
N
Nicolargo 已提交
40
            'esxi': {'percent': '1.3.6.1.2.1.25.3.3.1.2'},
N
Nicolargo 已提交
41 42 43
            '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 已提交
44

45
# Define the history items list
46
# - 'name' define the stat identifier
N
Nicolargo 已提交
47
# - 'y_unit' define the Y label
48 49 50 51 52 53
items_history_list = [{'name': 'user',
                       'description': 'User CPU usage',
                       'y_unit': '%'},
                      {'name': 'system',
                       'description': 'System CPU usage',
                       'y_unit': '%'}]
54

A
PEP 257  
Alessio Sergi 已提交
55

N
Nicolargo 已提交
56
class Plugin(GlancesPlugin):
A
Alessio Sergi 已提交
57 58 59 60
    """Glances CPU plugin.

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

63
    def __init__(self, args=None, config=None):
A
PEP 257  
Alessio Sergi 已提交
64
        """Init the CPU plugin."""
65 66 67
        super(Plugin, self).__init__(args=args,
                                     config=config,
                                     items_history_list=items_history_list)
A
Alessio Sergi 已提交
68 69 70 71

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

72 73 74 75 76 77
        # Call CorePlugin in order to display the core number
        try:
            self.nb_log_core = CorePlugin(args=self.args).update()["log"]
        except Exception:
            self.nb_log_core = 1

78 79 80 81
        # Force a first update because we need two update to have the first stat
        self.update()
        self.refresh_timer.set(0)

82
    @GlancesPlugin._check_decorator
83
    @GlancesPlugin._log_result_decorator
A
Alessio Sergi 已提交
84
    def update(self):
A
PEP 257  
Alessio Sergi 已提交
85
        """Update CPU stats using the input method."""
86
        # Grab stats into self.stats
87
        if self.input_method == 'local':
88
            stats = self.update_local()
89
        elif self.input_method == 'snmp':
90 91 92 93 94 95
            stats = self.update_snmp()
        else:
            stats = self.get_init_value()

        # Update the stats
        self.stats = stats
A
Alessio Sergi 已提交
96 97 98

        return self.stats

99
    def update_local(self):
A
Alessio Sergi 已提交
100
        """Update CPU stats using psutil."""
101 102 103 104 105
        # Grab CPU stats using psutil's cpu_percent and cpu_times_percent
        # Get all possible values for CPU stats: user, system, idle,
        # nice (UNIX), iowait (Linux), irq (Linux, FreeBSD), steal (Linux 2.6.11+)
        # The following stats are returned by the API but not displayed in the UI:
        # softirq (Linux), guest (Linux 2.6.24+), guest_nice (Linux 3.2.0+)
106 107 108 109 110

        # Init new stats
        stats = self.get_init_value()

        stats['total'] = cpu_percent.get()
111 112 113 114
        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):
115
                stats[stat] = getattr(cpu_times_percent, stat)
116

A
Alessio Sergi 已提交
117
        # Additional CPU stats (number of events not as a %; psutil>=4.1.0)
118 119 120 121
        # ctx_switches: number of context switches (voluntary + involuntary) per second
        # interrupts: number of interrupts per second
        # soft_interrupts: number of software interrupts per second. Always set to 0 on Windows and SunOS.
        # syscalls: number of system calls since boot. Always set to 0 on Linux.
A
Alessio Sergi 已提交
122 123 124 125 126 127 128 129 130 131
        cpu_stats = psutil.cpu_stats()
        # By storing time data we enable Rx/s and Tx/s calculations in the
        # XML/RPC API, which would otherwise be overly difficult work
        # for users of the API
        time_since_update = getTimeSinceLastUpdate('cpu')

        # Previous CPU stats are stored in the cpu_stats_old variable
        if not hasattr(self, 'cpu_stats_old'):
            # First call, we init the cpu_stats_old var
            self.cpu_stats_old = cpu_stats
132
        else:
A
Alessio Sergi 已提交
133 134
            for stat in cpu_stats._fields:
                if getattr(cpu_stats, stat) is not None:
135
                    stats[stat] = getattr(cpu_stats, stat) - getattr(self.cpu_stats_old, stat)
A
Alessio Sergi 已提交
136

137
            stats['time_since_update'] = time_since_update
A
Alessio Sergi 已提交
138 139

            # Core number is needed to compute the CTX switch limit
140
            stats['cpucore'] = self.nb_log_core
A
Alessio Sergi 已提交
141 142 143

            # Save stats to compute next step
            self.cpu_stats_old = cpu_stats
144

145 146
        return stats

147 148
    def update_snmp(self):
        """Update CPU stats using SNMP."""
149 150 151 152

        # Init new stats
        stats = self.get_init_value()

153 154 155 156 157 158 159 160 161 162 163 164
        # Update stats using SNMP
        if self.short_system_name in ('windows', 'esxi'):
            # Windows or VMWare ESXi
            # 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:
                cpu_stats = self.get_stats_snmp(snmp_oid=snmp_oid[self.short_system_name],
                                                bulk=True)
            except KeyError:
                self.reset()

            # Iter through CPU and compute the idle CPU stats
165 166
            stats['nb_log_core'] = 0
            stats['idle'] = 0
167 168
            for c in cpu_stats:
                if c.startswith('percent'):
169 170 171 172 173 174
                    stats['idle'] += float(cpu_stats['percent.3'])
                    stats['nb_log_core'] += 1
            if stats['nb_log_core'] > 0:
                stats['idle'] = stats['idle'] / stats['nb_log_core']
            stats['idle'] = 100 - stats['idle']
            stats['total'] = 100 - stats['idle']
175 176 177 178

        else:
            # Default behavor
            try:
179
                stats = self.get_stats_snmp(
180 181
                    snmp_oid=snmp_oid[self.short_system_name])
            except KeyError:
182
                stats = self.get_stats_snmp(
183 184
                    snmp_oid=snmp_oid['default'])

185
            if stats['idle'] == '':
186 187 188 189
                self.reset()
                return self.stats

            # Convert SNMP stats to float
190 191 192 193 194
            for key in iterkeys(stats):
                stats[key] = float(stats[key])
            stats['total'] = 100 - stats['idle']

        return stats
195

196
    def update_views(self):
A
PEP 257  
Alessio Sergi 已提交
197
        """Update stats views."""
198
        # Call the father's method
A
Alessio Sergi 已提交
199
        super(Plugin, self).update_views()
200 201 202

        # Add specifics informations
        # Alert and log
203
        for key in ['user', 'system', 'iowait', 'total']:
204 205 206
            if key in self.stats:
                self.views[key]['decoration'] = self.get_alert_log(self.stats[key], header=key)
        # Alert only
207
        for key in ['steal']:
208 209
            if key in self.stats:
                self.views[key]['decoration'] = self.get_alert(self.stats[key], header=key)
210 211 212 213
        # Alert only but depend on Core number
        for key in ['ctx_switches']:
            if key in self.stats:
                self.views[key]['decoration'] = self.get_alert(self.stats[key], maximum=100 * self.stats['cpucore'], header=key)
214
        # Optional
215
        for key in ['nice', 'irq', 'idle', 'steal', 'ctx_switches', 'interrupts', 'soft_interrupts', 'syscalls']:
216 217 218
            if key in self.stats:
                self.views[key]['optional'] = True

219
    def msg_curse(self, args=None, max_width=None):
A
PEP 257  
Alessio Sergi 已提交
220
        """Return the list to display in the UI."""
A
Alessio Sergi 已提交
221 222 223
        # Init the return message
        ret = []

224
        # Only process if stats exist and plugin not disable
N
nicolargo 已提交
225
        if not self.stats or self.args.percpu or self.is_disable():
N
Nicolas Hennion 已提交
226 227
            return ret

A
Alessio Sergi 已提交
228
        # Build the string message
N
Nicolargo 已提交
229 230
        # If user stat is not here, display only idle / total CPU usage (for
        # exemple on Windows OS)
231
        idle_tag = 'user' not in self.stats
232

A
Alessio Sergi 已提交
233
        # Header
234
        msg = '{}'.format('CPU')
A
Alessio Sergi 已提交
235
        ret.append(self.curse_add_line(msg, "TITLE"))
236 237 238 239 240 241 242 243
        trend_user = self.get_trend('user')
        trend_system = self.get_trend('system')
        if trend_user is None or trend_user is None:
            trend_cpu = None
        else:
            trend_cpu = trend_user + trend_system
        msg = ' {:4}'.format(self.trend_msg(trend_cpu))
        ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
244
        # Total CPU usage
245
        msg = '{:5.1f}%'.format(self.stats['total'])
246 247 248 249 250 251 252 253 254 255 256 257
        ret.append(self.curse_add_line(
            msg, self.get_views(key='total', option='decoration')))
        # if idle_tag:
        #     ret.append(self.curse_add_line(
        #         msg, self.get_views(key='total', option='decoration')))
        # else:
        #     ret.append(self.curse_add_line(msg))
        # Idle CPU
        if 'idle' in self.stats and not idle_tag:
            msg = '  {:8}'.format('idle:')
            ret.append(self.curse_add_line(msg))
            msg = '{:5.1f}%'.format(self.stats['idle'])
258
            ret.append(self.curse_add_line(msg))
259 260
        # ctx_switches
        if 'ctx_switches' in self.stats:
261
            msg = '  {:8}'.format('ctx_sw:')
262
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='ctx_switches', option='optional')))
263
            msg = '{:>5}'.format(self.auto_unit(int(self.stats['ctx_switches'] // self.stats['time_since_update']),
N
nicolargo 已提交
264
                                                min_symbol='K'))
265 266 267 268
            ret.append(self.curse_add_line(
                msg, self.get_views(key='ctx_switches', option='decoration'),
                optional=self.get_views(key='ctx_switches', option='optional')))

A
Alessio Sergi 已提交
269 270 271
        # New line
        ret.append(self.curse_new_line())
        # User CPU
272
        if 'user' in self.stats:
273
            msg = '{:8}'.format('user:')
A
Alessio Sergi 已提交
274
            ret.append(self.curse_add_line(msg))
275
            msg = '{:5.1f}%'.format(self.stats['user'])
N
Nicolargo 已提交
276
            ret.append(self.curse_add_line(
277
                msg, self.get_views(key='user', option='decoration')))
278
        elif 'idle' in self.stats:
279
            msg = '{:8}'.format('idle:')
280
            ret.append(self.curse_add_line(msg))
281
            msg = '{:5.1f}%'.format(self.stats['idle'])
282
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
283 284
        # IRQ CPU
        if 'irq' in self.stats:
285
            msg = '  {:8}'.format('irq:')
286
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='irq', option='optional')))
287
            msg = '{:5.1f}%'.format(self.stats['irq'])
288
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='irq', option='optional')))
289 290
        # interrupts
        if 'interrupts' in self.stats:
291
            msg = '  {:8}'.format('inter:')
292
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='interrupts', option='optional')))
293
            msg = '{:>5}'.format(int(self.stats['interrupts'] // self.stats['time_since_update']))
294 295
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='interrupts', option='optional')))

A
Alessio Sergi 已提交
296 297 298
        # New line
        ret.append(self.curse_new_line())
        # System CPU
299
        if 'system' in self.stats and not idle_tag:
300
            msg = '{:8}'.format('system:')
A
Alessio Sergi 已提交
301
            ret.append(self.curse_add_line(msg))
302
            msg = '{:5.1f}%'.format(self.stats['system'])
N
Nicolargo 已提交
303
            ret.append(self.curse_add_line(
304
                msg, self.get_views(key='system', option='decoration')))
305
        else:
306
            msg = '{:8}'.format('core:')
307
            ret.append(self.curse_add_line(msg))
308
            msg = '{:>6}'.format(self.stats['nb_log_core'])
309
            ret.append(self.curse_add_line(msg))
310 311 312
        # Nice CPU
        if 'nice' in self.stats:
            msg = '  {:8}'.format('nice:')
N
Nicolargo 已提交
313
            ret.append(self.curse_add_line(
314 315 316 317
                msg, optional=self.get_views(key='nice', option='optional')))
            msg = '{:5.1f}%'.format(self.stats['nice'])
            ret.append(self.curse_add_line(
                msg, optional=self.get_views(key='nice', option='optional')))
318 319
        # soft_interrupts
        if 'soft_interrupts' in self.stats:
320
            msg = '  {:8}'.format('sw_int:')
321
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='soft_interrupts', option='optional')))
322
            msg = '{:>5}'.format(int(self.stats['soft_interrupts'] // self.stats['time_since_update']))
323 324
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='soft_interrupts', option='optional')))

A
Alessio Sergi 已提交
325 326
        # New line
        ret.append(self.curse_new_line())
327 328 329 330 331 332 333 334 335
        # IOWait CPU
        if 'iowait' in self.stats:
            msg = '{:8}'.format('iowait:')
            ret.append(self.curse_add_line(
                msg, optional=self.get_views(key='iowait', option='optional')))
            msg = '{:5.1f}%'.format(self.stats['iowait'])
            ret.append(self.curse_add_line(
                msg, self.get_views(key='iowait', option='decoration'),
                optional=self.get_views(key='iowait', option='optional')))
A
Alessio Sergi 已提交
336 337
        # Steal CPU usage
        if 'steal' in self.stats:
338
            msg = '  {:8}'.format('steal:')
339
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='steal', option='optional')))
340
            msg = '{:5.1f}%'.format(self.stats['steal'])
N
Nicolargo 已提交
341
            ret.append(self.curse_add_line(
342 343
                msg, self.get_views(key='steal', option='decoration'),
                optional=self.get_views(key='steal', option='optional')))
344 345 346
        # syscalls
        # syscalls: number of system calls since boot. Always set to 0 on Linux. (do not display)
        if 'syscalls' in self.stats and not LINUX:
347
            msg = '  {:8}'.format('syscal:')
348
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='syscalls', option='optional')))
349
            msg = '{:>5}'.format(int(self.stats['syscalls'] // self.stats['time_since_update']))
350
            ret.append(self.curse_add_line(msg, optional=self.get_views(key='syscalls', option='optional')))
A
Alessio Sergi 已提交
351 352 353

        # Return the message with decoration
        return ret