Skip to content

Commit 44bf3b3

Browse files
committed
initial commit
0 parents  commit 44bf3b3

13 files changed

+277
-0
lines changed

.gitignore

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
*.py[cod]
2+
__pycache__
3+
4+
# C extensions
5+
*.so
6+
7+
# Packages
8+
*.egg
9+
*.egg-info
10+
dist
11+
build
12+
eggs
13+
parts
14+
var
15+
sdist
16+
develop-eggs
17+
.installed.cfg
18+
lib
19+
lib64
20+
21+
# Installer logs
22+
pip-log.txt
23+
24+
# Unit test / coverage reports
25+
.coverage
26+
.tox
27+
nosetests.xml
28+
29+
# Mr Developer
30+
.mr.developer.cfg
31+
.project
32+
.pydevproject
33+
34+
# Backup versions of files [Kw’s scheme]
35+
o.*
36+
37+
# GitHub token file
38+
.pypt/gh-token

.travis.yml

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
language: python
2+
sudo: false
3+
python:
4+
- "2.7"
5+
- "3.3"
6+
- "3.4"
7+
- "3.5"
8+
- "pypy"
9+
10+
install:
11+
- "pip install -r requirements.txt"
12+
- "pip install ."
13+
14+
script:
15+
- "py.test --cov cache_fallback --cov-report term-missing tests/"
16+
17+
notifications:
18+
email:
19+
on_success: change
20+
on_failure: always

AUTHORS

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
Jakub S

LICENSE

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
MIT License
2+
3+
Copyright (c) 2016 Jakub S.
4+
5+
Permission is hereby granted, free of charge, to any person obtaining a copy
6+
of this software and associated documentation files (the "Software"), to deal
7+
in the Software without restriction, including without limitation the rights
8+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9+
copies of the Software, and to permit persons to whom the Software is
10+
furnished to do so, subject to the following conditions:
11+
12+
The above copyright notice and this permission notice shall be included in all
13+
copies or substantial portions of the Software.
14+
15+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21+
SOFTWARE.

MANIFEST.in

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
graft cache_fallback
2+
graft docs
3+
graft tests
4+
include README.rst AUTHORS LICENSE CHANGELOG.rst setup.py setup.cfg requirements.txt
5+
global-exclude __pycache__ *.pyc

README.rst

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
==============================================================================
2+
Django Cache Fallback.
3+
==============================================================================
4+
:Info: This is the README file for Django Cache Fallback.
5+
:Author: Jakub S
6+
7+
.. index: README
8+
.. image:: https://travis-ci.org/Kub-AT/django-cache-fallback.svg?branch=master
9+
:target: https://travis-ci.org/Kub-AT/django-cache-fallback
10+
11+
PURPOSE
12+
-------
13+
Allows you to set fallback cache backend.
14+
The data is not shared between cache backends.
15+
Example: Memcached is not available, backend switch to fallback. Site may slow down (cache have to be set)
16+
but will not rise an error (watch your logs)
17+
18+
INSTALLATION
19+
------------
20+
pip install django-cache-fallback
21+
22+
INSTALLED_APPS = (
23+
...
24+
'cache_fallback',
25+
)
26+
27+
USAGE
28+
-----
29+
Usage example PyLibMCCache + LocMemCache
30+
31+
.. code:: python
32+
33+
CACHES = {
34+
'default': {
35+
'BACKEND': 'cache_fallback.FallbackCache',
36+
},
37+
38+
'main_cache': {
39+
'BACKEND': 'django.core.cache.backends.memcached.PyLibMCCache',
40+
'LOCATION': '/tmp/memcached.sock',
41+
'TIMEOUT': 500,
42+
},
43+
'fallback_cache': {
44+
'BACKEND': 'django.core.cache.backends.locmem.LocMemCache',
45+
'LOCATION': 'unique'
46+
}
47+
48+
}
49+
50+
NOTES
51+
-----

cache_fallback/__init__.py

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
## -*- coding: utf-8 -*-
2+
from __future__ import unicode_literals
3+
4+
from .cache import FallbackCache
5+
6+
__title__ = 'Django Cache Fallback'
7+
8+
__all__ = (FallbackCache, )
9+

cache_fallback/cache.py

