
    jjs                        U 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 d dlmZ d dlmZmZmZmZmZmZmZ d dlmZmZmZ d dlmZmZmZm Z m!Z!m"Z" d	d
l#m$Z$m%Z%m&Z&m'Z' d	dl(m)Z) d	dl*m+Z+m,Z, d	dl-m.Z.m/Z/m0Z0m1Z1m2Z2m3Z3m4Z4m5Z5 d	dl6m7Z7 d	dl8m9Z9m:Z:m;Z;m<Z<m=Z=m>Z>m?Z?m@Z@ e:rd dlAZAdZBdZCdZDdZEdZF eG            ZHeee&e'e%f                  eId<   e:re%ZJne&ZJ G d d          ZK G d d          ZL G d d          ZM G d deM          ZN G d deN          ZO G d  d!eM          ZP G d" d#eO          ZQ G d$ d%eN          ZRd&ZSd' ZTeUeVeVeTeTeWeUeUeTeVd(
ZXd) ZY G d* d+          ZZ G d, d-eZ          Z[dS ).    N)abstractmethod)chain)EmptyFull	LifoQueue)time)AnyCallableDictListOptionalTypeUnion)parse_qsunquoteurlparse)
CacheEntryCacheEntryStatusCacheFactoryCacheFactoryInterfaceCacheInterfaceCacheKey   )Encoder_HiredisParser_RESP2Parser_RESP3Parser)	NoBackoff)CredentialProvider"UsernamePasswordCredentialProvider)AuthenticationError$AuthenticationWrongNumberOfArgsErrorChildDeadlockedErrorConnectionError	DataError
RedisErrorResponseErrorTimeoutError)Retry)CRYPTOGRAPHY_AVAILABLEHIREDIS_AVAILABLESSL_AVAILABLEcompare_versionsensure_stringformat_error_messageget_lib_versionstr_if_bytes   *   $s   
       DefaultParserc                       e Zd ZdefdZdS )HiredisRespSerializerargsc                 "   g }t          |d         t                    rEt          |d                                                                                   |dd         z   }n<d|d         v r2t          |d                                                   |dd         z   }	 |                    t          j        |                     nG# t          $ r: t          j
                    \  }}}t          |                              |          w xY w|S 2Pack a series of arguments into the Redis protocolr   r   N    )
