| 
									
										
										
										
											2023-08-24 19:21:44 +02:00
										 |  |  | import sys | 
					
						
							|  |  |  | import threading | 
					
						
							|  |  |  | import time | 
					
						
							|  |  |  | import unittest | 
					
						
							|  |  |  | from concurrent import futures | 
					
						
							| 
									
										
										
										
											2023-09-05 17:56:30 +03:00
										 |  |  | from test import support | 
					
						
							| 
									
										
										
										
											2023-08-24 19:21:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | from .util import ( | 
					
						
							|  |  |  |     CANCELLED_FUTURE, CANCELLED_AND_NOTIFIED_FUTURE, EXCEPTION_FUTURE, | 
					
						
							|  |  |  |     SUCCESSFUL_FUTURE, | 
					
						
							|  |  |  |     create_executor_tests, setup_module, | 
					
						
							|  |  |  |     BaseTestCase, ThreadPoolMixin, | 
					
						
							|  |  |  |     ProcessPoolForkMixin, ProcessPoolForkserverMixin, ProcessPoolSpawnMixin) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def mul(x, y): | 
					
						
							|  |  |  |     return x * y | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def sleep_and_raise(t): | 
					
						
							|  |  |  |     time.sleep(t) | 
					
						
							|  |  |  |     raise Exception('this is an exception') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class WaitTests: | 
					
						
							|  |  |  |     def test_20369(self): | 
					
						
							|  |  |  |         # See https://bugs.python.org/issue20369 | 
					
						
							|  |  |  |         future = self.executor.submit(time.sleep, 1.5) | 
					
						
							|  |  |  |         done, not_done = futures.wait([future, future], | 
					
						
							|  |  |  |                             return_when=futures.ALL_COMPLETED) | 
					
						
							|  |  |  |         self.assertEqual({future}, done) | 
					
						
							|  |  |  |         self.assertEqual(set(), not_done) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_first_completed(self): | 
					
						
							|  |  |  |         future1 = self.executor.submit(mul, 21, 2) | 
					
						
							|  |  |  |         future2 = self.executor.submit(time.sleep, 1.5) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         done, not_done = futures.wait( | 
					
						
							|  |  |  |                 [CANCELLED_FUTURE, future1, future2], | 
					
						
							|  |  |  |                  return_when=futures.FIRST_COMPLETED) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(set([future1]), done) | 
					
						
							|  |  |  |         self.assertEqual(set([CANCELLED_FUTURE, future2]), not_done) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_first_completed_some_already_completed(self): | 
					
						
							|  |  |  |         future1 = self.executor.submit(time.sleep, 1.5) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         finished, pending = futures.wait( | 
					
						
							|  |  |  |                  [CANCELLED_AND_NOTIFIED_FUTURE, SUCCESSFUL_FUTURE, future1], | 
					
						
							|  |  |  |                  return_when=futures.FIRST_COMPLETED) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual( | 
					
						
							|  |  |  |                 set([CANCELLED_AND_NOTIFIED_FUTURE, SUCCESSFUL_FUTURE]), | 
					
						
							|  |  |  |                 finished) | 
					
						
							|  |  |  |         self.assertEqual(set([future1]), pending) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-05 17:56:30 +03:00
										 |  |  |     @support.requires_resource('walltime') | 
					
						
							| 
									
										
										
										
											2023-08-24 19:21:44 +02:00
										 |  |  |     def test_first_exception(self): | 
					
						
							|  |  |  |         future1 = self.executor.submit(mul, 2, 21) | 
					
						
							|  |  |  |         future2 = self.executor.submit(sleep_and_raise, 1.5) | 
					
						
							|  |  |  |         future3 = self.executor.submit(time.sleep, 3) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         finished, pending = futures.wait( | 
					
						
							|  |  |  |                 [future1, future2, future3], | 
					
						
							|  |  |  |                 return_when=futures.FIRST_EXCEPTION) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(set([future1, future2]), finished) | 
					
						
							|  |  |  |         self.assertEqual(set([future3]), pending) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_first_exception_some_already_complete(self): | 
					
						
							|  |  |  |         future1 = self.executor.submit(divmod, 21, 0) | 
					
						
							|  |  |  |         future2 = self.executor.submit(time.sleep, 1.5) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         finished, pending = futures.wait( | 
					
						
							|  |  |  |                 [SUCCESSFUL_FUTURE, | 
					
						
							|  |  |  |                  CANCELLED_FUTURE, | 
					
						
							|  |  |  |                  CANCELLED_AND_NOTIFIED_FUTURE, | 
					
						
							|  |  |  |                  future1, future2], | 
					
						
							|  |  |  |                 return_when=futures.FIRST_EXCEPTION) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(set([SUCCESSFUL_FUTURE, | 
					
						
							|  |  |  |                               CANCELLED_AND_NOTIFIED_FUTURE, | 
					
						
							|  |  |  |                               future1]), finished) | 
					
						
							|  |  |  |         self.assertEqual(set([CANCELLED_FUTURE, future2]), pending) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_first_exception_one_already_failed(self): | 
					
						
							|  |  |  |         future1 = self.executor.submit(time.sleep, 2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         finished, pending = futures.wait( | 
					
						
							|  |  |  |                  [EXCEPTION_FUTURE, future1], | 
					
						
							|  |  |  |                  return_when=futures.FIRST_EXCEPTION) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(set([EXCEPTION_FUTURE]), finished) | 
					
						
							|  |  |  |         self.assertEqual(set([future1]), pending) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_all_completed(self): | 
					
						
							|  |  |  |         future1 = self.executor.submit(divmod, 2, 0) | 
					
						
							|  |  |  |         future2 = self.executor.submit(mul, 2, 21) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         finished, pending = futures.wait( | 
					
						
							|  |  |  |                 [SUCCESSFUL_FUTURE, | 
					
						
							|  |  |  |                  CANCELLED_AND_NOTIFIED_FUTURE, | 
					
						
							|  |  |  |                  EXCEPTION_FUTURE, | 
					
						
							|  |  |  |                  future1, | 
					
						
							|  |  |  |                  future2], | 
					
						
							|  |  |  |                 return_when=futures.ALL_COMPLETED) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(set([SUCCESSFUL_FUTURE, | 
					
						
							|  |  |  |                               CANCELLED_AND_NOTIFIED_FUTURE, | 
					
						
							|  |  |  |                               EXCEPTION_FUTURE, | 
					
						
							|  |  |  |                               future1, | 
					
						
							|  |  |  |                               future2]), finished) | 
					
						
							|  |  |  |         self.assertEqual(set(), pending) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_timeout(self): | 
					
						
							| 
									
										
										
										
											2023-09-28 15:21:15 +02:00
										 |  |  |         short_timeout = 0.050 | 
					
						
							|  |  |  |         long_timeout = short_timeout * 10 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         future = self.executor.submit(time.sleep, long_timeout) | 
					
						
							| 
									
										
										
										
											2023-08-24 19:21:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         finished, pending = futures.wait( | 
					
						
							|  |  |  |                 [CANCELLED_AND_NOTIFIED_FUTURE, | 
					
						
							|  |  |  |                  EXCEPTION_FUTURE, | 
					
						
							|  |  |  |                  SUCCESSFUL_FUTURE, | 
					
						
							| 
									
										
										
										
											2023-09-28 15:21:15 +02:00
										 |  |  |                  future], | 
					
						
							|  |  |  |                 timeout=short_timeout, | 
					
						
							| 
									
										
										
										
											2023-08-24 19:21:44 +02:00
										 |  |  |                 return_when=futures.ALL_COMPLETED) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(set([CANCELLED_AND_NOTIFIED_FUTURE, | 
					
						
							|  |  |  |                               EXCEPTION_FUTURE, | 
					
						
							| 
									
										
										
										
											2023-09-28 15:21:15 +02:00
										 |  |  |                               SUCCESSFUL_FUTURE]), | 
					
						
							|  |  |  |                          finished) | 
					
						
							|  |  |  |         self.assertEqual(set([future]), pending) | 
					
						
							| 
									
										
										
										
											2023-08-24 19:21:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ThreadPoolWaitTests(ThreadPoolMixin, WaitTests, BaseTestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_pending_calls_race(self): | 
					
						
							|  |  |  |         # Issue #14406: multi-threaded race condition when waiting on all | 
					
						
							|  |  |  |         # futures. | 
					
						
							|  |  |  |         event = threading.Event() | 
					
						
							|  |  |  |         def future_func(): | 
					
						
							|  |  |  |             event.wait() | 
					
						
							|  |  |  |         oldswitchinterval = sys.getswitchinterval() | 
					
						
							| 
									
										
										
										
											2024-04-05 21:57:36 +01:00
										 |  |  |         support.setswitchinterval(1e-6) | 
					
						
							| 
									
										
										
										
											2023-08-24 19:21:44 +02:00
										 |  |  |         try: | 
					
						
							|  |  |  |             fs = {self.executor.submit(future_func) for i in range(100)} | 
					
						
							|  |  |  |             event.set() | 
					
						
							|  |  |  |             futures.wait(fs, return_when=futures.ALL_COMPLETED) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             sys.setswitchinterval(oldswitchinterval) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | create_executor_tests(globals(), WaitTests, | 
					
						
							|  |  |  |                       executor_mixins=(ProcessPoolForkMixin, | 
					
						
							|  |  |  |                                        ProcessPoolForkserverMixin, | 
					
						
							|  |  |  |                                        ProcessPoolSpawnMixin)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def setUpModule(): | 
					
						
							|  |  |  |     setup_module() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							|  |  |  |     unittest.main() |