+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
# -*- coding: utf-8 -*-
2+
from __future__ import unicode_literals
3+
4+
import logging
5+
from django.core import signals
6+
from django.core.cache.backends.base import BaseCache
7+
8+
logging.basicConfig()
9+
logger = logging.getLogger(__name__)
10+
11+
12+
13+
def get_cache(backend, **kwargs):
14+
"""
15+
Compatibilty wrapper for getting Django's cache backend instance
16+
17+
original source:
18+
https://github.com/vstoykov/django-imagekit/commit/c26f8a0538778969a64ee471ce99b25a04865a8e
19+
"""
20+
from django.core import cache
21+
22+
# Django < 1.7
23+
if not hasattr(cache, '_create_cache'):
24+
return cache.get_cache(backend, **kwargs)
25+
26+
cache = cache._create_cache(backend, **kwargs)
27+
# Some caches -- python-memcached in particular -- need to do a cleanup at the
28+
# end of a request cycle. If not implemented in a particular backend
29+
# cache.close is a no-op. Not available in Django 1.5
30+
if hasattr(cache, 'close'):
31+
signals.request_finished.connect(cache.close)
32+
return cache
33+
34+
35+
class FallbackCache(BaseCache):
36+
_cache = None
37+
_cache_fallback = None
38+
39+
def __init__(self, host, *args, **kwargs):
40+
BaseCache.__init__(self, *args, **kwargs)
41+
self._cache = get_cache('main_cache')
42+
self._cache_fallback = get_cache('fallback_cache')
43+
44+
def _call_with_fallback(self, method, *args, **kwargs):
45+
try:
46+
return getattr(self._cache, method)(*args, **kwargs)
47+
except Exception as e:
48+
logger.exception(e)
49+
return getattr(self._cache_fallback, method)(*args, **kwargs)
50+
51+
def add(self, key, value, timeout=None, version=None):
52+
return self._call_with_fallback('add', key, value, timeout=None, version=None)
53+
54+
def get(self, key, default=None, version=None):
55+
return self._call_with_fallback('get', key, default=default, version=version)
56+
57+
def set(self, key, value, timeout=None, version=None, client=None):
58+
return self._call_with_fallback('set', key, value, timeout=timeout, version=version)
59+
60+
def delete(self, key, version=None):
61+
return self._call_with_fallback('delete', key, version=version)
62+
63+
def clear(self):
64+
return self._call_with_fallback('clear')

requirements.txt

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
django
2+
pytest
3+
coverage
4+
pytest-cov

setup.cfg

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
[wheel]
2+
universal = 1
3+
4+
[tool:pytest]
5+
norecursedirs = .git
6+
addopts = --cov=cache_fallback --cov-report term-missing
7+
8+
[coverage:run]
9+
branch = True
10+
omit = tests/*

setup.py

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
#!/usr/bin/env python
2+
# -*- coding: utf-8 -*-
3+
import sys
4+
import io
5+
from setuptools import setup, find_packages
6+
7+
8+
with open('requirements.txt') as f:
9+
requirements = [line.strip() for line in f.readlines()]
10+
11+
12+
setup(
13+
name='django-cache-fallback',
14+
version='0.1.0',
15+
description='Django Cache Fallback',
16+
keywords='django-cache-fallback, django cache, multiple cache, fallback cache',
17+
author='Jakub S',
18+
author_email='@',
19+
url='https://github.com/Kub-AT/django-cache-fallback',
20+
license='3-clause BSD',
21+
long_description=io.open('./README.rst', 'r', encoding='utf-8').read(),
22+
platforms='any',
23+
zip_safe=False,
24+
# http://pypi.python.org/pypi?%3Aaction=list_classifiers
25+
# 'Development Status :: 4 - Beta',
26+
classifiers=[
27+
'Programming Language :: Python',
28+
'Programming Language :: Python :: 2',
29+
'Programming Language :: Python :: 2.7',
30+
'Programming Language :: Python :: 3',
31+
'Programming Language :: Python :: 3.3',
32+
'Programming Language :: Python :: 3.4',
33+
'Programming Language :: Python :: 3.5'
34+
],
35+
packages=find_packages(exclude=('tests',)),
36+
include_package_data=True,
37+
install_requires=requirements,
38+
)

tests/__init__.py

Whitespace-only changes.

tests/test_sanity.py

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
#!/usr/bin/env python
2+
# -*- coding: utf-8 -*-
3+
from __future__ import unicode_literals
4+
5+
import cache_fallback
6+
import cache_fallback.cache
7+
8+
9+
def test_true():
10+
"""Test if True is truthy."""
11+
assert True
12+
13+
def test_import():
14+
"""Test imports."""
15+
cache_fallback
16+
cache_fallback.cache

0 commit comments

Comments
 (0)