
    =}`e                        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mZmZmZmZmZ ej*                  ej,                  ej.                  ej0                  dZdZdZdZdZd	Zd
ZdZ dZ!dZ"dZ#dZ$dZ%dZ&dZ'dZ(dZ)dZ*dZ+dZ,dZ-dZ.dZ/dZ0dZ1dZ2dZ3dZ4dZ5dZ6d Z7d!Z8d"Z9d#Z:d$Z;d%Z<d&Z=d'Z>d(Z?d)Z@d*ZAd+ZBd,ZCd-ZDd.ZEd/ZFe/e;e0e<e1e=e2e>e3e?e4e@e5eAe6eBe7eCe8eDe9eEe:eFiZGdZHdZIdZJdZKdZLdZMdZNdZOdZPd ZQd!ZRd"ZSd#ZTd0ZUd1ZVd2ZWd3ZXd4ZYd5ZZd6Z[d7Z\d8Z]d9Z^d:Z_d;Z`d<Zad=Zbd>Zcd7Zdd8Zed9Zfd:Zgd;Zhd<Zid=Zjd>ZkdZldZmdZnej                  d>ej                  d=ej                  d?ej                  d<ej                  d@iZt eudA etj                         D              Zwej                  d>ej                  d?ej                  d=iZ{ eudB e{j                         D              Z|d Z}d>Z~d=ZdCZdDZdEZdFZ G dG dHe      Z G dI dJe      Z G dK dLe      Z G dM dNe      ZedOk(  rd dlZ edPdQ      Zej                  j                  dRj                  e             ej                  j                  dS       ej                  dT       ej                          ej                  j                  dUj                  ej!                  d@                   ej#                          yy# e$ r
 d dl	m
Z Y w xY w# e$ r d dlZY w xY w)V    )absolute_importN)
SerialBaseSerialExceptionto_bytes	iterbytesPortNotOpenErrorTimeout)debuginfowarningerror                                                s                ,                     	   
         e   f   g   h   i   j   k   l   m   n   o   p                        @                            c              #   *   K   | ]  \  }}||f  y wN .0kvs      0/usr/lib/python3/dist-packages/serial/rfc2217.py	<genexpr>rQ      s     !PTQ1a&!P   c              #   *   K   | ]  \  }}||f  y wrJ   rK   rL   s      rP   rQ   rQ      s     "RdaAq6"RrR   	REQUESTEDACTIVEINACTIVEREALLY_INACTIVEc                   &    e Zd ZdZ	 ddZd Zd Zy)TelnetOptionz@Manage a single telnet option, keeps track of DO/DONT WILL/WONT.Nc
                     || _         || _        || _        || _        || _        || _        || _        || _        d| _        |	| _	        y)a;          Initialize option.
        :param connection: connection used to transmit answers
        :param name: a readable name for debug outputs
        :param send_yes: what to send when option is to be enabled.
        :param send_no: what to send when option is to be disabled.
        :param ack_yes: what to expect when remote agrees on option.
        :param ack_no: what to expect when remote disagrees on option.
        :param initial_state: options initialized with REQUESTED are tried to
            be enabled on startup. use INACTIVE for all others.
        FN)

connectionnameoptionsend_yessend_noack_yesack_nostateactiveactivation_callback)
selfr[   r\   r]   r^   r_   r`   ra   initial_staterd   s
             rP   __init__zTelnetOption.__init__   sK     %	 "
#6     c                 &    dj                  |       S )zString for debug outputsz{o.name}:{o.active}({o.state}))oformatre   s    rP   __repr__zTelnetOption.__repr__  s    /666>>rh   c                    || j                   k(  r$| j                  t        u r/t        | _        d| _        | j
                  | j                          y| j                  t        u ry| j                  t        u r_t        | _        | j                  j                  | j                  | j                         d| _        | j
                  n| j                          y| j                  t        u r1| j                  j                  | j                  | j                         yt        dj                  |             yy|| j                  k(  r| j                  t        u rt        | _        d| _        y| j                  t        u rCt        | _        | j                  j                  | j                  | j                         d| _        y| j                  t        u ry| j                  t        u ryt        dj                  |             y)zo        A DO/DONT/WILL/WONT was received for this option, update state and
        answer when needed.
        TNzoption in illegal state {!r}F)r`   rb   rT   rU   rc   rd   rV   r[   telnet_send_optionr^   r]   rW   r_   
ValueErrorrl   ra   re   commands     rP   process_incomingzTelnetOption.process_incoming  s   
 dll"zzY&#
"++7,,.v%x'#
224==$++N"++7,,..224<<M !?!F!Ft!LMM 8 8 #zzY&%
#v%%
224<<M#x'. !?!F!Ft!LMM $rh   rJ   )__name__
__module____qualname____doc__rg   rn   rt   rK   rh   rP   rY   rY      s    J =A70?$Nrh   rY   c                   H    e Zd ZdZd	dZd Zd Zd Z ee      Z	d
dZ
d Zy)TelnetSubnegotiationz    A object to handle subnegotiation of options. In this case actually
    sub-sub options for RFC 2217. It is used to track com port options.
    Nc                 h    ||}|| _         || _        || _        d | _        || _        t
        | _        y rJ   )r[   r\   r]   value
ack_optionrV   rb   )re   r[   r\   r]   r}   s        rP   rg   zTelnetSubnegotiation.__init__9  s7    J$	
$
rh   c                 &    dj                  |       S )zString for debug outputs.z{sn.name}:{sn.state})snrk   rm   s    rP   rn   zTelnetSubnegotiation.__repr__C  s    %,,,55rh   c                 H   || _         t        | _        | j                  j	                  | j
                  | j                          | j                  j                  rJ| j                  j                  j                  dj                  | j                  | j                                yy)z        Request a change of the value. a request is sent to the server. if
        the client needs to know if the change is performed he has to check the
        state of this object.
        zSB Requesting {} -> {!r}N)
r|   rT   rb   r[   rfc2217_send_subnegotiationr]   loggerr
   rl   r\   )re   r|   s     rP   setzTelnetSubnegotiation.setG  sq     

