Compare commits
7 Commits
a6d59c8c71
...
6fc1e073c0
Author | SHA1 | Date | |
---|---|---|---|
6fc1e073c0 | |||
56351ee700 | |||
f45c5b20c5 | |||
7a2b7add04 | |||
ae0c2ef4e9 | |||
d35c176aab | |||
9444405453 |
@ -1,14 +1,15 @@
|
||||
# Parameters file for Surveillance Bot
|
||||
datapath: '/data/SDS/' # SC3 Datapath
|
||||
networks: ['1Y', 'HA']
|
||||
stations: '*'
|
||||
locations: '*'
|
||||
channels: ['EX1', 'EX2', 'EX3', 'VEI'] # Specify SOH channels, currently supported EX[1-3] and VEI
|
||||
stations_blacklist: ['TEST', 'EREA']
|
||||
datapath: "/data/SDS/" # SC3 Datapath
|
||||
networks: ["1Y", "HA"]
|
||||
stations: "*"
|
||||
locations: "*"
|
||||
channels: ["EX1", "EX2", "EX3", "VEI"] # Specify SOH channels, currently supported EX[1-3] and VEI
|
||||
channel_names: ["Temperature (°)", "230V/12V Status (V)", "Router/Charger State (V)", "Logger Voltage (V)"] # names for plotting (optional)
|
||||
stations_blacklist: ["TEST", "EREA"]
|
||||
networks_blacklist: []
|
||||
interval: 60 # Perform checks every x seconds
|
||||
n_track: 120 # wait number of intervals after FAIL before performing an action (i.e. send mail)
|
||||
timespan: 3 # Check data of the recent x days
|
||||
n_track: 120 # wait n_track * intervals before performing an action (i.e. send mail/end highlight status)
|
||||
timespan: 7 # Check data of the recent x days
|
||||
verbosity: 0
|
||||
track_changes: True # tracks all changes since GUI startup by text highlighting (GUI only)
|
||||
warn_count: False # show number of warnings and errors in table
|
||||
@ -16,17 +17,11 @@ dt_thresh: [300, 1800] # threshold (s) for timing delay colourisation (yello
|
||||
html_figures: True # Create html figure directory and links
|
||||
reread_parameters: True # reread parameters file (change parameters on runtime, not for itself/GUI refresh/datapath)
|
||||
|
||||
# add links to html table with specified key as column and value as relative link, interpretable string parameters:
|
||||
# nw (e.g. 1Y), st (e.g. GR01A), nwst_id (e.g. 1Y.GR01A)
|
||||
# can also be empty!
|
||||
add_links:
|
||||
slmon: {"URL": "{nw}_{st}.html", "text": "show"} # for example: slmon: {"URL": "{nw}_{st}.html", "text": "link"}
|
||||
|
||||
POWBOX:
|
||||
pb_ok: 1 # Voltage for PowBox OK
|
||||
pb_SOH2: # PowBox channel 2 voltage translations
|
||||
-1: {"230V": "PBox under 1V", "12V": "PBox under 1V"}
|
||||
1: {"230V": 'OK', "12V": "OK"}
|
||||
1: {"230V": "OK", "12V": "OK"}
|
||||
2: {"230V": "OFF", "12V": "OK"}
|
||||
3: {"230V": "OK", "12V": "overvoltage"}
|
||||
4: {"230V": "OK", "12V": "undervoltage"}
|
||||
@ -49,13 +44,19 @@ THRESHOLDS:
|
||||
high_volt: 14.8 # max voltage for over voltage warning
|
||||
unclassified: 5 # min voltage samples not classified for warning
|
||||
|
||||
# E-mail notifications
|
||||
EMAIL:
|
||||
mailserver: 'localhost'
|
||||
addresses: ['marcel.paffrath@rub.de', 'kasper.fischer@rub.de'] # list of mail addresses for info mails
|
||||
sender: 'webmaster@geophysik.ruhr-uni-bochum.de' # mail sender
|
||||
# add links to html table with specified key as column and value as relative link, interpretable string parameters:
|
||||
# nw (e.g. 1Y), st (e.g. GR01A), nwst_id (e.g. 1Y.GR01A)
|
||||
# optional!
|
||||
add_links:
|
||||
slmon: {"URL": "{nw}_{st}.html", "text": "show"} # for example: slmon: {"URL": "path/{nw}_{st}.html", "text": "link"}
|
||||
|
||||
# Factor for channel to SI-units (for plotting)
|
||||
# E-mail notifications (optional)
|
||||
EMAIL:
|
||||
mailserver: "localhost"
|
||||
addresses: ["marcel.paffrath@rub.de", "kasper.fischer@rub.de"] # list of mail addresses for info mails
|
||||
sender: "webmaster@geophysik.ruhr-uni-bochum.de" # mail sender
|
||||
|
||||
# Factor for channel to SI-units (for plotting, optional)
|
||||
CHANNEL_UNITS:
|
||||
EX1: 1e-6
|
||||
EX2: 1e-6
|
||||
@ -63,7 +64,8 @@ CHANNEL_UNITS:
|
||||
VEI: 1e-3
|
||||
|
||||
# Transform channel for plotting, perform arithmetic operations in given order, e.g.: PBox EX1 V to deg C: 20 * x -20
|
||||
# optional!
|
||||
CHANNEL_TRANSFORM:
|
||||
EX1:
|
||||
- ['*', 20]
|
||||
- ['-', 20]
|
||||
- ["*", 20]
|
||||
- ["-", 20]
|
98
survBot.py
98
survBot.py
@ -19,7 +19,7 @@ from obspy.clients.filesystem.sds import Client
|
||||
|
||||
from write_utils import write_html_text, write_html_row, write_html_footer, write_html_header, get_print_title_str, \
|
||||
init_html_table, finish_html_table
|
||||
from utils import get_bg_color, modify_stream_for_plot
|
||||
from utils import get_bg_color, modify_stream_for_plot, annotate_trace_axes
|
||||
|
||||
try:
|
||||
import smtplib
|
||||
@ -337,15 +337,22 @@ class SurveillanceBot(object):
|
||||
fnout = self.get_fig_path_abs(nwst_id)
|
||||
st = self.data.get(nwst_id)
|
||||
if st:
|
||||
st = modify_stream_for_plot(st, parameters=self.parameters)
|
||||
st.plot(fig=fig, show=False, draw=False, block=False, equal_scale=False, method='full')
|
||||
ax = fig.axes[0]
|
||||
ax.set_title(f'Plot refreshed at (UTC) {UTCDateTime.now().strftime("%Y-%m-%d %H:%M:%S")}. '
|
||||
f'Refreshed hourly or on FAIL status.')
|
||||
fig.savefig(fnout, dpi=150., bbox_inches='tight')
|
||||
# TODO: this section might fail, adding try-except block for analysis and to prevent program from crashing
|
||||
try:
|
||||
st = modify_stream_for_plot(st, parameters=self.parameters)
|
||||
st.plot(fig=fig, show=False, draw=False, block=False, equal_scale=False, method='full')
|
||||
annotate_trace_axes(fig, self.parameters, self.verbosity)
|
||||
except Exception as e:
|
||||
print(f'Could not generate plot for {nwst_id}:')
|
||||
print(traceback.format_exc())
|
||||
if len(fig.axes) > 0:
|
||||
ax = fig.axes[0]
|
||||
ax.set_title(f'Plot refreshed at (UTC) {UTCDateTime.now().strftime("%Y-%m-%d %H:%M:%S")}. '
|
||||
f'Refreshed hourly or on FAIL status.')
|
||||
fig.savefig(fnout, dpi=150., bbox_inches='tight')
|
||||
plt.close(fig)
|
||||
|
||||
def write_html_table(self, default_color='#e6e6e6'):
|
||||
def write_html_table(self, default_color='#e6e6e6', default_header_color='#999'):
|
||||
self.check_html_dir()
|
||||
fnout = pjoin(self.outpath_html, 'survBot_out.html')
|
||||
if not fnout:
|
||||
@ -361,16 +368,18 @@ class SurveillanceBot(object):
|
||||
# add columns for additional links
|
||||
for key in self.add_links:
|
||||
header.insert(-1, key)
|
||||
header_items = [dict(text='Station', color=default_color)]
|
||||
header_items = [dict(text='Station', color=default_header_color)]
|
||||
for check_key in header:
|
||||
item = dict(text=check_key, color=default_color)
|
||||
item = dict(text=check_key, color=default_header_color)
|
||||
header_items.append(item)
|
||||
write_html_row(outfile, header_items, html_key='th')
|
||||
|
||||
# Write all cells
|
||||
for nwst_id in self.station_list:
|
||||
fig_name = self.get_fig_path_rel(nwst_id)
|
||||
col_items = [dict(text=nwst_id.rstrip('.'), color=default_color, hyperlink=fig_name)]
|
||||
nwst_id_str = nwst_id.rstrip('.')
|
||||
col_items = [dict(text=nwst_id_str, color=default_color, hyperlink=fig_name,
|
||||
bold=True, tooltip=f'Show plot of {nwst_id_str}')]
|
||||
for check_key in header:
|
||||
if check_key in self.keys:
|
||||
status_dict = self.analysis_results.get(nwst_id)
|
||||
@ -388,7 +397,8 @@ class SurveillanceBot(object):
|
||||
if not type(message) in [str]:
|
||||
message = str(message) + deg_str
|
||||
|
||||
item = dict(text=str(message), tooltip=str(detailed_message), color=bg_color)
|
||||
item = dict(text=str(message), tooltip=str(detailed_message), color=bg_color,
|
||||
blink=status.is_active)
|
||||
elif check_key in self.add_links:
|
||||
value = self.add_links.get(check_key).get('URL')
|
||||
link_text = self.add_links.get(check_key).get('text')
|
||||
@ -510,25 +520,30 @@ class StationQC(object):
|
||||
current_status.count += count
|
||||
else:
|
||||
current_status = new_error
|
||||
# refresh plot (using parent class) if error is new and not on program-startup
|
||||
if self.search_previous_errors(key, n_errors=1):
|
||||
# if error is new and not on program-startup set active and refresh plot (using parent class)
|
||||
if self.search_previous_errors(key, n_errors=1) is True:
|
||||
self.parent.write_html_figure(self.nwst_id)
|
||||
|
||||
self._update_status(key, current_status, detailed_message, last_occurrence)
|
||||
|
||||
if self.verbosity:
|
||||
self.print(f'{UTCDateTime()}: {detailed_message}', flush=False)
|
||||
|
||||
# do not send error mail if this is the first run (e.g. program startup) or state was already error (unchanged)
|
||||
if self.search_previous_errors(key):
|
||||
self.send_mail(key, detailed_message)
|
||||
if self.search_previous_errors(key) is True:
|
||||
self.send_mail(key, status_type='FAIL', additional_message=detailed_message)
|
||||
# set status to "inactive" after sending info mail
|
||||
current_status.is_active = False
|
||||
elif self.search_previous_errors(key) == 'active':
|
||||
current_status.is_active = True
|
||||
|
||||
self._update_status(key, current_status, detailed_message, last_occurrence)
|
||||
|
||||
def search_previous_errors(self, key, n_errors=None):
|
||||
"""
|
||||
Check n_track + 1 previous statuses for errors.
|
||||
If first item in list is no error but all others are return True (first time n_track errors appeared --
|
||||
if ALL n_track + 1 are error: error is old)
|
||||
In all other cases return True.
|
||||
If first item in list is no error but all others are return True
|
||||
(first time n_track errors appeared if ALL n_track + 1 are error: error is old)
|
||||
If last item is error but not all items are error yet return keyword 'active' -> error active, no message sent
|
||||
In all other cases return False.
|
||||
This also prevents sending status (e.g. mail) in case of program startup
|
||||
"""
|
||||
if n_errors is not None:
|
||||
@ -540,10 +555,12 @@ class StationQC(object):
|
||||
# if first entry was no error but all others are, return True (-> new Fail n_track times)
|
||||
if not previous_errors[0] and all(previous_errors[1:]):
|
||||
return True
|
||||
else:
|
||||
return False
|
||||
# in case previous_errors exists, last item is error but not all items are error, error still active
|
||||
elif previous_errors and previous_errors[-1] and not all(previous_errors):
|
||||
return 'active'
|
||||
return False
|
||||
|
||||
def send_mail(self, key, message):
|
||||
def send_mail(self, key, status_type, additional_message=''):
|
||||
""" Send info mail using parameters specified in parameters file """
|
||||
if not mail_functionality:
|
||||
if self.verbosity:
|
||||
@ -561,12 +578,10 @@ class StationQC(object):
|
||||
if self.verbosity:
|
||||
print('Mail sender or addresses not correctly defined. Return')
|
||||
return
|
||||
n_track = self.parameters.get('n_track')
|
||||
interval = self.parameters.get('interval')
|
||||
dt = timedelta(seconds=n_track * interval)
|
||||
text = f'{key} FAIL status longer than {dt}: ' + message
|
||||
dt = self.get_dt_for_action()
|
||||
text = f'{key}: Status {status_type} longer than {dt}: ' + additional_message
|
||||
msg = MIMEText(text)
|
||||
msg['Subject'] = f'new FAIL status on station {self.nwst_id}'
|
||||
msg['Subject'] = f'new message on station {self.nwst_id}'
|
||||
msg['From'] = sender
|
||||
msg['To'] = ', '.join(addresses)
|
||||
|
||||
@ -575,6 +590,12 @@ class StationQC(object):
|
||||
s.sendmail(sender, addresses, msg.as_string())
|
||||
s.quit()
|
||||
|
||||
def get_dt_for_action(self):
|
||||
n_track = self.parameters.get('n_track')
|
||||
interval = self.parameters.get('interval')
|
||||
dt = timedelta(seconds=n_track * interval)
|
||||
return dt
|
||||
|
||||
def status_other(self, detailed_message, status_message, last_occurrence=None, count=1):
|
||||
key = 'other'
|
||||
new_status = StatusOther(count=count, messages=[status_message])
|
||||
@ -593,13 +614,15 @@ class StationQC(object):
|
||||
|
||||
self.status_dict[key] = current_status
|
||||
|
||||
def activity_check(self):
|
||||
def activity_check(self, key='last_active'):
|
||||
self.last_active = self.last_activity()
|
||||
if not self.last_active:
|
||||
status = StatusError()
|
||||
else:
|
||||
message = timedelta(seconds=int(self.program_starttime - self.last_active))
|
||||
status = Status(message=message)
|
||||
dt_active = timedelta(seconds=int(self.program_starttime - self.last_active))
|
||||
status = Status(message=dt_active)
|
||||
self.check_for_inactive_message(key, dt_active)
|
||||
|
||||
self.status_dict['last active'] = status
|
||||
|
||||
def last_activity(self):
|
||||
@ -611,6 +634,12 @@ class StationQC(object):
|
||||
if len(endtimes) > 0:
|
||||
return max(endtimes)
|
||||
|
||||
def check_for_inactive_message(self, key, dt_active):
|
||||
dt_action = self.get_dt_for_action()
|
||||
interval = self.parameters.get('interval')
|
||||
if dt_action <= dt_active < dt_action + timedelta(seconds=interval):
|
||||
self.send_mail(key, status_type='Inactive')
|
||||
|
||||
def start(self):
|
||||
self.analyse_channels()
|
||||
|
||||
@ -889,7 +918,9 @@ class StationQC(object):
|
||||
|
||||
|
||||
class Status(object):
|
||||
def __init__(self, message='-', detailed_messages=None, count: int = 0, last_occurrence=None, show_count=True):
|
||||
def __init__(self, message=None, detailed_messages=None, count: int = 0, last_occurrence=None, show_count=True):
|
||||
if message is None:
|
||||
message = '-'
|
||||
if detailed_messages is None:
|
||||
detailed_messages = []
|
||||
self.show_count = show_count
|
||||
@ -901,6 +932,7 @@ class Status(object):
|
||||
self.is_warn = None
|
||||
self.is_error = None
|
||||
self.is_other = False
|
||||
self.is_active = False
|
||||
|
||||
def set_warn(self):
|
||||
self.is_warn = True
|
||||
|
@ -34,7 +34,7 @@ from obspy import UTCDateTime
|
||||
|
||||
from survBot import SurveillanceBot
|
||||
from write_utils import *
|
||||
from utils import get_bg_color, modify_stream_for_plot
|
||||
from utils import get_bg_color, modify_stream_for_plot, annotate_trace_axes
|
||||
|
||||
try:
|
||||
from rest_api.utils import get_station_iccid
|
||||
@ -316,6 +316,7 @@ class MainWindow(QtWidgets.QMainWindow):
|
||||
self.plot_widget.setWindowTitle(nwst_id)
|
||||
st = modify_stream_for_plot(st, parameters=self.parameters)
|
||||
st.plot(equal_scale=False, method='full', block=False, fig=self.plot_widget.canvas.fig)
|
||||
annotate_trace_axes(fig=self.plot_widget.canvas.fig, parameters=self.parameters)
|
||||
self.plot_widget.show()
|
||||
|
||||
def notification(self, text):
|
||||
|
20
utils.py
20
utils.py
@ -102,3 +102,23 @@ def transform_trace(data, transf):
|
||||
raise IOError(f'Unknown arithmethic operator string: {operator_str}')
|
||||
|
||||
return data
|
||||
|
||||
|
||||
def annotate_trace_axes(fig, parameters, verbosity=0):
|
||||
"""
|
||||
Adds channel names to y-axis if defined in parameters.
|
||||
Can get mixed up if channel order in stream and channel names defined in parameters.yaml differ, but it is
|
||||
difficult to assess the correct order from Obspy plotting routing.
|
||||
"""
|
||||
names = parameters.get('channel_names')
|
||||
if not names: # or not len(st.traces):
|
||||
return
|
||||
if not len(names) == len(fig.axes):
|
||||
if verbosity:
|
||||
print('Mismatch in axis and label lengths. Not adding plot labels')
|
||||
return
|
||||
for channel_name, ax in zip(names, fig.axes):
|
||||
if channel_name:
|
||||
ax.set_ylabel(channel_name)
|
||||
|
||||
|
||||
|
@ -1,12 +1,15 @@
|
||||
from datetime import timedelta
|
||||
|
||||
|
||||
def write_html_table_title(fobj, parameters):
|
||||
title = get_print_title_str(parameters)
|
||||
fobj.write(f'<h3>{title}</h3>\n')
|
||||
|
||||
|
||||
def write_html_text(fobj, text):
|
||||
fobj.write(f'<p>{text}</p>\n')
|
||||
|
||||
|
||||
def write_html_header(fobj, refresh_rate=10):
|
||||
header = ['<!DOCTYPE html>',
|
||||
'<html>',
|
||||
@ -24,35 +27,44 @@ def write_html_header(fobj, refresh_rate=10):
|
||||
for item in header:
|
||||
fobj.write(item + '\n')
|
||||
|
||||
|
||||
def init_html_table(fobj):
|
||||
fobj.write('<table style="width:100%">\n')
|
||||
|
||||
|
||||
def finish_html_table(fobj):
|
||||
fobj.write('</table>\n')
|
||||
|
||||
|
||||
def write_html_footer(fobj):
|
||||
footer = ['</body>',
|
||||
'</html>']
|
||||
for item in footer:
|
||||
fobj.write(item + '\n')
|
||||
|
||||
|
||||
def write_html_row(fobj, items, html_key='td'):
|
||||
default_space = ' '
|
||||
fobj.write(default_space + '<tr>\n')
|
||||
for item in items:
|
||||
text = item.get('text')
|
||||
if item.get('bold'):
|
||||
text = '<b>' + text + '</b>'
|
||||
if item.get('italic'):
|
||||
text = '<i>' + text + '</i>'
|
||||
tooltip = item.get('tooltip')
|
||||
color = item.get('color')
|
||||
# check for black background of headers (shouldnt happen anymore)
|
||||
color = '#e6e6e6' if color == '#000000' else color
|
||||
hyperlink = item.get('hyperlink')
|
||||
image_str = f'<a href="{hyperlink}">' if hyperlink else ''
|
||||
fobj.write(2 * default_space + f'<{html_key} bgcolor="{color}" title="{tooltip}"> {image_str}'
|
||||
blink_str = f' class="blink-bg"' if item.get('blink') else ''
|
||||
fobj.write(2 * default_space + f'<{html_key}{blink_str} bgcolor="{color}" title="{tooltip}"> {image_str}'
|
||||
+ text + f'</{html_key}>\n')
|
||||
fobj.write(default_space + '</tr>\n')
|
||||
|
||||
|
||||
def get_print_title_str(parameters):
|
||||
timespan = parameters.get('timespan') * 24 * 3600
|
||||
tdelta_str = str(timedelta(seconds=int(timespan))).replace(', 0:00:00', '')
|
||||
return f'Analysis table of router quality within the last {tdelta_str}'
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user