django-components/tests/test_component_cache.py
Oliver Haas e0b718c314
test: test to illustrate bug in component caching when using include tag (#1135)
* test: test to illustrate bug in component caching when using include tag

* fix: add cleanup for render context in component rendering

* refactor: clarify cleanup comment in component rendering logic

* refactor: fix linter errors

* test: formatting and unnecessary test setup

---------

Co-authored-by: Juro Oravec <juraj.oravec.josefson@gmail.com>
2025-04-20 11:53:06 +02:00

235 lines
7.1 KiB
Python

import time
from typing import Any
from django.core.cache import caches
from django.template import Template
from django.template.context import Context
from django_components import Component, register
from django_components.testing import djc_test
from .testutils import setup_test_config
setup_test_config({"autodiscover": False})
# Common settings for all tests
@djc_test(
django_settings={
"CACHES": {
"default": {
"BACKEND": "django.core.cache.backends.locmem.LocMemCache",
},
},
},
)
class TestComponentCache:
def test_cache_enabled(self):
did_call_get = False
class TestComponent(Component):
template = "Hello"
class Cache:
enabled = True
def get_context_data(self, **kwargs: Any):
nonlocal did_call_get
did_call_get = True
return {}
# First render
component = TestComponent()
result = component.render()
assert did_call_get
assert result == "Hello"
# Check if the cache entry is set
cache_key = component.cache.get_cache_key()
assert cache_key == "components:cache:TestComponent_c9770f::"
assert component.cache.get_entry(cache_key) == "<!-- _RENDERED TestComponent_c9770f,ca1bc3e,, -->Hello"
assert caches["default"].get(cache_key) == "<!-- _RENDERED TestComponent_c9770f,ca1bc3e,, -->Hello"
# Second render
did_call_get = False
component.render()
# get_context_data not called because the cache entry was returned
assert not did_call_get
assert result == "Hello"
def test_cache_disabled(self):
did_call_get = False
class TestComponent(Component):
template = "Hello"
class Cache:
enabled = False
def get_context_data(self, **kwargs: Any):
nonlocal did_call_get
did_call_get = True
return {}
# First render
component = TestComponent()
result = component.render()
assert did_call_get
assert result == "Hello"
# Check if the cache entry is not set
cache_instance = component.cache
cache_key = cache_instance.get_cache_key()
assert cache_instance.get_entry(cache_key) is None
# Second render
did_call_get = False
result = component.render()
# get_context_data IS called because the cache is NOT used
assert did_call_get
assert result == "Hello"
def test_cache_ttl(self):
class TestComponent(Component):
template = "Hello"
class Cache:
enabled = True
ttl = 0.1 # .1 seconds TTL
component = TestComponent()
component.render()
cache_instance = component.cache
cache_key = cache_instance.get_cache_key()
assert cache_instance.get_entry(cache_key) == "<!-- _RENDERED TestComponent_42aca9,ca1bc3e,, -->Hello"
# Wait for TTL to expire
time.sleep(0.2)
assert cache_instance.get_entry(cache_key) is None
@djc_test(
django_settings={
"CACHES": {
"default": {
"BACKEND": "django.core.cache.backends.locmem.LocMemCache",
"LOCATION": "default",
},
"custom": {
"BACKEND": "django.core.cache.backends.locmem.LocMemCache",
"LOCATION": "custom",
},
},
},
)
def test_custom_cache_name(self):
class TestComponent(Component):
template = "Hello"
class Cache:
enabled = True
cache_name = "custom"
component = TestComponent()
component.render()
assert component.cache.get_cache() is caches["custom"]
assert (
component.cache.get_entry("components:cache:TestComponent_90ef7a::")
== "<!-- _RENDERED TestComponent_90ef7a,ca1bc3e,, -->Hello"
) # noqa: E501
def test_cache_by_input(self):
class TestComponent(Component):
template = "Hello {{ input }}"
class Cache:
enabled = True
def get_context_data(self, input, **kwargs: Any):
return {"input": input}
component = TestComponent()
component.render(
kwargs={"input": "world"},
)
component.render(
kwargs={"input": "cake"},
)
# Check if the cache entry is set
cache = caches["default"]
assert len(cache._cache) == 2
assert (
component.cache.get_entry("components:cache:TestComponent_648b95::input-world")
== "<!-- _RENDERED TestComponent_648b95,ca1bc3e,, -->Hello world"
) # noqa: E501
assert (
component.cache.get_entry("components:cache:TestComponent_648b95::input-cake")
== "<!-- _RENDERED TestComponent_648b95,ca1bc3f,, -->Hello cake"
) # noqa: E501
def test_cache_input_hashing(self):
class TestComponent(Component):
template = "Hello"
class Cache:
enabled = True
component = TestComponent()
component.render(args=(1, 2), kwargs={"key": "value"})
# The key consists of `component._class_hash`, hashed args, and hashed kwargs
expected_key = "1,2:key-value"
assert component.cache.hash(1, 2, key="value") == expected_key
def test_override_hash_methods(self):
class TestComponent(Component):
template = "Hello"
class Cache:
enabled = True
def hash(self, *args, **kwargs):
# Custom hash method for args and kwargs
return "custom-args-and-kwargs"
def get_context_data(self, *args, **kwargs: Any):
return {}
component = TestComponent()
component.render(args=(1, 2), kwargs={"key": "value"})
# The key should use the custom hash methods
expected_key = "components:cache:TestComponent_28880f:custom-args-and-kwargs"
assert component.cache.get_cache_key(1, 2, key="value") == expected_key
def test_cached_component_inside_include(self):
@register("test_component")
class TestComponent(Component):
template = "Hello"
class Cache:
enabled = True
template = Template("""
{% extends "test_cached_component_inside_include_base.html" %}
{% block content %}
THIS_IS_IN_ACTUAL_TEMPLATE_SO_SHOULD_NOT_BE_OVERRIDDEN
{% endblock %}
""")
result = template.render(Context({}))
assert "THIS_IS_IN_BASE_TEMPLATE_SO_SHOULD_BE_OVERRIDDEN" not in result
assert "THIS_IS_IN_ACTUAL_TEMPLATE_SO_SHOULD_NOT_BE_OVERRIDDEN" in result
result_cached = template.render(Context({}))
assert "THIS_IS_IN_BASE_TEMPLATE_SO_SHOULD_BE_OVERRIDDEN" not in result_cached
assert "THIS_IS_IN_ACTUAL_TEMPLATE_SO_SHOULD_NOT_BE_OVERRIDDEN" in result_cached