Source code for geoh5py.shared.concatenation.object
# ''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
# 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
from typing import TYPE_CHECKING
from ...data import Data
from ...groups.property_group import GroupTypeEnum
from ...objects import ObjectBase
from ..utils import INV_KEY_MAP
from .concatenated import Concatenated
from .data import ConcatenatedData
from .property_group import ConcatenatedPropertyGroup
if TYPE_CHECKING:
from ..entity import Entity
from .concatenator import Concatenator
[docs]
class ConcatenatedObject(Concatenated, ObjectBase):
def __init__(self, **kwargs):
if kwargs.get("parent") is None:
raise UserWarning(
"Creating a concatenated object must have a parent "
"of type Concatenator."
)
self._parent: Concatenator
self._property_groups: list | None = None
super().__init__(**kwargs)
[docs]
def create_property_group(
self,
name=None,
property_group_type: GroupTypeEnum | str = GroupTypeEnum.INTERVAL,
**kwargs,
) -> ConcatenatedPropertyGroup:
"""
Create a new :obj:`~geoh5py.groups.property_group.PropertyGroup`.
:param name: Name of the property group.
:param property_group_type: Type of property group.
:param kwargs: Any arguments taken by the
:obj:`~geoh5py.groups.property_group.PropertyGroup` class.
:return: A new :obj:`~geoh5py.groups.property_group.PropertyGroup`
"""
if self._property_groups is not None and name in [
pg.name for pg in self._property_groups
]:
raise KeyError(f"A Property Group with name '{name}' already exists.")
prop_group = ConcatenatedPropertyGroup(
self, name=name, property_group_type=property_group_type, **kwargs
)
return prop_group
def _fetch_concatenated_children(self):
"""
Method to generate concatenated children.
"""
attr = self.concatenator.get_concatenated_attributes(self.uid).copy()
for key, value in attr.items():
if "Property:" in key:
child_data = self.workspace.get_entity(uuid.UUID(value))[0]
if child_data is None:
attributes: dict = self.concatenator.get_concatenated_attributes(
value
).copy()
attributes["parent"] = self
self.workspace.create_from_concatenation(attributes)
elif not isinstance(child_data, ConcatenatedPropertyGroup):
self.add_children([child_data])
[docs]
def get_entity(self, name: str | uuid.UUID) -> list[Entity | None]:
"""
Get a child :obj:`~geoh5py.data.data.Data` by name.
:param name: Name of the target child data.
:return: A list of children Data objects
"""
if not any(child for child in self.children if isinstance(child, Data)):
self._fetch_concatenated_children()
if isinstance(name, uuid.UUID):
entity_list = [child for child in self.children if child.uid == name]
else:
entity_list = [child for child in self.children if child.name == name]
if not entity_list:
return [None]
return entity_list
[docs]
def get_data_list(self, attribute="name"):
"""
Lazy loading of data names from concatenated attributes.
"""
data_list = [
attr.replace("Property:", "").replace("\u2044", "/")
for attr in self.concatenator.get_concatenated_attributes(self.uid)
if "Property:" in attr
]
return data_list
@property
def parent(self) -> Concatenator:
return self._parent
@parent.setter
def parent(self, parent: Concatenator):
if not hasattr(parent, "add_children"):
raise ValueError(
"The 'parent' of a concatenated Object must have an "
"'add_children' method."
)
parent.add_children([self])
self._parent = parent
@property
def property_groups(self) -> list | None:
if self._property_groups is None:
property_groups = self.concatenator.fetch_values(self, "property_group_ids")
if property_groups is None or isinstance(self, ConcatenatedData):
property_groups = []
for key in property_groups:
attributes = {
INV_KEY_MAP.get(key, key): val
for key, val in self.concatenator.get_concatenated_attributes(
key
).items()
}
self.fetch_property_group(**attributes, on_file=True)
property_groups = [
child
for child in self.children
if isinstance(child, ConcatenatedPropertyGroup)
]
if len(property_groups) > 0:
self._property_groups = property_groups
return self._property_groups
[docs]
def remove_children(
self, children: list | Concatenated | ConcatenatedPropertyGroup
):
"""
Remove children from object.
This method calls the ObjectBase parent class to remove children from the
object children, but also deletes the children from the workspace.
:param children: List of children to remove.
"""
if not isinstance(children, list):
children = [children]
for child in children:
if child not in self._children:
continue
self.concatenator.remove_entity(child)
self._children.remove(child)