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

22
import os
23
import shlex
N
nicolargo 已提交
24
import copy
A
Alessio Sergi 已提交
25 26
from datetime import timedelta

27
from glances.logger import logger
28
from glances.globals import WINDOWS
29
from glances.processes import glances_processes, sort_stats
30
from glances.plugins.glances_core import Plugin as CorePlugin
A
Alessio Sergi 已提交
31
from glances.plugins.glances_plugin import GlancesPlugin
A
Alessio Sergi 已提交
32

D
desbma 已提交
33

34 35 36 37
def seconds_to_hms(input_seconds):
    """Convert seconds to human-readable time."""
    minutes, seconds = divmod(input_seconds, 60)
    hours, minutes = divmod(minutes, 60)
38

39 40 41 42 43
    hours = int(hours)
    minutes = int(minutes)
    seconds = str(int(seconds)).zfill(2)

    return hours, minutes, seconds
44 45


46 47
def split_cmdline(cmdline):
    """Return path, cmd and arguments for a process cmdline."""
48 49
    path, cmd = os.path.split(cmdline[0])
    arguments = ' '.join(cmdline[1:])
50 51 52
    return path, cmd, arguments


A
Alessio Sergi 已提交
53
class Plugin(GlancesPlugin):
A
PEP 257  
Alessio Sergi 已提交
54
    """Glances' processes plugin.
A
Alessio Sergi 已提交
55 56 57 58

    stats is a list
    """

59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90
    # Define the header layout of the processes list columns
    layout_header = {
        'cpu': '{:<6} ',
        'mem': '{:<5} ',
        'virt': '{:<5} ',
        'res': '{:<5} ',
        'pid': '{:>{width}} ',
        'user': '{:<10} ',
        'time': '{:>8} ',
        'thread': '{:<3} ',
        'nice': '{:>3} ',
        'status': '{:>1} ',
        'ior': '{:>4} ',
        'iow': '{:<4} ',
        'command': '{}',
    }

    # Define the stat layout of the processes list columns
    layout_stat = {
        'cpu': '{:<6.1f} ',
        'mem': '{:<5.1f} ',
        'virt': '{:<5} ',
        'res': '{:<5} ',
        'pid': '{:>{width}} ',
        'user': '{:<10} ',
        'time': '{:>8} ',
        'thread': '{:<3} ',
        'nice': '{:>3} ',
        'status': '{:>1} ',
        'ior': '{:>4} ',
        'iow': '{:<4} ',
        'command': '{}',
91
        'name': '[{}]'
92 93
    }

94
    def __init__(self, args=None):
A
PEP 257  
Alessio Sergi 已提交
95
        """Init the plugin."""
96 97
        super(Plugin, self).__init__(args=args,
                                     stats_init_value=[])
A
Alessio Sergi 已提交
98 99 100 101

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

102 103 104
        # Trying to display proc time
        self.tag_proc_time = True

105 106 107 108 109 110
        # Call CorePlugin to get the core number (needed when not in IRIX mode / Solaris mode)
        try:
            self.nb_log_core = CorePlugin(args=self.args).update()["log"]
        except Exception:
            self.nb_log_core = 0

111
        # Get the max values (dict)
N
nicolargo 已提交
112
        self.max_values = copy.deepcopy(glances_processes.max_values())
113

114 115 116 117
        # Get the maximum PID number
        # Use to optimize space (see https://github.com/nicolargo/glances/issues/959)
        self.pid_max = glances_processes.pid_max

118
        # Note: 'glances_processes' is already init in the processes.py script
119

120
    def get_key(self):
A
PEP 257  
Alessio Sergi 已提交
121
        """Return the key of the list."""
122 123
        return 'pid'

124
    def update(self):
A
PEP 257  
Alessio Sergi 已提交
125
        """Update processes stats using the input method."""
126 127
        # Init new stats
        stats = self.get_init_value()
128

129
        if self.input_method == 'local':
130 131 132
            # Update stats using the standard system lib
            # Note: Update is done in the processcount plugin
            # Just return the processes list
133
            stats = glances_processes.getlist()
134

135
        elif self.input_method == 'snmp':
N
Nicolargo 已提交
136
            # No SNMP grab for processes
137
            pass
138

