main.py 22.3 KB
Newer Older
N
Nicolas Hennion 已提交
1 2
# -*- coding: utf-8 -*-
#
3
# This file is part of Glances.
N
Nicolas Hennion 已提交
4
#
N
nicolargo 已提交
5
# Copyright (C) 2016 Nicolargo <nicolas@nicolargo.com>
N
Nicolas Hennion 已提交
6 7 8 9 10 11 12 13 14 15 16 17 18
#
# 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 已提交
19 20

"""Glances main class."""
N
Nicolas Hennion 已提交
21

N
Nicolas Hennion 已提交
22
import argparse
23
import os
24
import sys
25
import tempfile
N
Nicolas Hennion 已提交
26

27
from glances import __appname__, __version__, psutil_version
N
nicolargo 已提交
28
from glances.compat import input
29
from glances.config import Config
30
from glances.globals import LINUX, WINDOWS
31
from glances.logger import logger
N
Nicolas Hennion 已提交
32

N
Nicolas Hennion 已提交
33

34
class GlancesMain(object):
A
PEP 257  
Alessio Sergi 已提交
35 36

    """Main class to manage Glances instance."""
N
Nicolas Hennion 已提交
37

N
Nicolas Hennion 已提交
38 39
    # Default stats' refresh time is 3 seconds
    refresh_time = 3
N
Nicolas Hennion 已提交
40
    # Set the default cache lifetime to 1 second (only for server)
A
Alessio Sergi 已提交
41
    cached_time = 1
N
Nicolas Hennion 已提交
42 43 44 45
    # By default, Glances is ran in standalone mode (no client/server)
    client_tag = False
    # Server TCP port number (default is 61209)
    server_port = 61209
N
Nicolas Hennion 已提交
46 47
    # Web Server TCP port number (default is 61208)
    web_server_port = 61208
N
Nicolas Hennion 已提交
48 49 50 51
    # Default username/password for client/server mode
    username = "glances"
    password = ""

N
Nicolargo 已提交
52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
    # Exemple of use
    example_of_use = "\
Examples of use:\n\
\n\
Monitor local machine (standalone mode):\n\
  $ glances\n\
\n\
Monitor local machine with the Web interface (Web UI):\n\
  $ glances -w\n\
  Glances web server started on http://0.0.0.0:61208/\n\
\n\
Monitor local machine and export stats to a CSV file (standalone mode):\n\
  $ glances --export-csv\n\
\n\
Monitor local machine and export stats to a InfluxDB server with 5s refresh time (standalone mode):\n\
N
Nicolargo 已提交
67
  $ glances -t 5 --export-influxdb\n\
N
Nicolargo 已提交
68 69 70 71 72 73 74 75 76 77 78 79 80 81
\n\
Start a Glances server (server mode):\n\
  $ glances -s\n\
\n\
Connect Glances to a Glances server (client mode):\n\
  $ glances -c <ip_server>\n\
\n\
Connect Glances to a Glances server and export stats to a StatsD server (client mode):\n\
  $ glances -c <ip_server> --export-statsd\n\
\n\
Start the client browser (browser mode):\n\
  $ glances --browser\n\
    "

N
Nicolas Hennion 已提交
82
    def __init__(self):
A
Alessio Sergi 已提交
83
        """Manage the command line arguments."""
84
        # Read the command line arguments
A
Alessio Sergi 已提交
85 86 87 88
        self.args = self.parse_args()

    def init_args(self):
        """Init all the command line arguments."""
89
        version = "Glances v" + __version__ + " with psutil v" + psutil_version
90
        parser = argparse.ArgumentParser(
91
            prog=__appname__,
N
Nicolargo 已提交
92 93 94
            conflict_handler='resolve',
            formatter_class=argparse.RawDescriptionHelpFormatter,
            epilog=self.example_of_use)
95
        parser.add_argument(
96
            '-V', '--version', action='version', version=version)
N
Nicolas Hennion 已提交
97
        parser.add_argument('-d', '--debug', action='store_true', default=False,
A
Alessio Sergi 已提交
98
                            dest='debug', help='enable debug mode')
