mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1075 lines
		
	
	
		
			37 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			1075 lines
		
	
	
		
			37 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """
 | |
| Tests for django test runner
 | |
| """
 | |
| 
 | |
| import collections.abc
 | |
| import multiprocessing
 | |
| import os
 | |
| import sys
 | |
| import unittest
 | |
| from unittest import mock
 | |
| 
 | |
| from admin_scripts.tests import AdminScriptTestCase
 | |
| 
 | |
| from django import db
 | |
| from django.conf import settings
 | |
| from django.core.exceptions import ImproperlyConfigured
 | |
| from django.core.management import call_command
 | |
| from django.core.management.base import CommandError, SystemCheckError
 | |
| from django.test import SimpleTestCase, TransactionTestCase, skipUnlessDBFeature
 | |
| from django.test.runner import (
 | |
|     DiscoverRunner,
 | |
|     Shuffler,
 | |
|     _init_worker,
 | |
|     reorder_test_bin,
 | |
|     reorder_tests,
 | |
|     shuffle_tests,
 | |
| )
 | |
| from django.test.testcases import connections_support_transactions
 | |
| from django.test.utils import (
 | |
|     captured_stderr,
 | |
|     dependency_ordered,
 | |
|     get_unique_databases_and_mirrors,
 | |
|     iter_test_cases,
 | |
| )
 | |
| from django.utils.version import PY312
 | |
| 
 | |
| from .models import B, Person, Through
 | |
| 
 | |
| 
 | |
| class MySuite:
 | |
|     def __init__(self):
 | |
|         self.tests = []
 | |
| 
 | |
|     def addTest(self, test):
 | |
|         self.tests.append(test)
 | |
| 
 | |
|     def __iter__(self):
 | |
|         yield from self.tests
 | |
| 
 | |
| 
 | |
| class TestSuiteTests(SimpleTestCase):
 | |
|     def build_test_suite(self, test_classes, suite=None, suite_class=None):
 | |
|         if suite_class is None:
 | |
|             suite_class = unittest.TestSuite
 | |
|         if suite is None:
 | |
|             suite = suite_class()
 | |
| 
 | |
|         loader = unittest.defaultTestLoader
 | |
|         for test_class in test_classes:
 | |
|             tests = loader.loadTestsFromTestCase(test_class)
 | |
|             subsuite = suite_class()
 | |
|             # Only use addTest() to simplify testing a custom TestSuite.
 | |
|             for test in tests:
 | |
|                 subsuite.addTest(test)
 | |
|             suite.addTest(subsuite)
 | |
| 
 | |
|         return suite
 | |
| 
 | |
|     def make_test_suite(self, suite=None, suite_class=None):
 | |
|         class Tests1(unittest.TestCase):
 | |
|             def test1(self):
 | |
|                 pass
 | |
| 
 | |
|             def test2(self):
 | |
|                 pass
 | |
| 
 | |
|         class Tests2(unittest.TestCase):
 | |
|             def test1(self):
 | |
|                 pass
 | |
| 
 | |
|             def test2(self):
 | |
|                 pass
 | |
| 
 | |
|         return self.build_test_suite(
 | |
|             (Tests1, Tests2),
 | |
|             suite=suite,
 | |
|             suite_class=suite_class,
 | |
|         )
 | |
| 
 | |
|     def assertTestNames(self, tests, expected):
 | |
|         # Each test.id() has a form like the following:
 | |
|         # "test_runner.tests.IterTestCasesTests.test_iter_test_cases.<locals>.Tests1.test1".
 | |
|         # It suffices to check only the last two parts.
 | |
|         names = [".".join(test.id().split(".")[-2:]) for test in tests]
 | |
|         self.assertEqual(names, expected)
 | |
| 
 | |
|     def test_iter_test_cases_basic(self):
 | |
|         suite = self.make_test_suite()
 | |
|         tests = iter_test_cases(suite)
 | |
|         self.assertTestNames(
 | |
|             tests,
 | |
|             expected=[
 | |
|                 "Tests1.test1",
 | |
|                 "Tests1.test2",
 | |
|                 "Tests2.test1",
 | |
|                 "Tests2.test2",
 | |
|             ],
 | |
|         )
 | |
| 
 | |
|     def test_iter_test_cases_string_input(self):
 | |
|         msg = (
 | |
|             "Test 'a' must be a test case or test suite not string (was found "
 | |
|             "in 'abc')."
 | |
|         )
 | |
|         with self.assertRaisesMessage(TypeError, msg):
 | |
|             list(iter_test_cases("abc"))
 | |
| 
 | |
|     def test_iter_test_cases_iterable_of_tests(self):
 | |
|         class Tests(unittest.TestCase):
 | |
|             def test1(self):
 | |
|                 pass
 | |
| 
 | |
|             def test2(self):
 | |
|                 pass
 | |
| 
 | |
|         tests = list(unittest.defaultTestLoader.loadTestsFromTestCase(Tests))
 | |
|         actual_tests = iter_test_cases(tests)
 | |
|         self.assertTestNames(
 | |
|             actual_tests,
 | |
|             expected=[
 | |
|                 "Tests.test1",
 | |
|                 "Tests.test2",
 | |
|             ],
 | |
|         )
 | |
| 
 | |
|     def test_iter_test_cases_custom_test_suite_class(self):
 | |
|         suite = self.make_test_suite(suite_class=MySuite)
 | |
|         tests = iter_test_cases(suite)
 | |
|         self.assertTestNames(
 | |
|             tests,
 | |
|             expected=[
 | |
|                 "Tests1.test1",
 | |
|                 "Tests1.test2",
 | |
|                 "Tests2.test1",
 | |
|                 "Tests2.test2",
 | |
|             ],
 | |
|         )
 | |
| 
 | |
|     def test_iter_test_cases_mixed_test_suite_classes(self):
 | |
|         suite = self.make_test_suite(suite=MySuite())
 | |
|         child_suite = list(suite)[0]
 | |
|         self.assertNotIsInstance(child_suite, MySuite)
 | |
|         tests = list(iter_test_cases(suite))
 | |
