Compare commits
	
		
			10 Commits
		
	
	
		
			908535fcc8
			...
			477727018f
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 477727018f | |||
| 9d9ced7c83 | |||
| 305ab25ab2 | |||
| b875e63f83 | |||
| 20635e3433 | |||
| a6c792b271 | |||
| 5632bab07b | |||
| d7cbbe6876 | |||
| 3a384fd7b5 | |||
| c736048811 | 
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -211,3 +211,4 @@ flycheck_*.el | |||||||
| /network-security.data | /network-security.data | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | /__simulate_fail.json | ||||||
|  | |||||||
							
								
								
									
										9
									
								
								mailing_list.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								mailing_list.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | # specify mail addresses and station network ids for which information mails shall be sent, e.g.: | ||||||
|  | # "mail.address@provider.com, mail.address2@provider2.com": | ||||||
|  | #   - 1Y.GR01 | ||||||
|  | #   - 1Y.GR02 | ||||||
|  | # "mail.address3@provder.com": | ||||||
|  | #   - 1Y.GR03 | ||||||
|  | 
 | ||||||
|  | #"kasper.fischer@rub.de": | ||||||
|  | #    - 1Y.GR01 | ||||||
| @ -7,7 +7,7 @@ stations_blacklist: ["TEST", "EREA"]      # exclude these stations | |||||||
| networks_blacklist: []                    # exclude these networks | networks_blacklist: []                    # exclude these networks | ||||||
| interval: 60               # Perform checks every x seconds | interval: 60               # Perform checks every x seconds | ||||||
| n_track: 300               # wait n_track * intervals before performing an action (i.e. send mail/end highlight status) | n_track: 300               # wait n_track * intervals before performing an action (i.e. send mail/end highlight status) | ||||||
| timespan: 7                # Check data of the recent x days | timespan: 3                # Check data of the recent x days | ||||||
| verbosity: 0               # verbosity flag | verbosity: 0               # verbosity flag | ||||||
| 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) | ||||||
| warn_count: False           # show number of warnings and errors in table | warn_count: False           # show number of warnings and errors in table | ||||||
| @ -129,3 +129,5 @@ EMAIL: | |||||||
|   sender: "webmaster@geophysik.ruhr-uni-bochum.de" # mail sender |   sender: "webmaster@geophysik.ruhr-uni-bochum.de" # mail sender | ||||||
|   stations_blacklist: ['GR33']                     # do not send emails for specific stations |   stations_blacklist: ['GR33']                     # do not send emails for specific stations | ||||||
|   networks_blacklist: []                           # do not send emails for specific network |   networks_blacklist: []                           # do not send emails for specific network | ||||||
|  |   # specify recipients for single stations in a yaml: key = email-address, val = station list (e.g. [1Y.GR01, 1Y.GR02]) | ||||||
|  |   external_mail_list: "mailing_list.yaml" | ||||||
|  | |||||||
							
								
								
									
										247
									
								
								survBot.py
									
									
									
									
									
								
							
							
						
						
									
										247
									
								
								survBot.py
									
									
									
									
									
								
							| @ -5,9 +5,12 @@ __version__ = '0.1' | |||||||
| __author__ = 'Marcel Paffrath' | __author__ = 'Marcel Paffrath' | ||||||
| 
 | 
 | ||||||
| import os | import os | ||||||
|  | import io | ||||||
|  | import copy | ||||||
| import traceback | import traceback | ||||||
| import yaml | import yaml | ||||||
| import argparse | import argparse | ||||||
|  | import json | ||||||
| 
 | 
 | ||||||
| import time | import time | ||||||
| from datetime import timedelta | from datetime import timedelta | ||||||
| @ -17,13 +20,14 @@ import matplotlib.pyplot as plt | |||||||
| 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 write_html_text, write_html_row, write_html_footer, write_html_header, get_print_title_str, \ | from write_utils import get_html_text, get_html_row, html_footer, get_html_header, get_print_title_str, \ | ||||||
|     init_html_table, finish_html_table |     init_html_table, finish_html_table, get_mail_html_header, add_html_image | ||||||
| from utils import get_bg_color, modify_stream_for_plot, set_axis_yticks, set_axis_color, plot_axis_thresholds | from utils import get_bg_color, modify_stream_for_plot, set_axis_yticks, set_axis_color, plot_axis_thresholds | ||||||
| 
 | 
 | ||||||
| try: | try: | ||||||
|     import smtplib |     import smtplib | ||||||
|     from email.mime.text import MIMEText |     from email.message import EmailMessage | ||||||
|  |     from email.utils import make_msgid | ||||||
| 
 | 
 | ||||||
