Cache a django view that has URL parameters

孤者浪人 提交于 2020-01-10 08:24:23

问题


I have recently implemented Django's excellent cache framework. However from what I understand Django will not cache a view that is passed parameters in a get request. I have an Ajax view that is passed get parameters that I would like to cache for X seconds, what would be an easy way to do this?

In psuedo code I currently have a URL:

http://mysites/ajaxthing/?user=foo&items=10

I would like to cache any this url as long as it has the same get parameters.

I'm currently using the cache decorators in my view:

myview(stuff)

myview = cache_page(myview, 60 * 3)

I did read about django's vary headers but it went a little over my head, and I'm not even sure its the correct solution


回答1:


Right, vary headers is not the correct solution, it's used when you want to cache based on client request headers like user-agent etc.

You'll need to use low-level API or template fragment caching. It depends on your views really.

With low-level API it looks something like this:

from django.core.cache import cache

def get_user(request):
    user_id = request.GET.get("user_id")
    user = cache.get("user_id_%s"%user_id)
    if user is None:
        user = User.objects.get(pk=user_id)
        cache.set("user_id_%s"%user_id, user, 10*60) # 10 minutes
    ...
    ..
    .



回答2:


Yes, you can use django-view-cache-utils, here is code for your case:

from view_cache_utils import cache_page_with_prefix
from django.utils.hashcompat import md5_constructor
...
@cache_page_with_prefix(60*15, lambda request: md5_constructor(request.get_full_path()).hexdigest())
def my_view(request):
    ...



回答3:


It appears that you no longer need to do anything more complicated than placing @cache_page([length of time]) above your View function you are trying to cache, irrespective of whether you have parameters in the URL.

For example, if you have a url like:

http://example.com/user/some_user_id

Your view function in views.py would look something like this:

from django.views.decorators.cache import cache_page
...

@cache_page(60 * 10)
def get_user_detail(request, user_id=None):
    ...
    return render(...)



回答4:


This should be no longer an issue in Django 1.3+. See: https://docs.djangoproject.com/en/dev/topics/cache/#using-vary-headers




回答5:


From my reading of the source code and empirical testing, the @cache_page decorator natively handles GET parameters correctly (at least in Django 2.2).

Digging through the source:

  1. The decorator is defined in django.views.decorators.cache
  2. It calls django.utils.decorators.decorator_from_middleware_with_args()
  3. Which calls django.utils.decorators.make_middleware_decorator()
  4. Which is a silly level of complex. A veritable onion of functions returning functions. The important bit is it calls middleware.process_request() where 'middleware' is django.middleware.cache.CacheMiddleware.
  5. Which calls django.utils.cache.get_cache_key() to generate the cache key.
  6. Which calls django.utils.cache._generate_cache_header_key().
  7. Which calls request.build_absolute_uri() where 'request' is django.http.request.HttpRequest
  8. Which calls django.http.request.HttpRequest.get_full_path()
  9. Which calls django.http.request.HttpRequest._get_full_path()
  10. Which, finally, includes self.META.get('QUERY_STRING', '') in the string it returns.

On the flip side, when the response is complete a similar path goes through middleware.process_response() which eventually calls django.utils.cache._generate_cache_header_key() to determine where to store the response in the cache.

Empirically, I can see that requests to the decorated view are being cached and the response changes when the GET parameters change.




回答6:


a bit late, but you can use django-view-cache-utils for that.



来源:https://stackoverflow.com/questions/1590372/cache-a-django-view-that-has-url-parameters

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!