A
Alessio Sergi 已提交
99
        parser.add_argument('-C', '--config', dest='conf_file',
A
Alessio Sergi 已提交
100
                            help='path to the configuration file')
N
Nicolargo 已提交
101
        # Enable or disable option on startup
N
nicolargo 已提交
102 103 104 105
        parser.add_argument('--disable-alert', action='store_true', default=False,
                            dest='disable_alert', help='disable alert module')
        parser.add_argument('--disable-amps', action='store_true', default=False,
                            dest='disable_amps', help='disable applications monitoring process (AMP) module')
106 107
        parser.add_argument('--disable-cpu', action='store_true', default=False,
                            dest='disable_cpu', help='disable CPU module')
A
Alessio Sergi 已提交
108
        parser.add_argument('--disable-diskio', action='store_true', default=False,
A
Alessio Sergi 已提交
109
                            dest='disable_diskio', help='disable disk I/O module')
N
nicolargo 已提交
110 111 112 113
        parser.add_argument('--disable-docker', action='store_true', default=False,
                            dest='disable_docker', help='disable Docker module')
        parser.add_argument('--disable-folders', action='store_true', default=False,
                            dest='disable_folders', help='disable folder module')
A
Alessio Sergi 已提交
114
        parser.add_argument('--disable-fs', action='store_true', default=False,
A
Alessio Sergi 已提交
115
                            dest='disable_fs', help='disable filesystem module')
116
        parser.add_argument('--disable-hddtemp', action='store_true', default=False,
A
Alessio Sergi 已提交
117
                            dest='disable_hddtemp', help='disable HD temperature module')
N
nicolargo 已提交
118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133
        parser.add_argument('--disable-ip', action='store_true', default=False,
                            dest='disable_ip', help='disable IP module')
        parser.add_argument('--disable-irq', action='store_true', default=False,
                            dest='disable_irq', help='disable IRQ module'),
        parser.add_argument('--disable-load', action='store_true', default=False,
                            dest='disable_load', help='disable load module')
        parser.add_argument('--disable-mem', action='store_true', default=False,
                            dest='disable_mem', help='disable memory module')
        parser.add_argument('--disable-memswap', action='store_true', default=False,
                            dest='disable_memswap', help='disable memory swap module')
        parser.add_argument('--disable-network', action='store_true', default=False,
                            dest='disable_network', help='disable network module')
        parser.add_argument('--disable-ports', action='store_true', default=False,
                            dest='disable_ports', help='disable ports scanner module')
        parser.add_argument('--disable-process', action='store_true', default=False,
                            dest='disable_process', help='disable process module')
N
Nicolargo 已提交
134
        parser.add_argument('--disable-raid', action='store_true', default=False,
A
Alessio Sergi 已提交
135
                            dest='disable_raid', help='disable RAID module')
N
nicolargo 已提交
136 137 138 139
        parser.add_argument('--disable-sensors', action='store_true', default=False,
                            dest='disable_sensors', help='disable sensors module')
        parser.add_argument('--disable-wifi', action='store_true', default=False,
                            dest='disable_wifi', help='disable wifi module')
N
nicolargo 已提交
140 141 142 143
        parser.add_argument('-0', '--disable-irix', action='store_true', default=False,
                            dest='disable_irix', help='task\'s cpu usage will be divided by the total number of CPUs')
        parser.add_argument('-1', '--percpu', action='store_true', default=False,
                            dest='percpu', help='start Glances in per CPU mode')
144
        parser.add_argument('-2', '--disable-left-sidebar', action='store_true',
A
Alessio Sergi 已提交
145
                            default=False, dest='disable_left_sidebar',
146
                            help='disable network, disk I/O, FS and sensors modules')
N
nicolargo 已提交
147 148 149 150 151 152 153
        parser.add_argument('-3', '--disable-quicklook', action='store_true', default=False,
                            dest='disable_quicklook', help='disable quick look module')
        parser.add_argument('-4', '--full-quicklook', action='store_true', default=False,
                            dest='full_quicklook', help='disable all but quick look and load')
        parser.add_argument('-5', '--disable-top', action='store_true',
                            default=False, dest='disable_top',
                            help='disable top menu (QL, CPU, MEM, SWAP and LOAD)')
