Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Expire a view-cache in Django?

The @cache_page decorator is awesome. But for my blog I would like to keep a page in cache until someone comments on a post. This sounds like a great idea as people rarely comment so keeping the pages in memcached while nobody comments would be great. I'm thinking that someone must have had this problem before? And this is different than caching per url.

So a solution I'm thinking of is:

@cache_page( 60 * 15, "blog" ); def blog( request ) ... 

And then I'd keep a list of all cache keys used for the blog view and then have way of expire the "blog" cache space. But I'm not super experienced with Django so I'm wondering if someone knows a better way of doing this?

like image 524
Nixarn Avatar asked Feb 15 '10 19:02

Nixarn


2 Answers

This solution works for django versions before 1.7

Here's a solution I wrote to do just what you're talking about on some of my own projects:

def expire_view_cache(view_name, args=[], namespace=None, key_prefix=None):     """     This function allows you to invalidate any view-level cache.          view_name: view function you wish to invalidate or it's named url pattern         args: any arguments passed to the view function         namepace: optioal, if an application namespace is needed         key prefix: for the @cache_page decorator for the function (if any)     """     from django.core.urlresolvers import reverse     from django.http import HttpRequest     from django.utils.cache import get_cache_key     from django.core.cache import cache     # create a fake request object     request = HttpRequest()     # Loookup the request path:     if namespace:         view_name = namespace + ":" + view_name     request.path = reverse(view_name, args=args)     # get cache key, expire if the cached item exists:     key = get_cache_key(request, key_prefix=key_prefix)     if key:         if cache.get(key):             # Delete the cache entry.               #             # Note that there is a possible race condition here, as another              # process / thread may have refreshed the cache between             # the call to cache.get() above, and the cache.set(key, None)              # below.  This may lead to unexpected performance problems under              # severe load.             cache.set(key, None, 0)         return True     return False 

Django keys these caches of the view request, so what this does is creates a fake request object for the cached view, uses that to fetch the cache key, then expires it.

To use it in the way you're talking about, try something like:

from django.db.models.signals import post_save from blog.models import Entry  def invalidate_blog_index(sender, **kwargs):     expire_view_cache("blog")  post_save.connect(invalidate_portfolio_index, sender=Entry) 

So basically, when ever a blog Entry object is saved, invalidate_blog_index is called and the cached view is expired. NB: haven't tested this extensively, but it's worked fine for me so far.

like image 92
mazelife Avatar answered Sep 28 '22 02:09

mazelife


The cache_page decorator will use CacheMiddleware in the end which will generate a cache key based on the request (look at django.utils.cache.get_cache_key) and the key_prefix ("blog" in your case). Note that "blog" is only a prefix, not the whole cache key.

You can get notified via django's post_save signal when a comment is saved, then you can try to build the cache key for the appropriate page(s) and finally say cache.delete(key).

However this requires the cache_key, which is constructed with the request for the previously cached view. This request object is not available when a comment is saved. You could construct the cache key without the proper request object, but this construction happens in a function marked as private (_generate_cache_header_key), so you are not supposed to use this function directly. However, you could build an object that has a path attribute that is the same as for the original cached view and Django wouldn't notice, but I don't recommend that.

The cache_page decorator abstracts caching quite a bit for you and makes it hard to delete a certain cache object directly. You could make up your own keys and handle them in the same way, but this requires some more programming and is not as abstract as the cache_page decorator.

You will also have to delete multiple cache objects when your comments are displayed in multiple views (i.e. index page with comment counts and individual blog entry pages).

To sum up: Django does time based expiration of cache keys for you, but custom deletion of cache keys at the right time is more tricky.

like image 32
stefanw Avatar answered Sep 28 '22 03:09

stefanw