|         self.assertEqual(len(tests), 4)
 | |
|         self.assertNotIsInstance(tests[0], unittest.TestSuite)
 | |
| 
 | |
|     def make_tests(self):
 | |
|         """Return an iterable of tests."""
 | |
|         suite = self.make_test_suite()
 | |
|         return list(iter_test_cases(suite))
 | |
| 
 | |
|     def test_shuffle_tests(self):
 | |
|         tests = self.make_tests()
 | |
|         # Choose a seed that shuffles both the classes and methods.
 | |
|         shuffler = Shuffler(seed=9)
 | |
|         shuffled_tests = shuffle_tests(tests, shuffler)
 | |
|         self.assertIsInstance(shuffled_tests, collections.abc.Iterator)
 | |
|         self.assertTestNames(
 | |
|             shuffled_tests,
 | |
|             expected=[
 | |
|                 "Tests2.test1",
 | |
|                 "Tests2.test2",
 | |
|                 "Tests1.test2",
 | |
|                 "Tests1.test1",
 | |
|             ],
 | |
|         )
 | |
| 
 | |
|     def test_reorder_test_bin_no_arguments(self):
 | |
|         tests = self.make_tests()
 | |
|         reordered_tests = reorder_test_bin(tests)
 | |
|         self.assertIsInstance(reordered_tests, collections.abc.Iterator)
 | |
|         self.assertTestNames(
 | |
|             reordered_tests,
 | |
|             expected=[
 | |
|                 "Tests1.test1",
 | |
|                 "Tests1.test2",
 | |
|                 "Tests2.test1",
 | |
|                 "Tests2.test2",
 | |
|             ],
 | |
|         )
 | |
| 
 | |
|     def test_reorder_test_bin_reverse(self):
 | |
|         tests = self.make_tests()
 | |
|         reordered_tests = reorder_test_bin(tests, reverse=True)
 | |
|         self.assertIsInstance(reordered_tests, collections.abc.Iterator)
 | |
|         self.assertTestNames(
 | |
|             reordered_tests,
 | |
|             expected=[
 | |
|                 "Tests2.test2",
 | |
|                 "Tests2.test1",
 | |
|                 "Tests1.test2",
 | |
|                 "Tests1.test1",
 | |
|             ],
 | |
|         )
 | |
| 
 | |
|     def test_reorder_test_bin_random(self):
 | |
|         tests = self.make_tests()
 | |
|         # Choose a seed that shuffles both the classes and methods.
 | |
|         shuffler = Shuffler(seed=9)
 | |
|         reordered_tests = reorder_test_bin(tests, shuffler=shuffler)
 | |
|         self.assertIsInstance(reordered_tests, collections.abc.Iterator)
 | |
|         self.assertTestNames(
 | |
|             reordered_tests,
 | |
|             expected=[
 | |
|                 "Tests2.test1",
 | |
|                 "Tests2.test2",
 | |
|                 "Tests1.test2",
 | |
|                 "Tests1.test1",
 | |
|             ],
 | |
|         )
 | |
| 
 | |
|     def test_reorder_test_bin_random_and_reverse(self):
 | |
|         tests = self.make_tests()
 | |
|         # Choose a seed that shuffles both the classes and methods.
 | |
|         shuffler = Shuffler(seed=9)
 | |
|         reordered_tests = reorder_test_bin(tests, shuffler=shuffler, reverse=True)
 | |
|         self.assertIsInstance(reordered_tests, collections.abc.Iterator)
 | |
|         self.assertTestNames(
 | |
|             reordered_tests,
 | |
|             expected=[
 | |
|                 "Tests1.test1",
 | |
|                 "Tests1.test2",
 | |
|                 "Tests2.test2",
 | |
|                 "Tests2.test1",
 | |
|             ],
 | |
|         )
 | |
| 
 | |
|     def test_reorder_tests_same_type_consecutive(self):
 | |
|         """Tests of the same type are made consecutive."""
 | |
|         tests = self.make_tests()
 | |
|         # Move the last item to the front.
 | |
|         tests.insert(0, tests.pop())
 | |
|         self.assertTestNames(
 | |
|             tests,
 | |
|             expected=[
 | |
|                 "Tests2.test2",
 | |
|                 "Tests1.test1",
 | |
|                 "Tests1.test2",
 | |
|                 "Tests2.test1",
 | |
|             ],
 | |
|         )
 | |
|         reordered_tests = reorder_tests(tests, classes=[])
 | |
|         self.assertTestNames(
 | |
|             reordered_tests,
 | |
|             expected=[
 | |
|                 "Tests2.test2",
 | |
|                 "Tests2.test1",
 | |
|                 "Tests1.test1",
 | |
|                 "Tests1.test2",
 | |
|             ],
 | |
|         )
 | |
| 
 | |
|     def test_reorder_tests_random(self):
 | |
|         tests = self.make_tests()
 | |
|         # Choose a seed that shuffles both the classes and methods.
 | |
|         shuffler = Shuffler(seed=9)
 | |
|         reordered_tests = reorder_tests(tests, classes=[], shuffler=shuffler)
 | |
|         self.assertIsInstance(reordered_tests, collections.abc.Iterator)
 | |
|         self.assertTestNames(
 | |
|             reordered_tests,
 | |
|             expected=[
 | |
|                 "Tests2.test1",
 | |
|                 "Tests2.test2",
 | |
|                 "Tests1.test2",
 | |
|                 "Tests1.test1",
 | |
|             ],
 | |
|         )
 | |
| 
 | |
|     def test_reorder_tests_random_mixed_classes(self):
 | |
|         tests = self.make_tests()
 | |
|         # Move the last item to the front.
 | |
|         tests.insert(0, tests.pop())
 | |
|         shuffler = Shuffler(seed=9)
 | |
|         self.assertTestNames(
 | |
|             tests,
 | |
|             expected=[
 | |
|                 "Tests2.test2",
 | |
|                 "Tests1.test1",
 | |
|                 "Tests1.test2",
 | |
|                 "Tests2.test1",
 | |
|             ],
 | |
|         )
 | |
