"""
Absfuyu: Performance
--------------------
Performance Check
Version: 5.1.0
Date updated: 10/03/2025 (dd/mm/yyyy)
Feature:
--------
- measure_performance
- function_debug
- retry
- var_check
- Checker
"""
# Module level
# ---------------------------------------------------------------------------
__all__ = [
# Wrapper
"function_debug",
"measure_performance",
"retry",
# Class
"Checker",
]
# Library
# ---------------------------------------------------------------------------
import time
import tracemalloc
from collections.abc import Callable
from functools import wraps
from inspect import getsource
from typing import Any, ParamSpec, TypeVar
from absfuyu.core import deprecated, versionadded, versionchanged
from absfuyu.dxt import ListNoDunder
# Type
# ---------------------------------------------------------------------------
P = ParamSpec("P") # Parameter type
R = TypeVar("R") # Return type
# Function
# ---------------------------------------------------------------------------
[docs]
@versionadded("3.2.0")
def function_debug(f: Callable[P, R]) -> Callable[P, R]:
"""
Print the function signature and return value
Parameters
----------
func : Callable
Function to debug
Returns
-------
Callable
A decorated function
Usage
-----
Use this as a decorator (``@function_debug``)
Example:
--------
>>> @function_debug
>>> def test(a: int, b: int):
... return a + b
>>> test(6, 8)
Calling test(6, 8)
test() returned 14
"""
@wraps(f)
def wrapper(*args: P.args, **kwargs: P.kwargs) -> R:
# Get all parameters inputed
args_repr = [repr(a) for a in args]
kwargs_repr = [f"{k}={repr(v)}" for k, v in kwargs.items()]
signature = ", ".join(args_repr + kwargs_repr)
# Output
print(f"Calling {f.__name__}({signature})")
# logger.debug(f"Calling {func.__name__}({signature})")
value = f(*args, **kwargs)
print(f"{f.__name__}() returned {repr(value)}")
# logger.debug(f"{func.__name__}() returned {repr(value)}")
return value
return wrapper
[docs]
@versionadded("3.2.0")
def retry(retries: int, delay: float = 1):
"""
Attempt to call a function, if it fails, try again with a specified delay.
Parameters
----------
retries : int
The max amount of retries you want for the function call
delay : int
The delay (in seconds) between each function retry
Returns
-------
Callable
A decorated function
Usage
-----
Use this as a decorator (``@retry``)
Example:
--------
>>> @retry(retries=3, delay=1)
>>> def test() -> None:
... time.sleep(1)
... raise Exception("Function error")
>>> test()
Running (1): test()
Error: Exception('Function error') -> Retrying...
Running (2): test()
Error: Exception('Function error') -> Retrying...
Running (3): test()
Error: Exception('Function error').
"test()" failed after 3 retries.
"""
# Condition
if retries < 1 or delay <= 0:
raise ValueError("retries must be >= 1, delay must be >= 0")
def decorator(f: Callable[P, R]) -> Callable[P, R]:
@wraps(f)
def wrapper(*args: P.args, **kwargs: P.kwargs):
for i in range(1, retries + 1):
try:
print(f"Running ({i}): {f.__name__}()")
return f(*args, **kwargs)
except Exception as e:
# Break out of the loop if the max amount of retries is exceeded
if i == retries:
print(f"Error: {repr(e)}.")
print(f'"{f.__name__}()" failed after {retries} retries.')
break
else:
print(f"Error: {repr(e)} -> Retrying...")
time.sleep(
delay
) # Add a delay before running the next iteration
return wrapper
return decorator
# Class
# ---------------------------------------------------------------------------
[docs]
@deprecated("5.1.0", reason="Use `absfuyu.tools.inspector` instead")
class Checker:
"""
Check a variable
Parameters
----------
variable : Any
Variable that needed to check
Example:
--------
>>> test = "test"
>>> Checker(test).check()
{'name': None, 'value': 'test', 'class': <class 'str'>, 'id': ...}
"""
def __init__(self, variable: Any) -> None:
self.item_to_check = variable
def __str__(self) -> str:
return self.item_to_check.__str__() # type: ignore
def __repr__(self) -> str:
return f"{self.__class__.__name__}({self.item_to_check})"
@property
def name(self) -> Any | None:
"""``__name__`` of variable (if any)"""
try:
return self.item_to_check.__name__
except Exception:
return None
@property
def value(self) -> Any:
"""Value of the variable"""
return self.item_to_check
@property
def docstring(self) -> str | None:
"""``__doc__`` of variable (if any)"""
return self.item_to_check.__doc__ # type: ignore
@property
def class_(self) -> Any:
"""``class()`` of variable"""
return type(self.item_to_check)
@property
def id_(self) -> int:
"""``id()`` of variable"""
return id(self.item_to_check)
@property
def dir_(self) -> list[str]:
"""``dir()`` of variable"""
# return self.item_to_check.__dir__()
return ListNoDunder(self.item_to_check.__dir__())
@property
def source(self) -> str | None:
"""Source code of variable (if available)"""
try:
return getsource(self.item_to_check)
except Exception:
return None
[docs]
def check(self, full: bool = False) -> dict[str, Any]:
"""
Check the variable
Parameters
----------
full : bool
| ``True``: Shows full detail
| ``False``: Hides ``dir``, ``docstring`` and source code
| Default: ``False``
Returns
-------
dict[str, Any]
Check result
"""
out = {
"name": self.name,
"value": self.value,
"class": self.class_,
"id": self.id_,
}
if full:
out["dir"] = self.dir_
out["docs"] = self.docstring
out["source"] = self.source
return out