
    Mh>                       d Z ddlmZ ddlZddlZddlZddlZddlZddlZddl	Z	ddl
Z
ddlmZ ddlmZ ddlmZmZmZ erddlmZ 	 ddlmZ d'dZn"# e$ r 	 ddlmZ n# e$ r eZY nw xY wdZY nw xY wddlmZ dZdZ G d d          Z d(dZ!d)d*dZ" G d d          Z# G d d e#          Z$ G d! d"e%          Z& G d# d$e#          Z' G d% d&e#          Z(dS )+zMTerminal management for exposing terminals to a web interface using Tornado.
    )annotationsN)deque)futures)TYPE_CHECKINGAny	Coroutine)
TermSocket)PtyProcessUnicodereturnNonec                 X    t          j         t           j        t           j                   dS )z-A prexec function to set up a signal handler.N)signalSIGPIPESIG_DFL     T/var/www/html/test/jupyter/venv/lib/python3.11/site-packages/terminado/management.py
preexec_fnr      s    fnfn55555r   )
PtyProcess)IOLoopPYXTERM_zxterm-256colorc                  f    e Zd ZU dZded<   ddd
ZddZej        fddZ	ej        fddZ
dddZdS )PtyWithClientsz%A pty object with associated clients.
str | None	term_nameNargvr   envdict[str, str] | Nonecwdc                    g | _         t          g d          | _        ||pg |d}t          
t          |d<   t	          j        d	i || _         t          j        d          d          | j        _	        dS )
zInitialize the pty.  )maxlen)r   r   r   Nr   zutf-8replace)errorsr   )
clientsr   read_bufferr   r
   spawnptyproccodecsgetincrementaldecoderdecoder)selfr   r   r   kwargss        r   __init__zPtyWithClients.__init__2   s    "$',R'='='=syb==!#-F< (.8888  Ev;GDDIVVVr   r   r   c                    dx}}| j         D ] }|j        \  }}|||k     r|}|||k     r|}!|dk    s|dk    rdS | j                                        \  }}||f||fk    r| j                            ||           dS dS )a  Set the terminal size to that of the smallest client dimensions.

        A terminal not using the full space available is much nicer than a
        terminal trying to use more than the available space, so we keep it
        sized to the smallest client.
        i'  N)r%   sizer(   
getwinsize
setwinsize)r,   minrowsmincolsclientrowscolss         r   resize_to_smallestz!PtyWithClients.resize_to_smallest@   s     "!'l 	 	FJD$D7NND7NNew%//F\,,..
d$<GW---L##GW55555 .-r   sigintc                :    | j                             |           dS )z'Send a signal to the process in the ptyN)r(   kill)r,   r9   s     r   r<   zPtyWithClients.killV   s    #r   c                    t           j        dk    r| j                            |          S t          j        | j        j                  }t          j        ||           dS )z<Send a signal to the process group of the process in the ptyntN)osnamer(   r<   getpgidpidkillpg)r,   r9   pgids      r   rC   zPtyWithClients.killpgZ   sN    7d??<$$S)))z$,*++
	$tr   Fforceboolc                   K   t           j        dk    rt          j        t          j        g}n.t          j        t          j        t          j        t          j        g}t          j                    }d fd} j	        
                                sdS 	 |D ]C}                     |            |             d{V   j	        
                                s dS D|rV                     t          j                    |             d{V  t           j	        
                                           S dS # t          $ r:  |             d{V  t           j	        
                                           cY S w xY w)	zThis forces a child process to terminate. It starts nicely with
        SIGHUP and SIGINT. If "force" is True then moves onto SIGKILL. This
        returns True if the child was terminated. This returns False if the
        child could not be terminated.r>   r   Coroutine[Any, Any, None]c                 @    t          j         j        j                  S )z/Sleep to allow the terminal to exit gracefully.)asynciosleepr(   delayafterterminater,   s   r   rK   z'PtyWithClients.terminate.<locals>.sleepn   s    =!ABBBr   TNF)r   rH   )r?   r@   r   SIGINTSIGTERMSIGHUPSIGCONTr   currentr(   isaliver<   SIGKILLrF   OSError)r,   rE   signals_rK   r9   s   `     r   	terminatezPtyWithClients.terminateb   s     
 7d??}fn5GG}fnfmV^TGN	C 	C 	C 	C 	C 	C |##%% 	4	4    		#egg|++--  44  8		&.)))egg 4 4 6 667775 	4 	4 	4
 %''MMMMMMMDL0022233333	4s   AD0 AD0 0AE43E4)NN)r   r   r   r   r   r   r   r   )r9   r:   r   r   )r9   r:   r   r   F)rE   rF   r   rF   )__name__
