glances_processlist.py 26.8 KB
Newer Older
A
Alessio Sergi 已提交
1 2
# -*- coding: utf-8 -*-
#
3
# This file is part of Glances.
A
Alessio Sergi 已提交
4
#
5
# Copyright (C) 2015 Nicolargo <nicolas@nicolargo.com>
A
Alessio Sergi 已提交
6 7 8 9 10 11 12 13 14 15 16 17 18 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 operator
23
import os
A
Alessio Sergi 已提交
24 25
from datetime import timedelta

26
from glances.compat import iteritems
A
Alessio Sergi 已提交
27
from glances.globals import WINDOWS
28 29
from glances.logger import logger
from glances.processes import glances_processes
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 38 39 40 41 42 43 44
def convert_timedelta(delta):
    """Convert timedelta to human-readable time."""
    days, total_seconds = delta.days, delta.seconds
    hours = days * 24 + total_seconds // 3600
    minutes = (total_seconds % 3600) // 60
    seconds = str(total_seconds % 60).zfill(2)
    microseconds = str(delta.microseconds)[:2].zfill(2)

    return hours, minutes, seconds, microseconds


A
Alessio Sergi 已提交
45
class Plugin(GlancesPlugin):
A
PEP 257  
Alessio Sergi 已提交
46 47

    """Glances' processes plugin.
A
Alessio Sergi 已提交
48 49 50 51

    stats is a list
    """

52
    def __init__(self, args=None):
A
PEP 257  
Alessio Sergi 已提交
53
        """Init the plugin."""
A
Alessio Sergi 已提交
54
        super(Plugin, self).__init__(args=args)
A
Alessio Sergi 已提交
55 56 57 58

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

59 60 61
        # Trying to display proc time
        self.tag_proc_time = True

62 63 64 65 66 67
        # 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

68
        # Note: 'glances_processes' is already init in the processes.py script
69

70
    def get_key(self):
A
PEP 257  
Alessio Sergi 已提交
71
        """Return the key of the list."""
72 73
        return 'pid'

74
    def reset(self):
A
PEP 257  
Alessio Sergi 已提交
75
        """Reset/init the stats."""
76
        self.stats = []
77 78

    def update(self):
A
PEP 257  
Alessio Sergi 已提交
79
        """Update processes stats using the input method."""
80 81
        # Reset stats
        self.reset()
82

83
        if self.input_method == 'local':
84 85 86
            # Update stats using the standard system lib
            # Note: Update is done in the processcount plugin
            # Just return the processes list
87
            if glances_processes.is_tree_enabled():
D
desbma 已提交
88 89 90
                self.stats = glances_processes.gettree()
            else:
                self.stats = glances_processes.getlist()
91
        elif self.input_method == 'snmp':
N
Nicolargo 已提交
92
            # No SNMP grab for processes
93
            pass
94

95
        return self.stats
96

97
    def get_process_tree_curses_data(self, node, args, first_level=True, max_node_count=None):
A
PEP 257  
Alessio Sergi 已提交
98
        """Get curses data to display for a process tree."""
D
desbma 已提交
99
        ret = []
100
        node_count = 0
N
nicolargo 已提交
101
        if not node.is_root and ((max_node_count is None) or (max_node_count > 0)):
D
desbma 已提交
102
            node_data = self.get_process_curses_data(node.stats, False, args)
103
            node_count += 1
D
desbma 已提交
104
            ret.extend(node_data)
A
Alessio Sergi 已提交
105
        for child in node.iter_children():
106
            # stop if we have enough nodes to display
N
nicolargo 已提交
107
            if max_node_count is not None and node_count >= max_node_count:
108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
                break

            if max_node_count is None:
                children_max_node_count = None
            else:
                children_max_node_count = max_node_count - node_count
            child_data = self.get_process_tree_curses_data(child,
                                                           args,
                                                           first_level=node.is_root,
                                                           max_node_count=children_max_node_count)
            if max_node_count is None:
                node_count += len(child)
            else:
                node_count += min(children_max_node_count, len(child))

