deprecated.py 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. # -*- coding: utf-8 -*-
  2. """Deprecation utilities."""
  3. import warnings
  4. from typing import Any, Callable, Optional
  5. from vine.utils import wraps
  6. from celery.exceptions import CPendingDeprecationWarning, CDeprecationWarning
  7. __all__ = ['Callable', 'Property', 'warn']
  8. PENDING_DEPRECATION_FMT = """
  9. {description} is scheduled for deprecation in \
  10. version {deprecation} and removal in version v{removal}. \
  11. {alternative}
  12. """
  13. DEPRECATION_FMT = """
  14. {description} is deprecated and scheduled for removal in
  15. version {removal}. {alternative}
  16. """
  17. def warn(description: Optional[str]=None,
  18. deprecation: Optional[str]=None,
  19. removal: Optional[str]=None,
  20. alternative: Optional[str]=None,
  21. stacklevel: int=2) -> None:
  22. """Warn of (pending) deprecation."""
  23. ctx = {'description': description,
  24. 'deprecation': deprecation, 'removal': removal,
  25. 'alternative': alternative}
  26. if deprecation is not None:
  27. w = CPendingDeprecationWarning(PENDING_DEPRECATION_FMT.format(**ctx))
  28. else:
  29. w = CDeprecationWarning(DEPRECATION_FMT.format(**ctx))
  30. warnings.warn(w, stacklevel=stacklevel)
  31. def Callable(deprecation: Optional[str]=None,
  32. removal: Optional[str]=None,
  33. alternative: Optional[str]=None,
  34. description: Optional[str]=None) -> Callable:
  35. """Decorator for deprecated functions.
  36. A deprecation warning will be emitted when the function is called.
  37. Arguments:
  38. deprecation (str): Version that marks first deprecation, if this
  39. argument isn't set a ``PendingDeprecationWarning`` will be
  40. emitted instead.
  41. removal (str): Future version when this feature will be removed.
  42. alternative (str): Instructions for an alternative solution (if any).
  43. description (str): Description of what's being deprecated.
  44. """
  45. def _inner(fun):
  46. @wraps(fun)
  47. def __inner(*args, **kwargs):
  48. from .imports import qualname
  49. warn(description=description or qualname(fun),
  50. deprecation=deprecation,
  51. removal=removal,
  52. alternative=alternative,
  53. stacklevel=3)
  54. return fun(*args, **kwargs)
  55. return __inner
  56. return _inner
  57. def Property(deprecation: Optional[str]=None,
  58. removal: Optional[str]=None,
  59. alternative: Optional[str]=None,
  60. description: Optional[str]=None) -> Callable:
  61. def _inner(fun: Callable) -> Any:
  62. """Decorator for deprecated properties."""
  63. return _deprecated_property(
  64. fun, deprecation=deprecation, removal=removal,
  65. alternative=alternative, description=description or fun.__name__)
  66. return _inner
  67. class _deprecated_property:
  68. def __init__(self,
  69. fget: Optional[Callable]=None,
  70. fset: Optional[Callable]=None,
  71. fdel: Optional[Callable]=None,
  72. doc: Optional[str]=None,
  73. **depreinfo) -> None:
  74. self.__get = fget
  75. self.__set = fset
  76. self.__del = fdel
  77. self.__name__, self.__module__, self.__doc__ = (
  78. fget.__name__, fget.__module__, fget.__doc__,
  79. )
  80. self.depreinfo = depreinfo
  81. self.depreinfo.setdefault('stacklevel', 3)
  82. def __get__(self, obj: Any, type: Optional[Any]=None) -> Any:
  83. if obj is None:
  84. return self
  85. warn(**self.depreinfo)
  86. return self.__get(obj)
  87. def __set__(self, obj: Any, value: Any) -> Any:
  88. if obj is None:
  89. return self
  90. if self.__set is None:
  91. raise AttributeError('cannot set attribute')
  92. warn(**self.depreinfo)
  93. self.__set(obj, value)
  94. def __delete__(self, obj: Any) -> Any:
  95. if obj is None:
  96. return self
  97. if self.__del is None:
  98. raise AttributeError('cannot delete attribute')
  99. warn(**self.depreinfo)
  100. self.__del(obj)
  101. def setter(self, fset: Callable) -> Any:
  102. return self.__class__(self.__get, fset, self.__del, **self.depreinfo)
  103. def deleter(self, fdel: Callable) -> Any:
  104. return self.__class__(self.__get, self.__set, fdel, **self.depreinfo)