2011-03-28 13:58:43 +08:00
|
|
|
from __future__ import with_statement
|
|
|
|
|
2010-10-12 11:33:19 +08:00
|
|
|
import sys
|
|
|
|
|
2010-10-17 12:26:47 +08:00
|
|
|
from django.test import TestCase, skipUnlessDBFeature, skipIfDBFeature
|
|
|
|
|
2011-01-20 12:47:47 +08:00
|
|
|
from models import Person
|
2010-10-17 12:26:47 +08:00
|
|
|
|
2010-10-12 11:33:19 +08:00
|
|
|
|
2010-10-17 12:26:47 +08:00
|
|
|
class SkippingTestCase(TestCase):
|
2011-01-20 12:47:47 +08:00
|
|
|
def test_skip_unless_db_feature(self):
|
|
|
|
"A test that might be skipped is actually called."
|
|
|
|
# Total hack, but it works, just want an attribute that's always true.
|
|
|
|
@skipUnlessDBFeature("__class__")
|
|
|
|
def test_func():
|
|
|
|
raise ValueError
|
|
|
|
|
|
|
|
self.assertRaises(ValueError, test_func)
|
|
|
|
|
|
|
|
|
|
|
|
class AssertNumQueriesTests(TestCase):
|
2011-04-02 21:27:17 +08:00
|
|
|
urls = 'regressiontests.test_utils.urls'
|
|
|
|
|
2010-10-31 16:29:07 +08:00
|
|
|
def test_assert_num_queries(self):
|
|
|
|
def test_func():
|
|
|
|
raise ValueError
|
|
|
|
|
|
|
|
self.assertRaises(ValueError,
|
|
|
|
self.assertNumQueries, 2, test_func
|
|
|
|
)
|
|
|
|
|
2011-01-20 12:47:47 +08:00
|
|
|
def test_assert_num_queries_with_client(self):
|
|
|
|
person = Person.objects.create(name='test')
|
2010-10-17 12:26:47 +08:00
|
|
|
|
2011-01-20 12:47:47 +08:00
|
|
|
self.assertNumQueries(
|
|
|
|
1,
|
|
|
|
self.client.get,
|
|
|
|
"/test_utils/get_person/%s/" % person.pk
|
|
|
|
)
|
2010-10-17 12:26:47 +08:00
|
|
|
|
2011-01-20 12:47:47 +08:00
|
|
|
self.assertNumQueries(
|
|
|
|
1,
|
|
|
|
self.client.get,
|
|
|
|
"/test_utils/get_person/%s/" % person.pk
|
|
|
|
)
|
2010-10-17 12:26:47 +08:00
|
|
|
|
2011-01-20 12:47:47 +08:00
|
|
|
def test_func():
|
|
|
|
self.client.get("/test_utils/get_person/%s/" % person.pk)
|
|
|
|
self.client.get("/test_utils/get_person/%s/" % person.pk)
|
|
|
|
self.assertNumQueries(2, test_func)
|
2010-11-11 23:06:20 +08:00
|
|
|
|
2011-03-28 13:58:43 +08:00
|
|
|
class AssertNumQueriesContextManagerTests(TestCase):
|
2011-04-02 21:27:17 +08:00
|
|
|
urls = 'regressiontests.test_utils.urls'
|
|
|
|
|
2011-03-28 13:58:43 +08:00
|
|
|
def test_simple(self):
|
|
|
|
with self.assertNumQueries(0):
|
|
|
|
pass
|
|
|
|
|
|
|
|
with self.assertNumQueries(1):
|
|
|
|
Person.objects.count()
|
|
|
|
|
|
|
|
with self.assertNumQueries(2):
|
|
|
|
Person.objects.count()
|
|
|
|
Person.objects.count()
|
|
|
|
|
|
|
|
def test_failure(self):
|
|
|
|
with self.assertRaises(AssertionError) as exc_info:
|
|
|
|
with self.assertNumQueries(2):
|
|
|
|
Person.objects.count()
|
|
|
|
self.assertIn("1 queries executed, 2 expected", str(exc_info.exception))
|
|
|
|
|
|
|
|
with self.assertRaises(TypeError):
|
|
|
|
with self.assertNumQueries(4000):
|
|
|
|
raise TypeError
|
|
|
|
|
|
|
|
def test_with_client(self):
|
|
|
|
person = Person.objects.create(name="test")
|
|
|
|
|
|
|
|
with self.assertNumQueries(1):
|
|
|
|
self.client.get("/test_utils/get_person/%s/" % person.pk)
|
|
|
|
|
|
|
|
with self.assertNumQueries(1):
|
|
|
|
self.client.get("/test_utils/get_person/%s/" % person.pk)
|
|
|
|
|
|
|
|
with self.assertNumQueries(2):
|
|
|
|
self.client.get("/test_utils/get_person/%s/" % person.pk)
|
|
|
|
self.client.get("/test_utils/get_person/%s/" % person.pk)
|
|
|
|
|
2011-01-20 12:47:47 +08:00
|
|
|
|
|
|
|
class SaveRestoreWarningState(TestCase):
|
2010-11-11 23:06:20 +08:00
|
|
|
def test_save_restore_warnings_state(self):
|
|
|
|
"""
|
|
|
|
Ensure save_warnings_state/restore_warnings_state work correctly.
|
|
|
|
"""
|
|
|
|
# In reality this test could be satisfied by many broken implementations
|
|
|
|
# of save_warnings_state/restore_warnings_state (e.g. just
|
|
|
|
# warnings.resetwarnings()) , but it is difficult to test more.
|
|
|
|
import warnings
|
|
|
|
self.save_warnings_state()
|
|
|
|
|
|
|
|
class MyWarning(Warning):
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Add a filter that causes an exception to be thrown, so we can catch it
|
|
|
|
warnings.simplefilter("error", MyWarning)
|
|
|
|
self.assertRaises(Warning, lambda: warnings.warn("warn", MyWarning))
|
|
|
|
|
|
|
|
# Now restore.
|
|
|
|
self.restore_warnings_state()
|
|
|
|
# After restoring, we shouldn't get an exception. But we don't want a
|
|
|
|
# warning printed either, so we have to silence the warning.
|
|
|
|
warnings.simplefilter("ignore", MyWarning)
|
|
|
|
warnings.warn("warn", MyWarning)
|
|
|
|
|
|
|
|
# Remove the filter we just added.
|
|
|
|
self.restore_warnings_state()
|
|
|
|
|
2010-10-12 11:33:19 +08:00
|
|
|
__test__ = {"API_TEST": r"""
|
2008-07-19 22:46:55 +08:00
|
|
|
# Some checks of the doctest output normalizer.
|
2010-10-12 11:33:19 +08:00
|
|
|
# Standard doctests do fairly
|
2008-07-19 22:46:55 +08:00
|
|
|
>>> from django.utils import simplejson
|
|
|
|
>>> from django.utils.xmlutils import SimplerXMLGenerator
|
|
|
|
>>> from StringIO import StringIO
|
|
|
|
|
|
|
|
>>> def produce_long():
|
|
|
|
... return 42L
|
|
|
|
|
|
|
|
>>> def produce_int():
|
|
|
|
... return 42
|
|
|
|
|
|
|
|
>>> def produce_json():
|
|
|
|
... return simplejson.dumps(['foo', {'bar': ('baz', None, 1.0, 2), 'whiz': 42}])
|
|
|
|
|
|
|
|
>>> def produce_xml():
|
|
|
|
... stream = StringIO()
|
|
|
|
... xml = SimplerXMLGenerator(stream, encoding='utf-8')
|
|
|
|
... xml.startDocument()
|
|
|
|
... xml.startElement("foo", {"aaa" : "1.0", "bbb": "2.0"})
|
|
|
|
... xml.startElement("bar", {"ccc" : "3.0"})
|
|
|
|
... xml.characters("Hello")
|
|
|
|
... xml.endElement("bar")
|
|
|
|
... xml.startElement("whiz", {})
|
|
|
|
... xml.characters("Goodbye")
|
|
|
|
... xml.endElement("whiz")
|
|
|
|
... xml.endElement("foo")
|
|
|
|
... xml.endDocument()
|
|
|
|
... return stream.getvalue()
|
|
|
|
|
2008-07-20 13:46:41 +08:00
|
|
|
>>> def produce_xml_fragment():
|
|
|
|
... stream = StringIO()
|
|
|
|
... xml = SimplerXMLGenerator(stream, encoding='utf-8')
|
2008-07-20 16:47:10 +08:00
|
|
|
... xml.startElement("foo", {"aaa": "1.0", "bbb": "2.0"})
|
2008-07-20 13:46:41 +08:00
|
|
|
... xml.characters("Hello")
|
|
|
|
... xml.endElement("foo")
|
2008-07-20 16:47:10 +08:00
|
|
|
... xml.startElement("bar", {"ccc": "3.0", "ddd": "4.0"})
|
2008-07-20 13:46:41 +08:00
|
|
|
... xml.endElement("bar")
|
|
|
|
... return stream.getvalue()
|
|
|
|
|
2008-07-19 22:46:55 +08:00
|
|
|
# Long values are normalized and are comparable to normal integers ...
|
|
|
|
>>> produce_long()
|
|
|
|
42
|
|
|
|
|
|
|
|
# ... and vice versa
|
|
|
|
>>> produce_int()
|
|
|
|
42L
|
|
|
|
|
|
|
|
# JSON output is normalized for field order, so it doesn't matter
|
|
|
|
# which order json dictionary attributes are listed in output
|
|
|
|
>>> produce_json()
|
|
|
|
'["foo", {"bar": ["baz", null, 1.0, 2], "whiz": 42}]'
|
|
|
|
|
|
|
|
>>> produce_json()
|
|
|
|
'["foo", {"whiz": 42, "bar": ["baz", null, 1.0, 2]}]'
|
|
|
|
|
2010-10-12 11:33:19 +08:00
|
|
|
# XML output is normalized for attribute order, so it doesn't matter
|
2008-07-19 22:46:55 +08:00
|
|
|
# which order XML element attributes are listed in output
|
|
|
|
>>> produce_xml()
|
|
|
|
'<?xml version="1.0" encoding="UTF-8"?>\n<foo aaa="1.0" bbb="2.0"><bar ccc="3.0">Hello</bar><whiz>Goodbye</whiz></foo>'
|
|
|
|
|
|
|
|
>>> produce_xml()
|
|
|
|
'<?xml version="1.0" encoding="UTF-8"?>\n<foo bbb="2.0" aaa="1.0"><bar ccc="3.0">Hello</bar><whiz>Goodbye</whiz></foo>'
|
|
|
|
|
2008-07-20 13:46:41 +08:00
|
|
|
>>> produce_xml_fragment()
|
2008-07-20 16:47:10 +08:00
|
|
|
'<foo aaa="1.0" bbb="2.0">Hello</foo><bar ccc="3.0" ddd="4.0"></bar>'
|
2008-07-20 13:46:41 +08:00
|
|
|
|
|
|
|
>>> produce_xml_fragment()
|
2008-07-20 16:47:10 +08:00
|
|
|
'<foo bbb="2.0" aaa="1.0">Hello</foo><bar ddd="4.0" ccc="3.0"></bar>'
|
2008-07-19 22:46:55 +08:00
|
|
|
|
2010-10-12 11:33:19 +08:00
|
|
|
"""}
|