Source code for icclim._core.model.registry

"""Contain the Registry class, a fancy enum replacement."""

from __future__ import annotations

from abc import ABC
from copy import deepcopy
from typing import Generic, TypeVar

from icclim.exception import InvalidIcclimArgumentError

T = TypeVar("T")


[docs] class Registry(Generic[T], ABC): """ Registry classes acts as fancy enums. It allows to easily store and find constants of similar type. Registries are namespaces, so there is no need to instantiate it or any of its subclasses, every item is a class attribute. Notes ----- Registries are not meant to store large collections, they are just fancy lookup tables for items with aliases and no case sensitivity. """ _item_class: type # runtime type for the generic `T` @classmethod
[docs] def lookup(cls, query: T | str) -> T: """ Look up an item in the registry. Parameters ---------- query : T or str The item to look up. It can be either an instance of the item class or a string. Returns ------- T The found item. Raises ------ InvalidIcclimArgumentError If the item is not found in the registry. Notes ----- This method performs a case-insensitive lookup. It first checks if the query is an instance of the item class, and if so, returns a deep copy of the query. """ if isinstance(query, cls._item_class): return deepcopy(query) if isinstance(query, str): q = query.upper() for key, item in cls.catalog().items(): if q == key.upper() or q in cls.get_item_aliases(item): return deepcopy(item) msg = ( f"Unknown {cls._item_class.__qualname__}: '{query}'. " f"Use one of {cls.every_aliases()}." ) raise InvalidIcclimArgumentError(msg)
@classmethod
[docs] def lookup_no_error(cls, query: T | str) -> T | None: """ Also look up an item in the registry, but return None if not found. Parameters ---------- query : T or str The item to look up. It can be either an instance of the item class or a string. Returns ------- T or None The found item, or None if not found. """ try: return cls.lookup(query) except InvalidIcclimArgumentError: return None
@classmethod
[docs] def every_aliases(cls) -> list[T]: """ Return a list of all aliases for items in the registry. Returns ------- list[T] A list of all aliases for items in the registry. """ return list(map(cls.get_item_aliases, list(cls.catalog().values())))
@staticmethod
[docs] def get_item_aliases(item: T) -> list[str]: """ Get the aliases for the given item. Parameters ---------- item : T The item to get aliases for. Returns ------- list[str] A list of aliases for the item. Notes ----- Should be overridden in subclasses. """ return [item.name.upper()]
@classmethod
[docs] def catalog(cls) -> dict[str, T]: """ Return a dictionary of all items in the registry. Returns ------- dict[str, T] A dictionary containing all items in the registry, where the keys are the item names and the values are the item instances. """ return {k: v for k, v in cls.__dict__.items() if isinstance(v, cls._item_class)}
@classmethod
[docs] def values(cls) -> list[T]: """ Return a list of all items in the registry. Returns ------- list[T] A list containing all items in the registry. """ return [v for v in cls.__dict__.values() if isinstance(v, cls._item_class)]