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

N
Nicolas Hennion 已提交
26
# Import Glances libs
27
from glances.core.glances_globals import glances_processes, is_linux, is_bsd, is_mac, is_windows
A
Alessio Sergi 已提交
28
from glances.plugins.glances_plugin import GlancesPlugin
A
Alessio Sergi 已提交
29 30 31


class Plugin(GlancesPlugin):
A
PEP 257  
Alessio Sergi 已提交
32 33

    """Glances' processes plugin.
A
Alessio Sergi 已提交
34 35 36 37

    stats is a list
    """

38
    def __init__(self, args=None):
A
PEP 257  
Alessio Sergi 已提交
39
        """Init the plugin."""
40
        GlancesPlugin.__init__(self, args=args)
A
Alessio Sergi 已提交
41 42 43 44

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

45
        # Note: 'glances_processes' is already init in the glances_processes.py script
46

47
    def reset(self):
A
PEP 257  
Alessio Sergi 已提交
48
        """Reset/init the stats."""
49
        self.stats = []
50 51

    def update(self):
A
PEP 257  
Alessio Sergi 已提交
52
        """Update processes stats using the input method."""
53 54
        # Reset stats
        self.reset()
55

56
        if self.get_input() == 'local':
57 58 59
            # Update stats using the standard system lib
            # Note: Update is done in the processcount plugin
            # Just return the processes list
N
Nicolargo 已提交
60
            self.stats = glances_processes.getlist()            
61
        elif self.get_input() == 'snmp':
N
Nicolargo 已提交
62
            # No SNMP grab for processes
63
            pass
64

65
        return self.stats
66

A
Alessio Sergi 已提交
67
    def msg_curse(self, args=None):
A
PEP 257  
Alessio Sergi 已提交
68
        """Return the dict to display in the curse interface."""
A
Alessio Sergi 已提交
69 70 71
        # Init the return message
        ret = []

72
        # Only process if stats exist and display plugin enable...
73
        if self.stats == [] or args.disable_process:
74 75
            return ret

A
Alessio Sergi 已提交
76
        # Compute the sort key
N
Nicolargo 已提交
77 78
        if glances_processes.getmanualsortkey() is None:
            process_sort_key = glances_processes.getautosortkey()
A
Alessio Sergi 已提交
79
        else:
N
Nicolargo 已提交
80
            process_sort_key = glances_processes.getmanualsortkey()
81
        sort_style = 'SORT'
A
Alessio Sergi 已提交
82 83

        # Header
A
Alessio Sergi 已提交
84
        msg = '{0:>6}'.format(_("CPU%"))
A
Alessio Sergi 已提交
85
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'cpu_percent' else 'DEFAULT'))
A
Alessio Sergi 已提交
86
        msg = '{0:>6}'.format(_("MEM%"))
A
Alessio Sergi 已提交
87
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'memory_percent' else 'DEFAULT'))
A
Alessio Sergi 已提交
88
        msg = '{0:>6}'.format(_("VIRT"))
A
Alessio Sergi 已提交
89
        ret.append(self.curse_add_line(msg, optional=True))
A
Alessio Sergi 已提交
90
        msg = '{0:>6}'.format(_("RES"))
A
Alessio Sergi 已提交
91
        ret.append(self.curse_add_line(msg, optional=True))
A
Alessio Sergi 已提交
92
        msg = '{0:>6}'.format(_("PID"))
93
        ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
94
        msg = ' {0:10}'.format(_("USER"))
95
        ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
96
        msg = '{0:>4}'.format(_("NI"))
97
        ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
98
        msg = '{0:>2}'.format(_("S"))
99
        ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
100
        msg = '{0:>9}'.format(_("TIME+"))
A
Alessio Sergi 已提交
101
        ret.append(self.curse_add_line(msg, optional=True))
A
Alessio Sergi 已提交
102
        msg = '{0:>6}'.format(_("IOR/s"))
103
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'io_counters' else 'DEFAULT', optional=True, additional=True))
A
Alessio Sergi 已提交
104
        msg = '{0:>6}'.format(_("IOW/s"))
105
        ret.append(self.curse_add_line(msg, sort_style if process_sort_key == 'io_counters' else 'DEFAULT', optional=True, additional=True))
A
Alessio Sergi 已提交
106
        msg = ' {0:8}'.format(_("Command"))
107
        ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
108 109 110 111 112

        # Trying to display proc time
        tag_proc_time = True

        # Loop over processes (sorted by the sort key previously compute)
113
        for p in self.sortlist(process_sort_key):
A
Alessio Sergi 已提交
114 115
            ret.append(self.curse_new_line())
            # CPU
N
Nicolargo 已提交
116 117 118 119 120 121 122
            if 'cpu_percent' in p:
                msg = '{0:>6.1f}'.format(p['cpu_percent'])
                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))
A
Alessio Sergi 已提交
123
            # MEM
N
Nicolargo 已提交
124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142
            if 'memory_percent' in p:
                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:
                # 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))
A
Alessio Sergi 已提交
143
            # PID
A
Alessio Sergi 已提交
144
            msg = '{0:>6}'.format(p['pid'])
145
            ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
146
            # USER
N
Nicolargo 已提交
147 148 149 150 151 152 153
            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))
A
Alessio Sergi 已提交
154
            # NICE
N
Nicolargo 已提交
155 156 157 158 159 160 161 162 163 164
            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))
165
            else:
N
Nicolargo 已提交
166
                msg = '{0:>5}'.format('?')
167
                ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
168
            # STATUS