__module____qualname____doc____annotations__r.   r8   r   rO   r<   rC   rX   r   r   r   r   r   -   s         //W W W W W6 6 6 6, %n      !'     #4 #4 #4 #4 #4 #4 #4r   r   targetr   changesc                v    |                                 D ]#\  }}||                     |d           || |<   $dS )z<Like dict.update(), but remove keys where the value is None.N)itemspop)r`   ra   kvs       r   _update_removingrg      sN      19JJq$F1II	 r   皙?fdr:   timeoutfloatlist[tuple[int, int]]c                B   t           j        dk    rrt          j                    }|                    | t          j        t          j        z  t          j        z  t          j        z             |                    |dz            S t          j        | gg g |          \  }}}|S )zIPoll using poll() on posix systems and select() elsewhere (e.g., Windows)posixr!   )	r?   r@   selectpollregisterPOLLINPOLLPRIPOLLHUPPOLLERR)ri   rj   pollerrrW   s        r   _pollrx      s    	w'.?&.P	
 	
 	
 {{7T>***mRD"b'22GAq!Hr   c                      e Zd ZdZ	 	 	 	 	 d)d*dZ	 	 	 	 d+d,dZd-dZd.dZd.dZd/d0d!Z	d.d"Z
d/d1d$Zd2d&Zd3d'Zd3d(ZdS )4TermManagerBasez"Base class for a terminal manager. Nshell_commandstr
server_urlterm_settingsr   	extra_envioloopblocking_io_executorc                L   || _         || _        |pi | _        || _        t	          j        t                    | _        i | _        |"d| _	        t          j        d          | _        nd| _	        || _        |,t          j        d| j        j         dt           d	           dS dS )
Initialize the manager.NF   )max_workersTzSetting z!.ioloop is deprecated and ignored   )
stacklevel)r|   r~   r   r   logging	getLoggerr[   log
ptys_by_fd!_blocking_io_executor_is_externalr   ThreadPoolExecutorr   warningswarn	__class__DeprecationWarning)r,   r|   r~   r   r   r   r   s          r   r.   zTermManagerBase.__init__   s     +$*0b"$X..57'5:D2(/(Bq(Q(Q(QD%%59D2(<D%MU4>2UUU"      r      P   r   heightr:   width	winheightwinwidthr-   r   dict[str, str]c                   t           j                                        }| j                            dt
                    |d<   d||fz  }|r|r
