декораторы в стандартной библиотеке Python (специально @deprecated)

128

Мне нужно пометить подпрограммы как устаревшие, но, очевидно, нет стандартного декоратора библиотеки для устаревания. Мне известны рецепты этого и модуль предупреждений, но мой вопрос: почему нет стандартного декоратора библиотеки для этой (общей) задачи?

Дополнительный вопрос: есть ли вообще в стандартной библиотеке стандартные декораторы?

Стефано Борини
источник
13
теперь есть пакет устаревания
мюон
11
Я понимаю, как это сделать, но пришел сюда, чтобы понять, почему этого нет в std lib (как я предполагаю, это случай OP), и не вижу хорошего ответа на фактический вопрос
SwimBikeRun
4
Почему так часто случается, что на вопросы приходят десятки ответов, которые даже не пытаются ответить на вопрос, и активно игнорируют такие вещи, как «Я знаю рецепты»? Это сводит с ума!
Catskul
1
@Catskul из-за поддельных интернет-точек.
Стефано Борини
1
Вы можете использовать устаревшую библиотеку.
Laurent LAPORTE

Ответы:

59

Вот фрагмент, модифицированный из тех, что цитирует Леандро:

import warnings
import functools

def deprecated(func):
    """This is a decorator which can be used to mark functions
    as deprecated. It will result in a warning being emitted
    when the function is used."""
    @functools.wraps(func)
    def new_func(*args, **kwargs):
        warnings.simplefilter('always', DeprecationWarning)  # turn off filter
        warnings.warn("Call to deprecated function {}.".format(func.__name__),
                      category=DeprecationWarning,
                      stacklevel=2)
        warnings.simplefilter('default', DeprecationWarning)  # reset filter
        return func(*args, **kwargs)
    return new_func

# Examples

@deprecated
def some_old_function(x, y):
    return x + y

class SomeClass:
    @deprecated
    def some_old_method(self, x, y):
        return x + y

Потому что в некоторых интерпретаторах первое открытое решение (без обработки фильтра) может привести к подавлению предупреждения.

Патрицио Бертони
источник
14
Почему бы не использовать, functools.wrapsа не задавать такие имя и документ?
Максимилиан
1
@Maximilian: Отредактировано, чтобы добавить это, чтобы будущие копипастеры этого кода тоже не ошиблись
Эрик
17
Не люблю побочный эффект (включение / выключение фильтра). Решать это - не задача декоратора.
Kentzo
1
Включение и выключение фильтра может вызвать ошибку bugs.python.org/issue29672
gerrit 07
4
не отвечает на актуальный вопрос.
Catskul
45

Вот еще одно решение:

Этот декоратор ( фактически фабрика декораторов ) позволяет вам указать причину . Также более полезно помочь разработчику диагностировать проблему, указав имя исходного файла и номер строки .

EDIT : этот код использует рекомендацию Zero: он заменяет warnings.warn_explicitстроку на warnings.warn(msg, category=DeprecationWarning, stacklevel=2), которая печатает сайт вызова функции, а не сайт определения функции. Это упрощает отладку.

EDIT2 : эта версия позволяет разработчику указывать необязательное сообщение «причина».

import functools
import inspect
import warnings

string_types = (type(b''), type(u''))


def deprecated(reason):
    """
    This is a decorator which can be used to mark functions
    as deprecated. It will result in a warning being emitted
    when the function is used.
    """

    if isinstance(reason, string_types):

        # The @deprecated is used with a 'reason'.
        #
        # .. code-block:: python
        #
        #    @deprecated("please, use another function")
        #    def old_function(x, y):
        #      pass

        def decorator(func1):

            if inspect.isclass(func1):
                fmt1 = "Call to deprecated class {name} ({reason})."
            else:
                fmt1 = "Call to deprecated function {name} ({reason})."

            @functools.wraps(func1)
            def new_func1(*args, **kwargs):
                warnings.simplefilter('always', DeprecationWarning)
                warnings.warn(
                    fmt1.format(name=func1.__name__, reason=reason),
                    category=DeprecationWarning,
                    stacklevel=2
                )
                warnings.simplefilter('default', DeprecationWarning)
                return func1(*args, **kwargs)

            return new_func1

        return decorator

    elif inspect.isclass(reason) or inspect.isfunction(reason):

        # The @deprecated is used without any 'reason'.
        #
        # .. code-block:: python
        #
        #    @deprecated
        #    def old_function(x, y):
        #      pass

        func2 = reason

        if inspect.isclass(func2):
            fmt2 = "Call to deprecated class {name}."
        else:
            fmt2 = "Call to deprecated function {name}."

        @functools.wraps(func2)
        def new_func2(*args, **kwargs):
            warnings.simplefilter('always', DeprecationWarning)
            warnings.warn(
                fmt2.format(name=func2.__name__),
                category=DeprecationWarning,
                stacklevel=2
            )
            warnings.simplefilter('default', DeprecationWarning)
            return func2(*args, **kwargs)

        return new_func2

    else:
        raise TypeError(repr(type(reason)))

