Django Rest Framework:泡菜响应



我想做的是构建一个自定义版本的cache_page,在那里我可以更好地控制缓存密钥,但我甚至被我的响应的基本缓存所困扰:

from django.core.cache import cache
from rest_framework import viewsets
from rest_framework.response import Response
from app import models
class BaseViewSet(viewsets.GenericViewSet):
    queryset = models.Items.objects.all()
    def get_queryset(self):
        return models.Items.objects.all()
    def list(self, request, **kwargs):
        response = Response({})
        cache.set('test', response, 10)
        return response

其中我的settings.py的相关部分设置为:

REST_FRAMEWORK = {
    'DEFAULT_RENDERER_CLASSES': [
        'rest_framework.renderers.JSONRenderer',
    ],
    'DEFAULT_VERSIONING_CLASS': 'rest_framework.versioning.NamespaceVersioning'
}
CACHES = {
    "default": {
        "BACKEND": "django_redis.cache.RedisCache",
        "LOCATION": f"redis://127.0.0.1:6729/1",
    },
}

当我尝试调用端点时,我得到:

django.template.response.ContentNotRenderedError: The response content must be rendered before it can be pickled.

然后,如果我将行更改为:

cache.set('test', response.render(), 10)

我得到:

AssertionError: .accepted_renderer not set on Response

(如果我设置了渲染器,它会抱怨接受的媒体,然后上下文,最后以TypeError: 'bytes' object is not callable失败(

尽管API调用本身在没有缓存的情况下运行良好。

cache_page实际上工作得很好,所以我知道可以缓存响应,但我不知道我缺少了什么。

我通过让视图返回django.http.JsonResponse而不是rest_framework.response.Response对象来解决这个问题。问题是,rest_framework Response与中介类型无关,而且据我所知,它依赖于在视图的调用链中设置渲染器和可接受的媒体值,在它离开处理程序之后。由于JsonResponse只处理json,所以它不必经历这个";内容协商";。

cache_page

因此而工作:

if hasattr(response, 'render') and callable(response.render):
            response.add_post_render_callback(lambda r: self.cache.set(cache_key, r, timeout))
        else:
            self.cache.set(cache_key, response, timeout)

相关内容

  • 没有找到相关文章

最新更新