mirror of
				https://github.com/python/cpython.git
				synced 2025-11-03 19:34:08 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1120 lines
		
	
	
	
		
			35 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			1120 lines
		
	
	
	
		
			35 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
#
 | 
						|
# Module providing the `SyncManager` class for dealing
 | 
						|
# with shared objects
 | 
						|
#
 | 
						|
# multiprocessing/managers.py
 | 
						|
#
 | 
						|
# Copyright (c) 2006-2008, R Oudkerk
 | 
						|
# Licensed to PSF under a Contributor Agreement.
 | 
						|
#
 | 
						|
 | 
						|
__all__ = [ 'BaseManager', 'SyncManager', 'BaseProxy', 'Token' ]
 | 
						|
 | 
						|
#
 | 
						|
# Imports
 | 
						|
#
 | 
						|
 | 
						|
import sys
 | 
						|
import threading
 | 
						|
import array
 | 
						|
import queue
 | 
						|
 | 
						|
from traceback import format_exc
 | 
						|
from multiprocessing import Process, current_process, active_children, Pool, util, connection
 | 
						|
from multiprocessing.process import AuthenticationString
 | 
						|
from multiprocessing.forking import Popen, ForkingPickler
 | 
						|
from time import time as _time
 | 
						|
 | 
						|
#
 | 
						|
# Register some things for pickling
 | 
						|
#
 | 
						|
 | 
						|
def reduce_array(a):
 | 
						|
    return array.array, (a.typecode, a.tobytes())
 | 
						|
ForkingPickler.register(array.array, reduce_array)
 | 
						|
 | 
						|
view_types = [type(getattr({}, name)()) for name in ('items','keys','values')]
 | 
						|
if view_types[0] is not list:       # only needed in Py3.0
 | 
						|
    def rebuild_as_list(obj):
 | 
						|
        return list, (list(obj),)
 | 
						|
    for view_type in view_types:
 | 
						|
        ForkingPickler.register(view_type, rebuild_as_list)
 | 
						|
        import copyreg
 | 
						|
        copyreg.pickle(view_type, rebuild_as_list)
 | 
						|
 | 
						|
#
 | 
						|
# Type for identifying shared objects
 | 
						|
#
 | 
						|
 | 
						|
class Token(object):
 | 
						|
    '''
 | 
						|
    Type to uniquely indentify a shared object
 | 
						|
    '''
 | 
						|
    __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):
 | 
						|
        return 'Token(typeid=%r, address=%r, id=%r)' % \
 | 
						|
               (self.typeid, self.address, self.id)
 | 
						|
 | 
						|
#
 | 
						|