N
Nicolargo 已提交
154 155
        parser.add_argument('--disable-history', action='store_true', default=False,
                            dest='disable_history', help='disable stats history')
156
        parser.add_argument('--disable-bold', action='store_true', default=False,
A
Alessio Sergi 已提交
157
                            dest='disable_bold', help='disable bold mode in the terminal')
158
        parser.add_argument('--disable-bg', action='store_true', default=False,
159
                            dest='disable_bg', help='disable background colors in the terminal')
N
Nicolargo 已提交
160
        parser.add_argument('--enable-process-extended', action='store_true', default=False,
A
Alessio Sergi 已提交
161
                            dest='enable_process_extended', help='enable extended stats on top process')
162
        # Export modules feature
N
Nicolargo 已提交
163
        parser.add_argument('--export-graph', action='store_true', default=None,
N
nicolargo 已提交
164
                            dest='export_graph', help='export stats to graphs')
N
Nicolargo 已提交
165
        parser.add_argument('--path-graph', default=tempfile.gettempdir(),
N
nicolargo 已提交
166
                            dest='path_graph', help='set the export path for graphs (default is {0})'.format(tempfile.gettempdir()))
167
        parser.add_argument('--export-csv', default=None,
A
Alessio Sergi 已提交
168
                            dest='export_csv', help='export stats to a CSV file')
N
Nicolargo 已提交
169
        parser.add_argument('--export-influxdb', action='store_true', default=False,
170
                            dest='export_influxdb', help='export stats to an InfluxDB server (influxdb lib needed)')
171 172
        parser.add_argument('--export-cassandra', action='store_true', default=False,
                            dest='export_cassandra', help='export stats to a Cassandra or Scylla server (cassandra lib needed)')
173
        parser.add_argument('--export-opentsdb', action='store_true', default=False,
174
                            dest='export_opentsdb', help='export stats to an OpenTSDB server (potsdb lib needed)')
N
Nicolargo 已提交
175
        parser.add_argument('--export-statsd', action='store_true', default=False,
176 177 178
                            dest='export_statsd', help='export stats to a StatsD server (statsd lib needed)')
        parser.add_argument('--export-elasticsearch', action='store_true', default=False,
                            dest='export_elasticsearch', help='export stats to an ElasticSearch server (elasticsearch lib needed)')
A
Alessio Sergi 已提交
179
        parser.add_argument('--export-rabbitmq', action='store_true', default=False,
180
                            dest='export_rabbitmq', help='export stats to rabbitmq broker (pika lib needed)')
181 182
        parser.add_argument('--export-riemann', action='store_true', default=False,
                            dest='export_riemann', help='export stats to riemann broker (bernhard lib needed)')
N
nicolargo 已提交
183 184
        parser.add_argument('--export-couchdb', action='store_true', default=False,
                            dest='export_couchdb', help='export stats to a CouchDB server (couch lib needed)')
N
nicolargo 已提交
185 186
        parser.add_argument('--export-zeromq', action='store_true', default=False,
                            dest='export_zeromq', help='export stats to a ZeroMQ server (pyzmq lib needed)')
N
Nicolargo 已提交
187 188
        # Client/Server option
        parser.add_argument('-c', '--client', dest='client',
A
Alessio Sergi 已提交
189
                            help='connect to a Glances server by IPv4/IPv6 address or hostname')
N
Nicolargo 已提交
190
        parser.add_argument('-s', '--server', action='store_true', default=False,
A
Alessio Sergi 已提交
191
                            dest='server', help='run Glances in server mode')
192
        parser.add_argument('--browser', action='store_true', default=False,
A
Alessio Sergi 已提交
193
                            dest='browser', help='start the client browser (list of servers)')
194
        parser.add_argument('--disable-autodiscover', action='store_true', default=False,
A
Alessio Sergi 已提交
195
                            dest='disable_autodiscover', help='disable autodiscover feature')
