1
#!/usr/bin/python3 -bbI
2
2
# -*- mode: python; coding: utf-8 -*-
4
4
# Mandos Monitor - Control and monitor the Mandos server
6
# Copyright © 2009-2019 Teddy Hogeborn
7
# Copyright © 2009-2019 Björn Påhlsson
9
# This file is part of Mandos.
11
# Mandos is free software: you can redistribute it and/or modify it
12
# under the terms of the GNU General Public License as published by
6
# Copyright © 2009-2016 Teddy Hogeborn
7
# Copyright © 2009-2016 Björn Påhlsson
9
# This program is free software: you can redistribute it and/or modify
10
# it under the terms of the GNU General Public License as published by
13
11
# the Free Software Foundation, either version 3 of the License, or
14
12
# (at your option) any later version.
16
# Mandos is distributed in the hope that it will be useful, but
17
# WITHOUT ANY WARRANTY; without even the implied warranty of
14
# This program is distributed in the hope that it will be useful,
15
# but WITHOUT ANY WARRANTY; without even the implied warranty of
18
16
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19
17
# GNU General Public License for more details.
21
19
# You should have received a copy of the GNU General Public License
22
# along with Mandos. If not, see <http://www.gnu.org/licenses/>.
20
# along with this program. If not, see
21
# <http://www.gnu.org/licenses/>.
24
23
# Contact the authors at <mandos@recompile.se>.
26
26
from __future__ import (division, absolute_import, print_function,
30
29
from future_builtins import *
31
30
except ImportError:
41
38
import urwid.curses_display
49
50
if sys.version_info.major == 2:
54
# Show warnings by default
55
if not sys.warnoptions:
56
warnings.simplefilter("default")
58
log = logging.getLogger(os.path.basename(sys.argv[0]))
59
logging.basicConfig(level="NOTSET", # Show all messages
60
format="%(message)s") # Show basic log messages
62
logging.captureWarnings(True) # Show warnings via the logging system
64
locale.setlocale(locale.LC_ALL, "")
66
logging.getLogger("dbus.proxies").setLevel(logging.CRITICAL)
53
locale.setlocale(locale.LC_ALL, '')
55
logging.getLogger('dbus.proxies').setLevel(logging.CRITICAL)
68
57
# Some useful constants
69
domain = "se.recompile"
70
server_interface = domain + ".Mandos"
71
client_interface = domain + ".Mandos.Client"
58
domain = 'se.recompile'
59
server_interface = domain + '.Mandos'
60
client_interface = domain + '.Mandos.Client'
75
64
dbus.OBJECT_MANAGER_IFACE
134
123
def __init__(self, server_proxy_object=None, update_hook=None,
135
delete_hook=None, **kwargs):
124
delete_hook=None, logger=None, **kwargs):
136
125
# Called on update
137
126
self.update_hook = update_hook
138
127
# Called on delete
139
128
self.delete_hook = delete_hook
140
129
# Mandos Server proxy object
141
130
self.server_proxy_object = server_proxy_object
143
134
self._update_timer_callback_tag = None
180
172
if flag and self._update_timer_callback_tag is None:
181
173
# Will update the shown timer value every second
182
self._update_timer_callback_tag = (
183
GLib.timeout_add(1000,
184
glib_safely(self.update_timer)))
174
self._update_timer_callback_tag = (GLib.timeout_add
185
177
elif not (flag or self._update_timer_callback_tag is None):
186
178
GLib.source_remove(self._update_timer_callback_tag)
187
179
self._update_timer_callback_tag = None
189
181
def checker_completed(self, exitstatus, condition, command):
190
182
if exitstatus == 0:
191
log.debug('Checker for client %s (command "%s")'
192
" succeeded", self.properties["Name"], command)
183
self.logger('Checker for client {} (command "{}")'
184
' succeeded'.format(self.properties["Name"],
196
189
if os.WIFEXITED(condition):
197
log.info('Checker for client %s (command "%s") failed'
198
" with exit code %d", self.properties["Name"],
199
command, os.WEXITSTATUS(condition))
190
self.logger('Checker for client {} (command "{}") failed'
192
.format(self.properties["Name"], command,
193
os.WEXITSTATUS(condition)))
200
194
elif os.WIFSIGNALED(condition):
201
log.info('Checker for client %s (command "%s") was'
202
" killed by signal %d", self.properties["Name"],
203
command, os.WTERMSIG(condition))
195
self.logger('Checker for client {} (command "{}") was'
196
' killed by signal {}'
197
.format(self.properties["Name"], command,
198
os.WTERMSIG(condition)))
206
201
def checker_started(self, command):
207
202
"""Server signals that a checker started."""
208
log.debug('Client %s started checker "%s"',
209
self.properties["Name"], command)
203
self.logger('Client {} started checker "{}"'
204
.format(self.properties["Name"],
211
207
def got_secret(self):
212
log.info("Client %s received its secret",
213
self.properties["Name"])
208
self.logger('Client {} received its secret'
209
.format(self.properties["Name"]))
215
211
def need_approval(self, timeout, default):
217
message = "Client %s needs approval within %f seconds"
213
message = 'Client {} needs approval within {} seconds'
219
message = "Client %s will get its secret in %f seconds"
220
log.info(message, self.properties["Name"], timeout/1000)
215
message = 'Client {} will get its secret in {} seconds'
216
self.logger(message.format(self.properties["Name"],
222
219
def rejected(self, reason):
223
log.info("Client %s was rejected; reason: %s",
224
self.properties["Name"], reason)
220
self.logger('Client {} was rejected; reason: {}'
221
.format(self.properties["Name"], reason))
226
223
def selectable(self):
227
224
"""Make this a "selectable" widget.
281
278
timer = datetime.timedelta(0)
283
280
expires = (datetime.datetime.strptime
284
(expires, "%Y-%m-%dT%H:%M:%S.%f"))
281
(expires, '%Y-%m-%dT%H:%M:%S.%f'))
285
282
timer = max(expires - datetime.datetime.utcnow(),
286
283
datetime.timedelta())
287
message = ("A checker has failed! Time until client"
284
message = ('A checker has failed! Time until client'
289
286
.format(str(timer).rsplit(".", 1)[0]))
290
287
self.using_timer(True)
402
class UserInterface(object):
416
403
"""This is the entire user interface - the whole screen
417
404
with boxes, lists of client widgets, etc.
419
def __init__(self, max_log_length=1000):
406
def __init__(self, max_log_length=1000, log_level=1):
420
407
DBusGMainLoop(set_as_default=True)
422
409
self.screen = urwid.curses_display.Screen()
468
457
self.log_visible = True
469
458
self.log_wrap = "any"
471
self.loghandler = UILogHandler(self)
474
self.add_log_line(("bold",
475
"Mandos Monitor version " + version))
476
self.add_log_line(("bold", "q: Quit ?: Help"))
461
self.log_message_raw(("bold",
462
"Mandos Monitor version " + version))
463
self.log_message_raw(("bold",
478
self.busname = domain + ".Mandos"
466
self.busname = domain + '.Mandos'
479
467
self.main_loop = GLib.MainLoop()
481
def client_not_found(self, key_id, address):
482
log.info("Client with address %s and key ID %s could"
483
" not be found", address, key_id)
469
def client_not_found(self, fingerprint, address):
470
self.log_message("Client with address {} and fingerprint {}"
471
" could not be found"
472
.format(address, fingerprint))
485
474
def rebuild(self):
486
475
"""This rebuilds the User Interface.
497
486
self.uilist.append(self.logbox)
498
487
self.topwidget = urwid.Pile(self.uilist)
500
def add_log_line(self, markup):
489
def log_message(self, message, level=1):
490
"""Log message formatted with timestamp"""
491
if level < self.log_level:
493
timestamp = datetime.datetime.now().isoformat()
494
self.log_message_raw("{}: {}".format(timestamp, message),
497
def log_message_raw(self, markup, level=1):
498
"""Add a log message to the log buffer."""
499
if level < self.log_level:
501
501
self.log.append(urwid.Text(markup, wrap=self.log_wrap))
502
502
if self.max_log_length:
503
503
if len(self.log) > self.max_log_length:
504
del self.log[0:(len(self.log) - self.max_log_length)]
505
self.logbox.set_focus(len(self.logbox.body.contents)-1,
504
del self.log[0:len(self.log)-self.max_log_length-1]
505
self.logbox.set_focus(len(self.logbox.body.contents),
506
506
coming_from="above")
593
592
mandos_clients = (self.mandos_serv
594
593
.GetAllClientsWithProperties())
595
594
if not mandos_clients:
596
log.warning("Note: Server has no clients.")
595
self.log_message_raw(("bold",
596
"Note: Server has no clients."))
597
597
except dbus.exceptions.DBusException:
598
log.warning("Note: No Mandos server running.")
598
self.log_message_raw(("bold",
599
"Note: No Mandos server running."))
599
600
mandos_clients = dbus.Dictionary()
601
602
(self.mandos_serv
621
622
proxy_object=client_proxy_object,
622
623
properties=client,
623
624
update_hook=self.refresh,
624
delete_hook=self.remove_client),
625
delete_hook=self.remove_client,
626
logger=self.log_message),
628
self._input_callback_tag = (
630
GLib.IOChannel.unix_new(sys.stdin.fileno()),
631
GLib.PRIORITY_DEFAULT, GLib.IO_IN,
632
glib_safely(self.process_input)))
630
self._input_callback_tag = (GLib.io_add_watch
633
634
self.main_loop.run()
634
635
# Main loop has finished, we should close everything now
635
636
GLib.source_remove(self._input_callback_tag)
636
with warnings.catch_warnings():
637
warnings.simplefilter("ignore", BytesWarning)
641
640
self.main_loop.quit()
642
log.removeHandler(self.loghandler)
643
log.propagate = self.orig_log_propagate
645
642
def process_input(self, source, condition):
646
643
keys = self.screen.get_input()
706
704
self.topwidget.set_focus(self.logbox)
709
if log.level < logging.INFO:
710
log.setLevel(logging.INFO)
711
log.info("Verbose mode: Off")
707
if self.log_level == 0:
709
self.log_message("Verbose mode: Off")
713
log.setLevel(logging.NOTSET)
714
log.info("Verbose mode: On")
712
self.log_message("Verbose mode: On")
715
713
# elif (key == "end" or key == "meta >" or key == "G"
716
714
# or key == ">"):
717
715
# pass # xxx end-of-buffer
739
class UILogHandler(logging.Handler):
740
def __init__(self, ui, *args, **kwargs):
742
super(UILogHandler, self).__init__(*args, **kwargs)
744
logging.Formatter("%(asctime)s: %(message)s"))
745
def emit(self, record):
746
msg = self.format(record)
747
if record.levelno > logging.INFO:
749
self.ui.add_log_line(msg)
752
737
ui = UserInterface()
755
740
except KeyboardInterrupt:
756
with warnings.catch_warnings():
757
warnings.filterwarnings("ignore", "", BytesWarning)
760
with warnings.catch_warnings():
761
warnings.filterwarnings("ignore", "", BytesWarning)
742
except Exception as e:
743
ui.log_message(str(e))