Вы можете использовать этот декоратор для функций , методов и классов .

Вот простой пример:

@deprecated("use another function")
def some_old_function(x, y):
    return x + y


class SomeClass(object):
    @deprecated("use another method")
    def some_old_method(self, x, y):
        return x + y


@deprecated("use another class")
class SomeOldClass(object):
    pass


some_old_function(5, 3)
SomeClass().some_old_method(8, 9)
SomeOldClass()

Ты получишь:

deprecated_example.py:59: DeprecationWarning: Call to deprecated function or method some_old_function (use another function).
  some_old_function(5, 3)
deprecated_example.py:60: DeprecationWarning: Call to deprecated function or method some_old_method (use another method).
  SomeClass().some_old_method(8, 9)
deprecated_example.py:61: DeprecationWarning: Call to deprecated class SomeOldClass (use another class).
  SomeOldClass()

EDIT3: этот декоратор теперь является частью устаревшей библиотеки:

Новый стабильный релиз v1.2.10 🎉

Лоран ЛАПОРТ
источник
6
Хорошо работает - я предпочитаю заменять warn_explicitстроку, warnings.warn(msg, category=DeprecationWarning, stacklevel=2)которая печатает сайт вызова функции, а не сайт определения функции. Это упрощает отладку.
Zero
Здравствуйте, я хотел бы использовать ваш фрагмент кода в библиотеке под лицензией GPLv3 . Хотели бы вы перелицензировать свой код под GPLv3 или какой-либо другой разрешающей лицензией , чтобы я мог сделать это на законных основаниях?
gerrit 07
1
@LaurentLAPORTE Я знаю. CC-BY-SO не разрешает использование в рамках GPLv3 (из-за битов общего доступа), поэтому я спрашиваю, готовы ли вы выпустить этот код специально дополнительно под лицензией, совместимой с GPL. Если нет, это нормально, и я не буду использовать ваш код.
gerrit 09
2
не отвечает на актуальный вопрос.
Catskul
15

Как подсказал мюон , вы можете установить deprecationпакет для этого.

deprecationБиблиотека предоставляет deprecatedдекоратор и fail_if_not_removedдекоратор для ваших тестов.

Монтаж

pip install deprecation

Пример использования

import deprecation

@deprecation.deprecated(deprecated_in="1.0", removed_in="2.0",
                        current_version=__version__,
                        details="Use the bar function instead")
def foo():
    """Do some stuff"""
    return 1

См. Http://deprecation.readthedocs.io/ для полной документации.

Stevoisiak
источник
4
не отвечает на актуальный вопрос.
Catskul
1
Обратите внимание, что PyCharm не распознает это
cz
12

Я предполагаю, что причина в том, что код Python не может обрабатываться статически (как это делается для компиляторов C ++), вы не можете получить предупреждение об использовании некоторых вещей до его фактического использования. Я не думаю, что это хорошая идея, чтобы спамить пользователю вашего скрипта кучей сообщений «Предупреждение: этот разработчик этого скрипта использует устаревший API».

Обновление: но вы можете создать декоратор, который преобразует исходную функцию в другую. Новая функция будет отмечать / проверять переключатель, сообщая, что эта функция уже была вызвана, и будет отображать сообщение только при включении переключателя. И / или при выходе он может распечатать список всех устаревших функций, используемых в программе.

