tests.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. """
  2. Tests for django test runner
  3. """
  4. from __future__ import unicode_literals
  5. from importlib import import_module
  6. from optparse import make_option
  7. import sys
  8. import unittest
  9. from django.core.exceptions import ImproperlyConfigured
  10. from django.core.management import call_command
  11. from django import db
  12. from django.test import runner, TestCase, TransactionTestCase, skipUnlessDBFeature
  13. from django.test.testcases import connections_support_transactions
  14. from django.test.utils import IgnoreAllDeprecationWarningsMixin
  15. from admin_scripts.tests import AdminScriptTestCase
  16. from .models import Person
  17. TEST_APP_OK = 'test_runner.valid_app.models'
  18. TEST_APP_ERROR = 'test_runner_invalid_app.models'
  19. class DependencyOrderingTests(unittest.TestCase):
  20. def test_simple_dependencies(self):
  21. raw = [
  22. ('s1', ('s1_db', ['alpha'])),
  23. ('s2', ('s2_db', ['bravo'])),
  24. ('s3', ('s3_db', ['charlie'])),
  25. ]
  26. dependencies = {
  27. 'alpha': ['charlie'],
  28. 'bravo': ['charlie'],
  29. }
  30. ordered = runner.dependency_ordered(raw, dependencies=dependencies)
  31. ordered_sigs = [sig for sig,value in ordered]
  32. self.assertIn('s1', ordered_sigs)
  33. self.assertIn('s2', ordered_sigs)
  34. self.assertIn('s3', ordered_sigs)
  35. self.assertLess(ordered_sigs.index('s3'), ordered_sigs.index('s1'))
  36. self.assertLess(ordered_sigs.index('s3'), ordered_sigs.index('s2'))
  37. def test_chained_dependencies(self):
  38. raw = [
  39. ('s1', ('s1_db', ['alpha'])),
  40. ('s2', ('s2_db', ['bravo'])),
  41. ('s3', ('s3_db', ['charlie'])),
  42. ]
  43. dependencies = {
  44. 'alpha': ['bravo'],
  45. 'bravo': ['charlie'],
  46. }
  47. ordered = runner.dependency_ordered(raw, dependencies=dependencies)
  48. ordered_sigs = [sig for sig,value in ordered]
  49. self.assertIn('s1', ordered_sigs)
  50. self.assertIn('s2', ordered_sigs)
  51. self.assertIn('s3', ordered_sigs)
  52. # Explicit dependencies
  53. self.assertLess(ordered_sigs.index('s2'), ordered_sigs.index('s1'))
  54. self.assertLess(ordered_sigs.index('s3'), ordered_sigs.index('s2'))
  55. # Implied dependencies
  56. self.assertLess(ordered_sigs.index('s3'), ordered_sigs.index('s1'))
  57. def test_multiple_dependencies(self):
  58. raw = [
  59. ('s1', ('s1_db', ['alpha'])),
  60. ('s2', ('s2_db', ['bravo'])),
  61. ('s3', ('s3_db', ['charlie'])),
  62. ('s4', ('s4_db', ['delta'])),
  63. ]
  64. dependencies = {
  65. 'alpha': ['bravo','delta'],
  66. 'bravo': ['charlie'],
  67. 'delta': ['charlie'],
  68. }
  69. ordered = runner.dependency_ordered(raw, dependencies=dependencies)
  70. ordered_sigs = [sig for sig,aliases in ordered]
  71. self.assertIn('s1', ordered_sigs)
  72. self.assertIn('s2', ordered_sigs)
  73. self.assertIn('s3', ordered_sigs)
  74. self.assertIn('s4', ordered_sigs)
  75. # Explicit dependencies
  76. self.assertLess(ordered_sigs.index('s2'), ordered_sigs.index('s1'))
  77. self.assertLess(ordered_sigs.index('s4'), ordered_sigs.index('s1'))
  78. self.assertLess(ordered_sigs.index('s3'), ordered_sigs.index('s2'))
  79. self.assertLess(ordered_sigs.index('s3'), ordered_sigs.index('s4'))
  80. # Implicit dependencies
  81. self.assertLess(ordered_sigs.index('s3'), ordered_sigs.index('s1'))
  82. def test_circular_dependencies(self):
  83. raw = [
  84. ('s1', ('s1_db', ['alpha'])),
  85. ('s2', ('s2_db', ['bravo'])),
  86. ]
  87. dependencies = {
  88. 'bravo': ['alpha'],
  89. 'alpha': ['bravo'],
  90. }
  91. self.assertRaises(ImproperlyConfigured, runner.dependency_ordered, raw, dependencies=dependencies)
  92. def test_own_alias_dependency(self):
  93. raw = [
  94. ('s1', ('s1_db', ['alpha', 'bravo']))
  95. ]
  96. dependencies = {
  97. 'alpha': ['bravo']
  98. }
  99. with self.assertRaises(ImproperlyConfigured):
  100. runner.dependency_ordered(raw, dependencies=dependencies)
  101. # reordering aliases shouldn't matter
  102. raw = [
  103. ('s1', ('s1_db', ['bravo', 'alpha']))
  104. ]
  105. with self.assertRaises(ImproperlyConfigured):
  106. runner.dependency_ordered(raw, dependencies=dependencies)
  107. class MockTestRunner(object):
  108. invoked = False
  109. def __init__(self, *args, **kwargs):
  110. pass
  111. def run_tests(self, test_labels, extra_tests=None, **kwargs):
  112. MockTestRunner.invoked = True
  113. class ManageCommandTests(unittest.TestCase):
  114. def test_custom_test_runner(self):
  115. call_command('test', 'sites',
  116. testrunner='test_runner.tests.MockTestRunner')
  117. self.assertTrue(MockTestRunner.invoked,
  118. "The custom test runner has not been invoked")
  119. def test_bad_test_runner(self):
  120. with self.assertRaises(AttributeError):
  121. call_command('test', 'sites',
  122. testrunner='test_runner.NonExistentRunner')
  123. class CustomOptionsTestRunner(runner.DiscoverRunner):
  124. option_list = (
  125. make_option('--option_a','-a', action='store', dest='option_a', default='1'),
  126. make_option('--option_b','-b', action='store', dest='option_b', default='2'),
  127. make_option('--option_c','-c', action='store', dest='option_c', default='3'),
  128. )
  129. def __init__(self, verbosity=1, interactive=True, failfast=True, option_a=None, option_b=None, option_c=None, **kwargs):
  130. super(CustomOptionsTestRunner, self).__init__(verbosity=verbosity, interactive=interactive,
  131. failfast=failfast)
  132. self.option_a = option_a
  133. self.option_b = option_b
  134. self.option_c = option_c
  135. def run_tests(self, test_labels, extra_tests=None, **kwargs):
  136. print("%s:%s:%s" % (self.option_a, self.option_b, self.option_c))
  137. class CustomTestRunnerOptionsTests(AdminScriptTestCase):
  138. def setUp(self):
  139. settings = {
  140. 'TEST_RUNNER': '\'test_runner.tests.CustomOptionsTestRunner\'',
  141. }
  142. self.write_settings('settings.py', sdict=settings)
  143. def tearDown(self):
  144. self.remove_settings('settings.py')
  145. def test_default_options(self):
  146. args = ['test', '--settings=test_project.settings']
  147. out, err = self.run_django_admin(args)
  148. self.assertNoOutput(err)
  149. self.assertOutput(out, '1:2:3')
  150. def test_default_and_given_options(self):
  151. args = ['test', '--settings=test_project.settings', '--option_b=foo']
  152. out, err = self.run_django_admin(args)
  153. self.assertNoOutput(err)
  154. self.assertOutput(out, '1:foo:3')
  155. def test_option_name_and_value_separated(self):
  156. args = ['test', '--settings=test_project.settings', '--option_b', 'foo']
  157. out, err = self.run_django_admin(args)
  158. self.assertNoOutput(err)
  159. self.assertOutput(out, '1:foo:3')
  160. def test_all_options_given(self):
  161. args = ['test', '--settings=test_project.settings', '--option_a=bar',
  162. '--option_b=foo', '--option_c=31337']
  163. out, err = self.run_django_admin(args)
  164. self.assertNoOutput(err)
  165. self.assertOutput(out, 'bar:foo:31337')
  166. class Ticket17477RegressionTests(AdminScriptTestCase):
  167. def setUp(self):
  168. self.write_settings('settings.py')
  169. def tearDown(self):
  170. self.remove_settings('settings.py')
  171. def test_ticket_17477(self):
  172. """'manage.py help test' works after r16352."""
  173. args = ['help', 'test']
  174. out, err = self.run_manage(args)
  175. self.assertNoOutput(err)
  176. class ModulesTestsPackages(IgnoreAllDeprecationWarningsMixin, unittest.TestCase):
  177. def test_get_tests(self):
  178. "Check that the get_tests helper function can find tests in a directory"
  179. from django.test.simple import get_tests
  180. module = import_module(TEST_APP_OK)
  181. tests = get_tests(module)
  182. self.assertIsInstance(tests, type(module))
  183. def test_import_error(self):
  184. "Test for #12658 - Tests with ImportError's shouldn't fail silently"
  185. from django.test.simple import get_tests
  186. module = import_module(TEST_APP_ERROR)
  187. self.assertRaises(ImportError, get_tests, module)
  188. class Sqlite3InMemoryTestDbs(TestCase):
  189. available_apps = []
  190. @unittest.skipUnless(all(db.connections[conn].vendor == 'sqlite' for conn in db.connections),
  191. "This is an sqlite-specific issue")
  192. def test_transaction_support(self):
  193. """Ticket #16329: sqlite3 in-memory test databases"""
  194. old_db_connections = db.connections
  195. for option in ('NAME', 'TEST_NAME'):
  196. try:
  197. db.connections = db.ConnectionHandler({
  198. 'default': {
  199. 'ENGINE': 'django.db.backends.sqlite3',
  200. option: ':memory:',
  201. },
  202. 'other': {
  203. 'ENGINE': 'django.db.backends.sqlite3',
  204. option: ':memory:',
  205. },
  206. })
  207. other = db.connections['other']
  208. runner.DiscoverRunner(verbosity=0).setup_databases()
  209. msg = "DATABASES setting '%s' option set to sqlite3's ':memory:' value shouldn't interfere with transaction support detection." % option
  210. # Transaction support should be properly initialised for the 'other' DB
  211. self.assertTrue(other.features.supports_transactions, msg)
  212. # And all the DBs should report that they support transactions
  213. self.assertTrue(connections_support_transactions(), msg)
  214. finally:
  215. db.connections = old_db_connections
  216. class DummyBackendTest(unittest.TestCase):
  217. def test_setup_databases(self):
  218. """
  219. Test that setup_databases() doesn't fail with dummy database backend.
  220. """
  221. runner_instance = runner.DiscoverRunner(verbosity=0)
  222. old_db_connections = db.connections
  223. try:
  224. db.connections = db.ConnectionHandler({})
  225. old_config = runner_instance.setup_databases()
  226. runner_instance.teardown_databases(old_config)
  227. except Exception as e:
  228. self.fail("setup_databases/teardown_databases unexpectedly raised "
  229. "an error: %s" % e)
  230. finally:
  231. db.connections = old_db_connections
  232. class AliasedDefaultTestSetupTest(unittest.TestCase):
  233. def test_setup_aliased_default_database(self):
  234. """
  235. Test that setup_datebases() doesn't fail when 'default' is aliased
  236. """
  237. runner_instance = runner.DiscoverRunner(verbosity=0)
  238. old_db_connections = db.connections
  239. try:
  240. db.connections = db.ConnectionHandler({
  241. 'default': {
  242. 'NAME': 'dummy'
  243. },
  244. 'aliased': {
  245. 'NAME': 'dummy'
  246. }
  247. })
  248. old_config = runner_instance.setup_databases()
  249. runner_instance.teardown_databases(old_config)
  250. except Exception as e:
  251. self.fail("setup_databases/teardown_databases unexpectedly raised "
  252. "an error: %s" % e)
  253. finally:
  254. db.connections = old_db_connections
  255. class AliasedDatabaseTeardownTest(unittest.TestCase):
  256. def test_setup_aliased_databases(self):
  257. from django.db.backends.dummy.base import DatabaseCreation
  258. runner_instance = runner.DiscoverRunner(verbosity=0)
  259. old_db_connections = db.connections
  260. old_destroy_test_db = DatabaseCreation.destroy_test_db
  261. old_create_test_db = DatabaseCreation.create_test_db
  262. try:
  263. destroyed_names = []
  264. DatabaseCreation.destroy_test_db = lambda self, old_database_name, verbosity=1: destroyed_names.append(old_database_name)
  265. DatabaseCreation.create_test_db = lambda self, verbosity=1, autoclobber=False: self._get_test_db_name()
  266. db.connections = db.ConnectionHandler({
  267. 'default': {
  268. 'ENGINE': 'django.db.backends.dummy',
  269. 'NAME': 'dbname',
  270. },
  271. 'other': {
  272. 'ENGINE': 'django.db.backends.dummy',
  273. 'NAME': 'dbname',
  274. }
  275. })
  276. old_config = runner_instance.setup_databases()
  277. runner_instance.teardown_databases(old_config)
  278. self.assertEqual(destroyed_names.count('dbname'), 1)
  279. finally:
  280. DatabaseCreation.create_test_db = old_create_test_db
  281. DatabaseCreation.destroy_test_db = old_destroy_test_db
  282. db.connections = old_db_connections
  283. class DeprecationDisplayTest(AdminScriptTestCase):
  284. # tests for 19546
  285. def setUp(self):
  286. settings = {
  287. 'DATABASES': '{"default": {"ENGINE":"django.db.backends.sqlite3", "NAME":":memory:"}}'
  288. }
  289. self.write_settings('settings.py', sdict=settings)
  290. def tearDown(self):
  291. self.remove_settings('settings.py')
  292. def test_runner_deprecation_verbosity_default(self):
  293. args = ['test', '--settings=test_project.settings', 'test_runner_deprecation_app']
  294. out, err = self.run_django_admin(args)
  295. self.assertIn("DeprecationWarning: warning from test", err)
  296. self.assertIn("DeprecationWarning: module-level warning from deprecation_app", err)
  297. def test_runner_deprecation_verbosity_zero(self):
  298. args = ['test', '--settings=settings', '--verbosity=0']
  299. out, err = self.run_django_admin(args)
  300. self.assertFalse("DeprecationWarning: warning from test" in err)
  301. class AutoIncrementResetTest(TransactionTestCase):
  302. """
  303. Here we test creating the same model two times in different test methods,
  304. and check that both times they get "1" as their PK value. That is, we test
  305. that AutoField values start from 1 for each transactional test case.
  306. """
  307. available_apps = ['test_runner']
  308. reset_sequences = True
  309. @skipUnlessDBFeature('supports_sequence_reset')
  310. def test_autoincrement_reset1(self):
  311. p = Person.objects.create(first_name='Jack', last_name='Smith')
  312. self.assertEqual(p.pk, 1)
  313. @skipUnlessDBFeature('supports_sequence_reset')
  314. def test_autoincrement_reset2(self):
  315. p = Person.objects.create(first_name='Jack', last_name='Smith')
  316. self.assertEqual(p.pk, 1)