D
desbma 已提交
123
            if not node.is_root:
D
desbma 已提交
124
                child_data = self.add_tree_decoration(child_data, child is node.children[-1], first_level)
D
desbma 已提交
125 126 127
            ret.extend(child_data)
        return ret

D
desbma 已提交
128
    def add_tree_decoration(self, child_data, is_last_child, first_level):
A
PEP 257  
Alessio Sergi 已提交
129
        """Add tree curses decoration and indentation to a subtree."""
D
desbma 已提交
130 131 132
        # find process command indices in messages
        pos = []
        for i, m in enumerate(child_data):
N
nicolargo 已提交
133
            if m["msg"] == "\n" and m is not child_data[-1]:
D
desbma 已提交
134 135 136 137
                # new line pos + 12
                # TODO find a way to get rid of hardcoded 12 value
                pos.append(i + 12)

D
desbma 已提交
138 139 140 141 142 143 144 145 146 147 148
        # add new curses items for tree decoration
        new_child_data = []
        new_pos = []
        for i, m in enumerate(child_data):
            if i in pos:
                new_pos.append(len(new_child_data))
                new_child_data.append(self.curse_add_line(""))
            new_child_data.append(m)
        child_data = new_child_data
        pos = new_pos

D
desbma 已提交
149 150 151 152 153
        # draw node prefix
        if is_last_child:
            prefix = "└─"
        else:
            prefix = "├─"
D
desbma 已提交
154
        child_data[pos[0]]["msg"] = prefix
D
desbma 已提交
155 156 157

        # add indentation
        for i in pos:
D
desbma 已提交
158
            spacing = 2
D
desbma 已提交
159
            if first_level:
D
desbma 已提交
160 161 162 163 164
                spacing = 1
            elif is_last_child and (i is not pos[0]):
                # compensate indentation for missing '│' char
                spacing = 3
            child_data[i]["msg"] = "%s%s" % (" " * spacing, child_data[i]["msg"])
D
desbma 已提交
165 166 167 168 169 170 171 172

        if not is_last_child:
            # add '│' tree decoration
            for i in pos[1:]:
                old_str = child_data[i]["msg"]
                if first_level:
                    child_data[i]["msg"] = " │" + old_str[2:]
                else:
D
desbma 已提交
173
                    child_data[i]["msg"] = old_str[:2] + "│" + old_str[3:]
D
desbma 已提交
174 175
        return child_data

D
desbma 已提交
176
    def get_process_curses_data(self, p, first, args):
A
PEP 257  
Alessio Sergi 已提交
177
        """Get curses data to display for a process."""
178
        ret = [self.curse_new_line()]
D
desbma 已提交
179 180
        # CPU
        if 'cpu_percent' in p and p['cpu_percent'] is not None and p['cpu_percent'] != '':
181 182 183 184
            if args.disable_irix and self.nb_log_core != 0:
                msg = '{0:>6.1f}'.format(p['cpu_percent'] / float(self.nb_log_core))
            else:
                msg = '{0:>6.1f}'.format(p['cpu_percent'])
