mirror of
https://github.com/django-components/django-components.git
synced 2025-07-08 01:45:00 +00:00

* 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>
235 lines
7.1 KiB
Python
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
|