|     mail_functionality = True |     mail_functionality = True | ||||||
| except ImportError: | except ImportError: | ||||||
| @ -49,17 +53,23 @@ def read_yaml(file_path, n_read=3): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def nsl_from_id(nwst_id): | def nsl_from_id(nwst_id): | ||||||
|  |     nwst_id = get_full_seed_id(nwst_id) | ||||||
|     network, station, location = nwst_id.split('.') |     network, station, location = nwst_id.split('.') | ||||||
|     return dict(network=network, station=station, location=location) |     return dict(network=network, station=station, location=location) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def get_full_seed_id(nwst_id): | ||||||
|  |     seed_id = '{}.{}.{}'.format(*nwst_id.split('.'), '') | ||||||
|  |     return seed_id | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| def get_nwst_id(trace): | def get_nwst_id(trace): | ||||||
|     stats = trace.stats |     stats = trace.stats | ||||||
|     return f'{stats.network}.{stats.station}.'  # {stats.location}' |     return f'{stats.network}.{stats.station}.'  # {stats.location}' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def fancy_timestr(dt, thresh=600, modif='+'): | def fancy_timestr(dt, thresh=600, modif='+'): | ||||||
|     if dt > timedelta(seconds=thresh): |     if isinstance(dt, timedelta) and dt > timedelta(seconds=thresh): | ||||||
|         value = f'{modif} ' + str(dt) + f' {modif}' |         value = f'{modif} ' + str(dt) + f' {modif}' | ||||||
|     else: |     else: | ||||||
|         value = str(dt) |         value = str(dt) | ||||||
| @ -78,7 +88,7 @@ class SurveillanceBot(object): | |||||||
|         self.outpath_html = outpath_html |         self.outpath_html = outpath_html | ||||||
|         self.filenames = [] |         self.filenames = [] | ||||||
|         self.filenames_wf_data = [] |         self.filenames_wf_data = [] | ||||||
|         self.filenames_read = [] |         self.filenames_read_last_modif = {} | ||||||
|         self.station_list = [] |         self.station_list = [] | ||||||
|         self.analysis_print_list = [] |         self.analysis_print_list = [] | ||||||
|         self.analysis_results = {} |         self.analysis_results = {} | ||||||
| @ -90,6 +100,8 @@ class SurveillanceBot(object): | |||||||
|         self.status_message = '' |         self.status_message = '' | ||||||
|         self.html_fig_dir = 'figures' |         self.html_fig_dir = 'figures' | ||||||
| 
 | 
 | ||||||
|  |         self.active_figures = {} | ||||||
|  | 
 | ||||||
|         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() | ||||||
| 
 | 
 | ||||||
| @ -156,12 +168,11 @@ class SurveillanceBot(object): | |||||||
|                         if channel in channels_wf_data: |                         if channel in channels_wf_data: | ||||||
|                             self.filenames_wf_data += fnames |                             self.filenames_wf_data += fnames | ||||||
| 
 | 
 | ||||||
|     def read_data(self, re_read_at_hour=1, daily_overlap=2): |     def read_data(self, re_read_at_hour=1): | ||||||
|         ''' |         ''' | ||||||
|         read data method reads new data into self.stream |         read data method reads new data into self.stream | ||||||
| 
 | 
 | ||||||
|         :param re_read_at_hour: update archive at specified hour each day (hours up to 24) |         :param re_read_at_hour: update archive at specified hour each day (hours up to 24) | ||||||
|         :param daily_overlap: re-read data of previous day until specified hour (hours up to 24) |  | ||||||
|         ''' |         ''' | ||||||
|         self.data = {} |         self.data = {} | ||||||
| 
 | 
 | ||||||
| @ -169,15 +180,17 @@ class SurveillanceBot(object): | |||||||
|         curr_time = UTCDateTime() |         curr_time = UTCDateTime() | ||||||
|         current_day = curr_time.julday |         current_day = curr_time.julday | ||||||
|         current_hour = curr_time.hour |         current_hour = curr_time.hour | ||||||
|         yesterday = (curr_time - 24. * 3600.).julday |  | ||||||
|         if re_read_at_hour is not False and current_day != self.current_day and current_hour == re_read_at_hour: |         if re_read_at_hour is not False and current_day != self.current_day and current_hour == re_read_at_hour: | ||||||
|             self.filenames_read = [] |             self.filenames_read_last_modif = {} | ||||||
|             self.dataStream = Stream() |             self.dataStream = Stream() | ||||||
|             self.current_day = current_day |             self.current_day = current_day | ||||||
| 
 | 
 | ||||||