D
desbma 已提交
185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226
            ret.append(self.curse_add_line(msg,
                                           self.get_alert(p['cpu_percent'], header="cpu")))
        else:
            msg = '{0:>6}'.format('?')
            ret.append(self.curse_add_line(msg))
        # MEM
        if 'memory_percent' in p and p['memory_percent'] is not None and p['memory_percent'] != '':
            msg = '{0:>6.1f}'.format(p['memory_percent'])
            ret.append(self.curse_add_line(msg,
                                           self.get_alert(p['memory_percent'], header="mem")))
        else:
            msg = '{0:>6}'.format('?')
            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
            msg = '{0:>6}'.format(self.auto_unit(p['memory_info'][1], low_precision=False))
            ret.append(self.curse_add_line(msg, optional=True))
            # RSS
            msg = '{0:>6}'.format(self.auto_unit(p['memory_info'][0], low_precision=False))
            ret.append(self.curse_add_line(msg, optional=True))
        else:
            msg = '{0:>6}'.format('?')
            ret.append(self.curse_add_line(msg))
            ret.append(self.curse_add_line(msg))
        # PID
        msg = '{0:>6}'.format(p['pid'])
        ret.append(self.curse_add_line(msg))
        # USER
        if 'username' in p:
            # docker internal users are displayed as ints only, therefore str()
            msg = ' {0:9}'.format(str(p['username'])[:9])
            ret.append(self.curse_add_line(msg))
        else:
            msg = ' {0:9}'.format('?')
            ret.append(self.curse_add_line(msg))
        # NICE
        if 'nice' in p:
            nice = p['nice']
            if nice is None:
                nice = '?'
            msg = '{0:>5}'.format(nice)
A
Alessio Sergi 已提交
227 228
            if isinstance(nice, int) and ((WINDOWS and nice != 32) or
                                          (not WINDOWS and nice != 0)):
D
desbma 已提交
229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248
                ret.append(self.curse_add_line(msg, decoration='NICE'))
            else:
                ret.append(self.curse_add_line(msg))
        else:
            msg = '{0:>5}'.format('?')
            ret.append(self.curse_add_line(msg))
        # STATUS
        if 'status' in p:
            status = p['status']
            msg = '{0:>2}'.format(status)
            if status == 'R':
                ret.append(self.curse_add_line(msg, decoration='STATUS'))
            else:
                ret.append(self.curse_add_line(msg))
        else:
            msg = '{0:>2}'.format('?')
            ret.append(self.curse_add_line(msg))
        # TIME+
        if self.tag_proc_time:
            try:
249
                delta = timedelta(seconds=sum(p['cpu_times']))
250 251
            except (OverflowError, TypeError) as e:
                # Catch OverflowError on some Amazon EC2 server
D
desbma 已提交
252
                # See https://github.com/nicolargo/glances/issues/87
253 254
                # Also catch TypeError on Mac OS X
                # See: https://github.com/nicolargo/glances/issues/622
255
                logger.debug("Cannot get TIME+ ({0})".format(e))
D
desbma 已提交
256 257
                self.tag_proc_time = False
            else:
258 259
                hours, minutes, seconds, microseconds = convert_timedelta(delta)
                if hours:
260 261
                    msg = '{0:>4}h'.format(hours)
                    ret.append(self.curse_add_line(msg, decoration='CPU_TIME', optional=True))
262
                    msg = '{0}:{1}'.format(str(minutes).zfill(2), seconds)
263
                else:
264
                    msg = '{0:>4}:{1}.{2}'.format(minutes, seconds, microseconds)
D
desbma 已提交
265
        else:
266
            msg = '{0:>10}'.format('?')
D
desbma 已提交
267 268 269 270
        ret.append(self.curse_add_line(msg, optional=True))
        # IO read/write
        if 'io_counters' in p:
            # IO read
271
            io_rs = int((p['io_counters'][0] - p['io_counters'][2]) / p['time_since_update'])
D
desbma 已提交
272 273 274
            if io_rs == 0:
                msg = '{0:>6}'.format("0")
            else:
275
                msg = '{0:>6}'.format(self.auto_unit(io_rs, low_precision=True))
D
desbma 已提交
276 277
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
            # IO write
278
            io_ws = int((p['io_counters'][1] - p['io_counters'][3]) / p['time_since_update'])
D
desbma 已提交
279 280 281
            if io_ws == 0:
                msg = '{0:>6}'.format("0")
            else:
282
                msg = '{0:>6}'.format(self.auto_unit(io_ws, low_precision=True))
D
desbma 已提交
283 284 285 286 287 288 289
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
        else:
            msg = '{0:>6}'.format("?")
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
            ret.append(self.curse_add_line(msg, optional=True, additional=True))

        # Command line
