2006-12-06 03:48:46 +08:00
|
|
|
# -*- coding: utf-8 -*-
|
2013-07-30 01:19:04 +08:00
|
|
|
from __future__ import unicode_literals
|
2011-09-16 09:16:25 +08:00
|
|
|
|
2007-07-23 12:45:01 +08:00
|
|
|
import os
|
2009-03-31 15:23:50 +08:00
|
|
|
import sys
|
2013-07-01 20:22:27 +08:00
|
|
|
import unittest
|
2007-07-23 12:45:01 +08:00
|
|
|
|
2006-08-27 21:59:47 +08:00
|
|
|
from django import template
|
2014-07-22 01:38:34 +08:00
|
|
|
from django.contrib.auth.models import Group
|
2008-08-05 22:16:13 +08:00
|
|
|
from django.core import urlresolvers
|
2015-02-10 02:19:34 +08:00
|
|
|
from django.template import (
|
|
|
|
Context, RequestContext, Template, TemplateSyntaxError,
|
2015-02-14 17:21:06 +08:00
|
|
|
base as template_base, engines, loader,
|
2015-02-10 02:19:34 +08:00
|
|
|
)
|
2015-02-22 03:13:26 +08:00
|
|
|
from django.test import RequestFactory, SimpleTestCase, override_settings
|
2012-12-08 18:13:52 +08:00
|
|
|
from django.utils._os import upath
|
2013-11-03 05:34:05 +08:00
|
|
|
|
2015-02-24 21:59:05 +08:00
|
|
|
TEMPLATES_DIR = os.path.join(os.path.dirname(upath(__file__)), 'templates')
|
2014-12-18 05:10:57 +08:00
|
|
|
|
|
|
|
|
2015-02-24 21:59:05 +08:00
|
|
|
class TemplateTests(SimpleTestCase):
|
2013-08-31 03:08:40 +08:00
|
|
|
|
2015-02-15 22:42:05 +08:00
|
|
|
@override_settings(DEBUG=True)
|
2013-08-31 03:08:40 +08:00
|
|
|
def test_string_origin(self):
|
2014-11-16 03:58:26 +08:00
|
|
|
template = Template('string template')
|
|
|
|
self.assertEqual(template.origin.source, 'string template')
|
2013-08-31 03:08:40 +08:00
|
|
|
|
2015-02-24 21:59:05 +08:00
|
|
|
|
2014-12-04 04:36:17 +08:00
|
|
|
class TemplateRegressionTests(SimpleTestCase):
|
2013-06-06 20:27:00 +08:00
|
|
|
|
2008-09-01 02:28:06 +08:00
|
|
|
def test_token_smart_split(self):
|
|
|
|
# Regression test for #7027
|
2014-11-11 04:40:26 +08:00
|
|
|
token = template_base.Token(template_base.TOKEN_BLOCK, 'sometag _("Page not found") value|yesno:_("yes,no")')
|
2008-09-01 02:28:06 +08:00
|
|
|
split = token.split_contents()
|
|
|
|
self.assertEqual(split, ["sometag", '_("Page not found")', 'value|yesno:_("yes,no")'])
|
|
|
|
|
2015-02-15 22:42:05 +08:00
|
|
|
@override_settings(SETTINGS_MODULE=None, DEBUG=True)
|
2009-04-02 06:46:46 +08:00
|
|
|
def test_url_reverse_no_settings_module(self):
|
2009-04-05 03:34:52 +08:00
|
|
|
# Regression test for #9005
|
2009-04-02 06:46:46 +08:00
|
|
|
t = Template('{% url will_not_match %}')
|
|
|
|
c = Context()
|
2011-09-16 09:16:25 +08:00
|
|
|
with self.assertRaises(urlresolvers.NoReverseMatch):
|
|
|
|
t.render(c)
|
|
|
|
|
2014-12-15 06:13:03 +08:00
|
|
|
@override_settings(
|
|
|
|
TEMPLATES=[{
|
|
|
|
'BACKEND': 'django.template.backends.django.DjangoTemplates',
|
|
|
|
'OPTIONS': {'string_if_invalid': '%s is invalid'},
|
|
|
|
}],
|
|
|
|
SETTINGS_MODULE='also_something',
|
|
|
|
)
|
2013-02-24 00:41:30 +08:00
|
|
|
def test_url_reverse_view_name(self):
|
|
|
|
# Regression test for #19827
|
|
|
|
t = Template('{% url will_not_match %}')
|
|
|
|
c = Context()
|
|
|
|
try:
|
|
|
|
t.render(c)
|
|
|
|
except urlresolvers.NoReverseMatch:
|
|
|
|
tb = sys.exc_info()[2]
|
|
|
|
depth = 0
|
|
|
|
while tb.tb_next is not None:
|
|
|
|
tb = tb.tb_next
|
|
|
|
depth += 1
|
2014-10-28 18:02:56 +08:00
|
|
|
self.assertGreater(depth, 5,
|
2013-02-24 00:41:30 +08:00
|
|
|
"The traceback context was lost when reraising the traceback. See #19827")
|
|
|
|
|
2015-02-15 22:42:05 +08:00
|
|
|
@override_settings(DEBUG=True)
|
2011-09-16 09:16:25 +08:00
|
|
|
def test_no_wrapped_exception(self):
|
|
|
|
"""
|
|
|
|
The template system doesn't wrap exceptions, but annotates them.
|
|
|
|
Refs #16770
|
|
|
|
"""
|
|
|
|
c = Context({"coconuts": lambda: 42 / 0})
|
|
|
|
t = Template("{{ coconuts }}")
|
|
|
|
with self.assertRaises(ZeroDivisionError) as cm:
|
|
|
|
t.render(c)
|
|
|
|
|
|
|
|
self.assertEqual(cm.exception.django_template_source[1], (0, 14))
|
2009-04-05 03:34:52 +08:00
|
|
|
|
2010-02-22 07:38:33 +08:00
|
|
|
def test_invalid_block_suggestion(self):
|
|
|
|
# See #7876
|
|
|
|
try:
|
2013-10-19 20:31:38 +08:00
|
|
|
Template("{% if 1 %}lala{% endblock %}{% endif %}")
|
2012-04-29 00:09:37 +08:00
|
|
|
except TemplateSyntaxError as e:
|
2011-12-10 06:13:27 +08:00
|
|
|
self.assertEqual(e.args[0], "Invalid block tag: 'endblock', expected 'elif', 'else' or 'endif'")
|
2010-02-22 07:38:33 +08:00
|
|
|
|
2013-02-23 20:13:44 +08:00
|
|
|
def test_ifchanged_concurrency(self):
|
|
|
|
# Tests for #15849
|
|
|
|
template = Template('[0{% for x in foo %},{% with var=get_value %}{% ifchanged %}{{ var }}{% endifchanged %}{% endwith %}{% endfor %}]')
|
|
|
|
|
|
|
|
# Using generator to mimic concurrency.
|
|
|
|
# The generator is not passed to the 'for' loop, because it does a list(values)
|
|
|
|
# instead, call gen.next() in the template to control the generator.
|
|
|
|
def gen():
|
|
|
|
yield 1
|
|
|
|
yield 2
|
|
|
|
# Simulate that another thread is now rendering.
|
|
|
|
# When the IfChangeNode stores state at 'self' it stays at '3' and skip the last yielded value below.
|
|
|
|
iter2 = iter([1, 2, 3])
|
|
|
|
output2 = template.render(Context({'foo': range(3), 'get_value': lambda: next(iter2)}))
|
2014-11-27 08:41:27 +08:00
|
|
|
self.assertEqual(output2, '[0,1,2,3]', 'Expected [0,1,2,3] in second parallel template, got {}'.format(output2))
|
2013-02-23 20:13:44 +08:00
|
|
|
yield 3
|
|
|
|
|
|
|
|
gen1 = gen()
|
|
|
|
output1 = template.render(Context({'foo': range(3), 'get_value': lambda: next(gen1)}))
|
2014-11-27 08:41:27 +08:00
|
|
|
self.assertEqual(output1, '[0,1,2,3]', 'Expected [0,1,2,3] in first template, got {}'.format(output1))
|
2013-02-23 20:13:44 +08:00
|
|
|
|
2013-03-25 16:10:32 +08:00
|
|
|
def test_cache_regression_20130(self):
|
|
|
|
t = Template('{% load cache %}{% cache 1 regression_20130 %}foo{% endcache %}')
|
|
|
|
cachenode = t.nodelist[1]
|
|
|
|
self.assertEqual(cachenode.fragment_name, 'regression_20130')
|
|
|
|
|
2013-10-04 05:36:21 +08:00
|
|
|
@override_settings(CACHES={
|
|
|
|
'default': {
|
|
|
|
'BACKEND': 'django.core.cache.backends.locmem.LocMemCache',
|
|
|
|
'LOCATION': 'default',
|
|
|
|
},
|
|
|
|
'template_fragments': {
|
|
|
|
'BACKEND': 'django.core.cache.backends.locmem.LocMemCache',
|
|
|
|
'LOCATION': 'fragments',
|
|
|
|
},
|
|
|
|
})
|
|
|
|
def test_cache_fragment_cache(self):
|
|
|
|
"""
|
|
|
|
When a cache called "template_fragments" is present, the cache tag
|
|
|
|
will use it in preference to 'default'
|
|
|
|
"""
|
|
|
|
t1 = Template('{% load cache %}{% cache 1 fragment %}foo{% endcache %}')
|
|
|
|
t2 = Template('{% load cache %}{% cache 1 fragment using="default" %}bar{% endcache %}')
|
|
|
|
|
|
|
|
ctx = Context()
|
|
|
|
o1 = t1.render(ctx)
|
|
|
|
o2 = t2.render(ctx)
|
|
|
|
|
|
|
|
self.assertEqual(o1, 'foo')
|
2013-10-19 06:49:24 +08:00
|
|
|
self.assertEqual(o2, 'bar')
|
2013-10-04 05:36:21 +08:00
|
|
|
|
|
|
|
def test_cache_missing_backend(self):
|
|
|
|
"""
|
|
|
|
When a cache that doesn't exist is specified, the cache tag will
|
|
|
|
raise a TemplateSyntaxError
|
|
|
|
'"""
|
|
|
|
t = Template('{% load cache %}{% cache 1 backend using="unknown" %}bar{% endcache %}')
|
|
|
|
|
|
|
|
ctx = Context()
|
|
|
|
with self.assertRaises(TemplateSyntaxError):
|
|
|
|
t.render(ctx)
|
|
|
|
|
2013-02-24 23:49:28 +08:00
|
|
|
def test_ifchanged_render_once(self):
|
|
|
|
""" Test for ticket #19890. The content of ifchanged template tag was
|
|
|
|
rendered twice."""
|
2014-03-28 04:24:19 +08:00
|
|
|
template = Template('{% ifchanged %}{% cycle "1st time" "2nd time" %}{% endifchanged %}')
|
2013-02-25 05:47:14 +08:00
|
|
|
output = template.render(Context({}))
|
|
|
|
self.assertEqual(output, '1st time')
|
2013-02-24 23:49:28 +08:00
|
|
|
|
2013-05-30 15:25:58 +08:00
|
|
|
def test_super_errors(self):
|
|
|
|
"""
|
|
|
|
Test behavior of the raise errors into included blocks.
|
|
|
|
See #18169
|
|
|
|
"""
|
|
|
|
t = loader.get_template('included_content.html')
|
|
|
|
with self.assertRaises(urlresolvers.NoReverseMatch):
|
2015-01-08 22:03:43 +08:00
|
|
|
t.render()
|
2013-05-30 15:25:58 +08:00
|
|
|
|
2014-07-22 01:38:34 +08:00
|
|
|
def test_debug_tag_non_ascii(self):
|
|
|
|
"""
|
|
|
|
Test non-ASCII model representation in debug output (#23060).
|
|
|
|
"""
|
|
|
|
Group.objects.create(name="清風")
|
|
|
|
c1 = Context({"objs": Group.objects.all()})
|
|
|
|
t1 = Template('{% debug %}')
|
|
|
|
self.assertIn("清風", t1.render(c1))
|
|
|
|
|
2015-02-14 17:21:06 +08:00
|
|
|
def test_extends_generic_template(self):
|
|
|
|
"""
|
|
|
|
{% extends %} accepts django.template.backends.django.Template (#24338).
|
|
|
|
"""
|
|
|
|
parent = engines['django'].from_string(
|
|
|
|
'{% block content %}parent{% endblock %}')
|
|
|
|
child = engines['django'].from_string(
|
|
|
|
'{% extends parent %}{% block content %}child{% endblock %}')
|
|
|
|
self.assertEqual(child.render({'parent': parent}), 'child')
|
|
|
|
|
2013-06-06 20:27:00 +08:00
|
|
|
|
2012-04-09 21:24:57 +08:00
|
|
|
class RequestContextTests(unittest.TestCase):
|
2011-04-17 12:52:31 +08:00
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
self.fake_request = RequestFactory().get('/')
|
|
|
|
|
2014-12-18 05:10:57 +08:00
|
|
|
@override_settings(TEMPLATES=[{
|
|
|
|
'BACKEND': 'django.template.backends.django.DjangoTemplates',
|
|
|
|
'OPTIONS': {
|
|
|
|
'loaders': [
|
|
|
|
('django.template.loaders.locmem.Loader', {
|
|
|
|
'child': '{{ var|default:"none" }}',
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
},
|
|
|
|
}])
|
2011-04-17 12:52:31 +08:00
|
|
|
def test_include_only(self):
|
|
|
|
"""
|
|
|
|
Regression test for #15721, ``{% include %}`` and ``RequestContext``
|
|
|
|
not playing together nicely.
|
|
|
|
"""
|
|
|
|
ctx = RequestContext(self.fake_request, {'var': 'parent'})
|
|
|
|
self.assertEqual(
|
|
|
|
template.Template('{% include "child" %}').render(ctx),
|
|
|
|
'parent'
|
|
|
|
)
|
|
|
|
self.assertEqual(
|
|
|
|
template.Template('{% include "child" only %}').render(ctx),
|
|
|
|
'none'
|
|
|
|
)
|
2013-08-29 12:08:01 +08:00
|
|
|
|
|
|
|
def test_stack_size(self):
|
|
|
|
"""
|
|
|
|
Regression test for #7116, Optimize RequetsContext construction
|
|
|
|
"""
|
|
|
|
ctx = RequestContext(self.fake_request, {})
|
|
|
|
# The stack should now contain 3 items:
|
|
|
|
# [builtins, supplied context, context processor]
|
|
|
|
self.assertEqual(len(ctx.dicts), 3)
|
2013-08-28 06:50:11 +08:00
|
|
|
|
2014-02-15 20:45:52 +08:00
|
|
|
def test_context_comparable(self):
|
|
|
|
test_data = {'x': 'y', 'v': 'z', 'd': {'o': object, 'a': 'b'}}
|
|
|
|
|
|
|
|
# test comparing RequestContext to prevent problems if somebody
|
|
|
|
# adds __eq__ in the future
|
|
|
|
request = RequestFactory().get('/')
|
|
|
|
|
2014-02-17 11:54:36 +08:00
|
|
|
self.assertEqual(
|
2015-02-18 05:49:59 +08:00
|
|
|
RequestContext(request, dict_=test_data),
|
|
|
|
RequestContext(request, dict_=test_data))
|