mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Refs #31949 -- Made @vary_on_(cookie/headers) decorators work with async functions.
This commit is contained in:
		
				
					committed by
					
						 Mariusz Felisiak
						Mariusz Felisiak
					
				
			
			
				
	
			
			
			
						parent
						
							fb1c763506
						
					
				
				
					commit
					b7a17b0ea0
				
			| @@ -1,5 +1,7 @@ | |||||||
| from functools import wraps | from functools import wraps | ||||||
|  |  | ||||||
|  | from asgiref.sync import iscoroutinefunction | ||||||
|  |  | ||||||
| from django.utils.cache import patch_vary_headers | from django.utils.cache import patch_vary_headers | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -16,13 +18,21 @@ def vary_on_headers(*headers): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def decorator(func): |     def decorator(func): | ||||||
|         @wraps(func) |         if iscoroutinefunction(func): | ||||||
|         def inner_func(*args, **kwargs): |  | ||||||
|             response = func(*args, **kwargs) |  | ||||||
|             patch_vary_headers(response, headers) |  | ||||||
|             return response |  | ||||||
|  |  | ||||||
|         return inner_func |             async def _view_wrapper(request, *args, **kwargs): | ||||||
|  |                 response = await func(request, *args, **kwargs) | ||||||
|  |                 patch_vary_headers(response, headers) | ||||||
|  |                 return response | ||||||
|  |  | ||||||
|  |         else: | ||||||
|  |  | ||||||
|  |             def _view_wrapper(request, *args, **kwargs): | ||||||
|  |                 response = func(request, *args, **kwargs) | ||||||
|  |                 patch_vary_headers(response, headers) | ||||||
|  |                 return response | ||||||
|  |  | ||||||
|  |         return wraps(func)(_view_wrapper) | ||||||
|  |  | ||||||
|     return decorator |     return decorator | ||||||
|  |  | ||||||
|   | |||||||
| @@ -268,6 +268,8 @@ Decorators | |||||||
|   * :func:`~django.views.decorators.http.require_GET` |   * :func:`~django.views.decorators.http.require_GET` | ||||||
|   * :func:`~django.views.decorators.http.require_POST` |   * :func:`~django.views.decorators.http.require_POST` | ||||||
|   * :func:`~django.views.decorators.http.require_safe` |   * :func:`~django.views.decorators.http.require_safe` | ||||||
|  |   * :func:`~django.views.decorators.vary.vary_on_cookie` | ||||||
|  |   * :func:`~django.views.decorators.vary.vary_on_headers` | ||||||
|   * ``xframe_options_deny()`` |   * ``xframe_options_deny()`` | ||||||
|   * ``xframe_options_sameorigin()`` |   * ``xframe_options_sameorigin()`` | ||||||
|   * ``xframe_options_exempt()`` |   * ``xframe_options_exempt()`` | ||||||
|   | |||||||
| @@ -94,6 +94,8 @@ view functions: | |||||||
| * :func:`~django.views.decorators.http.require_GET` | * :func:`~django.views.decorators.http.require_GET` | ||||||
| * :func:`~django.views.decorators.http.require_POST` | * :func:`~django.views.decorators.http.require_POST` | ||||||
| * :func:`~django.views.decorators.http.require_safe` | * :func:`~django.views.decorators.http.require_safe` | ||||||
|  | * :func:`~django.views.decorators.vary.vary_on_cookie` | ||||||
|  | * :func:`~django.views.decorators.vary.vary_on_headers` | ||||||
| * ``xframe_options_deny()`` | * ``xframe_options_deny()`` | ||||||
| * ``xframe_options_sameorigin()`` | * ``xframe_options_sameorigin()`` | ||||||
| * ``xframe_options_exempt()`` | * ``xframe_options_exempt()`` | ||||||
|   | |||||||
| @@ -115,6 +115,10 @@ caching based on specific request headers. | |||||||
|  |  | ||||||
| .. function:: vary_on_cookie(func) | .. function:: vary_on_cookie(func) | ||||||
|  |  | ||||||
|  |     .. versionchanged:: 5.0 | ||||||
|  |  | ||||||
|  |         Support for wrapping asynchronous view functions was added. | ||||||
|  |  | ||||||
| .. function:: vary_on_headers(*headers) | .. function:: vary_on_headers(*headers) | ||||||
|  |  | ||||||
|     The ``Vary`` header defines which request headers a cache mechanism should take |     The ``Vary`` header defines which request headers a cache mechanism should take | ||||||
| @@ -122,6 +126,10 @@ caching based on specific request headers. | |||||||
|  |  | ||||||
|     See :ref:`using vary headers <using-vary-headers>`. |     See :ref:`using vary headers <using-vary-headers>`. | ||||||
|  |  | ||||||
|  |     .. versionchanged:: 5.0 | ||||||
|  |  | ||||||
|  |         Support for wrapping asynchronous view functions was added. | ||||||
|  |  | ||||||
| .. module:: django.views.decorators.cache | .. module:: django.views.decorators.cache | ||||||
|  |  | ||||||
| Caching | Caching | ||||||
|   | |||||||
							
								
								
									
										69
									
								
								tests/decorators/test_vary.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								tests/decorators/test_vary.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | |||||||