139 140 141 142 143 144 145
        # Update the stats
        self.stats = stats

        # Get the max values (dict)
        # Use Deep copy to avoid change between update and display
        self.max_values = copy.deepcopy(glances_processes.max_values())

146
        return self.stats
147

148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167
    def get_nice_alert(self, value):
        """Return the alert relative to the Nice configuration list"""
        value = str(value)
        try:
            if value in self.get_limit('nice_critical'):
                return 'CRITICAL'
        except KeyError:
            pass
        try:
            if value in self.get_limit('nice_warning'):
                return 'WARNING'
        except KeyError:
            pass
        try:
            if value in self.get_limit('nice_careful'):
                return 'CAREFUL'
        except KeyError:
            pass
        return 'DEFAULT'

D
desbma 已提交
168
    def get_process_curses_data(self, p, first, args):
169
        """Get curses data to display for a process.
170

171 172 173
        - p is the process to display
        - first is a tag=True if the process is the first on the list
        """
174
        ret = [self.curse_new_line()]
D
desbma 已提交
175 176
        # CPU
        if 'cpu_percent' in p and p['cpu_percent'] is not None and p['cpu_percent'] != '':
177
            if args.disable_irix and self.nb_log_core != 0:
178
                msg = self.layout_stat['cpu'].format(p['cpu_percent'] / float(self.nb_log_core))
179
            else:
180
                msg = self.layout_stat['cpu'].format(p['cpu_percent'])
181 182
            alert = self.get_alert(p['cpu_percent'],
                                   highlight_zero=False,
N
Nicolargo 已提交
183
                                   is_max=(p['cpu_percent'] == self.max_values['cpu_percent']),
184 185
                                   header="cpu")
            ret.append(self.curse_add_line(msg, alert))
D
desbma 已提交
186
        else:
187
            msg = self.layout_header['cpu'].format('?')
D
desbma 已提交
188 189 190
            ret.append(self.curse_add_line(msg))
        # MEM
        if 'memory_percent' in p and p['memory_percent'] is not None and p['memory_percent'] != '':
191
            msg = self.layout_stat['mem'].format(p['memory_percent'])
192 193
            alert = self.get_alert(p['memory_percent'],
                                   highlight_zero=False,
N
Nicolargo 已提交
194
                                   is_max=(p['memory_percent'] == self.max_values['memory_percent']),
195 196
                                   header="mem")
            ret.append(self.curse_add_line(msg, alert))
D
desbma 已提交
197
        else:
198
            msg = self.layout_header['mem'].format('?')
D
desbma 已提交
199 200 201 202
            ret.append(self.curse_add_line(msg))
        # VMS/RSS
        if 'memory_info' in p and p['memory_info'] is not None and p['memory_info'] != '':
            # VMS
203
            msg = self.layout_stat['virt'].format(self.auto_unit(p['memory_info'][1], low_precision=False))
D
desbma 已提交
204 205
            ret.append(self.curse_add_line(msg, optional=True))
            # RSS
206
            msg = self.layout_stat['res'].format(self.auto_unit(p['memory_info'][0], low_precision=False))
D
desbma 已提交
207 208
            ret.append(self.curse_add_line(msg, optional=True))
        else:
209
            msg = self.layout_header['virt'].format('?')
D
desbma 已提交
210
            ret.append(self.curse_add_line(msg))
211
            msg = self.layout_header['res'].format('?')
D
desbma 已提交
212 213
            ret.append(self.curse_add_line(msg))
        # PID
214
        msg = self.layout_stat['pid'].format(p['pid'], width=self.__max_pid_size())
D
desbma 已提交
215 216 217 218
        ret.append(self.curse_add_line(msg))
        # USER
        if 'username' in p:
            # docker internal users are displayed as ints only, therefore str()
219
            # Correct issue #886 on Windows OS
220 221 222 223 224 225 226
            msg = self.layout_stat['user'].format(str(p['username'])[:9])
            ret.append(self.curse_add_line(msg))
        else:
            msg = self.layout_header['user'].format('?')
            ret.append(self.curse_add_line(msg))
        # TIME+
        try:
227 228
            # Sum user and system time
            user_system_time = p['cpu_times'][0] + p['cpu_times'][1]
