U .cTR@sdZddlZddlZddlZddlZddlZddlZddlZddlZddl Z ddl Z ddl Z ddl Z ddl Z ddlZddlZddZddZd*ddZGd d d ejZd Zd ZGd dde jZiZedZedZGdddZGdddZddZGdddZ Gddde j!eZ"GdddeZ#GdddZ$ddZ%dd Z&Gd!d"d"Z'd#d$Z(e)d%krdd&l*m+Z+e+d'd(d)dS)+aRPC 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. NcCst|}t|tjst|S)z*Return code object from marshal string ms.)marshalloads isinstancetypesCodeTypeAssertionError)mscor 0/opt/alt/python38/lib64/python3.8/idlelib/rpc.py unpickle_code-s r cCs$t|tjstt|}t|ffS)zBReturn unpickle function and tuple with marshalled co code object.)rrrrrdumpsr )r rr r r pickle_code3s rcCs$t}t||}|||S)z.Return pickled (or marshalled) string for obj.)ioBytesIO CodePicklerdumpgetvalue)objZprotocolfpr r r r 9s  r c@seZdZejeiejZdS)rN)__name__ __module__ __qualname__rrrcopyregdispatch_tabler r r r rBsri z 127.0.0.1c@s6eZdZd ddZddZddZdd Zd d ZdS) RPCServerNcCs |dkr t}tj|||dSN) RPCHandler socketserver TCPServer__init__)selfaddrZ handlerclassr r r r!KszRPCServer.__init__cCsdS)z@Override TCPServer method, no bind() phase for connecting entityNr r"r r r server_bindPszRPCServer.server_bindcCs|j|jdS)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_addressr$r r r server_activateTszRPCServer.server_activatecCs |j|jfS)z:Override TCPServer method, return already connected socket)r&r'r$r r r get_request]szRPCServer.get_requestcCszWntk rYntj}td|dtd|dtdtj|dtd||dtdt||dtj |dtd|dtd|dt d YnXd S) 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. z) ----------------------------------------filezUnhandled server exception!z Thread: %szClient Address: z Request: z# *** Unrecoverable, server exiting!z(----------------------------------------rN) SystemExitsys __stderr__print threadingcurrent_threadnamerepr traceback print_excos_exit)r"requestZclient_addressZerfr r r handle_erroras     zRPCServer.handle_error)N)rrrr!r%r(r)r9r r r r rIs   rc@seZdZdZd;ddZddZddZd d Zd d Zd dZ ddZ ddZ ddZ ddZ ddZddZddZddZdd Zd!d"Zd#d$Zd%d&Zd'd(Zd)d*Zd+Zd,ZdZd-d.Zd/d0Zd1d2Zd3d4Zd5d6Zd7d8Z d9d:Z!dS)<SocketIOrNcCs@t|_|dk r||_||_|dkr*t}||_i|_i|_dSr) r0r1 sockthread debuggingsock objecttableobjtable responsescvars)r"r=r?r<r r r r!s zSocketIO.__init__cCs |j}d|_|dk r|dSr)r=close)r"r=r r r rBszSocketIO.closecCstddS)z!override for specific exit actionrN)r6r7r$r r r exithookszSocketIO.exithookcGsN|js dS|jdttj}|D]}|dt|}q&t|tjddS)N r*) r<locationstrr0r1r2r/r-r.)r"argssar r r debugs zSocketIO.debugcCs||j|<dSr)r?)r"oidobjectr r r registerszSocketIO.registercCs&z |j|=Wntk r YnXdSr)r?KeyErrorr"rKr r r unregisters zSocketIO.unregisterc Cs|d|z|\}\}}}}Wntk r6YdSX||jkrPdd|ffS|j|}|dkrxi} t|| d| fS|dkri} t|| d| fSt||sdd|ffSt||} zd|d kr| ||} t| trt | } d| fWS|d kr t || ||ffWd Sdd |fWSWnt k r4Ynt k rJYn|tk r`Ynftk r} zd | fWYSd} ~ XYn8d}t||| |ftjdtjtjdYdSXdS)Nz localcall:)ERRORzBad request formatrQzUnknown object id: %r __methods__OK__attributes__zUnsupported method name: %rCALLQUEUE)QUEUEDNzUnsupported message type: %sCALLEXCzU*** Internal Error: rpc.py:SocketIO.localcall() Object: %s Method: %s Args: %s r*) EXCEPTIONN)rJ TypeErrorr? _getmethods_getattributeshasattrgetattrr RemoteObject remoteref request_queueZputr,KeyboardInterruptOSError Exceptionr/r-r.r4r5)r"seqr8howrK methodnamerGkwargsrmethods attributesmethodZretZexmsgr r r localcallsT           zSocketIO.localcallcCs(|d|||||||}||S)Nzremotecall:asynccall: )rJ asynccall asyncreturnr"rKrgrGrhrer r r remotecallszSocketIO.remotecallcCs(|d|||||||}||S)Nzremotequeue:asyncqueue: )rJ asyncqueuerorpr r r remotequeueszSocketIO.remotequeuecCs`d||||ff}|}t|jkr8t}||j|<|d||||||||f|S)NrUz asynccall:%d:newseqr0r1r;Z ConditionrArJ putmessager"rKrgrGrhr8recvarr r r rns zSocketIO.asynccallcCs`d||||ff}|}t|jkr8t}||j|<|d||||||||f|S)NrVzasyncqueue:%d:rtrwr r r rrs zSocketIO.asyncqueuecCs6|d||j|dd}|d||||S)Nz#asyncreturn:%d:call getresponse(): 皙?)waitzasyncreturn:%d:response: )rJ getresponsedecoderesponse)r"reresponser r r roszSocketIO.asyncreturncCs|\}}|dkr|S|dkr dS|dkr6|ddS|dkrT|d|dS|dkrp|d|t||d kr|d ||t||dS) NrSrWrYzdecoderesponse: EXCEPTIONEOFzdecoderesponse: EOFrQzdecoderesponse: Internal ERROR:rXzdecoderesponse: Call Exception:)rJdecode_interrupthook RuntimeError SystemError)r"r}rfwhatr r r r|s&    zSocketIO.decoderesponsecCstdS)N)EOFErrorr$r r r rszSocketIO.decode_interrupthookcCs8z|jdddWn tk r2|dYdSXdS)zListen on socket until I/O not ready or EOF pollresponse() will loop looking for seq number None, which never comes, and exit on EOFError. Nry)myseqrzzmainloop:return)r{rrJr$r r r mainloops  zSocketIO.mainloopcCs6|||}|dk r2|\}}|dkr2|||f}|S)NrS) _getresponse_proxify)r"rrzr}rfrr r r r{"s  zSocketIO.getresponsecCs4t|trt||jSt|tr0tt|j|S|Sr)r RemoteProxyRPCProxyrKlistmapr)r"rr r r r*s    zSocketIO._proxifycCs|d|t|jkr6|||}|dk r|Sqn^|j|}|||jkr\|qH|j|}|d||f|j|=|j|=| |SdS)Nz_getresponse:myseq:z-_getresponse:%s: thread woke up: response: %s) rJr0r1r; pollresponserAacquirer@rzrelease)r"rrzr}rxr r r r2s"      zSocketIO._getresponsecCs|jd|_}|S)N)nextseq)r"rer r r ruHszSocketIO.newseqc Cs|d|dz t|}Wn,tjk rJtdt|tjdYnXt dt ||}t |dkrz0t g|j gg\}}}|j |dt}Wn ttfk rtdYnX||d}q`dS)Nzputmessage:%d:rzCannot pickle:r* .   I r:c@s eZdZdS)r_N)rrrr r r r r_sr_cCst|}|t|<t|Sr)idr>r)rrKr r r r`sr`c@seZdZddZdS)rcCs ||_dSr)rKrOr r r r!szRemoteProxy.__init__N)rrrr!r r r r rsrc@s,eZdZdZdZddZddZddZd S) rFz#ScCs(||_t||tj||||dSr)Zcurrent_handlerr:r!rBaseRequestHandler)r"r=r#Zsvrr r r r!s zRPCHandler.__init__cCs |dS)z(handle() method required by socketserverN)rr$r r r handleszRPCHandler.handlecCs t||SrrrOr r r get_remote_proxyszRPCHandler.get_remote_proxyN)rrrr<rEr!rrr r r r rs rc@s:eZdZdZdZdZejejfddZ ddZ dd Z d S) RPCClientFz#CrcCs*t|||_|j||jddSNr)r&listening_sockZbindZlisten)r"addressZfamilytyper r r r!s zRPCClient.__init__cCsV|j\}}|jr$td|tjd|dtkr>t||ntd|tjdt dS)Nz****** Connection request from r*rz** Invalid host: ) racceptr<r/r-r. LOCALHOSTr:r!rc)r"Z working_sockrr r r rs zRPCClient.acceptcCs t||SrrrOr r r r"szRPCClient.get_remote_proxyN) rrrr<rErr&ZAF_INETZ SOCK_STREAMr!rrr r r r r s  rc@s4eZdZdZdZddZddZddZdd ZdS) rNcCs||_||_dSr)sockiorK)r"rrKr r r r!+szRPCProxy.__init__cCsp|jdkr||j|r.t|j|j|S|jdkr@|||jkrd|j|jd|fi}|St |dS)N__getattribute__) _RPCProxy__methods_RPCProxy__getmethodsr MethodProxyrrK_RPCProxy__attributes_RPCProxy__getattributesrqr)r"r2valuer r r __getattr__/s     zRPCProxy.__getattr__cCs|j|jddi|_dS)NrTr )rrqrKrr$r r r Z__getattributes=s  zRPCProxy.__getattributescCs|j|jddi|_dS)NrRr )rrqrKrr$r r r Z __getmethodsAs  zRPCProxy.__getmethods) rrrrrr!rrrr r r r r&s rcCsLt|D]}t||}t|rd||<qt|trH|jD]}t||q8dSr)dirr^callablerr __bases__r[)rrir2attrsuperr r r r[Es     r[cCs,t|D]}t||}t|sd||<qdSr)rr^r)rrjr2rr r r r\Ps  r\c@seZdZddZddZdS)rcCs||_||_||_dSr)rrKr2)r"rrKr2r r r r!YszMethodProxy.__init__cOs|j|j|j||}|Sr)rrqrKr2)r"rGrhrr r r __call__^szMethodProxy.__call__N)rrrr!rr r r r rWsrcCs~|dkr dSdt_t|}ztj|Wn<tk rfd}||d}||d}tj|YnXtjd|t_dS)z9Override standard display hook to use non-locale encodingNasciibackslashreplacestrict ) builtins_r3r-stdoutwriteUnicodeEncodeErrorencodedecode)rtextencodingbytesr r r displayhookfs   r__main__)mainzidlelib.idle_test.test_rpcr) verbosity)N),__doc__rrrrr6rrrr&rrr-r0r4rr rr ZPicklerrrrr rr>ZQueuerarr:r_r`rrrrrr[r\rrrZunittestrr r r r sR 2  k