33DKKL??!!OO""(()C)J)J499VZV`V`)ab "rh   c                     | j                   t        k(  r$t        dj                  | j                              | j                   t
        k(  S )z{        Check if answer from server has been received. when server rejects
        the change, raise a ValueError.
        z%remote rejected value for option {!r})rb   rW   rq   rl   r\   rU   rm   s    rP   is_readyzTelnetSubnegotiation.is_readyS  s:    
 ::(DKKDIIVWWzzV##rh   c                     t        |      }|j                         s7t        j                  d       | j	                         ry|j                         s7t        dj                  | j                              )z        Wait until the subnegotiation has been acknowledged or timeout. It
        can also throw a value error when the answer from the server does not
        match the value sent.
        皙?z%timeout while waiting for option {!r}N)r	   expiredtimesleepr   r   rl   r\   )re   timeouttimeout_timers      rP   waitzTelnetSubnegotiation.wait^  s`      ('')JJt}}  '')
 ""I"P"PQUQZQZ"[\\rh   c                 >   | j                   |dt        | j                          k(  rt        | _        nt        | _        | j
                  j                  rK| j
                  j                  j                  dj                  | j                  || j                               yy)z        Check an incoming subnegotiation block. The parameter already has
        cut off the header like sub option number and com port option value.
        NzSB Answer {} -> {!r} -> {})
r|   lenrU   rb   rW   r[   r   r
   rl   r\   )re   	suboptions     rP   check_answerz!TelnetSubnegotiation.check_answerl  sv    
 ::#3C

O44DJ )DJ??!!OO""(()E)L)LTYYXacgcmcm)no "rh   rJ   )rG   )ru   rv   rw   rx   rg   rn   r   r   propertyrc   r   r   rK   rh   rP   rz   rz   3  s4    
6
c$ hF]prh   rz   c                   2    e Zd ZdZdZ fdZd Zd Zd Zd Z	e
d        Zd!d	Zd
 Zd Zd Zd Zd Zd Ze
d        Ze
d        Ze
d        Ze
d        Ze
d        Zej0                  d        Zd Zd Zd Zd Zd Zd Zd"dZd Z d Z!d Z"d  Z# xZ$S )#Serialz<Serial port implementation for RFC 2217 remote serial ports.)2   K   n            i,  iX  i  i  i`	  i  i%  i K  i   i     c                    d | _         d | _        d| _        d | _        t	        d      | _        d| _        d | _        d | _        d| _	        d| _
        d| _        d | _        d | _        d | _        d | _        t!        t"        | J  |i | y )Nr   FrG   )_thread_socket
_linestate_modemstater	   _modemstate_timeout_remote_suspend_flow_write_lockr   _ignore_set_control_answer_poll_modem_state_network_timeout_telnet_options_rfc2217_port_settings_rfc2217_options_read_buffersuperr   rg   )re   argskwargs	__class__s      rP   rg   zSerial.__init__  s    #*2; $)!*/'!& !#&*# $ fd$d5f5rh   c                 
   d| _         d| _        d| _        d| _        | j                  t        d      | j                  rt        d      	 t        j                  | j                  | j                        d      | _        | j                  j                  t        j                  t        j                  d       t#        j"                         | _        t'        j(                         | _        t-        | d
t.        t0        t2        t4        t6        t8              t-        | dt:        t0        t2        t4        t6        t<              g}t-        | dt>        t4        t6        t0        t2        t<              t-        | dt@        t0        t2        t4        t6        t<              t-        | dt@        t4        t6        t0        t2        t<              t-        | dt.        t4        t6        t0        t2        t8              t-        | dt:        t4        t6        t0        t2        t<              g|z   | _!        tE        | dtF        tH              tE        | dtJ        tL              tE        | dtN        tP              tE        | dtR        tT              d| _+        tE        | dtX        tZ              tE        | dt\        t^              d| _0        | j`                  jc                  | jV                         d| _2        d| _3        ti        d      | _5        d| _6        d| _        t'        jn                  | jp                        | _9        | jr                  ju                  d       | jr                  jw                  dj!                  | j                               | jr                  jy                          	 | jB                  D ];  }|jz                  t<        u s| j}                  |j~                  |j                         = ti        | j                        }|j                         sLt        j                  d       t        d |D              t        d  |D              k(  rn+|j                         sLt        d!j!                  |            | j                   r4| j                   j                  d"j!                  | jB                               | j                          | j                  s| j                          | j                  s| j                          | j                          | j                          y# t        $ r1}d| _        t        d	j!                  | j                  |            d}~ww xY w#  | j                           xY w)#zx        Open port with current settings. This may throw a SerialException
        if the port cannot be opened.
        NFrG   z.Port must be configured before it can be used.zPort is already open.rH   )r   rF   zCould not open port {}: {}	we-BINARY
we-RFC2217ECHOwe-SGAthey-SGAthey-BINARYthey-RFC2217baudratedatasizeparitystopsize)r   r   r   r   purgecontrol)r   r   r   r   T)targetz&pySerial RFC 2217 reader thread for {}r   c              3   4   K   | ]  }|j                     y wrJ   rc   rM   rj   s     rP   rQ   zSerial.open.<locals>.<genexpr>  s     ;Aqxx;   c              3   B   K   | ]  }|j                   t        k7    y wrJ   )rb   rV   r   s     rP   rQ   zSerial.open.<locals>.<genexpr>  s     Br[\177hCVBrs   z;Remote does not seem to support RFC2217 or BINARY mode {!r}zNegotiated options: {})Nr   r   r   r   _portr   is_opensocketcreate_connectionfrom_urlportstrr   
setsockoptIPPROTO_TCPTCP_NODELAY	Exceptionrl   Queuer   	threadingLockr   rY   BINARYWILLWONTDODONTrV   COM_PORT_OPTIONrT   r   SGAr   rz   SET_BAUDRATESERVER_SET_BAUDRATESET_DATASIZESERVER_SET_DATASIZE
SET_PARITYSERVER_SET_PARITYSET_STOPSIZESERVER_SET_STOPSIZEr   
PURGE_DATASERVER_PURGE_DATASET_CONTROLSERVER_SET_CONTROLr   updater   r   r	   r   r   Thread_telnet_read_loopr   	setDaemonsetNamestartrb   rp   r^   r]   r   r   r   sumr   _reconfigure_port_dsrdtr_update_dtr_state_rtscts_update_rts_statereset_input_bufferreset_output_bufferclose)re   msgmandadory_optionsr]   r   s        rP   openzSerial.open  s   
 */'!& !::!"RSS<<!"9::	Z!33DMM$,,4OYZ[DLLL##F$6$68J8JAN "KKM %>>+ {FD$D(S|_dD"dT]^
 vtRtT9MxdD"dINz3D$iP}fb$dHU~D$PTV_` 
   -T:|M`a,T:|M`a,T8zM^_,T:|M`a	'
# -T7zM^_,T9{M_`!
 	$$T%@%@A#*2; $)! ''t/E/EFt$ELLTZZXY	.. L<<9,++FOOV]]KL d334Goo'

