glances_processlist.py 27.3 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) 2019 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, config=None):
A
PEP 257  
Alessio Sergi 已提交
95
        """Init the plugin."""
96
        super(Plugin, self).__init__(args=args,
97
                                     config=config,
98
                                     stats_init_value=[])
A
Alessio Sergi 已提交
99 100 101 102

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

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

106 107 108 109 110 111
        # 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

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

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

119 120 121
        # Set the default sort key if it is defined in the configuration file
        if 'processlist' in config.as_dict() and 'sort_key' in config.as_dict()['processlist']:
            logger.debug('Configuration overwrites processes sort key by {}'.format(config.as_dict()['processlist']['sort_key']))
N
nicolargo 已提交
122
            glances_processes.set_sort_key(config.as_dict()['processlist']['sort_key'], False)
123

124
        # Note: 'glances_processes' is already init in the processes.py script
125

126
    def get_key(self):
A
PEP 257  
Alessio Sergi 已提交
127
        """Return the key of the list."""
128 129
        return 'pid'

130
    def update(self):
A
PEP 257  
Alessio Sergi 已提交
131
        """Update processes stats using the input method."""
132 133
        # Init new stats
        stats = self.get_init_value()
134

135
        if self.input_method == 'local':
136 137 138
            # Update stats using the standard system lib
            # Note: Update is done in the processcount plugin
            # Just return the processes list
139
            stats = glances_processes.getlist()
140

141
        elif self.input_method == 'snmp':
N
Nicolargo 已提交
142
            # No SNMP grab for processes
143
            pass
144

145 146 147 148 149 150 151
        # 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())

152
        return self.stats
153

154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173
    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 已提交
174
    def get_process_curses_data(self, p, first, args):
175
        """Get curses data to display for a process.
176

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

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

        # Add extended stats but only for the top processes
344
        if first and 'extended_stats' in p and args.enable_process_extended:
D
desbma 已提交
345 346 347 348 349
            # 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 已提交
350
                msg = xpad + 'CPU affinity: ' + str(len(p['cpu_affinity'])) + ' cores'
D
desbma 已提交
351 352
                ret.append(self.curse_add_line(msg, splittable=True))
            # Second line is memory info
353 354
            if 'memory_info' in p and \
               p['memory_info'] is not None:
D
desbma 已提交
355
                ret.append(self.curse_new_line())
356
                msg = '{}Memory info: {}'.format(xpad, p['memory_info'])
D
desbma 已提交
357
                if 'memory_swap' in p and p['memory_swap'] is not None:
358
                    msg += ' swap ' + self.auto_unit(p['memory_swap'], low_precision=False)
D
desbma 已提交
359 360 361 362
                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:
363
                msg += str(p['num_threads']) + ' threads '
D
desbma 已提交
364
            if 'num_fds' in p and p['num_fds'] is not None:
365
                msg += str(p['num_fds']) + ' files '
D
desbma 已提交
366
            if 'num_handles' in p and p['num_handles'] is not None:
367
                msg += str(p['num_handles']) + ' handles '
D
desbma 已提交
368
            if 'tcp' in p and p['tcp'] is not None:
369
                msg += str(p['tcp']) + ' TCP '
D
desbma 已提交
370
            if 'udp' in p and p['udp'] is not None:
371
                msg += str(p['udp']) + ' UDP'
D
desbma 已提交
372 373
            if msg != '':
                ret.append(self.curse_new_line())
A
Alessio Sergi 已提交
374
                msg = xpad + 'Open: ' + msg
D
desbma 已提交
375 376
                ret.append(self.curse_add_line(msg, splittable=True))
            # Fouth line is IO nice level (only Linux and Windows OS)
377 378 379
            if 'ionice' in p and \
               p['ionice'] is not None \
               and hasattr(p['ionice'], 'ioclass'):
D
desbma 已提交
380
                ret.append(self.curse_new_line())
A
Alessio Sergi 已提交
381 382
                msg = xpad + 'IO nice: '
                k = 'Class is '
D
desbma 已提交
383 384 385
                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 已提交
386
                if WINDOWS:
D
desbma 已提交
387 388 389 390 391
                    if v == 0:
                        msg += k + 'Very Low'
                    elif v == 1:
                        msg += k + 'Low'
                    elif v == 2:
A
Alessio Sergi 已提交
392
                        msg += 'No specific I/O priority'
D
desbma 已提交
393 394 395 396
                    else:
                        msg += k + str(v)
                else:
                    if v == 0:
A
Alessio Sergi 已提交
397
                        msg += 'No specific I/O priority'
D
desbma 已提交
398 399 400 401 402 403 404 405 406 407 408
                    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 已提交
409
                    msg += ' (value %s/7)' % str(p['ionice'].value)
D
desbma 已提交
410 411 412 413
                ret.append(self.curse_add_line(msg, splittable=True))

        return ret

414
    def msg_curse(self, args=None, max_width=None):
A
PEP 257  
Alessio Sergi 已提交
415
        """Return the dict to display in the curse interface."""
A
Alessio Sergi 已提交
416 417 418
        # Init the return message
        ret = []

419
        # Only process if stats exist and display plugin enable...
D
desbma 已提交
420
        if not self.stats or args.disable_process:
421 422
            return ret

A
Alessio Sergi 已提交
423
        # Compute the sort key
424
        process_sort_key = glances_processes.sort_key
A
Alessio Sergi 已提交
425 426

        # Header
427 428 429
        self.__msg_curse_header(ret, process_sort_key, args)

        # Process list
N
nicolargo 已提交
430 431 432 433 434 435 436 437 438 439 440 441 442
        # 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)
443 444 445 446 447

        # Return the message with decoration
        return ret

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

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

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

N
nicolargo 已提交
487 488 489 490
        * 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
491 492
        """
        ret.append(self.curse_new_line())