229 230 231 232 233 234 235 236 237
        except (OverflowError, TypeError) as e:
            # Catch OverflowError on some Amazon EC2 server
            # See https://github.com/nicolargo/glances/issues/87
            # Also catch TypeError on macOS
            # See: https://github.com/nicolargo/glances/issues/622
            # logger.debug("Cannot get TIME+ ({})".format(e))
            msg = self.layout_header['time'].format('?')
            ret.append(self.curse_add_line(msg, optional=True))
        else:
238
            hours, minutes, seconds = seconds_to_hms(user_system_time)
239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257
            if hours > 99:
                msg = '{:<7}h'.format(hours)
            elif 0 < hours < 100:
                msg = '{}h{}:{}'.format(hours, minutes, seconds)
            else:
                msg = '{}:{}'.format(minutes, seconds)
            msg = self.layout_stat['time'].format(msg)
            if hours > 0:
                ret.append(self.curse_add_line(msg,
                                               decoration='CPU_TIME',
                                               optional=True))
            else:
                ret.append(self.curse_add_line(msg, optional=True))
        # THREAD
        if 'num_threads' in p:
            num_threads = p['num_threads']
            if num_threads is None:
                num_threads = '?'
            msg = self.layout_stat['thread'].format(num_threads)
D
desbma 已提交
258 259
            ret.append(self.curse_add_line(msg))
        else:
260
            msg = self.layout_header['thread'].format('?')
D
desbma 已提交
261 262 263 264 265 266
            ret.append(self.curse_add_line(msg))
        # NICE
        if 'nice' in p:
            nice = p['nice']
            if nice is None:
                nice = '?'
267
            msg = self.layout_stat['nice'].format(nice)
268 269
            ret.append(self.curse_add_line(msg,
                                           decoration=self.get_nice_alert(nice)))
D
desbma 已提交
270
        else:
271
            msg = self.layout_header['nice'].format('?')
D
desbma 已提交
272 273 274 275
            ret.append(self.curse_add_line(msg))
        # STATUS
        if 'status' in p:
            status = p['status']
276
            msg = self.layout_stat['status'].format(status)
D
desbma 已提交
277 278 279 280 281
            if status == 'R':
                ret.append(self.curse_add_line(msg, decoration='STATUS'))
            else:
                ret.append(self.curse_add_line(msg))
        else:
282
            msg = self.layout_header['status'].format('?')
D
desbma 已提交
283 284
            ret.append(self.curse_add_line(msg))
        # IO read/write
285
        if 'io_counters' in p and p['io_counters'][4] == 1 and p['time_since_update'] != 0:
N
nicolargo 已提交
286
            # Display rate if stats is available and io_tag ([4]) == 1
D
desbma 已提交
287
            # IO read
288
            io_rs = int((p['io_counters'][0] - p['io_counters'][2]) / p['time_since_update'])
D
desbma 已提交
289
            if io_rs == 0:
290
                msg = self.layout_stat['ior'].format("0")
D
desbma 已提交
291
            else:
292 293
                msg = self.layout_stat['ior'].format(self.auto_unit(io_rs,
                                                                    low_precision=True))
D
desbma 已提交
294 295
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
            # IO write
296
            io_ws = int((p['io_counters'][1] - p['io_counters'][3]) / p['time_since_update'])
D
desbma 已提交
297
            if io_ws == 0:
298
                msg = self.layout_stat['iow'].format("0")
D
desbma 已提交
299
            else:
300 301
                msg = self.layout_stat['iow'].format(self.auto_unit(io_ws,
                                                                    low_precision=True))
D
desbma 已提交
302 303
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
        else:
304
            msg = self.layout_header['ior'].format("?")
D
desbma 已提交
305
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
306
            msg = self.layout_header['iow'].format("?")
D
desbma 已提交
307 308 309
            ret.append(self.curse_add_line(msg, optional=True, additional=True))

        # Command line
310 311
        # If no command line for the process is available, fallback to
        # the bare process name instead
N
nicolargo 已提交
312 313 314 315
        if 'cmdline' in p:
            cmdline = p['cmdline']
        else:
            cmdline = '?'
316
        try:
A
Alessio Sergi 已提交
317
            if cmdline:
318
                path, cmd, arguments = split_cmdline(cmdline)
319
                if os.path.isdir(path) and not args.process_short_name:
320
                    msg = self.layout_stat['command'].format(path) + os.sep