196
        parser.add_argument('-p', '--port', default=None, type=int, dest='port',
197
                            help='define the client/server TCP port [default: {}]'.format(self.server_port))
N
Nicolargo 已提交
198
        parser.add_argument('-B', '--bind', default='0.0.0.0', dest='bind_address',
A
Alessio Sergi 已提交
199
                            help='bind server to the given IPv4/IPv6 address or hostname')
200 201
        parser.add_argument('--username', action='store_true', default=False, dest='username_prompt',
                            help='define a client/server username')
A
Alessio Sergi 已提交
202
        parser.add_argument('--password', action='store_true', default=False, dest='password_prompt',
A
Alessio Sergi 已提交
203
                            help='define a client/server password')
204
        parser.add_argument('--snmp-community', default='public', dest='snmp_community',
A
Alessio Sergi 已提交
205
                            help='SNMP community')
A
Alessio Sergi 已提交
206
        parser.add_argument('--snmp-port', default=161, type=int,
A
Alessio Sergi 已提交
207
                            dest='snmp_port', help='SNMP port')
N
Nicolargo 已提交
208
        parser.add_argument('--snmp-version', default='2c', dest='snmp_version',
A
Alessio Sergi 已提交
209
                            help='SNMP version (1, 2c or 3)')
N
Nicolargo 已提交
210
        parser.add_argument('--snmp-user', default='private', dest='snmp_user',
A
Alessio Sergi 已提交
211
                            help='SNMP username (only for SNMPv3)')
N
Nicolargo 已提交
212
        parser.add_argument('--snmp-auth', default='password', dest='snmp_auth',
A
Alessio Sergi 已提交
213
                            help='SNMP authentication key (only for SNMPv3)')
214
        parser.add_argument('--snmp-force', action='store_true', default=False,
A
Alessio Sergi 已提交
215
                            dest='snmp_force', help='force SNMP mode')
216
        parser.add_argument('-t', '--time', default=self.refresh_time, type=float,
217
                            dest='time', help='set refresh time in seconds [default: {} sec]'.format(self.refresh_time))
A
Alessio Sergi 已提交
218
        parser.add_argument('-w', '--webserver', action='store_true', default=False,
A
Alessio Sergi 已提交
219
                            dest='webserver', help='run Glances in web server mode (bottle needed)')
220 221
        parser.add_argument('--cached-time', default=self.cached_time, type=int,
                            dest='cached_time', help='set the server cache time [default: {} sec]'.format(self.cached_time))
222 223
        parser.add_argument('--open-web-browser', action='store_true', default=False,
                            dest='open_web_browser', help='try to open the Web UI in the default Web browser')
224
        # Display options
N
nicolargo 已提交
225
        parser.add_argument('-q', '--quiet', default=False, action='store_true',
A
Alessio Sergi 已提交
226
                            dest='quiet', help='do not display the curses interface')
N
Nicolargo 已提交
227
        parser.add_argument('-f', '--process-filter', default=None, type=str,
A
Alessio Sergi 已提交
228
                            dest='process_filter', help='set the process filter pattern (regular expression)')
229
        parser.add_argument('--process-short-name', action='store_true', default=False,
A
Alessio Sergi 已提交
230
                            dest='process_short_name', help='force short name for processes name')
A
Alessio Sergi 已提交
231
        if not WINDOWS:
232
            parser.add_argument('--hide-kernel-threads', action='store_true', default=False,
A
Alessio Sergi 已提交
233
                                dest='no_kernel_threads', help='hide kernel threads in process list')
A
Alessio Sergi 已提交
234
        if LINUX:
235 236
            parser.add_argument('--tree', action='store_true', default=False,
                                dest='process_tree', help='display processes as a tree')
N
Nicolargo 已提交
237
        parser.add_argument('-b', '--byte', action='store_true', default=False,
A
Alessio Sergi 已提交
238
                            dest='byte', help='display network rate in byte per second')
239 240
        parser.add_argument('--diskio-show-ramfs', action='store_true', default=False,
                            dest='diskio_show_ramfs', help='show RAM Fs in the DiskIO plugin')