isinstancestrtupleencodesplitappendhiredispack_command	TypeErrorsysexc_infor%   with_traceback)selfr9   output_value	tracebacks         `/lsinfo/ai/hellotax_ai/data_center/backend/venv/lib/python3.11/site-packages/redis/connection.pypackzHiredisRespSerializer.packE   s    d1gs## 	5a))//1122T!""X=DDT!W__a))DH4D	=MM'.t445555 	= 	= 	="%,..AuiE""11)<<<	= s    'C ADN)__name__
__module____qualname__r   rP    r4   rO   r8   r8   D   s/        $      r4   r8   c                       e Zd ZddZd ZdS )PythonRespSerializerreturnNc                 "    || _         || _        d S N)_buffer_cutoffrA   )rJ   buffer_cutoffrA   s      rO   __init__zPythonRespSerializer.__init__W   s    +r4   c           	      \   g }t          |d         t                    rEt          |d                                                                                   |dd         z   }n<d|d         v r2t          |d                                                   |dd         z   }t
                              t          t          t          |                                                    t          f          }| j
        }t          | j        |          D ]}t          |          }t          |          |k    s||k    st          |t                    ryt
                              |t          t          |                                          t          f          }|                    |           |                    |           t          }t
                              |t          t          |                                          t          |t          f          }	|                    |           |S r;   )r>   r?   r@   rA   rB   	SYM_EMPTYjoinSYM_STARlenSYM_CRLFrZ   map
memoryview
SYM_DOLLARrC   )rJ   r9   rK   buffr[   arg
arg_lengths          rO   rP   zPythonRespSerializer.pack[   s    d1gs## 	5a))//1122T!""X=DDT!W__a))DH4D~~xSYY)>)>)@)@(KLL+t{D)) 	 	C SJD		M))--c:.. . !~~:s:'='='?'?J  d###c""" ~~"J..00  	 	 	dr4   rW   N)rQ   rR   rS   r\   rP   rT   r4   rO   rV   rV   V   s7           + + + + +r4   rV   c                      e Zd Zed             Zed             Zed             Zed             Zed             Zed             Z	ed             Z
ed             Zedd
            Zed             Zedd            Ze	 dd	ddd            Zed             Zed             Zeedeeeef         eeef         f         fd                        ZdS )ConnectionInterfacec                     d S rY   rT   rJ   s    rO   repr_pieceszConnectionInterface.repr_pieces       r4   c                     d S rY   rT   rJ   callbacks     rO   register_connect_callbackz-ConnectionInterface.register_connect_callback   ro   r4   c                     d S rY   rT   rq   s     rO   deregister_connect_callbackz/ConnectionInterface.deregister_connect_callback   ro   r4   c                     d S rY   rT   rJ   parser_classs     rO   
set_parserzConnectionInterface.set_parser   ro   r4   c                     d S rY   rT   rm   s    rO   connectzConnectionInterface.connect   ro   r4   c                     d S rY   rT   rm   s    rO   
on_connectzConnectionInterface.on_connect   ro   r4   c                     d S rY   rT   rJ   r9   s     rO   
disconnectzConnectionInterface.disconnect   ro   r4   c                     d S rY   rT   rm   s    rO   check_healthz ConnectionInterface.check_health   ro   r4   Tc                     d S rY   rT   rJ   commandr   s      rO   send_packed_commandz'ConnectionInterface.send_packed_command   ro   r4   c                     d S rY   rT   rJ   r9   kwargss      rO   send_commandz ConnectionInterface.send_command   ro   r4   r   c                     d S rY   rT   rJ   timeouts     rO   can_readzConnectionInterface.can_read   ro   r4   Fdisconnect_on_errorpush_requestc                    d S rY   rT   )rJ   disable_decodingr   r   s       rO   read_responsez!ConnectionInterface.read_response   s	     	r4   c                     d S rY   rT   r   s     rO   rE   z ConnectionInterface.pack_command   ro   r4   c                     d S rY   rT   rJ   commandss     rO   pack_commandsz!ConnectionInterface.pack_commands   ro   r4   rW   c                     d S rY   rT   rm   s    rO   handshake_metadataz&ConnectionInterface.handshake_metadata   s	     	r4   NTr   F)rQ   rR   rS   r   rn   rs   ru   ry   r{   r}   r   r   r   r   r   r   rE   r   propertyr   r   bytesr?   r   rT   r4   rO   rk   rk      s         ^   ^   ^   ^   ^   ^   ^   ^    ^   ^    ^   !    ^   ^   ^ E$ue|*<d38n*L$M    ^ X  r4   rk   c            *          e Zd ZdZdddddedddedddd e            ddddd	dfd
edee	         dee
         dee
         dede	de	dedededee	         dee	         dee	         dee	         deedf         deeg df                  dee         dee         deeg df                  f&dZd Zed             Zd  Zd! Zd" Zd# Zd$ Zd% Zed&             Zed'             Zd( Zd) Zd* Zd+ Zd, Z d- Z!d;d/Z"d0 Z#d<d1Z$	 d=d.dd2d3Z%d4 Z&d5 Z'd6epe	fd7Z(e)d6ee*e+e+f         e*e	e	f         f         fd8            Z,e,j-        d9ee*e+e+f         e*e	e	f         f         fd:            Z,dS )>AbstractConnectionz0Manages communication to and from a Redis serverr   NFutf-8stricti   zredis-pyr5   dbpasswordsocket_timeoutsocket_connect_timeoutretry_on_timeoutencodingencoding_errorsdecode_responsessocket_read_sizehealth_check_intervalclient_namelib_namelib_versionusernameretryredis_connect_funccredential_providerprotocolcommand_packerc                 X   |s|r|t          d          t          j                    | _        || _        || _        || _        || _        || _        || _	        || _
        || _        ||}|| _        || _        |t          u rg }|r|                    t                      || _        |s|rX|"t%          t'                      d          | _        nt+          j        |          | _        | j                            |           n!t%          t'                      d          | _        || _        d| _        || _        t7          |||	          | _        d| _        d| _        || _        |                      |
           g | _!        d| _"        	 tG          |          }n/# tH          $ r
 tJ          }Y ntL          $ r tO          d          w xY w|dk     s|dk    rtO          d	          || _(        n'# |dk     s|dk    rtO          d	          || _(        w xY w| )                    |          | _*        dS )
a2  
        Initialize a new Connection.
        To specify a retry policy for specific errors, first set
        `retry_on_error` to a list of the error/s to retry on, then set
        `retry` to a valid `Retry` object.
        To retry on TimeoutError, `retry_on_timeout` can also be set to `True`.
        Nz'username' and 'password' cannot be passed along with 'credential_provider'. Please provide only one of the following arguments: 
1. 'password' and (optional) 'username'
2. 'credential_provider'r   r   ip  zprotocol must be an integerr5      zprotocol must be either 2 or 3)+r%   osgetpidpidr   r   r   r   r   r   r   r   r   r   SENTINELrC   r(   retry_on_errorr)   r   r   copydeepcopyupdate_supported_errorsr   next_health_checkr   r   encoderr   _sock_socket_read_sizery   _connect_callbacksrZ   intrF   DEFAULT_RESP_VERSION
ValueErrorr$   r   _construct_command_packer_command_packer)rJ   r   r   r   r   r   r   r   r   r   rx   r   r   r   r   r   r   r   r   r   r   r   ps                          rO   r\   zAbstractConnection.__init__   st   >  	 	&9&E+   9;;& &#6   ,!)%3"&<# 0X%%N 	0!!,///, 		/N 		/}"9;;22

 "]511
J..~>>>>y{{A..DJ%:"!""4x:JKK"&
!1%%%"$"
	HAA 	% 	% 	%$AAA 	A 	A 	A!"?@@@	A 1uuA%&FGGGDMM 1uuA%&FGGGDM#==nMMs*   F G) G(G) *GG) )$Hc                     d                     d |                                 D                       }d| j        j         d| j        j         d| dS )N,c                 "    g | ]\  }}| d | S )=rT   ).0kvs      rO   
<listcomp>z/AbstractConnection.__repr__.<locals>.<listcomp>,  s&    HHHTQjjQjjHHHr4   <.()>)r_   rn   	__class__rR   rQ   )rJ   	repr_argss     rO   __repr__zAbstractConnection.__repr__+  sY    HHHHT5E5E5G5GHHHII	V4>,VVt~/FVVVVVVr4   c                     d S rY   rT   rm   s    rO   rn   zAbstractConnection.repr_pieces/  ro   r4   c                 R    	 |                                   d S # t          $ r Y d S w xY wrY   )r   	Exceptionrm   s    rO   __del__zAbstractConnection.__del__3  s?    	OO 	 	 	DD	s    
&&c                 r    ||S t           rt                      S t          | j        | j        j                  S rY   )r+   r8   rV   rZ   r   rA   )rJ   packers     rO   r   z,AbstractConnection._construct_command_packer9  s:    M 	R(***'(;T\=PQQQr4   c                 x    t          j        |          }|| j        vr| j                            |           dS dS )a^  
        Register a callback to be called when the connection is established either
        initially or reconnected.  This allows listeners to issue commands that
        are ephemeral to the connection, for example pub/sub subscription or
        key tracking.  The callback must be a _method_ and will be kept as
        a weak reference.
        N)weakref
WeakMethodr   rC   )rJ   rr   wms      rO   rs   z,AbstractConnection.register_connect_callbackA  sG     ))T,,,#**2..... -,r4   c                     	 | j                             t          j        |                     dS # t          $ r Y dS w xY w)z
        De-register a previously registered callback.  It will no-longer receive
        notifications on connection events.  Calling this is not required when the
        listener goes away, since the callbacks are kept as weak methods.
        N)r   remover   r   r   rq   s     rO   ru   z.AbstractConnection.deregister_connect_callbackM  sQ    	#**7+=h+G+GHHHHH 	 	 	DD	s   ,0 
>>c                 2     || j                   | _        dS )z
        Creates a new instance of parser_class with socket size:
        _socket_read_size and assigns it to the parser for the connection
        :param parser_class: The required parser class
        )r   N)r   _parserrw   s     rO   ry   zAbstractConnection.set_parserX  s     $|T5KLLLr4   c                 *     j         rdS 	  j                             fd fd          }nQ# t          j        $ r t          d          t          $ r'}t                               |                    d}~ww xY w| _         	  j	         
                                 n 	                                n## t          $ r                                    w xY wd  j        D              _         j        D ]} |            }|r |            dS )z5Connects to the Redis server if not already connectedNc                  ,                                      S rY   )_connectrm   s   rO   <lambda>z,AbstractConnection.connect.<locals>.<lambda>f  s     r4   c                 .                         |           S rY   r   )errorrJ   s    rO   r   z,AbstractConnection.connect.<locals>.<lambda>f  s    tu7M7M r4   zTimeout connecting to serverc                 &    g | ]} |            |S rT   rT   )r   refs     rO   r   z.AbstractConnection.connect.<locals>.<listcomp>}  s#    "S"S"S3SSUU"S3"S"S"Sr4   )r   r   call_with_retrysocketr   r(   OSErrorr$   _error_messager   r}   r&   r   r   )rJ   socker   rr   s   `    rO   r{   zAbstractConnection.connect`  sp   : 	F	::--'''')M)M)M)M DD ~ 	? 	? 	?=>>> 	: 	: 	:!$"5"5a"8"8999	: 

	&.!!!! ''--- 	 	 	OO	 #T"S$2I"S"S"S* 	 	CsuuH 	 	s!   !. 'A<"A77A<1B9 9 Cc                     d S rY   rT   rm   s    rO   r   zAbstractConnection._connect  ro   r4   c                     d S rY   rT   rm   s    rO   _host_errorzAbstractConnection._host_error  ro   r4   c                 F    t          |                                 |          S rY   )r/   r   )rJ   	exceptions     rO   r   z!AbstractConnection._error_message  s    #D$4$4$6$6	BBBr4   c                    | j                             |            | j         }d}| j        s| j        s| j        r5| j        pt          | j        | j                  }|                                }|r| j        dvrt          | j         t                    rE| 
                    t                     |j        | j         _        | j                             |            t          |          dk    r
d|d         g} | j        d| j        dg|R   |                                 | _        n|r | j        dg|R dd	i 	 |                                 }nB# t"          $ r5 |                     d|d
         d	           |                                 }Y nw xY wt%          |          dk    rt'          d          n| j        dvrt          | j         t                    rE| 
                    t                     |j        | j         _        | j                             |            |                     d| j                   |                                 | _        | j                            d          | j        k    r2| j                            d          | j        k    rt+          d          | j        rP|                     dd| j                   t%          |                                           dk    rt+          d          	 | j        r1|                     ddd| j                   |                                  n# t0          $ r Y nw xY w	 | j        r1|                     ddd| j                   |                                  n# t0          $ r Y nw xY w| j        rO|                     d| j                   t%          |                                           dk    rt+          d          dS dS )z=Initialize the connection, authenticate and select a databaseN)r5   2r   defaultr   HELLOAUTHr   Fr   OKzInvalid Username or Passwords   protoprotozInvalid RESP versionCLIENTSETNAMEzError setting client nameSETINFOzLIB-NAMEzLIB-VERSELECTzInvalid Database)r   r}   r   r   r   r    get_credentialsr   r>   r   ry   r   EXCEPTION_CLASSESra   r   r   r   r"   r1   r!   getr$   r   r   r'   r   r   )rJ   parser	auth_argscred_providerauth_responses        rO   r}   zAbstractConnection.on_connect  s3   %%%	# 	8 	8 	8( T5dmT]SS  &5577I  -	>h66$,55 .---171I.''---9~~""&	!5	Dgt}fIyIIII&*&8&8&:&:D##
  	> DfEyEEEuEEE5 $ 2 2 4 47 5 5 5
 !!&)B-e!LLL $ 2 2 4 45 M**d22)*HIII 3 ](**$,55 .---171I.''---gt}555&*&8&8&:&:D#'++H55FF+//88DMII%&<===  	Ch	43CDDDD..0011T99%&ABBB	} %!!(Iz4=QQQ""$$$ 	 	 	D		 %!!(Iy$BRSSS""$$$ 	 	 	D	 7 	:h000D..0011T99%&8999	: 	:99s6   9E <FF<8L5 5
MM8M? ?
NNc                 P   | j                                          | j        }d| _        |dS t          j                    | j        k    r8	 |                    t          j                   n# t          t          f$ r Y nw xY w	 |                                 dS # t          $ r Y dS w xY w)z!Disconnects from the Redis serverN)r   on_disconnectr   r   r   r   shutdownr   	SHUT_RDWRr   rF   close)rJ   r9   	conn_socks      rO   r   zAbstractConnection.disconnect  s    ""$$$J	
F9;;$(""""6#34444Y'   	OO 	 	 	DD	s$   	A) )A=<A=B 
B%$B%c                     |                      dd           t          |                                           dk    rt          d          dS )z Send PING, expect PONG in returnPINGFr  PONGz#Bad response from PING health checkN)r   r1   r   r$   rm   s    rO   
_send_pingzAbstractConnection._send_ping  sQ    &u555**,,--77!"GHHH 87r4   c                 .    |                                   dS )z Function to call when PING failsNr   )rJ   r   s     rO   _ping_failedzAbstractConnection._ping_failed      r4   c                     | j         r>t                      | j        k    r)| j                            | j        | j                   dS dS dS )z3Check the health of the connection with a PING/PONGN)r   r   r   r   r   r  r  rm   s    rO   r   zAbstractConnection.check_health  sT    % 	K$&&43I*I*IJ&&t8IJJJJJ	K 	K*I*Ir4   Tc                 r   | j         s|                                  |r|                                  	 t          |t                    r|g}|D ]}| j                             |           dS # t          j        $ r$ |                                  t          d          t          $ rq}|                                  t          |j                  dk    rd|j        d         }}n|j        d         }|j        d         }t          d| d| d          d}~wt          $ r |                                   w xY w)	z2Send an already packed command to the Redis serverzTimeout writing to socketr   UNKNOWNr   zError z while writing to socket. r   N)r   r{   r   r>   r?   sendallr   r   r   r(   r   ra   r9   r$   BaseException)rJ   r   r   itemr   errnoerrmsgs          rO   r   z&AbstractConnection.send_packed_command  sk   z 	LLNNN 	 	'3'' $") ) )
""4(((() )~ 	< 	< 	<OO:;;; 	W 	W 	WOO16{{a )16!9vq	!"U5"U"UF"U"U"UVVV 	 	 	
 OO	s   7A, ,;D6'A,D#D6c                 v    |                       | j        j        | |                    dd                     dS )z+Pack and send a command to the Redis serverr   Tr  N)r   r   rP   r  r   s      rO   r   zAbstractConnection.send_command*  sJ      %D %t,ND99 	! 	
 	
 	
 	
 	
r4   c                    | j         }|s|                                  |                                 }	 | j                            |          S # t
          $ r3}|                                  t          d| d|j                   d}~ww xY w)z8Poll the socket to see if there's data that can be read.Error while reading from z: N)	r   r{   r   r   r   r   r   r$   r9   )rJ   r   r   
host_errorr   s        rO   r   zAbstractConnection.can_read1  s    z 	LLNNN%%''
	V<((111 	V 	V 	VOO!"Tj"T"TAF"T"TUUU	Vs   A 
B
.BB
r   c                r   |                                  }	 | j        dv r$t          s| j                            ||          }n| j                            |          }n# t
          j        $ r) |r|                                  t          d|           t          $ r5}|r|                                  t          d| d|j                   d}~wt          $ r |r|                                   w xY w| j        rt                      | j        z   | _        t!          |t"                    r		 |# ~w xY w|S )z0Read the response from a previously sent command)3r   )r   r   )r   zTimeout reading from r)  z : N)r   r   r+   r   r   r   r   r   r(   r   r$   r9   r#  r   r   r   r>   r'   )rJ   r   r   r   r*  responser   s          rO   r   z AbstractConnection.read_response?  s    %%''
	}((1B(<55%5L 6    <55GW5XX~ 	E 	E 	E" "!!!CzCCDDD 	 	 	" "!!!!GJGGqvGG    	 	 	 # "!!!	 % 	I%)VVd.H%HD"h.. 	s%   AA A C40C%C4/D1 1D4c                       | j         j        | S )r<   )r   rP   r   s     rO   rE   zAbstractConnection.pack_commandm  s    (t#($//r4   c                    g }g }d}| j         }|D ]} | j        j        | D ]}t          |          }||k    s||k    st	          |t
                    r3|r-|                    t                              |                     d}g }||k    st	          |t
                    r|                    |           |                    |           ||z  }|r-|                    t                              |                     |S )z.Pack multiple commands into the Redis protocolr   )	rZ   r   rP   ra   r>   rd   rC   r^   r_   )	rJ   r   rK   piecesbuffer_lengthr[   cmdchunkchunklens	            rO   r   z AbstractConnection.pack_commandsq  s(   + 	. 	.C2-2C8 . .u::!M11-//!%44 0  >innV&<&<===$%MFm++z%/L/L+MM%((((MM%(((!X-MM!.$  	2MM)..00111r4   rW   c                     | j         S rY   )r   rm   s    rO   get_protocolzAbstractConnection.get_protocol  s
    }r4   c                     | j         S rY   _handshake_metadatarm   s    rO   r   z%AbstractConnection.handshake_metadata  s    ''r4   rM   c                     || _         d S rY   r8  )rJ   rM   s     rO   r   z%AbstractConnection.handshake_metadata  s    #(   r4   r   r   r   ).rQ   rR   rS   __doc__r   r6   r0   r   r   r?   floatboolr   r	   r
   r   r\   r   r   rn   r   r   rs   ru   ry   r{   r   r   r   r}   r   r  r  r   r   r   r   r   rE   r   r6  r   r   r   r   setterrT   r4   rO   r   r      s       66 "&*.26!&'!&" %%&%)",%4_%6%6"&"&;?<@"#7;-XN XNXN 3-XN !	XN
 !)XN XN XN XN XN XN  #XN c]XN 3-XN  c]!XN" 3-#XN$ S$Y%XN& %Xb$h%78'XN( &&89)XN* 3-+XN, !"d(!34-XN XN XN XNtW W W   ^  R R R
/ 
/ 
/	 	 	M M M! ! !F   ^   ^C C CX: X: X:t  (I I I  K K K
   >
 
 
V V V V  , !, , , , ,\0 0 0  <cjS     (E$ue|*<d38n*L$M ( ( ( X( )d5%<.@$sCx..P(Q ) ) ) ) ) )r4   r   c                   @     e Zd ZdZ	 	 	 	 	 d fd	Zd Zd	 Zd
 Z xZS )
Connectionz4Manages TCP communication to and from a Redis server	localhost  FNr   c                     || _         t          |          | _        || _        |pi | _        || _         t                      j        di | d S NrT   )hostr   portsocket_keepalivesocket_keepalive_optionssocket_typesuperr\   )rJ   rE  rF  rG  rH  rI  r   r   s          rO   r\   zConnection.__init__  sY     	II	 0(@(FB%&""6"""""r4   c                     d| j         fd| j        fd| j        fg}| j        r|                    d| j        f           |S )NrE  rF  r   r   )rE  rF  r   r   rC   rJ   r0  s     rO   rn   zConnection.repr_pieces  sL    49%	':T47OL 	=MM=$*:;<<<r4   c                    d}t          j        | j        | j        | j        t           j                  D ]<}|\  }}}}}d}	 t          j         |||          }|                    t           j        t           j        d           | j	        rk|                    t           j
        t           j        d           | j                                        D ]&\  }	}
|                    t           j        |	|
           '|                    | j                   |                    |           |                    | j                   |c S # t$          $ r#}|}||                                 Y d}~6d}~ww xY w||t%          d          )zCreate a TCP socket connectionNr   z)socket.getaddrinfo returned an empty list)r   getaddrinforE  rF  rI  SOCK_STREAM
setsockoptIPPROTO_TCPTCP_NODELAYrG  
SOL_SOCKETSO_KEEPALIVErH  items
settimeoutr   r{   r   r   r  )rJ   errresfamilysocktyper  	canonnamesocket_addressr   r   r   rL   s               rO   r   zConnection._connect  s   
 %Ity$"2F4F
 
 	! 	!C BE>FHeYD!}VXu== 2F4FJJJ ( BOOF$5v7JANNN $ = C C E E B B1(:AqAAAA  ;<<< ^,,,  3444 ! ! !#JJLLL!
 ?IABBBs   C=E
E.E))E.c                 $    | j          d| j         S )N:)rE  rF  rm   s    rO   r   zConnection._host_error  s    )))di)))r4   )rA  rB  FNr   	rQ   rR   rS   r;  r\   rn   r   r   __classcell__r   s   @rO   r@  r@    s        :: !%# # # # # #   'C 'C 'CR* * * * * * *r4   r@  c                   J   e Zd ZdZdZdZdededej	        fdZ
d Zd	 Zd
 Zd Zd Zd Zd Zd Zd#dZd Zd$dZ	 d%ddddZd Zd Zedeeeef         eeef         f         fd            Zd Zd Z deddfdZ!d  Z"d!e#eee$e#e                  f                  fd"Z%dS )&CacheProxyConnections   fooz7.4.0redisconncache	pool_lockc                 F   t          j                    | _        || _        | j        j        | _        | j        j        | _        | j        j        | _        || _        || _        t          j
                    | _        d | _        d | _        |                     | j                   d S rY   )r   r   r   _connr   rE  rF  
_pool_lock_cache	threadingLock_cache_lock_current_command_cache_key_current_optionsrs   _enable_tracking_callback)rJ   re  rf  rg  s       rO   r\   zCacheProxyConnection.__init__  s     9;;
Z%
JO	JO	#$>++*.' $&&t'EFFFFFr4   c                 4    | j                                         S rY   )ri  rn   rm   s    rO   rn   z CacheProxyConnection.repr_pieces  s    z%%'''r4   c                 :    | j                             |           d S rY   )ri  rs   rq   s     rO   rs   z.CacheProxyConnection.register_connect_callback  s    
,,X66666r4   c                 :    | j                             |           d S rY   )ri  ru   rq   s     rO   ru   z0CacheProxyConnection.deregister_connect_callback  s    
..x88888r4   c                 :    | j                             |           d S rY   )ri  ry   rw   s     rO   ry   zCacheProxyConnection.set_parser  s    
l+++++r4   c                    | j                                          | j         j                            dd           }| | j         j                            dd           }| j         j                            dd           }| | j         j                            dd           }||t	          d          t          |          }t          |          }|| j        k    st          || j                  dk    rt	          d          d S )Ns   serverservers   versionversionz0Cannot retrieve information about server versionr   ziTo maximize compatibility with all Redis products, client-side caching is supported by Redis 7.4 or later)	ri  r{   r   r  r$   r.   DEFAULT_SERVER_NAMEr-   MIN_ALLOWED_VERSION)rJ   server_name
server_vers      rO   r{   zCacheProxyConnection.connect  s   
j377	4HH*7;;HdKKKZ266z4HH
6::9dKKJ!3!"TUUU":..
#K00 4333
D,DEEJJ!{   KJr4   c                 8    | j                                          d S rY   )ri  r}   rm   s    rO   r}   zCacheProxyConnection.on_connect  s    
r4   c                     | j         5  | j                                         d d d            n# 1 swxY w Y    | j        j        |  d S rY   )rn  rk  flushri  r   r   s     rO   r   zCacheProxyConnection.disconnect  s     	  	 K	  	  	  	  	  	  	  	  	  	  	  	  	  	  	 
t$$$$s   .22c                 8    | j                                          d S rY   )ri  r   rm   s    rO   r   z!CacheProxyConnection.check_health#  s    
!!!!!r4   Tc                 :    | j                             |           d S rY   )ri  r   r   s      rO   r   z(CacheProxyConnection.send_packed_command&  s      	
&&w/////r4   c           	      T   |                                   | j        5  | j                            t	          |d         d                    s'd | _         | j        j        |i | 	 d d d            d S 	 d d d            n# 1 swxY w Y   |                    d          t          d          t	          |d         t          |                    d                              | _        | j        5  | j                            | j                  r| j                            | j                  }|j        | j        k    rl| j        5  |j                                        r4|j                            d           |j                                        4d d d            n# 1 swxY w Y   	 d d d            d S | j                            t!          | j        | j        t$          j        | j                             d d d            n# 1 swxY w Y    | j        j        |i | d S )	Nr   rT   )r   
redis_keyskeyszCannot create cache key.Tr   )	cache_keycache_valuestatusconnection_ref)_process_pending_invalidationsrn  rk  is_cachabler   ro  ri  r   r  r   r@   r  rj  r   r   setr   DUMMY_CACHE_VALUEr   IN_PROGRESS)rJ   r9   r   entrys       rO   r   z!CacheProxyConnection.send_command+  s6   ++--- 	 	 ;**8DGPR+S+S+STT 26/'
'8888	 	 	 	 	 	 	 		 	 	 	 	 	 	 	 	 	 	 	 	 	 	 ::f%7888 +3Gfjj.@.@(A(A+
 +
 +
'  	 	 {t>?? (GHH'4:55 R R#2;;== R!0>>D>QQQ $2;;== RR R R R R R R R R R R R R R R 	 	 	 	 	 	 	 	  KOO"= $ 6+7#':	    !	 	 	 	 	 	 	 	 	 	 	 	 	 	 	6 	 
000000sQ   A
B  BB2AHAF"H"F&	&H)F&	*H;AHHHr   c                 6    | j                             |          S rY   )ri  r   r   s     rO   r   zCacheProxyConnection.can_read[  s    z""7+++r4   Fr   c                    | j         5  | j        | j                            | j                  t| j                            | j                  j        t
          j        k    rBt          j        | j                            | j                  j	                  cd d d            S d d d            n# 1 swxY w Y   | j
                            |||          }| j         5  | j        |cd d d            S |.| j                            | j        g           |cd d d            S | j                            | j                  }|2t
          j        |_        ||_	        | j                            |           d d d            n# 1 swxY w Y   |S )N)r   r   r   )rn  ro  rk  r  r  r   r  r   r   r  ri  r   delete_by_cache_keysVALIDr  )rJ   r   r   r   r-  cache_entrys         rO   r   z"CacheProxyConnection.read_response^  sS     
	 
	 /;KOOD$CDDPKOOD$CDDK#/0 0 }KOOD$CDDP 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 
	 :++- 3% , 
 
  	- 	-.6	- 	- 	- 	- 	- 	- 	- 	-
 00$2Q1RSSS	- 	- 	- 	- 	- 	- 	- 	- +//$*IJJK &%5%;"*2',,,!	- 	- 	- 	- 	- 	- 	- 	- 	- 	- 	- 	- 	- 	- 	-$ s1   BB//B36B3	F4#F$AFF
Fc                       | j         j        | S rY   )ri  rE   r   s     rO   rE   z!CacheProxyConnection.pack_command  s    &tz&--r4   c                 6    | j                             |          S rY   )ri  r   r   s     rO   r   z"CacheProxyConnection.pack_commands  s    z''111r4   rW   c                     | j         j        S rY   )ri  r   rm   s    rO   r   z'CacheProxyConnection.handshake_metadata  s    z,,r4   c                 8    | j                                          d S rY   )ri  r   rm   s    rO   r   zCacheProxyConnection._connect  s    
r4   c                 8    | j                                          d S rY   )ri  r   rm   s    rO   r   z CacheProxyConnection._host_error  s    
     r4   Nc                     |                     ddd           |                                 |j                            | j                   d S )Nr  TRACKINGON)r   r   r   set_invalidation_push_handler_on_invalidation_callback)rJ   re  s     rO   rq  z.CacheProxyConnection._enable_tracking_callback  sK    (J5552243QRRRRRr4   c                     |                                  r1| j                            d           |                                  /d S d S )NTr  )r   ri  r   rm   s    rO   r  z3CacheProxyConnection._process_pending_invalidations  sQ    mmoo 	8J$$$$777 mmoo 	8 	8 	8 	8 	8r4   datac                     | j         5  |d         | j                                         n | j                            |d                    d d d            d S # 1 swxY w Y   d S )Nr   )rn  rk  r  delete_by_redis_keys)rJ   r  s     rO   r  z.CacheProxyConnection._on_invalidation_callback  s     	: 	:Aw!!####00a999	: 	: 	: 	: 	: 	: 	: 	: 	: 	: 	: 	: 	: 	: 	: 	: 	: 	:s   AAAAr   r   r   )&rQ   rR   rS   r  rz  ry  rk   r   rl  rm  r\   rn   rs   ru   ry   r{   r}   r   r   r   r   r   r   rE   r   r   r   r   r   r?   r   r   r   rq  r  r   r   r  rT   r4   rO   rc  rc    s"       !!G!G G >	G G G G$( ( (7 7 79 9 9, , ,  .     % % %
" " "0 0 0 0
.1 .1 .1`, , , ,  %'=APU' ' ' ' 'R. . .2 2 2 -E$ue|*<d38n*L$M - - - X-  ! ! !S.A Sd S S S S
8 8 8:d5htE{>S9S3T.U : : : : : :r4   rc  c                   P     e Zd ZdZ	 	 	 	 	 	 	 	 	 	 	 	 	 	 d fd	Z fdZd Z xZS )	SSLConnectionzManages SSL connections to and from the Redis server(s).
    This class extends the Connection class, adding SSL functionality, and making
    use of ssl.SSLContext (https://docs.python.org/3/library/ssl.html#ssl.SSLContext)
    NrequiredFc                    t           st          d          || _        || _        |t          j        }nWt          |t                    rBt          j        t          j        t          j	        d}||vrt          d|           ||         }|| _
        || _        || _        || _        || _        || _        |	| _        |
