Source code for geoh5py.data.numeric_data
# Copyright (c) 2024 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
from abc import ABC, abstractmethod
from warnings import warn
import numpy as np
from .data import Data, PrimitiveTypeEnum
from .data_association_enum import DataAssociationEnum
[docs]
class NumericData(Data, ABC):
"""
Data container for floats values
"""
@property
@abstractmethod
def ndv(self):
"""No-data-value"""
[docs]
@abstractmethod
def format_type(self, values: np.ndarray) -> np.ndarray:
"""
Check if the type of values is valid and convert it to right dtype.
:param values: numpy array to modify.
:return: the formatted values.
"""
@property
def values(self) -> np.ndarray | None:
"""
:return: values: An array of values
"""
if getattr(self, "_values", None) is None:
values = self.workspace.fetch_values(self)
if isinstance(values, (np.ndarray, type(None))):
self._values = self.format_values(values)
return self._values
@values.setter
def values(self, values: np.ndarray | None):
if not isinstance(values, (np.ndarray, type(None))):
raise TypeError(
f"Input 'values' for {self} must be of type {np.ndarray} or None."
)
self._values = self.format_values(values)
self.workspace.update_attribute(self, "values")
[docs]
def format_length(self, values: np.ndarray) -> np.ndarray:
"""
Check for possible mismatch between the length of values
:param values: the values to check.
:return: the values with the right length.
"""
if self.n_values is None:
return values
if len(values) < self.n_values:
full_vector = np.ones(self.n_values, dtype=values.dtype) * self.nan_value
full_vector[: len(np.ravel(values))] = np.ravel(values)
return full_vector
if (
len(values) > self.n_values
and self.association is not DataAssociationEnum.OBJECT
):
raise ValueError(
f"Input 'values' of shape({self.n_values},) expected. "
f"Array of shape{values.shape} provided.)"
)
return values
[docs]
def format_values(self, values: np.ndarray | None) -> np.ndarray:
"""
Check for possible mismatch between the length of values
stored and the expected number of cells or vertices.
:param values: Array of values to check
:returns: values: An array of float values of length n_values or None
"""
if values is None:
return values
if not isinstance(values, np.ndarray):
raise TypeError("Input 'values' must be a numpy array.")
if values.ndim > 1:
values = np.ravel(values)
warn("Input 'values' converted to a 1D array.")
# change nan values to nan_value
values[np.isnan(values)] = self.nan_value
# check the length of the values
values = self.format_length(values)
# check the value type
values = self.format_type(values)
return values