|         reordered_tests = reorder_tests(tests, classes=[], shuffler=shuffler)
 | |
|         self.assertTestNames(
 | |
|             reordered_tests,
 | |
|             expected=[
 | |
|                 "Tests2.test1",
 | |
|                 "Tests2.test2",
 | |
|                 "Tests1.test2",
 | |
|                 "Tests1.test1",
 | |
|             ],
 | |
|         )
 | |
| 
 | |
|     def test_reorder_tests_reverse_with_duplicates(self):
 | |
|         class Tests1(unittest.TestCase):
 | |
|             def test1(self):
 | |
|                 pass
 | |
| 
 | |
|         class Tests2(unittest.TestCase):
 | |
|             def test2(self):
 | |
|                 pass
 | |
| 
 | |
|             def test3(self):
 | |
|                 pass
 | |
| 
 | |
|         suite = self.build_test_suite((Tests1, Tests2))
 | |
|         subsuite = list(suite)[0]
 | |
|         suite.addTest(subsuite)
 | |
|         tests = list(iter_test_cases(suite))
 | |
|         self.assertTestNames(
 | |
|             tests,
 | |
|             expected=[
 | |
|                 "Tests1.test1",
 | |
|                 "Tests2.test2",
 | |
|                 "Tests2.test3",
 | |
|                 "Tests1.test1",
 | |
|             ],
 | |
|         )
 | |
|         reordered_tests = reorder_tests(tests, classes=[])
 | |
|         self.assertTestNames(
 | |
|             reordered_tests,
 | |
|             expected=[
 | |
|                 "Tests1.test1",
 | |
|                 "Tests2.test2",
 | |
|                 "Tests2.test3",
 | |
|             ],
 | |
|         )
 | |
|         reordered_tests = reorder_tests(tests, classes=[], reverse=True)
 | |
|         self.assertTestNames(
 | |
|             reordered_tests,
 | |
|             expected=[
 | |
|                 "Tests2.test3",
 | |
|                 "Tests2.test2",
 | |
|                 "Tests1.test1",
 | |
|             ],
 | |
|         )
 | |
| 
 | |
| 
 | |
| class DependencyOrderingTests(unittest.TestCase):
 | |
|     def test_simple_dependencies(self):
 | |
|         raw = [
 | |
|             ("s1", ("s1_db", ["alpha"])),
 | |
|             ("s2", ("s2_db", ["bravo"])),
 | |
|             ("s3", ("s3_db", ["charlie"])),
 | |
|         ]
 | |
|         dependencies = {
 | |
|             "alpha": ["charlie"],
 | |
|             "bravo": ["charlie"],
 | |
|         }
 | |
| 
 | |
|         ordered = dependency_ordered(raw, dependencies=dependencies)
 | |
|         ordered_sigs = [sig for sig, value in ordered]
 | |
| 
 | |
|         self.assertIn("s1", ordered_sigs)
 | |
|         self.assertIn("s2", ordered_sigs)
 | |
|         self.assertIn("s3", ordered_sigs)
 | |
|         self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s1"))
 | |
|         self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s2"))
 | |
| 
 | |
|     def test_chained_dependencies(self):
 | |
|         raw = [
 | |
|             ("s1", ("s1_db", ["alpha"])),
 | |
|             ("s2", ("s2_db", ["bravo"])),
 | |
|             ("s3", ("s3_db", ["charlie"])),
 | |
|         ]
 | |
|         dependencies = {
 | |
|             "alpha": ["bravo"],
 | |
|             "bravo": ["charlie"],
 | |
|         }
 | |
| 
 | |
|         ordered = dependency_ordered(raw, dependencies=dependencies)
 | |
|         ordered_sigs = [sig for sig, value in ordered]
 | |
| 
 | |
|         self.assertIn("s1", ordered_sigs)
 | |
|         self.assertIn("s2", ordered_sigs)
 | |
|         self.assertIn("s3", ordered_sigs)
 | |
| 
 | |
|         # Explicit dependencies
 | |
|         self.assertLess(ordered_sigs.index("s2"), ordered_sigs.index("s1"))
 | |
|         self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s2"))
 | |
| 
 | |
|         # Implied dependencies
 | |
|         self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s1"))
 | |
| 
 | |
|     def test_multiple_dependencies(self):
 | |
|         raw = [
 | |
|             ("s1", ("s1_db", ["alpha"])),
 | |
|             ("s2", ("s2_db", ["bravo"])),
 | |
|             ("s3", ("s3_db", ["charlie"])),
 | |
|             ("s4", ("s4_db", ["delta"])),
 | |
|         ]
 | |
|         dependencies = {
 | |
|             "alpha": ["bravo", "delta"],
 | |
|             "bravo": ["charlie"],
 | |
|             "delta": ["charlie"],
 | |
|         }
 | |
| 
 | |
|         ordered = dependency_ordered(raw, dependencies=dependencies)
 | |
|         ordered_sigs = [sig for sig, aliases in ordered]
 | |
| 
 | |
|         self.assertIn("s1", ordered_sigs)
 | |
|         self.assertIn("s2", ordered_sigs)
 | |
|         self.assertIn("s3", ordered_sigs)
 | |
|         self.assertIn("s4", ordered_sigs)
 | |
| 
 | |
|         # Explicit dependencies
 | |
|         self.assertLess(ordered_sigs.index("s2"), ordered_sigs.index("s1"))
 | |
|         self.assertLess(ordered_sigs.index("s4"), ordered_sigs.index("s1"))
 | |
|         self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s2"))
 | |
|         self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s4"))
 | |
| 
 | |
|         # Implicit dependencies
 | |
|         self.assertLess(ordered_sigs.index("s3"), ordered_sigs.index("s1"))
 | |
| 
 | |
|     def test_circular_dependencies(self):
 | |
|         raw = [
 | |
|             ("s1", ("s1_db", ["alpha"])),
 | |
|             ("s2", ("s2_db", ["bravo"])),
 | |
|         ]
 | |
|         dependencies = {
 | |
|             "bravo": ["alpha"],
 | |
|             "alpha": ["bravo"],
 | |
|         }
 | |
| 
 | |