| _        || _        || _        || _        || _         t-                      j        di | dS )a  Constructor

        Args:
            ssl_keyfile: Path to an ssl private key. Defaults to None.
            ssl_certfile: Path to an ssl certificate. Defaults to None.
            ssl_cert_reqs: The string value for the SSLContext.verify_mode (none, optional, required). Defaults to "required".
            ssl_ca_certs: The path to a file of concatenated CA certificates in PEM format. Defaults to None.
            ssl_ca_data: Either an ASCII string of one or more PEM-encoded certificates or a bytes-like object of DER-encoded certificates.
            ssl_check_hostname: If set, match the hostname during the SSL handshake. Defaults to False.
            ssl_ca_path: The path to a directory containing several CA certificates in PEM format. Defaults to None.
            ssl_password: Password for unlocking an encrypted private key. Defaults to None.

            ssl_validate_ocsp: If set, perform a full ocsp validation (i.e not a stapled verification)
            ssl_validate_ocsp_stapled: If set, perform a validation on a stapled ocsp response
            ssl_ocsp_context: A fully initialized OpenSSL.SSL.Context object to be used in verifying the ssl_ocsp_expected_cert
            ssl_ocsp_expected_cert: A PEM armoured string containing the expected certificate to be returned from the ocsp verification service.
            ssl_min_version: The lowest supported SSL version. It affects the supported SSL versions of the SSLContext. None leaves the default provided by ssl module.
            ssl_ciphers: A string listing the ciphers that are allowed to be used. Defaults to None, which means that the default ciphers are used. See https://docs.python.org/3/library/ssl.html#ssl.SSLContext.set_ciphers for more information.

        Raises:
            RedisError
        z$Python wasn't built with SSL supportN)noneoptionalr  z+Invalid SSL Certificate Requirements Flag: rT   )r,   r&   keyfilecertfilessl	CERT_NONEr>   r?   CERT_OPTIONALCERT_REQUIRED	cert_reqsca_certsca_dataca_pathcheck_hostnamecertificate_passwordssl_validate_ocspssl_validate_ocsp_stapledssl_ocsp_contextssl_ocsp_expected_certssl_min_versionssl_ciphersrJ  r\   )rJ   ssl_keyfilessl_certfilessl_cert_reqsssl_ca_certsssl_ca_datassl_check_hostnamessl_ca_pathssl_passwordr  r  r  r  r  r  r   	CERT_REQSr   s                    rO   r\   zSSLConnection.__init__  s   P  	ECDDD"$ MMMs++ 
	5-- I
 I-- Q-QQ   &m4M&$""0$0!!2)B& 0&<#.&""6"""""r4   c                     t                                                      }	 |                     |          S # t          t          f$ r |                                  w xY w)zN
        Wrap the socket with SSL support, handling potential errors.
        )rJ  r   _wrap_socket_with_sslr   r&   r  )rJ   r   r   s     rO   r   zSSLConnection._connect  sc     ww!!	--d333$ 	 	 	JJLLL	s	   8 'Ac                 H   t          j                    }| j        |_        | j        |_        | j        s| j        r'|                    | j        | j        | j                   | j	        | j
        | j        '|                    | j	        | j
        | j                   | j        | j        |_        | j        r|                    | j                   | j        du rt$          du rt'          d          | j        r| j        rt'          d          |                    || j                  }| j        rd	dl}d