D
desbma 已提交
321
                    ret.append(self.curse_add_line(msg, splittable=True))
322
                    ret.append(self.curse_add_line(cmd, decoration='PROCESS', splittable=True))
D
desbma 已提交
323
                else:
324
                    msg = self.layout_stat['command'].format(cmd)
325
                    ret.append(self.curse_add_line(msg, decoration='PROCESS', splittable=True))
326
                if arguments:
327
                    msg = ' ' + self.layout_stat['command'].format(arguments)
328
                    ret.append(self.curse_add_line(msg, splittable=True))
329
            else:
330
                msg = self.layout_stat['name'].format(p['name'])
331
                ret.append(self.curse_add_line(msg, splittable=True))
332 333 334
        except (TypeError, UnicodeEncodeError) as e:
            # Avoid crach after running fine for several hours #1335
            logger.debug("Can not decode command line '{}' ({})".format(cmdline, e))
335
            ret.append(self.curse_add_line('', splittable=True))
D
desbma 已提交
336 337

        # Add extended stats but only for the top processes
338
        if first and 'extended_stats' in p and args.enable_process_extended:
D
desbma 已提交
339 340 341 342 343
            # Left padding
            xpad = ' ' * 13
            # First line is CPU affinity
            if 'cpu_affinity' in p and p['cpu_affinity'] is not None:
                ret.append(self.curse_new_line())
A
Alessio Sergi 已提交
344
                msg = xpad + 'CPU affinity: ' + str(len(p['cpu_affinity'])) + ' cores'
D
desbma 已提交
345 346
                ret.append(self.curse_add_line(msg, splittable=True))
            # Second line is memory info
347 348
            if 'memory_info' in p and \
               p['memory_info'] is not None:
D
desbma 已提交
349
                ret.append(self.curse_new_line())
350
                msg = '{}Memory info: {}'.format(xpad, p['memory_info'])
D
desbma 已提交
351
                if 'memory_swap' in p and p['memory_swap'] is not None:
352
                    msg += ' swap ' + self.auto_unit(p['memory_swap'], low_precision=False)
D
desbma 已提交
353 354 355 356
                ret.append(self.curse_add_line(msg, splittable=True))
            # Third line is for open files/network sessions
            msg = ''
            if 'num_threads' in p and p['num_threads'] is not None:
357
                msg += str(p['num_threads']) + ' threads '
D
desbma 已提交
358
            if 'num_fds' in p and p['num_fds'] is not None:
359
                msg += str(p['num_fds']) + ' files '
D
desbma 已提交
360
            if 'num_handles' in p and p['num_handles'] is not None:
361
                msg += str(p['num_handles']) + ' handles '
D
desbma 已提交
362
            if 'tcp' in p and p['tcp'] is not None:
363
                msg += str(p['tcp']) + ' TCP '
D
desbma 已提交
364
            if 'udp' in p and p['udp'] is not None:
365
                msg += str(p['udp']) + ' UDP'
D
desbma 已提交
366 367
            if msg != '':
                ret.append(self.curse_new_line())
A
Alessio Sergi 已提交
368
                msg = xpad + 'Open: ' + msg
D
desbma 已提交
369 370
                ret.append(self.curse_add_line(msg, splittable=True))
            # Fouth line is IO nice level (only Linux and Windows OS)
371 372 373
            if 'ionice' in p and \
               p['ionice'] is not None \
               and hasattr(p['ionice'], 'ioclass'):
D
desbma 已提交
374
                ret.append(self.curse_new_line())
A
Alessio Sergi 已提交
375 376
                msg = xpad + 'IO nice: '
                k = 'Class is '
D
desbma 已提交
377 378 379
                v = p['ionice'].ioclass
                # Linux: The scheduling class. 0 for none, 1 for real time, 2 for best-effort, 3 for idle.
                # Windows: On Windows only ioclass is used and it can be set to 2 (normal), 1 (low) or 0 (very low).
A
Alessio Sergi 已提交
380
                if WINDOWS:
D
desbma 已提交
381 382 383 384 385
                    if v == 0:
                        msg += k + 'Very Low'
                    elif v == 1:
                        msg += k + 'Low'
                    elif v == 2:
A
Alessio Sergi 已提交
386
                        msg += 'No specific I/O priority'
