Compare commits
	
		
			4 Commits
		
	
	
		
			c8c3aff2fb
			...
			7d5f9cf516
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 7d5f9cf516 | |||
| b17ee1288c | |||
| bf82148449 | |||
| 174a8e0823 | 
| @ -3,7 +3,6 @@ datapath: "/data/SDS/"                    # SC3 Datapath | |||||||
| networks: ["1Y", "HA"]                    # select networks, list or str | networks: ["1Y", "HA"]                    # select networks, list or str | ||||||
| stations: "*"                             # select stations, list or str | stations: "*"                             # select stations, list or str | ||||||
| locations: "*"                            # select locations, list or str | locations: "*"                            # select locations, list or str | ||||||
| channels: ["EX1", "EX2", "EX3", "VEI", "LCQ"]    # Specify SOH channels, currently supported EX[1-3], VEI and LCQ |  | ||||||
| stations_blacklist: ["TEST", "EREA"]      # exclude these stations | 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 | ||||||
| @ -43,9 +42,73 @@ THRESHOLDS: | |||||||
|   low_volt: 12                 # min voltage for low voltage warning |   low_volt: 12                 # min voltage for low voltage warning | ||||||
|   high_volt: 14.8              # max voltage for over voltage warning |   high_volt: 14.8              # max voltage for over voltage warning | ||||||
|   unclassified: 5              # min voltage samples not classified for warning |   unclassified: 5              # min voltage samples not classified for warning | ||||||
|  |   max_vm_warn: 1.5             # threshold for mass offset (warn), fail) | ||||||
|  |   max_vm_fail: 2.5             # threshold for mass offset (warn), fail) | ||||||
|   clockquality_warn: 90        # clock quality ranges from 0 % to 100 % with 100 % being the best level |   clockquality_warn: 90        # clock quality ranges from 0 % to 100 % with 100 % being the best level | ||||||
|   clockquality_fail: 70 |   clockquality_fail: 70 | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | # ---------------------------------- Specification of input channels --------------------------------------------------- | ||||||
|  | # Currently supported: EX[1-3], VEI, VM[1-3], LCQ | ||||||
|  | # | ||||||
|  | # For each channel a factor 'unit' for unit conversion (e.g. to SI) can be provided, as well as a 'name' | ||||||
|  | # and 'ticks' [ymin, ymax, ystep] for plotting. | ||||||
|  | # 'warn' and 'fail' plot horizontal lines in corresponding colors (can be str in TRESHOLDS, int/float or iterable) | ||||||
|  | # | ||||||
|  | # 'transform' can be provided for plotting to perform arithmetic operations in given order, e.g.: | ||||||
|  | #  transform: - ["*", 20] | ||||||
|  | #             - ["-", 20] | ||||||
|  | #  --> PBox EX1 V to deg C: 20 * x -20 | ||||||
|  | CHANNELS: | ||||||
|  |   EX1: | ||||||
|  |     unit: 1e-6 | ||||||
|  |     name: "Temperature (°C)" | ||||||
|  |     ticks: [-10, 50, 10] | ||||||
|  |     transform: | ||||||
|  |       - ["*", 20] | ||||||
|  |       - ["-", 20] | ||||||
|  |     warn: "max_temp" | ||||||
|  |   EX2: | ||||||
|  |     unit: 1e-6 | ||||||
|  |     name: "230V/12V (V)" | ||||||
|  |     ticks: [1, 5, 1] | ||||||
|  |     warn: [2, 3, 4, 4.5, 5] | ||||||
|  |   EX3: | ||||||
|  |     unit: 1e-6 | ||||||
|  |     name: "Rout/Charge (V)" | ||||||
|  |     ticks: [1, 5, 1] | ||||||
|  |     warn: [2, 2.5, 3, 4, 5] | ||||||
|  |   VEI: | ||||||
|  |     unit: 1e-3 | ||||||
|  |     name: "Logger (V)" | ||||||
|  |     ticks: [9, 15, 1] | ||||||
|  |     warn: ["low_volt", "high_volt"] | ||||||
|  |     fail: 10.5 | ||||||
|  |   VM1: | ||||||
|  |     unit: 1e-6 | ||||||
|  |     name: "Mass 1 (V)" | ||||||
|  |     ticks: [-2.5, 2.5, 1] | ||||||
|  |     warn: [-1.5, 1.5] | ||||||
|  |     fail: [-2.5, 2.5] | ||||||
|  |   VM2: | ||||||
|  |     unit: 1e-6 | ||||||
|  |     name: "Mass 2 (V)" | ||||||
|  |     ticks: [-2.5, 2.5, 1] | ||||||
|  |     warn: [-1.5, 1.5] | ||||||
|  |     fail: [-2.5, 2.5] | ||||||
|  |   VM3: | ||||||
|  |     unit: 1e-6 | ||||||
|  |     name: "Mass 3 (V)" | ||||||
|  |     ticks: [-2.5, 2.5, 1] | ||||||
|  |     warn: [-1.5, 1.5] | ||||||
|  |     fail: [-2.5, 2.5] | ||||||
|  |   LCQ: | ||||||
|  |     name: "Clock Q (%)" | ||||||
|  |     ticks: [0, 100, 20] | ||||||
|  |     warn: "clockquality_warn" | ||||||
|  |     fail: "clockquality_fail" | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| # ---------------------------------------- OPTIONAL PARAMETERS --------------------------------------------------------- | # ---------------------------------------- OPTIONAL PARAMETERS --------------------------------------------------------- | ||||||
| 
 | 
 | ||||||
