2023-08-24 14:21:44 -03:00
|
|
|
import sys
|
|
|
|
import threading
|
|
|
|
import time
|
|
|
|
import unittest
|
|
|
|
from concurrent import futures
|
2023-09-05 11:56:30 -03:00
|
|
|
from test import support
|
2023-08-24 14:21:44 -03: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 11:56:30 -03:00
|
|
|
@support.requires_resource('walltime')
|
2023-08-24 14:21:44 -03: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 10:21:15 -03:00
|
|
|
short_timeout = 0.050
|
|
|
|
long_timeout = short_timeout * 10
|
|
|
|
|
|
|
|
future = self.executor.submit(time.sleep, long_timeout)
|
2023-08-24 14:21:44 -03:00
|
|
|
|
|
|
|
finished, pending = futures.wait(
|
|
|
|
[CANCELLED_AND_NOTIFIED_FUTURE,
|
|
|
|
EXCEPTION_FUTURE,
|
|
|
|
SUCCESSFUL_FUTURE,
|
2023-09-28 10:21:15 -03:00
|
|
|
future],
|
|
|
|
timeout=short_timeout,
|
2023-08-24 14:21:44 -03:00
|
|
|
return_when=futures.ALL_COMPLETED)
|
|
|
|
|
|
|
|
self.assertEqual(set([CANCELLED_AND_NOTIFIED_FUTURE,
|
|
|
|
EXCEPTION_FUTURE,
|
2023-09-28 10:21:15 -03:00
|
|
|
SUCCESSFUL_FUTURE]),
|
|
|
|
finished)
|
|
|
|
self.assertEqual(set([future]), pending)
|
2023-08-24 14:21:44 -03: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 17:57:36 -03:00
|
|
|
support.setswitchinterval(1e-6)
|
2023-08-24 14:21:44 -03: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()
|