mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	Fixed #22102 -- Made SimpleTestCase tests run before unittest.TestCase ones
Thanks aptiko for the reporti and Tim Graham for the review.
This commit is contained in:
		| @@ -6,7 +6,7 @@ from unittest import TestSuite, defaultTestLoader | |||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.core.exceptions import ImproperlyConfigured | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.test import TestCase | from django.test import SimpleTestCase, TestCase | ||||||
| from django.test.utils import setup_test_environment, teardown_test_environment | from django.test.utils import setup_test_environment, teardown_test_environment | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -18,7 +18,7 @@ class DiscoverRunner(object): | |||||||
|     test_suite = TestSuite |     test_suite = TestSuite | ||||||
|     test_runner = unittest.TextTestRunner |     test_runner = unittest.TextTestRunner | ||||||
|     test_loader = defaultTestLoader |     test_loader = defaultTestLoader | ||||||
|     reorder_by = (TestCase, ) |     reorder_by = (TestCase, SimpleTestCase) | ||||||
|     option_list = ( |     option_list = ( | ||||||
|         make_option('-t', '--top-level-directory', |         make_option('-t', '--top-level-directory', | ||||||
|             action='store', dest='top_level', default=None, |             action='store', dest='top_level', default=None, | ||||||
|   | |||||||
| @@ -206,13 +206,13 @@ the Django test runner reorders tests in the following way: | |||||||
|  |  | ||||||
| * All :class:`~django.test.TestCase` subclasses are run first. | * All :class:`~django.test.TestCase` subclasses are run first. | ||||||
|  |  | ||||||
| * Then, all other unittests (including :class:`unittest.TestCase`, | * Then, all other Django-based tests (test cases based on | ||||||
|   :class:`~django.test.SimpleTestCase` and |   :class:`~django.test.SimpleTestCase`, including | ||||||
|   :class:`~django.test.TransactionTestCase`) are run with no particular |   :class:`~django.test.TransactionTestCase`) are run with no particular | ||||||
|   ordering guaranteed nor enforced among them. |   ordering guaranteed nor enforced among them. | ||||||
|  |  | ||||||
| * Then any other tests (e.g. doctests) that may alter the database without | * Then any other :class:`unittest.TestCase` tests (including doctests) that may | ||||||
|   restoring it to its original state are run. |   alter the database without restoring it to its original state are run. | ||||||
|  |  | ||||||
| .. note:: | .. note:: | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										45
									
								
								tests/test_discovery_sample/doctests.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								tests/test_discovery_sample/doctests.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | |||||||
|  | """ | ||||||
|  | Doctest example from the official Python documentation. | ||||||
|  | https://docs.python.org/3/library/doctest.html | ||||||
|  | """ | ||||||
|  |  | ||||||
|  | def factorial(n): | ||||||
|  |     """Return the factorial of n, an exact integer >= 0. | ||||||
|  |  | ||||||
|  |     >>> [factorial(n) for n in range(6)] | ||||||
|  |     [1, 1, 2, 6, 24, 120] | ||||||
|  |     >>> factorial(30) | ||||||
|  |     265252859812191058636308480000000 | ||||||
|  |     >>> factorial(-1) | ||||||
|  |     Traceback (most recent call last): | ||||||
|  |         ... | ||||||
|  |     ValueError: n must be >= 0 | ||||||
|  |  | ||||||
|  |     Factorials of floats are OK, but the float must be an exact integer: | ||||||
|  |     >>> factorial(30.1) | ||||||
|  |     Traceback (most recent call last): | ||||||
|  |         ... | ||||||
|  |     ValueError: n must be exact integer | ||||||
|  |     >>> factorial(30.0) | ||||||
|  |     265252859812191058636308480000000 | ||||||
|  |  | ||||||
|  |     It must also not be ridiculously large: | ||||||
|  |     >>> factorial(1e100) | ||||||
|  |     Traceback (most recent call last): | ||||||
|  |         ... | ||||||
|  |     OverflowError: n too large | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     import math | ||||||
|  |     if not n >= 0: | ||||||
|  |         raise ValueError("n must be >= 0") | ||||||
|  |     if math.floor(n) != n: | ||||||
|  |         raise ValueError("n must be exact integer") | ||||||
|  |     if n+1 == n:  # catch a value like 1e300 | ||||||
|  |         raise OverflowError("n too large") | ||||||
|  |     result = 1 | ||||||
|  |     factor = 2 | ||||||
|  |     while factor <= n: | ||||||
|  |         result *= factor | ||||||
|  |         factor += 1 | ||||||
|  |     return result | ||||||
| @@ -1,6 +1,9 @@ | |||||||
|  | import doctest | ||||||
| from unittest import TestCase | from unittest import TestCase | ||||||
|  |  | ||||||
| from django.test import TestCase as DjangoTestCase | from django.test import SimpleTestCase, TestCase as DjangoTestCase | ||||||
|  |  | ||||||
|  | from . import doctests | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestVanillaUnittest(TestCase): | class TestVanillaUnittest(TestCase): | ||||||
| @@ -15,5 +18,17 @@ class TestDjangoTestCase(DjangoTestCase): | |||||||
|         self.assertEqual(1, 1) |         self.assertEqual(1, 1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestZimpleTestCase(SimpleTestCase): | ||||||
|  |     # Z is used to trick this test case to appear after Vanilla in default suite | ||||||
|  |  | ||||||
|  |     def test_sample(self): | ||||||
|  |         self.assertEqual(1, 1) | ||||||
|  |  | ||||||
|  |  | ||||||
| class EmptyTestCase(TestCase): | class EmptyTestCase(TestCase): | ||||||
|     pass |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def load_tests(loader, tests, ignore): | ||||||
|  |     tests.addTests(doctest.DocTestSuite(doctests)) | ||||||
|  |     return tests | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ class DiscoverRunnerTest(TestCase): | |||||||
|             ["test_discovery_sample.tests_sample"], |             ["test_discovery_sample.tests_sample"], | ||||||
|         ).countTestCases() |         ).countTestCases() | ||||||
|  |  | ||||||
|         self.assertEqual(count, 2) |         self.assertEqual(count, 4) | ||||||
|  |  | ||||||
|     def test_dotted_test_class_vanilla_unittest(self): |     def test_dotted_test_class_vanilla_unittest(self): | ||||||
|         count = DiscoverRunner().build_suite( |         count = DiscoverRunner().build_suite( | ||||||
| @@ -61,7 +61,7 @@ class DiscoverRunnerTest(TestCase): | |||||||
|                 ["test_discovery_sample/"], |                 ["test_discovery_sample/"], | ||||||
|             ).countTestCases() |             ).countTestCases() | ||||||
|  |  | ||||||
|         self.assertEqual(count, 3) |         self.assertEqual(count, 5) | ||||||
|  |  | ||||||
|     def test_empty_label(self): |     def test_empty_label(self): | ||||||
|         """ |         """ | ||||||
| @@ -103,6 +103,20 @@ class DiscoverRunnerTest(TestCase): | |||||||
|  |  | ||||||
|         self.assertEqual(count, 0) |         self.assertEqual(count, 0) | ||||||
|  |  | ||||||
|  |     def test_testcase_ordering(self): | ||||||
|  |         suite = DiscoverRunner().build_suite(["test_discovery_sample/"]) | ||||||
|  |         tc_names = [case.__class__.__name__ for case in suite._tests] | ||||||
|  |         self.assertEqual( | ||||||
|  |             suite._tests[0].__class__.__name__, | ||||||
|  |             'TestDjangoTestCase', | ||||||
|  |             msg="TestDjangoTestCase should be the first test case") | ||||||
|  |         self.assertEqual( | ||||||
|  |             suite._tests[1].__class__.__name__, | ||||||
|  |             'TestZimpleTestCase', | ||||||
|  |             msg="TestZimpleTestCase should be the second test case") | ||||||
|  |         # All others can follow in unspecified order, including doctests | ||||||
|  |         self.assertIn('DocTestCase', [t.__class__.__name__ for t in suite._tests[2:]]) | ||||||
|  |  | ||||||
|     def test_overrideable_test_suite(self): |     def test_overrideable_test_suite(self): | ||||||
|         self.assertEqual(DiscoverRunner().test_suite, TestSuite) |         self.assertEqual(DiscoverRunner().test_suite, TestSuite) | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user