| # add links to html table with specified key as column and value as relative link, interpretable string parameters: | # add links to html table with specified key as column and value as relative link, interpretable string parameters: | ||||||
| @ -62,26 +125,3 @@ 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 | ||||||
| 
 |  | ||||||
| # names for plotting of the above defined parameter "channels" in the same order |  | ||||||
| channel_names: ["Clock Quality (%)", "Temperature (°C)", "230V/12V Status (V)", "Router/Charger State (V)", "Logger Voltage (V)"] # names for plotting (optional) |  | ||||||
| # specify y-ticks (and ylims) giving, (ymin, ymax, step) for each of the above channels (0: default) |  | ||||||
| CHANNEL_TICKS: |  | ||||||
|   - [0, 100, 20] |  | ||||||
|   - [-10, 50, 10] |  | ||||||
|   - [1, 5, 1] |  | ||||||
|   - [1, 5, 1] |  | ||||||
|   - [9, 15, 1] |  | ||||||
| 
 |  | ||||||
| # Factor for channel to SI-units (for plotting) |  | ||||||
| CHANNEL_UNITS: |  | ||||||
|   EX1: 1e-6 |  | ||||||
|   EX2: 1e-6 |  | ||||||
|   EX3: 1e-6 |  | ||||||
|   VEI: 1e-3 |  | ||||||
| 
 |  | ||||||
| # Transform channel for plotting, perform arithmetic operations in given order, e.g.: PBox EX1 V to deg C: 20 * x -20 |  | ||||||
| CHANNEL_TRANSFORM: |  | ||||||
|   EX1: |  | ||||||
|     - ["*", 20] |  | ||||||
|     - ["-", 20] |  | ||||||
							
								
								
									
										88
									
								
								survBot.py
									
									
									
									
									
								
							
							
						
						
									
										88
									
								
								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, \ | 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 |     init_html_table, finish_html_table | ||||||
| from utils import get_bg_color, modify_stream_for_plot, trace_ylabels, trace_yticks | from utils import get_bg_color, modify_stream_for_plot, trace_yticks, trace_thresholds | ||||||
| 
 | 
 | ||||||
| try: | try: | ||||||
|     import smtplib |     import smtplib | ||||||
| @ -36,9 +36,16 @@ CLR = "\x1B[0K" | |||||||
| deg_str = '\N{DEGREE SIGN}C' | deg_str = '\N{DEGREE SIGN}C' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def read_yaml(file_path): | def read_yaml(file_path, n_read=3): | ||||||
|  |     for index in range(n_read): | ||||||
|  |         try: | ||||||
|             with open(file_path, "r") as f: |             with open(file_path, "r") as f: | ||||||
|         return yaml.safe_load(f) |                 params = yaml.safe_load(f) | ||||||
|  |         except Exception as e: | ||||||
|  |             print(f'Could not read parameters file: {e}.\nWill try again {n_read - index - 1} time(s).') | ||||||
|  |             time.sleep(10) | ||||||
|  |             continue | ||||||
|  |         return params | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def nsl_from_id(nwst_id): | def nsl_from_id(nwst_id): | ||||||
| @ -61,7 +68,7 @@ def fancy_timestr(dt, thresh=600, modif='+'): | |||||||
| 
 | 
 | ||||||