|d||fz  z  }||t          dz   <   t          |          |d<   t          |          |d<   | j        r| j        |t          dz   <   | j	        rt          || j	                   |                    d	i           }|r%t          |t                    rt          ||           |S )
z@Build the environment variables for the process in the terminal.typeTERMz%dx%dz;%dx%d
DIMENSIONSCOLUMNSLINESURLr   )r?   environcopyr   getDEFAULT_TERM_TYPE
ENV_PREFIXr}   r~   r   rg   
isinstancedict)	r,   r   r   r   r   r-   r   
dimensionsterm_envs	            r   make_term_envzTermManagerBase.make_term_env   s
    joo (,,V5FGGFv.
 	;	 	;(h	%:::J)3J%&UI6{{G? 	6&*oC
U"#> 	2S$.111::k2.. 	,
8T22 	,S(+++
r   r   c                    | j                                         }| j        |d<   |                    |           |d         } | j        di |}|                    dd          }t          |||          S )z?Make a new terminal, return a :class:`PtyWithClients` instance.r|   r   Nr   )r   r   r|   updater   r   r   )r,   r-   optionsr   r   r   s         r   new_terminalzTermManagerBase.new_terminal   s|    $))++#'#5 v' d ++7++kk%&&dC---r   ptywclientsr   c                    |j         j        }|| j        |<   t          j                    }|                    || j        |j                   dS )zBConnect a terminal to the tornado event loop to read data from it.N)r(   ri   r   r   rR   add_handlerpty_readREAD)r,   r   ri   loops       r   start_readingzTermManagerBase.start_reading   sH     #)~T]DI66666r   c                    |j         j        }| j                            d|           | j        |= t          j                                        |           |j                                          dS )zCalled when the pty has closed.zEOF on FD %d; stopping readingN)	r(   ri   r   infor   r   rR   remove_handlerclose)r,   r   ri   s      r   on_eofzTermManagerBase.on_eof   si      #6;;;OB''+++ 	!!#####r   ri   eventsc                   t          |d          s| j                            d|           dS | j        |         }	 |                     |           |j                            d          }|j                            |           |j	        D ]}|
                    |           dS # t          $ r7 |                     |           |j	        D ]}|                                 Y dS w xY w)z>Called by the event loop when there is pty data ready to read.rh   )rj   zSpurious pty_read() on fd %sNi   )rx   r   debugr   pre_pty_read_hookr(   readr&   appendr%   on_pty_readEOFErrorr   on_pty_died)r,   ri   r   r   sr5   s         r   r   zTermManagerBase.pty_read   s    R%%% 	HNN92>>>Fob)		%"";///#((//A#**1---%- & &""1%%%%& & 	% 	% 	%KK$$$%- % %""$$$$% % %	%s   A(B' '=C('C(c                    dS )zQHook before pty read, subclass can patch something into ptywclients when pty_readNr   )r,   r   s     r   r   z!TermManagerBase.pre_pty_read_hook        r   url_componentc                    t           )a7  Override in a subclass to give a terminal to a new websocket connection

        The :class:`TermSocket` handler works with zero or one URL components
        (capturing groups in the URL spec regex). If it receives one, it is
        passed as the ``url_component`` parameter; otherwise, this is None.
        )NotImplementedErrorr,   r   s     r   get_terminalzTermManagerBase.get_terminal  s
     "!r   	websocketc                    dS )z=Override this to e.g. kill terminals on client disconnection.Nr   r,   r   s     r   client_disconnectedz#TermManagerBase.client_disconnected  r   r   c                   K   |                                   d{V  | j        s| j                            dd           dS dS )zShutdown the manager.NFT)waitcancel_futures)kill_allr   r   shutdownrM   s    r   r   zTermManagerBase.shutdown  s`      mmoo5 	P%..E$.OOOOO	P 	Pr   c                   K   g }| j                                         D ]&} |j        |                    d                     '|rt	          j        |  d{V  dS dS )Kill all terminals.TrE   N)r   valuesr   rX   rJ   gather)r,   r   terms      r   r   zTermManagerBase.kill_all$  s      O**,, 	7 	7DGN4>>>556666 	+.'**********	+ 	+r   )r{   NNNN)r|   r}   r~   r}   r   r   r   r   r   r   r   r   )r   r   r   r   )r   r:   r   r:   r   r:   r   r:   r-   r   r   r   )r-   r   r   r   r   r   r   r   N)ri   r:   r   r   r   r   r   r   r   r   )r   r   r   r   rY   )r[   r\   r]   r^   r.   r   r   r   r   r   r   r   r   r   r   r   r   r   rz   rz      s7       ,,
 !$(    D     @. . . .7 7 7 7	$ 	$ 	$ 	$% % % % %$` ` ` `" " " " "L L L LP P P P+ + + + + +r   rz   c                  :     e Zd ZdZd fdZddd
Zd fdZ xZS )SingleTermManagerz9All connections to the websocket share a common terminal.r-   r   r   r   c                H     t                      j        di | d| _        dS r   Nr   )superr.   terminal)r,   r-   r   s     r   r.   zSingleTermManager.__init__1  s+    ""6"""/3r   Nr   r   c                    | j         3|                                 | _         |                     | j                    | j         S )z "Get the singleton terminal.)r   r   r   r   s     r   r   zSingleTermManager.get_terminal6  s;    =  --//DMt}---}r   c                f   K   t                                                       d{V  d| _        dS )zKill the singletone terminal.N)r   r   r   r,   r   s    r   r   zSingleTermManager.kill_all=  s:      gg         r   )r-   r   r   r   r   r   rY   )r[   r\   r]   r^   r.   r   r   __classcell__r   s   @r   r   r   .  sz        CC4 4 4 4 4 4
             r   r   c                  "    e Zd ZdZd
dZddZd	S )MaxTerminalsReachedz;An error raised when we exceed the max number of terminals.max_terminalsr:   r   r   c                    || _         dS )zInitialize the error.Nr   )r,   r   s     r   r.   zMaxTerminalsReached.__init__F  s    *r   r}   c                    d| j         z  S )z'The string representation of the error.z$Cannot create more than %d terminalsr   rM   s    r   __str__zMaxTerminalsReached.__str__J  s    58JJJr   N)r   r:   r   r   )r   r}   )r[   r\   r]   r^   r.   r   r   r   r   r   r   C  sH        EE+ + + +K K K K K Kr   r   c                  8     e Zd ZdZdd fd	ZdddZddZ xZS )UniqueTermManagerz-Give each websocket a unique terminal to use.Nr   
int | Noner-   r   r   r   c                H     t                      j        di | || _        dS r   )r   r.   r   r,   r   r-   r   s      r   r.   zUniqueTermManager.__init__R  s.    ""6"""*r   r   r   c                    | j         r1t          | j                  | j         k    rt          | j                   |                                 }|                     |           |S )z Get a terminal from the manager.)r   lenr   r   r   r   )r,   r   r   s      r   r   zUniqueTermManager.get_terminalW  s_     	:#do"6"6$:L"L"L%d&8999  ""4   r   r   r	   c                2   | j                             d           |j        rut          j        dk    r?|j                                         |                     |j        j        j                   dS |j        	                    t          j                   dS dS )z-Send terminal SIGHUP when client disconnects.z-Websocket closed, sending SIGHUP to terminal.r>   N)r   r   r   r?   r@   r<   r   r(   ri   rC   r   rP   r   s     r   r   z%UniqueTermManager.client_disconnected`  s    EFFF 	5w$"''))) i08;<<<%%fm44444	5 	5r   r   )r   r   r-   r   r   r   r   )r   r	   r   r   )r[   r\   r]   r^   r.   r   r   r   r   s   @r   r   r   O  su        77+ + + + + + +
    
