3  fR@sdZddlZddlZddlZddlZddlZddlZddlZddlZddl Z ddl Z ddl Z ddl Z ddl Z ddlZddlZddZddZd+ddZGd d d ejZd,Zd ZGddde jZiZejdZejdZGdddeZGdddeZddZ GdddeZ!Gddde j"eZ#GdddeZ$GdddeZ%ddZ&d d!Z'Gd"d#d#eZ(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. NcCstj|}t|tjst|S)z*Return code object from marshal string ms.)marshalloads isinstancetypesCodeTypeAssertionError)mscor 0/opt/alt/python36/lib64/python3.6/idlelib/rpc.py unpickle_code-s r cCs$t|tjsttj|}t|ffS)zBReturn unpickle function and tuple with marshalled co code object.)rrrrrdumpsr )r rr r r pickle_code3s rcCs$tj}t||}|j||jS)z.Return pickled (or marshalled) string for obj.)ioBytesIO CodePicklerdumpgetvalue)objZprotocolfpr r r r 9s  r c@s"eZdZejeiZejejdS)rN) __name__ __module__ __qualname__rrrdispatch_tableupdatecopyregr r r r rBs riz 127.0.0.1c@s6eZdZd ddZddZddZdd Zd d ZdS) RPCServerNcCs |dkr t}tjj|||dS)N) RPCHandler socketserver TCPServer__init__)selfaddrZ handlerclassr r r r"LszRPCServer.__init__cCsdS)z@Override TCPServer method, no bind() phase for connecting entityNr )r#r r r server_bindQszRPCServer.server_bindcCs|jj|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_address)r#r r r server_activateUszRPCServer.server_activatecCs |j|jfS)z:Override TCPServer method, return already connected socket)r&r')r#r r r get_request^szRPCServer.get_requestc CsyWntk rYntj}tddd|dtd|dtdtjj|dtd||dtdt||dtj |dtd |dtdd|dt j 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.  -()filezUnhandled server exception!z Thread: %szClient Address: z Request: z# *** Unrecoverable, server exiting!rN) SystemExitsys __stderr__print threadingcurrent_threadnamerepr traceback print_excos_exit)r#requestZclient_addressZerfr r r handle_errorbs   zRPCServer.handle_error)N)rrrr"r%r(r)r;r r r r rJs   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@tj|_|dk r||_||_|dkr*t}||_i|_i|_dS)N) r2r3 sockthread debuggingsock objecttableobjtable responsescvars)r#r?rAr>r r r r"s zSocketIO.__init__cCs |j}d|_|dk r|jdS)N)r?close)r#r?r r r rDszSocketIO.closecCstjddS)z!override for specific exit actionrN)r8r9)r#r r r exithookszSocketIO.exithookcGsR|js dS|jdttjj}x|D]}|dt|}q(Wt|tjddS)N )r-) r>locationstrr2r3r4r1r/r0)r#argssar r r debugs  zSocketIO.debugcCs||j|<dS)N)rA)r#oidobjectr r r registerszSocketIO.registerc Cs&y |j|=Wntk r YnXdS)N)rAKeyError)r#rMr r r unregisters zSocketIO.unregistercCs|jd|y|\}\}}}}Wntk r4dSX||jkrNdd|ffS|j|}|dkrvi} t|| d| fS|dkri} t|| d| fSt||sdd|ffSt||} y^|d kr| ||} t| trt | } d| fS|d krt j || ||ffdSdd |fSWnt k r,Ynt k rBYnntk rXYnXtk r|} zd | fSd} ~ Xn4d}t||| |ftjdtjtjddSdS)Nz localcall:ERRORBad request formatzUnknown object id: %r __methods__OK__attributes__zUnsupported method name: %rCALLQUEUEQUEUEDzUnsupported message type: %sCALLEXCzU*** Internal Error: rpc.py:SocketIO.localcall() Object: %s Method: %s Args: %s )r- EXCEPTION)rRrS)rYN)r[N)rL TypeErrorrA _getmethods_getattributeshasattrgetattrr RemoteObject remoteref request_queueZputr.KeyboardInterruptOSError Exceptionr1r/r0r6r7)r#seqr:howrM methodnamerIkwargsrmethods attributesmethodZretZexmsgr r r localcallsT          zSocketIO.localcallcCs(|jd|||j||||}|j|S)Nzremotecall:asynccall: )rL asynccall asyncreturn)r#rMrirIrjrgr r r remotecallszSocketIO.remotecallcCs(|jd|||j||||}|j|S)Nzremotequeue:asyncqueue: )rL asyncqueuerq)r#rMrirIrjrgr r r remotequeueszSocketIO.remotequeuecCs`d||||ff}|j}tj|jkr8tj}||j|<|jd||||||j||f|S)NrWz asynccall:%d:)newseqr2r3r= ConditionrCrL putmessage)r#rMrirIrjr:rgcvarr r r rps zSocketIO.asynccallcCs`d||||ff}|j}tj|jkr8tj}||j|<|jd||||||j||f|S)NrXzasyncqueue:%d:)rur2r3r=rvrCrLrw)r#rMrirIrjr:rgrxr r r rss zSocketIO.asyncqueuecCs6|jd||j|dd}|jd|||j|S)Nz#asyncreturn:%d:call getresponse(): g?)waitzasyncreturn:%d:response: )rL getresponsedecoderesponse)r#rgresponser r r rqszSocketIO.asyncreturncCs|\}}|dkr|S|dkr dS|dkr6|jddS|dkrT|jd|jdS|dkrp|jd|t||d kr|jd ||t||dS) NrUrYr[zdecoderesponse: EXCEPTIONEOFzdecoderesponse: EOFrRzdecoderesponse: Internal ERROR:rZzdecoderesponse: Call Exception:)rLdecode_interrupthook RuntimeError SystemError)r#r|rhwhatr r r r{s&    zSocketIO.decoderesponsecCstdS)N)EOFError)r#r r r r~szSocketIO.decode_interrupthookc Cs6y|jdddWntk r0|jddSXdS)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. Ng?)myseqryzmainloop:return)rzrrL)r#r r r mainloops  zSocketIO.mainloopcCs6|j||}|dk r2|\}}|dkr2||j|f}|S)NrU) _getresponse_proxify)r#rryr|rhrr r r rz#s  zSocketIO.getresponsecCs4t|trt||jSt|tr0tt|j|S|S)N)r RemoteProxyRPCProxyrMlistmapr)r#rr r r r+s    zSocketIO._proxifycCs|jd|tj|jkr:x|j||}|dk r|SqWnb|j|}|jx||jkrb|jqNW|j|}|jd||f|j|=|j|=|j |SdS)Nz_getresponse:myseq:z-_getresponse:%s: thread woke up: response: %s) rLr2r3r= pollresponserCacquirerBryrelease)r#rryr|rxr r r r3s"        zSocketIO._getresponsecCs|jd|_}|S)N)nextseq)r#rgr r r ruIszSocketIO.newseqcCs|jd|dy t|}Wn,tjk rJtdt|tjdYnXtj dt ||}xnt |dkry0t j g|j gg\}}}|j j |dt}Wn ttfk rtdYnX||d}qbWdS)Nzputmessage:%d:rzCannot pickle:)r-z .   I r<c@s eZdZdS)raN)rrrr r r r rasracCst|}|t|<t|S)N)idr@r)rrMr r r rbsrbc@seZdZddZdS)rcCs ||_dS)N)rM)r#rMr r r r"szRemoteProxy.__init__N)rrrr"r r r r rsrc@s,eZdZdZdZddZddZddZd S) rFz#ScCs(||_tj||tjj||||dS)N)Zcurrent_handlerr<r"r BaseRequestHandler)r#r?r$Zsvrr r r r"s zRPCHandler.__init__cCs |jdS)z(handle() method required by socketserverN)r)r#r r r handleszRPCHandler.handlecCs t||S)N)r)r#rMr r r get_remote_proxy szRPCHandler.get_remote_proxyN)rrrr>rGr"rrr r r r rs rc@s:eZdZdZdZdZejejfddZ ddZ dd Z d S) RPCClientFz#CrcCs*tj|||_|jj||jjddS)Nr)r&listening_sockZbindZlisten)r#addressZfamilytyper r r r"s zRPCClient.__init__cCsV|jj\}}|jr$td|tjd|dtkr>tj||ntd|tjdt dS)Nz****** Connection request from )r-rz** Invalid host: ) racceptr>r1r/r0 LOCALHOSTr<r"re)r#Z working_sockrr r r rs zRPCClient.acceptcCs t||S)N)r)r#rMr r r r#szRPCClient.get_remote_proxyN) rrrr>rGrr&ZAF_INETZ SOCK_STREAMr"rrr r r r r s  rc@s4eZdZdZdZddZddZddZdd ZdS) rNcCs||_||_dS)N)sockiorM)r#rrMr r r r",szRPCProxy.__init__cCsp|jdkr|j|jj|r.t|j|j|S|jdkr@|j||jkrd|jj|jd|fi}|St |dS)N__getattribute__) _RPCProxy__methods_RPCProxy__getmethodsr MethodProxyrrM_RPCProxy__attributes_RPCProxy__getattributesrrr)r#r4valuer r r __getattr__0s      zRPCProxy.__getattr__cCs|jj|jdfi|_dS)NrV)rrrrMr)r#r r r Z__getattributes>s zRPCProxy.__getattributescCs|jj|jdfi|_dS)NrT)rrrrMr)r#r r r Z __getmethodsBs zRPCProxy.__getmethods) rrrrrr"rrrr r r r r's rcCsTx*t|D]}t||}t|r d||<q Wt|trPx|jD]}t||q>WdS)Nr)dirr`callablerr __bases__r])rrkr4attrsuperr r r r]Fs    r]cCs0x*t|D]}t||}t|s d||<q WdS)Nr)rr`r)rrlr4rr r r r^Qs r^c@seZdZddZddZdS)rcCs||_||_||_dS)N)rrMr4)r#rrMr4r r r r"ZszMethodProxy.__init__cOs|jj|j|j||}|S)N)rrrrMr4)r#rIrjrr r r __call___szMethodProxy.__call__N)rrrr"rr r r r rXsrc Cs~|dkr dSdt_t|}ytjj|Wn<tk rfd}|j|d}|j|d}tjj|YnXtjjd|t_dS)z9Override standard display hook to use non-locale encodingNasciibackslashreplacestrictr*) builtins_r5r/stdoutwriteUnicodeEncodeErrorencodedecode)rtextencodingbytesr r r displayhookgs   r__main__)mainzidlelib.idle_test.test_rpcr) verbosity)Ni )-__doc__rrrrr8rrrr&r rr/r2r6rr rr ZPicklerrrrr!rr@ZQueuercrrNr<rarbrrrrrr]r^rrrZunittestrr r r r sR 2  k