Skip to content

Commit

Permalink
Introduce SingletonMeta metaclass and refactor singleton decorator …
Browse files Browse the repository at this point in the history
…to a metaclass instead of a function decorator
  • Loading branch information
glrs committed Dec 5, 2024
1 parent cb217b2 commit 6beb56c
Showing 1 changed file with 27 additions and 14 deletions.
41 changes: 27 additions & 14 deletions lib/core_utils/singleton_decorator.py
Original file line number Diff line number Diff line change
@@ -1,25 +1,38 @@
from typing import Any, Callable, Dict, Type, TypeVar
from typing import Any, Dict, Type

T = TypeVar("T")

class SingletonMeta(type):
"""
A metaclass that creates a singleton instance of a class.
"""

_instances: Dict[type, Any] = {}

def singleton(cls: Type[T]) -> Callable[..., T]:
"""Decorator to make a class a singleton.
def __call__(cls, *args: Any, **kwargs: Any) -> Any:
if cls not in cls._instances:
cls._instances[cls] = super().__call__(*args, **kwargs)
return cls._instances[cls]

Ensures that only one instance of the class is created. Subsequent
calls to the class will return the same instance.

def singleton(cls: Type[Any]) -> Type[Any]:
"""
Decorator to make a class a singleton by setting its metaclass to SingletonMeta.
Args:
cls (Type[T]): The class to be decorated.
cls (Type[Any]): The class to be decorated.
Returns:
Callable[..., T]: A function that returns the singleton instance of the class.
Type[Any]: The singleton class with SingletonMeta as its metaclass.
"""
instances: Dict[Type[T], T] = {}

def get_instance(*args: Any, **kwargs: Any) -> T:
if cls not in instances:
instances[cls] = cls(*args, **kwargs)
return instances[cls]
# Create a new class with SingletonMeta as its metaclass
class SingletonClass(cls, metaclass=SingletonMeta):
pass

# Preserve class metadata
SingletonClass.__name__ = cls.__name__
SingletonClass.__doc__ = cls.__doc__
SingletonClass.__module__ = cls.__module__
SingletonClass.__annotations__ = getattr(cls, "__annotations__", {})

return get_instance
return SingletonClass

0 comments on commit 6beb56c

Please sign in to comment.