290 291 292 293
        # If no command line for the process is available, fallback to
        # the bare process name instead
        cmdline = p['cmdline']
        argument = ' '.join(cmdline.split()[1:])
294
        try:
295 296
            if cmdline == '':
                msg = ' {0}'.format(p['name'])
297 298
                ret.append(self.curse_add_line(msg, splittable=True))
            elif args.process_short_name:
299
                msg = ' {0}'.format(p['name'])
300
                ret.append(self.curse_add_line(msg, decoration='PROCESS', splittable=True))
301
                msg = ' {0}'.format(argument)
302 303
                ret.append(self.curse_add_line(msg, splittable=True))
            else:
304 305 306 307
                cmd = cmdline.split()[0]
                path, basename = os.path.split(cmd)
                if os.path.isdir(path):
                    msg = ' {0}'.format(path) + os.sep
D
desbma 已提交
308
                    ret.append(self.curse_add_line(msg, splittable=True))
309
                    ret.append(self.curse_add_line(basename, decoration='PROCESS', splittable=True))
D
desbma 已提交
310
                else:
311 312 313 314
                    msg = ' {0}'.format(basename)
                    ret.append(self.curse_add_line(msg, decoration='PROCESS', splittable=True))
                msg = ' {0}'.format(argument)
                ret.append(self.curse_add_line(msg, splittable=True))
315
        except UnicodeEncodeError:
316
            ret.append(self.curse_add_line('', splittable=True))
D
desbma 已提交
317 318 319 320 321 322 323 324 325 326

        # Add extended stats but only for the top processes
        # !!! CPU consumption ???
        # TODO: extended stats into the web interface
        if first and 'extended_stats' in p:
            # 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 已提交
327
                msg = xpad + 'CPU affinity: ' + str(len(p['cpu_affinity'])) + ' cores'
D
desbma 已提交
328 329 330 331
                ret.append(self.curse_add_line(msg, splittable=True))
            # Second line is memory info
            if 'memory_info_ex' in p and p['memory_info_ex'] is not None:
                ret.append(self.curse_new_line())
A
Alessio Sergi 已提交
332
                msg = xpad + 'Memory info: '
A
Alessio Sergi 已提交
333
                for k, v in iteritems(p['memory_info_ex']._asdict()):
D
desbma 已提交
334 335 336 337
                    # Ignore rss and vms (already displayed)
                    if k not in ['rss', 'vms'] and v is not None:
                        msg += k + ' ' + self.auto_unit(v, low_precision=False) + ' '
                if 'memory_swap' in p and p['memory_swap'] is not None:
A
Alessio Sergi 已提交
338
                    msg += 'swap ' + self.auto_unit(p['memory_swap'], low_precision=False)
D
desbma 已提交
339 340 341 342
                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:
A
Alessio Sergi 已提交
343
                msg += 'threads ' + str(p['num_threads']) + ' '
D
desbma 已提交
344
            if 'num_fds' in p and p['num_fds'] is not None:
A
Alessio Sergi 已提交
345
                msg += 'files ' + str(p['num_fds']) + ' '
D
desbma 已提交
346
            if 'num_handles' in p and p['num_handles'] is not None:
A
Alessio Sergi 已提交
347
                msg += 'handles ' + str(p['num_handles']) + ' '
D
desbma 已提交
348
            if 'tcp' in p and p['tcp'] is not None:
A
Alessio Sergi 已提交
349
                msg += 'TCP ' + str(p['tcp']) + ' '
D
desbma 已提交
350
            if 'udp' in p and p['udp'] is not None:
A
Alessio Sergi 已提交
351
                msg += 'UDP ' + str(p['udp']) + ' '
D
desbma 已提交
352 353
            if msg != '':
                ret.append(self.curse_new_line())
A
Alessio Sergi 已提交
354
                msg = xpad + 'Open: ' + msg
