Compare commits

...

4 Commits

7 changed files with 221 additions and 160 deletions

View File

@ -1,16 +1,17 @@
# Parameters file for Surveillance Bot # Parameters file for Surveillance Bot
datapath: '/data/SDS/' # SC3 Datapath datapath: '/data/SDS/' # SC3 Datapath
outpath_html: '/home/marcel/tmp/survBot_out.html' # output of HTML table
networks: ['1Y', 'HA'] networks: ['1Y', 'HA']
stations: '*' stations: '*'
locations: '*' locations: '*'
channels: ['EX1', 'EX2', 'EX3', 'VEI'] # Specify SOH channels, currently supported EX[1-3] and VEI channels: ['EX1', 'EX2', 'EX3', 'VEI'] # Specify SOH channels, currently supported EX[1-3] and VEI
stations_blacklist: ['TEST', 'EREA'] stations_blacklist: ['TEST', 'EREA']
networks_blacklist: [] networks_blacklist: []
interval: 20 # Perform checks every x seconds interval: 60 # Perform checks every x seconds
timespan: 7 # Check data of the recent x days timespan: 7 # Check data of the recent x days
verbosity: 0 verbosity: 0
reread_parameters: True # reread parameters file (change parameters on runtime, not for itself/GUI refresh/datapath)
track_changes: True # tracks all changes since GUI startup by text highlighting (GUI only) track_changes: True # tracks all changes since GUI startup by text highlighting (GUI only)
dt_thresh: [300, 1800] # threshold (s) for timing delay colourisation (yellow/red)
POWBOX: POWBOX:
pb_ok: 1 # Voltage for PowBox OK pb_ok: 1 # Voltage for PowBox OK

12
submit_bot.sh Normal file → Executable file
View File

@ -2,15 +2,19 @@
ulimit -s 8192 ulimit -s 8192
#$ -l low #$ -l low
#$ -l os=*stretch #$ -l h_vmem=5G
#$ -cwd #$ -cwd
#$ -pe smp 1 #$ -pe smp 1
##$ -q "*@minos15" #$ -N survBot_bg
export PYTHONPATH="$PYTHONPATH:/home/marcel/git/"
export PYTHONPATH="$PYTHONPATH:/home/marcel/git/code_base/" export PYTHONPATH="$PYTHONPATH:/home/marcel/git/code_base/"
source /opt/anaconda3/etc/profile.d/conda.sh source /opt/anaconda3/etc/profile.d/conda.sh
conda activate py37 conda activate py37
python survBot.py # environment variables for numpy to prevent multi threading
export MKL_NUM_THREADS=1
export NUMEXPR_NUM_THREADS=1
export OMP_NUM_THREADS=1
python survBot.py -html '/home/marcel/public_html/survBot_out.html'

View File

