glances_processlist.py 27.0 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."""

N
Nicolas Hennion 已提交
22
# Import sys libs
23
import operator
24
import os
A
Alessio Sergi 已提交
25 26
from datetime import timedelta

N
Nicolas Hennion 已提交
27
# Import Glances libs
A
Alessio Sergi 已提交
28
from glances.core.compat import iteritems
29
from glances.core.glances_logging import logger
30
from glances.plugins.glances_core import Plugin as CorePlugin
A
Alessio Sergi 已提交
31 32
from glances.core.glances_globals import is_windows
from glances.core.glances_processes import glances_processes
A
Alessio Sergi 已提交
33
from glances.plugins.glances_plugin import GlancesPlugin
A
Alessio Sergi 已提交
34

D
desbma 已提交
35

36 37 38 39 40 41 42 43 44 45 46
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 已提交
47
class Plugin(GlancesPlugin):
A
PEP 257  
Alessio Sergi 已提交
48 49

    """Glances' processes plugin.
A
Alessio Sergi 已提交
50 51 52 53

    stats is a list
    """

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

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

61 62 63
        # Trying to display proc time
        self.tag_proc_time = True

64 65 66 67 68 69
        # 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

70
        # Note: 'glances_processes' is already init in the glances_processes.py script
71

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

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

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

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

97
        return self.stats
98

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

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

D
desbma 已提交
140 141 142 143 144 145 146 147 148 149 150
        # 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 已提交
151 152 153 154 155
        # draw node prefix
        if is_last_child:
            prefix = "└─"
        else:
            prefix = "├─"
D
desbma 已提交
156
        child_data[pos[0]]["msg"] = prefix
D
desbma 已提交
157 158 159

        # add indentation
        for i in pos:
D
desbma 已提交
160
            spacing = 2
D
desbma 已提交
161
            if first_level:
D
desbma 已提交
162 163 164 165 166
                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 已提交
167 168 169 170 171 172 173 174

        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 已提交
175
                    child_data[i]["msg"] = old_str[:2] + "│" + old_str[3:]
D
desbma 已提交
176 177
        return child_data

D
desbma 已提交
178
    def get_process_curses_data(self, p, first, args):
A
PEP 257  
Alessio Sergi 已提交
179
        """Get curses data to display for a process."""
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 184 185 186
            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 已提交
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 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250
            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)
            if isinstance(nice, int) and ((is_windows and nice != 32) or
                                          (not is_windows and nice != 0)):
                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:
251
                delta = timedelta(seconds=sum(p['cpu_times']))
252 253
            except (OverflowError, TypeError) as e:
                # Catch OverflowError on some Amazon EC2 server
D
desbma 已提交
254
                # See https://github.com/nicolargo/glances/issues/87
255 256
                # Also catch TypeError on Mac OS X
                # See: https://github.com/nicolargo/glances/issues/622
257
                logger.debug("Cannot get TIME+ ({0})".format(e))
D
desbma 已提交
258 259
                self.tag_proc_time = False
            else:
260 261
                hours, minutes, seconds, microseconds = convert_timedelta(delta)
                if hours:
262 263
                    msg = '{0:>4}h'.format(hours)
                    ret.append(self.curse_add_line(msg, decoration='CPU_TIME', optional=True))
264
                    msg = '{0}:{1}'.format(str(minutes).zfill(2), seconds)
265
                else:
266
                    msg = '{0:>4}:{1}.{2}'.format(minutes, seconds, microseconds)
D
desbma 已提交
267
        else:
268
            msg = '{0:>10}'.format('?')
D
desbma 已提交
269 270 271 272
        ret.append(self.curse_add_line(msg, optional=True))
        # IO read/write
        if 'io_counters' in p:
            # IO read
273
            io_rs = int((p['io_counters'][0] - p['io_counters'][2]) / p['time_since_update'])
D
desbma 已提交
274 275 276
            if io_rs == 0:
                msg = '{0:>6}'.format("0")
            else:
277
                msg = '{0:>6}'.format(self.auto_unit(io_rs, low_precision=True))
D
desbma 已提交
278 279
            ret.append(self.curse_add_line(msg, optional=True, additional=True))
            # IO write
280
            io_ws = int((p['io_counters'][1] - p['io_counters'][3]) / p['time_since_update'])
D
desbma 已提交
281 282 283
            if io_ws == 0:
                msg = '{0:>6}'.format("0")
            else:
284
                msg = '{0:>6}'.format(self.auto_unit(io_ws, low_precision=True))
D
desbma 已提交
285 286 287 288 289 290 291 292 293 294
            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
        # If no command line for the process is available, fallback to
        # the bare process name instead
        cmdline = p['cmdline']
295 296
        argument = ' '.join(cmdline.split()[1:])
        try:
297 298 299 300
            if cmdline == '':
                msg = ' {0}'.format(p['name'])
                ret.append(self.curse_add_line(msg, splittable=True))
            elif args.process_short_name:
301 302 303 304 305
                msg = ' {0}'.format(p['name'])
                ret.append(self.curse_add_line(msg, decoration='PROCESS', splittable=True))
                msg = ' {0}'.format(argument)
                ret.append(self.curse_add_line(msg, splittable=True))
            else:
D
desbma 已提交
306 307 308 309 310 311 312 313 314
                cmd = cmdline.split()[0]
                path, basename = os.path.split(cmd)
                if os.path.isdir(path):
                    msg = ' {0}'.format(path) + os.sep
                    ret.append(self.curse_add_line(msg, splittable=True))
                    ret.append(self.curse_add_line(basename, decoration='PROCESS', splittable=True))
                else:
                    msg = ' {0}'.format(basename)
                    ret.append(self.curse_add_line(msg, decoration='PROCESS', splittable=True))
315
                msg = ' {0}'.format(argument)
D
desbma 已提交
316
                ret.append(self.curse_add_line(msg, splittable=True))
317
        except UnicodeEncodeError:
318
            ret.append(self.curse_add_line('', splittable=True))
D
desbma 已提交
319 320 321 322 323 324 325 326 327 328

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

        return ret

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

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

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

        # Header
407 408 409 410 411 412 413 414 415 416 417 418 419 420 421
        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:
422 423 424
                if args.reset_minmax_tag:
                    args.reset_minmax_tag = not args.reset_minmax_tag
                    self.__mmm_reset()
425
                self.__msg_curse_sum(ret, args=args)
N
nicolargo 已提交
426 427
                self.__msg_curse_sum(ret, mmm='min', args=args)
                self.__msg_curse_sum(ret, mmm='max', args=args)
428 429 430 431 432 433 434 435 436 437

        # 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'

438 439 440 441 442 443
        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 已提交
444
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'cpu_percent' else 'DEFAULT'))
A
Alessio Sergi 已提交
445
        msg = '{0:>6}'.format('MEM%')
A
Alessio Sergi 已提交
446
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'memory_percent' else 'DEFAULT'))
A
Alessio Sergi 已提交
447
        msg = '{0:>6}'.format('VIRT')
A
Alessio Sergi 已提交
448
        ret.append(self.curse_add_line(msg, optional=True))
A
Alessio Sergi 已提交
449
        msg = '{0:>6}'.format('RES')
A
Alessio Sergi 已提交
450
        ret.append(self.curse_add_line(msg, optional=True))
A
Alessio Sergi 已提交
451
        msg = '{0:>6}'.format('PID')
452
        ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
453
        msg = ' {0:10}'.format('USER')
A
Alessio Sergi 已提交
454
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'username' else 'DEFAULT'))
A
Alessio Sergi 已提交
455
        msg = '{0:>4}'.format('NI')
456
        ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
457
        msg = '{0:>2}'.format('S')
458
        ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
459
        msg = '{0:>10}'.format('TIME+')
460
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'cpu_times' else 'DEFAULT', optional=True))
A
Alessio Sergi 已提交
461
        msg = '{0:>6}'.format('IOR/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:>6}'.format('IOW/s')
464
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'io_counters' else 'DEFAULT', optional=True, additional=True))
A
Alessio Sergi 已提交
465
        msg = ' {0:8}'.format('Command')
466
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'name' else 'DEFAULT'))
A
Alessio Sergi 已提交
467

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

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

N
nicolargo 已提交
568
    def __sum_stats(self, key, indice=None, mmm=None):
569 570
        """
        Return the sum of the stats value for the given key
N
nicolargo 已提交
571 572
        * indice: If indice is set, get the p[key][indice]
        * mmm: display min, max, mean or current (if mmm=None)
573
        """
N
nicolargo 已提交
574
        # Compute stats summary
575 576 577 578 579 580
        ret = 0
        for p in self.stats:
            if indice is None:
                ret += p[key]
            else:
                ret += p[key][indice]
N
nicolargo 已提交
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 609 610

        # 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 已提交
611
        return ret
612

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

619 620 621
        tree = glances_processes.is_tree_enabled()

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

A
Alessio Sergi 已提交
645
        return self.stats