2013-04-30 13:09:08 +00:00
|
|
|
from ..utils import get_singleton, sanitize_cache_key
|
2013-01-31 08:51:29 +00:00
|
|
|
from django.core.cache import get_cache
|
|
|
|
|
from django.core.exceptions import ImproperlyConfigured
|
|
|
|
|
|
|
|
|
|
|
2013-03-15 03:34:27 +00:00
|
|
|
class CacheFileState(object):
|
|
|
|
|
EXISTS = 'exists'
|
|
|
|
|
PENDING = 'pending'
|
|
|
|
|
DOES_NOT_EXIST = 'does_not_exist'
|
|
|
|
|
|
|
|
|
|
|
2013-02-05 00:39:25 +00:00
|
|
|
def get_default_cachefile_backend():
|
2013-01-31 08:51:29 +00:00
|
|
|
"""
|
|
|
|
|
Get the default file backend.
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
from django.conf import settings
|
2013-02-05 00:39:25 +00:00
|
|
|
return get_singleton(settings.IMAGEKIT_DEFAULT_CACHEFILE_BACKEND,
|
2013-01-31 08:51:29 +00:00
|
|
|
'file backend')
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class InvalidFileBackendError(ImproperlyConfigured):
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
2013-03-15 03:36:21 +00:00
|
|
|
class AbstractCacheFileBackend(object):
|
|
|
|
|
"""
|
|
|
|
|
An abstract cache file backend. This isn't used by any internal classes and
|
|
|
|
|
is included simply to illustrate the minimum interface of a cache file
|
|
|
|
|
backend for users who wish to implement their own.
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
def generate(self, file, force=False):
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
def exists(self, file):
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
|
2013-01-31 08:51:29 +00:00
|
|
|
class CachedFileBackend(object):
|
2013-03-15 05:05:53 +00:00
|
|
|
existence_check_timeout = 5
|
|
|
|
|
"""
|
|
|
|
|
The number of seconds to wait before rechecking to see if the file exists.
|
|
|
|
|
If the image is found to exist, that information will be cached using the
|
|
|
|
|
timeout specified in your CACHES setting (which should be very high).
|
|
|
|
|
However, when the file does not exist, you probably want to check again
|
|
|
|
|
in a relatively short amount of time. This attribute allows you to do that.
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
2013-01-31 08:51:29 +00:00
|
|
|
@property
|
|
|
|
|
def cache(self):
|
|
|
|
|
if not getattr(self, '_cache', None):
|
|
|
|
|
from django.conf import settings
|
|
|
|
|
self._cache = get_cache(settings.IMAGEKIT_CACHE_BACKEND)
|
|
|
|
|
return self._cache
|
|
|
|
|
|
|
|
|
|
def get_key(self, file):
|
|
|
|
|
from django.conf import settings
|
2013-04-30 13:09:08 +00:00
|
|
|
return sanitize_cache_key('%s%s-state' %
|
|
|
|
|
(settings.IMAGEKIT_CACHE_PREFIX, file.name))
|
2013-01-31 08:51:29 +00:00
|
|
|
|
2013-03-15 03:34:27 +00:00
|
|
|
def get_state(self, file):
|
2013-01-31 08:51:29 +00:00
|
|
|
key = self.get_key(file)
|
2013-03-15 03:34:27 +00:00
|
|
|
state = self.cache.get(key)
|
|
|
|
|
if state is None:
|
2013-03-15 02:58:28 +00:00
|
|
|
exists = self._exists(file)
|
2013-03-15 03:34:27 +00:00
|
|
|
state = CacheFileState.EXISTS if exists else CacheFileState.DOES_NOT_EXIST
|
|
|
|
|
self.set_state(file, state)
|
|
|
|
|
return state
|
|
|
|
|
|
|
|
|
|
def set_state(self, file, state):
|
|
|
|
|
key = self.get_key(file)
|
2013-03-15 05:05:53 +00:00
|
|
|
if state is CacheFileState.DOES_NOT_EXIST:
|
|
|
|
|
self.cache.set(key, state, self.existence_check_timeout)
|
|
|
|
|
else:
|
|
|
|
|
self.cache.set(key, state)
|
2013-01-31 08:51:29 +00:00
|
|
|
|
2013-03-15 03:34:27 +00:00
|
|
|
def exists(self, file):
|
|
|
|
|
return self.get_state(file) is CacheFileState.EXISTS
|
|
|
|
|
|
|
|
|
|
def generate(self, file, force=False):
|
2013-04-03 02:55:44 +00:00
|
|
|
raise NotImplementedError
|
2013-04-03 02:37:52 +00:00
|
|
|
|
|
|
|
|
def generate_now(self, file, force=False):
|
|
|
|
|
if force or self.get_state(file) is CacheFileState.DOES_NOT_EXIST:
|
2013-03-15 03:34:27 +00:00
|
|
|
file._generate()
|
2013-03-15 04:30:58 +00:00
|
|
|
self.set_state(file, CacheFileState.EXISTS)
|
2013-01-31 08:51:29 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
class Simple(CachedFileBackend):
|
|
|
|
|
"""
|
|
|
|
|
The most basic file backend. The storage is consulted to see if the file
|
2013-03-15 03:34:27 +00:00
|
|
|
exists. Files are generated synchronously.
|
2013-01-31 08:51:29 +00:00
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
2013-04-03 02:55:44 +00:00
|
|
|
def generate(self, file, force=False):
|
|
|
|
|
self.generate_now(file, force=force)
|
2013-03-15 03:34:27 +00:00
|
|
|
|
2013-03-15 02:58:28 +00:00
|
|
|
def _exists(self, file):
|
2013-03-15 04:49:24 +00:00
|
|
|
return bool(getattr(file, '_file', None)
|
|
|
|
|
or file.storage.exists(file.name))
|
2013-04-03 02:29:53 +00:00
|
|
|
|
|
|
|
|
|
2013-04-03 02:55:44 +00:00
|
|
|
def _generate_file(backend, file, force=False):
|
|
|
|
|
backend.generate_now(file, force=force)
|
2013-04-03 02:29:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
import celery
|
|
|
|
|
except ImportError:
|
|
|
|
|
pass
|
|
|
|
|
else:
|
|
|
|
|
_generate_file = celery.task(ignore_result=True)(_generate_file)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Async(Simple):
|
|
|
|
|
"""
|
|
|
|
|
A backend that uses Celery to generate the images.
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
|
try:
|
|
|
|
|
import celery
|
|
|
|
|
except ImportError:
|
|
|
|
|
raise ImproperlyConfigured('You must install celery to use'
|
|
|
|
|
' imagekit.cachefiles.backend.Async.')
|
|
|
|
|
super(Async, self).__init__(*args, **kwargs)
|
|
|
|
|
|
2013-04-03 02:55:44 +00:00
|
|
|
def generate(self, file, force=False):
|
2013-04-03 02:29:53 +00:00
|
|
|
self.set_state(file, CacheFileState.PENDING)
|
2013-04-03 02:55:44 +00:00
|
|
|
_generate_file.delay(self, file, force=force)
|