|         # add all data to current stream |         # add all data to current stream | ||||||
|         for filename in self.filenames: |         for filename in self.filenames: | ||||||
|             if filename in self.filenames_read: |             # if file already read and last modification time is the same as of last read operation: continue | ||||||
|  |             if self.filenames_read_last_modif.get(filename) == os.path.getmtime(filename): | ||||||
|  |                 if self.verbosity > 0: | ||||||
|  |                     print('Continue on file', filename) | ||||||
|                 continue |                 continue | ||||||
|             try: |             try: | ||||||
|                 # read only header of wf_data |                 # read only header of wf_data | ||||||
| @ -185,11 +198,7 @@ class SurveillanceBot(object): | |||||||
|                     st_new = read(filename, headonly=True) |                     st_new = read(filename, headonly=True) | ||||||
|                 else: |                 else: | ||||||
|                     st_new = read(filename, dtype=float) |                     st_new = read(filename, dtype=float) | ||||||
|                 # add file to read filenames to prevent re-reading in case it is not the current day (or end of |                 self.filenames_read_last_modif[filename] = os.path.getmtime(filename) | ||||||
|                 # previous day) |  | ||||||
|                 if not filename.endswith(f'{current_day:03}') and not ( |  | ||||||
|                         filename.endswith(f'{yesterday:03}') and current_hour <= daily_overlap): |  | ||||||
|                     self.filenames_read.append(filename) |  | ||||||
|             except Exception as e: |             except Exception as e: | ||||||
|                 print(f'Could not read file {filename}:', e) |                 print(f'Could not read file {filename}:', e) | ||||||
|                 continue |                 continue | ||||||
| @ -357,13 +366,13 @@ class SurveillanceBot(object): | |||||||
|         for nwst_id in self.station_list: |         for nwst_id in self.station_list: | ||||||
|             self.write_html_figure(nwst_id) |             self.write_html_figure(nwst_id) | ||||||
| 
 | 
 | ||||||
|     def write_html_figure(self, nwst_id): |     def write_html_figure(self, nwst_id, save_bytes=False): | ||||||
|         """ Write figure for html for specified station """ |         """ Write figure for html for specified station """ | ||||||
|         self.check_fig_dir() |         self.check_fig_dir() | ||||||
| 
 | 
 | ||||||
|         fig = plt.figure(figsize=(16, 9)) |         fig = plt.figure(figsize=(16, 9)) | ||||||
|         fnout = self.get_fig_path_abs(nwst_id) |         fnames_out = [self.get_fig_path_abs(nwst_id), io.BytesIO()] | ||||||
|         st = self.data.get(nwst_id) |         st = self.data.get(get_full_seed_id(nwst_id)) | ||||||
|         if st: |         if st: | ||||||
|             # TODO: this section failed once, adding try-except block for analysis and to prevent program from crashing |             # TODO: this section failed once, adding try-except block for analysis and to prevent program from crashing | ||||||
|             try: |             try: | ||||||
| @ -385,51 +394,50 @@ class SurveillanceBot(object): | |||||||
|                              f'Refreshed hourly or on FAIL status.') |                              f'Refreshed hourly or on FAIL status.') | ||||||
|                 for ax in fig.axes: |                 for ax in fig.axes: | ||||||
|                     ax.grid(True, alpha=0.1) |                     ax.grid(True, alpha=0.1) | ||||||
|  |                 for fnout in fnames_out: | ||||||
|                     fig.savefig(fnout, dpi=150., bbox_inches='tight') |                     fig.savefig(fnout, dpi=150., bbox_inches='tight') | ||||||
|  |                 # if needed save figure as virtual object (e.g. for mailing) | ||||||
|  |                 if save_bytes: | ||||||
|  |                     fnames_out[-1].seek(0) | ||||||
|  |                     self.active_figures[nwst_id] = fnames_out[-1] | ||||||
|         plt.close(fig) |         plt.close(fig) | ||||||
| 
 | 
 | ||||||
|     def write_html_table(self, default_color='#e6e6e6', default_header_color='#999', hide_keys_mobile=('other')): |     def get_html_class(self, hide_keys_mobile=None, status=None, check_key=None): | ||||||
| 
 |  | ||||||
|         def get_html_class(status=None, check_key=None): |  | ||||||
|         """ helper function for html class if a certain condition is fulfilled """ |         """ helper function for html class if a certain condition is fulfilled """ | ||||||
|         html_class = None |         html_class = None | ||||||
|         if status and status.is_active: |         if status and status.is_active: | ||||||
|             html_class = 'blink-bg' |             html_class = 'blink-bg' | ||||||
|             if check_key in hide_keys_mobile: |         if hide_keys_mobile and check_key in hide_keys_mobile: | ||||||
|             html_class = 'hidden-mobile' |             html_class = 'hidden-mobile' | ||||||
|         return html_class |         return html_class | ||||||
| 
 | 
 | ||||||
