metric / deprecation_utils.py
Elron's picture
Upload folder using huggingface_hub
24df49f verified
raw
history blame
4.38 kB
import functools
import warnings
from .error_utils import UnitxtWarning
from .settings_utils import get_constants, get_settings
constants = get_constants()
settings = get_settings()
class DeprecationError(Exception):
"""Custom exception for deprecated versions."""
pass
def compare_versions(version1, version2):
"""Compare two semantic versioning strings and determine their relationship.
Parameters:
version1 (str):
The first version string to compare.
version2 (str):
The second version string to compare.
Returns:
int: -1 if version1 < version2, 1 if version1 > version2, 0 if equal.
Example:
.. code-block:: text
>>> compare_versions("1.2.0", "1.2.3")
-1
>>> compare_versions("1.3.0", "1.2.8")
1
>>> compare_versions("1.0.0", "1.0.0")
0
"""
parts1 = [int(part) for part in version1.split(".")]
parts2 = [int(part) for part in version2.split(".")]
length_difference = len(parts1) - len(parts2)
if length_difference > 0:
parts2.extend([0] * length_difference)
elif length_difference < 0:
parts1.extend([0] * (-length_difference))
for part1, part2 in zip(parts1, parts2):
if part1 < part2:
return -1
if part1 > part2:
return 1
return 0
def depraction_wrapper(obj, version, alt_text):
"""A wrapper function for deprecation handling, issuing warnings or errors based on version comparison.
Args:
obj (callable): The object to be wrapped, typically a function or class method.
version (str): The version at which the object becomes deprecated.
alt_text (str): Additional text to display, usually suggests an alternative.
Returns:
callable: A wrapped version of the original object that checks for deprecation.
"""
@functools.wraps(obj)
def wrapper(*args, **kwargs):
if constants.version < version:
if settings.default_verbosity in ["debug", "info", "warning"]:
warnings.warn(
f"{obj.__name__} is deprecated.{alt_text}",
DeprecationWarning,
stacklevel=2,
)
elif constants.version >= version:
raise DeprecationError(f"{obj.__name__} is no longer supported.{alt_text}")
return obj(*args, **kwargs)
return wrapper
def deprecation(version, alternative=None, msg=None):
"""Decorator for marking functions or class methods as deprecated.
Args:
version (str): The version at which the function or method becomes deprecated.
alternative (str, optional): Suggested alternative to the deprecated functionality.
msg (str, optional): Additional message regarding the deprecation reason or alternatives.
Returns:
callable: A decorator that can be applied to functions or class methods.
"""
def decorator(obj):
alt_text = f" Use {alternative} instead." if alternative is not None else ""
alt_text += msg if msg is not None else ""
if callable(obj):
func = obj
elif hasattr(obj, "__init__"):
func = obj.__init__
else:
raise ValueError("Unsupported object type for deprecation.")
return depraction_wrapper(func, version, alt_text)
return decorator
def init_warning(msg=""):
# Decorator that raises warning when class is initialized
def decorator(initiated_class):
UnitxtWarning(msg)
return initiated_class
return decorator
def warn_on_call(warning_type=UserWarning, msg=""):
def decorator(obj):
if isinstance(obj, type):
original_init = obj.__init__
@functools.wraps(original_init)
def new_init(self, *args, **kwargs):
warnings.warn(msg, warning_type, stacklevel=2)
original_init(self, *args, **kwargs)
obj.__init__ = new_init
return obj
if callable(obj):
@functools.wraps(obj)
def wrapper(*args, **kwargs):
warnings.warn(msg, warning_type, stacklevel=2)
return obj(*args, **kwargs)
return wrapper
raise TypeError("This decorator can only be applied to classes or functions.")
return decorator