From 448708a8a8cdd3269e81bf72dce29da392e9a553 Mon Sep 17 00:00:00 2001 From: spencer <swilbur@usgs.gov> Date: Tue, 24 Sep 2024 15:06:44 -0600 Subject: [PATCH 1/3] Adding a few changes to FDSNFActory regarding the FDSNNoDataException. I also created a private function to contain the logic that removes sensitivity/response info for traces. I added all avaiable stations that transmit geomgagnetic data for networks, IU, N4, and US to the VariometerMetadata.py module. --- geomagio/VariometerMetadata.py | 202 +++++++++++++++++++++++++++++---- geomagio/edge/FDSNFactory.py | 80 +++++++------ 2 files changed, 221 insertions(+), 61 deletions(-) diff --git a/geomagio/VariometerMetadata.py b/geomagio/VariometerMetadata.py index cdffb271..3dfab0e6 100644 --- a/geomagio/VariometerMetadata.py +++ b/geomagio/VariometerMetadata.py @@ -30,8 +30,8 @@ DEFAULT_ASL_METADATA = { "ANMO": { "metadata": { "elevation": 1820, - "geodetic_latitude": 34.946, - "geodetic_longitude": -106.457, + "geodetic_latitude": 34.94591, + "geodetic_longitude": -106.4572, "station_name": "Albuquerque", "agency_name": "United States Geological Survey (USGS)", "declination_base": None, @@ -41,8 +41,8 @@ DEFAULT_ASL_METADATA = { "CASY": { "metadata": { "elevation": 10, - "geodetic_latitude": -66.279, - "geodetic_longitude": 110.535, + "geodetic_latitude": -66.2792, + "geodetic_longitude": 110.5354, "station_name": "Casey", "agency_name": "United States Geological Survey (USGS)", "declination_base": None, @@ -52,8 +52,8 @@ DEFAULT_ASL_METADATA = { "COLA": { "metadata": { "elevation": 200, - "geodetic_latitude": 64.874, - "geodetic_longitude": -147.862, + "geodetic_latitude": 64.8736, + "geodetic_longitude": -147.8616, "station_name": "College Outpost", "agency_name": "United States Geological Survey (USGS)", "declination_base": None, @@ -63,20 +63,31 @@ DEFAULT_ASL_METADATA = { "COR": { "metadata": { "elevation": 110, - "geodetic_latitude": 44.586, - "geodetic_longitude": -123.305, + "geodetic_latitude": 44.5855, + "geodetic_longitude": -123.3046, "station_name": "Corvallis", "agency_name": "United States Geological Survey (USGS)", "declination_base": None, }, "interval_specific": DEFAULT_INTERVAL_SPECIFIC, }, - "QSPA": { + "KBS": { "metadata": { - "elevation": 2850, - "geodetic_latitude": -89.929, - "geodetic_longitude": 144.438, - "station_name": "South Pole Remote Earth Science Observatory", + "elevation": 90, + "geodetic_latitude": 78.9154, + "geodetic_longitude": 11.9385, + "station_name": "Ny-Alesund", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "KONO": { + "metadata": { + "elevation": 558, + "geodetic_latitude": 59.6521, + "geodetic_longitude": 9.5946, + "station_name": "Kongsberg", "agency_name": "United States Geological Survey (USGS)", "declination_base": None, }, @@ -85,8 +96,8 @@ DEFAULT_ASL_METADATA = { "RSSD": { "metadata": { "elevation": 2090, - "geodetic_latitude": 44.121, - "geodetic_longitude": -104.036, + "geodetic_latitude": 44.1212, + "geodetic_longitude": -104.0359, "station_name": "Black Hills", "agency_name": "United States Geological Survey (USGS)", "declination_base": None, @@ -96,8 +107,8 @@ DEFAULT_ASL_METADATA = { "SBA": { "metadata": { "elevation": 50, - "geodetic_latitude": -77.849, - "geodetic_longitude": 166.757, + "geodetic_latitude": -77.8492, + "geodetic_longitude": 166.7572, "station_name": "Scott Base", "agency_name": "United States Geological Survey (USGS)", "declination_base": None, @@ -107,20 +118,163 @@ DEFAULT_ASL_METADATA = { "SFJD": { "metadata": { "elevation": 330, - "geodetic_latitude": 66.996, - "geodetic_longitude": -50.621, + "geodetic_latitude": 66.9961, + "geodetic_longitude": -50.6207, "station_name": "Sondre Stromfjord", "agency_name": "United States Geological Survey (USGS)", "declination_base": None, }, "interval_specific": DEFAULT_INTERVAL_SPECIFIC, }, - "SPA": { + "SSPA": { + "metadata": { + "elevation": 270, + "geodetic_latitude": 40.6358, + "geodetic_longitude": -77.8876, + "station_name": "Standing Stone, Pennsylvania", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "QSPA": { + "metadata": { + "elevation": 2850, + "geodetic_latitude": -89.9289, + "geodetic_longitude": 144.4382, + "station_name": "South Pole Remote Earth Science Observatory", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "E62A": { + "metadata": { + "elevation": 365, + "geodetic_latitude": 46.6201, + "geodetic_longitude": -69.5227, + "station_name": "Eastport", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "E46A": { + "metadata": { + "elevation": 269, + "geodetic_latitude": 46.3665, + "geodetic_longitude": -84.3062, + "station_name": "Lake Superior", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "PAB": { + "metadata": { + "elevation": 950, + "geodetic_latitude": 39.5446, + "geodetic_longitude": -4.3499, + "station_name": "San Pablo, Spain", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "WCI": { + "metadata": { + "elevation": 210, + "geodetic_latitude": 38.2289, + "geodetic_longitude": -86.2939, + "station_name": "Wyandotte Cave, Indiana, USA", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "KSU1": { + "metadata": { + "elevation": 317, + "geodetic_latitude": 39.1009, + "geodetic_longitude": -96.6094, + "station_name": "Kansas State University", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "EYMN": { + "metadata": { + "elevation": 475, + "geodetic_latitude": 47.9462, + "geodetic_longitude": -91.4953, + "station_name": "Ely, Minnesota, USA", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "GOGA": { + "metadata": { + "elevation": 150, + "geodetic_latitude": 33.41476, + "geodetic_longitude": -83.47327, + "station_name": "Godfrey, GA, USA", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "DGMT": { + "metadata": { + "elevation": 646, + "geodetic_latitude": 48.4702, + "geodetic_longitude": -104.1959, + "station_name": "Dagmar Montana, USA", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "DWPF": { + "metadata": { + "elevation": 30, + "geodetic_latitude": 28.1103, + "geodetic_longitude": -81.4327, + "station_name": "Disney Wilderness Preserve, Florida, USA", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "ECSD": { + "metadata": { + "elevation": 478, + "geodetic_latitude": 43.7337, + "geodetic_longitude": -96.6141, + "station_name": "EROS Data Center, Sioux Falls, South Dakota, USA", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "HRV": { + "metadata": { + "elevation": 200, + "geodetic_latitude": 42.5064, + "geodetic_longitude": -71.5583, + "station_name": "Adam Dziewonski Observatory (Oak Ridge), Massachusetts, USA", + "agency_name": "United States Geological Survey (USGS)", + "declination_base": None, + }, + "interval_specific": DEFAULT_INTERVAL_SPECIFIC, + }, + "O20A": { "metadata": { - "elevation": 2927, - "geodetic_latitude": -90.0, - "geodetic_longitude": 0.0, - "station_name": "South Pole", + "elevation": 1915, + "geodetic_latitude": 40.1348, + "geodetic_longitude": -108.2416, + "station_name": "White River City, CO, USA", "agency_name": "United States Geological Survey (USGS)", "declination_base": None, }, diff --git a/geomagio/edge/FDSNFactory.py b/geomagio/edge/FDSNFactory.py index afd9de17..b4109d4d 100644 --- a/geomagio/edge/FDSNFactory.py +++ b/geomagio/edge/FDSNFactory.py @@ -256,6 +256,13 @@ class FDSNFactory(TimeseriesFactory): endtime=endtime + half_delta, attach_response=True, ) + # Perfrom calibrations to properly remove response/sensitivity + data = self._calibrate_sensitivty_and_response(data) + + if channel in channel_rotations: + data = self._rotate_and_return_requested_channel( + data, sncl, starttime, endtime, channel + ) except FDSNNoDataException: data = Stream() @@ -282,43 +289,8 @@ class FDSNFactory(TimeseriesFactory): TimeseriesUtility.pad_and_trim_trace( trace=data[0], starttime=starttime, endtime=endtime ) - if channel in channel_rotations: - data = self._rotate_and_return_requested_channel( - data, sncl, starttime, endtime, channel - ) - - for trace in data: - if hasattr(trace.stats, "response") and trace.stats.response is not None: - response = trace.stats.response - if self.remove_sensitivity is not None: - if self.remove_sensitivity_flag == "remove_sensitivity": - trace.remove_sensitivity() - elif self.remove_sensitivity_flag == "remove_response": - trace.remove_response() - else: - raise ValueError( - f"Warning: Unrecognized remove_sensitivity value '{self.remove_sensitivity}'. No sensitivity or response removal applied." - ) - else: - if response.instrument_sensitivity: - if response.instrument_sensitivity.input_units == "T": - # some ASL stations produce data in Teslas, not nanoteslas - response.instrument_sensitivity.value /= 1e9 - response.instrument_sensitivity.input_units = "nT" - # apply total gain/sensitivity, ignore any frequency response - trace.remove_sensitivity() - else: - # according to fdsn docs, if no instrument_sensitivity, - # the response must be an instrument_polynomial - # https://docs.fdsn.org/projects/stationxml/en/latest/reference.html#instrumentsensitivity - - # Geomag Program observatories use 2nd order instrument_polynomial - # to capture both offset and scale, but no frequency response - data.remove_response() - - else: - self._set_metadata(data, observatory, channel, type, interval) + self._set_metadata(data, observatory, channel, type, interval) return data @@ -403,7 +375,7 @@ class FDSNFactory(TimeseriesFactory): # Initialize FDSN client and get the necessary data FDSN = self.Client - # Determine if the requested channel is X, Y, or Z + # Determine if the requested channel is X, Y, or inv = FDSN.get_stations( network=sncl.network, @@ -436,3 +408,37 @@ class FDSNFactory(TimeseriesFactory): return Stream() return rotated_stream + + def _calibrate_sensitivty_and_response(self, data: Stream) -> Stream: + + for trace in data: + if hasattr(trace.stats, "response") and trace.stats.response is not None: + response = trace.stats.response + if self.remove_sensitivity is not None: + if self.remove_sensitivity_flag == "remove_sensitivity": + trace.remove_sensitivity() + + elif self.remove_sensitivity_flag == "remove_response": + trace.remove_response() + else: + raise ValueError( + f"Warning: Unrecognized remove_sensitivity value '{self.remove_sensitivity}'. No sensitivity or response removal applied." + ) + else: + if response.instrument_sensitivity: + if response.instrument_sensitivity.input_units == "T": + # some ASL stations produce data in Teslas, not nanoteslas + response.instrument_sensitivity.value /= 1e9 + response.instrument_sensitivity.input_units = "nT" + # apply total gain/sensitivity, ignore any frequency response + trace.remove_sensitivity() + + else: + # according to fdsn docs, if no instrument_sensitivity, + # the response must be an instrument_polynomial + # https://docs.fdsn.org/projects/stationxml/en/latest/reference.html#instrumentsensitivity + + # Geomag Program observatories use 2nd order instrument_polynomial + # to capture both offset and scale, but no frequency response + data.remove_response() + return data -- GitLab From f98589290b27c09df9a760ec995568ea25c6ae4c Mon Sep 17 00:00:00 2001 From: spencer <swilbur@usgs.gov> Date: Thu, 26 Sep 2024 09:02:57 -0600 Subject: [PATCH 2/3] I removed an unecessary call to FilterApiQuery from within Data.py. I also added validation to the FilterApiQuery.py. Rather than using super I opted to add the validation to this file beacuse sampling_period was ignored from the base class i.e. DataApiQuery and this is why the REQUEST_LIMIT was being ignored. --- geomagio/api/ws/DataApiQuery.py | 12 +----------- geomagio/api/ws/FilterApiQuery.py | 26 +++++++++++++++++++++++++- geomagio/api/ws/data.py | 1 - 3 files changed, 26 insertions(+), 13 deletions(-) diff --git a/geomagio/api/ws/DataApiQuery.py b/geomagio/api/ws/DataApiQuery.py index a1716691..6cba83df 100644 --- a/geomagio/api/ws/DataApiQuery.py +++ b/geomagio/api/ws/DataApiQuery.py @@ -12,7 +12,7 @@ from .Observatory import OBSERVATORY_INDEX, ASL_OBSERVATORY_INDEX DEFAULT_ELEMENTS = ["X", "Y", "Z", "F"] -REQUEST_LIMIT = 345600 +REQUEST_LIMIT = 3456000 # Increased the request limit by 10x what was decided by Jeremy VALID_ELEMENTS = [e.id for e in ELEMENTS] @@ -149,13 +149,3 @@ class DataApiQuery(BaseModel): raise ValueError(f"Request exceeds limit ({samples} > {REQUEST_LIMIT})") # otherwise okay return values - - -# The new class inheriting everything except input/output_sampling period from DataApiQuery -class FilterDataApiQuery(DataApiQuery): - input_sampling_period: SamplingPeriod = SamplingPeriod.SECOND - output_sampling_period: SamplingPeriod = SamplingPeriod.MINUTE - - # Remove inherited fields that we don't need - class Config: - fields = {"sampling_period": {"exclude": True}, "data_host": {"exclude": True}} diff --git a/geomagio/api/ws/FilterApiQuery.py b/geomagio/api/ws/FilterApiQuery.py index 2eb960ac..dcae69ac 100644 --- a/geomagio/api/ws/FilterApiQuery.py +++ b/geomagio/api/ws/FilterApiQuery.py @@ -1,4 +1,5 @@ -from .DataApiQuery import DataApiQuery, SamplingPeriod +from .DataApiQuery import DataApiQuery, SamplingPeriod, REQUEST_LIMIT +from pydantic import root_validator """This script contains the class inheriting everything except input/output_sampling period from the DataApiQuery class. This is where more specific functionailty @@ -12,3 +13,26 @@ class FilterApiQuery(DataApiQuery): # Remove inherited fields that we don't need for this specific endpoint class Config: fields = {"sampling_period": {"exclude": True}} + + @root_validator + def validate_combinations(cls, values): + starttime, endtime, elements, format, input_sampling_period = ( + values.get("starttime"), + values.get("endtime"), + values.get("elements"), + values.get("format"), + values.get("input_sampling_period"), + ) + if len(elements) > 4 and format == "iaga2002": + raise ValueError("No more than four elements allowed for iaga2002 format.") + if starttime > endtime: + raise ValueError("Starttime must be before endtime.") + + # Calculate the number of samples based on the input sampling period + samples = int(len(elements) * (endtime - starttime) / input_sampling_period) + + # Validate the request size + if samples > REQUEST_LIMIT: + raise ValueError(f"Request exceeds limit ({samples} > {REQUEST_LIMIT})") + + return values diff --git a/geomagio/api/ws/data.py b/geomagio/api/ws/data.py index 3cde0746..40a09da6 100644 --- a/geomagio/api/ws/data.py +++ b/geomagio/api/ws/data.py @@ -13,7 +13,6 @@ from .Observatory import ASL_OBSERVATORY_INDEX from .DataApiQuery import ( DEFAULT_ELEMENTS, DataApiQuery, - FilterDataApiQuery, DataType, OutputFormat, SamplingPeriod, -- GitLab From 7ed080e4c0b3ebf94994dd8aecbc140a2f545fae Mon Sep 17 00:00:00 2001 From: spencer <swilbur@usgs.gov> Date: Thu, 26 Sep 2024 09:24:54 -0600 Subject: [PATCH 3/3] Updated data_test.py to not retrieve FilterApiQuery from DataApiQuery. Honestly not sure how these are just now causing issues when all my previous commits have contained this code. --- test/api_test/ws_test/data_test.py | 1 - 1 file changed, 1 deletion(-) diff --git a/test/api_test/ws_test/data_test.py b/test/api_test/ws_test/data_test.py index f56f7429..e0b731ef 100644 --- a/test/api_test/ws_test/data_test.py +++ b/test/api_test/ws_test/data_test.py @@ -10,7 +10,6 @@ from geomagio.api.ws.DataApiQuery import ( DataApiQuery, OutputFormat, SamplingPeriod, - FilterDataApiQuery, ) -- GitLab