|         self.check_html_dir() |     def make_html_table_header(self, default_header_color, hide_keys_mobile=None, add_links=True): | ||||||
|         fnout = pjoin(self.outpath_html, 'survBot_out.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 |         # First write header items | ||||||
|         header = self.keys.copy() |         header = self.keys.copy() | ||||||
|         # add columns for additional links |         # add columns for additional links | ||||||
|  |         if add_links: | ||||||
|             for key in self.add_links: |             for key in self.add_links: | ||||||
|                 header.insert(-1, key) |                 header.insert(-1, key) | ||||||
|  | 
 | ||||||
|         header_items = [dict(text='Station', color=default_header_color)] |         header_items = [dict(text='Station', color=default_header_color)] | ||||||
|         for check_key in header: |         for check_key in header: | ||||||
|                     html_class = get_html_class(check_key=check_key) |             html_class = self.get_html_class(hide_keys_mobile, check_key=check_key) | ||||||
|             item = dict(text=check_key, color=default_header_color, html_class=html_class) |             item = dict(text=check_key, color=default_header_color, html_class=html_class) | ||||||
|             header_items.append(item) |             header_items.append(item) | ||||||
|                 write_html_row(outfile, header_items, html_key='th') |  | ||||||
| 
 | 
 | ||||||
|                 # Write all cells |         return header, header_items | ||||||
|                 for nwst_id in self.station_list: | 
 | ||||||
|  |     def get_html_row_items(self, status_dict, nwst_id, header, default_color, hide_keys_mobile=None, | ||||||
|  |                            hyperlinks=True): | ||||||
|  |         ''' create a html table row for the different keys ''' | ||||||
|  | 
 | ||||||
|         fig_name = self.get_fig_path_rel(nwst_id) |         fig_name = self.get_fig_path_rel(nwst_id) | ||||||
|         nwst_id_str = nwst_id.rstrip('.') |         nwst_id_str = nwst_id.rstrip('.') | ||||||
|                     col_items = [dict(text=nwst_id_str, color=default_color, hyperlink=fig_name, |         col_items = [dict(text=nwst_id_str, color=default_color, hyperlink=fig_name if hyperlinks else None, | ||||||
|                           bold=True, tooltip=f'Show plot of {nwst_id_str}')] |                           bold=True, tooltip=f'Show plot of {nwst_id_str}')] | ||||||
|  | 
 | ||||||
|         for check_key in header: |         for check_key in header: | ||||||
|             if check_key in self.keys: |             if check_key in self.keys: | ||||||
|                             status_dict = self.analysis_results.get(nwst_id) |  | ||||||
|                 status = status_dict.get(check_key) |                 status = status_dict.get(check_key) | ||||||
|                 message, detailed_message = status.get_status_str() |                 message, detailed_message = status.get_status_str() | ||||||
| 
 | 
 | ||||||
| @ -444,7 +452,7 @@ class SurveillanceBot(object): | |||||||
|                     if not type(message) in [str]: |                     if not type(message) in [str]: | ||||||
|                         message = str(message) + deg_str |                         message = str(message) + deg_str | ||||||
| 
 | 
 | ||||||
|                             html_class = get_html_class(status=status, check_key=check_key) |                 html_class = self.get_html_class(hide_keys_mobile, status=status, check_key=check_key) | ||||||
|                 item = dict(text=str(message), tooltip=str(detailed_message), color=bg_color, |                 item = dict(text=str(message), tooltip=str(detailed_message), color=bg_color, | ||||||
|                             html_class=html_class) |                             html_class=html_class) | ||||||
|             elif check_key in self.add_links: |             elif check_key in self.add_links: | ||||||
| @ -455,14 +463,42 @@ class SurveillanceBot(object): | |||||||
|                 nw, st = nwst_id.split('.')[:2] |                 nw, st = nwst_id.split('.')[:2] | ||||||
|                 hyperlink_dict = dict(nw=nw, st=st, nwst_id=nwst_id) |                 hyperlink_dict = dict(nw=nw, st=st, nwst_id=nwst_id) | ||||||
|                 link = value.format(**hyperlink_dict) |                 link = value.format(**hyperlink_dict) | ||||||
|                             item = dict(text=link_text, tooltip=link, hyperlink=link, color=default_color) |                 item = dict(text=link_text, tooltip=link, hyperlink=link if hyperlinks else None, color=default_color) | ||||||
|  |             else: | ||||||
|  |                 item = dict(text='', tooltip='') | ||||||
|             col_items.append(item) |             col_items.append(item) | ||||||
| 
 | 
 | ||||||
|                     write_html_row(outfile, col_items) |         return col_items | ||||||
|  | 
 | ||||||
|  |     def write_html_table(self, default_color='#e6e6e6', default_header_color='#999999', hide_keys_mobile=('other',)): | ||||||
|  |         self.check_html_dir() | ||||||
|  |         fnout = pjoin(self.outpath_html, 'survBot_out.html') | ||||||
|  |         if not fnout: | ||||||
|  |             return | ||||||
|  |         try: | ||||||
|  |             with open(fnout, 'w') as outfile: | ||||||
|  |                 outfile.write(get_html_header(self.refresh_period)) | ||||||
|  | 
 | ||||||
|  |                 # write_html_table_title(self.parameters) | ||||||
|  |                 outfile.write(init_html_table()) | ||||||
|  | 
 | ||||||
|  |                 # write html header row | ||||||
|  |                 header, header_items = self.make_html_table_header(default_header_color, hide_keys_mobile) | ||||||
|  |                 html_row = get_html_row(header_items, html_key='th') | ||||||
|  |                 outfile.write(html_row) | ||||||
|  | 
 | ||||||
|  |                 # Write all cells (row after row) | ||||||
|  |                 for nwst_id in self.station_list: | ||||||
|  |                     # get list with column-wise items to write as a html row | ||||||
|  |                     status_dict = self.analysis_results.get(nwst_id) | ||||||
|  |                     col_items = self.get_html_row_items(status_dict, nwst_id, header, default_color, hide_keys_mobile) | ||||||
|  |                     outfile.write(get_html_row(col_items)) | ||||||
|  | 
 | ||||||
|  |                 outfile.write(finish_html_table()) | ||||||
|  | 
 | ||||||
|  |                 outfile.write(get_html_text(self.status_message)) | ||||||
|  |                 outfile.write(html_footer()) | ||||||
| 
 | 
 | ||||||
|                 finish_html_table(outfile) |  | ||||||
|                 write_html_text(outfile, self.status_message) |  | ||||||
|                 write_html_footer(outfile) |  | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|             print(f'Could not write HTML table to {fnout}:') |             print(f'Could not write HTML table to {fnout}:') | ||||||
|             print(traceback.format_exc()) |             print(traceback.format_exc()) | ||||||
| @ -505,7 +541,8 @@ class StationQC(object): | |||||||
|         self.network = nsl.get('network') |         self.network = nsl.get('network') | ||||||
|         self.station = nsl.get('station') |         self.station = nsl.get('station') | ||||||
|         self.location = nsl.get('location') |         self.location = nsl.get('location') | ||||||
|         self.parameters = parameters |         # make a copy of parameters object to prevent accidental changes | ||||||
|  |         self.parameters = copy.deepcopy(parameters) | ||||||
|         self.program_starttime = starttime |         self.program_starttime = starttime | ||||||
|         self.verbosity = verbosity |         self.verbosity = verbosity | ||||||
|         self.last_active = False |         self.last_active = False | ||||||
| @ -567,6 +604,7 @@ class StationQC(object): | |||||||
|         # self.status_dict[key] = current_status_message + status_message |         # self.status_dict[key] = current_status_message + status_message | ||||||
| 
 | 
 | ||||||
|     def error(self, key, detailed_message, last_occurrence=None, count=1): |     def error(self, key, detailed_message, last_occurrence=None, count=1): | ||||||
|  |         send_mail = False | ||||||
|         new_error = StatusError(count=count, show_count=self.parameters.get('warn_count')) |         new_error = StatusError(count=count, show_count=self.parameters.get('warn_count')) | ||||||
|         current_status = self.status_dict.get(key) |         current_status = self.status_dict.get(key) | ||||||
|         if current_status.is_error: |         if current_status.is_error: | ||||||
| @ -574,21 +612,24 @@ class StationQC(object): | |||||||
|         else: |         else: | ||||||
|             current_status = new_error |             current_status = new_error | ||||||
|             # if error is new and not on program-startup set active and refresh plot (using parent class) |             # 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: |             if self.status_track.get(key) and not self.status_track.get(key)[-1]: | ||||||
|                 self.parent.write_html_figure(self.nwst_id) |                 self.parent.write_html_figure(self.nwst_id, save_bytes=True) | ||||||
| 
 | 
 | ||||||
|         if self.verbosity: |         if self.verbosity: | ||||||
|             self.print(f'{UTCDateTime()}: {detailed_message}', flush=False) |             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) |         # 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) is True: |         if self.search_previous_errors(key) is True: | ||||||
|             self.send_mail(key, status_type='FAIL', additional_message=detailed_message) |             send_mail = True | ||||||
|             # set status to "inactive" after sending info mail |             # set status to "inactive" when info mail is sent | ||||||
|             current_status.is_active = False |             current_status.is_active = False | ||||||
|         elif self.search_previous_errors(key) == 'active': |         elif self.search_previous_errors(key) == 'active': | ||||||
|             current_status.is_active = True |             current_status.is_active = True | ||||||
| 
 | 
 | ||||||
