tests.py 37 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067
  1. """
  2. Tests for django test runner
  3. """
  4. import collections.abc
  5. import multiprocessing
  6. import os
  7. import sys
  8. import unittest
  9. from unittest import mock
  10. from admin_scripts.tests import AdminScriptTestCase
  11. from django import db
  12. from django.conf import settings
  13. from django.core.exceptions import ImproperlyConfigured
  14. from django.core.management import call_command
  15. from django.core.management.base import SystemCheckError
  16. from django.test import SimpleTestCase, TransactionTestCase, skipUnlessDBFeature
  17. from django.test.runner import (
  18. DiscoverRunner,
  19. Shuffler,
  20. _init_worker,
  21. reorder_test_bin,
  22. reorder_tests,
  23. shuffle_tests,
  24. )
  25. from django.test.testcases import connections_support_transactions
  26. from django.test.utils import (
  27. captured_stderr,
  28. dependency_ordered,
  29. get_unique_databases_and_mirrors,
  30. iter_test_cases,
  31. )
  32. from .models import B, Person, Through
  33. class MySuite:
  34. def __init__(self):
  35. self.tests = []
  36. def addTest(self, test):
  37. self.tests.append(test)
  38. def __iter__(self):
  39. yield from self.tests
  40. class TestSuiteTests(SimpleTestCase):
  41. def build_test_suite(self, test_classes, suite=None, suite_class=None):
  42. if suite_class is None:
  43. suite_class = unittest.TestSuite
  44. if suite is None:
  45. suite = suite_class()
  46. loader = unittest.defaultTestLoader
  47. for test_class in test_classes:
  48. tests = loader.loadTestsFromTestCase(test_class)
  49. subsuite = suite_class()
  50. # Only use addTest() to simplify testing a custom TestSuite.
  51. for test in tests:
  52. subsuite.addTest(test)
  53. suite.addTest(subsuite)
  54. return suite
  55. def make_test_suite(self, suite=None, suite_class=None):
  56. class Tests1(unittest.TestCase):
  57. def test1(self):
  58. pass
  59. def test2(self):
  60. pass
  61. class Tests2(unittest.TestCase):
  62. def test1(self):
  63. pass
  64. def test2(self):
  65. pass
  66. return self.build_test_suite(
  67. (Tests1, Tests2),
  68. suite=suite,
  69. suite_class=suite_class,
  70. )
  71. def assertTestNames(self, tests, expected):
  72. # Each test.id() has a form like the following:
  73. # "test_runner.tests.IterTestCasesTests.test_iter_test_cases.<locals>.Tests1.test1".
  74. # It suffices to check only the last two parts.
  75. names = [".".join(test.id().split(".")[-2:]) for test in tests]
  76. self.assertEqual(names, expected)
  77. def test_iter_test_cases_basic(self):
  78. suite = self.make_test_suite()
  79. tests = iter_test_cases(suite)
  80. self.assertTestNames(
  81. tests,
  82. expected=[
  83. "Tests1.test1",
  84. "Tests1.test2",
  85. "Tests2.test1",
  86. "Tests2.test2",
  87. ],
  88. )
  89. def test_iter_test_cases_string_input(self):
  90. msg = (
  91. "Test 'a' must be a test case or test suite not string (was found "
  92. "in 'abc')."
  93. )
  94. with self.assertRaisesMessage(TypeError, msg):
  95. list(iter_test_cases("abc"))
  96. def test_iter_test_cases_iterable_of_tests(self):
  97. class Tests(unittest.TestCase):
  98. def test1(self):
  99. pass
  100. def test2(self):
  101. pass
  102. tests = list(unittest.defaultTestLoader.loadTestsFromTestCase(Tests))
  103. actual_tests = iter_test_cases(tests)
  104. self.assertTestNames(
  105. actual_tests,
  106. expected=[
  107. "Tests.test1",
  108. "Tests.test2",
  109. ],
  110. )
  111. def test_iter_test_cases_custom_test_suite_class(self):
  112. suite = self.make_test_suite(suite_class=MySuite)
  113. tests = iter_test_cases(suite)
  114. self.assertTestNames(
  115. tests,
  116. expected=[
  117. "Tests1.test1",
  118. "Tests1.test2",
  119. "Tests2.test1",
  120. "Tests2.test2",
  121. ],
  122. )
  123. def test_iter_test_cases_mixed_test_suite_classes(self):
  124. suite = self.make_test_suite(suite=MySuite())
  125. child_suite = list(suite)[0]
  126. self.assertNotIsInstance(child_suite, MySuite)
  127. tests = list(iter_test_cases(suite))
  128. self.assertEqual(len(tests), 4)
  129. self.assertNotIsInstance(tests[0], unittest.TestSuite)
  130. def make_tests(self):
  131. """Return an iterable of tests."""
  132. suite = self.make_test_suite()
  133. return list(iter_test_cases(suite))
  134. def test_shuffle_tests(self):
  135. tests = self.make_tests()
  136. # Choose a seed that shuffles both the classes and methods.
  137. shuffler = Shuffler(seed=9)
  138. shuffled_tests = shuffle_tests(tests, shuffler)
  139. self.assertIsInstance(shuffled_tests, collections.abc.Iterator)
  140. self.assertTestNames(
  141. shuffled_tests,
  142. expected=[
  143. "Tests2.test1",
  144. "Tests2.test2",
  145. "Tests1.test2",
  146. "Tests1.test1",
  147. ],
  148. )
  149. def test_reorder_test_bin_no_arguments(self):
  150. tests = self.make_tests()
  151. reordered_tests = reorder_test_bin(tests)
  152. self.assertIsInstance(reordered_tests, collections.abc.Iterator)
  153. self.assertTestNames(
  154. reordered_tests,
  155. expected=[
  156. "Tests1.test1",
  157. "Tests1.test2",
  158. "Tests2.test1",
  159. "Tests2.test2",
  160. ],
  161. )
  162. def test_reorder_test_bin_reverse(self):
  163. tests = self.make_tests()
  164. reordered_tests = reorder_test_bin(tests, reverse=True)
  165. self.assertIsInstance(reordered_tests, collections.abc.Iterator)
  166. self.assertTestNames(
  167. reordered_tests,
  168. expected=[
  169. "Tests2.test2",
  170. "Tests2.test1",
  171. "Tests1.test2",
  172. "Tests1.test1",
  173. ],
  174. )
  175. def test_reorder_test_bin_random(self):
  176. tests = self.make_tests()
  177. # Choose a seed that shuffles both the classes and methods.
  178. shuffler = Shuffler(seed=9)
  179. reordered_tests = reorder_test_bin(tests, shuffler=shuffler)
  180. self.assertIsInstance(reordered_tests, collections.abc.Iterator)
  181. self.assertTestNames(
  182. reordered_tests,
  183. expected=[
  184. "Tests2.test1",
  185. "Tests2.test2",
  186. "Tests1.test2",
  187. "Tests1.test1",
  188. ],
  189. )
  190. def test_reorder_test_bin_random_and_reverse(self):
  191. tests = self.make_tests()
  192. # Choose a seed that shuffles both the classes and methods.
  193. shuffler = Shuffler(seed=9)
  194. reordered_tests = reorder_test_bin(tests, shuffler=shuffler, reverse=True)
  195. self.assertIsInstance(reordered_tests, collections.abc.Iterator)
  196. self.assertTestNames(
  197. reordered_tests,
  198. expected=[
  199. "Tests1.test1",
  200. "Tests1.test2",
  201. "Tests2.test2",
  202. "Tests2.test1",
  203. ],
  204. )
  205. def test_reorder_tests_same_type_consecutive(self):
  206. """Tests of the same type are made consecutive."""
  207. tests = self.make_tests()
  208. # Move the last item to the front.
  209. tests.insert(0, tests.pop())
  210. self.assertTestNames(
  211. tests,
  212. expected=[
  213. "Tests2.test2",
  214. "Tests1.test1",
  215. "Tests1.test2",
  216. "Tests2.test1",
  217. ],
  218. )
  219. reordered_tests = reorder_tests(tests, classes=[])
  220. self.assertTestNames(
  221. reordered_tests,
  222. expected=[
  223. "Tests2.test2",
  224. "Tests2.test1",
  225. "Tests1.test1",
  226. "Tests1.test2",
  227. ],
  228. )
  229. def test_reorder_tests_random(self):
  230. tests = self.make_tests()
  231. # Choose a seed that shuffles both the classes and methods.
  232. shuffler = Shuffler(seed=9)
  233. reordered_tests = reorder_tests(tests, classes=[], shuffler=shuffler)
  234. self.assertIsInstance(reordered_tests, collections.abc.Iterator)
  235. self.assertTestNames(
  236. reordered_tests,
  237. expected=[
  238. "Tests2.test1",
  239. "Tests2.test2",
  240. "Tests1.test2",
  241. "Tests1.test1",
  242. ],
  243. )
  244. def test_reorder_tests_random_mixed_classes(self):
  245. tests = self.make_tests()
  246. # Move the last item to the front.
  247. tests.insert(0, tests.pop())
  248. shuffler = Shuffler(seed=9)
  249. self.assertTestNames(
  250. tests,
  251. expected=[
  252. "Tests2.test2",
  253. "Tests1.test1",
  254. "Tests1.test2",
  255. "Tests2.test1",
  256. ],
  257. )
  258. reordered_tests = reorder_tests(tests, classes=[], shuffler=shuffler)
  259. self.assertTestNames(
  260. reordered_tests,
  261. expected=[
  262. "Tests2.test1",
  263. "Tests2.test2",
  264. "Tests1.test2",
  265. "Tests1.test1",
  266. ],
  267. )
  268. def test_reorder_tests_reverse_with_duplicates(self):
  269. class Tests1(unittest.TestCase):
  270. def test1(self):
  271. pass
  272. class Tests2(unittest.TestCase):
  273. def test2(self):
  274. pass
  275. def test3(self):
  276. pass
  277. suite = self.build_test_suite((Tests1, Tests2))
  278. subsuite = list(suite)[0]
  279. suite.addTest(subsuite)
  280. tests = list(iter_test_cases(suite))
  281. self.assertTestNames(
  282. tests,
  283. expected=[
  284. "Tests1.test1",
  285. "Tests2.test2",
  286. "Tests2.test3",
  287. "Tests1.test1",
  288. ],
  289. )
  290. reordered_tests = reorder_tests(tests, classes=[])
  291. self.assertTestNames(
  292. reordered_tests,
  293. expected=[
  294. "Tests1.test1",
  295. "Tests2.test2",
  296. "Tests2.test3",
  297. ],
  298. )
  299. reordered_tests = reorder_tests(tests, classes=[], reverse=True)
  300. self.assertTestNames(
  301. reordered_tests,
  302. expected=[
  303. "Tests2.test3",
  304. "Tests2.test2",
  305. "Tests1.test1",
  306. ],
  307. )
  308. class DependencyOrderingTests(unittest.TestCase):
  309. def test_simple_dependencies(self):
  310. raw = [
  311. ("s1", ("s1_db", ["alpha"])),
  312. ("s2", ("s2_db", ["bravo"])),
  313. ("s3", ("s3_db", ["charlie"])),
  314. ]
  315. dependencies = {
  316. "alpha": ["charlie"],
  317. "bravo": ["charlie"],
  318. }
  319. ordered = dependency_ordered(raw, dependencies=dependencies)
  320. ordered_sigs = [sig for sig, value in ordered]
  321. self.assertIn("s1", ordered_sigs)
  322. self.assertIn("s2", ordered_sigs)
  323. self.assertIn("s3", ordered_sigs)
  324. self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s1"))
  325. self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s2"))
  326. def test_chained_dependencies(self):
  327. raw = [
  328. ("s1", ("s1_db", ["alpha"])),
  329. ("s2", ("s2_db", ["bravo"])),
  330. ("s3", ("s3_db", ["charlie"])),
  331. ]
  332. dependencies = {
  333. "alpha": ["bravo"],
  334. "bravo": ["charlie"],
  335. }
  336. ordered = dependency_ordered(raw, dependencies=dependencies)
  337. ordered_sigs = [sig for sig, value in ordered]
  338. self.assertIn("s1", ordered_sigs)
  339. self.assertIn("s2", ordered_sigs)
  340. self.assertIn("s3", ordered_sigs)
  341. # Explicit dependencies
  342. self.assertLess(ordered_sigs.index("s2"), ordered_sigs.index("s1"))
  343. self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s2"))
  344. # Implied dependencies
  345. self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s1"))
  346. def test_multiple_dependencies(self):
  347. raw = [
  348. ("s1", ("s1_db", ["alpha"])),
  349. ("s2", ("s2_db", ["bravo"])),
  350. ("s3", ("s3_db", ["charlie"])),
  351. ("s4", ("s4_db", ["delta"])),
  352. ]
  353. dependencies = {
  354. "alpha": ["bravo", "delta"],
  355. "bravo": ["charlie"],
  356. "delta": ["charlie"],
  357. }
  358. ordered = dependency_ordered(raw, dependencies=dependencies)
  359. ordered_sigs = [sig for sig, aliases in ordered]
  360. self.assertIn("s1", ordered_sigs)
  361. self.assertIn("s2", ordered_sigs)
  362. self.assertIn("s3", ordered_sigs)
  363. self.assertIn("s4", ordered_sigs)
  364. # Explicit dependencies
  365. self.assertLess(ordered_sigs.index("s2"), ordered_sigs.index("s1"))
  366. self.assertLess(ordered_sigs.index("s4"), ordered_sigs.index("s1"))
  367. self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s2"))
  368. self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s4"))
  369. # Implicit dependencies
  370. self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s1"))
  371. def test_circular_dependencies(self):
  372. raw = [
  373. ("s1", ("s1_db", ["alpha"])),
  374. ("s2", ("s2_db", ["bravo"])),
  375. ]
  376. dependencies = {
  377. "bravo": ["alpha"],
  378. "alpha": ["bravo"],
  379. }
  380. with self.assertRaises(ImproperlyConfigured):
  381. dependency_ordered(raw, dependencies=dependencies)
  382. def test_own_alias_dependency(self):
  383. raw = [("s1", ("s1_db", ["alpha", "bravo"]))]
  384. dependencies = {"alpha": ["bravo"]}
  385. with self.assertRaises(ImproperlyConfigured):
  386. dependency_ordered(raw, dependencies=dependencies)
  387. # reordering aliases shouldn't matter
  388. raw = [("s1", ("s1_db", ["bravo", "alpha"]))]
  389. with self.assertRaises(ImproperlyConfigured):
  390. dependency_ordered(raw, dependencies=dependencies)
  391. class MockTestRunner:
  392. def __init__(self, *args, **kwargs):
  393. if parallel := kwargs.get("parallel"):
  394. sys.stderr.write(f"parallel={parallel}")
  395. if durations := kwargs.get("durations"):
  396. sys.stderr.write(f"durations={durations}")
  397. MockTestRunner.run_tests = mock.Mock(return_value=[])
  398. class ManageCommandTests(unittest.TestCase):
  399. def test_custom_test_runner(self):
  400. call_command("test", "sites", testrunner="test_runner.tests.MockTestRunner")
  401. MockTestRunner.run_tests.assert_called_with(("sites",))
  402. def test_bad_test_runner(self):
  403. with self.assertRaises(AttributeError):
  404. call_command("test", "sites", testrunner="test_runner.NonexistentRunner")
  405. def test_time_recorded(self):
  406. with captured_stderr() as stderr:
  407. call_command(
  408. "test",
  409. "--timing",
  410. "sites",
  411. testrunner="test_runner.tests.MockTestRunner",
  412. )
  413. self.assertIn("Total run took", stderr.getvalue())
  414. def test_durations(self):
  415. with captured_stderr() as stderr:
  416. call_command(
  417. "test",
  418. "--durations=10",
  419. "sites",
  420. testrunner="test_runner.tests.MockTestRunner",
  421. )
  422. self.assertIn("durations=10", stderr.getvalue())
  423. # Isolate from the real environment.
  424. @mock.patch.dict(os.environ, {}, clear=True)
  425. @mock.patch.object(multiprocessing, "cpu_count", return_value=12)
  426. class ManageCommandParallelTests(SimpleTestCase):
  427. @mock.patch.object(multiprocessing, "get_start_method", return_value="fork")
  428. def test_parallel_default(self, *mocked_objects):
  429. with captured_stderr() as stderr:
  430. call_command(
  431. "test",
  432. "--parallel",
  433. testrunner="test_runner.tests.MockTestRunner",
  434. )
  435. self.assertIn("parallel=12", stderr.getvalue())
  436. @mock.patch.object(multiprocessing, "get_start_method", return_value="fork")
  437. def test_parallel_auto(self, *mocked_objects):
  438. with captured_stderr() as stderr:
  439. call_command(
  440. "test",
  441. "--parallel=auto",
  442. testrunner="test_runner.tests.MockTestRunner",
  443. )
  444. self.assertIn("parallel=12", stderr.getvalue())
  445. def test_no_parallel(self, *mocked_objects):
  446. with captured_stderr() as stderr:
  447. call_command("test", testrunner="test_runner.tests.MockTestRunner")
  448. # Parallel is disabled by default.
  449. self.assertEqual(stderr.getvalue(), "")
  450. @mock.patch.object(multiprocessing, "get_start_method", return_value="spawn")
  451. def test_parallel_spawn(self, *mocked_objects):
  452. with captured_stderr() as stderr:
  453. call_command(
  454. "test",
  455. "--parallel=auto",
  456. testrunner="test_runner.tests.MockTestRunner",
  457. )
  458. self.assertIn("parallel=1", stderr.getvalue())
  459. @mock.patch.object(multiprocessing, "get_start_method", return_value="spawn")
  460. def test_no_parallel_spawn(self, *mocked_objects):
  461. with captured_stderr() as stderr:
  462. call_command(
  463. "test",
  464. testrunner="test_runner.tests.MockTestRunner",
  465. )
  466. self.assertEqual(stderr.getvalue(), "")
  467. @mock.patch.dict(os.environ, {"DJANGO_TEST_PROCESSES": "7"})
  468. @mock.patch.object(multiprocessing, "get_start_method", return_value="fork")
  469. def test_no_parallel_django_test_processes_env(self, *mocked_objects):
  470. with captured_stderr() as stderr:
  471. call_command("test", testrunner="test_runner.tests.MockTestRunner")
  472. self.assertEqual(stderr.getvalue(), "")
  473. @mock.patch.dict(os.environ, {"DJANGO_TEST_PROCESSES": "invalid"})
  474. @mock.patch.object(multiprocessing, "get_start_method", return_value="fork")
  475. def test_django_test_processes_env_non_int(self, *mocked_objects):
  476. with self.assertRaises(ValueError):
  477. call_command(
  478. "test",
  479. "--parallel",
  480. testrunner="test_runner.tests.MockTestRunner",
  481. )
  482. @mock.patch.dict(os.environ, {"DJANGO_TEST_PROCESSES": "7"})
  483. @mock.patch.object(multiprocessing, "get_start_method", return_value="fork")
  484. def test_django_test_processes_parallel_default(self, *mocked_objects):
  485. for parallel in ["--parallel", "--parallel=auto"]:
  486. with self.subTest(parallel=parallel):
  487. with captured_stderr() as stderr:
  488. call_command(
  489. "test",
  490. parallel,
  491. testrunner="test_runner.tests.MockTestRunner",
  492. )
  493. self.assertIn("parallel=7", stderr.getvalue())
  494. class CustomTestRunnerOptionsSettingsTests(AdminScriptTestCase):
  495. """
  496. Custom runners can add command line arguments. The runner is specified
  497. through a settings file.
  498. """
  499. def setUp(self):
  500. super().setUp()
  501. settings = {
  502. "TEST_RUNNER": "'test_runner.runner.CustomOptionsTestRunner'",
  503. }
  504. self.write_settings("settings.py", sdict=settings)
  505. def test_default_options(self):
  506. args = ["test", "--settings=test_project.settings"]
  507. out, err = self.run_django_admin(args)
  508. self.assertNoOutput(err)
  509. self.assertOutput(out, "1:2:3")
  510. def test_default_and_given_options(self):
  511. args = ["test", "--settings=test_project.settings", "--option_b=foo"]
  512. out, err = self.run_django_admin(args)
  513. self.assertNoOutput(err)
  514. self.assertOutput(out, "1:foo:3")
  515. def test_option_name_and_value_separated(self):
  516. args = ["test", "--settings=test_project.settings", "--option_b", "foo"]
  517. out, err = self.run_django_admin(args)
  518. self.assertNoOutput(err)
  519. self.assertOutput(out, "1:foo:3")
  520. def test_all_options_given(self):
  521. args = [
  522. "test",
  523. "--settings=test_project.settings",
  524. "--option_a=bar",
  525. "--option_b=foo",
  526. "--option_c=31337",
  527. ]
  528. out, err = self.run_django_admin(args)
  529. self.assertNoOutput(err)
  530. self.assertOutput(out, "bar:foo:31337")
  531. class CustomTestRunnerOptionsCmdlineTests(AdminScriptTestCase):
  532. """
  533. Custom runners can add command line arguments when the runner is specified
  534. using --testrunner.
  535. """
  536. def setUp(self):
  537. super().setUp()
  538. self.write_settings("settings.py")
  539. def test_testrunner_option(self):
  540. args = [
  541. "test",
  542. "--testrunner",
  543. "test_runner.runner.CustomOptionsTestRunner",
  544. "--option_a=bar",
  545. "--option_b=foo",
  546. "--option_c=31337",
  547. ]
  548. out, err = self.run_django_admin(args, "test_project.settings")
  549. self.assertNoOutput(err)
  550. self.assertOutput(out, "bar:foo:31337")
  551. def test_testrunner_equals(self):
  552. args = [
  553. "test",
  554. "--testrunner=test_runner.runner.CustomOptionsTestRunner",
  555. "--option_a=bar",
  556. "--option_b=foo",
  557. "--option_c=31337",
  558. ]
  559. out, err = self.run_django_admin(args, "test_project.settings")
  560. self.assertNoOutput(err)
  561. self.assertOutput(out, "bar:foo:31337")
  562. def test_no_testrunner(self):
  563. args = ["test", "--testrunner"]
  564. out, err = self.run_django_admin(args, "test_project.settings")
  565. self.assertIn("usage", err)
  566. self.assertNotIn("Traceback", err)
  567. self.assertNoOutput(out)
  568. class NoInitializeSuiteTestRunnerTests(SimpleTestCase):
  569. @mock.patch.object(multiprocessing, "get_start_method", return_value="spawn")
  570. @mock.patch(
  571. "django.test.runner.ParallelTestSuite.initialize_suite",
  572. side_effect=Exception("initialize_suite() is called."),
  573. )
  574. def test_no_initialize_suite_test_runner(self, *mocked_objects):
  575. """
  576. The test suite's initialize_suite() method must always be called when
  577. using spawn. It cannot rely on a test runner implementation.
  578. """
  579. class NoInitializeSuiteTestRunner(DiscoverRunner):
  580. def setup_test_environment(self, **kwargs):
  581. return
  582. def setup_databases(self, **kwargs):
  583. return
  584. def run_checks(self, databases):
  585. return
  586. def teardown_databases(self, old_config, **kwargs):
  587. return
  588. def teardown_test_environment(self, **kwargs):
  589. return
  590. def run_suite(self, suite, **kwargs):
  591. kwargs = self.get_test_runner_kwargs()
  592. runner = self.test_runner(**kwargs)
  593. return runner.run(suite)
  594. with self.assertRaisesMessage(Exception, "initialize_suite() is called."):
  595. runner = NoInitializeSuiteTestRunner(
  596. verbosity=0, interactive=False, parallel=2
  597. )
  598. runner.run_tests(
  599. [
  600. "test_runner_apps.sample.tests_sample.TestDjangoTestCase",
  601. "test_runner_apps.simple.tests",
  602. ]
  603. )
  604. class TestRunnerInitializerTests(SimpleTestCase):
  605. # Raise an exception to don't actually run tests.
  606. @mock.patch.object(
  607. multiprocessing, "Pool", side_effect=Exception("multiprocessing.Pool()")
  608. )
  609. def test_no_initialize_suite_test_runner(self, mocked_pool):
  610. class StubTestRunner(DiscoverRunner):
  611. def setup_test_environment(self, **kwargs):
  612. return
  613. def setup_databases(self, **kwargs):
  614. return
  615. def run_checks(self, databases):
  616. return
  617. def teardown_databases(self, old_config, **kwargs):
  618. return
  619. def teardown_test_environment(self, **kwargs):
  620. return
  621. def run_suite(self, suite, **kwargs):
  622. kwargs = self.get_test_runner_kwargs()
  623. runner = self.test_runner(**kwargs)
  624. return runner.run(suite)
  625. runner = StubTestRunner(
  626. verbosity=0, interactive=False, parallel=2, debug_mode=True
  627. )
  628. with self.assertRaisesMessage(Exception, "multiprocessing.Pool()"):
  629. runner.run_tests(
  630. [
  631. "test_runner_apps.sample.tests_sample.TestDjangoTestCase",
  632. "test_runner_apps.simple.tests",
  633. ]
  634. )
  635. # Initializer must be a function.
  636. self.assertIs(mocked_pool.call_args.kwargs["initializer"], _init_worker)
  637. initargs = mocked_pool.call_args.kwargs["initargs"]
  638. self.assertEqual(len(initargs), 7)
  639. self.assertEqual(initargs[5], True) # debug_mode
  640. self.assertEqual(initargs[6], {db.DEFAULT_DB_ALIAS}) # Used database aliases.
  641. class Ticket17477RegressionTests(AdminScriptTestCase):
  642. def setUp(self):
  643. super().setUp()
  644. self.write_settings("settings.py")
  645. def test_ticket_17477(self):
  646. """'manage.py help test' works after r16352."""
  647. args = ["help", "test"]
  648. out, err = self.run_manage(args)
  649. self.assertNoOutput(err)
  650. class SQLiteInMemoryTestDbs(TransactionTestCase):
  651. available_apps = ["test_runner"]
  652. databases = {"default", "other"}
  653. @unittest.skipUnless(
  654. all(db.connections[conn].vendor == "sqlite" for conn in db.connections),
  655. "This is an sqlite-specific issue",
  656. )
  657. def test_transaction_support(self):
  658. # Assert connections mocking is appropriately applied by preventing
  659. # any attempts at calling create_test_db on the global connection
  660. # objects.
  661. for connection in db.connections.all():
  662. create_test_db = mock.patch.object(
  663. connection.creation,
  664. "create_test_db",
  665. side_effect=AssertionError(
  666. "Global connection object shouldn't be manipulated."
  667. ),
  668. )
  669. create_test_db.start()
  670. self.addCleanup(create_test_db.stop)
  671. for option_key, option_value in (
  672. ("NAME", ":memory:"),
  673. ("TEST", {"NAME": ":memory:"}),
  674. ):
  675. tested_connections = db.ConnectionHandler(
  676. {
  677. "default": {
  678. "ENGINE": "django.db.backends.sqlite3",
  679. option_key: option_value,
  680. },
  681. "other": {
  682. "ENGINE": "django.db.backends.sqlite3",
  683. option_key: option_value,
  684. },
  685. }
  686. )
  687. with mock.patch("django.test.utils.connections", new=tested_connections):
  688. other = tested_connections["other"]
  689. try:
  690. new_test_connections = DiscoverRunner(verbosity=0).setup_databases()
  691. msg = (
  692. f"DATABASES setting '{option_key}' option set to sqlite3's "
  693. "':memory:' value shouldn't interfere with transaction support "
  694. "detection."
  695. )
  696. # Transaction support is properly initialized for the
  697. # 'other' DB.
  698. self.assertTrue(other.features.supports_transactions, msg)
  699. # And all the DBs report that they support transactions.
  700. self.assertTrue(connections_support_transactions(), msg)
  701. finally:
  702. for test_connection, _, _ in new_test_connections:
  703. test_connection._close()
  704. class DummyBackendTest(unittest.TestCase):
  705. def test_setup_databases(self):
  706. """
  707. setup_databases() doesn't fail with dummy database backend.
  708. """
  709. tested_connections = db.ConnectionHandler({})
  710. with mock.patch("django.test.utils.connections", new=tested_connections):
  711. runner_instance = DiscoverRunner(verbosity=0)
  712. old_config = runner_instance.setup_databases()
  713. runner_instance.teardown_databases(old_config)
  714. class AliasedDefaultTestSetupTest(unittest.TestCase):
  715. def test_setup_aliased_default_database(self):
  716. """
  717. setup_databases() doesn't fail when 'default' is aliased
  718. """
  719. tested_connections = db.ConnectionHandler(
  720. {"default": {"NAME": "dummy"}, "aliased": {"NAME": "dummy"}}
  721. )
  722. with mock.patch("django.test.utils.connections", new=tested_connections):
  723. runner_instance = DiscoverRunner(verbosity=0)
  724. old_config = runner_instance.setup_databases()
  725. runner_instance.teardown_databases(old_config)
  726. class SetupDatabasesTests(unittest.TestCase):
  727. def setUp(self):
  728. self.runner_instance = DiscoverRunner(verbosity=0)
  729. def test_setup_aliased_databases(self):
  730. tested_connections = db.ConnectionHandler(
  731. {
  732. "default": {
  733. "ENGINE": "django.db.backends.dummy",
  734. "NAME": "dbname",
  735. },
  736. "other": {
  737. "ENGINE": "django.db.backends.dummy",
  738. "NAME": "dbname",
  739. },
  740. }
  741. )
  742. with mock.patch(
  743. "django.db.backends.dummy.base.DatabaseWrapper.creation_class"
  744. ) as mocked_db_creation:
  745. with mock.patch("django.test.utils.connections", new=tested_connections):
  746. old_config = self.runner_instance.setup_databases()
  747. self.runner_instance.teardown_databases(old_config)
  748. mocked_db_creation.return_value.destroy_test_db.assert_called_once_with(
  749. "dbname", 0, False
  750. )
  751. def test_setup_test_database_aliases(self):
  752. """
  753. The default database must be the first because data migrations
  754. use the default alias by default.
  755. """
  756. tested_connections = db.ConnectionHandler(
  757. {
  758. "other": {
  759. "ENGINE": "django.db.backends.dummy",
  760. "NAME": "dbname",
  761. },
  762. "default": {
  763. "ENGINE": "django.db.backends.dummy",
  764. "NAME": "dbname",
  765. },
  766. }
  767. )
  768. with mock.patch("django.test.utils.connections", new=tested_connections):
  769. test_databases, _ = get_unique_databases_and_mirrors()
  770. self.assertEqual(
  771. test_databases,
  772. {
  773. ("", "", "django.db.backends.dummy", "test_dbname"): (
  774. "dbname",
  775. ["default", "other"],
  776. ),
  777. },
  778. )
  779. def test_destroy_test_db_restores_db_name(self):
  780. tested_connections = db.ConnectionHandler(
  781. {
  782. "default": {
  783. "ENGINE": settings.DATABASES[db.DEFAULT_DB_ALIAS]["ENGINE"],
  784. "NAME": "xxx_test_database",
  785. },
  786. }
  787. )
  788. # Using the real current name as old_name to not mess with the test suite.
  789. old_name = settings.DATABASES[db.DEFAULT_DB_ALIAS]["NAME"]
  790. with mock.patch("django.db.connections", new=tested_connections):
  791. tested_connections["default"].creation.destroy_test_db(
  792. old_name, verbosity=0, keepdb=True
  793. )
  794. self.assertEqual(
  795. tested_connections["default"].settings_dict["NAME"], old_name
  796. )
  797. def test_serialization(self):
  798. tested_connections = db.ConnectionHandler(
  799. {
  800. "default": {
  801. "ENGINE": "django.db.backends.dummy",
  802. },
  803. }
  804. )
  805. with mock.patch(
  806. "django.db.backends.dummy.base.DatabaseWrapper.creation_class"
  807. ) as mocked_db_creation:
  808. with mock.patch("django.test.utils.connections", new=tested_connections):
  809. self.runner_instance.setup_databases()
  810. mocked_db_creation.return_value.create_test_db.assert_called_once_with(
  811. verbosity=0, autoclobber=False, serialize=False, keepdb=False
  812. )
  813. mocked_db_creation.return_value.serialize_db_to_string.assert_called_once_with()
  814. @skipUnlessDBFeature("supports_sequence_reset")
  815. class AutoIncrementResetTest(TransactionTestCase):
  816. """
  817. Creating the same models in different test methods receive the same PK
  818. values since the sequences are reset before each test method.
  819. """
  820. available_apps = ["test_runner"]
  821. reset_sequences = True
  822. def _test(self):
  823. # Regular model
  824. p = Person.objects.create(first_name="Jack", last_name="Smith")
  825. self.assertEqual(p.pk, 1)
  826. # Auto-created many-to-many through model
  827. p.friends.add(Person.objects.create(first_name="Jacky", last_name="Smith"))
  828. self.assertEqual(p.friends.through.objects.first().pk, 1)
  829. # Many-to-many through model
  830. b = B.objects.create()
  831. t = Through.objects.create(person=p, b=b)
  832. self.assertEqual(t.pk, 1)
  833. def test_autoincrement_reset1(self):
  834. self._test()
  835. def test_autoincrement_reset2(self):
  836. self._test()
  837. class EmptyDefaultDatabaseTest(unittest.TestCase):
  838. def test_empty_default_database(self):
  839. """
  840. An empty default database in settings does not raise an ImproperlyConfigured
  841. error when running a unit test that does not use a database.
  842. """
  843. tested_connections = db.ConnectionHandler({"default": {}})
  844. with mock.patch("django.db.connections", new=tested_connections):
  845. connection = tested_connections[db.utils.DEFAULT_DB_ALIAS]
  846. self.assertEqual(
  847. connection.settings_dict["ENGINE"], "django.db.backends.dummy"
  848. )
  849. connections_support_transactions()
  850. class RunTestsExceptionHandlingTests(unittest.TestCase):
  851. def test_run_checks_raises(self):
  852. """
  853. Teardown functions are run when run_checks() raises SystemCheckError.
  854. """
  855. with (
  856. mock.patch("django.test.runner.DiscoverRunner.setup_test_environment"),
  857. mock.patch("django.test.runner.DiscoverRunner.setup_databases"),
  858. mock.patch("django.test.runner.DiscoverRunner.build_suite"),
  859. mock.patch(
  860. "django.test.runner.DiscoverRunner.run_checks",
  861. side_effect=SystemCheckError,
  862. ),
  863. mock.patch(
  864. "django.test.runner.DiscoverRunner.teardown_databases"
  865. ) as teardown_databases,
  866. mock.patch(
  867. "django.test.runner.DiscoverRunner.teardown_test_environment"
  868. ) as teardown_test_environment,
  869. ):
  870. runner = DiscoverRunner(verbosity=0, interactive=False)
  871. with self.assertRaises(SystemCheckError):
  872. runner.run_tests(
  873. ["test_runner_apps.sample.tests_sample.TestDjangoTestCase"]
  874. )
  875. self.assertTrue(teardown_databases.called)
  876. self.assertTrue(teardown_test_environment.called)
  877. def test_run_checks_raises_and_teardown_raises(self):
  878. """
  879. SystemCheckError is surfaced when run_checks() raises SystemCheckError
  880. and teardown databases() raises ValueError.
  881. """
  882. with (
  883. mock.patch("django.test.runner.DiscoverRunner.setup_test_environment"),
  884. mock.patch("django.test.runner.DiscoverRunner.setup_databases"),
  885. mock.patch("django.test.runner.DiscoverRunner.build_suite"),
  886. mock.patch(
  887. "django.test.runner.DiscoverRunner.run_checks",
  888. side_effect=SystemCheckError,
  889. ),
  890. mock.patch(
  891. "django.test.runner.DiscoverRunner.teardown_databases",
  892. side_effect=ValueError,
  893. ) as teardown_databases,
  894. mock.patch(
  895. "django.test.runner.DiscoverRunner.teardown_test_environment"
  896. ) as teardown_test_environment,
  897. ):
  898. runner = DiscoverRunner(verbosity=0, interactive=False)
  899. with self.assertRaises(SystemCheckError):
  900. runner.run_tests(
  901. ["test_runner_apps.sample.tests_sample.TestDjangoTestCase"]
  902. )
  903. self.assertTrue(teardown_databases.called)
  904. self.assertFalse(teardown_test_environment.called)
  905. def test_run_checks_passes_and_teardown_raises(self):
  906. """
  907. Exceptions on teardown are surfaced if no exceptions happen during
  908. run_checks().
  909. """
  910. with (
  911. mock.patch("django.test.runner.DiscoverRunner.setup_test_environment"),
  912. mock.patch("django.test.runner.DiscoverRunner.setup_databases"),
  913. mock.patch("django.test.runner.DiscoverRunner.build_suite"),
  914. mock.patch("django.test.runner.DiscoverRunner.run_checks"),
  915. mock.patch(
  916. "django.test.runner.DiscoverRunner.teardown_databases",
  917. side_effect=ValueError,
  918. ) as teardown_databases,
  919. mock.patch(
  920. "django.test.runner.DiscoverRunner.teardown_test_environment"
  921. ) as teardown_test_environment,
  922. ):
  923. runner = DiscoverRunner(verbosity=0, interactive=False)
  924. with self.assertRaises(ValueError):
  925. # Suppress the output when running TestDjangoTestCase.
  926. with mock.patch("sys.stderr"):
  927. runner.run_tests(
  928. ["test_runner_apps.sample.tests_sample.TestDjangoTestCase"]
  929. )
  930. self.assertTrue(teardown_databases.called)
  931. self.assertFalse(teardown_test_environment.called)