我想做的是构建一个自定义版本的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)