241 242
        parser.add_argument('--diskio-iops', action='store_true', default=False,
                            dest='diskio_iops', help='show IO per second in the DiskIO plugin')
N
nicolargo 已提交
243 244
        parser.add_argument('--fahrenheit', action='store_true', default=False,
                            dest='fahrenheit', help='display temperature in Fahrenheit (default is Celsius)')
A
Alessio Sergi 已提交
245
        parser.add_argument('--fs-free-space', action='store_true', default=False,
A
Alessio Sergi 已提交
246
                            dest='fs_free_space', help='display FS free space instead of used')
N
Nicolargo 已提交
247
        parser.add_argument('--theme-white', action='store_true', default=False,
A
Alessio Sergi 已提交
248
                            dest='theme_white', help='optimize display colors for white background')
249 250 251
        # Globals options
        parser.add_argument('--disable-check-update', action='store_true', default=False,
                            dest='disable_check_update', help='disable online Glances version ckeck')
A
Alessio Sergi 已提交
252 253 254 255 256 257
        return parser

    def parse_args(self):
        """Parse command line arguments."""
        args = self.init_args().parse_args()

258
        # Load the configuration file, if it exists
A
Alessio Sergi 已提交
259 260
        self.config = Config(args.conf_file)

N
Nicolas Hennion 已提交
261 262 263 264 265
        # Debug mode
        if args.debug:
            from logging import DEBUG
            logger.setLevel(DEBUG)

266
        # Client/server Port
267 268 269 270
        if args.port is None:
            if args.webserver:
                args.port = self.web_server_port
            else:
271
                args.port = self.server_port
272

273 274 275 276
        # Autodiscover
        if args.disable_autodiscover:
            logger.info("Auto discover mode is disabled")

277
        # In web server mode
A
Alessio Sergi 已提交
278
        if args.webserver:
279
            args.process_short_name = True
N
Nicolas Hennion 已提交
280 281

        # Server or client login/password
282 283 284 285 286 287 288 289 290 291 292 293 294 295
        if args.username_prompt:
            # Every username needs a password
            args.password_prompt = True
            # Prompt username
            if args.server:
                args.username = self.__get_username(description='Define the Glances server username: ')
            elif args.webserver:
                args.username = self.__get_username(description='Define the Glances webserver username: ')
            elif args.client:
                args.username = self.__get_username(description='Enter the Glances server username: ')
        else:
            # Default user name is 'glances'
            args.username = self.username

296
        if args.password_prompt:
N
Nicolargo 已提交
297
            # Interactive or file password
A
Alessio Sergi 已提交
298
            if args.server:
N
Nicolas Hennion 已提交
299
                args.password = self.__get_password(
300
                    description='Define the Glances server password ({} username): '.format(args.username),
301 302
                    confirm=True,
                    username=args.username)
303 304
            elif args.webserver:
                args.password = self.__get_password(
305
                    description='Define the Glances webserver password ({} username): '.format(args.username),
306 307
                    confirm=True,
                    username=args.username)
A
Alessio Sergi 已提交
308
            elif args.client:
N
Nicolas Hennion 已提交
309
                args.password = self.__get_password(
310
                    description='Enter the Glances server password ({} username): '.format(args.username),
311 312
                    clear=True,
                    username=args.username)
N
Nicolas Hennion 已提交
313 314 315 316
        else:
            # Default is no password
            args.password = self.password

A
Alessio Sergi 已提交
317 318 319 320 321 322 323
        # By default help is hidden
        args.help_tag = False

        # Display Rx and Tx, not the sum for the network
        args.network_sum = False
        args.network_cumul = False

324 325
        # Manage full quicklook option
        if args.full_quicklook:
326
            logger.info("Disable QuickLook menu")
327 328 329
            args.disable_quicklook = False
            args.disable_cpu = True
            args.disable_mem = True
B
Beau Hastings 已提交
330
            args.disable_memswap = True
331
            args.disable_load = False
332 333 334 335 336 337 338

        # Manage disable_top option
        if args.disable_top:
            logger.info("Disable top menu")
            args.disable_quicklook = True
            args.disable_cpu = True
            args.disable_mem = True