D
desbma 已提交
355 356 357 358
                ret.append(self.curse_add_line(msg, splittable=True))
            # Fouth line is IO nice level (only Linux and Windows OS)
            if 'ionice' in p and p['ionice'] is not None:
                ret.append(self.curse_new_line())
A
Alessio Sergi 已提交
359 360
                msg = xpad + 'IO nice: '
                k = 'Class is '
D
desbma 已提交
361 362 363
                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 已提交
364
                if WINDOWS:
D
desbma 已提交
365 366 367 368 369
                    if v == 0:
                        msg += k + 'Very Low'
                    elif v == 1:
                        msg += k + 'Low'
                    elif v == 2:
A
Alessio Sergi 已提交
370
                        msg += 'No specific I/O priority'
D
desbma 已提交
371 372 373 374
                    else:
                        msg += k + str(v)
                else:
                    if v == 0:
A
Alessio Sergi 已提交
375
                        msg += 'No specific I/O priority'
D
desbma 已提交
376 377 378 379 380 381 382 383 384 385 386
                    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 已提交
387
                    msg += ' (value %s/7)' % str(p['ionice'].value)
D
desbma 已提交
388 389 390 391
                ret.append(self.curse_add_line(msg, splittable=True))

        return ret

A
Alessio Sergi 已提交
392
    def msg_curse(self, args=None):
A
PEP 257  
Alessio Sergi 已提交
393
        """Return the dict to display in the curse interface."""
A
Alessio Sergi 已提交
394 395 396
        # Init the return message
        ret = []

397
        # Only process if stats exist and display plugin enable...
D
desbma 已提交
398
        if not self.stats or args.disable_process:
399 400
            return ret

A
Alessio Sergi 已提交
401
        # Compute the sort key
402
        process_sort_key = glances_processes.sort_key
A
Alessio Sergi 已提交
403 404

        # Header
405 406 407 408 409 410 411 412 413 414 415 416 417 418 419
        self.__msg_curse_header(ret, process_sort_key, args)

        # Process list
        if glances_processes.is_tree_enabled():
            ret.extend(self.get_process_tree_curses_data(
                self.sort_stats(process_sort_key), args, first_level=True,
                max_node_count=glances_processes.max_processes))
        else:
            # 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:
420 421 422
                if args.reset_minmax_tag:
                    args.reset_minmax_tag = not args.reset_minmax_tag
                    self.__mmm_reset()
423
                self.__msg_curse_sum(ret, args=args)
N
nicolargo 已提交
424 425
                self.__msg_curse_sum(ret, mmm='min', args=args)
                self.__msg_curse_sum(ret, mmm='max', args=args)
426 427 428 429 430 431 432 433 434 435

        # Return the message with decoration
        return ret

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

436 437 438 439 440 441
        if args.disable_irix and 0 < self.nb_log_core < 10:
            msg = '{0:>6}'.format('CPU%/' + str(self.nb_log_core))
        elif args.disable_irix and self.nb_log_core != 0:
            msg = '{0:>6}'.format('CPU%/C')
        else:
            msg = '{0:>6}'.format('CPU%')
A
Alessio Sergi 已提交
442
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'cpu_percent' else 'DEFAULT'))
A
Alessio Sergi 已提交
443
        msg = '{0:>6}'.format('MEM%')
A
Alessio Sergi 已提交
444
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'memory_percent' else 'DEFAULT'))
A
Alessio Sergi 已提交
445
        msg = '{0:>6}'.format('VIRT')
A
Alessio Sergi 已提交
446
        ret.append(self.curse_add_line(msg, optional=True))
A
Alessio Sergi 已提交
447
        msg = '{0:>6}'.format('RES')
A
Alessio Sergi 已提交
448
        ret.append(self.curse_add_line(msg, optional=True))
A
Alessio Sergi 已提交
449
        msg = '{0:>6}'.format('PID')
450
        ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
451
        msg = ' {0:10}'.format('USER')