N
Nicolargo 已提交
169 170 171 172 173 174 175
            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))
176
            else:
N
Nicolargo 已提交
177
                msg = '{0:>2}'.format('?')
178
                ret.append(self.curse_add_line(msg))
A
Alessio Sergi 已提交
179
            # TIME+
180
            if tag_proc_time:
A
Alessio Sergi 已提交
181 182 183 184 185 186 187
                try:
                    dtime = timedelta(seconds=sum(p['cpu_times']))
                except Exception:
                    # Catched on some Amazon EC2 server
                    # See https://github.com/nicolargo/glances/issues/87
                    tag_proc_time = False
                else:
A
Alessio Sergi 已提交
188
                    msg = '{0}:{1}.{2}'.format(str(dtime.seconds // 60 % 60),
A
Alessio Sergi 已提交
189 190 191
                                               str(dtime.seconds % 60).zfill(2),
                                               str(dtime.microseconds)[:2].zfill(2))
            else:
A
Alessio Sergi 已提交
192 193
                msg = ' '
            msg = '{0:>9}'.format(msg)
A
Alessio Sergi 已提交
194
            ret.append(self.curse_add_line(msg, optional=True))
N
Nicolas Hennion 已提交
195
            # IO read/write
196
            if 'io_counters' in p:
N
Nicolas Hennion 已提交
197 198
                # IO read
                io_rs = (p['io_counters'][0] - p['io_counters'][2]) / p['time_since_update']
199
                if io_rs == 0:
A
Alessio Sergi 已提交
200
                    msg = '{0:>6}'.format("0")
N
Nicolas Hennion 已提交
201
                else:
A
Alessio Sergi 已提交
202
                    msg = '{0:>6}'.format(self.auto_unit(io_rs, low_precision=False))
203
                ret.append(self.curse_add_line(msg, optional=True, additional=True))
N
Nicolas Hennion 已提交
204 205
                # IO write
                io_ws = (p['io_counters'][1] - p['io_counters'][3]) / p['time_since_update']
206
                if io_ws == 0:
A
Alessio Sergi 已提交
207
                    msg = '{0:>6}'.format("0")
N
Nicolas Hennion 已提交
208
                else:
A
Alessio Sergi 已提交
209
                    msg = '{0:>6}'.format(self.auto_unit(io_ws, low_precision=False))
210
                ret.append(self.curse_add_line(msg, optional=True, additional=True))
A
Alessio Sergi 已提交
211
            else:
A
Alessio Sergi 已提交
212
                msg = '{0:>6}'.format("?")
213 214
                ret.append(self.curse_add_line(msg, optional=True, additional=True))
                ret.append(self.curse_add_line(msg, optional=True, additional=True))
215

A
Alessio Sergi 已提交
216
            # Command line
217 218 219 220
            # If no command line for the process is available, fallback to
            # the bare process name instead
            cmdline = p['cmdline']
            if cmdline == "":
A
Alessio Sergi 已提交
221
                msg = ' {0}'.format(p['name'])
222
                ret.append(self.curse_add_line(msg, splittable=True))
223 224 225 226 227 228
            else:
                try:
                    cmd = cmdline.split()[0]
                    args = ' '.join(cmdline.split()[1:])
                    path, basename = os.path.split(cmd)
                    if os.path.isdir(path):
A
Alessio Sergi 已提交
229
                        msg = ' {0}'.format(path) + os.sep
230 231
                        ret.append(self.curse_add_line(msg, splittable=True))
                        ret.append(self.curse_add_line(basename, decoration='PROCESS', splittable=True))
232
                    else:
A
Alessio Sergi 已提交
233
                        msg = ' {0}'.format(basename)
234
                        ret.append(self.curse_add_line(msg, decoration='PROCESS', splittable=True))
235
                    msg = " {0}".format(args)
236
                    ret.append(self.curse_add_line(msg, splittable=True))
237
                except UnicodeEncodeError:
238
                    ret.append(self.curse_add_line("", splittable=True))
A
Alessio Sergi 已提交
239 240 241

        # Return the message with decoration
        return ret
242 243

    def sortlist(self, sortedby=None):
A
PEP 257  
Alessio Sergi 已提交
244
        """Return the stats sorted by sortedby variable."""
245
        if sortedby is None:
246 247 248 249
            # No need to sort...
            return self.stats

        sortedreverse = True
250
        if sortedby == 'name':
251 252
            sortedreverse = False

253
        if sortedby == 'io_counters':
254 255 256 257 258 259 260 261 262 263 264 265 266 267 268
            # Specific case for io_counters
            # Sum of io_r + io_w
            try:
                # Sort process by IO rate (sum IO read + IO write)
                listsorted = sorted(self.stats,
                                    key=lambda process: process[sortedby][0] -
                                    process[sortedby][2] + process[sortedby][1] -
                                    process[sortedby][3],
                                    reverse=sortedreverse)
            except Exception:
                listsorted = sorted(self.stats,
                                    key=lambda process: process['cpu_percent'],
                                    reverse=sortedreverse)
        else:
            # Others sorts
N
Nicolargo 已提交
269 270 271 272 273 274 275 276
            try:
                listsorted = sorted(self.stats,
                                    key=lambda process: process[sortedby],
                                    reverse=sortedreverse)
            except KeyError:
                listsorted = sorted(self.stats,
                                    key=lambda process: process['name'],
                                    reverse=False)
277 278 279

        self.stats = listsorted

A
Alessio Sergi 已提交
280
        return self.stats