|  | from asgiref.sync import iscoroutinefunction | ||||||
|  |  | ||||||
|  | from django.http import HttpRequest, HttpResponse | ||||||
|  | from django.test import SimpleTestCase | ||||||
|  | from django.views.decorators.vary import vary_on_cookie, vary_on_headers | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class VaryOnHeadersTests(SimpleTestCase): | ||||||
|  |     def test_wrapped_sync_function_is_not_coroutine_function(self): | ||||||
|  |         def sync_view(request): | ||||||
|  |             return HttpResponse() | ||||||
|  |  | ||||||
|  |         wrapped_view = vary_on_headers()(sync_view) | ||||||
|  |         self.assertIs(iscoroutinefunction(wrapped_view), False) | ||||||
|  |  | ||||||
|  |     def test_wrapped_async_function_is_coroutine_function(self): | ||||||
|  |         async def async_view(request): | ||||||
|  |             return HttpResponse() | ||||||
|  |  | ||||||
|  |         wrapped_view = vary_on_headers()(async_view) | ||||||
|  |         self.assertIs(iscoroutinefunction(wrapped_view), True) | ||||||
|  |  | ||||||
|  |     def test_vary_on_headers_decorator(self): | ||||||
|  |         @vary_on_headers("Header", "Another-header") | ||||||
|  |         def sync_view(request): | ||||||
|  |             return HttpResponse() | ||||||
|  |  | ||||||
|  |         response = sync_view(HttpRequest()) | ||||||
|  |         self.assertEqual(response.get("Vary"), "Header, Another-header") | ||||||
|  |  | ||||||
|  |     async def test_vary_on_headers_decorator_async_view(self): | ||||||
|  |         @vary_on_headers("Header", "Another-header") | ||||||
|  |         async def async_view(request): | ||||||
|  |             return HttpResponse() | ||||||
|  |  | ||||||
|  |         response = await async_view(HttpRequest()) | ||||||
|  |         self.assertEqual(response.get("Vary"), "Header, Another-header") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class VaryOnCookieTests(SimpleTestCase): | ||||||
|  |     def test_wrapped_sync_function_is_not_coroutine_function(self): | ||||||
|  |         def sync_view(request): | ||||||
|  |             return HttpResponse() | ||||||
|  |  | ||||||
|  |         wrapped_view = vary_on_cookie(sync_view) | ||||||
|  |         self.assertIs(iscoroutinefunction(wrapped_view), False) | ||||||
|  |  | ||||||
|  |     def test_wrapped_async_function_is_coroutine_function(self): | ||||||
|  |         async def async_view(request): | ||||||
|  |             return HttpResponse() | ||||||
|  |  | ||||||
|  |         wrapped_view = vary_on_cookie(async_view) | ||||||
|  |         self.assertIs(iscoroutinefunction(wrapped_view), True) | ||||||
|  |  | ||||||
|  |     def test_vary_on_cookie_decorator(self): | ||||||
|  |         @vary_on_cookie | ||||||
|  |         def sync_view(request): | ||||||
|  |             return HttpResponse() | ||||||
|  |  | ||||||
|  |         response = sync_view(HttpRequest()) | ||||||
|  |         self.assertEqual(response.get("Vary"), "Cookie") | ||||||
|  |  | ||||||
|  |     async def test_vary_on_cookie_decorator_async_view(self): | ||||||
|  |         @vary_on_cookie | ||||||
|  |         async def async_view(request): | ||||||
|  |             return HttpResponse() | ||||||
|  |  | ||||||
|  |         response = await async_view(HttpRequest()) | ||||||
|  |         self.assertEqual(response.get("Vary"), "Cookie") | ||||||
		Reference in New Issue
	
	Block a user