A
Alessio Sergi 已提交
452
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'username' else 'DEFAULT'))
A
Alessio Sergi 已提交
453
        msg = '{0:>4}'.format('NI')
454
        ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
455
        msg = '{0:>2}'.format('S')
456
        ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
457
        msg = '{0:>10}'.format('TIME+')
458
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'cpu_times' else 'DEFAULT', optional=True))
459
        msg = '{0:>6}'.format('R/s')
460
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'io_counters' else 'DEFAULT', optional=True, additional=True))
461
        msg = '{0:>6}'.format('W/s')
462
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'io_counters' else 'DEFAULT', optional=True, additional=True))
A
Alessio Sergi 已提交
463
        msg = ' {0:8}'.format('Command')
464
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'name' else 'DEFAULT'))
A
Alessio Sergi 已提交
465

N
nicolargo 已提交
466
    def __msg_curse_sum(self, ret, sep_char='_', mmm=None, args=None):
467 468
        """
        Build the sum message (only when filter is on) and add it to the ret dict
N
nicolargo 已提交
469 470 471 472
        * 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
473 474
        """
        ret.append(self.curse_new_line())
N
nicolargo 已提交
475 476 477
        if mmm is None:
            ret.append(self.curse_add_line(sep_char * 69))
            ret.append(self.curse_new_line())
478
        # CPU percent sum
N
nicolargo 已提交
479 480 481
        msg = '{0:>6.1f}'.format(self.__sum_stats('cpu_percent', mmm=mmm))
        ret.append(self.curse_add_line(msg,
                                       decoration=self.__mmm_deco(mmm)))
482
        # MEM percent sum
N
nicolargo 已提交
483 484 485
        msg = '{0:>6.1f}'.format(self.__sum_stats('memory_percent', mmm=mmm))
        ret.append(self.curse_add_line(msg,
                                       decoration=self.__mmm_deco(mmm)))
486 487 488
        # 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 已提交
489 490 491 492
            msg = '{0:>6}'.format(self.auto_unit(self.__sum_stats('memory_info', indice=1, mmm=mmm), low_precision=False))
            ret.append(self.curse_add_line(msg,
                                           decoration=self.__mmm_deco(mmm),
                                           optional=True))
493
            # RSS
N
nicolargo 已提交
494 495 496 497
            msg = '{0:>6}'.format(self.auto_unit(self.__sum_stats('memory_info', indice=0, mmm=mmm), low_precision=False))
            ret.append(self.curse_add_line(msg,
                                           decoration=self.__mmm_deco(mmm),
                                           optional=True))
D
desbma 已提交
498
        else:
499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517
            msg = '{0:>6}'.format('')
            ret.append(self.curse_add_line(msg))
            ret.append(self.curse_add_line(msg))
        # PID
        msg = '{0:>6}'.format('')
        ret.append(self.curse_add_line(msg))
        # USER
        msg = ' {0:9}'.format('')
        ret.append(self.curse_add_line(msg))
        # NICE
        msg = '{0:>5}'.format('')
        ret.append(self.curse_add_line(msg))
        # STATUS
        msg = '{0:>2}'.format('')
        ret.append(self.curse_add_line(msg))
        # TIME+
        msg = '{0:>10}'.format('')
        ret.append(self.curse_add_line(msg, optional=True))
        # IO read/write
N
nicolargo 已提交
518
        if 'io_counters' in self.stats[0] and mmm is None:
519
            # IO read
N
nicolargo 已提交
520
            io_rs = int((self.__sum_stats('io_counters', 0) - self.__sum_stats('io_counters', indice=2, mmm=mmm)) / self.stats[0]['time_since_update'])
521 522 523 524
            if io_rs == 0:
                msg = '{0:>6}'.format('0')
            else:
                msg = '{0:>6}'.format(self.auto_unit(io_rs, low_precision=True))
N
nicolargo 已提交
525 526 527
            ret.append(self.curse_add_line(msg,
                                           decoration=self.__mmm_deco(mmm),
                                           optional=True, additional=True))