D
desbma 已提交
387 388 389 390
                    else:
                        msg += k + str(v)
                else:
                    if v == 0:
A
Alessio Sergi 已提交
391
                        msg += 'No specific I/O priority'
D
desbma 已提交
392 393 394 395 396 397 398 399 400 401 402
                    elif v == 1:
                        msg += k + 'Real Time'
                    elif v == 2:
                        msg += k + 'Best Effort'
                    elif v == 3:
                        msg += k + 'IDLE'
                    else:
                        msg += k + str(v)
                #  value is a number which goes from 0 to 7.
                # The higher the value, the lower the I/O priority of the process.
                if hasattr(p['ionice'], 'value') and p['ionice'].value != 0:
A
Alessio Sergi 已提交
403
                    msg += ' (value %s/7)' % str(p['ionice'].value)
D
desbma 已提交
404 405 406 407
                ret.append(self.curse_add_line(msg, splittable=True))

        return ret

408
    def msg_curse(self, args=None, max_width=None):
A
PEP 257  
Alessio Sergi 已提交
409
        """Return the dict to display in the curse interface."""
A
Alessio Sergi 已提交
410 411 412
        # Init the return message
        ret = []

413
        # Only process if stats exist and display plugin enable...
D
desbma 已提交
414
        if not self.stats or args.disable_process:
415 416
            return ret

A
Alessio Sergi 已提交
417
        # Compute the sort key
418
        process_sort_key = glances_processes.sort_key
A
Alessio Sergi 已提交
419 420

        # Header
421 422 423
        self.__msg_curse_header(ret, process_sort_key, args)

        # Process list
N
nicolargo 已提交
424 425 426 427 428 429 430 431 432 433 434 435 436
        # Loop over processes (sorted by the sort key previously compute)
        first = True
        for p in self.__sort_stats(process_sort_key):
            ret.extend(self.get_process_curses_data(p, first, args))
            # End of extended stats
            first = False
        if glances_processes.process_filter is not None:
            if args.reset_minmax_tag:
                args.reset_minmax_tag = not args.reset_minmax_tag
                self.__mmm_reset()
            self.__msg_curse_sum(ret, args=args)
            self.__msg_curse_sum(ret, mmm='min', args=args)
            self.__msg_curse_sum(ret, mmm='max', args=args)
437 438 439 440 441

        # Return the message with decoration
        return ret

    def __msg_curse_header(self, ret, process_sort_key, args=None):
442
        """Build the header and add it to the ret dict."""
443 444
        sort_style = 'SORT'

445
        if args.disable_irix and 0 < self.nb_log_core < 10:
446
            msg = self.layout_header['cpu'].format('CPU%/' + str(self.nb_log_core))
447
        elif args.disable_irix and self.nb_log_core != 0:
448
            msg = self.layout_header['cpu'].format('CPU%/C')
449
        else:
450
            msg = self.layout_header['cpu'].format('CPU%')
A
Alessio Sergi 已提交
451
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'cpu_percent' else 'DEFAULT'))
452
        msg = self.layout_header['mem'].format('MEM%')
A
Alessio Sergi 已提交
453
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'memory_percent' else 'DEFAULT'))
454
        msg = self.layout_header['virt'].format('VIRT')
A
Alessio Sergi 已提交
455
        ret.append(self.curse_add_line(msg, optional=True))
456
        msg = self.layout_header['res'].format('RES')
A
Alessio Sergi 已提交
457
        ret.append(self.curse_add_line(msg, optional=True))
458
        msg = self.layout_header['pid'].format('PID', width=self.__max_pid_size())
459
        ret.append(self.curse_add_line(msg))
460
        msg = self.layout_header['user'].format('USER')
A
Alessio Sergi 已提交
461
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'username' else 'DEFAULT'))
462 463 464
        msg = self.layout_header['time'].format('TIME+')
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'cpu_times' else 'DEFAULT', optional=True))
        msg = self.layout_header['thread'].format('THR')
465
        ret.append(self.curse_add_line(msg))
466
        msg = self.layout_header['nice'].format('NI')
467
        ret.append(self.curse_add_line(msg))
468 469 470
        msg = self.layout_header['status'].format('S')
        ret.append(self.curse_add_line(msg))
        msg = self.layout_header['ior'].format('R/s')
