Source code for geoh5py.objects.surveys.electromagnetics.tipper

# ''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
#  Copyright (c) 2025 Mira Geoscience Ltd.                                     '
#                                                                              '
#  This file is part of geoh5py.                                               '
#                                                                              '
#  geoh5py is free software: you can redistribute it and/or modify             '
#  it under the terms of the GNU Lesser General Public License as published by '
#  the Free Software Foundation, either version 3 of the License, or           '
#  (at your option) any later version.                                         '
#                                                                              '
#  geoh5py is distributed in the hope that it will be useful,                  '
#  but WITHOUT ANY WARRANTY; without even the implied warranty of              '
#  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the               '
#  GNU Lesser General Public License for more details.                         '
#                                                                              '
#  You should have received a copy of the GNU Lesser General Public License    '
#  along with geoh5py.  If not, see <https://www.gnu.org/licenses/>.           '
# ''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''


from __future__ import annotations

import uuid

import numpy as np

from geoh5py.data import IntegerData, ReferencedData
from geoh5py.objects.curve import Curve
from geoh5py.objects.points import Points

from .base import FEMSurvey


# pylint: disable=too-many-ancestors


[docs] class TipperSurvey(FEMSurvey): """ Base tipper survey class. """ __INPUT_TYPE = ["Rx and base stations"] _base_stations = None _receivers = None def __init__( self, base_stations: TipperBaseStations | None = None, **kwargs, ): self._base_stations = base_stations super().__init__( **kwargs, ) @property def base_stations(self) -> TipperBaseStations | None: """The base station entity""" if isinstance(self, TipperBaseStations): return self if getattr(self, "_base_stations", None) is None: if ( self.metadata is not None and "Base stations" in self.metadata["EM Dataset"] ): base_station = self.metadata["EM Dataset"]["Base stations"] base_station_entity = self.workspace.get_entity(base_station)[0] if isinstance(base_station_entity, TipperBaseStations): self._base_stations = base_station_entity return self._base_stations @base_stations.setter def base_stations(self, base: TipperBaseStations): if not isinstance(base, (TipperBaseStations, type(None))): raise TypeError( f"Input `base_stations` must be of type '{TipperBaseStations}' or None" ) if isinstance(self, TipperBaseStations): raise AttributeError( f"The 'base_station' attribute cannot be set on class {TipperBaseStations}." ) if base.tx_id_property is not None: self.edit_em_metadata({"Tx ID tx property": base.tx_id_property.uid}) if isinstance( self.tx_id_property, ReferencedData | IntegerData ) and isinstance(base.tx_id_property, ReferencedData | IntegerData): self.tx_id_property.entity_type = base.tx_id_property.entity_type self._base_stations = base self.edit_em_metadata({"Base stations": base.uid})
[docs] def copy_from_extent( self, extent: np.ndarray, parent=None, *, copy_children: bool = True, clear_cache: bool = False, inverse: bool = False, **kwargs, ) -> TipperReceivers | TipperBaseStations | None: """ Sub-class extension of :func:`~geoh5py.shared.entity.Entity.copy_from_extent`. """ indices = self.mask_by_extent(extent, inverse=inverse) if indices is None: return None new_entity = self.copy( parent=parent, copy_children=copy_children, clear_cache=clear_cache, mask=indices, **kwargs, ) return new_entity
@property def default_input_types(self) -> list[str]: """Choice of survey creation types.""" return self.__INPUT_TYPE @property def default_receiver_type(self): """ :return: Transmitter class """ return TipperReceivers @property def default_transmitter_type(self): """ :return: Transmitter class """ return type(None) @property def base_receiver_type(self): return Curve @property def base_transmitter_type(self): return Points @property def default_metadata(self) -> dict: """ :return: Default unique identifier """ return { "EM Dataset": { "Base stations": None, "Channels": [], "Input type": "Rx and base stations", "Property groups": [], "Receivers": None, "Survey type": "ZTEM", "Unit": "Hertz (Hz)", } } @property def default_units(self) -> list[str]: """Accepted time units. Must be one of "Seconds (s)", "Milliseconds (ms)", "Microseconds (us)" or "Nanoseconds (ns)" """ return self.__UNITS def _format_transmitter_ids(self, values, attributes): """ Format transmitter ids. :param values: Array of transmitter ids. :param attributes: Attributes dictionary for the new Data. """ if self.complement is not None and self.complement.tx_id_property is not None: attributes["entity_type"] = self.complement.tx_id_property.entity_type else: value_map = { ind: f"Base station {ind}" for ind in np.unique(values.astype(np.int32)) } value_map[0] = "Unknown" attributes.update( { "primitive_type": "REFERENCED", "value_map": value_map, "association": "VERTEX", } )
[docs] class TipperReceivers(TipperSurvey, Curve): # pylint: disable=too-many-ancestors """ A z-tipper EM survey object. """ _TYPE_UID = uuid.UUID("{0b639533-f35b-44d8-92a8-f70ecff3fd26}") __TYPE = "Receivers" _default_name = "Tipper rx" @property def complement(self): return self.base_stations @property def type(self): """Survey element type""" return self.__TYPE
[docs] class TipperBaseStations(TipperSurvey, Points): """ A z-tipper EM survey object. """ _TYPE_UID = uuid.UUID("{f495cd13-f09b-4a97-9212-2ea392aeb375}") __TYPE = "Base stations" _default_name = "Tipper base" _minimum_vertices = 1 @property def complement(self): return self.receivers @property def type(self): """Survey element type""" return self.__TYPE