| class SurveillanceBot(object): | class SurveillanceBot(object): | ||||||
|     def __init__(self, parameter_path, outpath_html=None): |     def __init__(self, parameter_path, outpath_html=None): | ||||||
|         self.keys = ['last active', '230V', '12V', 'router', 'charger', 'voltage', 'clock', 'temp', 'other'] |         self.keys = ['last active', '230V', '12V', 'router', 'charger', 'voltage', 'mass', 'temp', 'other'] | ||||||
|         self.parameter_path = parameter_path |         self.parameter_path = parameter_path | ||||||
|         self.update_parameters() |         self.update_parameters() | ||||||
|         self.starttime = UTCDateTime() |         self.starttime = UTCDateTime() | ||||||
| @ -85,6 +92,8 @@ class SurveillanceBot(object): | |||||||
| 
 | 
 | ||||||
|     def update_parameters(self): |     def update_parameters(self): | ||||||
|         self.parameters = read_yaml(self.parameter_path) |         self.parameters = read_yaml(self.parameter_path) | ||||||
|  |         # add channels to list in parameters dicitonary | ||||||
|  |         self.parameters['channels'] = list(self.parameters.get('CHANNELS').keys()) | ||||||
|         self.reread_parameters = self.parameters.get('reread_parameters') |         self.reread_parameters = self.parameters.get('reread_parameters') | ||||||
|         self.dt_thresh = [int(val) for val in self.parameters.get('dt_thresh')] |         self.dt_thresh = [int(val) for val in self.parameters.get('dt_thresh')] | ||||||
|         self.verbosity = self.parameters.get('verbosity') |         self.verbosity = self.parameters.get('verbosity') | ||||||
| @ -242,7 +251,7 @@ class SurveillanceBot(object): | |||||||
|     def get_station_delay(self, nwst_id): |     def get_station_delay(self, nwst_id): | ||||||
|         """ try to get station delay from SDS archive using client""" |         """ try to get station delay from SDS archive using client""" | ||||||
|         locations = ['', '0', '00'] |         locations = ['', '0', '00'] | ||||||
|         channels = ['HHZ', 'HHE', 'HHN', 'VEI', 'EX1', 'EX2', 'EX3'] |         channels = ['HHZ', 'HHE', 'HHN'] + self.parameters.get('channels') | ||||||
|         network, station = nwst_id.split('.')[:2] |         network, station = nwst_id.split('.')[:2] | ||||||
| 
 | 
 | ||||||
|         times = [] |         times = [] | ||||||
| @ -341,8 +350,9 @@ class SurveillanceBot(object): | |||||||
|             try: |             try: | ||||||
|                 st = modify_stream_for_plot(st, parameters=self.parameters) |                 st = modify_stream_for_plot(st, parameters=self.parameters) | ||||||
|                 st.plot(fig=fig, show=False, draw=False, block=False, equal_scale=False, method='full') |                 st.plot(fig=fig, show=False, draw=False, block=False, equal_scale=False, method='full') | ||||||
|                 trace_ylabels(fig, self.parameters, self.verbosity) |                 # trace_ylabels(fig, self.parameters, self.verbosity) | ||||||
|                 trace_yticks(fig, self.parameters, self.verbosity) |                 trace_yticks(fig, self.parameters, self.verbosity) | ||||||
|  |                 trace_thresholds(fig, self.parameters, self.verbosity) | ||||||
|             except Exception as e: |             except Exception as e: | ||||||
|                 print(f'Could not generate plot for {nwst_id}:') |                 print(f'Could not generate plot for {nwst_id}:') | ||||||
|                 print(traceback.format_exc()) |                 print(traceback.format_exc()) | ||||||
| @ -434,6 +444,9 @@ class SurveillanceBot(object): | |||||||
|             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()) | ||||||
| 
 | 
 | ||||||
|  |         if self.verbosity: | ||||||
|  |             print(f'Wrote html table to {fnout}') | ||||||
|  | 
 | ||||||
|     def update_status_message(self): |     def update_status_message(self): | ||||||
|         timespan = timedelta(seconds=int(self.parameters.get('timespan') * 24 * 3600)) |         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")} | ' \ |         self.status_message = f'Program starttime (UTC) {self.starttime.strftime("%Y-%m-%d %H:%M:%S")} | ' \ | ||||||
| @ -685,7 +698,8 @@ class StationQC(object): | |||||||
|         self.pb_temp_analysis() |         self.pb_temp_analysis() | ||||||
|         self.pb_power_analysis() |         self.pb_power_analysis() | ||||||
|         self.pb_rout_charge_analysis() |         self.pb_rout_charge_analysis() | ||||||
|         self.clock_quality_analysis() |         self.mass_analysis() | ||||||
|  |         #self.clock_quality_analysis() | ||||||
| 
 | 
 | ||||||
