Author: jacob
Date: 2009-04-01 11:17:38 -0500 (Wed, 01 Apr 2009)
New Revision: 10328

Modified:
   django/trunk/django/contrib/auth/decorators.py
   django/trunk/tests/regressiontests/decorators/tests.py
Log:
Fixed #9474: user_passes_test may now be applied multiple times.

Modified: django/trunk/django/contrib/auth/decorators.py
===================================================================
--- django/trunk/django/contrib/auth/decorators.py      2009-04-01 16:06:22 UTC 
(rev 10327)
+++ django/trunk/django/contrib/auth/decorators.py      2009-04-01 16:17:38 UTC 
(rev 10328)
@@ -56,8 +56,19 @@
         self.test_func = test_func
         self.login_url = login_url
         self.redirect_field_name = redirect_field_name
-        update_wrapper(self, view_func)
         
+        # We can't blindly apply update_wrapper because it udpates __dict__ 
and 
+        # if the view function is already a _CheckLogin object then 
+        # self.test_func and friends will get stomped. However, we also can't 
+        # *not* update the wrapper's dict because then view function attributes
+        # don't get updated into the wrapper. So we need to split the
+        # difference: don't let update_wrapper update __dict__, but then update
+        # the (parts of) __dict__ that we care about ourselves.
+        update_wrapper(self, view_func, updated=())
+        for k in view_func.__dict__:
+            if k not in self.__dict__:
+                self.__dict__[k] = view_func.__dict__[k]
+
     def __get__(self, obj, cls=None):
         view_func = self.view_func.__get__(obj, cls)
         return _CheckLogin(view_func, self.test_func, self.login_url, 
self.redirect_field_name)

Modified: django/trunk/tests/regressiontests/decorators/tests.py
===================================================================
--- django/trunk/tests/regressiontests/decorators/tests.py      2009-04-01 
16:06:22 UTC (rev 10327)
+++ django/trunk/tests/regressiontests/decorators/tests.py      2009-04-01 
16:17:38 UTC (rev 10328)
@@ -29,6 +29,7 @@
 fully_decorated = never_cache(fully_decorated)
 
 # django.contrib.auth.decorators
+# Apply user_passes_test twice to check #9474
 fully_decorated = user_passes_test(lambda u:True)(fully_decorated)
 fully_decorated = login_required(fully_decorated)
 fully_decorated = permission_required('change_world')(fully_decorated)
@@ -54,3 +55,33 @@
             self.assertEquals(fully_decorated.__name__, 'fully_decorated')
         self.assertEquals(fully_decorated.__doc__, 'Expected __doc__')
         self.assertEquals(fully_decorated.__dict__['anything'], 'Expected 
__dict__')
+
+    def test_user_passes_test_composition(self):
+        """
+        Test that the user_passes_test decorator can be applied multiple times
+        (#9474).
+        """
+        def test1(user):
+            user.decorators_applied.append('test1')
+            return True
+            
+        def test2(user):
+            user.decorators_applied.append('test2')
+            return True
+            
+        def callback(request):
+            return request.user.decorators_applied
+
+        callback = user_passes_test(test1)(callback)
+        callback = user_passes_test(test2)(callback)
+        
+        class DummyUser(object): pass
+        class DummyRequest(object): pass
+        
+        request = DummyRequest()
+        request.user = DummyUser()
+        request.user.decorators_applied = []
+        response = callback(request)
+        
+        self.assertEqual(response, ['test2', 'test1'])
+        


--~--~---------~--~----~------------~-------~--~----~
You received this message because you are subscribed to the Google Groups 
"Django updates" group.
To post to this group, send email to django-updates@googlegroups.com
To unsubscribe from this group, send email to 
django-updates+unsubscr...@googlegroups.com
For more options, visit this group at 
http://groups.google.com/group/django-updates?hl=en
-~----------~----~----~----~------~----~------~--~---

Reply via email to