528
            # IO write
N
nicolargo 已提交
529
            io_ws = int((self.__sum_stats('io_counters', 1) - self.__sum_stats('io_counters', indice=3, mmm=mmm)) / self.stats[0]['time_since_update'])
530 531 532 533
            if io_ws == 0:
                msg = '{0:>6}'.format('0')
            else:
                msg = '{0:>6}'.format(self.auto_unit(io_ws, low_precision=True))
N
nicolargo 已提交
534 535 536
            ret.append(self.curse_add_line(msg,
                                           decoration=self.__mmm_deco(mmm),
                                           optional=True, additional=True))
537 538 539 540
        else:
            msg = '{0:>6}'.format('')
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
N
nicolargo 已提交
541
        if mmm is None:
542
            msg = ' < {0}'.format('current')
N
nicolargo 已提交
543 544
            ret.append(self.curse_add_line(msg, optional=True))
        else:
545 546 547
            msg = ' < {0}'.format(mmm)
            ret.append(self.curse_add_line(msg, optional=True))
            msg = ' (\'M\' to reset)'
N
nicolargo 已提交
548 549 550 551 552 553 554 555 556 557
            ret.append(self.curse_add_line(msg, optional=True))

    def __mmm_deco(self, mmm):
        """
        Return the decoration string for the current mmm status
        """
        if mmm is not None:
            return 'DEFAULT'
        else:
            return 'FILTER'
N
Nicolargo 已提交
558

559 560 561 562 563 564 565
    def __mmm_reset(self):
        """
        Reset the MMM stats
        """
        self.mmm_min = {}
        self.mmm_max = {}

N
nicolargo 已提交
566
    def __sum_stats(self, key, indice=None, mmm=None):
567 568
        """
        Return the sum of the stats value for the given key
N
nicolargo 已提交
569 570
        * indice: If indice is set, get the p[key][indice]
        * mmm: display min, max, mean or current (if mmm=None)
571
        """
N
nicolargo 已提交
572
        # Compute stats summary
573 574 575 576 577 578
        ret = 0
        for p in self.stats:
            if indice is None:
                ret += p[key]
            else:
                ret += p[key][indice]
N
nicolargo 已提交
579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608

        # 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 已提交
609
        return ret
610

A
Alessio Sergi 已提交
611
    def sort_stats(self, sortedby=None):
A
PEP 257  
Alessio Sergi 已提交
612
        """Return the stats sorted by sortedby variable."""
613
        if sortedby is None:
614 615 616
            # No need to sort...
            return self.stats

617 618 619
        tree = glances_processes.is_tree_enabled()

        if sortedby == 'io_counters' and not tree:
620 621 622 623
            # Specific case for io_counters
            # Sum of io_r + io_w
            try:
                # Sort process by IO rate (sum IO read + IO write)
D
desbma 已提交
624 625 626
                self.stats.sort(key=lambda process: process[sortedby][0] -
                                process[sortedby][2] + process[sortedby][1] -
                                process[sortedby][3],
A
Alessio Sergi 已提交
627
                                reverse=glances_processes.sort_reverse)
628
            except Exception:
629
                self.stats.sort(key=operator.itemgetter('cpu_percent'),
A
Alessio Sergi 已提交
630
                                reverse=glances_processes.sort_reverse)
631 632
        else:
            # Others sorts
633
            if tree:
A
Alessio Sergi 已提交
634
                self.stats.set_sorting(sortedby, glances_processes.sort_reverse)
635 636
            else:
                try:
637
                    self.stats.sort(key=operator.itemgetter(sortedby),
A
Alessio Sergi 已提交
638
                                    reverse=glances_processes.sort_reverse)
639
                except (KeyError, TypeError):
640
                    self.stats.sort(key=operator.itemgetter('name'),
D
desbma 已提交
641
                                    reverse=False)
642

A
Alessio Sergi 已提交
643
        return self.stats