|         with self.assertRaises(ImproperlyConfigured):
 | |
|             dependency_ordered(raw, dependencies=dependencies)
 | |
| 
 | |
|     def test_own_alias_dependency(self):
 | |
|         raw = [("s1", ("s1_db", ["alpha", "bravo"]))]
 | |
|         dependencies = {"alpha": ["bravo"]}
 | |
| 
 | |
|         with self.assertRaises(ImproperlyConfigured):
 | |
|             dependency_ordered(raw, dependencies=dependencies)
 | |
| 
 | |
|         # reordering aliases shouldn't matter
 | |
|         raw = [("s1", ("s1_db", ["bravo", "alpha"]))]
 | |
| 
 | |
|         with self.assertRaises(ImproperlyConfigured):
 | |
|             dependency_ordered(raw, dependencies=dependencies)
 | |
| 
 | |
| 
 | |
| class MockTestRunner:
 | |
|     def __init__(self, *args, **kwargs):
 | |
|         if parallel := kwargs.get("parallel"):
 | |
|             sys.stderr.write(f"parallel={parallel}")
 | |
|         if durations := kwargs.get("durations"):
 | |
|             sys.stderr.write(f"durations={durations}")
 | |
| 
 | |
| 
 | |
| MockTestRunner.run_tests = mock.Mock(return_value=[])
 | |
| 
 | |
| 
 | |
| class ManageCommandTests(unittest.TestCase):
 | |
|     def test_custom_test_runner(self):
 | |
|         call_command("test", "sites", testrunner="test_runner.tests.MockTestRunner")
 | |
|         MockTestRunner.run_tests.assert_called_with(("sites",))
 | |
| 
 | |
|     def test_bad_test_runner(self):
 | |
|         with self.assertRaises(AttributeError):
 | |
|             call_command("test", "sites", testrunner="test_runner.NonexistentRunner")
 | |
| 
 | |
|     def test_time_recorded(self):
 | |
|         with captured_stderr() as stderr:
 | |
|             call_command(
 | |
|                 "test",
 | |
|                 "--timing",
 | |
|                 "sites",
 | |
|                 testrunner="test_runner.tests.MockTestRunner",
 | |
|             )
 | |
|         self.assertIn("Total run took", stderr.getvalue())
 | |
| 
 | |
|     @unittest.skipUnless(PY312, "unittest --durations option requires Python 3.12")
 | |
|     def test_durations(self):
 | |
|         with captured_stderr() as stderr:
 | |
|             call_command(
 | |
|                 "test",
 | |
|                 "--durations=10",
 | |
|                 "sites",
 | |
|                 testrunner="test_runner.tests.MockTestRunner",
 | |
|             )
 | |
|         self.assertIn("durations=10", stderr.getvalue())
 | |
| 
 | |
|     @unittest.skipIf(PY312, "unittest --durations option requires Python 3.12")
 | |
|     def test_durations_lt_py312(self):
 | |
|         msg = "Error: unrecognized arguments: --durations=10"
 | |
|         with self.assertRaises(CommandError, msg=msg):
 | |
|             call_command(
 | |
|                 "test",
 | |
|                 "--durations=10",
 | |
|                 "sites",
 | |
|                 testrunner="test_runner.tests.MockTestRunner",
 | |
|             )
 | |
| 
 | |
| 
 | |
| # Isolate from the real environment.
 | |
| @mock.patch.dict(os.environ, {}, clear=True)
 | |
| @mock.patch.object(multiprocessing, "cpu_count", return_value=12)
 | |
| class ManageCommandParallelTests(SimpleTestCase):
 | |
|     def test_parallel_default(self, *mocked_objects):
 | |
|         with captured_stderr() as stderr:
 | |
|             call_command(
 | |
|                 "test",
 | |
|                 "--parallel",
 | |
|                 testrunner="test_runner.tests.MockTestRunner",
 | |
|             )
 | |
|         self.assertIn("parallel=12", stderr.getvalue())
 | |
| 
 | |
|     def test_parallel_auto(self, *mocked_objects):
 | |
|         with captured_stderr() as stderr:
 | |
|             call_command(
 | |
|                 "test",
 | |
|                 "--parallel=auto",
 | |
|                 testrunner="test_runner.tests.MockTestRunner",
 | |
|             )
 | |
|         self.assertIn("parallel=12", stderr.getvalue())
 | |
| 
 | |
|     def test_no_parallel(self, *mocked_objects):
 | |
|         with captured_stderr() as stderr:
 | |
|             call_command("test", testrunner="test_runner.tests.MockTestRunner")
 | |
|         # Parallel is disabled by default.
 | |
|         self.assertEqual(stderr.getvalue(), "")
 | |
| 
 | |
|     @mock.patch.object(multiprocessing, "get_start_method", return_value="spawn")
 | |
|     def test_parallel_spawn(self, *mocked_objects):
 | |
|         with captured_stderr() as stderr:
 | |
|             call_command(
 | |
|                 "test",
 | |
|                 "--parallel=auto",
 | |
|                 testrunner="test_runner.tests.MockTestRunner",
 | |
|             )
 | |
|         self.assertIn("parallel=1", stderr.getvalue())
 | |
| 
 | |
|     @mock.patch.object(multiprocessing, "get_start_method", return_value="spawn")
 | |
|     def test_no_parallel_spawn(self, *mocked_objects):
 | |
|         with captured_stderr() as stderr:
 | |
|             call_command(
 | |
|                 "test",
 | |
|                 testrunner="test_runner.tests.MockTestRunner",
 | |
|             )
 | |
|         self.assertEqual(stderr.getvalue(), "")
 | |
| 
 | |
|     @mock.patch.dict(os.environ, {"DJANGO_TEST_PROCESSES": "7"})
 | |
|     def test_no_parallel_django_test_processes_env(self, *mocked_objects):
 | |
|         with captured_stderr() as stderr:
 | |
|             call_command("test", testrunner="test_runner.tests.MockTestRunner")
 | |
|         self.assertEqual(stderr.getvalue(), "")
 | |
| 
 | |