471
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'io_counters' else 'DEFAULT', optional=True, additional=True))
472
        msg = self.layout_header['iow'].format('W/s')
473
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'io_counters' else 'DEFAULT', optional=True, additional=True))
474
        msg = self.layout_header['command'].format('Command')
475
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'name' else 'DEFAULT'))
A
Alessio Sergi 已提交
476

N
nicolargo 已提交
477
    def __msg_curse_sum(self, ret, sep_char='_', mmm=None, args=None):
478
        """
479 480
        Build the sum message (only when filter is on) and add it to the ret dict.

N
nicolargo 已提交
481 482 483 484
        * ret: list of string where the message is added
        * sep_char: define the line separation char
        * mmm: display min, max, mean or current (if mmm=None)
        * args: Glances args
485 486
        """
        ret.append(self.curse_new_line())
N
nicolargo 已提交
487 488 489
        if mmm is None:
            ret.append(self.curse_add_line(sep_char * 69))
            ret.append(self.curse_new_line())
490
        # CPU percent sum
N
nicolargo 已提交
491
        msg = self.layout_stat['cpu'].format(self.__sum_stats('cpu_percent', mmm=mmm))
N
nicolargo 已提交
492 493
        ret.append(self.curse_add_line(msg,
                                       decoration=self.__mmm_deco(mmm)))
494
        # MEM percent sum
N
nicolargo 已提交
495
        msg = self.layout_stat['mem'].format(self.__sum_stats('memory_percent', mmm=mmm))
N
nicolargo 已提交
496 497
        ret.append(self.curse_add_line(msg,
                                       decoration=self.__mmm_deco(mmm)))
498 499 500
        # VIRT and RES memory sum
        if 'memory_info' in self.stats[0] and self.stats[0]['memory_info'] is not None and self.stats[0]['memory_info'] != '':
            # VMS
N
nicolargo 已提交
501
            msg = self.layout_stat['virt'].format(self.auto_unit(self.__sum_stats('memory_info', indice=1, mmm=mmm), low_precision=False))
N
nicolargo 已提交
502 503 504
            ret.append(self.curse_add_line(msg,
                                           decoration=self.__mmm_deco(mmm),
                                           optional=True))
505
            # RSS
N
nicolargo 已提交
506
            msg = self.layout_stat['res'].format(self.auto_unit(self.__sum_stats('memory_info', indice=0, mmm=mmm), low_precision=False))
N
nicolargo 已提交
507 508 509
            ret.append(self.curse_add_line(msg,
                                           decoration=self.__mmm_deco(mmm),
                                           optional=True))
D
desbma 已提交
510
        else:
N
nicolargo 已提交
511
            msg = self.layout_header['virt'].format('')
512
            ret.append(self.curse_add_line(msg))
N
nicolargo 已提交
513
            msg = self.layout_header['res'].format('')
514 515
            ret.append(self.curse_add_line(msg))
        # PID
N
nicolargo 已提交
516
        msg = self.layout_header['pid'].format('', width=self.__max_pid_size())
517 518
        ret.append(self.curse_add_line(msg))
        # USER
N
nicolargo 已提交
519 520 521 522 523 524 525
        msg = self.layout_header['user'].format('')
        ret.append(self.curse_add_line(msg))
        # TIME+
        msg = self.layout_header['time'].format('')
        ret.append(self.curse_add_line(msg, optional=True))
        # THREAD
        msg = self.layout_header['thread'].format('')
526 527
        ret.append(self.curse_add_line(msg))
        # NICE
N
nicolargo 已提交
528
        msg = self.layout_header['nice'].format('')
529 530
        ret.append(self.curse_add_line(msg))
        # STATUS
N
nicolargo 已提交
531
        msg = self.layout_header['status'].format('')
532 533
        ret.append(self.curse_add_line(msg))
        # IO read/write
N
nicolargo 已提交
534
        if 'io_counters' in self.stats[0] and mmm is None:
535
            # IO read
N
nicolargo 已提交
536
            io_rs = int((self.__sum_stats('io_counters', 0) - self.__sum_stats('io_counters', indice=2, mmm=mmm)) / self.stats[0]['time_since_update'])
537
            if io_rs == 0:
N
nicolargo 已提交
538
                msg = self.layout_stat['ior'].format('0')
539
            else:
N
nicolargo 已提交
540
                msg = self.layout_stat['ior'].format(self.auto_unit(io_rs, low_precision=True))
N
nicolargo 已提交
541 542 543
            ret.append(self.curse_add_line(msg,
                                           decoration=self.__mmm_deco(mmm),
                                           optional=True, additional=True))
544
            # IO write
N
nicolargo 已提交
545
            io_ws = int((self.__sum_stats('io_counters', 1) - self.__sum_stats('io_counters', indice=3, mmm=mmm)) / self.stats[0]['time_since_update'])
546
            if io_ws == 0:
N
nicolargo 已提交
547
                msg = self.layout_stat['iow'].format('0')
548
            else:
N
nicolargo 已提交
549
                msg = self.layout_stat['iow'].format(self.auto_unit(io_ws, low_precision=True))
N
nicolargo 已提交
550 551 552
            ret.append(self.curse_add_line(msg,
                                           decoration=self.__mmm_deco(mmm),
                                           optional=True, additional=True))
553
        else:
N
nicolargo 已提交
554
            msg = self.layout_header['ior'].format('')
555
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
N
nicolargo 已提交
556
            msg = self.layout_header['iow'].format('')
557
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
N
nicolargo 已提交
558
        if mmm is None:
559
            msg = ' < {}'.format('current')
N
nicolargo 已提交
560 561
            ret.append(self.curse_add_line(msg, optional=True))
        else:
562
            msg = ' < {}'.format(mmm)
563 564
            ret.append(self.curse_add_line(msg, optional=True))
            msg = ' (\'M\' to reset)'
N
nicolargo 已提交
565 566 567
            ret.append(self.curse_add_line(msg, optional=True))

    def __mmm_deco(self, mmm):
568
        """Return the decoration string for the current mmm status."""
N
nicolargo 已提交
569 570 571 572
        if mmm is not None:
            return 'DEFAULT'
        else:
            return 'FILTER'
N
Nicolargo 已提交
573

574
    def __mmm_reset(self):
575
        """Reset the MMM stats."""
576 577 578
        self.mmm_min = {}
        self.mmm_max = {}

N
nicolargo 已提交
579
    def __sum_stats(self, key, indice=None, mmm=None):
580 581
        """Return the sum of the stats value for the given key.

N
nicolargo 已提交
582 583
        * indice: If indice is set, get the p[key][indice]
        * mmm: display min, max, mean or current (if mmm=None)
584
        """
N
nicolargo 已提交
585
        # Compute stats summary
586 587
        ret = 0
        for p in self.stats:
588 589 590
            if key not in p:
                # Correct issue #1188
                continue
591 592 593
            if p[key] is None:
                # Correct https://github.com/nicolargo/glances/issues/1105#issuecomment-363553788
                continue
594 595 596 597
            if indice is None:
                ret += p[key]
            else:
                ret += p[key][indice]
N
nicolargo 已提交
598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627

        # Manage Min/Max/Mean
        mmm_key = self.__mmm_key(key, indice)
        if mmm == 'min':
            try:
                if self.mmm_min[mmm_key] > ret:
                    self.mmm_min[mmm_key] = ret
            except AttributeError:
                self.mmm_min = {}
                return 0
            except KeyError:
                self.mmm_min[mmm_key] = ret
            ret = self.mmm_min[mmm_key]
        elif mmm == 'max':
            try:
                if self.mmm_max[mmm_key] < ret:
                    self.mmm_max[mmm_key] = ret
            except AttributeError:
                self.mmm_max = {}
                return 0
            except KeyError:
                self.mmm_max[mmm_key] = ret
            ret = self.mmm_max[mmm_key]

        return ret

    def __mmm_key(self, key, indice):
        ret = key
        if indice is not None:
            ret += str(indice)
A
Alessio Sergi 已提交
628
        return ret
629

630
    def __sort_stats(self, sortedby=None):
631
        """Return the stats (dict) sorted by (sortedby)."""
632 633
        return sort_stats(self.stats, sortedby,
                          reverse=glances_processes.sort_reverse)
634 635

    def __max_pid_size(self):
636
        """Return the maximum PID size in number of char."""
637 638 639 640 641
        if self.pid_max is not None:
            return len(str(self.pid_max))
        else:
            # By default return 5 (corresponding to 99999 PID number)
            return 5