-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
f7ad180
commit 474fc79
Showing
11 changed files
with
199 additions
and
24 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -2,5 +2,6 @@ poetry.lock | |
|
||
*.pyc | ||
.coverage | ||
.python-version | ||
.tox | ||
redis*.db | ||
redis*.db |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,7 +1,101 @@ | ||
# django-cache-mock | ||
|
||
Use in-process mocks to avoid setting up external caches for Django during | ||
development. | ||
|
||
Django has a limited built-in `django.core.cache.backends.locmem.LocMemCache`, | ||
to help development, but Django do some magic to always give you a working | ||
connection. | ||
|
||
I have some reasons to abuse Django cache this way: | ||
|
||
* Thread safety: Django spin one connection per thread to avoid issues with | ||
thread unsafe drivers. | ||
* Good defaults: Django run connections with good defaults. | ||
* Connection reuse: Django already have a pool running and in most cases it is | ||
better to use it. | ||
|
||
## Install | ||
|
||
```shell | ||
$ pip install django-cache-mock | ||
``` | ||
|
||
Also, it is possible to install with the backends you want. | ||
|
||
For `mockcache`, it installs a fork of the original package because it doesn´t | ||
work for new versions of Python. | ||
|
||
```shell | ||
$ pip install django-cache-mock[mockcache] | ||
$ pip install django-cache-mock[fakeredis] | ||
$ pip install django-cache-mock[redislite] | ||
``` | ||
|
||
## How to use | ||
|
||
In your Django settings you already have `CACHES` defined. | ||
|
||
For `memcached`, it's something like that: | ||
|
||
```python | ||
CACHES = { | ||
"default": { | ||
"BACKEND": "django.core.cache.backends.memcached.PyMemcacheCache", | ||
"LOCATION": os.getenv("MEMCACHED_HOSTS"), | ||
"OPTIONS": { | ||
"no_delay": True, | ||
"ignore_exc": True, | ||
"max_pool_size": 4, | ||
"use_pooling": True, | ||
}, | ||
}, | ||
} | ||
``` | ||
|
||
Just make a call to `django_cache_mock.patch` to replace with a mock backend. | ||
|
||
**The lib will patch only when cache LOCATION is not defined.** | ||
|
||
```python | ||
import django_cache_mock | ||
|
||
if DEBUG: # Apply it only in debug mode to be extra careful. | ||
django_cache_mock.patch(CACHES, "default", "mockcache") | ||
``` | ||
|
||
This patch replace cache with a mocked one. For mockcache, | ||
|
||
## Custom cache options | ||
|
||
The `patch` function accepts custom params. It can be used to override mock | ||
behaviours, like the db file `redislite` will use, defined by `LOCATION`: | ||
|
||
```python | ||
django_cache_mock.patch(CACHES, "default", "redislite", {"LOCATION": "data/redis.db"}) | ||
``` | ||
|
||
## How to access connections | ||
|
||
To get Django memcached and redis clients from cache: | ||
|
||
```python | ||
from django.core.cache import caches | ||
|
||
def give_me_memcached(): | ||
return caches["memcached"]._cache | ||
|
||
# for django.core.cache.backends.redis | ||
def give_me_primary_redis(): | ||
return caches["redis"]._cache.get_client() | ||
|
||
def give_me_secondary_redis(): | ||
return caches["redis"]._cache.get_client(write=False) | ||
|
||
# for django-redis | ||
def give_me_primary_redis(): | ||
return caches["redis"].client.get_client() | ||
|
||
def give_me_secondary_redis(): | ||
return caches["redis"].client.get_client(write=False) | ||
``` | ||
pip install django-cache-mock | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
from django_cache_mock.mock import patch | ||
|
||
__all__ = ("patch",) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
import logging | ||
|
||
SUPPORTED_BACKENDS = { | ||
"mockcache": "django_cache_mock.backends.memcached.MockcacheCache", | ||
"fakeredis": "django_cache_mock.backends.redis.FakeRedisCache", | ||
"redislite": "django_cache_mock.backends.redis.RedisLiteCache", | ||
} | ||
|
||
logger = logging.getLogger(__name__) | ||
|
||
|
||
def patch(caches, cache_alias, backend, params=None, *, force=False): | ||
current_config = caches[cache_alias] | ||
location = current_config.get("LOCATION") | ||
if location and not force: | ||
logger.debug(f"Skipped cache {cache_alias} patch because LOCATION is defined.") | ||
return False | ||
|
||
if params is None: | ||
params = {} | ||
|
||
params["BACKEND"] = SUPPORTED_BACKENDS[backend] | ||
logger.info(f"Cache {cache_alias} mocked with {backend}.") | ||
logger.debug(f"{params=}.") | ||
caches[cache_alias] = params | ||
return True |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
from unittest.mock import sentinel | ||
|
||
import pytest | ||
|
||
from django_cache_mock import patch | ||
|
||
|
||
def test_do_not_patch_when_location_is_defined(): | ||
caches = {"default": {"LOCATION": sentinel.location}} | ||
patch(caches, "default", "mockcache") | ||
assert caches["default"]["LOCATION"] == sentinel.location | ||
|
||
|
||
def test_patch_forced_when_location_is_defined(): | ||
caches = {"default": {"LOCATION": sentinel.location}} | ||
patch(caches, "default", "mockcache", force=True) | ||
expected_backend = "django_cache_mock.backends.memcached.MockcacheCache" | ||
assert caches["default"]["BACKEND"] == expected_backend | ||
|
||
|
||
def test_fail_for_cache_unknown(): | ||
with pytest.raises(KeyError): | ||
patch({}, "default", "mockcache") | ||
|
||
|
||
def test_fail_for_backend_unknown(): | ||
with pytest.raises(KeyError): | ||
patch({"default": {}}, "default", "fakefake") | ||
|
||
|
||
def test_custom_params(): | ||
caches = {"default": {"FOO": sentinel.foo}} | ||
patch(caches, "default", "mockcache", {"BAR": sentinel.bar}) | ||
assert "FOO" not in caches["default"] | ||
assert caches["default"]["BAR"] == sentinel.bar |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
import pytest | ||
|
||
from tests.thread_with_exceptions import Thread | ||
|
||
|
||
def test_thread_error(): | ||
"""Just a sane check that _Thread really raises exceptions.""" | ||
thread = Thread(target=lambda: 1 / 0) | ||
thread.start() | ||
|
||
with pytest.raises(ZeroDivisionError): | ||
thread.join() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
import threading | ||
|
||
|
||
class Thread(threading.Thread): | ||
def run(self): | ||
self.exc = None | ||
try: | ||
super().run() | ||
except BaseException as exc: | ||
self.exc = exc | ||
|
||
def join(self): | ||
threading.Thread.join(self) | ||
if self.exc: | ||
raise self.exc |