#29024: `TestContextDecorator` never exits if `setUp` fails in tests ---------------------------------------------+------------------------ Reporter: Anthony King | Owner: nobody Type: Bug | Status: new Component: Testing framework | Version: 1.11 Severity: Normal | Keywords: Triage Stage: Unreviewed | Has patch: 0 Needs documentation: 0 | Needs tests: 0 Patch needs improvement: 0 | Easy pickings: 1 UI/UX: 0 | ---------------------------------------------+------------------------ when using `TestContextDecorator` as a class decorator, `enable` is called just before `Test.setUp`, and `disable` is called after `Test.tearDown`.
unfortunately, `tearDown` is not called in the event of a failure inside `setUp`. This can result in unexpected behaviour. Even though this class is not documented, there are decorators that use it that are. example: {{{#!python class example_decorator(TestContextDecorator): some_var = False def enable(self): self.__class__.some_var = True def disable(self): self.__class__.some_var = False class Example1TestCase(TestCase): def test_var_is_false(self): # some_var will be False self.assertFalse(example_decorator.some_var) @example_decorator() class Example2TestCase(TestCase): def setUp(self): raise Exception def test_var_is_true(self): # won't be hit due to exception in setUp self.assertTrue(example_decorator.some_var) class Example3TestCase(TestCase): def test_var_is_false(self): # some_var will be True now due to no cleanup in Example2TestCase self.assertFalse(example_decorator.some_var) }}} output: {{{ .EF ====================================================================== ERROR: test_var_is_true (sharefile.tests.Example2TestCase) ---------------------------------------------------------------------- ... Traceback ====================================================================== FAIL: test_var_is_false (sharefile.tests.Example3TestCase) ---------------------------------------------------------------------- ... AssertionError: True is not false Ran 3 tests in 0.007s FAILED (failures=1, errors=1) }}} There are 2 potential fixes here: - decorate `setUpClass` and `tearDownClass` - use `addCleanup` inside the `setUp` decorator `addCleanup` will always run, and will only ever be registered if the context manager was successfully enabled. It would also be useful to have this class documented, however that's out of scope of this ticket. -- Ticket URL: <https://code.djangoproject.com/ticket/29024> Django <https://code.djangoproject.com/> The Web framework for perfectionists with deadlines. -- You received this message because you are subscribed to the Google Groups "Django updates" group. To unsubscribe from this group and stop receiving emails from it, send an email to django-updates+unsubscr...@googlegroups.com. To post to this group, send email to django-updates@googlegroups.com. To view this discussion on the web visit https://groups.google.com/d/msgid/django-updates/052.3ccb058a9ed903ace6fc26e54008982a%40djangoproject.com. For more options, visit https://groups.google.com/d/optout.