2018-10-14 02:01:35 -04:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
2021-07-18 17:03:57 -04:00
|
|
|
# log.py - PVC daemon logger functions
|
2018-10-14 02:01:35 -04:00
|
|
|
# Part of the Parallel Virtual Cluster (PVC) system
|
|
|
|
#
|
2021-03-25 17:01:55 -04:00
|
|
|
# Copyright (C) 2018-2021 Joshua M. Boniface <joshua@boniface.me>
|
2018-10-14 02:01:35 -04:00
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
2021-03-25 16:57:17 -04:00
|
|
|
# the Free Software Foundation, version 3.
|
2018-10-14 02:01:35 -04:00
|
|
|
#
|
|
|
|
# This program 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 General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
#
|
|
|
|
###############################################################################
|
|
|
|
|
2021-07-18 17:03:57 -04:00
|
|
|
from collections import deque
|
|
|
|
from threading import Thread
|
|
|
|
from queue import Queue
|
|
|
|
from datetime import datetime
|
2021-07-19 11:45:58 -04:00
|
|
|
from time import sleep
|
2021-07-18 17:03:57 -04:00
|
|
|
|
|
|
|
from daemon_lib.zkhandler import ZKHandler
|
2018-10-14 02:01:35 -04:00
|
|
|
|
2020-11-07 14:45:24 -05:00
|
|
|
|
2018-10-14 02:01:35 -04:00
|
|
|
class Logger(object):
|
|
|
|
# Define a logger class for a daemon instance
|
|
|
|
# Keeps record of where to log, and is passed messages which are
|
|
|
|
# formatted in various ways based off secondary characteristics.
|
|
|
|
|
|
|
|
# ANSII colours for output
|
|
|
|
fmt_red = '\033[91m'
|
|
|
|
fmt_green = '\033[92m'
|
|
|
|
fmt_yellow = '\033[93m'
|
2020-08-17 12:46:52 -04:00
|
|
|
fmt_blue = '\033[94m'
|
2018-10-14 02:01:35 -04:00
|
|
|
fmt_purple = '\033[95m'
|
2020-08-17 12:46:52 -04:00
|
|
|
fmt_cyan = '\033[96m'
|
|
|
|
fmt_white = '\033[97m'
|
2020-11-07 12:10:24 -05:00
|
|
|
fmt_bold = '\033[1m'
|
2018-10-14 02:01:35 -04:00
|
|
|
fmt_end = '\033[0m'
|
|
|
|
|
2019-07-10 21:39:25 -04:00
|
|
|
last_colour = ''
|
|
|
|
last_prompt = ''
|
|
|
|
|
|
|
|
# Format maps
|
|
|
|
format_map_colourized = {
|
2020-11-06 22:29:49 -05:00
|
|
|
# Colourized formatting with chevron prompts (log_colours = True)
|
2020-11-07 14:52:39 -05:00
|
|
|
'o': {'colour': fmt_green, 'prompt': '>>> '},
|
|
|
|
'e': {'colour': fmt_red, 'prompt': '>>> '},
|
|
|
|
'w': {'colour': fmt_yellow, 'prompt': '>>> '},
|
|
|
|
't': {'colour': fmt_purple, 'prompt': '>>> '},
|
|
|
|
'i': {'colour': fmt_blue, 'prompt': '>>> '},
|
|
|
|
's': {'colour': fmt_cyan, 'prompt': '>>> '},
|
|
|
|
'd': {'colour': fmt_white, 'prompt': '>>> '},
|
2020-11-07 12:57:42 -05:00
|
|
|
'x': {'colour': last_colour, 'prompt': last_prompt}
|
2019-07-10 21:39:25 -04:00
|
|
|
}
|
|
|
|
format_map_textual = {
|
2020-11-06 22:29:49 -05:00
|
|
|
# Uncolourized formatting with text prompts (log_colours = False)
|
2020-11-07 12:57:42 -05:00
|
|
|
'o': {'colour': '', 'prompt': 'ok: '},
|
|
|
|
'e': {'colour': '', 'prompt': 'failed: '},
|
|
|
|
'w': {'colour': '', 'prompt': 'warning: '},
|
|
|
|
't': {'colour': '', 'prompt': 'tick: '},
|
|
|
|
'i': {'colour': '', 'prompt': 'info: '},
|
|
|
|
's': {'colour': '', 'prompt': 'system: '},
|
|
|
|
'd': {'colour': '', 'prompt': 'debug: '},
|
|
|
|
'x': {'colour': '', 'prompt': last_prompt}
|
2019-07-10 21:39:25 -04:00
|
|
|
}
|
|
|
|
|
2018-10-14 02:01:35 -04:00
|
|
|
# Initialization of instance
|
|
|
|
def __init__(self, config):
|
|
|
|
self.config = config
|
2019-07-10 21:39:25 -04:00
|
|
|
|
2019-07-10 22:20:24 -04:00
|
|
|
if self.config['file_logging']:
|
2018-10-14 02:01:35 -04:00
|
|
|
self.logfile = self.config['log_directory'] + '/pvc.log'
|
|
|
|
# We open the logfile for the duration of our session, but have a hup function
|
2021-06-01 18:50:26 -04:00
|
|
|
self.writer = open(self.logfile, 'a', buffering=0)
|
2019-07-10 21:39:25 -04:00
|
|
|
|
|
|
|
self.last_colour = ''
|
|
|
|
self.last_prompt = ''
|
2019-06-25 22:31:04 -04:00
|
|
|
|
2021-07-18 17:03:57 -04:00
|
|
|
if self.config['zookeeper_logging']:
|
2021-07-19 11:45:58 -04:00
|
|
|
self.zookeeper_queue = Queue()
|
|
|
|
self.zookeeper_logger = ZookeeperLogger(self.config, self.zookeeper_queue)
|
2021-07-18 17:03:57 -04:00
|
|
|
self.zookeeper_logger.start()
|
|
|
|
|
2018-10-14 02:01:35 -04:00
|
|
|
# Provide a hup function to close and reopen the writer
|
|
|
|
def hup(self):
|
|
|
|
self.writer.close()
|
|
|
|
self.writer = open(self.logfile, 'a', buffering=0)
|
|
|
|
|
2021-07-18 19:53:00 -04:00
|
|
|
# Provide a termination function so all messages are flushed before terminating the main daemon
|
|
|
|
def terminate(self):
|
|
|
|
if self.config['file_logging']:
|
|
|
|
self.writer.close()
|
|
|
|
if self.config['zookeeper_logging']:
|
2021-07-19 11:45:58 -04:00
|
|
|
self.out("Waiting 15s for Zookeeper message queue to drain", state='s')
|
|
|
|
|
|
|
|
tick_count = 0
|
|
|
|
while not self.zookeeper_queue.empty():
|
|
|
|
sleep(0.5)
|
|
|
|
tick_count += 1
|
|
|
|
if tick_count > 30:
|
|
|
|
break
|
|
|
|
|
2021-07-18 19:53:00 -04:00
|
|
|
self.zookeeper_logger.stop()
|
|
|
|
self.zookeeper_logger.join()
|
|
|
|
|
2018-10-14 02:01:35 -04:00
|
|
|
# Output function
|
2019-07-10 21:39:25 -04:00
|
|
|
def out(self, message, state=None, prefix=''):
|
2018-10-14 02:01:35 -04:00
|
|
|
|
|
|
|
# Get the date
|
2019-07-10 21:39:25 -04:00
|
|
|
if self.config['log_dates']:
|
2021-07-18 19:00:54 -04:00
|
|
|
date = '{} '.format(datetime.now().strftime('%Y/%m/%d %H:%M:%S.%f'))
|
2019-07-10 21:39:25 -04:00
|
|
|
else:
|
|
|
|
date = ''
|
|
|
|
|
|
|
|
# Get the format map
|
|
|
|
if self.config['log_colours']:
|
|
|
|
format_map = self.format_map_colourized
|
|
|
|
endc = Logger.fmt_end
|
2018-10-14 02:01:35 -04:00
|
|
|
else:
|
2019-07-10 21:39:25 -04:00
|
|
|
format_map = self.format_map_textual
|
|
|
|
endc = ''
|
|
|
|
|
|
|
|
# Define an undefined state as 'x'; no date in these prompts
|
|
|
|
if not state:
|
|
|
|
state = 'x'
|
2018-10-14 02:01:35 -04:00
|
|
|
date = ''
|
2019-06-25 22:31:04 -04:00
|
|
|
|
2019-07-10 21:39:25 -04:00
|
|
|
# Get colour and prompt from the map
|
|
|
|
colour = format_map[state]['colour']
|
|
|
|
prompt = format_map[state]['prompt']
|
|
|
|
|
|
|
|
# Append space and separator to prefix
|
2018-10-14 02:01:35 -04:00
|
|
|
if prefix != '':
|
|
|
|
prefix = prefix + ' - '
|
2019-06-25 22:31:04 -04:00
|
|
|
|
2019-07-10 21:39:25 -04:00
|
|
|
# Assemble message string
|
2018-10-14 02:01:35 -04:00
|
|
|
message = colour + prompt + endc + date + prefix + message
|
2019-07-10 21:39:25 -04:00
|
|
|
|
|
|
|
# Log to stdout
|
|
|
|
if self.config['stdout_logging']:
|
|
|
|
print(message)
|
|
|
|
|
|
|
|
# Log to file
|
|
|
|
if self.config['file_logging']:
|
2018-10-14 02:01:35 -04:00
|
|
|
self.writer.write(message + '\n')
|
2019-07-10 21:39:25 -04:00
|
|
|
|
2021-07-18 17:03:57 -04:00
|
|
|
# Log to Zookeeper
|
|
|
|
if self.config['zookeeper_logging']:
|
2021-07-19 11:45:58 -04:00
|
|
|
self.zookeeper_queue.put(message)
|
2021-07-18 17:03:57 -04:00
|
|
|
|
2019-07-10 21:39:25 -04:00
|
|
|
# Set last message variables
|
2018-10-14 02:01:35 -04:00
|
|
|
self.last_colour = colour
|
2019-07-10 21:39:25 -04:00
|
|
|
self.last_prompt = prompt
|
2021-07-18 17:03:57 -04:00
|
|
|
|
|
|
|
|
|
|
|
class ZookeeperLogger(Thread):
|
|
|
|
"""
|
|
|
|
Defines a threaded writer for Zookeeper locks. Threading prevents the blocking of other
|
|
|
|
daemon events while the records are written. They will be eventually-consistent
|
|
|
|
"""
|
2021-07-19 11:45:58 -04:00
|
|
|
def __init__(self, config, zookeeper_queue):
|
2021-07-18 17:03:57 -04:00
|
|
|
self.config = config
|
|
|
|
self.node = self.config['node']
|
|
|
|
self.max_lines = self.config['node_log_lines']
|
2021-07-19 11:45:58 -04:00
|
|
|
self.zookeeper_queue = zookeeper_queue
|
|
|
|
self.connected = False
|
2021-07-18 19:53:00 -04:00
|
|
|
self.running = False
|
2021-07-19 11:45:58 -04:00
|
|
|
self.zkhandler = None
|
2021-07-18 17:03:57 -04:00
|
|
|
Thread.__init__(self, args=(), kwargs=None)
|
|
|
|
|
|
|
|
def start_zkhandler(self):
|
|
|
|
# We must open our own dedicated Zookeeper instance because we can't guarantee one already exists when this starts
|
|
|
|
if self.zkhandler is not None:
|
|
|
|
try:
|
|
|
|
self.zkhandler.disconnect()
|
|
|
|
except Exception:
|
|
|
|
pass
|
2021-07-19 11:45:58 -04:00
|
|
|
|
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
self.zkhandler = ZKHandler(self.config, logger=None)
|
|
|
|
self.zkhandler.connect(persistent=True)
|
|
|
|
break
|
|
|
|
except Exception:
|
|
|
|
sleep(0.5)
|
|
|
|
continue
|
|
|
|
|
|
|
|
self.connected = True
|
|
|
|
|
|
|
|
# Ensure the root keys for this are instantiated
|
|
|
|
self.zkhandler.write([
|
|
|
|
('base.logs', ''),
|
|
|
|
(('logs', self.node), '')
|
|
|
|
])
|
2021-07-18 17:03:57 -04:00
|
|
|
|
|
|
|
def run(self):
|
2021-07-19 11:45:58 -04:00
|
|
|
while not self.connected:
|
|
|
|
self.start_zkhandler()
|
|
|
|
sleep(1)
|
|
|
|
|
2021-07-18 19:53:00 -04:00
|
|
|
self.running = True
|
|
|
|
# Get the logs that are currently in Zookeeper and populate our deque
|
2021-07-18 20:58:14 -04:00
|
|
|
raw_logs = self.zkhandler.read(('logs.messages', self.node))
|
|
|
|
if raw_logs is None:
|
|
|
|
raw_logs = ''
|
|
|
|
logs = deque(raw_logs.split('\n'), self.max_lines)
|
2021-07-18 19:53:00 -04:00
|
|
|
while self.running:
|
|
|
|
# Get a new message
|
|
|
|
try:
|
2021-07-19 11:45:58 -04:00
|
|
|
message = self.zookeeper_queue.get(timeout=1)
|
2021-07-18 19:53:00 -04:00
|
|
|
if not message:
|
|
|
|
continue
|
|
|
|
except Exception:
|
|
|
|
continue
|
2021-07-18 17:03:57 -04:00
|
|
|
|
|
|
|
if not self.config['log_dates']:
|
|
|
|
# We want to log dates here, even if the log_dates config is not set
|
2021-07-18 19:00:54 -04:00
|
|
|
date = '{} '.format(datetime.now().strftime('%Y/%m/%d %H:%M:%S.%f'))
|
2021-07-18 17:03:57 -04:00
|
|
|
else:
|
|
|
|
date = ''
|
2021-07-18 19:53:00 -04:00
|
|
|
# Add the message to the deque
|
2021-07-18 17:03:57 -04:00
|
|
|
logs.append(f'{date}{message}')
|
2021-07-18 19:53:00 -04:00
|
|
|
# Write the updated messages into Zookeeper
|
2021-07-18 17:03:57 -04:00
|
|
|
self.zkhandler.write([(('logs.messages', self.node), '\n'.join(logs))])
|
2021-07-18 19:53:00 -04:00
|
|
|
return
|
|
|
|
|
|
|
|
def stop(self):
|
|
|
|
self.running = False
|