mirror of
				https://github.com/django/django.git
				synced 2025-11-03 21:25:09 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			482 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			482 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
from functools import update_wrapper, wraps
 | 
						|
from unittest import TestCase
 | 
						|
 | 
						|
from django.contrib.admin.views.decorators import staff_member_required
 | 
						|
from django.contrib.auth.decorators import (
 | 
						|
    login_required, permission_required, user_passes_test,
 | 
						|
)
 | 
						|
from django.http import HttpRequest, HttpResponse, HttpResponseNotAllowed
 | 
						|
from django.middleware.clickjacking import XFrameOptionsMiddleware
 | 
						|
from django.test import SimpleTestCase
 | 
						|
from django.utils.decorators import method_decorator
 | 
						|
from django.utils.functional import keep_lazy, keep_lazy_text, lazy
 | 
						|
from django.utils.safestring import mark_safe
 | 
						|
from django.views.decorators.cache import (
 | 
						|
    cache_control, cache_page, never_cache,
 | 
						|
)
 | 
						|
from django.views.decorators.clickjacking import (
 | 
						|
    xframe_options_deny, xframe_options_exempt, xframe_options_sameorigin,
 | 
						|
)
 | 
						|
from django.views.decorators.http import (
 | 
						|
    condition, require_GET, require_http_methods, require_POST, require_safe,
 | 
						|
)
 | 
						|
from django.views.decorators.vary import vary_on_cookie, vary_on_headers
 | 
						|
 | 
						|
 | 
						|
def fully_decorated(request):
 | 
						|
    """Expected __doc__"""
 | 
						|
    return HttpResponse('<html><body>dummy</body></html>')
 | 
						|
 | 
						|
 | 
						|
fully_decorated.anything = "Expected __dict__"
 | 
						|
 | 
						|
 | 
						|
def compose(*functions):
 | 
						|
    # compose(f, g)(*args, **kwargs) == f(g(*args, **kwargs))
 | 
						|
    functions = list(reversed(functions))
 | 
						|
 | 
						|
    def _inner(*args, **kwargs):
 | 
						|
        result = functions[0](*args, **kwargs)
 | 
						|
        for f in functions[1:]:
 | 
						|
            result = f(result)
 | 
						|
        return result
 | 
						|
    return _inner
 | 
						|
 | 
						|
 | 
						|
full_decorator = compose(
 | 
						|
    # django.views.decorators.http
 | 
						|
    require_http_methods(["GET"]),
 | 
						|
    require_GET,
 | 
						|
    require_POST,
 | 
						|
    require_safe,
 | 
						|
    condition(lambda r: None, lambda r: None),
 | 
						|
 | 
						|
    # django.views.decorators.vary
 | 
						|
    vary_on_headers('Accept-language'),
 | 
						|
    vary_on_cookie,
 | 
						|
 | 
						|
    # django.views.decorators.cache
 | 
						|
    cache_page(60 * 15),
 | 
						|
    cache_control(private=True),
 | 
						|
    never_cache,
 | 
						|
 | 
						|
    # django.contrib.auth.decorators
 | 
						|
    # Apply user_passes_test twice to check #9474
 | 
						|
    user_passes_test(lambda u: True),
 | 
						|
    login_required,
 | 
						|
    permission_required('change_world'),
 | 
						|
 | 
						|
    # django.contrib.admin.views.decorators
 | 
						|
    staff_member_required,
 | 
						|
 | 
						|
    # django.utils.functional
 | 
						|
    keep_lazy(HttpResponse),
 | 
						|
    keep_lazy_text,
 | 
						|
    lazy,
 | 
						|
 | 
						|
    # django.utils.safestring
 | 
						|
    mark_safe,
 | 
						|
)
 | 
						|
 | 
						|
fully_decorated = full_decorator(fully_decorated)
 | 
						|
 | 
						|
 | 
						|