@ -1,10 +1,13 @@
#! /usr/bin/env python #!/usr/bin/env python
# -*- coding: utf-8 -*-
__version__ = '0.1' __version__ = '0.1'
__author__ = 'Marcel Paffrath' __author__ = 'Marcel Paffrath'
import os import os
import traceback
import yaml import yaml
import argparse
import time import time
from datetime import timedelta from datetime import timedelta
@ -13,7 +16,9 @@ import numpy as np
from obspy import read, UTCDateTime, Stream from obspy import read, UTCDateTime, Stream
from obspy.clients.filesystem.sds import Client from obspy.clients.filesystem.sds import Client
from write_utils import get_print_title_str 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
pjoin = os.path.join pjoin = os.path.join
UP = "\x1B[{length}A" UP = "\x1B[{length}A"
@ -45,27 +50,35 @@ def fancy_timestr(dt, thresh=600, modif='+'):
class SurveillanceBot(object): class SurveillanceBot(object):
def __init__(self, parameter_path): def __init__(self, parameter_path, outpath_html=None):
self.keys = ['last active', '230V', '12V', 'router', 'charger', 'voltage', 'temp', 'other'] self.keys = ['last active', '230V', '12V', 'router', 'charger', 'voltage', 'temp', 'other']
self.parameters = read_yaml(parameter_path) self.parameter_path = parameter_path
self.transform_parameters() self.update_parameters()
self.starttime = UTCDateTime() self.starttime = UTCDateTime()
self.verbosity = self.parameters.get('verbosity') self.outpath_html = outpath_html
self.filenames = [] self.filenames = []
self.filenames_read = [] self.filenames_read = []
self.station_list = [] self.station_list = []
self.analysis_print_list = [] self.analysis_print_list = []
self.analysis_results = {} self.analysis_results = {}
self.stations_blacklist = self.parameters.get('stations_blacklist')
self.networks_blacklist = self.parameters.get('networks_blacklist')
self.dataStream = Stream() self.dataStream = Stream()
self.data = {} self.data = {}
self.print_count = 0 self.print_count = 0
self.refresh_period = 0 self.status_message = ''
self.cl = Client(self.parameters.get('datapath')) # TODO: Check if this has to be loaded again on update self.cl = Client(self.parameters.get('datapath')) # TODO: Check if this has to be loaded again on update
self.get_stations() self.get_stations()
def update_parameters(self):
self.parameters = read_yaml(self.parameter_path)
self.reread_parameters = self.parameters.get('reread_parameters')
self.dt_thresh = [int(val) for val in self.parameters.get('dt_thresh')]
self.verbosity = self.parameters.get('verbosity')
self.stations_blacklist = self.parameters.get('stations_blacklist')
self.networks_blacklist = self.parameters.get('networks_blacklist')
self.refresh_period = self.parameters.get('interval')
self.transform_parameters()
def transform_parameters(self): def transform_parameters(self):
for key in ['networks', 'stations', 'locations', 'channels']: for key in ['networks', 'stations', 'locations', 'channels']:
parameter = self.parameters.get(key) parameter = self.parameters.get(key)
@ -131,9 +144,11 @@ class SurveillanceBot(object):
self.data[st_id].append(trace) self.data[st_id].append(trace)
def execute_qc(self): def execute_qc(self):
self.starttime = UTCDateTime() if self.reread_parameters:
self.update_parameters()
self.get_filenames() self.get_filenames()
self.read_data() self.read_data()
qc_starttime = UTCDateTime()
self.analysis_print_list = [] self.analysis_print_list = []
self.analysis_results = {} self.analysis_results = {}
@ -141,7 +156,7 @@ class SurveillanceBot(object):
stream = self.data.get(st_id) stream = self.data.get(st_id)
if stream: if stream:
nsl = nsl_from_id(st_id) nsl = nsl_from_id(st_id)
station_qc = StationQC(stream, nsl, self.parameters, self.keys, self.starttime, self.verbosity, station_qc = StationQC(stream, nsl, self.parameters, self.keys, qc_starttime, self.verbosity,
print_func=self.print) print_func=self.print)
analysis_print_result = station_qc.return_print_analysis() analysis_print_result = station_qc.return_print_analysis()
station_dict, warn_dict = station_qc.return_analysis() station_dict, warn_dict = station_qc.return_analysis()
@ -150,6 +165,8 @@ class SurveillanceBot(object):
station_dict, warn_dict = self.get_no_data_station(st_id) station_dict, warn_dict = self.get_no_data_station(st_id)
self.analysis_print_list.append(analysis_print_result) self.analysis_print_list.append(analysis_print_result)
self.analysis_results[st_id] = (station_dict, warn_dict) self.analysis_results[st_id] = (station_dict, warn_dict)
self.update_status_message()
return 'ok' return 'ok'
def get_no_data_station(self, st_id, no_data='-', to_print=False): def get_no_data_station(self, st_id, no_data='-', to_print=False):
@ -186,10 +203,6 @@ class SurveillanceBot(object):
if len(times) > 0: if len(times) > 0:
return min(times) return min(times)
def print_analysis_html(self, filename):
with open(filename, 'w') as outfile:
pass
def print_analysis(self): def print_analysis(self):
self.print(200 * '+') self.print(200 * '+')
title_str = get_print_title_str(self.parameters) title_str = get_print_title_str(self.parameters)
@ -201,19 +214,22 @@ class SurveillanceBot(object):
for items in self.analysis_print_list: for items in self.analysis_print_list:
self.console_print(items) self.console_print(items)
def start(self, refresh_period=30): def start(self):
''' '''
Perform qc periodically. Perform qc periodically.
:param refresh_period: Update every x seconds :param refresh_period: Update every x seconds
:return: :return:
''' '''
self.refresh_period = refresh_period
status = 'ok' status = 'ok'
while status == 'ok' and self.refresh_period > 0: while status == 'ok' and self.refresh_period > 0:
status = self.execute_qc() status = self.execute_qc()
self.print_analysis() if self.outpath_html:
self.write_html_table()
else:
self.print_analysis()
time.sleep(self.refresh_period) time.sleep(self.refresh_period)
self.clear_prints() if not self.outpath_html:
self.clear_prints()
def console_print(self, itemlist, str_len=21, sep='|', seplen=3): def console_print(self, itemlist, str_len=21, sep='|', seplen=3):
assert len(sep) <= seplen, f'Make sure seperator has less than {seplen} characters' assert len(sep) <= seplen, f'Make sure seperator has less than {seplen} characters'
@ -224,6 +240,60 @@ class SurveillanceBot(object):
string += item.center(str_len) + sr string += item.center(str_len) + sr
self.print(string, flush=False) self.print(string, flush=False)
def write_html_table(self, default_color='#e6e6e6'):
fnout = self.outpath_html
if not fnout:
return
try:
with open(fnout, 'w') as outfile:
write_html_header(outfile, self.refresh_period)
#write_html_table_title(outfile, self.parameters)
init_html_table(outfile)
# First write header items
header_items = [dict(text='Station', color=default_color)]
for check_key in self.keys:
item = dict(text=check_key, color=default_color)
header_items.append(item)
write_html_row(outfile, header_items, html_key='th')
# Write all cells
for st_id in self.station_list:
col_items = [dict(text=st_id.rstrip('.'), color=default_color)]
for check_key in self.keys:
status_dict, detailed_dict = self.analysis_results.get(st_id)
status = status_dict.get(check_key)
# get background color
dt_thresh = [timedelta(seconds=sec) for sec in self.dt_thresh]
bg_color = get_bg_color(check_key, status, dt_thresh, hex=True)
if not bg_color:
bg_color = default_color
# add degree sign for temp
if check_key == 'temp':
if not type(status) in [str]:
status = str(status) + deg_str
item = dict(text=str(status), tooltip=str(detailed_dict.get(check_key)),
color=bg_color)
col_items.append(item)
write_html_row(outfile, col_items)
finish_html_table(outfile)
write_html_text(outfile, self.status_message)
write_html_footer(outfile)
except Exception as e:
print(f'Could not write HTML table to {fnout}:')
print(traceback.format_exc())
def update_status_message(self):
timespan = timedelta(seconds=int(self.parameters.get('timespan') * 24 * 3600))
self.status_message = f'Program starttime (UTC) {self.starttime.strftime("%Y-%m-%d %H:%M:%S")} | ' \
f'Current time (UTC) {UTCDateTime().strftime("%Y-%m-%d %H:%M:%S")} | ' \
f'Refresh period: {self.refresh_period}s | '\
f'Showing data of last {timespan}'
def print(self, string, **kwargs): def print(self, string, **kwargs):
clear_end = CLR + '\n' clear_end = CLR + '\n'
n_nl = string.count('\n') n_nl = string.count('\n')
@ -270,12 +340,20 @@ class StationQC(object):
if message: if message:
self.detailed_status_dict[key] = message self.detailed_status_dict[key] = message
def warn(self, key, message, status_message='WARN'): def warn(self, key, detailed_message, status_message='WARN'):
self.detailed_status_dict[key] = message # update detailed status if already existing
self.status_dict[key] = status_message current_message = self.detailed_status_dict.get(key)
current_message = '' if current_message in [None, '-'] else current_message + ' | '
self.detailed_status_dict[key] = current_message + detailed_message
# this is becoming a little bit too complicated (adding warnings to existing)
current_status_message = self.status_dict.get(key)
current_status_message = '' if current_status_message in [None, 'OK', '-'] else current_status_message + ' | '
self.status_dict[key] = current_status_message + status_message
# change this to something more useful, SMS/EMAIL/PUSH # change this to something more useful, SMS/EMAIL/PUSH
if self.verbosity: if self.verbosity:
self.print(f'{UTCDateTime()}: {message}', flush=False) self.print(f'{UTCDateTime()}: {detailed_message}', flush=False)
# warnings.warn(message) # warnings.warn(message)
def error(self, key, message): def error(self, key, message):
@ -357,18 +435,21 @@ class StationQC(object):
self.status_ok(key, message=f'U={(voltage[-1])}V') self.status_ok(key, message=f'U={(voltage[-1])}V')
return return
# try calculate number of voltage peaks from gaps between indices n_overvolt = 0
n_overvolt = len(np.where(np.diff(overvolt) > 1)[0]) + 1 n_undervolt = 0
n_undervolt = len(np.where(np.diff(undervolt) > 1)[0]) + 1
warn_message = f'Trace {trace.get_id()}:' warn_message = f'Trace {trace.get_id()}:'
if len(overvolt) > 0: if len(overvolt) > 0:
# try calculate number of voltage peaks from gaps between indices
n_overvolt = len(np.where(np.diff(overvolt) > 1)[0]) + 1
warn_message += f' {n_overvolt}x Voltage over {high_volt}V' \ warn_message += f' {n_overvolt}x Voltage over {high_volt}V' \
+ self.get_last_occurrence_timestring(trace, overvolt) + self.get_last_occurrence_timestring(trace, overvolt)
if len(undervolt) > 0: if len(undervolt) > 0:
# try calculate number of voltage peaks from gaps between indices
n_undervolt = len(np.where(np.diff(undervolt) > 1)[0]) + 1
warn_message += f' {n_undervolt}x Voltage under {low_volt}V ' \ warn_message += f' {n_undervolt}x Voltage under {low_volt}V ' \
+ self.get_last_occurrence_timestring(trace, undervolt) + self.get_last_occurrence_timestring(trace, undervolt)
self.warn(key, message=warn_message, status_message='WARN ({})'.format(n_overvolt + n_undervolt)) self.warn(key, detailed_message=warn_message, status_message='WARN ({})'.format(n_overvolt + n_undervolt))
def pb_temp_analysis(self, channel='EX1'): def pb_temp_analysis(self, channel='EX1'):
""" Analyse PowBox temperature output. """ """ Analyse PowBox temperature output. """
@ -398,9 +479,9 @@ class StationQC(object):
if len(t_check) > 0: if len(t_check) > 0:
self.warn(key=key, self.warn(key=key,
status_message=cur_temp, status_message=cur_temp,
message=f'Trace {trace.get_id()}: ' detailed_message=f'Trace {trace.get_id()}: '
f'Temperature over {max_temp}\N{DEGREE SIGN} at {trace.get_id()}!' f'Temperature over {max_temp}\N{DEGREE SIGN} at {trace.get_id()}!'
+ self.get_last_occurrence_timestring(trace, t_check)) + self.get_last_occurrence_timestring(trace, t_check))
else: else:
self.status_ok(key, self.status_ok(key,
status_message=cur_temp, status_message=cur_temp,
@ -416,7 +497,8 @@ class StationQC(object):
if self.verbosity > 1: if self.verbosity > 1:
self.print(40 * '-') self.print(40 * '-')
self.print('Performing PowBox 12V/230V check (EX2)', flush=False) self.print('Performing PowBox 12V/230V check (EX2)', flush=False)
voltage_check, voltage_dict, last_val = self.pb_voltage_ok(trace, voltage, pb_dict_key, warn_keys=keys) voltage_check, voltage_dict, last_val = self.pb_voltage_ok(trace, voltage, pb_dict_key, channel=channel,
warn_keys=keys)
if voltage_check: if voltage_check:
for key in keys: for key in keys:
self.status_ok(key) self.status_ok(key)
@ -437,7 +519,8 @@ class StationQC(object):
if self.verbosity > 1: if self.verbosity > 1:
self.print(40 * '-') self.print(40 * '-')
self.print('Performing PowBox Router/Charger check (EX3)', flush=False) self.print('Performing PowBox Router/Charger check (EX3)', flush=False)
voltage_check, voltage_dict, last_val = self.pb_voltage_ok(trace, voltage, pb_dict_key, warn_keys=keys) voltage_check, voltage_dict, last_val = self.pb_voltage_ok(trace, voltage, pb_dict_key, channel=channel,
warn_keys=keys)
if voltage_check: if voltage_check:
for key in keys: for key in keys:
self.status_ok(key) self.status_ok(key)
@ -459,9 +542,9 @@ class StationQC(object):
# try calculate number of voltage peaks from gaps between indices # try calculate number of voltage peaks from gaps between indices
n_occurrences = len(np.where(np.diff(ind_array) > 1)[0]) + 1 n_occurrences = len(np.where(np.diff(ind_array) > 1)[0]) + 1
self.warn(key=key, self.warn(key=key,
message=f'Trace {trace.get_id()}: ' detailed_message=f'Trace {trace.get_id()}: '
f'Found {n_occurrences} occurrence(s) of {volt_lvl}V: {key}: {message}' f'Found {n_occurrences} occurrence(s) of {volt_lvl}V: {key}: {message}'
+ self.get_last_occurrence_timestring(trace, ind_array), + self.get_last_occurrence_timestring(trace, ind_array),
status_message='WARN ({})'.format(n_occurrences)) status_message='WARN ({})'.format(n_occurrences))
if last_val != 1: if last_val != 1:
self.error(key, message=f'Last PowBox voltage state {last_val}V: {message}') self.error(key, message=f'Last PowBox voltage state {last_val}V: {message}')
@ -482,7 +565,7 @@ class StationQC(object):
return return
return trace return trace
def pb_voltage_ok(self, trace, voltage, pb_dict_key, warn_keys): def pb_voltage_ok(self, trace, voltage, pb_dict_key, warn_keys, channel=None):
""" """
Checks if voltage level is ok everywhere and returns True. If it is not okay it returns a dictionary Checks if voltage level is ok everywhere and returns True. If it is not okay it returns a dictionary
with each voltage value associated to the different steps specified in POWBOX > pb_steps. Also raises with each voltage value associated to the different steps specified in POWBOX > pb_steps. Also raises
@ -508,10 +591,10 @@ class StationQC(object):
n_occurrences = len(np.where(np.diff(under) > 1)[0]) + 1 n_occurrences = len(np.where(np.diff(under) > 1)[0]) + 1
for key in warn_keys: for key in warn_keys:
self.warn(key=key, self.warn(key=key,
message=f'Trace {trace.get_id()}: ' detailed_message=f'Trace {trace.get_id()}: '
f'Voltage below {pb_ok}V {len(under)} times. ' f'Voltage below {pb_ok}V in {len(under)} samples, {n_occurrences} time(s). '
f'Mean voltage: {np.mean(voltage)}' f'Mean voltage: {np.mean(voltage):.2}'
+ self.get_last_occurrence_timestring(trace, under), + self.get_last_occurrence_timestring(trace, under),
status_message='WARN ({})'.format(n_occurrences)) status_message='WARN ({})'.format(n_occurrences))
# Get voltage levels for classification # Get voltage levels for classification
@ -539,10 +622,10 @@ class StationQC(object):
n_unclassified = len(unclassified_indices) n_unclassified = len(unclassified_indices)
max_uncl = self.parameters.get('THRESHOLDS').get('unclassified') max_uncl = self.parameters.get('THRESHOLDS').get('unclassified')
if max_uncl and n_unclassified > max_uncl: if max_uncl and n_unclassified > max_uncl:
self.warn(key='other', message=f'Trace {trace.get_id()}: ' self.warn(key='other', detailed_message=f'Trace {trace.get_id()}: '
f'{n_unclassified}/{len(all_indices)} ' f'{n_unclassified}/{len(all_indices)} '
f'unclassified voltage values in channel {trace.get_id()}', f'unclassified voltage values in channel {trace.get_id()}',
status_message=f'{n_unclassified} UNCLASSIFIED') status_message=f'{channel}: {n_unclassified} uncl.')
return False, voltage_dict, last_val return False, voltage_dict, last_val
@ -552,5 +635,9 @@ class StationQC(object):
if __name__ == '__main__': if __name__ == '__main__':
survBot = SurveillanceBot(parameter_path='parameters.yaml') parser = argparse.ArgumentParser(description='Call survBot')
survBot.start(refresh_period=30) parser.add_argument('-html', dest='html_filename', default=None, help='filename for HTML output')
args = parser.parse_args()
survBot = SurveillanceBot(parameter_path='parameters.yaml', outpath_html=args.html_filename)
survBot.start()

View File

@ -1,4 +1,5 @@
#! /usr/bin/env python #!/usr/bin/env python
# -*- coding: utf-8 -*-
""" """
GUI overlay for the main survBot to show quality control of different stations specified in parameters.yaml file. GUI overlay for the main survBot to show quality control of different stations specified in parameters.yaml file.
""" """
@ -9,7 +10,6 @@ __author__ = 'Marcel Paffrath'
import os import os
import sys import sys
import traceback import traceback
import argparse
try: try:
from PySide2 import QtGui, QtCore, QtWidgets from PySide2 import QtGui, QtCore, QtWidgets
@ -35,6 +35,7 @@ from obspy import UTCDateTime
from survBot import SurveillanceBot from survBot import SurveillanceBot
from write_utils import * from write_utils import *
from utils import get_bg_color
try: try:
from rest_api.utils import get_station_iccid from rest_api.utils import get_station_iccid
@ -77,24 +78,14 @@ class Thread(QtCore.QThread):
class MainWindow(QtWidgets.QMainWindow): class MainWindow(QtWidgets.QMainWindow):
def __init__(self, parameters='parameters.yaml', dt_thresh=(300, 1800)): def __init__(self, parameters='parameters.yaml'):
""" """
Main window of survBot GUI. Main window of survBot GUI.
:param parameters: Parameters dictionary file (yaml format) :param parameters: Parameters dictionary file (yaml format)
:param dt_thresh: threshold for timing delay colourisation (yellow/red)
""" """
super(MainWindow, self).__init__() super(MainWindow, self).__init__()
# some GUI default colors
self.colors_dict = {'FAIL': (255, 50, 0, 255),
'NO DATA': (255, 255, 125, 255),
'WARN': (255, 255, 125, 255),
'WARNX': lambda x: (min([255, 200 + x**2]), 255, 125, 255),
'OK': (125, 255, 125, 255),
'undefined': (230, 230, 230, 255)}
# init some attributes # init some attributes
self.dt_thresh = dt_thresh
self.last_mouse_loc = None self.last_mouse_loc = None
self.status_message = '' self.status_message = ''
self.starttime = UTCDateTime() self.starttime = UTCDateTime()
@ -109,6 +100,7 @@ class MainWindow(QtWidgets.QMainWindow):
self.survBot = SurveillanceBot(parameter_path=parameters) self.survBot = SurveillanceBot(parameter_path=parameters)
self.parameters = self.survBot.parameters self.parameters = self.survBot.parameters
self.refresh_period = self.parameters.get('interval') self.refresh_period = self.parameters.get('interval')
self.dt_thresh = [int(val) for val in self.parameters.get('dt_thresh')]
# create thread that is used to update # create thread that is used to update
self.thread = Thread(parent=self, runnable=self.survBot.execute_qc) self.thread = Thread(parent=self, runnable=self.survBot.execute_qc)
@ -183,41 +175,6 @@ class MainWindow(QtWidgets.QMainWindow):
self.last_mouse_loc = event.pos() self.last_mouse_loc = event.pos()
return super(QtWidgets.QMainWindow, self).eventFilter(object, event) return super(QtWidgets.QMainWindow, self).eventFilter(object, event)
def write_html_table(self):
fnout = self.parameters.get('outpath_html')
if not fnout:
return
try:
with open(fnout, 'w') as outfile:
write_html_header(outfile)
#write_html_table_title(outfile, self.parameters)
init_html_table(outfile)
nrows = self.table.rowCount()
ncolumns = self.table.columnCount()
# add header item 0 fix default black bg color for headers
station_header = QtWidgets.QTableWidgetItem(text='Station')
station_header.setText('Station')
header_items = [station_header]
for column in range(ncolumns):
hheader = self.table.horizontalHeaderItem(column)
header_items.append(hheader)
write_html_row(outfile, header_items, html_key='th')
for row in range(nrows):
vheader = self.table.verticalHeaderItem(row)
col_items = [vheader]
for column in range(ncolumns):
col_items.append(self.table.item(row, column))
write_html_row(outfile, col_items)
finish_html_table(outfile)
write_html_text(outfile, self.status_message)
write_html_footer(outfile)
except Exception as e:
print(f'Could not write HTML table to {fnout}:')
print(e)
def sms_context_menu(self, row_ind): def sms_context_menu(self, row_ind):
""" Open a context menu when left-clicking vertical header item """ """ Open a context menu when left-clicking vertical header item """
header_item = self.table.verticalHeaderItem(row_ind) header_item = self.table.verticalHeaderItem(row_ind)
@ -262,36 +219,27 @@ class MainWindow(QtWidgets.QMainWindow):
def fill_status_bar(self): def fill_status_bar(self):
""" Set status bar text """ """ Set status bar text """
timespan = timedelta(seconds=int(self.parameters.get('timespan') * 24 * 3600)) self.status_message = self.survBot.status_message
self.status_message = f'Program starttime (UTC) {self.starttime.strftime("%Y-%m-%d %H:%M:%S")} | ' \
f'Current time (UTC) {UTCDateTime().strftime("%Y-%m-%d %H:%M:%S")} | ' \
f'Refresh period: {self.refresh_period}s | '\
f'Showing data of last {timespan}'
status_bar = self.statusBar() status_bar = self.statusBar()
status_bar.showMessage(self.status_message) status_bar.showMessage(self.status_message)
def fill_table(self): def fill_table(self):
""" Fills the table with most recent information. Executed after execute_qc thread is done or on refresh. """ """ Fills the table with most recent information. Executed after execute_qc thread is done or on refresh. """
# fill status bar first with new time
self.fill_status_bar()
for col_ind, check_key in enumerate(self.survBot.keys): for col_ind, check_key in enumerate(self.survBot.keys):
for row_ind, st_id in enumerate(self.survBot.station_list): for row_ind, st_id in enumerate(self.survBot.station_list):
status_dict, detailed_dict = self.survBot.analysis_results.get(st_id) status_dict, detailed_dict = self.survBot.analysis_results.get(st_id)
status = status_dict.get(check_key) status = status_dict.get(check_key)
detailed_message = detailed_dict.get(check_key) detailed_message = detailed_dict.get(check_key)
if check_key == 'last active':
bg_color = self.get_time_delay_color(status) dt_thresh = [timedelta(seconds=sec) for sec in self.dt_thresh]
elif check_key == 'temp': bg_color = get_bg_color(check_key, status, dt_thresh)
bg_color = self.get_temp_color(status) if check_key == 'temp':
if not type(status) in [str]: if not type(status) in [str]:
status = str(status) + deg_str status = str(status) + deg_str
else:
statussplit = status.split(' ')
if len(statussplit) > 1 and statussplit[0] == 'WARN':
x = int(status.split(' ')[-1].lstrip('(').rstrip(')'))
bg_color = self.colors_dict.get('WARNX')(x)
else:
bg_color = self.colors_dict.get(status)
if not bg_color:
bg_color = self.colors_dict.get('undefined')
# Continue if nothing changed # Continue if nothing changed
text = str(status) text = str(status)
@ -330,26 +278,6 @@ class MainWindow(QtWidgets.QMainWindow):
# table filling/refreshing done, set clear_on_refresh to False # table filling/refreshing done, set clear_on_refresh to False
self.clear_on_refresh = False self.clear_on_refresh = False
# write html output if parameter is set
self.write_html_table()
def get_time_delay_color(self, dt):
""" Set color of time delay after thresholds specified in self.dt_thresh """
dt_thresh = [timedelta(seconds=sec) for sec in self.dt_thresh]
if dt < dt_thresh[0]:
return self.colors_dict.get('OK')
elif dt_thresh[0] <= dt < dt_thresh[1]:
return self.colors_dict.get('WARN')
return self.colors_dict.get('FAIL')
def get_temp_color(self, temp, vmin=-10, vmax=60, cmap='coolwarm'):
""" Get an rgba temperature value back from specified cmap, linearly interpolated between vmin and vmax. """
if type(temp) in [str]:
return self.colors_dict.get('undefined')
cmap = matplotlib.cm.get_cmap(cmap)
val = (temp - vmin) / (vmax - vmin)
rgba = [int(255 * c) for c in cmap(val)]
return rgba
def set_font_bold(self, item): def set_font_bold(self, item):
""" Set item font bold """ """ Set item font bold """

View File

@ -1,14 +0,0 @@
#!/bin/bash
ulimit -s 8192
#$ -l os=*stretch
##$ -cwd
#$ -pe smp 1
##$ -q "*@minos15"
export PYTHONPATH="$PYTHONPATH:/home/marcel/git/"
source /opt/anaconda3/etc/profile.d/conda.sh
conda activate py37
python /home/marcel/git/survBot/survBotGUI.py

51
utils.py Normal file
View File

@ -0,0 +1,51 @@
#!/usr/bin/env python
# -*- coding: utf-8 -*-
import matplotlib
def get_bg_color(check_key, status, dt_thresh=None, hex=False):
if check_key == 'last active':
bg_color = get_time_delay_color(status, dt_thresh)
elif check_key == 'temp':
bg_color = get_temp_color(status)
else:
statussplit = status.split(' ')
if len(statussplit) > 1 and statussplit[0] == 'WARN':
x = int(status.split(' ')[-1].lstrip('(').rstrip(')'))
bg_color = get_color('WARNX')(x)
else:
bg_color = get_color(status)
if not bg_color:
bg_color = get_color('undefined')
if hex:
bg_color = '#{:02x}{:02x}{:02x}'.format(*bg_color[:3])
return bg_color
def get_color(key):
# some GUI default colors
colors_dict = {'FAIL': (255, 50, 0, 255),
'NO DATA': (255, 255, 125, 255),
'WARN': (255, 255, 125, 255),
'WARNX': lambda x: (min([255, 200 + x ** 2]), 255, 125, 255),
'OK': (125, 255, 125, 255),
'undefined': (230, 230, 230, 255)}
return colors_dict.get(key)
def get_time_delay_color(dt, dt_thresh):
""" Set color of time delay after thresholds specified in self.dt_thresh """
if dt < dt_thresh[0]:
return get_color('OK')
elif dt_thresh[0] <= dt < dt_thresh[1]:
return get_color('WARN')
return get_color('FAIL')
def get_temp_color(temp, vmin=-10, vmax=60, cmap='coolwarm'):
""" Get an rgba temperature value back from specified cmap, linearly interpolated between vmin and vmax. """
if type(temp) in [str]:
return get_color('undefined')
cmap = matplotlib.cm.get_cmap(cmap)
val = (temp - vmin) / (vmax - vmin)
rgba = [int(255 * c) for c in cmap(val)]
return rgba

View File

@ -7,15 +7,17 @@ def write_html_table_title(fobj, parameters):
def write_html_text(fobj, text): def write_html_text(fobj, text):
fobj.write(f'<p>{text}</p>\n') fobj.write(f'<p>{text}</p>\n')
def write_html_header(fobj): def write_html_header(fobj, refresh_rate=10):
header = ['<!DOCTYPE html>', header = ['<!DOCTYPE html>',
'<html>', '<html>',
'<style>', f'<meta http-equiv="refresh" content="{refresh_rate}" >',
'table, th, td {', '<meta charset="utf-8">',
'border:1px solid black;',
'}',
'</style>',
'<body>'] '<body>']
# style = ['<style>',
# 'table, th, td {',
# 'border:1px solid black;',
# '}',
# '</style>',]
for item in header: for item in header:
fobj.write(item + '\n') fobj.write(item + '\n')
@ -32,14 +34,16 @@ def write_html_footer(fobj):
fobj.write(item + '\n') fobj.write(item + '\n')
def write_html_row(fobj, items, html_key='td'): def write_html_row(fobj, items, html_key='td'):
fobj.write('<tr>\n') default_space = ' '
fobj.write(default_space + '<tr>\n')
for item in items: for item in items:
text = item.text() text = item.get('text')
color = item.backgroundColor().name() tooltip = item.get('tooltip')
# fix for black background of headers color = item.get('color')
# check for black background of headers (shouldnt happen anymore)
color = '#e6e6e6' if color == '#000000' else color color = '#e6e6e6' if color == '#000000' else color
fobj.write(f'<{html_key} bgcolor="{color}">' + text + f'</{html_key}>\n') fobj.write(2 * default_space + f'<{html_key} bgcolor="{color}" title="{tooltip}">' + text + f'</{html_key}>\n')
fobj.write('</tr>\n') fobj.write(default_space + '</tr>\n')
def get_print_title_str(parameters): def get_print_title_str(parameters):
timespan = parameters.get('timespan') * 24 * 3600 timespan = parameters.get('timespan') * 24 * 3600