|  |         # first update status, then send mail | ||||||
|         self._update_status(key, current_status, detailed_message, last_occurrence) |         self._update_status(key, current_status, detailed_message, last_occurrence) | ||||||
|  |         if send_mail: | ||||||
|  |             self.send_mail(key, status_type='FAIL', additional_message=detailed_message) | ||||||
| 
 | 
 | ||||||
|     def search_previous_errors(self, key, n_errors=None): |     def search_previous_errors(self, key, n_errors=None): | ||||||
|         """ |         """ | ||||||
| @ -602,20 +643,33 @@ class StationQC(object): | |||||||
|         if n_errors is None: |         if n_errors is None: | ||||||
|             n_errors = self.parameters.get('n_track') |             n_errors = self.parameters.get('n_track') | ||||||
| 
 | 
 | ||||||
|         # +1 to check whether n_errors +1 was no error (error is new) |         # +1 to check whether n_errors + 1 was no error (error is new) | ||||||
|         n_errors += 1 |         n_errors += 1 | ||||||
| 
 | 
 | ||||||
|  |         # simulate an error specified in json file (dictionary: {nwst_id: key} ) | ||||||
|  |         if self._simulated_error_check(key) is True: | ||||||
|  |             print(f'Simulating Error on {self.nwst_id}, {key}') | ||||||
|  |             return True | ||||||
|  | 
 | ||||||