dlm} | j        Y|j                            |j        j                  }|                    | j                   |                    | j                   n| j        }|                     || j!                   |j        "                    |tG          j#                              }|$                                 |%                    | j        | j&        f           |'                                 |(                                 |S | j        du rOt$          rHd
dlm)}  ||| j        | j&        | j	                  }	|	*                                r|S tW          d          |S )z
        Wraps the socket with SSL support.

        Args:
            sock: The plain socket to wrap with SSL.

        Returns:
            An SSL wrapped socket.
        )r  r  r   N)cafilecapathcadataTFzcryptography is not installed.zKEither an OCSP staple or pure OCSP connection must be validated - not both.)server_hostnamer   r   )ocsp_staple_verifier)OCSPVerifierzocsp validation error),r  create_default_contextr  r  verify_moder  r  load_cert_chainr  r  r  r  load_verify_locationsr  minimum_versionr  set_ciphersr  r*   r&   r  wrap_socketrE  OpenSSLocspr  r  SSLContextSSLv23_METHODuse_certificate_fileuse_privatekey_fileset_ocsp_client_callbackr  r@  r   request_ocspr{   rF  do_handshaker  r  is_validr$   )
rJ   r   contextsslsockr  r  
staple_ctxconr  os
             rO   r  z#SSLConnection._wrap_socket_with_ssl  s    ,..!%!4"n= 	DL 	##2 $    M%|'|'))}T\$, *    +&*&:G# 	2 0111!T)).D.M.M=>>>) 	d.D 	  
 %%dDI%FF ) 	NNN222222 $,$[001JKK
//>>>..t|<<<<!2
//$d&A  
 +((V]__EECKKDI.///LLNNNN !T)).D)******WdiDMJJAzz|| ?%&=>>>r4   )NNr  NNFNNFFNNNN)rQ   rR   rS   r;  r\   r   r  r`  ra  s   @rO   r  r    s            "'#F# F# F# F# F# F#P	 	 	 	 	M M M M M M Mr4   r  c                   6     e Zd ZdZd fd	Zd Zd Zd Z xZS )	UnixDomainSocketConnectionz4Manages UDS communication to and from a Redis server Nc                 V     t                      j        di | || _        || _        d S rD  )rJ  r\   pathr   )rJ   r  r   r   r   s       rO   r\   z#UnixDomainSocketConnection.__init__U  s5    ""6"""	,r4   c                 p    d| j         fd| j        fg}| j        r|                    d| j        f           |S )Nr  r   r   )r  r   r   rC   rL  s     rO   rn   z&UnixDomainSocketConnection.repr_piecesZ  sC    49%dg7 	=MM=$*:;<<<r4   c                 >   t          j         t           j        t           j                  }|                    | j                   	 |                    | j                   n## t          $ r |                                  w xY w|                    | j	                   |S )z&Create a Unix domain socket connection)
r   AF_UNIXrO  rV  r   r{   r  r   r  r   )rJ   r   s     rO   r   z#UnixDomainSocketConnection._connect`  s    }V^V-?@@3444	LL#### 	 	 	JJLLL	 	+,,,s   A    B c                     | j         S rY   )r  rm   s    rO   r   z&UnixDomainSocketConnection._host_errorm  s
    yr4   )r  Nr_  ra  s   @rO   r  r  R  sp        ::- - - - - -
          r4   r  )0FFALSENNOc                     | | dk    rd S t          | t                    r|                                 t          v rdS t	          |           S )Nr  F)r>   r?   upperFALSE_STRINGSr=  )rM   s    rO   to_boolr  t  sI    }t% %++--="@"@u;;r4   )