оны
источник
3
И вы должны иметь возможность указать устаревание, когда функция импортируется из модуля . Декоратор подойдет для этого.
Януш Ленар
@JanuszLenar, это предупреждение будет отображаться, даже если мы не будем использовать эту устаревшую функцию. Но я думаю, что могу обновить свой ответ некоторым намеком.
оны
8

Вы можете создать файл utils

import warnings

def deprecated(message):
  def deprecated_decorator(func):
      def deprecated_func(*args, **kwargs):
          warnings.warn("{} is a deprecated function. {}".format(func.__name__, message),
                        category=DeprecationWarning,
                        stacklevel=2)
          warnings.simplefilter('default', DeprecationWarning)
          return func(*args, **kwargs)
      return deprecated_func
  return deprecated_decorator

А затем импортируйте декоратор устаревания следующим образом:

from .utils import deprecated

@deprecated("Use method yyy instead")
def some_method()"
 pass
Эрика Дсуза
источник
Спасибо, я использую это, чтобы отправить пользователя в нужное место, а не просто показывать сообщение об устаревании!
German Attanasio
3
не отвечает на актуальный вопрос.
Catskul
2

ОБНОВЛЕНИЕ: я думаю, что лучше, когда мы показываем DeprecationWarning только первый раз для каждой строки кода и когда мы можем отправить какое-то сообщение:

import inspect
import traceback
import warnings
import functools

import time


def deprecated(message: str = ''):
    """
    This is a decorator which can be used to mark functions
    as deprecated. It will result in a warning being emitted
    when the function is used first time and filter is set for show DeprecationWarning.
    """
    def decorator_wrapper(func):
        @functools.wraps(func)
        def function_wrapper(*args, **kwargs):
            current_call_source = '|'.join(traceback.format_stack(inspect.currentframe()))
            if current_call_source not in function_wrapper.last_call_source:
                warnings.warn("Function {} is now deprecated! {}".format(func.__name__, message),
                              category=DeprecationWarning, stacklevel=2)
                function_wrapper.last_call_source.add(current_call_source)

            return func(*args, **kwargs)

        function_wrapper.last_call_source = set()

        return function_wrapper
    return decorator_wrapper


@deprecated('You must use my_func2!')
def my_func():
    time.sleep(.1)
    print('aaa')
    time.sleep(.1)


def my_func2():
    print('bbb')


warnings.simplefilter('always', DeprecationWarning)  # turn off filter
print('before cycle')
for i in range(5):
    my_func()
print('after cycle')
my_func()
my_func()
my_func()

Результат:

before cycle
C:/Users/adr-0/OneDrive/Projects/Python/test/unit1.py:45: DeprecationWarning: Function my_func is now deprecated! You must use my_func2!
aaa
aaa
aaa
aaa
aaa
after cycle
C:/Users/adr-0/OneDrive/Projects/Python/test/unit1.py:47: DeprecationWarning: Function my_func is now deprecated! You must use my_func2!
aaa
C:/Users/adr-0/OneDrive/Projects/Python/test/unit1.py:48: DeprecationWarning: Function my_func is now deprecated! You must use my_func2!
aaa
C:/Users/adr-0/OneDrive/Projects/Python/test/unit1.py:49: DeprecationWarning: Function my_func is now deprecated! You must use my_func2!
aaa

Process finished with exit code 0

Мы можем просто щелкнуть по пути предупреждения и перейти к строке в PyCharm.

ADR
источник
2
не отвечает на актуальный вопрос.
Catskul
0

Дополнение к этому ответу Стивена Васкелларо :

Если вы используете Anaconda, сначала установите deprecationпакет:

conda install -c conda-forge deprecation 

Затем вставьте следующее в верхнюю часть файла

import deprecation

@deprecation.deprecated(deprecated_in="1.0", removed_in="2.0",
                    current_version=__version__,
                    details="Use the bar function instead")
def foo():
    """Do some stuff"""
    return 1

См. Http://deprecation.readthedocs.io/ для полной документации.

omerbp
источник
4
не отвечает на актуальный вопрос.
Catskul