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

gh-91090: Make started multiprocessing.Process instances and started multiprocessing.managers.BaseManager instances serialisable #31701

Open
wants to merge 26 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
d628fc1
Update popen_spawn_posix.py
geryogam Mar 5, 2022
e369d5e
Update popen_fork.py
geryogam Mar 5, 2022
590f446
Update popen_forkserver.py
geryogam Mar 5, 2022
941e4c0
Update popen_spawn_win32.py
geryogam Mar 5, 2022
e2a93dc
Update managers.py
geryogam Mar 5, 2022
2513628
Update popen_spawn_win32.py
geryogam Mar 5, 2022
c55302b
Update popen_fork.py
geryogam Mar 5, 2022
409349f
Update popen_forkserver.py
geryogam Mar 5, 2022
2eeeeb6
Update popen_spawn_posix.py
geryogam Mar 5, 2022
ade0b2a
Add unit tests
geryogam Nov 29, 2022
71eb4e1
Add a NEWS entry
geryogam Nov 30, 2022
1419e0d
Make State instances pickable with the old text protocol
geryogam Nov 30, 2022
1e53631
Add restoration of spawning_popen
geryogam Nov 30, 2022
56f0bb8
Revert "Update popen_spawn_posix.py"
geryogam Nov 30, 2022
9f76108
Revert "Update popen_forkserver.py"
geryogam Nov 30, 2022
5c7719d
Revert "Update popen_fork.py"
geryogam Nov 30, 2022
24b53f1
Revert "Update popen_spawn_win32.py"
geryogam Nov 30, 2022
04eac3c
Revert "Update managers.py"
geryogam Nov 30, 2022
9638a36
Revert "Update popen_spawn_win32.py"
geryogam Nov 30, 2022
242d027
Revert "Update popen_forkserver.py"
geryogam Nov 30, 2022
dbbbe1b
Revert "Update popen_fork.py"
geryogam Nov 30, 2022
88056fe
Revert "Update popen_spawn_posix.py"
geryogam Nov 30, 2022
f055d11
Use a simpler implementation of serialisability based on __getstate__…
geryogam Nov 30, 2022
3c6f5da
Restore finalizer attribute names
geryogam Nov 30, 2022
fe89f2d
Move __getstate__/__setstate__ to the right Popen classes
geryogam Nov 30, 2022
c2ab61e
Remove an empty line introduced by mistake in the last commit
geryogam Nov 30, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 15 additions & 0 deletions Lib/multiprocessing/managers.py
Original file line number Diff line number Diff line change
Expand Up @@ -476,6 +476,12 @@ class State(object):
STARTED = 1
SHUTDOWN = 2

def __getstate__(self):
return self.value

def __setstate__(self, state):
self.value = state

#
# Mapping from serializer name to Listener and Client types
#
Expand Down Expand Up @@ -731,6 +737,15 @@ def temp(self, /, *args, **kwds):
temp.__name__ = typeid
setattr(cls, typeid, temp)

def __getstate__(self):
state = vars(self).copy()
state['shutdown'] = state['shutdown']._key
return state

def __setstate__(self, state):
vars(self).update(state)
self.shutdown = util._finalizer_registry[self.shutdown]

#
# Subclass of set which get cleared after a fork
#
Expand Down
9 changes: 9 additions & 0 deletions Lib/multiprocessing/popen_fork.py
Original file line number Diff line number Diff line change
Expand Up @@ -81,3 +81,12 @@ def _launch(self, process_obj):
def close(self):
if self.finalizer is not None:
self.finalizer()

def __getstate__(self):
state = vars(self).copy()
state['finalizer'] = state['finalizer']._key
return state

def __setstate__(self, state):
vars(self).update(state)
self.finalizer = util._finalizer_registry[self.finalizer]
9 changes: 9 additions & 0 deletions Lib/multiprocessing/popen_spawn_win32.py
Original file line number Diff line number Diff line change
Expand Up @@ -129,3 +129,12 @@ def terminate(self):

def close(self):
self.finalizer()

def __getstate__(self):
state = vars(self).copy()
state['finalizer'] = state['finalizer']._key
return state

def __setstate__(self, state):
vars(self).update(state)
self.finalizer = util._finalizer_registry[self.finalizer]
38 changes: 37 additions & 1 deletion Lib/test/_test_multiprocessing.py
Original file line number Diff line number Diff line change
Expand Up @@ -807,6 +807,23 @@ def test_forkserver_sigkill(self):
if os.name != 'nt':
self.check_forkserver_death(signal.SIGKILL)

def test_process_pickling(self):
if self.TYPE == 'threads':
self.skipTest(f'test not appropriate for {self.TYPE}')
for proto in range(pickle.HIGHEST_PROTOCOL + 1):
with self.subTest(proto=proto):
proc = self.Process()
proc.start()
# Calling set_spawning_popen with a value other than None
# allows pickling the authentication keys of processes
# (.authkey), which is prevented by default in
# AuthenticationString for security reasons.
multiprocessing.context.set_spawning_popen(0)
serialized_proc = pickle.dumps(proc, protocol=proto)
multiprocessing.context.set_spawning_popen(None)
deserialized_proc = pickle.loads(serialized_proc)
self.assertIsInstance(deserialized_proc, self.Process)


#
#
Expand Down Expand Up @@ -2920,7 +2937,26 @@ def test_mymanager_context_prestarted(self):
manager.start()
with manager:
self.common(manager)
self.assertEqual(manager._process.exitcode, 0)
# bpo-30356: BaseManager._finalize_manager() sends SIGTERM
# to the manager process if it takes longer than 1 second to stop,
# which happens on slow buildbots.
self.assertIn(manager._process.exitcode, (0, -signal.SIGTERM))

def test_mymanager_pickling(self):
for proto in range(pickle.HIGHEST_PROTOCOL + 1):
with self.subTest(proto=proto):
manager = MyManager()
manager.start()
# Calling set_spawning_popen with a value other than None
# allows pickling the authentication keys of processes
# (.authkey), which is prevented by default in
# AuthenticationString for security reasons.
multiprocessing.context.set_spawning_popen(0)
serialized_manager = pickle.dumps(manager, protocol=proto)
multiprocessing.context.set_spawning_popen(None)
deserialized_manager = pickle.loads(serialized_manager)
self.assertIsInstance(deserialized_manager, MyManager)
manager.shutdown()

def common(self, manager):
foo = manager.Foo()
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
Make started :class:`multiprocessing.Process` instances and started
:class:`multiprocessing.managers.BaseManager` instances serialisable. Patch by
Géry Ogam.