4 ;):;;sBr`qBr?rr oo'
 &QXXYjkm m{{  !9!@!@AUAU!VW ""$<<&&(<<&&(##%$$&Y  	ZDL!">"E"EdllTW"XYY	ZZ	JJLs3   A.T !U  'BU   CU  	T=,T88T= Uc                    | j                   t        d      | j                  t        d      d| j                  cxk  rdk  s&n t        dj                  | j                              | j                  d   j                  t        j                  d| j                               | j                  d	   j                  t        j                  d
| j                               | j                  d   j                  t        j                  d
t        | j                                  | j                  d   j                  t        j                  d
t        | j                                  | j                  j!                         }| j"                  r*| j"                  j%                  dj                  |             t'        | j(                        }|j+                         sEt-        j.                  d       t1        d |D              t3        |      k(  rn+|j+                         sEt        dj                  |            | j"                  r*| j"                  j5                  dj                  |             | j6                  r| j8                  rt        d      | j6                  r| j;                  t<               y| j8                  r| j;                  t>               y| j;                  t@               y)z,Set communication parameters on opened port.NzCan only operate on open portsz(write_timeout is currently not supportedr   l        zinvalid baudrate: {!r}r      !Ir      !Br   r   zNegotiating settings: {}r   c              3   4   K   | ]  }|j                     y wrJ   r   r   s     rP   rQ   z+Serial._reconfigure_port.<locals>.<genexpr>  s     +188+r   z7Remote does not accept parameter change (RFC2217): {!r}zNegotiated settings: {}z-xonxoff and rtscts together are not supported)!r   r   _write_timeoutNotImplementedError	_baudraterq   rl   r   r   structpack	_bytesizeRFC2217_PARITY_MAP_parityRFC2217_STOPBIT_MAP	_stopbitsvaluesr   r
   r	   r   r   r   r   r   r   r   r   _xonxoffrfc2217_set_controlSET_CONTROL_USE_HW_FLOW_CONTROLSET_CONTROL_USE_SW_FLOW_CONTROLSET_CONTROL_USE_NO_FLOW_CONTROL)re   itemsr   s      rP   r   zSerial._reconfigure_port  s,   <<!"BCC
 *%&PQQ
 4>>+G+5<<T^^LMM##J/33FKKt~~4VW##J/33FKKt~~4VW##H-11&++eEWX\XdXdEe2fg##J/33FKKGZ[_[i[iGj4kl ++224;;KK8??FG$//0//#JJt+U++s5z9 //#
 ""["b"bch"ijj;;KK6==eDE<<DMMLMM\\$$%DE]]$$%DE$$%DErh   c                 T   d| _         | j                  rD	 | j                  j                  t        j                         | j                  j                          | j                  r7| j                  j                  d       d| _        t        j                  d       d| _        y#  Y OxY w)z
Close portF   N333333?)
r   r   shutdownr   	SHUT_RDWRr   r   joinr   r   rm   s    rP   r   zSerial.close  s{    <<%%f&6&67""$ <<LLa DLJJsOs   AB# #B'c                 `   t        j                  |      }|j                  dk7  r$t        dj	                  |j                              	 t        j
                  |j                  d      j                         D ]  \  }}|dk(  rot        j                          t        j                  d      | _        | j                  j                  t        |d             | j                  j                  d       z|dk(  rd| _        |d	k(  rd| _        |d
k(  rt#        |d         | _        t'        dj	                  |             d|j(                  cxk  rdk  st'        d       t'        d      	 |j*                  |j(                  fS # t&        $ r}t        dj	                  |            d}~ww xY w)zu        extract host and port from an URL string, other settings are extracted
        an stored in instance
        rfc2217zqexpected a string in the form "rfc2217://<host>:<port>[?option[&option...]]": not starting with rfc2217:// ({!r})TloggingzpySerial.rfc2217r   zenabled loggingign_set_control
poll_modemr   zunknown option: {!r}i   zport not in range 0...65535zPexpected a string in the form "rfc2217://<host>:<port>[?option[&option...]]": {}N)urlparseurlsplitschemer   rl   parse_qsqueryr
  r  basicConfig	getLoggerr   setLevelLOGGER_LEVELSr
   r   r   floatr   rq   porthostname)re   urlpartsr]   r  es         rP   r   zSerial.from_url0  s   
 !!#&<<9$!66<fU\\6JL L	P"*"3"3EKK"F"L"L"N LY&'')")"3"34F"GDKKK((vay)ABKK%%&78006:D3|+-1D*y(,1&),<D)$%;%B%B6%JKKL 

*U* !>?? + !>?? + 

++	  	P!EEKVAYP P	Ps   
D"F 	F-F((F-c                 b    | j                   s
t               | j                  j                         S )z9Return the number of bytes currently in the input buffer.)r   r   r   qsizerm   s    rP   
in_waitingzSerial.in_waitingU  s)     ||"$$  &&((rh   c                     | j                   s
t               t               }	 t        | j                        }t        |      |k  r| j                  | j                  j                         st        d      | j                  j                  d|j                               }|t        |      S ||z  }|j                         rnt        |      |k  rt        |      S # t        j                  $ r Y t        |      S w xY w)z        Read size bytes from the serial port. If a timeout is set it may
        return less characters as requested. With no timeout it will block
        until the requested number of bytes is read.
        z&connection failed (reader thread died)T)r   r   	bytearrayr	   _timeoutr   r   is_aliver   r   get	time_leftbytesr   r   Empty)re   sizedatar   bufs        rP   readzSerial.read\  s     ||"$${	dmm,Gd)d"<<'t||/D/D/F)*RSS''++D'2C2C2EF; ;&??$ d)d" T{ {{ 	T{	s   B
