| Viewing file:  deprecations.py (6.96 KB)      -rw-r--r-- Select action/file-type:
 
  (+) |  (+) |  (+) | Code (+) | Session (+) |  (+) | SDB (+) |  (+) |  (+) |  (+) |  (+) |  (+) | 
 
# util/deprecations.py# Copyright (C) 2005-2019 the SQLAlchemy authors and contributors
 # <see AUTHORS file>
 #
 # This module is part of SQLAlchemy and is released under
 # the MIT License: http://www.opensource.org/licenses/mit-license.php
 
 """Helpers related to deprecation of functions, methods, classes, other
 functionality."""
 
 import re
 import warnings
 
 from . import compat
 from .langhelpers import decorator
 from .langhelpers import inject_docstring_text
 from .langhelpers import inject_param_text
 from .. import exc
 
 
 def warn_deprecated(msg, stacklevel=3):
 warnings.warn(msg, exc.SADeprecationWarning, stacklevel=stacklevel)
 
 
 def warn_pending_deprecation(msg, stacklevel=3):
 warnings.warn(msg, exc.SAPendingDeprecationWarning, stacklevel=stacklevel)
 
 
 def deprecated_cls(version, message, constructor="__init__"):
 header = ".. deprecated:: %s %s" % (version, (message or ""))
 
 def decorate(cls):
 return _decorate_cls_with_warning(
 cls,
 constructor,
 exc.SADeprecationWarning,
 message % dict(func=constructor),
 header,
 )
 
 return decorate
 
 
 def deprecated(version, message=None, add_deprecation_to_docstring=True):
 """Decorates a function and issues a deprecation warning on use.
 
 :param version:
 Issue version in the warning.
 
 :param message:
 If provided, issue message in the warning.  A sensible default
 is used if not provided.
 
 :param add_deprecation_to_docstring:
 Default True.  If False, the wrapped function's __doc__ is left
 as-is.  If True, the 'message' is prepended to the docs if
 provided, or sensible default if message is omitted.
 
 """
 
 if add_deprecation_to_docstring:
 header = ".. deprecated:: %s %s" % (version, (message or ""))
 else:
 header = None
 
 if message is None:
 message = "Call to deprecated function %(func)s"
 
 def decorate(fn):
 return _decorate_with_warning(
 fn,
 exc.SADeprecationWarning,
 message % dict(func=fn.__name__),
 header,
 )
 
 return decorate
 
 
 def deprecated_params(**specs):
 """Decorates a function to warn on use of certain parameters.
 
 e.g. ::
 
 @deprecated_params(
 weak_identity_map=(
 "0.7",
 "the :paramref:`.Session.weak_identity_map parameter "
 "is deprecated."
 )
 
 )
 
 """
 
 messages = {}
 for param, (version, message) in specs.items():
 messages[param] = _sanitize_restructured_text(message)
 
 def decorate(fn):
 spec = compat.inspect_getfullargspec(fn)
 if spec.defaults is not None:
 defaults = dict(
 zip(
 spec.args[(len(spec.args) - len(spec.defaults)) :],
 spec.defaults,
 )
 )
 check_defaults = set(defaults).intersection(messages)
 check_kw = set(messages).difference(defaults)
 else:
 check_defaults = ()
 check_kw = set(messages)
 
 @decorator
 def warned(fn, *args, **kwargs):
 for m in check_defaults:
 if kwargs[m] != defaults[m]:
 warnings.warn(
 messages[m], exc.SADeprecationWarning, stacklevel=3
 )
 for m in check_kw:
 if m in kwargs:
 warnings.warn(
 messages[m], exc.SADeprecationWarning, stacklevel=3
 )
 
 return fn(*args, **kwargs)
 
 doc = fn.__doc__ is not None and fn.__doc__ or ""
 if doc:
 doc = inject_param_text(
 doc,
 {
 param: ".. deprecated:: %s %s" % (version, (message or ""))
 for param, (version, message) in specs.items()
 },
 )
 decorated = warned(fn)
 decorated.__doc__ = doc
 return decorated
 
 return decorate
 
 
 def pending_deprecation(
 version, message=None, add_deprecation_to_docstring=True
 ):
 """Decorates a function and issues a pending deprecation warning on use.
 
 :param version:
 An approximate future version at which point the pending deprecation
 will become deprecated.  Not used in messaging.
 
 :param message:
 If provided, issue message in the warning.  A sensible default
 is used if not provided.
 
 :param add_deprecation_to_docstring:
 Default True.  If False, the wrapped function's __doc__ is left
 as-is.  If True, the 'message' is prepended to the docs if
 provided, or sensible default if message is omitted.
 """
 
 if add_deprecation_to_docstring:
 header = ".. deprecated:: %s (pending) %s" % (version, (message or ""))
 else:
 header = None
 
 if message is None:
 message = "Call to deprecated function %(func)s"
 
 def decorate(fn):
 return _decorate_with_warning(
 fn,
 exc.SAPendingDeprecationWarning,
 message % dict(func=fn.__name__),
 header,
 )
 
 return decorate
 
 
 def deprecated_option_value(parameter_value, default_value, warning_text):
 if parameter_value is None:
 return default_value
 else:
 warn_deprecated(warning_text)
 return parameter_value
 
 
 def _sanitize_restructured_text(text):
 def repl(m):
 type_, name = m.group(1, 2)
 if type_ in ("func", "meth"):
 name += "()"
 return name
 
 return re.sub(r"\:(\w+)\:`~?\.?(.+?)`", repl, text)
 
 
 def _decorate_cls_with_warning(
 cls, constructor, wtype, message, docstring_header=None
 ):
 doc = cls.__doc__ is not None and cls.__doc__ or ""
 if docstring_header is not None:
 docstring_header %= dict(func=constructor)
 
 doc = inject_docstring_text(doc, docstring_header, 1)
 
 if type(cls) is type:
 clsdict = dict(cls.__dict__)
 clsdict["__doc__"] = doc
 cls = type(cls.__name__, cls.__bases__, clsdict)
 constructor_fn = clsdict[constructor]
 else:
 cls.__doc__ = doc
 constructor_fn = getattr(cls, constructor)
 
 setattr(
 cls,
 constructor,
 _decorate_with_warning(constructor_fn, wtype, message, None),
 )
 
 return cls
 
 
 def _decorate_with_warning(func, wtype, message, docstring_header=None):
 """Wrap a function with a warnings.warn and augmented docstring."""
 
 message = _sanitize_restructured_text(message)
 
 @decorator
 def warned(fn, *args, **kwargs):
 warnings.warn(message, wtype, stacklevel=3)
 return fn(*args, **kwargs)
 
 doc = func.__doc__ is not None and func.__doc__ or ""
 if docstring_header is not None:
 docstring_header %= dict(func=func.__name__)
 
 doc = inject_docstring_text(doc, docstring_header, 1)
 
 decorated = warned(func)
 decorated.__doc__ = doc
 decorated._sa_warn = lambda: warnings.warn(message, wtype, stacklevel=3)
 return decorated
 
 |