# ''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
# 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/>. '
# ''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
# pylint: disable=R0904
from __future__ import annotations
import uuid
from abc import ABC, abstractmethod
from typing import TYPE_CHECKING
import numpy as np
from ..shared.utils import map_attributes, str2uuid, str_json_to_dict, to_list
from .entity_type import EntityType
if TYPE_CHECKING: # pragma: no cover
from .. import shared
from ..shared.entity_container import EntityContainer
from ..workspace import Workspace
DEFAULT_CRS = {"Code": "Unknown", "Name": "Unknown"}
[docs]
class Entity(ABC): # pylint: disable=too-many-instance-attributes
"""
Base entity class for Objects, Groups and Data.
:param entity_type: Entity type registered by the Workspace.
:param allow_delete: Entity can be deleted from the workspace.
:param allow_move: Entity can change :obj:`~geoh5py.shared.entity.Entity.parent`
:param allow_rename: Entity can change name
:param clipping_ids: List of clipping uuids
:param metadata: Metadata attached to the entity.
:param name: Name of the entity
:param on_file: Whether this Entity is already stored on
:obj:`~geoh5py.workspace.workspace.Workspace.h5file`.
:param partially_hidden: Whether this Entity is partially hidden.
:param parent: Parent entity.
:param public: Whether this Entity is accessible in the workspace tree and other parts
of the user interface in ANALYST.
:param uid: Unique identifier of the entity.
:param visible: Whether the Entity is visible in camera (checked in ANALYST object tree).
"""
_attribute_map: dict = {
"Allow delete": "allow_delete",
"Allow move": "allow_move",
"Allow rename": "allow_rename",
"Clipping IDs": "clipping_ids: list | None",
"ID": "uid",
"Name": "name",
"Partially hidden": "partially_hidden",
"Public": "public",
"Visible": "visible",
}
_default_name: str | None = None
def __init__( # pylint: disable=too-many-arguments
self,
*,
entity_type: shared.EntityType | None = None,
allow_delete: bool = True,
allow_move: bool = True,
allow_rename: bool = True,
clipping_ids: list[uuid.UUID] | None = None,
metadata: dict | None = None,
name: str | None = None,
on_file: bool = False,
partially_hidden: bool = False,
parent: EntityContainer | None = None,
public: bool = True,
uid: uuid.UUID | None = None,
visible: bool = True,
**kwargs,
):
self.on_file = on_file
self.name = name or self._default_name or type(self).__name__
self._entity_type = self.validate_entity_type(entity_type)
self.uid: uuid.UUID = uid or uuid.uuid4()
self.allow_delete = allow_delete
self.allow_move = allow_move
self.allow_rename = allow_rename
self.clipping_ids = clipping_ids
self.metadata = metadata
self.parent = parent or self.workspace.root
self.partially_hidden = partially_hidden
self.public = public
self.visible = visible
# TODO Deprecate in favor of explicit attribute setter
map_attributes(self, **kwargs)
self.workspace.register(self)
@property
def allow_delete(self) -> bool:
"""
:obj:`bool` Entity can be deleted from the workspace.
"""
return self._allow_delete
@allow_delete.setter
def allow_delete(self, value: bool):
self._allow_delete = value
if self.on_file:
self.workspace.update_attribute(self, "attributes")
@property
def allow_move(self) -> bool:
"""
:obj:`bool` Entity can change :obj:`~geoh5py.shared.entity.Entity.parent`
"""
return self._allow_move
@allow_move.setter
def allow_move(self, value: bool):
self._allow_move = value
if self.on_file:
self.workspace.update_attribute(self, "attributes")
@property
def allow_rename(self) -> bool:
"""
:obj:`bool` Entity can change name
"""
return self._allow_rename
@allow_rename.setter
def allow_rename(self, value: bool):
self._allow_rename = value
if self.on_file:
self.workspace.update_attribute(self, "attributes")
@property
def attribute_map(self) -> dict:
"""
:obj:`dict` Correspondence map between property names used in geoh5py and
geoh5.
"""
return self._attribute_map
@property
def clipping_ids(self) -> list[uuid.UUID] | None:
"""
List of clipping uuids.
"""
return self._clipping_ids
@clipping_ids.setter
def clipping_ids(self, value: list | None):
msg = (
"Input clipping_ids must be a list of uuid.UUID or None of Slicer objects. "
f"Provided value of type '{type(value)}'."
)
if value is not None:
verified_values = []
for val in to_list(value):
val = self.workspace.get_entity(str2uuid(val))[0]
if getattr(val, "_default_name", None) != "Slicer":
raise TypeError(msg)
verified_values.append(val.uid)
value = verified_values
self._clipping_ids = value
if self.on_file:
self.workspace.update_attribute(self, "attributes")
@property
def coordinate_reference_system(self) -> dict:
"""
Coordinate reference system attached to the entity.
"""
coordinate_reference_system = DEFAULT_CRS
if self.metadata is not None and "Coordinate Reference System" in self.metadata:
coordinate_reference_system = self.metadata[
"Coordinate Reference System"
].get("Current", DEFAULT_CRS)
return coordinate_reference_system
@coordinate_reference_system.setter
def coordinate_reference_system(self, value: dict):
# assert value is a dictionary containing "Code" and "Name" keys
if not isinstance(value, dict):
raise TypeError("Input coordinate reference system must be a dictionary")
if value.keys() != {"Code", "Name"}:
raise KeyError(
"Input coordinate reference system must only contain a 'Code' and 'Name' keys"
)
# get the actual coordinate reference system
coordinate_reference_system = {
"Current": value,
"Previous": self.coordinate_reference_system,
}
# update the metadata
self.metadata = {"Coordinate Reference System": coordinate_reference_system}
[docs]
@classmethod
def create(cls, workspace, **kwargs):
"""
Function to create an entity.
:param workspace: Workspace to be added to.
:param kwargs: List of keyword arguments defining the properties of a class.
:return entity: Registered Entity to the workspace.
"""
entity_type_kwargs = (
{"uid": kwargs.pop("entity_type_uid")}
if "entity_type_uid" in kwargs
else {}
)
new_object = workspace.create_entity(
cls,
entity=kwargs,
entity_type=entity_type_kwargs,
)
return new_object
[docs]
@classmethod
@abstractmethod
def default_type_uid(cls) -> uuid.UUID | None:
"""Abstract method to return the default type uid for the class."""
@property
@abstractmethod
def entity_type(self):
"""Abstract property to get the entity type of the entity."""
[docs]
@classmethod
def find_or_create_type(cls, workspace: Workspace, **kwargs) -> EntityType:
"""
Find or create a type instance for a given object class.
:param workspace: Target :obj:`~geoh5py.workspace.workspace.Workspace`.
:return: The ObjectType instance for the given object class.
"""
kwargs["entity_class"] = cls
return EntityType.find_or_create(workspace, **kwargs)
[docs]
@classmethod
def fix_up_name(cls, name: str) -> str:
"""If the given name is not a valid one, transforms it to make it valid
:return: a valid name built from the given name. It simply returns the given name
if it was already valid.
"""
# TODO: implement an actual fixup
# (possibly it has to be abstract with different implementations per Entity type)
return name
[docs]
@abstractmethod
def mask_by_extent(
self, extent: np.ndarray, inverse: bool = False
) -> np.ndarray | None:
"""
Get a mask array from coordinate extent.
:param extent: Bounding box extent coordinates defined by either:
- obj:`numpy.ndarray` of shape (2, 3)
3D coordinate: [[west, south, bottom], [east, north, top]]
- obj:`numpy.ndarray` of shape (2, 2)
Horizontal coordinates: [[west, south], [east, north]].
:param inverse: Return the complement of the mask extent. Default to False
:return: Array of bool defining the vertices or cell centers
within the mask extent, or None if no intersection.
"""
@property
def metadata(self) -> dict | None:
"""
Metadata attached to the entity.
To update the metadata, use the setter method.
To remove the metadata, set it to None.
"""
if getattr(self, "_metadata", None) is None:
value = self.workspace.fetch_metadata(self)
self._metadata = self.validate_metadata(value)
return self._metadata
@metadata.setter
def metadata(self, value: dict | np.ndarray | bytes | None):
self._metadata = self.validate_metadata(value)
if self.on_file:
self.workspace.update_attribute(self, "metadata")
[docs]
@abstractmethod
def validate_entity_type(self, entity_type):
"""
Validate the entity type.
"""
@property
def name(self) -> str:
"""
:obj:`str` Name of the entity
"""
return self._name
@name.setter
def name(self, new_name: str):
self._name = self.fix_up_name(new_name)
if self.on_file:
self.workspace.update_attribute(self, "attributes")
@property
def on_file(self) -> bool:
"""
Whether this Entity is already stored on
:obj:`~geoh5py.workspace.workspace.Workspace.h5file`.
"""
return self._on_file
@on_file.setter
def on_file(self, value: bool):
if not isinstance(value, bool):
raise TypeError("Attribute 'on_file' must be of type bool.")
self._on_file = value
@property
def parent(self):
return self._parent
@parent.setter
def parent(self, parent: EntityContainer):
current_parent: EntityContainer | None = getattr(self, "_parent", None)
self._parent = parent
parent.add_children([self])
if current_parent is not None and current_parent != self._parent:
current_parent.remove_children([self])
self.workspace.save_entity(self)
@property
def partially_hidden(self) -> bool:
"""
Whether this Entity is partially hidden.
"""
return self._partially_hidden
@partially_hidden.setter
def partially_hidden(self, value: bool):
self._partially_hidden = value
if self.on_file:
self.workspace.update_attribute(self, "attributes")
@property
def public(self) -> bool:
"""
Whether this Entity is accessible in the workspace tree and other parts
of the the user interface in ANALYST.
"""
return self._public
@public.setter
def public(self, value: bool):
self._public = value
if self.on_file:
self.workspace.update_attribute(self, "attributes")
@property
def uid(self) -> uuid.UUID:
return self._uid
@uid.setter
def uid(self, uid: str | uuid.UUID):
uid = str2uuid(uid)
if not isinstance(uid, uuid.UUID):
raise TypeError("Input uid must be a string or uuid.UUID.")
self._uid = uid
@property
def visible(self) -> bool:
"""
Whether the Entity is visible in camera (checked in ANALYST object tree).
"""
return self._visible
@visible.setter
def visible(self, value: bool):
self._visible = value
if self.on_file:
self.workspace.update_attribute(self, "attributes")
@property
def workspace(self) -> Workspace:
"""
:obj:`~geoh5py.workspace.workspace.Workspace` to which the Entity belongs to.
"""
return self.entity_type.workspace