| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2019-02-23 22:08:16 -06:00
										 |  |  | # Module providing manager classes for dealing | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | # with shared objects | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # multiprocessing/managers.py | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2010-12-14 01:38:16 +00:00
										 |  |  | # Copyright (c) 2006-2008, R Oudkerk | 
					
						
							| 
									
										
										
										
											2012-04-30 12:13:55 +01:00
										 |  |  | # Licensed to PSF under a Contributor Agreement. | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-23 22:08:16 -06:00
										 |  |  | __all__ = [ 'BaseManager', 'SyncManager', 'BaseProxy', 'Token', | 
					
						
							|  |  |  |             'SharedMemoryManager' ] | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Imports | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | import threading | 
					
						
							| 
									
										
										
										
											2019-05-10 20:42:35 +02:00
										 |  |  | import signal | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | import array | 
					
						
							|  |  |  | import queue | 
					
						
							| 
									
										
										
										
											2018-07-06 13:51:52 +02:00
										 |  |  | import time | 
					
						
							| 
									
										
										
										
											2020-04-10 17:46:36 +03:00
										 |  |  | import types | 
					
						
							| 
									
										
										
										
											2019-05-13 21:15:32 +02:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2019-02-23 22:08:16 -06:00
										 |  |  | from os import getpid | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | from traceback import format_exc | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from . import connection | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  | from .context import reduction, get_spawning_popen, ProcessError | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | from . import pool | 
					
						
							|  |  |  | from . import process | 
					
						
							|  |  |  | from . import util | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  | from . import get_context | 
					
						
							| 
									
										
										
										
											2019-02-23 22:08:16 -06:00
										 |  |  | try: | 
					
						
							|  |  |  |     from . import shared_memory | 
					
						
							|  |  |  |     HAS_SHMEM = True | 
					
						
							|  |  |  | except ImportError: | 
					
						
							|  |  |  |     HAS_SHMEM = False | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Register some things for pickling | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def reduce_array(a): | 
					
						
							| 
									
										
										
										
											2010-09-01 20:29:34 +00:00
										 |  |  |     return array.array, (a.typecode, a.tobytes()) | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | reduction.register(array.array, reduce_array) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | view_types = [type(getattr({}, name)()) for name in ('items','keys','values')] | 
					
						
							| 
									
										
										
										
											2008-06-25 13:04:48 +00:00
										 |  |  | if view_types[0] is not list:       # only needed in Py3.0 | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     def rebuild_as_list(obj): | 
					
						
							|  |  |  |         return list, (list(obj),) | 
					
						
							|  |  |  |     for view_type in view_types: | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |         reduction.register(view_type, rebuild_as_list) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Type for identifying shared objects | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Token(object): | 
					
						
							|  |  |  |     '''
 | 
					
						
							| 
									
										
										
										
											2020-04-18 14:58:29 +08:00
										 |  |  |     Type to uniquely identify a shared object | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     '''
 | 
					
						
							|  |  |  |     __slots__ = ('typeid', 'address', 'id') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, typeid, address, id): | 
					
						
							|  |  |  |         (self.typeid, self.address, self.id) = (typeid, address, id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __getstate__(self): | 
					
						
							|  |  |  |         return (self.typeid, self.address, self.id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __setstate__(self, state): | 
					
						
							|  |  |  |         (self.typeid, self.address, self.id) = state | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							| 
									
										
										
										
											2014-07-25 23:36:00 +03:00
										 |  |  |         return '%s(typeid=%r, address=%r, id=%r)' % \ | 
					
						
							|  |  |  |                (self.__class__.__name__, self.typeid, self.address, self.id) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Function for communication with a manager's server process | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def dispatch(c, id, methodname, args=(), kwds={}): | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     Send a message to manager using connection `c` and return response | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     c.send((id, methodname, args, kwds)) | 
					
						
							|  |  |  |     kind, result = c.recv() | 
					
						
							|  |  |  |     if kind == '#RETURN': | 
					
						
							|  |  |  |         return result | 
					
						
							|  |  |  |     raise convert_to_error(kind, result) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def convert_to_error(kind, result): | 
					
						
							|  |  |  |     if kind == '#ERROR': | 
					
						
							|  |  |  |         return result | 
					
						
							| 
									
										
										
										
											2017-08-12 10:37:09 -05:00
										 |  |  |     elif kind in ('#TRACEBACK', '#UNSERIALIZABLE'): | 
					
						
							|  |  |  |         if not isinstance(result, str): | 
					
						
							|  |  |  |             raise TypeError( | 
					
						
							|  |  |  |                 "Result {0!r} (kind '{1}') type is {2}, not str".format( | 
					
						
							|  |  |  |                     result, kind, type(result))) | 
					
						
							|  |  |  |         if kind == '#UNSERIALIZABLE': | 
					
						
							|  |  |  |             return RemoteError('Unserializable message: %s\n' % result) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return RemoteError(result) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2017-08-12 10:37:09 -05:00
										 |  |  |         return ValueError('Unrecognized message type {!r}'.format(kind)) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class RemoteError(Exception): | 
					
						
							|  |  |  |     def __str__(self): | 
					
						
							|  |  |  |         return ('\n' + '-'*75 + '\n' + str(self.args[0]) + '-'*75) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Functions for finding the method names of an object | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def all_methods(obj): | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     Return a list of names of methods of `obj` | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     temp = [] | 
					
						
							|  |  |  |     for name in dir(obj): | 
					
						
							|  |  |  |         func = getattr(obj, name) | 
					
						
							| 
									
										
										
										
											2011-10-28 14:45:05 +02:00
										 |  |  |         if callable(func): | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             temp.append(name) | 
					
						
							|  |  |  |     return temp | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def public_methods(obj): | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     Return a list of names of methods of `obj` which do not start with '_' | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     return [name for name in all_methods(obj) if name[0] != '_'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Server which is run in a process controlled by a manager | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Server(object): | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     Server class which runs in a process controlled by a manager object | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     public = ['shutdown', 'create', 'accept_connection', 'get_methods', | 
					
						
							|  |  |  |               'debug_info', 'number_of_objects', 'dummy', 'incref', 'decref'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, registry, address, authkey, serializer): | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  |         if not isinstance(authkey, bytes): | 
					
						
							|  |  |  |             raise TypeError( | 
					
						
							|  |  |  |                 "Authkey {0!r} is type {1!s}, not bytes".format( | 
					
						
							|  |  |  |                     authkey, type(authkey))) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         self.registry = registry | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |         self.authkey = process.AuthenticationString(authkey) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         Listener, Client = listener_client[serializer] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # do authentication later | 
					
						
							| 
									
										
										
										
											2011-12-23 19:06:48 +01:00
										 |  |  |         self.listener = Listener(address=address, backlog=16) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         self.address = self.listener.address | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-21 02:15:48 +00:00
										 |  |  |         self.id_to_obj = {'0': (None, ())} | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         self.id_to_refcount = {} | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  |         self.id_to_local_proxy_obj = {} | 
					
						
							|  |  |  |         self.mutex = threading.Lock() | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def serve_forever(self): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Run the server forever | 
					
						
							|  |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2012-06-14 15:30:10 +01:00
										 |  |  |         self.stop_event = threading.Event() | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |         process.current_process()._manager_server = self | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2012-06-14 15:30:10 +01:00
										 |  |  |             accepter = threading.Thread(target=self.accepter) | 
					
						
							|  |  |  |             accepter.daemon = True | 
					
						
							|  |  |  |             accepter.start() | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2012-06-14 15:30:10 +01:00
										 |  |  |                 while not self.stop_event.is_set(): | 
					
						
							|  |  |  |                     self.stop_event.wait(1) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             except (KeyboardInterrupt, SystemExit): | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  |         finally: | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  |             if sys.stdout != sys.__stdout__: # what about stderr? | 
					
						
							| 
									
										
										
										
											2012-06-14 15:30:10 +01:00
										 |  |  |                 util.debug('resetting stdout, stderr') | 
					
						
							|  |  |  |                 sys.stdout = sys.__stdout__ | 
					
						
							|  |  |  |                 sys.stderr = sys.__stderr__ | 
					
						
							|  |  |  |             sys.exit(0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def accepter(self): | 
					
						
							|  |  |  |         while True: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 c = self.listener.accept() | 
					
						
							| 
									
										
										
										
											2012-12-25 16:47:37 +02:00
										 |  |  |             except OSError: | 
					
						
							| 
									
										
										
										
											2012-06-14 15:30:10 +01:00
										 |  |  |                 continue | 
					
						
							|  |  |  |             t = threading.Thread(target=self.handle_request, args=(c,)) | 
					
						
							|  |  |  |             t.daemon = True | 
					
						
							|  |  |  |             t.start() | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-16 19:42:34 +02:00
										 |  |  |     def _handle_request(self, c): | 
					
						
							|  |  |  |         request = None | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         try: | 
					
						
							|  |  |  |             connection.deliver_challenge(c, self.authkey) | 
					
						
							|  |  |  |             connection.answer_challenge(c, self.authkey) | 
					
						
							|  |  |  |             request = c.recv() | 
					
						
							|  |  |  |             ignore, funcname, args, kwds = request | 
					
						
							|  |  |  |             assert funcname in self.public, '%r unrecognized' % funcname | 
					
						
							|  |  |  |             func = getattr(self, funcname) | 
					
						
							|  |  |  |         except Exception: | 
					
						
							|  |  |  |             msg = ('#TRACEBACK', format_exc()) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 result = func(c, *args, **kwds) | 
					
						
							|  |  |  |             except Exception: | 
					
						
							|  |  |  |                 msg = ('#TRACEBACK', format_exc()) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 msg = ('#RETURN', result) | 
					
						
							| 
									
										
										
										
											2021-04-16 19:42:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         try: | 
					
						
							|  |  |  |             c.send(msg) | 
					
						
							|  |  |  |         except Exception as e: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 c.send(('#TRACEBACK', format_exc())) | 
					
						
							|  |  |  |             except Exception: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  |             util.info('Failure to send message: %r', msg) | 
					
						
							|  |  |  |             util.info(' ... request was %r', request) | 
					
						
							|  |  |  |             util.info(' ... exception was %r', e) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-16 19:42:34 +02:00
										 |  |  |     def handle_request(self, conn): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Handle a new connection | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             self._handle_request(conn) | 
					
						
							|  |  |  |         except SystemExit: | 
					
						
							|  |  |  |             # Server.serve_client() calls sys.exit(0) on EOF | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             conn.close() | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def serve_client(self, conn): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Handle requests from the proxies in a particular process/thread | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         util.debug('starting server thread to service %r', | 
					
						
							| 
									
										
										
										
											2008-08-18 18:09:21 +00:00
										 |  |  |                    threading.current_thread().name) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         recv = conn.recv | 
					
						
							|  |  |  |         send = conn.send | 
					
						
							|  |  |  |         id_to_obj = self.id_to_obj | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-14 15:30:10 +01:00
										 |  |  |         while not self.stop_event.is_set(): | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 methodname = obj = None | 
					
						
							|  |  |  |                 request = recv() | 
					
						
							|  |  |  |                 ident, methodname, args, kwds = request | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  |                 try: | 
					
						
							|  |  |  |                     obj, exposed, gettypeid = id_to_obj[ident] | 
					
						
							|  |  |  |                 except KeyError as ke: | 
					
						
							|  |  |  |                     try: | 
					
						
							|  |  |  |                         obj, exposed, gettypeid = \ | 
					
						
							|  |  |  |                             self.id_to_local_proxy_obj[ident] | 
					
						
							| 
									
										
										
										
											2019-11-19 21:34:03 +00:00
										 |  |  |                     except KeyError: | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  |                         raise ke | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if methodname not in exposed: | 
					
						
							|  |  |  |                     raise AttributeError( | 
					
						
							|  |  |  |                         'method %r of %r object is not in exposed=%r' % | 
					
						
							|  |  |  |                         (methodname, type(obj), exposed) | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 function = getattr(obj, methodname) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     res = function(*args, **kwds) | 
					
						
							|  |  |  |                 except Exception as e: | 
					
						
							|  |  |  |                     msg = ('#ERROR', e) | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     typeid = gettypeid and gettypeid.get(methodname, None) | 
					
						
							|  |  |  |                     if typeid: | 
					
						
							|  |  |  |                         rident, rexposed = self.create(conn, typeid, res) | 
					
						
							|  |  |  |                         token = Token(typeid, self.address, rident) | 
					
						
							|  |  |  |                         msg = ('#PROXY', (rexposed, token)) | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         msg = ('#RETURN', res) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             except AttributeError: | 
					
						
							|  |  |  |                 if methodname is None: | 
					
						
							|  |  |  |                     msg = ('#TRACEBACK', format_exc()) | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     try: | 
					
						
							|  |  |  |                         fallback_func = self.fallback_mapping[methodname] | 
					
						
							|  |  |  |                         result = fallback_func( | 
					
						
							|  |  |  |                             self, conn, ident, obj, *args, **kwds | 
					
						
							|  |  |  |                             ) | 
					
						
							|  |  |  |                         msg = ('#RETURN', result) | 
					
						
							|  |  |  |                     except Exception: | 
					
						
							|  |  |  |                         msg = ('#TRACEBACK', format_exc()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             except EOFError: | 
					
						
							|  |  |  |                 util.debug('got EOF -- exiting thread serving %r', | 
					
						
							| 
									
										
										
										
											2008-08-18 18:09:21 +00:00
										 |  |  |                            threading.current_thread().name) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |                 sys.exit(0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             except Exception: | 
					
						
							|  |  |  |                 msg = ('#TRACEBACK', format_exc()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     send(msg) | 
					
						
							| 
									
										
										
										
											2019-11-19 21:34:03 +00:00
										 |  |  |                 except Exception: | 
					
						
							| 
									
										
										
										
											2016-09-08 14:40:36 -05:00
										 |  |  |                     send(('#UNSERIALIZABLE', format_exc())) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             except Exception as e: | 
					
						
							|  |  |  |                 util.info('exception in thread serving %r', | 
					
						
							| 
									
										
										
										
											2008-08-18 18:09:21 +00:00
										 |  |  |                         threading.current_thread().name) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |                 util.info(' ... message was %r', msg) | 
					
						
							|  |  |  |                 util.info(' ... exception was %r', e) | 
					
						
							|  |  |  |                 conn.close() | 
					
						
							|  |  |  |                 sys.exit(1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def fallback_getvalue(self, conn, ident, obj): | 
					
						
							|  |  |  |         return obj | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def fallback_str(self, conn, ident, obj): | 
					
						
							|  |  |  |         return str(obj) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def fallback_repr(self, conn, ident, obj): | 
					
						
							|  |  |  |         return repr(obj) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fallback_mapping = { | 
					
						
							|  |  |  |         '__str__':fallback_str, | 
					
						
							|  |  |  |         '__repr__':fallback_repr, | 
					
						
							|  |  |  |         '#GETVALUE':fallback_getvalue | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def dummy(self, c): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def debug_info(self, c): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Return some info --- useful to spot problems with refcounting | 
					
						
							|  |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  |         # Perhaps include debug info about 'c'? | 
					
						
							| 
									
										
										
										
											2014-05-25 14:12:12 +01:00
										 |  |  |         with self.mutex: | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             result = [] | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  |             keys = list(self.id_to_refcount.keys()) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             keys.sort() | 
					
						
							|  |  |  |             for ident in keys: | 
					
						
							| 
									
										
										
										
											2009-01-21 02:15:48 +00:00
										 |  |  |                 if ident != '0': | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |                     result.append('  %s:       refcount=%s\n    %s' % | 
					
						
							|  |  |  |                                   (ident, self.id_to_refcount[ident], | 
					
						
							|  |  |  |                                    str(self.id_to_obj[ident][0])[:75])) | 
					
						
							|  |  |  |             return '\n'.join(result) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def number_of_objects(self, c): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Number of shared objects | 
					
						
							|  |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  |         # Doesn't use (len(self.id_to_obj) - 1) as we shouldn't count ident='0' | 
					
						
							|  |  |  |         return len(self.id_to_refcount) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def shutdown(self, c): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Shutdown this process | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2012-06-14 15:30:10 +01:00
										 |  |  |             util.debug('manager received shutdown message') | 
					
						
							|  |  |  |             c.send(('#RETURN', None)) | 
					
						
							|  |  |  |         except: | 
					
						
							|  |  |  |             import traceback | 
					
						
							|  |  |  |             traceback.print_exc() | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         finally: | 
					
						
							| 
									
										
										
										
											2012-06-14 15:30:10 +01:00
										 |  |  |             self.stop_event.set() | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-05 18:22:31 +03:00
										 |  |  |     def create(self, c, typeid, /, *args, **kwds): | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Create a new shared object and return its id | 
					
						
							|  |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2014-05-25 14:12:12 +01:00
										 |  |  |         with self.mutex: | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             callable, exposed, method_to_typeid, proxytype = \ | 
					
						
							|  |  |  |                       self.registry[typeid] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if callable is None: | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  |                 if kwds or (len(args) != 1): | 
					
						
							|  |  |  |                     raise ValueError( | 
					
						
							|  |  |  |                         "Without callable, must have one non-keyword argument") | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |                 obj = args[0] | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 obj = callable(*args, **kwds) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if exposed is None: | 
					
						
							|  |  |  |                 exposed = public_methods(obj) | 
					
						
							|  |  |  |             if method_to_typeid is not None: | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  |                 if not isinstance(method_to_typeid, dict): | 
					
						
							|  |  |  |                     raise TypeError( | 
					
						
							|  |  |  |                         "Method_to_typeid {0!r}: type {1!s}, not dict".format( | 
					
						
							|  |  |  |                             method_to_typeid, type(method_to_typeid))) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |                 exposed = list(exposed) + list(method_to_typeid) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             ident = '%x' % id(obj)  # convert to string because xmlrpclib | 
					
						
							|  |  |  |                                     # only has 32 bit signed integers | 
					
						
							|  |  |  |             util.debug('%r callable returned object with id %r', typeid, ident) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             self.id_to_obj[ident] = (obj, set(exposed), method_to_typeid) | 
					
						
							|  |  |  |             if ident not in self.id_to_refcount: | 
					
						
							| 
									
										
										
										
											2008-09-02 19:12:20 +00:00
										 |  |  |                 self.id_to_refcount[ident] = 0 | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.incref(c, ident) | 
					
						
							|  |  |  |         return ident, tuple(exposed) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def get_methods(self, c, token): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Return the methods of the shared object indicated by token | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         return tuple(self.id_to_obj[token.id][1]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def accept_connection(self, c, name): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Spawn a new thread to serve this connection | 
					
						
							|  |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2008-08-18 18:09:21 +00:00
										 |  |  |         threading.current_thread().name = name | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         c.send(('#RETURN', None)) | 
					
						
							|  |  |  |         self.serve_client(c) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def incref(self, c, ident): | 
					
						
							| 
									
										
										
										
											2014-05-25 14:12:12 +01:00
										 |  |  |         with self.mutex: | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  |             try: | 
					
						
							|  |  |  |                 self.id_to_refcount[ident] += 1 | 
					
						
							|  |  |  |             except KeyError as ke: | 
					
						
							|  |  |  |                 # If no external references exist but an internal (to the | 
					
						
							|  |  |  |                 # manager) still does and a new external reference is created | 
					
						
							|  |  |  |                 # from it, restore the manager's tracking of it from the | 
					
						
							|  |  |  |                 # previously stashed internal ref. | 
					
						
							|  |  |  |                 if ident in self.id_to_local_proxy_obj: | 
					
						
							|  |  |  |                     self.id_to_refcount[ident] = 1 | 
					
						
							|  |  |  |                     self.id_to_obj[ident] = \ | 
					
						
							|  |  |  |                         self.id_to_local_proxy_obj[ident] | 
					
						
							|  |  |  |                     obj, exposed, gettypeid = self.id_to_obj[ident] | 
					
						
							|  |  |  |                     util.debug('Server re-enabled tracking & INCREF %r', ident) | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     raise ke | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def decref(self, c, ident): | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  |         if ident not in self.id_to_refcount and \ | 
					
						
							|  |  |  |             ident in self.id_to_local_proxy_obj: | 
					
						
							|  |  |  |             util.debug('Server DECREF skipping %r', ident) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-25 14:12:12 +01:00
										 |  |  |         with self.mutex: | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  |             if self.id_to_refcount[ident] <= 0: | 
					
						
							|  |  |  |                 raise AssertionError( | 
					
						
							|  |  |  |                     "Id {0!s} ({1!r}) has refcount {2:n}, not 1+".format( | 
					
						
							|  |  |  |                         ident, self.id_to_obj[ident], | 
					
						
							|  |  |  |                         self.id_to_refcount[ident])) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             self.id_to_refcount[ident] -= 1 | 
					
						
							|  |  |  |             if self.id_to_refcount[ident] == 0: | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  |                 del self.id_to_refcount[ident] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ident not in self.id_to_refcount: | 
					
						
							|  |  |  |             # Two-step process in case the object turns out to contain other | 
					
						
							|  |  |  |             # proxy objects (e.g. a managed list of managed lists). | 
					
						
							|  |  |  |             # Otherwise, deleting self.id_to_obj[ident] would trigger the | 
					
						
							|  |  |  |             # deleting of the stored value (another managed object) which would | 
					
						
							|  |  |  |             # in turn attempt to acquire the mutex that is already held here. | 
					
						
							|  |  |  |             self.id_to_obj[ident] = (None, (), None)  # thread-safe | 
					
						
							|  |  |  |             util.debug('disposing of obj with id %r', ident) | 
					
						
							|  |  |  |             with self.mutex: | 
					
						
							|  |  |  |                 del self.id_to_obj[ident] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Class to represent state of a manager | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class State(object): | 
					
						
							|  |  |  |     __slots__ = ['value'] | 
					
						
							|  |  |  |     INITIAL = 0 | 
					
						
							|  |  |  |     STARTED = 1 | 
					
						
							|  |  |  |     SHUTDOWN = 2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Mapping from serializer name to Listener and Client types | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | listener_client = { | 
					
						
							|  |  |  |     'pickle' : (connection.Listener, connection.Client), | 
					
						
							|  |  |  |     'xmlrpclib' : (connection.XmlListener, connection.XmlClient) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Definition of BaseManager | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class BaseManager(object): | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     Base class for managers | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     _registry = {} | 
					
						
							|  |  |  |     _Server = Server | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  |     def __init__(self, address=None, authkey=None, serializer='pickle', | 
					
						
							|  |  |  |                  ctx=None): | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         if authkey is None: | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |             authkey = process.current_process().authkey | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         self._address = address     # XXX not final address if eg ('', 0) | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |         self._authkey = process.AuthenticationString(authkey) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         self._state = State() | 
					
						
							|  |  |  |         self._state.value = State.INITIAL | 
					
						
							|  |  |  |         self._serializer = serializer | 
					
						
							|  |  |  |         self._Listener, self._Client = listener_client[serializer] | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  |         self._ctx = ctx or get_context() | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def get_server(self): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Return server object with serve_forever() method and address attribute | 
					
						
							|  |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  |         if self._state.value != State.INITIAL: | 
					
						
							|  |  |  |             if self._state.value == State.STARTED: | 
					
						
							|  |  |  |                 raise ProcessError("Already started server") | 
					
						
							|  |  |  |             elif self._state.value == State.SHUTDOWN: | 
					
						
							|  |  |  |                 raise ProcessError("Manager has shut down") | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 raise ProcessError( | 
					
						
							|  |  |  |                     "Unknown state {!r}".format(self._state.value)) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         return Server(self._registry, self._address, | 
					
						
							|  |  |  |                       self._authkey, self._serializer) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def connect(self): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Connect manager object to the server process | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Listener, Client = listener_client[self._serializer] | 
					
						
							|  |  |  |         conn = Client(self._address, authkey=self._authkey) | 
					
						
							|  |  |  |         dispatch(conn, None, 'dummy') | 
					
						
							|  |  |  |         self._state.value = State.STARTED | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merged revisions 70912,70944,70968,71033,71041,71208,71263,71286,71395-71396,71405-71406,71485,71492,71494 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r70912 | georg.brandl | 2009-03-31 17:35:46 -0500 (Tue, 31 Mar 2009) | 1 line
  #5617: add a handy function to print a unicode string to gdbinit.
........
  r70944 | georg.brandl | 2009-03-31 23:32:39 -0500 (Tue, 31 Mar 2009) | 1 line
  #5631: add upload to list of possible commands, which is presented in --help-commands.
........
  r70968 | michael.foord | 2009-04-01 13:25:38 -0500 (Wed, 01 Apr 2009) | 1 line
  Adding Wing project file
........
  r71033 | brett.cannon | 2009-04-01 22:34:53 -0500 (Wed, 01 Apr 2009) | 3 lines
  Fix two issues introduced by issue #71031 by changing the signature of
  PyImport_AppendInittab() to take a const char *.
........
  r71041 | jesse.noller | 2009-04-02 00:17:26 -0500 (Thu, 02 Apr 2009) | 1 line
  Add custom initializer argument to multiprocess.Manager*, courtesy of lekma
........
  r71208 | michael.foord | 2009-04-04 20:15:01 -0500 (Sat, 04 Apr 2009) | 4 lines
  Change the way unittest.TestSuite use their tests to always access them through iteration. Non behavior changing, this allows you to create custom subclasses that override __iter__.
  Issue #5693
........
  r71263 | michael.foord | 2009-04-05 14:19:28 -0500 (Sun, 05 Apr 2009) | 4 lines
  Adding assertIs and assertIsNot methods to unittest.TestCase
  Issue #2578
........
  r71286 | tarek.ziade | 2009-04-05 17:04:38 -0500 (Sun, 05 Apr 2009) | 1 line
  added a simplest test to distutils.spawn._nt_quote_args
........
  r71395 | benjamin.peterson | 2009-04-08 08:27:29 -0500 (Wed, 08 Apr 2009) | 1 line
  these must be installed to correctly run tests
........
  r71396 | benjamin.peterson | 2009-04-08 08:29:41 -0500 (Wed, 08 Apr 2009) | 1 line
  fix syntax
........
  r71405 | andrew.kuchling | 2009-04-09 06:22:47 -0500 (Thu, 09 Apr 2009) | 1 line
  Add items
........
  r71406 | andrew.kuchling | 2009-04-09 06:23:36 -0500 (Thu, 09 Apr 2009) | 1 line
  Typo fixes
........
  r71485 | andrew.kuchling | 2009-04-11 11:12:23 -0500 (Sat, 11 Apr 2009) | 1 line
  Add various items
........
  r71492 | georg.brandl | 2009-04-11 13:19:27 -0500 (Sat, 11 Apr 2009) | 1 line
  Take credit for a patch of mine.
........
  r71494 | benjamin.peterson | 2009-04-11 14:31:00 -0500 (Sat, 11 Apr 2009) | 1 line
  ignore py3_test_grammar when compiling the library
........
											
										 
											2009-04-11 20:45:40 +00:00
										 |  |  |     def start(self, initializer=None, initargs=()): | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Spawn a server process for this manager object | 
					
						
							|  |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  |         if self._state.value != State.INITIAL: | 
					
						
							|  |  |  |             if self._state.value == State.STARTED: | 
					
						
							|  |  |  |                 raise ProcessError("Already started server") | 
					
						
							|  |  |  |             elif self._state.value == State.SHUTDOWN: | 
					
						
							|  |  |  |                 raise ProcessError("Manager has shut down") | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 raise ProcessError( | 
					
						
							|  |  |  |                     "Unknown state {!r}".format(self._state.value)) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-28 14:45:05 +02:00
										 |  |  |         if initializer is not None and not callable(initializer): | 
					
						
							| 
									
										
											  
											
												Merged revisions 70912,70944,70968,71033,71041,71208,71263,71286,71395-71396,71405-71406,71485,71492,71494 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r70912 | georg.brandl | 2009-03-31 17:35:46 -0500 (Tue, 31 Mar 2009) | 1 line
  #5617: add a handy function to print a unicode string to gdbinit.
........
  r70944 | georg.brandl | 2009-03-31 23:32:39 -0500 (Tue, 31 Mar 2009) | 1 line
  #5631: add upload to list of possible commands, which is presented in --help-commands.
........
  r70968 | michael.foord | 2009-04-01 13:25:38 -0500 (Wed, 01 Apr 2009) | 1 line
  Adding Wing project file
........
  r71033 | brett.cannon | 2009-04-01 22:34:53 -0500 (Wed, 01 Apr 2009) | 3 lines
  Fix two issues introduced by issue #71031 by changing the signature of
  PyImport_AppendInittab() to take a const char *.
........
  r71041 | jesse.noller | 2009-04-02 00:17:26 -0500 (Thu, 02 Apr 2009) | 1 line
  Add custom initializer argument to multiprocess.Manager*, courtesy of lekma
........
  r71208 | michael.foord | 2009-04-04 20:15:01 -0500 (Sat, 04 Apr 2009) | 4 lines
  Change the way unittest.TestSuite use their tests to always access them through iteration. Non behavior changing, this allows you to create custom subclasses that override __iter__.
  Issue #5693
........
  r71263 | michael.foord | 2009-04-05 14:19:28 -0500 (Sun, 05 Apr 2009) | 4 lines
  Adding assertIs and assertIsNot methods to unittest.TestCase
  Issue #2578
........
  r71286 | tarek.ziade | 2009-04-05 17:04:38 -0500 (Sun, 05 Apr 2009) | 1 line
  added a simplest test to distutils.spawn._nt_quote_args
........
  r71395 | benjamin.peterson | 2009-04-08 08:27:29 -0500 (Wed, 08 Apr 2009) | 1 line
  these must be installed to correctly run tests
........
  r71396 | benjamin.peterson | 2009-04-08 08:29:41 -0500 (Wed, 08 Apr 2009) | 1 line
  fix syntax
........
  r71405 | andrew.kuchling | 2009-04-09 06:22:47 -0500 (Thu, 09 Apr 2009) | 1 line
  Add items
........
  r71406 | andrew.kuchling | 2009-04-09 06:23:36 -0500 (Thu, 09 Apr 2009) | 1 line
  Typo fixes
........
  r71485 | andrew.kuchling | 2009-04-11 11:12:23 -0500 (Sat, 11 Apr 2009) | 1 line
  Add various items
........
  r71492 | georg.brandl | 2009-04-11 13:19:27 -0500 (Sat, 11 Apr 2009) | 1 line
  Take credit for a patch of mine.
........
  r71494 | benjamin.peterson | 2009-04-11 14:31:00 -0500 (Sat, 11 Apr 2009) | 1 line
  ignore py3_test_grammar when compiling the library
........
											
										 
											2009-04-11 20:45:40 +00:00
										 |  |  |             raise TypeError('initializer must be a callable') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         # pipe over which we will retrieve address of server | 
					
						
							|  |  |  |         reader, writer = connection.Pipe(duplex=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # spawn process which runs a server | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  |         self._process = self._ctx.Process( | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             target=type(self)._run_server, | 
					
						
							|  |  |  |             args=(self._registry, self._address, self._authkey, | 
					
						
							| 
									
										
											  
											
												Merged revisions 70912,70944,70968,71033,71041,71208,71263,71286,71395-71396,71405-71406,71485,71492,71494 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r70912 | georg.brandl | 2009-03-31 17:35:46 -0500 (Tue, 31 Mar 2009) | 1 line
  #5617: add a handy function to print a unicode string to gdbinit.
........
  r70944 | georg.brandl | 2009-03-31 23:32:39 -0500 (Tue, 31 Mar 2009) | 1 line
  #5631: add upload to list of possible commands, which is presented in --help-commands.
........
  r70968 | michael.foord | 2009-04-01 13:25:38 -0500 (Wed, 01 Apr 2009) | 1 line
  Adding Wing project file
........
  r71033 | brett.cannon | 2009-04-01 22:34:53 -0500 (Wed, 01 Apr 2009) | 3 lines
  Fix two issues introduced by issue #71031 by changing the signature of
  PyImport_AppendInittab() to take a const char *.
........
  r71041 | jesse.noller | 2009-04-02 00:17:26 -0500 (Thu, 02 Apr 2009) | 1 line
  Add custom initializer argument to multiprocess.Manager*, courtesy of lekma
........
  r71208 | michael.foord | 2009-04-04 20:15:01 -0500 (Sat, 04 Apr 2009) | 4 lines
  Change the way unittest.TestSuite use their tests to always access them through iteration. Non behavior changing, this allows you to create custom subclasses that override __iter__.
  Issue #5693
........
  r71263 | michael.foord | 2009-04-05 14:19:28 -0500 (Sun, 05 Apr 2009) | 4 lines
  Adding assertIs and assertIsNot methods to unittest.TestCase
  Issue #2578
........
  r71286 | tarek.ziade | 2009-04-05 17:04:38 -0500 (Sun, 05 Apr 2009) | 1 line
  added a simplest test to distutils.spawn._nt_quote_args
........
  r71395 | benjamin.peterson | 2009-04-08 08:27:29 -0500 (Wed, 08 Apr 2009) | 1 line
  these must be installed to correctly run tests
........
  r71396 | benjamin.peterson | 2009-04-08 08:29:41 -0500 (Wed, 08 Apr 2009) | 1 line
  fix syntax
........
  r71405 | andrew.kuchling | 2009-04-09 06:22:47 -0500 (Thu, 09 Apr 2009) | 1 line
  Add items
........
  r71406 | andrew.kuchling | 2009-04-09 06:23:36 -0500 (Thu, 09 Apr 2009) | 1 line
  Typo fixes
........
  r71485 | andrew.kuchling | 2009-04-11 11:12:23 -0500 (Sat, 11 Apr 2009) | 1 line
  Add various items
........
  r71492 | georg.brandl | 2009-04-11 13:19:27 -0500 (Sat, 11 Apr 2009) | 1 line
  Take credit for a patch of mine.
........
  r71494 | benjamin.peterson | 2009-04-11 14:31:00 -0500 (Sat, 11 Apr 2009) | 1 line
  ignore py3_test_grammar when compiling the library
........
											
										 
											2009-04-11 20:45:40 +00:00
										 |  |  |                   self._serializer, writer, initializer, initargs), | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             ) | 
					
						
							|  |  |  |         ident = ':'.join(str(i) for i in self._process._identity) | 
					
						
							| 
									
										
										
										
											2008-08-19 19:17:39 +00:00
										 |  |  |         self._process.name = type(self).__name__  + '-' + ident | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         self._process.start() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # get address of server | 
					
						
							|  |  |  |         writer.close() | 
					
						
							|  |  |  |         self._address = reader.recv() | 
					
						
							|  |  |  |         reader.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # register a finalizer | 
					
						
							|  |  |  |         self._state.value = State.STARTED | 
					
						
							|  |  |  |         self.shutdown = util.Finalize( | 
					
						
							|  |  |  |             self, type(self)._finalize_manager, | 
					
						
							|  |  |  |             args=(self._process, self._address, self._authkey, | 
					
						
							|  |  |  |                   self._state, self._Client), | 
					
						
							|  |  |  |             exitpriority=0 | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							| 
									
										
											  
											
												Merged revisions 70912,70944,70968,71033,71041,71208,71263,71286,71395-71396,71405-71406,71485,71492,71494 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r70912 | georg.brandl | 2009-03-31 17:35:46 -0500 (Tue, 31 Mar 2009) | 1 line
  #5617: add a handy function to print a unicode string to gdbinit.
........
  r70944 | georg.brandl | 2009-03-31 23:32:39 -0500 (Tue, 31 Mar 2009) | 1 line
  #5631: add upload to list of possible commands, which is presented in --help-commands.
........
  r70968 | michael.foord | 2009-04-01 13:25:38 -0500 (Wed, 01 Apr 2009) | 1 line
  Adding Wing project file
........
  r71033 | brett.cannon | 2009-04-01 22:34:53 -0500 (Wed, 01 Apr 2009) | 3 lines
  Fix two issues introduced by issue #71031 by changing the signature of
  PyImport_AppendInittab() to take a const char *.
........
  r71041 | jesse.noller | 2009-04-02 00:17:26 -0500 (Thu, 02 Apr 2009) | 1 line
  Add custom initializer argument to multiprocess.Manager*, courtesy of lekma
........
  r71208 | michael.foord | 2009-04-04 20:15:01 -0500 (Sat, 04 Apr 2009) | 4 lines
  Change the way unittest.TestSuite use their tests to always access them through iteration. Non behavior changing, this allows you to create custom subclasses that override __iter__.
  Issue #5693
........
  r71263 | michael.foord | 2009-04-05 14:19:28 -0500 (Sun, 05 Apr 2009) | 4 lines
  Adding assertIs and assertIsNot methods to unittest.TestCase
  Issue #2578
........
  r71286 | tarek.ziade | 2009-04-05 17:04:38 -0500 (Sun, 05 Apr 2009) | 1 line
  added a simplest test to distutils.spawn._nt_quote_args
........
  r71395 | benjamin.peterson | 2009-04-08 08:27:29 -0500 (Wed, 08 Apr 2009) | 1 line
  these must be installed to correctly run tests
........
  r71396 | benjamin.peterson | 2009-04-08 08:29:41 -0500 (Wed, 08 Apr 2009) | 1 line
  fix syntax
........
  r71405 | andrew.kuchling | 2009-04-09 06:22:47 -0500 (Thu, 09 Apr 2009) | 1 line
  Add items
........
  r71406 | andrew.kuchling | 2009-04-09 06:23:36 -0500 (Thu, 09 Apr 2009) | 1 line
  Typo fixes
........
  r71485 | andrew.kuchling | 2009-04-11 11:12:23 -0500 (Sat, 11 Apr 2009) | 1 line
  Add various items
........
  r71492 | georg.brandl | 2009-04-11 13:19:27 -0500 (Sat, 11 Apr 2009) | 1 line
  Take credit for a patch of mine.
........
  r71494 | benjamin.peterson | 2009-04-11 14:31:00 -0500 (Sat, 11 Apr 2009) | 1 line
  ignore py3_test_grammar when compiling the library
........
											
										 
											2009-04-11 20:45:40 +00:00
										 |  |  |     def _run_server(cls, registry, address, authkey, serializer, writer, | 
					
						
							|  |  |  |                     initializer=None, initargs=()): | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Create a server, report its address and run it | 
					
						
							|  |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2019-05-10 20:42:35 +02:00
										 |  |  |         # bpo-36368: protect server process from KeyboardInterrupt signals | 
					
						
							|  |  |  |         signal.signal(signal.SIGINT, signal.SIG_IGN) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merged revisions 70912,70944,70968,71033,71041,71208,71263,71286,71395-71396,71405-71406,71485,71492,71494 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk
........
  r70912 | georg.brandl | 2009-03-31 17:35:46 -0500 (Tue, 31 Mar 2009) | 1 line
  #5617: add a handy function to print a unicode string to gdbinit.
........
  r70944 | georg.brandl | 2009-03-31 23:32:39 -0500 (Tue, 31 Mar 2009) | 1 line
  #5631: add upload to list of possible commands, which is presented in --help-commands.
........
  r70968 | michael.foord | 2009-04-01 13:25:38 -0500 (Wed, 01 Apr 2009) | 1 line
  Adding Wing project file
........
  r71033 | brett.cannon | 2009-04-01 22:34:53 -0500 (Wed, 01 Apr 2009) | 3 lines
  Fix two issues introduced by issue #71031 by changing the signature of
  PyImport_AppendInittab() to take a const char *.
........
  r71041 | jesse.noller | 2009-04-02 00:17:26 -0500 (Thu, 02 Apr 2009) | 1 line
  Add custom initializer argument to multiprocess.Manager*, courtesy of lekma
........
  r71208 | michael.foord | 2009-04-04 20:15:01 -0500 (Sat, 04 Apr 2009) | 4 lines
  Change the way unittest.TestSuite use their tests to always access them through iteration. Non behavior changing, this allows you to create custom subclasses that override __iter__.
  Issue #5693
........
  r71263 | michael.foord | 2009-04-05 14:19:28 -0500 (Sun, 05 Apr 2009) | 4 lines
  Adding assertIs and assertIsNot methods to unittest.TestCase
  Issue #2578
........
  r71286 | tarek.ziade | 2009-04-05 17:04:38 -0500 (Sun, 05 Apr 2009) | 1 line
  added a simplest test to distutils.spawn._nt_quote_args
........
  r71395 | benjamin.peterson | 2009-04-08 08:27:29 -0500 (Wed, 08 Apr 2009) | 1 line
  these must be installed to correctly run tests
........
  r71396 | benjamin.peterson | 2009-04-08 08:29:41 -0500 (Wed, 08 Apr 2009) | 1 line
  fix syntax
........
  r71405 | andrew.kuchling | 2009-04-09 06:22:47 -0500 (Thu, 09 Apr 2009) | 1 line
  Add items
........
  r71406 | andrew.kuchling | 2009-04-09 06:23:36 -0500 (Thu, 09 Apr 2009) | 1 line
  Typo fixes
........
  r71485 | andrew.kuchling | 2009-04-11 11:12:23 -0500 (Sat, 11 Apr 2009) | 1 line
  Add various items
........
  r71492 | georg.brandl | 2009-04-11 13:19:27 -0500 (Sat, 11 Apr 2009) | 1 line
  Take credit for a patch of mine.
........
  r71494 | benjamin.peterson | 2009-04-11 14:31:00 -0500 (Sat, 11 Apr 2009) | 1 line
  ignore py3_test_grammar when compiling the library
........
											
										 
											2009-04-11 20:45:40 +00:00
										 |  |  |         if initializer is not None: | 
					
						
							|  |  |  |             initializer(*initargs) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         # create server | 
					
						
							|  |  |  |         server = cls._Server(registry, address, authkey, serializer) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # inform parent process of the server's address | 
					
						
							|  |  |  |         writer.send(server.address) | 
					
						
							|  |  |  |         writer.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # run the manager | 
					
						
							|  |  |  |         util.info('manager serving at %r', server.address) | 
					
						
							|  |  |  |         server.serve_forever() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-01 11:00:15 +03:00
										 |  |  |     def _create(self, typeid, /, *args, **kwds): | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Create a new shared object; return the token and exposed tuple | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         assert self._state.value == State.STARTED, 'server not yet started' | 
					
						
							|  |  |  |         conn = self._Client(self._address, authkey=self._authkey) | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             id, exposed = dispatch(conn, None, 'create', (typeid,)+args, kwds) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             conn.close() | 
					
						
							|  |  |  |         return Token(typeid, self._address, id), exposed | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def join(self, timeout=None): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Join the manager process (if it has been spawned) | 
					
						
							|  |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2012-05-03 18:29:02 +01:00
										 |  |  |         if self._process is not None: | 
					
						
							|  |  |  |             self._process.join(timeout) | 
					
						
							|  |  |  |             if not self._process.is_alive(): | 
					
						
							|  |  |  |                 self._process = None | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _debug_info(self): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Return some info about the servers shared objects and connections | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         conn = self._Client(self._address, authkey=self._authkey) | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return dispatch(conn, None, 'debug_info') | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             conn.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _number_of_objects(self): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Return the number of shared objects | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         conn = self._Client(self._address, authkey=self._authkey) | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return dispatch(conn, None, 'number_of_objects') | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             conn.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __enter__(self): | 
					
						
							| 
									
										
										
										
											2012-06-18 21:29:30 +01:00
										 |  |  |         if self._state.value == State.INITIAL: | 
					
						
							|  |  |  |             self.start() | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  |         if self._state.value != State.STARTED: | 
					
						
							|  |  |  |             if self._state.value == State.INITIAL: | 
					
						
							|  |  |  |                 raise ProcessError("Unable to start server") | 
					
						
							|  |  |  |             elif self._state.value == State.SHUTDOWN: | 
					
						
							|  |  |  |                 raise ProcessError("Manager has shut down") | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 raise ProcessError( | 
					
						
							|  |  |  |                     "Unknown state {!r}".format(self._state.value)) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __exit__(self, exc_type, exc_val, exc_tb): | 
					
						
							|  |  |  |         self.shutdown() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @staticmethod | 
					
						
							|  |  |  |     def _finalize_manager(process, address, authkey, state, _Client): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Shutdown the manager process; will be registered as a finalizer | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         if process.is_alive(): | 
					
						
							|  |  |  |             util.info('sending shutdown message to manager') | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 conn = _Client(address, authkey=authkey) | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     dispatch(conn, None, 'shutdown') | 
					
						
							|  |  |  |                 finally: | 
					
						
							|  |  |  |                     conn.close() | 
					
						
							|  |  |  |             except Exception: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-15 20:08:29 +01:00
										 |  |  |             process.join(timeout=1.0) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             if process.is_alive(): | 
					
						
							|  |  |  |                 util.info('manager still alive') | 
					
						
							|  |  |  |                 if hasattr(process, 'terminate'): | 
					
						
							|  |  |  |                     util.info('trying to `terminate()` manager process') | 
					
						
							|  |  |  |                     process.terminate() | 
					
						
							|  |  |  |                     process.join(timeout=0.1) | 
					
						
							|  |  |  |                     if process.is_alive(): | 
					
						
							|  |  |  |                         util.info('manager still alive after terminate') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         state.value = State.SHUTDOWN | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             del BaseProxy._address_to_local[address] | 
					
						
							|  |  |  |         except KeyError: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-19 08:40:32 +02:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def address(self): | 
					
						
							|  |  |  |         return self._address | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def register(cls, typeid, callable=None, proxytype=None, exposed=None, | 
					
						
							|  |  |  |                  method_to_typeid=None, create_method=True): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Register a typeid with the manager type | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         if '_registry' not in cls.__dict__: | 
					
						
							|  |  |  |             cls._registry = cls._registry.copy() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if proxytype is None: | 
					
						
							|  |  |  |             proxytype = AutoProxy | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         exposed = exposed or getattr(proxytype, '_exposed_', None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         method_to_typeid = method_to_typeid or \ | 
					
						
							|  |  |  |                            getattr(proxytype, '_method_to_typeid_', None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if method_to_typeid: | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  |             for key, value in list(method_to_typeid.items()): # isinstance? | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |                 assert type(key) is str, '%r is not a string' % key | 
					
						
							|  |  |  |                 assert type(value) is str, '%r is not a string' % value | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         cls._registry[typeid] = ( | 
					
						
							|  |  |  |             callable, exposed, method_to_typeid, proxytype | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if create_method: | 
					
						
							| 
									
										
										
										
											2019-06-01 11:00:15 +03:00
										 |  |  |             def temp(self, /, *args, **kwds): | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |                 util.debug('requesting creation of a shared %r object', typeid) | 
					
						
							|  |  |  |                 token, exp = self._create(typeid, *args, **kwds) | 
					
						
							|  |  |  |                 proxy = proxytype( | 
					
						
							|  |  |  |                     token, self._serializer, manager=self, | 
					
						
							|  |  |  |                     authkey=self._authkey, exposed=exp | 
					
						
							|  |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2008-09-02 19:12:20 +00:00
										 |  |  |                 conn = self._Client(token.address, authkey=self._authkey) | 
					
						
							|  |  |  |                 dispatch(conn, None, 'decref', (token.id,)) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |                 return proxy | 
					
						
							|  |  |  |             temp.__name__ = typeid | 
					
						
							|  |  |  |             setattr(cls, typeid, temp) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Subclass of set which get cleared after a fork | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ProcessLocalSet(set): | 
					
						
							|  |  |  |     def __init__(self): | 
					
						
							|  |  |  |         util.register_after_fork(self, lambda obj: obj.clear()) | 
					
						
							|  |  |  |     def __reduce__(self): | 
					
						
							|  |  |  |         return type(self), () | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Definition of BaseProxy | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class BaseProxy(object): | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     A base for proxies of shared objects | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     _address_to_local = {} | 
					
						
							|  |  |  |     _mutex = util.ForkAwareThreadLock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, token, serializer, manager=None, | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  |                  authkey=None, exposed=None, incref=True, manager_owned=False): | 
					
						
							| 
									
										
										
										
											2014-05-25 14:12:12 +01:00
										 |  |  |         with BaseProxy._mutex: | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             tls_idset = BaseProxy._address_to_local.get(token.address, None) | 
					
						
							|  |  |  |             if tls_idset is None: | 
					
						
							|  |  |  |                 tls_idset = util.ForkAwareLocal(), ProcessLocalSet() | 
					
						
							|  |  |  |                 BaseProxy._address_to_local[token.address] = tls_idset | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # self._tls is used to record the connection used by this | 
					
						
							|  |  |  |         # thread to communicate with the manager at token.address | 
					
						
							|  |  |  |         self._tls = tls_idset[0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # self._idset is used to record the identities of all shared | 
					
						
							|  |  |  |         # objects for which the current process owns references and | 
					
						
							|  |  |  |         # which are in the manager at token.address | 
					
						
							|  |  |  |         self._idset = tls_idset[1] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._token = token | 
					
						
							|  |  |  |         self._id = self._token.id | 
					
						
							|  |  |  |         self._manager = manager | 
					
						
							|  |  |  |         self._serializer = serializer | 
					
						
							|  |  |  |         self._Client = listener_client[serializer][1] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  |         # Should be set to True only when a proxy object is being created | 
					
						
							|  |  |  |         # on the manager server; primary use case: nested proxy objects. | 
					
						
							|  |  |  |         # RebuildProxy detects when a proxy is being created on the manager | 
					
						
							|  |  |  |         # and sets this value appropriately. | 
					
						
							|  |  |  |         self._owned_by_manager = manager_owned | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         if authkey is not None: | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |             self._authkey = process.AuthenticationString(authkey) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         elif self._manager is not None: | 
					
						
							|  |  |  |             self._authkey = self._manager._authkey | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |             self._authkey = process.current_process().authkey | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if incref: | 
					
						
							|  |  |  |             self._incref() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         util.register_after_fork(self, BaseProxy._after_fork) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _connect(self): | 
					
						
							|  |  |  |         util.debug('making connection to manager') | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |         name = process.current_process().name | 
					
						
							| 
									
										
										
										
											2008-08-18 18:09:21 +00:00
										 |  |  |         if threading.current_thread().name != 'MainThread': | 
					
						
							|  |  |  |             name += '|' + threading.current_thread().name | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         conn = self._Client(self._token.address, authkey=self._authkey) | 
					
						
							|  |  |  |         dispatch(conn, None, 'accept_connection', (name,)) | 
					
						
							|  |  |  |         self._tls.connection = conn | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _callmethod(self, methodname, args=(), kwds={}): | 
					
						
							|  |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2020-04-18 14:58:29 +08:00
										 |  |  |         Try to call a method of the referent and return a copy of the result | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             conn = self._tls.connection | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             util.debug('thread %r does not own a connection', | 
					
						
							| 
									
										
										
										
											2008-08-18 18:09:21 +00:00
										 |  |  |                        threading.current_thread().name) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             self._connect() | 
					
						
							|  |  |  |             conn = self._tls.connection | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         conn.send((self._id, methodname, args, kwds)) | 
					
						
							|  |  |  |         kind, result = conn.recv() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if kind == '#RETURN': | 
					
						
							|  |  |  |             return result | 
					
						
							|  |  |  |         elif kind == '#PROXY': | 
					
						
							|  |  |  |             exposed, token = result | 
					
						
							|  |  |  |             proxytype = self._manager._registry[token.typeid][-1] | 
					
						
							| 
									
										
										
										
											2013-07-02 13:37:43 +01:00
										 |  |  |             token.address = self._token.address | 
					
						
							| 
									
										
										
										
											2008-09-02 19:12:20 +00:00
										 |  |  |             proxy = proxytype( | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |                 token, self._serializer, manager=self._manager, | 
					
						
							|  |  |  |                 authkey=self._authkey, exposed=exposed | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2008-09-02 19:12:20 +00:00
										 |  |  |             conn = self._Client(token.address, authkey=self._authkey) | 
					
						
							|  |  |  |             dispatch(conn, None, 'decref', (token.id,)) | 
					
						
							|  |  |  |             return proxy | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         raise convert_to_error(kind, result) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _getvalue(self): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Get a copy of the value of the referent | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         return self._callmethod('#GETVALUE') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _incref(self): | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  |         if self._owned_by_manager: | 
					
						
							|  |  |  |             util.debug('owned_by_manager skipped INCREF of %r', self._token.id) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         conn = self._Client(self._token.address, authkey=self._authkey) | 
					
						
							|  |  |  |         dispatch(conn, None, 'incref', (self._id,)) | 
					
						
							|  |  |  |         util.debug('INCREF %r', self._token.id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._idset.add(self._id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         state = self._manager and self._manager._state | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._close = util.Finalize( | 
					
						
							|  |  |  |             self, BaseProxy._decref, | 
					
						
							|  |  |  |             args=(self._token, self._authkey, state, | 
					
						
							|  |  |  |                   self._tls, self._idset, self._Client), | 
					
						
							|  |  |  |             exitpriority=10 | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @staticmethod | 
					
						
							|  |  |  |     def _decref(token, authkey, state, tls, idset, _Client): | 
					
						
							|  |  |  |         idset.discard(token.id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # check whether manager is still alive | 
					
						
							|  |  |  |         if state is None or state.value == State.STARTED: | 
					
						
							|  |  |  |             # tell manager this process no longer cares about referent | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 util.debug('DECREF %r', token.id) | 
					
						
							|  |  |  |                 conn = _Client(token.address, authkey=authkey) | 
					
						
							|  |  |  |                 dispatch(conn, None, 'decref', (token.id,)) | 
					
						
							|  |  |  |             except Exception as e: | 
					
						
							|  |  |  |                 util.debug('... decref failed %s', e) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             util.debug('DECREF %r -- manager already shutdown', token.id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # check whether we can close this thread's connection because | 
					
						
							|  |  |  |         # the process owns no more references to objects for this manager | 
					
						
							|  |  |  |         if not idset and hasattr(tls, 'connection'): | 
					
						
							|  |  |  |             util.debug('thread %r has no more proxies so closing conn', | 
					
						
							| 
									
										
										
										
											2008-08-18 18:09:21 +00:00
										 |  |  |                        threading.current_thread().name) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             tls.connection.close() | 
					
						
							|  |  |  |             del tls.connection | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _after_fork(self): | 
					
						
							|  |  |  |         self._manager = None | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             self._incref() | 
					
						
							|  |  |  |         except Exception as e: | 
					
						
							|  |  |  |             # the proxy may just be for a manager which has shutdown | 
					
						
							|  |  |  |             util.info('incref failed: %s' % e) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __reduce__(self): | 
					
						
							|  |  |  |         kwds = {} | 
					
						
							| 
									
										
										
										
											2016-09-09 18:03:10 -05:00
										 |  |  |         if get_spawning_popen() is not None: | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |             kwds['authkey'] = self._authkey | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if getattr(self, '_isauto', False): | 
					
						
							|  |  |  |             kwds['exposed'] = self._exposed_ | 
					
						
							|  |  |  |             return (RebuildProxy, | 
					
						
							|  |  |  |                     (AutoProxy, self._token, self._serializer, kwds)) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return (RebuildProxy, | 
					
						
							|  |  |  |                     (type(self), self._token, self._serializer, kwds)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __deepcopy__(self, memo): | 
					
						
							|  |  |  |         return self._getvalue() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							| 
									
										
										
										
											2014-07-25 23:36:00 +03:00
										 |  |  |         return '<%s object, typeid %r at %#x>' % \ | 
					
						
							|  |  |  |                (type(self).__name__, self._token.typeid, id(self)) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __str__(self): | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Return representation of the referent (or a fall-back if that fails) | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return self._callmethod('__repr__') | 
					
						
							|  |  |  |         except Exception: | 
					
						
							|  |  |  |             return repr(self)[:-1] + "; '__str__()' failed>" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Function used for unpickling | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def RebuildProxy(func, token, serializer, kwds): | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     Function used for unpickling proxy objects. | 
					
						
							|  |  |  |     '''
 | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |     server = getattr(process.current_process(), '_manager_server', None) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     if server and server.address == token.address: | 
					
						
							| 
									
										
										
										
											2016-09-07 18:48:01 -05:00
										 |  |  |         util.debug('Rebuild a proxy owned by manager, token=%r', token) | 
					
						
							|  |  |  |         kwds['manager_owned'] = True | 
					
						
							|  |  |  |         if token.id not in server.id_to_local_proxy_obj: | 
					
						
							|  |  |  |             server.id_to_local_proxy_obj[token.id] = \ | 
					
						
							|  |  |  |                 server.id_to_obj[token.id] | 
					
						
							|  |  |  |     incref = ( | 
					
						
							|  |  |  |         kwds.pop('incref', True) and | 
					
						
							|  |  |  |         not getattr(process.current_process(), '_inheriting', False) | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |     return func(token, serializer, incref=incref, **kwds) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Functions to create proxies and proxy types | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def MakeProxyType(name, exposed, _cache={}): | 
					
						
							|  |  |  |     '''
 | 
					
						
							| 
									
										
										
										
											2016-04-17 08:32:47 +03:00
										 |  |  |     Return a proxy type whose methods are given by `exposed` | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     '''
 | 
					
						
							|  |  |  |     exposed = tuple(exposed) | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         return _cache[(name, exposed)] | 
					
						
							|  |  |  |     except KeyError: | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     dic = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for meth in exposed: | 
					
						
							| 
									
										
										
										
											2019-06-01 11:00:15 +03:00
										 |  |  |         exec('''def %s(self, /, *args, **kwds):
 | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         return self._callmethod(%r, args, kwds)''' % (meth, meth), dic)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ProxyType = type(name, (BaseProxy,), dic) | 
					
						
							|  |  |  |     ProxyType._exposed_ = exposed | 
					
						
							|  |  |  |     _cache[(name, exposed)] = ProxyType | 
					
						
							|  |  |  |     return ProxyType | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def AutoProxy(token, serializer, manager=None, authkey=None, | 
					
						
							| 
									
										
										
										
											2021-07-02 05:45:02 +02:00
										 |  |  |               exposed=None, incref=True, manager_owned=False): | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     '''
 | 
					
						
							|  |  |  |     Return an auto-proxy for `token` | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     _Client = listener_client[serializer][1] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if exposed is None: | 
					
						
							|  |  |  |         conn = _Client(token.address, authkey=authkey) | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             exposed = dispatch(conn, None, 'get_methods', (token,)) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             conn.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if authkey is None and manager is not None: | 
					
						
							|  |  |  |         authkey = manager._authkey | 
					
						
							|  |  |  |     if authkey is None: | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |         authkey = process.current_process().authkey | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ProxyType = MakeProxyType('AutoProxy[%s]' % token.typeid, exposed) | 
					
						
							|  |  |  |     proxy = ProxyType(token, serializer, manager=manager, authkey=authkey, | 
					
						
							| 
									
										
										
										
											2021-07-02 05:45:02 +02:00
										 |  |  |                       incref=incref, manager_owned=manager_owned) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     proxy._isauto = True | 
					
						
							|  |  |  |     return proxy | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Types/callables which we will register with SyncManager | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Namespace(object): | 
					
						
							| 
									
										
										
										
											2019-06-01 11:00:15 +03:00
										 |  |  |     def __init__(self, /, **kwds): | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |         self.__dict__.update(kwds) | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         items = list(self.__dict__.items()) | 
					
						
							|  |  |  |         temp = [] | 
					
						
							|  |  |  |         for name, value in items: | 
					
						
							|  |  |  |             if not name.startswith('_'): | 
					
						
							|  |  |  |                 temp.append('%s=%r' % (name, value)) | 
					
						
							|  |  |  |         temp.sort() | 
					
						
							| 
									
										
										
										
											2014-07-25 23:36:00 +03:00
										 |  |  |         return '%s(%s)' % (self.__class__.__name__, ', '.join(temp)) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class Value(object): | 
					
						
							|  |  |  |     def __init__(self, typecode, value, lock=True): | 
					
						
							|  |  |  |         self._typecode = typecode | 
					
						
							|  |  |  |         self._value = value | 
					
						
							|  |  |  |     def get(self): | 
					
						
							|  |  |  |         return self._value | 
					
						
							|  |  |  |     def set(self, value): | 
					
						
							|  |  |  |         self._value = value | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return '%s(%r, %r)'%(type(self).__name__, self._typecode, self._value) | 
					
						
							|  |  |  |     value = property(get, set) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def Array(typecode, sequence, lock=True): | 
					
						
							|  |  |  |     return array.array(typecode, sequence) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Proxy types used by SyncManager | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class IteratorProxy(BaseProxy): | 
					
						
							| 
									
										
										
										
											2008-06-25 13:04:48 +00:00
										 |  |  |     _exposed_ = ('__next__', 'send', 'throw', 'close') | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     def __iter__(self): | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  |     def __next__(self, *args): | 
					
						
							|  |  |  |         return self._callmethod('__next__', args) | 
					
						
							|  |  |  |     def send(self, *args): | 
					
						
							|  |  |  |         return self._callmethod('send', args) | 
					
						
							|  |  |  |     def throw(self, *args): | 
					
						
							|  |  |  |         return self._callmethod('throw', args) | 
					
						
							|  |  |  |     def close(self, *args): | 
					
						
							|  |  |  |         return self._callmethod('close', args) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class AcquirerProxy(BaseProxy): | 
					
						
							|  |  |  |     _exposed_ = ('acquire', 'release') | 
					
						
							| 
									
										
										
										
											2012-05-06 16:45:02 +01:00
										 |  |  |     def acquire(self, blocking=True, timeout=None): | 
					
						
							|  |  |  |         args = (blocking,) if timeout is None else (blocking, timeout) | 
					
						
							|  |  |  |         return self._callmethod('acquire', args) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     def release(self): | 
					
						
							|  |  |  |         return self._callmethod('release') | 
					
						
							|  |  |  |     def __enter__(self): | 
					
						
							|  |  |  |         return self._callmethod('acquire') | 
					
						
							|  |  |  |     def __exit__(self, exc_type, exc_val, exc_tb): | 
					
						
							|  |  |  |         return self._callmethod('release') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ConditionProxy(AcquirerProxy): | 
					
						
							| 
									
										
										
										
											2008-06-11 19:14:14 +00:00
										 |  |  |     _exposed_ = ('acquire', 'release', 'wait', 'notify', 'notify_all') | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     def wait(self, timeout=None): | 
					
						
							|  |  |  |         return self._callmethod('wait', (timeout,)) | 
					
						
							| 
									
										
										
										
											2017-07-04 08:59:22 +02:00
										 |  |  |     def notify(self, n=1): | 
					
						
							|  |  |  |         return self._callmethod('notify', (n,)) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     def notify_all(self): | 
					
						
							| 
									
										
										
										
											2008-06-11 19:14:14 +00:00
										 |  |  |         return self._callmethod('notify_all') | 
					
						
							| 
									
										
										
										
											2012-04-17 18:45:57 +02:00
										 |  |  |     def wait_for(self, predicate, timeout=None): | 
					
						
							|  |  |  |         result = predicate() | 
					
						
							|  |  |  |         if result: | 
					
						
							|  |  |  |             return result | 
					
						
							|  |  |  |         if timeout is not None: | 
					
						
							| 
									
										
										
										
											2018-07-06 13:51:52 +02:00
										 |  |  |             endtime = time.monotonic() + timeout | 
					
						
							| 
									
										
										
										
											2012-04-17 18:45:57 +02:00
										 |  |  |         else: | 
					
						
							|  |  |  |             endtime = None | 
					
						
							|  |  |  |             waittime = None | 
					
						
							|  |  |  |         while not result: | 
					
						
							|  |  |  |             if endtime is not None: | 
					
						
							| 
									
										
										
										
											2018-07-06 13:51:52 +02:00
										 |  |  |                 waittime = endtime - time.monotonic() | 
					
						
							| 
									
										
										
										
											2012-04-17 18:45:57 +02:00
										 |  |  |                 if waittime <= 0: | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  |             self.wait(waittime) | 
					
						
							|  |  |  |             result = predicate() | 
					
						
							|  |  |  |         return result | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class EventProxy(BaseProxy): | 
					
						
							| 
									
										
										
										
											2008-06-12 17:02:47 +00:00
										 |  |  |     _exposed_ = ('is_set', 'set', 'clear', 'wait') | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     def is_set(self): | 
					
						
							| 
									
										
										
										
											2008-06-12 17:02:47 +00:00
										 |  |  |         return self._callmethod('is_set') | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     def set(self): | 
					
						
							|  |  |  |         return self._callmethod('set') | 
					
						
							|  |  |  |     def clear(self): | 
					
						
							|  |  |  |         return self._callmethod('clear') | 
					
						
							|  |  |  |     def wait(self, timeout=None): | 
					
						
							|  |  |  |         return self._callmethod('wait', (timeout,)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-15 18:26:07 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | class BarrierProxy(BaseProxy): | 
					
						
							|  |  |  |     _exposed_ = ('__getattribute__', 'wait', 'abort', 'reset') | 
					
						
							|  |  |  |     def wait(self, timeout=None): | 
					
						
							|  |  |  |         return self._callmethod('wait', (timeout,)) | 
					
						
							|  |  |  |     def abort(self): | 
					
						
							|  |  |  |         return self._callmethod('abort') | 
					
						
							|  |  |  |     def reset(self): | 
					
						
							|  |  |  |         return self._callmethod('reset') | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def parties(self): | 
					
						
							|  |  |  |         return self._callmethod('__getattribute__', ('parties',)) | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def n_waiting(self): | 
					
						
							|  |  |  |         return self._callmethod('__getattribute__', ('n_waiting',)) | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def broken(self): | 
					
						
							|  |  |  |         return self._callmethod('__getattribute__', ('broken',)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | class NamespaceProxy(BaseProxy): | 
					
						
							|  |  |  |     _exposed_ = ('__getattribute__', '__setattr__', '__delattr__') | 
					
						
							|  |  |  |     def __getattr__(self, key): | 
					
						
							|  |  |  |         if key[0] == '_': | 
					
						
							|  |  |  |             return object.__getattribute__(self, key) | 
					
						
							|  |  |  |         callmethod = object.__getattribute__(self, '_callmethod') | 
					
						
							|  |  |  |         return callmethod('__getattribute__', (key,)) | 
					
						
							|  |  |  |     def __setattr__(self, key, value): | 
					
						
							|  |  |  |         if key[0] == '_': | 
					
						
							|  |  |  |             return object.__setattr__(self, key, value) | 
					
						
							|  |  |  |         callmethod = object.__getattribute__(self, '_callmethod') | 
					
						
							|  |  |  |         return callmethod('__setattr__', (key, value)) | 
					
						
							|  |  |  |     def __delattr__(self, key): | 
					
						
							|  |  |  |         if key[0] == '_': | 
					
						
							|  |  |  |             return object.__delattr__(self, key) | 
					
						
							|  |  |  |         callmethod = object.__getattribute__(self, '_callmethod') | 
					
						
							|  |  |  |         return callmethod('__delattr__', (key,)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ValueProxy(BaseProxy): | 
					
						
							|  |  |  |     _exposed_ = ('get', 'set') | 
					
						
							|  |  |  |     def get(self): | 
					
						
							|  |  |  |         return self._callmethod('get') | 
					
						
							|  |  |  |     def set(self, value): | 
					
						
							|  |  |  |         return self._callmethod('set', (value,)) | 
					
						
							|  |  |  |     value = property(get, set) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-10 17:46:36 +03:00
										 |  |  |     __class_getitem__ = classmethod(types.GenericAlias) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | BaseListProxy = MakeProxyType('BaseListProxy', ( | 
					
						
							| 
									
										
										
										
											2012-05-29 12:01:45 +01:00
										 |  |  |     '__add__', '__contains__', '__delitem__', '__getitem__', '__len__', | 
					
						
							|  |  |  |     '__mul__', '__reversed__', '__rmul__', '__setitem__', | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     'append', 'count', 'extend', 'index', 'insert', 'pop', 'remove', | 
					
						
							|  |  |  |     'reverse', 'sort', '__imul__' | 
					
						
							| 
									
										
										
										
											2012-05-29 12:01:45 +01:00
										 |  |  |     )) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | class ListProxy(BaseListProxy): | 
					
						
							|  |  |  |     def __iadd__(self, value): | 
					
						
							|  |  |  |         self._callmethod('extend', (value,)) | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  |     def __imul__(self, value): | 
					
						
							|  |  |  |         self._callmethod('__imul__', (value,)) | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DictProxy = MakeProxyType('DictProxy', ( | 
					
						
							| 
									
										
										
										
											2018-09-17 14:24:01 +03:00
										 |  |  |     '__contains__', '__delitem__', '__getitem__', '__iter__', '__len__', | 
					
						
							| 
									
										
										
										
											2019-02-12 01:37:24 +01:00
										 |  |  |     '__setitem__', 'clear', 'copy', 'get', 'items', | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     'keys', 'pop', 'popitem', 'setdefault', 'update', 'values' | 
					
						
							|  |  |  |     )) | 
					
						
							| 
									
										
										
										
											2018-09-17 14:24:01 +03:00
										 |  |  | DictProxy._method_to_typeid_ = { | 
					
						
							|  |  |  |     '__iter__': 'Iterator', | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ArrayProxy = MakeProxyType('ArrayProxy', ( | 
					
						
							| 
									
										
										
										
											2012-05-29 12:01:45 +01:00
										 |  |  |     '__len__', '__getitem__', '__setitem__' | 
					
						
							|  |  |  |     )) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-23 12:30:54 +00:00
										 |  |  | BasePoolProxy = MakeProxyType('PoolProxy', ( | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     'apply', 'apply_async', 'close', 'imap', 'imap_unordered', 'join', | 
					
						
							| 
									
										
										
										
											2014-03-23 12:30:54 +00:00
										 |  |  |     'map', 'map_async', 'starmap', 'starmap_async', 'terminate', | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     )) | 
					
						
							| 
									
										
										
										
											2014-03-23 12:30:54 +00:00
										 |  |  | BasePoolProxy._method_to_typeid_ = { | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     'apply_async': 'AsyncResult', | 
					
						
							|  |  |  |     'map_async': 'AsyncResult', | 
					
						
							| 
									
										
										
										
											2011-12-21 11:03:24 +01:00
										 |  |  |     'starmap_async': 'AsyncResult', | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  |     'imap': 'Iterator', | 
					
						
							|  |  |  |     'imap_unordered': 'Iterator' | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-23 12:30:54 +00:00
										 |  |  | class PoolProxy(BasePoolProxy): | 
					
						
							|  |  |  |     def __enter__(self): | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  |     def __exit__(self, exc_type, exc_val, exc_tb): | 
					
						
							|  |  |  |         self.terminate() | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Definition of SyncManager | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class SyncManager(BaseManager): | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  |     Subclass of `BaseManager` which supports a number of shared object types. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     The types registered are those intended for the synchronization | 
					
						
							|  |  |  |     of threads, plus `dict`, `list` and `Namespace`. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     The `multiprocessing.Manager()` function creates started instances of | 
					
						
							|  |  |  |     this class. | 
					
						
							|  |  |  |     '''
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SyncManager.register('Queue', queue.Queue) | 
					
						
							|  |  |  | SyncManager.register('JoinableQueue', queue.Queue) | 
					
						
							|  |  |  | SyncManager.register('Event', threading.Event, EventProxy) | 
					
						
							|  |  |  | SyncManager.register('Lock', threading.Lock, AcquirerProxy) | 
					
						
							|  |  |  | SyncManager.register('RLock', threading.RLock, AcquirerProxy) | 
					
						
							|  |  |  | SyncManager.register('Semaphore', threading.Semaphore, AcquirerProxy) | 
					
						
							|  |  |  | SyncManager.register('BoundedSemaphore', threading.BoundedSemaphore, | 
					
						
							|  |  |  |                      AcquirerProxy) | 
					
						
							|  |  |  | SyncManager.register('Condition', threading.Condition, ConditionProxy) | 
					
						
							| 
									
										
										
										
											2012-06-15 18:26:07 +01:00
										 |  |  | SyncManager.register('Barrier', threading.Barrier, BarrierProxy) | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | SyncManager.register('Pool', pool.Pool, PoolProxy) | 
					
						
							| 
									
										
										
										
											2008-06-11 16:44:04 +00:00
										 |  |  | SyncManager.register('list', list, ListProxy) | 
					
						
							|  |  |  | SyncManager.register('dict', dict, DictProxy) | 
					
						
							|  |  |  | SyncManager.register('Value', Value, ValueProxy) | 
					
						
							|  |  |  | SyncManager.register('Array', Array, ArrayProxy) | 
					
						
							|  |  |  | SyncManager.register('Namespace', Namespace, NamespaceProxy) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # types returned by methods of PoolProxy | 
					
						
							|  |  |  | SyncManager.register('Iterator', proxytype=IteratorProxy, create_method=False) | 
					
						
							|  |  |  | SyncManager.register('AsyncResult', create_method=False) | 
					
						
							| 
									
										
										
										
											2019-02-23 22:08:16 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Definition of SharedMemoryManager and SharedMemoryServer | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if HAS_SHMEM: | 
					
						
							|  |  |  |     class _SharedMemoryTracker: | 
					
						
							|  |  |  |         "Manages one or more shared memory segments." | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def __init__(self, name, segment_names=[]): | 
					
						
							|  |  |  |             self.shared_memory_context_name = name | 
					
						
							|  |  |  |             self.segment_names = segment_names | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def register_segment(self, segment_name): | 
					
						
							|  |  |  |             "Adds the supplied shared memory block name to tracker." | 
					
						
							|  |  |  |             util.debug(f"Register segment {segment_name!r} in pid {getpid()}") | 
					
						
							|  |  |  |             self.segment_names.append(segment_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def destroy_segment(self, segment_name): | 
					
						
							|  |  |  |             """Calls unlink() on the shared memory block with the supplied name
 | 
					
						
							|  |  |  |             and removes it from the list of blocks being tracked."""
 | 
					
						
							|  |  |  |             util.debug(f"Destroy segment {segment_name!r} in pid {getpid()}") | 
					
						
							|  |  |  |             self.segment_names.remove(segment_name) | 
					
						
							|  |  |  |             segment = shared_memory.SharedMemory(segment_name) | 
					
						
							|  |  |  |             segment.close() | 
					
						
							|  |  |  |             segment.unlink() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def unlink(self): | 
					
						
							|  |  |  |             "Calls destroy_segment() on all tracked shared memory blocks." | 
					
						
							|  |  |  |             for segment_name in self.segment_names[:]: | 
					
						
							|  |  |  |                 self.destroy_segment(segment_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def __del__(self): | 
					
						
							|  |  |  |             util.debug(f"Call {self.__class__.__name__}.__del__ in {getpid()}") | 
					
						
							|  |  |  |             self.unlink() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def __getstate__(self): | 
					
						
							|  |  |  |             return (self.shared_memory_context_name, self.segment_names) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def __setstate__(self, state): | 
					
						
							|  |  |  |             self.__init__(*state) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     class SharedMemoryServer(Server): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         public = Server.public + \ | 
					
						
							|  |  |  |                  ['track_segment', 'release_segment', 'list_segments'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def __init__(self, *args, **kwargs): | 
					
						
							|  |  |  |             Server.__init__(self, *args, **kwargs) | 
					
						
							| 
									
										
										
										
											2020-03-09 13:48:01 +00:00
										 |  |  |             address = self.address | 
					
						
							|  |  |  |             # The address of Linux abstract namespaces can be bytes | 
					
						
							|  |  |  |             if isinstance(address, bytes): | 
					
						
							|  |  |  |                 address = os.fsdecode(address) | 
					
						
							| 
									
										
										
										
											2019-02-23 22:08:16 -06:00
										 |  |  |             self.shared_memory_context = \ | 
					
						
							| 
									
										
										
										
											2020-03-09 13:48:01 +00:00
										 |  |  |                 _SharedMemoryTracker(f"shm_{address}_{getpid()}") | 
					
						
							| 
									
										
										
										
											2019-02-23 22:08:16 -06:00
										 |  |  |             util.debug(f"SharedMemoryServer started by pid {getpid()}") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-05 18:22:31 +03:00
										 |  |  |         def create(self, c, typeid, /, *args, **kwargs): | 
					
						
							| 
									
										
										
										
											2019-02-23 22:08:16 -06:00
										 |  |  |             """Create a new distributed-shared object (not backed by a shared
 | 
					
						
							|  |  |  |             memory block) and return its id to be used in a Proxy Object."""
 | 
					
						
							|  |  |  |             # Unless set up as a shared proxy, don't make shared_memory_context | 
					
						
							|  |  |  |             # a standard part of kwargs.  This makes things easier for supplying | 
					
						
							|  |  |  |             # simple functions. | 
					
						
							|  |  |  |             if hasattr(self.registry[typeid][-1], "_shared_memory_proxy"): | 
					
						
							|  |  |  |                 kwargs['shared_memory_context'] = self.shared_memory_context | 
					
						
							| 
									
										
										
										
											2019-06-05 18:22:31 +03:00
										 |  |  |             return Server.create(self, c, typeid, *args, **kwargs) | 
					
						
							| 
									
										
										
										
											2019-02-23 22:08:16 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |         def shutdown(self, c): | 
					
						
							|  |  |  |             "Call unlink() on all tracked shared memory, terminate the Server." | 
					
						
							|  |  |  |             self.shared_memory_context.unlink() | 
					
						
							|  |  |  |             return Server.shutdown(self, c) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def track_segment(self, c, segment_name): | 
					
						
							|  |  |  |             "Adds the supplied shared memory block name to Server's tracker." | 
					
						
							|  |  |  |             self.shared_memory_context.register_segment(segment_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def release_segment(self, c, segment_name): | 
					
						
							|  |  |  |             """Calls unlink() on the shared memory block with the supplied name
 | 
					
						
							|  |  |  |             and removes it from the tracker instance inside the Server."""
 | 
					
						
							|  |  |  |             self.shared_memory_context.destroy_segment(segment_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def list_segments(self, c): | 
					
						
							|  |  |  |             """Returns a list of names of shared memory blocks that the Server
 | 
					
						
							|  |  |  |             is currently tracking."""
 | 
					
						
							|  |  |  |             return self.shared_memory_context.segment_names | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     class SharedMemoryManager(BaseManager): | 
					
						
							|  |  |  |         """Like SyncManager but uses SharedMemoryServer instead of Server.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         It provides methods for creating and returning SharedMemory instances | 
					
						
							|  |  |  |         and for creating a list-like object (ShareableList) backed by shared | 
					
						
							|  |  |  |         memory.  It also provides methods that create and return Proxy Objects | 
					
						
							|  |  |  |         that support synchronization across processes (i.e. multi-process-safe | 
					
						
							|  |  |  |         locks and semaphores). | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         _Server = SharedMemoryServer | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def __init__(self, *args, **kwargs): | 
					
						
							| 
									
										
										
										
											2019-05-13 21:15:32 +02:00
										 |  |  |             if os.name == "posix": | 
					
						
							|  |  |  |                 # bpo-36867: Ensure the resource_tracker is running before | 
					
						
							|  |  |  |                 # launching the manager process, so that concurrent | 
					
						
							|  |  |  |                 # shared_memory manipulation both in the manager and in the | 
					
						
							|  |  |  |                 # current process does not create two resource_tracker | 
					
						
							|  |  |  |                 # processes. | 
					
						
							|  |  |  |                 from . import resource_tracker | 
					
						
							|  |  |  |                 resource_tracker.ensure_running() | 
					
						
							| 
									
										
										
										
											2019-02-23 22:08:16 -06:00
										 |  |  |             BaseManager.__init__(self, *args, **kwargs) | 
					
						
							|  |  |  |             util.debug(f"{self.__class__.__name__} created by pid {getpid()}") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def __del__(self): | 
					
						
							|  |  |  |             util.debug(f"{self.__class__.__name__}.__del__ by pid {getpid()}") | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def get_server(self): | 
					
						
							|  |  |  |             'Better than monkeypatching for now; merge into Server ultimately' | 
					
						
							|  |  |  |             if self._state.value != State.INITIAL: | 
					
						
							|  |  |  |                 if self._state.value == State.STARTED: | 
					
						
							|  |  |  |                     raise ProcessError("Already started SharedMemoryServer") | 
					
						
							|  |  |  |                 elif self._state.value == State.SHUTDOWN: | 
					
						
							|  |  |  |                     raise ProcessError("SharedMemoryManager has shut down") | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     raise ProcessError( | 
					
						
							|  |  |  |                         "Unknown state {!r}".format(self._state.value)) | 
					
						
							|  |  |  |             return self._Server(self._registry, self._address, | 
					
						
							|  |  |  |                                 self._authkey, self._serializer) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def SharedMemory(self, size): | 
					
						
							|  |  |  |             """Returns a new SharedMemory instance with the specified size in
 | 
					
						
							|  |  |  |             bytes, to be tracked by the manager."""
 | 
					
						
							|  |  |  |             with self._Client(self._address, authkey=self._authkey) as conn: | 
					
						
							|  |  |  |                 sms = shared_memory.SharedMemory(None, create=True, size=size) | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     dispatch(conn, None, 'track_segment', (sms.name,)) | 
					
						
							|  |  |  |                 except BaseException as e: | 
					
						
							|  |  |  |                     sms.unlink() | 
					
						
							|  |  |  |                     raise e | 
					
						
							|  |  |  |             return sms | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def ShareableList(self, sequence): | 
					
						
							|  |  |  |             """Returns a new ShareableList instance populated with the values
 | 
					
						
							|  |  |  |             from the input sequence, to be tracked by the manager."""
 | 
					
						
							|  |  |  |             with self._Client(self._address, authkey=self._authkey) as conn: | 
					
						
							|  |  |  |                 sl = shared_memory.ShareableList(sequence) | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     dispatch(conn, None, 'track_segment', (sl.shm.name,)) | 
					
						
							|  |  |  |                 except BaseException as e: | 
					
						
							|  |  |  |                     sl.shm.unlink() | 
					
						
							|  |  |  |                     raise e | 
					
						
							|  |  |  |             return sl |