# 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
 | 
						|
    elif kind == '#TRACEBACK':
 | 
						|
        assert type(result) is str
 | 
						|
        return  RemoteError(result)
 | 
						|
    elif kind == '#UNSERIALIZABLE':
 | 
						|
        assert type(result) is str
 | 
						|
        return RemoteError('Unserializable message: %s\n' % result)
 | 
						|
    else:
 | 
						|
        return ValueError('Unrecognized message type')
 | 
						|
 | 
						|
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)
 | 
						|
        if callable(func):
 | 
						|
            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):
 | 
						|
        assert isinstance(authkey, bytes)
 | 
						|
        self.registry = registry
 | 
						|
        self.authkey = AuthenticationString(authkey)
 | 
						|
        Listener, Client = listener_client[serializer]
 | 
						|
 | 
						|
        # do authentication later
 | 
						|
        self.listener = Listener(address=address, backlog=16)
 | 
						|
        self.address = self.listener.address
 | 
						|
 | 
						|
        self.id_to_obj = {'0': (None, ())}
 | 
						|
        self.id_to_refcount = {}
 | 
						|
        self.mutex = threading.RLock()
 | 
						|
 | 
						|
    def serve_forever(self):
 | 
						|
        '''
 | 
						|
        Run the server forever
 | 
						|
        '''
 | 
						|
        self.stop_event = threading.Event()
 | 
						|
        current_process()._manager_server = self
 | 
						|
        try:
 | 
						|
            accepter = threading.Thread(target=self.accepter)
 | 
						|
            accepter.daemon = True
 | 
						|
            accepter.start()
 | 
						|
            try:
 | 
						|
                while not self.stop_event.is_set():
 | 
						|
                    self.stop_event.wait(1)
 | 
						|
            except (KeyboardInterrupt, SystemExit):
 | 
						|
                pass
 | 
						|
        finally:
 | 
						|
            if sys.stdout != sys.__stdout__:
 | 
						|
                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()
 | 
						|
            except OSError:
 | 
						|
                continue
 | 
						|
            t = threading.Thread(target=self.handle_request, args=(c,))
 | 
						|
            t.daemon = True
 | 
						|
            t.start()
 | 
						|
 | 
						|
    def handle_request(self, c):
 | 
						|
        '''
 | 
						|
        Handle a new connection
 | 
						|
        '''
 | 
						|
        funcname = result = request = None
 | 
						|
        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)
 | 
						|
        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)
 | 
						|
 | 
						|
        c.close()
 | 
						|
 | 
						|
    def serve_client(self, conn):
 | 
						|
        '''
 | 
						|
        Handle requests from the proxies in a particular process/thread
 | 
						|
        '''
 | 
						|
        util.debug('starting server thread to service %r',
 | 
						|
                   threading.current_thread().name)
 | 
						|
 | 
						|
        recv = conn.recv
 | 
						|
        send = conn.send
 | 
						|
        id_to_obj = self.id_to_obj
 | 
						|
 | 
						|
        while not self.stop_event.is_set():
 | 
						|
 | 
						|
            try:
 | 
						|
                methodname = obj = None
 | 
						|
                request = recv()
 | 
						|
                ident, methodname, args, kwds = request
 | 
						|
                obj, exposed, gettypeid = id_to_obj[ident]
 | 
						|
 | 
						|
                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',
 | 
						|
                           threading.current_thread().name)
 | 
						|
                sys.exit(0)
 | 
						|
 | 
						|
            except Exception:
 | 
						|
                msg = ('#TRACEBACK', format_exc())
 | 
						|
 | 
						|
            try:
 | 
						|
                try:
 | 
						|
                    send(msg)
 | 
						|
                except Exception as e:
 | 
						|
                    send(('#UNSERIALIZABLE', repr(msg)))
 | 
						|
            except Exception as e:
 | 
						|
                util.info('exception in thread serving %r',
 | 
						|
                        threading.current_thread().name)
 | 
						|
                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
 | 
						|
        '''
 | 
						|
        self.mutex.acquire()
 | 
						|
        try:
 | 
						|
            result = []
 | 
						|
            keys = list(self.id_to_obj.keys())
 | 
						|
            keys.sort()
 | 
						|
            for ident in keys:
 | 
						|
                if ident != '0':
 | 
						|
                    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)
 | 
						|
        finally:
 | 
						|
            self.mutex.release()
 | 
						|
 | 
						|
    def number_of_objects(self, c):
 | 
						|
        '''
 | 
						|
        Number of shared objects
 | 
						|
        '''
 | 
						|
        return len(self.id_to_obj) - 1      # don't count ident='0'
 | 
						|
 | 
						|
    def shutdown(self, c):
 | 
						|
        '''
 | 
						|
        Shutdown this process
 | 
						|
        '''
 | 
						|
        try:
 | 
						|
            util.debug('manager received shutdown message')
 | 
						|
            c.send(('#RETURN', None))
 | 
						|
        except:
 | 
						|
            import traceback
 | 
						|
            traceback.print_exc()
 | 
						|
        finally:
 | 
						|
            self.stop_event.set()
 | 
						|
 | 
						|
    def create(self, c, typeid, *args, **kwds):
 | 
						|
        '''
 | 
						|
        Create a new shared object and return its id
 | 
						|
        '''
 | 
						|
        self.mutex.acquire()
 | 
						|
        try:
 | 
						|
            callable, exposed, method_to_typeid, proxytype = \
 | 
						|
                      self.registry[typeid]
 | 
						|
 | 
						|
            if callable is None:
 | 
						|
                assert len(args) == 1 and not kwds
 | 
						|
                obj = args[0]
 | 
						|
            else:
 | 
						|
                obj = callable(*args, **kwds)
 | 
						|
 | 
						|
            if exposed is None:
 | 
						|
                exposed = public_methods(obj)
 | 
						|
            if method_to_typeid is not None:
 | 
						|
                assert type(method_to_typeid) is dict
 | 
						|
                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:
 | 
						|
                self.id_to_refcount[ident] = 0
 | 
						|
            # increment the reference count immediately, to avoid
 | 
						|
            # this object being garbage collected before a Proxy
 | 
						|
            # object for it can be created.  The caller of create()
 | 
						|
            # is responsible for doing a decref once the Proxy object
 | 
						|
            # has been created.
 | 
						|
            self.incref(c, ident)
 | 
						|
            return ident, tuple(exposed)
 | 
						|
        finally:
 | 
						|
            self.mutex.release()
 | 
						|
 | 
						|
    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
 | 
						|
        '''
 | 
						|
        threading.current_thread().name = name
 | 
						|
        c.send(('#RETURN', None))
 | 
						|
        self.serve_client(c)
 | 
						|
 | 
						|
    def incref(self, c, ident):
 | 
						|
        self.mutex.acquire()
 | 
						|
        try:
 | 
						|
            self.id_to_refcount[ident] += 1
 | 
						|
        finally:
 | 
						|
            self.mutex.release()
 | 
						|
 | 
						|
    def decref(self, c, ident):
 | 
						|
        self.mutex.acquire()
 | 
						|
        try:
 | 
						|
            assert self.id_to_refcount[ident] >= 1
 | 
						|
            self.id_to_refcount[ident] -= 1
 | 
						|
            if self.id_to_refcount[ident] == 0:
 | 
						|
                del self.id_to_obj[ident], self.id_to_refcount[ident]
 | 
						|
                util.debug('disposing of obj with id %r', ident)
 | 
						|
        finally:
 | 
						|
            self.mutex.release()
 | 
						|
 | 
						|