class DecoratorsTest(TestCase):
 | 
						|
 | 
						|
    def test_attributes(self):
 | 
						|
        """
 | 
						|
        Built-in decorators set certain attributes of the wrapped function.
 | 
						|
        """
 | 
						|
        self.assertEqual(fully_decorated.__name__, 'fully_decorated')
 | 
						|
        self.assertEqual(fully_decorated.__doc__, 'Expected __doc__')
 | 
						|
        self.assertEqual(fully_decorated.__dict__['anything'], 'Expected __dict__')
 | 
						|
 | 
						|
    def test_user_passes_test_composition(self):
 | 
						|
        """
 | 
						|
        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:
 | 
						|
            pass
 | 
						|
 | 
						|
        class DummyRequest:
 | 
						|
            pass
 | 
						|
 | 
						|
        request = DummyRequest()
 | 
						|
        request.user = DummyUser()
 | 
						|
        request.user.decorators_applied = []
 | 
						|
        response = callback(request)
 | 
						|
 | 
						|
        self.assertEqual(response, ['test2', 'test1'])
 | 
						|
 | 
						|
    def test_cache_page(self):
 | 
						|
        def my_view(request):
 | 
						|
            return "response"
 | 
						|
        my_view_cached = cache_page(123)(my_view)
 | 
						|
        self.assertEqual(my_view_cached(HttpRequest()), "response")
 | 
						|
        my_view_cached2 = cache_page(123, key_prefix="test")(my_view)
 | 
						|
        self.assertEqual(my_view_cached2(HttpRequest()), "response")
 | 
						|
 | 
						|
    def test_require_safe_accepts_only_safe_methods(self):
 | 
						|
        """
 | 
						|
        Test for the require_safe decorator.
 | 
						|
        A view returns either a response or an exception.
 | 
						|
        Refs #15637.
 | 
						|
        """
 | 
						|
        def my_view(request):
 | 
						|
            return HttpResponse("OK")
 | 
						|
        my_safe_view = require_safe(my_view)
 | 
						|
        request = HttpRequest()
 | 
						|
        request.method = 'GET'
 | 
						|
        self.assertIsInstance(my_safe_view(request), HttpResponse)
 | 
						|
        request.method = 'HEAD'
 | 
						|
        self.assertIsInstance(my_safe_view(request), HttpResponse)
 | 
						|
        request.method = 'POST'
 | 
						|
        self.assertIsInstance(my_safe_view(request), HttpResponseNotAllowed)
 | 
						|
        request.method = 'PUT'
 | 
						|
        self.assertIsInstance(my_safe_view(request), HttpResponseNotAllowed)
 | 
						|
        request.method = 'DELETE'
 | 
						|
        self.assertIsInstance(my_safe_view(request), HttpResponseNotAllowed)
 | 
						|
 | 
						|
 | 
						|
# For testing method_decorator, a decorator that assumes a single argument.
 | 
						|
# We will get type arguments if there is a mismatch in the number of arguments.
 | 
						|
def simple_dec(func):
 | 
						|
    def wrapper(arg):
 | 
						|
        return func("test:" + arg)
 | 
						|
    return wraps(func)(wrapper)
 | 
						|
 | 
						|
 | 
						|
simple_dec_m = method_decorator(simple_dec)
 | 
						|
 | 
						|
 | 
						|
# For testing method_decorator, two decorators that add an attribute to the function
 | 
						|
def myattr_dec(func):
 | 
						|
    def wrapper(*args, **kwargs):
 | 
						|
        return func(*args, **kwargs)
 | 
						|
    wrapper.myattr = True
 | 
						|
    return wrapper
 | 
						|
 | 
						|
 | 
						|
myattr_dec_m = method_decorator(myattr_dec)
 | 
						|
 | 
						|
 | 
						|
def myattr2_dec(func):
 | 
						|
    def wrapper(*args, **kwargs):
 | 
						|
        return func(*args, **kwargs)
 | 
						|
    wrapper.myattr2 = True
 | 
						|
    return wrapper
 | 
						|
 | 
						|
 | 
						|
myattr2_dec_m = method_decorator(myattr2_dec)
 | 
						|
 | 
						|
 | 
						|
class ClsDec:
 | 
						|
    def __init__(self, myattr):
 | 
						|
        self.myattr = myattr
 | 
						|
 | 
						|
    def __call__(self, f):
 | 
						|
 | 
						|
        def wrapped():
 | 
						|
            return f() and self.myattr
 | 
						|
        return update_wrapper(wrapped, f)
 | 
						|
 | 
						|
 | 
						|
class MethodDecoratorTests(SimpleTestCase):
 | 
						|
    """
 | 
						|
    Tests for method_decorator
 | 
						|
    """
 | 
						|
    def test_preserve_signature(self):
 | 
						|
        class Test:
 | 
						|
            @simple_dec_m
 | 
						|
            def say(self, arg):
 | 
						|
                return arg
 | 
						|
 | 
						|
        self.assertEqual("test:hello", Test().say("hello"))
 | 
						|
 | 
						|
    def test_preserve_attributes(self):
 | 
						|
        # Sanity check myattr_dec and myattr2_dec
 | 
						|
        @myattr_dec
 | 
						|
        def func():
 | 
						|
            pass
 | 
						|
        self.assertIs(getattr(func, 'myattr', False), True)
 | 
						|
 | 
						|
        @myattr2_dec
 | 
						|
        def func():
 | 
						|
            pass
 | 
						|
        self.assertIs(getattr(func, 'myattr2', False), True)
 | 
						|
 | 
						|
        @myattr_dec
 | 
						|
        @myattr2_dec
 | 
						|
        def func():
 | 
						|
            pass
 | 
						|
 | 
						|
        self.assertIs(getattr(func, 'myattr', False), True)
 | 
						|
        self.assertIs(getattr(func, 'myattr2', False), False)
 | 
						|
 | 
						|
        # Decorate using method_decorator() on the method.
 | 
						|
        class TestPlain:
 | 
						|
            @myattr_dec_m
 | 
						|
            @myattr2_dec_m
 | 
						|
            def method(self):
 | 
						|
                "A method"
 | 
						|
                pass
 | 
						|
 | 
						|
        # Decorate using method_decorator() on both the class and the method.
 | 
						|
        # The decorators applied to the methods are applied before the ones
 | 
						|
        # applied to the class.
 | 
						|
        @method_decorator(myattr_dec_m, "method")
 | 
						|
        class TestMethodAndClass:
 | 
						|
            @method_decorator(myattr2_dec_m)
 | 
						|
            def method(self):
 | 
						|
                "A method"
 | 
						|
                pass
 | 
						|
 | 
						|
        # Decorate using an iterable of function decorators.
 | 
						|
        @method_decorator((myattr_dec, myattr2_dec), 'method')
 | 
						|
        class TestFunctionIterable:
 | 
						|
            def method(self):
 | 
						|
                "A method"
 | 
						|
                pass
 | 
						|
 | 
						|
        # Decorate using an iterable of method decorators.
 | 
						|
        decorators = (myattr_dec_m, myattr2_dec_m)
 | 
						|
 | 
						|
        @method_decorator(decorators, "method")
 | 
						|
        class TestMethodIterable:
 | 
						|
            def method(self):
 | 
						|
                "A method"
 | 
						|
                pass
 | 
						|
 | 
						|
        tests = (TestPlain, TestMethodAndClass, TestFunctionIterable, TestMethodIterable)
 | 
						|
        for Test in tests:
 | 
						|
            with self.subTest(Test=Test):
 | 
						|
                self.assertIs(getattr(Test().method, 'myattr', False), True)
 | 
						|
                self.assertIs(getattr(Test().method, 'myattr2', False), True)
 | 
						|
                self.assertIs(getattr(Test.method, 'myattr', False), True)
 | 
						|
                self.assertIs(getattr(Test.method, 'myattr2', False), True)
 | 
						|
                self.assertEqual(Test.method.__doc__, 'A method')
 | 
						|
                self.assertEqual(Test.method.__name__, 'method')
 | 
						|
 | 
						|
    def test_new_attribute(self):
 | 
						|
        """A decorator that sets a new attribute on the method."""
 | 
						|
        def decorate(func):
 | 
						|
            func.x = 1
 | 
						|
            return func
 | 
						|
 | 
						|
        class MyClass:
 | 
						|
            @method_decorator(decorate)
 | 
						|
            def method(self):
 | 
						|
                return True
 | 
						|
 | 
						|
        obj = MyClass()
 | 
						|
        self.assertEqual(obj.method.x, 1)
 | 
						|
        self.assertIs(obj.method(), True)
 | 
						|
 | 
						|
    def test_bad_iterable(self):
 | 
						|
        decorators = {myattr_dec_m, myattr2_dec_m}
 | 
						|
        msg = "'set' object is not subscriptable"
 | 
						|
        with self.assertRaisesMessage(TypeError, msg):
 | 
						|
            @method_decorator(decorators, "method")
 | 
						|
            class TestIterable:
 | 
						|
                def method(self):
 | 
						|
                    "A method"
 | 
						|
                    pass
 | 
						|
 | 
						|
    # Test for argumented decorator
 | 
						|
    def test_argumented(self):
 | 
						|
        class Test:
 | 
						|
            @method_decorator(ClsDec(False))
 | 
						|
            def method(self):
 | 
						|
                return True
 | 
						|
 | 
						|
        self.assertIs(Test().method(), False)
 | 
						|
 | 
						|
    def test_descriptors(self):
 | 
						|
 | 
						|
        def original_dec(wrapped):
 | 
						|
            def _wrapped(arg):
 | 
						|
                return wrapped(arg)
 | 
						|
 | 
						|
            return _wrapped
 | 
						|
 | 
						|
        method_dec = method_decorator(original_dec)
 | 
						|
 | 
						|
        class bound_wrapper:
 | 
						|
            def __init__(self, wrapped):
 | 
						|
                self.wrapped = wrapped
 | 
						|
                self.__name__ = wrapped.__name__
 | 
						|
 | 
						|
            def __call__(self, arg):
 | 
						|
                return self.wrapped(arg)
 | 
						|
 | 
						|
            def __get__(self, instance, cls=None):
 | 
						|
                return self
 | 
						|
 | 
						|
        class descriptor_wrapper:
 | 
						|
            def __init__(self, wrapped):
 | 
						|
                self.wrapped = wrapped
 | 
						|
                self.__name__ = wrapped.__name__
 | 
						|
 | 
						|
            def __get__(self, instance, cls=None):
 | 
						|
                return bound_wrapper(self.wrapped.__get__(instance, cls))
 | 
						|
 | 
						|
        class Test:
 | 
						|
            @method_dec
 | 
						|
            @descriptor_wrapper
 | 
						|
            def method(self, arg):
 | 
						|
                return arg
 | 
						|
 | 
						|
        self.assertEqual(Test().method(1), 1)
 | 
						|
 | 
						|
    def test_class_decoration(self):
 | 
						|
        """
 | 
						|
        @method_decorator can be used to decorate a class and its methods.
 | 
						|
        """
 | 
						|
        def deco(func):
 | 
						|
            def _wrapper(*args, **kwargs):
 | 
						|
                return True
 | 
						|
            return _wrapper
 | 
						|
 | 
						|
        @method_decorator(deco, name="method")
 | 
						|
        class Test:
 | 
						|
            def method(self):
 | 
						|
                return False
 | 
						|
 | 
						|
        self.assertTrue(Test().method())
 | 
						|
 | 
						|
    def test_tuple_of_decorators(self):
 | 
						|
        """
 | 
						|
        @method_decorator can accept a tuple of decorators.
 | 
						|
        """
 | 
						|
        def add_question_mark(func):
 | 
						|
            def _wrapper(*args, **kwargs):
 | 
						|
                return func(*args, **kwargs) + "?"
 | 
						|
            return _wrapper
 | 
						|
 | 
						|
        def add_exclamation_mark(func):
 | 
						|
            def _wrapper(*args, **kwargs):
 | 
						|
                return func(*args, **kwargs) + "!"
 | 
						|
            return _wrapper
 | 
						|
 | 
						|
        # The order should be consistent with the usual order in which
 | 
						|
        # decorators are applied, e.g.
 | 
						|
        #    @add_exclamation_mark
 | 
						|
        #    @add_question_mark
 | 
						|
        #    def func():
 | 
						|
        #        ...
 | 
						|
        decorators = (add_exclamation_mark, add_question_mark)
 | 
						|
 | 
						|
        @method_decorator(decorators, name="method")
 | 
						|
        class TestFirst:
 | 
						|
            def method(self):
 | 
						|
                return "hello world"
 | 
						|
 | 
						|
        class TestSecond:
 | 
						|
            @method_decorator(decorators)
 | 
						|
            def method(self):
 | 
						|
                return "hello world"
 | 
						|
 | 
						|
        self.assertEqual(TestFirst().method(), "hello world?!")
 | 
						|
        self.assertEqual(TestSecond().method(), "hello world?!")
 | 
						|
 | 
						|
    def test_invalid_non_callable_attribute_decoration(self):
 | 
						|
        """
 | 
						|
        @method_decorator on a non-callable attribute raises an error.
 | 
						|
        """
 | 
						|
        msg = (
 | 
						|
            "Cannot decorate 'prop' as it isn't a callable attribute of "
 | 
						|
            "<class 'Test'> (1)"
 | 
						|
        )
 | 
						|
        with self.assertRaisesMessage(TypeError, msg):
 | 
						|
            @method_decorator(lambda: None, name="prop")
 | 
						|
            class Test:
 | 
						|
                prop = 1
 | 
						|
 | 
						|
                @classmethod
 | 
						|
                def __module__(cls):
 | 
						|
                    return "tests"
 | 
						|
 | 
						|
    def test_invalid_method_name_to_decorate(self):
 | 
						|
        """
 | 
						|
        @method_decorator on a nonexistent method raises an error.
 | 
						|
        """
 | 
						|
        msg = (
 | 
						|
            "The keyword argument `name` must be the name of a method of the "
 | 
						|
            "decorated class: <class 'Test'>. Got 'nonexistent_method' instead"
 | 
						|
        )
 | 
						|
        with self.assertRaisesMessage(ValueError, msg):
 | 
						|
            @method_decorator(lambda: None, name='nonexistent_method')
 | 
						|
            class Test:
 | 
						|
                @classmethod
 | 
						|
                def __module__(cls):
 | 
						|
                    return "tests"
 | 
						|
 | 
						|
 | 
						|
class XFrameOptionsDecoratorsTests(TestCase):
 | 
						|
    """
 | 
						|
    Tests for the X-Frame-Options decorators.
 | 
						|
    """
 | 
						|
    def test_deny_decorator(self):
 | 
						|
        """
 | 
						|
        Ensures @xframe_options_deny properly sets the X-Frame-Options header.
 | 
						|
        """
 | 
						|
        @xframe_options_deny
 | 
						|
        def a_view(request):
 | 
						|
            return HttpResponse()
 | 
						|
        r = a_view(HttpRequest())
 | 
						|
        self.assertEqual(r['X-Frame-Options'], 'DENY')
 | 
						|
 | 
						|
    def test_sameorigin_decorator(self):
 | 
						|
        """
 | 
						|
        Ensures @xframe_options_sameorigin properly sets the X-Frame-Options
 | 
						|
        header.
 | 
						|
        """
 | 
						|
        @xframe_options_sameorigin
 | 
						|
        def a_view(request):
 | 
						|
            return HttpResponse()
 | 
						|
        r = a_view(HttpRequest())
 | 
						|
        self.assertEqual(r['X-Frame-Options'], 'SAMEORIGIN')
 | 
						|
 | 
						|
    def test_exempt_decorator(self):
 | 
						|
        """
 | 
						|
        Ensures @xframe_options_exempt properly instructs the
 | 
						|
        XFrameOptionsMiddleware to NOT set the header.
 | 
						|
        """
 | 
						|
        @xframe_options_exempt
 | 
						|
        def a_view(request):
 | 
						|
            return HttpResponse()
 | 
						|
        req = HttpRequest()
 | 
						|
        resp = a_view(req)
 | 
						|
        self.assertIsNone(resp.get('X-Frame-Options', None))
 | 
						|
        self.assertTrue(resp.xframe_options_exempt)
 | 
						|
 | 
						|
        # Since the real purpose of the exempt decorator is to suppress
 | 
						|
        # the middleware's functionality, let's make sure it actually works...
 | 
						|
        r = XFrameOptionsMiddleware().process_response(req, resp)
 | 
						|
        self.assertIsNone(r.get('X-Frame-Options', None))
 | 
						|
 | 
						|
 | 
						|
class NeverCacheDecoratorTest(TestCase):
 | 
						|
    def test_never_cache_decorator(self):
 | 
						|
        @never_cache
 | 
						|
        def a_view(request):
 | 
						|
            return HttpResponse()
 | 
						|
        r = a_view(HttpRequest())
 | 
						|
        self.assertEqual(
 | 
						|
            set(r['Cache-Control'].split(', ')),
 | 
						|
            {'max-age=0', 'no-cache', 'no-store', 'must-revalidate', 'private'},
 | 
						|
        )
 |