|     @mock.patch.dict(os.environ, {"DJANGO_TEST_PROCESSES": "invalid"})
 | |
|     def test_django_test_processes_env_non_int(self, *mocked_objects):
 | |
|         with self.assertRaises(ValueError):
 | |
|             call_command(
 | |
|                 "test",
 | |
|                 "--parallel",
 | |
|                 testrunner="test_runner.tests.MockTestRunner",
 | |
|             )
 | |
| 
 | |
|     @mock.patch.dict(os.environ, {"DJANGO_TEST_PROCESSES": "7"})
 | |
|     def test_django_test_processes_parallel_default(self, *mocked_objects):
 | |
|         for parallel in ["--parallel", "--parallel=auto"]:
 | |
|             with self.subTest(parallel=parallel):
 | |
|                 with captured_stderr() as stderr:
 | |
|                     call_command(
 | |
|                         "test",
 | |
|                         parallel,
 | |
|                         testrunner="test_runner.tests.MockTestRunner",
 | |
|                     )
 | |
|                 self.assertIn("parallel=7", stderr.getvalue())
 | |
| 
 | |
| 
 | |
| class CustomTestRunnerOptionsSettingsTests(AdminScriptTestCase):
 | |
|     """
 | |
|     Custom runners can add command line arguments. The runner is specified
 | |
|     through a settings file.
 | |
|     """
 | |
| 
 | |
|     def setUp(self):
 | |
|         super().setUp()
 | |
|         settings = {
 | |
|             "TEST_RUNNER": "'test_runner.runner.CustomOptionsTestRunner'",
 | |
|         }
 | |
|         self.write_settings("settings.py", sdict=settings)
 | |
| 
 | |
|     def test_default_options(self):
 | |
|         args = ["test", "--settings=test_project.settings"]
 | |
|         out, err = self.run_django_admin(args)
 | |
|         self.assertNoOutput(err)
 | |
|         self.assertOutput(out, "1:2:3")
 | |
| 
 | |
|     def test_default_and_given_options(self):
 | |
|         args = ["test", "--settings=test_project.settings", "--option_b=foo"]
 | |
|         out, err = self.run_django_admin(args)
 | |
|         self.assertNoOutput(err)
 | |
|         self.assertOutput(out, "1:foo:3")
 | |
| 
 | |
|     def test_option_name_and_value_separated(self):
 | |
|         args = ["test", "--settings=test_project.settings", "--option_b", "foo"]
 | |
|         out, err = self.run_django_admin(args)
 | |
|         self.assertNoOutput(err)
 | |
|         self.assertOutput(out, "1:foo:3")
 | |
| 
 | |
|     def test_all_options_given(self):
 | |
|         args = [
 | |
|             "test",
 | |
|             "--settings=test_project.settings",
 | |
|             "--option_a=bar",
 | |
|             "--option_b=foo",
 | |
|             "--option_c=31337",
 | |
|         ]
 | |
|         out, err = self.run_django_admin(args)
 | |
|         self.assertNoOutput(err)
 | |
|         self.assertOutput(out, "bar:foo:31337")
 | |
| 
 | |
| 
 | |
| class CustomTestRunnerOptionsCmdlineTests(AdminScriptTestCase):
 | |
|     """
 | |
|     Custom runners can add command line arguments when the runner is specified
 | |
|     using --testrunner.
 | |
|     """
 | |
| 
 | |
|     def setUp(self):
 | |
|         super().setUp()
 | |
|         self.write_settings("settings.py")
 | |
| 
 | |
|     def test_testrunner_option(self):
 | |
|         args = [
 | |
|             "test",
 | |
|             "--testrunner",
 | |
|             "test_runner.runner.CustomOptionsTestRunner",
 | |
|             "--option_a=bar",
 | |
|             "--option_b=foo",
 | |
|             "--option_c=31337",
 | |
|         ]
 | |
|         out, err = self.run_django_admin(args, "test_project.settings")
 | |
|         self.assertNoOutput(err)
 | |
|         self.assertOutput(out, "bar:foo:31337")
 | |
| 
 | |
|     def test_testrunner_equals(self):
 | |
|         args = [
 | |
|             "test",
 | |
|             "--testrunner=test_runner.runner.CustomOptionsTestRunner",
 | |
|             "--option_a=bar",
 | |
|             "--option_b=foo",
 | |
|             "--option_c=31337",
 | |
|         ]
 | |
|         out, err = self.run_django_admin(args, "test_project.settings")
 | |
|         self.assertNoOutput(err)
 | |
|         self.assertOutput(out, "bar:foo:31337")
 | |
| 
 | |
|     def test_no_testrunner(self):
 | |
|         args = ["test", "--testrunner"]
 | |
|         out, err = self.run_django_admin(args, "test_project.settings")
 | |
|         self.assertIn("usage", err)
 | |
|         self.assertNotIn("Traceback", err)
 | |
|         self.assertNoOutput(out)
 | |
| 
 | |
| 
 | |
| class NoInitializeSuiteTestRunnerTests(SimpleTestCase):
 | |
|     @mock.patch.object(multiprocessing, "get_start_method", return_value="spawn")
 | |
|     @mock.patch(
 | |
|         "django.test.runner.ParallelTestSuite.initialize_suite",
 | |
|         side_effect=Exception("initialize_suite() is called."),
 | |
|     )
 | |
|     def test_no_initialize_suite_test_runner(self, *mocked_objects):
 | |
|         """
 | |
|         The test suite's initialize_suite() method must always be called when
 | |
|         using spawn. It cannot rely on a test runner implementation.
 | |
|         """
 | |
| 
 | |
|         class NoInitializeSuiteTestRunner(DiscoverRunner):
 | |
|             def setup_test_environment(self, **kwargs):
 | |
|                 return
 | |
| 
 | |
|             def setup_databases(self, **kwargs):
 | |
|                 return
 | |
| 
 | |
|             def run_checks(self, databases):
 | |
|                 return
 | |
| 
 | |
|             def teardown_databases(self, old_config, **kwargs):
 | |
|                 return
 | |
| 
 | |
|             def teardown_test_environment(self, **kwargs):
 | |