|         previous_errors = self.status_track.get(key) |         previous_errors = self.status_track.get(key) | ||||||
|         # only if error list is filled n_track times |         # only if error list is filled n_track times | ||||||
|         if previous_errors and len(previous_errors) == n_errors: |         if previous_errors and len(previous_errors) == n_errors: | ||||||
|             # if first entry was no error but all others are, return True (-> new Fail n_track times) |             # 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:]): |             if not previous_errors[0] and all(previous_errors[1:]): | ||||||
|                 return True |                 return True | ||||||
|         # in case previous_errors exists, last item is error but not all items are error, error still active |         # in case previous_errors exist, 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): |         if previous_errors and previous_errors[-1] and not all(previous_errors): | ||||||
|             return 'active' |             return 'active' | ||||||
|         return False |         return False | ||||||
| 
 | 
 | ||||||
|  |     def _simulated_error_check(self, key, fname='simulate_fail.json'): | ||||||
|  |         if not os.path.isfile(fname): | ||||||
|  |             return | ||||||
|  |         with open(fname) as fid: | ||||||
|  |             d = json.load(fid) | ||||||
|  |         if d.get(self.nwst_id) == key: | ||||||
|  |             return True | ||||||
|  | 
 | ||||||
|     def send_mail(self, key, status_type, additional_message=''): |     def send_mail(self, key, status_type, additional_message=''): | ||||||
|         """ Send info mail using parameters specified in parameters file """ |         """ Send info mail using parameters specified in parameters file """ | ||||||
|         if not mail_functionality: |         if not mail_functionality: | ||||||
| @ -643,23 +697,93 @@ class StationQC(object): | |||||||
| 
 | 
 | ||||||
|         sender = mail_params.get('sender') |         sender = mail_params.get('sender') | ||||||
|         addresses = mail_params.get('addresses') |         addresses = mail_params.get('addresses') | ||||||
|  |         add_addresses = self.get_additional_mail_recipients(mail_params) | ||||||
|  |         if add_addresses: | ||||||
|  |             # create copy of addresses ( [:] ) to prevent changing original, general list with addresses | ||||||
|  |             addresses = addresses[:] + list(add_addresses) | ||||||
|         server = mail_params.get('mailserver') |         server = mail_params.get('mailserver') | ||||||
|         if not sender or not addresses: |         if not sender or not addresses: | ||||||
|             if self.verbosity: |             if self.verbosity: | ||||||
|                 print('Mail sender or addresses not correctly defined. Return') |                 print('Mail sender or addresses not (correctly) defined. Return') | ||||||
|             return |             return | ||||||
|         dt = self.get_dt_for_action() |         dt = self.get_dt_for_action() | ||||||
|         text = f'{key}: Status {status_type} longer than {dt}: ' + additional_message |         text = f'{key}: Status {status_type} longer than {dt}: ' + additional_message | ||||||
|         msg = MIMEText(text) | 
 | ||||||
|  |         msg = EmailMessage() | ||||||
|  | 
 | ||||||
|         msg['Subject'] = f'new message on station {self.nwst_id}' |         msg['Subject'] = f'new message on station {self.nwst_id}' | ||||||
|         msg['From'] = sender |         msg['From'] = sender | ||||||
|         msg['To'] = ', '.join(addresses) |         msg['To'] = ', '.join(addresses) | ||||||
| 
 | 
 | ||||||
|  |         msg.set_content(text) | ||||||
|  | 
 | ||||||
|  |         # html mail version | ||||||
|  |         html_str = self.add_html_mail_body(text) | ||||||
|  |         msg.add_alternative(html_str, subtype='html') | ||||||
|  | 
 | ||||||
|         # send message via SMTP server |         # send message via SMTP server | ||||||
|         s = smtplib.SMTP(server) |         s = smtplib.SMTP(server) | ||||||
|         s.sendmail(sender, addresses, msg.as_string()) |         s.send_message(msg) | ||||||
|         s.quit() |         s.quit() | ||||||
| 
 | 
 | ||||||
|  |     def add_html_mail_body(self, text, default_color='#e6e6e6'): | ||||||
|  |         parent = self.parent | ||||||
|  | 
 | ||||||
|  |         header, header_items = parent.make_html_table_header('#999999', add_links=False) | ||||||
|  |         col_items = parent.get_html_row_items(self.status_dict, self.nwst_id, header, default_color, hyperlinks=False) | ||||||
|  | 
 | ||||||
|  |         # set general status text | ||||||
|  |         html_str = get_html_text(text) | ||||||
|  | 
 | ||||||
|  |         # init html header and table | ||||||
|  |         html_str += get_mail_html_header() | ||||||
|  |         html_str += init_html_table() | ||||||
|  | 
 | ||||||
|  |         # add table header and row of current station | ||||||
|  |         html_str += get_html_row(header_items, html_key='th') | ||||||
|  |         html_str += get_html_row(col_items) | ||||||
|  | 
 | ||||||
|  |         html_str += finish_html_table() | ||||||
|  | 
 | ||||||
|  |         if self.nwst_id in self.parent.active_figures.keys(): | ||||||
|  |             fid = self.parent.active_figures.pop(self.nwst_id) | ||||||
|  |             html_str += add_html_image(img_data=fid.read()) | ||||||
|  | 
 | ||||||
|  |         html_str += html_footer() | ||||||
|  | 
 | ||||||
