Decorador de método de classe com argumentos próprios?


154

Como passo um campo de classe para um decorador em um método de classe como argumento? O que eu quero fazer é algo como:

class Client(object):
    def __init__(self, url):
        self.url = url

    @check_authorization("some_attr", self.url)
    def get(self):
        do_work()

Ele reclama que o eu não existe para passar self.urlpara o decorador. Existe uma maneira de contornar isso?


É um decorador personalizado sobre o qual você tem controle ou sobre o qual não pode mudar?
Joel Cornett

É o meu decorador, então eu tenho total controle sobre ele
Mark

Ele é chamado antes de init eu acho que é o problema ...
Joran Beasley

7
O problema é que o eu não existe no momento da definição da função. Você precisa transformá-lo em uma função parcial.
Antimony

Respostas:


208

Sim. Em vez de transmitir o atributo de instância no momento da definição da classe, verifique-o em tempo de execução:

def check_authorization(f):
    def wrapper(*args):
        print args[0].url
        return f(*args)
    return wrapper

class Client(object):
    def __init__(self, url):
        self.url = url

    @check_authorization
    def get(self):
        print 'get'

>>> Client('http://www.google.com').get()
http://www.google.com
get

O decorador intercepta os argumentos do método; o primeiro argumento é a instância, então ele lê o atributo disso. Você pode passar o nome do atributo como uma sequência para o decorador e usá- getattrlo se não desejar codificar o nome do atributo:

def check_authorization(attribute):
    def _check_authorization(f):
        def wrapper(self, *args):
            print getattr(self, attribute)
            return f(self, *args)
        return wrapper
    return _check_authorization

38
from re import search
from functools import wraps

def is_match(_lambda, pattern):
    def wrapper(f):
        @wraps(f)
        def wrapped(self, *f_args, **f_kwargs):
            if callable(_lambda) and search(pattern, (_lambda(self) or '')): 
                f(self, *f_args, **f_kwargs)
        return wrapped
    return wrapper

class MyTest(object):

    def __init__(self):
        self.name = 'foo'
        self.surname = 'bar'

    @is_match(lambda x: x.name, 'foo')
    @is_match(lambda x: x.surname, 'foo')
    def my_rule(self):
        print 'my_rule : ok'

    @is_match(lambda x: x.name, 'foo')
    @is_match(lambda x: x.surname, 'bar')
    def my_rule2(self):
        print 'my_rule2 : ok'



test = MyTest()
test.my_rule()
test.my_rule2()

ouput: my_rule2: ok


@raphael Nesta configuração, não consigo acessar _lambda ou padrão. Como posso remediar isso.
Jonathan

1
@ Rafael: Como posso fazer o mesmo para um método de classe, pois aqui todos os métodos são métodos de instância.
Apurva Kunkulol 12/03/2019

38

Um exemplo mais conciso pode ser o seguinte:

#/usr/bin/env python3
from functools import wraps

def wrapper(method):
    @wraps(method)
    def _impl(self, *method_args, **method_kwargs):
        method_output = method(self, *method_args, **method_kwargs)
        return method_output + "!"
    return _impl

class Foo:
    @wrapper
    def bar(self, word):
        return word

f = Foo()
result = f.bar("kitty")
print(result)

Qual será impresso:

kitty!

6

Outra opção seria abandonar o açúcar sintático e decorar na __init__sala de aula.

def countdown(number):
    def countdown_decorator(func):
        def func_wrapper():
            for index in reversed(range(1, number+1)):
                print("{}".format(index))
            func()
        return func_wrapper
    return countdown_decorator

class MySuperClass():
    def __init__(self, number):
        self.number = number
        self.do_thing = countdown(number)(self.do_thing)

    def do_thing(self):
        print('im doing stuff!')


myclass = MySuperClass(3)

myclass.do_thing()

o que imprimiria

3
2
1
im doing stuff!

4

Você não pode. Não há selfno corpo da classe, porque não existe instância. Você precisaria passar, digamos, um strcontendo o nome do atributo para pesquisar na instância, o que a função retornada pode executar ou usar um método completamente diferente.

Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.