424
423
.format(self.name)))
427
def call_pipe(connection, # : multiprocessing.Connection
428
func, *args, **kwargs):
429
"""This function is meant to be called by multiprocessing.Process
431
This function runs func(*args, **kwargs), and writes the resulting
432
return value on the provided multiprocessing.Connection.
434
connection.send(func(*args, **kwargs))
437
427
class Client(object):
438
428
"""A representation of a client host served by this server.
465
455
last_checker_status: integer between 0 and 255 reflecting exit
466
456
status of last checker. -1 reflects crashed
467
457
checker, -2 means no checker completed yet.
468
last_checker_signal: The signal which killed the last checker, if
469
last_checker_status is -1
470
458
last_enabled: datetime.datetime(); (UTC) or None
471
459
name: string; from the config file, used in log messages and
472
460
D-Bus identifiers
646
634
# Also start a new checker *right now*.
647
635
self.start_checker()
649
def checker_callback(self, source, condition, connection,
637
def checker_callback(self, pid, condition, command):
651
638
"""The checker has completed, so take appropriate actions."""
652
639
self.checker_callback_tag = None
653
640
self.checker = None
654
# Read return code from connection (see call_pipe)
655
returncode = connection.recv()
659
self.last_checker_status = returncode
660
self.last_checker_signal = None
641
if os.WIFEXITED(condition):
642
self.last_checker_status = os.WEXITSTATUS(condition)
661
643
if self.last_checker_status == 0:
662
644
logger.info("Checker for %(name)s succeeded",
666
648
logger.info("Checker for %(name)s failed", vars(self))
668
650
self.last_checker_status = -1
669
self.last_checker_signal = -returncode
670
651
logger.warning("Checker for %(name)s crashed?",
674
654
def checked_ok(self):
675
655
"""Assert that the client has been seen, alive and well."""
676
656
self.last_checked_ok = datetime.datetime.utcnow()
677
657
self.last_checker_status = 0
678
self.last_checker_signal = None
679
658
self.bump_timeout()
681
660
def bump_timeout(self, timeout=None):
707
686
# than 'timeout' for the client to be disabled, which is as it
710
if self.checker is not None and not self.checker.is_alive():
711
logger.warning("Checker was not alive; joining")
689
# If a checker exists, make sure it is not a zombie
691
pid, status = os.waitpid(self.checker.pid, os.WNOHANG)
692
except AttributeError:
694
except OSError as error:
695
if error.errno != errno.ECHILD:
699
logger.warning("Checker was a zombie")
700
gobject.source_remove(self.checker_callback_tag)
701
self.checker_callback(pid, status,
702
self.current_checker_command)
714
703
# Start a new checker if needed
715
704
if self.checker is None:
716
705
# Escape attributes for the shell
726
715
return True # Try again later
727
716
self.current_checker_command = command
728
logger.info("Starting checker %r for %s", command,
730
# We don't need to redirect stdout and stderr, since
731
# in normal mode, that is already done by daemon(),
732
# and in debug mode we don't want to. (Stdin is
733
# always replaced by /dev/null.)
734
# The exception is when not debugging but nevertheless
735
# running in the foreground; use the previously
737
popen_args = { "close_fds": True,
740
if (not self.server_settings["debug"]
741
and self.server_settings["foreground"]):
742
popen_args.update({"stdout": wnull,
744
pipe = multiprocessing.Pipe(duplex = False)
745
self.checker = multiprocessing.Process(
747
args = (pipe[1], subprocess.call, command),
750
self.checker_callback_tag = gobject.io_add_watch(
751
pipe[0].fileno(), gobject.IO_IN,
752
self.checker_callback, pipe[0], command)
718
logger.info("Starting checker %r for %s", command,
720
# We don't need to redirect stdout and stderr, since
721
# in normal mode, that is already done by daemon(),
722
# and in debug mode we don't want to. (Stdin is
723
# always replaced by /dev/null.)
724
# The exception is when not debugging but nevertheless
725
# running in the foreground; use the previously
728
if (not self.server_settings["debug"]
729
and self.server_settings["foreground"]):
730
popen_args.update({"stdout": wnull,
732
self.checker = subprocess.Popen(command,
737
except OSError as error:
738
logger.error("Failed to start subprocess",
741
self.checker_callback_tag = gobject.child_watch_add(
742
self.checker.pid, self.checker_callback, data=command)
743
# The checker may have completed before the gobject
744
# watch was added. Check for this.
746
pid, status = os.waitpid(self.checker.pid, os.WNOHANG)
747
except OSError as error:
748
if error.errno == errno.ECHILD:
749
# This should never happen
750
logger.error("Child process vanished",
755
gobject.source_remove(self.checker_callback_tag)
756
self.checker_callback(pid, status, command)
753
757
# Re-run this periodically if run by gobject.timeout_add
1099
1110
interface_names.add(alt_interface)
1100
1111
# Is this a D-Bus signal?
1101
1112
if getattr(attribute, "_dbus_is_signal", False):
1102
if sys.version_info.major == 2:
1103
# Extract the original non-method undecorated
1104
# function by black magic
1105
nonmethod_func = (dict(
1106
zip(attribute.func_code.co_freevars,
1107
attribute.__closure__))
1108
["func"].cell_contents)
1110
nonmethod_func = attribute
1113
# Extract the original non-method undecorated
1114
# function by black magic
1115
nonmethod_func = (dict(
1116
zip(attribute.func_code.co_freevars,
1117
attribute.__closure__))
1118
["func"].cell_contents)
1111
1119
# Create a new, but exactly alike, function
1112
1120
# object, and decorate it to be a new D-Bus signal
1113
1121
# with the alternate D-Bus interface name
1114
if sys.version_info.major == 2:
1115
new_function = types.FunctionType(
1116
nonmethod_func.func_code,
1117
nonmethod_func.func_globals,
1118
nonmethod_func.func_name,
1119
nonmethod_func.func_defaults,
1120
nonmethod_func.func_closure)
1122
new_function = types.FunctionType(
1123
nonmethod_func.__code__,
1124
nonmethod_func.__globals__,
1125
nonmethod_func.__name__,
1126
nonmethod_func.__defaults__,
1127
nonmethod_func.__closure__)
1128
1122
new_function = (dbus.service.signal(
1130
attribute._dbus_signature)(new_function))
1123
alt_interface, attribute._dbus_signature)
1124
(types.FunctionType(
1125
nonmethod_func.func_code,
1126
nonmethod_func.func_globals,
1127
nonmethod_func.func_name,
1128
nonmethod_func.func_defaults,
1129
nonmethod_func.func_closure)))
1131
1130
# Copy annotations, if any
1133
1132
new_function._dbus_annotations = dict(
1356
1355
DBusObjectWithProperties.__del__(self, *args, **kwargs)
1357
1356
Client.__del__(self, *args, **kwargs)
1359
def checker_callback(self, source, condition,
1360
connection, command, *args, **kwargs):
1361
ret = Client.checker_callback(self, source, condition,
1362
connection, command, *args,
1364
exitstatus = self.last_checker_status
1358
def checker_callback(self, pid, condition, command,
1360
self.checker_callback_tag = None
1362
if os.WIFEXITED(condition):
1363
exitstatus = os.WEXITSTATUS(condition)
1366
1364
# Emit D-Bus signal
1367
1365
self.CheckerCompleted(dbus.Int16(exitstatus),
1366
dbus.Int64(condition),
1369
1367
dbus.String(command))
1371
1369
# Emit D-Bus signal
1372
1370
self.CheckerCompleted(dbus.Int16(-1),
1374
self.last_checker_signal),
1371
dbus.Int64(condition),
1375
1372
dbus.String(command))
1374
return Client.checker_callback(self, pid, condition, command,
1378
1377
def start_checker(self, *args, **kwargs):
1379
1378
old_checker_pid = getattr(self.checker, "pid", None)
2183
2181
# avoid excessive use of external libraries.
2185
2183
# New type for defining tokens, syntax, and semantics all-in-one
2184
Token = collections.namedtuple("Token",
2185
("regexp", # To match token; if
2186
# "value" is not None,
2187
# must have a "group"
2189
"value", # datetime.timedelta or
2191
"followers")) # Tokens valid after
2186
2193
Token = collections.namedtuple("Token", (
2187
2194
"regexp", # To match token; if "value" is not None, must have
2188
2195
# a "group" containing digits