import sys import unittest import textwrap from test.support.testcase import ExceptionIsLikeMixin class TestInvalidExceptStar(unittest.TestCase): def test_mixed_except_and_except_star_is_syntax_error(self): errors = [ "try: pass\nexcept ValueError: pass\nexcept* TypeError: pass\n", "try: pass\nexcept* ValueError: pass\nexcept TypeError: pass\n", "try: pass\nexcept ValueError as e: pass\nexcept* TypeError: pass\n", "try: pass\nexcept* ValueError as e: pass\nexcept TypeError: pass\n", "try: pass\nexcept ValueError: pass\nexcept* TypeError as e: pass\n", "try: pass\nexcept* ValueError: pass\nexcept TypeError as e: pass\n", "try: pass\nexcept ValueError: pass\nexcept*: pass\n", "try: pass\nexcept* ValueError: pass\nexcept: pass\n", ] for err in errors: with self.assertRaises(SyntaxError): compile(err, "", "exec") def test_except_star_ExceptionGroup_is_runtime_error_single(self): with self.assertRaises(TypeError): try: raise OSError("blah") except* ExceptionGroup as e: pass def test_except_star_ExceptionGroup_is_runtime_error_tuple(self): with self.assertRaises(TypeError): try: raise ExceptionGroup("eg", [ValueError(42)]) except* (TypeError, ExceptionGroup): pass def test_except_star_invalid_exception_type(self): with self.assertRaises(TypeError): try: raise ValueError except* 42: pass with self.assertRaises(TypeError): try: raise ValueError except* (ValueError, 42): pass class TestBreakContinueReturnInExceptStarBlock(unittest.TestCase): MSG = (r"'break', 'continue' and 'return'" r" cannot appear in an except\* block") def check_invalid(self, src): with self.assertRaisesRegex(SyntaxError, self.MSG): compile(textwrap.dedent(src), "", "exec") def test_break_in_except_star(self): self.check_invalid( """ try: raise ValueError except* Exception as e: break """) self.check_invalid( """ for i in range(5): try: pass except* Exception as e: if i == 2: break """) self.check_invalid( """ for i in range(5): try: pass except* Exception as e: if i == 2: break finally: return 0 """) def test_continue_in_except_star_block_invalid(self): self.check_invalid( """ for i in range(5): try: raise ValueError except* Exception as e: continue """) self.check_invalid( """ for i in range(5): try: pass except* Exception as e: if i == 2: continue """) self.check_invalid( """ for i in range(5): try: pass except* Exception as e: if i == 2: continue finally: return 0 """) def test_return_in_except_star_block_invalid(self): self.check_invalid( """ def f(): try: raise ValueError except* Exception as e: return 42 """) self.check_invalid( """ def f(): try: pass except* Exception as e: return 42 finally: finished = True """) def test_break_continue_in_except_star_block_valid(self): try: raise ValueError(42) except* Exception as e: count = 0 for i in range(5): if i == 0: continue if i == 4: break count += 1 self.assertEqual(count, 3) self.assertEqual(i, 4) exc = e self.assertIsInstance(exc, ExceptionGroup) def test_return_in_except_star_block_valid(self): try: raise ValueError(42) except* Exception as e: def f(x): return 2*x r = f(3) exc = e self.assertEqual(r, 6) self.assertIsInstance(exc, ExceptionGroup) class ExceptStarTest(ExceptionIsLikeMixin, unittest.TestCase): def assertMetadataEqual(self, e1, e2): if e1 is None or e2 is None: self.assertTrue(e1 is None and e2 is None) else: self.assertEqual(e1.__context__, e2.__context__) self.assertEqual(e1.__cause__, e2.__cause__) self.assertEqual(e1.__traceback__, e2.__traceback__) def assertMetadataNotEqual(self, e1, e2): if e1 is None or e2 is None: self.assertNotEqual(e1, e2) else: return not (e1.__context__ == e2.__context__ and e1.__cause__ == e2.__cause__ and e1.__traceback__ == e2.__traceback__) class TestExceptStarSplitSemantics(ExceptStarTest): def doSplitTestNamed(self, exc, T, match_template, rest_template): initial_sys_exception = sys.exception() sys_exception = match = rest = None try: try: raise exc except* T as e: sys_exception = sys.exception() match = e except BaseException as e: rest = e self.assertEqual(sys_exception, match) self.assertExceptionIsLike(match, match_template) self.assertExceptionIsLike(rest, rest_template) self.assertEqual(sys.exception(), initial_sys_exception) def doSplitTestUnnamed(self, exc, T, match_template, rest_template): initial_sys_exception = sys.exception() sys_exception = match = rest = None try: try: raise exc except* T: sys_exception = match = sys.exception() else: if rest_template: self.fail("Exception not raised") except BaseException as e: rest = e self.assertExceptionIsLike(match, match_template) self.assertExceptionIsLike(rest, rest_template) self.assertEqual(sys.exception(), initial_sys_exception) def doSplitTestInExceptHandler(self, exc, T, match_template, rest_template): try: raise ExceptionGroup('eg', [TypeError(1), ValueError(2)]) except Exception: self.doSplitTestNamed(exc, T, match_template, rest_template) self.doSplitTestUnnamed(exc, T, match_template, rest_template) def doSplitTestInExceptStarHandler(self, exc, T, match_template, rest_template): try: raise ExceptionGroup('eg', [TypeError(1), ValueError(2)]) except* Exception: self.doSplitTestNamed(exc, T, match_template, rest_template) self.doSplitTestUnnamed(exc, T, match_template, rest_template) def doSplitTest(self, exc, T, match_template, rest_template): self.doSplitTestNamed(exc, T, match_template, rest_template) self.doSplitTestUnnamed(exc, T, match_template, rest_template) self.doSplitTestInExceptHandler(exc, T, match_template, rest_template) self.doSplitTestInExceptStarHandler(exc, T, match_template, rest_template) def test_no_match_single_type(self): self.doSplitTest( ExceptionGroup("test1", [ValueError("V"), TypeError("T")]), SyntaxError, None, ExceptionGroup("test1", [ValueError("V"), TypeError("T")])) def test_match_single_type(self): self.doSplitTest( ExceptionGroup("test2", [ValueError("V1"), ValueError("V2")]), ValueError, ExceptionGroup("test2", [ValueError("V1"), ValueError("V2")]), None) def test_match_single_type_partial_match(self): self.doSplitTest( ExceptionGroup( "test3", [ValueError("V1"), OSError("OS"), ValueError("V2")]), ValueError, ExceptionGroup("test3", [ValueError("V1"), ValueError("V2")]), ExceptionGroup("test3", [OSError("OS")])) def test_match_single_type_nested(self): self.doSplitTest( ExceptionGroup( "g1", [ ValueError("V1"), OSError("OS1"), ExceptionGroup( "g2", [ OSError("OS2"), ValueError("V2"), TypeError("T")])]), ValueError, ExceptionGroup( "g1", [ ValueError("V1"), ExceptionGroup("g2", [ValueError("V2")])]), ExceptionGroup("g1", [ OSError("OS1"), ExceptionGroup("g2", [ OSError("OS2"), TypeError("T")])])) def test_match_type_tuple_nested(self): self.doSplitTest( ExceptionGroup( "h1", [ ValueError("V1"), OSError("OS1"), ExceptionGroup( "h2", [OSError("OS2"), ValueError("V2"), TypeError("T")])]), (ValueError, TypeError), ExceptionGroup( "h1", [ ValueError("V1"), ExceptionGroup("h2", [ValueError("V2"), TypeError("T")])]), ExceptionGroup( "h1", [ OSError("OS1"), ExceptionGroup("h2", [OSError("OS2")])])) def test_empty_groups_removed(self): self.doSplitTest( ExceptionGroup( "eg", [ ExceptionGroup("i1", [ValueError("V1")]), ExceptionGroup("i2", [ValueError("V2"), TypeError("T1")]), ExceptionGroup("i3", [TypeError("T2")])]), TypeError, ExceptionGroup("eg", [ ExceptionGroup("i2", [TypeError("T1")]), ExceptionGroup("i3", [TypeError("T2")])]), ExceptionGroup("eg", [ ExceptionGroup("i1", [ValueError("V1")]), ExceptionGroup("i2", [ValueError("V2")])])) def test_singleton_groups_are_kept(self): self.doSplitTest( ExceptionGroup("j1", [ ExceptionGroup("j2", [ ExceptionGroup("j3", [ValueError("V1")]), ExceptionGroup("j4", [TypeError("T")])])]), TypeError, ExceptionGroup( "j1", [ExceptionGroup("j2", [ExceptionGroup("j4", [TypeError("T")])])]), ExceptionGroup( "j1", [ExceptionGroup("j2", [ExceptionGroup("j3", [ValueError("V1")])])])) def test_naked_exception_matched_wrapped1(self): self.doSplitTest( ValueError("V"), ValueError, ExceptionGroup("", [ValueError("V")]), None) def test_naked_exception_matched_wrapped2(self): self.doSplitTest( ValueError("V"), Exception, ExceptionGroup("", [ValueError("V")]), None) def test_exception_group_except_star_Exception_not_wrapped(self): self.doSplitTest( ExceptionGroup("eg", [ValueError("V")]), Exception, ExceptionGroup("eg", [ValueError("V")]), None) def test_plain_exception_not_matched(self): self.doSplitTest( ValueError("V"), TypeError, None, ValueError("V")) def test_match__supertype(self): self.doSplitTest( ExceptionGroup("st", [BlockingIOError("io"), TypeError("T")]), OSError, ExceptionGroup("st", [BlockingIOError("io")]), ExceptionGroup("st", [TypeError("T")])) def test_multiple_matches_named(self): try: raise ExceptionGroup("mmn", [OSError("os"), BlockingIOError("io")]) except* BlockingIOError as e: self.assertExceptionIsLike(e, ExceptionGroup("mmn", [BlockingIOError("io")])) except* OSError as e: self.assertExceptionIsLike(e, ExceptionGroup("mmn", [OSError("os")])) else: self.fail("Exception not raised") def test_multiple_matches_unnamed(self): try: raise ExceptionGroup("mmu", [OSError("os"), BlockingIOError("io")]) except* BlockingIOError: e = sys.exception() self.assertExceptionIsLike(e, ExceptionGroup("mmu", [BlockingIOError("io")])) except* OSError: e = sys.exception() self.assertExceptionIsLike(e, ExceptionGroup("mmu", [OSError("os")])) else: self.fail("Exception not raised") def test_first_match_wins_named(self): try: raise ExceptionGroup("fst", [BlockingIOError("io")]) except* OSError as e: self.assertExceptionIsLike(e, ExceptionGroup("fst", [BlockingIOError("io")])) except* BlockingIOError: self.fail("Should have been matched as OSError") else: self.fail("Exception not raised") def test_first_match_wins_unnamed(self): try: raise ExceptionGroup("fstu", [BlockingIOError("io")]) except* OSError: e = sys.exception() self.assertExceptionIsLike(e, ExceptionGroup("fstu", [BlockingIOError("io")])) except* BlockingIOError: pass else: self.fail("Exception not raised") def test_nested_except_stars(self): try: raise ExceptionGroup("n", [BlockingIOError("io")]) except* BlockingIOError: try: raise ExceptionGroup("n", [ValueError("io")]) except* ValueError: pass else: self.fail("Exception not raised") e = sys.exception() self.assertExceptionIsLike(e, ExceptionGroup("n", [BlockingIOError("io")])) else: self.fail("Exception not raised") def test_nested_in_loop(self): for _ in range(2): try: raise ExceptionGroup("nl", [BlockingIOError("io")]) except* BlockingIOError: pass else: self.fail("Exception not raised") class TestExceptStarReraise(ExceptStarTest): def test_reraise_all_named(self): try: try: raise ExceptionGroup( "eg", [TypeError(1), ValueError(2), OSError(3)]) except* TypeError as e: raise except* ValueError as e: raise # OSError not handled except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("eg", [TypeError(1), ValueError(2), OSError(3)])) def test_reraise_all_unnamed(self): try: try: raise ExceptionGroup( "eg", [TypeError(1), ValueError(2), OSError(3)]) except* TypeError: raise except* ValueError: raise # OSError not handled except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("eg", [TypeError(1), ValueError(2), OSError(3)])) def test_reraise_some_handle_all_named(self): try: try: raise ExceptionGroup( "eg", [TypeError(1), ValueError(2), OSError(3)]) except* TypeError as e: raise except* ValueError as e: pass # OSError not handled except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("eg", [TypeError(1), OSError(3)])) def test_reraise_partial_handle_all_unnamed(self): try: try: raise ExceptionGroup( "eg", [TypeError(1), ValueError(2)]) except* TypeError: raise except* ValueError: pass except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("eg", [TypeError(1)])) def test_reraise_partial_handle_some_named(self): try: try: raise ExceptionGroup( "eg", [TypeError(1), ValueError(2), OSError(3)]) except* TypeError as e: raise except* ValueError as e: pass # OSError not handled except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("eg", [TypeError(1), OSError(3)])) def test_reraise_partial_handle_some_unnamed(self): try: try: raise ExceptionGroup( "eg", [TypeError(1), ValueError(2), OSError(3)]) except* TypeError: raise except* ValueError: pass except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("eg", [TypeError(1), OSError(3)])) def test_reraise_plain_exception_named(self): try: try: raise ValueError(42) except* ValueError as e: raise except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("", [ValueError(42)])) def test_reraise_plain_exception_unnamed(self): try: try: raise ValueError(42) except* ValueError: raise except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("", [ValueError(42)])) class TestExceptStarRaise(ExceptStarTest): def test_raise_named(self): orig = ExceptionGroup("eg", [ValueError(1), OSError(2)]) try: try: raise orig except* OSError as e: raise TypeError(3) except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup( "", [TypeError(3), ExceptionGroup("eg", [ValueError(1)])])) self.assertExceptionIsLike( exc.exceptions[0].__context__, ExceptionGroup("eg", [OSError(2)])) self.assertMetadataNotEqual(orig, exc) self.assertMetadataEqual(orig, exc.exceptions[0].__context__) def test_raise_unnamed(self): orig = ExceptionGroup("eg", [ValueError(1), OSError(2)]) try: try: raise orig except* OSError: raise TypeError(3) except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup( "", [TypeError(3), ExceptionGroup("eg", [ValueError(1)])])) self.assertExceptionIsLike( exc.exceptions[0].__context__, ExceptionGroup("eg", [OSError(2)])) self.assertMetadataNotEqual(orig, exc) self.assertMetadataEqual(orig, exc.exceptions[0].__context__) def test_raise_handle_all_raise_one_named(self): orig = ExceptionGroup("eg", [TypeError(1), ValueError(2)]) try: try: raise orig except* (TypeError, ValueError) as e: raise SyntaxError(3) except SyntaxError as e: exc = e self.assertExceptionIsLike(exc, SyntaxError(3)) self.assertExceptionIsLike( exc.__context__, ExceptionGroup("eg", [TypeError(1), ValueError(2)])) self.assertMetadataNotEqual(orig, exc) self.assertMetadataEqual(orig, exc.__context__) def test_raise_handle_all_raise_one_unnamed(self): orig = ExceptionGroup("eg", [TypeError(1), ValueError(2)]) try: try: raise orig except* (TypeError, ValueError) as e: raise SyntaxError(3) except SyntaxError as e: exc = e self.assertExceptionIsLike(exc, SyntaxError(3)) self.assertExceptionIsLike( exc.__context__, ExceptionGroup("eg", [TypeError(1), ValueError(2)])) self.assertMetadataNotEqual(orig, exc) self.assertMetadataEqual(orig, exc.__context__) def test_raise_handle_all_raise_two_named(self): orig = ExceptionGroup("eg", [TypeError(1), ValueError(2)]) try: try: raise orig except* TypeError as e: raise SyntaxError(3) except* ValueError as e: raise SyntaxError(4) except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("", [SyntaxError(3), SyntaxError(4)])) self.assertExceptionIsLike( exc.exceptions[0].__context__, ExceptionGroup("eg", [TypeError(1)])) self.assertExceptionIsLike( exc.exceptions[1].__context__, ExceptionGroup("eg", [ValueError(2)])) self.assertMetadataNotEqual(orig, exc) self.assertMetadataEqual(orig, exc.exceptions[0].__context__) self.assertMetadataEqual(orig, exc.exceptions[1].__context__) def test_raise_handle_all_raise_two_unnamed(self): orig = ExceptionGroup("eg", [TypeError(1), ValueError(2)]) try: try: raise orig except* TypeError: raise SyntaxError(3) except* ValueError: raise SyntaxError(4) except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("", [SyntaxError(3), SyntaxError(4)])) self.assertExceptionIsLike( exc.exceptions[0].__context__, ExceptionGroup("eg", [TypeError(1)])) self.assertExceptionIsLike( exc.exceptions[1].__context__, ExceptionGroup("eg", [ValueError(2)])) self.assertMetadataNotEqual(orig, exc) self.assertMetadataEqual(orig, exc.exceptions[0].__context__) self.assertMetadataEqual(orig, exc.exceptions[1].__context__) class TestExceptStarRaiseFrom(ExceptStarTest): def test_raise_named(self): orig = ExceptionGroup("eg", [ValueError(1), OSError(2)]) try: try: raise orig except* OSError as e: raise TypeError(3) from e except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup( "", [TypeError(3), ExceptionGroup("eg", [ValueError(1)])])) self.assertExceptionIsLike( exc.exceptions[0].__context__, ExceptionGroup("eg", [OSError(2)])) self.assertExceptionIsLike( exc.exceptions[0].__cause__, ExceptionGroup("eg", [OSError(2)])) self.assertMetadataNotEqual(orig, exc) self.assertMetadataEqual(orig, exc.exceptions[0].__context__) self.assertMetadataEqual(orig, exc.exceptions[0].__cause__) self.assertMetadataNotEqual(orig, exc.exceptions[1].__context__) self.assertMetadataNotEqual(orig, exc.exceptions[1].__cause__) def test_raise_unnamed(self): orig = ExceptionGroup("eg", [ValueError(1), OSError(2)]) try: try: raise orig except* OSError: e = sys.exception() raise TypeError(3) from e except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup( "", [TypeError(3), ExceptionGroup("eg", [ValueError(1)])])) self.assertExceptionIsLike( exc.exceptions[0].__context__, ExceptionGroup("eg", [OSError(2)])) self.assertExceptionIsLike( exc.exceptions[0].__cause__, ExceptionGroup("eg", [OSError(2)])) self.assertMetadataNotEqual(orig, exc) self.assertMetadataEqual(orig, exc.exceptions[0].__context__) self.assertMetadataEqual(orig, exc.exceptions[0].__cause__) self.assertMetadataNotEqual(orig, exc.exceptions[1].__context__) self.assertMetadataNotEqual(orig, exc.exceptions[1].__cause__) def test_raise_handle_all_raise_one_named(self): orig = ExceptionGroup("eg", [TypeError(1), ValueError(2)]) try: try: raise orig except* (TypeError, ValueError) as e: raise SyntaxError(3) from e except SyntaxError as e: exc = e self.assertExceptionIsLike(exc, SyntaxError(3)) self.assertExceptionIsLike( exc.__context__, ExceptionGroup("eg", [TypeError(1), ValueError(2)])) self.assertExceptionIsLike( exc.__cause__, ExceptionGroup("eg", [TypeError(1), ValueError(2)])) self.assertMetadataNotEqual(orig, exc) self.assertMetadataEqual(orig, exc.__context__) self.assertMetadataEqual(orig, exc.__cause__) def test_raise_handle_all_raise_one_unnamed(self): orig = ExceptionGroup("eg", [TypeError(1), ValueError(2)]) try: try: raise orig except* (TypeError, ValueError) as e: e = sys.exception() raise SyntaxError(3) from e except SyntaxError as e: exc = e self.assertExceptionIsLike(exc, SyntaxError(3)) self.assertExceptionIsLike( exc.__context__, ExceptionGroup("eg", [TypeError(1), ValueError(2)])) self.assertExceptionIsLike( exc.__cause__, ExceptionGroup("eg", [TypeError(1), ValueError(2)])) self.assertMetadataNotEqual(orig, exc) self.assertMetadataEqual(orig, exc.__context__) self.assertMetadataEqual(orig, exc.__cause__) def test_raise_handle_all_raise_two_named(self): orig = ExceptionGroup("eg", [TypeError(1), ValueError(2)]) try: try: raise orig except* TypeError as e: raise SyntaxError(3) from e except* ValueError as e: raise SyntaxError(4) from e except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("", [SyntaxError(3), SyntaxError(4)])) self.assertExceptionIsLike( exc.exceptions[0].__context__, ExceptionGroup("eg", [TypeError(1)])) self.assertExceptionIsLike( exc.exceptions[0].__cause__, ExceptionGroup("eg", [TypeError(1)])) self.assertExceptionIsLike( exc.exceptions[1].__context__, ExceptionGroup("eg", [ValueError(2)])) self.assertExceptionIsLike( exc.exceptions[1].__cause__, ExceptionGroup("eg", [ValueError(2)])) self.assertMetadataNotEqual(orig, exc) self.assertMetadataEqual(orig, exc.exceptions[0].__context__) self.assertMetadataEqual(orig, exc.exceptions[0].__cause__) def test_raise_handle_all_raise_two_unnamed(self): orig = ExceptionGroup("eg", [TypeError(1), ValueError(2)]) try: try: raise orig except* TypeError: e = sys.exception() raise SyntaxError(3) from e except* ValueError: e = sys.exception() raise SyntaxError(4) from e except ExceptionGroup as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("", [SyntaxError(3), SyntaxError(4)])) self.assertExceptionIsLike( exc.exceptions[0].__context__, ExceptionGroup("eg", [TypeError(1)])) self.assertExceptionIsLike( exc.exceptions[0].__cause__, ExceptionGroup("eg", [TypeError(1)])) self.assertExceptionIsLike( exc.exceptions[1].__context__, ExceptionGroup("eg", [ValueError(2)])) self.assertExceptionIsLike( exc.exceptions[1].__cause__, ExceptionGroup("eg", [ValueError(2)])) self.assertMetadataNotEqual(orig, exc) self.assertMetadataEqual(orig, exc.exceptions[0].__context__) self.assertMetadataEqual(orig, exc.exceptions[0].__cause__) self.assertMetadataEqual(orig, exc.exceptions[1].__context__) self.assertMetadataEqual(orig, exc.exceptions[1].__cause__) class TestExceptStarExceptionGroupSubclass(ExceptStarTest): def test_except_star_EG_subclass(self): class EG(ExceptionGroup): def __new__(cls, message, excs, code): obj = super().__new__(cls, message, excs) obj.code = code return obj def derive(self, excs): return EG(self.message, excs, self.code) try: try: try: try: raise TypeError(2) except TypeError as te: raise EG("nested", [te], 101) from None except EG as nested: try: raise ValueError(1) except ValueError as ve: raise EG("eg", [ve, nested], 42) except* ValueError as eg: veg = eg except EG as eg: teg = eg self.assertIsInstance(veg, EG) self.assertIsInstance(teg, EG) self.assertIsInstance(teg.exceptions[0], EG) self.assertMetadataEqual(veg, teg) self.assertEqual(veg.code, 42) self.assertEqual(teg.code, 42) self.assertEqual(teg.exceptions[0].code, 101) def test_falsy_exception_group_subclass(self): class FalsyEG(ExceptionGroup): def __bool__(self): return False def derive(self, excs): return FalsyEG(self.message, excs) try: try: raise FalsyEG("eg", [TypeError(1), ValueError(2)]) except *TypeError as e: tes = e raise except *ValueError as e: ves = e pass except Exception as e: exc = e for e in [tes, ves, exc]: self.assertFalse(e) self.assertIsInstance(e, FalsyEG) self.assertExceptionIsLike(exc, FalsyEG("eg", [TypeError(1)])) self.assertExceptionIsLike(tes, FalsyEG("eg", [TypeError(1)])) self.assertExceptionIsLike(ves, FalsyEG("eg", [ValueError(2)])) def test_exception_group_subclass_with_bad_split_func(self): # see gh-128049. class BadEG1(ExceptionGroup): def split(self, *args): return "NOT A 2-TUPLE!" class BadEG2(ExceptionGroup): def split(self, *args): return ("NOT A 2-TUPLE!",) eg_list = [ (BadEG1("eg", [OSError(123), ValueError(456)]), r"split must return a tuple, not str"), (BadEG2("eg", [OSError(123), ValueError(456)]), r"split must return a 2-tuple, got tuple of size 1") ] for eg_class, msg in eg_list: with self.assertRaisesRegex(TypeError, msg) as m: try: raise eg_class except* ValueError: pass except* OSError: pass self.assertExceptionIsLike(m.exception.__context__, eg_class) # we allow tuples of length > 2 for backwards compatibility class WeirdEG(ExceptionGroup): def split(self, *args): return super().split(*args) + ("anything", 123456, None) try: raise WeirdEG("eg", [OSError(123), ValueError(456)]) except* OSError as e: oeg = e except* ValueError as e: veg = e self.assertExceptionIsLike(oeg, WeirdEG("eg", [OSError(123)])) self.assertExceptionIsLike(veg, WeirdEG("eg", [ValueError(456)])) class TestExceptStarCleanup(ExceptStarTest): def test_sys_exception_restored(self): try: try: raise ValueError(42) except: try: raise TypeError(int) except* Exception: pass 1/0 except Exception as e: exc = e self.assertExceptionIsLike(exc, ZeroDivisionError('division by zero')) self.assertExceptionIsLike(exc.__context__, ValueError(42)) self.assertEqual(sys.exception(), None) class TestExceptStar_WeirdLeafExceptions(ExceptStarTest): # Test that except* works when leaf exceptions are # unhashable or have a bad custom __eq__ class UnhashableExc(ValueError): __hash__ = None class AlwaysEqualExc(ValueError): def __eq__(self, other): return True class NeverEqualExc(ValueError): def __eq__(self, other): return False class BrokenEqualExc(ValueError): def __eq__(self, other): raise RuntimeError() def setUp(self): self.bad_types = [self.UnhashableExc, self.AlwaysEqualExc, self.NeverEqualExc, self.BrokenEqualExc] def except_type(self, eg, type): match, rest = None, None try: try: raise eg except* type as e: match = e except Exception as e: rest = e return match, rest def test_catch_unhashable_leaf_exception(self): for Bad in self.bad_types: with self.subTest(Bad): eg = ExceptionGroup("eg", [TypeError(1), Bad(2)]) match, rest = self.except_type(eg, Bad) self.assertExceptionIsLike( match, ExceptionGroup("eg", [Bad(2)])) self.assertExceptionIsLike( rest, ExceptionGroup("eg", [TypeError(1)])) def test_propagate_unhashable_leaf(self): for Bad in self.bad_types: with self.subTest(Bad): eg = ExceptionGroup("eg", [TypeError(1), Bad(2)]) match, rest = self.except_type(eg, TypeError) self.assertExceptionIsLike( match, ExceptionGroup("eg", [TypeError(1)])) self.assertExceptionIsLike( rest, ExceptionGroup("eg", [Bad(2)])) def test_catch_nothing_unhashable_leaf(self): for Bad in self.bad_types: with self.subTest(Bad): eg = ExceptionGroup("eg", [TypeError(1), Bad(2)]) match, rest = self.except_type(eg, OSError) self.assertIsNone(match) self.assertExceptionIsLike(rest, eg) def test_catch_everything_unhashable_leaf(self): for Bad in self.bad_types: with self.subTest(Bad): eg = ExceptionGroup("eg", [TypeError(1), Bad(2)]) match, rest = self.except_type(eg, Exception) self.assertExceptionIsLike(match, eg) self.assertIsNone(rest) def test_reraise_unhashable_leaf(self): for Bad in self.bad_types: with self.subTest(Bad): eg = ExceptionGroup( "eg", [TypeError(1), Bad(2), ValueError(3)]) try: try: raise eg except* TypeError: pass except* Bad: raise except Exception as e: exc = e self.assertExceptionIsLike( exc, ExceptionGroup("eg", [Bad(2), ValueError(3)])) class TestExceptStar_WeirdExceptionGroupSubclass(ExceptStarTest): # Test that except* works with exception groups that are # unhashable or have a bad custom __eq__ class UnhashableEG(ExceptionGroup): __hash__ = None def derive(self, excs): return type(self)(self.message, excs) class AlwaysEqualEG(ExceptionGroup): def __eq__(self, other): return True def derive(self, excs): return type(self)(self.message, excs) class NeverEqualEG(ExceptionGroup): def __eq__(self, other): return False def derive(self, excs): return type(self)(self.message, excs) class BrokenEqualEG(ExceptionGroup): def __eq__(self, other): raise RuntimeError() def derive(self, excs): return type(self)(self.message, excs) def setUp(self): self.bad_types = [self.UnhashableEG, self.AlwaysEqualEG, self.NeverEqualEG, self.BrokenEqualEG] def except_type(self, eg, type): match, rest = None, None try: try: raise eg except* type as e: match = e except Exception as e: rest = e return match, rest def test_catch_some_unhashable_exception_group_subclass(self): for BadEG in self.bad_types: with self.subTest(BadEG): eg = BadEG("eg", [TypeError(1), BadEG("nested", [ValueError(2)])]) match, rest = self.except_type(eg, TypeError) self.assertExceptionIsLike(match, BadEG("eg", [TypeError(1)])) self.assertExceptionIsLike(rest, BadEG("eg", [BadEG("nested", [ValueError(2)])])) def test_catch_none_unhashable_exception_group_subclass(self): for BadEG in self.bad_types: with self.subTest(BadEG): eg = BadEG("eg", [TypeError(1), BadEG("nested", [ValueError(2)])]) match, rest = self.except_type(eg, OSError) self.assertIsNone(match) self.assertExceptionIsLike(rest, eg) def test_catch_all_unhashable_exception_group_subclass(self): for BadEG in self.bad_types: with self.subTest(BadEG): eg = BadEG("eg", [TypeError(1), BadEG("nested", [ValueError(2)])]) match, rest = self.except_type(eg, Exception) self.assertExceptionIsLike(match, eg) self.assertIsNone(rest) def test_reraise_unhashable_eg(self): for BadEG in self.bad_types: with self.subTest(BadEG): eg = BadEG("eg", [TypeError(1), ValueError(2), BadEG("nested", [ValueError(3), OSError(4)])]) try: try: raise eg except* ValueError: pass except* OSError: raise except Exception as e: exc = e self.assertExceptionIsLike( exc, BadEG("eg", [TypeError(1), BadEG("nested", [OSError(4)])])) if __name__ == '__main__': unittest.main()