C -$C C=<C=c                    | j                   s
t               | j                  5  	 | j                  j	                  t        |      j                  t        t                     	 ddd       t        |      S # t        j                  $ r}t        dj                  |            d}~ww xY w# 1 sw Y   t        |      S xY w)z        Output the given byte string over the serial port. Can block if the
        connection is blocked. May raise SerialException if the connection is
        closed.
        z$connection failed (socket error): {}N)r   r   r   r   sendallr   replaceIACIAC_DOUBLEDr   r   r   rl   r   )re   r1  r$  s      rP   writezSerial.writet  s     ||"$$ 	XX$$Xd^%;%;C%MN	X
 4y << X%&L&S&STU&VWWX	X
 4ys(   B*<A55B'B""B''B**B=c                     | j                   s
t               | j                  t               | j                  j                         r7| j                  j                  d       | j                  j                         r6yy)z9Clear input buffer, discarding all that is in the buffer.FN)r   r   rfc2217_send_purgePURGE_RECEIVE_BUFFERr   r&  r,  rm   s    rP   r   zSerial.reset_input_buffer  s[    ||"$$ 45%%'!!%( %%'rh   c                 Z    | j                   s
t               | j                  t               y)zs        Clear output buffer, aborting the current output and
        discarding all that is in the buffer.
        N)r   r   r;  PURGE_TRANSMIT_BUFFERrm   s    rP   r   zSerial.reset_output_buffer  s#    
 ||"$$ 56rh   c                 &   | j                   s
t               | j                  r8| j                  j                  dj	                  | j
                  rdnd             | j
                  r| j                  t               y| j                  t               y)z[        Set break: Controls TXD. When active, to transmitting is
        possible.
        zset BREAK to {}rc   inactiveN)	r   r   r   r   rl   _break_stater  SET_CONTROL_BREAK_ONSET_CONTROL_BREAK_OFFrm   s    rP   _update_break_statezSerial._update_break_state  sk    
 ||"$$;;KK.55$BSBShYcde$$%9:$$%:;rh   c                 &   | j                   s
t               | j                  r8| j                  j                  dj	                  | j
                  rdnd             | j
                  r| j                  t               y| j                  t               y)z*Set terminal status line: Request To Send.zset RTS to {}rc   r@  N)	r   r   r   r   rl   
_rts_stater  SET_CONTROL_RTS_ONSET_CONTROL_RTS_OFFrm   s    rP   r   zSerial._update_rts_state  d    ||"$$;;KK_33HU_`a??$$%78$$%89rh   c                 &   | j                   s
t               | j                  r8| j                  j                  dj	                  | j
                  rdnd             | j
                  r| j                  t               y| j                  t               y)z.Set terminal status line: Data Terminal Ready.zset DTR to {}rc   r@  N)	r   r   r   r   rl   
_dtr_stater  SET_CONTROL_DTR_ONSET_CONTROL_DTR_OFFrm   s    rP   r   zSerial._update_dtr_state  rI  rh   c                 n    | j                   s
t               t        | j                         t        z        S )z)Read terminal status line: Clear To Send.)r   r   boolget_modem_stateMODEMSTATE_MASK_CTSrm   s    rP   ctsz
Serial.cts  .     ||"$$D((*-@@AArh   c                 n    | j                   s
t               t        | j                         t        z        S )z*Read terminal status line: Data Set Ready.)r   r   rO  rP  MODEMSTATE_MASK_DSRrm   s    rP   dsrz
Serial.dsr  rS  rh   c                 n    | j                   s
t               t        | j                         t        z        S )z*Read terminal status line: Ring Indicator.)r   r   rO  rP  MODEMSTATE_MASK_RIrm   s    rP   riz	Serial.ri  .     ||"$$D((*-??@@rh   c                 n    | j                   s
t               t        | j                         t        z        S )z*Read terminal status line: Carrier Detect.)r   r   rO  rP  MODEMSTATE_MASK_CDrm   s    rP   cdz	Serial.cd  rZ  rh   c                     | j                   S )z Get the current timeout setting.)r*  rm   s    rP   r   zSerial.timeout  s     }}rh   c                     |%	 |dz    |dk  rt        dj                  |            || _        y# t         $ r t        dj                  |            w xY w)zChange timeout setting.NrF   zNot a valid timeout: {!r}r   )	TypeErrorrq   rl   r*  re   r   s     rP   r   zSerial.timeout  sh     N! { !<!C!CG!LMM	  N !<!C!CG!LMMNs	   0 $Ac                    t         }d}	 | j                  r	 | j                  j                  d      }|s| j                  j                  d       nDt        |      D ]'  }|t         k(  r4|t        k(  rt        }|||z  }%| j                  j                  |       A|t        k(  r|t        k(  r2|
|t        z  }n| j                  j                  t               t         }|t        k(  rt!               }t         }|t"        k(  r#| j%                  t'        |             d}t         }|t(        t*        t,        t.        fv r	|}t0        }| j3                  |       t         }|t0        k(  s| j5                  |       t         }* | j                  r| j                  r| j                  j                  d       yy# t        j
                  $ r Y t        j                  $ r[}| j                  r*| j                  j                  dj                  |             | j                  j                  d       Y d}~d}~ww xY w# | j                  r| j                  j                  d       w w xY w)zRead loop for the socket.Ni   z!socket error in reader thread: {}zread thread terminated)M_NORMALr   r   recvr   r   r   r   r
   rl   r   putr   r7  