|                 return
 | |
| 
 | |
|             def run_suite(self, suite, **kwargs):
 | |
|                 kwargs = self.get_test_runner_kwargs()
 | |
|                 runner = self.test_runner(**kwargs)
 | |
|                 return runner.run(suite)
 | |
| 
 | |
|         with self.assertRaisesMessage(Exception, "initialize_suite() is called."):
 | |
|             runner = NoInitializeSuiteTestRunner(
 | |
|                 verbosity=0, interactive=False, parallel=2
 | |
|             )
 | |
|             runner.run_tests(
 | |
|                 [
 | |
|                     "test_runner_apps.sample.tests_sample.TestDjangoTestCase",
 | |
|                     "test_runner_apps.simple.tests",
 | |
|                 ]
 | |
|             )
 | |
| 
 | |
| 
 | |
| class TestRunnerInitializerTests(SimpleTestCase):
 | |
|     # Raise an exception to don't actually run tests.
 | |
|     @mock.patch.object(
 | |
|         multiprocessing, "Pool", side_effect=Exception("multiprocessing.Pool()")
 | |
|     )
 | |
|     def test_no_initialize_suite_test_runner(self, mocked_pool):
 | |
|         class StubTestRunner(DiscoverRunner):
 | |
|             def setup_test_environment(self, **kwargs):
 | |
|                 return
 | |
| 
 | |
|             def setup_databases(self, **kwargs):
 | |
|                 return
 | |
| 
 | |
|             def run_checks(self, databases):
 | |
|                 return
 | |
| 
 | |
|             def teardown_databases(self, old_config, **kwargs):
 | |
|                 return
 | |
| 
 | |
|             def teardown_test_environment(self, **kwargs):
 | |
|                 return
 | |
| 
 | |
|             def run_suite(self, suite, **kwargs):
 | |
|                 kwargs = self.get_test_runner_kwargs()
 | |
|                 runner = self.test_runner(**kwargs)
 | |
|                 return runner.run(suite)
 | |
| 
 | |
|         runner = StubTestRunner(
 | |
|             verbosity=0, interactive=False, parallel=2, debug_mode=True
 | |
|         )
 | |
|         with self.assertRaisesMessage(Exception, "multiprocessing.Pool()"):
 | |
|             runner.run_tests(
 | |
|                 [
 | |
|                     "test_runner_apps.sample.tests_sample.TestDjangoTestCase",
 | |
|                     "test_runner_apps.simple.tests",
 | |
|                 ]
 | |
|             )
 | |
|         # Initializer must be a function.
 | |
|         self.assertIs(mocked_pool.call_args.kwargs["initializer"], _init_worker)
 | |
|         initargs = mocked_pool.call_args.kwargs["initargs"]
 | |
|         self.assertEqual(len(initargs), 7)
 | |
|         self.assertEqual(initargs[5], True)  # debug_mode
 | |
|         self.assertEqual(initargs[6], {db.DEFAULT_DB_ALIAS})  # Used database aliases.
 | |
| 
 | |
| 
 | |
| class Ticket17477RegressionTests(AdminScriptTestCase):
 | |
|     def setUp(self):
 | |
|         super().setUp()
 | |
|         self.write_settings("settings.py")
 | |
| 
 | |
|     def test_ticket_17477(self):
 | |
|         """'manage.py help test' works after r16352."""
 | |
|         args = ["help", "test"]
 | |
|         out, err = self.run_manage(args)
 | |
|         self.assertNoOutput(err)
 | |
| 
 | |
| 
 | |
| class SQLiteInMemoryTestDbs(TransactionTestCase):
 | |
|     available_apps = ["test_runner"]
 | |
|     databases = {"default", "other"}
 | |
| 
 | |
|     @unittest.skipUnless(
 | |
|         all(db.connections[conn].vendor == "sqlite" for conn in db.connections),
 | |
|         "This is an sqlite-specific issue",
 | |
|     )
 | |
|     def test_transaction_support(self):
 | |
|         # Assert connections mocking is appropriately applied by preventing
 | |
|         # any attempts at calling create_test_db on the global connection
 | |
|         # objects.
 | |
|         for connection in db.connections.all():
 | |
|             create_test_db = mock.patch.object(
 | |
|                 connection.creation,
 | |
|                 "create_test_db",
 | |
|                 side_effect=AssertionError(
 | |
|                     "Global connection object shouldn't be manipulated."
 | |
|                 ),
 | |
|             )
 | |
|             create_test_db.start()
 | |
|             self.addCleanup(create_test_db.stop)
 | |
|         for option_key, option_value in (
 | |
|             ("NAME", ":memory:"),
 | |
|             ("TEST", {"NAME": ":memory:"}),
 | |
|         ):
 | |
|             tested_connections = db.ConnectionHandler(
 | |
|                 {
 | |
|                     "default": {
 | |
|                         "ENGINE": "django.db.backends.sqlite3",
 | |
|                         option_key: option_value,
 | |
|                     },
 | |
|                     "other": {
 | |
|                         "ENGINE": "django.db.backends.sqlite3",
 | |
|                         option_key: option_value,
 | |
|                     },
 | |
|                 }
 | |
|             )
 | |
|             with mock.patch("django.test.utils.connections", new=tested_connections):
 | |
|                 other = tested_connections["other"]
 | |
|                 try:
 | |
|                     new_test_connections = DiscoverRunner(verbosity=0).setup_databases()
 | |
|                     msg = (
 | |
|                         f"DATABASES setting '{option_key}' option set to sqlite3's "
 | |
|                         "':memory:' value shouldn't interfere with transaction support "
 | |
|                         "detection."
 | |
|                     )
 | |
|                     # Transaction support is properly initialized for the
 | |
|                     # 'other' DB.
 | |
|                     self.assertTrue(other.features.supports_transactions, msg)
 | |
|                     # And all the DBs report that they support transactions.
 | |
|                     self.assertTrue(connections_support_transactions(), msg)
 | |
|                 finally:
 | |
|                     for test_connection, _, _ in new_test_connections:
 | |
|                         test_connection._close()
 | |
| 
 | |
