This commit is contained in:
Daler 2024-01-20 14:56:30 +05:00 committed by GitHub
commit 349b870e2c
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 55 additions and 3 deletions

View File

@ -8,6 +8,7 @@ The wrapped request then offers a richer API, in particular :
- full support of PUT method, including support for file uploads
- form overloading of HTTP method, content type and content
"""
import contextvars
import io
import sys
from contextlib import contextmanager
@ -149,6 +150,8 @@ class Request:
authenticating the request's user.
"""
_context_instance = contextvars.ContextVar('request')
def __init__(self, request, parsers=None, authenticators=None,
negotiator=None, parser_context=None):
assert isinstance(request, HttpRequest), (
@ -458,3 +461,11 @@ class Request:
# Hack to allow our exception handler to force choice of
# plaintext or html error responses.
self._request.is_ajax = lambda: value
@classmethod
def get_current(cls, default=None):
return cls._context_instance.get(default)
@classmethod
def set_current(cls, value):
return cls._context_instance.set(value)

View File

@ -28,7 +28,8 @@ from django.utils.translation import gettext_lazy as _
from rest_framework.compat import postgres_fields
from rest_framework.exceptions import ErrorDetail, ValidationError
from rest_framework.fields import get_error_detail
from rest_framework.fields import get_error_detail, set_value
from rest_framework.request import Request
from rest_framework.settings import api_settings
from rest_framework.utils import html, model_meta, representation
from rest_framework.utils.field_mapping import (
@ -115,6 +116,12 @@ class BaseSerializer(Field):
self.partial = kwargs.pop('partial', False)
self._context = kwargs.pop('context', {})
kwargs.pop('many', None)
if isinstance(self._context, dict) and ('request' not in self._context):
request = Request.get_current()
if request:
self._context['request'] = request
super().__init__(**kwargs)
def __new__(cls, *args, **kwargs):

View File

@ -489,9 +489,11 @@ class APIView(View):
"""
self.args = args
self.kwargs = kwargs
self.headers = self.default_response_headers # deprecate?
request = self.initialize_request(request, *args, **kwargs)
self.request = request
self.headers = self.default_response_headers # deprecate?
Request.set_current(request)
try:
self.initial(request, *args, **kwargs)

View File

@ -637,6 +637,8 @@ class Test2555Regression:
nested = NestedSerializer()
serializer = ParentSerializer(data={}, context={'foo': 'bar'})
serializer.context.pop('request', None)
serializer.fields['nested'].context.pop('request', None)
assert serializer.context == {'foo': 'bar'}
assert serializer.fields['nested'].context == {'foo': 'bar'}

View File

@ -1,9 +1,12 @@
import copy
import sys
import pytest
from django.test import TestCase
from rest_framework import status
from rest_framework import serializers, status
from rest_framework.decorators import api_view
from rest_framework.request import Request
from rest_framework.response import Response
from rest_framework.settings import APISettings, api_settings
from rest_framework.test import APIRequestFactory
@ -22,6 +25,16 @@ class BasicView(APIView):
return Response({'method': 'POST', 'data': request.data})
class ViewWithSerializer(APIView):
serializer_class = serializers.Serializer
def get(self, request, *args, **kwargs):
serializer = self.serializer_class()
response = Response()
response.serializer = serializer
return response
@api_view(['GET', 'POST', 'PUT', 'PATCH'])
def basic_view(request):
if request.method == 'GET':
@ -136,3 +149,20 @@ class TestCustomSettings(TestCase):
response = self.view(request)
assert response.status_code == 400
assert response.data == {'error': 'SyntaxError'}
class TestRequestContextVar(TestCase):
def setUp(self):
self.view = ViewWithSerializer.as_view()
@pytest.mark.skipif(
sys.version_info < (3, 7),
reason="contextvars require Python 3.7 or higher",
)
def test_request_context_var(self):
request = factory.get('/', content_type='application/json')
response = self.view(request)
assert Request.get_current() is not None
assert Request.get_current().method == request.method
assert Request.get_current().path == request.path
assert response.serializer.context['request'] is Request.get_current()