|  |         return html_str | ||||||
|  | 
 | ||||||
|  |     def get_additional_mail_recipients(self, mail_params): | ||||||
|  |         """ return additional recipients from external mail list if this station (self.nwst_id) is specified """ | ||||||
|  |         eml_filename = mail_params.get('external_mail_list') | ||||||
|  |         if eml_filename: | ||||||
|  |             # try to open file | ||||||
|  |             try: | ||||||
|  |                 with open(eml_filename, 'r') as fid: | ||||||
|  |                     address_dict = yaml.safe_load(fid) | ||||||
|  | 
 | ||||||
|  |                 for address, nwst_ids in address_dict.items(): | ||||||
|  |                     if self.nwst_id in nwst_ids: | ||||||
|  |                         yield address | ||||||
|  |             # file not existing | ||||||
|  |             except FileNotFoundError as e: | ||||||
|  |                 if self.verbosity: | ||||||
|  |                     print(e) | ||||||
|  |             # no dictionary | ||||||
|  |             except AttributeError as e: | ||||||
|  |                 if self.verbosity: | ||||||
|  |                     print(f'Could not read dictionary from file {eml_filename}: {e}') | ||||||
|  |             # other exceptions | ||||||
|  |             except Exception as e: | ||||||
|  |                 if self.verbosity: | ||||||
|  |                     print(f'Could not open file {eml_filename}: {e}') | ||||||
|  |         # no file specified | ||||||
|  |         else: | ||||||
|  |             if self.verbosity: | ||||||
|  |                 print('No external mail list set.') | ||||||
|  | 
 | ||||||
|  |         return [] | ||||||
|  | 
 | ||||||
|     def get_dt_for_action(self): |     def get_dt_for_action(self): | ||||||
|         n_track = self.parameters.get('n_track') |         n_track = self.parameters.get('n_track') | ||||||
|         interval = self.parameters.get('interval') |         interval = self.parameters.get('interval') | ||||||
| @ -735,6 +859,13 @@ class StationQC(object): | |||||||
|         # activity check should be done last for useful status output (e.g. email) |         # activity check should be done last for useful status output (e.g. email) | ||||||
|         self.activity_check() |         self.activity_check() | ||||||
| 
 | 
 | ||||||
|  |         self._simulate_error() | ||||||
|  | 
 | ||||||
|  |     def _simulate_error(self): | ||||||
|  |         for key in self.keys: | ||||||
|  |             if self._simulated_error_check(key): | ||||||
|  |                 self.error(key, 'SIMULATED ERROR') | ||||||
|  | 
 | ||||||
