Я создаю промежуточное ПО Django для блокировки пользователя, когда он(а) подвергается дросселированию более 5 раз, но я получаю ContentNotRenderedError
.
Traceback (most recent call last):
File "/home/raptor/Application/utilities/anaconda3/envs/slic4rapi/lib/python3.8/site-packages/django/core/handlers/exception.py", line 47, in inner
response = get_response(request)
File "/home/raptor/Application/utilities/anaconda3/envs/slic4rapi/lib/python3.8/site-packages/django/utils/deprecation.py", line 119, in __call__
response = self.process_response(request, response)
File "/home/raptor/Application/utilities/anaconda3/envs/slic4rapi/lib/python3.8/site-packages/django/middleware/common.py", line 113, in process_response
response.headers['Content-Length'] = str(len(response.content))
File "/home/raptor/Application/utilities/anaconda3/envs/slic4rapi/lib/python3.8/site-packages/django/template/response.py", line 126, in content
raise ContentNotRenderedError(
django.template.response.ContentNotRenderedError: The response content must be rendered before it can be accessed.
[22/Mar/2022 11:55:03] "GET /api/v1/userdetail/ HTTP/1.1" 500 84321
class BlockMiddleware:
def __init__(self, get_response):
self.get_response = get_response
# some other variables
def __call__(self, request):
# handle_blocking will return "None" if user can access application else "rest_framework.response.Response" object
blocking_res = self.handle_blocking(request)
if blocking_res:
return blocking_res
response = self.get_response(request)
# logic for counting how many throttles have left
# then
if throttles_left <= 0:
return Response(
data='User is blocked due to exceeding throttles limit.',
status=status.HTTP_403_FORBIDDEN
)
else:
return response
Пример возврата функции handle_blocking
:
return Response(
data='User is blocked, please contact the support team.',
status=status.HTTP_403_FORBIDDEN
)
Это работает, когда я удаляю промежуточное ПО и не использую класс Response
(вместо этого возвращаю self.get_response(request)).
Я не могу понять ошибку. Что я делаю не так?
Если вы действительно хотите вернуть экземпляр Response
, вам нужно установить некоторые свойства перед его возвратом:
from rest_framework.renderers import JSONRenderer
class BlockMiddleware:
def __init__(self, get_response):
self.get_response = get_response
def __call__(self, request):
blocking_res = self.handle_blocking(request)
if blocking_res:
return blocking_res
response = self.get_response(request)
if throttles_left <= 0:
response = Response(
data='User is blocked due to exceeding throttles limit.',
status=status.HTTP_403_FORBIDDEN
)
response.accepted_renderer = JSONRenderer()
response.accepted_media_type = "application/json"
response.renderer_context = {}
response.render()
return response
else:
return response
Вам нужно сделать то же самое и в вашей реализации handle_blocking
, например:
from rest_framework.renderers import JSONRenderer
response = Response(
data='User is blocked, please contact the support team.',
status=status.HTTP_403_FORBIDDEN
)
response.accepted_renderer = JSONRenderer()
response.accepted_media_type = "application/json"
response.renderer_context = {}
response.render()
return response
Эти свойства обычно устанавливаются декоратором api_view
, который нельзя использовать с промежуточным программным обеспечением, поэтому вы должны установить их вручную.
@rain Я отредактировал свой ответ, теперь он работает с Response
Не работает, вызывает ту же ошибку, но да, это происходит из-за декоратора api_view
. Пожалуйста, верните свой ответ, согласившись с тем, что это было бы более уместно.
@rain Попробуйте вызвать render
экземпляр Response
после установки других свойств (я обновил свой ответ, чтобы показать это)
Да, это сработало. Можете ли вы поделиться какой-либо ссылкой на документ / ссылку, которую вы использовали?
@дождь Отлично. Конечно, я прочитал документацию о Объект ответа, а затем попытался установить различные свойства Response
.
Привет, спасибо за ответ. Я уже пробовал
django.http.JsonResponse
, и он работал, но проблема вrest_framework.response.Response
, у меня есть другое промежуточное ПО, и там оно работает нормально.