|     def return_print_analysis(self): |     def return_print_analysis(self): | ||||||
|         items = [self.nwst_id] |         items = [self.nwst_id] | ||||||
| @ -739,7 +753,7 @@ class StationQC(object): | |||||||
|         if len(clockQuality_warn) > 0: |         if len(clockQuality_warn) > 0: | ||||||
|             # try calculate number of warn peaks from gaps between indices |             # try calculate number of warn peaks from gaps between indices | ||||||
|             n_qc_warn = self.calc_occurrences(clockQuality_warn) |             n_qc_warn = self.calc_occurrences(clockQuality_warn) | ||||||
|             detailed_message = warn_message + f' {n_qc_warn}x Qlock Quality less then {clockQuality_warn_level}' \ |             detailed_message = warn_message + f' {n_qc_warn}x Qlock Quality less then {clockQuality_warn_level}%' \ | ||||||
|                                + self.get_last_occurrence_timestring(trace, clockQuality_warn) |                                + self.get_last_occurrence_timestring(trace, clockQuality_warn) | ||||||
|             self.warn(key, detailed_message=detailed_message, count=n_qc_warn, |             self.warn(key, detailed_message=detailed_message, count=n_qc_warn, | ||||||
|                       last_occurrence=self.get_last_occurrence(trace, clockQuality_warn)) |                       last_occurrence=self.get_last_occurrence(trace, clockQuality_warn)) | ||||||
| @ -747,7 +761,7 @@ class StationQC(object): | |||||||
|         if len(clockQuality_fail) > 0: |         if len(clockQuality_fail) > 0: | ||||||
|             # try calculate number of fail peaks from gaps between indices |             # try calculate number of fail peaks from gaps between indices | ||||||
|             n_qc_fail = self.calc_occurrences(clockQuality_fail) |             n_qc_fail = self.calc_occurrences(clockQuality_fail) | ||||||
|             detailed_message = warn_message + f' {n_qc_fail}x Qlock Quality less then {clockQuality_fail_level}V ' \ |             detailed_message = warn_message + f' {n_qc_fail}x Qlock Quality less then {clockQuality_fail_level}%' \ | ||||||
|                                + self.get_last_occurrence_timestring(trace, clockQuality_fail) |                                + self.get_last_occurrence_timestring(trace, clockQuality_fail) | ||||||
|             self.error(key, detailed_message=detailed_message, count=n_qc_fail, |             self.error(key, detailed_message=detailed_message, count=n_qc_fail, | ||||||
|                       last_occurrence=self.get_last_occurrence(trace, clockQuality_fail)) |                       last_occurrence=self.get_last_occurrence(trace, clockQuality_fail)) | ||||||
| @ -757,7 +771,8 @@ class StationQC(object): | |||||||
|         key = 'voltage' |         key = 'voltage' | ||||||
|         st = self.stream.select(channel=channel) |         st = self.stream.select(channel=channel) | ||||||
|         trace = self.get_trace(st, key) |         trace = self.get_trace(st, key) | ||||||
|         if not trace: return |         if not trace: | ||||||
|  |             return | ||||||
|         voltage = trace.data * 1e-3 |         voltage = trace.data * 1e-3 | ||||||
|         low_volt = self.parameters.get('THRESHOLDS').get('low_volt') |         low_volt = self.parameters.get('THRESHOLDS').get('low_volt') | ||||||
|         high_volt = self.parameters.get('THRESHOLDS').get('high_volt') |         high_volt = self.parameters.get('THRESHOLDS').get('high_volt') | ||||||
| @ -795,14 +810,15 @@ class StationQC(object): | |||||||
|         key = 'temp' |         key = 'temp' | ||||||
|         st = self.stream.select(channel=channel) |         st = self.stream.select(channel=channel) | ||||||
|         trace = self.get_trace(st, key) |         trace = self.get_trace(st, key) | ||||||
|         if not trace: return |         if not trace: | ||||||
|  |             return | ||||||
|         voltage = trace.data * 1e-6 |         voltage = trace.data * 1e-6 | ||||||
|         thresholds = self.parameters.get('THRESHOLDS') |         thresholds = self.parameters.get('THRESHOLDS') | ||||||
|         temp = 20. * voltage - 20 |         temp = 20. * voltage - 20 | ||||||
|         # average temp |         # average temp | ||||||
|         timespan = min([self.parameters.get('timespan') * 24 * 3600, int(len(temp) / trace.stats.sampling_rate)]) |         timespan = min([self.parameters.get('timespan') * 24 * 3600, int(len(temp) / trace.stats.sampling_rate)]) | ||||||
|         nsamp_av = int(trace.stats.sampling_rate) * timespan |         nsamp_av = int(trace.stats.sampling_rate) * timespan | ||||||
|         av_temp_str = str(round(np.mean(temp[-nsamp_av:]), 1)) + deg_str |         av_temp_str = str(round(np.nanmean(temp[-nsamp_av:]), 1)) + deg_str | ||||||
|         # dt of average |         # dt of average | ||||||
|         dt_t_str = str(timedelta(seconds=int(timespan))).replace(', 0:00:00', '') |         dt_t_str = str(timedelta(seconds=int(timespan))).replace(', 0:00:00', '') | ||||||
|         # current temp |         # current temp | ||||||
| @ -810,7 +826,7 @@ class StationQC(object): | |||||||
|         if self.verbosity > 1: |         if self.verbosity > 1: | ||||||
|             self.print(40 * '-') |             self.print(40 * '-') | ||||||
|             self.print('Performing PowBox temperature check (EX1)', flush=False) |             self.print('Performing PowBox temperature check (EX1)', flush=False) | ||||||
|             self.print(f'Average temperature at {np.mean(temp)}\N{DEGREE SIGN}', flush=False) |             self.print(f'Average temperature at {np.nanmean(temp)}\N{DEGREE SIGN}', flush=False) | ||||||
|             self.print(f'Peak temperature at {max(temp)}\N{DEGREE SIGN}', flush=False) |             self.print(f'Peak temperature at {max(temp)}\N{DEGREE SIGN}', flush=False) | ||||||
|             self.print(f'Min temperature at {min(temp)}\N{DEGREE SIGN}', flush=False) |             self.print(f'Min temperature at {min(temp)}\N{DEGREE SIGN}', flush=False) | ||||||
|         max_temp = thresholds.get('max_temp') |         max_temp = thresholds.get('max_temp') | ||||||
| @ -826,6 +842,52 @@ class StationQC(object): | |||||||
|                            status_message=cur_temp, |                            status_message=cur_temp, | ||||||
|                            detailed_message=f'Average temperature of last {dt_t_str}: {av_temp_str}') |                            detailed_message=f'Average temperature of last {dt_t_str}: {av_temp_str}') | ||||||
| 
 | 
 | ||||||