| 
 | |
| class DummyBackendTest(unittest.TestCase):
 | |
|     def test_setup_databases(self):
 | |
|         """
 | |
|         setup_databases() doesn't fail with dummy database backend.
 | |
|         """
 | |
|         tested_connections = db.ConnectionHandler({})
 | |
|         with mock.patch("django.test.utils.connections", new=tested_connections):
 | |
|             runner_instance = DiscoverRunner(verbosity=0)
 | |
|             old_config = runner_instance.setup_databases()
 | |
|             runner_instance.teardown_databases(old_config)
 | |
| 
 | |
| 
 | |
| class AliasedDefaultTestSetupTest(unittest.TestCase):
 | |
|     def test_setup_aliased_default_database(self):
 | |
|         """
 | |
|         setup_databases() doesn't fail when 'default' is aliased
 | |
|         """
 | |
|         tested_connections = db.ConnectionHandler(
 | |
|             {"default": {"NAME": "dummy"}, "aliased": {"NAME": "dummy"}}
 | |
|         )
 | |
|         with mock.patch("django.test.utils.connections", new=tested_connections):
 | |
|             runner_instance = DiscoverRunner(verbosity=0)
 | |
|             old_config = runner_instance.setup_databases()
 | |
|             runner_instance.teardown_databases(old_config)
 | |
| 
 | |
| 
 | |
| class SetupDatabasesTests(unittest.TestCase):
 | |
|     def setUp(self):
 | |
|         self.runner_instance = DiscoverRunner(verbosity=0)
 | |
| 
 | |
|     def test_setup_aliased_databases(self):
 | |
|         tested_connections = db.ConnectionHandler(
 | |
|             {
 | |
|                 "default": {
 | |
|                     "ENGINE": "django.db.backends.dummy",
 | |
|                     "NAME": "dbname",
 | |
|                 },
 | |
|                 "other": {
 | |
|                     "ENGINE": "django.db.backends.dummy",
 | |
|                     "NAME": "dbname",
 | |
|                 },
 | |
|             }
 | |
|         )
 | |
| 
 | |
|         with mock.patch(
 | |
|             "django.db.backends.dummy.base.DatabaseWrapper.creation_class"
 | |
|         ) as mocked_db_creation:
 | |
|             with mock.patch("django.test.utils.connections", new=tested_connections):
 | |
|                 old_config = self.runner_instance.setup_databases()
 | |
|                 self.runner_instance.teardown_databases(old_config)
 | |
|         mocked_db_creation.return_value.destroy_test_db.assert_called_once_with(
 | |
|             "dbname", 0, False
 | |
|         )
 | |
| 
 | |
|     def test_setup_test_database_aliases(self):
 | |
|         """
 | |
|         The default database must be the first because data migrations
 | |
|         use the default alias by default.
 | |
|         """
 | |
|         tested_connections = db.ConnectionHandler(
 | |
|             {
 | |
|                 "other": {
 | |
|                     "ENGINE": "django.db.backends.dummy",
 | |
|                     "NAME": "dbname",
 | |
|                 },
 | |
|                 "default": {
 | |
|                     "ENGINE": "django.db.backends.dummy",
 | |
|                     "NAME": "dbname",
 | |
|                 },
 | |
|             }
 | |
|         )
 | |
|         with mock.patch("django.test.utils.connections", new=tested_connections):
 | |
|             test_databases, _ = get_unique_databases_and_mirrors()
 | |
|             self.assertEqual(
 | |
|                 test_databases,
 | |
|                 {
 | |
|                     ("", "", "django.db.backends.dummy", "test_dbname"): (
 | |
|                         "dbname",
 | |
|                         ["default", "other"],
 | |
|                     ),
 | |
|                 },
 | |
|             )
 | |
| 
 | |
|     def test_destroy_test_db_restores_db_name(self):
 | |
|         tested_connections = db.ConnectionHandler(
 | |
|             {
 | |
|                 "default": {
 | |
|                     "ENGINE": settings.DATABASES[db.DEFAULT_DB_ALIAS]["ENGINE"],
 | |
|                     "NAME": "xxx_test_database",
 | |
|                 },
 | |
|             }
 | |
|         )
 | |
|         # Using the real current name as old_name to not mess with the test suite.
 | |
|         old_name = settings.DATABASES[db.DEFAULT_DB_ALIAS]["NAME"]
 | |
|         with mock.patch("django.db.connections", new=tested_connections):
 | |
|             tested_connections["default"].creation.destroy_test_db(
 | |
|                 old_name, verbosity=0, keepdb=True
 | |
|             )
 | |
|             self.assertEqual(
 | |
|                 tested_connections["default"].settings_dict["NAME"], old_name
 | |
|             )
 | |
| 
 | |
|     def test_serialization(self):
 | |
|         tested_connections = db.ConnectionHandler(
 | |
|             {
 | |
|                 "default": {
 | |
|                     "ENGINE": "django.db.backends.dummy",
 | |
|                 },
 | |
|             }
 | |
|         )
 | |
|         with mock.patch(
 | |
|             "django.db.backends.dummy.base.DatabaseWrapper.creation_class"
 | |
|         ) as mocked_db_creation:
 | |
|             with mock.patch("django.test.utils.connections", new=tested_connections):
 | |
|                 self.runner_instance.setup_databases()
 | |
|         mocked_db_creation.return_value.create_test_db.assert_called_once_with(
 | |
|             verbosity=0, autoclobber=False, serialize=True, keepdb=False
 | |
|         )
 | |
| 
 | |
| 
 | |
| @skipUnlessDBFeature("supports_sequence_reset")
 | |
| class AutoIncrementResetTest(TransactionTestCase):
 | |
|     """
 | |
|     Creating the same models in different test methods receive the same PK
 | |
|     values since the sequences are reset before each test method.
 | |
|     """
 | |
| 
 | |
|     available_apps = ["test_runner"]
 | |
| 
 | |
|     reset_sequences = True
 | |
| 
 | |
|     def _test(self):
 | |
|         # Regular model
 | |
|         p = Person.objects.create(first_name="Jack", last_name="Smith")
 | |
