Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

ProcessPoolExecutor swallows falsy Exceptions #132063

Open
graingert opened this issue Apr 4, 2025 · 6 comments
Open

ProcessPoolExecutor swallows falsy Exceptions #132063

graingert opened this issue Apr 4, 2025 · 6 comments
Labels
3.12 bugs and security fixes 3.13 bugs and security fixes 3.14 new features, bugs and security fixes stdlib Python modules in the Lib dir topic-multiprocessing type-bug An unexpected behavior, bug, or error

Comments

@graingert
Copy link
Contributor

graingert commented Apr 4, 2025

Bug report

Bug description:

import sys
import concurrent.futures

class MyException(Exception):

    def __bool__(self):
        return False


def raiser():
    raise MyException("foo")

def main():
    with concurrent.futures.ProcessPoolExecutor() as p:
        print(p.submit(raiser).result())

if __name__ == "__main__":
    sys.exit(main())

This program prints None, but should fail and print a traceback

CPython versions tested on:

CPython main branch, 3.14, 3.13, 3.12, 3.11, 3.10, 3.9

Operating systems tested on:

Linux

@graingert graingert added stdlib Python modules in the Lib dir type-bug An unexpected behavior, bug, or error labels Apr 4, 2025
@tomasr8 tomasr8 added 3.12 bugs and security fixes topic-multiprocessing 3.13 bugs and security fixes 3.14 new features, bugs and security fixes labels Apr 4, 2025
@YvesDup
Copy link
Contributor

YvesDup commented Apr 4, 2025

I think there is something wrong here in the __get_result(self) method of Future class:

if self._exception:
try:
raise self._exception
finally:
# Break a reference cycle with the exception in self._exception
self = None
else:
return self._result

The test should be: if self._exception is not None: because if self._exception calls the __bool__ method of exception.

Edit: same problem here on line 443

# work_item can be None if another process terminated (see above)
if work_item is not None:
if result_item.exception:
work_item.future.set_exception(result_item.exception)
else:
work_item.future.set_result(result_item.result)

@picnixz
Copy link
Member

picnixz commented Apr 4, 2025

The test should be: if self._exception is not None: because if self._exception calls the bool method of exception.

Well spotted. Indeed, this should be changed to is not None. More generally, are there other places in the stdlib (or in mp-related code) where we have this issue?

@YvesDup
Copy link
Contributor

YvesDup commented Apr 4, 2025

More generally, are there other places in the stdlib

I saw these tests this morning:

May I open a PR ?

@picnixz
Copy link
Member

picnixz commented Apr 4, 2025

Unless they can make a silly bug, I think it's fine to keep them. However, unless @graingert wants to work on the issue itself, you may first fix the __get_result method first.

@YvesDup
Copy link
Contributor

YvesDup commented Apr 4, 2025

It's mandatory to work on both files /Lib/concurrent/futures/process.py and /Lib/concurrent/futures/_base.pyto fix this issue.

This should fix the same issue with a ThreadPoolExcecutor.

@graingert
Copy link
Contributor Author

graingert commented Apr 4, 2025 via email

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
3.12 bugs and security fixes 3.13 bugs and security fixes 3.14 new features, bugs and security fixes stdlib Python modules in the Lib dir topic-multiprocessing type-bug An unexpected behavior, bug, or error
Projects
None yet
Development

No branches or pull requests

4 participants