From 440a2a97e9a8420ddf6186d4e948662bb3015e7d Mon Sep 17 00:00:00 2001 From: Jacob Kaplan-Moss Date: Mon, 1 Aug 2005 19:09:07 +0000 Subject: [PATCH] Added framework for writing non-model-based tests, and added tests for cache and templates from the old django/tests location (which has been removed). git-svn-id: http://code.djangoproject.com/svn/django/trunk@367 bcc190cf-cafb-0310-a4f2-bffc1f526a37 --- django/tests/cache_tests.py | 119 --- django/tests/template_inheritance.py | 102 --- django/tests/template_tests.py | 707 ------------------ .../tests => tests/othertests}/__init__.py | 0 tests/othertests/cache.py | 60 ++ tests/othertests/templates.py | 209 ++++++ tests/runtests.py | 31 +- tests/testapp/templatetags/__init__.py | 0 tests/testapp/templatetags/testtags.py | 15 + 9 files changed, 313 insertions(+), 930 deletions(-) delete mode 100644 django/tests/cache_tests.py delete mode 100644 django/tests/template_inheritance.py delete mode 100644 django/tests/template_tests.py rename {django/tests => tests/othertests}/__init__.py (100%) create mode 100644 tests/othertests/cache.py create mode 100644 tests/othertests/templates.py create mode 100644 tests/testapp/templatetags/__init__.py create mode 100644 tests/testapp/templatetags/testtags.py 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 = '

12

Last

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('') - 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('') - 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 %}{% 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 %}{% 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