Refs #2333 - Modified runtests script to use new testing framework. Migrated existing tests to use Django testing framework. All the 'othertests' have been migrated into 'regressiontests', and converted into doctests/unittests, as appropriate.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@3661 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
parent
77ab11be45
commit
97b9ad73b4
2665
tests/doctest.py
2665
tests/doctest.py
File diff suppressed because it is too large
Load Diff
|
@ -13,8 +13,7 @@ class Article(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.headline
|
return self.headline
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS': """
|
||||||
|
|
||||||
# No articles are in the system yet.
|
# No articles are in the system yet.
|
||||||
>>> Article.objects.all()
|
>>> Article.objects.all()
|
||||||
[]
|
[]
|
||||||
|
@ -314,14 +313,14 @@ AttributeError: Manager isn't accessible via Article instances
|
||||||
>>> Article.objects.all()
|
>>> Article.objects.all()
|
||||||
[<Article: Article 6>, <Article: Default headline>, <Article: Article 7>, <Article: Updated article 8>]
|
[<Article: Article 6>, <Article: Default headline>, <Article: Article 7>, <Article: Updated article 8>]
|
||||||
|
|
||||||
"""
|
"""}
|
||||||
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
|
|
||||||
building_docs = getattr(settings, 'BUILDING_DOCS', False)
|
building_docs = getattr(settings, 'BUILDING_DOCS', False)
|
||||||
|
|
||||||
if building_docs or settings.DATABASE_ENGINE == 'postgresql':
|
if building_docs or settings.DATABASE_ENGINE == 'postgresql':
|
||||||
API_TESTS += """
|
__test__['API_TESTS'] += """
|
||||||
# In PostgreSQL, microsecond-level precision is available.
|
# In PostgreSQL, microsecond-level precision is available.
|
||||||
>>> a9 = Article(headline='Article 9', pub_date=datetime(2005, 7, 31, 12, 30, 45, 180))
|
>>> a9 = Article(headline='Article 9', pub_date=datetime(2005, 7, 31, 12, 30, 45, 180))
|
||||||
>>> a9.save()
|
>>> a9.save()
|
||||||
|
@ -330,7 +329,7 @@ datetime.datetime(2005, 7, 31, 12, 30, 45, 180)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
if building_docs or settings.DATABASE_ENGINE == 'mysql':
|
if building_docs or settings.DATABASE_ENGINE == 'mysql':
|
||||||
API_TESTS += """
|
__test__['API_TESTS'] += """
|
||||||
# In MySQL, microsecond-level precision isn't available. You'll lose
|
# In MySQL, microsecond-level precision isn't available. You'll lose
|
||||||
# microsecond-level precision once the data is saved.
|
# microsecond-level precision once the data is saved.
|
||||||
>>> a9 = Article(headline='Article 9', pub_date=datetime(2005, 7, 31, 12, 30, 45, 180))
|
>>> a9 = Article(headline='Article 9', pub_date=datetime(2005, 7, 31, 12, 30, 45, 180))
|
||||||
|
@ -339,7 +338,7 @@ if building_docs or settings.DATABASE_ENGINE == 'mysql':
|
||||||
datetime.datetime(2005, 7, 31, 12, 30, 45)
|
datetime.datetime(2005, 7, 31, 12, 30, 45)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
API_TESTS += """
|
__test__['API_TESTS'] += """
|
||||||
|
|
||||||
# You can manually specify the primary key when creating a new object.
|
# You can manually specify the primary key when creating a new object.
|
||||||
>>> a101 = Article(id=101, headline='Article 101', pub_date=datetime(2005, 7, 31, 12, 30, 45))
|
>>> a101 = Article(id=101, headline='Article 101', pub_date=datetime(2005, 7, 31, 12, 30, 45))
|
||||||
|
|
|
@ -23,7 +23,7 @@ class Person(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> a = Person(name='Adrian', gender='M')
|
>>> a = Person(name='Adrian', gender='M')
|
||||||
>>> a.save()
|
>>> a.save()
|
||||||
>>> s = Person(name='Sara', gender='F')
|
>>> s = Person(name='Sara', gender='F')
|
||||||
|
@ -36,4 +36,4 @@ API_TESTS = """
|
||||||
'Male'
|
'Male'
|
||||||
>>> s.get_gender_display()
|
>>> s.get_gender_display()
|
||||||
'Female'
|
'Female'
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -15,7 +15,7 @@ class Person(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return '%s %s' % (self.first_name, self.last_name)
|
return '%s %s' % (self.first_name, self.last_name)
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create a Person.
|
# Create a Person.
|
||||||
>>> p = Person(first_name='John', last_name='Smith')
|
>>> p = Person(first_name='John', last_name='Smith')
|
||||||
>>> p.save()
|
>>> p.save()
|
||||||
|
@ -50,4 +50,4 @@ AttributeError: 'Person' object has no attribute 'firstname'
|
||||||
Traceback (most recent call last):
|
Traceback (most recent call last):
|
||||||
...
|
...
|
||||||
AttributeError: 'Person' object has no attribute 'last'
|
AttributeError: 'Person' object has no attribute 'last'
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -58,7 +58,7 @@ class Car(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> p1 = Person(first_name='Bugs', last_name='Bunny', fun=True)
|
>>> p1 = Person(first_name='Bugs', last_name='Bunny', fun=True)
|
||||||
>>> p1.save()
|
>>> p1.save()
|
||||||
>>> p2 = Person(first_name='Droopy', last_name='Dog', fun=False)
|
>>> p2 = Person(first_name='Droopy', last_name='Dog', fun=False)
|
||||||
|
@ -104,4 +104,4 @@ True
|
||||||
# to the first manager defined in the class. In this case, it's "cars".
|
# to the first manager defined in the class. In this case, it's "cars".
|
||||||
>>> Car._default_manager.order_by('name')
|
>>> Car._default_manager.order_by('name')
|
||||||
[<Car: Corvette>, <Car: Neon>]
|
[<Car: Corvette>, <Car: Neon>]
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -36,7 +36,7 @@ class Article(models.Model):
|
||||||
# positional arguments to Article().
|
# positional arguments to Article().
|
||||||
return [self.__class__(*row) for row in cursor.fetchall()]
|
return [self.__class__(*row) for row in cursor.fetchall()]
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create a couple of Articles.
|
# Create a couple of Articles.
|
||||||
>>> from datetime import date
|
>>> from datetime import date
|
||||||
>>> a = Article(id=None, headline='Area man programs in Python', pub_date=date(2005, 7, 27))
|
>>> a = Article(id=None, headline='Area man programs in Python', pub_date=date(2005, 7, 27))
|
||||||
|
@ -55,4 +55,4 @@ False
|
||||||
[<Article: Area man programs in Python>]
|
[<Article: Area man programs in Python>]
|
||||||
>>> b.articles_from_same_day_2()
|
>>> b.articles_from_same_day_2()
|
||||||
[<Article: Area man programs in Python>]
|
[<Article: Area man programs in Python>]
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -27,7 +27,7 @@ class Business(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> dan = Employee(employee_code='ABC123', first_name='Dan', last_name='Jones')
|
>>> dan = Employee(employee_code='ABC123', first_name='Dan', last_name='Jones')
|
||||||
>>> dan.save()
|
>>> dan.save()
|
||||||
>>> Employee.objects.all()
|
>>> Employee.objects.all()
|
||||||
|
@ -88,4 +88,4 @@ DoesNotExist: Employee matching query does not exist.
|
||||||
>>> Business.objects.filter(employees__first_name__startswith='Fran')
|
>>> Business.objects.filter(employees__first_name__startswith='Fran')
|
||||||
[<Business: Sears>]
|
[<Business: Sears>]
|
||||||
|
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -10,7 +10,7 @@ from django.db import models
|
||||||
class Empty(models.Model):
|
class Empty(models.Model):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> m = Empty()
|
>>> m = Empty()
|
||||||
>>> m.id
|
>>> m.id
|
||||||
>>> m.save()
|
>>> m.save()
|
||||||
|
@ -23,4 +23,4 @@ True
|
||||||
>>> existing = Empty(m.id)
|
>>> existing = Empty(m.id)
|
||||||
>>> existing.save()
|
>>> existing.save()
|
||||||
|
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -19,7 +19,7 @@ class Article(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.headline
|
return self.headline
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> from datetime import datetime
|
>>> from datetime import datetime
|
||||||
|
|
||||||
# No articles are in the system yet.
|
# No articles are in the system yet.
|
||||||
|
@ -48,4 +48,4 @@ API_TESTS = """
|
||||||
>>> d = now - a.pub_date
|
>>> d = now - a.pub_date
|
||||||
>>> d.seconds < 5
|
>>> d.seconds < 5
|
||||||
True
|
True
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -53,7 +53,7 @@ class Mineral(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create the world in 7 lines of code...
|
# Create the world in 7 lines of code...
|
||||||
>>> lion = Animal(common_name="Lion", latin_name="Panthera leo")
|
>>> lion = Animal(common_name="Lion", latin_name="Panthera leo")
|
||||||
>>> platypus = Animal(common_name="Platypus", latin_name="Ornithorhynchus anatinus")
|
>>> platypus = Animal(common_name="Platypus", latin_name="Ornithorhynchus anatinus")
|
||||||
|
@ -105,4 +105,4 @@ API_TESTS = """
|
||||||
[<TaggedItem: shiny>]
|
[<TaggedItem: shiny>]
|
||||||
>>> TaggedItem.objects.filter(content_type__pk=ctype.id, object_id=quartz.id)
|
>>> TaggedItem.objects.filter(content_type__pk=ctype.id, object_id=quartz.id)
|
||||||
[<TaggedItem: clearish>]
|
[<TaggedItem: clearish>]
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -29,7 +29,7 @@ class Person(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Because no Articles exist yet, get_latest() raises ArticleDoesNotExist.
|
# Because no Articles exist yet, get_latest() raises ArticleDoesNotExist.
|
||||||
>>> Article.objects.latest()
|
>>> Article.objects.latest()
|
||||||
Traceback (most recent call last):
|
Traceback (most recent call last):
|
||||||
|
@ -76,4 +76,4 @@ AssertionError: latest() requires either a field_name parameter or 'get_latest_b
|
||||||
|
|
||||||
>>> Person.objects.latest('birthday')
|
>>> Person.objects.latest('birthday')
|
||||||
<Person: Stephanie>
|
<Person: Stephanie>
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -15,7 +15,7 @@ class Person(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return '%s %s' % (self.first_name, self.last_name)
|
return '%s %s' % (self.first_name, self.last_name)
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Acting as a divine being, create an Person.
|
# Acting as a divine being, create an Person.
|
||||||
>>> from datetime import date
|
>>> from datetime import date
|
||||||
>>> p = Person(first_name='John', last_name='Lennon', birthday=date(1940, 10, 9))
|
>>> p = Person(first_name='John', last_name='Lennon', birthday=date(1940, 10, 9))
|
||||||
|
@ -49,4 +49,4 @@ True
|
||||||
False
|
False
|
||||||
>>> Person.objects.count()
|
>>> Person.objects.count()
|
||||||
2
|
2
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -78,7 +78,7 @@ class SelfClashM2M(models.Model):
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
error_log = """invalid_models.fielderrors: "charfield": CharFields require a "maxlength" attribute.
|
model_errors = """invalid_models.fielderrors: "charfield": CharFields require a "maxlength" attribute.
|
||||||
invalid_models.fielderrors: "floatfield": FloatFields require a "decimal_places" attribute.
|
invalid_models.fielderrors: "floatfield": FloatFields require a "decimal_places" attribute.
|
||||||
invalid_models.fielderrors: "floatfield": FloatFields require a "max_digits" attribute.
|
invalid_models.fielderrors: "floatfield": FloatFields require a "max_digits" attribute.
|
||||||
invalid_models.fielderrors: "filefield": FileFields require an "upload_to" attribute.
|
invalid_models.fielderrors: "filefield": FileFields require an "upload_to" attribute.
|
||||||
|
|
|
@ -15,7 +15,7 @@ class Article(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.headline
|
return self.headline
|
||||||
|
|
||||||
API_TESTS = r"""
|
__test__ = {'API_TESTS':r"""
|
||||||
# Create a couple of Articles.
|
# Create a couple of Articles.
|
||||||
>>> from datetime import datetime
|
>>> from datetime import datetime
|
||||||
>>> a1 = Article(headline='Article 1', pub_date=datetime(2005, 7, 26))
|
>>> a1 = Article(headline='Article 1', pub_date=datetime(2005, 7, 26))
|
||||||
|
@ -191,4 +191,4 @@ DoesNotExist: Article matching query does not exist.
|
||||||
>>> Article.objects.filter(headline__contains='\\')
|
>>> Article.objects.filter(headline__contains='\\')
|
||||||
[<Article: Article with \ backslash>]
|
[<Article: Article with \ backslash>]
|
||||||
|
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -21,7 +21,7 @@ class Issue(models.Model):
|
||||||
ordering = ('num',)
|
ordering = ('num',)
|
||||||
|
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> Issue.objects.all()
|
>>> Issue.objects.all()
|
||||||
[]
|
[]
|
||||||
>>> r = User(username='russell')
|
>>> r = User(username='russell')
|
||||||
|
@ -62,4 +62,4 @@ API_TESTS = """
|
||||||
[<Issue: 1>, <Issue: 2>, <Issue: 3>]
|
[<Issue: 1>, <Issue: 2>, <Issue: 3>]
|
||||||
>>> Issue.objects.filter(Q(client=r.id) | Q(cc__id__exact=r.id))
|
>>> Issue.objects.filter(Q(client=r.id) | Q(cc__id__exact=r.id))
|
||||||
[<Issue: 1>, <Issue: 2>, <Issue: 3>]
|
[<Issue: 1>, <Issue: 2>, <Issue: 3>]
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -34,7 +34,7 @@ class Writer(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return '%s (%s)' % (self.reporter, self.position)
|
return '%s (%s)' % (self.reporter, self.position)
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create a few Reporters.
|
# Create a few Reporters.
|
||||||
>>> r1 = Reporter(first_name='John', last_name='Smith')
|
>>> r1 = Reporter(first_name='John', last_name='Smith')
|
||||||
>>> r1.save()
|
>>> r1.save()
|
||||||
|
@ -65,4 +65,4 @@ API_TESTS = """
|
||||||
<Article: This is a test>
|
<Article: This is a test>
|
||||||
>>> r1.writer_set.all()
|
>>> r1.writer_set.all()
|
||||||
[<Writer: John Smith (Main writer)>]
|
[<Writer: John Smith (Main writer)>]
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -28,7 +28,7 @@ class Article(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.headline
|
return self.headline
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> from datetime import datetime
|
>>> from datetime import datetime
|
||||||
|
|
||||||
>>> c1 = Category(name='Sports')
|
>>> c1 = Category(name='Sports')
|
||||||
|
@ -76,4 +76,4 @@ API_TESTS = """
|
||||||
[]
|
[]
|
||||||
>>> c4.secondary_article_set.all()
|
>>> c4.secondary_article_set.all()
|
||||||
[<Article: Area man steals>, <Article: Area man runs>]
|
[<Article: Area man steals>, <Article: Area man runs>]
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -22,7 +22,7 @@ class Person(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> a = Person(name='Anne')
|
>>> a = Person(name='Anne')
|
||||||
>>> a.save()
|
>>> a.save()
|
||||||
>>> b = Person(name='Bill')
|
>>> b = Person(name='Bill')
|
||||||
|
@ -189,4 +189,4 @@ API_TESTS = """
|
||||||
>>> d.stalkers.all()
|
>>> d.stalkers.all()
|
||||||
[<Person: Chuck>]
|
[<Person: Chuck>]
|
||||||
|
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -19,7 +19,7 @@ class Category(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create a few Category objects.
|
# Create a few Category objects.
|
||||||
>>> r = Category(id=None, name='Root category', parent=None)
|
>>> r = Category(id=None, name='Root category', parent=None)
|
||||||
>>> r.save()
|
>>> r.save()
|
||||||
|
@ -37,4 +37,4 @@ None
|
||||||
[]
|
[]
|
||||||
>>> c.parent
|
>>> c.parent
|
||||||
<Category: Root category>
|
<Category: Root category>
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -17,7 +17,7 @@ class Person(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.full_name
|
return self.full_name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create two Person objects -- the mom and dad in our family.
|
# Create two Person objects -- the mom and dad in our family.
|
||||||
>>> dad = Person(full_name='John Smith Senior', mother=None, father=None)
|
>>> dad = Person(full_name='John Smith Senior', mother=None, father=None)
|
||||||
>>> dad.save()
|
>>> dad.save()
|
||||||
|
@ -40,4 +40,4 @@ API_TESTS = """
|
||||||
[]
|
[]
|
||||||
>>> kid.fathers_child_set.all()
|
>>> kid.fathers_child_set.all()
|
||||||
[]
|
[]
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -21,7 +21,7 @@ class Album(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> from django.utils.datastructures import MultiValueDict
|
>>> from django.utils.datastructures import MultiValueDict
|
||||||
|
|
||||||
# Create a Musician object via the default AddManipulator.
|
# Create a Musician object via the default AddManipulator.
|
||||||
|
@ -88,4 +88,4 @@ True
|
||||||
<Album: Ultimate Ella>
|
<Album: Ultimate Ella>
|
||||||
>>> a2.release_date
|
>>> a2.release_date
|
||||||
datetime.date(2005, 2, 13)
|
datetime.date(2005, 2, 13)
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -28,7 +28,7 @@ class Article(models.Model):
|
||||||
class Meta:
|
class Meta:
|
||||||
ordering = ('headline',)
|
ordering = ('headline',)
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create a couple of Publications.
|
# Create a couple of Publications.
|
||||||
>>> p1 = Publication(id=None, title='The Python Journal')
|
>>> p1 = Publication(id=None, title='The Python Journal')
|
||||||
>>> p1.save()
|
>>> p1.save()
|
||||||
|
@ -231,4 +231,4 @@ API_TESTS = """
|
||||||
>>> p1.article_set.all()
|
>>> p1.article_set.all()
|
||||||
[<Article: NASA uses Python>]
|
[<Article: NASA uses Python>]
|
||||||
|
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -25,7 +25,7 @@ class Article(models.Model):
|
||||||
class Meta:
|
class Meta:
|
||||||
ordering = ('headline',)
|
ordering = ('headline',)
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create a few Reporters.
|
# Create a few Reporters.
|
||||||
>>> r = Reporter(first_name='John', last_name='Smith', email='john@example.com')
|
>>> r = Reporter(first_name='John', last_name='Smith', email='john@example.com')
|
||||||
>>> r.save()
|
>>> r.save()
|
||||||
|
@ -263,4 +263,4 @@ TypeError: Cannot resolve keyword 'reporter_id' into field
|
||||||
>>> Article.objects.all()
|
>>> Article.objects.all()
|
||||||
[]
|
[]
|
||||||
|
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -23,7 +23,7 @@ class Article(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.headline
|
return self.headline
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create a Reporter.
|
# Create a Reporter.
|
||||||
>>> r = Reporter(name='John Smith')
|
>>> r = Reporter(name='John Smith')
|
||||||
>>> r.save()
|
>>> r.save()
|
||||||
|
@ -121,4 +121,4 @@ DoesNotExist: <Article: Fourth> is not related to <Reporter: John Smith>.
|
||||||
>>> Article.objects.filter(reporter__isnull=True)
|
>>> Article.objects.filter(reporter__isnull=True)
|
||||||
[<Article: First>, <Article: Fourth>]
|
[<Article: First>, <Article: Fourth>]
|
||||||
|
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -26,7 +26,7 @@ class ItalianRestaurant(Restaurant):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return "%s the italian restaurant" % self.name
|
return "%s the italian restaurant" % self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Make sure Restaurant has the right fields in the right order.
|
# Make sure Restaurant has the right fields in the right order.
|
||||||
>>> [f.name for f in Restaurant._meta.fields]
|
>>> [f.name for f in Restaurant._meta.fields]
|
||||||
['id', 'name', 'address', 'serves_hot_dogs', 'serves_pizza']
|
['id', 'name', 'address', 'serves_hot_dogs', 'serves_pizza']
|
||||||
|
@ -50,4 +50,4 @@ API_TESTS = """
|
||||||
>>> ir.save()
|
>>> ir.save()
|
||||||
|
|
||||||
|
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -14,7 +14,7 @@ class Child(Model):
|
||||||
name = CharField(maxlength=100)
|
name = CharField(maxlength=100)
|
||||||
parent = ForeignKey(Parent)
|
parent = ForeignKey(Parent)
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create a Parent
|
# Create a Parent
|
||||||
>>> q = Parent(name='Elizabeth')
|
>>> q = Parent(name='Elizabeth')
|
||||||
>>> q.save()
|
>>> q.save()
|
||||||
|
@ -29,4 +29,4 @@ API_TESTS = """
|
||||||
|
|
||||||
>>> q.delete()
|
>>> q.delete()
|
||||||
|
|
||||||
"""
|
"""}
|
|
@ -30,7 +30,7 @@ class Waiter(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return "%s the waiter at %s" % (self.name, self.restaurant)
|
return "%s the waiter at %s" % (self.name, self.restaurant)
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create a couple of Places.
|
# Create a couple of Places.
|
||||||
>>> p1 = Place(name='Demon Dogs', address='944 W. Fullerton')
|
>>> p1 = Place(name='Demon Dogs', address='944 W. Fullerton')
|
||||||
>>> p1.save()
|
>>> p1.save()
|
||||||
|
@ -151,4 +151,4 @@ DoesNotExist: Restaurant matching query does not exist.
|
||||||
# Delete the restaurant; the waiter should also be removed
|
# Delete the restaurant; the waiter should also be removed
|
||||||
>>> r = Restaurant.objects.get(pk=1)
|
>>> r = Restaurant.objects.get(pk=1)
|
||||||
>>> r.delete()
|
>>> r.delete()
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -23,7 +23,7 @@ class Article(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.headline
|
return self.headline
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> from datetime import datetime
|
>>> from datetime import datetime
|
||||||
>>> from django.db.models import Q
|
>>> from django.db.models import Q
|
||||||
|
|
||||||
|
@ -101,4 +101,4 @@ API_TESTS = """
|
||||||
[<Article: Hello>]
|
[<Article: Hello>]
|
||||||
>>> Article.objects.complex_filter(Q(pk=1) | Q(pk=2))
|
>>> Article.objects.complex_filter(Q(pk=1) | Q(pk=2))
|
||||||
[<Article: Hello>, <Article: Goodbye>]
|
[<Article: Hello>, <Article: Goodbye>]
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -24,7 +24,7 @@ class Article(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.headline
|
return self.headline
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create a couple of Articles.
|
# Create a couple of Articles.
|
||||||
>>> from datetime import datetime
|
>>> from datetime import datetime
|
||||||
>>> a1 = Article(headline='Article 1', pub_date=datetime(2005, 7, 26))
|
>>> a1 = Article(headline='Article 1', pub_date=datetime(2005, 7, 26))
|
||||||
|
@ -64,4 +64,4 @@ API_TESTS = """
|
||||||
# don't know what order the output will be in.
|
# don't know what order the output will be in.
|
||||||
>>> Article.objects.order_by('?')
|
>>> Article.objects.order_by('?')
|
||||||
[...]
|
[...]
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -15,7 +15,7 @@ class Article(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.headline
|
return self.headline
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# prepare a list of objects for pagination
|
# prepare a list of objects for pagination
|
||||||
>>> from datetime import datetime
|
>>> from datetime import datetime
|
||||||
>>> for x in range(1, 10):
|
>>> for x in range(1, 10):
|
||||||
|
@ -64,4 +64,4 @@ True
|
||||||
>>> paginator.last_on_page(1)
|
>>> paginator.last_on_page(1)
|
||||||
9
|
9
|
||||||
|
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -20,7 +20,7 @@ class Person(models.Model):
|
||||||
|
|
||||||
full_name_2 = property(_get_full_name, _set_full_name)
|
full_name_2 = property(_get_full_name, _set_full_name)
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> a = Person(first_name='John', last_name='Lennon')
|
>>> a = Person(first_name='John', last_name='Lennon')
|
||||||
>>> a.save()
|
>>> a.save()
|
||||||
>>> a.full_name
|
>>> a.full_name
|
||||||
|
@ -37,4 +37,4 @@ AttributeError: can't set attribute
|
||||||
>>> a2.save()
|
>>> a2.save()
|
||||||
>>> a2.first_name
|
>>> a2.first_name
|
||||||
'Paul'
|
'Paul'
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -24,7 +24,7 @@ class Thing(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.when
|
return self.when
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> import datetime
|
>>> import datetime
|
||||||
>>> day1 = datetime.date(2005, 1, 1)
|
>>> day1 = datetime.date(2005, 1, 1)
|
||||||
>>> day2 = datetime.date(2006, 2, 2)
|
>>> day2 = datetime.date(2006, 2, 2)
|
||||||
|
@ -53,4 +53,4 @@ b
|
||||||
|
|
||||||
>>> Thing.objects.filter(where__month=1)
|
>>> Thing.objects.filter(where__month=1)
|
||||||
[<Thing: a>]
|
[<Thing: a>]
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -27,7 +27,7 @@ class Choice(models.Model):
|
||||||
def __str(self):
|
def __str(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> john = User(name="John Doe")
|
>>> john = User(name="John Doe")
|
||||||
>>> john.save()
|
>>> john.save()
|
||||||
>>> jim = User(name="Jim Bo")
|
>>> jim = User(name="Jim Bo")
|
||||||
|
@ -56,4 +56,4 @@ API_TESTS = """
|
||||||
Traceback (most recent call last):
|
Traceback (most recent call last):
|
||||||
...
|
...
|
||||||
TypeError: Cannot resolve keyword 'choice' into field
|
TypeError: Cannot resolve keyword 'choice' into field
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -24,7 +24,7 @@ class Person(models.Model):
|
||||||
super(Person, self).delete() # Call the "real" delete() method
|
super(Person, self).delete() # Call the "real" delete() method
|
||||||
print "After deletion"
|
print "After deletion"
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> p1 = Person(first_name='John', last_name='Smith')
|
>>> p1 = Person(first_name='John', last_name='Smith')
|
||||||
>>> p1.save()
|
>>> p1.save()
|
||||||
Before save
|
Before save
|
||||||
|
@ -39,4 +39,4 @@ After deletion
|
||||||
|
|
||||||
>>> Person.objects.all()
|
>>> Person.objects.all()
|
||||||
[]
|
[]
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -37,7 +37,7 @@ class Article(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.headline
|
return self.headline
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create some data:
|
# Create some data:
|
||||||
>>> from datetime import datetime
|
>>> from datetime import datetime
|
||||||
>>> sports = Category(name="Sports")
|
>>> sports = Category(name="Sports")
|
||||||
|
@ -118,4 +118,4 @@ API_TESTS = """
|
||||||
>>> Article.objects.all()
|
>>> Article.objects.all()
|
||||||
[<Article: Just kidding; I love TV poker>, <Article: Time to reform copyright>]
|
[<Article: Just kidding; I love TV poker>, <Article: Time to reform copyright>]
|
||||||
|
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -17,7 +17,7 @@ class Article(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.headline
|
return self.headline
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Create an Article.
|
# Create an Article.
|
||||||
>>> from datetime import datetime
|
>>> from datetime import datetime
|
||||||
>>> a = Article(headline='Area man programs in Python', pub_date=datetime(2005, 7, 28))
|
>>> a = Article(headline='Area man programs in Python', pub_date=datetime(2005, 7, 28))
|
||||||
|
@ -28,4 +28,4 @@ API_TESTS = """
|
||||||
|
|
||||||
>>> a
|
>>> a
|
||||||
<Article: Area man programs in Python>
|
<Article: Area man programs in Python>
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -17,16 +17,16 @@ class Reporter(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return "%s %s" % (self.first_name, self.last_name)
|
return "%s %s" % (self.first_name, self.last_name)
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
>>> from django.db import connection, transaction
|
>>> from django.db import connection, transaction
|
||||||
"""
|
"""}
|
||||||
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
|
|
||||||
building_docs = getattr(settings, 'BUILDING_DOCS', False)
|
building_docs = getattr(settings, 'BUILDING_DOCS', False)
|
||||||
|
|
||||||
if building_docs or settings.DATABASE_ENGINE != 'mysql':
|
if building_docs or settings.DATABASE_ENGINE != 'mysql':
|
||||||
API_TESTS += """
|
__test__['API_TESTS'] += """
|
||||||
# the default behavior is to autocommit after each save() action
|
# the default behavior is to autocommit after each save() action
|
||||||
>>> def create_a_reporter_then_fail(first, last):
|
>>> def create_a_reporter_then_fail(first, last):
|
||||||
... a = Reporter(first_name=first, last_name=last)
|
... a = Reporter(first_name=first, last_name=last)
|
||||||
|
|
|
@ -20,7 +20,7 @@ class Person(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
|
|
||||||
>>> import datetime
|
>>> import datetime
|
||||||
>>> valid_params = {
|
>>> valid_params = {
|
||||||
|
@ -146,4 +146,4 @@ u'john@example.com'
|
||||||
>>> p.validate()
|
>>> p.validate()
|
||||||
{'email': ['Enter a valid e-mail address.']}
|
{'email': ['Enter a valid e-mail address.']}
|
||||||
|
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -1,60 +0,0 @@
|
||||||
# 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-existent 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
|
|
|
@ -1,70 +0,0 @@
|
||||||
# Quick tests for the markup templatetags (django.contrib.markup)
|
|
||||||
|
|
||||||
from django.template import Template, Context, add_to_builtins
|
|
||||||
import re
|
|
||||||
|
|
||||||
add_to_builtins('django.contrib.markup.templatetags.markup')
|
|
||||||
|
|
||||||
# find out if markup modules are installed and tailor the test appropriately
|
|
||||||
try:
|
|
||||||
import textile
|
|
||||||
except ImportError:
|
|
||||||
textile = None
|
|
||||||
|
|
||||||
try:
|
|
||||||
import markdown
|
|
||||||
except ImportError:
|
|
||||||
markdown = None
|
|
||||||
|
|
||||||
try:
|
|
||||||
import docutils
|
|
||||||
except ImportError:
|
|
||||||
docutils = None
|
|
||||||
|
|
||||||
# simple examples 'cause this isn't actually testing the markup, just
|
|
||||||
# that the filters work as advertised
|
|
||||||
|
|
||||||
### test textile
|
|
||||||
|
|
||||||
textile_content = """Paragraph 1
|
|
||||||
|
|
||||||
Paragraph 2 with "quotes" and @code@"""
|
|
||||||
|
|
||||||
t = Template("{{ textile_content|textile }}")
|
|
||||||
rendered = t.render(Context(locals())).strip()
|
|
||||||
if textile:
|
|
||||||
assert rendered == """<p>Paragraph 1</p>
|
|
||||||
|
|
||||||
<p>Paragraph 2 with “quotes” and <code>code</code></p>"""
|
|
||||||
else:
|
|
||||||
assert rendered == textile_content
|
|
||||||
|
|
||||||
### test markdown
|
|
||||||
|
|
||||||
markdown_content = """Paragraph 1
|
|
||||||
|
|
||||||
## An h2"""
|
|
||||||
|
|
||||||
t = Template("{{ markdown_content|markdown }}")
|
|
||||||
rendered = t.render(Context(locals())).strip()
|
|
||||||
if markdown:
|
|
||||||
pattern = re.compile("""<p>Paragraph 1\s*</p>\s*<h2>\s*An h2</h2>""")
|
|
||||||
assert pattern.match(rendered)
|
|
||||||
else:
|
|
||||||
assert rendered == markdown_content
|
|
||||||
|
|
||||||
### test rest
|
|
||||||
|
|
||||||
rest_content = """Paragraph 1
|
|
||||||
|
|
||||||
Paragraph 2 with a link_
|
|
||||||
|
|
||||||
.. _link: http://www.example.com/"""
|
|
||||||
|
|
||||||
t = Template("{{ rest_content|restructuredtext }}")
|
|
||||||
rendered = t.render(Context(locals())).strip()
|
|
||||||
if docutils:
|
|
||||||
assert rendered =="""<p>Paragraph 1</p>
|
|
||||||
<p>Paragraph 2 with a <a class="reference" href="http://www.example.com/">link</a></p>"""
|
|
||||||
else:
|
|
||||||
assert rendered == rest_content
|
|
|
@ -1,634 +0,0 @@
|
||||||
from django.conf import settings
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
|
||||||
# When running this file in isolation, we need to set up the configuration
|
|
||||||
# before importing 'template'.
|
|
||||||
settings.configure()
|
|
||||||
|
|
||||||
from django import template
|
|
||||||
from django.template import loader
|
|
||||||
from django.utils.translation import activate, deactivate, install
|
|
||||||
from django.utils.tzinfo import LocalTimezone
|
|
||||||
from datetime import datetime, timedelta
|
|
||||||
import traceback
|
|
||||||
|
|
||||||
#################################
|
|
||||||
# Custom template tag for tests #
|
|
||||||
#################################
|
|
||||||
|
|
||||||
register = template.Library()
|
|
||||||
|
|
||||||
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:])
|
|
||||||
|
|
||||||
register.tag("echo", do_echo)
|
|
||||||
|
|
||||||
template.libraries['django.templatetags.testtags'] = register
|
|
||||||
|
|
||||||
#####################################
|
|
||||||
# Helper objects for template tests #
|
|
||||||
#####################################
|
|
||||||
|
|
||||||
class SomeException(Exception):
|
|
||||||
silent_variable_failure = True
|
|
||||||
|
|
||||||
class SomeOtherException(Exception):
|
|
||||||
pass
|
|
||||||
|
|
||||||
class SomeClass:
|
|
||||||
def __init__(self):
|
|
||||||
self.otherclass = OtherClass()
|
|
||||||
|
|
||||||
def method(self):
|
|
||||||
return "SomeClass.method"
|
|
||||||
|
|
||||||
def method2(self, o):
|
|
||||||
return o
|
|
||||||
|
|
||||||
def method3(self):
|
|
||||||
raise SomeException
|
|
||||||
|
|
||||||
def method4(self):
|
|
||||||
raise SomeOtherException
|
|
||||||
|
|
||||||
class OtherClass:
|
|
||||||
def method(self):
|
|
||||||
return "OtherClass.method"
|
|
||||||
|
|
||||||
# NOW and NOW_tz are used by timesince tag tests.
|
|
||||||
NOW = datetime.now()
|
|
||||||
NOW_tz = datetime.now(LocalTimezone(datetime.now()))
|
|
||||||
|
|
||||||
# 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", {}, "asINVALIDdf"),
|
|
||||||
|
|
||||||
# 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()}, "INVALID"),
|
|
||||||
|
|
||||||
# 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"}}, "INVALID"),
|
|
||||||
|
|
||||||
# Fail silently when accessing a non-simple method
|
|
||||||
'basic-syntax20': ("{{ var.method2 }}", {"var": SomeClass()}, "INVALID"),
|
|
||||||
|
|
||||||
# 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),
|
|
||||||
|
|
||||||
# Chained filters, with an argument to the first one
|
|
||||||
'basic-syntax29': ('{{ var|removetags:"b i"|upper|lower }}', {"var": "<b><i>Yes</i></b>"}, "yes"),
|
|
||||||
|
|
||||||
# Escaped string as argument
|
|
||||||
'basic-syntax30': (r'{{ var|default_if_none:" endquote\" hah" }}', {"var": None}, ' endquote" hah'),
|
|
||||||
|
|
||||||
# Variable as argument
|
|
||||||
'basic-syntax31': (r'{{ var|default_if_none:var2 }}', {"var": None, "var2": "happy"}, 'happy'),
|
|
||||||
|
|
||||||
# Default argument testing
|
|
||||||
'basic-syntax32': (r'{{ var|yesno:"yup,nup,mup" }} {{ var|yesno }}', {"var": True}, 'yup yes'),
|
|
||||||
|
|
||||||
# Fail silently for methods that raise an exception with a "silent_variable_failure" attribute
|
|
||||||
'basic-syntax33': (r'1{{ var.method3 }}2', {"var": SomeClass()}, "1INVALID2"),
|
|
||||||
|
|
||||||
# In methods that raise an exception without a "silent_variable_attribute" set to True,
|
|
||||||
# the exception propogates
|
|
||||||
'basic-syntax34': (r'1{{ var.method4 }}2', {"var": SomeClass()}, SomeOtherException),
|
|
||||||
|
|
||||||
# Escaped backslash in argument
|
|
||||||
'basic-syntax35': (r'{{ var|default_if_none:"foo\bar" }}', {"var": None}, r'foo\bar'),
|
|
||||||
|
|
||||||
# Escaped backslash using known escape char
|
|
||||||
'basic-syntax35': (r'{{ var|default_if_none:"foo\now" }}', {"var": None}, r'foo\now'),
|
|
||||||
|
|
||||||
### COMMENT TAG ###########################################################
|
|
||||||
'comment-tag01': ("{% comment %}this is hidden{% endcomment %}hello", {}, "hello"),
|
|
||||||
'comment-tag02': ("{% comment %}this is hidden{% endcomment %}hello{% comment %}foo{% endcomment %}", {}, "hello"),
|
|
||||||
|
|
||||||
# Comment tag can contain invalid stuff.
|
|
||||||
'comment-tag03': ("foo{% comment %} {% if %} {% endcomment %}", {}, "foo"),
|
|
||||||
'comment-tag04': ("foo{% comment %} {% endblock %} {% endcomment %}", {}, "foo"),
|
|
||||||
'comment-tag05': ("foo{% comment %} {% somerandomtag %} {% endcomment %}", {}, "foo"),
|
|
||||||
|
|
||||||
### CYCLE TAG #############################################################
|
|
||||||
'cycle01': ('{% cycle a %}', {}, template.TemplateSyntaxError),
|
|
||||||
'cycle02': ('{% cycle a,b,c as abc %}{% cycle abc %}', {}, 'ab'),
|
|
||||||
'cycle03': ('{% cycle a,b,c as abc %}{% cycle abc %}{% cycle abc %}', {}, 'abc'),
|
|
||||||
'cycle04': ('{% cycle a,b,c as abc %}{% cycle abc %}{% cycle abc %}{% cycle abc %}', {}, 'abca'),
|
|
||||||
'cycle05': ('{% cycle %}', {}, template.TemplateSyntaxError),
|
|
||||||
'cycle06': ('{% cycle a %}', {}, template.TemplateSyntaxError),
|
|
||||||
'cycle07': ('{% cycle a,b,c as foo %}{% cycle bar %}', {}, template.TemplateSyntaxError),
|
|
||||||
|
|
||||||
### EXCEPTIONS ############################################################
|
|
||||||
|
|
||||||
# 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),
|
|
||||||
|
|
||||||
### FILTER TAG ############################################################
|
|
||||||
'filter01': ('{% filter upper %}{% endfilter %}', {}, ''),
|
|
||||||
'filter02': ('{% filter upper %}django{% endfilter %}', {}, 'DJANGO'),
|
|
||||||
'filter03': ('{% filter upper|lower %}django{% endfilter %}', {}, 'django'),
|
|
||||||
|
|
||||||
### FIRSTOF TAG ###########################################################
|
|
||||||
'firstof01': ('{% firstof a b c %}', {'a':0,'b':0,'c':0}, ''),
|
|
||||||
'firstof02': ('{% firstof a b c %}', {'a':1,'b':0,'c':0}, '1'),
|
|
||||||
'firstof03': ('{% firstof a b c %}', {'a':0,'b':2,'c':0}, '2'),
|
|
||||||
'firstof04': ('{% firstof a b c %}', {'a':0,'b':0,'c':3}, '3'),
|
|
||||||
'firstof05': ('{% firstof a b c %}', {'a':1,'b':2,'c':3}, '1'),
|
|
||||||
'firstof06': ('{% firstof %}', {}, template.TemplateSyntaxError),
|
|
||||||
|
|
||||||
### FOR TAG ###############################################################
|
|
||||||
'for-tag01': ("{% for val in values %}{{ val }}{% endfor %}", {"values": [1, 2, 3]}, "123"),
|
|
||||||
'for-tag02': ("{% for val in values reversed %}{{ val }}{% endfor %}", {"values": [1, 2, 3]}, "321"),
|
|
||||||
'for-tag-vars01': ("{% for val in values %}{{ forloop.counter }}{% endfor %}", {"values": [6, 6, 6]}, "123"),
|
|
||||||
'for-tag-vars02': ("{% for val in values %}{{ forloop.counter0 }}{% endfor %}", {"values": [6, 6, 6]}, "012"),
|
|
||||||
'for-tag-vars03': ("{% for val in values %}{{ forloop.revcounter }}{% endfor %}", {"values": [6, 6, 6]}, "321"),
|
|
||||||
'for-tag-vars04': ("{% for val in values %}{{ forloop.revcounter0 }}{% endfor %}", {"values": [6, 6, 6]}, "210"),
|
|
||||||
|
|
||||||
### IF TAG ################################################################
|
|
||||||
'if-tag01': ("{% if foo %}yes{% else %}no{% endif %}", {"foo": True}, "yes"),
|
|
||||||
'if-tag02': ("{% if foo %}yes{% else %}no{% endif %}", {"foo": False}, "no"),
|
|
||||||
'if-tag03': ("{% if foo %}yes{% else %}no{% endif %}", {}, "no"),
|
|
||||||
|
|
||||||
# AND
|
|
||||||
'if-tag-and01': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'yes'),
|
|
||||||
'if-tag-and02': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'no'),
|
|
||||||
'if-tag-and03': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'no'),
|
|
||||||
'if-tag-and04': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'no'),
|
|
||||||
'if-tag-and05': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'foo': False}, 'no'),
|
|
||||||
'if-tag-and06': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'bar': False}, 'no'),
|
|
||||||
'if-tag-and07': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'foo': True}, 'no'),
|
|
||||||
'if-tag-and08': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'bar': True}, 'no'),
|
|
||||||
|
|
||||||
# OR
|
|
||||||
'if-tag-or01': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'yes'),
|
|
||||||
'if-tag-or02': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'yes'),
|
|
||||||
'if-tag-or03': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'yes'),
|
|
||||||
'if-tag-or04': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'no'),
|
|
||||||
'if-tag-or05': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'foo': False}, 'no'),
|
|
||||||
'if-tag-or06': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'bar': False}, 'no'),
|
|
||||||
'if-tag-or07': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'foo': True}, 'yes'),
|
|
||||||
'if-tag-or08': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'bar': True}, 'yes'),
|
|
||||||
|
|
||||||
# TODO: multiple ORs
|
|
||||||
|
|
||||||
# NOT
|
|
||||||
'if-tag-not01': ("{% if not foo %}no{% else %}yes{% endif %}", {'foo': True}, 'yes'),
|
|
||||||
'if-tag-not02': ("{% if not %}yes{% else %}no{% endif %}", {'foo': True}, 'no'),
|
|
||||||
'if-tag-not03': ("{% if not %}yes{% else %}no{% endif %}", {'not': True}, 'yes'),
|
|
||||||
'if-tag-not04': ("{% if not not %}no{% else %}yes{% endif %}", {'not': True}, 'yes'),
|
|
||||||
'if-tag-not05': ("{% if not not %}no{% else %}yes{% endif %}", {}, 'no'),
|
|
||||||
|
|
||||||
'if-tag-not06': ("{% if foo and not bar %}yes{% else %}no{% endif %}", {}, 'no'),
|
|
||||||
'if-tag-not07': ("{% if foo and not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'no'),
|
|
||||||
'if-tag-not08': ("{% if foo and not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'yes'),
|
|
||||||
'if-tag-not09': ("{% if foo and not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'no'),
|
|
||||||
'if-tag-not10': ("{% if foo and not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'no'),
|
|
||||||
|
|
||||||
'if-tag-not11': ("{% if not foo and bar %}yes{% else %}no{% endif %}", {}, 'no'),
|
|
||||||
'if-tag-not12': ("{% if not foo and bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'no'),
|
|
||||||
'if-tag-not13': ("{% if not foo and bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'no'),
|
|
||||||
'if-tag-not14': ("{% if not foo and bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'yes'),
|
|
||||||
'if-tag-not15': ("{% if not foo and bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'no'),
|
|
||||||
|
|
||||||
'if-tag-not16': ("{% if foo or not bar %}yes{% else %}no{% endif %}", {}, 'yes'),
|
|
||||||
'if-tag-not17': ("{% if foo or not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'yes'),
|
|
||||||
'if-tag-not18': ("{% if foo or not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'yes'),
|
|
||||||
'if-tag-not19': ("{% if foo or not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'no'),
|
|
||||||
'if-tag-not20': ("{% if foo or not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'yes'),
|
|
||||||
|
|
||||||
'if-tag-not21': ("{% if not foo or bar %}yes{% else %}no{% endif %}", {}, 'yes'),
|
|
||||||
'if-tag-not22': ("{% if not foo or bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'yes'),
|
|
||||||
'if-tag-not23': ("{% if not foo or bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'no'),
|
|
||||||
'if-tag-not24': ("{% if not foo or bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'yes'),
|
|
||||||
'if-tag-not25': ("{% if not foo or bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'yes'),
|
|
||||||
|
|
||||||
'if-tag-not26': ("{% if not foo and not bar %}yes{% else %}no{% endif %}", {}, 'yes'),
|
|
||||||
'if-tag-not27': ("{% if not foo and not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'no'),
|
|
||||||
'if-tag-not28': ("{% if not foo and not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'no'),
|
|
||||||
'if-tag-not29': ("{% if not foo and not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'no'),
|
|
||||||
'if-tag-not30': ("{% if not foo and not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'yes'),
|
|
||||||
|
|
||||||
'if-tag-not31': ("{% if not foo or not bar %}yes{% else %}no{% endif %}", {}, 'yes'),
|
|
||||||
'if-tag-not32': ("{% if not foo or not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'no'),
|
|
||||||
'if-tag-not33': ("{% if not foo or not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'yes'),
|
|
||||||
'if-tag-not34': ("{% if not foo or not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'yes'),
|
|
||||||
'if-tag-not35': ("{% if not foo or not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'yes'),
|
|
||||||
|
|
||||||
# AND and OR raises a TemplateSyntaxError
|
|
||||||
'if-tag-error01': ("{% if foo or bar and baz %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, template.TemplateSyntaxError),
|
|
||||||
'if-tag-error02': ("{% if foo and %}yes{% else %}no{% endif %}", {'foo': True}, template.TemplateSyntaxError),
|
|
||||||
'if-tag-error03': ("{% if foo or %}yes{% else %}no{% endif %}", {'foo': True}, template.TemplateSyntaxError),
|
|
||||||
'if-tag-error04': ("{% if not foo and %}yes{% else %}no{% endif %}", {'foo': True}, template.TemplateSyntaxError),
|
|
||||||
'if-tag-error05': ("{% if not foo or %}yes{% else %}no{% endif %}", {'foo': True}, template.TemplateSyntaxError),
|
|
||||||
|
|
||||||
### IFCHANGED TAG #########################################################
|
|
||||||
'ifchanged01': ('{% for n in num %}{% ifchanged %}{{ n }}{% endifchanged %}{% endfor %}', { 'num': (1,2,3) }, '123'),
|
|
||||||
'ifchanged02': ('{% for n in num %}{% ifchanged %}{{ n }}{% endifchanged %}{% endfor %}', { 'num': (1,1,3) }, '13'),
|
|
||||||
'ifchanged03': ('{% for n in num %}{% ifchanged %}{{ n }}{% endifchanged %}{% endfor %}', { 'num': (1,1,1) }, '1'),
|
|
||||||
|
|
||||||
### IFEQUAL TAG ###########################################################
|
|
||||||
'ifequal01': ("{% ifequal a b %}yes{% endifequal %}", {"a": 1, "b": 2}, ""),
|
|
||||||
'ifequal02': ("{% ifequal a b %}yes{% endifequal %}", {"a": 1, "b": 1}, "yes"),
|
|
||||||
'ifequal03': ("{% ifequal a b %}yes{% else %}no{% endifequal %}", {"a": 1, "b": 2}, "no"),
|
|
||||||
'ifequal04': ("{% ifequal a b %}yes{% else %}no{% endifequal %}", {"a": 1, "b": 1}, "yes"),
|
|
||||||
'ifequal05': ("{% ifequal a 'test' %}yes{% else %}no{% endifequal %}", {"a": "test"}, "yes"),
|
|
||||||
'ifequal06': ("{% ifequal a 'test' %}yes{% else %}no{% endifequal %}", {"a": "no"}, "no"),
|
|
||||||
'ifequal07': ('{% ifequal a "test" %}yes{% else %}no{% endifequal %}', {"a": "test"}, "yes"),
|
|
||||||
'ifequal08': ('{% ifequal a "test" %}yes{% else %}no{% endifequal %}', {"a": "no"}, "no"),
|
|
||||||
'ifequal09': ('{% ifequal a "test" %}yes{% else %}no{% endifequal %}', {}, "no"),
|
|
||||||
'ifequal10': ('{% ifequal a b %}yes{% else %}no{% endifequal %}', {}, "yes"),
|
|
||||||
|
|
||||||
# SMART SPLITTING
|
|
||||||
'ifequal-split01': ('{% ifequal a "test man" %}yes{% else %}no{% endifequal %}', {}, "no"),
|
|
||||||
'ifequal-split02': ('{% ifequal a "test man" %}yes{% else %}no{% endifequal %}', {'a': 'foo'}, "no"),
|
|
||||||
'ifequal-split03': ('{% ifequal a "test man" %}yes{% else %}no{% endifequal %}', {'a': 'test man'}, "yes"),
|
|
||||||
'ifequal-split04': ("{% ifequal a 'test man' %}yes{% else %}no{% endifequal %}", {'a': 'test man'}, "yes"),
|
|
||||||
'ifequal-split05': ("{% ifequal a 'i \"love\" you' %}yes{% else %}no{% endifequal %}", {'a': ''}, "no"),
|
|
||||||
'ifequal-split06': ("{% ifequal a 'i \"love\" you' %}yes{% else %}no{% endifequal %}", {'a': 'i "love" you'}, "yes"),
|
|
||||||
'ifequal-split07': ("{% ifequal a 'i \"love\" you' %}yes{% else %}no{% endifequal %}", {'a': 'i love you'}, "no"),
|
|
||||||
'ifequal-split08': (r"{% ifequal a 'I\'m happy' %}yes{% else %}no{% endifequal %}", {'a': "I'm happy"}, "yes"),
|
|
||||||
'ifequal-split09': (r"{% ifequal a 'slash\man' %}yes{% else %}no{% endifequal %}", {'a': r"slash\man"}, "yes"),
|
|
||||||
'ifequal-split10': (r"{% ifequal a 'slash\man' %}yes{% else %}no{% endifequal %}", {'a': r"slashman"}, "no"),
|
|
||||||
|
|
||||||
### IFNOTEQUAL TAG ########################################################
|
|
||||||
'ifnotequal01': ("{% ifnotequal a b %}yes{% endifnotequal %}", {"a": 1, "b": 2}, "yes"),
|
|
||||||
'ifnotequal02': ("{% ifnotequal a b %}yes{% endifnotequal %}", {"a": 1, "b": 1}, ""),
|
|
||||||
'ifnotequal03': ("{% ifnotequal a b %}yes{% else %}no{% endifnotequal %}", {"a": 1, "b": 2}, "yes"),
|
|
||||||
'ifnotequal04': ("{% ifnotequal a b %}yes{% else %}no{% endifnotequal %}", {"a": 1, "b": 1}, "no"),
|
|
||||||
|
|
||||||
### INCLUDE TAG ###########################################################
|
|
||||||
'include01': ('{% include "basic-syntax01" %}', {}, "something cool"),
|
|
||||||
'include02': ('{% include "basic-syntax02" %}', {'headline': 'Included'}, "Included"),
|
|
||||||
'include03': ('{% include template_name %}', {'template_name': 'basic-syntax02', 'headline': 'Included'}, "Included"),
|
|
||||||
'include04': ('a{% include "nonexistent" %}b', {}, "ab"),
|
|
||||||
|
|
||||||
### INHERITANCE ###########################################################
|
|
||||||
|
|
||||||
# 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_'),
|
|
||||||
|
|
||||||
# Two-level inheritance with {{ block.super }}
|
|
||||||
'inheritance20': ("{% extends 'inheritance01' %}{% block first %}{{ block.super }}a{% endblock %}", {}, '1_a3_'),
|
|
||||||
|
|
||||||
# Three-level inheritance with {{ block.super }} from parent
|
|
||||||
'inheritance21': ("{% extends 'inheritance02' %}{% block first %}{{ block.super }}a{% endblock %}", {}, '12a34'),
|
|
||||||
|
|
||||||
# Three-level inheritance with {{ block.super }} from grandparent
|
|
||||||
'inheritance22': ("{% extends 'inheritance04' %}{% block first %}{{ block.super }}a{% endblock %}", {}, '1_a3_'),
|
|
||||||
|
|
||||||
# Three-level inheritance with {{ block.super }} from parent and grandparent
|
|
||||||
'inheritance23': ("{% extends 'inheritance20' %}{% block first %}{{ block.super }}b{% endblock %}", {}, '1_ab3_'),
|
|
||||||
|
|
||||||
# Inheritance from local context without use of template loader
|
|
||||||
'inheritance24': ("{% extends context_template %}{% block first %}2{% endblock %}{% block second %}4{% endblock %}", {'context_template': template.Template("1{% block first %}_{% endblock %}3{% block second %}_{% endblock %}")}, '1234'),
|
|
||||||
|
|
||||||
# Inheritance from local context with variable parent template
|
|
||||||
'inheritance25': ("{% extends context_template.1 %}{% block first %}2{% endblock %}{% block second %}4{% endblock %}", {'context_template': [template.Template("Wrong"), template.Template("1{% block first %}_{% endblock %}3{% block second %}_{% endblock %}")]}, '1234'),
|
|
||||||
|
|
||||||
### I18N ##################################################################
|
|
||||||
|
|
||||||
# {% spaceless %} tag
|
|
||||||
'spaceless01': ("{% spaceless %} <b> <i> text </i> </b> {% endspaceless %}", {}, "<b> <i> text </i> </b>"),
|
|
||||||
'spaceless02': ("{% spaceless %} <b> \n <i> text </i> \n </b> {% endspaceless %}", {}, "<b> <i> text </i> </b>"),
|
|
||||||
'spaceless03': ("{% spaceless %}<b><i>text</i></b>{% endspaceless %}", {}, "<b><i>text</i></b>"),
|
|
||||||
|
|
||||||
# simple translation of a string delimited by '
|
|
||||||
'i18n01': ("{% load i18n %}{% trans 'xxxyyyxxx' %}", {}, "xxxyyyxxx"),
|
|
||||||
|
|
||||||
# simple translation of a string delimited by "
|
|
||||||
'i18n02': ('{% load i18n %}{% trans "xxxyyyxxx" %}', {}, "xxxyyyxxx"),
|
|
||||||
|
|
||||||
# simple translation of a variable
|
|
||||||
'i18n03': ('{% load i18n %}{% blocktrans %}{{ anton }}{% endblocktrans %}', {'anton': 'xxxyyyxxx'}, "xxxyyyxxx"),
|
|
||||||
|
|
||||||
# simple translation of a variable and filter
|
|
||||||
'i18n04': ('{% load i18n %}{% blocktrans with anton|lower as berta %}{{ berta }}{% endblocktrans %}', {'anton': 'XXXYYYXXX'}, "xxxyyyxxx"),
|
|
||||||
|
|
||||||
# simple translation of a string with interpolation
|
|
||||||
'i18n05': ('{% load i18n %}{% blocktrans %}xxx{{ anton }}xxx{% endblocktrans %}', {'anton': 'yyy'}, "xxxyyyxxx"),
|
|
||||||
|
|
||||||
# simple translation of a string to german
|
|
||||||
'i18n06': ('{% load i18n %}{% trans "Page not found" %}', {'LANGUAGE_CODE': 'de'}, "Seite nicht gefunden"),
|
|
||||||
|
|
||||||
# translation of singular form
|
|
||||||
'i18n07': ('{% load i18n %}{% blocktrans count number as counter %}singular{% plural %}plural{% endblocktrans %}', {'number': 1}, "singular"),
|
|
||||||
|
|
||||||
# translation of plural form
|
|
||||||
'i18n08': ('{% load i18n %}{% blocktrans count number as counter %}singular{% plural %}plural{% endblocktrans %}', {'number': 2}, "plural"),
|
|
||||||
|
|
||||||
# simple non-translation (only marking) of a string to german
|
|
||||||
'i18n09': ('{% load i18n %}{% trans "Page not found" noop %}', {'LANGUAGE_CODE': 'de'}, "Page not found"),
|
|
||||||
|
|
||||||
# translation of a variable with a translated filter
|
|
||||||
'i18n10': ('{{ bool|yesno:_("ja,nein") }}', {'bool': True}, 'ja'),
|
|
||||||
|
|
||||||
# translation of a variable with a non-translated filter
|
|
||||||
'i18n11': ('{{ bool|yesno:"ja,nein" }}', {'bool': True}, 'ja'),
|
|
||||||
|
|
||||||
# usage of the get_available_languages tag
|
|
||||||
'i18n12': ('{% load i18n %}{% get_available_languages as langs %}{% for lang in langs %}{% ifequal lang.0 "de" %}{{ lang.0 }}{% endifequal %}{% endfor %}', {}, 'de'),
|
|
||||||
|
|
||||||
# translation of a constant string
|
|
||||||
'i18n13': ('{{ _("Page not found") }}', {'LANGUAGE_CODE': 'de'}, 'Seite nicht gefunden'),
|
|
||||||
|
|
||||||
### MULTILINE #############################################################
|
|
||||||
|
|
||||||
'multiline01': ("""
|
|
||||||
Hello,
|
|
||||||
boys.
|
|
||||||
How
|
|
||||||
are
|
|
||||||
you
|
|
||||||
gentlemen.
|
|
||||||
""",
|
|
||||||
{},
|
|
||||||
"""
|
|
||||||
Hello,
|
|
||||||
boys.
|
|
||||||
How
|
|
||||||
are
|
|
||||||
you
|
|
||||||
gentlemen.
|
|
||||||
"""),
|
|
||||||
|
|
||||||
### REGROUP TAG ###########################################################
|
|
||||||
'regroup01': ('{% regroup data by bar as grouped %}' + \
|
|
||||||
'{% for group in grouped %}' + \
|
|
||||||
'{{ group.grouper }}:' + \
|
|
||||||
'{% for item in group.list %}' + \
|
|
||||||
'{{ item.foo }}' + \
|
|
||||||
'{% endfor %},' + \
|
|
||||||
'{% endfor %}',
|
|
||||||
{'data': [ {'foo':'c', 'bar':1},
|
|
||||||
{'foo':'d', 'bar':1},
|
|
||||||
{'foo':'a', 'bar':2},
|
|
||||||
{'foo':'b', 'bar':2},
|
|
||||||
{'foo':'x', 'bar':3} ]},
|
|
||||||
'1:cd,2:ab,3:x,'),
|
|
||||||
|
|
||||||
# Test for silent failure when target variable isn't found
|
|
||||||
'regroup02': ('{% regroup data by bar as grouped %}' + \
|
|
||||||
'{% for group in grouped %}' + \
|
|
||||||
'{{ group.grouper }}:' + \
|
|
||||||
'{% for item in group.list %}' + \
|
|
||||||
'{{ item.foo }}' + \
|
|
||||||
'{% endfor %},' + \
|
|
||||||
'{% endfor %}',
|
|
||||||
{}, 'INVALID:INVALIDINVALIDINVALIDINVALIDINVALIDINVALIDINVALID,'),
|
|
||||||
|
|
||||||
### TEMPLATETAG TAG #######################################################
|
|
||||||
'templatetag01': ('{% templatetag openblock %}', {}, '{%'),
|
|
||||||
'templatetag02': ('{% templatetag closeblock %}', {}, '%}'),
|
|
||||||
'templatetag03': ('{% templatetag openvariable %}', {}, '{{'),
|
|
||||||
'templatetag04': ('{% templatetag closevariable %}', {}, '}}'),
|
|
||||||
'templatetag05': ('{% templatetag %}', {}, template.TemplateSyntaxError),
|
|
||||||
'templatetag06': ('{% templatetag foo %}', {}, template.TemplateSyntaxError),
|
|
||||||
'templatetag07': ('{% templatetag openbrace %}', {}, '{'),
|
|
||||||
'templatetag08': ('{% templatetag closebrace %}', {}, '}'),
|
|
||||||
'templatetag09': ('{% templatetag openbrace %}{% templatetag openbrace %}', {}, '{{'),
|
|
||||||
'templatetag10': ('{% templatetag closebrace %}{% templatetag closebrace %}', {}, '}}'),
|
|
||||||
|
|
||||||
### WIDTHRATIO TAG ########################################################
|
|
||||||
'widthratio01': ('{% widthratio a b 0 %}', {'a':50,'b':100}, '0'),
|
|
||||||
'widthratio02': ('{% widthratio a b 100 %}', {'a':0,'b':0}, ''),
|
|
||||||
'widthratio03': ('{% widthratio a b 100 %}', {'a':0,'b':100}, '0'),
|
|
||||||
'widthratio04': ('{% widthratio a b 100 %}', {'a':50,'b':100}, '50'),
|
|
||||||
'widthratio05': ('{% widthratio a b 100 %}', {'a':100,'b':100}, '100'),
|
|
||||||
|
|
||||||
# 62.5 should round to 63
|
|
||||||
'widthratio06': ('{% widthratio a b 100 %}', {'a':50,'b':80}, '63'),
|
|
||||||
|
|
||||||
# 71.4 should round to 71
|
|
||||||
'widthratio07': ('{% widthratio a b 100 %}', {'a':50,'b':70}, '71'),
|
|
||||||
|
|
||||||
# Raise exception if we don't have 3 args, last one an integer
|
|
||||||
'widthratio08': ('{% widthratio %}', {}, template.TemplateSyntaxError),
|
|
||||||
'widthratio09': ('{% widthratio a b %}', {'a':50,'b':100}, template.TemplateSyntaxError),
|
|
||||||
'widthratio10': ('{% widthratio a b 100.0 %}', {'a':50,'b':100}, template.TemplateSyntaxError),
|
|
||||||
|
|
||||||
### NOW TAG ########################################################
|
|
||||||
# Simple case
|
|
||||||
'now01' : ('{% now "j n Y"%}', {}, str(datetime.now().day) + ' ' + str(datetime.now().month) + ' ' + str(datetime.now().year)),
|
|
||||||
|
|
||||||
# Check parsing of escaped and special characters
|
|
||||||
'now02' : ('{% now "j "n" Y"%}', {}, template.TemplateSyntaxError),
|
|
||||||
# 'now03' : ('{% now "j \"n\" Y"%}', {}, str(datetime.now().day) + '"' + str(datetime.now().month) + '"' + str(datetime.now().year)),
|
|
||||||
# 'now04' : ('{% now "j \nn\n Y"%}', {}, str(datetime.now().day) + '\n' + str(datetime.now().month) + '\n' + str(datetime.now().year))
|
|
||||||
|
|
||||||
### TIMESINCE TAG ##################################################
|
|
||||||
# Default compare with datetime.now()
|
|
||||||
'timesince01' : ('{{ a|timesince }}', {'a':datetime.now() + timedelta(minutes=-1, seconds = -10)}, '1 minute'),
|
|
||||||
'timesince02' : ('{{ a|timesince }}', {'a':(datetime.now() - timedelta(days=1, minutes = 1))}, '1 day'),
|
|
||||||
'timesince03' : ('{{ a|timesince }}', {'a':(datetime.now() -
|
|
||||||
timedelta(hours=1, minutes=25, seconds = 10))}, '1 hour, 25 minutes'),
|
|
||||||
|
|
||||||
# Compare to a given parameter
|
|
||||||
'timesince04' : ('{{ a|timesince:b }}', {'a':NOW + timedelta(days=2), 'b':NOW + timedelta(days=1)}, '1 day'),
|
|
||||||
'timesince05' : ('{{ a|timesince:b }}', {'a':NOW + timedelta(days=2, minutes=1), 'b':NOW + timedelta(days=2)}, '1 minute'),
|
|
||||||
|
|
||||||
# Check that timezone is respected
|
|
||||||
'timesince06' : ('{{ a|timesince:b }}', {'a':NOW_tz + timedelta(hours=8), 'b':NOW_tz}, '8 hours'),
|
|
||||||
|
|
||||||
### TIMEUNTIL TAG ##################################################
|
|
||||||
# Default compare with datetime.now()
|
|
||||||
'timeuntil01' : ('{{ a|timeuntil }}', {'a':datetime.now() + timedelta(minutes=2, seconds = 10)}, '2 minutes'),
|
|
||||||
'timeuntil02' : ('{{ a|timeuntil }}', {'a':(datetime.now() + timedelta(days=1, seconds = 10))}, '1 day'),
|
|
||||||
'timeuntil03' : ('{{ a|timeuntil }}', {'a':(datetime.now() + timedelta(hours=8, minutes=10, seconds = 10))}, '8 hours, 10 minutes'),
|
|
||||||
|
|
||||||
# Compare to a given parameter
|
|
||||||
'timeuntil04' : ('{{ a|timeuntil:b }}', {'a':NOW - timedelta(days=1), 'b':NOW - timedelta(days=2)}, '1 day'),
|
|
||||||
'timeuntil05' : ('{{ a|timeuntil:b }}', {'a':NOW - timedelta(days=2), 'b':NOW - timedelta(days=2, minutes=1)}, '1 minute'),
|
|
||||||
}
|
|
||||||
|
|
||||||
def test_template_loader(template_name, template_dirs=None):
|
|
||||||
"A custom template loader that loads the unit-test templates."
|
|
||||||
try:
|
|
||||||
return (TEMPLATE_TESTS[template_name][0] , "test:%s" % template_name)
|
|
||||||
except KeyError:
|
|
||||||
raise template.TemplateDoesNotExist, template_name
|
|
||||||
|
|
||||||
def run_tests(verbosity=0, standalone=False):
|
|
||||||
# Register our custom template loader.
|
|
||||||
old_template_loaders = loader.template_source_loaders
|
|
||||||
loader.template_source_loaders = [test_template_loader]
|
|
||||||
|
|
||||||
failed_tests = []
|
|
||||||
tests = TEMPLATE_TESTS.items()
|
|
||||||
tests.sort()
|
|
||||||
|
|
||||||
# Turn TEMPLATE_DEBUG off, because tests assume that.
|
|
||||||
old_td, settings.TEMPLATE_DEBUG = settings.TEMPLATE_DEBUG, False
|
|
||||||
# Set TEMPLATE_STRING_IF_INVALID to a known string
|
|
||||||
old_invalid, settings.TEMPLATE_STRING_IF_INVALID = settings.TEMPLATE_STRING_IF_INVALID, 'INVALID'
|
|
||||||
|
|
||||||
for name, vals in tests:
|
|
||||||
install()
|
|
||||||
if 'LANGUAGE_CODE' in vals[1]:
|
|
||||||
activate(vals[1]['LANGUAGE_CODE'])
|
|
||||||
else:
|
|
||||||
activate('en-us')
|
|
||||||
try:
|
|
||||||
output = 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:
|
|
||||||
traceback.print_exc()
|
|
||||||
print "Template test: %s -- FAILED. Got %s, exception: %s" % (name, e.__class__, e)
|
|
||||||
failed_tests.append(name)
|
|
||||||
continue
|
|
||||||
if 'LANGUAGE_CODE' in vals[1]:
|
|
||||||
deactivate()
|
|
||||||
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)
|
|
||||||
loader.template_source_loaders = old_template_loaders
|
|
||||||
deactivate()
|
|
||||||
settings.TEMPLATE_DEBUG = old_td
|
|
||||||
settings.TEMPLATE_STRING_IF_INVALID = old_invalid
|
|
||||||
|
|
||||||
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)
|
|
|
@ -0,0 +1,71 @@
|
||||||
|
# Unit tests for cache framework
|
||||||
|
# Uses whatever cache backend is set in the test settings file.
|
||||||
|
|
||||||
|
from django.core.cache import cache
|
||||||
|
import time, unittest
|
||||||
|
|
||||||
|
# functions/classes for complex data type tests
|
||||||
|
def f():
|
||||||
|
return 42
|
||||||
|
class C:
|
||||||
|
def m(n):
|
||||||
|
return 24
|
||||||
|
|
||||||
|
class Cache(unittest.TestCase):
|
||||||
|
def test_simple(self):
|
||||||
|
# simple set/get
|
||||||
|
cache.set("key", "value")
|
||||||
|
self.assertEqual(cache.get("key"), "value")
|
||||||
|
|
||||||
|
def test_non_existent(self):
|
||||||
|
# get with non-existent keys
|
||||||
|
self.assertEqual(cache.get("does not exist"), None)
|
||||||
|
self.assertEqual(cache.get("does not exist", "bang!"), "bang!")
|
||||||
|
|
||||||
|
def test_get_many(self):
|
||||||
|
# get_many
|
||||||
|
cache.set('a', 'a')
|
||||||
|
cache.set('b', 'b')
|
||||||
|
cache.set('c', 'c')
|
||||||
|
cache.set('d', 'd')
|
||||||
|
self.assertEqual(cache.get_many(['a', 'c', 'd']), {'a' : 'a', 'c' : 'c', 'd' : 'd'})
|
||||||
|
self.assertEqual(cache.get_many(['a', 'b', 'e']), {'a' : 'a', 'b' : 'b'})
|
||||||
|
|
||||||
|
def test_delete(self):
|
||||||
|
# delete
|
||||||
|
cache.set("key1", "spam")
|
||||||
|
cache.set("key2", "eggs")
|
||||||
|
self.assertEqual(cache.get("key1"), "spam")
|
||||||
|
cache.delete("key1")
|
||||||
|
self.assertEqual(cache.get("key1"), None)
|
||||||
|
self.assertEqual(cache.get("key2"), "eggs")
|
||||||
|
|
||||||
|
def test_has_key(self):
|
||||||
|
# has_key
|
||||||
|
cache.set("hello", "goodbye")
|
||||||
|
self.assertEqual(cache.has_key("hello"), True)
|
||||||
|
self.assertEqual(cache.has_key("goodbye"), False)
|
||||||
|
|
||||||
|
def test_data_types(self):
|
||||||
|
# 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)
|
||||||
|
self.assertEqual(cache.get(key), value)
|
||||||
|
|
||||||
|
def test_expiration(self):
|
||||||
|
# expiration
|
||||||
|
cache.set('expire', 'very quickly', 1)
|
||||||
|
time.sleep(2)
|
||||||
|
self.assertEqual(cache.get("expire"), None)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
unittest.main()
|
|
@ -1,7 +1,7 @@
|
||||||
# Unit tests for typecast functions in django.db.backends.util
|
# Unit tests for typecast functions in django.db.backends.util
|
||||||
|
|
||||||
from django.db.backends import util as typecasts
|
from django.db.backends import util as typecasts
|
||||||
import datetime
|
import datetime, unittest
|
||||||
|
|
||||||
TEST_CASES = {
|
TEST_CASES = {
|
||||||
'typecast_date': (
|
'typecast_date': (
|
||||||
|
@ -45,7 +45,12 @@ TEST_CASES = {
|
||||||
),
|
),
|
||||||
}
|
}
|
||||||
|
|
||||||
for k, v in TEST_CASES.items():
|
class DBTypeCasts(unittest.TestCase):
|
||||||
for inpt, expected in v:
|
def test_typeCasts(self):
|
||||||
got = getattr(typecasts, k)(inpt)
|
for k, v in TEST_CASES.items():
|
||||||
assert got == expected, "In %s: %r doesn't match %r. Got %r instead." % (k, inpt, expected, got)
|
for inpt, expected in v:
|
||||||
|
got = getattr(typecasts, k)(inpt)
|
||||||
|
assert got == expected, "In %s: %r doesn't match %r. Got %r instead." % (k, inpt, expected, got)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
unittest.main()
|
|
@ -7,7 +7,7 @@ from django.db import models
|
||||||
class Simple(models.Model):
|
class Simple(models.Model):
|
||||||
name = models.CharField(maxlength = 50)
|
name = models.CharField(maxlength = 50)
|
||||||
|
|
||||||
API_TESTS = ""
|
__test__ = {'API_TESTS':""}
|
||||||
|
|
||||||
# NOTE: The format of the included SQL file for this test suite is important.
|
# NOTE: The format of the included SQL file for this test suite is important.
|
||||||
# It must end with a trailing newline in order to test the fix for #2161.
|
# It must end with a trailing newline in order to test the fix for #2161.
|
||||||
|
|
|
@ -10,4 +10,4 @@ class Second(models.Model):
|
||||||
# created (the field names being lower-cased versions of their opposite
|
# created (the field names being lower-cased versions of their opposite
|
||||||
# classes is important here).
|
# classes is important here).
|
||||||
|
|
||||||
API_TESTS = ""
|
__test__ = {'API_TESTS':""}
|
||||||
|
|
|
@ -0,0 +1,69 @@
|
||||||
|
# Quick tests for the markup templatetags (django.contrib.markup)
|
||||||
|
|
||||||
|
from django.template import Template, Context, add_to_builtins
|
||||||
|
import re
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
add_to_builtins('django.contrib.markup.templatetags.markup')
|
||||||
|
|
||||||
|
class Templates(unittest.TestCase):
|
||||||
|
def test_textile(self):
|
||||||
|
try:
|
||||||
|
import textile
|
||||||
|
except ImportError:
|
||||||
|
textile = None
|
||||||
|
|
||||||
|
textile_content = """Paragraph 1
|
||||||
|
|
||||||
|
Paragraph 2 with "quotes" and @code@"""
|
||||||
|
|
||||||
|
t = Template("{{ textile_content|textile }}")
|
||||||
|
rendered = t.render(Context(locals())).strip()
|
||||||
|
if textile:
|
||||||
|
self.assertEqual(rendered, """<p>Paragraph 1</p>
|
||||||
|
|
||||||
|
<p>Paragraph 2 with “quotes” and <code>code</code></p>""")
|
||||||
|
else:
|
||||||
|
self.assertEqual(rendered, textile_content)
|
||||||
|
|
||||||
|
def test_markdown(self):
|
||||||
|
try:
|
||||||
|
import markdown
|
||||||
|
except ImportError:
|
||||||
|
markdown = None
|
||||||
|
|
||||||
|
markdown_content = """Paragraph 1
|
||||||
|
|
||||||
|
## An h2"""
|
||||||
|
|
||||||
|
t = Template("{{ markdown_content|markdown }}")
|
||||||
|
rendered = t.render(Context(locals())).strip()
|
||||||
|
if markdown:
|
||||||
|
pattern = re.compile("""<p>Paragraph 1\s*</p>\s*<h2>\s*An h2</h2>""")
|
||||||
|
self.assert_(pattern.match(rendered))
|
||||||
|
else:
|
||||||
|
self.assertEqual(rendered, markdown_content)
|
||||||
|
|
||||||
|
def test_docutils(self):
|
||||||
|
try:
|
||||||
|
import docutils
|
||||||
|
except ImportError:
|
||||||
|
docutils = None
|
||||||
|
|
||||||
|
rest_content = """Paragraph 1
|
||||||
|
|
||||||
|
Paragraph 2 with a link_
|
||||||
|
|
||||||
|
.. _link: http://www.example.com/"""
|
||||||
|
|
||||||
|
t = Template("{{ rest_content|restructuredtext }}")
|
||||||
|
rendered = t.render(Context(locals())).strip()
|
||||||
|
if docutils:
|
||||||
|
self.assertEqual(rendered, """<p>Paragraph 1</p>
|
||||||
|
<p>Paragraph 2 with a <a class="reference" href="http://www.example.com/">link</a></p>""")
|
||||||
|
else:
|
||||||
|
self.assertEqual(rendered, rest_content)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
unittest.main()
|
|
@ -22,7 +22,7 @@ class Favorites(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return "Favorites for %s" % self.name
|
return "Favorites for %s" % self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Regression test for #1064 and #1506: Check that we create models via the m2m
|
# Regression test for #1064 and #1506: Check that we create models via the m2m
|
||||||
# relation if the remote model has a OneToOneField.
|
# relation if the remote model has a OneToOneField.
|
||||||
>>> p1 = Place(name='Demon Dogs', address='944 W. Fullerton')
|
>>> p1 = Place(name='Demon Dogs', address='944 W. Fullerton')
|
||||||
|
@ -34,4 +34,4 @@ API_TESTS = """
|
||||||
>>> f.restaurants = [r]
|
>>> f.restaurants = [r]
|
||||||
>>> f.restaurants.all()
|
>>> f.restaurants.all()
|
||||||
[<Restaurant: Demon Dogs the restaurant>]
|
[<Restaurant: Demon Dogs the restaurant>]
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -34,7 +34,7 @@ class Base(models.Model):
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return "Base %s" % self.name
|
return "Base %s" % self.name
|
||||||
|
|
||||||
API_TESTS = """
|
__test__ = {'API_TESTS':"""
|
||||||
# Regression test for #1661 and #1662: Check that string form referencing of models works,
|
# Regression test for #1661 and #1662: Check that string form referencing of models works,
|
||||||
# both as pre and post reference, on all RelatedField types.
|
# both as pre and post reference, on all RelatedField types.
|
||||||
|
|
||||||
|
@ -66,4 +66,4 @@ API_TESTS = """
|
||||||
|
|
||||||
>>> child1.parent
|
>>> child1.parent
|
||||||
<Base: Base Base1>
|
<Base: Base Base1>
|
||||||
"""
|
"""}
|
||||||
|
|
|
@ -0,0 +1,621 @@
|
||||||
|
from django.conf import settings
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
# When running this file in isolation, we need to set up the configuration
|
||||||
|
# before importing 'template'.
|
||||||
|
settings.configure()
|
||||||
|
|
||||||
|
from django import template
|
||||||
|
from django.template import loader
|
||||||
|
from django.utils.translation import activate, deactivate, install
|
||||||
|
from django.utils.tzinfo import LocalTimezone
|
||||||
|
from datetime import datetime, timedelta
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
#################################
|
||||||
|
# Custom template tag for tests #
|
||||||
|
#################################
|
||||||
|
|
||||||
|
register = template.Library()
|
||||||
|
|
||||||
|
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:])
|
||||||
|
|
||||||
|
register.tag("echo", do_echo)
|
||||||
|
|
||||||
|
template.libraries['django.templatetags.testtags'] = register
|
||||||
|
|
||||||
|
#####################################
|
||||||
|
# Helper objects for template tests #
|
||||||
|
#####################################
|
||||||
|
|
||||||
|
class SomeException(Exception):
|
||||||
|
silent_variable_failure = True
|
||||||
|
|
||||||
|
class SomeOtherException(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
class SomeClass:
|
||||||
|
def __init__(self):
|
||||||
|
self.otherclass = OtherClass()
|
||||||
|
|
||||||
|
def method(self):
|
||||||
|
return "SomeClass.method"
|
||||||
|
|
||||||
|
def method2(self, o):
|
||||||
|
return o
|
||||||
|
|
||||||
|
def method3(self):
|
||||||
|
raise SomeException
|
||||||
|
|
||||||
|
def method4(self):
|
||||||
|
raise SomeOtherException
|
||||||
|
|
||||||
|
class OtherClass:
|
||||||
|
def method(self):
|
||||||
|
return "OtherClass.method"
|
||||||
|
|
||||||
|
class Templates(unittest.TestCase):
|
||||||
|
def test_templates(self):
|
||||||
|
# NOW and NOW_tz are used by timesince tag tests.
|
||||||
|
NOW = datetime.now()
|
||||||
|
NOW_tz = datetime.now(LocalTimezone(datetime.now()))
|
||||||
|
|
||||||
|
# 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", {}, "asINVALIDdf"),
|
||||||
|
|
||||||
|
# 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()}, "INVALID"),
|
||||||
|
|
||||||
|
# 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"}}, "INVALID"),
|
||||||
|
|
||||||
|
# Fail silently when accessing a non-simple method
|
||||||
|
'basic-syntax20': ("{{ var.method2 }}", {"var": SomeClass()}, "INVALID"),
|
||||||
|
|
||||||
|
# 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),
|
||||||
|
|
||||||
|
# Chained filters, with an argument to the first one
|
||||||
|
'basic-syntax29': ('{{ var|removetags:"b i"|upper|lower }}', {"var": "<b><i>Yes</i></b>"}, "yes"),
|
||||||
|
|
||||||
|
# Escaped string as argument
|
||||||
|
'basic-syntax30': (r'{{ var|default_if_none:" endquote\" hah" }}', {"var": None}, ' endquote" hah'),
|
||||||
|
|
||||||
|
# Variable as argument
|
||||||
|
'basic-syntax31': (r'{{ var|default_if_none:var2 }}', {"var": None, "var2": "happy"}, 'happy'),
|
||||||
|
|
||||||
|
# Default argument testing
|
||||||
|
'basic-syntax32': (r'{{ var|yesno:"yup,nup,mup" }} {{ var|yesno }}', {"var": True}, 'yup yes'),
|
||||||
|
|
||||||
|
# Fail silently for methods that raise an exception with a "silent_variable_failure" attribute
|
||||||
|
'basic-syntax33': (r'1{{ var.method3 }}2', {"var": SomeClass()}, "1INVALID2"),
|
||||||
|
|
||||||
|
# In methods that raise an exception without a "silent_variable_attribute" set to True,
|
||||||
|
# the exception propogates
|
||||||
|
'basic-syntax34': (r'1{{ var.method4 }}2', {"var": SomeClass()}, SomeOtherException),
|
||||||
|
|
||||||
|
# Escaped backslash in argument
|
||||||
|
'basic-syntax35': (r'{{ var|default_if_none:"foo\bar" }}', {"var": None}, r'foo\bar'),
|
||||||
|
|
||||||
|
# Escaped backslash using known escape char
|
||||||
|
'basic-syntax35': (r'{{ var|default_if_none:"foo\now" }}', {"var": None}, r'foo\now'),
|
||||||
|
|
||||||
|
### COMMENT TAG ###########################################################
|
||||||
|
'comment-tag01': ("{% comment %}this is hidden{% endcomment %}hello", {}, "hello"),
|
||||||
|
'comment-tag02': ("{% comment %}this is hidden{% endcomment %}hello{% comment %}foo{% endcomment %}", {}, "hello"),
|
||||||
|
|
||||||
|
# Comment tag can contain invalid stuff.
|
||||||
|
'comment-tag03': ("foo{% comment %} {% if %} {% endcomment %}", {}, "foo"),
|
||||||
|
'comment-tag04': ("foo{% comment %} {% endblock %} {% endcomment %}", {}, "foo"),
|
||||||
|
'comment-tag05': ("foo{% comment %} {% somerandomtag %} {% endcomment %}", {}, "foo"),
|
||||||
|
|
||||||
|
### CYCLE TAG #############################################################
|
||||||
|
'cycle01': ('{% cycle a %}', {}, template.TemplateSyntaxError),
|
||||||
|
'cycle02': ('{% cycle a,b,c as abc %}{% cycle abc %}', {}, 'ab'),
|
||||||
|
'cycle03': ('{% cycle a,b,c as abc %}{% cycle abc %}{% cycle abc %}', {}, 'abc'),
|
||||||
|
'cycle04': ('{% cycle a,b,c as abc %}{% cycle abc %}{% cycle abc %}{% cycle abc %}', {}, 'abca'),
|
||||||
|
'cycle05': ('{% cycle %}', {}, template.TemplateSyntaxError),
|
||||||
|
'cycle06': ('{% cycle a %}', {}, template.TemplateSyntaxError),
|
||||||
|
'cycle07': ('{% cycle a,b,c as foo %}{% cycle bar %}', {}, template.TemplateSyntaxError),
|
||||||
|
|
||||||
|
### EXCEPTIONS ############################################################
|
||||||
|
|
||||||
|
# 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),
|
||||||
|
|
||||||
|
### FILTER TAG ############################################################
|
||||||
|
'filter01': ('{% filter upper %}{% endfilter %}', {}, ''),
|
||||||
|
'filter02': ('{% filter upper %}django{% endfilter %}', {}, 'DJANGO'),
|
||||||
|
'filter03': ('{% filter upper|lower %}django{% endfilter %}', {}, 'django'),
|
||||||
|
|
||||||
|
### FIRSTOF TAG ###########################################################
|
||||||
|
'firstof01': ('{% firstof a b c %}', {'a':0,'b':0,'c':0}, ''),
|
||||||
|
'firstof02': ('{% firstof a b c %}', {'a':1,'b':0,'c':0}, '1'),
|
||||||
|
'firstof03': ('{% firstof a b c %}', {'a':0,'b':2,'c':0}, '2'),
|
||||||
|
'firstof04': ('{% firstof a b c %}', {'a':0,'b':0,'c':3}, '3'),
|
||||||
|
'firstof05': ('{% firstof a b c %}', {'a':1,'b':2,'c':3}, '1'),
|
||||||
|
'firstof06': ('{% firstof %}', {}, template.TemplateSyntaxError),
|
||||||
|
|
||||||
|
### FOR TAG ###############################################################
|
||||||
|
'for-tag01': ("{% for val in values %}{{ val }}{% endfor %}", {"values": [1, 2, 3]}, "123"),
|
||||||
|
'for-tag02': ("{% for val in values reversed %}{{ val }}{% endfor %}", {"values": [1, 2, 3]}, "321"),
|
||||||
|
'for-tag-vars01': ("{% for val in values %}{{ forloop.counter }}{% endfor %}", {"values": [6, 6, 6]}, "123"),
|
||||||
|
'for-tag-vars02': ("{% for val in values %}{{ forloop.counter0 }}{% endfor %}", {"values": [6, 6, 6]}, "012"),
|
||||||
|
'for-tag-vars03': ("{% for val in values %}{{ forloop.revcounter }}{% endfor %}", {"values": [6, 6, 6]}, "321"),
|
||||||
|
'for-tag-vars04': ("{% for val in values %}{{ forloop.revcounter0 }}{% endfor %}", {"values": [6, 6, 6]}, "210"),
|
||||||
|
|
||||||
|
### IF TAG ################################################################
|
||||||
|
'if-tag01': ("{% if foo %}yes{% else %}no{% endif %}", {"foo": True}, "yes"),
|
||||||
|
'if-tag02': ("{% if foo %}yes{% else %}no{% endif %}", {"foo": False}, "no"),
|
||||||
|
'if-tag03': ("{% if foo %}yes{% else %}no{% endif %}", {}, "no"),
|
||||||
|
|
||||||
|
# AND
|
||||||
|
'if-tag-and01': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'yes'),
|
||||||
|
'if-tag-and02': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'no'),
|
||||||
|
'if-tag-and03': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'no'),
|
||||||
|
'if-tag-and04': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'no'),
|
||||||
|
'if-tag-and05': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'foo': False}, 'no'),
|
||||||
|
'if-tag-and06': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'bar': False}, 'no'),
|
||||||
|
'if-tag-and07': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'foo': True}, 'no'),
|
||||||
|
'if-tag-and08': ("{% if foo and bar %}yes{% else %}no{% endif %}", {'bar': True}, 'no'),
|
||||||
|
|
||||||
|
# OR
|
||||||
|
'if-tag-or01': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'yes'),
|
||||||
|
'if-tag-or02': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'yes'),
|
||||||
|
'if-tag-or03': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'yes'),
|
||||||
|
'if-tag-or04': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'no'),
|
||||||
|
'if-tag-or05': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'foo': False}, 'no'),
|
||||||
|
'if-tag-or06': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'bar': False}, 'no'),
|
||||||
|
'if-tag-or07': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'foo': True}, 'yes'),
|
||||||
|
'if-tag-or08': ("{% if foo or bar %}yes{% else %}no{% endif %}", {'bar': True}, 'yes'),
|
||||||
|
|
||||||
|
# TODO: multiple ORs
|
||||||
|
|
||||||
|
# NOT
|
||||||
|
'if-tag-not01': ("{% if not foo %}no{% else %}yes{% endif %}", {'foo': True}, 'yes'),
|
||||||
|
'if-tag-not02': ("{% if not %}yes{% else %}no{% endif %}", {'foo': True}, 'no'),
|
||||||
|
'if-tag-not03': ("{% if not %}yes{% else %}no{% endif %}", {'not': True}, 'yes'),
|
||||||
|
'if-tag-not04': ("{% if not not %}no{% else %}yes{% endif %}", {'not': True}, 'yes'),
|
||||||
|
'if-tag-not05': ("{% if not not %}no{% else %}yes{% endif %}", {}, 'no'),
|
||||||
|
|
||||||
|
'if-tag-not06': ("{% if foo and not bar %}yes{% else %}no{% endif %}", {}, 'no'),
|
||||||
|
'if-tag-not07': ("{% if foo and not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'no'),
|
||||||
|
'if-tag-not08': ("{% if foo and not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'yes'),
|
||||||
|
'if-tag-not09': ("{% if foo and not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'no'),
|
||||||
|
'if-tag-not10': ("{% if foo and not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'no'),
|
||||||
|
|
||||||
|
'if-tag-not11': ("{% if not foo and bar %}yes{% else %}no{% endif %}", {}, 'no'),
|
||||||
|
'if-tag-not12': ("{% if not foo and bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'no'),
|
||||||
|
'if-tag-not13': ("{% if not foo and bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'no'),
|
||||||
|
'if-tag-not14': ("{% if not foo and bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'yes'),
|
||||||
|
'if-tag-not15': ("{% if not foo and bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'no'),
|
||||||
|
|
||||||
|
'if-tag-not16': ("{% if foo or not bar %}yes{% else %}no{% endif %}", {}, 'yes'),
|
||||||
|
'if-tag-not17': ("{% if foo or not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'yes'),
|
||||||
|
'if-tag-not18': ("{% if foo or not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'yes'),
|
||||||
|
'if-tag-not19': ("{% if foo or not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'no'),
|
||||||
|
'if-tag-not20': ("{% if foo or not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'yes'),
|
||||||
|
|
||||||
|
'if-tag-not21': ("{% if not foo or bar %}yes{% else %}no{% endif %}", {}, 'yes'),
|
||||||
|
'if-tag-not22': ("{% if not foo or bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'yes'),
|
||||||
|
'if-tag-not23': ("{% if not foo or bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'no'),
|
||||||
|
'if-tag-not24': ("{% if not foo or bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'yes'),
|
||||||
|
'if-tag-not25': ("{% if not foo or bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'yes'),
|
||||||
|
|
||||||
|
'if-tag-not26': ("{% if not foo and not bar %}yes{% else %}no{% endif %}", {}, 'yes'),
|
||||||
|
'if-tag-not27': ("{% if not foo and not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'no'),
|
||||||
|
'if-tag-not28': ("{% if not foo and not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'no'),
|
||||||
|
'if-tag-not29': ("{% if not foo and not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'no'),
|
||||||
|
'if-tag-not30': ("{% if not foo and not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'yes'),
|
||||||
|
|
||||||
|
'if-tag-not31': ("{% if not foo or not bar %}yes{% else %}no{% endif %}", {}, 'yes'),
|
||||||
|
'if-tag-not32': ("{% if not foo or not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': True}, 'no'),
|
||||||
|
'if-tag-not33': ("{% if not foo or not bar %}yes{% else %}no{% endif %}", {'foo': True, 'bar': False}, 'yes'),
|
||||||
|
'if-tag-not34': ("{% if not foo or not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': True}, 'yes'),
|
||||||
|
'if-tag-not35': ("{% if not foo or not bar %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, 'yes'),
|
||||||
|
|
||||||
|
# AND and OR raises a TemplateSyntaxError
|
||||||
|
'if-tag-error01': ("{% if foo or bar and baz %}yes{% else %}no{% endif %}", {'foo': False, 'bar': False}, template.TemplateSyntaxError),
|
||||||
|
'if-tag-error02': ("{% if foo and %}yes{% else %}no{% endif %}", {'foo': True}, template.TemplateSyntaxError),
|
||||||
|
'if-tag-error03': ("{% if foo or %}yes{% else %}no{% endif %}", {'foo': True}, template.TemplateSyntaxError),
|
||||||
|
'if-tag-error04': ("{% if not foo and %}yes{% else %}no{% endif %}", {'foo': True}, template.TemplateSyntaxError),
|
||||||
|
'if-tag-error05': ("{% if not foo or %}yes{% else %}no{% endif %}", {'foo': True}, template.TemplateSyntaxError),
|
||||||
|
|
||||||
|
### IFCHANGED TAG #########################################################
|
||||||
|
'ifchanged01': ('{% for n in num %}{% ifchanged %}{{ n }}{% endifchanged %}{% endfor %}', { 'num': (1,2,3) }, '123'),
|
||||||
|
'ifchanged02': ('{% for n in num %}{% ifchanged %}{{ n }}{% endifchanged %}{% endfor %}', { 'num': (1,1,3) }, '13'),
|
||||||
|
'ifchanged03': ('{% for n in num %}{% ifchanged %}{{ n }}{% endifchanged %}{% endfor %}', { 'num': (1,1,1) }, '1'),
|
||||||
|
|
||||||
|
### IFEQUAL TAG ###########################################################
|
||||||
|
'ifequal01': ("{% ifequal a b %}yes{% endifequal %}", {"a": 1, "b": 2}, ""),
|
||||||
|
'ifequal02': ("{% ifequal a b %}yes{% endifequal %}", {"a": 1, "b": 1}, "yes"),
|
||||||
|
'ifequal03': ("{% ifequal a b %}yes{% else %}no{% endifequal %}", {"a": 1, "b": 2}, "no"),
|
||||||
|
'ifequal04': ("{% ifequal a b %}yes{% else %}no{% endifequal %}", {"a": 1, "b": 1}, "yes"),
|
||||||
|
'ifequal05': ("{% ifequal a 'test' %}yes{% else %}no{% endifequal %}", {"a": "test"}, "yes"),
|
||||||
|
'ifequal06': ("{% ifequal a 'test' %}yes{% else %}no{% endifequal %}", {"a": "no"}, "no"),
|
||||||
|
'ifequal07': ('{% ifequal a "test" %}yes{% else %}no{% endifequal %}', {"a": "test"}, "yes"),
|
||||||
|
'ifequal08': ('{% ifequal a "test" %}yes{% else %}no{% endifequal %}', {"a": "no"}, "no"),
|
||||||
|
'ifequal09': ('{% ifequal a "test" %}yes{% else %}no{% endifequal %}', {}, "no"),
|
||||||
|
'ifequal10': ('{% ifequal a b %}yes{% else %}no{% endifequal %}', {}, "yes"),
|
||||||
|
|
||||||
|
# SMART SPLITTING
|
||||||
|
'ifequal-split01': ('{% ifequal a "test man" %}yes{% else %}no{% endifequal %}', {}, "no"),
|
||||||
|
'ifequal-split02': ('{% ifequal a "test man" %}yes{% else %}no{% endifequal %}', {'a': 'foo'}, "no"),
|
||||||
|
'ifequal-split03': ('{% ifequal a "test man" %}yes{% else %}no{% endifequal %}', {'a': 'test man'}, "yes"),
|
||||||
|
'ifequal-split04': ("{% ifequal a 'test man' %}yes{% else %}no{% endifequal %}", {'a': 'test man'}, "yes"),
|
||||||
|
'ifequal-split05': ("{% ifequal a 'i \"love\" you' %}yes{% else %}no{% endifequal %}", {'a': ''}, "no"),
|
||||||
|
'ifequal-split06': ("{% ifequal a 'i \"love\" you' %}yes{% else %}no{% endifequal %}", {'a': 'i "love" you'}, "yes"),
|
||||||
|
'ifequal-split07': ("{% ifequal a 'i \"love\" you' %}yes{% else %}no{% endifequal %}", {'a': 'i love you'}, "no"),
|
||||||
|
'ifequal-split08': (r"{% ifequal a 'I\'m happy' %}yes{% else %}no{% endifequal %}", {'a': "I'm happy"}, "yes"),
|
||||||
|
'ifequal-split09': (r"{% ifequal a 'slash\man' %}yes{% else %}no{% endifequal %}", {'a': r"slash\man"}, "yes"),
|
||||||
|
'ifequal-split10': (r"{% ifequal a 'slash\man' %}yes{% else %}no{% endifequal %}", {'a': r"slashman"}, "no"),
|
||||||
|
|
||||||
|
### IFNOTEQUAL TAG ########################################################
|
||||||
|
'ifnotequal01': ("{% ifnotequal a b %}yes{% endifnotequal %}", {"a": 1, "b": 2}, "yes"),
|
||||||
|
'ifnotequal02': ("{% ifnotequal a b %}yes{% endifnotequal %}", {"a": 1, "b": 1}, ""),
|
||||||
|
'ifnotequal03': ("{% ifnotequal a b %}yes{% else %}no{% endifnotequal %}", {"a": 1, "b": 2}, "yes"),
|
||||||
|
'ifnotequal04': ("{% ifnotequal a b %}yes{% else %}no{% endifnotequal %}", {"a": 1, "b": 1}, "no"),
|
||||||
|
|
||||||
|
### INCLUDE TAG ###########################################################
|
||||||
|
'include01': ('{% include "basic-syntax01" %}', {}, "something cool"),
|
||||||
|
'include02': ('{% include "basic-syntax02" %}', {'headline': 'Included'}, "Included"),
|
||||||
|
'include03': ('{% include template_name %}', {'template_name': 'basic-syntax02', 'headline': 'Included'}, "Included"),
|
||||||
|
'include04': ('a{% include "nonexistent" %}b', {}, "ab"),
|
||||||
|
|
||||||
|
### INHERITANCE ###########################################################
|
||||||
|
|
||||||
|
# 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_'),
|
||||||
|
|
||||||
|
# Two-level inheritance with {{ block.super }}
|
||||||
|
'inheritance20': ("{% extends 'inheritance01' %}{% block first %}{{ block.super }}a{% endblock %}", {}, '1_a3_'),
|
||||||
|
|
||||||
|
# Three-level inheritance with {{ block.super }} from parent
|
||||||
|
'inheritance21': ("{% extends 'inheritance02' %}{% block first %}{{ block.super }}a{% endblock %}", {}, '12a34'),
|
||||||
|
|
||||||
|
# Three-level inheritance with {{ block.super }} from grandparent
|
||||||
|
'inheritance22': ("{% extends 'inheritance04' %}{% block first %}{{ block.super }}a{% endblock %}", {}, '1_a3_'),
|
||||||
|
|
||||||
|
# Three-level inheritance with {{ block.super }} from parent and grandparent
|
||||||
|
'inheritance23': ("{% extends 'inheritance20' %}{% block first %}{{ block.super }}b{% endblock %}", {}, '1_ab3_'),
|
||||||
|
|
||||||
|
# Inheritance from local context without use of template loader
|
||||||
|
'inheritance24': ("{% extends context_template %}{% block first %}2{% endblock %}{% block second %}4{% endblock %}", {'context_template': template.Template("1{% block first %}_{% endblock %}3{% block second %}_{% endblock %}")}, '1234'),
|
||||||
|
|
||||||
|
# Inheritance from local context with variable parent template
|
||||||
|
'inheritance25': ("{% extends context_template.1 %}{% block first %}2{% endblock %}{% block second %}4{% endblock %}", {'context_template': [template.Template("Wrong"), template.Template("1{% block first %}_{% endblock %}3{% block second %}_{% endblock %}")]}, '1234'),
|
||||||
|
|
||||||
|
### I18N ##################################################################
|
||||||
|
|
||||||
|
# {% spaceless %} tag
|
||||||
|
'spaceless01': ("{% spaceless %} <b> <i> text </i> </b> {% endspaceless %}", {}, "<b> <i> text </i> </b>"),
|
||||||
|
'spaceless02': ("{% spaceless %} <b> \n <i> text </i> \n </b> {% endspaceless %}", {}, "<b> <i> text </i> </b>"),
|
||||||
|
'spaceless03': ("{% spaceless %}<b><i>text</i></b>{% endspaceless %}", {}, "<b><i>text</i></b>"),
|
||||||
|
|
||||||
|
# simple translation of a string delimited by '
|
||||||
|
'i18n01': ("{% load i18n %}{% trans 'xxxyyyxxx' %}", {}, "xxxyyyxxx"),
|
||||||
|
|
||||||
|
# simple translation of a string delimited by "
|
||||||
|
'i18n02': ('{% load i18n %}{% trans "xxxyyyxxx" %}', {}, "xxxyyyxxx"),
|
||||||
|
|
||||||
|
# simple translation of a variable
|
||||||
|
'i18n03': ('{% load i18n %}{% blocktrans %}{{ anton }}{% endblocktrans %}', {'anton': 'xxxyyyxxx'}, "xxxyyyxxx"),
|
||||||
|
|
||||||
|
# simple translation of a variable and filter
|
||||||
|
'i18n04': ('{% load i18n %}{% blocktrans with anton|lower as berta %}{{ berta }}{% endblocktrans %}', {'anton': 'XXXYYYXXX'}, "xxxyyyxxx"),
|
||||||
|
|
||||||
|
# simple translation of a string with interpolation
|
||||||
|
'i18n05': ('{% load i18n %}{% blocktrans %}xxx{{ anton }}xxx{% endblocktrans %}', {'anton': 'yyy'}, "xxxyyyxxx"),
|
||||||
|
|
||||||
|
# simple translation of a string to german
|
||||||
|
'i18n06': ('{% load i18n %}{% trans "Page not found" %}', {'LANGUAGE_CODE': 'de'}, "Seite nicht gefunden"),
|
||||||
|
|
||||||
|
# translation of singular form
|
||||||
|
'i18n07': ('{% load i18n %}{% blocktrans count number as counter %}singular{% plural %}plural{% endblocktrans %}', {'number': 1}, "singular"),
|
||||||
|
|
||||||
|
# translation of plural form
|
||||||
|
'i18n08': ('{% load i18n %}{% blocktrans count number as counter %}singular{% plural %}plural{% endblocktrans %}', {'number': 2}, "plural"),
|
||||||
|
|
||||||
|
# simple non-translation (only marking) of a string to german
|
||||||
|
'i18n09': ('{% load i18n %}{% trans "Page not found" noop %}', {'LANGUAGE_CODE': 'de'}, "Page not found"),
|
||||||
|
|
||||||
|
# translation of a variable with a translated filter
|
||||||
|
'i18n10': ('{{ bool|yesno:_("ja,nein") }}', {'bool': True}, 'ja'),
|
||||||
|
|
||||||
|
# translation of a variable with a non-translated filter
|
||||||
|
'i18n11': ('{{ bool|yesno:"ja,nein" }}', {'bool': True}, 'ja'),
|
||||||
|
|
||||||
|
# usage of the get_available_languages tag
|
||||||
|
'i18n12': ('{% load i18n %}{% get_available_languages as langs %}{% for lang in langs %}{% ifequal lang.0 "de" %}{{ lang.0 }}{% endifequal %}{% endfor %}', {}, 'de'),
|
||||||
|
|
||||||
|
# translation of a constant string
|
||||||
|
'i18n13': ('{{ _("Page not found") }}', {'LANGUAGE_CODE': 'de'}, 'Seite nicht gefunden'),
|
||||||
|
|
||||||
|
### MULTILINE #############################################################
|
||||||
|
|
||||||
|
'multiline01': ("""
|
||||||
|
Hello,
|
||||||
|
boys.
|
||||||
|
How
|
||||||
|
are
|
||||||
|
you
|
||||||
|
gentlemen.
|
||||||
|
""",
|
||||||
|
{},
|
||||||
|
"""
|
||||||
|
Hello,
|
||||||
|
boys.
|
||||||
|
How
|
||||||
|
are
|
||||||
|
you
|
||||||
|
gentlemen.
|
||||||
|
"""),
|
||||||
|
|
||||||
|
### REGROUP TAG ###########################################################
|
||||||
|
'regroup01': ('{% regroup data by bar as grouped %}' + \
|
||||||
|
'{% for group in grouped %}' + \
|
||||||
|
'{{ group.grouper }}:' + \
|
||||||
|
'{% for item in group.list %}' + \
|
||||||
|
'{{ item.foo }}' + \
|
||||||
|
'{% endfor %},' + \
|
||||||
|
'{% endfor %}',
|
||||||
|
{'data': [ {'foo':'c', 'bar':1},
|
||||||
|
{'foo':'d', 'bar':1},
|
||||||
|
{'foo':'a', 'bar':2},
|
||||||
|
{'foo':'b', 'bar':2},
|
||||||
|
{'foo':'x', 'bar':3} ]},
|
||||||
|
'1:cd,2:ab,3:x,'),
|
||||||
|
|
||||||
|
# Test for silent failure when target variable isn't found
|
||||||
|
'regroup02': ('{% regroup data by bar as grouped %}' + \
|
||||||
|
'{% for group in grouped %}' + \
|
||||||
|
'{{ group.grouper }}:' + \
|
||||||
|
'{% for item in group.list %}' + \
|
||||||
|
'{{ item.foo }}' + \
|
||||||
|
'{% endfor %},' + \
|
||||||
|
'{% endfor %}',
|
||||||
|
{}, 'INVALID:INVALIDINVALIDINVALIDINVALIDINVALIDINVALIDINVALID,'),
|
||||||
|
|
||||||
|
### TEMPLATETAG TAG #######################################################
|
||||||
|
'templatetag01': ('{% templatetag openblock %}', {}, '{%'),
|
||||||
|
'templatetag02': ('{% templatetag closeblock %}', {}, '%}'),
|
||||||
|
'templatetag03': ('{% templatetag openvariable %}', {}, '{{'),
|
||||||
|
'templatetag04': ('{% templatetag closevariable %}', {}, '}}'),
|
||||||
|
'templatetag05': ('{% templatetag %}', {}, template.TemplateSyntaxError),
|
||||||
|
'templatetag06': ('{% templatetag foo %}', {}, template.TemplateSyntaxError),
|
||||||
|
'templatetag07': ('{% templatetag openbrace %}', {}, '{'),
|
||||||
|
'templatetag08': ('{% templatetag closebrace %}', {}, '}'),
|
||||||
|
'templatetag09': ('{% templatetag openbrace %}{% templatetag openbrace %}', {}, '{{'),
|
||||||
|
'templatetag10': ('{% templatetag closebrace %}{% templatetag closebrace %}', {}, '}}'),
|
||||||
|
|
||||||
|
### WIDTHRATIO TAG ########################################################
|
||||||
|
'widthratio01': ('{% widthratio a b 0 %}', {'a':50,'b':100}, '0'),
|
||||||
|
'widthratio02': ('{% widthratio a b 100 %}', {'a':0,'b':0}, ''),
|
||||||
|
'widthratio03': ('{% widthratio a b 100 %}', {'a':0,'b':100}, '0'),
|
||||||
|
'widthratio04': ('{% widthratio a b 100 %}', {'a':50,'b':100}, '50'),
|
||||||
|
'widthratio05': ('{% widthratio a b 100 %}', {'a':100,'b':100}, '100'),
|
||||||
|
|
||||||
|
# 62.5 should round to 63
|
||||||
|
'widthratio06': ('{% widthratio a b 100 %}', {'a':50,'b':80}, '63'),
|
||||||
|
|
||||||
|
# 71.4 should round to 71
|
||||||
|
'widthratio07': ('{% widthratio a b 100 %}', {'a':50,'b':70}, '71'),
|
||||||
|
|
||||||
|
# Raise exception if we don't have 3 args, last one an integer
|
||||||
|
'widthratio08': ('{% widthratio %}', {}, template.TemplateSyntaxError),
|
||||||
|
'widthratio09': ('{% widthratio a b %}', {'a':50,'b':100}, template.TemplateSyntaxError),
|
||||||
|
'widthratio10': ('{% widthratio a b 100.0 %}', {'a':50,'b':100}, template.TemplateSyntaxError),
|
||||||
|
|
||||||
|
### NOW TAG ########################################################
|
||||||
|
# Simple case
|
||||||
|
'now01' : ('{% now "j n Y"%}', {}, str(datetime.now().day) + ' ' + str(datetime.now().month) + ' ' + str(datetime.now().year)),
|
||||||
|
|
||||||
|
# Check parsing of escaped and special characters
|
||||||
|
'now02' : ('{% now "j "n" Y"%}', {}, template.TemplateSyntaxError),
|
||||||
|
# 'now03' : ('{% now "j \"n\" Y"%}', {}, str(datetime.now().day) + '"' + str(datetime.now().month) + '"' + str(datetime.now().year)),
|
||||||
|
# 'now04' : ('{% now "j \nn\n Y"%}', {}, str(datetime.now().day) + '\n' + str(datetime.now().month) + '\n' + str(datetime.now().year))
|
||||||
|
|
||||||
|
### TIMESINCE TAG ##################################################
|
||||||
|
# Default compare with datetime.now()
|
||||||
|
'timesince01' : ('{{ a|timesince }}', {'a':datetime.now() + timedelta(minutes=-1, seconds = -10)}, '1 minute'),
|
||||||
|
'timesince02' : ('{{ a|timesince }}', {'a':(datetime.now() - timedelta(days=1, minutes = 1))}, '1 day'),
|
||||||
|
'timesince03' : ('{{ a|timesince }}', {'a':(datetime.now() -
|
||||||
|
timedelta(hours=1, minutes=25, seconds = 10))}, '1 hour, 25 minutes'),
|
||||||
|
|
||||||
|
# Compare to a given parameter
|
||||||
|
'timesince04' : ('{{ a|timesince:b }}', {'a':NOW + timedelta(days=2), 'b':NOW + timedelta(days=1)}, '1 day'),
|
||||||
|
'timesince05' : ('{{ a|timesince:b }}', {'a':NOW + timedelta(days=2, minutes=1), 'b':NOW + timedelta(days=2)}, '1 minute'),
|
||||||
|
|
||||||
|
# Check that timezone is respected
|
||||||
|
'timesince06' : ('{{ a|timesince:b }}', {'a':NOW_tz + timedelta(hours=8), 'b':NOW_tz}, '8 hours'),
|
||||||
|
|
||||||
|
### TIMEUNTIL TAG ##################################################
|
||||||
|
# Default compare with datetime.now()
|
||||||
|
'timeuntil01' : ('{{ a|timeuntil }}', {'a':datetime.now() + timedelta(minutes=2, seconds = 10)}, '2 minutes'),
|
||||||
|
'timeuntil02' : ('{{ a|timeuntil }}', {'a':(datetime.now() + timedelta(days=1, seconds = 10))}, '1 day'),
|
||||||
|
'timeuntil03' : ('{{ a|timeuntil }}', {'a':(datetime.now() + timedelta(hours=8, minutes=10, seconds = 10))}, '8 hours, 10 minutes'),
|
||||||
|
|
||||||
|
# Compare to a given parameter
|
||||||
|
'timeuntil04' : ('{{ a|timeuntil:b }}', {'a':NOW - timedelta(days=1), 'b':NOW - timedelta(days=2)}, '1 day'),
|
||||||
|
'timeuntil05' : ('{{ a|timeuntil:b }}', {'a':NOW - timedelta(days=2), 'b':NOW - timedelta(days=2, minutes=1)}, '1 minute'),
|
||||||
|
}
|
||||||
|
|
||||||
|
# Register our custom template loader.
|
||||||
|
def test_template_loader(template_name, template_dirs=None):
|
||||||
|
"A custom template loader that loads the unit-test templates."
|
||||||
|
try:
|
||||||
|
return (TEMPLATE_TESTS[template_name][0] , "test:%s" % template_name)
|
||||||
|
except KeyError:
|
||||||
|
raise template.TemplateDoesNotExist, template_name
|
||||||
|
|
||||||
|
old_template_loaders = loader.template_source_loaders
|
||||||
|
loader.template_source_loaders = [test_template_loader]
|
||||||
|
|
||||||
|
failures = []
|
||||||
|
tests = TEMPLATE_TESTS.items()
|
||||||
|
tests.sort()
|
||||||
|
|
||||||
|
# Turn TEMPLATE_DEBUG off, because tests assume that.
|
||||||
|
old_td, settings.TEMPLATE_DEBUG = settings.TEMPLATE_DEBUG, False
|
||||||
|
|
||||||
|
# Set TEMPLATE_STRING_IF_INVALID to a known string
|
||||||
|
old_invalid, settings.TEMPLATE_STRING_IF_INVALID = settings.TEMPLATE_STRING_IF_INVALID, 'INVALID'
|
||||||
|
|
||||||
|
for name, vals in tests:
|
||||||
|
install()
|
||||||
|
if 'LANGUAGE_CODE' in vals[1]:
|
||||||
|
activate(vals[1]['LANGUAGE_CODE'])
|
||||||
|
else:
|
||||||
|
activate('en-us')
|
||||||
|
try:
|
||||||
|
output = loader.get_template(name).render(template.Context(vals[1]))
|
||||||
|
except Exception, e:
|
||||||
|
if e.__class__ != vals[2]:
|
||||||
|
failures.append("Template test: %s -- FAILED. Got %s, exception: %s" % (name, e.__class__, e))
|
||||||
|
continue
|
||||||
|
if 'LANGUAGE_CODE' in vals[1]:
|
||||||
|
deactivate()
|
||||||
|
if output != vals[2]:
|
||||||
|
failures.append("Template test: %s -- FAILED. Expected %r, got %r" % (name, vals[2], output))
|
||||||
|
loader.template_source_loaders = old_template_loaders
|
||||||
|
deactivate()
|
||||||
|
settings.TEMPLATE_DEBUG = old_td
|
||||||
|
settings.TEMPLATE_STRING_IF_INVALID = old_invalid
|
||||||
|
|
||||||
|
self.assertEqual(failures, [])
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
unittest.main()
|
|
@ -1,7 +1,7 @@
|
||||||
"Unit tests for reverse URL lookup"
|
"Unit tests for reverse URL lookup"
|
||||||
|
|
||||||
from django.core.urlresolvers import reverse_helper, NoReverseMatch
|
from django.core.urlresolvers import reverse_helper, NoReverseMatch
|
||||||
import re
|
import re, unittest
|
||||||
|
|
||||||
test_data = (
|
test_data = (
|
||||||
('^places/(\d+)/$', 'places/3/', [3], {}),
|
('^places/(\d+)/$', 'places/3/', [3], {}),
|
||||||
|
@ -25,23 +25,15 @@ test_data = (
|
||||||
('^people/(?P<state>\w\w)/(\w+)/$', 'people/il/adrian/', ['adrian'], {'state': 'il'}),
|
('^people/(?P<state>\w\w)/(\w+)/$', 'people/il/adrian/', ['adrian'], {'state': 'il'}),
|
||||||
)
|
)
|
||||||
|
|
||||||
def run_tests(verbosity=0):
|
class URLPatternReverse(unittest.TestCase):
|
||||||
for regex, expected, args, kwargs in test_data:
|
def test_urlpattern_reverse(self):
|
||||||
passed = True
|
for regex, expected, args, kwargs in test_data:
|
||||||
try:
|
try:
|
||||||
got = reverse_helper(re.compile(regex), *args, **kwargs)
|
got = reverse_helper(re.compile(regex), *args, **kwargs)
|
||||||
except NoReverseMatch, e:
|
except NoReverseMatch, e:
|
||||||
if expected != NoReverseMatch:
|
self.assertEqual(expected, NoReverseMatch)
|
||||||
passed, got = False, str(e)
|
else:
|
||||||
else:
|
self.assertEquals(got, expected)
|
||||||
if got != expected:
|
|
||||||
passed, got = False, got
|
|
||||||
if passed and verbosity:
|
|
||||||
print "Passed: %s" % regex
|
|
||||||
elif not passed:
|
|
||||||
print "REVERSE LOOKUP FAILED: %s" % regex
|
|
||||||
print " Got: %s" % got
|
|
||||||
print " Expected: %r" % expected
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
run_tests(1)
|
run_tests(1)
|
|
@ -1,23 +1,10 @@
|
||||||
#!/usr/bin/env python
|
#!/usr/bin/env python
|
||||||
|
|
||||||
import os, re, sys, time, traceback
|
import os, sys, traceback
|
||||||
|
import unittest
|
||||||
# doctest is included in the same package as this module, because this testing
|
|
||||||
# framework uses features only available in the Python 2.4 version of doctest,
|
|
||||||
# and Django aims to work with Python 2.3+.
|
|
||||||
import doctest
|
|
||||||
|
|
||||||
MODEL_TESTS_DIR_NAME = 'modeltests'
|
MODEL_TESTS_DIR_NAME = 'modeltests'
|
||||||
OTHER_TESTS_DIR = "othertests"
|
|
||||||
REGRESSION_TESTS_DIR_NAME = 'regressiontests'
|
REGRESSION_TESTS_DIR_NAME = 'regressiontests'
|
||||||
TEST_DATABASE_NAME = 'django_test_db'
|
|
||||||
|
|
||||||
error_list = []
|
|
||||||
def log_error(model_name, title, description):
|
|
||||||
error_list.append({
|
|
||||||
'title': "%r module: %s" % (model_name, title),
|
|
||||||
'description': description,
|
|
||||||
})
|
|
||||||
|
|
||||||
MODEL_TEST_DIR = os.path.join(os.path.dirname(__file__), MODEL_TESTS_DIR_NAME)
|
MODEL_TEST_DIR = os.path.join(os.path.dirname(__file__), MODEL_TESTS_DIR_NAME)
|
||||||
REGRESSION_TEST_DIR = os.path.join(os.path.dirname(__file__), REGRESSION_TESTS_DIR_NAME)
|
REGRESSION_TEST_DIR = os.path.join(os.path.dirname(__file__), REGRESSION_TESTS_DIR_NAME)
|
||||||
|
@ -37,258 +24,101 @@ def get_test_models():
|
||||||
models = []
|
models = []
|
||||||
for loc, dirpath in (MODEL_TESTS_DIR_NAME, MODEL_TEST_DIR), (REGRESSION_TESTS_DIR_NAME, REGRESSION_TEST_DIR):
|
for loc, dirpath in (MODEL_TESTS_DIR_NAME, MODEL_TEST_DIR), (REGRESSION_TESTS_DIR_NAME, REGRESSION_TEST_DIR):
|
||||||
for f in os.listdir(dirpath):
|
for f in os.listdir(dirpath):
|
||||||
if f.startswith('__init__') or f.startswith('.') or f.startswith('sql'):
|
if f.startswith('__init__') or f.startswith('.') or f.startswith('sql') or f.startswith('invalid'):
|
||||||
continue
|
continue
|
||||||
models.append((loc, f))
|
models.append((loc, f))
|
||||||
return models
|
return models
|
||||||
|
|
||||||
class DjangoDoctestRunner(doctest.DocTestRunner):
|
def get_invalid_models():
|
||||||
def __init__(self, verbosity_level, *args, **kwargs):
|
models = []
|
||||||
self.verbosity_level = verbosity_level
|
for loc, dirpath in (MODEL_TESTS_DIR_NAME, MODEL_TEST_DIR), (REGRESSION_TESTS_DIR_NAME, REGRESSION_TEST_DIR):
|
||||||
doctest.DocTestRunner.__init__(self, *args, **kwargs)
|
for f in os.listdir(dirpath):
|
||||||
self._checker = DjangoDoctestOutputChecker()
|
if f.startswith('__init__') or f.startswith('.') or f.startswith('sql'):
|
||||||
self.optionflags = doctest.ELLIPSIS
|
|
||||||
|
|
||||||
def report_start(self, out, test, example):
|
|
||||||
if self.verbosity_level > 1:
|
|
||||||
out(" >>> %s\n" % example.source.strip())
|
|
||||||
|
|
||||||
def report_failure(self, out, test, example, got):
|
|
||||||
log_error(test.name, "API test failed",
|
|
||||||
"Code: %r\nLine: %s\nExpected: %r\nGot: %r" % (example.source.strip(), example.lineno, example.want, got))
|
|
||||||
|
|
||||||
def report_unexpected_exception(self, out, test, example, exc_info):
|
|
||||||
from django.db import transaction
|
|
||||||
tb = ''.join(traceback.format_exception(*exc_info)[1:])
|
|
||||||
log_error(test.name, "API test raised an exception",
|
|
||||||
"Code: %r\nLine: %s\nException: %s" % (example.source.strip(), example.lineno, tb))
|
|
||||||
# Rollback, in case of database errors. Otherwise they'd have
|
|
||||||
# side effects on other tests.
|
|
||||||
transaction.rollback_unless_managed()
|
|
||||||
|
|
||||||
normalize_long_ints = lambda s: re.sub(r'(?<![\w])(\d+)L(?![\w])', '\\1', s)
|
|
||||||
|
|
||||||
class DjangoDoctestOutputChecker(doctest.OutputChecker):
|
|
||||||
def check_output(self, want, got, optionflags):
|
|
||||||
ok = doctest.OutputChecker.check_output(self, want, got, optionflags)
|
|
||||||
|
|
||||||
# Doctest does an exact string comparison of output, which means long
|
|
||||||
# integers aren't equal to normal integers ("22L" vs. "22"). The
|
|
||||||
# following code normalizes long integers so that they equal normal
|
|
||||||
# integers.
|
|
||||||
if not ok:
|
|
||||||
return normalize_long_ints(want) == normalize_long_ints(got)
|
|
||||||
return ok
|
|
||||||
|
|
||||||
class TestRunner:
|
|
||||||
def __init__(self, verbosity_level=0, which_tests=None):
|
|
||||||
self.verbosity_level = verbosity_level
|
|
||||||
self.which_tests = which_tests
|
|
||||||
|
|
||||||
def output(self, required_level, message):
|
|
||||||
if self.verbosity_level > required_level - 1:
|
|
||||||
print message
|
|
||||||
|
|
||||||
def run_tests(self):
|
|
||||||
from django.conf import settings
|
|
||||||
|
|
||||||
# An empty access of the settings to force the default options to be
|
|
||||||
# installed prior to assigning to them.
|
|
||||||
settings.INSTALLED_APPS
|
|
||||||
|
|
||||||
# Manually set INSTALLED_APPS to point to the test models.
|
|
||||||
settings.INSTALLED_APPS = ALWAYS_INSTALLED_APPS + ['.'.join(a) for a in get_test_models()]
|
|
||||||
|
|
||||||
# Manually set DEBUG and USE_I18N.
|
|
||||||
settings.DEBUG = False
|
|
||||||
settings.USE_I18N = True
|
|
||||||
|
|
||||||
from django.db import connection
|
|
||||||
from django.core import management
|
|
||||||
import django.db.models
|
|
||||||
|
|
||||||
# Determine which models we're going to test.
|
|
||||||
test_models = get_test_models()
|
|
||||||
if 'othertests' in self.which_tests:
|
|
||||||
self.which_tests.remove('othertests')
|
|
||||||
run_othertests = True
|
|
||||||
if not self.which_tests:
|
|
||||||
test_models = []
|
|
||||||
else:
|
|
||||||
run_othertests = not self.which_tests
|
|
||||||
|
|
||||||
if self.which_tests:
|
|
||||||
# Only run the specified tests.
|
|
||||||
bad_models = [m for m in self.which_tests if (MODEL_TESTS_DIR_NAME, m) not in test_models and (REGRESSION_TESTS_DIR_NAME, m) not in test_models]
|
|
||||||
if bad_models:
|
|
||||||
sys.stderr.write("Models not found: %s\n" % bad_models)
|
|
||||||
sys.exit(1)
|
|
||||||
else:
|
|
||||||
all_tests = []
|
|
||||||
for test in self.which_tests:
|
|
||||||
for loc in MODEL_TESTS_DIR_NAME, REGRESSION_TESTS_DIR_NAME:
|
|
||||||
if (loc, test) in test_models:
|
|
||||||
all_tests.append((loc, test))
|
|
||||||
test_models = all_tests
|
|
||||||
|
|
||||||
self.output(0, "Running tests with database %r" % settings.DATABASE_ENGINE)
|
|
||||||
|
|
||||||
# If we're using SQLite, it's more convenient to test against an
|
|
||||||
# in-memory database.
|
|
||||||
if settings.DATABASE_ENGINE == "sqlite3":
|
|
||||||
global TEST_DATABASE_NAME
|
|
||||||
TEST_DATABASE_NAME = ":memory:"
|
|
||||||
else:
|
|
||||||
# Create the test database and connect to it. We need to autocommit
|
|
||||||
# if the database supports it because PostgreSQL doesn't allow
|
|
||||||
# CREATE/DROP DATABASE statements within transactions.
|
|
||||||
cursor = connection.cursor()
|
|
||||||
self._set_autocommit(connection)
|
|
||||||
self.output(1, "Creating test database")
|
|
||||||
try:
|
|
||||||
cursor.execute("CREATE DATABASE %s" % TEST_DATABASE_NAME)
|
|
||||||
except Exception, e:
|
|
||||||
sys.stderr.write("Got an error creating the test database: %s\n" % e)
|
|
||||||
confirm = raw_input("It appears the test database, %s, already exists. Type 'yes' to delete it, or 'no' to cancel: " % TEST_DATABASE_NAME)
|
|
||||||
if confirm == 'yes':
|
|
||||||
cursor.execute("DROP DATABASE %s" % TEST_DATABASE_NAME)
|
|
||||||
cursor.execute("CREATE DATABASE %s" % TEST_DATABASE_NAME)
|
|
||||||
else:
|
|
||||||
print "Tests cancelled."
|
|
||||||
return
|
|
||||||
connection.close()
|
|
||||||
old_database_name = settings.DATABASE_NAME
|
|
||||||
settings.DATABASE_NAME = TEST_DATABASE_NAME
|
|
||||||
|
|
||||||
# Initialize the test database.
|
|
||||||
cursor = connection.cursor()
|
|
||||||
|
|
||||||
from django.db.models.loading import load_app
|
|
||||||
# Install the core always installed apps
|
|
||||||
for app in ALWAYS_INSTALLED_APPS:
|
|
||||||
self.output(1, "Installing contrib app %s" % app)
|
|
||||||
mod = load_app(app)
|
|
||||||
management.install(mod)
|
|
||||||
|
|
||||||
# Run the tests for each test model.
|
|
||||||
self.output(1, "Running app tests")
|
|
||||||
for model_dir, model_name in test_models:
|
|
||||||
self.output(1, "%s model: Importing" % model_name)
|
|
||||||
try:
|
|
||||||
mod = load_app(model_dir + '.' + model_name)
|
|
||||||
except Exception, e:
|
|
||||||
log_error(model_name, "Error while importing", ''.join(traceback.format_exception(*sys.exc_info())[1:]))
|
|
||||||
continue
|
continue
|
||||||
|
if f.startswith('invalid'):
|
||||||
|
models.append((loc, f))
|
||||||
|
return models
|
||||||
|
|
||||||
|
class InvalidModelTestCase(unittest.TestCase):
|
||||||
|
def __init__(self, model_label):
|
||||||
|
unittest.TestCase.__init__(self)
|
||||||
|
self.model_label = model_label
|
||||||
|
|
||||||
|
def runTest(self):
|
||||||
|
from django.core import management
|
||||||
|
from django.db.models.loading import load_app
|
||||||
|
from cStringIO import StringIO
|
||||||
|
|
||||||
if not getattr(mod, 'error_log', None):
|
try:
|
||||||
# Model is not marked as an invalid model
|
module = load_app(self.model_label)
|
||||||
self.output(1, "%s.%s model: Installing" % (model_dir, model_name))
|
except Exception, e:
|
||||||
management.install(mod)
|
self.fail('Unable to load invalid model module')
|
||||||
|
|
||||||
|
s = StringIO()
|
||||||
|
count = management.get_validation_errors(s, module)
|
||||||
|
s.seek(0)
|
||||||
|
error_log = s.read()
|
||||||
|
actual = error_log.split('\n')
|
||||||
|
expected = module.model_errors.split('\n')
|
||||||
|
|
||||||
# Run the API tests.
|
unexpected = [err for err in actual if err not in expected]
|
||||||
p = doctest.DocTestParser()
|
missing = [err for err in expected if err not in actual]
|
||||||
test_namespace = dict([(m._meta.object_name, m) \
|
|
||||||
for m in django.db.models.get_models(mod)])
|
|
||||||
dtest = p.get_doctest(mod.API_TESTS, test_namespace, model_name, None, None)
|
|
||||||
# Manually set verbose=False, because "-v" command-line parameter
|
|
||||||
# has side effects on doctest TestRunner class.
|
|
||||||
runner = DjangoDoctestRunner(verbosity_level=verbosity_level, verbose=False)
|
|
||||||
self.output(1, "%s.%s model: Running tests" % (model_dir, model_name))
|
|
||||||
runner.run(dtest, clear_globs=True, out=sys.stdout.write)
|
|
||||||
else:
|
|
||||||
# Check that model known to be invalid is invalid for the right reasons.
|
|
||||||
self.output(1, "%s.%s model: Validating" % (model_dir, model_name))
|
|
||||||
|
|
||||||
from cStringIO import StringIO
|
self.assert_(not unexpected, "Unexpected Errors: " + '\n'.join(unexpected))
|
||||||
s = StringIO()
|
self.assert_(not missing, "Missing Errors: " + '\n'.join(missing))
|
||||||
count = management.get_validation_errors(s, mod)
|
|
||||||
s.seek(0)
|
|
||||||
error_log = s.read()
|
|
||||||
actual = error_log.split('\n')
|
|
||||||
expected = mod.error_log.split('\n')
|
|
||||||
|
|
||||||
unexpected = [err for err in actual if err not in expected]
|
def django_tests(verbosity, tests_to_run):
|
||||||
missing = [err for err in expected if err not in actual]
|
from django.conf import settings
|
||||||
|
from django.db.models.loading import get_apps, load_app
|
||||||
if unexpected or missing:
|
old_installed_apps = settings.INSTALLED_APPS
|
||||||
unexpected_log = '\n'.join(unexpected)
|
|
||||||
missing_log = '\n'.join(missing)
|
# load all the ALWAYS_INSTALLED_APPS
|
||||||
log_error(model_name,
|
settings.INSTALLED_APPS = ALWAYS_INSTALLED_APPS
|
||||||
"Validator found %d validation errors, %d expected" % (count, len(expected) - 1),
|
get_apps()
|
||||||
"Missing errors:\n%s\n\nUnexpected errors:\n%s" % (missing_log, unexpected_log))
|
|
||||||
|
test_models = []
|
||||||
if run_othertests:
|
# Load all the test model apps
|
||||||
# Run the non-model tests in the other tests dir
|
for model_dir, model_name in get_test_models():
|
||||||
self.output(1, "Running other tests")
|
model_label = '.'.join([model_dir, model_name])
|
||||||
other_tests_dir = os.path.join(os.path.dirname(__file__), OTHER_TESTS_DIR)
|
try:
|
||||||
test_modules = [f[:-3] for f in os.listdir(other_tests_dir) if f.endswith('.py') and not f.startswith('__init__')]
|
# if the model was named on the command line, or
|
||||||
for module in test_modules:
|
# no models were named (i.e., run all), import
|
||||||
self.output(1, "%s module: Importing" % module)
|
# this model and add it to the list to test.
|
||||||
try:
|
if not tests_to_run or model_name in tests_to_run:
|
||||||
mod = __import__("othertests." + module, '', '', [''])
|
if verbosity >= 1:
|
||||||
except Exception, e:
|
print "Importing model %s" % model_name
|
||||||
log_error(module, "Error while importing", ''.join(traceback.format_exception(*sys.exc_info())[1:]))
|
mod = load_app(model_label)
|
||||||
continue
|
settings.INSTALLED_APPS.append(model_label)
|
||||||
if mod.__doc__:
|
test_models.append(mod)
|
||||||
p = doctest.DocTestParser()
|
except Exception, e:
|
||||||
dtest = p.get_doctest(mod.__doc__, mod.__dict__, module, None, None)
|
sys.stderr.write("Error while importing %s:" % model_name + ''.join(traceback.format_exception(*sys.exc_info())[1:]))
|
||||||
runner = DjangoDoctestRunner(verbosity_level=verbosity_level, verbose=False)
|
continue
|
||||||
self.output(1, "%s module: running 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: running 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
|
|
||||||
# connected to it.
|
|
||||||
if settings.DATABASE_ENGINE != "sqlite3":
|
|
||||||
connection.close()
|
|
||||||
settings.DATABASE_NAME = old_database_name
|
|
||||||
cursor = connection.cursor()
|
|
||||||
self.output(1, "Deleting test database")
|
|
||||||
self._set_autocommit(connection)
|
|
||||||
time.sleep(1) # To avoid "database is being accessed by other users" errors.
|
|
||||||
cursor.execute("DROP DATABASE %s" % TEST_DATABASE_NAME)
|
|
||||||
|
|
||||||
# Display output.
|
|
||||||
if error_list:
|
|
||||||
for d in error_list:
|
|
||||||
print
|
|
||||||
print d['title']
|
|
||||||
print "=" * len(d['title'])
|
|
||||||
print d['description']
|
|
||||||
print "%s error%s:" % (len(error_list), len(error_list) != 1 and 's' or '')
|
|
||||||
else:
|
|
||||||
print "All tests passed."
|
|
||||||
|
|
||||||
def _set_autocommit(self, connection):
|
|
||||||
"""
|
|
||||||
Make sure a connection is in autocommit mode.
|
|
||||||
"""
|
|
||||||
if hasattr(connection.connection, "autocommit"):
|
|
||||||
connection.connection.autocommit(True)
|
|
||||||
elif hasattr(connection.connection, "set_isolation_level"):
|
|
||||||
connection.connection.set_isolation_level(0)
|
|
||||||
|
|
||||||
|
# Add tests for invalid models
|
||||||
|
extra_tests = []
|
||||||
|
for model_dir, model_name in get_invalid_models():
|
||||||
|
model_label = '.'.join([model_dir, model_name])
|
||||||
|
if not tests_to_run or model_name in tests_to_run:
|
||||||
|
extra_tests.append(InvalidModelTestCase(model_label))
|
||||||
|
|
||||||
|
# Run the test suite, including the extra validation tests.
|
||||||
|
from django.test.simple import run_tests
|
||||||
|
run_tests(test_models, verbosity, extra_tests=extra_tests)
|
||||||
|
|
||||||
|
# Restore the old INSTALLED_APPS setting
|
||||||
|
settings.INSTALLED_APPS = old_installed_apps
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
from optparse import OptionParser
|
from optparse import OptionParser
|
||||||
usage = "%prog [options] [model model model ...]"
|
usage = "%prog [options] [model model model ...]"
|
||||||
parser = OptionParser(usage=usage)
|
parser = OptionParser(usage=usage)
|
||||||
parser.add_option('-v', help='How verbose should the output be? Choices are 0, 1 and 2, where 2 is most verbose. Default is 0.',
|
parser.add_option('-v','--verbosity', action='store', dest='verbosity', default='0',
|
||||||
type='choice', choices=['0', '1', '2'])
|
type='choice', choices=['0', '1', '2'],
|
||||||
|
help='Verbosity level; 0=minimal output, 1=normal output, 2=all output')
|
||||||
parser.add_option('--settings',
|
parser.add_option('--settings',
|
||||||
help='Python path to settings module, e.g. "myproject.settings". If this isn\'t provided, the DJANGO_SETTINGS_MODULE environment variable will be used.')
|
help='Python path to settings module, e.g. "myproject.settings". If this isn\'t provided, the DJANGO_SETTINGS_MODULE environment variable will be used.')
|
||||||
options, args = parser.parse_args()
|
options, args = parser.parse_args()
|
||||||
verbosity_level = 0
|
|
||||||
if options.v:
|
|
||||||
verbosity_level = int(options.v)
|
|
||||||
if options.settings:
|
if options.settings:
|
||||||
os.environ['DJANGO_SETTINGS_MODULE'] = options.settings
|
os.environ['DJANGO_SETTINGS_MODULE'] = options.settings
|
||||||
t = TestRunner(verbosity_level, args)
|
|
||||||
t.run_tests()
|
django_tests(int(options.verbosity), args)
|
||||||
|
|
Loading…
Reference in New Issue