3

我想要做的是构建一个自定义版本,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实际上工作正常,所以我知道可以缓存响应,但我不知道我错过了什么。

4

1 回答 1

1

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

于 2020-09-17T21:57:05.400 回答