|         self.assertEqual(p.pk, 1)
 | |
|         # Auto-created many-to-many through model
 | |
|         p.friends.add(Person.objects.create(first_name="Jacky", last_name="Smith"))
 | |
|         self.assertEqual(p.friends.through.objects.first().pk, 1)
 | |
|         # Many-to-many through model
 | |
|         b = B.objects.create()
 | |
|         t = Through.objects.create(person=p, b=b)
 | |
|         self.assertEqual(t.pk, 1)
 | |
| 
 | |
|     def test_autoincrement_reset1(self):
 | |
|         self._test()
 | |
| 
 | |
|     def test_autoincrement_reset2(self):
 | |
|         self._test()
 | |
| 
 | |
| 
 | |
| class EmptyDefaultDatabaseTest(unittest.TestCase):
 | |
|     def test_empty_default_database(self):
 | |
|         """
 | |
|         An empty default database in settings does not raise an ImproperlyConfigured
 | |
|         error when running a unit test that does not use a database.
 | |
|         """
 | |
|         tested_connections = db.ConnectionHandler({"default": {}})
 | |
|         with mock.patch("django.db.connections", new=tested_connections):
 | |
|             connection = tested_connections[db.utils.DEFAULT_DB_ALIAS]
 | |
|             self.assertEqual(
 | |
|                 connection.settings_dict["ENGINE"], "django.db.backends.dummy"
 | |
|             )
 | |
|             connections_support_transactions()
 | |
| 
 | |
| 
 | |
| class RunTestsExceptionHandlingTests(unittest.TestCase):
 | |
|     def test_run_checks_raises(self):
 | |
|         """
 | |
|         Teardown functions are run when run_checks() raises SystemCheckError.
 | |
|         """
 | |
|         with (
 | |
|             mock.patch("django.test.runner.DiscoverRunner.setup_test_environment"),
 | |
|             mock.patch("django.test.runner.DiscoverRunner.setup_databases"),
 | |
|             mock.patch("django.test.runner.DiscoverRunner.build_suite"),
 | |
|             mock.patch(
 | |
|                 "django.test.runner.DiscoverRunner.run_checks",
 | |
|                 side_effect=SystemCheckError,
 | |
|             ),
 | |
|             mock.patch(
 | |
|                 "django.test.runner.DiscoverRunner.teardown_databases"
 | |
|             ) as teardown_databases,
 | |
|             mock.patch(
 | |
|                 "django.test.runner.DiscoverRunner.teardown_test_environment"
 | |
|             ) as teardown_test_environment,
 | |
|         ):
 | |
|             runner = DiscoverRunner(verbosity=0, interactive=False)
 | |
|             with self.assertRaises(SystemCheckError):
 | |
|                 runner.run_tests(
 | |
|                     ["test_runner_apps.sample.tests_sample.TestDjangoTestCase"]
 | |
|                 )
 | |
|             self.assertTrue(teardown_databases.called)
 | |
|             self.assertTrue(teardown_test_environment.called)
 | |
| 
 | |
|     def test_run_checks_raises_and_teardown_raises(self):
 | |
|         """
 | |
|         SystemCheckError is surfaced when run_checks() raises SystemCheckError
 | |
|         and teardown databases() raises ValueError.
 | |
|         """
 | |
|         with (
 | |
|             mock.patch("django.test.runner.DiscoverRunner.setup_test_environment"),
 | |
|             mock.patch("django.test.runner.DiscoverRunner.setup_databases"),
 | |
|             mock.patch("django.test.runner.DiscoverRunner.build_suite"),
 | |
|             mock.patch(
 | |
|                 "django.test.runner.DiscoverRunner.run_checks",
 | |
|                 side_effect=SystemCheckError,
 | |
|             ),
 | |
|             mock.patch(
 | |
|                 "django.test.runner.DiscoverRunner.teardown_databases",
 | |
|                 side_effect=ValueError,
 | |
|             ) as teardown_databases,
 | |
|             mock.patch(
 | |
|                 "django.test.runner.DiscoverRunner.teardown_test_environment"
 | |
|             ) as teardown_test_environment,
 | |
|         ):
 | |
|             runner = DiscoverRunner(verbosity=0, interactive=False)
 | |
|             with self.assertRaises(SystemCheckError):
 | |
|                 runner.run_tests(
 | |
|                     ["test_runner_apps.sample.tests_sample.TestDjangoTestCase"]
 | |
|                 )
 | |
|             self.assertTrue(teardown_databases.called)
 | |
|             self.assertFalse(teardown_test_environment.called)
 | |
| 
 | |
|     def test_run_checks_passes_and_teardown_raises(self):
 | |
|         """
 | |
|         Exceptions on teardown are surfaced if no exceptions happen during
 | |
|         run_checks().
 | |
|         """
 | |
|         with (
 | |
|             mock.patch("django.test.runner.DiscoverRunner.setup_test_environment"),
 | |
|             mock.patch("django.test.runner.DiscoverRunner.setup_databases"),
 | |
|             mock.patch("django.test.runner.DiscoverRunner.build_suite"),
 | |
|             mock.patch("django.test.runner.DiscoverRunner.run_checks"),
 | |
|             mock.patch(
 | |
|                 "django.test.runner.DiscoverRunner.teardown_databases",
 | |
|                 side_effect=ValueError,
 | |
|             ) as teardown_databases,
 | |
|             mock.patch(
 | |
|                 "django.test.runner.DiscoverRunner.teardown_test_environment"
 | |
|             ) as teardown_test_environment,
 | |
|         ):
 | |
|             runner = DiscoverRunner(verbosity=0, interactive=False)
 | |
|             with self.assertRaises(ValueError):
 | |
|                 # Suppress the output when running TestDjangoTestCase.
 | |
|                 with mock.patch("sys.stderr"):
 | |
|                     runner.run_tests(
 | |
|                         ["test_runner_apps.sample.tests_sample.TestDjangoTestCase"]
 | |
|                     )
 | |
|             self.assertTrue(teardown_databases.called)
 | |
|             self.assertFalse(teardown_test_environment.called)
 |