|  |     def mass_analysis(self, channels=('VM1', 'VM2', 'VM3'), n_samp_mean=10): | ||||||
|  |         """ Analyse datalogger mass channels. """ | ||||||
|  |         key = 'mass' | ||||||
|  | 
 | ||||||
|  |         # build stream with all channels | ||||||
|  |         st = Stream() | ||||||
|  |         for channel in channels: | ||||||
|  |             st += self.stream.select(channel=channel).copy() | ||||||
|  |         st.merge() | ||||||
|  | 
 | ||||||
|  |         # return if there are no three components | ||||||
|  |         if not len(st) == 3: | ||||||
|  |             return | ||||||
|  | 
 | ||||||
|  |         # correct for channel unit | ||||||
|  |         for trace in st: | ||||||
|  |             trace.data = trace.data * 1e-6  # hardcoded, change this? | ||||||
|  | 
 | ||||||
|  |         # calculate average of absolute maximum of mass offset of last n_samp_mean | ||||||
|  |         last_values = np.array([trace.data[-n_samp_mean:] for trace in st]) | ||||||
|  |         last_val_mean = np.nanmean(last_values, axis=1) | ||||||
|  |         common_highest_val = np.nanmax(abs(last_val_mean)) | ||||||
|  |         common_highest_val = round(common_highest_val, 1) | ||||||
|  | 
 | ||||||
|  |         # get thresholds for WARN (max_vm_warn) and FAIL (max_vm_fail) | ||||||
|  |         thresholds = self.parameters.get('THRESHOLDS') | ||||||
|  |         max_vm_warn = thresholds.get('max_vm_warn') | ||||||
|  |         max_vm_fail = thresholds.get('max_vm_fail') | ||||||
|  |         if not max_vm_warn or not max_vm_fail: | ||||||
|  |             return | ||||||
|  | 
 | ||||||
