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) 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
        # Note: 'glances_processes' is already init in the processes.py script
120

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

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

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

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

140 141 142 143 144 145 146
        # 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())

147
        return self.stats
148

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

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

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

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

        return ret

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

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

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

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

        # Process list
N
nicolargo 已提交
425 426 427 428 429 430 431 432 433 434 435 436 437
        # 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)
438 439 440 441 442

        # Return the message with decoration
        return ret

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

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

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

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

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

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

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

N
nicolargo 已提交
583 584
        * indice: If indice is set, get the p[key][indice]
        * mmm: display min, max, mean or current (if mmm=None)
585
        """
N
nicolargo 已提交
586
        # Compute stats summary
587 588
        ret = 0
        for p in self.stats:
589 590 591
            if key not in p:
                # Correct issue #1188
                continue
592 593 594
            if p[key] is None:
                # Correct https://github.com/nicolargo/glances/issues/1105#issuecomment-363553788
                continue
595 596 597 598
            if indice is None:
                ret += p[key]
            else:
                ret += p[key][indice]
N
nicolargo 已提交
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 628

        # 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 已提交
629
        return ret
630

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

    def __max_pid_size(self):
637
        """Return the maximum PID size in number of char."""
638 639 640 641 642
        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