r   r   r   rG  r   r   max_connectionsr   r  r   c                 \   |                      d          s9|                      d          s$|                      d          st          d          t          |           } i }t          | j                                                  D ]\  }}|rt          |          dk    rnt          |d                   }t          	                    |          }|r8	  ||          ||<   [# t          t          f$ r t          d| d          w xY w|||<   | j        rt          | j                  |d<   | j        rt          | j                  |d	<   | j        d
k    r)| j        rt          | j                  |d<   t          |d<   n| j        rt          | j                  |d<   | j        rt%          | j                  |d<   | j        rUd|vrQ	 t%          t          | j                                      dd                    |d<   n# t(          t          f$ r Y nw xY w| j        dk    r
t*          |d<   |S )Nzredis://z	rediss://zunix://zRRedis URL must specify one of the following schemes (redis://, rediss://, unix://)r   zInvalid value for 'z' in connection URL.r   r   unixr  connection_classrE  rF  r   /r  rediss)
startswithr   r   r   queryrU  ra   r   URL_QUERY_ARGUMENT_PARSERSr  rF   r   r   schemer  r  hostnamerF  r   replaceAttributeErrorr  )urlr   namerM   r  s        rO   	parse_urlr    su   z""
>>+&&
 >>)$$

 5
 
 	

 3--CF	**0022 
