#29253: method_decorator behaves differently with list argument
------------------------------------------+----------------------------
               Reporter:  Chris Jerdonek  |          Owner:  nobody
                   Type:  Bug             |         Status:  new
              Component:  Utilities       |        Version:  2.0
               Severity:  Normal          |       Keywords:  decorators
           Triage Stage:  Unreviewed      |      Has patch:  0
    Needs documentation:  0               |    Needs tests:  0
Patch needs improvement:  0               |  Easy pickings:  0
                  UI/UX:  0               |
------------------------------------------+----------------------------
 The [https://docs.djangoproject.com/en/2.0/topics/class-based-views/intro
 /#decorating-the-class docs say] that the "list" form of
 `@method_decorator` is equivalent to invoking it multiple times:

 {{{#!python
 decorators = [never_cache, login_required]

 @method_decorator(decorators, name='dispatch')
 class ProtectedView(TemplateView):
     template_name = 'secret.html'

 @method_decorator(never_cache, name='dispatch')
 @method_decorator(login_required, name='dispatch')
 class ProtectedView(TemplateView):
     template_name = 'secret.html'
 }}}

 However, it appears there is a slight difference in behavior. For example:

 {{{#!python
 from django.utils.decorators import method_decorator
 from django.views.decorators.csrf import csrf_exempt

 def my_decorator(func):
     def new_func(*args, **kwargs):
         func(*args, **kwargs)

     return new_func

 @method_decorator(my_decorator, name='dispatch')
 @method_decorator(csrf_exempt, name='dispatch')
 class View1:

     def dispatch(self):
         pass

 @method_decorator([my_decorator, csrf_exempt], name='dispatch')
 class View2:

     def dispatch(self):
         pass

 print(hasattr(View1.dispatch, 'csrf_exempt'))
 print(hasattr(View2.dispatch, 'csrf_exempt'))
 }}}

 results in the output--

 {{{
 True
 False
 }}}

 It appears this is because `method_decorator()` takes a
 
[https://github.com/django/django/blob/ee7f51c66dfc1700ff065dfeb5fe2388cc2e9619/django/utils/decorators.py#L51
 short cut] when processing a list. It doesn't carry over the attributes of
 the decorated function like it does in the normal case.

-- 
Ticket URL: <https://code.djangoproject.com/ticket/29253>
Django <https://code.djangoproject.com/>
The Web framework for perfectionists with deadlines.

-- 
You received this message because you are subscribed to the Google Groups 
"Django updates" group.
To unsubscribe from this group and stop receiving emails from it, send an email 
to [email protected].
To post to this group, send email to [email protected].
To view this discussion on the web visit 
https://groups.google.com/d/msgid/django-updates/052.8040820e390467ce0c78dad8547e97b5%40djangoproject.com.
For more options, visit https://groups.google.com/d/optout.

Reply via email to