M_IAC_SEENSBr)  SE_telnet_process_subnegotiationr.  r   r   r   r   M_NEGOTIATE_telnet_process_command_telnet_negotiate_option)re   moder   r1  r$  bytetelnet_commands          rP   r   zSerial._telnet_read_loop  s   	<	<,,<<,,T2D %%))$/%dO ((Dx'3;#-D  )4 )T 1	 $ 1 1 5 5d ;+3;  )4 )S 0	 $ 1 1 5 5c :#+D!RZ(1I#+D!RZ ??i@PQ(,I#+D!b$d%;;-1N#.D !88>#+D,55ndK'Q((! ,,t {{!!":; o ~~  || {{))*M*T*TUV*WX%%))$/f {{!!":; sG   I F? D8I -'I ?II I'AH=8I =II *I/c                 r    | j                   r+| j                   j                  dj                  |             yyz1Process commands other than DO, DONT, WILL, WONT.zignoring Telnet command: {!r}Nr   r   rl   rr   s     rP   rk  zSerial._telnet_process_command/  .     ;;KK ? F Fw OP rh   c                 T   d}| j                   D ]%  }|j                  |k(  s|j                  |       d}' |sp|t        k(  s	|t        k(  r]| j                  |t        k(  rt        nt        |       | j                  r+| j                  j                  dj                  |             yyyyz&Process incoming DO, DONT, WILL, WONT.FTzrejected Telnet option: {!r}Nr   r]   rt   r   r   rp   r   r   r   r   rl   re   rs   r]   knownitems        rP   rl  zSerial._telnet_negotiate_option5       (( 	D {{f$%%g.	  $'R-''DdVT;;KK''(F(M(Mf(UV  #0 rh   c                    |dd t         k(  r|dd t        k(  rct        |      dk\  rUt        |dd       | _        | j
                  rn| j
                  j                  dj                  | j                               y
|dd t        k(  r}t        |      dk\  rot        |dd       | _	        | j
                  r4| j
                  j                  dj                  | j                               | j                  j                  d       y
|dd t        k(  rd| _        y
|dd t        k(  rd	| _        y
| j                  j!                         D ]3  }|j"                  |dd k(  s|j%                  t'        |dd
               y
 | j
                  r+| j
                  j)                  dj                  |             y
y
y
| j
                  r+| j
                  j)                  dj                  |             y
y
);Process subnegotiation, the data between IAC SB and IAC SE.r   rF   rE   rG   zNOTIFY_LINESTATE: {}NOTIFY_MODEMSTATE: {}r  TFNzignoring COM_PORT_OPTION: {!r}zignoring subnegotiation: {!r})r   SERVER_NOTIFY_LINESTATEr   ordr   r   r   rl   SERVER_NOTIFY_MODEMSTATEr   r   restartFLOWCONTROL_SUSPENDr   FLOWCONTROL_RESUMEr   r  r}   r   r.  r   )re   r   ry  s      rP   ri  z%Serial._telnet_process_subnegotiationH  s   Qq>_,1~!88S^q=P"%i!n"5;;KK$$%;%B%B4??%ST1Q#;;IRS@S#&y1~#6 ;;KK$$%<%C%CDDTDT%UV((0051Q#66,0)1Q#55,1) 1188: `D)Aa.8))%	!"*>?	` {{++,L,S,ST],^_ #' , {{##$C$J$J9$UV rh   c                 |    | j                   5  | j                  j                  |       ddd       y# 1 sw Y   yxY w)zGinternal socket write with no data escaping. used to send telnet stuff.N)r   r   r5  )re   r1  s     rP   _internal_raw_writezSerial._internal_raw_writeh  s2     	'LL  &	' 	' 	's   2;c                 :    | j                  t        |z   |z          yzSend DO, DONT, WILL, WONT.N)r  r7  re   actionr]   s      rP   rp   zSerial.telnet_send_optionm  s      v!67rh   c                     |j                  t        t              }| j                  t        t        z   t
        z   |z   |z   t        z   t        z          y)z%Subnegotiation of RFC2217 parameters.N)r6  r7  r8  r  rg  r   rh  re   r]   r|   s      rP   r   z"Serial.rfc2217_send_subnegotiationq  s>    c;/  rO!;f!Du!Ls!RUW!WXrh   c                 z    | j                   d   }|j                  |       |j                  | j                         y)z~        Send purge request to the remote.
        (PURGE_RECEIVE_BUFFER / PURGE_TRANSMIT_BUFFER / PURGE_BOTH_BUFFERS)
        r   N)r   r   r   r   re   r|   ry  s      rP   r;  zSerial.rfc2217_send_purgev  s1    
 $$W-		$''(rh   c                     | j                   d   }|j                  |       | j                  rt        j                  d       y|j                  | j                         y)z)transmit change of control line to remoter   g?N)r   r   r   r   r   r   r   r  s      rP   r  zSerial.rfc2217_set_control  sE    $$Y/** JJsOIId++,rh   c                      y)z`        check if server is ready to receive data. block for some time when
        not.
        NrK   rm   s    rP   rfc2217_flow_server_readyz Serial.rfc2217_flow_server_ready  s    rh   c                 t   | j                   r| j                  j                         r| j                  r| j                  j	                  d       | j                  t               t        | j                        }|j                         sAt        j                  d       | j                  j                         sn8|j                         sA| j                  r| j                  j                  d       | j                  3| j                  r| j                  j	                  d       | j                  S t        d      )z        get last modem state (cached value. If value is "old", request a new
        one. This cache helps that we don't issue to many requests when e.g. all
        status lines, one after the other is queried by the user (CTS, DSR
        etc.)
        zpolling modem stater   zpoll for modem state failedzusing cached modem statez!remote sends no NOTIFY_MODEMSTATE)r   r   r   r   r
   r   NOTIFY_MODEMSTATEr	   r   r   r   r   r   r   ra  s     rP   rP  zSerial.get_modem_state  s     !!d&>&>&F&F&H{{!!"78,,->?d334Goo'

4  //779 oo' ;;KK''(EF
 '{{!!"<=### ""EFFrh   )rF   rh   )%ru   rv   rw   rx   	BAUDRATESrg   r   r   r   r   r   r'  r3  r9  r   r   rD  r   r   rR  rV  rY  r]  r   setterr   rk  rl  ri  r  rp   r   r;  r  r  rP  __classcell__)r   s   @rP   r   r   z  s%   F4I6$_B*FX"!,J ) )0)7<	:	: B B B B A A A A   ^^	  	  @<HQW&W@'
8Y
)
-!Grh   r   c                   R    e Zd ZdZddZd Zd ZddZddZd Z	d	 Z
d
 Zd Zd Zy)PortManagerz    This class manages the state of Telnet and RFC 2217. It needs a serial
    instance and a connection to work with. Connection is expected to implement
    a (thread safe) write function, that writes the string to the network.
    Nc                    || _         || _        || _        d| _        t        | _        d | _        d | _        d| _        d | _	        d| _
        t        | dt        t        t        t        t         t"              t        | dt$        t        t        t        t         t"              t        | dt$        t        t         t        t        t&              t        | dt(        t        t        t        t         t&              t        | dt(        t        t         t        t        t"              t        | d	t*        t        t        t        t         t"        | j,                  	      t        | d
t*        t        t         t        t        t&        | j,                  	      g| _        | j                  r| j                  j1                  d       | j.                  D ];  }|j2                  t"        u s| j5                  |j6                  |j8                         = y )NF   r   r   r   r   r   r   r   r   z*requesting initial Telnet/RFC 2217 options)serialr[   r   _client_is_rfc2217rc  rm  r   ro  modemstate_masklast_modemstatelinstate_maskrY   r   r   r   r   r   rT   r   rV   r   r   
_client_okr   r
   rb   rp   r^   r]   )re   serial_portr[   r   r]   s        rP   rg   zPortManager.__init__  sc   !$"' 	"  ## vtT4T9MxdD"dINz3D$hO{FD$D(S}fb$dIV|_dD"dT]_c_n_no~D$PTV^`d`o`op 
 ;;KKJK** 	HF||y(''G	Hrh   c                     d| _         | j                  r| j                  j                  d       | j                  d       y)a          callback of telnet option. It gets called when option is activated.
        This one here is used to detect when the client agrees on RFC 2217. A
        flag is set so that other functions like check_modem_lines know if the
        client is OK.
        Tzclient accepts RFC 2217force_notificationN)r  r   r   check_modem_linesrm   s    rP   r  zPortManager._client_ok  s9     #';;KK67 	$7rh   c                 N    | j                   j                  t        |z   |z          yr  )r[   r9  r7  r  s      rP   rp   zPortManager.telnet_send_option  s    cFlV34rh   c                     |j                  t        t              }| j                  j	                  t        t
        z   t        z   |z   |z   t        z   t        z          y)z&Subnegotiation of RFC 2217 parameters.N)r6  r7  r8  r[   r9  rg  r   rh  r  s      rP   r   z'PortManager.rfc2217_send_subnegotiation  sB    c;/cBh86AEICORTTUrh   c                    | j                   j                  xr t        | j                   j                  xr t        z  | j                   j
                  xr t        z  | j                   j                  xr t        z  }|| j                  xs dz  }|t        z  r	|t        z  }|t        z  r	|t        z  }|t        z  r	|t        z  }|t        z  r	|t        z  }|| j                  k7  s|r| j                  r|| j                  z  s|rc| j!                  t"        t%        || j                  z  g             | j&                  r*| j&                  j)                  dj+                  |             |dz  | _	        yy)z        read control lines from serial port and compare the last value sent to remote.
        send updates on changes.
        r   r}     N)r  rR  rQ  rV  rU  rY  rX  r]  r\  r  MODEMSTATE_MASK_CTS_CHANGEMODEMSTATE_MASK_DSR_CHANGEMODEMSTATE_MASK_RI_CHANGEMODEMSTATE_MASK_CD_CHANGEr  r  r   r  r   r   r   rl   )re   r  
modemstatedeltass       rP   r  zPortManager.check_modem_lines  sU    [[__4!4[[__4!46[[^^2 24 [[^^2 24 	 t338q9''44J''44J&&33J&&33J ---1C''Z$:N:N-NTf00,j4+?+??@AC ;;KK$$%<%C%CJ%OP $.#4D  2Drh   c              #   h   K   t        |      D ]   }|t        k(  rt         t         | " yw)z        This generator function is for the user. All outgoing data has to be
        properly escaped, so that no IAC character in the data stream messes up
        the Telnet state machine in the server.

        socket.sendall(escape(data))
        N)r   r7  re   r1  rn  s      rP   escapezPortManager.escape(  s1      dO 	Ds{		
	s   02c              #   @  K   t        |      D ]  }| j                  t        k(  r<|t        k(  rt        | _        ,| j
                  | xj
                  |z  c_        N| S| j                  t        k(  r|t        k(  r2| j
                  | xj
                  |z  c_        n| t        | _        |t        k(  rt               | _        t        | _        |t        k(  r8| j                  t        | j
                               d| _        t        | _        |t        t        t        t        fv r|| _        t         | _        2| j#                  |       t        | _        P| j                  t         k(  se| j%                  | j                  |       t        | _         yw)a          Handle a bunch of incoming bytes. This is a generator. It will yield
        all characters not of interest for Telnet/RFC 2217.

        The idea is that the reader thread pushes data from the socket through
        this filter:

        for byte in filter(socket.recv(1024)):
            # do things like CR/LF conversion/whatever
            # and write data to the serial port
            serial.write(byte)

        (socket error handling code left as exercise for the reader)
        N)r   rm  rc  r7  rf  r   rg  r)  rh  ri  r.  r   r   r   r   ro  rj  rk  rl  r  s      rP   filterzPortManager.filter9  s1     dO (	%DyyH$3; *DI ~~1$."
j(3; ~~1$."
 (DIRZ%.[DN (DIRZ77dnn8MN%)DN (DIb$d33*.D' +DI 006 (DIk)--d.A.A4H$	Q(	%s   E/F3+Fc                 r    | j                   r+| j                   j                  dj                  |             yyrq  rr  rr   s     rP   rk  z#PortManager._telnet_process_commandt  rs  rh   c                 T   d}| j                   D ]%  }|j                  |k(  s|j                  |       d}' |sp|t        k(  s	|t        k(  r]| j                  |t        k(  rt        nt        |       | j                  r+| j                  j                  dj                  |             yyyyru  rv  rw  s        rP   rl  z$PortManager._telnet_negotiate_optionz  rz  rh   c           	         |dd t         k(  r| j                  r*| j                  j                  dj                  |             |dd t        k(  r| j
                  j                  }	 t        j                  d|dd       \  }|dk7  r|| j
                  _        | j                  rD| j                  j                  dj                  |rdnd	| j
                  j                               	 | j                  t        t        j                  d| j
                  j                               y|dd t        k(  r| j
                  j                   }	 t        j                  d|dd       \  }|dk7  r|| j
                  _        | j                  rD| j                  j                  dj                  |rdnd	| j
                  j                                	 | j                  t"        t        j                  d| j
                  j                                y|dd t$        k(  r| j
                  j&                  }	 t        j                  d|dd       d   }|dk7  rt(        |   | j
                  _        | j                  rD| j                  j                  dj                  |rdnd	| j
                  j&                               	 | j                  t*        t        j                  dt,        | j
                  j&                                  y|dd t.        k(  r| j
                  j0                  }	 t        j                  d|dd       d   }|dk7  rt2        |   | j
                  _        | j                  rD| j                  j                  dj                  |rdnd	| j
                  j0                               	 | j                  t4        t        j                  dt6        | j
                  j0                                  y|dd t8        k(  r|dd t:        k(  r}| j
                  j<                  r| j                  t>        t@               y| j
                  jB                  r| j                  t>        tD               y| j                  t>        tF               y|dd tF        k(  rdd| j
                  _        d| j
                  _!        | j                  r| j                  j                  d       | j                  t>        tF               y|dd t@        k(  rSd| j
                  _        | j                  r| j                  j                  d       | j                  t>        t@               y|dd tD        k(  rSd| j
                  _!        | j                  r| j                  j                  d       | j                  t>        tD               y|dd tH        k(  r(| j                  r| j                  jK                  d       y|dd tL        k(  rSd| j
                  _'        | j                  r| j                  j                  d       | j                  t>        tL               y|dd tP        k(  rSd| j
                  _'        | j                  r| j                  j                  d       | j                  t>        tP               y|dd tR        k(  r(| j                  r| j                  jK                  d       y|dd tT        k(  rSd| j
                  _+        | j                  r| j                  j                  d       | j                  t>        tT               y|dd tX        k(  rSd| j
                  _+        | j                  r| j                  j                  d       | j                  t>        tX               y|dd tZ        k(  r(| j                  r| j                  jK                  d       y|dd t\        k(  rSd| j
                  _/        | j                  r| j                  j                  d        | j                  t>        t\               y|dd t`        k(  rd| j
                  _/        | j                  r| j                  j                  d!       | j                  t>        t`               y|dd tb        k(  r!| j                  td        tg        dg             y|dd th        k(  r:| j                  r| j                  j                  d"       | jk                  d#       y|dd tl        k(  r/| j                  r| j                  j                  d$       d| _7        y|dd tp        k(  r/| j                  r| j                  j                  d%       d| _7        y|dd tr        k(  rUtu        |dd       | _;        | j                  r~| j                  j                  d&j                  | jv                               y|dd tx        k(  rUtu        |dd       | _=        | j                  r| j                  j                  d'j                  | jz                               y|dd t|        k(  r|dd t~        k(  r\| j
                  j                          | j                  r| j                  j                  d(       | j                  t        t~               y|dd t        k(  r\| j
                  j                          | j                  r| j                  j                  d)       | j                  t        t               y|dd t        k(  rv| j
                  j                          | j
                  j                          | j                  r| j                  j                  d*       | j                  t        t               y| j                  r{| j                  j                  d+j                  t        |dd                    y| j                  r7| j                  j                  d,j                  t        |dd                    yyyyyy| j                  r+| j                  jK                  d-j                  |             yy# t        $ rR}| j                  r*| j                  j                  d
j                  |             || j
                  _        Y d}~d}~ww xY w# t        $ rR}| j                  r*| j                  j                  dj                  |             || j
                  _        Y d}~?d}~ww xY w# t        $ rR}| j                  r*| j                  j                  dj                  |             || j
                  _        Y d}~
d}~ww xY w# t        $ rR}| j                  r*| j                  j                  dj                  |             || j
                  _        Y d}~
d}~ww xY w).r|  r   rF   zreceived COM_PORT_OPTION: {!r}rE   r      z{} baud rate: {}r   r,  zfailed to set baud rate: {}Nr   rG   z{} data size: {}zfailed to set data size: {}z{} parity: {}zfailed to set parity: {}z{} stop bits: {}zfailed to set stop bits: {}Fzchanged flow control to NoneTz changed flow control to XON/XOFFzchanged flow control to RTS/CTSz'requested break state - not implementedzchanged BREAK to activezchanged BREAK to inactivez%requested DTR state - not implementedzchanged DTR to activezchanged DTR to inactivez%requested RTS state - not implementedzchanged RTS to activezchanged RTS to inactivezrequest for modem stater  suspendresumezline state mask: 0x{:02x}zmodem state mask: 0x{:02x}zpurge inz	purge outz
purge bothzundefined PURGE_DATA: {!r}zundefined COM_PORT_OPTION: {!r}zunknown subnegotiation: {!r})Fr   r   r
   rl   r   r  r   r   unpackr   rq   r   r   r   r   r   bytesizer   r   r   RFC2217_REVERSE_PARITY_MAPr   r   r   stopbitsRFC2217_REVERSE_STOPBIT_MAPr   r  r   SET_CONTROL_REQ_FLOW_SETTINGxonxoffr   r  rtsctsr  r	  SET_CONTROL_REQ_BREAK_STATEr   rB  break_conditionrC  SET_CONTROL_REQ_DTRrL  dtrrM  SET_CONTROL_REQ_RTSrG  rtsrH  NOTIFY_LINESTATEr~  r   r  r  r  r   r  SET_LINESTATE_MASKr  r  SET_MODEMSTATE_MASKr  r   r<  r   r   r>  r   PURGE_BOTH_BUFFERSlist)re   r   backupr   r$  r   r   r  s           rP   ri  z*PortManager._telnet_process_subnegotiation  s
   Qq>_,{{!!"B"I"I)"TU1~---
x"(--y1~"FKX1}/7, {{(();)B)BH5Z_aealalauau)vw001DfkkRWY]YdYdYmYmFno1Q</--
x"(--y1~"FKX1}/7, {{(();)B)BH5Z_aealalauau)vw001DfkkRWY]YdYdYmYmFno1Q:-++
q#]]5)Aa.A!DF{-G-O* {{(()?)?UZ\`\g\g\n\n)op00%KK'9$++:L:L'MNP 1Q</--
x%}}UIaNCAFH1}/J8/T, {{(();)B)BH5Z_aealalauau)vw00'KK':4;;;O;O'PQS 1Q;.Qq>%AA{{**889KMlm++889KMlm889KMlmq^'FF*/DKK').DKK&{{(()GH445GIhiq^'FF*.DKK'{{(()KL445GIhiq^'FF)-DKK&{{(()JK445GIhiq^'BB{{++,UVq^';;26DKK/{{(()BC445GI]^q^'<<27DKK/{{(()DE445GI^_q^'::{{++,STq^'99&*DKKO{{(()@A445GI[\q^'::&+DKKO{{(()BC445GI\]q^'::{{++,STq^'99&*DKKO{{(()@A445GI[\q^'::&+DKKO{{(()BC445GI\] 1Q#3300+aSM# 1Q#44;;KK$$%>?&&$&?1Q#66;;KK$$Y/,0)1Q#55;;KK$$X.,1)1Q#55%(1Q%8";;KK$$%@%G%GHZHZ%[\1Q#66'*9Qq>':$;;KK$$%A%H%HI]I]%^_1Q:-Qq>%99KK224{{((4445FH\]q^'<<KK335{{((5445FH]^q^'99KK224KK335{{((6445FHZ[{{))*F*M*MdS\]^]_S`Na*bc;;KK%%&G&N&NtT]^_^`TaOb&cd  #) 	 A ;| {{##$B$I$I)$TU e " 2{{))*G*N*Nq*QR+1DKK((2 " 2{{))*G*N*Nq*QR+1DKK((2 " 0{{))*D*K*KA*NO)/DKK&&0  " 2{{))*G*N*Nq*QR+1DKK((2sd   '1q 
1r4 -9t 9u0 	r1Ar,,r14	t=At

t	u-Au((u-0	w9AwwrJ   r  )F)ru   rv   rw   rx   rg   r  rp   r   r  r  r  rk  rl  ri  rK   rh   rP   r  r    sB     HF8(5V 5H"7%vQW&~Vrh   r  __main__zrfc2217://localhost:7000r   z{}
z	write...
s   hello
z	read: {}
)
__future__r   r  r   r   r   r   r  ImportErrorurllib.parseparser   queuer  serial.serialutilr   r   r   r   r   r	   DEBUGINFOWARNINGERRORr  rh  NOPDMBRKIPAOAYTECELGArg  r   r   r   r   r7  r8  r   r   r   r   r   r   r   r   r   r  r  r  r  r  r  r   r   r   r   r   r   r~  r  SERVER_FLOWCONTROL_SUSPENDSERVER_FLOWCONTROL_RESUMESERVER_SET_LINESTATE_MASKSERVER_SET_MODEMSTATE_MASKr   RFC2217_ANSWER_MAPr  r	  r  r  r  rB  rC  r  rL  rM  r  rG  rH  SET_CONTROL_REQ_FLOW_SETTING_IN"SET_CONTROL_USE_NO_FLOW_CONTROL_IN!SET_CONTROL_USE_SW_FLOW_CONTOL_IN!SET_CONTROL_USE_HW_FLOW_CONTOL_IN SET_CONTROL_USE_DCD_FLOW_CONTROL SET_CONTROL_USE_DTR_FLOW_CONTROL SET_CONTROL_USE_DSR_FLOW_CONTROLLINESTATE_MASK_TIMEOUTLINESTATE_MASK_SHIFTREG_EMPTYLINESTATE_MASK_TRANSREG_EMPTYLINESTATE_MASK_BREAK_DETECTLINESTATE_MASK_FRAMING_ERRORLINESTATE_MASK_PARTIY_ERRORLINESTATE_MASK_OVERRUN_ERRORLINESTATE_MASK_DATA_READYr\  rX  rU  rQ  r  r  r  r  r<  r>  r  PARITY_NONE
PARITY_ODDPARITY_EVENPARITY_MARKPARITY_SPACEr   dictr
  r  STOPBITS_ONESTOPBITS_ONE_POINT_FIVESTOPBITS_TWOr  r  rc  rf  rj  rT   rU   rV   rW   objectrY   rz   r   r  ru   syssstdoutr9  rl   flushr3  r   rK   rh   rP   <module>r     sb  z '     $ ) ) ]]LL]]	  
  
      
     ! " $ # # $   %%!%#-/3113!   ' ") ") ") %         ") %, "$+ !$+ !#*  #*  #*    "  "                    
 
q


  "!P5G5M5M5O!PP  
""A
 
 #"R6I6O6O6Q"RR  
 		#CN6 CNLDp6 DpNzGZ zG@QV& QVj z)62AJJV]]1%&JJ\"GGJGGIJJ\((34GGI S(  $##$  s"   K K# K K #	K0/K0