N
nicolargo 已提交
493 494 495
        if mmm is None:
            ret.append(self.curse_add_line(sep_char * 69))
            ret.append(self.curse_new_line())
496
        # CPU percent sum
N
nicolargo 已提交
497
        msg = self.layout_stat['cpu'].format(self.__sum_stats('cpu_percent', mmm=mmm))
N
nicolargo 已提交
498 499
        ret.append(self.curse_add_line(msg,
                                       decoration=self.__mmm_deco(mmm)))
500
        # MEM percent sum
N
nicolargo 已提交
501
        msg = self.layout_stat['mem'].format(self.__sum_stats('memory_percent', mmm=mmm))
N
nicolargo 已提交
502 503
        ret.append(self.curse_add_line(msg,
                                       decoration=self.__mmm_deco(mmm)))
504 505 506
        # 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 已提交
507
            msg = self.layout_stat['virt'].format(self.auto_unit(self.__sum_stats('memory_info', indice=1, mmm=mmm), low_precision=False))
N
nicolargo 已提交
508 509 510
            ret.append(self.curse_add_line(msg,
                                           decoration=self.__mmm_deco(mmm),
                                           optional=True))
511
            # RSS
N
nicolargo 已提交
512
            msg = self.layout_stat['res'].format(self.auto_unit(self.__sum_stats('memory_info', indice=0, mmm=mmm), low_precision=False))
N
nicolargo 已提交
513 514 515
            ret.append(self.curse_add_line(msg,
                                           decoration=self.__mmm_deco(mmm),
                                           optional=True))
D
desbma 已提交
516
        else:
N
nicolargo 已提交
517
            msg = self.layout_header['virt'].format('')
518
            ret.append(self.curse_add_line(msg))
N
nicolargo 已提交
519
            msg = self.layout_header['res'].format('')
520 521
            ret.append(self.curse_add_line(msg))
        # PID
N
nicolargo 已提交
522
        msg = self.layout_header['pid'].format('', width=self.__max_pid_size())
523 524
        ret.append(self.curse_add_line(msg))
        # USER
N
nicolargo 已提交
525 526 527 528 529 530 531
        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('')
532 533
        ret.append(self.curse_add_line(msg))
        # NICE
N
nicolargo 已提交
534
        msg = self.layout_header['nice'].format('')
535 536
        ret.append(self.curse_add_line(msg))
        # STATUS
N
nicolargo 已提交
537
        msg = self.layout_header['status'].format('')
538 539
        ret.append(self.curse_add_line(msg))
        # IO read/write
N
nicolargo 已提交
540
        if 'io_counters' in self.stats[0] and mmm is None:
541
            # IO read
N
nicolargo 已提交
542
            io_rs = int((self.__sum_stats('io_counters', 0) - self.__sum_stats('io_counters', indice=2, mmm=mmm)) / self.stats[0]['time_since_update'])
543
            if io_rs == 0:
N
nicolargo 已提交
544
                msg = self.layout_stat['ior'].format('0')
545
            else:
N
nicolargo 已提交
546
                msg = self.layout_stat['ior'].format(self.auto_unit(io_rs, low_precision=True))
N
nicolargo 已提交
547 548 549
            ret.append(self.curse_add_line(msg,
                                           decoration=self.__mmm_deco(mmm),
                                           optional=True, additional=True))
550
            # IO write
N
nicolargo 已提交
551
            io_ws = int((self.__sum_stats('io_counters', 1) - self.__sum_stats('io_counters', indice=3, mmm=mmm)) / self.stats[0]['time_since_update'])
552
            if io_ws == 0:
N
nicolargo 已提交
553
                msg = self.layout_stat['iow'].format('0')
554
            else:
N
nicolargo 已提交
555
                msg = self.layout_stat['iow'].format(self.auto_unit(io_ws, low_precision=True))
N
nicolargo 已提交
556 557 558
            ret.append(self.curse_add_line(msg,
                                           decoration=self.__mmm_deco(mmm),
                                           optional=True, additional=True))
559
        else:
N
nicolargo 已提交
560
            msg = self.layout_header['ior'].format('')
561
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
N
nicolargo 已提交
562
            msg = self.layout_header['iow'].format('')
563
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
N
nicolargo 已提交
564
        if mmm is None:
565
            msg = ' < {}'.format('current')
N
nicolargo 已提交
566 567
            ret.append(self.curse_add_line(msg, optional=True))
        else:
568
            msg = ' < {}'.format(mmm)
569 570
            ret.append(self.curse_add_line(msg, optional=True))
            msg = ' (\'M\' to reset)'
N
nicolargo 已提交
571 572 573
            ret.append(self.curse_add_line(msg, optional=True))

    def __mmm_deco(self, mmm):
574
        """Return the decoration string for the current mmm status."""
N
nicolargo 已提交
575 576 577 578
        if mmm is not None:
            return 'DEFAULT'
        else:
            return 'FILTER'
N
Nicolargo 已提交
579

580
    def __mmm_reset(self):
581
        """Reset the MMM stats."""
582 583 584
        self.mmm_min = {}
        self.mmm_max = {}

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

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

        # 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 已提交
634
        return ret
635

636
    def __sort_stats(self, sortedby=None):
637
        """Return the stats (dict) sorted by (sortedby)."""
638 639
        return sort_stats(self.stats, sortedby,
                          reverse=glances_processes.sort_reverse)
640 641

    def __max_pid_size(self):
642
        """Return the maximum PID size in number of char."""
643 644 645 646 647
        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