#
 | 
						|
# 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
 | 
						|
 | 
						|
    def __init__(self, address=None, authkey=None, serializer='pickle'):
 | 
						|
        if authkey is None:
 | 
						|
            authkey = current_process().authkey
 | 
						|
        self._address = address     # XXX not final address if eg ('', 0)
 | 
						|
        self._authkey = AuthenticationString(authkey)
 | 
						|
        self._state = State()
 | 
						|
        self._state.value = State.INITIAL
 | 
						|
        self._serializer = serializer
 | 
						|
        self._Listener, self._Client = listener_client[serializer]
 | 
						|
 | 
						|
    def get_server(self):
 | 
						|
        '''
 | 
						|
        Return server object with serve_forever() method and address attribute
 | 
						|
        '''
 | 
						|
        assert self._state.value == State.INITIAL
 | 
						|
        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
 | 
						|
 | 
						|
    def start(self, initializer=None, initargs=()):
 | 
						|
        '''
 | 
						|
        Spawn a server process for this manager object
 | 
						|
        '''
 | 
						|
        assert self._state.value == State.INITIAL
 | 
						|
 | 
						|
        if initializer is not None and not callable(initializer):
 | 
						|
            raise TypeError('initializer must be a callable')
 | 
						|
 | 
						|
        # pipe over which we will retrieve address of server
 | 
						|
        reader, writer = connection.Pipe(duplex=False)
 | 
						|
 | 
						|
        # spawn process which runs a server
 | 
						|
        self._process = Process(
 | 
						|
            target=type(self)._run_server,
 | 
						|
            args=(self._registry, self._address, self._authkey,
 | 
						|
                  self._serializer, writer, initializer, initargs),
 | 
						|
            )
 | 
						|
        ident = ':'.join(str(i) for i in self._process._identity)
 | 
						|
        self._process.name = type(self).__name__  + '-' + ident
 | 
						|
        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
 | 
						|
    def _run_server(cls, registry, address, authkey, serializer, writer,
 | 
						|
                    initializer=None, initargs=()):
 | 
						|
        '''
 | 
						|
        Create a server, report its address and run it
 | 
						|
        '''
 | 
						|
        if initializer is not None:
 | 
						|
            initializer(*initargs)
 | 
						|
 | 
						|
        # 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()
 | 
						|
 | 
						|
    def _create(self, typeid, *args, **kwds):
 | 
						|
        '''
 | 
						|
        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)
 | 
						|
        '''
 | 
						|
        if self._process is not None:
 | 
						|
            self._process.join(timeout)
 | 
						|
            if not self._process.is_alive():
 | 
						|
                self._process = None
 | 
						|
 | 
						|
    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):
 | 
						|
        if self._state.value == State.INITIAL:
 | 
						|
            self.start()
 | 
						|
        assert self._state.value == State.STARTED
 | 
						|
        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
 | 
						|
 | 
						|
            process.join(timeout=1.0)
 | 
						|
            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
 | 
						|
 | 
						|
    address = property(lambda self: self._address)
 | 
						|
 | 
						|
    @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:
 | 
						|
            for key, value in list(method_to_typeid.items()):
 | 
						|
                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:
 | 
						|
            def temp(self, *args, **kwds):
 | 
						|
                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
 | 
						|
                    )
 | 
						|
                conn = self._Client(token.address, authkey=self._authkey)
 | 
						|
                dispatch(conn, None, 'decref', (token.id,))
 | 
						|
                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,
 | 
						|
                 authkey=None, exposed=None, incref=True):
 | 
						|
        BaseProxy._mutex.acquire()
 | 
						|
        try:
 | 
						|
            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
 | 
						|
        finally:
 | 
						|
            BaseProxy._mutex.release()
 | 
						|
 | 
						|
        # 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]
 | 
						|
 | 
						|
        if authkey is not None:
 | 
						|
            self._authkey = AuthenticationString(authkey)
 | 
						|
        elif self._manager is not None:
 | 
						|
            self._authkey = self._manager._authkey
 | 
						|
        else:
 | 
						|
            self._authkey = current_process().authkey
 | 
						|
 | 
						|
        if incref:
 | 
						|
            self._incref()
 | 
						|
 | 
						|
        util.register_after_fork(self, BaseProxy._after_fork)
 | 
						|
 | 
						|
    def _connect(self):
 | 
						|
        util.debug('making connection to manager')
 | 
						|
        name = current_process().name
 | 
						|
        if threading.current_thread().name != 'MainThread':
 | 
						|
            name += '|' + threading.current_thread().name
 | 
						|
        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={}):
 | 
						|
        '''
 | 
						|
        Try to call a method of the referrent and return a copy of the result
 | 
						|
        '''
 | 
						|
        try:
 | 
						|
            conn = self._tls.connection
 | 
						|
        except AttributeError:
 | 
						|
            util.debug('thread %r does not own a connection',
 | 
						|
                       threading.current_thread().name)
 | 
						|
            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]
 | 
						|
            proxy = proxytype(
 | 
						|
                token, self._serializer, manager=self._manager,
 | 
						|
                authkey=self._authkey, exposed=exposed
 | 
						|
                )
 | 
						|
            conn = self._Client(token.address, authkey=self._authkey)
 | 
						|
            dispatch(conn, None, 'decref', (token.id,))
 | 
						|
            return proxy
 | 
						|
        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):
 | 
						|
        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',
 | 
						|
                       threading.current_thread().name)
 | 
						|
            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 = {}
 | 
						|
        if Popen.thread_is_spawning():
 | 
						|
            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):
 | 
						|
        return '<%s object, typeid %r at %s>' % \
 | 
						|
               (type(self).__name__, self._token.typeid, '0x%x' % id(self))
 | 
						|
 | 
						|
    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.
 | 
						|
 | 
						|
    If possible the shared object is returned, or otherwise a proxy for it.
 | 
						|
    '''
 | 
						|
    server = getattr(current_process(), '_manager_server', None)
 | 
						|
 | 
						|
    if server and server.address == token.address:
 | 
						|
        return server.id_to_obj[token.id][0]
 | 
						|
    else:
 | 
						|
        incref = (
 | 
						|
            kwds.pop('incref', True) and
 | 
						|
            not getattr(current_process(), '_inheriting', False)
 | 
						|
            )
 | 
						|
        return func(token, serializer, incref=incref, **kwds)
 | 
						|
 | 
						|
#
 | 
						|
# Functions to create proxies and proxy types
 | 
						|
#
 | 
						|
 | 
						|
def MakeProxyType(name, exposed, _cache={}):
 | 
						|
    '''
 | 
						|
    Return an proxy type whose methods are given by `exposed`
 | 
						|
    '''
 | 
						|
    exposed = tuple(exposed)
 | 
						|
    try:
 | 
						|
        return _cache[(name, exposed)]
 | 
						|
    except KeyError:
 | 
						|
        pass
 | 
						|
 | 
						|
    dic = {}
 | 
						|
 | 
						|
    for meth in exposed:
 | 
						|
        exec('''def %s(self, *args, **kwds):
 | 
						|
        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,
 | 
						|
              exposed=None, incref=True):
 | 
						|
    '''
 | 
						|
    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:
 | 
						|
        authkey = current_process().authkey
 | 
						|
 | 
						|
    ProxyType = MakeProxyType('AutoProxy[%s]' % token.typeid, exposed)
 | 
						|
    proxy = ProxyType(token, serializer, manager=manager, authkey=authkey,
 | 
						|
                      incref=incref)
 | 
						|
    proxy._isauto = True
 | 
						|
    return proxy
 | 
						|
 | 
						|
#
 | 
						|
# Types/callables which we will register with SyncManager
 | 
						|
#
 | 
						|
 | 
						|
class Namespace(object):
 | 
						|
    def __init__(self, **kwds):
 | 
						|
        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()
 | 
						|
        return 'Namespace(%s)' % str.join(', ', temp)
 | 
						|
 | 
						|
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):
 | 
						|
    _exposed_ = ('__next__', 'send', 'throw', 'close')
 | 
						|
    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')
 | 
						|
    def acquire(self, blocking=True, timeout=None):
 | 
						|
        args = (blocking,) if timeout is None else (blocking, timeout)
 | 
						|
        return self._callmethod('acquire', args)
 | 
						|
    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):
 | 
						|
    _exposed_ = ('acquire', 'release', 'wait', 'notify', 'notify_all')
 | 
						|
    def wait(self, timeout=None):
 | 
						|
        return self._callmethod('wait', (timeout,))
 | 
						|
    def notify(self):
 | 
						|
        return self._callmethod('notify')
 | 
						|
    def notify_all(self):
 | 
						|
        return self._callmethod('notify_all')
 | 
						|
    def wait_for(self, predicate, timeout=None):
 | 
						|
        result = predicate()
 | 
						|
        if result:
 | 
						|
            return result
 | 
						|
        if timeout is not None:
 | 
						|
            endtime = _time() + timeout
 | 
						|
        else:
 | 
						|
            endtime = None
 | 
						|
            waittime = None
 | 
						|
        while not result:
 | 
						|
            if endtime is not None:
 | 
						|
                waittime = endtime - _time()
 | 
						|
                if waittime <= 0:
 | 
						|
                    break
 | 
						|
            self.wait(waittime)
 | 
						|
            result = predicate()
 | 
						|
        return result
 | 
						|
 | 
						|
 | 
						|
class EventProxy(BaseProxy):
 | 
						|
    _exposed_ = ('is_set', 'set', 'clear', 'wait')
 | 
						|
    def is_set(self):
 | 
						|
        return self._callmethod('is_set')
 | 
						|
    def set(self):
 | 
						|
        return self._callmethod('set')
 | 
						|
    def clear(self):
 | 
						|
        return self._callmethod('clear')
 | 
						|
    def wait(self, timeout=None):
 | 
						|
        return self._callmethod('wait', (timeout,))
 | 
						|
 | 
						|
 | 
						|
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',))
 | 
						|
 | 
						|
 | 
						|
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)
 | 
						|
 | 
						|
 | 
						|
BaseListProxy = MakeProxyType('BaseListProxy', (
 | 
						|
    '__add__', '__contains__', '__delitem__', '__getitem__', '__len__',
 | 
						|
    '__mul__', '__reversed__', '__rmul__', '__setitem__',
 | 
						|
    'append', 'count', 'extend', 'index', 'insert', 'pop', 'remove',
 | 
						|
    'reverse', 'sort', '__imul__'
 | 
						|
    ))
 | 
						|
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', (
 | 
						|
    '__contains__', '__delitem__', '__getitem__', '__len__',
 | 
						|
    '__setitem__', 'clear', 'copy', 'get', 'has_key', 'items',
 | 
						|
    'keys', 'pop', 'popitem', 'setdefault', 'update', 'values'
 | 
						|
    ))
 | 
						|
 | 
						|
 | 
						|
ArrayProxy = MakeProxyType('ArrayProxy', (
 | 
						|
    '__len__', '__getitem__', '__setitem__'
 | 
						|
    ))
 | 
						|
 | 
						|
 | 
						|
PoolProxy = MakeProxyType('PoolProxy', (
 | 
						|
    'apply', 'apply_async', 'close', 'imap', 'imap_unordered', 'join',
 | 
						|
    'map', 'map_async', 'starmap', 'starmap_async', 'terminate'
 | 
						|
    ))
 | 
						|
PoolProxy._method_to_typeid_ = {
 | 
						|
    'apply_async': 'AsyncResult',
 | 
						|
    'map_async': 'AsyncResult',
 | 
						|
    'starmap_async': 'AsyncResult',
 | 
						|
    'imap': 'Iterator',
 | 
						|
    'imap_unordered': 'Iterator'
 | 
						|
    }
 | 
						|
 | 
						|
#
 | 
						|
# 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)
 | 
						|
SyncManager.register('Barrier', threading.Barrier, BarrierProxy)
 | 
						|
SyncManager.register('Pool', Pool, PoolProxy)
 | 
						|
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)
 |