diff -r fae92299eb68 Doc/library/concurrent.futures.rst --- a/Doc/library/concurrent.futures.rst Sat Jan 25 13:27:53 2014 -0500 +++ b/Doc/library/concurrent.futures.rst Sat Jan 25 22:24:17 2014 -0500 @@ -371,7 +371,8 @@ Returns an iterator over the :class:`Future` instances (possibly created by different :class:`Executor` instances) given by *fs* that yields futures as - they complete (finished or were cancelled). Any futures that completed + they complete (finished or were cancelled). Any futures given by *fs* that + are duplicated will be returned once. Any futures that completed before :func:`as_completed` is called will be yielded first. The returned iterator raises a :exc:`TimeoutError` if :meth:`~iterator.__next__` is called and the result isn't available after *timeout* seconds from the diff -r fae92299eb68 Lib/concurrent/futures/_base.py --- a/Lib/concurrent/futures/_base.py Sat Jan 25 13:27:53 2014 -0500 +++ b/Lib/concurrent/futures/_base.py Sat Jan 25 22:24:17 2014 -0500 @@ -181,7 +181,8 @@ Returns: An iterator that yields the given Futures as they complete (finished or - cancelled). + cancelled). If any given Futures are duplicated, they will be returned + once. Raises: TimeoutError: If the entire result iterator could not be generated @@ -190,11 +191,12 @@ if timeout is not None: end_time = timeout + time.time() + fs = set(fs) with _AcquireFutures(fs): finished = set( f for f in fs if f._state in [CANCELLED_AND_NOTIFIED, FINISHED]) - pending = set(fs) - finished + pending = fs - finished waiter = _create_and_install_waiters(fs, _AS_COMPLETED) try: diff -r fae92299eb68 Lib/test/test_concurrent_futures.py --- a/Lib/test/test_concurrent_futures.py Sat Jan 25 13:27:53 2014 -0500 +++ b/Lib/test/test_concurrent_futures.py Sat Jan 25 22:24:17 2014 -0500 @@ -350,6 +350,13 @@ SUCCESSFUL_FUTURE]), completed_futures) + def test_duplicate_futures(self): + # Issue 20367. Duplicate futures should not raise exceptions or give + # duplicate responses. + future1 = self.executor.submit(time.sleep, 2) + completed = [f for f in futures.as_completed([future1,future1])] + self.assertEqual(len(completed), 1) + class ThreadPoolAsCompletedTests(ThreadPoolMixin, AsCompletedTests, unittest.TestCase): pass