mirror of
				https://github.com/python/cpython.git
				synced 2025-10-25 18:54:53 +00:00 
			
		
		
		
	GH-95704: Don't suppress errors from tasks when TG is cancelled (#95761)
When a task catches CancelledError and raises some other error, the other error should not silently be suppressed. Any scenario where a task crashes in cleanup upon cancellation will now result in an ExceptionGroup wrapping the crash(es) instead of propagating CancelledError and ignoring the side errors. NOTE: This represents a change in behavior (hence the need to change several tests). But it is only an edge case. Co-authored-by: Thomas Grainger <tagrain@gmail.com>
This commit is contained in:
		
							parent
							
								
									9b30b965f0
								
							
						
					
					
						commit
						f51f54f39d
					
				
					 3 changed files with 35 additions and 28 deletions
				
			
		|  | @ -230,29 +230,29 @@ async def runner(): | |||
| 
 | ||||
|         self.assertEqual(NUM, 15) | ||||
| 
 | ||||
|     async def test_cancellation_in_body(self): | ||||
|     async def test_taskgroup_08(self): | ||||
| 
 | ||||
|         async def foo(): | ||||
|             await asyncio.sleep(0.1) | ||||
|             1 / 0 | ||||
|             try: | ||||
|                 await asyncio.sleep(10) | ||||
|             finally: | ||||
|                 1 / 0 | ||||
| 
 | ||||
|         async def runner(): | ||||
|             async with taskgroups.TaskGroup() as g: | ||||
|                 for _ in range(5): | ||||
|                     g.create_task(foo()) | ||||
| 
 | ||||
|                 try: | ||||
|                     await asyncio.sleep(10) | ||||
|                 except asyncio.CancelledError: | ||||
|                     raise | ||||
|                 await asyncio.sleep(10) | ||||
| 
 | ||||
|         r = asyncio.create_task(runner()) | ||||
|         await asyncio.sleep(0.1) | ||||
| 
 | ||||
|         self.assertFalse(r.done()) | ||||
|         r.cancel() | ||||
|         with self.assertRaises(asyncio.CancelledError) as cm: | ||||
|         with self.assertRaises(ExceptionGroup) as cm: | ||||
|             await r | ||||
|         self.assertEqual(get_error_types(cm.exception), {ZeroDivisionError}) | ||||
| 
 | ||||
|     async def test_taskgroup_09(self): | ||||
| 
 | ||||
|  | @ -316,8 +316,10 @@ async def runner(): | |||
|     async def test_taskgroup_11(self): | ||||
| 
 | ||||
|         async def foo(): | ||||
|             await asyncio.sleep(0.1) | ||||
|             1 / 0 | ||||
|             try: | ||||
|                 await asyncio.sleep(10) | ||||
|             finally: | ||||
|                 1 / 0 | ||||
| 
 | ||||
|         async def runner(): | ||||
|             async with taskgroups.TaskGroup(): | ||||
|  | @ -325,24 +327,26 @@ async def runner(): | |||
|                     for _ in range(5): | ||||
|                         g2.create_task(foo()) | ||||
| 
 | ||||
|                     try: | ||||
|                         await asyncio.sleep(10) | ||||
|                     except asyncio.CancelledError: | ||||
|                         raise | ||||
|                     await asyncio.sleep(10) | ||||
| 
 | ||||
|         r = asyncio.create_task(runner()) | ||||
|         await asyncio.sleep(0.1) | ||||
| 
 | ||||
|         self.assertFalse(r.done()) | ||||
|         r.cancel() | ||||
|         with self.assertRaises(asyncio.CancelledError): | ||||
|         with self.assertRaises(ExceptionGroup) as cm: | ||||
|             await r | ||||
| 
 | ||||
|         self.assertEqual(get_error_types(cm.exception), {ExceptionGroup}) | ||||
|         self.assertEqual(get_error_types(cm.exception.exceptions[0]), {ZeroDivisionError}) | ||||
| 
 | ||||
|     async def test_taskgroup_12(self): | ||||
| 
 | ||||
|         async def foo(): | ||||
|             await asyncio.sleep(0.1) | ||||
|             1 / 0 | ||||
|             try: | ||||
|                 await asyncio.sleep(10) | ||||
|             finally: | ||||
|                 1 / 0 | ||||
| 
 | ||||
|         async def runner(): | ||||
|             async with taskgroups.TaskGroup() as g1: | ||||
|  | @ -352,19 +356,19 @@ async def runner(): | |||
|                     for _ in range(5): | ||||
|                         g2.create_task(foo()) | ||||
| 
 | ||||
|                     try: | ||||
|                         await asyncio.sleep(10) | ||||
|                     except asyncio.CancelledError: | ||||
|                         raise | ||||
|                     await asyncio.sleep(10) | ||||
| 
 | ||||
|         r = asyncio.create_task(runner()) | ||||
|         await asyncio.sleep(0.1) | ||||
| 
 | ||||
|         self.assertFalse(r.done()) | ||||
|         r.cancel() | ||||
|         with self.assertRaises(asyncio.CancelledError): | ||||
|         with self.assertRaises(ExceptionGroup) as cm: | ||||
|             await r | ||||
| 
 | ||||
|         self.assertEqual(get_error_types(cm.exception), {ExceptionGroup}) | ||||
|         self.assertEqual(get_error_types(cm.exception.exceptions[0]), {ZeroDivisionError}) | ||||
| 
 | ||||
|     async def test_taskgroup_13(self): | ||||
| 
 | ||||
|         async def crash_after(t): | ||||
|  | @ -424,8 +428,9 @@ async def runner(): | |||
| 
 | ||||
|         self.assertFalse(r.done()) | ||||
|         r.cancel() | ||||
|         with self.assertRaises(asyncio.CancelledError): | ||||
|         with self.assertRaises(ExceptionGroup) as cm: | ||||
|             await r | ||||
|         self.assertEqual(get_error_types(cm.exception), {ZeroDivisionError}) | ||||
| 
 | ||||
|     async def test_taskgroup_16(self): | ||||
| 
 | ||||
|  | @ -451,8 +456,9 @@ async def runner(): | |||
| 
 | ||||
|         self.assertFalse(r.done()) | ||||
|         r.cancel() | ||||
|         with self.assertRaises(asyncio.CancelledError): | ||||
|         with self.assertRaises(ExceptionGroup) as cm: | ||||
|             await r | ||||
|         self.assertEqual(get_error_types(cm.exception), {ZeroDivisionError}) | ||||
| 
 | ||||
|     async def test_taskgroup_17(self): | ||||
|         NUM = 0 | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Guido van Rossum
						Guido van Rossum