% 
%e 		%SZZ!^^E!H%%E/33D99F %W#)6%==F4LL!:. W W W$%U4%U%U%UVVVW  %t
| 3$S\22z
| 3$S\22z zV8 	/$SX..F6N%?!"" < 	3$S\22F6N8 	+ ]]F6N 8 	F**"738#4#4#<#<S"#E#EFFt"J/    :!!)6F%&Ms   C!!%D8H   HHc                       e Zd ZdZed             Zeddfdee         dee	         fdZ
deeffdZd	 Zdd
ZddZdeddfdZdefdZddZddZdddefdZddeddfdZddZd dZdS )!ConnectionPoola  
    Create a connection pool. ``If max_connections`` is set, then this
    object raises :py:class:`~redis.exceptions.ConnectionError` when the pool's
    limit is reached.

    By default, TCP connections are created unless ``connection_class``
    is specified. Use class:`.UnixDomainSocketConnection` for
    unix sockets.

    Any additional keyword arguments are passed to the constructor of
    ``connection_class``.
    c                 x    t          |          }d|v r|d         |d<   |                    |            | di |S )a  
        Return a connection pool configured from the given URL.

        For example::

            redis://[[username]:[password]]@localhost:6379/0
            rediss://[[username]:[password]]@localhost:6379/0
            unix://[username@]/path/to/socket.sock?db=0[&password=password]

        Three URL schemes are supported:

        - `redis://` creates a TCP socket connection. See more at:
          <https://www.iana.org/assignments/uri-schemes/prov/redis>
        - `rediss://` creates a SSL wrapped TCP socket connection. See more at:
          <https://www.iana.org/assignments/uri-schemes/prov/rediss>
        - ``unix://``: creates a Unix Domain Socket connection.

        The username, password, hostname, path and all querystring values
        are passed through urllib.parse.unquote in order to replace any
        percent-encoded values with their corresponding characters.

        There are several ways to specify a database number. The first value
        found will be used:

            1. A ``db`` querystring option, e.g. redis://localhost?db=0
            2. If using the redis:// or rediss:// schemes, the path argument
               of the url, e.g. redis://localhost/0
            3. A ``db`` keyword argument to this function.

        If none of these options are specified, the default db=0 is used.

        All querystring options are cast to their appropriate Python types.
        Boolean arguments can be specified with string values "True"/"False"
        or "Yes"/"No". Values that cannot be properly cast cause a
        ``ValueError`` to be raised. Once parsed, the querystring arguments
        and keyword arguments are passed to the ``ConnectionPool``'s
        class initializer. In the case of conflicting arguments, querystring
        arguments always win.
        r  rT   )r  update)clsr  r   url_optionss       rO   from_urlzConnectionPool.from_url  sP    R  nn''.45G.HK*+k"""s}}V}}r4   Nr  cache_factoryc                 P   |pd}t          |t                    r|dk     rt          d          || _        || _        || _        d | _        || _        |                    d          s|                    d          r|                    d          dvrt          d          | j                            d          }|,t          |t                    st          d	          || _        nd| j        | j                                        | _        n>t          | j                            d                                                    | _        |                    dd            |                    dd            t          j                    | _        |                                  d S )
Nl        r   z,"max_connections" must be a positive integercache_configrf  r   )r   r,  z4Client caching is only supported with RESP version 3z#Cache must implement CacheInterface)r>   r   r   r  connection_kwargsr  rf  _cache_factoryr  r&   r   	get_cacher   poprl  rm  
_fork_lockreset)rJ   r  r  r  r  rf  s         rO   r\   zConnectionPool.__init__  s    *2U/3// 	M?Q3F3FKLLL 0!2.
+  00 	"4E4I4I'4R4R 	" $$Z00@@ !WXXX*..w77E !%88 L$%JKKK"

&2!%!4!>!>!@!@DJJ!-.22>BB" "ikk J 	gt,,,nd333 $.**

r4   rW   c                     dt          |           j         dt          |           j         dt           | j        di | j                   dS )Nr   r   r   r   rT   )typerR   rQ   reprr  r  rm   s    rO   r   zConnectionPool.__repr__4  so    JT

% J JT

(; J J*T*DDT-CDDEEJ J J	
r4   c                 8    | j                             dd          S )z
        Returns:
            The RESP protocol version, or ``None`` if the protocol is not specified,
            in which case the server default will be used.
        r   N)r  r  rm   s    rO   r6  zConnectionPool.get_protocol:  s     %))*d;;;r4   c                     t          j                    | _        d| _        g | _        t                      | _        t          j                    | _	        d S )Nr   )
rl  rm  _lock_created_connections_available_connectionsr  _in_use_connectionsr   r   r   rm   s    rO   r  zConnectionPool.resetB  s@    ^%%
$%!&(##&55  9;;r4   c                 Z   | j         t          j                    k    r| j                            d          }|st
          	 | j         t          j                    k    r|                                  | j                                         d S # | j                                         w xY wd S )N   )r   )r   r   r   r  acquirer#   r  release)rJ   acquireds     rO   	_checkpidzConnectionPool._checkpidS  s    F 8ry{{""..q.99H +***8ry{{**JJLLL'')))))'')))) #"s   0B B(command_namer@  c                    |                                   | j        5  	 | j                                        }n$# t          $ r |                                 }Y nw xY w| j                            |           ddd           n# 1 swxY w Y   	 |                                 	 |	                                r| j
        t          d          nb# t          t          f$ rN |                                 |                                 |	                                rt          d          Y nw xY wn$# t          $ r |                     |            w xY w|S )zGet a connection from the poolNConnection has dataConnection not ready)r$  r  r  r  
IndexErrormake_connectionr  addr{   r   rf  r$   r   r   r#  r"  rJ   r%  r  options
connections        rO   get_connectionzConnectionPool.get_connection  s   Z 	5 	54!8<<>>

 4 4 4!1133


4$((444	5 	5 	5 	5 	5 	5 	5 	5 	5 	5 	5 	5 	5 	5 	5	   
B&&(( ATZ-?)*?@@@#W- B B B%%'''""$$$&&(( B)*@AAAB BB
  	 	 	 LL$$$		 sb   B8BABABB	BD1 #*C D1 AD-*D1 ,D--D1 1!Ec                     | j         }t          |                    dd          |                    dd          |                    dd                    S )z,Return an encoder based on encoding settingsr   r   r   r   r   F)r   r   r   )r  r   r  )rJ   r   s     rO   get_encoderzConnectionPool.get_encoder  sV    'ZZ
G44"JJ'8(CC#ZZ(:EBB
 
 
 	
r4   rk   c                     | j         | j        k    rt          d          | xj         dz  c_         | j        +t	           | j        di | j        | j        | j                  S  | j        di | j        S )zCreate a new connectionzToo many connectionsr   NrT   )r  r  r$   rf  rc  r  r  r  rm   s    rO   r*  zConnectionPool.make_connection  s    $(<<<!"8999!!Q&!!:!'%%??(>??TZ   %t$>>t'=>>>r4   r.  c                    |                                   | j        5  	 | j                            |           n# t          $ r Y nw xY w|                     |          r| j                            |           n2| xj        dz  c_        |	                                 	 ddd           dS 	 ddd           dS # 1 swxY w Y   dS )z(Releases the connection back to the poolr   N)
r$  r  r  r   KeyErrorowns_connectionr  rC   r  r   rJ   r.  s     rO   r"  zConnectionPool.release  sI   Z 	 	(//
;;;;    
 ##J// +22:>>>>
 ))Q.))%%'''!	 	 	 	 	 	 	 	 ?	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	s2   B99B9
AB9AAB99B= B=c                 "    |j         | j         k    S rY   )r   r6  s     rO   r5  zConnectionPool.owns_connection  s    ~))r4   Tinuse_connectionsc                     |                                   | j        5  |rt          | j        | j                  }n| j        }|D ]}|                                 	 ddd           dS # 1 swxY w Y   dS )z
        Disconnects connections in the pool

        If ``inuse_connections`` is True, disconnect connections that are
        current in use, potentially by other threads. Otherwise only disconnect
        connections that are idle in the pool.
        N)r$  r  r   r  r  r   )rJ   r8  connectionsr.  s       rO   r   zConnectionPool.disconnect  s     	Z 		( 		(  :#/1I  #9) ( (
%%''''(		( 		( 		( 		( 		( 		( 		( 		( 		( 		( 		( 		( 		( 		( 		( 		( 		( 		(s   >A((A,/A,c                 .    |                                   dS )z-Close the pool, disconnecting all connectionsNr   rm   s    rO   r  zConnectionPool.close  r  r4   r   r)   c                     | j                             d|i           | j        D ]	}||_        
| j        D ]	}||_        
d S )Nr   )r  r	  r  r   r  )rJ   r   re  s      rO   	set_retryzConnectionPool.set_retry  s^    %%w&6777/ 	 	DDJJ, 	 	DDJJ	 	r4   ri   )rW   rk   )r.  r@  rW   Nr   )r   r)   rW   N)rQ   rR   rS   r;  classmethodr  r@  r   r   r   r\   r?   r   r6  r  r$  r/  r   r1  r*  r"  r5  r=  r   r  r=  rT   r4   rO   r  r    s         . . [.d $)-9=	0 0 "#0   56	0 0 0 0d
3* 
 
 
 
< < <   "-* -* -* -*^3 \    B
W 
 
 
 
? ? ? ?   **, *3 * * * *( (D (D ( ( ( ((        r4   r  c                   J     e Zd ZdZddeef fd	Zd Zd Zd Z	d Z
d	 Z xZS )
BlockingConnectionPoola  
    Thread-safe blocking connection pool::

        >>> from redis.client import Redis
        >>> client = Redis(connection_pool=BlockingConnectionPool())

    It performs the same function as the default
    :py:class:`~redis.ConnectionPool` implementation, in that,
    it maintains a pool of reusable connections that can be shared by
    multiple redis clients (safely across threads if required).

    The difference is that, in the event that a client tries to get a
    connection from the pool when all of connections are in use, rather than
    raising a :py:class:`~redis.ConnectionError` (as the default
    :py:class:`~redis.ConnectionPool` implementation does), it
    makes the client wait ("blocks") for a specified number of seconds until
    a connection becomes available.

    Use ``max_connections`` to increase / decrease the pool size::

        >>> pool = BlockingConnectionPool(max_connections=10)

    Use ``timeout`` to tell it either how many seconds to wait for a connection
    to become available, or to block forever:

        >>> # Block forever.
        >>> pool = BlockingConnectionPool(timeout=None)

        >>> # Raise a ``ConnectionError`` after five seconds if a connection is
        >>> # not available.
        >>> pool = BlockingConnectionPool(timeout=5)
    2      c                 \    || _         || _         t                      j        d||d| d S )N)r  r  rT   )queue_classr   rJ  r\   )rJ   r  r   r  rD  r  r   s         rO   r\   zBlockingConnectionPool.__init__  sS     ' 	
-+	
 	
  	
 	
 	
 	
 	
r4   c                     |                      | j                  | _        	 	 | j                            d            n# t          $ r Y nw xY w-g | _        t          j                    | _        d S rY   )	rD  r  pool
put_nowaitr   _connectionsr   r   r   rm   s    rO   r  zBlockingConnectionPool.reset#  s    $$T%9::			$$T****   	  9;;s   = 
A
	A
c                     | j         ,t           | j        di | j        | j         | j                  }n | j        di | j        }| j                            |           |S )zMake a fresh connection.NrT   )rf  rc  r  r  r  rH  rC   r6  s     rO   r*  z&BlockingConnectionPool.make_connection;  sv    :!-%%??(>??TZ JJ /.HH1GHHJ  ,,,r4   c                 `   |                                   d}	 | j                            d| j                  }n# t          $ r t          d          w xY w||                                 }	 |                                 	 |                                rt          d          nb# t
          t          f$ rN |
                                 |                                 |                                rt          d          Y nw xY wn$# t          $ r |                     |            w xY w|S )a7  
        Get a connection, blocking for ``self.timeout`` until a connection
        is available from the pool.

        If the connection returned is ``None`` then creates a new connection.
        Because we use a last-in first-out queue, the existing connections
        (having been returned to the pool after the initial ``None`` values
        were added) will be returned before ``None`` values. This means we only
        create new connections when we need to, i.e.: the actual number of
        connections will only increase in response to demand.
        NT)blockr   zNo connection available.r'  r(  )r$  rF  r  r   r   r$   r*  r{   r   r   r   r#  r"  r,  s        rO   r/  z%BlockingConnectionPool.get_connectionF  sz    	 
	>T4<HHJJ 	> 	> 	> ""<===	> --//J	   
B&&(( A)*?@@@A#W- B B B%%'''""$$$&&(( B)*@AAAB BB
  	 	 	LL$$$	
 s:   !: A.D
 #B' &D
 'ADD
 DD
 
!D+c                    |                                   |                     |          s0|                                 | j                            d           dS 	 | j                            |           dS # t
          $ r Y dS w xY w)z)Releases the connection back to the pool.N)r$  r5  r   rF  rG  r   r6  s     rO   r"  zBlockingConnectionPool.releasez  s     	##J// 	
 !!###I  &&&F	I  ,,,,, 	 	 	 DD	s   A7 7
BBc                 j    |                                   | j        D ]}|                                 dS )z(Disconnects all connections in the pool.N)r$  rH  r   r6  s     rO   r   z!BlockingConnectionPool.disconnect  sC    + 	$ 	$J!!####	$ 	$r4   )rQ   rR   rS   r;  r@  r   r\   r  r*  r/  r"  r   r`  ra  s   @rO   r@  r@    s         F #
 
 
 
 
 
   0	 	 	2 2 2h  *$ $ $ $ $ $ $r4   r@  )\r   r   r   r  rG   rl  r   abcr   	itertoolsr   queuer   r   r   r   typingr	   r
   r   r   r   r   r   urllib.parser   r   r   redis.cacher   r   r   r   r   r   _parsersr   r   r   r   backoffr   credentialsr   r    
exceptionsr!   r"   r#   r$   r%   r&   r'   r(   r   r)   utilsr*   r+   r,   r-   r.   r/   r0   r1   rD   r`   re   rb   r^   r   objectr   __annotations__r6   r8   rV   rk   r   r@  rc  r  r  r  r  r   r<  listr  r  r  r@  rT   r4   rO   <module>r\     s    				  



 



                  ( ( ( ( ( ( ( ( ( (       C C C C C C C C C C C C C C C C C C 4 4 4 4 4 4 4 4 4 4                J I I I I I I I I I I I       O O O O O O O O	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	      	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	  NNN
	 688E,nDEF F F F !"MM M       $0 0 0 0 0 0 0 0fB B B B B B B BJJ) J) J) J) J), J) J) J)ZC* C* C* C* C*# C* C* C*LE: E: E: E: E:. E: E: E:Pf f f f fJ f f fR    !3   > /   # !  6 6 6rk k k k k k k k\	b$ b$ b$ b$ b$^ b$ b$ b$ b$ b$r4   