|     def return_print_analysis(self): |     def return_print_analysis(self): | ||||||
|         items = [self.nwst_id] |         items = [self.nwst_id] | ||||||
|         for key in self.keys: |         for key in self.keys: | ||||||
| @ -997,7 +1128,7 @@ class StationQC(object): | |||||||
|                     if message == 'OK': |                     if message == 'OK': | ||||||
|                         self.status_ok(key) |                         self.status_ok(key) | ||||||
|                         continue |                         continue | ||||||
|                     if volt_lvl > 1: |                     if volt_lvl != 1: | ||||||
|                         n_occurrences = self.calc_occurrences(ind_array) |                         n_occurrences = self.calc_occurrences(ind_array) | ||||||
|                         self.warn(key=key, |                         self.warn(key=key, | ||||||
|                                   detailed_message=f'Trace {trace.get_id()}: ' |                                   detailed_message=f'Trace {trace.get_id()}: ' | ||||||
| @ -1233,7 +1364,9 @@ class StatusOther(Status): | |||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     parser = argparse.ArgumentParser(description='Call survBot') |     parser = argparse.ArgumentParser(description='Call survBot') | ||||||
|     parser.add_argument('-html', dest='html_path', default=None, help='filepath for HTML output') |     parser.add_argument('-html', dest='html_path', default=None, help='filepath for HTML output') | ||||||
|  |     parser.add_argument('-parfile', dest='parfile', default='parameters.yaml', | ||||||
|  |                         help='parameter file (default: parameters.yaml)') | ||||||
|     args = parser.parse_args() |     args = parser.parse_args() | ||||||
| 
 | 
 | ||||||
|     survBot = SurveillanceBot(parameter_path='parameters.yaml', outpath_html=args.html_path) |     survBot = SurveillanceBot(parameter_path=args.parfile, outpath_html=args.html_path) | ||||||
|     survBot.start() |     survBot.start() | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								utils.py
									
									
									
									
									
								
							
							
						
						
									
										3
									
								
								utils.py
									
									
									
									
									
								
							| @ -47,6 +47,7 @@ def get_color_mpl(key): | |||||||
| 
 | 
 | ||||||
| def get_time_delay_color(dt, dt_thresh): | def get_time_delay_color(dt, dt_thresh): | ||||||
|     """ Set color of time delay after thresholds specified in self.dt_thresh """ |     """ Set color of time delay after thresholds specified in self.dt_thresh """ | ||||||
|  |     if isinstance(dt, type(dt_thresh[0])): | ||||||
|         if dt < dt_thresh[0]: |         if dt < dt_thresh[0]: | ||||||
|             return get_color('OK') |             return get_color('OK') | ||||||
|         elif dt_thresh[0] <= dt < dt_thresh[1]: |         elif dt_thresh[0] <= dt < dt_thresh[1]: | ||||||
| @ -155,7 +156,7 @@ def set_axis_ylabels(fig, parameters, verbosity=0): | |||||||
|             ax.set_ylabel(channel_name) |             ax.set_ylabel(channel_name) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def set_axis_color(fig, color='grey'): | def set_axis_color(fig, color='0.8'): | ||||||
|     """ |     """ | ||||||
|     Set all axes of figure to specific color |     Set all axes of figure to specific color | ||||||
|     """ |     """ | ||||||
|  | |||||||
| @ -1,16 +1,21 @@ | |||||||
|  | from base64 import b64encode | ||||||
| from datetime import timedelta | from datetime import timedelta | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def write_html_table_title(fobj, parameters): | def _convert_to_textstring(lst): | ||||||
|  |     return '\n'.join(lst) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def get_html_table_title(parameters): | ||||||
|     title = get_print_title_str(parameters) |     title = get_print_title_str(parameters) | ||||||
|     fobj.write(f'<h3>{title}</h3>\n') |     return f'<h3>{title}</h3>\n' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def write_html_text(fobj, text): | def get_html_text(text): | ||||||
|     fobj.write(f'<p>{text}</p>\n') |     return f'<p>{text}</p>\n' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def write_html_header(fobj, refresh_rate=10): | def get_html_header(refresh_rate=10): | ||||||
|     header = ['<!DOCTYPE html>', |     header = ['<!DOCTYPE html>', | ||||||
|               '<html>', |               '<html>', | ||||||
|               '<head>', |               '<head>', | ||||||
| @ -21,28 +26,42 @@ def write_html_header(fobj, refresh_rate=10): | |||||||
|               '<meta charset="utf-8">', |               '<meta charset="utf-8">', | ||||||
|               '<meta name="viewport" content="width=device-width, initial-scale=1">', |               '<meta name="viewport" content="width=device-width, initial-scale=1">', | ||||||
|               '<body>'] |               '<body>'] | ||||||
|     for item in header: |     header = _convert_to_textstring(header) | ||||||
|         fobj.write(item + '\n') |     return header | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def init_html_table(fobj): | def get_mail_html_header(): | ||||||
|     fobj.write('<table style="width:100%">\n') |     header = ['<html>', | ||||||
|  |               '<head>', | ||||||
|  |               '</head>', | ||||||
|  |               '<body>'] | ||||||
|  |     header = _convert_to_textstring(header) | ||||||
|  |     return header | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def finish_html_table(fobj): | def init_html_table(): | ||||||
|     fobj.write('</table>\n') |     return '<table style="width:100%">\n' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def write_html_footer(fobj): | def finish_html_table(): | ||||||
|  |     return '</table>\n' | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def html_footer(): | ||||||
|     footer = ['</body>', |     footer = ['</body>', | ||||||
|               '</html>'] |               '</html>'] | ||||||
|     for item in footer: |     footer = _convert_to_textstring(footer) | ||||||
|         fobj.write(item + '\n') |     return footer | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def write_html_row(fobj, items, html_key='td'): | def add_html_image(img_data, img_format='png'): | ||||||
|  |     return f"""<br>\n<img width="100%" src="data:image/{img_format};base64, {b64encode(img_data).decode('ascii')}">""" | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def get_html_row(items, html_key='td'): | ||||||
|  |     row_string = '' | ||||||
|     default_space = '  ' |     default_space = '  ' | ||||||
|     fobj.write(default_space + '<tr>\n') |     row_string += default_space + '<tr>\n' | ||||||
|     for item in items: |     for item in items: | ||||||
|         text = item.get('text') |         text = item.get('text') | ||||||
|         if item.get('bold'): |         if item.get('bold'): | ||||||
| @ -57,9 +76,10 @@ def write_html_row(fobj, items, html_key='td'): | |||||||
|         image_str = f'<a href="{hyperlink}">' if hyperlink else '' |         image_str = f'<a href="{hyperlink}">' if hyperlink else '' | ||||||
|         html_class = item.get('html_class') |         html_class = item.get('html_class') | ||||||
|         class_str = f' class="{html_class}"' if html_class else '' |         class_str = f' class="{html_class}"' if html_class else '' | ||||||
|         fobj.write(2 * default_space + f'<{html_key}{class_str} bgcolor="{color}" title="{tooltip}"> {image_str}' |         row_string += 2 * default_space + f'<{html_key}{class_str} bgcolor="{color}" title="{tooltip}"> {image_str}'\ | ||||||
|                    + text + f'</{html_key}>\n') |                     + text + f'</{html_key}>\n' | ||||||
|     fobj.write(default_space + '</tr>\n') |     row_string += default_space + '</tr>\n' | ||||||
|  |     return row_string | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def get_print_title_str(parameters): | def get_print_title_str(parameters): | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user