Source code for toponetx.classes.reportviews

"""Module with views.

Such as:
HyperEdgeView, CellView, SimplexView, NodeView.
"""

from abc import ABC, abstractmethod
from collections.abc import Collection, Hashable, Iterable, Iterator, Sequence
from itertools import chain
from typing import Any, Generic, Literal, TypeVar

from toponetx.classes.cell import Cell
from toponetx.classes.complex import Atom
from toponetx.classes.hyperedge import HyperEdge
from toponetx.classes.path import Path
from toponetx.classes.simplex import Simplex
from toponetx.classes.simplex_trie import SimplexTrie

__all__ = [
    "AtomView",
    "CellView",
    "ColoredHyperEdgeView",
    "HyperEdgeView",
    "NodeView",
    "PathView",
    "SimplexView",
]

T_Atom = TypeVar("T_Atom", bound=Atom)


[docs] class AtomView(ABC, Generic[T_Atom]): """Abstract class representing a read-only view on a collection of atoms.""" @abstractmethod def __contains__(self, atom: Any) -> bool: """Check if a given element is in the view. Parameters ---------- atom : Any The element to check. Returns ------- bool Whether the element is in the view. """ @abstractmethod def __getitem__(self, atom: Any) -> dict: """Get the attributes of a given element. Parameters ---------- atom : Any The element of interest. Returns ------- dict The attributes associated with the element. Raises ------ KeyError If the element is not in the view. """ @abstractmethod def __iter__(self) -> Iterator[T_Atom]: """Iterate over all elements in the view. Returns ------- Iterator Iterator to iterate over all elements in the view. """ @abstractmethod def __len__(self) -> int: """Return the number of elements in the view. Returns ------- int The number of elements in the view. """
[docs] class CellView(AtomView[Cell]): """A CellView class for cells of a CellComplex.""" # Dictionary to hold cells, with keys being the tuple that defines the cell, and # values being dictionaries of cell objects with different attributes _cells: dict[tuple[Hashable, ...], dict[int, Cell]]
[docs] def __init__(self) -> None: self._cells = {}
def __getitem__(self, cell: Any) -> dict[Hashable, Any]: """Return the attributes of a given cell. Parameters ---------- cell : Any The cell of interest. Returns ------- dict[Hashable, Any] The attributes associated with the cell. Raises ------ KeyError If the cell is not in the cell dictionary. """ if isinstance(cell, Cell): if cell.elements not in self._cells: raise KeyError( f"cell {cell!r} is not in the cell dictionary", ) # If there is only one cell with these elements, return its attributes if len(self._cells[cell.elements]) == 1: k = next(iter(self._cells[cell.elements].keys())) return self._cells[cell.elements][k]._attributes # If there are multiple cells with these elements, return the attributes of all cells return [ self._cells[cell.elements][c]._attributes for c in self._cells[cell.elements] ] # If a tuple or list is passed in, assume it represents a cell if isinstance(cell, Iterable): cell = tuple(cell) if cell in self._cells: if len(self._cells[cell]) == 1: k = next(iter(self._cells[cell].keys())) return self._cells[cell][k]._attributes return [self._cells[cell][c]._attributes for c in self._cells[cell]] raise KeyError(f"cell {cell} is not in the cell dictionary") raise TypeError("Input must be a tuple, list or a cell.")
[docs] def raw(self, cell: tuple | list | Cell) -> Cell | list[Cell]: """Index the raw cell objects analogous to the overall index of CellView. Parameters ---------- cell : tuple, list, or cell The cell of interest. Returns ------- Cell or list of Cells The raw Cell objects. If more than one cell with the same boundary exists, returns a list; otherwise a single cell. Raises ------ KeyError If the cell is not in the cell dictionary. """ if isinstance(cell, Cell): if cell.elements not in self._cells: raise KeyError(f"cell {cell!r} is not in the cell dictionary") # If there is only one cell with these elements, return its attributes if len(self._cells[cell.elements]) == 1: k = next(iter(self._cells[cell.elements].keys())) return self._cells[cell.elements][k] # If there are multiple cells with these elements, return the attributes of all cells return [self._cells[cell.elements][c] for c in self._cells[cell.elements]] # If a tuple or list is passed in, assume it represents a cell if isinstance(cell, tuple | list): cell = tuple(cell) if cell in self._cells: if len(self._cells[cell]) == 1: k = next(iter(self._cells[cell].keys())) return self._cells[cell][k] return [self._cells[cell][c] for c in self._cells[cell]] raise KeyError(f"cell {cell} is not in the cell dictionary") raise TypeError("Input must be a tuple, list or a cell.")
def __len__(self) -> int: """Return the number of cells in the cell view. Returns ------- int The number of cells in the cell view. """ return sum(len(self._cells[cell]) for cell in self._cells) def __iter__(self) -> Iterator[Cell]: """Iterate over all cells in the cell view. Returns ------- Iterator Iterator to iterate over all cells in the cell view. """ return iter( [ self._cells[cell][key] for cell in self._cells for key in self._cells[cell] ] ) def __contains__(self, atom: Any) -> bool: """Check if a given element is in the cell view. Parameters ---------- atom : Any The element to check. Returns ------- bool Whether the element is in the cell view. """ if not isinstance(atom, Cell | tuple | list): return False atom = Cell(atom) return any(atom.is_homotopic_to(x) for x in self._cells) def __repr__(self) -> str: """Return a string representation of the cell view. Returns ------- str The __repr__ representation of the cell view. """ return f"CellView({[self._cells[cell][key] for cell in self._cells for key in self._cells[cell]] })" def __str__(self) -> str: """Return a string representation of the cell view. Returns ------- str The __str__ representation of the cell view. """ return f"CellView({[self._cells[cell][key] for cell in self._cells for key in self._cells[cell]]})"
[docs] class ColoredHyperEdgeView(AtomView): """A class for viewing the cells/hyperedges of a colored hypergraph. Provides methods for accessing, and retrieving information about the cells/hyperedges of a complex. Examples -------- >>> hev = tnx.ColoredHyperEdgeView() """
[docs] def __init__(self) -> None: self.hyperedge_dict = {}
def __getitem__(self, atom: Any) -> dict[Hashable, Any]: """Return the user-defined attributes associated with the given hyperedge. Parameters ---------- atom : Any The hyperedge for which to return the associated user-defined attributes. Returns ------- dict[Hashable, Any] The user-defined attributes associated with the given atom. Raises ------ KeyError If the hyperedge does not exist. """ if isinstance(atom, Hashable) and not isinstance(atom, Collection): atom = (atom,) if len(atom) == 0: raise KeyError(f"Hyperedge {atom} is not in the complex.") if len(atom) == 2: if isinstance(atom, HyperEdge): hyperedge_elements = atom.elements key = 0 elif isinstance(atom[0], Iterable) and isinstance(atom[1], int): hyperedge_elements_ = atom[0] if not isinstance(hyperedge_elements_, HyperEdge): hyperedge_elements, key = atom else: _, key = atom hyperedge_elements = hyperedge_elements_.elements else: hyperedge_elements = atom key = 0 else: hyperedge_elements = atom key = 0 if not isinstance(hyperedge_elements, Iterable) or len(hyperedge_elements) == 0: raise KeyError(f"Hyperedge {atom} is not in the complex.") for i in self.allranks: if frozenset(hyperedge_elements) in self.hyperedge_dict[i]: return self.hyperedge_dict[i][frozenset(hyperedge_elements)][key] raise KeyError(f"Hyperedge {atom} is not in the complex.") @property def shape(self) -> tuple[int, ...]: """Compute shape. Returns ------- tuple[int, ...] The shape of the ColoredHyperEdge. """ shape = [] for i in self.allranks: sm = sum(len(self.hyperedge_dict[i][k]) for k in self.hyperedge_dict[i]) shape.append(sm) return tuple(shape) def __len__(self) -> int: """Compute the number of nodes. Returns ------- int The number of nodes in the ColoredHyperEdge. """ return sum(self.shape[1:]) def __iter__(self) -> Iterator: """Iterate over the hyperedges. Returns ------- Iterator The iterator to iterate over the hyperedges. """ lst = [] for r in self.hyperedge_dict: if r == 0: continue lst.extend( (he, k) for he in self.hyperedge_dict[r] for k in self.hyperedge_dict[r][he] ) return iter(lst) def __contains__(self, atom: Any) -> bool: """Check if hyperedge is in the hyperedges. Parameters ---------- atom : Any The hyperedge to check. Returns ------- bool Return `True` if the hyperedge is contained within the hyperedges. Notes ----- Assumption of input here hyperedge = ( elements of hyperedge, key of hyperedge) """ if isinstance(atom, Hashable) and not isinstance(atom, Collection): atom = (atom,) if len(atom) == 0: return False if len(atom) == 2: if isinstance(atom, HyperEdge): hyperedge_elements = atom.elements key = 0 elif isinstance(atom[0], Iterable) and isinstance(atom[1], int): hyperedge_elements_ = atom[0] if not isinstance(hyperedge_elements_, HyperEdge): hyperedge_elements, key = atom else: _, key = atom hyperedge_elements = hyperedge_elements_.elements else: hyperedge_elements = atom key = 0 else: hyperedge_elements = atom key = 0 if not isinstance(hyperedge_elements, Iterable) or len(hyperedge_elements) == 0: return False for i in self.allranks: if frozenset(hyperedge_elements) in self.hyperedge_dict[i]: return key in self.hyperedge_dict[i][frozenset(hyperedge_elements)] return False def __repr__(self) -> str: """Return string representation of hyperedges. Returns ------- str The __repr__ string representation of the hyperedges. """ return f"ColoredHyperEdgeView({[(tuple(x[0]),x[1]) for x in self]})" def __str__(self) -> str: """Return string representation of hyperedges. Returns ------- str The __str__ string representation of the hyperedges. """ return f"ColoredHyperEdgeView({[(tuple(x[0]),x[1]) for x in self]})"
[docs] def skeleton(self, rank: int, store_hyperedge_key: bool = True): """Skeleton of the complex. Parameters ---------- rank : int Rank of the skeleton. store_hyperedge_key : bool, default=True Whether to return the hyperedge key or not. Returns ------- list of frozensets The skeleton of rank `rank`. """ if rank not in self.hyperedge_dict: return [] if store_hyperedge_key: return sorted( [ (he, k) for he in self.hyperedge_dict[rank] for k in self.hyperedge_dict[rank][he] ] ) return sorted( [ he for he in self.hyperedge_dict[rank] for k in self.hyperedge_dict[rank][he] ] )
[docs] def get_rank(self, edge): """Get the rank of a given hyperedge. Parameters ---------- edge : Iterable, Hashable or ColoredHyperEdge The edge for which to get the rank. Returns ------- int The rank of the given colored hyperedge. """ if isinstance(edge, HyperEdge): if len(edge) == 0: return 0 for i in list(self.allranks): if frozenset(edge.elements) in self.hyperedge_dict[i]: return i raise KeyError(f"hyperedge {edge.elements} is not in the complex") if isinstance(edge, str): if frozenset({edge}) in self.hyperedge_dict[0]: return 0 raise KeyError(f"hyperedge {frozenset({edge})} is not in the complex") if isinstance(edge, Iterable): if len(edge) == 0: return 0 for i in list(self.allranks): if frozenset(edge) in self.hyperedge_dict[i]: return i raise KeyError(f"hyperedge {edge} is not in the complex") if isinstance(edge, Hashable) and not isinstance(edge, Iterable): if frozenset({edge}) in self.hyperedge_dict[0]: return 0 raise KeyError(f"hyperedge {frozenset({edge})} is not in the complex") return None
@property def allranks(self) -> list[int]: """All ranks. Returns ------- list[int] The sorted list of all ranks. """ return sorted(self.hyperedge_dict.keys())
[docs] class HyperEdgeView(AtomView): """A class for viewing the cells/hyperedges of a combinatorial complex. Provides methods for accessing, and retrieving information about the cells/hyperedges of a complex. Examples -------- >>> hev = tnx.HyperEdgeView() """
[docs] def __init__(self) -> None: self.hyperedge_dict = {}
@staticmethod def _to_frozen_set(hyperedge): """Convert a hyperedge into a frozen set. Parameters ---------- hyperedge : HyperEdge | Iterable | Hashable The hyperedge that is to be converted to a frozen set. Returns ------- frozenset Returns a frozenset of the elements contained in the hyperedge. """ if isinstance(hyperedge, HyperEdge): hyperedge_ = hyperedge.elements elif isinstance(hyperedge, Iterable): hyperedge_ = frozenset(hyperedge) elif isinstance(hyperedge, Hashable) and not isinstance(hyperedge, Iterable): hyperedge_ = frozenset([hyperedge]) return hyperedge_ def __getitem__(self, hyperedge: Any) -> dict: """Get item. Parameters ---------- hyperedge : Hashable or HyperEdge DESCRIPTION. Returns ------- dict or list or dicts Return dict of attributes associated with that hyperedges. """ hyperedge_ = HyperEdgeView._to_frozen_set(hyperedge) rank = self.get_rank(hyperedge_) return self.hyperedge_dict[rank][hyperedge_] @property def shape(self) -> tuple[int, ...]: """Compute shape. Returns ------- tuple[int, ...] A tuple representing the shape of the hyperedge. """ return tuple(len(self.hyperedge_dict[i]) for i in self.allranks) def __len__(self) -> int: """Compute the number of nodes. Returns ------- int The number of nodes present in the HyperEdgeView. """ return sum(self.shape) def __iter__(self) -> Iterator[HyperEdge]: """Iterate over the hyperedges. Returns ------- Iterator Iterator object over the hyperedges. """ return chain.from_iterable(self.hyperedge_dict.values()) def __contains__(self, atom: Collection) -> bool: """Check if e is in the hyperedges. Parameters ---------- atom : Collection The hyperedge that needs to be checked for containership in the HyperEdgeView. Returns ------- bool Returns `True` if the hyperedge e is contained within the HyperEdgeView, else return `False`. """ if len(self.hyperedge_dict) == 0: return False all_ranks = self.allranks if isinstance(atom, HyperEdge): if len(atom) == 0: return False for i in all_ranks: if frozenset(atom.elements) in self.hyperedge_dict[i]: return True return False if isinstance(atom, Iterable): if len(atom) == 0: return False return any(frozenset(atom) in self.hyperedge_dict[i] for i in all_ranks) if isinstance(atom, Hashable): return frozenset({atom}) in self.hyperedge_dict[0] return None def __repr__(self) -> str: """Return string representation of hyperedges. Returns ------- str The __repr__ string representation of HyperEdgeView. """ return f"HyperEdgeView({[tuple(x) for x in self]})" def __str__(self) -> str: """Return string representation of hyperedges. Returns ------- str The __str__ string representation of HyperEdgeView. """ return f"HyperEdgeView({[tuple(x) for x in self]})"
[docs] def skeleton( self, rank: int, level: Literal[ "equal", "upper", "up", "lower", "down", "uppereq", "upeq", "lowereq", "downeq", ] = "equal", ): """Skeleton of the complex. Parameters ---------- rank : int Rank of the skeleton. level : str, default="equal" Level of the skeleton. Returns ------- list of frozensets The skeleton of rank `rank`. """ if level == "equal": if rank in self.allranks: return sorted(self.hyperedge_dict[rank].keys()) return [] if level in {"upper", "up"}: elements = [] for rank_i in self.allranks: if rank_i > rank: elements = elements + list(self.hyperedge_dict[rank_i].keys()) return sorted(elements) if level in {"lower", "down"}: elements = [] for rank_i in self.allranks: if rank_i < rank: elements = elements + list(self.hyperedge_dict[rank_i].keys()) return sorted(elements) if level in {"uppereq", "upeq"}: elements = [] for rank_i in self.allranks: if rank_i >= rank: elements = elements + list(self.hyperedge_dict[rank_i].keys()) return sorted(elements) if level in {"lowereq", "downeq"}: elements = [] for rank_i in self.allranks: if rank_i <= rank: elements = elements + list(self.hyperedge_dict[rank_i].keys()) return sorted(elements) raise ValueError( "level must be 'equal', 'uppereq', 'lowereq', 'upeq', 'downeq', 'uppereq', 'lower', 'up', or 'down'" )
[docs] def get_rank(self, edge): """Get the rank of a hyperedge. Parameters ---------- edge : Iterable, Hashable or HyperEdge The edge for which to get the rank. Returns ------- int The rank of the given hyperedge. """ if isinstance(edge, HyperEdge): if len(edge) == 0: return 0 for i in list(self.allranks): if frozenset(edge.elements) in self.hyperedge_dict[i]: return i raise KeyError(f"hyperedge {edge.elements} is not in the complex") if isinstance(edge, str): if frozenset({edge}) in self.hyperedge_dict[0]: return 0 raise KeyError(f"hyperedge {frozenset({edge})} is not in the complex") if isinstance(edge, Iterable): if len(edge) == 0: return 0 for i in list(self.allranks): if frozenset(edge) in self.hyperedge_dict[i]: return i raise KeyError(f"hyperedge {edge} is not in the complex") if isinstance(edge, Hashable) and not isinstance(edge, Iterable): if frozenset({edge}) in self.hyperedge_dict[0]: return 0 raise KeyError(f"hyperedge {frozenset({edge})} is not in the complex") return None
@property def allranks(self): """All ranks. Returns ------- list[hashable] The sorted list of all ranks. """ return sorted(self.hyperedge_dict.keys()) def _get_lower_rank(self, rank): """Get a lower rank compared to given rank. Parameters ---------- rank : int The rank to be used to get a lower rank. Returns ------- int A rank below the current rank available in the HyperEdgeView. """ if len(self.allranks) == 0: return -1 ranks = sorted(self.allranks) if rank <= min(ranks) or rank >= max(ranks): return -1 return ranks[ranks.index(rank) - 1] def _get_higher_rank(self, rank): """Get a higher rank compared to given rank. Parameters ---------- rank : int The rank to be used to get a higher rank. Returns ------- int A rank above the current rank available in the HyperEdgeView. """ if len(self.allranks) == 0: return -1 ranks = sorted(self.allranks) if rank <= min(ranks) or rank >= max(ranks): return -1 return ranks[ranks.index(rank) + 1]
[docs] class SimplexView(AtomView[Simplex]): """Simplex View class. The SimplexView class is used to provide a view/read only information into a subset of the nodes in a simplex. These classes are used in conjunction with the SimplicialComplex class for view/read only purposes for simplices in simplicial complexes. Parameters ---------- simplex_trie : SimplexTrie A SimplexTrie instance containing the simplices in the simplex view. """
[docs] def __init__(self, simplex_trie: SimplexTrie) -> None: self._simplex_trie = simplex_trie
def __getitem__(self, simplex: Any) -> dict[Hashable, Any]: """Get the dictionary of attributes associated with the given simplex. Parameters ---------- simplex : tuple, list or Simplex A tuple or list of nodes representing a simplex. Returns ------- dict A dictionary of attributes associated with the given simplex. Raises ------ KeyError If the simplex is not in the simplex view. """ if isinstance(simplex, Hashable) and not isinstance(simplex, Iterable): simplex = (simplex,) else: simplex = tuple(sorted(simplex)) node = self._simplex_trie.find(simplex) if node is None: raise KeyError(f"Simplex {simplex} is not in the simplex view.") return node.attributes @property def shape(self) -> tuple[int, ...]: """Return the number of simplices in each dimension. Returns ------- tuple of ints A tuple of integers representing the number of simplices in each dimension. """ return tuple(self._simplex_trie.shape) def __len__(self) -> int: """Return the number of simplices in the SimplexView instance. Returns ------- int Returns the number of simplices in the SimplexView instance. """ return len(self._simplex_trie) def __iter__(self) -> Iterator: """Return an iterator over all simplices in the simplex view. Returns ------- Iterator Returns an iterator over all simplices in the simplex view. """ return iter(node.simplex for node in self._simplex_trie) def __contains__(self, atom: Any) -> bool: """Check if a simplex is in the simplex view. Parameters ---------- atom : Any The simplex to be checked for membership in the simplex view. Returns ------- bool True if the simplex is in the simplex view, False otherwise. Examples -------- Check if a node is in the simplex view: >>> view = tnx.SimplexView() >>> view.faces_dict.append({frozenset({1}): {"weight": 1}}) >>> view.max_dim = 0 >>> 1 in view True >>> 2 in view False Check if a simplex is in the simplex view: >>> view.faces_dict.append({frozenset({1, 2}): {"weight": 1}}) >>> view.max_dim = 1 >>> {1, 2} in view True >>> {1, 3} in view False >>> {1, 2, 3} in view False """ if isinstance(atom, Hashable) and not isinstance(atom, Iterable): atom = (atom,) else: atom = tuple(sorted(atom)) return atom in self._simplex_trie def __repr__(self) -> str: """Return string representation that can be used to recreate it. Returns ------- str Returns the __repr__ representation of the object. """ return f"SimplexView({[tuple(simplex.elements) for simplex in self._simplex_trie]})" def __str__(self) -> str: """Return detailed string representation of the simplex view. Returns ------- str Returns the __str__ representation of the object. """ return f"SimplexView({[tuple(simplex.elements) for simplex in self._simplex_trie]})"
[docs] class NodeView: """Node view class. Parameters ---------- nodes : dict[Hashable, Any] A dictionary of nodes with their attributes. cell_type : type The type of the cell. colored_nodes : bool, optional Whether or not the nodes are colored. """
[docs] def __init__( self, nodes: dict[Hashable, Any], cell_type, colored_nodes: bool = False ) -> None: self.nodes = nodes if cell_type is None: raise ValueError("cell_type cannot be None") self.cell_type = cell_type self.colored_nodes = colored_nodes
def __repr__(self) -> str: """Return string representation of nodes. Returns ------- str Returns the __repr__ representation of the object. """ return f"NodeView({list(map(tuple, self.nodes.keys()))})" def __iter__(self) -> Iterator: """Return an iterator over all nodes in the node view. Returns ------- Iterator Returns an iterator over all nodes in the node view. """ return iter(self.nodes) def __getitem__(self, cell): """Get item. Parameters ---------- cell : tuple list or AbstractCell or Simplex A cell. Returns ------- dict or list Dict of attributes associated with that cells. """ if isinstance(cell, Iterable): cell = frozenset(cell) if cell in self.nodes: if self.colored_nodes: return self.nodes[cell][0] return self.nodes[cell] elif isinstance(cell, Hashable): if cell in self: if self.colored_nodes: return self.nodes[frozenset({cell})][0] return self.nodes[frozenset({cell})] raise KeyError(f"input {cell} is not in the node set of the complex") def __len__(self) -> int: """Compute the number of nodes. Returns ------- int Returns the number of nodes. """ return len(self.nodes) def __contains__(self, e) -> bool: """Check if e is in the nodes. Parameters ---------- e : Hashable | Iterable The node to check for. Returns ------- bool Return `True` if e is contained in NodeView, else Return `False`. """ if isinstance(e, Hashable) and not isinstance(e, self.cell_type): return frozenset({e}) in self.nodes if isinstance(e, self.cell_type): return e.elements in self.nodes if isinstance(e, Iterable): if len(e) == 1: return frozenset(e) in self.nodes return None return False
[docs] class PathView(AtomView[Path]): """Path view class."""
[docs] def __init__(self) -> None: self.max_dim = -1 self.faces_dict = []
def __getitem__(self, path: Any) -> dict: """Get the dictionary of attributes associated with the given path. Parameters ---------- path : Any A tuple or list of nodes representing a path. It can also be a Path object. It can also be a single node represented by int or str. Returns ------- dict or list or dict A dictionary of attributes associated with the given path. Raises ------ KeyError If the path is not in this view. """ if isinstance(path, Path): path = path.elements if isinstance(path, Hashable) and not isinstance(path, Iterable): path = (path,) path = tuple(path) if path in self.faces_dict[len(path) - 1]: return self.faces_dict[len(path) - 1][path] raise KeyError(f"input {path} is not in the path dictionary") def __iter__(self) -> Iterator: """Return an iterator over all paths in the path view. Returns ------- Iterator Iterator over all paths in the paths view. """ return chain.from_iterable(self.faces_dict) def __contains__(self, atom: Any) -> bool: """Check if a path is in the path view. Parameters ---------- atom : Any The path to be checked for membership in the path view. Returns ------- bool True if the path is in the path view, False otherwise. """ if isinstance(atom, Sequence): atom = tuple(atom) if not 0 < len(atom) <= self.max_dim + 1: return False return atom in self.faces_dict[len(atom) - 1] if isinstance(atom, Path): atom = atom.elements if not 0 < len(atom) <= self.max_dim + 1: return False return atom in self.faces_dict[len(atom) - 1] if isinstance(atom, Hashable): return (atom,) in self.faces_dict[0] return False def __len__(self) -> int: """Return the number of simplices in the SimplexView instance. Returns ------- int Returns the number of simplices in the SimplexView instance. """ return sum(self.shape) def __repr__(self) -> str: """Return string representation that can be used to recreate it. Returns ------- str Returns the __repr__ representation of the object. """ all_paths: list[tuple[int | str, ...]] = [] for i in range(len(self.faces_dict)): all_paths += [tuple(j) for j in self.faces_dict[i]] return f"PathView({all_paths})" def __str__(self) -> str: """Return detailed string representation of the path view. Returns ------- str Returns the __str__ representation of the object. """ all_paths: list[tuple[int | str, ...]] = [] for i in range(len(self.faces_dict)): all_paths += [tuple(j) for j in self.faces_dict[i]] return f"PathView({all_paths})" @property def shape(self) -> tuple[int, ...]: """Return the number of paths in each dimension. Returns ------- tuple of ints A tuple of integers representing the number of paths in each dimension. """ return tuple(len(self.faces_dict[i]) for i in range(len(self.faces_dict)))