3 fR @ s d Z ddlZddlZddlZddlZddlZddlZddlZddlZddl Z ddl Z ddlZddlZddl Z ddlZddlZdd Zdd Zd+ddZG d d d ejZd,Zd ZG dd de jZi ZejdZejdZG dd deZG dd deZdd Z G dd deZ!G dd de j"eZ#G dd deZ$G dd deZ%dd Z&d d! Z'G d"d# d#eZ(d$d% Z)e*d&krdd'l+m,Z, e,d(d)d* dS )-a RPC Implementation, originally written for the Python Idle IDE For security reasons, GvR requested that Idle's Python execution server process connect to the Idle process, which listens for the connection. Since Idle has only one client per server, this was not a limitation. +---------------------------------+ +-------------+ | socketserver.BaseRequestHandler | | SocketIO | +---------------------------------+ +-------------+ ^ | register() | | | unregister()| | +-------------+ | ^ ^ | | | | + -------------------+ | | | | +-------------------------+ +-----------------+ | RPCHandler | | RPCClient | | [attribute of RPCServer]| | | +-------------------------+ +-----------------+ The RPCServer handler class is expected to provide register/unregister methods. RPCHandler inherits the mix-in class SocketIO, which provides these methods. See the Idle run.main() docstring for further information on how this was accomplished in Idle. Nc C s t j| }|S )z*Return code object from marshal string ms.)marshalloads)msco r 0/opt/alt/python36/lib64/python3.6/idlelib/rpc.py unpickle_code- s r c C s t j| }t|ffS )zBReturn unpickle function and tuple with marshalled co code object.)r dumpsr )r r r r r pickle_code3 s r c C s$ t j }t||}|j| |j S )z.Return pickled (or marshalled) string for obj.)ioBytesIOCodePicklerdumpgetvalue)objZprotocolfpr r r r 9 s r c @ s" e Zd ZejeiZejej dS )r N) __name__ __module____qualname__typesCodeTyper dispatch_tableupdatecopyregr r r r r B s r i z 127.0.0.1c @ s6 e Zd ZdddZdd Zdd Zdd Zd d ZdS ) RPCServerNc C s |d krt }tjj| || d S )N) RPCHandlersocketserver TCPServer__init__)selfaddrZhandlerclassr r r r L s zRPCServer.__init__c C s dS )z@Override TCPServer method, no bind() phase for connecting entityNr )r! r r r server_bindQ s zRPCServer.server_bindc C s | j j| j dS )zOverride TCPServer method, connect() instead of listen() Due to the reversed connection, self.server_address is actually the address of the Idle Client to which we are connecting. N)socketZconnectserver_address)r! r r r server_activateU s zRPCServer.server_activatec C s | j | jfS )z:Override TCPServer method, return already connected socket)r$ r% )r! r r r get_request^ s zRPCServer.get_requestc C s y W n t k r Y n tj}tddd |d td|d tdtj j |d td||d tdt||d tj |d td |d tdd |d t jd Y nX dS )zOverride TCPServer method Error message goes to __stderr__. No error message if exiting normally or socket raised EOF. Other exceptions not handled in server code will cause os._exit. -( )filezUnhandled server exception!z Thread: %szClient Address: z Request: z# *** Unrecoverable, server exiting!r N) SystemExitsys __stderr__print threadingcurrent_threadnamerepr traceback print_excos_exit)r! requestZclient_addressZerfr r r handle_errorb s zRPCServer.handle_error)N)r r r r r# r&