Source code for objwatch.wrappers.abc_wrapper
# MIT License
# Copyright (c) 2025 aeeeeeep
from types import FrameType
from typing import Any, List, Tuple
from abc import ABC, abstractmethod
from ..constants import Constants
from ..event_handls import EventHandls
[docs]
class ABCWrapper(ABC):
"""
Abstract base class for function wrappers to extend tracing and logging functionality.
"""
[docs]
def __init__(self):
# Class attribute to specify the function for processing sequence elements
# Subclasses can override this to provide custom sequence processing
self.format_sequence_func = None
[docs]
@abstractmethod
def wrap_call(self, func_name: str, frame: FrameType) -> str:
"""
Process and format the function call information.
Args:
func_name (str): Name of the function being called.
frame (FrameType): The current stack frame.
Returns:
str: Formatted call message.
"""
pass
[docs]
@abstractmethod
def wrap_return(self, func_name: str, result: Any) -> str:
"""
Process and format the function return information.
Args:
func_name (str): Name of the function returning.
result (Any): The result returned by the function.
Returns:
str: Formatted return message.
"""
pass
[docs]
@abstractmethod
def wrap_upd(self, old_value: Any, current_value: Any) -> Tuple[str, str]:
"""
Process and format the update information of a variable.
Args:
old_value (Any): The old value of the variable.
current_value (Any): The new value of the variable.
Returns:
Tuple[str, str]: Formatted old and new values.
"""
pass
[docs]
def _extract_args_kwargs(self, frame: FrameType) -> Tuple[List[Any], dict]:
"""
Extract positional and keyword arguments from the current frame.
Args:
frame (FrameType): The current stack frame.
Returns:
Tuple[List[Any], dict]: Lists of positional and keyword arguments.
"""
args: List[Any] = []
kwargs: dict = {}
code = frame.f_code
arg_names = code.co_varnames[: code.co_argcount]
for name in arg_names:
if name in frame.f_locals:
args.append(frame.f_locals[name])
if code.co_flags & 0x08: # CO_VARKEYWORDS
kwargs = {k: v for k, v in frame.f_locals.items() if k not in arg_names and not k.startswith('_')}
return args, kwargs
[docs]
def _format_args_kwargs(self, args: List[Any], kwargs: dict) -> str:
"""
Format positional and keyword arguments into a string.
Args:
args (List[Any]): List of positional arguments.
kwargs (dict): Dictionary of keyword arguments.
Returns:
str: Formatted arguments string.
"""
formatted_args = [f"'{i}':{self._format_value(arg)}" for i, arg in enumerate(args)]
formatted_kwargs = [f"'{k}':{self._format_value(v)}" for k, v in kwargs.items()]
call_msg = ', '.join(filter(None, formatted_args + formatted_kwargs))
return call_msg
[docs]
def _format_value(self, value: Any, is_return: bool = False) -> str:
"""
Format a value into a string.
Args:
value (Any): The value to format.
is_return (bool): Flag indicating if the value is a return value.
Returns:
str: Formatted value string.
"""
if isinstance(value, Constants.LOG_ELEMENT_TYPES):
formatted = f"{value}"
elif isinstance(value, Constants.LOG_SEQUENCE_TYPES):
formatted_sequence = EventHandls.format_sequence(value, func=self.format_sequence_func)
if formatted_sequence:
formatted = f"{formatted_sequence}"
else:
formatted = f"(type){type(value).__name__}"
else:
try:
formatted = f"(type){value.__name__}" # type: ignore
except:
formatted = f"(type){type(value).__name__}"
if is_return:
if isinstance(value, Constants.LOG_SEQUENCE_TYPES) and formatted:
return f"[{formatted}]"
return f"{formatted}"
return formatted
[docs]
def _format_return(self, result: Any) -> str:
"""
Format the return value of a function.
Args:
result (Any): The result returned by the function.
Returns:
str: Formatted return message.
"""
return_msg = self._format_value(result, is_return=True)
return return_msg