5 
5 
5 
5 
5 
5 
5 
5r   r   c                  z     e Zd ZdZdd fdZddZdZd dZd!dZe	j
        fd"dZd#d$dZd% fdZd& fdZ xZS )'NamedTermManagerzBShare terminals between websockets connected to the same endpoint.Nr   r   r-   r   r   c                V     t                      j        di | || _        i | _        dS r   )r   r.   r   	terminalsr   s      r   r.   zNamedTermManager.__init__p  s3    ""6"""*46r   r   r}   r   c                T   |J || j         v r| j         |         S | j        r1t          | j                   | j        k    rt          | j                  | j                            d|           |                                 }||_        || j         |<   |                     |           |S )z!Get or create a terminal by name.Nz$New terminal with specified name: %s)	r   r   r   r   r   r   r   r   r   )r,   r   r   s      r   r   zNamedTermManager.get_terminalv  s    $$$&&>),, 	:#dn"5"59K"K"K%d&8999 	<iHHH  """$(y!4   r   z%dr   c                d    t          j        d          D ]}| j        |z  }|| j        vr|c S d S )Nr   )start)	itertoolscountname_templater   )r,   nr@   s      r   _next_available_namez%NamedTermManager._next_available_name  sK    q))) 	 	A%)D4>)) *tr   tuple[str, PtyWithClients]c                    d|v r|d         n|                                  } | j        di |}| j                            d|           ||_        || j        |<   |                     |           ||fS )z3Create a new named terminal with an automatic name.r@   z$New terminal with automatic name: %sr   )r  r   r   r   r   r   r   )r,   r-   r@   r   s       r   new_named_terminalz#NamedTermManager.new_named_terminal  s    !'6!1!1vf~~t7P7P7R7R t **6**<dCCC#t4   Tzr   r@   r9   r:   c                J    | j         |         }|                    |           dS )zKill a terminal by name.N)r   r<   )r,   r@   r9   r   s       r   r<   zNamedTermManager.kill  s!    ~d#		#r   FrE   rF   c                \   K   | j         |         }|                    |           d{V  dS )zTerminate a terminal by name.r   N)r   rX   )r,   r@   rE   r   s       r   rX   zNamedTermManager.terminate  s>      ~d#nn5n)))))))))))r   r   c                    t                                          |           |j        }| j                            d|           |J | j                            |d           dS )z*Handle end of file for a pty with clients.zTerminal %s closedN)r   r   r   r   r   r   rd   )r,   r   r@   r   s      r   r   zNamedTermManager.on_eof  sb    {###$*D1114&&&&&r   c                f   K   t                                                       d{V  i | _        dS )r   N)r   r   r   r   s    r   r   zNamedTermManager.kill_all  s:      gg         r   r   )r   r   r-   r   r   r   )r   r}   r   r   )r   r   )r-   r   r   r  )r@   r}   r9   r:   r   r   rZ   )r@   r}   rE   rF   r   r   r   rY   )r[   r\   r]   r^   r.   r   r  r  r  r   rO   r<   rX   r   r   r   r   s   @r   r   r   m  s        LL7 7 7 7 7 7 7   $ M       *0     
* * * * *
' ' ' ' ' '         r   r   rY   )r`   r   ra   r   r   r   )rh   )ri   r:   rj   rk   r   rl   ))r^   
__future__r   rJ   r)   r   r   r?   ro   r   r   collectionsr   
concurrentr   typingr   r   r   terminado.websocketr	   
ptyprocessr
   r   ImportErrorwinptyr   objecttornado.ioloopr   r   r   r   rg   rx   rz   r   	Exceptionr   r   r   r   r   r   <module>r     s   
 # " " " " "        				                0 0 0 0 0 0 0 0 0 0 /......,,,,,,6 6 6 6 6    #::::::: # # #"#JJJ " ! ! ! ! !
 % X4 X4 X4 X4 X4 X4 X4 X4v   
 
 
 
 
M+ M+ M+ M+ M+ M+ M+ M+`       *	K 	K 	K 	K 	K) 	K 	K 	K5 5 5 5 5 5 5 5<C C C C C C C C C Cs6   
A A2A A2 A*'A2)A**A21A2