diff --git a/django/tests/cache_tests.py b/django/tests/cache_tests.py
deleted file mode 100644
index a9cce041fa..0000000000
--- a/django/tests/cache_tests.py
+++ /dev/null
@@ -1,119 +0,0 @@
-"""
-Unit tests for django.core.cache
-
-If you don't have memcached running on localhost port 11211, the memcached tests
-will fail.
-"""
-
-from django.core import cache
-import unittest
-import time
-
-# functions/classes for complex data type tests
-def f():
- return 42
-class C:
- def m(n):
- return 24
-
-class CacheBackendsTest(unittest.TestCase):
-
- def testBackends(self):
- sc = cache.get_cache('simple://')
- mc = cache.get_cache('memcached://127.0.0.1:11211/')
- self.failUnless(isinstance(sc, cache._SimpleCache))
- self.failUnless(isinstance(mc, cache._MemcachedCache))
-
- def testInvalidBackends(self):
- self.assertRaises(cache.InvalidCacheBackendError, cache.get_cache, 'nothing://foo/')
- self.assertRaises(cache.InvalidCacheBackendError, cache.get_cache, 'not a uri')
-
- def testDefaultTimeouts(self):
- sc = cache.get_cache('simple:///?timeout=15')
- mc = cache.get_cache('memcached://127.0.0.1:11211/?timeout=15')
- self.assertEquals(sc.default_timeout, 15)
- self.assertEquals(sc.default_timeout, 15)
-
-class SimpleCacheTest(unittest.TestCase):
-
- def setUp(self):
- self.cache = cache.get_cache('simple://')
-
- def testGetSet(self):
- self.cache.set('key', 'value')
- self.assertEqual(self.cache.get('key'), 'value')
-
- def testNonExistantKeys(self):
- self.assertEqual(self.cache.get('does not exist'), None)
- self.assertEqual(self.cache.get('does not exist', 'bang!'), 'bang!')
-
- def testGetMany(self):
- self.cache.set('a', 'a')
- self.cache.set('b', 'b')
- self.cache.set('c', 'c')
- self.cache.set('d', 'd')
- self.assertEqual(self.cache.get_many(['a', 'c', 'd']), {'a' : 'a', 'c' : 'c', 'd' : 'd'})
- self.assertEqual(self.cache.get_many(['a', 'b', 'e']), {'a' : 'a', 'b' : 'b'})
-
- def testDelete(self):
- self.cache.set('key1', 'spam')
- self.cache.set('key2', 'eggs')
- self.assertEqual(self.cache.get('key1'), 'spam')
- self.cache.delete('key1')
- self.assertEqual(self.cache.get('key1'), None)
- self.assertEqual(self.cache.get('key2'), 'eggs')
-
- def testHasKey(self):
- self.cache.set('hello', 'goodbye')
- self.assertEqual(self.cache.has_key('hello'), True)
- self.assertEqual(self.cache.has_key('goodbye'), False)
-
- def testDataTypes(self):
- items = {
- 'string' : 'this is a string',
- 'int' : 42,
- 'list' : [1, 2, 3, 4],
- 'tuple' : (1, 2, 3, 4),
- 'dict' : {'A': 1, 'B' : 2},
- 'function' : f,
- 'class' : C,
- }
- for (key, value) in items.items():
- self.cache.set(key, value)
- self.assertEqual(self.cache.get(key), value)
-
- def testExpiration(self):
- self.cache.set('expire', 'very quickly', 1)
- time.sleep(2)
- self.assertEqual(self.cache.get('expire'), None)
-
- def testCull(self):
- c = cache.get_cache('simple://?max_entries=9&cull_frequency=3')
- for i in range(10):
- c.set('culltest%i' % i, i)
- n = 0
- for i in range(10):
- if c.get('culltest%i' % i):
- n += 1
- self.assertEqual(n, 6)
-
- def testCullAll(self):
- c = cache.get_cache('simple://?max_entries=9&cull_frequency=0')
- for i in range(10):
- c.set('cullalltest%i' % i, i)
- for i in range(10):
- self.assertEqual(self.cache.get('cullalltest%i' % i), None)
-
-class MemcachedCacheTest(SimpleCacheTest):
-
- def setUp(self):
- self.cache = cache.get_cache('memcached://127.0.0.1:11211/')
-
- testCull = testCullAll = lambda s: None
-
-def tests():
- s = unittest.TestLoader().loadTestsFromName(__name__)
- unittest.TextTestRunner(verbosity=0).run(s)
-
-if __name__ == "__main__":
- tests()
diff --git a/django/tests/template_inheritance.py b/django/tests/template_inheritance.py
deleted file mode 100644
index 35de1457e3..0000000000
--- a/django/tests/template_inheritance.py
+++ /dev/null
@@ -1,102 +0,0 @@
-from django.core import template, template_loader
-
-# SYNTAX --
-# 'template_name': ('template contents', 'context dict', 'expected string output' or Exception class)
-TEMPLATE_TESTS = {
- # Standard template with no inheritance
- 'test01': ("1{% block first %}_{% endblock %}3{% block second %}_{% endblock %}", {}, '1_3_'),
-
- # Standard two-level inheritance
- 'test02': ("{% extends 'test01' %}{% block first %}2{% endblock %}{% block second %}4{% endblock %}", {}, '1234'),
-
- # Three-level with no redefinitions on third level
- 'test03': ("{% extends 'test02' %}", {}, '1234'),
-
- # Two-level with no redefinitions on second level
- 'test04': ("{% extends 'test01' %}", {}, '1_3_'),
-
- # Two-level with double quotes instead of single quotes
- 'test05': ('{% extends "test02" %}', {}, '1234'),
-
- # Three-level with variable parent-template name
- 'test06': ("{% extends foo %}", {'foo': 'test02'}, '1234'),
-
- # Two-level with one block defined, one block not defined
- 'test07': ("{% extends 'test01' %}{% block second %}5{% endblock %}", {}, '1_35'),
-
- # Three-level with one block defined on this level, two blocks defined next level
- 'test08': ("{% extends 'test02' %}{% block second %}5{% endblock %}", {}, '1235'),
-
- # Three-level with second and third levels blank
- 'test09': ("{% extends 'test04' %}", {}, '1_3_'),
-
- # Three-level with space NOT in a block -- should be ignored
- 'test10': ("{% extends 'test04' %} ", {}, '1_3_'),
-
- # Three-level with both blocks defined on this level, but none on second level
- 'test11': ("{% extends 'test04' %}{% block first %}2{% endblock %}{% block second %}4{% endblock %}", {}, '1234'),
-
- # Three-level with this level providing one and second level providing the other
- 'test12': ("{% extends 'test07' %}{% block first %}2{% endblock %}", {}, '1235'),
-
- # Three-level with this level overriding second level
- 'test13': ("{% extends 'test02' %}{% block first %}a{% endblock %}{% block second %}b{% endblock %}", {}, '1a3b'),
-
- # A block defined only in a child template shouldn't be displayed
- 'test14': ("{% extends 'test01' %}{% block newblock %}NO DISPLAY{% endblock %}", {}, '1_3_'),
-
- # A block within another block
- 'test15': ("{% extends 'test01' %}{% block first %}2{% block inner %}inner{% endblock %}{% endblock %}", {}, '12inner3_'),
-
- # A block within another block (level 2)
- 'test16': ("{% extends 'test15' %}{% block inner %}out{% endblock %}", {}, '12out3_'),
-
- # {% load %} tag (parent -- setup for test-exception04)
- 'test17': ("{% load polls.polls %}{% block first %}1234{% endblock %}", {}, '1234'),
-
- # {% load %} tag (standard usage, without inheritance)
- 'test18': ("{% load polls.polls %}{% voteratio choice poll 400 %}5678", {}, '05678'),
-
- # {% load %} tag (within a child template)
- 'test19': ("{% extends 'test01' %}{% block first %}{% load polls.polls %}{% voteratio choice poll 400 %}5678{% endblock %}", {}, '1056783_'),
-
- # Raise exception for invalid template name
- 'test-exception01': ("{% extends 'nonexistent' %}", {}, template.TemplateSyntaxError),
-
- # Raise exception for invalid template name (in variable)
- 'test-exception02': ("{% extends nonexistent %}", {}, template.TemplateSyntaxError),
-
- # Raise exception for extra {% extends %} tags
- 'test-exception03': ("{% extends 'test01' %}{% block first %}2{% endblock %}{% extends 'test16' %}", {}, template.TemplateSyntaxError),
-
- # Raise exception for custom tags used in child with {% load %} tag in parent, not in child
- 'test-exception04': ("{% extends 'test17' %}{% block first %}{% votegraph choice poll 400 %}5678{% endblock %}", {}, template.TemplateSyntaxError),
-}
-
-# This replaces the standard template_loader.
-def test_template_loader(template_name):
- try:
- return TEMPLATE_TESTS[template_name][0]
- except KeyError:
- raise template.TemplateDoesNotExist, template_name
-template_loader.load_template_source = test_template_loader
-
-def run_tests():
- tests = TEMPLATE_TESTS.items()
- tests.sort()
- for name, vals in tests:
- try:
- output = template_loader.get_template(name).render(template.Context(vals[1]))
- except Exception, e:
- if e.__class__ == vals[2]:
- print "%s -- Passed" % name
- else:
- print "%s -- FAILED. Got %s, exception: %s" % (name, e.__class__, e)
- continue
- if output == vals[2]:
- print "%s -- Passed" % name
- else:
- print "%s -- FAILED. Expected %r, got %r" % (name, vals[2], output)
-
-if __name__ == "__main__":
- run_tests()
diff --git a/django/tests/template_tests.py b/django/tests/template_tests.py
deleted file mode 100644
index 3b082818bd..0000000000
--- a/django/tests/template_tests.py
+++ /dev/null
@@ -1,707 +0,0 @@
-"""
-Unit tests for template.py
-
-These tests assume the following template syntax:
-
-FILTER_SEPARATOR = '|'
-VARIABLE_ATTRIBUTE_SEPARATOR = '.'
-BLOCK_TAG_START = '{%'
-BLOCK_TAG_END = '%}'
-VARIABLE_TAG_START = '{{'
-VARIABLE_TAG_END = '}}'
-"""
-
-from django.core import template
-import unittest
-
-class RandomSyntaxErrorsCheck(unittest.TestCase):
-
- def testTagsOnOneLine(self):
- "Tags straddling more than one line are not interpreted"
- c = template.Context({'key':'value'})
- t = template.Template('
{{key\n}}
')
- expected = '{{key\n}}
'
- self.assertEqual(expected, t.render(c))
-
-class PlainTextCheck(unittest.TestCase):
-
- def testPlainText(self):
- "Plain text should go through the template parser untouched"
- c = template.Context()
- t = template.Template('Success
')
- expected = 'Success
'
- self.assertEqual(expected, t.render(c))
-
-class VariableSubstitutionCheck(unittest.TestCase):
-
- def testSingleTag(self):
- "Variables should be replaced with their value in the current context"
- c = template.Context({'headline':'Success'})
- t = template.Template('{{headline}}
')
- expected = 'Success
'
- self.assertEqual(expected, t.render(c))
-
- def testDoubleTag(self):
- "More than one replacement variable is allowed in a template"
- c = template.Context({'firsttag':'it', 'secondtag':'worked'})
- t = template.Template('{{firsttag}} {{secondtag}}
')
- expected = 'it worked
'
- self.assertEqual(expected, t.render(c))
-
- def testNonexistentVariable(self):
- "Fail silently when a variable is not found in the current context"
- c = template.Context({})
- t = template.Template('{{unknownvar}}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testVariablesWithSpaces(self):
- "A replacement-variable tag may not contain more than one word"
- t = '{{multi word tag}}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testEmptyTag(self):
- "Raise TemplateSyntaxError for empty variable tags"
- t = '{{ }}'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
- t = '{{ }}'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testIntegerContextValue(self):
- "Accept integers as variable values"
- c = template.Context({'var':55})
- t = template.Template('{{var}}
')
- expected = '55
'
- self.assertEqual(expected, t.render(c))
-
- def textIntegerContextKey(self):
- "Accept integers as variable keys"
- c = template.Context({55:'var'})
- t = template.Template('{{55}}
')
- expected = 'var
'
- self.assertEqual(expected, t.render(c))
-
- def testVariableAttributeAccess1(self):
- "Attribute syntax allows a template to call an object's attribute"
- class AClass: pass
- obj = AClass()
- obj.att = 'attvalue'
- c = template.Context({'var':obj})
- t = template.Template('{{ var.att }}
')
- expected = 'attvalue
'
- self.assertEqual(expected, t.render(c))
-
- def testVariableAttributeAccess2(self):
- "Attribute syntax allows a template to call an object's attribute (with getattr defined)"
- class AClass:
- def __getattr__(self, attr):
- return "attvalue"
- obj = AClass()
- c = template.Context({'var':obj})
- t = template.Template('{{ var.att }}
')
- expected = 'attvalue
'
- self.assertEqual(expected, t.render(c))
-
- def testVariableAttributeAccessMultiple(self):
- "Multiple levels of attribute access are allowed"
- class AClass: pass
- obj = AClass()
- obj.article = AClass()
- obj.article.section = AClass()
- obj.article.section.title = 'Headline'
- c = template.Context({'obj':obj})
- t = template.Template('{{ obj.article.section.title }}
')
- expected = 'Headline
'
- self.assertEqual(expected, t.render(c))
-
- def testNonexistentVariableAttributeObject(self):
- "Fail silently when a variable's attribute isn't found"
- class AClass: pass
- obj = AClass()
- obj.att = 'attvalue'
- c = template.Context({'var':obj})
- t = template.Template('{{ var.nonexistentatt }}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testIllegalUnderscoreInVariableName(self):
- "Raise TemplateSyntaxError when trying to access a variable beginning with an underscore"
- t = '{{ var._att }}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
- t = '{{ _att }}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testIllegalCharacterInVariableName(self):
- "Raise TemplateSyntaxError when trying to access a variable containing an illegal character"
- t = '{{ (blah }}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
- t = '{{ (blah.test) }}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
- t = '{{ bl(ah.test) }}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testVariableAttributeDictionary(self):
- "Attribute syntax allows a template to call a dictionary key's value"
- obj = {'att':'attvalue'}
- c = template.Context({'var':obj})
- t = template.Template('{{ var.att }}
')
- expected = 'attvalue
'
- self.assertEqual(expected, t.render(c))
-
- def testNonexistentVariableAttributeDictionary(self):
- "Fail silently when a variable's dictionary key isn't found"
- obj = {'att':'attvalue'}
- c = template.Context({'var':obj})
- t = template.Template('{{ var.nonexistentatt }}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testVariableAttributeCallable(self):
- "Attribute syntax allows a template to call a simple method"
- class AClass:
- def hello(self): return 'hi'
- obj = AClass()
- c = template.Context({'var':obj})
- t = template.Template('{{ var.hello }}
')
- expected = 'hi
'
- self.assertEqual(expected, t.render(c))
-
- def testVariableAttributeCallableWrongArguments(self):
- "Fail silently when accessing a non-simple method"
- class AClass:
- def hello(self, name): return 'hi, %s' % name
- obj = AClass()
- c = template.Context({'var':obj})
- t = template.Template('{{ var.hello }}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
-class VariableFiltersCheck(unittest.TestCase):
-
- def setUp(self):
- self.c = template.Context({'var':'Hello There Programmer'})
-
- def tearDown(self):
- self.c = None
-
- def testUpper(self):
- "The 'upper' filter converts a string into all uppercase"
- t = template.Template('{{ var|upper }}
')
- expected = 'HELLO THERE PROGRAMMER
'
- self.assertEqual(expected, t.render(self.c))
-
- def testLower(self):
- "The 'lower' filter converts a string into all lowercase"
- t = template.Template('{{ var|lower }}
')
- expected = 'hello there programmer
'
- self.assertEqual(expected, t.render(self.c))
-
- def testUpperThenLower(self):
- "Filters may be applied in succession (upper|lower)"
- t = template.Template('{{ var|upper|lower }}
')
- expected = 'hello there programmer
'
- self.assertEqual(expected, t.render(self.c))
-
- def testLowerThenUpper(self):
- "Filters may be applied in succession (lower|upper)"
- t = template.Template('{{ var|lower|upper }}
')
- expected = 'HELLO THERE PROGRAMMER
'
- self.assertEqual(expected, t.render(self.c))
-
- def testSpaceBetweenVariableAndFilterPipe(self):
- "Raise TemplateSyntaxError for space between a variable and filter pipe"
- t = '{{ var |lower }}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testSpaceBetweenFilterPipeAndFilterName1(self):
- "Raise TemplateSyntaxError for space after a filter pipe"
- t = '{{ var| lower }}
'
- expected = 'Hello There Programmer
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testNonexistentFilter(self):
- "Raise TemplateSyntaxError for a nonexistent filter"
- t = '{{ var|nonexistentfilter }}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testDefaultFilter1(self):
- "Ignore the default argument when a variable passed through the 'default' filter already exists"
- c = template.Context({'var':'Variable'})
- t = template.Template('{{ var|default:"Default" }}
')
- expected = 'Variable
'
- self.assertEqual(expected, t.render(c))
-
- def testDefaultFilter2(self):
- "Use the default argument when a variable passed through the 'default' filter doesn't exist"
- c = template.Context({'var':'Variable'})
- t = template.Template('{{ nonvar|default:"Default" }}
')
- expected = 'Default
'
- self.assertEqual(expected, t.render(c))
-
- def testDefaultFilter3(self):
- "Use the default argument when a variable passed through the 'default' filter doesn't exist (spaces)"
- c = template.Context({'var':'Variable'})
- t = template.Template('{{ nonvar|default:"Default value" }}
')
- expected = 'Default value
'
- self.assertEqual(expected, t.render(c))
-
- def testDefaultFilter4(self):
- "Use the default argument when a variable passed through the 'default' filter doesn't exist (quoted)"
- c = template.Context({'var':'Variable'})
- t = template.Template('{{ nonvar|default:"Default \"quoted\" value" }}
')
- expected = 'Default "quoted" value
'
- self.assertEqual(expected, t.render(c))
-
- def testDefaultFilter4(self):
- "Use the default argument when a variable passed through the 'default' filter doesn't exist (escaped backslash)"
- c = template.Context({'var':'Variable'})
- t = template.Template('{{ nonvar|default:"Default \\\\ slash" }}
')
- expected = 'Default \\ slash
'
- self.assertEqual(expected, t.render(c))
-
- def testDefaultFilter4(self):
- "Use the default argument when a variable passed through the 'default' filter doesn't exist (single backslash)"
- t = '{{ nonvar|default:"Default \\ slash" }}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testIllegalCharacterInFilterName(self):
- "Raise TemplateSyntaxError when trying to access a filter containing an illegal character"
- t = '{{ blah|(lower) }}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
- t = '{{ blah|low(er) }}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
-class BlockTagCheck(unittest.TestCase):
-
- def testNonexistentTag(self):
- "Raise TemplateSyntaxError for invalid block tags"
- t = '{% not-a-tag %}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testEmptyTag(self):
- "Raise TemplateSyntaxError for empty block tags"
- t = '{% %}'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
-class FirstOfCheck(unittest.TestCase):
-
- def testFirstOfDisplaysFirstIfSet(self):
- "A firstof tag should display the first item if it evaluates to true somehow"
- c = template.Context({'first': 'one', 'second': 'two'})
- t = template.Template('{% firstof first second %}
')
- expected = 'one
'
- self.assertEqual(expected, t.render(c))
-
- def testFirstOfDisplaysSecondIfFirstIsFalse(self):
- "A firstof tag should display the second item if it evaluates to true and the first is false"
- c = template.Context({'first': '', 'second': 'two'})
- t = template.Template('{% firstof first second %}
')
- expected = 'two
'
- self.assertEqual(expected, t.render(c))
-
- def testFirstOfRaisesErrorIfEmpty(self):
- "A firstof tag should raise a syntax error if it doesn't have any arguments"
- t = '{% firstof %}'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testFirstOfDoesNothingIfAllAreFalse(self):
- "A firstof tag should display nothing if no arguments evaluate to true"
- c = template.Context({'first': '', 'second': False})
- t = template.Template('{% firstof first second third %}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testFirstOfWorksWithInts(self):
- "Can a firstof tag display an integer?"
- c = template.Context({'first': 1, 'second': False})
- t = template.Template('{% firstof first second %}
')
- expected = '1
'
- self.assertEqual(expected, t.render(c))
-
-class IfStatementCheck(unittest.TestCase):
-
- def testSingleIfStatementTrue(self):
- "An if statement should display its contents if the test evaluates true"
- c = template.Context({'test':True})
- t = template.Template('{% if test %}Yes{% endif %}
')
- expected = 'Yes
'
- self.assertEqual(expected, t.render(c))
-
- def testSingleIfStatementFalse(self):
- "An if statement should not display its contents if the test is false"
- c = template.Context({'test':False})
- t = template.Template('{% if test %}Should not see this{% endif %}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testNestedIfStatementTrueThenTrue(self):
- "Nested if statements should work properly (case 1)"
- c = template.Context({'test1':True, 'test2':True})
- t = template.Template('{% if test1 %} First {% if test2 %} Second {% endif %} First again {% endif %}
')
- expected = ' First Second First again
'
- self.assertEqual(expected, t.render(c))
-
- def testNestedIfStatementTrueThenFalse(self):
- "Nested if statements should work properly (case 2)"
- c = template.Context({'test1':True, 'test2':False})
- t = template.Template('{% if test1 %} First {% if test2 %} Second {% endif %} First again {% endif %}
')
- expected = ' First First again
'
- self.assertEqual(expected, t.render(c))
-
- def testNestedIfStatementFalseThenTrue(self):
- "Nested if statements should work properly (case 3)"
- c = template.Context({'test1':False, 'test2':True})
- t = template.Template('{% if test1 %} First {% if test2 %} Second {% endif %} First again {% endif %}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testNestedIfStatementFalseThenFalse(self):
- "Nested if statements should work properly (case 4)"
- c = template.Context({'test1':False, 'test2':False})
- t = template.Template('{% if test1 %} First {% if test2 %} Second {% endif %} First again {% endif %}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testElseIfTrue(self):
- "An else statement should not execute if the test evaluates to true"
- c = template.Context({'test':True})
- t = template.Template('{% if test %}Correct{% else %}Incorrect{% endif %}
')
- expected = 'Correct
'
- self.assertEqual(expected, t.render(c))
-
- def testElseIfFalse(self):
- "An else statement should execute if the test evaluates to false"
- c = template.Context({'test':False})
- t = template.Template('{% if test %}Incorrect{% else %}Correct{% endif %}
')
- expected = 'Correct
'
- self.assertEqual(expected, t.render(c))
-
- def testNonClosedIfTag(self):
- "Raise TemplateSyntaxError for non-closed 'if' tags"
- c = template.Context({'test':True})
- t = '{% if test %}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testNonexistentTest(self):
- "Fail silently when an if statement accesses a nonexistent test"
- c = template.Context({'var':'value'})
- t = template.Template('{% if nonexistent %}Hello{% endif %}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testIfTagNoArgs(self):
- "If statements must have one argument (case 1)"
- t = '{% if %}Hello{% endif %}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testIfTagManyArgs(self):
- "If statements must have one argument (case 2)"
- t = '{% if multiple tests %}Hello{% endif %}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testAttributeAccessInIfNode(self):
- "An if node should resolve a variable's attributes before checking it as a test"
- class AClass: pass
- obj = AClass()
- obj.article = AClass()
- obj.article.section = AClass()
- obj.article.section.title = 'Headline'
- c = template.Context({'obj':obj})
- t = template.Template('{% if obj.article.section.title %}Hello{% endif %}
')
- expected = 'Hello
'
- self.assertEqual(expected, t.render(c))
- t = template.Template('{% if obj.article.section.not_here %}Hello{% endif %}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testIfNot(self):
- "If statements supports 'not' as an optional argument"
- t = template.Template('{% if not a %}Not a{% endif %}')
- c = template.Context({'a': False})
- expected = 'Not a'
- self.assertEqual(expected, t.render(c))
- c['a'] = True
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testIfOr(self):
- "If statements support 'or'"
- t = template.Template('{% if a or b %}Hello{% endif %}')
- c = template.Context({'a': False, 'b': True})
- expected = 'Hello'
- self.assertEqual(expected, t.render(c))
- c['b'] = False
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testIfOrNot(self):
- "If statements support 'or' clauses with optional 'not's"
- t = template.Template('{% if a or not b or c%}Hello{% endif %}')
- c = template.Context({'a': False, 'b': False, 'c': False})
- expected = 'Hello'
- self.assertEqual(expected, t.render(c))
- c['b'] = True
- expected = ''
- self.assertEqual(expected, t.render(c))
-
-class ForLoopCheck(unittest.TestCase):
-
- def testNormalForLoop(self):
- "A for loop should work as expected, given one or more values"
- c = template.Context({'pieces': ('1', '2', '3')})
- t = template.Template('{% for piece in pieces %}{{ piece }}{% endfor %}
')
- expected = '123
'
- self.assertEqual(expected, t.render(c))
-
- def testBlankForLoop(self):
- "A for loop should work as expected, given an empty list"
- c = template.Context({'pieces': []})
- t = template.Template('{% for piece in pieces %}{{ piece }}{% endfor %}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testInvalidForTagFourWords(self):
- "Raise TemplateSyntaxError if a 'for' statement is not exactly 4 words"
- t = '{% for article %}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testInvalidForTagThirdWord(self):
- "Raise TemplateSyntaxError if 3rd word in a 'for' statement isn't 'in'"
- t = '{% for article NOTIN blah %}{% endfor %}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testNonClosedForTag(self):
- "Raise TemplateSyntaxError for non-closed 'for' tags"
- t = '{% for i in numbers %}{{ i }}
'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testNonexistentVariable1(self):
- "Fail silently in loops with nonexistent variables in defn"
- c = template.Context({'var':'value'})
- t = template.Template('{% for i in nonexistent %}
{{ var }}
{% endfor %}')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testNonexistentVariable2(self):
- "Raise TemplateSyntaxError in loops with nonexistent variables in loop"
- c = template.Context({'set':('val1', 'val2')})
- t = template.Template('{% for i in set %}
{{ nonexistent }}
{% endfor %}')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testAttributeAccessInForNode(self):
- "A for node should resolve a variable's attributes before looping through it"
- c = template.Context({'article': {'authors':('Simon', 'Adrian')}})
- t = template.Template('{% for i in article.authors %}{{ i }}{% endfor %}
')
- self.assertEqual('SimonAdrian
', t.render(c))
- t = template.Template('{% for i in article.nonexistent %}{{ i }}{% endfor %}
')
- self.assertEqual('', t.render(c))
-
- def testForLoopFirst(self):
- "A for loop's 'first' variable should work as expected"
- c = template.Context({'pieces': ('1', '2', '3')})
- t = template.Template('{% for piece in pieces %}{% if forloop.first %}First
{% endif %}{{ piece }}{% endfor %}
')
- expected = 'First
123
'
- self.assertEqual(expected, t.render(c))
-
- def testForLoopLast(self):
- "A for loop's 'last' variable should work as expected"
- c = template.Context({'pieces': ('1', '2', '3')})
- t = template.Template('{% for piece in pieces %}{% if forloop.last %}Last
{% endif %}{{ piece }}{% endfor %}
')
- expected = '12Last
3
'
- self.assertEqual(expected, t.render(c))
-
-class CycleNodeCheck(unittest.TestCase):
-
- def testNormalUsage(self):
- "A cycle tag should work as expected"
- c = template.Context({'set':range(10)})
- t = template.Template('{% for i in set %}{% cycle red, green %}-{{ i }} {% endfor %}')
- expected = 'red-0 green-1 red-2 green-3 red-4 green-5 red-6 green-7 red-8 green-9 '
- self.assertEqual(expected, t.render(c))
-
- def testNoArguments(self):
- "Raise TemplateSyntaxError in cycle tags with no arguments"
- t = '{% cycle %}'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testOneArgument(self):
- "Raise TemplateSyntaxError in cycle tags with only one argument"
- t = '{% cycle hello %}'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testExtraInitialSpaces(self):
- "Extra spaces around cycle tags and their arguments should be ignored"
- c = template.Context({'set':range(5)})
- t = template.Template('{% for i in set %}{% cycle red, green %}{% endfor %}')
- expected = 'redgreenredgreenred'
- self.assertEqual(expected, t.render(c))
-
-class TemplateTagNodeCheck(unittest.TestCase):
-
- def testNormalUsage(self):
- "A templatetag tag should work as expected"
- c = template.Context()
- t = template.Template('{% templatetag openblock %}{% templatetag closeblock %}{% templatetag openvariable %}{% templatetag closevariable %}')
- expected = '{%%}{{}}'
- self.assertEqual(expected, t.render(c))
-
- def testNoArguments(self):
- "Raise TemplateSyntaxError in templatetag tags with no arguments"
- t = '{% templatetag %}'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testTwoArguments(self):
- "Raise TemplateSyntaxError in templatetag tags with more than one argument"
- t = '{% templatetag hello goodbye %}'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
- t = '{% templatetag hello goodbye helloagain %}'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
- def testBadArgument(self):
- "Raise TemplateSyntaxError in templatetag tags with invalid arguments"
- t = '{% templatetag hello %}'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
-class PluginFilterCheck(unittest.TestCase):
-
- def custom_filter(self, value, arg):
- "Temporary filter used to verify the filter plugin system is working"
- return "_%s_%s_" % (value, arg)
-
- def testPluginFilter(self):
- "Plugin support allows for custom filters"
- template.register_filter('unittest', self.custom_filter, True)
- c = template.Context({'var':'value'})
- t = template.Template('{{ var|unittest:"hello" }}')
- expected = '_value_hello_'
- self.assertEqual(expected, t.render(c))
- template.unregister_filter('unittest')
-
- def testUnregisterPluginFilter(self):
- "Plugin support allows custom filters to be unregistered"
- template.register_filter('unittest', self.custom_filter, True)
- c = template.Context({'var':'value'})
- t = template.Template('{{ var|unittest:"hello" }}')
- rendered = t.render(c) # should run with no exception
- template.unregister_filter('unittest')
-
-class PluginTagCheck(unittest.TestCase):
-
- class CustomNode(template.Node):
- "Prints argument"
- def __init__(self, arg):
- self.arg = arg
-
- def render(self, context):
- return '_%s_' % self.arg
-
- def do_custom_node(self, parser, token):
- "Handle the 'unittest' custom tag"
- bits = token.contents.split()
- return self.CustomNode(bits[1])
-
- def testPluginTag(self):
- "Plugin support allows for custom tags"
- template.register_tag('unittest', self.do_custom_node)
- c = template.Context({})
- t = template.Template('{% unittest hello %}')
- expected = '_hello_'
- self.assertEqual(expected, t.render(c))
- template.unregister_tag('unittest')
-
- def testUnregisterPluginTag(self):
- "Plugin support allows custom tags to be unregistered"
- template.register_tag('unittest', self.do_custom_node)
- c = template.Context({})
- t = template.Template('{% unittest hello %}')
- rendered = t.render(c) # should run with no exception
- del(t)
- template.unregister_tag('unittest')
- t = '{% unittest hello %}'
- self.assertRaises(template.TemplateSyntaxError, template.Template, t)
-
-class ContextUsageCheck(unittest.TestCase):
-
- def testVariableContext2(self):
- "Variables should fall through additional block-level contexts"
- c = template.Context({'global':'out', 'set': ('1', '2', '3')})
- t = template.Template('{{ global }}
{% for i in set %}{{ i }} {{ global }}
{% endfor %}')
- expected = 'out
1 out
2 out
3 out
'
- self.assertEqual(expected, t.render(c))
-
- def testVariableContext2(self):
- "Variables set within a block statement override like-named variables within their scope"
- c = template.Context({'i':'out', 'set': ('1', '2', '3')})
- t = template.Template('{{ i }}
{% for i in set %}{{ i }}
{% endfor %}{{ i }}')
- expected = 'out
1
2
3
out'
- self.assertEqual(expected, t.render(c))
-
- def testVariableContextDelete(self):
- "Variables can be deleted from the current context"
- c = template.Context({'a':'first', 'b':'second'})
- del c['a']
- self.assertEqual(c.__repr__(), template.Context({'b':'second'}).__repr__())
-
- def testInvalidVariableContextDelete(self):
- "Raise KeyError if code tries to delete a variable that doesn't exist in the current context"
- c = template.Context({'a':'first'})
- self.assertRaises(KeyError, c.__delitem__, 'b')
-
-class AdvancedUsageCheck(unittest.TestCase):
-
- def testIfInsideFor(self):
- "An if statement should be executed repeatedly inside a for statement"
- c = template.Context({'set':(True, False, True, True, False)})
- t = template.Template('{% for i in set %}{% if i %}- 1
{% endif %}{% endfor %}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testIfElseInsideFor(self):
- "An if/else statement should be executed repeatedly inside a for statement"
- c = template.Context({'set':(True, False, True, True, False)})
- t = template.Template('{% for i in set %}- {% if i %}1{% else %}0{% endif %}
{% endfor %}
')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testForInsideIf_True(self):
- "A for loop inside an if statement should be executed if the test=true"
- c = template.Context({'test':True, 'set':('1', '2', '3')})
- t = template.Template('{% if test %}{% for i in set %}- {{ i }}
{% endfor %}
{% endif %}')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testForInsideIf_False(self):
- "A for loop inside an if statement shouldn't be executed if the test=false"
- c = template.Context({'test':False, 'set':('1', '2', '3')})
- t = template.Template('{% if test %}{% for i in set %}- {{ i }}
{% endfor %}
{% endif %}')
- expected = ''
- self.assertEqual(expected, t.render(c))
-
- def testForInsideIfInsideFor(self):
- "A for loop inside an if statement inside a for loop should work properly"
- c = template.Context({'set1': (True, False, False, False, True), 'set2': ('1', '2', '3')})
- t = template.Template('{% for i in set1 %}{% if i %}{% for j in set2 %}{{ j }}{% endfor %}{% endif %}{% endfor %}')
- expected = '123123'
- self.assertEqual(expected, t.render(c))
-
- def testMultipleRendersWhenCompiled(self):
- "A template can render multiple contexts without having to be recompiled"
- t = template.Template('{% for i in set1 %}{% if i %}{% for j in set2 %}{{ j }}{% endfor %}{% endif %}{% endfor %}')
- c = template.Context({'set1': (True, False, False, False, False), 'set2': ('1', '2', '3')})
- self.assertEqual('123', t.render(c))
- c = template.Context({'set1': (True, True, False, False, False), 'set2': ('1', '2', '3')})
- self.assertEqual('123123', t.render(c))
- c = template.Context({'set1': (True, True, True, False, False), 'set2': ('1', '2', '3')})
- self.assertEqual('123123123', t.render(c))
- c = template.Context({'set1': (True, True, True, True, False), 'set2': ('1', '2', '3')})
- self.assertEqual('123123123123', t.render(c))
- c = template.Context({'set1': (True, True, True, True, True), 'set2': ('1', '2', '3')})
- self.assertEqual('123123123123123', t.render(c))
-
-def tests():
- s = unittest.TestLoader().loadTestsFromName(__name__)
- unittest.TextTestRunner(verbosity=0).run(s)
-
-if __name__ == "__main__":
- tests()
diff --git a/django/tests/__init__.py b/tests/othertests/__init__.py
similarity index 100%
rename from django/tests/__init__.py
rename to tests/othertests/__init__.py
diff --git a/tests/othertests/cache.py b/tests/othertests/cache.py
new file mode 100644
index 0000000000..513e2186c4
--- /dev/null
+++ b/tests/othertests/cache.py
@@ -0,0 +1,60 @@
+# Unit tests for cache framework
+# Uses whatever cache backend is set in the test settings file.
+
+from django.core.cache import cache
+import time
+
+# functions/classes for complex data type tests
+def f():
+ return 42
+class C:
+ def m(n):
+ return 24
+
+# simple set/get
+cache.set("key", "value")
+assert cache.get("key") == "value"
+
+# get with non-existant keys
+assert cache.get("does not exist") is None
+assert cache.get("does not exist", "bang!") == "bang!"
+
+# get_many
+cache.set('a', 'a')
+cache.set('b', 'b')
+cache.set('c', 'c')
+cache.set('d', 'd')
+assert cache.get_many(['a', 'c', 'd']) == {'a' : 'a', 'c' : 'c', 'd' : 'd'}
+assert cache.get_many(['a', 'b', 'e']) == {'a' : 'a', 'b' : 'b'}
+
+# delete
+cache.set("key1", "spam")
+cache.set("key2", "eggs")
+assert cache.get("key1") == "spam"
+cache.delete("key1")
+assert cache.get("key1") is None
+assert cache.get("key2") == "eggs"
+
+# has_key
+cache.set("hello", "goodbye")
+assert cache.has_key("hello") == True
+assert cache.has_key("goodbye") == False
+
+# test data types
+stuff = {
+ 'string' : 'this is a string',
+ 'int' : 42,
+ 'list' : [1, 2, 3, 4],
+ 'tuple' : (1, 2, 3, 4),
+ 'dict' : {'A': 1, 'B' : 2},
+ 'function' : f,
+ 'class' : C,
+}
+for (key, value) in stuff.items():
+ cache.set(key, value)
+ assert cache.get(key) == value
+
+# expiration
+cache.set('expire', 'very quickly', 1)
+time.sleep(2)
+assert cache.get("expire") == None
\ No newline at end of file
diff --git a/tests/othertests/templates.py b/tests/othertests/templates.py
new file mode 100644
index 0000000000..451d3d880c
--- /dev/null
+++ b/tests/othertests/templates.py
@@ -0,0 +1,209 @@
+from django.core import template, template_loader
+
+# Helper objects for template tests
+class SomeClass:
+ def __init__(self):
+ self.otherclass = OtherClass()
+
+ def method(self):
+ return "SomeClass.method"
+
+ def method2(self, o):
+ return o
+
+class OtherClass:
+ def method(self):
+ return "OtherClass.method"
+
+# SYNTAX --
+# 'template_name': ('template contents', 'context dict', 'expected string output' or Exception class)
+TEMPLATE_TESTS = {
+
+ ### BASIC SYNTAX ##########################################################
+
+ # Plain text should go through the template parser untouched
+ 'basic-syntax01': ("something cool", {}, "something cool"),
+
+ # Variables should be replaced with their value in the current context
+ 'basic-syntax02': ("{{ headline }}", {'headline':'Success'}, "Success"),
+
+ # More than one replacement variable is allowed in a template
+ 'basic-syntax03': ("{{ first }} --- {{ second }}", {"first" : 1, "second" : 2}, "1 --- 2"),
+
+ # Fail silently when a variable is not found in the current context
+ 'basic-syntax04': ("as{{ missing }}df", {}, "asdf"),
+
+ # A variable may not contain more than one word
+ 'basic-syntax06': ("{{ multi word variable }}", {}, template.TemplateSyntaxError),
+
+ # Raise TemplateSyntaxError for empty variable tags
+ 'basic-syntax07': ("{{ }}", {}, template.TemplateSyntaxError),
+ 'basic-syntax08': ("{{ }}", {}, template.TemplateSyntaxError),
+
+ # Attribute syntax allows a template to call an object's attribute
+ 'basic-syntax09': ("{{ var.method }}", {"var": SomeClass()}, "SomeClass.method"),
+
+ # Multiple levels of attribute access are allowed
+ 'basic-syntax10': ("{{ var.otherclass.method }}", {"var": SomeClass()}, "OtherClass.method"),
+
+ # Fail silently when a variable's attribute isn't found
+ 'basic-syntax11': ("{{ var.blech }}", {"var": SomeClass()}, ""),
+
+ # Raise TemplateSyntaxError when trying to access a variable beginning with an underscore
+ 'basic-syntax12': ("{{ var.__dict__ }}", {"var": SomeClass()}, template.TemplateSyntaxError),
+
+ # Raise TemplateSyntaxError when trying to access a variable containing an illegal character
+ 'basic-syntax13': ("{{ va>r }}", {}, template.TemplateSyntaxError),
+ 'basic-syntax14': ("{{ (var.r) }}", {}, template.TemplateSyntaxError),
+ 'basic-syntax15': ("{{ sp%am }}", {}, template.TemplateSyntaxError),
+ 'basic-syntax16': ("{{ eggs! }}", {}, template.TemplateSyntaxError),
+ 'basic-syntax17': ("{{ moo? }}", {}, template.TemplateSyntaxError),
+
+ # Attribute syntax allows a template to call a dictionary key's value
+ 'basic-syntax18': ("{{ foo.bar }}", {"foo" : {"bar" : "baz"}}, "baz"),
+
+ # Fail silently when a variable's dictionary key isn't found
+ 'basic-syntax19': ("{{ foo.spam }}", {"foo" : {"bar" : "baz"}}, ""),
+
+ # Fail silently when accessing a non-simple method
+ 'basic-syntax20': ("{{ var.method2 }}", {"var": SomeClass()}, ""),
+
+ # Basic filter usage
+ 'basic-syntax21': ("{{ var|upper }}", {"var": "Django is the greatest!"}, "DJANGO IS THE GREATEST!"),
+
+ # Chained filters
+ 'basic-syntax22': ("{{ var|upper|lower }}", {"var": "Django is the greatest!"}, "django is the greatest!"),
+
+ # Raise TemplateSyntaxError for space between a variable and filter pipe
+ 'basic-syntax23': ("{{ var |upper }}", {}, template.TemplateSyntaxError),
+
+ # Raise TemplateSyntaxError for space after a filter pipe
+ 'basic-syntax24': ("{{ var| upper }}", {}, template.TemplateSyntaxError),
+
+ # Raise TemplateSyntaxError for a nonexistent filter
+ 'basic-syntax25': ("{{ var|does_not_exist }}", {}, template.TemplateSyntaxError),
+
+ # Raise TemplateSyntaxError when trying to access a filter containing an illegal character
+ 'basic-syntax26': ("{{ var|fil(ter) }}", {}, template.TemplateSyntaxError),
+
+ # Raise TemplateSyntaxError for invalid block tags
+ 'basic-syntax27': ("{% nothing_to_see_here %}", {}, template.TemplateSyntaxError),
+
+ # Raise TemplateSyntaxError for empty block tags
+ 'basic-syntax28': ("{% %}", {}, template.TemplateSyntaxError),
+
+ ### INHERITANCE TESTS #####################################################
+
+ # Standard template with no inheritance
+ 'inheritance01': ("1{% block first %}_{% endblock %}3{% block second %}_{% endblock %}", {}, '1_3_'),
+
+ # Standard two-level inheritance
+ 'inheritance02': ("{% extends 'inheritance01' %}{% block first %}2{% endblock %}{% block second %}4{% endblock %}", {}, '1234'),
+
+ # Three-level with no redefinitions on third level
+ 'inheritance03': ("{% extends 'inheritance02' %}", {}, '1234'),
+
+ # Two-level with no redefinitions on second level
+ 'inheritance04': ("{% extends 'inheritance01' %}", {}, '1_3_'),
+
+ # Two-level with double quotes instead of single quotes
+ 'inheritance05': ('{% extends "inheritance02" %}', {}, '1234'),
+
+ # Three-level with variable parent-template name
+ 'inheritance06': ("{% extends foo %}", {'foo': 'inheritance02'}, '1234'),
+
+ # Two-level with one block defined, one block not defined
+ 'inheritance07': ("{% extends 'inheritance01' %}{% block second %}5{% endblock %}", {}, '1_35'),
+
+ # Three-level with one block defined on this level, two blocks defined next level
+ 'inheritance08': ("{% extends 'inheritance02' %}{% block second %}5{% endblock %}", {}, '1235'),
+
+ # Three-level with second and third levels blank
+ 'inheritance09': ("{% extends 'inheritance04' %}", {}, '1_3_'),
+
+ # Three-level with space NOT in a block -- should be ignored
+ 'inheritance10': ("{% extends 'inheritance04' %} ", {}, '1_3_'),
+
+ # Three-level with both blocks defined on this level, but none on second level
+ 'inheritance11': ("{% extends 'inheritance04' %}{% block first %}2{% endblock %}{% block second %}4{% endblock %}", {}, '1234'),
+
+ # Three-level with this level providing one and second level providing the other
+ 'inheritance12': ("{% extends 'inheritance07' %}{% block first %}2{% endblock %}", {}, '1235'),
+
+ # Three-level with this level overriding second level
+ 'inheritance13': ("{% extends 'inheritance02' %}{% block first %}a{% endblock %}{% block second %}b{% endblock %}", {}, '1a3b'),
+
+ # A block defined only in a child template shouldn't be displayed
+ 'inheritance14': ("{% extends 'inheritance01' %}{% block newblock %}NO DISPLAY{% endblock %}", {}, '1_3_'),
+
+ # A block within another block
+ 'inheritance15': ("{% extends 'inheritance01' %}{% block first %}2{% block inner %}inner{% endblock %}{% endblock %}", {}, '12inner3_'),
+
+ # A block within another block (level 2)
+ 'inheritance16': ("{% extends 'inheritance15' %}{% block inner %}out{% endblock %}", {}, '12out3_'),
+
+ # {% load %} tag (parent -- setup for exception04)
+ 'inheritance17': ("{% load testtags %}{% block first %}1234{% endblock %}", {}, '1234'),
+
+ # {% load %} tag (standard usage, without inheritance)
+ 'inheritance18': ("{% load testtags %}{% echo this that theother %}5678", {}, 'this that theother5678'),
+
+ # {% load %} tag (within a child template)
+ 'inheritance19': ("{% extends 'inheritance01' %}{% block first %}{% load testtags %}{% echo 400 %}5678{% endblock %}", {}, '140056783_'),
+
+ ### EXCEPTION TESTS #######################################################
+
+ # Raise exception for invalid template name
+ 'exception01': ("{% extends 'nonexistent' %}", {}, template.TemplateSyntaxError),
+
+ # Raise exception for invalid template name (in variable)
+ 'exception02': ("{% extends nonexistent %}", {}, template.TemplateSyntaxError),
+
+ # Raise exception for extra {% extends %} tags
+ 'exception03': ("{% extends 'inheritance01' %}{% block first %}2{% endblock %}{% extends 'inheritance16' %}", {}, template.TemplateSyntaxError),
+
+ # Raise exception for custom tags used in child with {% load %} tag in parent, not in child
+ 'exception04': ("{% extends 'inheritance17' %}{% block first %}{% echo 400 %}5678{% endblock %}", {}, template.TemplateSyntaxError),
+}
+
+# This replaces the standard template_loader.
+def test_template_loader(template_name, template_dirs=None):
+ try:
+ return TEMPLATE_TESTS[template_name][0]
+ except KeyError:
+ raise template.TemplateDoesNotExist, template_name
+
+def run_tests(verbosity=0, standalone=False):
+ template_loader.load_template_source, old_template_loader = test_template_loader, template_loader.load_template_source
+ failed_tests = []
+ tests = TEMPLATE_TESTS.items()
+ tests.sort()
+ for name, vals in tests:
+ try:
+ output = template_loader.get_template(name).render(template.Context(vals[1]))
+ except Exception, e:
+ if e.__class__ == vals[2]:
+ if verbosity:
+ print "Template test: %s -- Passed" % name
+ else:
+ if verbosity:
+ print "Template test: %s -- FAILED. Got %s, exception: %s" % (name, e.__class__, e)
+ failed_tests.append(name)
+ continue
+ if output == vals[2]:
+ if verbosity:
+ print "Template test: %s -- Passed" % name
+ else:
+ if verbosity:
+ print "Template test: %s -- FAILED. Expected %r, got %r" % (name, vals[2], output)
+ failed_tests.append(name)
+ template_loader.load_template_source = old_template_loader
+
+ if failed_tests and not standalone:
+ msg = "Template tests %s failed." % failed_tests
+ if not verbosity:
+ msg += " Re-run tests with -v1 to see actual failures"
+ raise Exception, msg
+
+if __name__ == "__main__":
+ run_tests(1, True)
diff --git a/tests/runtests.py b/tests/runtests.py
index 495ff36879..2987b7db81 100755
--- a/tests/runtests.py
+++ b/tests/runtests.py
@@ -8,12 +8,13 @@ import os, sys, time, traceback
import doctest
APP_NAME = 'testapp'
+OTHER_TESTS_DIR = "othertests"
TEST_DATABASE_NAME = 'django_test_db'
error_list = []
def log_error(model_name, title, description):
error_list.append({
- 'title': "%r model: %s" % (model_name, title),
+ 'title': "%r module: %s" % (model_name, title),
'description': description,
})
@@ -91,6 +92,7 @@ class TestRunner:
management.init()
# Run the tests for each test model.
+ self.output(1, "Running app tests")
for model_name in get_test_models():
self.output(1, "%s model: Importing" % model_name)
try:
@@ -110,7 +112,32 @@ class TestRunner:
runner = DjangoDoctestRunner(verbosity_level=verbosity_level, verbose=False)
self.output(1, "%s model: Running tests" % model_name)
runner.run(dtest, clear_globs=True, out=sys.stdout.write)
-
+
+ # Run the non-model tests in the other tests dir
+ self.output(1, "Running other tests")
+ other_tests_dir = os.path.join(os.path.dirname(__file__), OTHER_TESTS_DIR)
+ test_modules = [f[:-3] for f in os.listdir(other_tests_dir) if f.endswith('.py') and not f.startswith('__init__')]
+ for module in test_modules:
+ self.output(1, "%s module: Importing" % module)
+ try:
+ mod = __import__("othertests." + module, '', '', [''])
+ except Exception, e:
+ log_error(module, "Error while importing", ''.join(traceback.format_exception(*sys.exc_info())[1:]))
+ continue
+ if mod.__doc__:
+ p = doctest.DocTestParser()
+ dtest = p.get_doctest(mod.__doc__, mod.__dict__, module, None, None)
+ runner = DjangoDoctestRunner(verbosity_level=verbosity_level, verbose=False)
+ self.output(1, "%s module: runing tests" % module)
+ runner.run(dtest, clear_globs=True, out=sys.stdout.write)
+ if hasattr(mod, "run_tests") and callable(mod.run_tests):
+ self.output(1, "%s module: runing tests" % module)
+ try:
+ mod.run_tests(verbosity_level)
+ except Exception, e:
+ log_error(module, "Exception running tests", ''.join(traceback.format_exception(*sys.exc_info())[1:]))
+ continue
+
# Unless we're using SQLite, remove the test database to clean up after
# ourselves. Connect to the previous database (not the test database)
# to do so, because it's not allowed to delete a database while being
diff --git a/tests/testapp/templatetags/__init__.py b/tests/testapp/templatetags/__init__.py
new file mode 100644
index 0000000000..e69de29bb2
diff --git a/tests/testapp/templatetags/testtags.py b/tests/testapp/templatetags/testtags.py
new file mode 100644
index 0000000000..7b755043fa
--- /dev/null
+++ b/tests/testapp/templatetags/testtags.py
@@ -0,0 +1,15 @@
+# Custom tag library used in conjunction with template tests
+
+from django.core import template
+
+class EchoNode(template.Node):
+ def __init__(self, contents):
+ self.contents = contents
+
+ def render(self, context):
+ return " ".join(self.contents)
+
+def do_echo(parser, token):
+ return EchoNode(token.contents.split()[1:])
+
+template.register_tag("echo", do_echo)
\ No newline at end of file