B
Beau Hastings 已提交
339
            args.disable_memswap = True
340
            args.disable_load = True
341

342 343 344
        # Control parameter and exit if it is not OK
        self.args = args

345
        # Export is only available in standalone or client mode (issue #614)
N
nicolargo 已提交
346 347 348 349 350 351 352 353
        export_tag = args.export_csv or \
                     args.export_elasticsearch or \
                     args.export_statsd or \
                     args.export_influxdb or \
                     args.export_cassandra or \
                     args.export_opentsdb or \
                     args.export_rabbitmq or \
                     args.export_couchdb
354 355 356 357
        if not (self.is_standalone() or self.is_client()) and export_tag:
            logger.critical("Export is only available in standalone or client mode")
            sys.exit(2)

358 359
        # Filter is only available in standalone mode
        if args.process_filter is not None and not self.is_standalone():
A
Alessio Sergi 已提交
360
            logger.critical("Process filter is only available in standalone mode")
361 362
            sys.exit(2)

363
        # Check graph output path
N
nicolargo 已提交
364 365 366
        if args.export_graph and args.path_graph is not None:
            if not os.access(args.path_graph, os.W_OK):
                logger.critical("Graphs output path {0} do not exist or is not writable".format(args.path_graph))
367
                sys.exit(2)
N
nicolargo 已提交
368
            logger.debug("Graphs output path is set to {0}".format(args.path_graph))
369

N
Nicolargo 已提交
370 371 372 373 374
        # For export graph, history is mandatory
        if args.export_graph and args.disable_history:
            logger.critical("Can not export graph if history is disabled")
            sys.exit(2)

375 376 377 378 379
        # Disable HDDTemp if sensors are disabled
        if args.disable_sensors:
            args.disable_hddtemp = True
            logger.debug("Sensors and HDDTemp are disabled")

N
Nicolas Hennion 已提交
380
        return args
N
Nicolas Hennion 已提交
381

382 383 384 385 386 387
    def __get_username(self, description=''):
        """Read a username from the command line.
        """
        return input(description)

    def __get_password(self, description='', confirm=False, clear=False, username='glances'):
A
PEP 257  
Alessio Sergi 已提交
388 389 390 391
        """Read a password from the command line.

        - if confirm = True, with confirmation
        - if clear = True, plain (clear password)
N
Nicolargo 已提交
392
        """
393
        from glances.password import GlancesPassword
394
        password = GlancesPassword(username=username)
N
Nicolargo 已提交
395
        return password.get_password(description, confirm, clear)
N
Nicolas Hennion 已提交
396

N
Nicolas Hennion 已提交
397
    def is_standalone(self):
A
PEP 257  
Alessio Sergi 已提交
398
        """Return True if Glances is running in standalone mode."""
399
        return not self.args.client and not self.args.browser and not self.args.server and not self.args.webserver
N
Nicolas Hennion 已提交
400 401

    def is_client(self):
A
PEP 257  
Alessio Sergi 已提交
402
        """Return True if Glances is running in client mode."""
403
        return (self.args.client or self.args.browser) and not self.args.server
404

405 406 407
    def is_client_browser(self):
        """Return True if Glances is running in client browser mode."""
        return self.args.browser and not self.args.server
N
Nicolas Hennion 已提交
408 409

    def is_server(self):
A
PEP 257  
Alessio Sergi 已提交
410
        """Return True if Glances is running in server mode."""
411
        return not self.args.client and self.args.server
N
Nicolas Hennion 已提交
412

413
    def is_webserver(self):
A
PEP 257  
Alessio Sergi 已提交
414
        """Return True if Glances is running in Web server mode."""
415
        return not self.args.client and self.args.webserver
N
Nicolas Hennion 已提交
416

417
    def get_config(self):
A
PEP 257  
Alessio Sergi 已提交
418
        """Return configuration file object."""
419
        return self.config
N
Nicolas Hennion 已提交
420 421

    def get_args(self):
A
PEP 257  
Alessio Sergi 已提交
422
        """Return the arguments."""
A
Alessio Sergi 已提交
423
        return self.args