|  |         # change status depending on common_highest_val | ||||||
|  |         if common_highest_val < max_vm_warn: | ||||||
|  |             self.status_ok(key, detailed_message=f'{common_highest_val}V') | ||||||
|  |         elif max_vm_warn <= common_highest_val < max_vm_fail: | ||||||
|  |             self.warn(key=key, | ||||||
|  |                       detailed_message=f'Warning raised for mass centering. Highest val {common_highest_val}V', ) | ||||||
|  |         else: | ||||||
|  |             self.error(key=key, | ||||||
|  |                       detailed_message=f'Fail status for mass centering. Highest val {common_highest_val}V',) | ||||||
|  | 
 | ||||||
|  |         if self.verbosity > 1: | ||||||
|  |             self.print(40 * '-') | ||||||
|  |             self.print('Performing mass position check', flush=False) | ||||||
|  |             self.print(f'Average mass position at {common_highest_val}', flush=False) | ||||||
|  | 
 | ||||||
|     def pb_power_analysis(self, channel='EX2', pb_dict_key='pb_SOH2'): |     def pb_power_analysis(self, channel='EX2', pb_dict_key='pb_SOH2'): | ||||||
|         """ Analyse EX2 channel of PowBox """ |         """ Analyse EX2 channel of PowBox """ | ||||||
|         keys = ['230V', '12V'] |         keys = ['230V', '12V'] | ||||||
|  | |||||||
| @ -34,7 +34,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, modify_stream_for_plot, trace_ylabels, trace_yticks | from utils import get_bg_color, modify_stream_for_plot, trace_yticks, trace_thresholds | ||||||
| 
 | 
 | ||||||
| try: | try: | ||||||
|     from rest_api.utils import get_station_iccid |     from rest_api.utils import get_station_iccid | ||||||
| @ -316,8 +316,9 @@ class MainWindow(QtWidgets.QMainWindow): | |||||||
|             self.plot_widget.setWindowTitle(nwst_id) |             self.plot_widget.setWindowTitle(nwst_id) | ||||||
|             st = modify_stream_for_plot(st, parameters=self.parameters) |             st = modify_stream_for_plot(st, parameters=self.parameters) | ||||||
|             st.plot(equal_scale=False, method='full', block=False, fig=self.plot_widget.canvas.fig) |             st.plot(equal_scale=False, method='full', block=False, fig=self.plot_widget.canvas.fig) | ||||||
|             trace_ylabels(fig=self.plot_widget.canvas.fig, parameters=self.parameters) |             # trace_ylabels(fig=self.plot_widget.canvas.fig, parameters=self.parameters) | ||||||
|             trace_yticks(fig=self.plot_widget.canvas.fig, parameters=self.parameters) |             trace_yticks(fig=self.plot_widget.canvas.fig, parameters=self.parameters) | ||||||
|  |             trace_thresholds(fig=self.plot_widget.canvas.fig, parameters=self.parameters) | ||||||
|             self.plot_widget.show() |             self.plot_widget.show() | ||||||
| 
 | 
 | ||||||
|     def notification(self, text): |     def notification(self, text): | ||||||
|  | |||||||
							
								
								
									
										119
									
								
								utils.py
									
									
									
									
									
								
							
							
						
						
									
										119
									
								
								utils.py
									
									
									
									
									
								
							| @ -2,6 +2,9 @@ | |||||||
| # -*- coding: utf-8 -*- | # -*- coding: utf-8 -*- | ||||||
| 
 | 
 | ||||||
| import matplotlib | import matplotlib | ||||||
|  | import numpy as np | ||||||
|  | 
 | ||||||
