2013-04-23 18:51:17 +02:00
|
|
|
from __future__ import absolute_import
|
|
|
|
|
2012-11-02 00:23:26 +01:00
|
|
|
from functools import wraps
|
|
|
|
|
2013-01-09 20:35:19 +01:00
|
|
|
from django.core.cache import cache as djcache
|
2013-03-11 16:23:34 +01:00
|
|
|
from django.core.cache import get_cache
|
2012-09-19 18:41:20 +02:00
|
|
|
|
2013-04-23 18:51:17 +02:00
|
|
|
from zephyr.lib.utils import statsd, statsd_key, make_safe_digest
|
2013-04-19 00:00:33 +02:00
|
|
|
import time
|
|
|
|
|
2013-05-10 16:57:06 +02:00
|
|
|
memcached_time_start = 0
|
2013-04-19 00:00:33 +02:00
|
|
|
memcached_total_time = 0
|
|
|
|
memcached_total_requests = 0
|
|
|
|
|
|
|
|
def get_memcached_time():
|
|
|
|
return memcached_total_time
|
|
|
|
|
|
|
|
def get_memcached_requests():
|
|
|
|
return memcached_total_requests
|
|
|
|
|
2013-05-10 16:57:06 +02:00
|
|
|
def memcached_stats_start():
|
|
|
|
global memcached_time_start
|
|
|
|
memcached_time_start = time.time()
|
|
|
|
|
|
|
|
def memcached_stats_finish():
|
|
|
|
global memcached_total_time
|
|
|
|
global memcached_total_requests
|
|
|
|
global memcached_time_start
|
|
|
|
memcached_total_requests += 1
|
|
|
|
memcached_total_time += (time.time() - memcached_time_start)
|
2013-04-16 22:58:21 +02:00
|
|
|
|
2013-05-11 15:50:02 +02:00
|
|
|
def cache_with_key(keyfunc, cache_name=None, timeout=None, with_statsd_key=None):
|
2012-09-19 18:41:20 +02:00
|
|
|
"""Decorator which applies Django caching to a function.
|
|
|
|
|
|
|
|
Decorator argument is a function which computes a cache key
|
|
|
|
from the original function's arguments. You are responsible
|
|
|
|
for avoiding collisions with other uses of this decorator or
|
|
|
|
other uses of caching."""
|
|
|
|
|
|
|
|
def decorator(func):
|
2012-11-02 00:23:26 +01:00
|
|
|
@wraps(func)
|
2012-09-19 18:41:20 +02:00
|
|
|
def func_with_caching(*args, **kwargs):
|
2013-05-10 16:57:06 +02:00
|
|
|
key = keyfunc(*args, **kwargs)
|
|
|
|
|
|
|
|
memcached_stats_start()
|
2013-03-13 18:36:58 +01:00
|
|
|
if cache_name is None:
|
|
|
|
cache_backend = djcache
|
|
|
|
else:
|
|
|
|
cache_backend = get_cache(cache_name)
|
|
|
|
|
|
|
|
val = cache_backend.get(key)
|
2013-05-10 16:57:06 +02:00
|
|
|
memcached_stats_finish()
|
2012-09-19 18:41:20 +02:00
|
|
|
|
2013-05-11 15:50:02 +02:00
|
|
|
extra = ""
|
|
|
|
if cache_name == 'database':
|
|
|
|
extra = ".dbcache"
|
|
|
|
|
|
|
|
if with_statsd_key is not None:
|
|
|
|
metric_key = with_statsd_key
|
2013-04-16 22:58:21 +02:00
|
|
|
else:
|
2013-05-11 15:50:02 +02:00
|
|
|
metric_key = statsd_key(key)
|
|
|
|
|
|
|
|
status = "hit" if val is not None else "miss"
|
|
|
|
statsd.incr("cache%s.%s.%s" % (extra, metric_key, status))
|
2013-04-16 22:58:21 +02:00
|
|
|
|
2012-09-19 18:41:20 +02:00
|
|
|
# Values are singleton tuples so that we can distinguish
|
|
|
|
# a result of None from a missing key.
|
|
|
|
if val is not None:
|
|
|
|
return val[0]
|
|
|
|
|
|
|
|
val = func(*args, **kwargs)
|
2013-04-19 00:00:33 +02:00
|
|
|
|
2013-05-10 16:57:06 +02:00
|
|
|
memcached_stats_start()
|
2013-03-15 19:51:19 +01:00
|
|
|
cache_backend.set(key, (val,), timeout=timeout)
|
2013-05-10 16:57:06 +02:00
|
|
|
memcached_stats_finish()
|
2013-04-19 00:00:33 +02:00
|
|
|
|
2012-09-19 18:41:20 +02:00
|
|
|
return val
|
|
|
|
|
|
|
|
return func_with_caching
|
|
|
|
|
|
|
|
return decorator
|
|
|
|
|
2013-04-22 16:29:57 +02:00
|
|
|
def cache_get_many(keys, cache_name=None):
|
2013-05-10 16:57:06 +02:00
|
|
|
memcached_stats_start()
|
2013-04-22 16:29:57 +02:00
|
|
|
if cache_name is None:
|
|
|
|
cache_backend = djcache
|
|
|
|
else:
|
|
|
|
cache_backend = get_cache(cache_name)
|
2013-05-10 16:57:06 +02:00
|
|
|
ret = cache_backend.get_many(keys)
|
|
|
|
memcached_stats_finish()
|
|
|
|
return ret
|
2013-04-22 16:29:57 +02:00
|
|
|
|
2013-04-25 20:41:54 +02:00
|
|
|
def cache_set_many(items, cache_name=None):
|
2013-05-10 16:57:06 +02:00
|
|
|
memcached_stats_start()
|
2013-04-25 20:41:54 +02:00
|
|
|
if cache_name is None:
|
|
|
|
cache_backend = djcache
|
|
|
|
else:
|
|
|
|
cache_backend = get_cache(cache_name)
|
2013-05-10 16:57:06 +02:00
|
|
|
ret = cache_backend.set_many(items)
|
|
|
|
memcached_stats_finish()
|
|
|
|
return ret
|
2013-04-25 20:41:54 +02:00
|
|
|
|
2012-09-19 18:41:20 +02:00
|
|
|
def cache(func):
|
|
|
|
"""Decorator which applies Django caching to a function.
|
|
|
|
|
|
|
|
Uses a key based on the function's name, filename, and
|
|
|
|
the repr() of its arguments."""
|
|
|
|
|
|
|
|
func_uniqifier = '%s-%s' % (func.func_code.co_filename, func.func_name)
|
|
|
|
|
2012-11-02 00:23:26 +01:00
|
|
|
@wraps(func)
|
2012-09-19 18:41:20 +02:00
|
|
|
def keyfunc(*args, **kwargs):
|
|
|
|
# Django complains about spaces because memcached rejects them
|
|
|
|
key = func_uniqifier + repr((args, kwargs))
|
|
|
|
return key.replace('-','--').replace(' ','-s')
|
|
|
|
|
|
|
|
return cache_with_key(keyfunc)(func)
|
2013-03-13 18:49:29 +01:00
|
|
|
|
|
|
|
def message_cache_key(message_id):
|
|
|
|
return "message:%d" % (message_id,)
|
|
|
|
|
2013-03-18 20:56:32 +01:00
|
|
|
def user_profile_by_email_cache_key(email):
|
2013-03-20 15:31:27 +01:00
|
|
|
# See the comment in zephyr/lib/avatar.py:gravatar_hash for why we
|
|
|
|
# are proactively encoding email addresses even though they will
|
|
|
|
# with high likelihood be ASCII-only for the foreseeable future.
|
|
|
|
return 'user_profile_by_email:%s' % (make_safe_digest(email),)
|
2013-03-13 18:49:29 +01:00
|
|
|
|
2013-03-18 17:10:45 +01:00
|
|
|
def user_profile_by_id_cache_key(user_profile_id):
|
|
|
|
return "user_profile_by_id:%s" % (user_profile_id,)
|
|
|
|
|
2013-03-15 21:17:32 +01:00
|
|
|
# Called by models.py to flush the user_profile cache whenever we save
|
|
|
|
# a user_profile object
|
|
|
|
def update_user_profile_cache(sender, **kwargs):
|
|
|
|
user_profile = kwargs['instance']
|
|
|
|
items_for_memcached = {}
|
2013-03-28 20:43:34 +01:00
|
|
|
items_for_memcached[user_profile_by_email_cache_key(user_profile.email)] = (user_profile,)
|
2013-03-18 17:10:45 +01:00
|
|
|
items_for_memcached[user_profile_by_id_cache_key(user_profile.id)] = (user_profile,)
|
2013-03-15 21:17:32 +01:00
|
|
|
djcache.set_many(items_for_memcached)
|
2013-04-05 00:13:03 +02:00
|
|
|
|
|
|
|
def status_dict_cache_key(user_profile):
|
|
|
|
return "status_dict:%d" % (user_profile.realm_id,)
|
|
|
|
|
|
|
|
def update_user_presence_cache(sender, **kwargs):
|
|
|
|
user_profile = kwargs['instance'].user_profile
|
|
|
|
djcache.delete(status_dict_cache_key(user_profile))
|