|  | from obspy import Stream | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def get_bg_color(check_key, status, dt_thresh=None, hex=False): | def get_bg_color(check_key, status, dt_thresh=None, hex=False): | ||||||
| @ -10,9 +13,11 @@ def get_bg_color(check_key, status, dt_thresh=None, hex=False): | |||||||
|         bg_color = get_time_delay_color(message, dt_thresh) |         bg_color = get_time_delay_color(message, dt_thresh) | ||||||
|     elif check_key == 'temp': |     elif check_key == 'temp': | ||||||
|         bg_color = get_temp_color(message) |         bg_color = get_temp_color(message) | ||||||
|  |     elif check_key == 'mass': | ||||||
|  |         bg_color = get_mass_color(message) | ||||||
|     else: |     else: | ||||||
|         if status.is_warn: |         if status.is_warn: | ||||||
|             bg_color = get_color('WARNX')(status.count) |             bg_color = get_warn_color(status.count) | ||||||
|         elif status.is_error: |         elif status.is_error: | ||||||
|             bg_color = get_color('FAIL') |             bg_color = get_color('FAIL') | ||||||
|         else: |         else: | ||||||
| @ -30,12 +35,16 @@ def get_color(key): | |||||||
|     colors_dict = {'FAIL': (255, 50, 0, 255), |     colors_dict = {'FAIL': (255, 50, 0, 255), | ||||||
|                    'NO DATA': (255, 255, 125, 255), |                    'NO DATA': (255, 255, 125, 255), | ||||||
|                    'WARN': (255, 255, 80, 255), |                    'WARN': (255, 255, 80, 255), | ||||||
|                    'WARNX': lambda x: (min([255, 200 + x ** 2]), 255, 80, 255), |  | ||||||
|                    'OK': (125, 255, 125, 255), |                    'OK': (125, 255, 125, 255), | ||||||
|                    'undefined': (230, 230, 230, 255)} |                    'undefined': (230, 230, 230, 255)} | ||||||
|     return colors_dict.get(key) |     return colors_dict.get(key) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def get_color_mpl(key): | ||||||
|  |     color_tup = get_color(key) | ||||||
|  |     return np.array([color/255. for color in color_tup]) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| 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 dt < dt_thresh[0]: |     if dt < dt_thresh[0]: | ||||||
| @ -45,6 +54,18 @@ def get_time_delay_color(dt, dt_thresh): | |||||||
|     return get_color('FAIL') |     return get_color('FAIL') | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def get_warn_color(count): | ||||||
|  |     color = (min([255, 200 + count ** 2]), 255, 80, 255) | ||||||
|  |     return color | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def get_mass_color(message): | ||||||
|  |     # can change this to something else if wanted. This way it always returns get_color (without warn count) | ||||||
|  |     if isinstance(message, (float, int)): | ||||||
|  |         return get_color('OK') | ||||||
|  |     return get_color(message) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| def get_temp_color(temp, vmin=-10, vmax=60, cmap='coolwarm'): | 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. """ |     """ Get an rgba temperature value back from specified cmap, linearly interpolated between vmin and vmax. """ | ||||||
|     if type(temp) in [str]: |     if type(temp) in [str]: | ||||||
| @ -55,29 +76,43 @@ def get_temp_color(temp, vmin=-10, vmax=60, cmap='coolwarm'): | |||||||
|     return rgba |     return rgba | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def modify_stream_for_plot(st, parameters): | def modify_stream_for_plot(input_stream, parameters): | ||||||
|     """ copy (if necessary) and modify stream for plotting """ |     """ copy (if necessary) and modify stream for plotting """ | ||||||
|     ch_units = parameters.get('CHANNEL_UNITS') |  | ||||||
|     ch_transf = parameters.get('CHANNEL_TRANSFORM') |  | ||||||
| 
 | 
 | ||||||
|     # if either of both are defined make copy |     # make a copy | ||||||
|     if ch_units or ch_transf: |     st = Stream() | ||||||
|         st = st.copy() |  | ||||||
| 
 | 
 | ||||||
|     # modify trace for plotting by multiplying unit factor (e.g. 1e-3 mV to V) |     channels_dict = parameters.get('CHANNELS') | ||||||
|     if ch_units: | 
 | ||||||
|         for tr in st: |     # iterate over all channels and put them to new stream in order | ||||||
|             channel = tr.stats.channel |     for index, ch_tup in enumerate(channels_dict.items()): | ||||||
|             unit_factor = ch_units.get(channel) |         # unpack tuple from items | ||||||
|  |         channel, channel_dict = ch_tup | ||||||
|  | 
 | ||||||
|  |         # get correct channel from stream | ||||||
|  |         st_sel = input_stream.select(channel=channel) | ||||||
|  |         # in case there are != 1 there is ambiguity | ||||||
|  |         if not len(st_sel) == 1: | ||||||
|  |             continue | ||||||
|  | 
 | ||||||
|  |         # make a copy to not modify original stream! | ||||||
|  |         tr = st_sel[0].copy() | ||||||
|  | 
 | ||||||
|  |         # multiply with conversion factor for unit | ||||||
|  |         unit_factor = channel_dict.get('unit') | ||||||
|         if unit_factor: |         if unit_factor: | ||||||
|             tr.data = tr.data * float(unit_factor) |             tr.data = tr.data * float(unit_factor) | ||||||
|     # modify trace for plotting by other arithmetic expressions | 
 | ||||||
|     if ch_transf: |         # apply transformations if provided | ||||||
|         for tr in st: |         transform = channel_dict.get('transform') | ||||||
|             channel = tr.stats.channel |         if transform: | ||||||
|             transf = ch_transf.get(channel) |             tr.data = transform_trace(tr.data, transform) | ||||||
|             if transf: | 
 | ||||||
|                 tr.data = transform_trace(tr.data, transf) |         # modify trace id to maintain plotting order | ||||||
|  |         name = channel_dict.get('name') | ||||||
|  |         tr.id = f'trace {index + 1}: {name} - {tr.id}' | ||||||
|  | 
 | ||||||
|  |         st.append(tr) | ||||||
| 
 | 
 | ||||||
|     return st |     return st | ||||||
| 
 | 
 | ||||||
| @ -107,10 +142,8 @@ def transform_trace(data, transf): | |||||||
| def trace_ylabels(fig, parameters, verbosity=0): | def trace_ylabels(fig, parameters, verbosity=0): | ||||||
|     """ |     """ | ||||||
|     Adds channel names to y-axis if defined in parameters. |     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') |     names = [channel.get('name') for channel in parameters.get('CHANNELS').values()] | ||||||
|     if not names: # or not len(st.traces): |     if not names: # or not len(st.traces): | ||||||
|         return |         return | ||||||
|     if not len(names) == len(fig.axes): |     if not len(names) == len(fig.axes): | ||||||
| @ -125,10 +158,8 @@ def trace_ylabels(fig, parameters, verbosity=0): | |||||||
| def trace_yticks(fig, parameters, verbosity=0): | def trace_yticks(fig, parameters, verbosity=0): | ||||||
|     """ |     """ | ||||||
|     Adds channel names to y-axis if defined in parameters. |     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. |  | ||||||
|     """ |     """ | ||||||
|     ticks = parameters.get('CHANNEL_TICKS') |     ticks = [channel.get('ticks') for channel in parameters.get('CHANNELS').values()] | ||||||
|     if not ticks: |     if not ticks: | ||||||
|         return |         return | ||||||
|     if not len(ticks) == len(fig.axes): |     if not len(ticks) == len(fig.axes): | ||||||
| @ -140,6 +171,38 @@ def trace_yticks(fig, parameters, verbosity=0): | |||||||
|             continue |             continue | ||||||
|         ymin, ymax, step = ytick_tripple |         ymin, ymax, step = ytick_tripple | ||||||
| 
 | 
 | ||||||
|         yticks = list(range(ymin, ymax + step, step)) |         yticks = list(np.arange(ymin, ymax + step, step)) | ||||||
|         ax.set_yticks(yticks) |         ax.set_yticks(yticks) | ||||||
|         ax.set_ylim(ymin - step, ymax + step) |         ax.set_ylim(ymin - 0.33 * step, ymax + 0.33 * step) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def trace_thresholds(fig, parameters, verbosity=0): | ||||||
|  |     """ | ||||||
|  |     Adds channel thresholds (warn, fail) to y-axis if defined in parameters. | ||||||
|  |     """ | ||||||
|  |     if verbosity > 0: | ||||||
|  |         print('Plotting trace thresholds') | ||||||
|  | 
 | ||||||
|  |     keys_colors = {'warn': dict(color=0.8 * get_color_mpl('WARN'), linestyle=(0, (5, 10)), alpha=0.5, linewidth=0.7), | ||||||
|  |                    'fail': dict(color=0.8 * get_color_mpl('FAIL'), linestyle='solid', alpha=0.5, linewidth=0.7)} | ||||||
|  | 
 | ||||||
|  |     for key, kwargs in keys_colors.items(): | ||||||
|  |         channel_threshold_list = [channel.get(key) for channel in parameters.get('CHANNELS').values()] | ||||||
|  |         if not channel_threshold_list: | ||||||
|  |             continue | ||||||
|  |         plot_threshold_lines(fig, channel_threshold_list, parameters, **kwargs) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def plot_threshold_lines(fig, channel_threshold_list, parameters, **kwargs): | ||||||
|  |     for channel_thresholds, ax in zip(channel_threshold_list, fig.axes): | ||||||
|  |         if not channel_thresholds: | ||||||
|  |             continue | ||||||
|  | 
 | ||||||
|  |         if not isinstance(channel_thresholds, (list, tuple)): | ||||||
|  |             channel_thresholds = [channel_thresholds] | ||||||
|  | 
 | ||||||
|  |         for warn_thresh in channel_thresholds: | ||||||
|  |             if isinstance(warn_thresh, str): | ||||||
|  |                 warn_thresh = parameters.get('THRESHOLDS').get(warn_thresh) | ||||||
|  |             if type(warn_thresh in (float, int)): | ||||||
|  |                 ax.axhline(warn_thresh, **kwargs) | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user