uawdijnntqw1x1x1
IP : 216.73.216.110
Hostname : 6.87.74.97.host.secureserver.net
Kernel : Linux 6.87.74.97.host.secureserver.net 4.18.0-553.83.1.el8_10.x86_64 #1 SMP Mon Nov 10 04:22:44 EST 2025 x86_64
Disable Function : None :)
OS : Linux
PATH:
/
home
/
emeraadmin
/
www
/
Classes
/
..
/
node_modules
/
react
/
..
/
raf
/
..
/
function-bind
/
..
/
..
/
4d695
/
urllib.zip
/
/
PK23�\�#�zz)__pycache__/__init__.cpython-36.opt-1.pycnu�[���3 ���i�@sdS)N�rrr�'/usr/lib64/python3.6/urllib/__init__.py�<module>sPK23�\�#�zz)__pycache__/__init__.cpython-36.opt-2.pycnu�[���3 ���i�@sdS)N�rrr�'/usr/lib64/python3.6/urllib/__init__.py�<module>sPK23�\�#�zz#__pycache__/__init__.cpython-36.pycnu�[���3 ���i�@sdS)N�rrr�'/usr/lib64/python3.6/urllib/__init__.py�<module>sPK23�\��d�� � &__pycache__/error.cpython-36.opt-1.pycnu�[���3 \Q �@sPdZddlZdddgZGdd�de�ZGdd�deejj�ZGdd�de�Z dS) a�Exception classes raised by urllib. The base exception class is URLError, which inherits from OSError. It doesn't define any behavior of its own, but is the base class for all exceptions defined in this package. HTTPError is an exception class that is also a valid HTTP response instance. It behaves this way because HTTP protocol errors are valid responses, with a status code, headers, and a body. In some contexts, an application may want to handle an exception like a regular response. �N�URLError� HTTPError�ContentTooShortErrorc@seZdZddd�Zdd�ZdS)rNcCs |f|_||_|dk r||_dS)N)�args�reason�filename)�selfrr�r �$/usr/lib64/python3.6/urllib/error.py�__init__szURLError.__init__cCs d|jS)Nz<urlopen error %s>)r)rr r r �__str__szURLError.__str__)N)�__name__� __module__�__qualname__rrr r r r rs c@sXeZdZdZejjjZdd�Zdd�Z dd�Z edd ��Zed d��Z e jdd��Z d S)rzBRaised when HTTP error occurs, but also acts like non-error returncCs:||_||_||_||_||_|dk r6|j||||�dS)N)�code�msg�hdrs�fpr�_HTTPError__super_init)rZurlrrrrr r r r'szHTTPError.__init__cCsd|j|jfS)NzHTTP Error %s: %s)rr)rr r r r4szHTTPError.__str__cCsd|j|jfS)Nz<HTTPError %s: %r>)rr)rr r r �__repr__7szHTTPError.__repr__cCs|jS)N)r)rr r r r<szHTTPError.reasoncCs|jS)N)r)rr r r �headers@szHTTPError.headerscCs ||_dS)N)r)rrr r r rDsN)r rr�__doc__�urllib�response� addinfourlrrrr�propertyrr�setterr r r r r#s c@seZdZdZdd�ZdS)rzDException raised when downloaded size does not match content-length.cCstj||�||_dS)N)rr�content)r�messagerr r r rKszContentTooShortError.__init__N)r rrrrr r r r rIs) rZurllib.responser�__all__�OSErrorrrrrrr r r r �<module>s &PK23�\���;!!&__pycache__/error.cpython-36.opt-2.pycnu�[���3 \Q �@sLddlZdddgZGdd�de�ZGdd�deejj�ZGdd�de�ZdS)�N�URLError� HTTPError�ContentTooShortErrorc@seZdZddd�Zdd�ZdS)rNcCs |f|_||_|dk r||_dS)N)�args�reason�filename)�selfrr�r �$/usr/lib64/python3.6/urllib/error.py�__init__szURLError.__init__cCs d|jS)Nz<urlopen error %s>)r)rr r r �__str__szURLError.__str__)N)�__name__� __module__�__qualname__rrr r r r rs c@sTeZdZejjjZdd�Zdd�Zdd�Z e dd��Ze d d ��Zej dd ��ZdS) rcCs:||_||_||_||_||_|dk r6|j||||�dS)N)�code�msg�hdrs�fpr�_HTTPError__super_init)rZurlrrrrr r r r'szHTTPError.__init__cCsd|j|jfS)NzHTTP Error %s: %s)rr)rr r r r4szHTTPError.__str__cCsd|j|jfS)Nz<HTTPError %s: %r>)rr)rr r r �__repr__7szHTTPError.__repr__cCs|jS)N)r)rr r r r<szHTTPError.reasoncCs|jS)N)r)rr r r �headers@szHTTPError.headerscCs ||_dS)N)r)rrr r r rDsN)r rr�urllib�response� addinfourlrrrr�propertyrr�setterr r r r r#s c@seZdZdd�ZdS)rcCstj||�||_dS)N)rr�content)r�messagerr r r rKszContentTooShortError.__init__N)r rrrr r r r rIs) Zurllib.responser�__all__�OSErrorrrrrrr r r r �<module>s &PK23�\��d�� � __pycache__/error.cpython-36.pycnu�[���3 \Q �@sPdZddlZdddgZGdd�de�ZGdd�deejj�ZGdd�de�Z dS) a�Exception classes raised by urllib. The base exception class is URLError, which inherits from OSError. It doesn't define any behavior of its own, but is the base class for all exceptions defined in this package. HTTPError is an exception class that is also a valid HTTP response instance. It behaves this way because HTTP protocol errors are valid responses, with a status code, headers, and a body. In some contexts, an application may want to handle an exception like a regular response. �N�URLError� HTTPError�ContentTooShortErrorc@seZdZddd�Zdd�ZdS)rNcCs |f|_||_|dk r||_dS)N)�args�reason�filename)�selfrr�r �$/usr/lib64/python3.6/urllib/error.py�__init__szURLError.__init__cCs d|jS)Nz<urlopen error %s>)r)rr r r �__str__szURLError.__str__)N)�__name__� __module__�__qualname__rrr r r r rs c@sXeZdZdZejjjZdd�Zdd�Z dd�Z edd ��Zed d��Z e jdd��Z d S)rzBRaised when HTTP error occurs, but also acts like non-error returncCs:||_||_||_||_||_|dk r6|j||||�dS)N)�code�msg�hdrs�fpr�_HTTPError__super_init)rZurlrrrrr r r r'szHTTPError.__init__cCsd|j|jfS)NzHTTP Error %s: %s)rr)rr r r r4szHTTPError.__str__cCsd|j|jfS)Nz<HTTPError %s: %r>)rr)rr r r �__repr__7szHTTPError.__repr__cCs|jS)N)r)rr r r r<szHTTPError.reasoncCs|jS)N)r)rr r r �headers@szHTTPError.headerscCs ||_dS)N)r)rrr r r rDsN)r rr�__doc__�urllib�response� addinfourlrrrr�propertyrr�setterr r r r r#s c@seZdZdZdd�ZdS)rzDException raised when downloaded size does not match content-length.cCstj||�||_dS)N)rr�content)r�messagerr r r rKszContentTooShortError.__init__N)r rrrrr r r r rIs) rZurllib.responser�__all__�OSErrorrrrrrr r r r �<module>s &PK23�\�I�����&__pycache__/parse.cpython-36.opt-1.pycnu�[���3 ���ie��@s�dZddlZddlZddlZddlZddlZddddddd d ddd ddddddddddgZddddddddd d!d"d#d$d%d&d'd(d)d*gZdddddd+dddd"d d!d,d#d$d%d-d'd(d&d.d/d0d)d*gZddd1d#ddd d!d$d%d2d3d"d&d4gZ dd1d5d6d+ddd,d2d3g Z ddddd d!d"dd$d%d2d3gZddd1ddd6ddd d!d,dd#g Zd7Z d8Zd9d:d;gZd<ZiZd=d>�Zd?Zd@ZdAdB�ZeefdCdD�ZeefdEdF�ZdGdH�ZGdIdJ�dJe�ZGdKdL�dLe�ZGdMdN�dNe�ZGdOdP�dPee�ZGdQdR�dRee�ZddSlmZeddT�Z eddU�Z!eddV�Z"dWe _dXe j#_dYe j$_dZe!_d[e!j%_d\e!j&_d]e!j'_d^e!j(_d_e!j$_d`e"_e!j%je"j%_e!j&je"j&_e!j'je"j'_dae"j)_e!j(je"j(_e!j$je"j$_eZ*Gdbd�de e�Z+Gdcd�de!e�Z,Gddd�de"e�Z-Gded�de e�Z.Gdfd�de!e�Z/Gdgd�de"e�Z0dhdi�Z1e1�[1d�dkd�Z2dldm�Z3d�dndo�Z4dpdq�Z5drds�Z6dtdu�Z7dvdw�Z8d�dxd�Z9dyd�Z:dzd�Z;d�d{d�Z<d|d�Z=d}Z>da?d~d�Z@ejAd�ZBd�d�d�ZCd�d�d �ZDGd�d��d�eE�ZFd�ZGdaHd�d�d�ZId�d�d�ZJeKd��ZLeMeL�ZNiZOGd�d��d�ejP�ZQd�d�d�ZRd�d�d �ZSd�d�d�ZTd�dddeSfd�d �ZUd�d��ZVd�d��ZWdaXd�d��ZYdaZd�d��Z[d�d��Z\d�d��Z]da^d�d��Z_d�d�d��Z`d�d��Zad�d��Zbd�d��Zcd�d��ZddS)�a3Parse (absolute and relative) URLs. urlparse module is based upon the following RFC specifications. RFC 3986 (STD66): "Uniform Resource Identifiers" by T. Berners-Lee, R. Fielding and L. Masinter, January 2005. RFC 2732 : "Format for Literal IPv6 Addresses in URL's by R.Hinden, B.Carpenter and L.Masinter, December 1999. RFC 2396: "Uniform Resource Identifiers (URI)": Generic Syntax by T. Berners-Lee, R. Fielding, and L. Masinter, August 1998. RFC 2368: "The mailto URL scheme", by P.Hoffman , L Masinter, J. Zawinski, July 1998. RFC 1808: "Relative Uniform Resource Locators", by R. Fielding, UC Irvine, June 1995. RFC 1738: "Uniform Resource Locators (URL)" by T. Berners-Lee, L. Masinter, M. McCahill, December 1994 RFC 3986 is considered the current standard and any future changes to urlparse module should conform with it. The urlparse module is currently not entirely compliant with this RFC due to defacto scenarios for parsing, and for backward compatibility purposes, some parsing quirks from older RFCs are retained. The testcases in test_urlparse.py provides a good indicator of parsing behavior. The WHATWG URL Parser spec should also be considered. We are not compliant with it either due to existing user code API behavior expectations (Hyrum's Law). It serves as a useful guide when making changes. �N�urlparse� urlunparse�urljoin� urldefrag�urlsplit� urlunsplit� urlencode�parse_qs� parse_qsl�quote� quote_plus�quote_from_bytes�unquote�unquote_plus�unquote_to_bytes�DefragResult�ParseResult�SplitResult�DefragResultBytes�ParseResultBytes�SplitResultBytes�Zftp�httpZgopherZnntpZimapZwais�fileZhttpsZshttpZmmsZprosperoZrtspZrtspuZsftpZsvnzsvn+sshZwsZwssZtelnetZsnewsZrsyncZnfsZgitzgit+sshZhdlZsipZsipsZtelZmailtoZnewszAabcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+-.z! � � � �cCstj�tj�dS)z,Clear the parse cache and the quoters cache.N)�_parse_cache�clear� _safe_quoters�r!r!�$/usr/lib64/python3.6/urllib/parse.py�clear_cache`sr#�ascii�strictcCs|S)Nr!)�objr!r!r"�_nooposr'cCs|j||�S)N)�encode)r&�encoding�errorsr!r!r"�_encode_resultrsr+cst��fdd�|D��S)Nc3s"|]}|r|j���ndVqdS)rN)�decode)�.0�x)r)r*r!r"� <genexpr>xsz_decode_args.<locals>.<genexpr>)�tuple)�argsr)r*r!)r)r*r"�_decode_argsvsr2cGsZt|dt�}x.|dd�D]}|rt|t�|krtd��qW|rL|tfSt|�tfS)Nr�z$Cannot mix str and non-str arguments)� isinstance�str� TypeErrorr'r2r+)r1Z str_input�argr!r!r"�_coerce_argszs r8c@seZdZdZfZddd�ZdS)�_ResultMixinStrz>Standard approach to encoding parsed results from str to bytesr$r%cs|j��fdd�|D��S)Nc3s|]}|j���VqdS)N)r()r-r.)r)r*r!r"r/�sz)_ResultMixinStr.encode.<locals>.<genexpr>)�_encoded_counterpart)�selfr)r*r!)r)r*r"r(�sz_ResultMixinStr.encodeN)r$r%)�__name__� __module__�__qualname__�__doc__� __slots__r(r!r!r!r"r9�sr9c@seZdZdZfZddd�ZdS)�_ResultMixinBytesz>Standard approach to decoding parsed results from bytes to strr$r%cs|j��fdd�|D��S)Nc3s|]}|j���VqdS)N)r,)r-r.)r)r*r!r"r/�sz+_ResultMixinBytes.decode.<locals>.<genexpr>)�_decoded_counterpart)r;r)r*r!)r)r*r"r,�sz_ResultMixinBytes.decodeN)r$r%)r<r=r>r?r@r,r!r!r!r"rA�srAc@sDeZdZdZfZedd��Zedd��Zedd��Zedd ��Z d S)�_NetlocResultMixinBasezHShared methods for the parsed result objects containing a netloc elementcCs |jdS)Nr)� _userinfo)r;r!r!r"�username�sz_NetlocResultMixinBase.usernamecCs |jdS)Nr3)rD)r;r!r!r"�password�sz_NetlocResultMixinBase.passwordcCsD|jd}|sdSt|t�r dnd}|j|�\}}}|j�||S)Nr�%�%)� _hostinfor4r5� partition�lower)r;�hostname� separatorZpercentZzoner!r!r"rL�s z_NetlocResultMixinBase.hostnamecCs@|jd}|dk r<t|d�}d|ko.dkns<td��|S)Nr3� ri��zPort out of range 0-65535)rI�int� ValueError)r;�portr!r!r"rQ�s z_NetlocResultMixinBase.portN) r<r=r>r?r@�propertyrErFrLrQr!r!r!r"rC�srCc@s(eZdZfZedd��Zedd��ZdS)�_NetlocResultMixinStrcCsD|j}|jd�\}}}|r4|jd�\}}}|s<d}nd}}||fS)N�@�:)�netloc� rpartitionrJ)r;rV�userinfo� have_info�hostinforE� have_passwordrFr!r!r"rD�sz_NetlocResultMixinStr._userinfocCsl|j}|jd�\}}}|jd�\}}}|rL|jd�\}}}|jd�\}}}n|jd�\}}}|sdd}||fS)NrT�[�]rU)rVrWrJ)r;rV�_rZ�have_open_br� bracketedrLrQr!r!r"rI�sz_NetlocResultMixinStr._hostinfoN)r<r=r>r@rRrDrIr!r!r!r"rS�srSc@s(eZdZfZedd��Zedd��ZdS)�_NetlocResultMixinBytescCsD|j}|jd�\}}}|r4|jd�\}}}|s<d}nd}}||fS)N�@�:)rVrWrJ)r;rVrXrYrZrEr[rFr!r!r"rD�sz!_NetlocResultMixinBytes._userinfocCsl|j}|jd�\}}}|jd�\}}}|rL|jd�\}}}|jd�\}}}n|jd�\}}}|sdd}||fS)Nrb�[�]rc)rVrWrJ)r;rVr^rZr_r`rLrQr!r!r"rI�sz!_NetlocResultMixinBytes._hostinfoN)r<r=r>r@rRrDrIr!r!r!r"ra�sra)� namedtuplezurl fragmentz!scheme netloc path query fragmentz(scheme netloc path params query fragmentz� DefragResult(url, fragment) A 2-tuple that contains the url without fragment identifier and the fragment identifier as a separate argument. z$The URL with no fragment identifier.z� Fragment identifier separated from URL, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. z� SplitResult(scheme, netloc, path, query, fragment) A 5-tuple that contains the different components of a URL. Similar to ParseResult, but does not split params. z%Specifies URL scheme for the request.z0 Network location where the request is made to. z@ The hierarchical path, such as the path to a file to download. z� The query component, that contains non-hierarchical data, that along with data in path component, identifies a resource in the scope of URI's scheme and network location. z� Fragment identifier, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. zr ParseResult(scheme, netloc, path, params, query, fragment) A 6-tuple that contains components of a parsed URL. z� Parameters for last path element used to dereference the URI in order to provide access to perform some operation on the resource. c@seZdZfZdd�ZdS)rcCs |jr|jd|jS|jSdS)N�#)�fragment�url)r;r!r!r"�geturlEszDefragResult.geturlN)r<r=r>r@rjr!r!r!r"rCsc@seZdZfZdd�ZdS)rcCst|�S)N)r)r;r!r!r"rjMszSplitResult.geturlN)r<r=r>r@rjr!r!r!r"rKsc@seZdZfZdd�ZdS)rcCst|�S)N)r)r;r!r!r"rjRszParseResult.geturlN)r<r=r>r@rjr!r!r!r"rPsc@seZdZfZdd�ZdS)rcCs |jr|jd|jS|jSdS)N�#)rhri)r;r!r!r"rjXszDefragResultBytes.geturlN)r<r=r>r@rjr!r!r!r"rVsc@seZdZfZdd�ZdS)rcCst|�S)N)r)r;r!r!r"rj`szSplitResultBytes.geturlN)r<r=r>r@rjr!r!r!r"r^sc@seZdZfZdd�ZdS)rcCst|�S)N)r)r;r!r!r"rjeszParseResultBytes.geturlN)r<r=r>r@rjr!r!r!r"rcscCs8ttfttfttff}x|D]\}}||_||_qWdS)N)rrrrrrr:rB)Z _result_pairsZ_decodedZ_encodedr!r!r"�_fix_result_transcodingis rlTc Csft||�\}}}t|||�}|\}}}}}|tkrHd|krHt|�\}}nd}t||||||�} || �S)a#Parse a URL into 6 components: <scheme>://<netloc>/<path>;<params>?<query>#<fragment> Return a 6-tuple: (scheme, netloc, path, params, query, fragment). Note that we don't break the components up in smaller bits (e.g. netloc is a single string) and we don't expand % escapes.�;r)r8r�uses_params�_splitparamsr) ri�scheme�allow_fragments�_coerce_resultZsplitresultrV�queryrh�params�resultr!r!r"rvscCsRd|kr,|jd|jd��}|dkr6|dfSn |jd�}|d|�||dd�fS)N�/rmrrr3)�find�rfind)ri�ir!r!r"ro�s rocCsLt|�}x*dD]"}|j||�}|dkrt||�}qW|||�||d�fS)Nz/?#r)�lenrw�min)ri�start�delim�cZwdelimr!r!r"�_splitnetloc�s rcCs�|stdd�|D��rdSddl}|jdd�}|jdd�}|jdd�}|jdd�}|jd |�}||krndSx(d D] }||krttd|dd ��qtWdS)Ncss|]}t|�dkVqdS)�N)�ord)r-r~r!r!r"r/�sz_checknetloc.<locals>.<genexpr>rrTrrUrg�?�NFKCz/?#@:znetloc 'z' contains invalid z#characters under NFKC normalization)�any�unicodedata�replace� normalizerP)rVr��nZnetloc2r~r!r!r"�_checknetloc�s r�cCsxtD]}|j|d�}qW|S)Nr)�_UNSAFE_URL_BYTES_TO_REMOVEr�)ri�br!r!r"�_remove_unsafe_bytes_from_url�s r�cCst|jd�d}|jd�\}}}|rX|r.td��|jd�\}}}|rh|jd�rhtd��n|jd�\}}}t|�dS)NrT�r\zInvalid IPv6 URLr]rU)rWrJrP� startswith�_check_bracketed_host)rVZhostname_and_portZbefore_bracketr_r`rLr^rQr!r!r"�_check_bracketed_netloc�s r�cCsB|jd�r tjd|�s>td��ntj|�}t|tj�r>td��dS)N�vz\Av[a-fA-F0-9]+\..+\ZzIPvFuture address is invalidz%An IPv4 address cannot be in brackets)r��re�matchrP� ipaddressZ ip_addressr4ZIPv4Address)rLZipr!r!r"r��s r�c Cs�t||�\}}}t|�}t|�}|jt�}|jt�}t|�}|||t|�t|�f}tj|d�}|rj||�St t�t kr|t�d}}}|jd�} | dk�r�|d| �dk�r~|d| �j �}|| dd�}|dd�dk�rt|d�\}}d |ko�d |k�sd |k�rd |k�rtd��|�r<d|k�r<|jdd�\}}d |k�rV|jd d�\}}t|�t|||||�} | t|<|| �Sxd|d| �D]}|tk�r�P�q�W|| dd�}|�s�tdd�|D���r�|d| �j �|}}|dd�dk�rPt|d�\}}d |k�rd |k�s,d |k�r4d |k�r4td��d |k�rPd |k�rPt|�|�rpd|k�rp|jdd�\}}d |k�r�|jd d�\}}t|�t|||||�} | t|<|| �S)aParse a URL into 5 components: <scheme>://<netloc>/<path>?<query>#<fragment> Return a 5-tuple: (scheme, netloc, path, query, fragment). Note that we don't break the components up in smaller bits (e.g. netloc is a single string) and we don't expand % escapes.NrrUrrr3r�z//r\r]zInvalid IPv6 URLrgr�css|]}|dkVqdS)� 0123456789Nr!)r-r~r!r!r"r/�szurlsplit.<locals>.<genexpr>)r8r��lstrip�_WHATWG_C0_CONTROL_OR_SPACE�strip�bool�typer�getrz�MAX_CACHE_SIZEr#rwrKrrP�splitr�r�scheme_charsr�r�) rirprqrr�key�cachedrVrsrhryr�r~�restr!r!r"r�sh cCs<t|�\}}}}}}}|r&d||f}|t|||||f��S)z�Put a parsed URL back together again. This may result in a slightly different, but equivalent URL, if the URL that was parsed originally had redundant delimiters, e.g. a ? with an empty query (the draft states that these are equivalent).z%s;%s)r8r)� componentsrprVrirtrsrhrrr!r!r"rscCs�t|�\}}}}}}|s4|r`|tkr`|dd�dkr`|rP|dd�dkrPd|}d|pXd|}|rp|d|}|r�|d|}|r�|d |}||�S) akCombine the elements of a tuple as returned by urlsplit() into a complete URL as a string. The data argument can be any five-item iterable. This may result in a slightly different, but equivalent URL, if the URL that was parsed originally had unnecessary delimiters (for example, a ? with an empty query; the RFC states that these are equivalent).Nr�z//r3rvrrUr�rg)r8�uses_netloc)r�rprVrirsrhrrr!r!r"rs cCs�|s|S|s|St||�\}}}t|d|�\}}}}}} t|||�\} }}} }}| |ks`| tkrh||�S| tkr�|r�|t| ||| ||f��S|}|r�| r�|}|} |s�|}|t| ||| ||f��S|jd�}|ddkr�|d=|dd�dk�r�|jd�}n(||jd�}td|dd ��|dd �<g}xX|D]P}|dk�rdy|j�Wntk �r`YnXn|dk�rt�q0n |j |��q0W|ddk�r�|j d�|t| |dj |��p�d| ||f��S) zaJoin a base URL and a possibly relative URL to form an absolute interpretation of the latter.rrvr3N�..�.���r�r�r�r�)r�r�)r8r� uses_relativer�rr��filter�pop� IndexError�append�join)�baserirqrrZbschemeZbnetlocZbpathZbparamsZbqueryZ bfragmentrprV�pathrtrsrhZ base_partsZsegmentsZ resolved_pathZsegr!r!r"r*sT c CsTt|�\}}d|kr>t|�\}}}}}}t|||||df�}nd}|}|t||��S)z�Removes any existing fragment from URL. Returns a tuple of the defragmented URL and the fragment. If the URL contained no fragments, the second element is the empty string. rgr)r8rrr) rirr�sr��p�a�qZfragZdefragr!r!r"rosZ0123456789ABCDEFabcdefcCs�|s|jdSt|t�r"|jd�}|jd�}t|�dkr<|S|dg}|j}tdkrbdd�tD�axb|dd�D]R}y(|t|dd ��||d d��Wqptk r�|d�||�YqpXqpWdj |�S) z,unquote_to_bytes('abc%20def') -> b'abc def'.�zutf-8rHr3rNcSs4i|],}tD]"}tt||d�g�||j��qqS)�)�_hexdig�bytesrOr()r-r�r�r!r!r"� <dictcomp>�sz$unquote_to_bytes.<locals>.<dictcomp>r�) r�r4r5r(rzr�� _hextobyter��KeyErrorr�)�string�bits�resr��itemr!r!r"r�s* z([-]+)�utf-8r�cCs�d|kr|j|S|dkrd}|dkr*d}tj|�}|dg}|j}x@tdt|�d�D],}|t||�j||��|||d�qVWdj|�S) a�Replace %xx escapes by their single-character equivalent. The optional encoding and errors parameters specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. By default, percent-encoded sequences are decoded with UTF-8, and invalid sequences are replaced by a placeholder character. unquote('abc%20def') -> 'abc def'. rGNzutf-8r�rr3r�r)r��_asciirer��rangerzrr,r�)r�r)r*r�r�r�ryr!r!r"r�s Fc CsRi}t|||||||d�}x2|D]*\} } | |kr@|| j| �q | g|| <q W|S)a�Parse a query given as a string argument. Arguments: qs: percent-encoded query string to be parsed keep_blank_values: flag indicating whether blank values in percent-encoded queries should be treated as blank strings. A true value indicates that blanks should be retained as blank strings. The default false value indicates that blank values are to be ignored and treated as if they were not included. strict_parsing: flag indicating what to do with parsing errors. If false (the default), errors are silently ignored. If true, errors raise a ValueError exception. encoding and errors: specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. max_num_fields: int. If set, then throws a ValueError if there are more than n fields read by parse_qsl(). Returns a dictionary. )r)r*�max_num_fieldsrM)r r�)�qs�keep_blank_values�strict_parsingr)r*r�rMZ parsed_result�pairs�name�valuer!r!r"r �sc@seZdZdZdS)�_QueryStringSeparatorWarningz>Warning for using default `separator` in parse_qs or parse_qslN)r<r=r>r?r!r!r!r"r��sr�z/etc/python/urllib.cfgcCs�t|�\}}t|t�r |jd�}|s6t|ttf�rF|dk rFtd��t�}|dk�rVt}d} |dkrvtj j | �}d} |dkr�ytt�}Wnt k r�YnJX|�:ddl}|jdd'd�} | j|�| j d | dd �}|aWdQRXt} |dk�rd|k�rddlm}|d-tdd�d}n:|dk�r,|}n*t|�dk�rVt| �d| �d�dd��|dk �r�||k�r�d|jd�|jd�}nd|j|�}||k�r�td��||k�r�dd�|jd�D�}nd d�|j|�D�}g}x�|D]�}|�r�|�r��q�|jd!d�}t|�dk�r>|�r(td"|f��|�r�|jd#�n�q�t|d��sR|�r�|djd$d%�}t|||d&�}||�}|djd$d%�}t|||d&�}||�}|j||f��q�W|S).a�Parse a query given as a string argument. Arguments: qs: percent-encoded query string to be parsed keep_blank_values: flag indicating whether blank values in percent-encoded queries should be treated as blank strings. A true value indicates that blanks should be retained as blank strings. The default false value indicates that blank values are to be ignored and treated as if they were not included. strict_parsing: flag indicating what to do with parsing errors. If false (the default), errors are silently ignored. If true, errors raise a ValueError exception. encoding and errors: specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. max_num_fields: int. If set, then throws a ValueError if there are more than n fields read by parse_qsl(). Returns a list, as G-d intended. r$Nz*Separator must be of type string or bytes.ZPYTHON_URLLIB_QS_SEPARATORzenvironment variablerrg)Z interpolationZcomment_prefixesr )Zfallbackrm)�warnz4The default separator of urllib.parse.parse_qsl and z1parse_qs was changed to '&' to avoid a web cache z"poisoning issue (CVE-2021-23336). z4By default, semicolons no longer act as query field zseparators. z3See https://access.redhat.com/articles/5860431 for z more details.r�)� stacklevel�&Zlegacyr3z (from z) must contain z1 character, or "legacy". See z<https://access.redhat.com/articles/5860431 for more details.zMax number of fields exceededcSs g|]}|jd�D]}|�qqS)rm)r�)r-�s1�s2r!r!r"� <listcomp>Lszparse_qsl.<locals>.<listcomp>cSsg|]}|�qSr!r!)r-r�r!r!r"r�Ns�=zbad query field: %rr�+� )r)r*)rgzeThe default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache z�The default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). z�The default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). By default, semicolons no longer act as query field z�The default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). By default, semicolons no longer act as query field separators. z�The default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). By default, semicolons no longer act as query field separators. See https://access.redhat.com/articles/5860431 for aThe default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). By default, semicolons no longer act as query field separators. See https://access.redhat.com/articles/5860431 for more details.)r8r4r�r,r5rP�object�_default_qs_separator�os�environr��open�_QS_SEPARATOR_CONFIG_FILENAME�FileNotFoundError�configparserZConfigParserZ read_file�warningsr�r�rz�countr�r�r�r)r�r�r�r)r*r�rMrrZ_legacyZenvvar_nameZ config_sourcerr��configr�� num_fieldsr��rZ name_valueZnvr�r�r!r!r"r �s� cCs|jdd�}t|||�S)z�Like unquote(), but also replace plus signs by spaces, as required for unquoting HTML form values. unquote_plus('%7e/abc+def') -> '~/abc def' r�r�)r�r)r�r)r*r!r!r"rfssAABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_.-c@s(eZdZdZdd�Zdd�Zdd�ZdS) �Quoterz�A mapping from bytes (in range(0,256)) to strings. String values are percent-encoded byte values, unless the key < 128, and in the "safe" set (either the specified safe set, or default set). cCstj|�|_dS)zsafe: bytes object.N)�_ALWAYS_SAFE�union�safe)r;r�r!r!r"�__init__~szQuoter.__init__cCsd|jjt|�fS)Nz<%s %r>)� __class__r<�dict)r;r!r!r"�__repr__�szQuoter.__repr__cCs(||jkrt|�ndj|�}|||<|S)Nz%{:02X})r��chr�format)r;r�r�r!r!r"�__missing__�szQuoter.__missing__N)r<r=r>r?r�r�r�r!r!r!r"r�vsr�rvcCsbt|t�r8|s|S|dkrd}|dkr*d}|j||�}n |dk rHtd��|dk rXtd��t||�S)a�quote('abc def') -> 'abc%20def' Each part of a URL, e.g. the path info, the query, etc., has a different set of reserved characters that must be quoted. RFC 2396 Uniform Resource Identifiers (URI): Generic Syntax lists the following reserved characters. reserved = ";" | "/" | "?" | ":" | "@" | "&" | "=" | "+" | "$" | "," Each of these characters is reserved in some component of a URL, but not necessarily in all of them. By default, the quote function is intended for quoting the path section of a URL. Thus, it will not encode '/'. This character is reserved, but in typical usage the quote function is being called on a path where the existing slash characters are used as reserved characters. string and safe may be either str or bytes objects. encoding and errors must not be specified if string is a bytes object. The optional encoding and errors parameters specify how to deal with non-ASCII characters, as accepted by the str.encode method. By default, encoding='utf-8' (characters are encoded with UTF-8), and errors='strict' (unsupported characters raise a UnicodeEncodeError). Nzutf-8r%z,quote() doesn't support 'encoding' for bytesz*quote() doesn't support 'errors' for bytes)r4r5r(r6r )r�r�r)r*r!r!r"r�s cCsdt|t�rd|ks$t|t�r2d|kr2t||||�St|t�rBd}nd}t|||||�}|jdd�S)z�Like quote(), but also replace ' ' with '+', as required for quoting HTML form values. Plus signs in the original string are escaped unless they are included in safe. It also does not have safe default to '/'. r�� r�)r4r5r�rr�)r�r�r)r*Zspacer!r!r"r�s cs�t|ttf�std��|sdSt|t�r6|jdd�}ntdd�|D��}|jt|�s^|j�Syt |�Wn&t k r�t|�jt |<�YnXdj �fdd�|D��S)z�Like quote(), but accepts a bytes object rather than a str, and does not perform string-to-bytes encoding. It always returns an ASCII string. quote_from_bytes(b'abc def?') -> 'abc%20def%3f' z!quote_from_bytes() expected bytesrr$�ignorecSsg|]}|dkr|�qS)�r!)r-r~r!r!r"r��sz$quote_from_bytes.<locals>.<listcomp>csg|]}�|��qSr!r!)r-�char)�quoterr!r"r��s)r4r�� bytearrayr6r5r(�rstrip�_ALWAYS_SAFE_BYTESr,r r�r��__getitem__r�)Zbsr�r!)r�r"r �s cCst|d�r|j�}nRy t|�r2t|dt�r2t�Wn0tk rdtj�\}}}td�j|��YnXg} |s�xr|D]j\} }t| t �r�|| |�} n|t | �|||�} t|t �r�|||�}n|t |�|||�}| j| d|�qtW�n,�x(|D�]\} }t| t ��r|| |�} n|t | �|||�} t|t ��rL|||�}| j| d|�q�t|t ��rz|||||�}| j| d|�q�yt|�}Wn:tk �r�|t |�|||�}| j| d|�Yq�XxJ|D]B} t| t ��r�|| |�} n|t | �|||�} | j| d| ��q�Wq�Wdj| �S)a^Encode a dict or sequence of two-element tuples into a URL query string. If any values in the query arg are sequences and doseq is true, each sequence element is converted to a separate parameter. If the query arg is a sequence of two-element tuples, the order of the parameters in the output will match the order of parameters in the input. The components of a query arg may each be either a string or a bytes type. The safe, encoding, and errors parameters are passed down to the function specified by quote_via (encoding and errors only if a component is a str). �itemsrz1not a valid non-string sequence or mapping objectr�r�) �hasattrr�rzr4r0r6�sys�exc_info�with_tracebackr�r5r�r�)rsZdoseqr�r)r*Z quote_viaZtyZva�tb�l�kr�r.Zeltr!r!r"r�sP cCsJt|t�rFy|jd�j�}Wn(tk rDtdt|�d��YnX|S)zto_bytes(u"URL") --> 'URL'.�ASCIIzURL z contains non-ASCII characters)r4r5r(r,�UnicodeError�repr)rir!r!r"�to_bytes/s r�cCs`t|�j�}|dd�dkr<|dd�dkr<|dd�j�}|dd�dkr\|dd�j�}|S) z8unwrap('<URL:type://host/path>') --> 'type://host/path'.Nr3�<�>�zURL:r�r�)r5r�)rir!r!r"�unwrap<s rcCsDtdkrtjdtj�atj|�}|r<|j�\}}|j�|fSd|fS)z:splittype('type:opaquestring') --> 'type', 'opaquestring'.Nz ([^/:]+):(.*))� _typeprogr��compile�DOTALLr��groupsrK)rir�rp�datar!r!r"� splittypeEs rcCsXtdkrtjdtj�atj|�}|rP|j�\}}|rH|ddkrHd|}||fSd|fS)z;splithost('//host[:port]/path') --> 'host[:port]', '/path'.Nz//([^/#?]*)(.*)rrv)� _hostprogr�rrr�r)rir�Z host_portr�r!r!r"� splithostRs r cCs |jd�\}}}|r|nd|fS)zJsplituser('user[:passwd]@host[:port]') --> 'user[:passwd]', 'host[:port]'.rTN)rW)�host�userr}r!r!r"� splituser`sr cCs |jd�\}}}||r|ndfS)z/splitpasswd('user:passwd') -> 'user', 'passwd'.rUN)rJ)rr}Zpasswdr!r!r"�splitpasswdesrcCsDtdkrtjdtj�atj|�}|r<|j�\}}|r<||fS|dfS)z*splitport('host:port') --> 'host', 'port'.Nz(.*):([0-9]*)$)� _portprogr�rrr�r)rr�rQr!r!r"� splitportls rr3cCsT|jd�\}}}|s|}n2|rLyt|�}Wntk rBd}YnX||fS||fS)z�Split host and port, returning numeric port. Return given default port if no ':' found; defaults to -1. Return numerical port if a valid number are found after ':'. Return None if ':' but not a valid number.rUN)rWrOrP)rZdefportr}rQZnportr!r!r"� splitnportys rcCs$|jd�\}}}|r||fS|dfS)z/splitquery('/path?query') --> '/path', 'query'.r�N)rW)rir�r}rsr!r!r"� splitquery�srcCs$|jd�\}}}|r||fS|dfS)z)splittag('/path#tag') --> '/path', 'tag'.rgN)rW)rir�r}�tagr!r!r"�splittag�srcCs|jd�}|d|dd�fS)zksplitattr('/path;attr1=value1;attr2=value2;...') -> '/path', ['attr1=value1', 'attr2=value2', ...].rmrr3N)r�)riZwordsr!r!r"� splitattr�s rcCs |jd�\}}}||r|ndfS)z-splitvalue('attr=value') --> 'attr', 'value'.r�N)rJ)�attrr}r�r!r!r"� splitvalue�sr)rT)r)rT)T)r�r�)FFr�r�NN)FFr�r�NN)r�r�)rvNN)rNN)rvr�)r�)er?r�r�r��collectionsr��__all__r�r�rnZnon_hierarchicalZ uses_queryZ uses_fragmentr�r�r�r�rr#Z_implicit_encodingZ_implicit_errorsr'r+r2r8r�r9rArCrSrarfZ_DefragResultBaseZ_SplitResultBaseZ_ParseResultBaserirhrprVr�rsrtZ ResultBaserrrrrrrlrrorr�r�r�r�rrrrrr�r�rrr�rr �RuntimeWarningr�r�r�r r� frozensetr�r�r�r �defaultdictr�rrr rr�rrrr r r rrrrrrrrr!r!r!r"�<module> s ! B E & y , O PK23�\>edd&__pycache__/parse.cpython-36.opt-2.pycnu�[���3 ���ie��@s�ddlZddlZddlZddlZddlZdddddddd d ddd dddddddddgZdddddddddd d!d"d#d$d%d&d'd(d)gZdddddd*dddd!dd d+d"d#d$d,d&d'd%d-d.d/d(d)gZddd0d"dddd d#d$d1d2d!d%d3gZdd0d4d5d*ddd+d1d2g Z dddddd d!dd#d$d1d2gZ ddd0ddd5dddd d+dd"g Zd6Zd7Z d8d9d:gZd;ZiZd<d=�Zd>Zd?Zd@dA�ZeefdBdC�ZeefdDdE�ZdFdG�ZGdHdI�dIe�ZGdJdK�dKe�ZGdLdM�dMe�ZGdNdO�dOee�ZGdPdQ�dQee�ZddRlmZeddS�ZeddT�Z eddU�Z!dVe_"dWej#_"dXej$_"dYe _"dZe j%_"d[e j&_"d\e j'_"d]e j(_"d^e j$_"d_e!_"e j%j"e!j%_"e j&j"e!j&_"e j'j"e!j'_"d`e!j)_"e j(j"e!j(_"e j$j"e!j$_"eZ*Gdad�dee�Z+Gdbd�de e�Z,Gdcd�de!e�Z-Gddd�dee�Z.Gded�de e�Z/Gdfd�de!e�Z0dgdh�Z1e1�[1d�djd�Z2dkdl�Z3d�dmdn�Z4dodp�Z5dqdr�Z6dsdt�Z7dudv�Z8d�dwd�Z9dxd�Z:dyd�Z;d�dzd�Z<d{d�Z=d|Z>da?d}d�Z@ejAd~�ZBd�d�d�ZCd�d�d �ZDGd�d��d�eE�ZFd�ZGdaHd�d�d �ZId�d�d�ZJeKd��ZLeMeL�ZNiZOGd�d��d�ejP�ZQd�d�d�ZRd�d�d�ZSd�d�d �ZTd�dddeSfd�d�ZUd�d��ZVd�d��ZWdaXd�d��ZYdaZd�d��Z[d�d��Z\d�d��Z]da^d�d��Z_d�d�d��Z`d�d��Zad�d��Zbd�d��Zcd�d��ZddS)��N�urlparse� urlunparse�urljoin� urldefrag�urlsplit� urlunsplit� urlencode�parse_qs� parse_qsl�quote� quote_plus�quote_from_bytes�unquote�unquote_plus�unquote_to_bytes�DefragResult�ParseResult�SplitResult�DefragResultBytes�ParseResultBytes�SplitResultBytes�Zftp�httpZgopherZnntpZimapZwais�fileZhttpsZshttpZmmsZprosperoZrtspZrtspuZsftpZsvnzsvn+sshZwsZwssZtelnetZsnewsZrsyncZnfsZgitzgit+sshZhdlZsipZsipsZtelZmailtoZnewszAabcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+-.z! � � � �cCstj�tj�dS)N)�_parse_cache�clear� _safe_quoters�r!r!�$/usr/lib64/python3.6/urllib/parse.py�clear_cache`sr#�ascii�strictcCs|S)Nr!)�objr!r!r"�_nooposr'cCs|j||�S)N)�encode)r&�encoding�errorsr!r!r"�_encode_resultrsr+cst��fdd�|D��S)Nc3s"|]}|r|j���ndVqdS)rN)�decode)�.0�x)r)r*r!r"� <genexpr>xsz_decode_args.<locals>.<genexpr>)�tuple)�argsr)r*r!)r)r*r"�_decode_argsvsr2cGsZt|dt�}x.|dd�D]}|rt|t�|krtd��qW|rL|tfSt|�tfS)Nr�z$Cannot mix str and non-str arguments)� isinstance�str� TypeErrorr'r2r+)r1Z str_input�argr!r!r"�_coerce_argszs r8c@seZdZfZddd�ZdS)�_ResultMixinStrr$r%cs|j��fdd�|D��S)Nc3s|]}|j���VqdS)N)r()r-r.)r)r*r!r"r/�sz)_ResultMixinStr.encode.<locals>.<genexpr>)�_encoded_counterpart)�selfr)r*r!)r)r*r"r(�sz_ResultMixinStr.encodeN)r$r%)�__name__� __module__�__qualname__� __slots__r(r!r!r!r"r9�sr9c@seZdZfZddd�ZdS)�_ResultMixinBytesr$r%cs|j��fdd�|D��S)Nc3s|]}|j���VqdS)N)r,)r-r.)r)r*r!r"r/�sz+_ResultMixinBytes.decode.<locals>.<genexpr>)�_decoded_counterpart)r;r)r*r!)r)r*r"r,�sz_ResultMixinBytes.decodeN)r$r%)r<r=r>r?r,r!r!r!r"r@�sr@c@s@eZdZfZedd��Zedd��Zedd��Zedd��Zd S) �_NetlocResultMixinBasecCs |jdS)Nr)� _userinfo)r;r!r!r"�username�sz_NetlocResultMixinBase.usernamecCs |jdS)Nr3)rC)r;r!r!r"�password�sz_NetlocResultMixinBase.passwordcCsD|jd}|sdSt|t�r dnd}|j|�\}}}|j�||S)Nr�%�%)� _hostinfor4r5� partition�lower)r;�hostname� separatorZpercentZzoner!r!r"rK�s z_NetlocResultMixinBase.hostnamecCs@|jd}|dk r<t|d�}d|ko.dkns<td��|S)Nr3� ri��zPort out of range 0-65535)rH�int� ValueError)r;�portr!r!r"rP�s z_NetlocResultMixinBase.portN) r<r=r>r?�propertyrDrErKrPr!r!r!r"rB�s rBc@s(eZdZfZedd��Zedd��ZdS)�_NetlocResultMixinStrcCsD|j}|jd�\}}}|r4|jd�\}}}|s<d}nd}}||fS)N�@�:)�netloc� rpartitionrI)r;rU�userinfo� have_info�hostinforD� have_passwordrEr!r!r"rC�sz_NetlocResultMixinStr._userinfocCsl|j}|jd�\}}}|jd�\}}}|rL|jd�\}}}|jd�\}}}n|jd�\}}}|sdd}||fS)NrS�[�]rT)rUrVrI)r;rU�_rY�have_open_br� bracketedrKrPr!r!r"rH�sz_NetlocResultMixinStr._hostinfoN)r<r=r>r?rQrCrHr!r!r!r"rR�srRc@s(eZdZfZedd��Zedd��ZdS)�_NetlocResultMixinBytescCsD|j}|jd�\}}}|r4|jd�\}}}|s<d}nd}}||fS)N�@�:)rUrVrI)r;rUrWrXrYrDrZrEr!r!r"rC�sz!_NetlocResultMixinBytes._userinfocCsl|j}|jd�\}}}|jd�\}}}|rL|jd�\}}}|jd�\}}}n|jd�\}}}|sdd}||fS)Nra�[�]rb)rUrVrI)r;rUr]rYr^r_rKrPr!r!r"rH�sz!_NetlocResultMixinBytes._hostinfoN)r<r=r>r?rQrCrHr!r!r!r"r`�sr`)� namedtuplezurl fragmentz!scheme netloc path query fragmentz(scheme netloc path params query fragmentz� DefragResult(url, fragment) A 2-tuple that contains the url without fragment identifier and the fragment identifier as a separate argument. z$The URL with no fragment identifier.z� Fragment identifier separated from URL, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. z� SplitResult(scheme, netloc, path, query, fragment) A 5-tuple that contains the different components of a URL. Similar to ParseResult, but does not split params. z%Specifies URL scheme for the request.z0 Network location where the request is made to. z@ The hierarchical path, such as the path to a file to download. z� The query component, that contains non-hierarchical data, that along with data in path component, identifies a resource in the scope of URI's scheme and network location. z� Fragment identifier, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. zr ParseResult(scheme, netloc, path, params, query, fragment) A 6-tuple that contains components of a parsed URL. z� Parameters for last path element used to dereference the URI in order to provide access to perform some operation on the resource. c@seZdZfZdd�ZdS)rcCs |jr|jd|jS|jSdS)N�#)�fragment�url)r;r!r!r"�geturlEszDefragResult.geturlN)r<r=r>r?rir!r!r!r"rCsc@seZdZfZdd�ZdS)rcCst|�S)N)r)r;r!r!r"riMszSplitResult.geturlN)r<r=r>r?rir!r!r!r"rKsc@seZdZfZdd�ZdS)rcCst|�S)N)r)r;r!r!r"riRszParseResult.geturlN)r<r=r>r?rir!r!r!r"rPsc@seZdZfZdd�ZdS)rcCs |jr|jd|jS|jSdS)N�#)rgrh)r;r!r!r"riXszDefragResultBytes.geturlN)r<r=r>r?rir!r!r!r"rVsc@seZdZfZdd�ZdS)rcCst|�S)N)r)r;r!r!r"ri`szSplitResultBytes.geturlN)r<r=r>r?rir!r!r!r"r^sc@seZdZfZdd�ZdS)rcCst|�S)N)r)r;r!r!r"rieszParseResultBytes.geturlN)r<r=r>r?rir!r!r!r"rcscCs8ttfttfttff}x|D]\}}||_||_qWdS)N)rrrrrrr:rA)Z _result_pairsZ_decodedZ_encodedr!r!r"�_fix_result_transcodingis rkTc Csft||�\}}}t|||�}|\}}}}}|tkrHd|krHt|�\}}nd}t||||||�} || �S)N�;r)r8r�uses_params�_splitparamsr) rh�scheme�allow_fragments�_coerce_resultZsplitresultrU�queryrg�params�resultr!r!r"rvscCsRd|kr,|jd|jd��}|dkr6|dfSn |jd�}|d|�||dd�fS)N�/rlrrr3)�find�rfind)rh�ir!r!r"rn�s rncCsLt|�}x*dD]"}|j||�}|dkrt||�}qW|||�||d�fS)Nz/?#r)�lenrv�min)rh�start�delim�cZwdelimr!r!r"�_splitnetloc�s r~cCs�|stdd�|D��rdSddl}|jdd�}|jdd�}|jdd�}|jdd�}|jd |�}||krndSx(d D] }||krttd|dd ��qtWdS)Ncss|]}t|�dkVqdS)�N)�ord)r-r}r!r!r"r/�sz_checknetloc.<locals>.<genexpr>rrSrrTrf�?�NFKCz/?#@:znetloc 'z' contains invalid z#characters under NFKC normalization)�any�unicodedata�replace� normalizerO)rUr��nZnetloc2r}r!r!r"�_checknetloc�s r�cCsxtD]}|j|d�}qW|S)Nr)�_UNSAFE_URL_BYTES_TO_REMOVEr�)rh�br!r!r"�_remove_unsafe_bytes_from_url�s r�cCst|jd�d}|jd�\}}}|rX|r.td��|jd�\}}}|rh|jd�rhtd��n|jd�\}}}t|�dS)NrS�r[zInvalid IPv6 URLr\rT)rVrIrO� startswith�_check_bracketed_host)rUZhostname_and_portZbefore_bracketr^r_rKr]rPr!r!r"�_check_bracketed_netloc�s r�cCsB|jd�r tjd|�s>td��ntj|�}t|tj�r>td��dS)N�vz\Av[a-fA-F0-9]+\..+\ZzIPvFuture address is invalidz%An IPv4 address cannot be in brackets)r��re�matchrO� ipaddressZ ip_addressr4ZIPv4Address)rKZipr!r!r"r��s r�c Cs�t||�\}}}t|�}t|�}|jt�}|jt�}t|�}|||t|�t|�f}tj|d�}|rj||�St t�t kr|t�d}}}|jd�} | dk�r�|d| �dk�r~|d| �j �}|| dd�}|dd�dk�rt|d�\}}d|ko�d |k�sd |k�rd|k�rtd ��|�r<d|k�r<|jdd�\}}d|k�rV|jdd�\}}t|�t|||||�} | t|<|| �Sxd|d| �D]}|tk�r�P�q�W|| dd�}|�s�td d�|D���r�|d| �j �|}}|dd�dk�rPt|d�\}}d|k�rd |k�s,d |k�r4d|k�r4td ��d|k�rPd |k�rPt|�|�rpd|k�rp|jdd�\}}d|k�r�|jdd�\}}t|�t|||||�} | t|<|| �S)NrrTrrr3r�z//r[r\zInvalid IPv6 URLrfr�css|]}|dkVqdS)� 0123456789Nr!)r-r}r!r!r"r/�szurlsplit.<locals>.<genexpr>)r8r��lstrip�_WHATWG_C0_CONTROL_OR_SPACE�strip�bool�typer�getry�MAX_CACHE_SIZEr#rvrJr~rO�splitr�r�scheme_charsr�r�) rhrorprq�key�cachedrUrrrgrxr�r}�restr!r!r"r�sh cCs<t|�\}}}}}}}|r&d||f}|t|||||f��S)Nz%s;%s)r8r)� componentsrorUrhrsrrrgrqr!r!r"rscCs�t|�\}}}}}}|s4|r`|tkr`|dd�dkr`|rP|dd�dkrPd|}d|pXd|}|rp|d|}|r�|d|}|r�|d|}||�S) Nr�z//r3rurrTr�rf)r8�uses_netloc)r�rorUrhrrrgrqr!r!r"rs cCs�|s|S|s|St||�\}}}t|d|�\}}}}}} t|||�\} }}} }}| |ks`| tkrh||�S| tkr�|r�|t| ||| ||f��S|}|r�| r�|}|} |s�|}|t| ||| ||f��S|jd�}|ddkr�|d=|dd�dk�r�|jd�}n(||jd�}td|dd��|dd �<g}xX|D]P}|dk�rdy|j�Wntk �r`YnXn|dk�rt�q0n |j |��q0W|d dk�r�|j d�|t| |dj |��p�d| ||f��S)Nrrur3�..�.���r�r�r�r�)r�r�)r8r� uses_relativer�rr��filter�pop� IndexError�append�join)�baserhrprqZbschemeZbnetlocZbpathZbparamsZbqueryZ bfragmentrorU�pathrsrrrgZ base_partsZsegmentsZ resolved_pathZsegr!r!r"r*sT c CsTt|�\}}d|kr>t|�\}}}}}}t|||||df�}nd}|}|t||��S)Nrfr)r8rrr) rhrq�sr��p�a�qZfragZdefragr!r!r"rosZ0123456789ABCDEFabcdefcCs�|s|jdSt|t�r"|jd�}|jd�}t|�dkr<|S|dg}|j}tdkrbdd�tD�axb|dd�D]R}y(|t|dd��||dd��Wqptk r�|d�||�YqpXqpWdj |�S) N�zutf-8rGr3rcSs4i|],}tD]"}tt||d�g�||j��qqS)�)�_hexdig�bytesrNr()r-r�r�r!r!r"� <dictcomp>�sz$unquote_to_bytes.<locals>.<dictcomp>r�) r�r4r5r(ryr�� _hextobyter��KeyErrorr�)�string�bits�resr��itemr!r!r"r�s* z([-]+)�utf-8r�cCs�d|kr|j|S|dkrd}|dkr*d}tj|�}|dg}|j}x@tdt|�d�D],}|t||�j||��|||d�qVWdj|�S)NrFzutf-8r�rr3r�r)r��_asciirer��rangeryrr,r�)r�r)r*r�r�r�rxr!r!r"r�s Fc CsRi}t|||||||d�}x2|D]*\} } | |kr@|| j| �q | g|| <q W|S)N)r)r*�max_num_fieldsrL)r r�)�qs�keep_blank_values�strict_parsingr)r*r�rLZ parsed_result�pairs�name�valuer!r!r"r �sc@seZdZdS)�_QueryStringSeparatorWarningN)r<r=r>r!r!r!r"r��sr�z/etc/python/urllib.cfgcCs�t|�\}}t|t�r |jd�}|s6t|ttf�rF|dk rFtd��t�}|dk�rVt}d} |dkrvtj j | �}d} |dkr�ytt�}Wnt k r�YnJX|�:ddl}|jdd&d�} | j|�| j d| dd �}|aWdQRXt} |dk�rd |k�rddlm}|d,tdd�d}n:|dk�r,|}n*t|�dk�rVt| �d| �d�dd��|dk �r�||k�r�d|jd�|jd �}nd|j|�}||k�r�td��||k�r�dd�|jd�D�}ndd�|j|�D�}g}x�|D]�}|�r�|�r��q�|jd d�}t|�dk�r>|�r(td!|f��|�r�|jd"�n�q�t|d��sR|�r�|djd#d$�}t|||d%�}||�}|djd#d$�}t|||d%�}||�}|j||f��q�W|S)-Nr$z*Separator must be of type string or bytes.ZPYTHON_URLLIB_QS_SEPARATORzenvironment variablerrf)Z interpolationZcomment_prefixesr )Zfallbackrl)�warnz4The default separator of urllib.parse.parse_qsl and z1parse_qs was changed to '&' to avoid a web cache z"poisoning issue (CVE-2021-23336). z4By default, semicolons no longer act as query field zseparators. z3See https://access.redhat.com/articles/5860431 for z more details.r�)� stacklevel�&Zlegacyr3z (from z) must contain z1 character, or "legacy". See z<https://access.redhat.com/articles/5860431 for more details.zMax number of fields exceededcSs g|]}|jd�D]}|�qqS)rl)r�)r-�s1�s2r!r!r"� <listcomp>Lszparse_qsl.<locals>.<listcomp>cSsg|]}|�qSr!r!)r-r�r!r!r"r�Ns�=zbad query field: %rr�+� )r)r*)rfzeThe default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache z�The default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). z�The default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). By default, semicolons no longer act as query field z�The default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). By default, semicolons no longer act as query field separators. z�The default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). By default, semicolons no longer act as query field separators. See https://access.redhat.com/articles/5860431 for aThe default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). By default, semicolons no longer act as query field separators. See https://access.redhat.com/articles/5860431 for more details.)r8r4r�r,r5rO�object�_default_qs_separator�os�environr��open�_QS_SEPARATOR_CONFIG_FILENAME�FileNotFoundError�configparserZConfigParserZ read_file�warningsr�r�ry�countr�r�r�r)r�r�r�r)r*r�rLrqZ_legacyZenvvar_nameZ config_sourcerr��configr�� num_fieldsr��rZ name_valueZnvr�r�r!r!r"r �s� cCs|jdd�}t|||�S)Nr�r�)r�r)r�r)r*r!r!r"rfssAABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_.-c@s$eZdZdd�Zdd�Zdd�ZdS)�QuotercCstj|�|_dS)N)�_ALWAYS_SAFE�union�safe)r;r�r!r!r"�__init__~szQuoter.__init__cCsd|jjt|�fS)Nz<%s %r>)� __class__r<�dict)r;r!r!r"�__repr__�szQuoter.__repr__cCs(||jkrt|�ndj|�}|||<|S)Nz%{:02X})r��chr�format)r;r�r�r!r!r"�__missing__�szQuoter.__missing__N)r<r=r>r�r�r�r!r!r!r"r�vsr�rucCsbt|t�r8|s|S|dkrd}|dkr*d}|j||�}n |dk rHtd��|dk rXtd��t||�S)Nzutf-8r%z,quote() doesn't support 'encoding' for bytesz*quote() doesn't support 'errors' for bytes)r4r5r(r6r )r�r�r)r*r!r!r"r�s cCsdt|t�rd|ks$t|t�r2d|kr2t||||�St|t�rBd}nd}t|||||�}|jdd�S)Nr�� r�)r4r5r�rr�)r�r�r)r*Zspacer!r!r"r�s cs�t|ttf�std��|sdSt|t�r6|jdd�}ntdd�|D��}|jt|�s^|j�Syt |�Wn&t k r�t|�jt |<�YnXdj �fdd�|D��S)Nz!quote_from_bytes() expected bytesrr$�ignorecSsg|]}|dkr|�qS)�r!)r-r}r!r!r"r��sz$quote_from_bytes.<locals>.<listcomp>csg|]}�|��qSr!r!)r-�char)�quoterr!r"r��s)r4r�� bytearrayr6r5r(�rstrip�_ALWAYS_SAFE_BYTESr,r r�r��__getitem__r�)Zbsr�r!)r�r"r �s cCst|d�r|j�}nRy t|�r2t|dt�r2t�Wn0tk rdtj�\}}}td�j|��YnXg} |s�xr|D]j\} }t| t �r�|| |�} n|t | �|||�} t|t �r�|||�}n|t |�|||�}| j| d|�qtW�n,�x(|D�]\} }t| t ��r|| |�} n|t | �|||�} t|t ��rL|||�}| j| d|�q�t|t ��rz|||||�}| j| d|�q�yt|�}Wn:tk �r�|t |�|||�}| j| d|�Yq�XxJ|D]B} t| t ��r�|| |�} n|t | �|||�} | j| d| ��q�Wq�Wdj| �S)N�itemsrz1not a valid non-string sequence or mapping objectr�r�) �hasattrr�ryr4r0r6�sys�exc_info�with_tracebackr�r5r�r�)rrZdoseqr�r)r*Z quote_viaZtyZva�tb�l�kr�r.Zeltr!r!r"r�sP cCsJt|t�rFy|jd�j�}Wn(tk rDtdt|�d��YnX|S)N�ASCIIzURL z contains non-ASCII characters)r4r5r(r,�UnicodeError�repr)rhr!r!r"�to_bytes/s r�cCs`t|�j�}|dd�dkr<|dd�dkr<|dd�j�}|dd�dkr\|dd�j�}|S)Nr3�<�>�zURL:r�r�)r5r�)rhr!r!r"�unwrap<s rcCsDtdkrtjdtj�atj|�}|r<|j�\}}|j�|fSd|fS)Nz ([^/:]+):(.*))� _typeprogr��compile�DOTALLr��groupsrJ)rhr�ro�datar!r!r"� splittypeEs rcCsXtdkrtjdtj�atj|�}|rP|j�\}}|rH|ddkrHd|}||fSd|fS)Nz//([^/#?]*)(.*)rru)� _hostprogr�rrr�r)rhr�Z host_portr�r!r!r"� splithostRs r cCs |jd�\}}}|r|nd|fS)NrS)rV)�host�userr|r!r!r"� splituser`srcCs |jd�\}}}||r|ndfS)NrT)rI)rr|Zpasswdr!r!r"�splitpasswdesr cCsDtdkrtjdtj�atj|�}|r<|j�\}}|r<||fS|dfS)Nz(.*):([0-9]*)$)� _portprogr�rrr�r)r r�rPr!r!r"� splitportls rr3cCsT|jd�\}}}|s|}n2|rLyt|�}Wntk rBd}YnX||fS||fS)NrT)rVrNrO)r Zdefportr|rPZnportr!r!r"� splitnportys rcCs$|jd�\}}}|r||fS|dfS)Nr�)rV)rhr�r|rrr!r!r"� splitquery�srcCs$|jd�\}}}|r||fS|dfS)Nrf)rV)rhr�r|�tagr!r!r"�splittag�srcCs|jd�}|d|dd�fS)Nrlrr3)r�)rhZwordsr!r!r"� splitattr�s rcCs |jd�\}}}||r|ndfS)Nr�)rI)�attrr|r�r!r!r"� splitvalue�sr)rT)r)rT)T)r�r�)FFr�r�NN)FFr�r�NN)r�r�)ruNN)rNN)rur�)r�)er�r�r��collectionsr��__all__r�r�rmZnon_hierarchicalZ uses_queryZ uses_fragmentr�r�r�r�rr#Z_implicit_encodingZ_implicit_errorsr'r+r2r8r�r9r@rBrRr`reZ_DefragResultBaseZ_SplitResultBaseZ_ParseResultBase�__doc__rhrgrorUr�rrrsZ ResultBaserrrrrrrkrrnr~r�r�r�r�rrrrrr�r�rrr�rr �RuntimeWarningr�r�r�r r� frozensetr�r�r�r �defaultdictr�rrr rr�rrrrr rr rrrrrrrr!r!r!r"�<module>"s ! B E & y , O PK23�\�I����� __pycache__/parse.cpython-36.pycnu�[���3 ���ie��@s�dZddlZddlZddlZddlZddlZddddddd d ddd ddddddddddgZddddddddd d!d"d#d$d%d&d'd(d)d*gZdddddd+dddd"d d!d,d#d$d%d-d'd(d&d.d/d0d)d*gZddd1d#ddd d!d$d%d2d3d"d&d4gZ dd1d5d6d+ddd,d2d3g Z ddddd d!d"dd$d%d2d3gZddd1ddd6ddd d!d,dd#g Zd7Z d8Zd9d:d;gZd<ZiZd=d>�Zd?Zd@ZdAdB�ZeefdCdD�ZeefdEdF�ZdGdH�ZGdIdJ�dJe�ZGdKdL�dLe�ZGdMdN�dNe�ZGdOdP�dPee�ZGdQdR�dRee�ZddSlmZeddT�Z eddU�Z!eddV�Z"dWe _dXe j#_dYe j$_dZe!_d[e!j%_d\e!j&_d]e!j'_d^e!j(_d_e!j$_d`e"_e!j%je"j%_e!j&je"j&_e!j'je"j'_dae"j)_e!j(je"j(_e!j$je"j$_eZ*Gdbd�de e�Z+Gdcd�de!e�Z,Gddd�de"e�Z-Gded�de e�Z.Gdfd�de!e�Z/Gdgd�de"e�Z0dhdi�Z1e1�[1d�dkd�Z2dldm�Z3d�dndo�Z4dpdq�Z5drds�Z6dtdu�Z7dvdw�Z8d�dxd�Z9dyd�Z:dzd�Z;d�d{d�Z<d|d�Z=d}Z>da?d~d�Z@ejAd�ZBd�d�d�ZCd�d�d �ZDGd�d��d�eE�ZFd�ZGdaHd�d�d�ZId�d�d�ZJeKd��ZLeMeL�ZNiZOGd�d��d�ejP�ZQd�d�d�ZRd�d�d �ZSd�d�d�ZTd�dddeSfd�d �ZUd�d��ZVd�d��ZWdaXd�d��ZYdaZd�d��Z[d�d��Z\d�d��Z]da^d�d��Z_d�d�d��Z`d�d��Zad�d��Zbd�d��Zcd�d��ZddS)�a3Parse (absolute and relative) URLs. urlparse module is based upon the following RFC specifications. RFC 3986 (STD66): "Uniform Resource Identifiers" by T. Berners-Lee, R. Fielding and L. Masinter, January 2005. RFC 2732 : "Format for Literal IPv6 Addresses in URL's by R.Hinden, B.Carpenter and L.Masinter, December 1999. RFC 2396: "Uniform Resource Identifiers (URI)": Generic Syntax by T. Berners-Lee, R. Fielding, and L. Masinter, August 1998. RFC 2368: "The mailto URL scheme", by P.Hoffman , L Masinter, J. Zawinski, July 1998. RFC 1808: "Relative Uniform Resource Locators", by R. Fielding, UC Irvine, June 1995. RFC 1738: "Uniform Resource Locators (URL)" by T. Berners-Lee, L. Masinter, M. McCahill, December 1994 RFC 3986 is considered the current standard and any future changes to urlparse module should conform with it. The urlparse module is currently not entirely compliant with this RFC due to defacto scenarios for parsing, and for backward compatibility purposes, some parsing quirks from older RFCs are retained. The testcases in test_urlparse.py provides a good indicator of parsing behavior. The WHATWG URL Parser spec should also be considered. We are not compliant with it either due to existing user code API behavior expectations (Hyrum's Law). It serves as a useful guide when making changes. �N�urlparse� urlunparse�urljoin� urldefrag�urlsplit� urlunsplit� urlencode�parse_qs� parse_qsl�quote� quote_plus�quote_from_bytes�unquote�unquote_plus�unquote_to_bytes�DefragResult�ParseResult�SplitResult�DefragResultBytes�ParseResultBytes�SplitResultBytes�Zftp�httpZgopherZnntpZimapZwais�fileZhttpsZshttpZmmsZprosperoZrtspZrtspuZsftpZsvnzsvn+sshZwsZwssZtelnetZsnewsZrsyncZnfsZgitzgit+sshZhdlZsipZsipsZtelZmailtoZnewszAabcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+-.z! � � � �cCstj�tj�dS)z,Clear the parse cache and the quoters cache.N)�_parse_cache�clear� _safe_quoters�r!r!�$/usr/lib64/python3.6/urllib/parse.py�clear_cache`sr#�ascii�strictcCs|S)Nr!)�objr!r!r"�_nooposr'cCs|j||�S)N)�encode)r&�encoding�errorsr!r!r"�_encode_resultrsr+cst��fdd�|D��S)Nc3s"|]}|r|j���ndVqdS)rN)�decode)�.0�x)r)r*r!r"� <genexpr>xsz_decode_args.<locals>.<genexpr>)�tuple)�argsr)r*r!)r)r*r"�_decode_argsvsr2cGsZt|dt�}x.|dd�D]}|rt|t�|krtd��qW|rL|tfSt|�tfS)Nr�z$Cannot mix str and non-str arguments)� isinstance�str� TypeErrorr'r2r+)r1Z str_input�argr!r!r"�_coerce_argszs r8c@seZdZdZfZddd�ZdS)�_ResultMixinStrz>Standard approach to encoding parsed results from str to bytesr$r%cs|j��fdd�|D��S)Nc3s|]}|j���VqdS)N)r()r-r.)r)r*r!r"r/�sz)_ResultMixinStr.encode.<locals>.<genexpr>)�_encoded_counterpart)�selfr)r*r!)r)r*r"r(�sz_ResultMixinStr.encodeN)r$r%)�__name__� __module__�__qualname__�__doc__� __slots__r(r!r!r!r"r9�sr9c@seZdZdZfZddd�ZdS)�_ResultMixinBytesz>Standard approach to decoding parsed results from bytes to strr$r%cs|j��fdd�|D��S)Nc3s|]}|j���VqdS)N)r,)r-r.)r)r*r!r"r/�sz+_ResultMixinBytes.decode.<locals>.<genexpr>)�_decoded_counterpart)r;r)r*r!)r)r*r"r,�sz_ResultMixinBytes.decodeN)r$r%)r<r=r>r?r@r,r!r!r!r"rA�srAc@sDeZdZdZfZedd��Zedd��Zedd��Zedd ��Z d S)�_NetlocResultMixinBasezHShared methods for the parsed result objects containing a netloc elementcCs |jdS)Nr)� _userinfo)r;r!r!r"�username�sz_NetlocResultMixinBase.usernamecCs |jdS)Nr3)rD)r;r!r!r"�password�sz_NetlocResultMixinBase.passwordcCsD|jd}|sdSt|t�r dnd}|j|�\}}}|j�||S)Nr�%�%)� _hostinfor4r5� partition�lower)r;�hostname� separatorZpercentZzoner!r!r"rL�s z_NetlocResultMixinBase.hostnamecCs@|jd}|dk r<t|d�}d|ko.dkns<td��|S)Nr3� ri��zPort out of range 0-65535)rI�int� ValueError)r;�portr!r!r"rQ�s z_NetlocResultMixinBase.portN) r<r=r>r?r@�propertyrErFrLrQr!r!r!r"rC�srCc@s(eZdZfZedd��Zedd��ZdS)�_NetlocResultMixinStrcCsD|j}|jd�\}}}|r4|jd�\}}}|s<d}nd}}||fS)N�@�:)�netloc� rpartitionrJ)r;rV�userinfo� have_info�hostinforE� have_passwordrFr!r!r"rD�sz_NetlocResultMixinStr._userinfocCsl|j}|jd�\}}}|jd�\}}}|rL|jd�\}}}|jd�\}}}n|jd�\}}}|sdd}||fS)NrT�[�]rU)rVrWrJ)r;rV�_rZ�have_open_br� bracketedrLrQr!r!r"rI�sz_NetlocResultMixinStr._hostinfoN)r<r=r>r@rRrDrIr!r!r!r"rS�srSc@s(eZdZfZedd��Zedd��ZdS)�_NetlocResultMixinBytescCsD|j}|jd�\}}}|r4|jd�\}}}|s<d}nd}}||fS)N�@�:)rVrWrJ)r;rVrXrYrZrEr[rFr!r!r"rD�sz!_NetlocResultMixinBytes._userinfocCsl|j}|jd�\}}}|jd�\}}}|rL|jd�\}}}|jd�\}}}n|jd�\}}}|sdd}||fS)Nrb�[�]rc)rVrWrJ)r;rVr^rZr_r`rLrQr!r!r"rI�sz!_NetlocResultMixinBytes._hostinfoN)r<r=r>r@rRrDrIr!r!r!r"ra�sra)� namedtuplezurl fragmentz!scheme netloc path query fragmentz(scheme netloc path params query fragmentz� DefragResult(url, fragment) A 2-tuple that contains the url without fragment identifier and the fragment identifier as a separate argument. z$The URL with no fragment identifier.z� Fragment identifier separated from URL, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. z� SplitResult(scheme, netloc, path, query, fragment) A 5-tuple that contains the different components of a URL. Similar to ParseResult, but does not split params. z%Specifies URL scheme for the request.z0 Network location where the request is made to. z@ The hierarchical path, such as the path to a file to download. z� The query component, that contains non-hierarchical data, that along with data in path component, identifies a resource in the scope of URI's scheme and network location. z� Fragment identifier, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. zr ParseResult(scheme, netloc, path, params, query, fragment) A 6-tuple that contains components of a parsed URL. z� Parameters for last path element used to dereference the URI in order to provide access to perform some operation on the resource. c@seZdZfZdd�ZdS)rcCs |jr|jd|jS|jSdS)N�#)�fragment�url)r;r!r!r"�geturlEszDefragResult.geturlN)r<r=r>r@rjr!r!r!r"rCsc@seZdZfZdd�ZdS)rcCst|�S)N)r)r;r!r!r"rjMszSplitResult.geturlN)r<r=r>r@rjr!r!r!r"rKsc@seZdZfZdd�ZdS)rcCst|�S)N)r)r;r!r!r"rjRszParseResult.geturlN)r<r=r>r@rjr!r!r!r"rPsc@seZdZfZdd�ZdS)rcCs |jr|jd|jS|jSdS)N�#)rhri)r;r!r!r"rjXszDefragResultBytes.geturlN)r<r=r>r@rjr!r!r!r"rVsc@seZdZfZdd�ZdS)rcCst|�S)N)r)r;r!r!r"rj`szSplitResultBytes.geturlN)r<r=r>r@rjr!r!r!r"r^sc@seZdZfZdd�ZdS)rcCst|�S)N)r)r;r!r!r"rjeszParseResultBytes.geturlN)r<r=r>r@rjr!r!r!r"rcscCs8ttfttfttff}x|D]\}}||_||_qWdS)N)rrrrrrr:rB)Z _result_pairsZ_decodedZ_encodedr!r!r"�_fix_result_transcodingis rlTc Csft||�\}}}t|||�}|\}}}}}|tkrHd|krHt|�\}}nd}t||||||�} || �S)a#Parse a URL into 6 components: <scheme>://<netloc>/<path>;<params>?<query>#<fragment> Return a 6-tuple: (scheme, netloc, path, params, query, fragment). Note that we don't break the components up in smaller bits (e.g. netloc is a single string) and we don't expand % escapes.�;r)r8r�uses_params�_splitparamsr) ri�scheme�allow_fragments�_coerce_resultZsplitresultrV�queryrh�params�resultr!r!r"rvscCsRd|kr,|jd|jd��}|dkr6|dfSn |jd�}|d|�||dd�fS)N�/rmrrr3)�find�rfind)ri�ir!r!r"ro�s rocCsLt|�}x*dD]"}|j||�}|dkrt||�}qW|||�||d�fS)Nz/?#r)�lenrw�min)ri�start�delim�cZwdelimr!r!r"�_splitnetloc�s rcCs�|stdd�|D��rdSddl}|jdd�}|jdd�}|jdd�}|jdd�}|jd |�}||krndSx(d D] }||krttd|dd ��qtWdS)Ncss|]}t|�dkVqdS)�N)�ord)r-r~r!r!r"r/�sz_checknetloc.<locals>.<genexpr>rrTrrUrg�?�NFKCz/?#@:znetloc 'z' contains invalid z#characters under NFKC normalization)�any�unicodedata�replace� normalizerP)rVr��nZnetloc2r~r!r!r"�_checknetloc�s r�cCsxtD]}|j|d�}qW|S)Nr)�_UNSAFE_URL_BYTES_TO_REMOVEr�)ri�br!r!r"�_remove_unsafe_bytes_from_url�s r�cCst|jd�d}|jd�\}}}|rX|r.td��|jd�\}}}|rh|jd�rhtd��n|jd�\}}}t|�dS)NrT�r\zInvalid IPv6 URLr]rU)rWrJrP� startswith�_check_bracketed_host)rVZhostname_and_portZbefore_bracketr_r`rLr^rQr!r!r"�_check_bracketed_netloc�s r�cCsB|jd�r tjd|�s>td��ntj|�}t|tj�r>td��dS)N�vz\Av[a-fA-F0-9]+\..+\ZzIPvFuture address is invalidz%An IPv4 address cannot be in brackets)r��re�matchrP� ipaddressZ ip_addressr4ZIPv4Address)rLZipr!r!r"r��s r�c Cs�t||�\}}}t|�}t|�}|jt�}|jt�}t|�}|||t|�t|�f}tj|d�}|rj||�St t�t kr|t�d}}}|jd�} | dk�r�|d| �dk�r~|d| �j �}|| dd�}|dd�dk�rt|d�\}}d |ko�d |k�sd |k�rd |k�rtd��|�r<d|k�r<|jdd�\}}d |k�rV|jd d�\}}t|�t|||||�} | t|<|| �Sxd|d| �D]}|tk�r�P�q�W|| dd�}|�s�tdd�|D���r�|d| �j �|}}|dd�dk�rPt|d�\}}d |k�rd |k�s,d |k�r4d |k�r4td��d |k�rPd |k�rPt|�|�rpd|k�rp|jdd�\}}d |k�r�|jd d�\}}t|�t|||||�} | t|<|| �S)aParse a URL into 5 components: <scheme>://<netloc>/<path>?<query>#<fragment> Return a 5-tuple: (scheme, netloc, path, query, fragment). Note that we don't break the components up in smaller bits (e.g. netloc is a single string) and we don't expand % escapes.NrrUrrr3r�z//r\r]zInvalid IPv6 URLrgr�css|]}|dkVqdS)� 0123456789Nr!)r-r~r!r!r"r/�szurlsplit.<locals>.<genexpr>)r8r��lstrip�_WHATWG_C0_CONTROL_OR_SPACE�strip�bool�typer�getrz�MAX_CACHE_SIZEr#rwrKrrP�splitr�r�scheme_charsr�r�) rirprqrr�key�cachedrVrsrhryr�r~�restr!r!r"r�sh cCs<t|�\}}}}}}}|r&d||f}|t|||||f��S)z�Put a parsed URL back together again. This may result in a slightly different, but equivalent URL, if the URL that was parsed originally had redundant delimiters, e.g. a ? with an empty query (the draft states that these are equivalent).z%s;%s)r8r)� componentsrprVrirtrsrhrrr!r!r"rscCs�t|�\}}}}}}|s4|r`|tkr`|dd�dkr`|rP|dd�dkrPd|}d|pXd|}|rp|d|}|r�|d|}|r�|d |}||�S) akCombine the elements of a tuple as returned by urlsplit() into a complete URL as a string. The data argument can be any five-item iterable. This may result in a slightly different, but equivalent URL, if the URL that was parsed originally had unnecessary delimiters (for example, a ? with an empty query; the RFC states that these are equivalent).Nr�z//r3rvrrUr�rg)r8�uses_netloc)r�rprVrirsrhrrr!r!r"rs cCs�|s|S|s|St||�\}}}t|d|�\}}}}}} t|||�\} }}} }}| |ks`| tkrh||�S| tkr�|r�|t| ||| ||f��S|}|r�| r�|}|} |s�|}|t| ||| ||f��S|jd�}|ddkr�|d=|dd�dk�r�|jd�}n(||jd�}td|dd ��|dd �<g}xX|D]P}|dk�rdy|j�Wntk �r`YnXn|dk�rt�q0n |j |��q0W|ddk�r�|j d�|t| |dj |��p�d| ||f��S) zaJoin a base URL and a possibly relative URL to form an absolute interpretation of the latter.rrvr3N�..�.���r�r�r�r�)r�r�)r8r� uses_relativer�rr��filter�pop� IndexError�append�join)�baserirqrrZbschemeZbnetlocZbpathZbparamsZbqueryZ bfragmentrprV�pathrtrsrhZ base_partsZsegmentsZ resolved_pathZsegr!r!r"r*sT c CsTt|�\}}d|kr>t|�\}}}}}}t|||||df�}nd}|}|t||��S)z�Removes any existing fragment from URL. Returns a tuple of the defragmented URL and the fragment. If the URL contained no fragments, the second element is the empty string. rgr)r8rrr) rirr�sr��p�a�qZfragZdefragr!r!r"rosZ0123456789ABCDEFabcdefcCs�|s|jdSt|t�r"|jd�}|jd�}t|�dkr<|S|dg}|j}tdkrbdd�tD�axb|dd�D]R}y(|t|dd ��||d d��Wqptk r�|d�||�YqpXqpWdj |�S) z,unquote_to_bytes('abc%20def') -> b'abc def'.�zutf-8rHr3rNcSs4i|],}tD]"}tt||d�g�||j��qqS)�)�_hexdig�bytesrOr()r-r�r�r!r!r"� <dictcomp>�sz$unquote_to_bytes.<locals>.<dictcomp>r�) r�r4r5r(rzr�� _hextobyter��KeyErrorr�)�string�bits�resr��itemr!r!r"r�s* z([-]+)�utf-8r�cCs�d|kr|j|S|dkrd}|dkr*d}tj|�}|dg}|j}x@tdt|�d�D],}|t||�j||��|||d�qVWdj|�S) a�Replace %xx escapes by their single-character equivalent. The optional encoding and errors parameters specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. By default, percent-encoded sequences are decoded with UTF-8, and invalid sequences are replaced by a placeholder character. unquote('abc%20def') -> 'abc def'. rGNzutf-8r�rr3r�r)r��_asciirer��rangerzrr,r�)r�r)r*r�r�r�ryr!r!r"r�s Fc CsRi}t|||||||d�}x2|D]*\} } | |kr@|| j| �q | g|| <q W|S)a�Parse a query given as a string argument. Arguments: qs: percent-encoded query string to be parsed keep_blank_values: flag indicating whether blank values in percent-encoded queries should be treated as blank strings. A true value indicates that blanks should be retained as blank strings. The default false value indicates that blank values are to be ignored and treated as if they were not included. strict_parsing: flag indicating what to do with parsing errors. If false (the default), errors are silently ignored. If true, errors raise a ValueError exception. encoding and errors: specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. max_num_fields: int. If set, then throws a ValueError if there are more than n fields read by parse_qsl(). Returns a dictionary. )r)r*�max_num_fieldsrM)r r�)�qs�keep_blank_values�strict_parsingr)r*r�rMZ parsed_result�pairs�name�valuer!r!r"r �sc@seZdZdZdS)�_QueryStringSeparatorWarningz>Warning for using default `separator` in parse_qs or parse_qslN)r<r=r>r?r!r!r!r"r��sr�z/etc/python/urllib.cfgcCs�t|�\}}t|t�r |jd�}|s6t|ttf�rF|dk rFtd��t�}|dk�rVt}d} |dkrvtj j | �}d} |dkr�ytt�}Wnt k r�YnJX|�:ddl}|jdd'd�} | j|�| j d | dd �}|aWdQRXt} |dk�rd|k�rddlm}|d-tdd�d}n:|dk�r,|}n*t|�dk�rVt| �d| �d�dd��|dk �r�||k�r�d|jd�|jd�}nd|j|�}||k�r�td��||k�r�dd�|jd�D�}nd d�|j|�D�}g}x�|D]�}|�r�|�r��q�|jd!d�}t|�dk�r>|�r(td"|f��|�r�|jd#�n�q�t|d��sR|�r�|djd$d%�}t|||d&�}||�}|djd$d%�}t|||d&�}||�}|j||f��q�W|S).a�Parse a query given as a string argument. Arguments: qs: percent-encoded query string to be parsed keep_blank_values: flag indicating whether blank values in percent-encoded queries should be treated as blank strings. A true value indicates that blanks should be retained as blank strings. The default false value indicates that blank values are to be ignored and treated as if they were not included. strict_parsing: flag indicating what to do with parsing errors. If false (the default), errors are silently ignored. If true, errors raise a ValueError exception. encoding and errors: specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. max_num_fields: int. If set, then throws a ValueError if there are more than n fields read by parse_qsl(). Returns a list, as G-d intended. r$Nz*Separator must be of type string or bytes.ZPYTHON_URLLIB_QS_SEPARATORzenvironment variablerrg)Z interpolationZcomment_prefixesr )Zfallbackrm)�warnz4The default separator of urllib.parse.parse_qsl and z1parse_qs was changed to '&' to avoid a web cache z"poisoning issue (CVE-2021-23336). z4By default, semicolons no longer act as query field zseparators. z3See https://access.redhat.com/articles/5860431 for z more details.r�)� stacklevel�&Zlegacyr3z (from z) must contain z1 character, or "legacy". See z<https://access.redhat.com/articles/5860431 for more details.zMax number of fields exceededcSs g|]}|jd�D]}|�qqS)rm)r�)r-�s1�s2r!r!r"� <listcomp>Lszparse_qsl.<locals>.<listcomp>cSsg|]}|�qSr!r!)r-r�r!r!r"r�Ns�=zbad query field: %rr�+� )r)r*)rgzeThe default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache z�The default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). z�The default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). By default, semicolons no longer act as query field z�The default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). By default, semicolons no longer act as query field separators. z�The default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). By default, semicolons no longer act as query field separators. See https://access.redhat.com/articles/5860431 for aThe default separator of urllib.parse.parse_qsl and parse_qs was changed to '&' to avoid a web cache poisoning issue (CVE-2021-23336). By default, semicolons no longer act as query field separators. See https://access.redhat.com/articles/5860431 for more details.)r8r4r�r,r5rP�object�_default_qs_separator�os�environr��open�_QS_SEPARATOR_CONFIG_FILENAME�FileNotFoundError�configparserZConfigParserZ read_file�warningsr�r�rz�countr�r�r�r)r�r�r�r)r*r�rMrrZ_legacyZenvvar_nameZ config_sourcerr��configr�� num_fieldsr��rZ name_valueZnvr�r�r!r!r"r �s� cCs|jdd�}t|||�S)z�Like unquote(), but also replace plus signs by spaces, as required for unquoting HTML form values. unquote_plus('%7e/abc+def') -> '~/abc def' r�r�)r�r)r�r)r*r!r!r"rfssAABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_.-c@s(eZdZdZdd�Zdd�Zdd�ZdS) �Quoterz�A mapping from bytes (in range(0,256)) to strings. String values are percent-encoded byte values, unless the key < 128, and in the "safe" set (either the specified safe set, or default set). cCstj|�|_dS)zsafe: bytes object.N)�_ALWAYS_SAFE�union�safe)r;r�r!r!r"�__init__~szQuoter.__init__cCsd|jjt|�fS)Nz<%s %r>)� __class__r<�dict)r;r!r!r"�__repr__�szQuoter.__repr__cCs(||jkrt|�ndj|�}|||<|S)Nz%{:02X})r��chr�format)r;r�r�r!r!r"�__missing__�szQuoter.__missing__N)r<r=r>r?r�r�r�r!r!r!r"r�vsr�rvcCsbt|t�r8|s|S|dkrd}|dkr*d}|j||�}n |dk rHtd��|dk rXtd��t||�S)a�quote('abc def') -> 'abc%20def' Each part of a URL, e.g. the path info, the query, etc., has a different set of reserved characters that must be quoted. RFC 2396 Uniform Resource Identifiers (URI): Generic Syntax lists the following reserved characters. reserved = ";" | "/" | "?" | ":" | "@" | "&" | "=" | "+" | "$" | "," Each of these characters is reserved in some component of a URL, but not necessarily in all of them. By default, the quote function is intended for quoting the path section of a URL. Thus, it will not encode '/'. This character is reserved, but in typical usage the quote function is being called on a path where the existing slash characters are used as reserved characters. string and safe may be either str or bytes objects. encoding and errors must not be specified if string is a bytes object. The optional encoding and errors parameters specify how to deal with non-ASCII characters, as accepted by the str.encode method. By default, encoding='utf-8' (characters are encoded with UTF-8), and errors='strict' (unsupported characters raise a UnicodeEncodeError). Nzutf-8r%z,quote() doesn't support 'encoding' for bytesz*quote() doesn't support 'errors' for bytes)r4r5r(r6r )r�r�r)r*r!r!r"r�s cCsdt|t�rd|ks$t|t�r2d|kr2t||||�St|t�rBd}nd}t|||||�}|jdd�S)z�Like quote(), but also replace ' ' with '+', as required for quoting HTML form values. Plus signs in the original string are escaped unless they are included in safe. It also does not have safe default to '/'. r�� r�)r4r5r�rr�)r�r�r)r*Zspacer!r!r"r�s cs�t|ttf�std��|sdSt|t�r6|jdd�}ntdd�|D��}|jt|�s^|j�Syt |�Wn&t k r�t|�jt |<�YnXdj �fdd�|D��S)z�Like quote(), but accepts a bytes object rather than a str, and does not perform string-to-bytes encoding. It always returns an ASCII string. quote_from_bytes(b'abc def?') -> 'abc%20def%3f' z!quote_from_bytes() expected bytesrr$�ignorecSsg|]}|dkr|�qS)�r!)r-r~r!r!r"r��sz$quote_from_bytes.<locals>.<listcomp>csg|]}�|��qSr!r!)r-�char)�quoterr!r"r��s)r4r�� bytearrayr6r5r(�rstrip�_ALWAYS_SAFE_BYTESr,r r�r��__getitem__r�)Zbsr�r!)r�r"r �s cCst|d�r|j�}nRy t|�r2t|dt�r2t�Wn0tk rdtj�\}}}td�j|��YnXg} |s�xr|D]j\} }t| t �r�|| |�} n|t | �|||�} t|t �r�|||�}n|t |�|||�}| j| d|�qtW�n,�x(|D�]\} }t| t ��r|| |�} n|t | �|||�} t|t ��rL|||�}| j| d|�q�t|t ��rz|||||�}| j| d|�q�yt|�}Wn:tk �r�|t |�|||�}| j| d|�Yq�XxJ|D]B} t| t ��r�|| |�} n|t | �|||�} | j| d| ��q�Wq�Wdj| �S)a^Encode a dict or sequence of two-element tuples into a URL query string. If any values in the query arg are sequences and doseq is true, each sequence element is converted to a separate parameter. If the query arg is a sequence of two-element tuples, the order of the parameters in the output will match the order of parameters in the input. The components of a query arg may each be either a string or a bytes type. The safe, encoding, and errors parameters are passed down to the function specified by quote_via (encoding and errors only if a component is a str). �itemsrz1not a valid non-string sequence or mapping objectr�r�) �hasattrr�rzr4r0r6�sys�exc_info�with_tracebackr�r5r�r�)rsZdoseqr�r)r*Z quote_viaZtyZva�tb�l�kr�r.Zeltr!r!r"r�sP cCsJt|t�rFy|jd�j�}Wn(tk rDtdt|�d��YnX|S)zto_bytes(u"URL") --> 'URL'.�ASCIIzURL z contains non-ASCII characters)r4r5r(r,�UnicodeError�repr)rir!r!r"�to_bytes/s r�cCs`t|�j�}|dd�dkr<|dd�dkr<|dd�j�}|dd�dkr\|dd�j�}|S) z8unwrap('<URL:type://host/path>') --> 'type://host/path'.Nr3�<�>�zURL:r�r�)r5r�)rir!r!r"�unwrap<s rcCsDtdkrtjdtj�atj|�}|r<|j�\}}|j�|fSd|fS)z:splittype('type:opaquestring') --> 'type', 'opaquestring'.Nz ([^/:]+):(.*))� _typeprogr��compile�DOTALLr��groupsrK)rir�rp�datar!r!r"� splittypeEs rcCsXtdkrtjdtj�atj|�}|rP|j�\}}|rH|ddkrHd|}||fSd|fS)z;splithost('//host[:port]/path') --> 'host[:port]', '/path'.Nz//([^/#?]*)(.*)rrv)� _hostprogr�rrr�r)rir�Z host_portr�r!r!r"� splithostRs r cCs |jd�\}}}|r|nd|fS)zJsplituser('user[:passwd]@host[:port]') --> 'user[:passwd]', 'host[:port]'.rTN)rW)�host�userr}r!r!r"� splituser`sr cCs |jd�\}}}||r|ndfS)z/splitpasswd('user:passwd') -> 'user', 'passwd'.rUN)rJ)rr}Zpasswdr!r!r"�splitpasswdesrcCsDtdkrtjdtj�atj|�}|r<|j�\}}|r<||fS|dfS)z*splitport('host:port') --> 'host', 'port'.Nz(.*):([0-9]*)$)� _portprogr�rrr�r)rr�rQr!r!r"� splitportls rr3cCsT|jd�\}}}|s|}n2|rLyt|�}Wntk rBd}YnX||fS||fS)z�Split host and port, returning numeric port. Return given default port if no ':' found; defaults to -1. Return numerical port if a valid number are found after ':'. Return None if ':' but not a valid number.rUN)rWrOrP)rZdefportr}rQZnportr!r!r"� splitnportys rcCs$|jd�\}}}|r||fS|dfS)z/splitquery('/path?query') --> '/path', 'query'.r�N)rW)rir�r}rsr!r!r"� splitquery�srcCs$|jd�\}}}|r||fS|dfS)z)splittag('/path#tag') --> '/path', 'tag'.rgN)rW)rir�r}�tagr!r!r"�splittag�srcCs|jd�}|d|dd�fS)zksplitattr('/path;attr1=value1;attr2=value2;...') -> '/path', ['attr1=value1', 'attr2=value2', ...].rmrr3N)r�)riZwordsr!r!r"� splitattr�s rcCs |jd�\}}}||r|ndfS)z-splitvalue('attr=value') --> 'attr', 'value'.r�N)rJ)�attrr}r�r!r!r"� splitvalue�sr)rT)r)rT)T)r�r�)FFr�r�NN)FFr�r�NN)r�r�)rvNN)rNN)rvr�)r�)er?r�r�r��collectionsr��__all__r�r�rnZnon_hierarchicalZ uses_queryZ uses_fragmentr�r�r�r�rr#Z_implicit_encodingZ_implicit_errorsr'r+r2r8r�r9rArCrSrarfZ_DefragResultBaseZ_SplitResultBaseZ_ParseResultBaserirhrprVr�rsrtZ ResultBaserrrrrrrlrrorr�r�r�r�rrrrrr�r�rrr�rr �RuntimeWarningr�r�r�r r� frozensetr�r�r�r �defaultdictr�rrr rr�rrrr r r rrrrrrrrr!r!r!r"�<module> s ! B E & y , O PK23�\�2`zz(__pycache__/request.cpython-36.opt-1.pycnu�[���3 ���i~��)@s�dZddlZddlZddlZddlZddlZddlZddlZddl Z ddl Z ddlZddlZddl Z ddlZddlZddlZddlZddlZddlmZmZmZddlmZmZmZmZmZmZmZmZm Z m!Z!m"Z"m#Z#m$Z$m%Z%m&Z&m'Z'm(Z(m)Z)ddl*m+Z+m,Z,yddl-Z-Wne.k �r&dZ/YnXdZ/dd d ddd ddddddddddddddddddd d!d"d#d$d%d&d'd(g!Z0d)e j1dd*�Z2da3dej4fddddd+�d,d�Z5d-d �Z6gZ7d~d.d%�Z8d/d&�Z9e j:d0e j;�Z<d1d2�Z=Gd3d�d�Z>Gd4d �d �Z?d5d!�Z@Gd6d �d �ZAGd7d�deA�ZBGd8d�deA�ZCGd9d�deA�ZDd:d;�ZEGd<d�deA�ZFGd=d�d�ZGGd>d�deG�ZHGd?d�deH�ZIGd@d�d�ZJGdAd�deJeA�ZKGdBd�deJeA�ZLejMZNGdCd�d�ZOGdDd�deAeO�ZPGdEd�deAeO�ZQGdFdG�dGeA�ZRGdHd�deR�ZSeTejUdI��r2GdJdK�dKeR�ZVe0jWdK�GdLd �d eA�ZXGdMd�deA�ZYdNdO�ZZdPdQ�Z[GdRd�deA�Z\dSdT�Z]GdUd�deA�Z^GdVd�de^�Z_GdWd�deA�Z`dXZaejbdYk�r�ddZlcmdZdmeZend[d#�Zdd\d"�ZeiZfGd]d'�d'�ZgGd^d(�d(eg�Zhdaid_d`�Zjdakdadb�Zldamdcdd�Zndaodedf�ZpGdgdh�dh�Zqdidj�Zrddkdl�Zsdmdn�Zte judok�r�ddplvmwZwmxZxdqdr�Zydsdt�Zzdudv�Z{dwd$�Z|n6ejbdYk�r�dxdy�Z}dzd$�Z|d{d|�Z~d}dv�Z{nerZ|esZ{dS)�a� An extensible library for opening URLs using a variety of protocols The simplest way to use this module is to call the urlopen function, which accepts a string containing a URL or a Request object (described below). It opens the URL and returns the results as file-like object; the returned object has some extra methods described below. The OpenerDirector manages a collection of Handler objects that do all the actual work. Each Handler implements a particular protocol or option. The OpenerDirector is a composite object that invokes the Handlers needed to open the requested URL. For example, the HTTPHandler performs HTTP GET and POST requests and deals with non-error returns. The HTTPRedirectHandler automatically deals with HTTP 301, 302, 303 and 307 redirect errors, and the HTTPDigestAuthHandler deals with digest authentication. urlopen(url, data=None) -- Basic usage is the same as original urllib. pass the url and optionally data to post to an HTTP URL, and get a file-like object back. One difference is that you can also pass a Request instance instead of URL. Raises a URLError (subclass of OSError); for HTTP errors, raises an HTTPError, which can also be treated as a valid response. build_opener -- Function that creates a new OpenerDirector instance. Will install the default handlers. Accepts one or more Handlers as arguments, either instances or Handler classes that it will instantiate. If one of the argument is a subclass of the default handler, the argument will be installed instead of the default. install_opener -- Installs a new opener as the default opener. objects of interest: OpenerDirector -- Sets up the User Agent as the Python-urllib client and manages the Handler classes, while dealing with requests and responses. Request -- An object that encapsulates the state of a request. The state can be as simple as the URL. It can also include extra HTTP headers, e.g. a User-Agent. BaseHandler -- internals: BaseHandler and parent _call_chain conventions Example usage: import urllib.request # set up authentication info authinfo = urllib.request.HTTPBasicAuthHandler() authinfo.add_password(realm='PDQ Application', uri='https://mahler:8092/site-updates.py', user='klem', passwd='geheim$parole') proxy_support = urllib.request.ProxyHandler({"http" : "http://ahad-haam:3128"}) # build a new opener that adds authentication and caching FTP handlers opener = urllib.request.build_opener(proxy_support, authinfo, urllib.request.CacheFTPHandler) # install it urllib.request.install_opener(opener) f = urllib.request.urlopen('http://www.python.org/') �N)�URLError� HTTPError�ContentTooShortError)�urlparse�urlsplit�urljoin�unwrap�quote�unquote� splittype� splithost� splitport� splituser�splitpasswd� splitattr� splitquery� splitvalue�splittag�to_bytes�unquote_to_bytes� urlunparse)� addinfourl�addclosehookFT�Request�OpenerDirector�BaseHandler�HTTPDefaultErrorHandler�HTTPRedirectHandler�HTTPCookieProcessor�ProxyHandler�HTTPPasswordMgr�HTTPPasswordMgrWithDefaultRealm�HTTPPasswordMgrWithPriorAuth�AbstractBasicAuthHandler�HTTPBasicAuthHandler�ProxyBasicAuthHandler�AbstractDigestAuthHandler�HTTPDigestAuthHandler�ProxyDigestAuthHandler�HTTPHandler�FileHandler� FTPHandler�CacheFTPHandler�DataHandler�UnknownHandler�HTTPErrorProcessor�urlopen�install_opener�build_opener�pathname2url�url2pathname� getproxies�urlretrieve� urlcleanup� URLopener�FancyURLopenerz%d.%d�)�cafile�capath� cadefault�contextc Cs�|s|s|rfddl}|jdtd�|dk r2td��ts>td��tjtjj||d�}t |d�}t |�} n0|r~t |d�}t |�} ntdkr�t �a} nt} | j|||�S) a$ Open the URL url, which can be either a string or a Request object. *data* must be an object specifying additional data to be sent to the server, or None if no such data is needed. See Request for details. urllib.request module uses HTTP/1.1 and includes a "Connection:close" header in its HTTP requests. The optional *timeout* parameter specifies a timeout in seconds for blocking operations like the connection attempt (if not specified, the global default timeout setting will be used). This only works for HTTP, HTTPS and FTP connections. If *context* is specified, it must be a ssl.SSLContext instance describing the various SSL options. See HTTPSConnection for more details. The optional *cafile* and *capath* parameters specify a set of trusted CA certificates for HTTPS requests. cafile should point to a single file containing a bundle of CA certificates, whereas capath should point to a directory of hashed certificate files. More information can be found in ssl.SSLContext.load_verify_locations(). The *cadefault* parameter is ignored. This function always returns an object which can work as a context manager and has methods such as * geturl() - return the URL of the resource retrieved, commonly used to determine if a redirect was followed * info() - return the meta-information of the page, such as headers, in the form of an email.message_from_string() instance (see Quick Reference to HTTP Headers) * getcode() - return the HTTP status code of the response. Raises URLError on errors. For HTTP and HTTPS URLs, this function returns a http.client.HTTPResponse object slightly modified. In addition to the three new methods above, the msg attribute contains the same information as the reason attribute --- the reason phrase returned by the server --- instead of the response headers as it is specified in the documentation for HTTPResponse. For FTP, file, and data URLs and requests explicitly handled by legacy URLopener and FancyURLopener classes, this function returns a urllib.response.addinfourl object. Note that None may be returned if no handler handles the request (though the default installed global OpenerDirector uses UnknownHandler to ensure this never happens). In addition, if proxy settings are detected (for example, when a *_proxy environment variable like http_proxy is set), ProxyHandler is default installed and makes sure the requests are handled through the proxy. rNzJcafile, capath and cadefault are deprecated, use a custom context instead.r:zDYou can't pass both context and any of cafile, capath, and cadefaultzSSL support not available)r;r<)r>) �warnings�warn�DeprecationWarning� ValueError� _have_ssl�sslZcreate_default_contextZPurposeZSERVER_AUTH�HTTPSHandlerr2�_opener�open) �url�data�timeoutr;r<r=r>r?Z https_handler�opener�rL�&/usr/lib64/python3.6/urllib/request.pyr0�s*< cCs|adS)N)rF)rKrLrLrMr1�scCs4t|�\}}tjt||����}|j�}|dkrD|rDtjj|�|fS|rTt|d�}nt j dd�}|j}tj |�|��||f} d } d}d}d} d |kr�t|d �}|r�|| | |�xB|j| �}|s�P|t|�7}|j|�| d7} |r�|| | |�q�WWdQRXWdQRX|dk�r0||k�r0td||f| ��| S)aW Retrieve a URL into a temporary location on disk. Requires a URL argument. If a filename is passed, it is used as the temporary file location. The reporthook argument should be a callable that accepts a block number, a read size, and the total file size of the URL target. The data argument should be valid URL encoded data. If a filename is passed and the URL points to a local resource, the result is a copy from local file to new file. Returns a tuple containing the path to the newly created data file as well as the resulting HTTPMessage object. �file�wbF)�deletei��rzcontent-lengthzContent-LengthNz1retrieval incomplete: got only %i out of %i bytesi ���)r� contextlib�closingr0�info�os�path�normpathrG�tempfileZNamedTemporaryFile�name�_url_tempfiles�append�int�read�len�writer)rH�filename� reporthookrIZurl_typerX�fp�headers�tfp�result�bs�sizer_�blocknum�blockrLrLrMr6�sD $cCsHx0tD](}ytj|�Wqtk r,YqXqWtdd�=trDdadS)z0Clean up temporary files from urlretrieve calls.N)r\rW�unlink�OSErrorrF)Z temp_filerLrLrMr7%s z:\d+$cCs<|j}t|�d}|dkr&|jdd�}tjd|d�}|j�S)z�Return request-host, as defined by RFC 2965. Variation from RFC: returned value is lowercased, for convenient comparison. rR��Host)�full_urlr� get_header�_cut_port_re�sub�lower)�requestrH�hostrLrLrM�request_host4srwc@s�eZdZdidddfdd�Zedd��Zejdd��Zejdd��Zed d ��Zejdd ��Zejdd ��Zd d�Z dd�Z dd�Zdd�Zdd�Z dd�Zdd�Zdd�Zd#dd�Zdd �Zd!d"�ZdS)$rNFc Csp||_i|_i|_d|_||_d|_x |j�D]\}}|j||�q.W|dkrVt|�}||_ ||_ |rl||_dS)N)rpre�unredirected_hdrs�_datarI�_tunnel_host�items� add_headerrw�origin_req_host�unverifiable�method) �selfrHrIrer}r~r�key�valuerLrLrM�__init__FszRequest.__init__cCs|jrdj|j|j�S|jS)Nz{}#{})�fragment�format� _full_url)r�rLrLrMrpXszRequest.full_urlcCs(t|�|_t|j�\|_|_|j�dS)N)rr�rr��_parse)r�rHrLrLrMrp^s cCsd|_d|_d|_dS)Nrn)r�r��selector)r�rLrLrMrpescCs|jS)N)ry)r�rLrLrMrIkszRequest.datacCs(||jkr$||_|jd�r$|jd�dS)NzContent-length)ry� has_header� remove_header)r�rIrLrLrMrIos cCs d|_dS)N)rI)r�rLrLrMrIyscCsNt|j�\|_}|jdkr(td|j��t|�\|_|_|jrJt|j�|_dS)Nzunknown url type: %r) rr��typerBrprrvr�r )r��restrLrLrMr�}s zRequest._parsecCs|jdk rdnd}t|d|�S)z3Return a string indicating the HTTP request method.N�POST�GETr)rI�getattr)r�Zdefault_methodrLrLrM� get_method�szRequest.get_methodcCs|jS)N)rp)r�rLrLrM�get_full_url�szRequest.get_full_urlcCs4|jdkr|jr|j|_n||_|j|_||_dS)N�https)r�rzrvrpr�)r�rvr�rLrLrM� set_proxy�s zRequest.set_proxycCs|j|jkS)N)r�rp)r�rLrLrM� has_proxy�szRequest.has_proxycCs||j|j�<dS)N)re� capitalize)r�r��valrLrLrMr|�szRequest.add_headercCs||j|j�<dS)N)rxr�)r�r�r�rLrLrM�add_unredirected_header�szRequest.add_unredirected_headercCs||jkp||jkS)N)rerx)r��header_namerLrLrMr��s zRequest.has_headercCs|jj||jj||��S)N)re�getrx)r�r��defaultrLrLrMrq�szRequest.get_headercCs |jj|d�|jj|d�dS)N)re�poprx)r�r�rLrLrMr��szRequest.remove_headercCs"|jj�}|j|j�t|j��S)N)rx�copy�updatere�listr{)r��hdrsrLrLrM�header_items�s zRequest.header_items)N)�__name__� __module__�__qualname__r��propertyrp�setter�deleterrIr�r�r�r�r�r|r�r�rqr�r�rLrLrLrMrDs( c@sNeZdZdd�Zdd�Zdd�Zdd�Zd ejfd d�Z ddd �Z dd�Zd S)rcCs6dt}d|fg|_g|_i|_i|_i|_i|_dS)NzPython-urllib/%sz User-agent)�__version__� addheaders�handlers�handle_open�handle_error�process_response�process_request)r�Zclient_versionrLrLrMr��szOpenerDirector.__init__cCsZt|d�stdt|���d}�xt|�D�]}|dkr:q*|jd�}|d|�}||dd�}|jd �r�|jd�|d}||dd�}yt|�}Wntk r�YnX|jj |i�} | |j|<n>|d kr�|}|j } n*|dkr�|}|j} n|dkr*|}|j} nq*| j |g�} | �r&tj| |�n | j|�d }q*W|�rVtj|j|�|j|�dS)N� add_parentz%expected BaseHandler instance, got %rF�redirect_request�do_open� proxy_open�_rR�errorrG�responseruT)r�r�r�)�hasattr� TypeErrorr��dir�find� startswithr^rBr�r�r�r�r�� setdefault�bisectZinsortr]r�r�)r��handlerZadded�meth�i�protocolZ condition�j�kind�lookupr�rLrLrM�add_handler�sJ zOpenerDirector.add_handlercCsdS)NrL)r�rLrLrM�close�szOpenerDirector.closec Gs<|j|f�}x*|D]"}t||�}||�}|dk r|SqWdS)N)r�r�) r��chainr�� meth_name�argsr�r��funcrgrLrLrM�_call_chain�s zOpenerDirector._call_chainNc Cs�t|t�rt||�}n|}|dk r(||_||_|j}|d}x(|jj|g�D]}t||�}||�}qLW|j ||�} |d}x*|j j|g�D]}t||�}||| �} q�W| S)NZ_requestZ _response)� isinstance�strrrIrJr�r�r�r��_openr�) r��fullurlrIrJ�reqr�r�Z processorr�r�rLrLrMrG�s" zOpenerDirector.opencCsP|j|jdd|�}|r|S|j}|j|j||d|�}|r>|S|j|jdd|�S)Nr�Zdefault_openr��unknown�unknown_open)r�r�r�)r�r�rIrgr�rLrLrMr�s zOpenerDirector._opencGs~|d kr,|jd}|d}d|}d}|}n|j}|d}d}|||f|}|j|�}|r^|S|rz|dd f|}|j|�SdS)N�httpr�r:z http_error_%srRZ_errorrr��http_error_default)r�r�)r�r�)r��protor��dictr�Zhttp_errZ orig_argsrgrLrLrMr�'s zOpenerDirector.error)N)r�r�r�r�r�r�r��socket�_GLOBAL_DEFAULT_TIMEOUTrGr�r�rLrLrLrMr�s/ c Gs�t�}ttttttttt g }t tjd�r2|j t�t�}xN|D]F}x@|D]8}t|t�rlt||�r�|j|�qHt||�rH|j|�qHWq>Wx|D]}|j|�q�Wx|D]}|j|��q�Wx&|D]}t|t�r�|�}|j|�q�W|S)a*Create an opener object from a list of handlers. The opener will use several default handlers, including support for HTTP, FTP and when applicable HTTPS. If any of the handlers passed as arguments are subclasses of the default handlers, the default handlers will not be used. �HTTPSConnection)rrr.r)rrr+r*r/r-r�r��clientr]rE�setr�r�� issubclass�add�remover�)r�rKZdefault_classes�skip�klassZcheck�hrLrLrMr2@s0 c@s(eZdZdZdd�Zdd�Zdd�ZdS) ri�cCs ||_dS)N)�parent)r�r�rLrLrMr�gszBaseHandler.add_parentcCsdS)NrL)r�rLrLrMr�jszBaseHandler.closecCst|d�sdS|j|jkS)N� handler_orderT)r�r�)r��otherrLrLrM�__lt__ns zBaseHandler.__lt__N)r�r�r�r�r�r�r�rLrLrLrMrdsc@s eZdZdZdZdd�ZeZdS)r/zProcess HTTP error responses.i�cCsJ|j|j|j�}}}d|ko*dknsF|jjd|||||�}|S)N��i,r�)�code�msgrVr�r�)r�rur�r�r�r�rLrLrM� http_response{s z HTTPErrorProcessor.http_responseN)r�r�r��__doc__r�r��https_responserLrLrLrMr/wsc@seZdZdd�ZdS)rcCst|j||||��dS)N)rrp)r�r�rdr�r�r�rLrLrMr��sz*HTTPDefaultErrorHandler.http_error_defaultN)r�r�r�r�rLrLrLrMr�sc@s4eZdZdZdZdd�Zdd�ZeZZZ dZ dS) r�� c sx|j�}|dkr|dkp&|dko&|dks:t|j||||��|jdd �}d�t�fdd �|jj�D��}t|||jdd�S)a�Return a Request or None in response to a redirect. This is called by the http_error_30x methods when a redirection response is received. If a redirection should take place, return a new Request to allow http_error_30x to perform the redirect. Otherwise, raise HTTPError if no-one else should try to handle this url. Return None if you can't but another Handler might. �-�.�/�3r��HEADr�� z%20�content-length�content-typec3s&|]\}}|j��kr||fVqdS)N)rt)�.0�k�v)�CONTENT_HEADERSrLrM� <genexpr>�sz7HTTPRedirectHandler.redirect_request.<locals>.<genexpr>T)rer}r~)r�r�r�r�)r�r�)r�r�r�)r�r�) r�rrp�replacer�rer{rr}) r�r�rdr�r�re�newurl�mZ newheadersrL)r�rMr��s z$HTTPRedirectHandler.redirect_requestc CsNd|kr|d}nd|kr$|d}ndSt|�}|jdkrRt||d||f||��|jrp|jrpt|�}d|d <t|�}t|d tj d�}t |j|�}|j||||||�}|dkr�dSt |d��r|j} |_| j|d �|jks�t| �|jk�rt|j||j|||��ni} |_|_| j|d �d| |<|j�|j�|jj||jd�S)N�location�urir�r��ftprnz+%s - Redirection to url '%s' is not allowed�/r:z iso-8859-1)�encoding�safe� redirect_dictrrR)rJ)r�r�r�rn)r�schemerrXZnetlocr�rr �stringZpunctuationrrpr�r�rr��max_repeatsr`�max_redirections�inf_msgr_r�r�rGrJ) r�r�rdr�r�rer��urlparts�newZvisitedrLrLrM�http_error_302�s@ z"HTTPRedirectHandler.http_error_302zoThe HTTP server returned a redirect error that would lead to an infinite loop. The last 30x error message was: N)r�r�r�rrr�r �http_error_301�http_error_303�http_error_307rrLrLrLrMr�s&<c Cs�t|�\}}|jd�s d}|}n:|jd�s6td|��|jdd�}|dkrNd}|d|�}t|�\}}|dk r|t|�\}}nd}}||||fS)aReturn (scheme, user, password, host/port) given a URL or an authority. If a URL is supplied, it must have an authority (host:port) component. According to RFC 3986, having an authority component means the URL must have two slashes after the scheme. r�Nz//zproxy URL with no authority: %rr:rRrS)rr�rBr�rr) �proxyrZr_scheme� authority�endZuserinfo�hostport�user�passwordrLrLrM�_parse_proxy�s rc@s"eZdZdZddd�Zdd�ZdS)r�dNcCsL|dkrt�}||_x2|j�D]&\}}t|d||||jfdd��qWdS)Nz%s_opencSs||||�S)NrL)�rrr�r�rLrLrM�<lambda>%sz'ProxyHandler.__init__.<locals>.<lambda>)r5�proxiesr{�setattrr�)r�rr�rHrLrLrMr�s zProxyHandler.__init__cCs�|j}t|�\}}}}|dkr"|}|jr6t|j�r6dS|rv|rvdt|�t|�f} tj| j��jd�} |j dd| �t|�}|j ||�||ks�|dkr�dS|jj||j d�SdS)Nz%s:%s�asciizProxy-authorizationzBasic r�)rJ)r�rrv�proxy_bypassr �base64� b64encode�encode�decoder|r�r�rGrJ)r�r�rr�Z orig_typeZ proxy_typerrrZ user_passZcredsrLrLrMr�(s zProxyHandler.proxy_open)N)r�r�r�r�r�r�rLrLrLrMrs c@s6eZdZdd�Zdd�Zdd�Zd dd �Zd d�ZdS)r cCs i|_dS)N)�passwd)r�rLrLrMr�FszHTTPPasswordMgr.__init__cs`t|t�r|g}|�jkr$i�j|<x6dD].�t��fdd�|D��}||f�j||<q*WdS)NTFcsg|]}�j|���qSrL)� reduce_uri)r��u)�default_portr�rLrM� <listcomp>Qsz0HTTPPasswordMgr.add_password.<locals>.<listcomp>)TF)r�r�r �tuple)r��realmr�rr �reduced_urirL)r#r�rM�add_passwordIs zHTTPPasswordMgr.add_passwordc Cs`|jj|i�}xLdD]D}|j||�}x2|j�D]&\}}x|D]}|j||�r<|Sq<Wq.WqWdS)NTF)TF)NN)r r�r!r{� is_suburi) r�r&�authuriZdomainsr#�reduced_authuriZurisZauthinfor�rLrLrM�find_user_passwordTs z"HTTPPasswordMgr.find_user_passwordTc Cs�t|�}|dr.|d}|d}|dp*d}nd}|}d}t|�\}}|r~|dkr~|dk r~ddd�j|�} | dk r~d || f}||fS) z@Accept authority or URI and extract only the authority and path.rRrr:r�N�Pi�)r�r�z%s:%d)rr r�) r�r�r#�partsrrrXrv�portZdportrLrLrMr!^s zHTTPPasswordMgr.reduce_uricCsR||krdS|d|dkr dStj|d|df�}t|�t|d�krNdSdS)zcCheck if test is below base in a URI tree Both args must be URIs in reduced form. TrFrR)� posixpath�commonprefixr`)r��base�test�commonrLrLrMr)uszHTTPPasswordMgr.is_suburiN)T)r�r�r�r�r(r,r!r)rLrLrLrMr Ds c@seZdZdd�ZdS)r!cCs0tj|||�\}}|dk r"||fStj|d|�S)N)r r,)r�r&r*rrrLrLrMr,�s z2HTTPPasswordMgrWithDefaultRealm.find_user_passwordN)r�r�r�r,rLrLrLrMr!�scs<eZdZ�fdd�Zd �fdd� Zddd�Zdd �Z�ZS)r"csi|_t�j||�dS)N)� authenticated�superr�)r�r��kwargs)� __class__rLrMr��sz%HTTPPasswordMgrWithPriorAuth.__init__Fcs<|j||�|dk r&t�jd|||�t�j||||�dS)N)�update_authenticatedr6r()r�r&r�rr �is_authenticated)r8rLrMr(�sz)HTTPPasswordMgrWithPriorAuth.add_passwordcCsFt|t�r|g}x0dD](}x"|D]}|j||�}||j|<q WqWdS)NTF)TF)r�r�r!r5)r�r�r:r#r"r'rLrLrMr9�s z1HTTPPasswordMgrWithPriorAuth.update_authenticatedcCsDx>dD]6}|j||�}x$|jD]}|j||�r|j|SqWqWdS)NTF)TF)r!r5r))r�r*r#r+r�rLrLrMr:�s z-HTTPPasswordMgrWithPriorAuth.is_authenticated)F)F)r�r�r�r�r(r9r:� __classcell__rLrL)r8rMr"�s c@sTeZdZejdej�Zddd�Zdd�Zdd�Z d d �Z dd�Zd d�ZeZ eZdS)r#z1(?:^|,)[ ]*([^ ,]+)[ ]+realm=(["']?)([^"']*)\2NcCs"|dkrt�}||_|jj|_dS)N)r r r()r�Zpassword_mgrrLrLrMr��sz!AbstractBasicAuthHandler.__init__ccstd}xFtjj|�D]6}|j�\}}}|d kr:tjdtd�||fVd}qW|sp|rb|j�d}nd}|dfVdS) NF�"�'zBasic Auth Realm was unquoted�Trrn)r<r=)r#�rx�finditer�groupsr?r@�UserWarning�split)r��headerZfound_challengeZmorr r&rLrLrM�_parse_realm�s z%AbstractBasicAuthHandler._parse_realmc Cs~|j|�}|sdSd}xL|D]D}x>|j|�D]0\}}|j�dkrF|}q,|dk r,|j|||�Sq,WqW|dk rztd|f��dS)N�basiczBAbstractBasicAuthHandler does not support the following scheme: %r)Zget_allrErt�retry_http_basic_authrB) r��authreqrvr�reZunsupportedrDrr&rLrLrM�http_error_auth_reqed�s z.AbstractBasicAuthHandler.http_error_auth_reqedcCs||jj||�\}}|dk rtd||f}dtj|j��jd�}|j|jd�|krTdS|j|j|�|j j ||jd�SdSdS)Nz%s:%szBasic r)rJ)r r,rrrrrq�auth_headerr�r�rGrJ)r�rvr�r&r�pw�raw�authrLrLrMrG�sz.AbstractBasicAuthHandler.retry_http_basic_authcCsxt|jd�s|jj|j�r"|S|jd�st|jjd|j�\}}dj||�j�}tj |�j �}|jddj|j���|S)Nr:� Authorizationz{0}:{1}zBasic {}) r�r r:rpr�r,r�rrZstandard_b64encoderr��strip)r�r�rr ZcredentialsZauth_strrLrLrM�http_requests z%AbstractBasicAuthHandler.http_requestcCsLt|jd�rHd|jko dknr8|jj|jd�n|jj|jd�|S)Nr:r�i,TF)r�r r�r9rp)r�r�r�rLrLrMr�s z&AbstractBasicAuthHandler.http_response)N)r�r�r��re�compile�Ir?r�rErIrGrPr�� https_requestr�rLrLrLrMr#�s c@seZdZdZdd�ZdS)r$rNcCs|j}|jd|||�}|S)Nzwww-authenticate)rprI)r�r�rdr�r�rerHr�rLrLrM�http_error_401 s z#HTTPBasicAuthHandler.http_error_401N)r�r�r�rJrUrLrLrLrMr$sc@seZdZdZdd�ZdS)r%zProxy-authorizationcCs|j}|jd|||�}|S)Nzproxy-authenticate)rvrI)r�r�rdr�r�rerr�rLrLrM�http_error_407+s z$ProxyBasicAuthHandler.http_error_407N)r�r�r�rJrVrLrLrLrMr%'sc@sNeZdZddd�Zdd�Zdd�Zdd �Zd d�Zdd �Zdd�Z dd�Z dS)r&NcCs4|dkrt�}||_|jj|_d|_d|_d|_dS)Nr)r r r(�retried�nonce_count� last_nonce)r�r rLrLrMr�Es z"AbstractDigestAuthHandler.__init__cCs d|_dS)Nr)rW)r�rLrLrM�reset_retry_countNsz+AbstractDigestAuthHandler.reset_retry_countcCs||j|d�}|jdkr*t|jdd|d��n|jd7_|rx|j�d}|j�dkr`|j||�S|j�dkrxtd|��dS) N�i�zdigest auth failedrRrZdigestrFzEAbstractDigestAuthHandler does not support the following scheme: '%s')r�rWrrprCrt�retry_http_digest_authrB)r�rJrvr�rerHrrLrLrMrIQs z/AbstractDigestAuthHandler.http_error_auth_reqedcCsz|jdd�\}}ttdt|���}|j||�}|rvd|}|jj|jd�|krRdS|j|j|�|j j ||jd�}|SdS)Nr�rRz Digest %s)rJ)rC�parse_keqv_list�filter�parse_http_list�get_authorizationrer�rJr�r�rGrJ)r�r�rM�tokenZ challenge�chalZauth_valZresprLrLrMr\esz0AbstractDigestAuthHandler.retry_http_digest_authcCs@d|j|tj�f}|jd�td�}tj|�j�}|dd�S)Nz %s:%s:%s:rrQ�)rX�timeZctimer�_randombytes�hashlib�sha1� hexdigest)r��nonce�s�b�digrLrLrM� get_cnonceqsz$AbstractDigestAuthHandler.get_cnoncecCs�y6|d}|d}|jd�}|jdd�}|jdd�}Wntk rJdSX|j|�\}} |dkrfdS|jj||j�\} }| dkr�dS|jdk r�|j|j|�}nd}d| ||f} d|j�|j f}|d k�r.||j kr�|jd 7_nd |_||_ d|j}|j|�}d||||||�f}| || �|�}n2|dk�rT| || �d|||�f�}nt d |��d| |||j |f}|�r�|d|7}|�r�|d|7}|d|7}|�r�|d||f7}|S)Nr&ri�qop� algorithm�MD5�opaquez%s:%s:%sz%s:%srMrRz%08xz%s:%s:%s:%s:%szqop '%s' is not supported.z>username="%s", realm="%s", nonce="%s", uri="%s", response="%s"z , opaque="%s"z , digest="%s"z, algorithm="%s"z, qop=auth, nc=%s, cnonce="%s")r��KeyError�get_algorithm_implsr r,rprI�get_entity_digestr�r�rYrXrmr)r�r�rbr&rirnrorq�H�KDrrKZentdigZA1ZA2ZncvalueZcnonceZnoncebitZrespdigr2rLrLrMr`|sV z+AbstractDigestAuthHandler.get_authorizationcsD|dkrdd��n|dkr$dd��ntd|���fdd�}�|fS)NrpcSstj|jd��j�S)Nr)rfZmd5rrh)�xrLrLrMr�sz?AbstractDigestAuthHandler.get_algorithm_impls.<locals>.<lambda>ZSHAcSstj|jd��j�S)Nr)rfrgrrh)rwrLrLrMr�sz.Unsupported digest authentication algorithm %rcs�d||f�S)Nz%s:%srL)rj�d)rurLrMr�s)rB)r�rorvrL)rurMrs�s z-AbstractDigestAuthHandler.get_algorithm_implscCsdS)NrL)r�rIrbrLrLrMrt�sz+AbstractDigestAuthHandler.get_entity_digest)N)r�r�r�r�rZrIr\rmr`rsrtrLrLrLrMr&:s < c@s eZdZdZdZdZdd�ZdS)r'z�An authentication protocol defined by RFC 2069 Digest authentication improves on basic authentication because it does not transmit passwords in the clear. rNi�cCs*t|j�d}|jd|||�}|j�|S)NrRzwww-authenticate)rrprIrZ)r�r�rdr�r�rerv�retryrLrLrMrU�s z$HTTPDigestAuthHandler.http_error_401N)r�r�r�r�rJr�rUrLrLrLrMr'�sc@seZdZdZdZdd�ZdS)r(zProxy-Authorizationi�cCs"|j}|jd|||�}|j�|S)Nzproxy-authenticate)rvrIrZ)r�r�rdr�r�rervryrLrLrMrV�s z%ProxyDigestAuthHandler.http_error_407N)r�r�r�rJr�rVrLrLrLrMr(�sc@s6eZdZd dd�Zdd�Zdd�Zdd �Zd d�ZdS)�AbstractHTTPHandlerrcCs ||_dS)N)�_debuglevel)r�� debuglevelrLrLrMr��szAbstractHTTPHandler.__init__cCs ||_dS)N)r{)r��levelrLrLrM�set_http_debuglevel�sz'AbstractHTTPHandler.set_http_debuglevelcCstjjj|j|j��S)N)r�r��HTTPConnection�_get_content_lengthrIr�)r�rurLrLrMr��sz'AbstractHTTPHandler._get_content_lengthcCs |j}|std��|jdk r�|j}t|t�r8d}t|��|jd�sN|jdd�|jd�r�|jd�r�|j|�}|dk r�|jdt|��n|jdd�|}|j �r�t |j�\}}t|�\}} |jd�s�|jd|�x2|j jD]&\} }| j�} |j| �s�|j| |�q�W|S) Nz no host givenz\POST data should be bytes, an iterable of bytes, or a file object. It cannot be of type str.zContent-typez!application/x-www-form-urlencodedzContent-lengthzTransfer-encodingZchunkedro)rvrrIr�r�r�r�r�r�r�rr�rr�r�r�)r�rurvrIr�Zcontent_lengthZsel_hostrZselZsel_pathr[r�rLrLrM�do_request_�s> zAbstractHTTPHandler.do_request_c s\|j}|std��||fd|ji|��}|j|j�t|j���jt�fdd�|jj �D���d�d<tdd��j �D���|j r�i}d}|�kr��|||<�|=|j|j |d �y`y&|j|j �|j|j�|jd �d�Wn,tk �r }zt|��WYdd}~XnX|j�} Wn|j��YnX|j�rF|jj�d|_|j�| _| j| _| S) z�Return an HTTPResponse object for the request, using http_class. http_class must implement the HTTPConnection API from http.client. z no host givenrJc3s"|]\}}|�kr||fVqdS)NrL)r�r�r�)rerLrMr�)sz.AbstractHTTPHandler.do_open.<locals>.<genexpr>r�� Connectioncss|]\}}|j�|fVqdS)N)�title)r�r[r�rLrLrMr�6szProxy-Authorization)rezTransfer-encoding)Zencode_chunkedN)rvrrJZset_debuglevelr{r�rxr�rer{rzZ set_tunnelrur�r�rIr�rm�getresponser�Zsockr�rH�reasonr�) r�Z http_classr�Zhttp_conn_argsrvr�Ztunnel_headersZproxy_auth_hdr�errrrL)rerMr�s@ " zAbstractHTTPHandler.do_openN)r)r�r�r�r�r~r�r�r�rLrLrLrMrz�s &rzc@seZdZdd�ZejZdS)r)cCs|jtjj|�S)N)r�r�r�r)r�r�rLrLrM� http_open`szHTTPHandler.http_openN)r�r�r�r�rzr�rPrLrLrLrMr)^sr�c@s$eZdZddd�Zdd�ZejZdS)rErNcCstj||�||_||_dS)N)rzr��_context�_check_hostname)r�r|r>�check_hostnamerLrLrMr�iszHTTPSHandler.__init__cCs|jtjj||j|jd�S)N)r>r�)r�r�r�r�r�r�)r�r�rLrLrM� https_opennszHTTPSHandler.https_open)rNN)r�r�r�r�r�rzr�rTrLrLrLrMrEgs rEc@s.eZdZddd�Zdd�Zdd�ZeZeZdS) rNcCs$ddl}|dkr|jj�}||_dS)Nr)Zhttp.cookiejar� cookiejarZ CookieJar)r�r�r�rLrLrMr�ws zHTTPCookieProcessor.__init__cCs|jj|�|S)N)r�Zadd_cookie_header)r�rurLrLrMrP}sz HTTPCookieProcessor.http_requestcCs|jj||�|S)N)r�Zextract_cookies)r�rur�rLrLrMr��sz!HTTPCookieProcessor.http_response)N)r�r�r�r�rPr�rTr�rLrLrLrMrvs c@seZdZdd�ZdS)r.cCs|j}td|��dS)Nzunknown url type: %s)r�r)r�r�r�rLrLrMr��szUnknownHandler.unknown_openN)r�r�r�r�rLrLrLrMr.�scCsRi}xH|D]@}|jdd�\}}|ddkrB|ddkrB|dd�}|||<q W|S)z>Parse list of key=value strings where keys are not duplicated.�=rRrr<rSrS)rC)�lZparsedZeltr�r�rLrLrMr]�s r]cCs�g}d}d}}xt|D]l}|r,||7}d}q|rV|dkr@d}qn|dkrLd}||7}q|dkrn|j|�d}q|dkrzd}||7}qW|r�|j|�dd�|D�S) apParse lists as described by RFC 2068 Section 2. In particular, parse comma-separated lists where the elements of the list may include quoted-strings. A quoted-string could contain a comma. A non-quoted string could have quotes in the middle. Neither commas nor quotes count if they are escaped. Only double-quotes count, not single-quotes. rnF�\Tr<�,cSsg|]}|j��qSrL)rO)r��partrLrLrMr$�sz#parse_http_list.<locals>.<listcomp>)r])rj�resr��escaper ZcurrLrLrMr_�s4 r_c@s(eZdZdd�ZdZdd�Zdd�ZdS)r*cCs\|j}|dd�dkrN|dd�dkrN|jrN|jdkrN|j|j�krXtd��n |j|�SdS)Nr:z//r>r�� localhostz-file:// scheme is supported only on localhost)r�rv� get_namesr�open_local_file)r�r�rHrLrLrM� file_open�s& zFileHandler.file_openNcCs`tjdkrZy*ttjd�dtjtj��d�t_Wn$tjk rXtjd�ft_YnXtjS)Nr�r:)r*�namesr%r��gethostbyname_ex�gethostname�gaierror� gethostbyname)r�rLrLrMr��s zFileHandler.get_namescCsddl}ddl}|j}|j}t|�}y�tj|�}|j}|jj |j dd�} |j|�d} |jd| pbd|| f�}|r~t |�\}}|s�|r�t|�|j�kr�|r�d||} nd|} tt|d�|| �SWn*tk r�}zt|��WYdd}~XnXtd��dS) NrT)�usegmtz6Content-type: %s Content-length: %d Last-modified: %s z text/plainzfile://�rbzfile not on local host)�email.utils� mimetypesrvr�r4rW�stat�st_size�utils� formatdate�st_mtime� guess_type�message_from_stringr �_safe_gethostbynamer�rrGrmr)r�r��emailr�rvrbZ localfile�statsri�modified�mtyperer/Zorigurl�exprLrLrMr��s0 zFileHandler.open_local_file)r�r�r�r�r�r�r�rLrLrLrMr*�s cCs&y tj|�Stjk r dSXdS)N)r�r�r�)rvrLrLrMr��s r�c@seZdZdd�Zdd�ZdS)r+cCs.ddl}ddl}|j}|s"td��t|�\}}|dkr>|j}nt|�}t|�\}}|rdt|�\}}nd}t |�}|pvd}|p~d}yt j|�}Wn*tk r�}zt|��WYdd}~XnXt |j�\} } | jd�}ttt |��}|dd�|d}}|�r|d�r|dd�}y�|j||||||j�} |�r8d�p:d}x:| D]2}t|�\}}|j�dk�rB|dk�rB|j�}�qBW| j||�\}}d}|j|j�d}|�r�|d |7}|dk �r�|dk�r�|d|7}tj|�}t|||j�S|jk �r(}z$td|�}|jtj �d��WYdd}~XnXdS)Nrzftp error: no host givenrnr�rRrS�Dr��a�Ar�rxzContent-type: %s zContent-length: %d z ftp error: %rr:rSrS)r�r�r�rSrxr�)!�ftplibr�rvrr �FTP_PORTr^rrr r�r�rmrr�rCr��map�connect_ftprJrrt�upper�retrfiler�rpr�r�r� all_errors�with_traceback�sys�exc_info)r�r�r�r�rvr/rr r�rX�attrs�dirsrN�fwr��attrr�rd�retrlenrer�r��excrLrLrM�ftp_open�s\ zFTPHandler.ftp_openc Cst||||||dd�S)NF)� persistent)� ftpwrapper)r�rr rvr/r�rJrLrLrMr�1szFTPHandler.connect_ftpN)r�r�r�r�r�rLrLrLrMr+�s5c@s<eZdZdd�Zdd�Zdd�Zdd�Zd d �Zdd�Zd S)r,cCs"i|_i|_d|_d|_d|_dS)Nr�<rc)�cacherJ�soonest�delay� max_conns)r�rLrLrMr�8s zCacheFTPHandler.__init__cCs ||_dS)N)r�)r��trLrLrM� setTimeout?szCacheFTPHandler.setTimeoutcCs ||_dS)N)r�)r�r�rLrLrM�setMaxConnsBszCacheFTPHandler.setMaxConnscCsr|||dj|�|f}||jkr4tj�|j|j|<n,t||||||�|j|<tj�|j|j|<|j�|j|S)Nr�)�joinr�rdr�rJr��check_cache)r�rr rvr/r�rJr�rLrLrMr�Es zCacheFTPHandler.connect_ftpcCs�tj�}|j|krTx@t|jj��D].\}}||kr"|j|j�|j|=|j|=q"Wtt|jj���|_t |j�|j kr�x6t|jj��D]$\}}||jkr�|j|=|j|=Pq�Wtt|jj���|_dS)N)rdr�r�rJr{r�r��min�valuesr`r�)r�r�r�r�rLrLrMr�Ps zCacheFTPHandler.check_cachecCs4x|jj�D]}|j�qW|jj�|jj�dS)N)r�r�r��clearrJ)r��connrLrLrM�clear_cacheds zCacheFTPHandler.clear_cacheN) r�r�r�r�r�r�r�r�r�rLrLrLrMr,5sc@seZdZdd�ZdS)r-cCs~|j}|jdd�\}}|jdd�\}}t|�}|jd�rNtj|�}|dd�}|sVd}tjd|t|�f�}t t j|�||�S) N�:rRr�z;base64�ztext/plain;charset=US-ASCIIz$Content-type: %s Content-length: %d i����)rprCr�endswithr�decodebytesr�r�r`r�io�BytesIO)r�r�rHrrIZ mediatypererLrLrM� data_openks zDataHandler.data_openN)r�r�r�r�rLrLrLrMr-jsr��nt)r4r3cCst|�S)zOS-specific conversion from a relative URL of the 'file' scheme to a file system path; not recommended for general use.)r )�pathnamerLrLrMr4�scCst|�S)zOS-specific conversion from a file system path to a relative URL of the 'file' scheme; not recommended for general use.)r )r�rLrLrMr3�sc@s�eZdZdZdZdeZd*dd�Zdd�Zdd �Z d d�Z dd �Zd+dd�Zd,dd�Z d-dd�Zd.dd�Zdd�Zd/dd�Zd0dd�Zdd�Zer�dd�Zd1d d!�Zd"d#�Zd$d%�Zd&d'�Zd2d(d)�ZdS)3r8a,Class to open URLs. This is a class rather than just a subroutine because we may need more than one set of global protocol-specific options. Note -- this is a base class for those who don't want the automatic handling of errors type 302 (relocated) and 401 (authorization needed).NzPython-urllib/%scKszdd|jji}tj|tdd�|dkr.t�}||_|jd�|_|jd�|_ d|j fd g|_g|_t j|_d|_t|_dS)NzW%(class)s style of invoking requests is deprecated. Use newer urlopen functions/methods�classr>)� stacklevel�key_file� cert_filez User-Agent�Accept�*/*)r�r�)r8r�r?r@rAr5rr�r�r��versionr��_URLopener__tempfilesrWrl�_URLopener__unlink� tempcache�ftpcache)r�rZx509r�rLrLrMr��szURLopener.__init__cCs|j�dS)N)r�)r�rLrLrM�__del__�szURLopener.__del__cCs|j�dS)N)�cleanup)r�rLrLrMr��szURLopener.closecCsZ|jrFx2|jD](}y|j|�Wqtk r4YqXqW|jdd�=|jrV|jj�dS)N)r�r�rmr�r�)r�rNrLrLrMr��s zURLopener.cleanupcGs|jj|�dS)zdAdd a header to be used by the HTTP interface only e.g. u.addheader('Accept', 'sound/basic')N)r�r])r�r�rLrLrM� addheader�szURLopener.addheadercCsntt|��}t|dd�}|jrL||jkrL|j|\}}t|d�}t|||�St|�\}}|s`d}||jkr�|j|}t|�\}} t| �\} }| |f}nd}d|}||_ |j dd�}t||�s�|d kr�|r�|j|||�S|j ||�Sy,|dk�rt||�|�St||�||�SWnVttfk �r.�Yn<tk �rh} ztd | �jtj�d��WYdd} ~ XnXdS)z6Use URLopener().open(file) instead of open(file, 'r').z%/:=&?~#+!$,;'@()*[]|)rr�rNNZopen_�-r�r�zsocket errorr:)rrr r�rGrrrrr�r�r��open_unknown_proxy�open_unknownr�rrrmr�r�r�)r�r�rIrbrerd�urltyperHr� proxyhostrvr�r[r�rLrLrMrG�s< zURLopener.opencCst|�\}}tdd|��dS)z/Overridable interface to open unknown URL type.z url errorzunknown url typeN)rrm)r�r�rIr�rHrLrLrMr�szURLopener.open_unknowncCs t|�\}}tdd||��dS)z/Overridable interface to open unknown URL type.z url errorzinvalid proxy for %sN)rrm)r�rr�rIr�rHrLrLrMr� szURLopener.open_unknown_proxyc Cs&tt|��}|jr&||jkr&|j|St|�\}}|dkr�|sH|dkr�y.|j|�}|j�}|j�tt|�d�|fSt k r�} zWYdd} ~ XnX|j ||�}�zH|j�} |r�t |d�}n|ddl}t|�\} }t|p�d�\} }t|p�d�\}} t |�pd�\}} tjj|�d}|j|�\}}|jj|�tj|d�}z�|| f}|jdk �r^||j|<d}d }d}d}d | k�r�t| d �}|�r�||||�xH|j|�}|�s�P|t|�7}|j|�|d7}|�r�||||��q�WWd|j�XWd|j�X|dk�r"||k�r"td||f|��|S)ztretrieve(url) returns (filename, headers) for a local object or (tempfilename, headers) for a remote object.NrNrRrOrrnirQzcontent-lengthzContent-Lengthz1retrieval incomplete: got only %i out of %i bytesi rS)rrr�rr�rVr�r4rrmrGrZrrrWrX�splitextZmkstempr�r]�fdopenr^r_r`rar)r�rHrbrcrIr�Zurl1rdr�r�rerfrZZgarbagerX�suffix�fdrgrhrir_rjrkrLrLrM�retrievesl zURLopener.retrievecCs(d}d}t|t�r<t|�\}}|r6t|�\}}t|�}|}nt|\}}t|�\}}t|�\} } | }d}| j�dkrvd}n:t| �\}} |r�t|�\}}|r�d| || f}t|�r�|}|s�tdd��|r�t|�}t j |j��jd�}nd}|�rt|�}t j |j��jd�}nd}||�} i}|�r*d||d<|�r<d||d <|�rJ||d <d|d<x|j D]\}}|||<�qZW|dk �r�d |d<| jd|||�n| jd||d�y| j�}Wn"tjjk �r�td��YnXd|jk�o�dkn�rt||jd||j�S|j||j|j|j|j|�SdS)a�Make an HTTP connection using connection_class. This is an internal method that should be called from open_http() or open_https(). Arguments: - connection_factory should take a host name and return an HTTPConnection instance. - url is the url to retrieval or a host, relative-path pair. - data is payload for a POST request or None. Nr�z %s://%s%sz http errorz no host givenrzBasic %szProxy-AuthorizationrNror�r�z!application/x-www-form-urlencodedzContent-Typer�r�)rez$http protocol error: bad status liner�i,zhttp:)r�r�rrr rrtrrmrrrrr�rur�r�r�Z BadStatusLinerZstatusrr�� http_errorrdr�)r�Zconnection_factoryrHrIZuser_passwdZproxy_passwdrvr�Zrealhostr�r�Z proxy_authrMZ http_connrerDr�r�rLrLrM�_open_generic_httpQsr zURLopener._open_generic_httpcCs|jtjj||�S)zUse HTTP protocol.)r�r�r�r)r�rHrIrLrLrM� open_http�szURLopener.open_httpc Csbd|}t||�rPt||�}|dkr6||||||�} n|||||||�} | rP| S|j|||||�S)z�Handle http errors. Derived class can override this, or provide specific handlers named http_error_DDD where DDD is the 3-digit error code.z http_error_%dN)r�r�r�) r�rHrd�errcode�errmsgrerIr[rrgrLrLrMr��s zURLopener.http_errorcCs|j�t||||d��dS)z>Default error handler: close the connection and raise OSError.N)r�r)r�rHrdr�r�rerLrLrMr��szURLopener.http_error_defaultcCstjj||j|jd�S)N)r�r�)r�r�r�r�r�)r�rvrLrLrM�_https_connection�szURLopener._https_connectioncCs|j|j||�S)zUse HTTPS protocol.)r�r�)r�rHrIrLrLrM� open_https�szURLopener.open_httpscCs^t|t�std��|dd�dkrP|dd�dkrP|dd�j�dkrPtd ��n |j|�SdS) z/Use local file or FTP depending on form of URL.zEfile error: proxy support for file protocol currently not implementedNr:z//r>r��z localhost/z-file:// scheme is supported only on localhost)r�r�rrtrBr�)r�rHrLrLrM� open_file�s 4 zURLopener.open_filecCs\ddl}ddl}t|�\}}t|�}ytj|�}Wn0tk rb}zt|j|j ��WYdd}~XnX|j } |jj|j dd�} |j|�d}|jd|p�d| | f�}|s�|} |dd�dkr�d |} tt|d �|| �St|�\}}|o�tj|�t�ft�k�rP|} |dd�dk�r d |} n|dd�dk�r>td |��tt|d �|| �Std��dS)zUse local file.rNT)r�z6Content-Type: %s Content-Length: %d Last-modified: %s z text/plainrRr�zfile://r�r:z./zAlocal file url may start with / or file:. Unknown url of type: %sz#local file error: not on local host)r�r�rr4rWr�rmr�strerrorrbr�r�r�r�r�r�rrGr r�r�r��thishostrB)r�rHr�r�rvrNZ localnamer��erir�r�reZurlfiler/rLrLrMr��s: zURLopener.open_local_filecCs�t|t�std��ddl}t|�\}}|s2td��t|�\}}t|�\}}|r\t|�\}}nd}t|�}t|ppd�}t|p|d�}t j |�}|s�ddl}|j}nt |�}t|�\}} t|�}|jd�} | dd�| d} }| o�| d�r�| dd�} | �r| d�rd| d<|||dj| �f}t|j�tk�rlx8t|j�D]*} | |k�r>|j| }|j| =|j��q>Wy�||jk�r�t||||| �|j|<|�s�d}nd }x:| D]2}t|�\}}|j�d k�r�|dk�r�|j�}�q�W|j|j||�\}}|jd|�d}d}|�r|d|7}|dk �r:|dk�r:|d|7}tj|�}t||d|�St�k �r�}z td|�j t!j"�d��WYdd}~XnXdS)zUse FTP protocol.zCftp error: proxy support for ftp protocol currently not implementedrNzftp error: no host givenrnr�rRr�rSr�r�r�r�rxzftp:zContent-Type: %s zContent-Length: %d zftp error %rr:rSrS)r�r�r�rSrxr�)#r�r�rr�rr rrr r�r�r�r�r^rrCr�r`r��MAXFTPCACHEr�r�r�rrtr�r�r�r�r�r� ftperrorsr�r�r�)r�rHr�rvrXr/rr r�r�r�rNr�r�r�r�r�r�rdr�r�rer�rLrLrM�open_ftp�sp zURLopener.open_ftpc Cs<t|t�std��y|jdd�\}}Wntk rDtdd��YnX|sNd}|jd�}|dkr�d ||d �kr�||dd �}|d |�}nd}g}|jdtj d tj tj����|jd|�|dkr�tj|j d��jd�}nt|�}|jdt|��|jd�|j|�dj|�}tj|�}tj|�}t|||�S)zUse "data" URL.zEdata error: proxy support for data protocol currently not implementedr�rRz data errorzbad data URLztext/plain;charset=US-ASCII�;rr�NrnzDate: %sz%a, %d %b %Y %H:%M:%S GMTzContent-type: %srrzlatin-1zContent-Length: %d� )r�r�rrCrBrm�rfindr]rdZstrftimeZgmtimerr�rrr r`r�r�r�r��StringIOr) r�rHrIr�Zsemirr�re�frLrLrM� open_data3s6 zURLopener.open_data)N)N)N)N)NNN)N)N)N)N)r�r�r�r�r�r�r�r�r�r�r�r�rGr�r�r�r�r�r�r�rCr�r�r�r�rrrLrLrLrMr8�s. $ B\ :c@s�eZdZdZdd�Zdd�Zd#dd�Zd d �Zd$dd�Zd%d d�Z d&dd�Z d'dd�Zd(dd�Zd)dd�Z d*dd�Zd+dd�Zd,dd�Zd-dd �Zd!d"�ZdS).r9z?Derived class with handlers for errors we can handle (perhaps).cOs(tj|f|�|�i|_d|_d|_dS)Nrr�)r8r�� auth_cache�tries�maxtries)r�r�r7rLrLrMr�`szFancyURLopener.__init__cCst||d||�S)z3Default error handling -- don't raise an exception.zhttp:)r)r�rHrdr�r�rerLrLrMr�fsz!FancyURLopener.http_error_defaultNc Csn|jd7_zR|jrJ|j|jkrJt|d�r4|j}n|j}|||dd|�S|j||||||�}|Sd|_XdS)z%Error 302 -- relocated (temporarily).rR�http_error_500i�z)Internal Server Error: Redirect RecursionNr)r rr�rr��redirect_internal) r�rHrdr�r�rerIr�rgrLrLrMr js zFancyURLopener.http_error_302c Csxd|kr|d}nd|kr$|d}ndS|j�t|jd||�}t|�}|jd krnt|||d|||��|j|�S) Nr�r�r�r�r�r�rnz( Redirection to url '%s' is not allowed.)r�r�r�rn)r�rr�rrrrG) r�rHrdr�r�rerIr�rrLrLrMr |s z FancyURLopener.redirect_internalcCs|j||||||�S)z*Error 301 -- also relocated (permanently).)r )r�rHrdr�r�rerIrLrLrMr�szFancyURLopener.http_error_301cCs|j||||||�S)z;Error 303 -- also relocated (essentially identical to 302).)r )r�rHrdr�r�rerIrLrLrMr�szFancyURLopener.http_error_303cCs2|dkr|j||||||�S|j|||||�SdS)z1Error 307 -- relocated, but turn POST into error.N)r r�)r�rHrdr�r�rerIrLrLrMr �szFancyURLopener.http_error_307Fc Cs�d|krtj||||||�|d}tjd|�} | sHtj||||||�| j�\} }| j�dkrttj||||||�|s�tj||||||�d|jd}|dkr�t||�||�St||�|||�SdS)z_Error 401 -- authentication required. This function supports Basic authentication only.zwww-authenticatez![ ]*([^ ]+)[ ]+realm="([^"]*)"rFZretry_�_basic_authN)r8r�rQ�matchrArtr�r�) r�rHrdr�r�rerIry�stuffrrr&r[rLrLrMrU�s& zFancyURLopener.http_error_401c Cs�d|krtj||||||�|d}tjd|�} | sHtj||||||�| j�\} }| j�dkrttj||||||�|s�tj||||||�d|jd}|dkr�t||�||�St||�|||�SdS)zeError 407 -- proxy authentication required. This function supports Basic authentication only.zproxy-authenticatez![ ]*([^ ]+)[ ]+realm="([^"]*)"rFZretry_proxy_rN)r8r�rQrrArtr�r�) r�rHrdr�r�rerIryrrrr&r[rLrLrMrV�s& zFancyURLopener.http_error_407cCs�t|�\}}d||}|jd}t|�\}} t| �\} } | jd�d}| |d�} |j| ||�\}} |pl| srdSdt|dd�t| dd�| f} d| | |jd<|dkr�|j|�S|j||�SdS)Nzhttp://r��@rRz%s:%s@%srn)r)rrrr��get_user_passwdr rG)r�rHr&rIrvr�r�rr�r�� proxyselectorr�rr rLrLrM�retry_proxy_http_basic_auth�s z*FancyURLopener.retry_proxy_http_basic_authcCs�t|�\}}d||}|jd}t|�\}} t| �\} } | jd�d}| |d�} |j| ||�\}} |pl| srdSdt|dd�t| dd�| f} d| | |jd<|dkr�|j|�S|j||�SdS)Nzhttps://r�rrRz%s:%s@%srn)r)rrrr�rr rG)r�rHr&rIrvr�r�rr�r�rr�rr rLrLrM�retry_proxy_https_basic_auth�s z+FancyURLopener.retry_proxy_https_basic_authc Cs�t|�\}}|jd�d}||d�}|j|||�\}}|p>|sDdSdt|dd�t|dd�|f}d||} |dkr�|j| �S|j| |�SdS)NrrRz%s:%s@%srn)rzhttp://)rr�rr rG) r�rHr&rIrvr�r�rr r�rLrLrMrG�s z$FancyURLopener.retry_http_basic_authc Cs�t|�\}}|jd�d}||d�}|j|||�\}}|p>|sDdSdt|dd�t|dd�|f}d||} |dkr�|j| �S|j| |�SdS)NrrRz%s:%s@%srn)rzhttps://)rr�rr rG) r�rHr&rIrvr�r�rr r�rLrLrM�retry_https_basic_auth s z%FancyURLopener.retry_https_basic_authrcCs`|d|j�}||jkr2|r(|j|=n |j|S|j||�\}}|sJ|rX||f|j|<||fS)Nr)rtr �prompt_user_passwd)r�rvr&r�r�rr rLrLrMr s zFancyURLopener.get_user_passwdcCsTddl}y,td||f�}|jd|||f�}||fStk rNt�dSXdS)z#Override this in a GUI environment!rNzEnter username for %s at %s: z#Enter password for %s in %s at %s: )NN)�getpass�input�KeyboardInterrupt�print)r�rvr&rrr rLrLrMr$ sz!FancyURLopener.prompt_user_passwd)N)N)N)N)NF)NF)N)N)N)N)r)r�r�r�r�r�r�r r rrr rUrVrrrGrrrrLrLrLrMr9]s$ cCstdkrtjd�atS)z8Return the IP address of the magic hostname 'localhost'.Nr�)� _localhostr�r�rLrLrLrMr�4 s r�cCsPtdkrLyttjtj��d�aWn(tjk rJttjd�d�aYnXtS)z,Return the IP addresses of the current host.Nr:r�)� _thishostr%r�r�r�r�rLrLrLrMr�< sr�cCstdkrddl}|jatS)z1Return the set of errors raised by the FTP class.Nr)� _ftperrorsr�r�)r�rLrLrMrG srcCstdkrtjd�atS)z%Return an empty email Message object.Nrn)� _noheadersr�r�rLrLrLrM� noheadersP s r c@sJeZdZdZddd�Zdd�Zdd �Zd d�Zdd �Zdd�Z dd�Z dS)r�z;Class used by open_ftp() for cache of open FTP connections.NTc CsX||_||_||_||_||_||_d|_||_y|j�Wn|j ��YnXdS)Nr) rr rvr/r�rJ�refcount� keepalive�initr�)r�rr rvr/r�rJr�rLrLrMr�] szftpwrapper.__init__cCs\ddl}d|_|j�|_|jj|j|j|j�|jj|j |j �dj|j�}|jj |�dS)Nrr�)r��busyZFTPr�Zconnectrvr/rJZloginrr r�r��cwd)r�r�Z_targetrLrLrMr#m s zftpwrapper.initc-Cs�ddl}|j�|dkr"d}d}nd|}d}y|jj|�Wn*|jk rh|j�|jj|�YnXd}|r�|r�yd|}|jj|�\}}WnR|jk r�}z4t|�dd�d kr�t d |�j tj�d��WYdd}~XnX|�s�|jjd�|�rn|jj �} zJy|jj|�Wn4|jk �rP}zt d |�|�WYdd}~XnXWd|jj| �Xd|}nd }|jj|�\}}d|_t|jd�|j�} |jd7_|j�| |fS)Nrrxr�zTYPE ArRzTYPE zRETR r>Z550z ftp error: %rr:zLIST ZLISTr�)rxr�)r��endtransferr�Zvoidcmdr�r#ZntransfercmdZ error_permr�rr�r�r��pwdr%r$r�makefile� file_closer!r�)r�rNr�r��cmd�isdirr�r�r�r'ZftpobjrLrLrMr�v sN $ zftpwrapper.retrfilecCs d|_dS)Nr)r$)r�rLrLrMr&� szftpwrapper.endtransfercCsd|_|jdkr|j�dS)NFr)r"r!� real_close)r�rLrLrMr�� s zftpwrapper.closecCs4|j�|jd8_|jdkr0|jr0|j�dS)NrRr)r&r!r"r,)r�rLrLrMr)� szftpwrapper.file_closecCs2|j�y|jj�Wnt�k r,YnXdS)N)r&r�r�r)r�rLrLrMr,� s zftpwrapper.real_close)NT)r�r�r�r�r�r#r�r&r�r)r,rLrLrLrMr�Z s -r�cCs�i}xBtjj�D]4\}}|j�}|r|dd�dkr|||dd�<qWdtjkr^|jdd�xXtjj�D]J\}}|dd�dkrj|j�}|r�|||dd �<qj|j|dd �d�qjW|S)aReturn a dictionary of scheme -> proxy server URL mappings. Scan the environment for variables named <scheme>_proxy; this seems to be the standard convention. If you need a different way, you can pass a proxies dictionary to the [Fancy]URLopener constructor. �N�_proxyZREQUEST_METHODr�i����i����i����i����i����)rW�environr{rtr�)rr[r�rLrLrM�getproxies_environment� s r0c Cs�|dkrt�}y|d}Wntk r.dSX|dkr<dSt|�\}}dd�|jd�D�}xP|D]H}|rb|jd �}tj|�}d |}tj||tj�s�tj||tj�rbdSqbWdS)z�Test if proxies should not be used for a particular host. Checks the proxy dict for the value of no_proxy, which should be a list of comma separated DNS suffixes, or '*' for all hosts. N�nor�*rRcSsg|]}|j��qSrL)rO)r�rrLrLrMr$� sz,proxy_bypass_environment.<locals>.<listcomp>r��.z (.+\.)?%s$) r0rrr rC�lstriprQr�rrS)rvrZno_proxy�hostonlyr/Z no_proxy_listr[�patternrLrLrM�proxy_bypass_environment� s& r7c Csddlm}t|�\}}dd�}d|kr4|dr4dSd}x�|jd f�D]�}|sPqFtjd |�}|dk �r|dkr�ytj|�}||�}Wntk r�wFYnX||jd��} |jd�} | dkr�d |jd�j d�d} nt | dd��} d| } || ?| | ?k�rdSqF|||�rFdSqFWdS)aj Return True iff this host shouldn't be accessed using a proxy This function uses the MacOSX framework SystemConfiguration to fetch the proxy information. proxy_settings come from _scproxy._get_proxy_settings or get mocked ie: { 'exclude_simple': bool, 'exceptions': ['foo.bar', '*.bar.com', '127.0.0.1', '10.1', '10.0/16'] } r)�fnmatchcSsh|jd�}ttt|��}t|�dkr<|ddddgdd�}|dd>|dd>B|dd>B|d BS) Nr3r�r�rRrcr:rQr>)rCr�r�r^r`)ZipAddrr.rLrLrM�ip2num s z,_proxy_bypass_macosx_sysconf.<locals>.ip2numr3Zexclude_simpleTN� exceptionsz(\d+(?:\.\d+)*)(/\d+)?rRr:rQ� F)r8r r�rQrr�r�rm�group�countr^)rv�proxy_settingsr8r5r/r:ZhostIPr�r�r2�maskrLrLrM�_proxy_bypass_macosx_sysconf� s: rA�darwin)�_get_proxy_settings�_get_proxiescCst�}t||�S)N)rCrA)rvr?rLrLrM�proxy_bypass_macosx_sysconf: srEcCst�S)z�Return a dictionary of scheme -> proxy server URL mappings. This function uses the MacOSX framework SystemConfiguration to fetch the proxy information. )rDrLrLrLrM�getproxies_macosx_sysconf> srFcCs t�}|rt||�St|�SdS)z�Return True, if host should be bypassed. Checks proxy settings gathered from the environment, if specified, or from the MacOSX framework SystemConfiguration. N)r0r7rE)rvrrLrLrMrH s rcCst�p t�S)N)r0rFrLrLrLrMr5U scCsi}yddl}Wntk r$|SXy�|j|jd�}|j|d�d}|r�t|j|d�d�}d|kr�x�|jd�D]4}|jdd�\}}tjd |�s�d ||f}|||<qrWn>|dd�dkr�||d <n$d||d <d||d<d||d<|j �Wnt ttfk �rYnX|S)zxReturn a dictionary of scheme -> proxy server URL mappings. Win32 uses the registry to store proxies. rNz;Software\Microsoft\Windows\CurrentVersion\Internet Settings�ProxyEnableZProxyServerr�rrRz^([^/:]+)://z%s://%sr[zhttp:r�z http://%sz https://%sr�zftp://%sr�) �winreg�ImportError�OpenKey�HKEY_CURRENT_USER�QueryValueExr�rCrQrZClosermrBr�)rrH�internetSettings�proxyEnableZproxyServer�pr�ZaddressrLrLrM�getproxies_registryZ s8 rPcCst�p t�S)z�Return a dictionary of scheme -> proxy server URL mappings. Returns settings gathered from the environment, if specified, or the registry. )r0rPrLrLrLrMr5� sc&Cs~yddl}Wntk r dSXy6|j|jd�}|j|d�d}t|j|d�d�}Wntk rldSX|sz|r~dSt|�\}}|g}y tj |�}||kr�|j |�Wntk r�YnXy tj|�}||kr�|j |�Wntk �r�YnX|jd�}xp|D]h} | dk�r*d|k�r*dS| j dd �} | j d d�} | j dd�} x$|D]} tj| | tj��rTdS�qTW�qWdS) Nrz;Software\Microsoft\Windows\CurrentVersion\Internet SettingsrGZ ProxyOverriderz<local>r3rRz\.r2z.*�?)rHrIrJrKrLr�rmr r�r�r]ZgetfqdnrCr�rQrrS)rvrHrMrNZ proxyOverrideZrawHostr/ZaddrZfqdnr3r�rLrLrM�proxy_bypass_registry� sR rRcCs t�}|rt||�St|�SdS)z�Return True, if host should be bypassed. Checks proxy settings gathered from the environment, if specified, or the registry. N)r0r7rR)rvrrLrLrMr� s )NNN)N)r�rr�r�rfZhttp.clientr�r�rWr0rQr�rr�rd�collectionsrZrTr?Zurllib.errorrrrZurllib.parserrrrr r rrr rrrrrrrrrZurllib.responserrrDrIrC�__all__�version_infor�rFr�r0r1r\r6r7rR�ASCIIrrrwrrr2rr/rrrrr r!r"r#r$r%�urandomrer&r'r(rzr)r�r�rEr]rr.r]r_r*r�r+r,r-rr[Z nturl2pathr4r3r�r8r9rr�rr�rrrr r�r0r7rA�platformZ_scproxyrCrDrErFrr5rPrRrLrLrLrM�<module>Ds�P T ?n$q*@ ov +3:5!AW _ #< - 2 PK23�\pl� 2�2�(__pycache__/request.cpython-36.opt-2.pycnu�[���3 ���i~��)@s�ddlZddlZddlZddlZddlZddlZddlZddlZddl Z ddl Z ddlZddlZddl Z ddlZddlZddlZddlZddlmZmZmZddlmZmZmZmZmZmZmZmZmZm Z m!Z!m"Z"m#Z#m$Z$m%Z%m&Z&m'Z'm(Z(ddl)m*Z*m+Z+yddl,Z,Wne-k �r"dZ.YnXdZ.ddd d ddd ddddddddddddddddddd d!d"d#d$d%d&d'g!Z/d(ej0dd)�Z1da2de j3fddddd*�d+d�Z4d,d�Z5gZ6d}d-d$�Z7d.d%�Z8e j9d/e j:�Z;d0d1�Z<Gd2d�d�Z=Gd3d�d�Z>d4d �Z?Gd5d �d �Z@Gd6d�de@�ZAGd7d �d e@�ZBGd8d�de@�ZCd9d:�ZDGd;d �d e@�ZEGd<d�d�ZFGd=d�deF�ZGGd>d�deG�ZHGd?d�d�ZIGd@d�deIe@�ZJGdAd�deIe@�ZKejLZMGdBd�d�ZNGdCd�de@eN�ZOGdDd�de@eN�ZPGdEdF�dFe@�ZQGdGd�deQ�ZReSejTdH��r.GdIdJ�dJeQ�ZUe/jVdJ�GdKd�de@�ZWGdLd�de@�ZXdMdN�ZYdOdP�ZZGdQd�de@�Z[dRdS�Z\GdTd�de@�Z]GdUd�de]�Z^GdVd�de@�Z_dWZ`ejadXk�r�ddYlbmcZcmdZdndZd"�Zcd[d!�ZdiZeGd\d&�d&�ZfGd]d'�d'ef�Zgdahd^d_�Zidajd`da�Zkdaldbdc�Zmdanddde�ZoGdfdg�dg�Zpdhdi�Zqd~djdk�Zrdldm�Zsejtdnk�r�ddolumvZvmwZwdpdq�Zxdrds�Zydtdu�Zzdvd#�Z{n6ejadXk�r�dwdx�Z|dyd#�Z{dzd{�Z}d|du�ZzneqZ{erZzdS)�N)�URLError� HTTPError�ContentTooShortError)�urlparse�urlsplit�urljoin�unwrap�quote�unquote� splittype� splithost� splitport� splituser�splitpasswd� splitattr� splitquery� splitvalue�splittag�to_bytes�unquote_to_bytes� urlunparse)� addinfourl�addclosehookFT�Request�OpenerDirector�BaseHandler�HTTPDefaultErrorHandler�HTTPRedirectHandler�HTTPCookieProcessor�ProxyHandler�HTTPPasswordMgr�HTTPPasswordMgrWithDefaultRealm�HTTPPasswordMgrWithPriorAuth�AbstractBasicAuthHandler�HTTPBasicAuthHandler�ProxyBasicAuthHandler�AbstractDigestAuthHandler�HTTPDigestAuthHandler�ProxyDigestAuthHandler�HTTPHandler�FileHandler� FTPHandler�CacheFTPHandler�DataHandler�UnknownHandler�HTTPErrorProcessor�urlopen�install_opener�build_opener�pathname2url�url2pathname� getproxies�urlretrieve� urlcleanup� URLopener�FancyURLopenerz%d.%d�)�cafile�capath� cadefault�contextc Cs�|s|s|rfddl}|jdtd�|dk r2td��ts>td��tjtjj||d�}t |d�}t |�} n0|r~t |d�}t |�} ntdkr�t �a} nt} | j|||�S)NrzJcafile, capath and cadefault are deprecated, use a custom context instead.r:zDYou can't pass both context and any of cafile, capath, and cadefaultzSSL support not available)r;r<)r>) �warnings�warn�DeprecationWarning� ValueError� _have_ssl�sslZcreate_default_contextZPurposeZSERVER_AUTH�HTTPSHandlerr2�_opener�open) �url�data�timeoutr;r<r=r>r?Z https_handler�opener�rL�&/usr/lib64/python3.6/urllib/request.pyr0�s*< cCs|adS)N)rF)rKrLrLrMr1�scCs4t|�\}}tjt||����}|j�}|dkrD|rDtjj|�|fS|rTt|d�}nt j dd�}|j}tj |�|��||f} d} d }d}d} d |kr�t|d �}|r�|| | |�xB|j| �}|s�P|t|�7}|j|�| d7} |r�|| | |�q�WWdQRXWdQRX|dk�r0||k�r0td||f| ��| S)N�file�wbF)�deletei��rzcontent-lengthzContent-Lengthz1retrieval incomplete: got only %i out of %i bytesi ���)r� contextlib�closingr0�info�os�path�normpathrG�tempfileZNamedTemporaryFile�name�_url_tempfiles�append�int�read�len�writer)rH�filename� reporthookrIZurl_typerX�fp�headers�tfp�result�bs�sizer_�blocknum�blockrLrLrMr6�sD $cCsHx0tD](}ytj|�Wqtk r,YqXqWtdd�=trDdadS)N)r\rW�unlink�OSErrorrF)Z temp_filerLrLrMr7%s z:\d+$cCs<|j}t|�d}|dkr&|jdd�}tjd|d�}|j�S)NrR��Host)�full_urlr� get_header�_cut_port_re�sub�lower)�requestrH�hostrLrLrM�request_host4srwc@s�eZdZdidddfdd�Zedd��Zejdd��Zejdd��Zed d ��Zejdd ��Zejdd ��Zd d�Z dd�Z dd�Zdd�Zdd�Z dd�Zdd�Zdd�Zd#dd�Zdd �Zd!d"�ZdS)$rNFc Csp||_i|_i|_d|_||_d|_x |j�D]\}}|j||�q.W|dkrVt|�}||_ ||_ |rl||_dS)N)rpre�unredirected_hdrs�_datarI�_tunnel_host�items� add_headerrw�origin_req_host�unverifiable�method) �selfrHrIrer}r~r�key�valuerLrLrM�__init__FszRequest.__init__cCs|jrdj|j|j�S|jS)Nz{}#{})�fragment�format� _full_url)r�rLrLrMrpXszRequest.full_urlcCs(t|�|_t|j�\|_|_|j�dS)N)rr�rr��_parse)r�rHrLrLrMrp^s cCsd|_d|_d|_dS)Nrn)r�r��selector)r�rLrLrMrpescCs|jS)N)ry)r�rLrLrMrIkszRequest.datacCs(||jkr$||_|jd�r$|jd�dS)NzContent-length)ry� has_header� remove_header)r�rIrLrLrMrIos cCs d|_dS)N)rI)r�rLrLrMrIyscCsNt|j�\|_}|jdkr(td|j��t|�\|_|_|jrJt|j�|_dS)Nzunknown url type: %r) rr��typerBrprrvr�r )r��restrLrLrMr�}s zRequest._parsecCs|jdk rdnd}t|d|�S)N�POST�GETr)rI�getattr)r�Zdefault_methodrLrLrM� get_method�szRequest.get_methodcCs|jS)N)rp)r�rLrLrM�get_full_url�szRequest.get_full_urlcCs4|jdkr|jr|j|_n||_|j|_||_dS)N�https)r�rzrvrpr�)r�rvr�rLrLrM� set_proxy�s zRequest.set_proxycCs|j|jkS)N)r�rp)r�rLrLrM� has_proxy�szRequest.has_proxycCs||j|j�<dS)N)re� capitalize)r�r��valrLrLrMr|�szRequest.add_headercCs||j|j�<dS)N)rxr�)r�r�r�rLrLrM�add_unredirected_header�szRequest.add_unredirected_headercCs||jkp||jkS)N)rerx)r��header_namerLrLrMr��s zRequest.has_headercCs|jj||jj||��S)N)re�getrx)r�r��defaultrLrLrMrq�szRequest.get_headercCs |jj|d�|jj|d�dS)N)re�poprx)r�r�rLrLrMr��szRequest.remove_headercCs"|jj�}|j|j�t|j��S)N)rx�copy�updatere�listr{)r��hdrsrLrLrM�header_items�s zRequest.header_items)N)�__name__� __module__�__qualname__r��propertyrp�setter�deleterrIr�r�r�r�r�r|r�r�rqr�r�rLrLrLrMrDs( c@sNeZdZdd�Zdd�Zdd�Zdd�Zd ejfd d�Z ddd �Z dd�Zd S)rcCs6dt}d|fg|_g|_i|_i|_i|_i|_dS)NzPython-urllib/%sz User-agent)�__version__� addheaders�handlers�handle_open�handle_error�process_response�process_request)r�Zclient_versionrLrLrMr��szOpenerDirector.__init__cCsZt|d�stdt|���d}�xt|�D�]}|dkr:q*|jd�}|d|�}||dd�}|jd �r�|jd�|d}||dd�}yt|�}Wntk r�YnX|jj |i�} | |j|<n>|d kr�|}|j } n*|dkr�|}|j} n|dkr*|}|j} nq*| j |g�} | �r&tj| |�n | j|�d }q*W|�rVtj|j|�|j|�dS)N� add_parentz%expected BaseHandler instance, got %rF�redirect_request�do_open� proxy_open�_rR�errorrG�responseruT)r�r�r�)�hasattr� TypeErrorr��dir�find� startswithr^rBr�r�r�r�r�� setdefault�bisectZinsortr]r�r�)r��handlerZadded�meth�i�protocolZ condition�j�kind�lookupr�rLrLrM�add_handler�sJ zOpenerDirector.add_handlercCsdS)NrL)r�rLrLrM�close�szOpenerDirector.closec Gs<|j|f�}x*|D]"}t||�}||�}|dk r|SqWdS)N)r�r�) r��chainr�� meth_name�argsr�r��funcrgrLrLrM�_call_chain�s zOpenerDirector._call_chainNc Cs�t|t�rt||�}n|}|dk r(||_||_|j}|d}x(|jj|g�D]}t||�}||�}qLW|j ||�} |d}x*|j j|g�D]}t||�}||| �} q�W| S)NZ_requestZ _response)� isinstance�strrrIrJr�r�r�r��_openr�) r��fullurlrIrJ�reqr�r�Z processorr�r�rLrLrMrG�s" zOpenerDirector.opencCsP|j|jdd|�}|r|S|j}|j|j||d|�}|r>|S|j|jdd|�S)Nr�Zdefault_openr��unknown�unknown_open)r�r�r�)r�r�rIrgr�rLrLrMr�s zOpenerDirector._opencGs~|d kr,|jd}|d}d|}d}|}n|j}|d}d}|||f|}|j|�}|r^|S|rz|dd f|}|j|�SdS)N�httpr�r:z http_error_%srRZ_errorrr��http_error_default)r�r�)r�r�)r��protor��dictr�Zhttp_errZ orig_argsrgrLrLrMr�'s zOpenerDirector.error)N)r�r�r�r�r�r�r��socket�_GLOBAL_DEFAULT_TIMEOUTrGr�r�rLrLrLrMr�s/ c Gs�t�}ttttttttt g }t tjd�r2|j t�t�}xN|D]F}x@|D]8}t|t�rlt||�r�|j|�qHt||�rH|j|�qHWq>Wx|D]}|j|�q�Wx|D]}|j|��q�Wx&|D]}t|t�r�|�}|j|�q�W|S)N�HTTPSConnection)rrr.r)rrr+r*r/r-r�r��clientr]rE�setr�r�� issubclass�add�remover�)r�rKZdefault_classes�skip�klassZcheck�hrLrLrMr2@s0 c@s(eZdZdZdd�Zdd�Zdd�ZdS) ri�cCs ||_dS)N)�parent)r�r�rLrLrMr�gszBaseHandler.add_parentcCsdS)NrL)r�rLrLrMr�jszBaseHandler.closecCst|d�sdS|j|jkS)N� handler_orderT)r�r�)r��otherrLrLrM�__lt__ns zBaseHandler.__lt__N)r�r�r�r�r�r�r�rLrLrLrMrdsc@seZdZdZdd�ZeZdS)r/i�cCsJ|j|j|j�}}}d|ko*dknsF|jjd|||||�}|S)N��i,r�)�code�msgrVr�r�)r�rur�r�r�r�rLrLrM� http_response{s z HTTPErrorProcessor.http_responseN)r�r�r�r�r��https_responserLrLrLrMr/wsc@seZdZdd�ZdS)rcCst|j||||��dS)N)rrp)r�r�rdr�r�r�rLrLrMr��sz*HTTPDefaultErrorHandler.http_error_defaultN)r�r�r�r�rLrLrLrMr�sc@s4eZdZdZdZdd�Zdd�ZeZZZ dZ dS) r�� c sx|j�}|dkr|dkp&|dko&|dks:t|j||||��|jdd �}d�t�fdd �|jj�D��}t|||jdd�S)N�-�.�/�3r��HEADr�� z%20�content-length�content-typec3s&|]\}}|j��kr||fVqdS)N)rt)�.0�k�v)�CONTENT_HEADERSrLrM� <genexpr>�sz7HTTPRedirectHandler.redirect_request.<locals>.<genexpr>T)rer}r~)r�r�r�r�)r�r�)r�r�r�)r�r�) r�rrp�replacer�rer{rr}) r�r�rdr�r�re�newurl�mZ newheadersrL)r�rMr��s z$HTTPRedirectHandler.redirect_requestc CsNd|kr|d}nd|kr$|d}ndSt|�}|jdkrRt||d||f||��|jrp|jrpt|�}d|d <t|�}t|d tj d�}t |j|�}|j||||||�}|dkr�dSt |d��r|j} |_| j|d �|jks�t| �|jk�rt|j||j|||��ni} |_|_| j|d �d| |<|j�|j�|jj||jd�S)N�location�urir�r��ftprnz+%s - Redirection to url '%s' is not allowed�/r:z iso-8859-1)�encoding�safe� redirect_dictrrR)rJ)r�r�r�rn)r�schemerrXZnetlocr�rr �stringZpunctuationrrpr�r�rr��max_repeatsr`�max_redirections�inf_msgr_r�r�rGrJ) r�r�rdr�r�rer��urlparts�newZvisitedrLrLrM�http_error_302�s@ z"HTTPRedirectHandler.http_error_302zoThe HTTP server returned a redirect error that would lead to an infinite loop. The last 30x error message was: N)r�r�r�rrr�r �http_error_301�http_error_303�http_error_307rrLrLrLrMr�s&<c Cs�t|�\}}|jd�s d}|}n:|jd�s6td|��|jdd�}|dkrNd}|d|�}t|�\}}|dk r|t|�\}}nd}}||||fS)Nr�z//zproxy URL with no authority: %rr:rRrS)rr�rBr�rr) �proxyrZr_scheme� authority�endZuserinfo�hostport�user�passwordrLrLrM�_parse_proxy�s rc@s"eZdZdZddd�Zdd�ZdS)r�dNcCsL|dkrt�}||_x2|j�D]&\}}t|d||||jfdd��qWdS)Nz%s_opencSs||||�S)NrL)�rr r�r�rLrLrM�<lambda>%sz'ProxyHandler.__init__.<locals>.<lambda>)r5�proxiesr{�setattrr�)r�rr�rHrLrLrMr�s zProxyHandler.__init__cCs�|j}t|�\}}}}|dkr"|}|jr6t|j�r6dS|rv|rvdt|�t|�f} tj| j��jd�} |j dd| �t|�}|j ||�||ks�|dkr�dS|jj||j d�SdS)Nz%s:%s�asciizProxy-authorizationzBasic r�)rJ)r�rrv�proxy_bypassr �base64� b64encode�encode�decoder|r�r�rGrJ)r�r�r r�Z orig_typeZ proxy_typerrrZ user_passZcredsrLrLrMr�(s zProxyHandler.proxy_open)N)r�r�r�r�r�r�rLrLrLrMrs c@s6eZdZdd�Zdd�Zdd�Zd dd �Zd d�ZdS)r cCs i|_dS)N)�passwd)r�rLrLrMr�FszHTTPPasswordMgr.__init__cs`t|t�r|g}|�jkr$i�j|<x6dD].�t��fdd�|D��}||f�j||<q*WdS)NTFcsg|]}�j|���qSrL)� reduce_uri)r��u)�default_portr�rLrM� <listcomp>Qsz0HTTPPasswordMgr.add_password.<locals>.<listcomp>)TF)r�r�r�tuple)r��realmr�rr�reduced_urirL)r"r�rM�add_passwordIs zHTTPPasswordMgr.add_passwordc Cs`|jj|i�}xLdD]D}|j||�}x2|j�D]&\}}x|D]}|j||�r<|Sq<Wq.WqWdS)NTF)TF)NN)rr�r r{� is_suburi) r�r%�authuriZdomainsr"�reduced_authuriZurisZauthinfor�rLrLrM�find_user_passwordTs z"HTTPPasswordMgr.find_user_passwordTc Cs�t|�}|dr.|d}|d}|dp*d}nd}|}d}t|�\}}|r~|dkr~|dk r~ddd�j|�} | dk r~d|| f}||fS) NrRrr:r��Pi�)r�r�z%s:%d)rr r�) r�r�r"�partsrrrXrv�portZdportrLrLrMr ^s zHTTPPasswordMgr.reduce_uricCsR||krdS|d|dkr dStj|d|df�}t|�t|d�krNdSdS)NTrFrR)� posixpath�commonprefixr`)r��base�test�commonrLrLrMr(uszHTTPPasswordMgr.is_suburiN)T)r�r�r�r�r'r+r r(rLrLrLrMr Ds c@seZdZdd�ZdS)r!cCs0tj|||�\}}|dk r"||fStj|d|�S)N)r r+)r�r%r)rrrLrLrMr+�s z2HTTPPasswordMgrWithDefaultRealm.find_user_passwordN)r�r�r�r+rLrLrLrMr!�scs<eZdZ�fdd�Zd �fdd� Zddd�Zdd �Z�ZS)r"csi|_t�j||�dS)N)� authenticated�superr�)r�r��kwargs)� __class__rLrMr��sz%HTTPPasswordMgrWithPriorAuth.__init__Fcs<|j||�|dk r&t�jd|||�t�j||||�dS)N)�update_authenticatedr5r')r�r%r�rr�is_authenticated)r7rLrMr'�sz)HTTPPasswordMgrWithPriorAuth.add_passwordcCsFt|t�r|g}x0dD](}x"|D]}|j||�}||j|<q WqWdS)NTF)TF)r�r�r r4)r�r�r9r"r!r&rLrLrMr8�s z1HTTPPasswordMgrWithPriorAuth.update_authenticatedcCsDx>dD]6}|j||�}x$|jD]}|j||�r|j|SqWqWdS)NTF)TF)r r4r()r�r)r"r*r�rLrLrMr9�s z-HTTPPasswordMgrWithPriorAuth.is_authenticated)F)F)r�r�r�r�r'r8r9� __classcell__rLrL)r7rMr"�s c@sTeZdZejdej�Zddd�Zdd�Zdd�Z d d �Z dd�Zd d�ZeZ eZdS)r#z1(?:^|,)[ ]*([^ ,]+)[ ]+realm=(["']?)([^"']*)\2NcCs"|dkrt�}||_|jj|_dS)N)r rr')r�Zpassword_mgrrLrLrMr��sz!AbstractBasicAuthHandler.__init__ccstd}xFtjj|�D]6}|j�\}}}|d kr:tjdtd�||fVd}qW|sp|rb|j�d}nd}|dfVdS) NF�"�'zBasic Auth Realm was unquoted�Trrn)r;r<)r#�rx�finditer�groupsr?r@�UserWarning�split)r��headerZfound_challengeZmorr r%rLrLrM�_parse_realm�s z%AbstractBasicAuthHandler._parse_realmc Cs~|j|�}|sdSd}xL|D]D}x>|j|�D]0\}}|j�dkrF|}q,|dk r,|j|||�Sq,WqW|dk rztd|f��dS)N�basiczBAbstractBasicAuthHandler does not support the following scheme: %r)Zget_allrDrt�retry_http_basic_authrB) r��authreqrvr�reZunsupportedrCrr%rLrLrM�http_error_auth_reqed�s z.AbstractBasicAuthHandler.http_error_auth_reqedcCs||jj||�\}}|dk rtd||f}dtj|j��jd�}|j|jd�|krTdS|j|j|�|j j ||jd�SdSdS)Nz%s:%szBasic r)rJ)rr+rrrrrq�auth_headerr�r�rGrJ)r�rvr�r%r�pw�raw�authrLrLrMrF�sz.AbstractBasicAuthHandler.retry_http_basic_authcCsxt|jd�s|jj|j�r"|S|jd�st|jjd|j�\}}dj||�j�}tj |�j �}|jddj|j���|S)Nr9� Authorizationz{0}:{1}zBasic {}) r�rr9rpr�r+r�rrZstandard_b64encoderr��strip)r�r�rrZcredentialsZauth_strrLrLrM�http_requests z%AbstractBasicAuthHandler.http_requestcCsLt|jd�rHd|jko dknr8|jj|jd�n|jj|jd�|S)Nr9r�i,TF)r�rr�r8rp)r�r�r�rLrLrMr�s z&AbstractBasicAuthHandler.http_response)N)r�r�r��re�compile�Ir>r�rDrHrFrOr�� https_requestr�rLrLrLrMr#�s c@seZdZdZdd�ZdS)r$rMcCs|j}|jd|||�}|S)Nzwww-authenticate)rprH)r�r�rdr�r�rerHr�rLrLrM�http_error_401 s z#HTTPBasicAuthHandler.http_error_401N)r�r�r�rIrTrLrLrLrMr$sc@seZdZdZdd�ZdS)r%zProxy-authorizationcCs|j}|jd|||�}|S)Nzproxy-authenticate)rvrH)r�r�rdr�r�rerr�rLrLrM�http_error_407+s z$ProxyBasicAuthHandler.http_error_407N)r�r�r�rIrUrLrLrLrMr%'sc@sNeZdZddd�Zdd�Zdd�Zdd �Zd d�Zdd �Zdd�Z dd�Z dS)r&NcCs4|dkrt�}||_|jj|_d|_d|_d|_dS)Nr)r rr'�retried�nonce_count� last_nonce)r�rrLrLrMr�Es z"AbstractDigestAuthHandler.__init__cCs d|_dS)Nr)rV)r�rLrLrM�reset_retry_countNsz+AbstractDigestAuthHandler.reset_retry_countcCs||j|d�}|jdkr*t|jdd|d��n|jd7_|rx|j�d}|j�dkr`|j||�S|j�dkrxtd|��dS) N�i�zdigest auth failedrRrZdigestrEzEAbstractDigestAuthHandler does not support the following scheme: '%s')r�rVrrprBrt�retry_http_digest_authrB)r�rIrvr�rerGrrLrLrMrHQs z/AbstractDigestAuthHandler.http_error_auth_reqedcCsz|jdd�\}}ttdt|���}|j||�}|rvd|}|jj|jd�|krRdS|j|j|�|j j ||jd�}|SdS)Nr�rRz Digest %s)rJ)rB�parse_keqv_list�filter�parse_http_list�get_authorizationrer�rIr�r�rGrJ)r�r�rL�tokenZ challenge�chalZauth_valZresprLrLrMr[esz0AbstractDigestAuthHandler.retry_http_digest_authcCs@d|j|tj�f}|jd�td�}tj|�j�}|dd�S)Nz %s:%s:%s:rrQ�)rW�timeZctimer�_randombytes�hashlib�sha1� hexdigest)r��nonce�s�b�digrLrLrM� get_cnonceqsz$AbstractDigestAuthHandler.get_cnoncecCs�y6|d}|d}|jd�}|jdd�}|jdd�}Wntk rJdSX|j|�\}} |dkrfdS|jj||j�\} }| dkr�dS|jdk r�|j|j|�}nd}d| ||f} d|j�|j f}|d k�r.||j kr�|jd 7_nd |_||_ d|j}|j|�}d||||||�f}| || �|�}n2|dk�rT| || �d|||�f�}nt d |��d| |||j |f}|�r�|d|7}|�r�|d|7}|d|7}|�r�|d||f7}|S)Nr%rh�qop� algorithm�MD5�opaquez%s:%s:%sz%s:%srLrRz%08xz%s:%s:%s:%s:%szqop '%s' is not supported.z>username="%s", realm="%s", nonce="%s", uri="%s", response="%s"z , opaque="%s"z , digest="%s"z, algorithm="%s"z, qop=auth, nc=%s, cnonce="%s")r��KeyError�get_algorithm_implsrr+rprI�get_entity_digestr�r�rXrWrlr)r�r�rar%rhrmrnrp�H�KDrrJZentdigZA1ZA2ZncvalueZcnonceZnoncebitZrespdigr1rLrLrMr_|sV z+AbstractDigestAuthHandler.get_authorizationcsD|dkrdd��n|dkr$dd��ntd|���fdd�}�|fS)NrocSstj|jd��j�S)Nr)reZmd5rrg)�xrLrLrMr�sz?AbstractDigestAuthHandler.get_algorithm_impls.<locals>.<lambda>ZSHAcSstj|jd��j�S)Nr)rerfrrg)rvrLrLrMr�sz.Unsupported digest authentication algorithm %rcs�d||f�S)Nz%s:%srL)ri�d)rtrLrMr�s)rB)r�rnrurL)rtrMrr�s z-AbstractDigestAuthHandler.get_algorithm_implscCsdS)NrL)r�rIrarLrLrMrs�sz+AbstractDigestAuthHandler.get_entity_digest)N)r�r�r�r�rYrHr[rlr_rrrsrLrLrLrMr&:s < c@seZdZdZdZdd�ZdS)r'rMi�cCs*t|j�d}|jd|||�}|j�|S)NrRzwww-authenticate)rrprHrY)r�r�rdr�r�rerv�retryrLrLrMrT�s z$HTTPDigestAuthHandler.http_error_401N)r�r�r�rIr�rTrLrLrLrMr'�sc@seZdZdZdZdd�ZdS)r(zProxy-Authorizationi�cCs"|j}|jd|||�}|j�|S)Nzproxy-authenticate)rvrHrY)r�r�rdr�r�rervrxrLrLrMrU�s z%ProxyDigestAuthHandler.http_error_407N)r�r�r�rIr�rUrLrLrLrMr(�sc@s6eZdZd dd�Zdd�Zdd�Zdd �Zd d�ZdS)�AbstractHTTPHandlerrcCs ||_dS)N)�_debuglevel)r�� debuglevelrLrLrMr��szAbstractHTTPHandler.__init__cCs ||_dS)N)rz)r��levelrLrLrM�set_http_debuglevel�sz'AbstractHTTPHandler.set_http_debuglevelcCstjjj|j|j��S)N)r�r��HTTPConnection�_get_content_lengthrIr�)r�rurLrLrMr�sz'AbstractHTTPHandler._get_content_lengthcCs |j}|std��|jdk r�|j}t|t�r8d}t|��|jd�sN|jdd�|jd�r�|jd�r�|j|�}|dk r�|jdt|��n|jdd�|}|j �r�t |j�\}}t|�\}} |jd�s�|jd|�x2|j jD]&\} }| j�} |j| �s�|j| |�q�W|S) Nz no host givenz\POST data should be bytes, an iterable of bytes, or a file object. It cannot be of type str.zContent-typez!application/x-www-form-urlencodedzContent-lengthzTransfer-encodingZchunkedro)rvrrIr�r�r�r�r�rr�rr�rr�r�r�)r�rurvrIr�Zcontent_lengthZsel_hostrZselZsel_pathr[r�rLrLrM�do_request_�s> zAbstractHTTPHandler.do_request_c s\|j}|std��||fd|ji|��}|j|j�t|j���jt�fdd�|jj �D���d�d<tdd��j �D���|j r�i}d}|�kr��|||<�|=|j|j |d �y`y&|j|j �|j|j�|jd �d�Wn,tk �r }zt|��WYdd}~XnX|j�} Wn|j��YnX|j�rF|jj�d|_|j�| _| j| _| S)Nz no host givenrJc3s"|]\}}|�kr||fVqdS)NrL)r�r�r�)rerLrMr�)sz.AbstractHTTPHandler.do_open.<locals>.<genexpr>r�� Connectioncss|]\}}|j�|fVqdS)N)�title)r�r[r�rLrLrMr�6szProxy-Authorization)rezTransfer-encoding)Zencode_chunked)rvrrJZset_debuglevelrzr�rxr�rer{rzZ set_tunnelrur�r�rIr�rm�getresponser�Zsockr�rH�reasonr�) r�Z http_classr�Zhttp_conn_argsrvr�Ztunnel_headersZproxy_auth_hdr�errrrL)rerMr�s@ " zAbstractHTTPHandler.do_openN)r)r�r�r�r�r}rr�r�rLrLrLrMry�s &ryc@seZdZdd�ZejZdS)r)cCs|jtjj|�S)N)r�r�r�r~)r�r�rLrLrM� http_open`szHTTPHandler.http_openN)r�r�r�r�ryr�rOrLrLrLrMr)^sr�c@s$eZdZddd�Zdd�ZejZdS)rErNcCstj||�||_||_dS)N)ryr��_context�_check_hostname)r�r{r>�check_hostnamerLrLrMr�iszHTTPSHandler.__init__cCs|jtjj||j|jd�S)N)r>r�)r�r�r�r�r�r�)r�r�rLrLrM� https_opennszHTTPSHandler.https_open)rNN)r�r�r�r�r�ryr�rSrLrLrLrMrEgs rEc@s.eZdZddd�Zdd�Zdd�ZeZeZdS) rNcCs$ddl}|dkr|jj�}||_dS)Nr)Zhttp.cookiejar� cookiejarZ CookieJar)r�r�r�rLrLrMr�ws zHTTPCookieProcessor.__init__cCs|jj|�|S)N)r�Zadd_cookie_header)r�rurLrLrMrO}sz HTTPCookieProcessor.http_requestcCs|jj||�|S)N)r�Zextract_cookies)r�rur�rLrLrMr��sz!HTTPCookieProcessor.http_response)N)r�r�r�r�rOr�rSr�rLrLrLrMrvs c@seZdZdd�ZdS)r.cCs|j}td|��dS)Nzunknown url type: %s)r�r)r�r�r�rLrLrMr��szUnknownHandler.unknown_openN)r�r�r�r�rLrLrLrMr.�scCsRi}xH|D]@}|jdd�\}}|ddkrB|ddkrB|dd�}|||<q W|S)N�=rRrr;rSrS)rB)�lZparsedZeltr�r�rLrLrMr\�s r\cCs�g}d}d}}xt|D]l}|r,||7}d}q|rV|dkr@d}qn|dkrLd}||7}q|dkrn|j|�d}q|dkrzd}||7}qW|r�|j|�dd�|D�S) NrnF�\Tr;�,cSsg|]}|j��qSrL)rN)r��partrLrLrMr#�sz#parse_http_list.<locals>.<listcomp>)r])ri�resr��escaper ZcurrLrLrMr^�s4 r^c@s(eZdZdd�ZdZdd�Zdd�ZdS)r*cCs\|j}|dd�dkrN|dd�dkrN|jrN|jdkrN|j|j�krXtd��n |j|�SdS)Nr:z//r=r�� localhostz-file:// scheme is supported only on localhost)r�rv� get_namesr�open_local_file)r�r�rHrLrLrM� file_open�s& zFileHandler.file_openNcCs`tjdkrZy*ttjd�dtjtj��d�t_Wn$tjk rXtjd�ft_YnXtjS)Nr�r:)r*�namesr$r��gethostbyname_ex�gethostname�gaierror� gethostbyname)r�rLrLrMr��s zFileHandler.get_namescCsddl}ddl}|j}|j}t|�}y�tj|�}|j}|jj |j dd�} |j|�d} |jd| pbd|| f�}|r~t |�\}}|s�|r�t|�|j�kr�|r�d||} nd|} tt|d�|| �SWn*tk r�}zt|��WYdd}~XnXtd��dS) NrT)�usegmtz6Content-type: %s Content-length: %d Last-modified: %s z text/plainzfile://�rbzfile not on local host)�email.utils� mimetypesrvr�r4rW�stat�st_size�utils� formatdate�st_mtime� guess_type�message_from_stringr �_safe_gethostbynamer�rrGrmr)r�r��emailr�rvrbZ localfile�statsri�modified�mtyperer.Zorigurl�exprLrLrMr��s0 zFileHandler.open_local_file)r�r�r�r�r�r�r�rLrLrLrMr*�s cCs&y tj|�Stjk r dSXdS)N)r�r�r�)rvrLrLrMr��s r�c@seZdZdd�Zdd�ZdS)r+cCs.ddl}ddl}|j}|s"td��t|�\}}|dkr>|j}nt|�}t|�\}}|rdt|�\}}nd}t |�}|pvd}|p~d}yt j|�}Wn*tk r�}zt|��WYdd}~XnXt |j�\} } | jd�}ttt |��}|dd�|d}}|�r|d�r|dd�}y�|j||||||j�} |�r8d�p:d}x:| D]2}t|�\}}|j�dk�rB|dk�rB|j�}�qBW| j||�\}}d}|j|j�d}|�r�|d |7}|dk �r�|dk�r�|d|7}tj|�}t|||j�S|jk �r(}z$td|�}|jtj �d��WYdd}~XnXdS)Nrzftp error: no host givenrnr�rRrR�Dr��a�Ar�rwzContent-type: %s zContent-length: %d z ftp error: %rr:rSrS)r�r�r�rRrwr�)!�ftplibr�rvrr �FTP_PORTr^rrr r�r�rmrr�rBr��map�connect_ftprJrrt�upper�retrfiler�rpr�r�r� all_errors�with_traceback�sys�exc_info)r�r�r�r�rvr.rrr�rX�attrs�dirsrN�fwr��attrr�rd�retrlenrer�r��excrLrLrM�ftp_open�s\ zFTPHandler.ftp_openc Cst||||||dd�S)NF)� persistent)� ftpwrapper)r�rrrvr.r�rJrLrLrMr�1szFTPHandler.connect_ftpN)r�r�r�r�r�rLrLrLrMr+�s5c@s<eZdZdd�Zdd�Zdd�Zdd�Zd d �Zdd�Zd S)r,cCs"i|_i|_d|_d|_d|_dS)Nr�<rb)�cacherJ�soonest�delay� max_conns)r�rLrLrMr�8s zCacheFTPHandler.__init__cCs ||_dS)N)r�)r��trLrLrM� setTimeout?szCacheFTPHandler.setTimeoutcCs ||_dS)N)r�)r�r�rLrLrM�setMaxConnsBszCacheFTPHandler.setMaxConnscCsr|||dj|�|f}||jkr4tj�|j|j|<n,t||||||�|j|<tj�|j|j|<|j�|j|S)Nr�)�joinr�rcr�rJr��check_cache)r�rrrvr.r�rJr�rLrLrMr�Es zCacheFTPHandler.connect_ftpcCs�tj�}|j|krTx@t|jj��D].\}}||kr"|j|j�|j|=|j|=q"Wtt|jj���|_t |j�|j kr�x6t|jj��D]$\}}||jkr�|j|=|j|=Pq�Wtt|jj���|_dS)N)rcr�r�rJr{r�r��min�valuesr`r�)r�r�r�r�rLrLrMr�Ps zCacheFTPHandler.check_cachecCs4x|jj�D]}|j�qW|jj�|jj�dS)N)r�r�r��clearrJ)r��connrLrLrM�clear_cacheds zCacheFTPHandler.clear_cacheN) r�r�r�r�r�r�r�r�r�rLrLrLrMr,5sc@seZdZdd�ZdS)r-cCs~|j}|jdd�\}}|jdd�\}}t|�}|jd�rNtj|�}|dd�}|sVd}tjd|t|�f�}t t j|�||�S) N�:rRr�z;base64�ztext/plain;charset=US-ASCIIz$Content-type: %s Content-length: %d i����)rprBr�endswithr�decodebytesr�r�r`r�io�BytesIO)r�r�rHrrIZ mediatypererLrLrM� data_openks zDataHandler.data_openN)r�r�r�r�rLrLrLrMr-jsr��nt)r4r3cCst|�S)N)r )�pathnamerLrLrMr4�scCst|�S)N)r )r�rLrLrMr3�sc@s�eZdZdZdeZd)dd�Zdd�Zdd�Zd d �Z dd�Z d*d d�Zd+dd�Zd,dd�Z d-dd�Zdd�Zd.dd�Zd/dd�Zdd�Zer�dd�Zd0dd �Zd!d"�Zd#d$�Zd%d&�Zd1d'd(�ZdS)2r8NzPython-urllib/%scKszdd|jji}tj|tdd�|dkr.t�}||_|jd�|_|jd�|_ d|j fd g|_g|_t j|_d|_t|_dS)NzW%(class)s style of invoking requests is deprecated. Use newer urlopen functions/methods�classr=)� stacklevel�key_file� cert_filez User-Agent�Accept�*/*)r�r�)r7r�r?r@rAr5rr�r�r��versionr��_URLopener__tempfilesrWrl�_URLopener__unlink� tempcache�ftpcache)r�rZx509r�rLrLrMr��szURLopener.__init__cCs|j�dS)N)r�)r�rLrLrM�__del__�szURLopener.__del__cCs|j�dS)N)�cleanup)r�rLrLrMr��szURLopener.closecCsZ|jrFx2|jD](}y|j|�Wqtk r4YqXqW|jdd�=|jrV|jj�dS)N)r�r�rmr�r�)r�rNrLrLrMr��s zURLopener.cleanupcGs|jj|�dS)N)r�r])r�r�rLrLrM� addheader�szURLopener.addheadercCsntt|��}t|dd�}|jrL||jkrL|j|\}}t|d�}t|||�St|�\}}|s`d}||jkr�|j|}t|�\}} t| �\} }| |f}nd}d|}||_ |j dd�}t||�s�|dkr�|r�|j|||�S|j ||�Sy,|dk�rt||�|�St||�||�SWnVttfk �r.�Yn<tk �rh} ztd | �jtj�d ��WYdd} ~ XnXdS)Nz%/:=&?~#+!$,;'@()*[]|)rr�rNZopen_�-r�r�zsocket errorr:)rrr r�rGrrrrr�r�r��open_unknown_proxy�open_unknownr�rrrmr�r�r�)r�r�rIrbrerd�urltyperHr � proxyhostrvr�r[r�rLrLrMrG�s< zURLopener.opencCst|�\}}tdd|��dS)Nz url errorzunknown url type)rrm)r�r�rIr�rHrLrLrMr�szURLopener.open_unknowncCs t|�\}}tdd||��dS)Nz url errorzinvalid proxy for %s)rrm)r�r r�rIr�rHrLrLrMr� szURLopener.open_unknown_proxyc Cs&tt|��}|jr&||jkr&|j|St|�\}}|dkr�|sH|dkr�y.|j|�}|j�}|j�tt|�d�|fSt k r�} zWYdd} ~ XnX|j ||�}�zH|j�} |r�t |d�}n|ddl}t|�\} }t|p�d�\} }t|p�d�\}} t |�pd�\}} tjj|�d}|j|�\}}|jj|�tj|d�}z�|| f}|jdk �r^||j|<d}d}d}d}d| k�r�t| d �}|�r�||||�xH|j|�}|�s�P|t|�7}|j|�|d7}|�r�||||��q�WWd|j�XWd|j�X|dk�r"||k�r"td ||f|��|S) NrNrRrOrrnirQzcontent-lengthzContent-Lengthz1retrieval incomplete: got only %i out of %i bytesi rS)rrr�rr�rVr�r4rrmrGrZrrrWrX�splitextZmkstempr�r]�fdopenr^r_r`rar)r�rHrbrcrIr�Zurl1rdr�r�rerfrZZgarbagerX�suffix�fdrgrhrir_rjrkrLrLrM�retrievesl zURLopener.retrievecCs(d}d}t|t�r<t|�\}}|r6t|�\}}t|�}|}nt|\}}t|�\}}t|�\} } | }d}| j�dkrvd}n:t| �\}} |r�t|�\}}|r�d| || f}t|�r�|}|s�tdd��|r�t|�}t j |j��jd�}nd}|�rt|�}t j |j��jd�}nd}||�} i}|�r*d||d<|�r<d||d<|�rJ||d <d |d<x|j D]\}}|||<�qZW|dk �r�d|d <| jd|||�n| jd||d�y| j�}Wn"tjjk �r�td��YnXd|jk�o�dkn�rt||jd||j�S|j||j|j|j|j|�SdS)Nr�z %s://%s%sz http errorz no host givenrzBasic %szProxy-AuthorizationrMror�r�z!application/x-www-form-urlencodedzContent-Typer�r�)rez$http protocol error: bad status liner�i,zhttp:)r�r�rrr rrtrrmrrrrr�rur�r�r�Z BadStatusLinerZstatusrr�� http_errorrdr�)r�Zconnection_factoryrHrIZuser_passwdZproxy_passwdrvr�Zrealhostr�r�Z proxy_authrLZ http_connrerCr�r�rLrLrM�_open_generic_httpQsr zURLopener._open_generic_httpcCs|jtjj||�S)N)r�r�r�r~)r�rHrIrLrLrM� open_http�szURLopener.open_httpc Csbd|}t||�rPt||�}|dkr6||||||�} n|||||||�} | rP| S|j|||||�S)Nz http_error_%d)r�r�r�) r�rHrd�errcode�errmsgrerIr[rrgrLrLrMr��s zURLopener.http_errorcCs|j�t||||d��dS)N)r�r)r�rHrdr�r�rerLrLrMr��szURLopener.http_error_defaultcCstjj||j|jd�S)N)r�r�)r�r�r�r�r�)r�rvrLrLrM�_https_connection�szURLopener._https_connectioncCs|j|j||�S)N)r�r�)r�rHrIrLrLrM� open_https�szURLopener.open_httpscCs^t|t�std��|dd�dkrP|dd�dkrP|dd�j�dkrPtd��n |j|�SdS) NzEfile error: proxy support for file protocol currently not implementedr:z//r=r��z localhost/z-file:// scheme is supported only on localhost)r�r�rrtrBr�)r�rHrLrLrM� open_file�s 4 zURLopener.open_filecCs\ddl}ddl}t|�\}}t|�}ytj|�}Wn0tk rb}zt|j|j ��WYdd}~XnX|j } |jj|j dd�} |j|�d}|jd|p�d| | f�}|s�|} |dd�dkr�d|} tt|d �|| �St|�\}}|o�tj|�t�ft�k�rP|} |dd�dk�r d|} n|dd �dk�r>td|��tt|d �|| �Std ��dS)NrT)r�z6Content-Type: %s Content-Length: %d Last-modified: %s z text/plainrRr�zfile://r�r:z./zAlocal file url may start with / or file:. Unknown url of type: %sz#local file error: not on local host)r�r�rr4rWr�rmr�strerrorrbr�r�r�r�r�r�rrGr r�r�r��thishostrB)r�rHr�r�rvrNZ localnamer��erir�r�reZurlfiler.rLrLrMr��s: zURLopener.open_local_filecCs�t|t�std��ddl}t|�\}}|s2td��t|�\}}t|�\}}|r\t|�\}}nd}t|�}t|ppd�}t|p|d�}t j |�}|s�ddl}|j}nt |�}t|�\}} t|�}|jd�} | dd�| d} }| o�| d�r�| dd�} | �r| d�rd| d<|||dj| �f}t|j�tk�rlx8t|j�D]*} | |k�r>|j| }|j| =|j��q>Wy�||jk�r�t||||| �|j|<|�s�d}nd}x:| D]2}t|�\}}|j�d k�r�|dk�r�|j�}�q�W|j|j||�\}}|jd|�d}d}|�r|d|7}|dk �r:|dk�r:|d|7}tj|�}t||d|�St�k �r�}z td|�j t!j"�d��WYdd}~XnXdS)NzCftp error: proxy support for ftp protocol currently not implementedrzftp error: no host givenrnr�rRr�rRr�r�r�r�rwzftp:zContent-Type: %s zContent-Length: %d zftp error %rr:rSrS)r�r�r�rRrwr�)#r�r�rr�rr rrr r�r�r�r�r^rrBr�r`r��MAXFTPCACHEr�r�r�rrtr�r�r�r�r�r� ftperrorsr�r�r�)r�rHr�rvrXr.rrr�r�r�rNr�r�r�r�r�r�rdr�r�rer�rLrLrM�open_ftp�sp zURLopener.open_ftpc Cs<t|t�std��y|jdd�\}}Wntk rDtdd��YnX|sNd}|jd�}|dkr�d ||d�kr�||dd�}|d|�}nd }g}|jdtj dtj tj����|jd |�|dkr�tj|j d��jd�}nt|�}|jdt|��|jd �|j|�dj|�}tj|�}tj|�}t|||�S)NzEdata error: proxy support for data protocol currently not implementedr�rRz data errorzbad data URLztext/plain;charset=US-ASCII�;rr�rnzDate: %sz%a, %d %b %Y %H:%M:%S GMTzContent-type: %srrzlatin-1zContent-Length: %d� )r�r�rrBrBrm�rfindr]rcZstrftimeZgmtimerr�rrr r`r�r�r�r��StringIOr) r�rHrIr�Zsemir�r�re�frLrLrM� open_data3s6 zURLopener.open_data)N)N)N)N)NNN)N)N)N)N)r�r�r�r�r�r�r�r�r�r�r�rGr�r�r�r�r�r�r�rCr�r�r�r�rrrLrLrLrMr8�s, $ B\ :c@s�eZdZdd�Zdd�Zd"dd�Zdd �Zd#d d�Zd$dd �Zd%dd�Z d&dd�Z d'dd�Zd(dd�Zd)dd�Z d*dd�Zd+dd�Zd,dd�Zd d!�ZdS)-r9cOs(tj|f|�|�i|_d|_d|_dS)Nrr�)r8r�� auth_cache�tries�maxtries)r�r�r6rLrLrMr�`szFancyURLopener.__init__cCst||d||�S)Nzhttp:)r)r�rHrdr�r�rerLrLrMr�fsz!FancyURLopener.http_error_defaultNc Csn|jd7_zR|jrJ|j|jkrJt|d�r4|j}n|j}|||dd|�S|j||||||�}|Sd|_XdS)NrR�http_error_500i�z)Internal Server Error: Redirect Recursionr)r r r�rr��redirect_internal) r�rHrdr�r�rerIr�rgrLrLrMr js zFancyURLopener.http_error_302c Csxd|kr|d}nd|kr$|d}ndS|j�t|jd||�}t|�}|jd krnt|||d|||��|j|�S) Nr�r�r�r�r�r�rnz( Redirection to url '%s' is not allowed.)r�r�r�rn)r�rr�rrrrG) r�rHrdr�r�rerIr�rrLrLrMr|s z FancyURLopener.redirect_internalcCs|j||||||�S)N)r )r�rHrdr�r�rerIrLrLrMr �szFancyURLopener.http_error_301cCs|j||||||�S)N)r )r�rHrdr�r�rerIrLrLrMr�szFancyURLopener.http_error_303cCs2|dkr|j||||||�S|j|||||�SdS)N)r r�)r�rHrdr�r�rerIrLrLrMr�szFancyURLopener.http_error_307Fc Cs�d|krtj||||||�|d}tjd|�} | sHtj||||||�| j�\} }| j�dkrttj||||||�|s�tj||||||�d|jd}|dkr�t||�||�St||�|||�SdS)Nzwww-authenticatez![ ]*([^ ]+)[ ]+realm="([^"]*)"rEZretry_�_basic_auth)r8r�rP�matchr@rtr�r�) r�rHrdr�r�rerIrx�stuffrrr%r[rLrLrMrT�s& zFancyURLopener.http_error_401c Cs�d|krtj||||||�|d}tjd|�} | sHtj||||||�| j�\} }| j�dkrttj||||||�|s�tj||||||�d|jd}|dkr�t||�||�St||�|||�SdS)Nzproxy-authenticatez![ ]*([^ ]+)[ ]+realm="([^"]*)"rEZretry_proxy_r )r8r�rPrr@rtr�r�) r�rHrdr�r�rerIrxrrrr%r[rLrLrMrU�s& zFancyURLopener.http_error_407cCs�t|�\}}d||}|jd}t|�\}} t| �\} } | jd�d}| |d�} |j| ||�\}} |pl| srdSdt|dd�t| dd�| f} d| | |jd<|dkr�|j|�S|j||�SdS)Nzhttp://r��@rRz%s:%s@%srn)r)rrrr��get_user_passwdr rG)r�rHr%rIrvr�r�r r�r�� proxyselectorr�rrrLrLrM�retry_proxy_http_basic_auth�s z*FancyURLopener.retry_proxy_http_basic_authcCs�t|�\}}d||}|jd}t|�\}} t| �\} } | jd�d}| |d�} |j| ||�\}} |pl| srdSdt|dd�t| dd�| f} d| | |jd<|dkr�|j|�S|j||�SdS)Nzhttps://r�rrRz%s:%s@%srn)r)rrrr�rr rG)r�rHr%rIrvr�r�r r�r�rr�rrrLrLrM�retry_proxy_https_basic_auth�s z+FancyURLopener.retry_proxy_https_basic_authc Cs�t|�\}}|jd�d}||d�}|j|||�\}}|p>|sDdSdt|dd�t|dd�|f}d||} |dkr�|j| �S|j| |�SdS)NrrRz%s:%s@%srn)rzhttp://)rr�rr rG) r�rHr%rIrvr�r�rrr�rLrLrMrF�s z$FancyURLopener.retry_http_basic_authc Cs�t|�\}}|jd�d}||d�}|j|||�\}}|p>|sDdSdt|dd�t|dd�|f}d||} |dkr�|j| �S|j| |�SdS)NrrRz%s:%s@%srn)rzhttps://)rr�rr rG) r�rHr%rIrvr�r�rrr�rLrLrM�retry_https_basic_auth s z%FancyURLopener.retry_https_basic_authrcCs`|d|j�}||jkr2|r(|j|=n |j|S|j||�\}}|sJ|rX||f|j|<||fS)Nr)rtr�prompt_user_passwd)r�rvr%r�r�rrrLrLrMr s zFancyURLopener.get_user_passwdcCsTddl}y,td||f�}|jd|||f�}||fStk rNt�dSXdS)NrzEnter username for %s at %s: z#Enter password for %s in %s at %s: )NN)�getpass�input�KeyboardInterrupt�print)r�rvr%rrrrLrLrMr$ sz!FancyURLopener.prompt_user_passwd)N)N)N)N)NF)NF)N)N)N)N)r)r�r�r�r�r�r rr rrrTrUrrrFrrrrLrLrLrMr9]s" cCstdkrtjd�atS)Nr�)� _localhostr�r�rLrLrLrMr�4 s r�cCsPtdkrLyttjtj��d�aWn(tjk rJttjd�d�aYnXtS)Nr:r�)� _thishostr$r�r�r�r�rLrLrLrMr�< sr�cCstdkrddl}|jatS)Nr)� _ftperrorsr�r�)r�rLrLrMrG srcCstdkrtjd�atS)Nrn)� _noheadersr�r�rLrLrLrM� noheadersP s rc@sFeZdZddd�Zdd�Zdd�Zd d �Zdd�Zd d�Zdd�Z dS)r�NTc CsX||_||_||_||_||_||_d|_||_y|j�Wn|j ��YnXdS)Nr) rrrvr.r�rJ�refcount� keepalive�initr�)r�rrrvr.r�rJr�rLrLrMr�] szftpwrapper.__init__cCs\ddl}d|_|j�|_|jj|j|j|j�|jj|j |j �dj|j�}|jj |�dS)Nrr�)r��busyZFTPr�Zconnectrvr.rJZloginrrr�r��cwd)r�r�Z_targetrLrLrMr"m s zftpwrapper.initc-Cs�ddl}|j�|dkr"d}d}nd|}d}y|jj|�Wn*|jk rh|j�|jj|�YnXd}|r�|r�yd|}|jj|�\}}WnR|jk r�}z4t|�dd�d kr�t d |�j tj�d��WYdd}~XnX|�s�|jjd�|�rn|jj �} zJy|jj|�Wn4|jk �rP}zt d |�|�WYdd}~XnXWd|jj| �Xd|}nd }|jj|�\}}d|_t|jd�|j�} |jd7_|j�| |fS)Nrrwr�zTYPE ArRzTYPE zRETR r=Z550z ftp error: %rr:zLIST ZLISTr�)rwr�)r��endtransferr�Zvoidcmdr�r"ZntransfercmdZ error_permr�rr�r�r��pwdr$r#r�makefile� file_closer r�)r�rNr�r��cmd�isdirr�r�r�r&ZftpobjrLrLrMr�v sN $ zftpwrapper.retrfilecCs d|_dS)Nr)r#)r�rLrLrMr%� szftpwrapper.endtransfercCsd|_|jdkr|j�dS)NFr)r!r � real_close)r�rLrLrMr�� s zftpwrapper.closecCs4|j�|jd8_|jdkr0|jr0|j�dS)NrRr)r%r r!r+)r�rLrLrMr(� szftpwrapper.file_closecCs2|j�y|jj�Wnt�k r,YnXdS)N)r%r�r�r)r�rLrLrMr+� s zftpwrapper.real_close)NT) r�r�r�r�r"r�r%r�r(r+rLrLrLrMr�Z s -r�cCs�i}xBtjj�D]4\}}|j�}|r|dd�dkr|||dd�<qWdtjkr^|jdd�xXtjj�D]J\}}|dd�dkrj|j�}|r�|||dd�<qj|j|dd �d�qjW|S) N��_proxyZREQUEST_METHODr�i����i����i����i����i����)rW�environr{rtr�)rr[r�rLrLrM�getproxies_environment� s r/c Cs�|dkrt�}y|d}Wntk r.dSX|dkr<dSt|�\}}dd�|jd�D�}xP|D]H}|rb|jd�}tj|�}d |}tj||tj�s�tj||tj�rbdSqbWdS) N�nor�*rRcSsg|]}|j��qSrL)rN)r�r rLrLrMr#� sz,proxy_bypass_environment.<locals>.<listcomp>r��.z (.+\.)?%s$) r/rqr rB�lstriprPr�rrR)rvrZno_proxy�hostonlyr.Z no_proxy_listr[�patternrLrLrM�proxy_bypass_environment� s& r6c Csddlm}t|�\}}dd�}d|kr4|dr4dSd}x�|jdf�D]�}|sPqFtjd |�}|dk �r|dkr�ytj|�}||�}Wntk r�wFYnX||jd ��} |jd�} | dkr�d|jd �j d�d } nt | d d��} d | } || ?| | ?k�rdSqF|||�rFdSqFWdS)Nr)�fnmatchcSsh|jd�}ttt|��}t|�dkr<|ddddgdd�}|dd>|dd>B|dd>B|d BS) Nr2r�r�rRrbr:rQr=)rBr�r�r^r`)ZipAddrr-rLrLrM�ip2num s z,_proxy_bypass_macosx_sysconf.<locals>.ip2numr2Zexclude_simpleT� exceptionsz(\d+(?:\.\d+)*)(/\d+)?rRr:rQ� F)r7r r�rPrr�r�rm�group�countr^)rv�proxy_settingsr7r4r.r9ZhostIPr�r�r1�maskrLrLrM�_proxy_bypass_macosx_sysconf� s: r@�darwin)�_get_proxy_settings�_get_proxiescCst�}t||�S)N)rBr@)rvr>rLrLrM�proxy_bypass_macosx_sysconf: srDcCst�S)N)rCrLrLrLrM�getproxies_macosx_sysconf> srEcCs t�}|rt||�St|�SdS)N)r/r6rD)rvrrLrLrMrH s rcCst�p t�S)N)r/rErLrLrLrMr5U scCsi}yddl}Wntk r$|SXy�|j|jd�}|j|d�d}|r�t|j|d�d�}d|kr�x�|jd�D]4}|jdd�\}}tjd|�s�d ||f}|||<qrWn>|dd �dkr�||d<n$d ||d<d||d<d||d<|j �Wnt ttfk �rYnX|S)Nrz;Software\Microsoft\Windows\CurrentVersion\Internet Settings�ProxyEnableZProxyServerr�rrRz^([^/:]+)://z%s://%srZzhttp:r�z http://%sz https://%sr�zftp://%sr�) �winreg�ImportError�OpenKey�HKEY_CURRENT_USER�QueryValueExr�rBrPrZClosermrBr�)rrG�internetSettings�proxyEnableZproxyServer�pr�ZaddressrLrLrM�getproxies_registryZ s8 rOcCst�p t�S)N)r/rOrLrLrLrMr5� sc&Cs~yddl}Wntk r dSXy6|j|jd�}|j|d�d}t|j|d�d�}Wntk rldSX|sz|r~dSt|�\}}|g}y tj |�}||kr�|j |�Wntk r�YnXy tj|�}||kr�|j |�Wntk �r�YnX|jd�}xp|D]h} | dk�r*d|k�r*dS| j dd �} | j d d�} | j dd�} x$|D]} tj| | tj��rTdS�qTW�qWdS) Nrz;Software\Microsoft\Windows\CurrentVersion\Internet SettingsrFZ ProxyOverriderz<local>r2rRz\.r1z.*�?)rGrHrIrJrKr�rmr r�r�r]ZgetfqdnrBr�rPrrR)rvrGrLrMZ proxyOverrideZrawHostr.ZaddrZfqdnr2r�rLrLrM�proxy_bypass_registry� sR rQcCs t�}|rt||�St|�SdS)N)r/r6rQ)rvrrLrLrMr� s )NNN)N)~rr�r�reZhttp.clientr�r�rWr/rPr�rr�rc�collectionsrZrTr?Zurllib.errorrrrZurllib.parserrrrr r rrr rrrrrrrrrZurllib.responserrrDrHrC�__all__�version_infor�rFr�r0r1r\r6r7rQ�ASCIIrrrwrrr2rr/rrrrr r!r"r#r$r%�urandomrdr&r'r(ryr)r�r�rEr]rr.r\r^r*r�r+r,r-r�r[Z nturl2pathr4r3r�r8r9rr�rr�rrrrr�r/r6r@�platformZ_scproxyrBrCrDrErr5rOrQrLrLrLrM�<module>Ts�P T ?n$q*@ ov +3:5!AW _ #< - 2 PK23�\���"__pycache__/request.cpython-36.pycnu�[���3 ���i~��)@s�dZddlZddlZddlZddlZddlZddlZddlZddl Z ddl Z ddlZddlZddl Z ddlZddlZddlZddlZddlZddlmZmZmZddlmZmZmZmZmZmZmZmZm Z m!Z!m"Z"m#Z#m$Z$m%Z%m&Z&m'Z'm(Z(m)Z)ddl*m+Z+m,Z,yddl-Z-Wne.k �r&dZ/YnXdZ/dd d ddd ddddddddddddddddddd d!d"d#d$d%d&d'd(g!Z0d)e j1dd*�Z2da3dej4fddddd+�d,d�Z5d-d �Z6gZ7d~d.d%�Z8d/d&�Z9e j:d0e j;�Z<d1d2�Z=Gd3d�d�Z>Gd4d �d �Z?d5d!�Z@Gd6d �d �ZAGd7d�deA�ZBGd8d�deA�ZCGd9d�deA�ZDd:d;�ZEGd<d�deA�ZFGd=d�d�ZGGd>d�deG�ZHGd?d�deH�ZIGd@d�d�ZJGdAd�deJeA�ZKGdBd�deJeA�ZLejMZNGdCd�d�ZOGdDd�deAeO�ZPGdEd�deAeO�ZQGdFdG�dGeA�ZRGdHd�deR�ZSeTejUdI��r2GdJdK�dKeR�ZVe0jWdK�GdLd �d eA�ZXGdMd�deA�ZYdNdO�ZZdPdQ�Z[GdRd�deA�Z\dSdT�Z]GdUd�deA�Z^GdVd�de^�Z_GdWd�deA�Z`dXZaejbdYk�r�ddZlcmdZdmeZend[d#�Zdd\d"�ZeiZfGd]d'�d'�ZgGd^d(�d(eg�Zhdaid_d`�Zjdakdadb�Zldamdcdd�Zndaodedf�ZpGdgdh�dh�Zqdidj�Zrddkdl�Zsdmdn�Zte judok�r�ddplvmwZwmxZxdqdr�Zydsdt�Zzdudv�Z{dwd$�Z|n6ejbdYk�r�dxdy�Z}dzd$�Z|d{d|�Z~d}dv�Z{nerZ|esZ{dS)�a� An extensible library for opening URLs using a variety of protocols The simplest way to use this module is to call the urlopen function, which accepts a string containing a URL or a Request object (described below). It opens the URL and returns the results as file-like object; the returned object has some extra methods described below. The OpenerDirector manages a collection of Handler objects that do all the actual work. Each Handler implements a particular protocol or option. The OpenerDirector is a composite object that invokes the Handlers needed to open the requested URL. For example, the HTTPHandler performs HTTP GET and POST requests and deals with non-error returns. The HTTPRedirectHandler automatically deals with HTTP 301, 302, 303 and 307 redirect errors, and the HTTPDigestAuthHandler deals with digest authentication. urlopen(url, data=None) -- Basic usage is the same as original urllib. pass the url and optionally data to post to an HTTP URL, and get a file-like object back. One difference is that you can also pass a Request instance instead of URL. Raises a URLError (subclass of OSError); for HTTP errors, raises an HTTPError, which can also be treated as a valid response. build_opener -- Function that creates a new OpenerDirector instance. Will install the default handlers. Accepts one or more Handlers as arguments, either instances or Handler classes that it will instantiate. If one of the argument is a subclass of the default handler, the argument will be installed instead of the default. install_opener -- Installs a new opener as the default opener. objects of interest: OpenerDirector -- Sets up the User Agent as the Python-urllib client and manages the Handler classes, while dealing with requests and responses. Request -- An object that encapsulates the state of a request. The state can be as simple as the URL. It can also include extra HTTP headers, e.g. a User-Agent. BaseHandler -- internals: BaseHandler and parent _call_chain conventions Example usage: import urllib.request # set up authentication info authinfo = urllib.request.HTTPBasicAuthHandler() authinfo.add_password(realm='PDQ Application', uri='https://mahler:8092/site-updates.py', user='klem', passwd='geheim$parole') proxy_support = urllib.request.ProxyHandler({"http" : "http://ahad-haam:3128"}) # build a new opener that adds authentication and caching FTP handlers opener = urllib.request.build_opener(proxy_support, authinfo, urllib.request.CacheFTPHandler) # install it urllib.request.install_opener(opener) f = urllib.request.urlopen('http://www.python.org/') �N)�URLError� HTTPError�ContentTooShortError)�urlparse�urlsplit�urljoin�unwrap�quote�unquote� splittype� splithost� splitport� splituser�splitpasswd� splitattr� splitquery� splitvalue�splittag�to_bytes�unquote_to_bytes� urlunparse)� addinfourl�addclosehookFT�Request�OpenerDirector�BaseHandler�HTTPDefaultErrorHandler�HTTPRedirectHandler�HTTPCookieProcessor�ProxyHandler�HTTPPasswordMgr�HTTPPasswordMgrWithDefaultRealm�HTTPPasswordMgrWithPriorAuth�AbstractBasicAuthHandler�HTTPBasicAuthHandler�ProxyBasicAuthHandler�AbstractDigestAuthHandler�HTTPDigestAuthHandler�ProxyDigestAuthHandler�HTTPHandler�FileHandler� FTPHandler�CacheFTPHandler�DataHandler�UnknownHandler�HTTPErrorProcessor�urlopen�install_opener�build_opener�pathname2url�url2pathname� getproxies�urlretrieve� urlcleanup� URLopener�FancyURLopenerz%d.%d�)�cafile�capath� cadefault�contextc Cs�|s|s|rfddl}|jdtd�|dk r2td��ts>td��tjtjj||d�}t |d�}t |�} n0|r~t |d�}t |�} ntdkr�t �a} nt} | j|||�S) a$ Open the URL url, which can be either a string or a Request object. *data* must be an object specifying additional data to be sent to the server, or None if no such data is needed. See Request for details. urllib.request module uses HTTP/1.1 and includes a "Connection:close" header in its HTTP requests. The optional *timeout* parameter specifies a timeout in seconds for blocking operations like the connection attempt (if not specified, the global default timeout setting will be used). This only works for HTTP, HTTPS and FTP connections. If *context* is specified, it must be a ssl.SSLContext instance describing the various SSL options. See HTTPSConnection for more details. The optional *cafile* and *capath* parameters specify a set of trusted CA certificates for HTTPS requests. cafile should point to a single file containing a bundle of CA certificates, whereas capath should point to a directory of hashed certificate files. More information can be found in ssl.SSLContext.load_verify_locations(). The *cadefault* parameter is ignored. This function always returns an object which can work as a context manager and has methods such as * geturl() - return the URL of the resource retrieved, commonly used to determine if a redirect was followed * info() - return the meta-information of the page, such as headers, in the form of an email.message_from_string() instance (see Quick Reference to HTTP Headers) * getcode() - return the HTTP status code of the response. Raises URLError on errors. For HTTP and HTTPS URLs, this function returns a http.client.HTTPResponse object slightly modified. In addition to the three new methods above, the msg attribute contains the same information as the reason attribute --- the reason phrase returned by the server --- instead of the response headers as it is specified in the documentation for HTTPResponse. For FTP, file, and data URLs and requests explicitly handled by legacy URLopener and FancyURLopener classes, this function returns a urllib.response.addinfourl object. Note that None may be returned if no handler handles the request (though the default installed global OpenerDirector uses UnknownHandler to ensure this never happens). In addition, if proxy settings are detected (for example, when a *_proxy environment variable like http_proxy is set), ProxyHandler is default installed and makes sure the requests are handled through the proxy. rNzJcafile, capath and cadefault are deprecated, use a custom context instead.r:zDYou can't pass both context and any of cafile, capath, and cadefaultzSSL support not available)r;r<)r>) �warnings�warn�DeprecationWarning� ValueError� _have_ssl�sslZcreate_default_contextZPurposeZSERVER_AUTH�HTTPSHandlerr2�_opener�open) �url�data�timeoutr;r<r=r>r?Z https_handler�opener�rL�&/usr/lib64/python3.6/urllib/request.pyr0�s*< cCs|adS)N)rF)rKrLrLrMr1�scCs4t|�\}}tjt||����}|j�}|dkrD|rDtjj|�|fS|rTt|d�}nt j dd�}|j}tj |�|��||f} d } d}d}d} d |kr�t|d �}|r�|| | |�xB|j| �}|s�P|t|�7}|j|�| d7} |r�|| | |�q�WWdQRXWdQRX|dk�r0||k�r0td||f| ��| S)aW Retrieve a URL into a temporary location on disk. Requires a URL argument. If a filename is passed, it is used as the temporary file location. The reporthook argument should be a callable that accepts a block number, a read size, and the total file size of the URL target. The data argument should be valid URL encoded data. If a filename is passed and the URL points to a local resource, the result is a copy from local file to new file. Returns a tuple containing the path to the newly created data file as well as the resulting HTTPMessage object. �file�wbF)�deletei��rzcontent-lengthzContent-LengthNz1retrieval incomplete: got only %i out of %i bytesi ���)r� contextlib�closingr0�info�os�path�normpathrG�tempfileZNamedTemporaryFile�name�_url_tempfiles�append�int�read�len�writer)rH�filename� reporthookrIZurl_typerX�fp�headers�tfp�result�bs�sizer_�blocknum�blockrLrLrMr6�sD $cCsHx0tD](}ytj|�Wqtk r,YqXqWtdd�=trDdadS)z0Clean up temporary files from urlretrieve calls.N)r\rW�unlink�OSErrorrF)Z temp_filerLrLrMr7%s z:\d+$cCs<|j}t|�d}|dkr&|jdd�}tjd|d�}|j�S)z�Return request-host, as defined by RFC 2965. Variation from RFC: returned value is lowercased, for convenient comparison. rR��Host)�full_urlr� get_header�_cut_port_re�sub�lower)�requestrH�hostrLrLrM�request_host4srwc@s�eZdZdidddfdd�Zedd��Zejdd��Zejdd��Zed d ��Zejdd ��Zejdd ��Zd d�Z dd�Z dd�Zdd�Zdd�Z dd�Zdd�Zdd�Zd#dd�Zdd �Zd!d"�ZdS)$rNFc Csp||_i|_i|_d|_||_d|_x |j�D]\}}|j||�q.W|dkrVt|�}||_ ||_ |rl||_dS)N)rpre�unredirected_hdrs�_datarI�_tunnel_host�items� add_headerrw�origin_req_host�unverifiable�method) �selfrHrIrer}r~r�key�valuerLrLrM�__init__FszRequest.__init__cCs|jrdj|j|j�S|jS)Nz{}#{})�fragment�format� _full_url)r�rLrLrMrpXszRequest.full_urlcCs(t|�|_t|j�\|_|_|j�dS)N)rr�rr��_parse)r�rHrLrLrMrp^s cCsd|_d|_d|_dS)Nrn)r�r��selector)r�rLrLrMrpescCs|jS)N)ry)r�rLrLrMrIkszRequest.datacCs(||jkr$||_|jd�r$|jd�dS)NzContent-length)ry� has_header� remove_header)r�rIrLrLrMrIos cCs d|_dS)N)rI)r�rLrLrMrIyscCsNt|j�\|_}|jdkr(td|j��t|�\|_|_|jrJt|j�|_dS)Nzunknown url type: %r) rr��typerBrprrvr�r )r��restrLrLrMr�}s zRequest._parsecCs|jdk rdnd}t|d|�S)z3Return a string indicating the HTTP request method.N�POST�GETr)rI�getattr)r�Zdefault_methodrLrLrM� get_method�szRequest.get_methodcCs|jS)N)rp)r�rLrLrM�get_full_url�szRequest.get_full_urlcCs4|jdkr|jr|j|_n||_|j|_||_dS)N�https)r�rzrvrpr�)r�rvr�rLrLrM� set_proxy�s zRequest.set_proxycCs|j|jkS)N)r�rp)r�rLrLrM� has_proxy�szRequest.has_proxycCs||j|j�<dS)N)re� capitalize)r�r��valrLrLrMr|�szRequest.add_headercCs||j|j�<dS)N)rxr�)r�r�r�rLrLrM�add_unredirected_header�szRequest.add_unredirected_headercCs||jkp||jkS)N)rerx)r��header_namerLrLrMr��s zRequest.has_headercCs|jj||jj||��S)N)re�getrx)r�r��defaultrLrLrMrq�szRequest.get_headercCs |jj|d�|jj|d�dS)N)re�poprx)r�r�rLrLrMr��szRequest.remove_headercCs"|jj�}|j|j�t|j��S)N)rx�copy�updatere�listr{)r��hdrsrLrLrM�header_items�s zRequest.header_items)N)�__name__� __module__�__qualname__r��propertyrp�setter�deleterrIr�r�r�r�r�r|r�r�rqr�r�rLrLrLrMrDs( c@sNeZdZdd�Zdd�Zdd�Zdd�Zd ejfd d�Z ddd �Z dd�Zd S)rcCs6dt}d|fg|_g|_i|_i|_i|_i|_dS)NzPython-urllib/%sz User-agent)�__version__� addheaders�handlers�handle_open�handle_error�process_response�process_request)r�Zclient_versionrLrLrMr��szOpenerDirector.__init__cCsZt|d�stdt|���d}�xt|�D�]}|dkr:q*|jd�}|d|�}||dd�}|jd �r�|jd�|d}||dd�}yt|�}Wntk r�YnX|jj |i�} | |j|<n>|d kr�|}|j } n*|dkr�|}|j} n|dkr*|}|j} nq*| j |g�} | �r&tj| |�n | j|�d }q*W|�rVtj|j|�|j|�dS)N� add_parentz%expected BaseHandler instance, got %rF�redirect_request�do_open� proxy_open�_rR�errorrG�responseruT)r�r�r�)�hasattr� TypeErrorr��dir�find� startswithr^rBr�r�r�r�r�� setdefault�bisectZinsortr]r�r�)r��handlerZadded�meth�i�protocolZ condition�j�kind�lookupr�rLrLrM�add_handler�sJ zOpenerDirector.add_handlercCsdS)NrL)r�rLrLrM�close�szOpenerDirector.closec Gs<|j|f�}x*|D]"}t||�}||�}|dk r|SqWdS)N)r�r�) r��chainr�� meth_name�argsr�r��funcrgrLrLrM�_call_chain�s zOpenerDirector._call_chainNc Cs�t|t�rt||�}n|}|dk r(||_||_|j}|d}x(|jj|g�D]}t||�}||�}qLW|j ||�} |d}x*|j j|g�D]}t||�}||| �} q�W| S)NZ_requestZ _response)� isinstance�strrrIrJr�r�r�r��_openr�) r��fullurlrIrJ�reqr�r�Z processorr�r�rLrLrMrG�s" zOpenerDirector.opencCsP|j|jdd|�}|r|S|j}|j|j||d|�}|r>|S|j|jdd|�S)Nr�Zdefault_openr��unknown�unknown_open)r�r�r�)r�r�rIrgr�rLrLrMr�s zOpenerDirector._opencGs~|d kr,|jd}|d}d|}d}|}n|j}|d}d}|||f|}|j|�}|r^|S|rz|dd f|}|j|�SdS)N�httpr�r:z http_error_%srRZ_errorrr��http_error_default)r�r�)r�r�)r��protor��dictr�Zhttp_errZ orig_argsrgrLrLrMr�'s zOpenerDirector.error)N)r�r�r�r�r�r�r��socket�_GLOBAL_DEFAULT_TIMEOUTrGr�r�rLrLrLrMr�s/ c Gs�t�}ttttttttt g }t tjd�r2|j t�t�}xN|D]F}x@|D]8}t|t�rlt||�r�|j|�qHt||�rH|j|�qHWq>Wx|D]}|j|�q�Wx|D]}|j|��q�Wx&|D]}t|t�r�|�}|j|�q�W|S)a*Create an opener object from a list of handlers. The opener will use several default handlers, including support for HTTP, FTP and when applicable HTTPS. If any of the handlers passed as arguments are subclasses of the default handlers, the default handlers will not be used. �HTTPSConnection)rrr.r)rrr+r*r/r-r�r��clientr]rE�setr�r�� issubclass�add�remover�)r�rKZdefault_classes�skip�klassZcheck�hrLrLrMr2@s0 c@s(eZdZdZdd�Zdd�Zdd�ZdS) ri�cCs ||_dS)N)�parent)r�r�rLrLrMr�gszBaseHandler.add_parentcCsdS)NrL)r�rLrLrMr�jszBaseHandler.closecCst|d�sdS|j|jkS)N� handler_orderT)r�r�)r��otherrLrLrM�__lt__ns zBaseHandler.__lt__N)r�r�r�r�r�r�r�rLrLrLrMrdsc@s eZdZdZdZdd�ZeZdS)r/zProcess HTTP error responses.i�cCsJ|j|j|j�}}}d|ko*dknsF|jjd|||||�}|S)N��i,r�)�code�msgrVr�r�)r�rur�r�r�r�rLrLrM� http_response{s z HTTPErrorProcessor.http_responseN)r�r�r��__doc__r�r��https_responserLrLrLrMr/wsc@seZdZdd�ZdS)rcCst|j||||��dS)N)rrp)r�r�rdr�r�r�rLrLrMr��sz*HTTPDefaultErrorHandler.http_error_defaultN)r�r�r�r�rLrLrLrMr�sc@s4eZdZdZdZdd�Zdd�ZeZZZ dZ dS) r�� c sx|j�}|dkr|dkp&|dko&|dks:t|j||||��|jdd �}d�t�fdd �|jj�D��}t|||jdd�S)a�Return a Request or None in response to a redirect. This is called by the http_error_30x methods when a redirection response is received. If a redirection should take place, return a new Request to allow http_error_30x to perform the redirect. Otherwise, raise HTTPError if no-one else should try to handle this url. Return None if you can't but another Handler might. �-�.�/�3r��HEADr�� z%20�content-length�content-typec3s&|]\}}|j��kr||fVqdS)N)rt)�.0�k�v)�CONTENT_HEADERSrLrM� <genexpr>�sz7HTTPRedirectHandler.redirect_request.<locals>.<genexpr>T)rer}r~)r�r�r�r�)r�r�)r�r�r�)r�r�) r�rrp�replacer�rer{rr}) r�r�rdr�r�re�newurl�mZ newheadersrL)r�rMr��s z$HTTPRedirectHandler.redirect_requestc CsNd|kr|d}nd|kr$|d}ndSt|�}|jdkrRt||d||f||��|jrp|jrpt|�}d|d <t|�}t|d tj d�}t |j|�}|j||||||�}|dkr�dSt |d��r|j} |_| j|d �|jks�t| �|jk�rt|j||j|||��ni} |_|_| j|d �d| |<|j�|j�|jj||jd�S)N�location�urir�r��ftprnz+%s - Redirection to url '%s' is not allowed�/r:z iso-8859-1)�encoding�safe� redirect_dictrrR)rJ)r�r�r�rn)r�schemerrXZnetlocr�rr �stringZpunctuationrrpr�r�rr��max_repeatsr`�max_redirections�inf_msgr_r�r�rGrJ) r�r�rdr�r�rer��urlparts�newZvisitedrLrLrM�http_error_302�s@ z"HTTPRedirectHandler.http_error_302zoThe HTTP server returned a redirect error that would lead to an infinite loop. The last 30x error message was: N)r�r�r�rrr�r �http_error_301�http_error_303�http_error_307rrLrLrLrMr�s&<c Cs�t|�\}}|jd�s d}|}n:|jd�s6td|��|jdd�}|dkrNd}|d|�}t|�\}}|dk r|t|�\}}nd}}||||fS)aReturn (scheme, user, password, host/port) given a URL or an authority. If a URL is supplied, it must have an authority (host:port) component. According to RFC 3986, having an authority component means the URL must have two slashes after the scheme. r�Nz//zproxy URL with no authority: %rr:rRrS)rr�rBr�rr) �proxyrZr_scheme� authority�endZuserinfo�hostport�user�passwordrLrLrM�_parse_proxy�s rc@s"eZdZdZddd�Zdd�ZdS)r�dNcCs^|dkrt�}t|d�s td��||_x2|j�D]&\}}t|d||||jfdd��q0WdS)N�keyszproxies must be a mappingz%s_opencSs||||�S)NrL)�rrr�r�rLrLrM�<lambda>%sz'ProxyHandler.__init__.<locals>.<lambda>)r5r��AssertionError�proxiesr{�setattrr�)r�rr�rHrLrLrMr�s zProxyHandler.__init__cCs�|j}t|�\}}}}|dkr"|}|jr6t|j�r6dS|rv|rvdt|�t|�f} tj| j��jd�} |j dd| �t|�}|j ||�||ks�|dkr�dS|jj||j d�SdS)Nz%s:%s�asciizProxy-authorizationzBasic r�)rJ)r�rrv�proxy_bypassr �base64� b64encode�encode�decoder|r�r�rGrJ)r�r�rr�Z orig_typeZ proxy_typerrrZ user_passZcredsrLrLrMr�(s zProxyHandler.proxy_open)N)r�r�r�r�r�r�rLrLrLrMrs c@s6eZdZdd�Zdd�Zdd�Zd dd �Zd d�ZdS)r cCs i|_dS)N)�passwd)r�rLrLrMr�FszHTTPPasswordMgr.__init__cs`t|t�r|g}|�jkr$i�j|<x6dD].�t��fdd�|D��}||f�j||<q*WdS)NTFcsg|]}�j|���qSrL)� reduce_uri)r��u)�default_portr�rLrM� <listcomp>Qsz0HTTPPasswordMgr.add_password.<locals>.<listcomp>)TF)r�r�r"�tuple)r��realmr�rr"�reduced_urirL)r%r�rM�add_passwordIs zHTTPPasswordMgr.add_passwordc Cs`|jj|i�}xLdD]D}|j||�}x2|j�D]&\}}x|D]}|j||�r<|Sq<Wq.WqWdS)NTF)TF)NN)r"r�r#r{� is_suburi) r�r(�authuriZdomainsr%�reduced_authuriZurisZauthinfor�rLrLrM�find_user_passwordTs z"HTTPPasswordMgr.find_user_passwordTc Cs�t|�}|dr.|d}|d}|dp*d}nd}|}d}t|�\}}|r~|dkr~|dk r~ddd�j|�} | dk r~d || f}||fS) z@Accept authority or URI and extract only the authority and path.rRrr:r�N�Pi�)r�r�z%s:%d)rr r�) r�r�r%�partsrrrXrv�portZdportrLrLrMr#^s zHTTPPasswordMgr.reduce_uricCsR||krdS|d|dkr dStj|d|df�}t|�t|d�krNdSdS)zcCheck if test is below base in a URI tree Both args must be URIs in reduced form. TrFrR)� posixpath�commonprefixr`)r��base�test�commonrLrLrMr+uszHTTPPasswordMgr.is_suburiN)T)r�r�r�r�r*r.r#r+rLrLrLrMr Ds c@seZdZdd�ZdS)r!cCs0tj|||�\}}|dk r"||fStj|d|�S)N)r r.)r�r(r,rrrLrLrMr.�s z2HTTPPasswordMgrWithDefaultRealm.find_user_passwordN)r�r�r�r.rLrLrLrMr!�scs<eZdZ�fdd�Zd �fdd� Zddd�Zdd �Z�ZS)r"csi|_t�j||�dS)N)� authenticated�superr�)r�r��kwargs)� __class__rLrMr��sz%HTTPPasswordMgrWithPriorAuth.__init__Fcs<|j||�|dk r&t�jd|||�t�j||||�dS)N)�update_authenticatedr8r*)r�r(r�rr"�is_authenticated)r:rLrMr*�sz)HTTPPasswordMgrWithPriorAuth.add_passwordcCsFt|t�r|g}x0dD](}x"|D]}|j||�}||j|<q WqWdS)NTF)TF)r�r�r#r7)r�r�r<r%r$r)rLrLrMr;�s z1HTTPPasswordMgrWithPriorAuth.update_authenticatedcCsDx>dD]6}|j||�}x$|jD]}|j||�r|j|SqWqWdS)NTF)TF)r#r7r+)r�r,r%r-r�rLrLrMr<�s z-HTTPPasswordMgrWithPriorAuth.is_authenticated)F)F)r�r�r�r�r*r;r<� __classcell__rLrL)r:rMr"�s c@sTeZdZejdej�Zddd�Zdd�Zdd�Z d d �Z dd�Zd d�ZeZ eZdS)r#z1(?:^|,)[ ]*([^ ,]+)[ ]+realm=(["']?)([^"']*)\2NcCs"|dkrt�}||_|jj|_dS)N)r r"r*)r�Zpassword_mgrrLrLrMr��sz!AbstractBasicAuthHandler.__init__ccstd}xFtjj|�D]6}|j�\}}}|d kr:tjdtd�||fVd}qW|sp|rb|j�d}nd}|dfVdS) NF�"�'zBasic Auth Realm was unquoted�Trrn)r>r?)r#�rx�finditer�groupsr?r@�UserWarning�split)r��headerZfound_challengeZmorr r(rLrLrM�_parse_realm�s z%AbstractBasicAuthHandler._parse_realmc Cs~|j|�}|sdSd}xL|D]D}x>|j|�D]0\}}|j�dkrF|}q,|dk r,|j|||�Sq,WqW|dk rztd|f��dS)N�basiczBAbstractBasicAuthHandler does not support the following scheme: %r)Zget_allrGrt�retry_http_basic_authrB) r��authreqrvr�reZunsupportedrFrr(rLrLrM�http_error_auth_reqed�s z.AbstractBasicAuthHandler.http_error_auth_reqedcCs||jj||�\}}|dk rtd||f}dtj|j��jd�}|j|jd�|krTdS|j|j|�|j j ||jd�SdSdS)Nz%s:%szBasic r)rJ)r"r.rrr r!rq�auth_headerr�r�rGrJ)r�rvr�r(r�pw�raw�authrLrLrMrI�sz.AbstractBasicAuthHandler.retry_http_basic_authcCsxt|jd�s|jj|j�r"|S|jd�st|jjd|j�\}}dj||�j�}tj |�j �}|jddj|j���|S)Nr<� Authorizationz{0}:{1}zBasic {}) r�r"r<rpr�r.r�r rZstandard_b64encoder!r��strip)r�r�rr"ZcredentialsZauth_strrLrLrM�http_requests z%AbstractBasicAuthHandler.http_requestcCsLt|jd�rHd|jko dknr8|jj|jd�n|jj|jd�|S)Nr<r�i,TF)r�r"r�r;rp)r�r�r�rLrLrMr�s z&AbstractBasicAuthHandler.http_response)N)r�r�r��re�compile�IrAr�rGrKrIrRr�� https_requestr�rLrLrLrMr#�s c@seZdZdZdd�ZdS)r$rPcCs|j}|jd|||�}|S)Nzwww-authenticate)rprK)r�r�rdr�r�rerHr�rLrLrM�http_error_401 s z#HTTPBasicAuthHandler.http_error_401N)r�r�r�rLrWrLrLrLrMr$sc@seZdZdZdd�ZdS)r%zProxy-authorizationcCs|j}|jd|||�}|S)Nzproxy-authenticate)rvrK)r�r�rdr�r�rerr�rLrLrM�http_error_407+s z$ProxyBasicAuthHandler.http_error_407N)r�r�r�rLrXrLrLrLrMr%'sc@sNeZdZddd�Zdd�Zdd�Zdd �Zd d�Zdd �Zdd�Z dd�Z dS)r&NcCs4|dkrt�}||_|jj|_d|_d|_d|_dS)Nr)r r"r*�retried�nonce_count� last_nonce)r�r"rLrLrMr�Es z"AbstractDigestAuthHandler.__init__cCs d|_dS)Nr)rY)r�rLrLrM�reset_retry_countNsz+AbstractDigestAuthHandler.reset_retry_countcCs||j|d�}|jdkr*t|jdd|d��n|jd7_|rx|j�d}|j�dkr`|j||�S|j�dkrxtd|��dS) N�i�zdigest auth failedrRrZdigestrHzEAbstractDigestAuthHandler does not support the following scheme: '%s')r�rYrrprErt�retry_http_digest_authrB)r�rLrvr�rerJrrLrLrMrKQs z/AbstractDigestAuthHandler.http_error_auth_reqedcCsz|jdd�\}}ttdt|���}|j||�}|rvd|}|jj|jd�|krRdS|j|j|�|j j ||jd�}|SdS)Nr�rRz Digest %s)rJ)rE�parse_keqv_list�filter�parse_http_list�get_authorizationrer�rLr�r�rGrJ)r�r�rO�tokenZ challenge�chalZauth_valZresprLrLrMr^esz0AbstractDigestAuthHandler.retry_http_digest_authcCs@d|j|tj�f}|jd�td�}tj|�j�}|dd�S)Nz %s:%s:%s:rrQ�)rZ�timeZctimer �_randombytes�hashlib�sha1� hexdigest)r��nonce�s�b�digrLrLrM� get_cnonceqsz$AbstractDigestAuthHandler.get_cnoncecCs�y6|d}|d}|jd�}|jdd�}|jdd�}Wntk rJdSX|j|�\}} |dkrfdS|jj||j�\} }| dkr�dS|jdk r�|j|j|�}nd}d| ||f} d|j�|j f}|d k�r.||j kr�|jd 7_nd |_||_ d|j}|j|�}d||||||�f}| || �|�}n2|dk�rT| || �d|||�f�}nt d |��d| |||j |f}|�r�|d|7}|�r�|d|7}|d|7}|�r�|d||f7}|S)Nr(rk�qop� algorithm�MD5�opaquez%s:%s:%sz%s:%srOrRz%08xz%s:%s:%s:%s:%szqop '%s' is not supported.z>username="%s", realm="%s", nonce="%s", uri="%s", response="%s"z , opaque="%s"z , digest="%s"z, algorithm="%s"z, qop=auth, nc=%s, cnonce="%s")r��KeyError�get_algorithm_implsr"r.rprI�get_entity_digestr�r�r[rZror)r�r�rdr(rkrprqrs�H�KDrrMZentdigZA1ZA2ZncvalueZcnonceZnoncebitZrespdigr4rLrLrMrb|sV z+AbstractDigestAuthHandler.get_authorizationcsD|dkrdd��n|dkr$dd��ntd|���fdd�}�|fS)NrrcSstj|jd��j�S)Nr)rhZmd5r rj)�xrLrLrMr�sz?AbstractDigestAuthHandler.get_algorithm_impls.<locals>.<lambda>ZSHAcSstj|jd��j�S)Nr)rhrir rj)ryrLrLrMr�sz.Unsupported digest authentication algorithm %rcs�d||f�S)Nz%s:%srL)rl�d)rwrLrMr�s)rB)r�rqrxrL)rwrMru�s z-AbstractDigestAuthHandler.get_algorithm_implscCsdS)NrL)r�rIrdrLrLrMrv�sz+AbstractDigestAuthHandler.get_entity_digest)N)r�r�r�r�r\rKr^rorbrurvrLrLrLrMr&:s < c@s eZdZdZdZdZdd�ZdS)r'z�An authentication protocol defined by RFC 2069 Digest authentication improves on basic authentication because it does not transmit passwords in the clear. rPi�cCs*t|j�d}|jd|||�}|j�|S)NrRzwww-authenticate)rrprKr\)r�r�rdr�r�rerv�retryrLrLrMrW�s z$HTTPDigestAuthHandler.http_error_401N)r�r�r�r�rLr�rWrLrLrLrMr'�sc@seZdZdZdZdd�ZdS)r(zProxy-Authorizationi�cCs"|j}|jd|||�}|j�|S)Nzproxy-authenticate)rvrKr\)r�r�rdr�r�rervr{rLrLrMrX�s z%ProxyDigestAuthHandler.http_error_407N)r�r�r�rLr�rXrLrLrLrMr(�sc@s6eZdZd dd�Zdd�Zdd�Zdd �Zd d�ZdS)�AbstractHTTPHandlerrcCs ||_dS)N)�_debuglevel)r�� debuglevelrLrLrMr��szAbstractHTTPHandler.__init__cCs ||_dS)N)r})r��levelrLrLrM�set_http_debuglevel�sz'AbstractHTTPHandler.set_http_debuglevelcCstjjj|j|j��S)N)r�r��HTTPConnection�_get_content_lengthrIr�)r�rurLrLrMr��sz'AbstractHTTPHandler._get_content_lengthcCs |j}|std��|jdk r�|j}t|t�r8d}t|��|jd�sN|jdd�|jd�r�|jd�r�|j|�}|dk r�|jdt|��n|jdd�|}|j �r�t |j�\}}t|�\}} |jd�s�|jd|�x2|j jD]&\} }| j�} |j| �s�|j| |�q�W|S) Nz no host givenz\POST data should be bytes, an iterable of bytes, or a file object. It cannot be of type str.zContent-typez!application/x-www-form-urlencodedzContent-lengthzTransfer-encodingZchunkedro)rvrrIr�r�r�r�r�r�r�rr�rr�r�r�)r�rurvrIr�Zcontent_lengthZsel_hostrZselZsel_pathr[r�rLrLrM�do_request_�s> zAbstractHTTPHandler.do_request_c s\|j}|std��||fd|ji|��}|j|j�t|j���jt�fdd�|jj �D���d�d<tdd��j �D���|j r�i}d}|�kr��|||<�|=|j|j |d �y`y&|j|j �|j|j�|jd �d�Wn,tk �r }zt|��WYdd}~XnX|j�} Wn|j��YnX|j�rF|jj�d|_|j�| _| j| _| S) z�Return an HTTPResponse object for the request, using http_class. http_class must implement the HTTPConnection API from http.client. z no host givenrJc3s"|]\}}|�kr||fVqdS)NrL)r�r�r�)rerLrMr�)sz.AbstractHTTPHandler.do_open.<locals>.<genexpr>r�� Connectioncss|]\}}|j�|fVqdS)N)�title)r�r[r�rLrLrMr�6szProxy-Authorization)rezTransfer-encoding)Zencode_chunkedN)rvrrJZset_debuglevelr}r�rxr�rer{rzZ set_tunnelrur�r�rIr�rm�getresponser�Zsockr�rH�reasonr�) r�Z http_classr�Zhttp_conn_argsrvr�Ztunnel_headersZproxy_auth_hdr�errrrL)rerMr�s@ " zAbstractHTTPHandler.do_openN)r)r�r�r�r�r�r�r�r�rLrLrLrMr|�s &r|c@seZdZdd�ZejZdS)r)cCs|jtjj|�S)N)r�r�r�r�)r�r�rLrLrM� http_open`szHTTPHandler.http_openN)r�r�r�r�r|r�rRrLrLrLrMr)^sr�c@s$eZdZddd�Zdd�ZejZdS)rErNcCstj||�||_||_dS)N)r|r��_context�_check_hostname)r�r~r>�check_hostnamerLrLrMr�iszHTTPSHandler.__init__cCs|jtjj||j|jd�S)N)r>r�)r�r�r�r�r�r�)r�r�rLrLrM� https_opennszHTTPSHandler.https_open)rNN)r�r�r�r�r�r|r�rVrLrLrLrMrEgs rEc@s.eZdZddd�Zdd�Zdd�ZeZeZdS) rNcCs$ddl}|dkr|jj�}||_dS)Nr)Zhttp.cookiejar� cookiejarZ CookieJar)r�r�r�rLrLrMr�ws zHTTPCookieProcessor.__init__cCs|jj|�|S)N)r�Zadd_cookie_header)r�rurLrLrMrR}sz HTTPCookieProcessor.http_requestcCs|jj||�|S)N)r�Zextract_cookies)r�rur�rLrLrMr��sz!HTTPCookieProcessor.http_response)N)r�r�r�r�rRr�rVr�rLrLrLrMrvs c@seZdZdd�ZdS)r.cCs|j}td|��dS)Nzunknown url type: %s)r�r)r�r�r�rLrLrMr��szUnknownHandler.unknown_openN)r�r�r�r�rLrLrLrMr.�scCsRi}xH|D]@}|jdd�\}}|ddkrB|ddkrB|dd�}|||<q W|S)z>Parse list of key=value strings where keys are not duplicated.�=rRrr>rSrS)rE)�lZparsedZeltr�r�rLrLrMr_�s r_cCs�g}d}d}}xt|D]l}|r,||7}d}q|rV|dkr@d}qn|dkrLd}||7}q|dkrn|j|�d}q|dkrzd}||7}qW|r�|j|�dd�|D�S) apParse lists as described by RFC 2068 Section 2. In particular, parse comma-separated lists where the elements of the list may include quoted-strings. A quoted-string could contain a comma. A non-quoted string could have quotes in the middle. Neither commas nor quotes count if they are escaped. Only double-quotes count, not single-quotes. rnF�\Tr>�,cSsg|]}|j��qSrL)rQ)r��partrLrLrMr&�sz#parse_http_list.<locals>.<listcomp>)r])rl�resr��escaper ZcurrLrLrMra�s4 rac@s(eZdZdd�ZdZdd�Zdd�ZdS)r*cCs\|j}|dd�dkrN|dd�dkrN|jrN|jdkrN|j|j�krXtd��n |j|�SdS)Nr:z//r@r�� localhostz-file:// scheme is supported only on localhost)r�rv� get_namesr�open_local_file)r�r�rHrLrLrM� file_open�s& zFileHandler.file_openNcCs`tjdkrZy*ttjd�dtjtj��d�t_Wn$tjk rXtjd�ft_YnXtjS)Nr�r:)r*�namesr'r��gethostbyname_ex�gethostname�gaierror� gethostbyname)r�rLrLrMr��s zFileHandler.get_namescCsddl}ddl}|j}|j}t|�}y�tj|�}|j}|jj |j dd�} |j|�d} |jd| pbd|| f�}|r~t |�\}}|s�|r�t|�|j�kr�|r�d||} nd|} tt|d�|| �SWn*tk r�}zt|��WYdd}~XnXtd��dS) NrT)�usegmtz6Content-type: %s Content-length: %d Last-modified: %s z text/plainzfile://�rbzfile not on local host)�email.utils� mimetypesrvr�r4rW�stat�st_size�utils� formatdate�st_mtime� guess_type�message_from_stringr �_safe_gethostbynamer�rrGrmr)r�r��emailr�rvrbZ localfile�statsri�modified�mtyperer1Zorigurl�exprLrLrMr��s0 zFileHandler.open_local_file)r�r�r�r�r�r�r�rLrLrLrMr*�s cCs&y tj|�Stjk r dSXdS)N)r�r�r�)rvrLrLrMr��s r�c@seZdZdd�Zdd�ZdS)r+cCs.ddl}ddl}|j}|s"td��t|�\}}|dkr>|j}nt|�}t|�\}}|rdt|�\}}nd}t |�}|pvd}|p~d}yt j|�}Wn*tk r�}zt|��WYdd}~XnXt |j�\} } | jd�}ttt |��}|dd�|d}}|�r|d�r|dd�}y�|j||||||j�} |�r8d�p:d}x:| D]2}t|�\}}|j�dk�rB|dk�rB|j�}�qBW| j||�\}}d}|j|j�d}|�r�|d |7}|dk �r�|dk�r�|d|7}tj|�}t|||j�S|jk �r(}z$td|�}|jtj �d��WYdd}~XnXdS)Nrzftp error: no host givenrnr�rRrU�Dr��a�Ar�rzzContent-type: %s zContent-length: %d z ftp error: %rr:rSrS)r�r�r�rUrzr�)!�ftplibr�rvrr �FTP_PORTr^rrr r�r�rmrr�rEr��map�connect_ftprJrrt�upper�retrfiler�rpr�r�r� all_errors�with_traceback�sys�exc_info)r�r�r�r�rvr1rr"r�rX�attrs�dirsrN�fwr��attrr�rd�retrlenrer�r��excrLrLrM�ftp_open�s\ zFTPHandler.ftp_openc Cst||||||dd�S)NF)� persistent)� ftpwrapper)r�rr"rvr1r�rJrLrLrMr�1szFTPHandler.connect_ftpN)r�r�r�r�r�rLrLrLrMr+�s5c@s<eZdZdd�Zdd�Zdd�Zdd�Zd d �Zdd�Zd S)r,cCs"i|_i|_d|_d|_d|_dS)Nr�<re)�cacherJ�soonest�delay� max_conns)r�rLrLrMr�8s zCacheFTPHandler.__init__cCs ||_dS)N)r�)r��trLrLrM� setTimeout?szCacheFTPHandler.setTimeoutcCs ||_dS)N)r�)r�r�rLrLrM�setMaxConnsBszCacheFTPHandler.setMaxConnscCsr|||dj|�|f}||jkr4tj�|j|j|<n,t||||||�|j|<tj�|j|j|<|j�|j|S)Nr�)�joinr�rfr�rJr��check_cache)r�rr"rvr1r�rJr�rLrLrMr�Es zCacheFTPHandler.connect_ftpcCs�tj�}|j|krTx@t|jj��D].\}}||kr"|j|j�|j|=|j|=q"Wtt|jj���|_t |j�|j kr�x6t|jj��D]$\}}||jkr�|j|=|j|=Pq�Wtt|jj���|_dS)N)rfr�r�rJr{r�r��min�valuesr`r�)r�r�r�r�rLrLrMr�Ps zCacheFTPHandler.check_cachecCs4x|jj�D]}|j�qW|jj�|jj�dS)N)r�r�r��clearrJ)r��connrLrLrM�clear_cacheds zCacheFTPHandler.clear_cacheN) r�r�r�r�r�r�r�r�r�rLrLrLrMr,5sc@seZdZdd�ZdS)r-cCs~|j}|jdd�\}}|jdd�\}}t|�}|jd�rNtj|�}|dd�}|sVd}tjd|t|�f�}t t j|�||�S) N�:rRr�z;base64�ztext/plain;charset=US-ASCIIz$Content-type: %s Content-length: %d i����)rprEr�endswithr�decodebytesr�r�r`r�io�BytesIO)r�r�rHrrIZ mediatypererLrLrM� data_openks zDataHandler.data_openN)r�r�r�r�rLrLrLrMr-jsr��nt)r4r3cCst|�S)zOS-specific conversion from a relative URL of the 'file' scheme to a file system path; not recommended for general use.)r )�pathnamerLrLrMr4�scCst|�S)zOS-specific conversion from a file system path to a relative URL of the 'file' scheme; not recommended for general use.)r )r�rLrLrMr3�sc@s�eZdZdZdZdeZd*dd�Zdd�Zdd �Z d d�Z dd �Zd+dd�Zd,dd�Z d-dd�Zd.dd�Zdd�Zd/dd�Zd0dd�Zdd�Zer�dd�Zd1d d!�Zd"d#�Zd$d%�Zd&d'�Zd2d(d)�ZdS)3r8a,Class to open URLs. This is a class rather than just a subroutine because we may need more than one set of global protocol-specific options. Note -- this is a base class for those who don't want the automatic handling of errors type 302 (relocated) and 401 (authorization needed).NzPython-urllib/%scKs�dd|jji}tj|tdd�|dkr.t�}t|d�s@td��||_|j d�|_ |j d�|_d |jfdg|_ g|_tj|_d|_t|_dS) NzW%(class)s style of invoking requests is deprecated. Use newer urlopen functions/methods�classr@)� stacklevelrzproxies must be a mapping�key_file� cert_filez User-Agent�Accept�*/*)r�r�)r:r�r?r@rAr5r�rrr�r�r��versionr��_URLopener__tempfilesrWrl�_URLopener__unlink� tempcache�ftpcache)r�rZx509r�rLrLrMr��szURLopener.__init__cCs|j�dS)N)r�)r�rLrLrM�__del__�szURLopener.__del__cCs|j�dS)N)�cleanup)r�rLrLrMr��szURLopener.closecCsZ|jrFx2|jD](}y|j|�Wqtk r4YqXqW|jdd�=|jrV|jj�dS)N)r�r�rmr�r�)r�rNrLrLrMr��s zURLopener.cleanupcGs|jj|�dS)zdAdd a header to be used by the HTTP interface only e.g. u.addheader('Accept', 'sound/basic')N)r�r])r�r�rLrLrM� addheader�szURLopener.addheadercCsntt|��}t|dd�}|jrL||jkrL|j|\}}t|d�}t|||�St|�\}}|s`d}||jkr�|j|}t|�\}} t| �\} }| |f}nd}d|}||_ |j dd�}t||�s�|d kr�|r�|j|||�S|j ||�Sy,|dk�rt||�|�St||�||�SWnVttfk �r.�Yn<tk �rh} ztd | �jtj�d��WYdd} ~ XnXdS)z6Use URLopener().open(file) instead of open(file, 'r').z%/:=&?~#+!$,;'@()*[]|)rr�rNNZopen_�-r�r�zsocket errorr:)rrr r�rGrrrrr�r�r��open_unknown_proxy�open_unknownr�rrrmr�r�r�)r�r�rIrbrerd�urltyperHr� proxyhostrvr�r[r�rLrLrMrG�s< zURLopener.opencCst|�\}}tdd|��dS)z/Overridable interface to open unknown URL type.z url errorzunknown url typeN)rrm)r�r�rIr�rHrLrLrMr�szURLopener.open_unknowncCs t|�\}}tdd||��dS)z/Overridable interface to open unknown URL type.z url errorzinvalid proxy for %sN)rrm)r�rr�rIr�rHrLrLrMr� szURLopener.open_unknown_proxyc Cs&tt|��}|jr&||jkr&|j|St|�\}}|dkr�|sH|dkr�y.|j|�}|j�}|j�tt|�d�|fSt k r�} zWYdd} ~ XnX|j ||�}�zH|j�} |r�t |d�}n|ddl}t|�\} }t|p�d�\} }t|p�d�\}} t |�pd�\}} tjj|�d}|j|�\}}|jj|�tj|d�}z�|| f}|jdk �r^||j|<d}d }d}d}d | k�r�t| d �}|�r�||||�xH|j|�}|�s�P|t|�7}|j|�|d7}|�r�||||��q�WWd|j�XWd|j�X|dk�r"||k�r"td||f|��|S)ztretrieve(url) returns (filename, headers) for a local object or (tempfilename, headers) for a remote object.NrNrRrOrrnirQzcontent-lengthzContent-Lengthz1retrieval incomplete: got only %i out of %i bytesi rS)rrr�rr�rVr�r4rrmrGrZrrrWrX�splitextZmkstempr�r]�fdopenr^r_r`rar)r�rHrbrcrIr�Zurl1rdr�r�rerfrZZgarbagerX�suffix�fdrgrhrir_rjrkrLrLrM�retrievesl zURLopener.retrievecCs(d}d}t|t�r<t|�\}}|r6t|�\}}t|�}|}nt|\}}t|�\}}t|�\} } | }d}| j�dkrvd}n:t| �\}} |r�t|�\}}|r�d| || f}t|�r�|}|s�tdd��|r�t|�}t j |j��jd�}nd}|�rt|�}t j |j��jd�}nd}||�} i}|�r*d||d<|�r<d||d <|�rJ||d <d|d<x|j D]\}}|||<�qZW|dk �r�d |d<| jd|||�n| jd||d�y| j�}Wn"tjjk �r�td��YnXd|jk�o�dkn�rt||jd||j�S|j||j|j|j|j|�SdS)a�Make an HTTP connection using connection_class. This is an internal method that should be called from open_http() or open_https(). Arguments: - connection_factory should take a host name and return an HTTPConnection instance. - url is the url to retrieval or a host, relative-path pair. - data is payload for a POST request or None. Nr�z %s://%s%sz http errorz no host givenrzBasic %szProxy-AuthorizationrPror�r�z!application/x-www-form-urlencodedzContent-Typer�r�)rez$http protocol error: bad status liner�i,zhttp:)r�r�rrr rrtrrmrrr r!r�rur�r�r�Z BadStatusLinerZstatusrr�� http_errorrdr�)r�Zconnection_factoryrHrIZuser_passwdZproxy_passwdrvr�Zrealhostr�r�Z proxy_authrOZ http_connrerFr�r�rLrLrM�_open_generic_httpQsr zURLopener._open_generic_httpcCs|jtjj||�S)zUse HTTP protocol.)r�r�r�r�)r�rHrIrLrLrM� open_http�szURLopener.open_httpc Csbd|}t||�rPt||�}|dkr6||||||�} n|||||||�} | rP| S|j|||||�S)z�Handle http errors. Derived class can override this, or provide specific handlers named http_error_DDD where DDD is the 3-digit error code.z http_error_%dN)r�r�r�) r�rHrd�errcode�errmsgrerIr[rrgrLrLrMr��s zURLopener.http_errorcCs|j�t||||d��dS)z>Default error handler: close the connection and raise OSError.N)r�r)r�rHrdr�r�rerLrLrMr��szURLopener.http_error_defaultcCstjj||j|jd�S)N)r�r�)r�r�r�r�r�)r�rvrLrLrM�_https_connection�szURLopener._https_connectioncCs|j|j||�S)zUse HTTPS protocol.)r�r�)r�rHrIrLrLrM� open_https�szURLopener.open_httpscCs^t|t�std��|dd�dkrP|dd�dkrP|dd�j�dkrPtd ��n |j|�SdS) z/Use local file or FTP depending on form of URL.zEfile error: proxy support for file protocol currently not implementedNr:z//r@r��z localhost/z-file:// scheme is supported only on localhost)r�r�rrtrBr�)r�rHrLrLrM� open_file�s 4 zURLopener.open_filecCs\ddl}ddl}t|�\}}t|�}ytj|�}Wn0tk rb}zt|j|j ��WYdd}~XnX|j } |jj|j dd�} |j|�d}|jd|p�d| | f�}|s�|} |dd�dkr�d |} tt|d �|| �St|�\}}|o�tj|�t�ft�k�rP|} |dd�dk�r d |} n|dd�dk�r>td |��tt|d �|| �Std��dS)zUse local file.rNT)r�z6Content-Type: %s Content-Length: %d Last-modified: %s z text/plainrRr�zfile://r�r:z./zAlocal file url may start with / or file:. Unknown url of type: %sz#local file error: not on local host)r�r�rr4rWr�rmr�strerrorrbr�r�r�r�r�r�rrGr r�r�r��thishostrB)r�rHr�r�rvrNZ localnamer��erir�r�reZurlfiler1rLrLrMr��s: zURLopener.open_local_filecCs�t|t�std��ddl}t|�\}}|s2td��t|�\}}t|�\}}|r\t|�\}}nd}t|�}t|ppd�}t|p|d�}t j |�}|s�ddl}|j}nt |�}t|�\}} t|�}|jd�} | dd�| d} }| o�| d�r�| dd�} | �r| d�rd| d<|||dj| �f}t|j�tk�rlx8t|j�D]*} | |k�r>|j| }|j| =|j��q>Wy�||jk�r�t||||| �|j|<|�s�d}nd }x:| D]2}t|�\}}|j�d k�r�|dk�r�|j�}�q�W|j|j||�\}}|jd|�d}d}|�r|d|7}|dk �r:|dk�r:|d|7}tj|�}t||d|�St�k �r�}z td|�j t!j"�d��WYdd}~XnXdS)zUse FTP protocol.zCftp error: proxy support for ftp protocol currently not implementedrNzftp error: no host givenrnr�rRr�rUr�r�r�r�rzzftp:zContent-Type: %s zContent-Length: %d zftp error %rr:rSrS)r�r�r�rUrzr�)#r�r�rr�rr rrr r�r�r�r�r^rrEr�r`r��MAXFTPCACHEr�r�r�rrtr�r�r�r�r�r� ftperrorsr�r�r�)r�rHr�rvrXr1rr"r�r�r�rNr�r�r�r�r�r�rdr�r�rer�rLrLrM�open_ftp�sp zURLopener.open_ftpc Cs<t|t�std��y|jdd�\}}Wntk rDtdd��YnX|sNd}|jd�}|dkr�d ||d �kr�||dd �}|d |�}nd}g}|jdtj d tj tj����|jd|�|dkr�tj|j d��jd�}nt|�}|jdt|��|jd�|j|�dj|�}tj|�}tj|�}t|||�S)zUse "data" URL.zEdata error: proxy support for data protocol currently not implementedr�rRz data errorzbad data URLztext/plain;charset=US-ASCII�;rr�NrnzDate: %sz%a, %d %b %Y %H:%M:%S GMTzContent-type: %srrzlatin-1zContent-Length: %d� )r�r�rrErBrm�rfindr]rfZstrftimeZgmtimerr�r r!r r`r�r�r�r��StringIOr) r�rHrIr�Zsemirr�re�frLrLrM� open_data3s6 zURLopener.open_data)N)N)N)N)NNN)N)N)N)N)r�r�r�r�r�r�r�r�r�r�r�r�rGr�r�r�r�r�r�r�rCr�r�r�r�rr rLrLrLrMr8�s. $ B\ :c@s�eZdZdZdd�Zdd�Zd#dd�Zd d �Zd$dd�Zd%d d�Z d&dd�Z d'dd�Zd(dd�Zd)dd�Z d*dd�Zd+dd�Zd,dd�Zd-dd �Zd!d"�ZdS).r9z?Derived class with handlers for errors we can handle (perhaps).cOs(tj|f|�|�i|_d|_d|_dS)Nrr�)r8r�� auth_cache�tries�maxtries)r�r�r9rLrLrMr�`szFancyURLopener.__init__cCst||d||�S)z3Default error handling -- don't raise an exception.zhttp:)r)r�rHrdr�r�rerLrLrMr�fsz!FancyURLopener.http_error_defaultNc Csn|jd7_zR|jrJ|j|jkrJt|d�r4|j}n|j}|||dd|�S|j||||||�}|Sd|_XdS)z%Error 302 -- relocated (temporarily).rR�http_error_500i�z)Internal Server Error: Redirect RecursionNr)rr r�rr��redirect_internal) r�rHrdr�r�rerIr�rgrLrLrMr js zFancyURLopener.http_error_302c Csxd|kr|d}nd|kr$|d}ndS|j�t|jd||�}t|�}|jd krnt|||d|||��|j|�S) Nr�r�r�r�r�r�rnz( Redirection to url '%s' is not allowed.)r�r�r�rn)r�rr�rrrrG) r�rHrdr�r�rerIr�rrLrLrMr|s z FancyURLopener.redirect_internalcCs|j||||||�S)z*Error 301 -- also relocated (permanently).)r )r�rHrdr�r�rerIrLrLrMr�szFancyURLopener.http_error_301cCs|j||||||�S)z;Error 303 -- also relocated (essentially identical to 302).)r )r�rHrdr�r�rerIrLrLrMr�szFancyURLopener.http_error_303cCs2|dkr|j||||||�S|j|||||�SdS)z1Error 307 -- relocated, but turn POST into error.N)r r�)r�rHrdr�r�rerIrLrLrMr �szFancyURLopener.http_error_307Fc Cs�d|krtj||||||�|d}tjd|�} | sHtj||||||�| j�\} }| j�dkrttj||||||�|s�tj||||||�d|jd}|dkr�t||�||�St||�|||�SdS)z_Error 401 -- authentication required. This function supports Basic authentication only.zwww-authenticatez![ ]*([^ ]+)[ ]+realm="([^"]*)"rHZretry_�_basic_authN)r8r�rS�matchrCrtr�r�) r�rHrdr�r�rerIr{�stuffrrr(r[rLrLrMrW�s& zFancyURLopener.http_error_401c Cs�d|krtj||||||�|d}tjd|�} | sHtj||||||�| j�\} }| j�dkrttj||||||�|s�tj||||||�d|jd}|dkr�t||�||�St||�|||�SdS)zeError 407 -- proxy authentication required. This function supports Basic authentication only.zproxy-authenticatez![ ]*([^ ]+)[ ]+realm="([^"]*)"rHZretry_proxy_rN)r8r�rSrrCrtr�r�) r�rHrdr�r�rerIr{rrrr(r[rLrLrMrX�s& zFancyURLopener.http_error_407cCs�t|�\}}d||}|jd}t|�\}} t| �\} } | jd�d}| |d�} |j| ||�\}} |pl| srdSdt|dd�t| dd�| f} d| | |jd<|dkr�|j|�S|j||�SdS)Nzhttp://r��@rRz%s:%s@%srn)r)rrrr��get_user_passwdr rG)r�rHr(rIrvr�r�rr�r�� proxyselectorr�rr"rLrLrM�retry_proxy_http_basic_auth�s z*FancyURLopener.retry_proxy_http_basic_authcCs�t|�\}}d||}|jd}t|�\}} t| �\} } | jd�d}| |d�} |j| ||�\}} |pl| srdSdt|dd�t| dd�| f} d| | |jd<|dkr�|j|�S|j||�SdS)Nzhttps://r�rrRz%s:%s@%srn)r)rrrr�rr rG)r�rHr(rIrvr�r�rr�r�rr�rr"rLrLrM�retry_proxy_https_basic_auth�s z+FancyURLopener.retry_proxy_https_basic_authc Cs�t|�\}}|jd�d}||d�}|j|||�\}}|p>|sDdSdt|dd�t|dd�|f}d||} |dkr�|j| �S|j| |�SdS)NrrRz%s:%s@%srn)rzhttp://)rr�rr rG) r�rHr(rIrvr�r�rr"r�rLrLrMrI�s z$FancyURLopener.retry_http_basic_authc Cs�t|�\}}|jd�d}||d�}|j|||�\}}|p>|sDdSdt|dd�t|dd�|f}d||} |dkr�|j| �S|j| |�SdS)NrrRz%s:%s@%srn)rzhttps://)rr�rr rG) r�rHr(rIrvr�r�rr"r�rLrLrM�retry_https_basic_auth s z%FancyURLopener.retry_https_basic_authrcCs`|d|j�}||jkr2|r(|j|=n |j|S|j||�\}}|sJ|rX||f|j|<||fS)Nr)rtr�prompt_user_passwd)r�rvr(r�r�rr"rLrLrMr s zFancyURLopener.get_user_passwdcCsTddl}y,td||f�}|jd|||f�}||fStk rNt�dSXdS)z#Override this in a GUI environment!rNzEnter username for %s at %s: z#Enter password for %s in %s at %s: )NN)�getpass�input�KeyboardInterrupt�print)r�rvr(rrr"rLrLrMr$ sz!FancyURLopener.prompt_user_passwd)N)N)N)N)NF)NF)N)N)N)N)r)r�r�r�r�r�r�r rrrr rWrXrrrIrrrrLrLrLrMr9]s$ cCstdkrtjd�atS)z8Return the IP address of the magic hostname 'localhost'.Nr�)� _localhostr�r�rLrLrLrMr�4 s r�cCsPtdkrLyttjtj��d�aWn(tjk rJttjd�d�aYnXtS)z,Return the IP addresses of the current host.Nr:r�)� _thishostr'r�r�r�r�rLrLrLrMr< srcCstdkrddl}|jatS)z1Return the set of errors raised by the FTP class.Nr)� _ftperrorsr�r�)r�rLrLrMrG srcCstdkrtjd�atS)z%Return an empty email Message object.Nrn)� _noheadersr�r�rLrLrLrM� noheadersP s r"c@sJeZdZdZddd�Zdd�Zdd �Zd d�Zdd �Zdd�Z dd�Z dS)r�z;Class used by open_ftp() for cache of open FTP connections.NTc CsX||_||_||_||_||_||_d|_||_y|j�Wn|j ��YnXdS)Nr) rr"rvr1r�rJ�refcount� keepalive�initr�)r�rr"rvr1r�rJr�rLrLrMr�] szftpwrapper.__init__cCs\ddl}d|_|j�|_|jj|j|j|j�|jj|j |j �dj|j�}|jj |�dS)Nrr�)r��busyZFTPr�Zconnectrvr1rJZloginrr"r�r��cwd)r�r�Z_targetrLrLrMr%m s zftpwrapper.initc-Cs�ddl}|j�|dkr"d}d}nd|}d}y|jj|�Wn*|jk rh|j�|jj|�YnXd}|r�|r�yd|}|jj|�\}}WnR|jk r�}z4t|�dd�d kr�t d |�j tj�d��WYdd}~XnX|�s�|jjd�|�rn|jj �} zJy|jj|�Wn4|jk �rP}zt d |�|�WYdd}~XnXWd|jj| �Xd|}nd }|jj|�\}}d|_t|jd�|j�} |jd7_|j�| |fS)Nrrzr�zTYPE ArRzTYPE zRETR r@Z550z ftp error: %rr:zLIST ZLISTr�)rzr�)r��endtransferr�Zvoidcmdr�r%ZntransfercmdZ error_permr�rr�r�r��pwdr'r&r�makefile� file_closer#r�)r�rNr�r��cmd�isdirr�r�r�r)ZftpobjrLrLrMr�v sN $ zftpwrapper.retrfilecCs d|_dS)Nr)r&)r�rLrLrMr(� szftpwrapper.endtransfercCsd|_|jdkr|j�dS)NFr)r$r#� real_close)r�rLrLrMr�� s zftpwrapper.closecCs4|j�|jd8_|jdkr0|jr0|j�dS)NrRr)r(r#r$r.)r�rLrLrMr+� szftpwrapper.file_closecCs2|j�y|jj�Wnt�k r,YnXdS)N)r(r�r�r)r�rLrLrMr.� s zftpwrapper.real_close)NT)r�r�r�r�r�r%r�r(r�r+r.rLrLrLrMr�Z s -r�cCs�i}xBtjj�D]4\}}|j�}|r|dd�dkr|||dd�<qWdtjkr^|jdd�xXtjj�D]J\}}|dd�dkrj|j�}|r�|||dd �<qj|j|dd �d�qjW|S)aReturn a dictionary of scheme -> proxy server URL mappings. Scan the environment for variables named <scheme>_proxy; this seems to be the standard convention. If you need a different way, you can pass a proxies dictionary to the [Fancy]URLopener constructor. �N�_proxyZREQUEST_METHODr�i����i����i����i����i����)rW�environr{rtr�)rr[r�rLrLrM�getproxies_environment� s r2c Cs�|dkrt�}y|d}Wntk r.dSX|dkr<dSt|�\}}dd�|jd�D�}xP|D]H}|rb|jd �}tj|�}d |}tj||tj�s�tj||tj�rbdSqbWdS)z�Test if proxies should not be used for a particular host. Checks the proxy dict for the value of no_proxy, which should be a list of comma separated DNS suffixes, or '*' for all hosts. N�nor�*rRcSsg|]}|j��qSrL)rQ)r�rrLrLrMr&� sz,proxy_bypass_environment.<locals>.<listcomp>r��.z (.+\.)?%s$) r2rtr rE�lstriprSr�rrU)rvrZno_proxy�hostonlyr1Z no_proxy_listr[�patternrLrLrM�proxy_bypass_environment� s& r9c Csddlm}t|�\}}dd�}d|kr4|dr4dSd}x�|jd f�D]�}|sPqFtjd |�}|dk �r|dkr�ytj|�}||�}Wntk r�wFYnX||jd��} |jd�} | dkr�d |jd�j d�d} nt | dd��} d| } || ?| | ?k�rdSqF|||�rFdSqFWdS)aj Return True iff this host shouldn't be accessed using a proxy This function uses the MacOSX framework SystemConfiguration to fetch the proxy information. proxy_settings come from _scproxy._get_proxy_settings or get mocked ie: { 'exclude_simple': bool, 'exceptions': ['foo.bar', '*.bar.com', '127.0.0.1', '10.1', '10.0/16'] } r)�fnmatchcSsh|jd�}ttt|��}t|�dkr<|ddddgdd�}|dd>|dd>B|dd>B|d BS) Nr5r�r�rRrer:rQr@)rEr�r�r^r`)ZipAddrr0rLrLrM�ip2num s z,_proxy_bypass_macosx_sysconf.<locals>.ip2numr5Zexclude_simpleTN� exceptionsz(\d+(?:\.\d+)*)(/\d+)?rRr:rQ� F)r:r r�rSrr�r�rm�group�countr^)rv�proxy_settingsr:r7r1r<ZhostIPr�r�r4�maskrLrLrM�_proxy_bypass_macosx_sysconf� s: rC�darwin)�_get_proxy_settings�_get_proxiescCst�}t||�S)N)rErC)rvrArLrLrM�proxy_bypass_macosx_sysconf: srGcCst�S)z�Return a dictionary of scheme -> proxy server URL mappings. This function uses the MacOSX framework SystemConfiguration to fetch the proxy information. )rFrLrLrLrM�getproxies_macosx_sysconf> srHcCs t�}|rt||�St|�SdS)z�Return True, if host should be bypassed. Checks proxy settings gathered from the environment, if specified, or from the MacOSX framework SystemConfiguration. N)r2r9rG)rvrrLrLrMrH s rcCst�p t�S)N)r2rHrLrLrLrMr5U scCsi}yddl}Wntk r$|SXy�|j|jd�}|j|d�d}|r�t|j|d�d�}d|kr�x�|jd�D]4}|jdd�\}}tjd |�s�d ||f}|||<qrWn>|dd�dkr�||d <n$d||d <d||d<d||d<|j �Wnt ttfk �rYnX|S)zxReturn a dictionary of scheme -> proxy server URL mappings. Win32 uses the registry to store proxies. rNz;Software\Microsoft\Windows\CurrentVersion\Internet Settings�ProxyEnableZProxyServerr�rrRz^([^/:]+)://z%s://%sr]zhttp:r�z http://%sz https://%sr�zftp://%sr�) �winreg�ImportError�OpenKey�HKEY_CURRENT_USER�QueryValueExr�rErSrZClosermrBr�)rrJ�internetSettings�proxyEnableZproxyServer�pr�ZaddressrLrLrM�getproxies_registryZ s8 rRcCst�p t�S)z�Return a dictionary of scheme -> proxy server URL mappings. Returns settings gathered from the environment, if specified, or the registry. )r2rRrLrLrLrMr5� sc&Cs~yddl}Wntk r dSXy6|j|jd�}|j|d�d}t|j|d�d�}Wntk rldSX|sz|r~dSt|�\}}|g}y tj |�}||kr�|j |�Wntk r�YnXy tj|�}||kr�|j |�Wntk �r�YnX|jd�}xp|D]h} | dk�r*d|k�r*dS| j dd �} | j d d�} | j dd�} x$|D]} tj| | tj��rTdS�qTW�qWdS) Nrz;Software\Microsoft\Windows\CurrentVersion\Internet SettingsrIZ ProxyOverriderz<local>r5rRz\.r4z.*�?)rJrKrLrMrNr�rmr r�r�r]ZgetfqdnrEr�rSrrU)rvrJrOrPZ proxyOverrideZrawHostr1ZaddrZfqdnr5r�rLrLrM�proxy_bypass_registry� sR rTcCs t�}|rt||�St|�SdS)z�Return True, if host should be bypassed. Checks proxy settings gathered from the environment, if specified, or the registry. N)r2r9rT)rvrrLrLrMr� s )NNN)N)r�rr�r�rhZhttp.clientr�r�rWr2rSr�rr�rf�collectionsrZrTr?Zurllib.errorrrrZurllib.parserrrrr r rrr rrrrrrrrrZurllib.responserrrDrKrC�__all__�version_infor�rFr�r0r1r\r6r7rT�ASCIIrrrwrrr2rr/rrrrr r!r"r#r$r%�urandomrgr&r'r(r|r)r�r�rEr]rr.r_rar*r�r+r,r-rr[Z nturl2pathr4r3r�r8r9rr�rrr rr!r"r�r2r9rC�platformZ_scproxyrErFrGrHrr5rRrTrLrLrLrM�<module>Ds�P T ?n$q*@ ov +3:5!AW _ #< - 2 PK23�\��P2��)__pycache__/response.cpython-36.opt-1.pycnu�[���3 \��@s^dZddlZddddgZGdd�dej�ZGdd�de�ZGd d�de�ZGd d�de�ZdS)aResponse classes used by urllib. The base class, addbase, defines a minimal file-like interface, including read() and readline(). The typical response object is an addinfourl instance, which defines an info() method that returns headers and a geturl() method that returns the url. �N�addbase�addclosehook�addinfo� addinfourlcs8eZdZdZ�fdd�Zdd�Zdd�Zdd �Z�ZS) rzOBase class for addinfo and addclosehook. Is a good idea for garbage collection.cs tt|�j|ddd�||_dS)Nz<urllib response>F)�delete)�superr�__init__�fp)�selfr )� __class__��'/usr/lib64/python3.6/urllib/response.pyrszaddbase.__init__cCsd|jjt|�|jfS)Nz<%s at %r whose fp = %r>)r�__name__�id�file)r rrr �__repr__szaddbase.__repr__cCs|jjrtd��|S)NzI/O operation on closed file)r �closed� ValueError)r rrr � __enter__szaddbase.__enter__cCs|j�dS)N)�close)r �type�value� tracebackrrr �__exit__!szaddbase.__exit__) r� __module__�__qualname__�__doc__rrrr� __classcell__rr)rr rs cs,eZdZdZ�fdd�Z�fdd�Z�ZS)rz*Class to add a close hook to an open file.cs tt|�j|�||_||_dS)N)rrr� closehook�hookargs)r r rr)rrr r(szaddclosehook.__init__cs>z(|j}|j}|r&d|_d|_||�Wdtt|�j�XdS)N)rrrrr)r rr)rrr r-szaddclosehook.close)rrrrrrrrr)rr r%scs(eZdZdZ�fdd�Zdd�Z�ZS)rz.class to add an info() method to an open file.cstt|�j|�||_dS)N)rrr�headers)r r r )rrr r<szaddinfo.__init__cCs|jS)N)r )r rrr �info@szaddinfo.info)rrrrrr!rrr)rr r9scs2eZdZdZd �fdd� Zdd�Zdd�Z�ZS) rz9class to add info() and geturl() methods to an open file.Ncs"tt|�j||�||_||_dS)N)rrr�url�code)r r r r"r#)rrr rGszaddinfourl.__init__cCs|jS)N)r#)r rrr �getcodeLszaddinfourl.getcodecCs|jS)N)r")r rrr �geturlOszaddinfourl.geturl)N)rrrrrr$r%rrr)rr rDs)rZtempfile�__all__Z_TemporaryFileWrapperrrrrrrrr �<module>sPK23�\3<�Z Z )__pycache__/response.cpython-36.opt-2.pycnu�[���3 \��@sZddlZddddgZGdd�dej�ZGdd�de�ZGdd�de�ZGd d�de�ZdS) �N�addbase�addclosehook�addinfo� addinfourlcs4eZdZ�fdd�Zdd�Zdd�Zdd�Z�ZS) rcs tt|�j|ddd�||_dS)Nz<urllib response>F)�delete)�superr�__init__�fp)�selfr )� __class__��'/usr/lib64/python3.6/urllib/response.pyrszaddbase.__init__cCsd|jjt|�|jfS)Nz<%s at %r whose fp = %r>)r�__name__�id�file)r rrr �__repr__szaddbase.__repr__cCs|jjrtd��|S)NzI/O operation on closed file)r �closed� ValueError)r rrr � __enter__szaddbase.__enter__cCs|j�dS)N)�close)r �type�value� tracebackrrr �__exit__!szaddbase.__exit__)r� __module__�__qualname__rrrr� __classcell__rr)rr rscs(eZdZ�fdd�Z�fdd�Z�ZS)rcs tt|�j|�||_||_dS)N)rrr� closehook�hookargs)r r rr)rrr r(szaddclosehook.__init__cs>z(|j}|j}|r&d|_d|_||�Wdtt|�j�XdS)N)rrrrr)r rr)rrr r-szaddclosehook.close)rrrrrrrr)rr r%scs$eZdZ�fdd�Zdd�Z�ZS)rcstt|�j|�||_dS)N)rrr�headers)r r r)rrr r<szaddinfo.__init__cCs|jS)N)r)r rrr �info@szaddinfo.info)rrrrr rrr)rr r9scs.eZdZd�fdd� Zdd�Zdd�Z�ZS) rNcs"tt|�j||�||_||_dS)N)rrr�url�code)r r rr!r")rrr rGszaddinfourl.__init__cCs|jS)N)r")r rrr �getcodeLszaddinfourl.getcodecCs|jS)N)r!)r rrr �geturlOszaddinfourl.geturl)N)rrrrr#r$rrr)rr rDs)Ztempfile�__all__Z_TemporaryFileWrapperrrrrrrrr �<module> s PK23�\��P2��#__pycache__/response.cpython-36.pycnu�[���3 \��@s^dZddlZddddgZGdd�dej�ZGdd�de�ZGd d�de�ZGd d�de�ZdS)aResponse classes used by urllib. The base class, addbase, defines a minimal file-like interface, including read() and readline(). The typical response object is an addinfourl instance, which defines an info() method that returns headers and a geturl() method that returns the url. �N�addbase�addclosehook�addinfo� addinfourlcs8eZdZdZ�fdd�Zdd�Zdd�Zdd �Z�ZS) rzOBase class for addinfo and addclosehook. Is a good idea for garbage collection.cs tt|�j|ddd�||_dS)Nz<urllib response>F)�delete)�superr�__init__�fp)�selfr )� __class__��'/usr/lib64/python3.6/urllib/response.pyrszaddbase.__init__cCsd|jjt|�|jfS)Nz<%s at %r whose fp = %r>)r�__name__�id�file)r rrr �__repr__szaddbase.__repr__cCs|jjrtd��|S)NzI/O operation on closed file)r �closed� ValueError)r rrr � __enter__szaddbase.__enter__cCs|j�dS)N)�close)r �type�value� tracebackrrr �__exit__!szaddbase.__exit__) r� __module__�__qualname__�__doc__rrrr� __classcell__rr)rr rs cs,eZdZdZ�fdd�Z�fdd�Z�ZS)rz*Class to add a close hook to an open file.cs tt|�j|�||_||_dS)N)rrr� closehook�hookargs)r r rr)rrr r(szaddclosehook.__init__cs>z(|j}|j}|r&d|_d|_||�Wdtt|�j�XdS)N)rrrrr)r rr)rrr r-szaddclosehook.close)rrrrrrrrr)rr r%scs(eZdZdZ�fdd�Zdd�Z�ZS)rz.class to add an info() method to an open file.cstt|�j|�||_dS)N)rrr�headers)r r r )rrr r<szaddinfo.__init__cCs|jS)N)r )r rrr �info@szaddinfo.info)rrrrrr!rrr)rr r9scs2eZdZdZd �fdd� Zdd�Zdd�Z�ZS) rz9class to add info() and geturl() methods to an open file.Ncs"tt|�j||�||_||_dS)N)rrr�url�code)r r r r"r#)rrr rGszaddinfourl.__init__cCs|jS)N)r#)r rrr �getcodeLszaddinfourl.getcodecCs|jS)N)r")r rrr �geturlOszaddinfourl.geturl)N)rrrrrr$r%rrr)rr rDs)rZtempfile�__all__Z_TemporaryFileWrapperrrrrrrrr �<module>sPK23�\� ���,__pycache__/robotparser.cpython-36.opt-1.pycnu�[���3 \�"�@s\dZddlZddlZddlZdgZejdd�ZGdd�d�ZGdd�d�Z Gd d �d �Z dS)a% robotparser.py Copyright (C) 2000 Bastian Kleineidam You can choose between two licenses when using this package: 1) GNU GPLv2 2) PSF license for Python 2.2 The robots.txt Exclusion Protocol is implemented as specified in http://www.robotstxt.org/norobots-rfc.txt �N�RobotFileParser�RequestRatezrequests secondsc@sjeZdZdZddd�Zdd�Zdd�Zd d �Zdd�Zd d�Z dd�Z dd�Zdd�Zdd�Z dd�ZdS)rzs This class provides a set of methods to read, parse and answer questions about a single robots.txt file. �cCs,g|_d|_d|_d|_|j|�d|_dS)NFr)�entries� default_entry�disallow_all� allow_all�set_url�last_checked)�self�url�r �*/usr/lib64/python3.6/urllib/robotparser.py�__init__s zRobotFileParser.__init__cCs|jS)z�Returns the time the robots.txt file was last fetched. This is useful for long-running web spiders that need to check for new robots.txt files periodically. )r )rr r r�mtime$szRobotFileParser.mtimecCsddl}|j�|_dS)zYSets the time the robots.txt file was last fetched to the current time. rN)�timer )rrr r r�modified-szRobotFileParser.modifiedcCs&||_tjj|�dd�\|_|_dS)z,Sets the URL referring to a robots.txt file.��N)r�urllib�parse�urlparse�host�path)rrr r rr 5szRobotFileParser.set_urlcCs�ytjj|j�}WnRtjjk rd}z2|jdkr:d|_n|jdkrT|jdkrTd|_WYdd}~XnX|j �}|j |jd�j��dS) z4Reads the robots.txt URL and feeds it to the parser.��Ti�i�Nzutf-8)rr) rZrequestZurlopenr�errorZ HTTPError�coderr�readr�decode� splitlines)r�f�err�rawr r rr:s zRobotFileParser.readcCs,d|jkr|jdkr(||_n|jj|�dS)N�*)� useragentsrr�append)r�entryr r r� _add_entryGs zRobotFileParser._add_entrycCs6d}t�}|j��x|D�]�}|sT|dkr8t�}d}n|dkrT|j|�t�}d}|jd�}|dkrr|d|�}|j�}|s�q|jdd�}t|�dkr|dj�j�|d<tj j |dj��|d<|ddk�r|dkr�|j|�t�}|jj|d�d}q|ddk�r4|dk�r|j jt|dd ��d}q|dd k�rh|dk�r|j jt|dd��d}q|ddk�r�|dk�r|dj�j��r�t|d�|_d}q|dd kr|dkr|djd�}t|�dk�r|dj�j��r|dj�j��rtt|d�t|d��|_d}qW|dk�r2|j|�dS)z�Parse the input lines from a robots.txt file. We allow that a user-agent: line is not preceded by one or more blank lines. rr��#N�:z user-agentZdisallowFZallowTzcrawl-delayzrequest-rate�/)�Entryrr(�find�strip�split�len�lowerrr�unquoter%r&� rulelines�RuleLine�isdigit�int�delayr�req_rate)r�lines�stater'�line�iZnumbersr r rrPsd zRobotFileParser.parsecCs�|jr dS|jrdS|jsdStjjtjj|��}tjjdd|j|j |j |jf�}tjj|�}|sfd}x"|j D]}|j|�rn|j|�SqnW|jr�|jj|�SdS)z=using the parsed robots.txt decide if useragent can fetch urlFTrr,)rrr rrrr3� urlunparserZparamsZqueryZfragment�quoter� applies_to� allowancer)r� useragentrZ parsed_urlr'r r r� can_fetch�s$ zRobotFileParser.can_fetchcCs4|j�sdSx|jD]}|j|�r|jSqW|jjS)N)rrr@r8r)rrBr'r r r�crawl_delay�s zRobotFileParser.crawl_delaycCs4|j�sdSx|jD]}|j|�r|jSqW|jjS)N)rrr@r9r)rrBr'r r r�request_rate�s zRobotFileParser.request_ratecCs0|j}|jdk r||jg}djtt|��dS)N� )rr�join�map�str)rrr r r�__str__�s zRobotFileParser.__str__N)r)�__name__� __module__�__qualname__�__doc__rrrr rr(rrCrDrErJr r r rrs Cc@s(eZdZdZdd�Zdd�Zdd�ZdS) r5zoA rule line is a single "Allow:" (allowance==True) or "Disallow:" (allowance==False) followed by a path.cCs>|dkr|rd}tjjtjj|��}tjj|�|_||_dS)NrT)rrr>rr?rrA)rrrAr r rr�s zRuleLine.__init__cCs|jdkp|j|j�S)Nr$)r� startswith)r�filenamer r rr@�szRuleLine.applies_tocCs|jr dndd|jS)NZAllowZDisallowz: )rAr)rr r rrJ�szRuleLine.__str__N)rKrLrMrNrr@rJr r r rr5�sr5c@s0eZdZdZdd�Zdd�Zdd�Zdd �Zd S)r-z?An entry has one or more user-agents and zero or more rulelinescCsg|_g|_d|_d|_dS)N)r%r4r8r9)rr r rr�szEntry.__init__cCs�g}x|jD]}|jd|���qW|jdk r@|jd|j���|jdk rj|j}|jd|j�d|j���|jtt|j ��|jd�dj |�S)NzUser-agent: z Crawl-delay: zRequest-rate: r,rrF)r%r&r8r9ZrequestsZseconds�extendrHrIr4rG)rZret�agentZrater r rrJ�s z Entry.__str__cCsF|jd�dj�}x.|jD]$}|dkr*dS|j�}||krdSqWdS)z2check if this entry applies to the specified agentr,rr$TF)r0r2r%)rrBrRr r rr@�szEntry.applies_tocCs$x|jD]}|j|�r|jSqWdS)zZPreconditions: - our agent applies to this entry - filename is URL decodedT)r4r@rA)rrPr<r r rrA�s zEntry.allowanceN)rKrLrMrNrrJr@rAr r r rr-�s r-)rN�collectionsZurllib.parserZurllib.request�__all__� namedtuplerrr5r-r r r r�<module>s2PK23�\w��33,__pycache__/robotparser.cpython-36.opt-2.pycnu�[���3 \�"�@sXddlZddlZddlZdgZejdd�ZGdd�d�ZGdd�d�ZGdd �d �Z dS) �N�RobotFileParser�RequestRatezrequests secondsc@sfeZdZddd�Zdd�Zdd�Zdd �Zd d�Zdd �Zdd�Z dd�Z dd�Zdd�Zdd�Z dS)r�cCs,g|_d|_d|_d|_|j|�d|_dS)NFr)�entries� default_entry�disallow_all� allow_all�set_url�last_checked)�self�url�r �*/usr/lib64/python3.6/urllib/robotparser.py�__init__s zRobotFileParser.__init__cCs|jS)N)r )rr r r�mtime$szRobotFileParser.mtimecCsddl}|j�|_dS)Nr)�timer )rrr r r�modified-szRobotFileParser.modifiedcCs&||_tjj|�dd�\|_|_dS)N��)r�urllib�parse�urlparse�host�path)rrr r rr 5szRobotFileParser.set_urlcCs�ytjj|j�}WnRtjjk rd}z2|jdkr:d|_n|jdkrT|jdkrTd|_WYdd}~XnX|j �}|j |jd�j��dS)N��Ti�i�zutf-8)rr) rZrequestZurlopenr�errorZ HTTPError�coderr�readr�decode� splitlines)r�f�err�rawr r rr:s zRobotFileParser.readcCs,d|jkr|jdkr(||_n|jj|�dS)N�*)� useragentsrr�append)r�entryr r r� _add_entryGs zRobotFileParser._add_entrycCs6d}t�}|j��x|D�]�}|sT|dkr8t�}d}n|dkrT|j|�t�}d}|jd�}|dkrr|d|�}|j�}|s�q|jdd�}t|�dkr|dj�j�|d<tj j |dj��|d<|ddk�r|dkr�|j|�t�}|jj|d�d}q|ddk�r4|dk�r|j jt|dd��d}q|dd k�rh|dk�r|j jt|dd ��d}q|ddk�r�|dk�r|dj�j��r�t|d�|_d}q|ddkr|dkr|djd �}t|�dk�r|dj�j��r|dj�j��rtt|d�t|d��|_d}qW|dk�r2|j|�dS)Nrr��#�:z user-agentZdisallowFZallowTzcrawl-delayzrequest-rate�/)�Entryrr(�find�strip�split�len�lowerrr�unquoter%r&� rulelines�RuleLine�isdigit�int�delayr�req_rate)r�lines�stater'�line�iZnumbersr r rrPsd zRobotFileParser.parsecCs�|jr dS|jrdS|jsdStjjtjj|��}tjjdd|j|j |j |jf�}tjj|�}|sfd}x"|j D]}|j|�rn|j|�SqnW|jr�|jj|�SdS)NFTrr,)rrr rrrr3� urlunparserZparamsZqueryZfragment�quoter� applies_to� allowancer)r� useragentrZ parsed_urlr'r r r� can_fetch�s$ zRobotFileParser.can_fetchcCs4|j�sdSx|jD]}|j|�r|jSqW|jjS)N)rrr@r8r)rrBr'r r r�crawl_delay�s zRobotFileParser.crawl_delaycCs4|j�sdSx|jD]}|j|�r|jSqW|jjS)N)rrr@r9r)rrBr'r r r�request_rate�s zRobotFileParser.request_ratecCs0|j}|jdk r||jg}djtt|��dS)N� )rr�join�map�str)rrr r r�__str__�s zRobotFileParser.__str__N)r)�__name__� __module__�__qualname__rrrr rr(rrCrDrErJr r r rrs Cc@s$eZdZdd�Zdd�Zdd�ZdS)r5cCs>|dkr|rd}tjjtjj|��}tjj|�|_||_dS)NrT)rrr>rr?rrA)rrrAr r rr�s zRuleLine.__init__cCs|jdkp|j|j�S)Nr$)r� startswith)r�filenamer r rr@�szRuleLine.applies_tocCs|jr dndd|jS)NZAllowZDisallowz: )rAr)rr r rrJ�szRuleLine.__str__N)rKrLrMrr@rJr r r rr5�sr5c@s,eZdZdd�Zdd�Zdd�Zdd�Zd S) r-cCsg|_g|_d|_d|_dS)N)r%r4r8r9)rr r rr�szEntry.__init__cCs�g}x|jD]}|jd|���qW|jdk r@|jd|j���|jdk rj|j}|jd|j�d|j���|jtt|j ��|jd�dj |�S)NzUser-agent: z Crawl-delay: zRequest-rate: r,rrF)r%r&r8r9ZrequestsZseconds�extendrHrIr4rG)rZret�agentZrater r rrJ�s z Entry.__str__cCsF|jd�dj�}x.|jD]$}|dkr*dS|j�}||krdSqWdS)Nr,rr$TF)r0r2r%)rrBrQr r rr@�szEntry.applies_tocCs$x|jD]}|j|�r|jSqWdS)NT)r4r@rA)rrOr<r r rrA�s zEntry.allowanceN)rKrLrMrrJr@rAr r r rr-�s r-) �collectionsZurllib.parserZurllib.request�__all__� namedtuplerrr5r-r r r r�<module> s2PK23�\� ���&__pycache__/robotparser.cpython-36.pycnu�[���3 \�"�@s\dZddlZddlZddlZdgZejdd�ZGdd�d�ZGdd�d�Z Gd d �d �Z dS)a% robotparser.py Copyright (C) 2000 Bastian Kleineidam You can choose between two licenses when using this package: 1) GNU GPLv2 2) PSF license for Python 2.2 The robots.txt Exclusion Protocol is implemented as specified in http://www.robotstxt.org/norobots-rfc.txt �N�RobotFileParser�RequestRatezrequests secondsc@sjeZdZdZddd�Zdd�Zdd�Zd d �Zdd�Zd d�Z dd�Z dd�Zdd�Zdd�Z dd�ZdS)rzs This class provides a set of methods to read, parse and answer questions about a single robots.txt file. �cCs,g|_d|_d|_d|_|j|�d|_dS)NFr)�entries� default_entry�disallow_all� allow_all�set_url�last_checked)�self�url�r �*/usr/lib64/python3.6/urllib/robotparser.py�__init__s zRobotFileParser.__init__cCs|jS)z�Returns the time the robots.txt file was last fetched. This is useful for long-running web spiders that need to check for new robots.txt files periodically. )r )rr r r�mtime$szRobotFileParser.mtimecCsddl}|j�|_dS)zYSets the time the robots.txt file was last fetched to the current time. rN)�timer )rrr r r�modified-szRobotFileParser.modifiedcCs&||_tjj|�dd�\|_|_dS)z,Sets the URL referring to a robots.txt file.��N)r�urllib�parse�urlparse�host�path)rrr r rr 5szRobotFileParser.set_urlcCs�ytjj|j�}WnRtjjk rd}z2|jdkr:d|_n|jdkrT|jdkrTd|_WYdd}~XnX|j �}|j |jd�j��dS) z4Reads the robots.txt URL and feeds it to the parser.��Ti�i�Nzutf-8)rr) rZrequestZurlopenr�errorZ HTTPError�coderr�readr�decode� splitlines)r�f�err�rawr r rr:s zRobotFileParser.readcCs,d|jkr|jdkr(||_n|jj|�dS)N�*)� useragentsrr�append)r�entryr r r� _add_entryGs zRobotFileParser._add_entrycCs6d}t�}|j��x|D�]�}|sT|dkr8t�}d}n|dkrT|j|�t�}d}|jd�}|dkrr|d|�}|j�}|s�q|jdd�}t|�dkr|dj�j�|d<tj j |dj��|d<|ddk�r|dkr�|j|�t�}|jj|d�d}q|ddk�r4|dk�r|j jt|dd ��d}q|dd k�rh|dk�r|j jt|dd��d}q|ddk�r�|dk�r|dj�j��r�t|d�|_d}q|dd kr|dkr|djd�}t|�dk�r|dj�j��r|dj�j��rtt|d�t|d��|_d}qW|dk�r2|j|�dS)z�Parse the input lines from a robots.txt file. We allow that a user-agent: line is not preceded by one or more blank lines. rr��#N�:z user-agentZdisallowFZallowTzcrawl-delayzrequest-rate�/)�Entryrr(�find�strip�split�len�lowerrr�unquoter%r&� rulelines�RuleLine�isdigit�int�delayr�req_rate)r�lines�stater'�line�iZnumbersr r rrPsd zRobotFileParser.parsecCs�|jr dS|jrdS|jsdStjjtjj|��}tjjdd|j|j |j |jf�}tjj|�}|sfd}x"|j D]}|j|�rn|j|�SqnW|jr�|jj|�SdS)z=using the parsed robots.txt decide if useragent can fetch urlFTrr,)rrr rrrr3� urlunparserZparamsZqueryZfragment�quoter� applies_to� allowancer)r� useragentrZ parsed_urlr'r r r� can_fetch�s$ zRobotFileParser.can_fetchcCs4|j�sdSx|jD]}|j|�r|jSqW|jjS)N)rrr@r8r)rrBr'r r r�crawl_delay�s zRobotFileParser.crawl_delaycCs4|j�sdSx|jD]}|j|�r|jSqW|jjS)N)rrr@r9r)rrBr'r r r�request_rate�s zRobotFileParser.request_ratecCs0|j}|jdk r||jg}djtt|��dS)N� )rr�join�map�str)rrr r r�__str__�s zRobotFileParser.__str__N)r)�__name__� __module__�__qualname__�__doc__rrrr rr(rrCrDrErJr r r rrs Cc@s(eZdZdZdd�Zdd�Zdd�ZdS) r5zoA rule line is a single "Allow:" (allowance==True) or "Disallow:" (allowance==False) followed by a path.cCs>|dkr|rd}tjjtjj|��}tjj|�|_||_dS)NrT)rrr>rr?rrA)rrrAr r rr�s zRuleLine.__init__cCs|jdkp|j|j�S)Nr$)r� startswith)r�filenamer r rr@�szRuleLine.applies_tocCs|jr dndd|jS)NZAllowZDisallowz: )rAr)rr r rrJ�szRuleLine.__str__N)rKrLrMrNrr@rJr r r rr5�sr5c@s0eZdZdZdd�Zdd�Zdd�Zdd �Zd S)r-z?An entry has one or more user-agents and zero or more rulelinescCsg|_g|_d|_d|_dS)N)r%r4r8r9)rr r rr�szEntry.__init__cCs�g}x|jD]}|jd|���qW|jdk r@|jd|j���|jdk rj|j}|jd|j�d|j���|jtt|j ��|jd�dj |�S)NzUser-agent: z Crawl-delay: zRequest-rate: r,rrF)r%r&r8r9ZrequestsZseconds�extendrHrIr4rG)rZret�agentZrater r rrJ�s z Entry.__str__cCsF|jd�dj�}x.|jD]$}|dkr*dS|j�}||krdSqWdS)z2check if this entry applies to the specified agentr,rr$TF)r0r2r%)rrBrRr r rr@�szEntry.applies_tocCs$x|jD]}|j|�r|jSqWdS)zZPreconditions: - our agent applies to this entry - filename is URL decodedT)r4r@rA)rrPr<r r rrA�s zEntry.allowanceN)rKrLrMrNrrJr@rAr r r rr-�s r-)rN�collectionsZurllib.parserZurllib.request�__all__� namedtuplerrr5r-r r r r�<module>s2PK23�\__init__.pynu�[���PK23�\�Ɇ�Q Q error.pynu�[���"""Exception classes raised by urllib. The base exception class is URLError, which inherits from OSError. It doesn't define any behavior of its own, but is the base class for all exceptions defined in this package. HTTPError is an exception class that is also a valid HTTP response instance. It behaves this way because HTTP protocol errors are valid responses, with a status code, headers, and a body. In some contexts, an application may want to handle an exception like a regular response. """ import urllib.response __all__ = ['URLError', 'HTTPError', 'ContentTooShortError'] class URLError(OSError): # URLError is a sub-type of OSError, but it doesn't share any of # the implementation. need to override __init__ and __str__. # It sets self.args for compatibility with other EnvironmentError # subclasses, but args doesn't have the typical format with errno in # slot 0 and strerror in slot 1. This may be better than nothing. def __init__(self, reason, filename=None): self.args = reason, self.reason = reason if filename is not None: self.filename = filename def __str__(self): return '<urlopen error %s>' % self.reason class HTTPError(URLError, urllib.response.addinfourl): """Raised when HTTP error occurs, but also acts like non-error return""" __super_init = urllib.response.addinfourl.__init__ def __init__(self, url, code, msg, hdrs, fp): self.code = code self.msg = msg self.hdrs = hdrs self.fp = fp self.filename = url # The addinfourl classes depend on fp being a valid file # object. In some cases, the HTTPError may not have a valid # file object. If this happens, the simplest workaround is to # not initialize the base classes. if fp is not None: self.__super_init(fp, hdrs, url, code) def __str__(self): return 'HTTP Error %s: %s' % (self.code, self.msg) def __repr__(self): return '<HTTPError %s: %r>' % (self.code, self.msg) # since URLError specifies a .reason attribute, HTTPError should also # provide this attribute. See issue13211 for discussion. @property def reason(self): return self.msg @property def headers(self): return self.hdrs @headers.setter def headers(self, headers): self.hdrs = headers class ContentTooShortError(URLError): """Exception raised when downloaded size does not match content-length.""" def __init__(self, message, content): URLError.__init__(self, message) self.content = content PK23�\�g��e�e�parse.pynu�[���"""Parse (absolute and relative) URLs. urlparse module is based upon the following RFC specifications. RFC 3986 (STD66): "Uniform Resource Identifiers" by T. Berners-Lee, R. Fielding and L. Masinter, January 2005. RFC 2732 : "Format for Literal IPv6 Addresses in URL's by R.Hinden, B.Carpenter and L.Masinter, December 1999. RFC 2396: "Uniform Resource Identifiers (URI)": Generic Syntax by T. Berners-Lee, R. Fielding, and L. Masinter, August 1998. RFC 2368: "The mailto URL scheme", by P.Hoffman , L Masinter, J. Zawinski, July 1998. RFC 1808: "Relative Uniform Resource Locators", by R. Fielding, UC Irvine, June 1995. RFC 1738: "Uniform Resource Locators (URL)" by T. Berners-Lee, L. Masinter, M. McCahill, December 1994 RFC 3986 is considered the current standard and any future changes to urlparse module should conform with it. The urlparse module is currently not entirely compliant with this RFC due to defacto scenarios for parsing, and for backward compatibility purposes, some parsing quirks from older RFCs are retained. The testcases in test_urlparse.py provides a good indicator of parsing behavior. The WHATWG URL Parser spec should also be considered. We are not compliant with it either due to existing user code API behavior expectations (Hyrum's Law). It serves as a useful guide when making changes. """ import re import os import sys import collections import ipaddress __all__ = ["urlparse", "urlunparse", "urljoin", "urldefrag", "urlsplit", "urlunsplit", "urlencode", "parse_qs", "parse_qsl", "quote", "quote_plus", "quote_from_bytes", "unquote", "unquote_plus", "unquote_to_bytes", "DefragResult", "ParseResult", "SplitResult", "DefragResultBytes", "ParseResultBytes", "SplitResultBytes"] # A classification of schemes. # The empty string classifies URLs with no scheme specified, # being the default value returned by “urlsplit” and “urlparse”. uses_relative = ['', 'ftp', 'http', 'gopher', 'nntp', 'imap', 'wais', 'file', 'https', 'shttp', 'mms', 'prospero', 'rtsp', 'rtspu', 'sftp', 'svn', 'svn+ssh', 'ws', 'wss'] uses_netloc = ['', 'ftp', 'http', 'gopher', 'nntp', 'telnet', 'imap', 'wais', 'file', 'mms', 'https', 'shttp', 'snews', 'prospero', 'rtsp', 'rtspu', 'rsync', 'svn', 'svn+ssh', 'sftp', 'nfs', 'git', 'git+ssh', 'ws', 'wss'] uses_params = ['', 'ftp', 'hdl', 'prospero', 'http', 'imap', 'https', 'shttp', 'rtsp', 'rtspu', 'sip', 'sips', 'mms', 'sftp', 'tel'] # These are not actually used anymore, but should stay for backwards # compatibility. (They are undocumented, but have a public-looking name.) non_hierarchical = ['gopher', 'hdl', 'mailto', 'news', 'telnet', 'wais', 'imap', 'snews', 'sip', 'sips'] uses_query = ['', 'http', 'wais', 'imap', 'https', 'shttp', 'mms', 'gopher', 'rtsp', 'rtspu', 'sip', 'sips'] uses_fragment = ['', 'ftp', 'hdl', 'http', 'gopher', 'news', 'nntp', 'wais', 'https', 'shttp', 'snews', 'file', 'prospero'] # Characters valid in scheme names scheme_chars = ('abcdefghijklmnopqrstuvwxyz' 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' '0123456789' '+-.') # Leading and trailing C0 control and space to be stripped per WHATWG spec. # == "".join([chr(i) for i in range(0, 0x20 + 1)]) _WHATWG_C0_CONTROL_OR_SPACE = '\x00\x01\x02\x03\x04\x05\x06\x07\x08\t\n\x0b\x0c\r\x0e\x0f\x10\x11\x12\x13\x14\x15\x16\x17\x18\x19\x1a\x1b\x1c\x1d\x1e\x1f ' # Unsafe bytes to be removed per WHATWG spec _UNSAFE_URL_BYTES_TO_REMOVE = ['\t', '\r', '\n'] # XXX: Consider replacing with functools.lru_cache MAX_CACHE_SIZE = 20 _parse_cache = {} def clear_cache(): """Clear the parse cache and the quoters cache.""" _parse_cache.clear() _safe_quoters.clear() # Helpers for bytes handling # For 3.2, we deliberately require applications that # handle improperly quoted URLs to do their own # decoding and encoding. If valid use cases are # presented, we may relax this by using latin-1 # decoding internally for 3.3 _implicit_encoding = 'ascii' _implicit_errors = 'strict' def _noop(obj): return obj def _encode_result(obj, encoding=_implicit_encoding, errors=_implicit_errors): return obj.encode(encoding, errors) def _decode_args(args, encoding=_implicit_encoding, errors=_implicit_errors): return tuple(x.decode(encoding, errors) if x else '' for x in args) def _coerce_args(*args): # Invokes decode if necessary to create str args # and returns the coerced inputs along with # an appropriate result coercion function # - noop for str inputs # - encoding function otherwise str_input = isinstance(args[0], str) for arg in args[1:]: # We special-case the empty string to support the # "scheme=''" default argument to some functions if arg and isinstance(arg, str) != str_input: raise TypeError("Cannot mix str and non-str arguments") if str_input: return args + (_noop,) return _decode_args(args) + (_encode_result,) # Result objects are more helpful than simple tuples class _ResultMixinStr(object): """Standard approach to encoding parsed results from str to bytes""" __slots__ = () def encode(self, encoding='ascii', errors='strict'): return self._encoded_counterpart(*(x.encode(encoding, errors) for x in self)) class _ResultMixinBytes(object): """Standard approach to decoding parsed results from bytes to str""" __slots__ = () def decode(self, encoding='ascii', errors='strict'): return self._decoded_counterpart(*(x.decode(encoding, errors) for x in self)) class _NetlocResultMixinBase(object): """Shared methods for the parsed result objects containing a netloc element""" __slots__ = () @property def username(self): return self._userinfo[0] @property def password(self): return self._userinfo[1] @property def hostname(self): hostname = self._hostinfo[0] if not hostname: return None # Scoped IPv6 address may have zone info, which must not be lowercased # like http://[fe80::822a:a8ff:fe49:470c%tESt]:1234/keys separator = '%' if isinstance(hostname, str) else b'%' hostname, percent, zone = hostname.partition(separator) return hostname.lower() + percent + zone @property def port(self): port = self._hostinfo[1] if port is not None: port = int(port, 10) if not ( 0 <= port <= 65535): raise ValueError("Port out of range 0-65535") return port class _NetlocResultMixinStr(_NetlocResultMixinBase, _ResultMixinStr): __slots__ = () @property def _userinfo(self): netloc = self.netloc userinfo, have_info, hostinfo = netloc.rpartition('@') if have_info: username, have_password, password = userinfo.partition(':') if not have_password: password = None else: username = password = None return username, password @property def _hostinfo(self): netloc = self.netloc _, _, hostinfo = netloc.rpartition('@') _, have_open_br, bracketed = hostinfo.partition('[') if have_open_br: hostname, _, port = bracketed.partition(']') _, _, port = port.partition(':') else: hostname, _, port = hostinfo.partition(':') if not port: port = None return hostname, port class _NetlocResultMixinBytes(_NetlocResultMixinBase, _ResultMixinBytes): __slots__ = () @property def _userinfo(self): netloc = self.netloc userinfo, have_info, hostinfo = netloc.rpartition(b'@') if have_info: username, have_password, password = userinfo.partition(b':') if not have_password: password = None else: username = password = None return username, password @property def _hostinfo(self): netloc = self.netloc _, _, hostinfo = netloc.rpartition(b'@') _, have_open_br, bracketed = hostinfo.partition(b'[') if have_open_br: hostname, _, port = bracketed.partition(b']') _, _, port = port.partition(b':') else: hostname, _, port = hostinfo.partition(b':') if not port: port = None return hostname, port from collections import namedtuple _DefragResultBase = namedtuple('DefragResult', 'url fragment') _SplitResultBase = namedtuple( 'SplitResult', 'scheme netloc path query fragment') _ParseResultBase = namedtuple( 'ParseResult', 'scheme netloc path params query fragment') _DefragResultBase.__doc__ = """ DefragResult(url, fragment) A 2-tuple that contains the url without fragment identifier and the fragment identifier as a separate argument. """ _DefragResultBase.url.__doc__ = """The URL with no fragment identifier.""" _DefragResultBase.fragment.__doc__ = """ Fragment identifier separated from URL, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. """ _SplitResultBase.__doc__ = """ SplitResult(scheme, netloc, path, query, fragment) A 5-tuple that contains the different components of a URL. Similar to ParseResult, but does not split params. """ _SplitResultBase.scheme.__doc__ = """Specifies URL scheme for the request.""" _SplitResultBase.netloc.__doc__ = """ Network location where the request is made to. """ _SplitResultBase.path.__doc__ = """ The hierarchical path, such as the path to a file to download. """ _SplitResultBase.query.__doc__ = """ The query component, that contains non-hierarchical data, that along with data in path component, identifies a resource in the scope of URI's scheme and network location. """ _SplitResultBase.fragment.__doc__ = """ Fragment identifier, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. """ _ParseResultBase.__doc__ = """ ParseResult(scheme, netloc, path, params, query, fragment) A 6-tuple that contains components of a parsed URL. """ _ParseResultBase.scheme.__doc__ = _SplitResultBase.scheme.__doc__ _ParseResultBase.netloc.__doc__ = _SplitResultBase.netloc.__doc__ _ParseResultBase.path.__doc__ = _SplitResultBase.path.__doc__ _ParseResultBase.params.__doc__ = """ Parameters for last path element used to dereference the URI in order to provide access to perform some operation on the resource. """ _ParseResultBase.query.__doc__ = _SplitResultBase.query.__doc__ _ParseResultBase.fragment.__doc__ = _SplitResultBase.fragment.__doc__ # For backwards compatibility, alias _NetlocResultMixinStr # ResultBase is no longer part of the documented API, but it is # retained since deprecating it isn't worth the hassle ResultBase = _NetlocResultMixinStr # Structured result objects for string data class DefragResult(_DefragResultBase, _ResultMixinStr): __slots__ = () def geturl(self): if self.fragment: return self.url + '#' + self.fragment else: return self.url class SplitResult(_SplitResultBase, _NetlocResultMixinStr): __slots__ = () def geturl(self): return urlunsplit(self) class ParseResult(_ParseResultBase, _NetlocResultMixinStr): __slots__ = () def geturl(self): return urlunparse(self) # Structured result objects for bytes data class DefragResultBytes(_DefragResultBase, _ResultMixinBytes): __slots__ = () def geturl(self): if self.fragment: return self.url + b'#' + self.fragment else: return self.url class SplitResultBytes(_SplitResultBase, _NetlocResultMixinBytes): __slots__ = () def geturl(self): return urlunsplit(self) class ParseResultBytes(_ParseResultBase, _NetlocResultMixinBytes): __slots__ = () def geturl(self): return urlunparse(self) # Set up the encode/decode result pairs def _fix_result_transcoding(): _result_pairs = ( (DefragResult, DefragResultBytes), (SplitResult, SplitResultBytes), (ParseResult, ParseResultBytes), ) for _decoded, _encoded in _result_pairs: _decoded._encoded_counterpart = _encoded _encoded._decoded_counterpart = _decoded _fix_result_transcoding() del _fix_result_transcoding def urlparse(url, scheme='', allow_fragments=True): """Parse a URL into 6 components: <scheme>://<netloc>/<path>;<params>?<query>#<fragment> Return a 6-tuple: (scheme, netloc, path, params, query, fragment). Note that we don't break the components up in smaller bits (e.g. netloc is a single string) and we don't expand % escapes.""" url, scheme, _coerce_result = _coerce_args(url, scheme) splitresult = urlsplit(url, scheme, allow_fragments) scheme, netloc, url, query, fragment = splitresult if scheme in uses_params and ';' in url: url, params = _splitparams(url) else: params = '' result = ParseResult(scheme, netloc, url, params, query, fragment) return _coerce_result(result) def _splitparams(url): if '/' in url: i = url.find(';', url.rfind('/')) if i < 0: return url, '' else: i = url.find(';') return url[:i], url[i+1:] def _splitnetloc(url, start=0): delim = len(url) # position of end of domain part of url, default is end for c in '/?#': # look for delimiters; the order is NOT important wdelim = url.find(c, start) # find first of this delim if wdelim >= 0: # if found delim = min(delim, wdelim) # use earliest delim position return url[start:delim], url[delim:] # return (domain, rest) def _checknetloc(netloc): if not netloc or not any(ord(c) > 127 for c in netloc): return # looking for characters like \u2100 that expand to 'a/c' # IDNA uses NFKC equivalence, so normalize for this check import unicodedata n = netloc.replace('@', '') # ignore characters already included n = n.replace(':', '') # but not the surrounding text n = n.replace('#', '') n = n.replace('?', '') netloc2 = unicodedata.normalize('NFKC', n) if n == netloc2: return for c in '/?#@:': if c in netloc2: raise ValueError("netloc '" + netloc + "' contains invalid " + "characters under NFKC normalization") def _remove_unsafe_bytes_from_url(url): for b in _UNSAFE_URL_BYTES_TO_REMOVE: url = url.replace(b, "") return url def _check_bracketed_netloc(netloc): # Note that this function must mirror the splitting # done in NetlocResultMixins._hostinfo(). hostname_and_port = netloc.rpartition('@')[2] before_bracket, have_open_br, bracketed = hostname_and_port.partition('[') if have_open_br: # No data is allowed before a bracket. if before_bracket: raise ValueError("Invalid IPv6 URL") hostname, _, port = bracketed.partition(']') # No data is allowed after the bracket but before the port delimiter. if port and not port.startswith(":"): raise ValueError("Invalid IPv6 URL") else: hostname, _, port = hostname_and_port.partition(':') _check_bracketed_host(hostname) # Valid bracketed hosts are defined in # https://www.rfc-editor.org/rfc/rfc3986#page-49 and https://url.spec.whatwg.org/ def _check_bracketed_host(hostname): if hostname.startswith('v'): if not re.match(r"\Av[a-fA-F0-9]+\..+\Z", hostname): raise ValueError(f"IPvFuture address is invalid") else: ip = ipaddress.ip_address(hostname) # Throws Value Error if not IPv6 or IPv4 if isinstance(ip, ipaddress.IPv4Address): raise ValueError(f"An IPv4 address cannot be in brackets") def urlsplit(url, scheme='', allow_fragments=True): """Parse a URL into 5 components: <scheme>://<netloc>/<path>?<query>#<fragment> Return a 5-tuple: (scheme, netloc, path, query, fragment). Note that we don't break the components up in smaller bits (e.g. netloc is a single string) and we don't expand % escapes.""" url, scheme, _coerce_result = _coerce_args(url, scheme) url = _remove_unsafe_bytes_from_url(url) scheme = _remove_unsafe_bytes_from_url(scheme) # Only lstrip url as some applications rely on preserving trailing space. # (https://url.spec.whatwg.org/#concept-basic-url-parser would strip both) url = url.lstrip(_WHATWG_C0_CONTROL_OR_SPACE) scheme = scheme.strip(_WHATWG_C0_CONTROL_OR_SPACE) allow_fragments = bool(allow_fragments) key = url, scheme, allow_fragments, type(url), type(scheme) cached = _parse_cache.get(key, None) if cached: return _coerce_result(cached) if len(_parse_cache) >= MAX_CACHE_SIZE: # avoid runaway growth clear_cache() netloc = query = fragment = '' i = url.find(':') if i > 0: if url[:i] == 'http': # optimize the common case scheme = url[:i].lower() url = url[i+1:] if url[:2] == '//': netloc, url = _splitnetloc(url, 2) if (('[' in netloc and ']' not in netloc) or (']' in netloc and '[' not in netloc)): raise ValueError("Invalid IPv6 URL") if allow_fragments and '#' in url: url, fragment = url.split('#', 1) if '?' in url: url, query = url.split('?', 1) _checknetloc(netloc) v = SplitResult(scheme, netloc, url, query, fragment) _parse_cache[key] = v return _coerce_result(v) for c in url[:i]: if c not in scheme_chars: break else: # make sure "url" is not actually a port number (in which case # "scheme" is really part of the path) rest = url[i+1:] if not rest or any(c not in '0123456789' for c in rest): # not a port number scheme, url = url[:i].lower(), rest if url[:2] == '//': netloc, url = _splitnetloc(url, 2) if (('[' in netloc and ']' not in netloc) or (']' in netloc and '[' not in netloc)): raise ValueError("Invalid IPv6 URL") if '[' in netloc and ']' in netloc: _check_bracketed_netloc(netloc) if allow_fragments and '#' in url: url, fragment = url.split('#', 1) if '?' in url: url, query = url.split('?', 1) _checknetloc(netloc) v = SplitResult(scheme, netloc, url, query, fragment) _parse_cache[key] = v return _coerce_result(v) def urlunparse(components): """Put a parsed URL back together again. This may result in a slightly different, but equivalent URL, if the URL that was parsed originally had redundant delimiters, e.g. a ? with an empty query (the draft states that these are equivalent).""" scheme, netloc, url, params, query, fragment, _coerce_result = ( _coerce_args(*components)) if params: url = "%s;%s" % (url, params) return _coerce_result(urlunsplit((scheme, netloc, url, query, fragment))) def urlunsplit(components): """Combine the elements of a tuple as returned by urlsplit() into a complete URL as a string. The data argument can be any five-item iterable. This may result in a slightly different, but equivalent URL, if the URL that was parsed originally had unnecessary delimiters (for example, a ? with an empty query; the RFC states that these are equivalent).""" scheme, netloc, url, query, fragment, _coerce_result = ( _coerce_args(*components)) if netloc or (scheme and scheme in uses_netloc and url[:2] != '//'): if url and url[:1] != '/': url = '/' + url url = '//' + (netloc or '') + url if scheme: url = scheme + ':' + url if query: url = url + '?' + query if fragment: url = url + '#' + fragment return _coerce_result(url) def urljoin(base, url, allow_fragments=True): """Join a base URL and a possibly relative URL to form an absolute interpretation of the latter.""" if not base: return url if not url: return base base, url, _coerce_result = _coerce_args(base, url) bscheme, bnetloc, bpath, bparams, bquery, bfragment = \ urlparse(base, '', allow_fragments) scheme, netloc, path, params, query, fragment = \ urlparse(url, bscheme, allow_fragments) if scheme != bscheme or scheme not in uses_relative: return _coerce_result(url) if scheme in uses_netloc: if netloc: return _coerce_result(urlunparse((scheme, netloc, path, params, query, fragment))) netloc = bnetloc if not path and not params: path = bpath params = bparams if not query: query = bquery return _coerce_result(urlunparse((scheme, netloc, path, params, query, fragment))) base_parts = bpath.split('/') if base_parts[-1] != '': # the last item is not a directory, so will not be taken into account # in resolving the relative path del base_parts[-1] # for rfc3986, ignore all base path should the first character be root. if path[:1] == '/': segments = path.split('/') else: segments = base_parts + path.split('/') # filter out elements that would cause redundant slashes on re-joining # the resolved_path segments[1:-1] = filter(None, segments[1:-1]) resolved_path = [] for seg in segments: if seg == '..': try: resolved_path.pop() except IndexError: # ignore any .. segments that would otherwise cause an IndexError # when popped from resolved_path if resolving for rfc3986 pass elif seg == '.': continue else: resolved_path.append(seg) if segments[-1] in ('.', '..'): # do some post-processing here. if the last segment was a relative dir, # then we need to append the trailing '/' resolved_path.append('') return _coerce_result(urlunparse((scheme, netloc, '/'.join( resolved_path) or '/', params, query, fragment))) def urldefrag(url): """Removes any existing fragment from URL. Returns a tuple of the defragmented URL and the fragment. If the URL contained no fragments, the second element is the empty string. """ url, _coerce_result = _coerce_args(url) if '#' in url: s, n, p, a, q, frag = urlparse(url) defrag = urlunparse((s, n, p, a, q, '')) else: frag = '' defrag = url return _coerce_result(DefragResult(defrag, frag)) _hexdig = '0123456789ABCDEFabcdef' _hextobyte = None def unquote_to_bytes(string): """unquote_to_bytes('abc%20def') -> b'abc def'.""" # Note: strings are encoded as UTF-8. This is only an issue if it contains # unescaped non-ASCII characters, which URIs should not. if not string: # Is it a string-like object? string.split return b'' if isinstance(string, str): string = string.encode('utf-8') bits = string.split(b'%') if len(bits) == 1: return string res = [bits[0]] append = res.append # Delay the initialization of the table to not waste memory # if the function is never called global _hextobyte if _hextobyte is None: _hextobyte = {(a + b).encode(): bytes([int(a + b, 16)]) for a in _hexdig for b in _hexdig} for item in bits[1:]: try: append(_hextobyte[item[:2]]) append(item[2:]) except KeyError: append(b'%') append(item) return b''.join(res) _asciire = re.compile('([\x00-\x7f]+)') def unquote(string, encoding='utf-8', errors='replace'): """Replace %xx escapes by their single-character equivalent. The optional encoding and errors parameters specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. By default, percent-encoded sequences are decoded with UTF-8, and invalid sequences are replaced by a placeholder character. unquote('abc%20def') -> 'abc def'. """ if '%' not in string: string.split return string if encoding is None: encoding = 'utf-8' if errors is None: errors = 'replace' bits = _asciire.split(string) res = [bits[0]] append = res.append for i in range(1, len(bits), 2): append(unquote_to_bytes(bits[i]).decode(encoding, errors)) append(bits[i + 1]) return ''.join(res) def parse_qs(qs, keep_blank_values=False, strict_parsing=False, encoding='utf-8', errors='replace', max_num_fields=None, separator=None): """Parse a query given as a string argument. Arguments: qs: percent-encoded query string to be parsed keep_blank_values: flag indicating whether blank values in percent-encoded queries should be treated as blank strings. A true value indicates that blanks should be retained as blank strings. The default false value indicates that blank values are to be ignored and treated as if they were not included. strict_parsing: flag indicating what to do with parsing errors. If false (the default), errors are silently ignored. If true, errors raise a ValueError exception. encoding and errors: specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. max_num_fields: int. If set, then throws a ValueError if there are more than n fields read by parse_qsl(). Returns a dictionary. """ parsed_result = {} pairs = parse_qsl(qs, keep_blank_values, strict_parsing, encoding=encoding, errors=errors, max_num_fields=max_num_fields, separator=separator) for name, value in pairs: if name in parsed_result: parsed_result[name].append(value) else: parsed_result[name] = [value] return parsed_result class _QueryStringSeparatorWarning(RuntimeWarning): """Warning for using default `separator` in parse_qs or parse_qsl""" # The default "separator" for parse_qsl can be specified in a config file. # It's cached after first read. _QS_SEPARATOR_CONFIG_FILENAME = '/etc/python/urllib.cfg' _default_qs_separator = None def parse_qsl(qs, keep_blank_values=False, strict_parsing=False, encoding='utf-8', errors='replace', max_num_fields=None, separator=None): """Parse a query given as a string argument. Arguments: qs: percent-encoded query string to be parsed keep_blank_values: flag indicating whether blank values in percent-encoded queries should be treated as blank strings. A true value indicates that blanks should be retained as blank strings. The default false value indicates that blank values are to be ignored and treated as if they were not included. strict_parsing: flag indicating what to do with parsing errors. If false (the default), errors are silently ignored. If true, errors raise a ValueError exception. encoding and errors: specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. max_num_fields: int. If set, then throws a ValueError if there are more than n fields read by parse_qsl(). Returns a list, as G-d intended. """ qs, _coerce_result = _coerce_args(qs) if isinstance(separator, bytes): separator = separator.decode('ascii') if (not separator or (not isinstance(separator, (str, bytes)))) and separator is not None: raise ValueError("Separator must be of type string or bytes.") # Used when both "&" and ";" act as separators. (Need a non-string value.) _legacy = object() if separator is None: global _default_qs_separator separator = _default_qs_separator envvar_name = 'PYTHON_URLLIB_QS_SEPARATOR' if separator is None: # Set default separator from environment variable separator = os.environ.get(envvar_name) config_source = 'environment variable' if separator is None: # Set default separator from the configuration file try: file = open(_QS_SEPARATOR_CONFIG_FILENAME) except FileNotFoundError: pass else: with file: import configparser config = configparser.ConfigParser( interpolation=None, comment_prefixes=('#', ), ) config.read_file(file) separator = config.get('parse_qs', envvar_name, fallback=None) _default_qs_separator = separator config_source = _QS_SEPARATOR_CONFIG_FILENAME if separator is None: # The default is '&', but warn if not specified explicitly if ';' in qs: from warnings import warn warn("The default separator of urllib.parse.parse_qsl and " + "parse_qs was changed to '&' to avoid a web cache " + "poisoning issue (CVE-2021-23336). " + "By default, semicolons no longer act as query field " + "separators. " + "See https://access.redhat.com/articles/5860431 for " + "more details.", _QueryStringSeparatorWarning, stacklevel=2) separator = '&' elif separator == 'legacy': separator = _legacy elif len(separator) != 1: raise ValueError( f'{envvar_name} (from {config_source}) must contain ' + '1 character, or "legacy". See ' + 'https://access.redhat.com/articles/5860431 for more details.' ) # If max_num_fields is defined then check that the number of fields # is less than max_num_fields. This prevents a memory exhaustion DOS # attack via post bodies with many fields. if max_num_fields is not None: if separator is _legacy: num_fields = 1 + qs.count('&') + qs.count(';') else: num_fields = 1 + qs.count(separator) if max_num_fields < num_fields: raise ValueError('Max number of fields exceeded') if separator is _legacy: pairs = [s2 for s1 in qs.split('&') for s2 in s1.split(';')] else: pairs = [s1 for s1 in qs.split(separator)] r = [] for name_value in pairs: if not name_value and not strict_parsing: continue nv = name_value.split('=', 1) if len(nv) != 2: if strict_parsing: raise ValueError("bad query field: %r" % (name_value,)) # Handle case of a control-name with no equal sign if keep_blank_values: nv.append('') else: continue if len(nv[1]) or keep_blank_values: name = nv[0].replace('+', ' ') name = unquote(name, encoding=encoding, errors=errors) name = _coerce_result(name) value = nv[1].replace('+', ' ') value = unquote(value, encoding=encoding, errors=errors) value = _coerce_result(value) r.append((name, value)) return r def unquote_plus(string, encoding='utf-8', errors='replace'): """Like unquote(), but also replace plus signs by spaces, as required for unquoting HTML form values. unquote_plus('%7e/abc+def') -> '~/abc def' """ string = string.replace('+', ' ') return unquote(string, encoding, errors) _ALWAYS_SAFE = frozenset(b'ABCDEFGHIJKLMNOPQRSTUVWXYZ' b'abcdefghijklmnopqrstuvwxyz' b'0123456789' b'_.-') _ALWAYS_SAFE_BYTES = bytes(_ALWAYS_SAFE) _safe_quoters = {} class Quoter(collections.defaultdict): """A mapping from bytes (in range(0,256)) to strings. String values are percent-encoded byte values, unless the key < 128, and in the "safe" set (either the specified safe set, or default set). """ # Keeps a cache internally, using defaultdict, for efficiency (lookups # of cached keys don't call Python code at all). def __init__(self, safe): """safe: bytes object.""" self.safe = _ALWAYS_SAFE.union(safe) def __repr__(self): # Without this, will just display as a defaultdict return "<%s %r>" % (self.__class__.__name__, dict(self)) def __missing__(self, b): # Handle a cache miss. Store quoted string in cache and return. res = chr(b) if b in self.safe else '%{:02X}'.format(b) self[b] = res return res def quote(string, safe='/', encoding=None, errors=None): """quote('abc def') -> 'abc%20def' Each part of a URL, e.g. the path info, the query, etc., has a different set of reserved characters that must be quoted. RFC 2396 Uniform Resource Identifiers (URI): Generic Syntax lists the following reserved characters. reserved = ";" | "/" | "?" | ":" | "@" | "&" | "=" | "+" | "$" | "," Each of these characters is reserved in some component of a URL, but not necessarily in all of them. By default, the quote function is intended for quoting the path section of a URL. Thus, it will not encode '/'. This character is reserved, but in typical usage the quote function is being called on a path where the existing slash characters are used as reserved characters. string and safe may be either str or bytes objects. encoding and errors must not be specified if string is a bytes object. The optional encoding and errors parameters specify how to deal with non-ASCII characters, as accepted by the str.encode method. By default, encoding='utf-8' (characters are encoded with UTF-8), and errors='strict' (unsupported characters raise a UnicodeEncodeError). """ if isinstance(string, str): if not string: return string if encoding is None: encoding = 'utf-8' if errors is None: errors = 'strict' string = string.encode(encoding, errors) else: if encoding is not None: raise TypeError("quote() doesn't support 'encoding' for bytes") if errors is not None: raise TypeError("quote() doesn't support 'errors' for bytes") return quote_from_bytes(string, safe) def quote_plus(string, safe='', encoding=None, errors=None): """Like quote(), but also replace ' ' with '+', as required for quoting HTML form values. Plus signs in the original string are escaped unless they are included in safe. It also does not have safe default to '/'. """ # Check if ' ' in string, where string may either be a str or bytes. If # there are no spaces, the regular quote will produce the right answer. if ((isinstance(string, str) and ' ' not in string) or (isinstance(string, bytes) and b' ' not in string)): return quote(string, safe, encoding, errors) if isinstance(safe, str): space = ' ' else: space = b' ' string = quote(string, safe + space, encoding, errors) return string.replace(' ', '+') def quote_from_bytes(bs, safe='/'): """Like quote(), but accepts a bytes object rather than a str, and does not perform string-to-bytes encoding. It always returns an ASCII string. quote_from_bytes(b'abc def\x3f') -> 'abc%20def%3f' """ if not isinstance(bs, (bytes, bytearray)): raise TypeError("quote_from_bytes() expected bytes") if not bs: return '' if isinstance(safe, str): # Normalize 'safe' by converting to bytes and removing non-ASCII chars safe = safe.encode('ascii', 'ignore') else: safe = bytes([c for c in safe if c < 128]) if not bs.rstrip(_ALWAYS_SAFE_BYTES + safe): return bs.decode() try: quoter = _safe_quoters[safe] except KeyError: _safe_quoters[safe] = quoter = Quoter(safe).__getitem__ return ''.join([quoter(char) for char in bs]) def urlencode(query, doseq=False, safe='', encoding=None, errors=None, quote_via=quote_plus): """Encode a dict or sequence of two-element tuples into a URL query string. If any values in the query arg are sequences and doseq is true, each sequence element is converted to a separate parameter. If the query arg is a sequence of two-element tuples, the order of the parameters in the output will match the order of parameters in the input. The components of a query arg may each be either a string or a bytes type. The safe, encoding, and errors parameters are passed down to the function specified by quote_via (encoding and errors only if a component is a str). """ if hasattr(query, "items"): query = query.items() else: # It's a bother at times that strings and string-like objects are # sequences. try: # non-sequence items should not work with len() # non-empty strings will fail this if len(query) and not isinstance(query[0], tuple): raise TypeError # Zero-length sequences of all types will get here and succeed, # but that's a minor nit. Since the original implementation # allowed empty dicts that type of behavior probably should be # preserved for consistency except TypeError: ty, va, tb = sys.exc_info() raise TypeError("not a valid non-string sequence " "or mapping object").with_traceback(tb) l = [] if not doseq: for k, v in query: if isinstance(k, bytes): k = quote_via(k, safe) else: k = quote_via(str(k), safe, encoding, errors) if isinstance(v, bytes): v = quote_via(v, safe) else: v = quote_via(str(v), safe, encoding, errors) l.append(k + '=' + v) else: for k, v in query: if isinstance(k, bytes): k = quote_via(k, safe) else: k = quote_via(str(k), safe, encoding, errors) if isinstance(v, bytes): v = quote_via(v, safe) l.append(k + '=' + v) elif isinstance(v, str): v = quote_via(v, safe, encoding, errors) l.append(k + '=' + v) else: try: # Is this a sufficient test for sequence-ness? x = len(v) except TypeError: # not a sequence v = quote_via(str(v), safe, encoding, errors) l.append(k + '=' + v) else: # loop over the sequence for elt in v: if isinstance(elt, bytes): elt = quote_via(elt, safe) else: elt = quote_via(str(elt), safe, encoding, errors) l.append(k + '=' + elt) return '&'.join(l) def to_bytes(url): """to_bytes(u"URL") --> 'URL'.""" # Most URL schemes require ASCII. If that changes, the conversion # can be relaxed. # XXX get rid of to_bytes() if isinstance(url, str): try: url = url.encode("ASCII").decode() except UnicodeError: raise UnicodeError("URL " + repr(url) + " contains non-ASCII characters") return url def unwrap(url): """unwrap('<URL:type://host/path>') --> 'type://host/path'.""" url = str(url).strip() if url[:1] == '<' and url[-1:] == '>': url = url[1:-1].strip() if url[:4] == 'URL:': url = url[4:].strip() return url _typeprog = None def splittype(url): """splittype('type:opaquestring') --> 'type', 'opaquestring'.""" global _typeprog if _typeprog is None: _typeprog = re.compile('([^/:]+):(.*)', re.DOTALL) match = _typeprog.match(url) if match: scheme, data = match.groups() return scheme.lower(), data return None, url _hostprog = None def splithost(url): """splithost('//host[:port]/path') --> 'host[:port]', '/path'.""" global _hostprog if _hostprog is None: _hostprog = re.compile('//([^/#?]*)(.*)', re.DOTALL) match = _hostprog.match(url) if match: host_port, path = match.groups() if path and path[0] != '/': path = '/' + path return host_port, path return None, url def splituser(host): """splituser('user[:passwd]@host[:port]') --> 'user[:passwd]', 'host[:port]'.""" user, delim, host = host.rpartition('@') return (user if delim else None), host def splitpasswd(user): """splitpasswd('user:passwd') -> 'user', 'passwd'.""" user, delim, passwd = user.partition(':') return user, (passwd if delim else None) # splittag('/path#tag') --> '/path', 'tag' _portprog = None def splitport(host): """splitport('host:port') --> 'host', 'port'.""" global _portprog if _portprog is None: _portprog = re.compile('(.*):([0-9]*)$', re.DOTALL) match = _portprog.match(host) if match: host, port = match.groups() if port: return host, port return host, None def splitnport(host, defport=-1): """Split host and port, returning numeric port. Return given default port if no ':' found; defaults to -1. Return numerical port if a valid number are found after ':'. Return None if ':' but not a valid number.""" host, delim, port = host.rpartition(':') if not delim: host = port elif port: try: nport = int(port) except ValueError: nport = None return host, nport return host, defport def splitquery(url): """splitquery('/path?query') --> '/path', 'query'.""" path, delim, query = url.rpartition('?') if delim: return path, query return url, None def splittag(url): """splittag('/path#tag') --> '/path', 'tag'.""" path, delim, tag = url.rpartition('#') if delim: return path, tag return url, None def splitattr(url): """splitattr('/path;attr1=value1;attr2=value2;...') -> '/path', ['attr1=value1', 'attr2=value2', ...].""" words = url.split(';') return words[0], words[1:] def splitvalue(attr): """splitvalue('attr=value') --> 'attr', 'value'.""" attr, delim, value = attr.partition('=') return attr, (value if delim else None) PK23�\��~@~�~� request.pynu�[���"""An extensible library for opening URLs using a variety of protocols The simplest way to use this module is to call the urlopen function, which accepts a string containing a URL or a Request object (described below). It opens the URL and returns the results as file-like object; the returned object has some extra methods described below. The OpenerDirector manages a collection of Handler objects that do all the actual work. Each Handler implements a particular protocol or option. The OpenerDirector is a composite object that invokes the Handlers needed to open the requested URL. For example, the HTTPHandler performs HTTP GET and POST requests and deals with non-error returns. The HTTPRedirectHandler automatically deals with HTTP 301, 302, 303 and 307 redirect errors, and the HTTPDigestAuthHandler deals with digest authentication. urlopen(url, data=None) -- Basic usage is the same as original urllib. pass the url and optionally data to post to an HTTP URL, and get a file-like object back. One difference is that you can also pass a Request instance instead of URL. Raises a URLError (subclass of OSError); for HTTP errors, raises an HTTPError, which can also be treated as a valid response. build_opener -- Function that creates a new OpenerDirector instance. Will install the default handlers. Accepts one or more Handlers as arguments, either instances or Handler classes that it will instantiate. If one of the argument is a subclass of the default handler, the argument will be installed instead of the default. install_opener -- Installs a new opener as the default opener. objects of interest: OpenerDirector -- Sets up the User Agent as the Python-urllib client and manages the Handler classes, while dealing with requests and responses. Request -- An object that encapsulates the state of a request. The state can be as simple as the URL. It can also include extra HTTP headers, e.g. a User-Agent. BaseHandler -- internals: BaseHandler and parent _call_chain conventions Example usage: import urllib.request # set up authentication info authinfo = urllib.request.HTTPBasicAuthHandler() authinfo.add_password(realm='PDQ Application', uri='https://mahler:8092/site-updates.py', user='klem', passwd='geheim$parole') proxy_support = urllib.request.ProxyHandler({"http" : "http://ahad-haam:3128"}) # build a new opener that adds authentication and caching FTP handlers opener = urllib.request.build_opener(proxy_support, authinfo, urllib.request.CacheFTPHandler) # install it urllib.request.install_opener(opener) f = urllib.request.urlopen('http://www.python.org/') """ # XXX issues: # If an authentication error handler that tries to perform # authentication for some reason but fails, how should the error be # signalled? The client needs to know the HTTP error code. But if # the handler knows that the problem was, e.g., that it didn't know # that hash algo that requested in the challenge, it would be good to # pass that information along to the client, too. # ftp errors aren't handled cleanly # check digest against correct (i.e. non-apache) implementation # Possible extensions: # complex proxies XXX not sure what exactly was meant by this # abstract factory for opener import base64 import bisect import email import hashlib import http.client import io import os import posixpath import re import socket import string import sys import time import collections import tempfile import contextlib import warnings from urllib.error import URLError, HTTPError, ContentTooShortError from urllib.parse import ( urlparse, urlsplit, urljoin, unwrap, quote, unquote, splittype, splithost, splitport, splituser, splitpasswd, splitattr, splitquery, splitvalue, splittag, to_bytes, unquote_to_bytes, urlunparse) from urllib.response import addinfourl, addclosehook # check for SSL try: import ssl except ImportError: _have_ssl = False else: _have_ssl = True __all__ = [ # Classes 'Request', 'OpenerDirector', 'BaseHandler', 'HTTPDefaultErrorHandler', 'HTTPRedirectHandler', 'HTTPCookieProcessor', 'ProxyHandler', 'HTTPPasswordMgr', 'HTTPPasswordMgrWithDefaultRealm', 'HTTPPasswordMgrWithPriorAuth', 'AbstractBasicAuthHandler', 'HTTPBasicAuthHandler', 'ProxyBasicAuthHandler', 'AbstractDigestAuthHandler', 'HTTPDigestAuthHandler', 'ProxyDigestAuthHandler', 'HTTPHandler', 'FileHandler', 'FTPHandler', 'CacheFTPHandler', 'DataHandler', 'UnknownHandler', 'HTTPErrorProcessor', # Functions 'urlopen', 'install_opener', 'build_opener', 'pathname2url', 'url2pathname', 'getproxies', # Legacy interface 'urlretrieve', 'urlcleanup', 'URLopener', 'FancyURLopener', ] # used in User-Agent header sent __version__ = '%d.%d' % sys.version_info[:2] _opener = None def urlopen(url, data=None, timeout=socket._GLOBAL_DEFAULT_TIMEOUT, *, cafile=None, capath=None, cadefault=False, context=None): '''Open the URL url, which can be either a string or a Request object. *data* must be an object specifying additional data to be sent to the server, or None if no such data is needed. See Request for details. urllib.request module uses HTTP/1.1 and includes a "Connection:close" header in its HTTP requests. The optional *timeout* parameter specifies a timeout in seconds for blocking operations like the connection attempt (if not specified, the global default timeout setting will be used). This only works for HTTP, HTTPS and FTP connections. If *context* is specified, it must be a ssl.SSLContext instance describing the various SSL options. See HTTPSConnection for more details. The optional *cafile* and *capath* parameters specify a set of trusted CA certificates for HTTPS requests. cafile should point to a single file containing a bundle of CA certificates, whereas capath should point to a directory of hashed certificate files. More information can be found in ssl.SSLContext.load_verify_locations(). The *cadefault* parameter is ignored. This function always returns an object which can work as a context manager and has methods such as * geturl() - return the URL of the resource retrieved, commonly used to determine if a redirect was followed * info() - return the meta-information of the page, such as headers, in the form of an email.message_from_string() instance (see Quick Reference to HTTP Headers) * getcode() - return the HTTP status code of the response. Raises URLError on errors. For HTTP and HTTPS URLs, this function returns a http.client.HTTPResponse object slightly modified. In addition to the three new methods above, the msg attribute contains the same information as the reason attribute --- the reason phrase returned by the server --- instead of the response headers as it is specified in the documentation for HTTPResponse. For FTP, file, and data URLs and requests explicitly handled by legacy URLopener and FancyURLopener classes, this function returns a urllib.response.addinfourl object. Note that None may be returned if no handler handles the request (though the default installed global OpenerDirector uses UnknownHandler to ensure this never happens). In addition, if proxy settings are detected (for example, when a *_proxy environment variable like http_proxy is set), ProxyHandler is default installed and makes sure the requests are handled through the proxy. ''' global _opener if cafile or capath or cadefault: import warnings warnings.warn("cafile, capath and cadefault are deprecated, use a " "custom context instead.", DeprecationWarning, 2) if context is not None: raise ValueError( "You can't pass both context and any of cafile, capath, and " "cadefault" ) if not _have_ssl: raise ValueError('SSL support not available') context = ssl.create_default_context(ssl.Purpose.SERVER_AUTH, cafile=cafile, capath=capath) https_handler = HTTPSHandler(context=context) opener = build_opener(https_handler) elif context: https_handler = HTTPSHandler(context=context) opener = build_opener(https_handler) elif _opener is None: _opener = opener = build_opener() else: opener = _opener return opener.open(url, data, timeout) def install_opener(opener): global _opener _opener = opener _url_tempfiles = [] def urlretrieve(url, filename=None, reporthook=None, data=None): """ Retrieve a URL into a temporary location on disk. Requires a URL argument. If a filename is passed, it is used as the temporary file location. The reporthook argument should be a callable that accepts a block number, a read size, and the total file size of the URL target. The data argument should be valid URL encoded data. If a filename is passed and the URL points to a local resource, the result is a copy from local file to new file. Returns a tuple containing the path to the newly created data file as well as the resulting HTTPMessage object. """ url_type, path = splittype(url) with contextlib.closing(urlopen(url, data)) as fp: headers = fp.info() # Just return the local path and the "headers" for file:// # URLs. No sense in performing a copy unless requested. if url_type == "file" and not filename: return os.path.normpath(path), headers # Handle temporary file setup. if filename: tfp = open(filename, 'wb') else: tfp = tempfile.NamedTemporaryFile(delete=False) filename = tfp.name _url_tempfiles.append(filename) with tfp: result = filename, headers bs = 1024*8 size = -1 read = 0 blocknum = 0 if "content-length" in headers: size = int(headers["Content-Length"]) if reporthook: reporthook(blocknum, bs, size) while True: block = fp.read(bs) if not block: break read += len(block) tfp.write(block) blocknum += 1 if reporthook: reporthook(blocknum, bs, size) if size >= 0 and read < size: raise ContentTooShortError( "retrieval incomplete: got only %i out of %i bytes" % (read, size), result) return result def urlcleanup(): """Clean up temporary files from urlretrieve calls.""" for temp_file in _url_tempfiles: try: os.unlink(temp_file) except OSError: pass del _url_tempfiles[:] global _opener if _opener: _opener = None # copied from cookielib.py _cut_port_re = re.compile(r":\d+$", re.ASCII) def request_host(request): """Return request-host, as defined by RFC 2965. Variation from RFC: returned value is lowercased, for convenient comparison. """ url = request.full_url host = urlparse(url)[1] if host == "": host = request.get_header("Host", "") # remove port, if present host = _cut_port_re.sub("", host, 1) return host.lower() class Request: def __init__(self, url, data=None, headers={}, origin_req_host=None, unverifiable=False, method=None): self.full_url = url self.headers = {} self.unredirected_hdrs = {} self._data = None self.data = data self._tunnel_host = None for key, value in headers.items(): self.add_header(key, value) if origin_req_host is None: origin_req_host = request_host(self) self.origin_req_host = origin_req_host self.unverifiable = unverifiable if method: self.method = method @property def full_url(self): if self.fragment: return '{}#{}'.format(self._full_url, self.fragment) return self._full_url @full_url.setter def full_url(self, url): # unwrap('<URL:type://host/path>') --> 'type://host/path' self._full_url = unwrap(url) self._full_url, self.fragment = splittag(self._full_url) self._parse() @full_url.deleter def full_url(self): self._full_url = None self.fragment = None self.selector = '' @property def data(self): return self._data @data.setter def data(self, data): if data != self._data: self._data = data # issue 16464 # if we change data we need to remove content-length header # (cause it's most probably calculated for previous value) if self.has_header("Content-length"): self.remove_header("Content-length") @data.deleter def data(self): self.data = None def _parse(self): self.type, rest = splittype(self._full_url) if self.type is None: raise ValueError("unknown url type: %r" % self.full_url) self.host, self.selector = splithost(rest) if self.host: self.host = unquote(self.host) def get_method(self): """Return a string indicating the HTTP request method.""" default_method = "POST" if self.data is not None else "GET" return getattr(self, 'method', default_method) def get_full_url(self): return self.full_url def set_proxy(self, host, type): if self.type == 'https' and not self._tunnel_host: self._tunnel_host = self.host else: self.type= type self.selector = self.full_url self.host = host def has_proxy(self): return self.selector == self.full_url def add_header(self, key, val): # useful for something like authentication self.headers[key.capitalize()] = val def add_unredirected_header(self, key, val): # will not be added to a redirected request self.unredirected_hdrs[key.capitalize()] = val def has_header(self, header_name): return (header_name in self.headers or header_name in self.unredirected_hdrs) def get_header(self, header_name, default=None): return self.headers.get( header_name, self.unredirected_hdrs.get(header_name, default)) def remove_header(self, header_name): self.headers.pop(header_name, None) self.unredirected_hdrs.pop(header_name, None) def header_items(self): hdrs = self.unredirected_hdrs.copy() hdrs.update(self.headers) return list(hdrs.items()) class OpenerDirector: def __init__(self): client_version = "Python-urllib/%s" % __version__ self.addheaders = [('User-agent', client_version)] # self.handlers is retained only for backward compatibility self.handlers = [] # manage the individual handlers self.handle_open = {} self.handle_error = {} self.process_response = {} self.process_request = {} def add_handler(self, handler): if not hasattr(handler, "add_parent"): raise TypeError("expected BaseHandler instance, got %r" % type(handler)) added = False for meth in dir(handler): if meth in ["redirect_request", "do_open", "proxy_open"]: # oops, coincidental match continue i = meth.find("_") protocol = meth[:i] condition = meth[i+1:] if condition.startswith("error"): j = condition.find("_") + i + 1 kind = meth[j+1:] try: kind = int(kind) except ValueError: pass lookup = self.handle_error.get(protocol, {}) self.handle_error[protocol] = lookup elif condition == "open": kind = protocol lookup = self.handle_open elif condition == "response": kind = protocol lookup = self.process_response elif condition == "request": kind = protocol lookup = self.process_request else: continue handlers = lookup.setdefault(kind, []) if handlers: bisect.insort(handlers, handler) else: handlers.append(handler) added = True if added: bisect.insort(self.handlers, handler) handler.add_parent(self) def close(self): # Only exists for backwards compatibility. pass def _call_chain(self, chain, kind, meth_name, *args): # Handlers raise an exception if no one else should try to handle # the request, or return None if they can't but another handler # could. Otherwise, they return the response. handlers = chain.get(kind, ()) for handler in handlers: func = getattr(handler, meth_name) result = func(*args) if result is not None: return result def open(self, fullurl, data=None, timeout=socket._GLOBAL_DEFAULT_TIMEOUT): # accept a URL or a Request object if isinstance(fullurl, str): req = Request(fullurl, data) else: req = fullurl if data is not None: req.data = data req.timeout = timeout protocol = req.type # pre-process request meth_name = protocol+"_request" for processor in self.process_request.get(protocol, []): meth = getattr(processor, meth_name) req = meth(req) response = self._open(req, data) # post-process response meth_name = protocol+"_response" for processor in self.process_response.get(protocol, []): meth = getattr(processor, meth_name) response = meth(req, response) return response def _open(self, req, data=None): result = self._call_chain(self.handle_open, 'default', 'default_open', req) if result: return result protocol = req.type result = self._call_chain(self.handle_open, protocol, protocol + '_open', req) if result: return result return self._call_chain(self.handle_open, 'unknown', 'unknown_open', req) def error(self, proto, *args): if proto in ('http', 'https'): # XXX http[s] protocols are special-cased dict = self.handle_error['http'] # https is not different than http proto = args[2] # YUCK! meth_name = 'http_error_%s' % proto http_err = 1 orig_args = args else: dict = self.handle_error meth_name = proto + '_error' http_err = 0 args = (dict, proto, meth_name) + args result = self._call_chain(*args) if result: return result if http_err: args = (dict, 'default', 'http_error_default') + orig_args return self._call_chain(*args) # XXX probably also want an abstract factory that knows when it makes # sense to skip a superclass in favor of a subclass and when it might # make sense to include both def build_opener(*handlers): """Create an opener object from a list of handlers. The opener will use several default handlers, including support for HTTP, FTP and when applicable HTTPS. If any of the handlers passed as arguments are subclasses of the default handlers, the default handlers will not be used. """ opener = OpenerDirector() default_classes = [ProxyHandler, UnknownHandler, HTTPHandler, HTTPDefaultErrorHandler, HTTPRedirectHandler, FTPHandler, FileHandler, HTTPErrorProcessor, DataHandler] if hasattr(http.client, "HTTPSConnection"): default_classes.append(HTTPSHandler) skip = set() for klass in default_classes: for check in handlers: if isinstance(check, type): if issubclass(check, klass): skip.add(klass) elif isinstance(check, klass): skip.add(klass) for klass in skip: default_classes.remove(klass) for klass in default_classes: opener.add_handler(klass()) for h in handlers: if isinstance(h, type): h = h() opener.add_handler(h) return opener class BaseHandler: handler_order = 500 def add_parent(self, parent): self.parent = parent def close(self): # Only exists for backwards compatibility pass def __lt__(self, other): if not hasattr(other, "handler_order"): # Try to preserve the old behavior of having custom classes # inserted after default ones (works only for custom user # classes which are not aware of handler_order). return True return self.handler_order < other.handler_order class HTTPErrorProcessor(BaseHandler): """Process HTTP error responses.""" handler_order = 1000 # after all other processing def http_response(self, request, response): code, msg, hdrs = response.code, response.msg, response.info() # According to RFC 2616, "2xx" code indicates that the client's # request was successfully received, understood, and accepted. if not (200 <= code < 300): response = self.parent.error( 'http', request, response, code, msg, hdrs) return response https_response = http_response class HTTPDefaultErrorHandler(BaseHandler): def http_error_default(self, req, fp, code, msg, hdrs): raise HTTPError(req.full_url, code, msg, hdrs, fp) class HTTPRedirectHandler(BaseHandler): # maximum number of redirections to any single URL # this is needed because of the state that cookies introduce max_repeats = 4 # maximum total number of redirections (regardless of URL) before # assuming we're in a loop max_redirections = 10 def redirect_request(self, req, fp, code, msg, headers, newurl): """Return a Request or None in response to a redirect. This is called by the http_error_30x methods when a redirection response is received. If a redirection should take place, return a new Request to allow http_error_30x to perform the redirect. Otherwise, raise HTTPError if no-one else should try to handle this url. Return None if you can't but another Handler might. """ m = req.get_method() if (not (code in (301, 302, 303, 307) and m in ("GET", "HEAD") or code in (301, 302, 303) and m == "POST")): raise HTTPError(req.full_url, code, msg, headers, fp) # Strictly (according to RFC 2616), 301 or 302 in response to # a POST MUST NOT cause a redirection without confirmation # from the user (of urllib.request, in this case). In practice, # essentially all clients do redirect in this case, so we do # the same. # Be conciliant with URIs containing a space. This is mainly # redundant with the more complete encoding done in http_error_302(), # but it is kept for compatibility with other callers. newurl = newurl.replace(' ', '%20') CONTENT_HEADERS = ("content-length", "content-type") newheaders = dict((k, v) for k, v in req.headers.items() if k.lower() not in CONTENT_HEADERS) return Request(newurl, headers=newheaders, origin_req_host=req.origin_req_host, unverifiable=True) # Implementation note: To avoid the server sending us into an # infinite loop, the request object needs to track what URLs we # have already seen. Do this by adding a handler-specific # attribute to the Request object. def http_error_302(self, req, fp, code, msg, headers): # Some servers (incorrectly) return multiple Location headers # (so probably same goes for URI). Use first header. if "location" in headers: newurl = headers["location"] elif "uri" in headers: newurl = headers["uri"] else: return # fix a possible malformed URL urlparts = urlparse(newurl) # For security reasons we don't allow redirection to anything other # than http, https or ftp. if urlparts.scheme not in ('http', 'https', 'ftp', ''): raise HTTPError( newurl, code, "%s - Redirection to url '%s' is not allowed" % (msg, newurl), headers, fp) if not urlparts.path and urlparts.netloc: urlparts = list(urlparts) urlparts[2] = "/" newurl = urlunparse(urlparts) # http.client.parse_headers() decodes as ISO-8859-1. Recover the # original bytes and percent-encode non-ASCII bytes, and any special # characters such as the space. newurl = quote( newurl, encoding="iso-8859-1", safe=string.punctuation) newurl = urljoin(req.full_url, newurl) # XXX Probably want to forget about the state of the current # request, although that might interact poorly with other # handlers that also use handler-specific request attributes new = self.redirect_request(req, fp, code, msg, headers, newurl) if new is None: return # loop detection # .redirect_dict has a key url if url was previously visited. if hasattr(req, 'redirect_dict'): visited = new.redirect_dict = req.redirect_dict if (visited.get(newurl, 0) >= self.max_repeats or len(visited) >= self.max_redirections): raise HTTPError(req.full_url, code, self.inf_msg + msg, headers, fp) else: visited = new.redirect_dict = req.redirect_dict = {} visited[newurl] = visited.get(newurl, 0) + 1 # Don't close the fp until we are sure that we won't use it # with HTTPError. fp.read() fp.close() return self.parent.open(new, timeout=req.timeout) http_error_301 = http_error_303 = http_error_307 = http_error_302 inf_msg = "The HTTP server returned a redirect error that would " \ "lead to an infinite loop.\n" \ "The last 30x error message was:\n" def _parse_proxy(proxy): """Return (scheme, user, password, host/port) given a URL or an authority. If a URL is supplied, it must have an authority (host:port) component. According to RFC 3986, having an authority component means the URL must have two slashes after the scheme. """ scheme, r_scheme = splittype(proxy) if not r_scheme.startswith("/"): # authority scheme = None authority = proxy else: # URL if not r_scheme.startswith("//"): raise ValueError("proxy URL with no authority: %r" % proxy) # We have an authority, so for RFC 3986-compliant URLs (by ss 3. # and 3.3.), path is empty or starts with '/' end = r_scheme.find("/", 2) if end == -1: end = None authority = r_scheme[2:end] userinfo, hostport = splituser(authority) if userinfo is not None: user, password = splitpasswd(userinfo) else: user = password = None return scheme, user, password, hostport class ProxyHandler(BaseHandler): # Proxies must be in front handler_order = 100 def __init__(self, proxies=None): if proxies is None: proxies = getproxies() assert hasattr(proxies, 'keys'), "proxies must be a mapping" self.proxies = proxies for type, url in proxies.items(): setattr(self, '%s_open' % type, lambda r, proxy=url, type=type, meth=self.proxy_open: meth(r, proxy, type)) def proxy_open(self, req, proxy, type): orig_type = req.type proxy_type, user, password, hostport = _parse_proxy(proxy) if proxy_type is None: proxy_type = orig_type if req.host and proxy_bypass(req.host): return None if user and password: user_pass = '%s:%s' % (unquote(user), unquote(password)) creds = base64.b64encode(user_pass.encode()).decode("ascii") req.add_header('Proxy-authorization', 'Basic ' + creds) hostport = unquote(hostport) req.set_proxy(hostport, proxy_type) if orig_type == proxy_type or orig_type == 'https': # let other handlers take care of it return None else: # need to start over, because the other handlers don't # grok the proxy's URL type # e.g. if we have a constructor arg proxies like so: # {'http': 'ftp://proxy.example.com'}, we may end up turning # a request for http://acme.example.com/a into one for # ftp://proxy.example.com/a return self.parent.open(req, timeout=req.timeout) class HTTPPasswordMgr: def __init__(self): self.passwd = {} def add_password(self, realm, uri, user, passwd): # uri could be a single URI or a sequence if isinstance(uri, str): uri = [uri] if realm not in self.passwd: self.passwd[realm] = {} for default_port in True, False: reduced_uri = tuple( [self.reduce_uri(u, default_port) for u in uri]) self.passwd[realm][reduced_uri] = (user, passwd) def find_user_password(self, realm, authuri): domains = self.passwd.get(realm, {}) for default_port in True, False: reduced_authuri = self.reduce_uri(authuri, default_port) for uris, authinfo in domains.items(): for uri in uris: if self.is_suburi(uri, reduced_authuri): return authinfo return None, None def reduce_uri(self, uri, default_port=True): """Accept authority or URI and extract only the authority and path.""" # note HTTP URLs do not have a userinfo component parts = urlsplit(uri) if parts[1]: # URI scheme = parts[0] authority = parts[1] path = parts[2] or '/' else: # host or host:port scheme = None authority = uri path = '/' host, port = splitport(authority) if default_port and port is None and scheme is not None: dport = {"http": 80, "https": 443, }.get(scheme) if dport is not None: authority = "%s:%d" % (host, dport) return authority, path def is_suburi(self, base, test): """Check if test is below base in a URI tree Both args must be URIs in reduced form. """ if base == test: return True if base[0] != test[0]: return False common = posixpath.commonprefix((base[1], test[1])) if len(common) == len(base[1]): return True return False class HTTPPasswordMgrWithDefaultRealm(HTTPPasswordMgr): def find_user_password(self, realm, authuri): user, password = HTTPPasswordMgr.find_user_password(self, realm, authuri) if user is not None: return user, password return HTTPPasswordMgr.find_user_password(self, None, authuri) class HTTPPasswordMgrWithPriorAuth(HTTPPasswordMgrWithDefaultRealm): def __init__(self, *args, **kwargs): self.authenticated = {} super().__init__(*args, **kwargs) def add_password(self, realm, uri, user, passwd, is_authenticated=False): self.update_authenticated(uri, is_authenticated) # Add a default for prior auth requests if realm is not None: super().add_password(None, uri, user, passwd) super().add_password(realm, uri, user, passwd) def update_authenticated(self, uri, is_authenticated=False): # uri could be a single URI or a sequence if isinstance(uri, str): uri = [uri] for default_port in True, False: for u in uri: reduced_uri = self.reduce_uri(u, default_port) self.authenticated[reduced_uri] = is_authenticated def is_authenticated(self, authuri): for default_port in True, False: reduced_authuri = self.reduce_uri(authuri, default_port) for uri in self.authenticated: if self.is_suburi(uri, reduced_authuri): return self.authenticated[uri] class AbstractBasicAuthHandler: # XXX this allows for multiple auth-schemes, but will stupidly pick # the last one with a realm specified. # allow for double- and single-quoted realm values # (single quotes are a violation of the RFC, but appear in the wild) rx = re.compile('(?:^|,)' # start of the string or ',' '[ \t]*' # optional whitespaces '([^ \t,]+)' # scheme like "Basic" '[ \t]+' # mandatory whitespaces # realm=xxx # realm='xxx' # realm="xxx" 'realm=(["\']?)([^"\']*)\\2', re.I) # XXX could pre-emptively send auth info already accepted (RFC 2617, # end of section 2, and section 1.2 immediately after "credentials" # production). def __init__(self, password_mgr=None): if password_mgr is None: password_mgr = HTTPPasswordMgr() self.passwd = password_mgr self.add_password = self.passwd.add_password def _parse_realm(self, header): # parse WWW-Authenticate header: accept multiple challenges per header found_challenge = False for mo in AbstractBasicAuthHandler.rx.finditer(header): scheme, quote, realm = mo.groups() if quote not in ['"', "'"]: warnings.warn("Basic Auth Realm was unquoted", UserWarning, 3) yield (scheme, realm) found_challenge = True if not found_challenge: if header: scheme = header.split()[0] else: scheme = '' yield (scheme, None) def http_error_auth_reqed(self, authreq, host, req, headers): # host may be an authority (without userinfo) or a URL with an # authority headers = headers.get_all(authreq) if not headers: # no header found return unsupported = None for header in headers: for scheme, realm in self._parse_realm(header): if scheme.lower() != 'basic': unsupported = scheme continue if realm is not None: # Use the first matching Basic challenge. # Ignore following challenges even if they use the Basic # scheme. return self.retry_http_basic_auth(host, req, realm) if unsupported is not None: raise ValueError("AbstractBasicAuthHandler does not " "support the following scheme: %r" % (scheme,)) def retry_http_basic_auth(self, host, req, realm): user, pw = self.passwd.find_user_password(realm, host) if pw is not None: raw = "%s:%s" % (user, pw) auth = "Basic " + base64.b64encode(raw.encode()).decode("ascii") if req.get_header(self.auth_header, None) == auth: return None req.add_unredirected_header(self.auth_header, auth) return self.parent.open(req, timeout=req.timeout) else: return None def http_request(self, req): if (not hasattr(self.passwd, 'is_authenticated') or not self.passwd.is_authenticated(req.full_url)): return req if not req.has_header('Authorization'): user, passwd = self.passwd.find_user_password(None, req.full_url) credentials = '{0}:{1}'.format(user, passwd).encode() auth_str = base64.standard_b64encode(credentials).decode() req.add_unredirected_header('Authorization', 'Basic {}'.format(auth_str.strip())) return req def http_response(self, req, response): if hasattr(self.passwd, 'is_authenticated'): if 200 <= response.code < 300: self.passwd.update_authenticated(req.full_url, True) else: self.passwd.update_authenticated(req.full_url, False) return response https_request = http_request https_response = http_response class HTTPBasicAuthHandler(AbstractBasicAuthHandler, BaseHandler): auth_header = 'Authorization' def http_error_401(self, req, fp, code, msg, headers): url = req.full_url response = self.http_error_auth_reqed('www-authenticate', url, req, headers) return response class ProxyBasicAuthHandler(AbstractBasicAuthHandler, BaseHandler): auth_header = 'Proxy-authorization' def http_error_407(self, req, fp, code, msg, headers): # http_error_auth_reqed requires that there is no userinfo component in # authority. Assume there isn't one, since urllib.request does not (and # should not, RFC 3986 s. 3.2.1) support requests for URLs containing # userinfo. authority = req.host response = self.http_error_auth_reqed('proxy-authenticate', authority, req, headers) return response # Return n random bytes. _randombytes = os.urandom class AbstractDigestAuthHandler: # Digest authentication is specified in RFC 2617. # XXX The client does not inspect the Authentication-Info header # in a successful response. # XXX It should be possible to test this implementation against # a mock server that just generates a static set of challenges. # XXX qop="auth-int" supports is shaky def __init__(self, passwd=None): if passwd is None: passwd = HTTPPasswordMgr() self.passwd = passwd self.add_password = self.passwd.add_password self.retried = 0 self.nonce_count = 0 self.last_nonce = None def reset_retry_count(self): self.retried = 0 def http_error_auth_reqed(self, auth_header, host, req, headers): authreq = headers.get(auth_header, None) if self.retried > 5: # Don't fail endlessly - if we failed once, we'll probably # fail a second time. Hm. Unless the Password Manager is # prompting for the information. Crap. This isn't great # but it's better than the current 'repeat until recursion # depth exceeded' approach <wink> raise HTTPError(req.full_url, 401, "digest auth failed", headers, None) else: self.retried += 1 if authreq: scheme = authreq.split()[0] if scheme.lower() == 'digest': return self.retry_http_digest_auth(req, authreq) elif scheme.lower() != 'basic': raise ValueError("AbstractDigestAuthHandler does not support" " the following scheme: '%s'" % scheme) def retry_http_digest_auth(self, req, auth): token, challenge = auth.split(' ', 1) chal = parse_keqv_list(filter(None, parse_http_list(challenge))) auth = self.get_authorization(req, chal) if auth: auth_val = 'Digest %s' % auth if req.headers.get(self.auth_header, None) == auth_val: return None req.add_unredirected_header(self.auth_header, auth_val) resp = self.parent.open(req, timeout=req.timeout) return resp def get_cnonce(self, nonce): # The cnonce-value is an opaque # quoted string value provided by the client and used by both client # and server to avoid chosen plaintext attacks, to provide mutual # authentication, and to provide some message integrity protection. # This isn't a fabulous effort, but it's probably Good Enough. s = "%s:%s:%s:" % (self.nonce_count, nonce, time.ctime()) b = s.encode("ascii") + _randombytes(8) dig = hashlib.sha1(b).hexdigest() return dig[:16] def get_authorization(self, req, chal): try: realm = chal['realm'] nonce = chal['nonce'] qop = chal.get('qop') algorithm = chal.get('algorithm', 'MD5') # mod_digest doesn't send an opaque, even though it isn't # supposed to be optional opaque = chal.get('opaque', None) except KeyError: return None H, KD = self.get_algorithm_impls(algorithm) if H is None: return None user, pw = self.passwd.find_user_password(realm, req.full_url) if user is None: return None # XXX not implemented yet if req.data is not None: entdig = self.get_entity_digest(req.data, chal) else: entdig = None A1 = "%s:%s:%s" % (user, realm, pw) A2 = "%s:%s" % (req.get_method(), # XXX selector: what about proxies and full urls req.selector) if qop == 'auth': if nonce == self.last_nonce: self.nonce_count += 1 else: self.nonce_count = 1 self.last_nonce = nonce ncvalue = '%08x' % self.nonce_count cnonce = self.get_cnonce(nonce) noncebit = "%s:%s:%s:%s:%s" % (nonce, ncvalue, cnonce, qop, H(A2)) respdig = KD(H(A1), noncebit) elif qop is None: respdig = KD(H(A1), "%s:%s" % (nonce, H(A2))) else: # XXX handle auth-int. raise URLError("qop '%s' is not supported." % qop) # XXX should the partial digests be encoded too? base = 'username="%s", realm="%s", nonce="%s", uri="%s", ' \ 'response="%s"' % (user, realm, nonce, req.selector, respdig) if opaque: base += ', opaque="%s"' % opaque if entdig: base += ', digest="%s"' % entdig base += ', algorithm="%s"' % algorithm if qop: base += ', qop=auth, nc=%s, cnonce="%s"' % (ncvalue, cnonce) return base def get_algorithm_impls(self, algorithm): # lambdas assume digest modules are imported at the top level if algorithm == 'MD5': H = lambda x: hashlib.md5(x.encode("ascii")).hexdigest() elif algorithm == 'SHA': H = lambda x: hashlib.sha1(x.encode("ascii")).hexdigest() # XXX MD5-sess else: raise ValueError("Unsupported digest authentication " "algorithm %r" % algorithm) KD = lambda s, d: H("%s:%s" % (s, d)) return H, KD def get_entity_digest(self, data, chal): # XXX not implemented yet return None class HTTPDigestAuthHandler(BaseHandler, AbstractDigestAuthHandler): """An authentication protocol defined by RFC 2069 Digest authentication improves on basic authentication because it does not transmit passwords in the clear. """ auth_header = 'Authorization' handler_order = 490 # before Basic auth def http_error_401(self, req, fp, code, msg, headers): host = urlparse(req.full_url)[1] retry = self.http_error_auth_reqed('www-authenticate', host, req, headers) self.reset_retry_count() return retry class ProxyDigestAuthHandler(BaseHandler, AbstractDigestAuthHandler): auth_header = 'Proxy-Authorization' handler_order = 490 # before Basic auth def http_error_407(self, req, fp, code, msg, headers): host = req.host retry = self.http_error_auth_reqed('proxy-authenticate', host, req, headers) self.reset_retry_count() return retry class AbstractHTTPHandler(BaseHandler): def __init__(self, debuglevel=0): self._debuglevel = debuglevel def set_http_debuglevel(self, level): self._debuglevel = level def _get_content_length(self, request): return http.client.HTTPConnection._get_content_length( request.data, request.get_method()) def do_request_(self, request): host = request.host if not host: raise URLError('no host given') if request.data is not None: # POST data = request.data if isinstance(data, str): msg = "POST data should be bytes, an iterable of bytes, " \ "or a file object. It cannot be of type str." raise TypeError(msg) if not request.has_header('Content-type'): request.add_unredirected_header( 'Content-type', 'application/x-www-form-urlencoded') if (not request.has_header('Content-length') and not request.has_header('Transfer-encoding')): content_length = self._get_content_length(request) if content_length is not None: request.add_unredirected_header( 'Content-length', str(content_length)) else: request.add_unredirected_header( 'Transfer-encoding', 'chunked') sel_host = host if request.has_proxy(): scheme, sel = splittype(request.selector) sel_host, sel_path = splithost(sel) if not request.has_header('Host'): request.add_unredirected_header('Host', sel_host) for name, value in self.parent.addheaders: name = name.capitalize() if not request.has_header(name): request.add_unredirected_header(name, value) return request def do_open(self, http_class, req, **http_conn_args): """Return an HTTPResponse object for the request, using http_class. http_class must implement the HTTPConnection API from http.client. """ host = req.host if not host: raise URLError('no host given') # will parse host:port h = http_class(host, timeout=req.timeout, **http_conn_args) h.set_debuglevel(self._debuglevel) headers = dict(req.unredirected_hdrs) headers.update(dict((k, v) for k, v in req.headers.items() if k not in headers)) # TODO(jhylton): Should this be redesigned to handle # persistent connections? # We want to make an HTTP/1.1 request, but the addinfourl # class isn't prepared to deal with a persistent connection. # It will try to read all remaining data from the socket, # which will block while the server waits for the next request. # So make sure the connection gets closed after the (only) # request. headers["Connection"] = "close" headers = dict((name.title(), val) for name, val in headers.items()) if req._tunnel_host: tunnel_headers = {} proxy_auth_hdr = "Proxy-Authorization" if proxy_auth_hdr in headers: tunnel_headers[proxy_auth_hdr] = headers[proxy_auth_hdr] # Proxy-Authorization should not be sent to origin # server. del headers[proxy_auth_hdr] h.set_tunnel(req._tunnel_host, headers=tunnel_headers) try: try: h.request(req.get_method(), req.selector, req.data, headers, encode_chunked=req.has_header('Transfer-encoding')) except OSError as err: # timeout error raise URLError(err) r = h.getresponse() except: h.close() raise # If the server does not send us a 'Connection: close' header, # HTTPConnection assumes the socket should be left open. Manually # mark the socket to be closed when this response object goes away. if h.sock: h.sock.close() h.sock = None r.url = req.get_full_url() # This line replaces the .msg attribute of the HTTPResponse # with .headers, because urllib clients expect the response to # have the reason in .msg. It would be good to mark this # attribute is deprecated and get then to use info() or # .headers. r.msg = r.reason return r class HTTPHandler(AbstractHTTPHandler): def http_open(self, req): return self.do_open(http.client.HTTPConnection, req) http_request = AbstractHTTPHandler.do_request_ if hasattr(http.client, 'HTTPSConnection'): class HTTPSHandler(AbstractHTTPHandler): def __init__(self, debuglevel=0, context=None, check_hostname=None): AbstractHTTPHandler.__init__(self, debuglevel) self._context = context self._check_hostname = check_hostname def https_open(self, req): return self.do_open(http.client.HTTPSConnection, req, context=self._context, check_hostname=self._check_hostname) https_request = AbstractHTTPHandler.do_request_ __all__.append('HTTPSHandler') class HTTPCookieProcessor(BaseHandler): def __init__(self, cookiejar=None): import http.cookiejar if cookiejar is None: cookiejar = http.cookiejar.CookieJar() self.cookiejar = cookiejar def http_request(self, request): self.cookiejar.add_cookie_header(request) return request def http_response(self, request, response): self.cookiejar.extract_cookies(response, request) return response https_request = http_request https_response = http_response class UnknownHandler(BaseHandler): def unknown_open(self, req): type = req.type raise URLError('unknown url type: %s' % type) def parse_keqv_list(l): """Parse list of key=value strings where keys are not duplicated.""" parsed = {} for elt in l: k, v = elt.split('=', 1) if v[0] == '"' and v[-1] == '"': v = v[1:-1] parsed[k] = v return parsed def parse_http_list(s): """Parse lists as described by RFC 2068 Section 2. In particular, parse comma-separated lists where the elements of the list may include quoted-strings. A quoted-string could contain a comma. A non-quoted string could have quotes in the middle. Neither commas nor quotes count if they are escaped. Only double-quotes count, not single-quotes. """ res = [] part = '' escape = quote = False for cur in s: if escape: part += cur escape = False continue if quote: if cur == '\\': escape = True continue elif cur == '"': quote = False part += cur continue if cur == ',': res.append(part) part = '' continue if cur == '"': quote = True part += cur # append last part if part: res.append(part) return [part.strip() for part in res] class FileHandler(BaseHandler): # Use local file or FTP depending on form of URL def file_open(self, req): url = req.selector if url[:2] == '//' and url[2:3] != '/' and (req.host and req.host != 'localhost'): if not req.host in self.get_names(): raise URLError("file:// scheme is supported only on localhost") else: return self.open_local_file(req) # names for the localhost names = None def get_names(self): if FileHandler.names is None: try: FileHandler.names = tuple( socket.gethostbyname_ex('localhost')[2] + socket.gethostbyname_ex(socket.gethostname())[2]) except socket.gaierror: FileHandler.names = (socket.gethostbyname('localhost'),) return FileHandler.names # not entirely sure what the rules are here def open_local_file(self, req): import email.utils import mimetypes host = req.host filename = req.selector localfile = url2pathname(filename) try: stats = os.stat(localfile) size = stats.st_size modified = email.utils.formatdate(stats.st_mtime, usegmt=True) mtype = mimetypes.guess_type(filename)[0] headers = email.message_from_string( 'Content-type: %s\nContent-length: %d\nLast-modified: %s\n' % (mtype or 'text/plain', size, modified)) if host: host, port = splitport(host) if not host or \ (not port and _safe_gethostbyname(host) in self.get_names()): if host: origurl = 'file://' + host + filename else: origurl = 'file://' + filename return addinfourl(open(localfile, 'rb'), headers, origurl) except OSError as exp: raise URLError(exp) raise URLError('file not on local host') def _safe_gethostbyname(host): try: return socket.gethostbyname(host) except socket.gaierror: return None class FTPHandler(BaseHandler): def ftp_open(self, req): import ftplib import mimetypes host = req.host if not host: raise URLError('ftp error: no host given') host, port = splitport(host) if port is None: port = ftplib.FTP_PORT else: port = int(port) # username/password handling user, host = splituser(host) if user: user, passwd = splitpasswd(user) else: passwd = None host = unquote(host) user = user or '' passwd = passwd or '' try: host = socket.gethostbyname(host) except OSError as msg: raise URLError(msg) path, attrs = splitattr(req.selector) dirs = path.split('/') dirs = list(map(unquote, dirs)) dirs, file = dirs[:-1], dirs[-1] if dirs and not dirs[0]: dirs = dirs[1:] try: fw = self.connect_ftp(user, passwd, host, port, dirs, req.timeout) type = file and 'I' or 'D' for attr in attrs: attr, value = splitvalue(attr) if attr.lower() == 'type' and \ value in ('a', 'A', 'i', 'I', 'd', 'D'): type = value.upper() fp, retrlen = fw.retrfile(file, type) headers = "" mtype = mimetypes.guess_type(req.full_url)[0] if mtype: headers += "Content-type: %s\n" % mtype if retrlen is not None and retrlen >= 0: headers += "Content-length: %d\n" % retrlen headers = email.message_from_string(headers) return addinfourl(fp, headers, req.full_url) except ftplib.all_errors as exp: exc = URLError('ftp error: %r' % exp) raise exc.with_traceback(sys.exc_info()[2]) def connect_ftp(self, user, passwd, host, port, dirs, timeout): return ftpwrapper(user, passwd, host, port, dirs, timeout, persistent=False) class CacheFTPHandler(FTPHandler): # XXX would be nice to have pluggable cache strategies # XXX this stuff is definitely not thread safe def __init__(self): self.cache = {} self.timeout = {} self.soonest = 0 self.delay = 60 self.max_conns = 16 def setTimeout(self, t): self.delay = t def setMaxConns(self, m): self.max_conns = m def connect_ftp(self, user, passwd, host, port, dirs, timeout): key = user, host, port, '/'.join(dirs), timeout if key in self.cache: self.timeout[key] = time.time() + self.delay else: self.cache[key] = ftpwrapper(user, passwd, host, port, dirs, timeout) self.timeout[key] = time.time() + self.delay self.check_cache() return self.cache[key] def check_cache(self): # first check for old ones t = time.time() if self.soonest <= t: for k, v in list(self.timeout.items()): if v < t: self.cache[k].close() del self.cache[k] del self.timeout[k] self.soonest = min(list(self.timeout.values())) # then check the size if len(self.cache) == self.max_conns: for k, v in list(self.timeout.items()): if v == self.soonest: del self.cache[k] del self.timeout[k] break self.soonest = min(list(self.timeout.values())) def clear_cache(self): for conn in self.cache.values(): conn.close() self.cache.clear() self.timeout.clear() class DataHandler(BaseHandler): def data_open(self, req): # data URLs as specified in RFC 2397. # # ignores POSTed data # # syntax: # dataurl := "data:" [ mediatype ] [ ";base64" ] "," data # mediatype := [ type "/" subtype ] *( ";" parameter ) # data := *urlchar # parameter := attribute "=" value url = req.full_url scheme, data = url.split(":",1) mediatype, data = data.split(",",1) # even base64 encoded data URLs might be quoted so unquote in any case: data = unquote_to_bytes(data) if mediatype.endswith(";base64"): data = base64.decodebytes(data) mediatype = mediatype[:-7] if not mediatype: mediatype = "text/plain;charset=US-ASCII" headers = email.message_from_string("Content-type: %s\nContent-length: %d\n" % (mediatype, len(data))) return addinfourl(io.BytesIO(data), headers, url) # Code move from the old urllib module MAXFTPCACHE = 10 # Trim the ftp cache beyond this size # Helper for non-unix systems if os.name == 'nt': from nturl2path import url2pathname, pathname2url else: def url2pathname(pathname): """OS-specific conversion from a relative URL of the 'file' scheme to a file system path; not recommended for general use.""" return unquote(pathname) def pathname2url(pathname): """OS-specific conversion from a file system path to a relative URL of the 'file' scheme; not recommended for general use.""" return quote(pathname) ftpcache = {} class URLopener: """Class to open URLs. This is a class rather than just a subroutine because we may need more than one set of global protocol-specific options. Note -- this is a base class for those who don't want the automatic handling of errors type 302 (relocated) and 401 (authorization needed).""" __tempfiles = None version = "Python-urllib/%s" % __version__ # Constructor def __init__(self, proxies=None, **x509): msg = "%(class)s style of invoking requests is deprecated. " \ "Use newer urlopen functions/methods" % {'class': self.__class__.__name__} warnings.warn(msg, DeprecationWarning, stacklevel=3) if proxies is None: proxies = getproxies() assert hasattr(proxies, 'keys'), "proxies must be a mapping" self.proxies = proxies self.key_file = x509.get('key_file') self.cert_file = x509.get('cert_file') self.addheaders = [('User-Agent', self.version), ('Accept', '*/*')] self.__tempfiles = [] self.__unlink = os.unlink # See cleanup() self.tempcache = None # Undocumented feature: if you assign {} to tempcache, # it is used to cache files retrieved with # self.retrieve(). This is not enabled by default # since it does not work for changing documents (and I # haven't got the logic to check expiration headers # yet). self.ftpcache = ftpcache # Undocumented feature: you can use a different # ftp cache by assigning to the .ftpcache member; # in case you want logically independent URL openers # XXX This is not threadsafe. Bah. def __del__(self): self.close() def close(self): self.cleanup() def cleanup(self): # This code sometimes runs when the rest of this module # has already been deleted, so it can't use any globals # or import anything. if self.__tempfiles: for file in self.__tempfiles: try: self.__unlink(file) except OSError: pass del self.__tempfiles[:] if self.tempcache: self.tempcache.clear() def addheader(self, *args): """Add a header to be used by the HTTP interface only e.g. u.addheader('Accept', 'sound/basic')""" self.addheaders.append(args) # External interface def open(self, fullurl, data=None): """Use URLopener().open(file) instead of open(file, 'r').""" fullurl = unwrap(to_bytes(fullurl)) fullurl = quote(fullurl, safe="%/:=&?~#+!$,;'@()*[]|") if self.tempcache and fullurl in self.tempcache: filename, headers = self.tempcache[fullurl] fp = open(filename, 'rb') return addinfourl(fp, headers, fullurl) urltype, url = splittype(fullurl) if not urltype: urltype = 'file' if urltype in self.proxies: proxy = self.proxies[urltype] urltype, proxyhost = splittype(proxy) host, selector = splithost(proxyhost) url = (host, fullurl) # Signal special case to open_*() else: proxy = None name = 'open_' + urltype self.type = urltype name = name.replace('-', '_') if not hasattr(self, name) or name == 'open_local_file': if proxy: return self.open_unknown_proxy(proxy, fullurl, data) else: return self.open_unknown(fullurl, data) try: if data is None: return getattr(self, name)(url) else: return getattr(self, name)(url, data) except (HTTPError, URLError): raise except OSError as msg: raise OSError('socket error', msg).with_traceback(sys.exc_info()[2]) def open_unknown(self, fullurl, data=None): """Overridable interface to open unknown URL type.""" type, url = splittype(fullurl) raise OSError('url error', 'unknown url type', type) def open_unknown_proxy(self, proxy, fullurl, data=None): """Overridable interface to open unknown URL type.""" type, url = splittype(fullurl) raise OSError('url error', 'invalid proxy for %s' % type, proxy) # External interface def retrieve(self, url, filename=None, reporthook=None, data=None): """retrieve(url) returns (filename, headers) for a local object or (tempfilename, headers) for a remote object.""" url = unwrap(to_bytes(url)) if self.tempcache and url in self.tempcache: return self.tempcache[url] type, url1 = splittype(url) if filename is None and (not type or type == 'file'): try: fp = self.open_local_file(url1) hdrs = fp.info() fp.close() return url2pathname(splithost(url1)[1]), hdrs except OSError as msg: pass fp = self.open(url, data) try: headers = fp.info() if filename: tfp = open(filename, 'wb') else: import tempfile garbage, path = splittype(url) garbage, path = splithost(path or "") path, garbage = splitquery(path or "") path, garbage = splitattr(path or "") suffix = os.path.splitext(path)[1] (fd, filename) = tempfile.mkstemp(suffix) self.__tempfiles.append(filename) tfp = os.fdopen(fd, 'wb') try: result = filename, headers if self.tempcache is not None: self.tempcache[url] = result bs = 1024*8 size = -1 read = 0 blocknum = 0 if "content-length" in headers: size = int(headers["Content-Length"]) if reporthook: reporthook(blocknum, bs, size) while 1: block = fp.read(bs) if not block: break read += len(block) tfp.write(block) blocknum += 1 if reporthook: reporthook(blocknum, bs, size) finally: tfp.close() finally: fp.close() # raise exception if actual size does not match content-length header if size >= 0 and read < size: raise ContentTooShortError( "retrieval incomplete: got only %i out of %i bytes" % (read, size), result) return result # Each method named open_<type> knows how to open that type of URL def _open_generic_http(self, connection_factory, url, data): """Make an HTTP connection using connection_class. This is an internal method that should be called from open_http() or open_https(). Arguments: - connection_factory should take a host name and return an HTTPConnection instance. - url is the url to retrieval or a host, relative-path pair. - data is payload for a POST request or None. """ user_passwd = None proxy_passwd= None if isinstance(url, str): host, selector = splithost(url) if host: user_passwd, host = splituser(host) host = unquote(host) realhost = host else: host, selector = url # check whether the proxy contains authorization information proxy_passwd, host = splituser(host) # now we proceed with the url we want to obtain urltype, rest = splittype(selector) url = rest user_passwd = None if urltype.lower() != 'http': realhost = None else: realhost, rest = splithost(rest) if realhost: user_passwd, realhost = splituser(realhost) if user_passwd: selector = "%s://%s%s" % (urltype, realhost, rest) if proxy_bypass(realhost): host = realhost if not host: raise OSError('http error', 'no host given') if proxy_passwd: proxy_passwd = unquote(proxy_passwd) proxy_auth = base64.b64encode(proxy_passwd.encode()).decode('ascii') else: proxy_auth = None if user_passwd: user_passwd = unquote(user_passwd) auth = base64.b64encode(user_passwd.encode()).decode('ascii') else: auth = None http_conn = connection_factory(host) headers = {} if proxy_auth: headers["Proxy-Authorization"] = "Basic %s" % proxy_auth if auth: headers["Authorization"] = "Basic %s" % auth if realhost: headers["Host"] = realhost # Add Connection:close as we don't support persistent connections yet. # This helps in closing the socket and avoiding ResourceWarning headers["Connection"] = "close" for header, value in self.addheaders: headers[header] = value if data is not None: headers["Content-Type"] = "application/x-www-form-urlencoded" http_conn.request("POST", selector, data, headers) else: http_conn.request("GET", selector, headers=headers) try: response = http_conn.getresponse() except http.client.BadStatusLine: # something went wrong with the HTTP status line raise URLError("http protocol error: bad status line") # According to RFC 2616, "2xx" code indicates that the client's # request was successfully received, understood, and accepted. if 200 <= response.status < 300: return addinfourl(response, response.msg, "http:" + url, response.status) else: return self.http_error( url, response.fp, response.status, response.reason, response.msg, data) def open_http(self, url, data=None): """Use HTTP protocol.""" return self._open_generic_http(http.client.HTTPConnection, url, data) def http_error(self, url, fp, errcode, errmsg, headers, data=None): """Handle http errors. Derived class can override this, or provide specific handlers named http_error_DDD where DDD is the 3-digit error code.""" # First check if there's a specific handler for this error name = 'http_error_%d' % errcode if hasattr(self, name): method = getattr(self, name) if data is None: result = method(url, fp, errcode, errmsg, headers) else: result = method(url, fp, errcode, errmsg, headers, data) if result: return result return self.http_error_default(url, fp, errcode, errmsg, headers) def http_error_default(self, url, fp, errcode, errmsg, headers): """Default error handler: close the connection and raise OSError.""" fp.close() raise HTTPError(url, errcode, errmsg, headers, None) if _have_ssl: def _https_connection(self, host): return http.client.HTTPSConnection(host, key_file=self.key_file, cert_file=self.cert_file) def open_https(self, url, data=None): """Use HTTPS protocol.""" return self._open_generic_http(self._https_connection, url, data) def open_file(self, url): """Use local file or FTP depending on form of URL.""" if not isinstance(url, str): raise URLError('file error: proxy support for file protocol currently not implemented') if url[:2] == '//' and url[2:3] != '/' and url[2:12].lower() != 'localhost/': raise ValueError("file:// scheme is supported only on localhost") else: return self.open_local_file(url) def open_local_file(self, url): """Use local file.""" import email.utils import mimetypes host, file = splithost(url) localname = url2pathname(file) try: stats = os.stat(localname) except OSError as e: raise URLError(e.strerror, e.filename) size = stats.st_size modified = email.utils.formatdate(stats.st_mtime, usegmt=True) mtype = mimetypes.guess_type(url)[0] headers = email.message_from_string( 'Content-Type: %s\nContent-Length: %d\nLast-modified: %s\n' % (mtype or 'text/plain', size, modified)) if not host: urlfile = file if file[:1] == '/': urlfile = 'file://' + file return addinfourl(open(localname, 'rb'), headers, urlfile) host, port = splitport(host) if (not port and socket.gethostbyname(host) in ((localhost(),) + thishost())): urlfile = file if file[:1] == '/': urlfile = 'file://' + file elif file[:2] == './': raise ValueError("local file url may start with / or file:. Unknown url of type: %s" % url) return addinfourl(open(localname, 'rb'), headers, urlfile) raise URLError('local file error: not on local host') def open_ftp(self, url): """Use FTP protocol.""" if not isinstance(url, str): raise URLError('ftp error: proxy support for ftp protocol currently not implemented') import mimetypes host, path = splithost(url) if not host: raise URLError('ftp error: no host given') host, port = splitport(host) user, host = splituser(host) if user: user, passwd = splitpasswd(user) else: passwd = None host = unquote(host) user = unquote(user or '') passwd = unquote(passwd or '') host = socket.gethostbyname(host) if not port: import ftplib port = ftplib.FTP_PORT else: port = int(port) path, attrs = splitattr(path) path = unquote(path) dirs = path.split('/') dirs, file = dirs[:-1], dirs[-1] if dirs and not dirs[0]: dirs = dirs[1:] if dirs and not dirs[0]: dirs[0] = '/' key = user, host, port, '/'.join(dirs) # XXX thread unsafe! if len(self.ftpcache) > MAXFTPCACHE: # Prune the cache, rather arbitrarily for k in list(self.ftpcache): if k != key: v = self.ftpcache[k] del self.ftpcache[k] v.close() try: if key not in self.ftpcache: self.ftpcache[key] = \ ftpwrapper(user, passwd, host, port, dirs) if not file: type = 'D' else: type = 'I' for attr in attrs: attr, value = splitvalue(attr) if attr.lower() == 'type' and \ value in ('a', 'A', 'i', 'I', 'd', 'D'): type = value.upper() (fp, retrlen) = self.ftpcache[key].retrfile(file, type) mtype = mimetypes.guess_type("ftp:" + url)[0] headers = "" if mtype: headers += "Content-Type: %s\n" % mtype if retrlen is not None and retrlen >= 0: headers += "Content-Length: %d\n" % retrlen headers = email.message_from_string(headers) return addinfourl(fp, headers, "ftp:" + url) except ftperrors() as exp: raise URLError('ftp error %r' % exp).with_traceback(sys.exc_info()[2]) def open_data(self, url, data=None): """Use "data" URL.""" if not isinstance(url, str): raise URLError('data error: proxy support for data protocol currently not implemented') # ignore POSTed data # # syntax of data URLs: # dataurl := "data:" [ mediatype ] [ ";base64" ] "," data # mediatype := [ type "/" subtype ] *( ";" parameter ) # data := *urlchar # parameter := attribute "=" value try: [type, data] = url.split(',', 1) except ValueError: raise OSError('data error', 'bad data URL') if not type: type = 'text/plain;charset=US-ASCII' semi = type.rfind(';') if semi >= 0 and '=' not in type[semi:]: encoding = type[semi+1:] type = type[:semi] else: encoding = '' msg = [] msg.append('Date: %s'%time.strftime('%a, %d %b %Y %H:%M:%S GMT', time.gmtime(time.time()))) msg.append('Content-type: %s' % type) if encoding == 'base64': # XXX is this encoding/decoding ok? data = base64.decodebytes(data.encode('ascii')).decode('latin-1') else: data = unquote(data) msg.append('Content-Length: %d' % len(data)) msg.append('') msg.append(data) msg = '\n'.join(msg) headers = email.message_from_string(msg) f = io.StringIO(msg) #f.fileno = None # needed for addinfourl return addinfourl(f, headers, url) class FancyURLopener(URLopener): """Derived class with handlers for errors we can handle (perhaps).""" def __init__(self, *args, **kwargs): URLopener.__init__(self, *args, **kwargs) self.auth_cache = {} self.tries = 0 self.maxtries = 10 def http_error_default(self, url, fp, errcode, errmsg, headers): """Default error handling -- don't raise an exception.""" return addinfourl(fp, headers, "http:" + url, errcode) def http_error_302(self, url, fp, errcode, errmsg, headers, data=None): """Error 302 -- relocated (temporarily).""" self.tries += 1 try: if self.maxtries and self.tries >= self.maxtries: if hasattr(self, "http_error_500"): meth = self.http_error_500 else: meth = self.http_error_default return meth(url, fp, 500, "Internal Server Error: Redirect Recursion", headers) result = self.redirect_internal(url, fp, errcode, errmsg, headers, data) return result finally: self.tries = 0 def redirect_internal(self, url, fp, errcode, errmsg, headers, data): if 'location' in headers: newurl = headers['location'] elif 'uri' in headers: newurl = headers['uri'] else: return fp.close() # In case the server sent a relative URL, join with original: newurl = urljoin(self.type + ":" + url, newurl) urlparts = urlparse(newurl) # For security reasons, we don't allow redirection to anything other # than http, https and ftp. # We are using newer HTTPError with older redirect_internal method # This older method will get deprecated in 3.3 if urlparts.scheme not in ('http', 'https', 'ftp', ''): raise HTTPError(newurl, errcode, errmsg + " Redirection to url '%s' is not allowed." % newurl, headers, fp) return self.open(newurl) def http_error_301(self, url, fp, errcode, errmsg, headers, data=None): """Error 301 -- also relocated (permanently).""" return self.http_error_302(url, fp, errcode, errmsg, headers, data) def http_error_303(self, url, fp, errcode, errmsg, headers, data=None): """Error 303 -- also relocated (essentially identical to 302).""" return self.http_error_302(url, fp, errcode, errmsg, headers, data) def http_error_307(self, url, fp, errcode, errmsg, headers, data=None): """Error 307 -- relocated, but turn POST into error.""" if data is None: return self.http_error_302(url, fp, errcode, errmsg, headers, data) else: return self.http_error_default(url, fp, errcode, errmsg, headers) def http_error_401(self, url, fp, errcode, errmsg, headers, data=None, retry=False): """Error 401 -- authentication required. This function supports Basic authentication only.""" if 'www-authenticate' not in headers: URLopener.http_error_default(self, url, fp, errcode, errmsg, headers) stuff = headers['www-authenticate'] match = re.match('[ \t]*([^ \t]+)[ \t]+realm="([^"]*)"', stuff) if not match: URLopener.http_error_default(self, url, fp, errcode, errmsg, headers) scheme, realm = match.groups() if scheme.lower() != 'basic': URLopener.http_error_default(self, url, fp, errcode, errmsg, headers) if not retry: URLopener.http_error_default(self, url, fp, errcode, errmsg, headers) name = 'retry_' + self.type + '_basic_auth' if data is None: return getattr(self,name)(url, realm) else: return getattr(self,name)(url, realm, data) def http_error_407(self, url, fp, errcode, errmsg, headers, data=None, retry=False): """Error 407 -- proxy authentication required. This function supports Basic authentication only.""" if 'proxy-authenticate' not in headers: URLopener.http_error_default(self, url, fp, errcode, errmsg, headers) stuff = headers['proxy-authenticate'] match = re.match('[ \t]*([^ \t]+)[ \t]+realm="([^"]*)"', stuff) if not match: URLopener.http_error_default(self, url, fp, errcode, errmsg, headers) scheme, realm = match.groups() if scheme.lower() != 'basic': URLopener.http_error_default(self, url, fp, errcode, errmsg, headers) if not retry: URLopener.http_error_default(self, url, fp, errcode, errmsg, headers) name = 'retry_proxy_' + self.type + '_basic_auth' if data is None: return getattr(self,name)(url, realm) else: return getattr(self,name)(url, realm, data) def retry_proxy_http_basic_auth(self, url, realm, data=None): host, selector = splithost(url) newurl = 'http://' + host + selector proxy = self.proxies['http'] urltype, proxyhost = splittype(proxy) proxyhost, proxyselector = splithost(proxyhost) i = proxyhost.find('@') + 1 proxyhost = proxyhost[i:] user, passwd = self.get_user_passwd(proxyhost, realm, i) if not (user or passwd): return None proxyhost = "%s:%s@%s" % (quote(user, safe=''), quote(passwd, safe=''), proxyhost) self.proxies['http'] = 'http://' + proxyhost + proxyselector if data is None: return self.open(newurl) else: return self.open(newurl, data) def retry_proxy_https_basic_auth(self, url, realm, data=None): host, selector = splithost(url) newurl = 'https://' + host + selector proxy = self.proxies['https'] urltype, proxyhost = splittype(proxy) proxyhost, proxyselector = splithost(proxyhost) i = proxyhost.find('@') + 1 proxyhost = proxyhost[i:] user, passwd = self.get_user_passwd(proxyhost, realm, i) if not (user or passwd): return None proxyhost = "%s:%s@%s" % (quote(user, safe=''), quote(passwd, safe=''), proxyhost) self.proxies['https'] = 'https://' + proxyhost + proxyselector if data is None: return self.open(newurl) else: return self.open(newurl, data) def retry_http_basic_auth(self, url, realm, data=None): host, selector = splithost(url) i = host.find('@') + 1 host = host[i:] user, passwd = self.get_user_passwd(host, realm, i) if not (user or passwd): return None host = "%s:%s@%s" % (quote(user, safe=''), quote(passwd, safe=''), host) newurl = 'http://' + host + selector if data is None: return self.open(newurl) else: return self.open(newurl, data) def retry_https_basic_auth(self, url, realm, data=None): host, selector = splithost(url) i = host.find('@') + 1 host = host[i:] user, passwd = self.get_user_passwd(host, realm, i) if not (user or passwd): return None host = "%s:%s@%s" % (quote(user, safe=''), quote(passwd, safe=''), host) newurl = 'https://' + host + selector if data is None: return self.open(newurl) else: return self.open(newurl, data) def get_user_passwd(self, host, realm, clear_cache=0): key = realm + '@' + host.lower() if key in self.auth_cache: if clear_cache: del self.auth_cache[key] else: return self.auth_cache[key] user, passwd = self.prompt_user_passwd(host, realm) if user or passwd: self.auth_cache[key] = (user, passwd) return user, passwd def prompt_user_passwd(self, host, realm): """Override this in a GUI environment!""" import getpass try: user = input("Enter username for %s at %s: " % (realm, host)) passwd = getpass.getpass("Enter password for %s in %s at %s: " % (user, realm, host)) return user, passwd except KeyboardInterrupt: print() return None, None # Utility functions _localhost = None def localhost(): """Return the IP address of the magic hostname 'localhost'.""" global _localhost if _localhost is None: _localhost = socket.gethostbyname('localhost') return _localhost _thishost = None def thishost(): """Return the IP addresses of the current host.""" global _thishost if _thishost is None: try: _thishost = tuple(socket.gethostbyname_ex(socket.gethostname())[2]) except socket.gaierror: _thishost = tuple(socket.gethostbyname_ex('localhost')[2]) return _thishost _ftperrors = None def ftperrors(): """Return the set of errors raised by the FTP class.""" global _ftperrors if _ftperrors is None: import ftplib _ftperrors = ftplib.all_errors return _ftperrors _noheaders = None def noheaders(): """Return an empty email Message object.""" global _noheaders if _noheaders is None: _noheaders = email.message_from_string("") return _noheaders # Utility classes class ftpwrapper: """Class used by open_ftp() for cache of open FTP connections.""" def __init__(self, user, passwd, host, port, dirs, timeout=None, persistent=True): self.user = user self.passwd = passwd self.host = host self.port = port self.dirs = dirs self.timeout = timeout self.refcount = 0 self.keepalive = persistent try: self.init() except: self.close() raise def init(self): import ftplib self.busy = 0 self.ftp = ftplib.FTP() self.ftp.connect(self.host, self.port, self.timeout) self.ftp.login(self.user, self.passwd) _target = '/'.join(self.dirs) self.ftp.cwd(_target) def retrfile(self, file, type): import ftplib self.endtransfer() if type in ('d', 'D'): cmd = 'TYPE A'; isdir = 1 else: cmd = 'TYPE ' + type; isdir = 0 try: self.ftp.voidcmd(cmd) except ftplib.all_errors: self.init() self.ftp.voidcmd(cmd) conn = None if file and not isdir: # Try to retrieve as a file try: cmd = 'RETR ' + file conn, retrlen = self.ftp.ntransfercmd(cmd) except ftplib.error_perm as reason: if str(reason)[:3] != '550': raise URLError('ftp error: %r' % reason).with_traceback( sys.exc_info()[2]) if not conn: # Set transfer mode to ASCII! self.ftp.voidcmd('TYPE A') # Try a directory listing. Verify that directory exists. if file: pwd = self.ftp.pwd() try: try: self.ftp.cwd(file) except ftplib.error_perm as reason: raise URLError('ftp error: %r' % reason) from reason finally: self.ftp.cwd(pwd) cmd = 'LIST ' + file else: cmd = 'LIST' conn, retrlen = self.ftp.ntransfercmd(cmd) self.busy = 1 ftpobj = addclosehook(conn.makefile('rb'), self.file_close) self.refcount += 1 conn.close() # Pass back both a suitably decorated object and a retrieval length return (ftpobj, retrlen) def endtransfer(self): self.busy = 0 def close(self): self.keepalive = False if self.refcount <= 0: self.real_close() def file_close(self): self.endtransfer() self.refcount -= 1 if self.refcount <= 0 and not self.keepalive: self.real_close() def real_close(self): self.endtransfer() try: self.ftp.close() except ftperrors(): pass # Proxy handling def getproxies_environment(): """Return a dictionary of scheme -> proxy server URL mappings. Scan the environment for variables named <scheme>_proxy; this seems to be the standard convention. If you need a different way, you can pass a proxies dictionary to the [Fancy]URLopener constructor. """ proxies = {} # in order to prefer lowercase variables, process environment in # two passes: first matches any, second pass matches lowercase only for name, value in os.environ.items(): name = name.lower() if value and name[-6:] == '_proxy': proxies[name[:-6]] = value # CVE-2016-1000110 - If we are running as CGI script, forget HTTP_PROXY # (non-all-lowercase) as it may be set from the web server by a "Proxy:" # header from the client # If "proxy" is lowercase, it will still be used thanks to the next block if 'REQUEST_METHOD' in os.environ: proxies.pop('http', None) for name, value in os.environ.items(): if name[-6:] == '_proxy': name = name.lower() if value: proxies[name[:-6]] = value else: proxies.pop(name[:-6], None) return proxies def proxy_bypass_environment(host, proxies=None): """Test if proxies should not be used for a particular host. Checks the proxy dict for the value of no_proxy, which should be a list of comma separated DNS suffixes, or '*' for all hosts. """ if proxies is None: proxies = getproxies_environment() # don't bypass, if no_proxy isn't specified try: no_proxy = proxies['no'] except KeyError: return 0 # '*' is special case for always bypass if no_proxy == '*': return 1 # strip port off host hostonly, port = splitport(host) # check if the host ends with any of the DNS suffixes no_proxy_list = [proxy.strip() for proxy in no_proxy.split(',')] for name in no_proxy_list: if name: name = name.lstrip('.') # ignore leading dots name = re.escape(name) pattern = r'(.+\.)?%s$' % name if (re.match(pattern, hostonly, re.I) or re.match(pattern, host, re.I)): return 1 # otherwise, don't bypass return 0 # This code tests an OSX specific data structure but is testable on all # platforms def _proxy_bypass_macosx_sysconf(host, proxy_settings): """ Return True iff this host shouldn't be accessed using a proxy This function uses the MacOSX framework SystemConfiguration to fetch the proxy information. proxy_settings come from _scproxy._get_proxy_settings or get mocked ie: { 'exclude_simple': bool, 'exceptions': ['foo.bar', '*.bar.com', '127.0.0.1', '10.1', '10.0/16'] } """ from fnmatch import fnmatch hostonly, port = splitport(host) def ip2num(ipAddr): parts = ipAddr.split('.') parts = list(map(int, parts)) if len(parts) != 4: parts = (parts + [0, 0, 0, 0])[:4] return (parts[0] << 24) | (parts[1] << 16) | (parts[2] << 8) | parts[3] # Check for simple host names: if '.' not in host: if proxy_settings['exclude_simple']: return True hostIP = None for value in proxy_settings.get('exceptions', ()): # Items in the list are strings like these: *.local, 169.254/16 if not value: continue m = re.match(r"(\d+(?:\.\d+)*)(/\d+)?", value) if m is not None: if hostIP is None: try: hostIP = socket.gethostbyname(hostonly) hostIP = ip2num(hostIP) except OSError: continue base = ip2num(m.group(1)) mask = m.group(2) if mask is None: mask = 8 * (m.group(1).count('.') + 1) else: mask = int(mask[1:]) mask = 32 - mask if (hostIP >> mask) == (base >> mask): return True elif fnmatch(host, value): return True return False if sys.platform == 'darwin': from _scproxy import _get_proxy_settings, _get_proxies def proxy_bypass_macosx_sysconf(host): proxy_settings = _get_proxy_settings() return _proxy_bypass_macosx_sysconf(host, proxy_settings) def getproxies_macosx_sysconf(): """Return a dictionary of scheme -> proxy server URL mappings. This function uses the MacOSX framework SystemConfiguration to fetch the proxy information. """ return _get_proxies() def proxy_bypass(host): """Return True, if host should be bypassed. Checks proxy settings gathered from the environment, if specified, or from the MacOSX framework SystemConfiguration. """ proxies = getproxies_environment() if proxies: return proxy_bypass_environment(host, proxies) else: return proxy_bypass_macosx_sysconf(host) def getproxies(): return getproxies_environment() or getproxies_macosx_sysconf() elif os.name == 'nt': def getproxies_registry(): """Return a dictionary of scheme -> proxy server URL mappings. Win32 uses the registry to store proxies. """ proxies = {} try: import winreg except ImportError: # Std module, so should be around - but you never know! return proxies try: internetSettings = winreg.OpenKey(winreg.HKEY_CURRENT_USER, r'Software\Microsoft\Windows\CurrentVersion\Internet Settings') proxyEnable = winreg.QueryValueEx(internetSettings, 'ProxyEnable')[0] if proxyEnable: # Returned as Unicode but problems if not converted to ASCII proxyServer = str(winreg.QueryValueEx(internetSettings, 'ProxyServer')[0]) if '=' in proxyServer: # Per-protocol settings for p in proxyServer.split(';'): protocol, address = p.split('=', 1) # See if address has a type:// prefix if not re.match('^([^/:]+)://', address): address = '%s://%s' % (protocol, address) proxies[protocol] = address else: # Use one setting for all protocols if proxyServer[:5] == 'http:': proxies['http'] = proxyServer else: proxies['http'] = 'http://%s' % proxyServer proxies['https'] = 'https://%s' % proxyServer proxies['ftp'] = 'ftp://%s' % proxyServer internetSettings.Close() except (OSError, ValueError, TypeError): # Either registry key not found etc, or the value in an # unexpected format. # proxies already set up to be empty so nothing to do pass return proxies def getproxies(): """Return a dictionary of scheme -> proxy server URL mappings. Returns settings gathered from the environment, if specified, or the registry. """ return getproxies_environment() or getproxies_registry() def proxy_bypass_registry(host): try: import winreg except ImportError: # Std modules, so should be around - but you never know! return 0 try: internetSettings = winreg.OpenKey(winreg.HKEY_CURRENT_USER, r'Software\Microsoft\Windows\CurrentVersion\Internet Settings') proxyEnable = winreg.QueryValueEx(internetSettings, 'ProxyEnable')[0] proxyOverride = str(winreg.QueryValueEx(internetSettings, 'ProxyOverride')[0]) # ^^^^ Returned as Unicode but problems if not converted to ASCII except OSError: return 0 if not proxyEnable or not proxyOverride: return 0 # try to make a host list from name and IP address. rawHost, port = splitport(host) host = [rawHost] try: addr = socket.gethostbyname(rawHost) if addr != rawHost: host.append(addr) except OSError: pass try: fqdn = socket.getfqdn(rawHost) if fqdn != rawHost: host.append(fqdn) except OSError: pass # make a check value list from the registry entry: replace the # '<local>' string by the localhost entry and the corresponding # canonical entry. proxyOverride = proxyOverride.split(';') # now check if we match one of the registry values. for test in proxyOverride: if test == '<local>': if '.' not in rawHost: return 1 test = test.replace(".", r"\.") # mask dots test = test.replace("*", r".*") # change glob sequence test = test.replace("?", r".") # change glob char for val in host: if re.match(test, val, re.I): return 1 return 0 def proxy_bypass(host): """Return True, if host should be bypassed. Checks proxy settings gathered from the environment, if specified, or the registry. """ proxies = getproxies_environment() if proxies: return proxy_bypass_environment(host, proxies) else: return proxy_bypass_registry(host) else: # By default use environment variables getproxies = getproxies_environment proxy_bypass = proxy_bypass_environment PK23�\��@��response.pynu�[���"""Response classes used by urllib. The base class, addbase, defines a minimal file-like interface, including read() and readline(). The typical response object is an addinfourl instance, which defines an info() method that returns headers and a geturl() method that returns the url. """ import tempfile __all__ = ['addbase', 'addclosehook', 'addinfo', 'addinfourl'] class addbase(tempfile._TemporaryFileWrapper): """Base class for addinfo and addclosehook. Is a good idea for garbage collection.""" # XXX Add a method to expose the timeout on the underlying socket? def __init__(self, fp): super(addbase, self).__init__(fp, '<urllib response>', delete=False) # Keep reference around as this was part of the original API. self.fp = fp def __repr__(self): return '<%s at %r whose fp = %r>' % (self.__class__.__name__, id(self), self.file) def __enter__(self): if self.fp.closed: raise ValueError("I/O operation on closed file") return self def __exit__(self, type, value, traceback): self.close() class addclosehook(addbase): """Class to add a close hook to an open file.""" def __init__(self, fp, closehook, *hookargs): super(addclosehook, self).__init__(fp) self.closehook = closehook self.hookargs = hookargs def close(self): try: closehook = self.closehook hookargs = self.hookargs if closehook: self.closehook = None self.hookargs = None closehook(*hookargs) finally: super(addclosehook, self).close() class addinfo(addbase): """class to add an info() method to an open file.""" def __init__(self, fp, headers): super(addinfo, self).__init__(fp) self.headers = headers def info(self): return self.headers class addinfourl(addinfo): """class to add info() and geturl() methods to an open file.""" def __init__(self, fp, headers, url, code=None): super(addinfourl, self).__init__(fp, headers) self.url = url self.code = code def getcode(self): return self.code def geturl(self): return self.url PK23�\�ኀ"�"robotparser.pynu�[���""" robotparser.py Copyright (C) 2000 Bastian Kleineidam You can choose between two licenses when using this package: 1) GNU GPLv2 2) PSF license for Python 2.2 The robots.txt Exclusion Protocol is implemented as specified in http://www.robotstxt.org/norobots-rfc.txt """ import collections import urllib.parse import urllib.request __all__ = ["RobotFileParser"] RequestRate = collections.namedtuple("RequestRate", "requests seconds") class RobotFileParser: """ This class provides a set of methods to read, parse and answer questions about a single robots.txt file. """ def __init__(self, url=''): self.entries = [] self.default_entry = None self.disallow_all = False self.allow_all = False self.set_url(url) self.last_checked = 0 def mtime(self): """Returns the time the robots.txt file was last fetched. This is useful for long-running web spiders that need to check for new robots.txt files periodically. """ return self.last_checked def modified(self): """Sets the time the robots.txt file was last fetched to the current time. """ import time self.last_checked = time.time() def set_url(self, url): """Sets the URL referring to a robots.txt file.""" self.url = url self.host, self.path = urllib.parse.urlparse(url)[1:3] def read(self): """Reads the robots.txt URL and feeds it to the parser.""" try: f = urllib.request.urlopen(self.url) except urllib.error.HTTPError as err: if err.code in (401, 403): self.disallow_all = True elif err.code >= 400 and err.code < 500: self.allow_all = True else: raw = f.read() self.parse(raw.decode("utf-8").splitlines()) def _add_entry(self, entry): if "*" in entry.useragents: # the default entry is considered last if self.default_entry is None: # the first default entry wins self.default_entry = entry else: self.entries.append(entry) def parse(self, lines): """Parse the input lines from a robots.txt file. We allow that a user-agent: line is not preceded by one or more blank lines. """ # states: # 0: start state # 1: saw user-agent line # 2: saw an allow or disallow line state = 0 entry = Entry() self.modified() for line in lines: if not line: if state == 1: entry = Entry() state = 0 elif state == 2: self._add_entry(entry) entry = Entry() state = 0 # remove optional comment and strip line i = line.find('#') if i >= 0: line = line[:i] line = line.strip() if not line: continue line = line.split(':', 1) if len(line) == 2: line[0] = line[0].strip().lower() line[1] = urllib.parse.unquote(line[1].strip()) if line[0] == "user-agent": if state == 2: self._add_entry(entry) entry = Entry() entry.useragents.append(line[1]) state = 1 elif line[0] == "disallow": if state != 0: entry.rulelines.append(RuleLine(line[1], False)) state = 2 elif line[0] == "allow": if state != 0: entry.rulelines.append(RuleLine(line[1], True)) state = 2 elif line[0] == "crawl-delay": if state != 0: # before trying to convert to int we need to make # sure that robots.txt has valid syntax otherwise # it will crash if line[1].strip().isdigit(): entry.delay = int(line[1]) state = 2 elif line[0] == "request-rate": if state != 0: numbers = line[1].split('/') # check if all values are sane if (len(numbers) == 2 and numbers[0].strip().isdigit() and numbers[1].strip().isdigit()): entry.req_rate = RequestRate(int(numbers[0]), int(numbers[1])) state = 2 if state == 2: self._add_entry(entry) def can_fetch(self, useragent, url): """using the parsed robots.txt decide if useragent can fetch url""" if self.disallow_all: return False if self.allow_all: return True # Until the robots.txt file has been read or found not # to exist, we must assume that no url is allowable. # This prevents false positives when a user erroneously # calls can_fetch() before calling read(). if not self.last_checked: return False # search for given user agent matches # the first match counts parsed_url = urllib.parse.urlparse(urllib.parse.unquote(url)) url = urllib.parse.urlunparse(('','',parsed_url.path, parsed_url.params,parsed_url.query, parsed_url.fragment)) url = urllib.parse.quote(url) if not url: url = "/" for entry in self.entries: if entry.applies_to(useragent): return entry.allowance(url) # try the default entry last if self.default_entry: return self.default_entry.allowance(url) # agent not found ==> access granted return True def crawl_delay(self, useragent): if not self.mtime(): return None for entry in self.entries: if entry.applies_to(useragent): return entry.delay return self.default_entry.delay def request_rate(self, useragent): if not self.mtime(): return None for entry in self.entries: if entry.applies_to(useragent): return entry.req_rate return self.default_entry.req_rate def __str__(self): entries = self.entries if self.default_entry is not None: entries = entries + [self.default_entry] return '\n'.join(map(str, entries)) + '\n' class RuleLine: """A rule line is a single "Allow:" (allowance==True) or "Disallow:" (allowance==False) followed by a path.""" def __init__(self, path, allowance): if path == '' and not allowance: # an empty value means allow all allowance = True path = urllib.parse.urlunparse(urllib.parse.urlparse(path)) self.path = urllib.parse.quote(path) self.allowance = allowance def applies_to(self, filename): return self.path == "*" or filename.startswith(self.path) def __str__(self): return ("Allow" if self.allowance else "Disallow") + ": " + self.path class Entry: """An entry has one or more user-agents and zero or more rulelines""" def __init__(self): self.useragents = [] self.rulelines = [] self.delay = None self.req_rate = None def __str__(self): ret = [] for agent in self.useragents: ret.append(f"User-agent: {agent}") if self.delay is not None: ret.append(f"Crawl-delay: {self.delay}") if self.req_rate is not None: rate = self.req_rate ret.append(f"Request-rate: {rate.requests}/{rate.seconds}") ret.extend(map(str, self.rulelines)) ret.append('') # for compatibility return '\n'.join(ret) def applies_to(self, useragent): """check if this entry applies to the specified agent""" # split the name token and make it lower case useragent = useragent.split("/")[0].lower() for agent in self.useragents: if agent == '*': # we have the catch-all agent return True agent = agent.lower() if agent in useragent: return True return False def allowance(self, filename): """Preconditions: - our agent applies to this entry - filename is URL decoded""" for line in self.rulelines: if line.applies_to(filename): return line.allowance return True PKS�\��ԉ�*__pycache__/__init__.cpython-312.opt-1.pycnu�[���� J�i���y)N�r��(/usr/lib64/python3.12/urllib/__init__.py�<module>rs�rPKS�\��ԉ�*__pycache__/__init__.cpython-312.opt-2.pycnu�[���� J�i���y)N�r��(/usr/lib64/python3.12/urllib/__init__.py�<module>rs�rPKS�\��ԉ�$__pycache__/__init__.cpython-312.pycnu�[���� J�i���y)N�r��(/usr/lib64/python3.12/urllib/__init__.py�<module>rs�rPKS�\�0 �BB'__pycache__/error.cpython-312.opt-1.pycnu�[���� T��ho ���dZddlZddlZgd�ZGd�de�ZGd�deejj�Z Gd�d e�Z y) a�Exception classes raised by urllib. The base exception class is URLError, which inherits from OSError. It doesn't define any behavior of its own, but is the base class for all exceptions defined in this package. HTTPError is an exception class that is also a valid HTTP response instance. It behaves this way because HTTP protocol errors are valid responses, with a status code, headers, and a body. In some contexts, an application may want to handle an exception like a regular response. �N)�URLError� HTTPError�ContentTooShortErrorc��eZdZdd�Zd�Zy)rNc�6�|f|_||_|�||_yy�N)�args�reason�filename)�selfr rs �%/usr/lib64/python3.12/urllib/error.py�__init__zURLError.__init__s$���G�� ������$�D�M� �c� �d|jzS)Nz<urlopen error %s>)r �rs r �__str__zURLError.__str__s��#�d�k�k�1�1rr)�__name__� __module__�__qualname__rr�rr rrs��%�2rrc��eZdZdZej jjZd�Zd�Z d�Z ed��Zed��Z e jd��Z y) rzBRaised when HTTP error occurs, but also acts like non-error returnc��||_||_||_||_||_|�tj�}|j||||�yr)�code�msg�hdrs�fpr�io�BytesIO�_HTTPError__super_init)r�urlrrrrs r rzHTTPError.__init__'sI���� ������ ������ � �:�����B����"�d�C��.rc�:�d|j�d|j��S)NzHTTP Error �: �rrrs r rzHTTPError.__str__1s��&*�i�i����:�:rc�<�d|j�d|j�d�S)Nz<HTTPError r"�>r#rs r �__repr__zHTTPError.__repr__4s��'+�y�y�$�(�(�;�;rc��|jSr)rrs r r zHTTPError.reason9s���x�x�rc��|jSr�rrs r �headerszHTTPError.headers=s���y�y�rc��||_yrr))rr*s r r*zHTTPError.headersAs ���� rN)rrr�__doc__�urllib�response� addinfourlrrrr&�propertyr r*�setterrrr rr#si��L��?�?�-�-�6�6�L�/�;�<� �������� �^�^���rrc��eZdZdZd�Zy)rzDException raised when downloaded size does not match content-length.c�>�tj||�||_yr)rr�content)r�messager4s r rzContentTooShortError.__init__Hs�����$��(���rN)rrrr,rrrr rrFs ��N�rr)r,r�urllib.responser-�__all__�OSErrorrr.r/rrrrr �<module>r9sH��� �� ;�� 2�w� 2� ��&�/�/�4�4� �F�8�rPKS�\��,��'__pycache__/error.cpython-312.opt-2.pycnu�[���� T��ho ��� ddlZddlZgd�ZGd�de�ZGd�deejj�ZGd�de�Z y) �N)�URLError� HTTPError�ContentTooShortErrorc��eZdZdd�Zd�Zy)rNc�6�|f|_||_|�||_yy�N)�args�reason�filename)�selfr rs �%/usr/lib64/python3.12/urllib/error.py�__init__zURLError.__init__s$���G�� ������$�D�M� �c� �d|jzS)Nz<urlopen error %s>)r �rs r �__str__zURLError.__str__s��#�d�k�k�1�1rr)�__name__� __module__�__qualname__rr�rr rrs��%�2rrc��eZdZ ejj jZd�Zd�Zd�Z e d��Ze d��Zejd��Zy)rc��||_||_||_||_||_|�tj�}|j||||�yr)�code�msg�hdrs�fpr�io�BytesIO�_HTTPError__super_init)r�urlrrrrs r rzHTTPError.__init__'sI���� ������ ������ � �:�����B����"�d�C��.rc�:�d|j�d|j��S)NzHTTP Error �: �rrrs r rzHTTPError.__str__1s��&*�i�i����:�:rc�<�d|j�d|j�d�S)Nz<HTTPError r"�>r#rs r �__repr__zHTTPError.__repr__4s��'+�y�y�$�(�(�;�;rc��|jSr)rrs r r zHTTPError.reason9s���x�x�rc��|jSr�rrs r �headerszHTTPError.headers=s���y�y�rc��||_yrr))rr*s r r*zHTTPError.headersAs ���� rN)rrr�urllib�response� addinfourlrrrr&�propertyr r*�setterrrr rr#si��L��?�?�-�-�6�6�L�/�;�<� �������� �^�^���rrc��eZdZ d�Zy)rc�>�tj||�||_yr)rr�content)r�messager3s r rzContentTooShortError.__init__Hs�����$��(���rN)rrrrrrr rrFs ��N�rr) r�urllib.responser,�__all__�OSErrorrr-r.rrrrr �<module>r8sH��� �� ;�� 2�w� 2� ��&�/�/�4�4� �F�8�rPKS�\�0 �BB!__pycache__/error.cpython-312.pycnu�[���� T��ho ���dZddlZddlZgd�ZGd�de�ZGd�deejj�Z Gd�d e�Z y) a�Exception classes raised by urllib. The base exception class is URLError, which inherits from OSError. It doesn't define any behavior of its own, but is the base class for all exceptions defined in this package. HTTPError is an exception class that is also a valid HTTP response instance. It behaves this way because HTTP protocol errors are valid responses, with a status code, headers, and a body. In some contexts, an application may want to handle an exception like a regular response. �N)�URLError� HTTPError�ContentTooShortErrorc��eZdZdd�Zd�Zy)rNc�6�|f|_||_|�||_yy�N)�args�reason�filename)�selfr rs �%/usr/lib64/python3.12/urllib/error.py�__init__zURLError.__init__s$���G�� ������$�D�M� �c� �d|jzS)Nz<urlopen error %s>)r �rs r �__str__zURLError.__str__s��#�d�k�k�1�1rr)�__name__� __module__�__qualname__rr�rr rrs��%�2rrc��eZdZdZej jjZd�Zd�Z d�Z ed��Zed��Z e jd��Z y) rzBRaised when HTTP error occurs, but also acts like non-error returnc��||_||_||_||_||_|�tj�}|j||||�yr)�code�msg�hdrs�fpr�io�BytesIO�_HTTPError__super_init)r�urlrrrrs r rzHTTPError.__init__'sI���� ������ ������ � �:�����B����"�d�C��.rc�:�d|j�d|j��S)NzHTTP Error �: �rrrs r rzHTTPError.__str__1s��&*�i�i����:�:rc�<�d|j�d|j�d�S)Nz<HTTPError r"�>r#rs r �__repr__zHTTPError.__repr__4s��'+�y�y�$�(�(�;�;rc��|jSr)rrs r r zHTTPError.reason9s���x�x�rc��|jSr�rrs r �headerszHTTPError.headers=s���y�y�rc��||_yrr))rr*s r r*zHTTPError.headersAs ���� rN)rrr�__doc__�urllib�response� addinfourlrrrr&�propertyr r*�setterrrr rr#si��L��?�?�-�-�6�6�L�/�;�<� �������� �^�^���rrc��eZdZdZd�Zy)rzDException raised when downloaded size does not match content-length.c�>�tj||�||_yr)rr�content)r�messager4s r rzContentTooShortError.__init__Hs�����$��(���rN)rrrr,rrrr rrFs ��N�rr)r,r�urllib.responser-�__all__�OSErrorrr.r/rrrrr �<module>r9sH��� �� ;�� 2�w� 2� ��&�/�/�4�4� �F�8�rPKS�\���t����'__pycache__/parse.cpython-312.opt-1.pycnu�[���� T��hl�����dZddlmZddlZddlZddlZddlZddlZddlZgd�Z gd�Z gd�Zgd�Zgd�Z gd �Zgd �ZdZdZgd �Zd�ZdZdZd�Zeefd�Zeefd�Zd�ZGd�de�ZGd�de�ZGd�de�ZGd�dee�ZGd�dee�Zedd �Z ed!d"�Z!ed#d$�Z"d%e _d&e jF_d'e jH_d(e!_d)e!jJ_d*e!jL_d+e!jN_d,e!jP_d-e!jH_d.e"_e!jJje"jJ_e!jLje"jL_e!jNje"jN_d/e"jR_e!jPje"jP_e!jHje"jH_eZ*Gd0�de e�Z+Gd1�d!e!e�Z,Gd2�d#e"e�Z-Gd3�d4e e�Z.Gd5�d6e!e�Z/Gd7�d8e"e�Z0d9�Z1e1�[1dtd<�Z2d=�Z3dud>�Z4d?�Z5d@�Z6dA�Z7ejpd;�B�dtdC��Z9dD�Z:dE�Z;dvdF�Z<dG�Z=dHZ>da?dI�Z@dJeAeBzeCzdKeAeBzfdL�ZDej�dM�ZFdN�ZGdwdO�ZH dxdQ�ZI dxdR�ZJdwdS�ZKeLdT�ZMeAeM�ZNdU�ZOGdV�dWeP�ZQdydX�ZRdzdY�ZSejpdZ��ZTd{d[�ZUdPd:ddeSfd\�ZVd]�ZWd^�ZXd_�ZYd`�ZZda[da�Z\db�Z]da^dc�Z_dd�Z`de�Zadf�Zbdg�Zcdh�Zddaedi�Zfd|dj�Zgd|dk�Zhdl�Zidm�Zjdn�Zkdo�Zldp�Zmdq�Zndr�Zods�Zpy)}a3Parse (absolute and relative) URLs. urlparse module is based upon the following RFC specifications. RFC 3986 (STD66): "Uniform Resource Identifiers" by T. Berners-Lee, R. Fielding and L. Masinter, January 2005. RFC 2732 : "Format for Literal IPv6 Addresses in URL's by R.Hinden, B.Carpenter and L.Masinter, December 1999. RFC 2396: "Uniform Resource Identifiers (URI)": Generic Syntax by T. Berners-Lee, R. Fielding, and L. Masinter, August 1998. RFC 2368: "The mailto URL scheme", by P.Hoffman , L Masinter, J. Zawinski, July 1998. RFC 1808: "Relative Uniform Resource Locators", by R. Fielding, UC Irvine, June 1995. RFC 1738: "Uniform Resource Locators (URL)" by T. Berners-Lee, L. Masinter, M. McCahill, December 1994 RFC 3986 is considered the current standard and any future changes to urlparse module should conform with it. The urlparse module is currently not entirely compliant with this RFC due to defacto scenarios for parsing, and for backward compatibility purposes, some parsing quirks from older RFCs are retained. The testcases in test_urlparse.py provides a good indicator of parsing behavior. The WHATWG URL Parser spec should also be considered. We are not compliant with it either due to existing user code API behavior expectations (Hyrum's Law). It serves as a useful guide when making changes. �)� namedtupleN)�urlparse� urlunparse�urljoin� urldefrag�urlsplit� urlunsplit� urlencode�parse_qs� parse_qsl�quote� quote_plus�quote_from_bytes�unquote�unquote_plus�unquote_to_bytes�DefragResult�ParseResult�SplitResult�DefragResultBytes�ParseResultBytes�SplitResultBytes)��ftp�http�gopher�nntp�imap�wais�file�https�shttp�mms�prospero�rtsp�rtsps�rtspu�sftp�svn�svn+ssh�ws�wss)rrrrr�telnetrrr r#r!r"�snewsr$r%r&r'�rsyncr)r*r(�nfs�gitzgit+sshr+r,z itms-services)rr�hdlr$rrr!r"r%r&r'�sip�sipsr#r(�tel) rr2�mailto�newsr-rrr.r3r4) rrrrr!r"r#rr%r&r'r3r4) rrr2rrr7rrr!r"r.r r$zAabcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+-.z! )� � � c�T�tj�tj�y)zDClear internal performance caches. Undocumented; some tests want it.N)r�cache_clear�_byte_quoter_factory���%/usr/lib64/python3.12/urllib/parse.py�clear_cacherA^s�������$�$�&r?�ascii�strictc��|S�Nr>)�objs r@�_nooprGls���Jr?c�&�|j||�SrE��encode)rF�encoding�errorss r@�_encode_resultrMos���:�:�h��'�'r?c�0���t��fd�|D��S)Nc3�J�K�|]}|r|j���nd���y�w)rN��decode��.0�xrKrLs ��r@� <genexpr>z_decode_args.<locals>.<genexpr>us$�����G�$�Q�q����(�F�+�b�8�$�s� #)�tuple)�argsrKrLs ``r@�_decode_argsrXss����G�$�G�G�Gr?c��t|dt�}|ddD]#}|s�t|t�|k7s�td��|r |tfzSt |�t fzS)Nr�z$Cannot mix str and non-str arguments)� isinstance�str� TypeErrorrGrXrM)rW� str_input�args r@�_coerce_argsr`wsg���4��7�C�(�I��A�B�x���:�c�3�'�9�4��B�C�C� � ��u�h������� 1�1�1r?c��eZdZdZdZdd�Zy)�_ResultMixinStrz>Standard approach to encoding parsed results from str to bytesr>c�8���|j��fd�|D��S)Nc3�B�K�|]}|j������y�wrErIrRs ��r@rUz)_ResultMixinStr.encode.<locals>.<genexpr>�������*T�t�!�1�8�8�H�f�+E�t���)�_encoded_counterpart��selfrKrLs ``r@rJz_ResultMixinStr.encode�����(�t�(�(�*T�t�*T�U�Ur?N�rBrC)�__name__� __module__�__qualname__�__doc__� __slots__rJr>r?r@rbrb����H��I�Vr?rbc��eZdZdZdZdd�Zy)�_ResultMixinBytesz>Standard approach to decoding parsed results from bytes to strr>c�8���|j��fd�|D��S)Nc3�B�K�|]}|j������y�wrErPrRs ��r@rUz+_ResultMixinBytes.decode.<locals>.<genexpr>�rerf)�_decoded_counterpartrhs ``r@rQz_ResultMixinBytes.decode�rjr?Nrk)rlrmrnrorprQr>r?r@rsrs�rqr?rsc�x�eZdZdZdZed��Zed��Zed��Zed��Z e ej�Z y)�_NetlocResultMixinBasezHShared methods for the parsed result objects containing a netloc elementr>c� �|jdS)Nr�� _userinfo�ris r@�usernamez_NetlocResultMixinBase.username�����~�~�a� � r?c� �|jdS)NrZrzr|s r@�passwordz_NetlocResultMixinBase.password�r~r?c��|jd}|syt|t�rdnd}|j|�\}}}|j �|z|zS)Nr�%�%)� _hostinfor[r\� partition�lower)ri�hostname� separator�percent�zones r@r�z_NetlocResultMixinBase.hostname�sV���>�>�!�$����&�h��4�C�$� �"*�"4�"4�Y�"?���'�4��~�~��'�)�D�0�0r?c���|jd}|�\|j�r|j�rt|�}nt d|����d|cxkrdkst d��t d��|S)NrZz+Port could not be cast to integer value as ri��zPort out of range 0-65535)r��isdigit�isascii�int� ValueError)ri�ports r@r�z_NetlocResultMixinBase.port�sr���~�~�a� �����|�|�~�$�,�,�.��4�y�� �#N�t�h�!W�X�X���&��&� �!<�=�=�'� �!<�=�=��r?N)rlrmrnrorp�propertyr}r�r�r��classmethod�types�GenericAlias�__class_getitem__r>r?r@rxrx�sk��R��I� �!��!��!��!��1��1�� �� �$�E�$6�$6�7�r?rxc�0�eZdZdZed��Zed��Zy)�_NetlocResultMixinStrr>c��|j}|jd�\}}}|r|jd�\}}}|sd}||fSdx}}||fS)N�@�:��netloc� rpartitionr��rir��userinfo� have_info�hostinfor}� have_passwordr�s r@r{z_NetlocResultMixinStr._userinfo�sh������(.�(9�(9�#�(>�%��)�X��08�0B�0B�3�0G�-�H�m�X� �����!�!�#'�&�H�x���!�!r?c��|j}|jd�\}}}|jd�\}}}|r+|jd�\}}}|jd�\}}}n|jd�\}}}|sd}||fS)Nr��[�]r�r��rir��_r��have_open_br� bracketedr�r�s r@r�z_NetlocResultMixinStr._hostinfo�s��������*�*�3�/���1�h�%-�%7�%7��%<�"��<��� )� 3� 3�C� 8��H�a������,�J�A�q�$� (� 2� 2�3� 7��H�a����D���~�r?N�rlrmrnrpr�r{r�r>r?r@r�r���-���I� � "�� "����r?r�c�0�eZdZdZed��Zed��Zy)�_NetlocResultMixinBytesr>c��|j}|jd�\}}}|r|jd�\}}}|sd}||fSdx}}||fS)N�@�:r�r�s r@r{z!_NetlocResultMixinBytes._userinfo�sh������(.�(9�(9�$�(?�%��)�X��08�0B�0B�4�0H�-�H�m�X� �����!�!�#'�&�H�x���!�!r?c��|j}|jd�\}}}|jd�\}}}|r+|jd�\}}}|jd�\}}}n|jd�\}}}|sd}||fS)Nr��[�]r�r�r�s r@r�z!_NetlocResultMixinBytes._hostinfo�s��������*�*�4�0���1�h�%-�%7�%7��%=�"��<��� )� 3� 3�D� 9��H�a������-�J�A�q�$� (� 2� 2�4� 8��H�a����D���~�r?Nr�r>r?r@r�r��r�r?r�rzurl fragmentrz!scheme netloc path query fragmentrz(scheme netloc path params query fragmentz� DefragResult(url, fragment) A 2-tuple that contains the url without fragment identifier and the fragment identifier as a separate argument. z$The URL with no fragment identifier.z� Fragment identifier separated from URL, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. z� SplitResult(scheme, netloc, path, query, fragment) A 5-tuple that contains the different components of a URL. Similar to ParseResult, but does not split params. z%Specifies URL scheme for the request.z0 Network location where the request is made to. z@ The hierarchical path, such as the path to a file to download. z� The query component, that contains non-hierarchical data, that along with data in path component, identifies a resource in the scope of URI's scheme and network location. z� Fragment identifier, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. zq ParseResult(scheme, netloc, path, params, query, fragment) A 6-tuple that contains components of a parsed URL. z� Parameters for last path element used to dereference the URI in order to provide access to perform some operation on the resource. c��eZdZdZd�Zy)rr>c�j�|jr|jdz|jzS|jS)N�#��fragment�urlr|s r@�geturlzDefragResult.geturlEs)���=�=��8�8�c�>�D�M�M�1�1��8�8�Or?N�rlrmrnrpr�r>r?r@rrC����I�r?c��eZdZdZd�Zy)rr>c��t|�SrE�r r|s r@r�zSplitResult.geturlM� ���$��r?Nr�r>r?r@rrK����I� r?c��eZdZdZd�Zy)rr>c��t|�SrE�rr|s r@r�zParseResult.geturlRr�r?Nr�r>r?r@rrPr�r?c��eZdZdZd�Zy)rr>c�j�|jr|jdz|jzS|jS)N�#r�r|s r@r�zDefragResultBytes.geturlXs)���=�=��8�8�d�?�T�]�]�2�2��8�8�Or?Nr�r>r?r@rrVr�r?rc��eZdZdZd�Zy)rr>c��t|�SrEr�r|s r@r�zSplitResultBytes.geturl`r�r?Nr�r>r?r@rr^r�r?rc��eZdZdZd�Zy)rr>c��t|�SrEr�r|s r@r�zParseResultBytes.geturler�r?Nr�r>r?r@rrcr�r?rc�z�ttfttftt ff}|D]\}}||_||_�yrE)rrrrrrrgrv)� _result_pairs�_decoded�_encodeds r@�_fix_result_transcodingr�isE�� �(�)� �&�'� �&�'��M� ,���(�(0��%�(0��%�,r?rTc��t||�\}}}t|||�}|\}}}}}|tvrd|vrt|�\}}nd}t ||||||�} || �S)a�Parse a URL into 6 components: <scheme>://<netloc>/<path>;<params>?<query>#<fragment> The result is a named 6-tuple with fields corresponding to the above. It is either a ParseResult or ParseResultBytes object, depending on the type of the url parameter. The username, password, hostname, and port sub-components of netloc can also be accessed as attributes of the returned object. The scheme argument provides the default value of the scheme component when no scheme is found in url. If allow_fragments is False, no attempt is made to separate the fragment component from the previous component, which can be either path or query. Note that % escapes are not expanded. �;r)r`r�uses_params�_splitparamsr) r��scheme�allow_fragments�_coerce_result�splitresultr��queryr��params�results r@rrvsu��(#/�s�F�";��C����3���8�K�+6�(�F�F�C��� ������"�3�'���V��� ����f�e�X� F�F��&�!�!r?c��d|vr*|jd|jd��}|dkr|dfS|jd�}|d|||dzdfS)N�/r�rrrZ)�find�rfind)r��is r@r�r��sY�� �s�{��H�H�S�#�)�)�C�.�)���q�5���7�N��H�H�S�M���r��7�C��!���I��r?c��t|�}dD]&}|j||�}|dk\s�t||�}�(|||||dfS)Nz/?#r)�lenr��min)r��start�delim�c�wdelims r@�_splitnetlocr��sR����H�E� �����!�U�#���Q�;���v�&�E���u�U��S���[�(�(r?c�0�|r|j�ryddl}|jdd�}|jdd�}|jdd�}|jdd�}|jd|�}||k(rydD]}||vs�t d |zd zdz��y)Nrr�rr�r��?�NFKCz/?#@:znetloc 'z' contains invalid z#characters under NFKC normalization)r��unicodedata�replace� normalizer�)r�r��n�netloc2r�s r@�_checknetlocr��s����V�^�^�%������s�B��A� � � �#�r��A� � � �#�r��A� � � �#�r��A��#�#�F�A�.�G��G�|�� ����<��Z�&�0�3H�H�B�C�D� D�r?c��|jd�d}|jd�\}}}|r@|rtd��|jd�\}}}|r1|jd�s td��|jd�\}}}t |�y)Nr��r��Invalid IPv6 URLr�r�)r�r�r�� startswith�_check_bracketed_host)r��hostname_and_port�before_bracketr�r�r�r�r�s r@�_check_bracketed_netlocr��s����)�)�#�.�q�1��.?�.I�.I�#�.N�+�N�L�)����/�0�0�%�/�/��4���!�T������,��/�0�0�-�7�7��<���!�T��(�#r?c���|jd�r"tjd|�std��yt j |�}t |tj�rtd��y)N�vz\Av[a-fA-F0-9]+\..+\ZzIPvFuture address is invalidz%An IPv4 address cannot be in brackets)r��re�matchr�� ipaddress� ip_addressr[�IPv4Address)r��ips r@r�r��sc�����3���x�x�0�(�;��;�=�=�<�� !� !�(� +���b�)�/�/�0��D�F�F�1r?)�typedc��t||�\}}}|jt�}|jt�}tD]&}|j|d�}|j|d�}�(t |�}dx}x}}|jd�}|dkDrU|dj�rB|dj�r/|d|D]} | tvs�n|d|j�||dzd}}|dddk(r=t|d�\}}d|vrd |vsd |vrd|vrtd ��d|vrd |vrt|�|rd|vr|jdd�\}}d|vr|jdd�\}}t!|�t#|||||�} || �S) a�Parse a URL into 5 components: <scheme>://<netloc>/<path>?<query>#<fragment> The result is a named 5-tuple with fields corresponding to the above. It is either a SplitResult or SplitResultBytes object, depending on the type of the url parameter. The username, password, hostname, and port sub-components of netloc can also be accessed as attributes of the returned object. The scheme argument provides the default value of the scheme component when no scheme is found in url. If allow_fragments is False, no attempt is made to separate the fragment component from the previous component, which can be either path or query. Note that % escapes are not expanded. rr�rNrZr��//r�r�r�r�r�)r`�lstrip�_WHATWG_C0_CONTROL_OR_SPACE�strip�_UNSAFE_URL_BYTES_TO_REMOVEr��boolr�r��isalpha�scheme_charsr�r�r�r��splitr�r)r�r�r�r��br�r�r�r�r�r�s r@rr�s���,#/�s�F�";��C����*�*�0� 1�C� �\�\�5� 6�F� (���k�k�!�R� ������2�&��)��?�+�O� "�"�F�"�U�X����� �A��1�u��Q����!�c�!�f�n�n�&6��R�a��A���$����b�q�'�-�-�/�3�q��s�t�9�C�F� �2�A�w�$��"�3��*���� �F�]�s�&�0����3�f�#4��/�0�0��&�=�S�F�]�#�F�+��3�#�:�� � �#�q�)� ��X� �c�z��Y�Y�s�A�&� ��U�����F�F�C���9�A��!��r?c �`�t|�\}}}}}}}|r|�d|��}|t|||||f��S)z�Put a parsed URL back together again. This may result in a slightly different, but equivalent URL, if the URL that was parsed originally had redundant delimiters, e.g. a ? with an empty query (the draft states that these are equivalent).r�)r`r )� componentsr�r�r�r�r�r�r�s r@rr sH��3?� �2K�A�F�F�C����.� ��f�%���*�f�f�c�5�(�%K�L�M�Mr?c���t|�\}}}}}}|r|r |dddk7rd|z}d|z|z}n'|dddk(rd|z}n|r|tvr|r|dddk(rd|z}|r|dz|z}|r|dz|z}|r|dz|z}||�S) akCombine the elements of a tuple as returned by urlsplit() into a complete URL as a string. The data argument can be any five-item iterable. This may result in a slightly different, but equivalent URL, if the URL that was parsed originally had unnecessary delimiters (for example, a ? with an empty query; the RFC states that these are equivalent).NrZr�rr�r�r�r�)r`�uses_netloc)rr�r�r�r�r�r�s r@r r s���+7� �*C�9�F�F�C���.� ��3�r��7�c�>��s��3��V�m�c�!�� �R�a��D���S�j�� �F�k�)�3�#�b�q�'�S�.��S�j�� ��s�l�S� ����C�i�%�����C�i�(�"���#��r?c ��|s|S|s|St||�\}}}t|d|�\}}}}}} t|||�\} }}} }}| |k7s| tvr||�S| tvr|r|t | ||| ||f��S|}|s!| s|}|} |s|}|t | ||| ||f��S|jd�}|ddk7r|d=|dddk(r|jd�}n&||jd�z}t d|dd�|ddg}|D]0}|dk(r |j��|dk(r� |j|��2|ddvr|jd�|t | |dj|�xsd| ||f��S#t$rY��wxYw) zaJoin a base URL and a possibly relative URL to form an absolute interpretation of the latter.rr����NrZ�..�.)rr)r`r� uses_relativerrr�filter�pop� IndexError�append�join)�baser�r�r��bscheme�bnetloc�bpath�bparams�bquery� bfragmentr�r��pathr�r�r�� base_parts�segments� resolved_path�segs r@rr/s���� ���� ,�T�3� 7��D�#�~��T�2��/�8�G�W�e�W�f�i� �S�'�?�3�2�F�F�D�&�%�����F�-�7��c�"�"� ����!�*�f�f�d�.4�e�X�.G�#H�I� I�����������E��j�&�&�$�*0�%��*C�D�E� E����S�!�J��"�~��� �r�N��B�Q�x�3���:�:�c�?���� � �3��/�� ��h�q��n�5���2���M����$�;� ��!�!�#� �C�Z��� � ��%����|�{�"� ���R� ��*�f�f�c�h�h��7�7��v�u�h�&8�9�:�:��� �� �s�E1�1 E=�<E=c��t|�\}}d|vr$t|�\}}}}}}t|||||df�}nd}|}|t||��S)z�Removes any existing fragment from URL. Returns a tuple of the defragmented URL and the fragment. If the URL contained no fragments, the second element is the empty string. r�r)r`rrr) r�r��sr��p�a�q�frag�defrags r@rrtsc��'�s�+��C�� �c�z�&�s�m���1�a��A�t��Q��1�a��B�/�0�������,�v�t�4�5�5r?�0123456789ABCDEFabcdefc�*�tt|��S)z,unquote_to_bytes('abc%20def') -> b'abc def'.)�bytes� _unquote_impl)�strings r@rr�s����v�&�'�'r?r3�returnc ��|s |jyt|t�r|jd�}|jd�}t |�dk(r|St|d�}|j}t�ItD��cic]7}tD],}||zj�tj||z���.�9c}}a|ddD] } |t|dd�||dd��"|Scc}}w#t$r|d�||�Y�FwxYw)Nr?�utf-8r�rZrr�)rr[r\rJr�� bytearray�extend� _hextobyte�_hexdigr1�fromhex�KeyError)r3�bits�resrr+r �items r@r2r2�s ��������&�#�����w�'���<�<���D� �4�y�A�~�� � �D��G� �C� �Z�Z�F���&�9�&�!��1��1�u�n�n�&�� � �a�!�e�(<�<�07�'�&�9� ��Q�R��� ��:�d�2�A�h�'�(��4���8����J��9��� ��4�L��4�L� �s�;<C%�C+�+D�Dz([-]+)c#��K�d}tj|�D]>}|j�\}}|||��t|d�j ||���|}�@||d��y�w)NrrZ)�_asciire�finditer�spanr2rQ)r3rKrL�previous_match_end�ascii_matchr��ends r@�_generate_unquoted_partsrG�st�������(�(��0�� �%�%�'� ��s��'��.�.��K��N�+�2�2�8�V�D�D� ��1��#�$� %�%�s�A!A#c���t|t�rt|�j||�Sd|vr|j|S|�d}|�d}djt |||��S)a�Replace %xx escapes by their single-character equivalent. The optional encoding and errors parameters specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. By default, percent-encoded sequences are decoded with UTF-8, and invalid sequences are replaced by a placeholder character. unquote('abc%20def') -> 'abc def'. r�r6r�r)r[r1r2rQrrrG�r3rKrLs r@rr�sh���&�%� ��V�$�+�+�H�f�=�=� �&������ ����� �~��� �7�7�+�F�H�f�E�F�Fr?Fc ��i}t|||||||��}|D]$\} } | |vr|| j| ��| g|| <�&|S)aXParse a query given as a string argument. Arguments: qs: percent-encoded query string to be parsed keep_blank_values: flag indicating whether blank values in percent-encoded queries should be treated as blank strings. A true value indicates that blanks should be retained as blank strings. The default false value indicates that blank values are to be ignored and treated as if they were not included. strict_parsing: flag indicating what to do with parsing errors. If false (the default), errors are silently ignored. If true, errors raise a ValueError exception. encoding and errors: specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. max_num_fields: int. If set, then throws a ValueError if there are more than n fields read by parse_qsl(). separator: str. The symbol to use for separating the query arguments. Defaults to &. Returns a dictionary. )rKrL�max_num_fieldsr�)rr)�qs�keep_blank_values�strict_parsingrKrLrKr�� parsed_result�pairs�name�values r@rr�sd��<�M��b�+�^�'��%3�y� J�E����e��=� ��$��&�&�u�-�#(�'�M�$�� � �r?c�X���|rt|ttf�std��t|t�r%t|t�st|d�}d}��fd�}n9|sgStt |��}t|t�rt|d�}d}d�}|sgS|�$d|j|�z} || krtd��g} |j |�D]V}|s|s�|j|�\}} }| s|rtd |����|s|s�4||�}||�}| j||f��X| S) aXParse a query given as a string argument. Arguments: qs: percent-encoded query string to be parsed keep_blank_values: flag indicating whether blank values in percent-encoded queries should be treated as blank strings. A true value indicates that blanks should be retained as blank strings. The default false value indicates that blank values are to be ignored and treated as if they were not included. strict_parsing: flag indicating what to do with parsing errors. If false (the default), errors are silently ignored. If true, errors raise a ValueError exception. encoding and errors: specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. max_num_fields: int. If set, then throws a ValueError if there are more than n fields read by parse_qsl(). separator: str. The symbol to use for separating the query arguments. Defaults to &. Returns a list, as G-d intended. z*Separator must be of type string or bytes.rB�=c� ��t|����S)N)rKrL)r)r)rKrLs ��r@�_unquotezparse_qsl.<locals>._unquotes�����H�V�D�Dr?�=c�8�t|jdd��S)N�+� )rr�)r)s r@rVzparse_qsl.<locals>._unquote$s��#�A�I�I�d�D�$9�:�:r?rZzMax number of fields exceededzbad query field: ) r[r\r1r�� memoryview�countrr�r)rLrMrNrKrLrKr��eqrV� num_fields�r� name_valuerQ�has_eqrRs `` r@rr�s5���<�J�y�3��,�?��E�F�F��"�c���)�S�)��I�w�/�I� �� E���I��:�b�>� "���i��%��i��1�I� �� ;��� � �!�����)�,�,� ��J�&��<�=�=� �A��h�h�y�)� ���",�"6�"6�r�":��D�&�%��n� �*�!F�G�G��)���~�� �������$���'�*� �Hr?c�@�|jdd�}t|||�S)z�Like unquote(), but also replace plus signs by spaces, as required for unquoting HTML form values. unquote_plus('%7e/abc+def') -> '~/abc def' �+� )r�rrIs r@rr>s#���^�^�C�� %�F��6�8�V�,�,r?sBABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_.-~c�z�|dk(r"tjdtd��tSt dt �d|����)N�QuoterzoDeprecated in 3.11. urllib.parse.Quoter will be removed in Python 3.14. It was not intended to be a public API.r��� stacklevelzmodule z has no attribute )�warnings�warn�DeprecationWarning�_Quoter�AttributeErrorrl)rQs r@�__getattr__rnMsC���x��� � �@�)�Q� 8��� �7�8�,�.@���I� J�Jr?c�"�eZdZdZd�Zd�Zd�Zy)rlz�A mapping from bytes numbers (in range(0,256)) to strings. String values are percent-encoded byte values, unless the key < 128, and in either of the specified safe set, or the always safe set. c�8�tj|�|_y)zsafe: bytes object.N)�_ALWAYS_SAFE�union�safe)rirss r@�__init__z_Quoter.__init__^s�� �&�&�t�,�� r?c� �dt|��d�S)Nz<Quoter �>)�dictr|s r@�__repr__z_Quoter.__repr__bs���$�t�*��q�)�)r?c�d�||jvrt|�ndj|�}|||<|S)Nz%{:02X})rs�chr�format)rir r>s r@�__missing__z_Quoter.__missing__es0���T�Y�Y��c�!�f�I�,<�,<�Q�,?����Q��� r?N)rlrmrnrortrxr|r>r?r@rlrlVs���-�*�r?rlc��t|t�r|s|S|�d}|�d}|j||�}n|�td��|�td��t ||�S)a�quote('abc def') -> 'abc%20def' Each part of a URL, e.g. the path info, the query, etc., has a different set of reserved characters that must be quoted. The quote function offers a cautious (not minimal) way to quote a string for most of these parts. RFC 3986 Uniform Resource Identifier (URI): Generic Syntax lists the following (un)reserved characters. unreserved = ALPHA / DIGIT / "-" / "." / "_" / "~" reserved = gen-delims / sub-delims gen-delims = ":" / "/" / "?" / "#" / "[" / "]" / "@" sub-delims = "!" / "$" / "&" / "'" / "(" / ")" / "*" / "+" / "," / ";" / "=" Each of the reserved characters is reserved in some component of a URL, but not necessarily in all of them. The quote function %-escapes all characters that are neither in the unreserved chars ("always safe") nor the additional chars set via the safe arg. The default for the safe arg is '/'. The character is reserved, but in typical usage the quote function is being called on a path where the existing slash characters are to be preserved. Python 3.7 updates from using RFC 2396 to RFC 3986 to quote URL strings. Now, "~" is included in the set of unreserved characters. string and safe may be either str or bytes objects. encoding and errors must not be specified if string is a bytes object. The optional encoding and errors parameters specify how to deal with non-ASCII characters, as accepted by the str.encode method. By default, encoding='utf-8' (characters are encoded with UTF-8), and errors='strict' (unsupported characters raise a UnicodeEncodeError). r6rCz,quote() doesn't support 'encoding' for bytesz*quote() doesn't support 'errors' for bytes)r[r\rJr]r)r3rsrKrLs r@r r ksm��N�&�#����M����H��>��F����x��0�����J�K�K����H�I�I��F�D�)�)r?c���t|t�rd|vst|t�rd|vrt||||�St|t�rd}nd}t|||z||�}|j dd�S)z�Like quote(), but also replace ' ' with '+', as required for quoting HTML form values. Plus signs in the original string are escaped unless they are included in safe. It also does not have safe default to '/'. rdrZrc)r[r\r1r r�)r3rsrKrL�spaces r@rr�sp�� �F�C� �S��%6� �F�E� "�t�6�'9��V�T�8�V�4�4��$������� �6�4�%�<��6� :�F��>�>�#�s�#�#r?c�,�t|�jSrE)rl�__getitem__)rss r@r=r=�s���4�=�$�$�$r?c�\�t|ttf�std��|syt|t�r|jdd�}nt|D�cgc] }|dks� |��c}�}|j t|z�s|j�St|�}t|�x}dkrdjt||��Stj|�}td||�D�cgc]#}djt|||||z����%}}dj|�Scc}wcc}w)z�Like quote(), but accepts a bytes object rather than a str, and does not perform string-to-bytes encoding. It always returns an ASCII string. quote_from_bytes(b'abc def?') -> 'abc%20def%3f' z!quote_from_bytes() expected bytesrrB�ignore�i@ r)r[r1r7r]r\rJ�rstrip�_ALWAYS_SAFE_BYTESrQr=r�r�map�math�isqrt�range)�bsrsr��quoter�bs_len� chunk_sizer��chunkss r@rr�s�� �b�5�)�,�-��;�<�<� ���$����{�{�7�H�-����1��A��S��a��1�2�� �9�9�'�$�.�/��y�y�{�� !�$� '�F��b�'���W�$��w�w�s�6�2��'�'��Z�Z��'� � ��F�J�7�9�7�a��'�'�#�f�b��1�Z�<�&8�9�:�7� �9��w�w�v����2��9s� D$�D$�)(D)c�D�t|d�r|j�}n% t|�rt|dt�st �g}|s~|D]x\}} t|t�r |||�}n|t|�|||�}t| t�r || |�} n|t| �|||�} |j|dz| z��zn�|D]�\}} t|t�r |||�}n|t|�|||�}t| t�r!|| |�} |j|dz| z��et| t�r#|| |||�} |j|dz| z��� t| �} | D]G}t|t�r |||�}n|t|�|||�}|j|dz|z��I��dj|�S#t $r}td�|�d}~wwxYw#t $r/|t| �|||�} |j|dz| z�Y��WwxYw)a^Encode a dict or sequence of two-element tuples into a URL query string. If any values in the query arg are sequences and doseq is true, each sequence element is converted to a separate parameter. If the query arg is a sequence of two-element tuples, the order of the parameters in the output will match the order of parameters in the input. The components of a query arg may each be either a string or a bytes type. The safe, encoding, and errors parameters are passed down to the function specified by quote_via (encoding and errors only if a component is a str). �itemsrz1not a valid non-string sequence or mapping objectNrT�&) �hasattrr�r�r[rVr]r1r\rr)r��doseqrsrKrL� quote_via�err�l�kr�rT�elts r@r r �s��"�u�g����� �� :��5�z�*�U�1�X�u�"=��� �A���D�A�q��!�U�#��a��&���c�!�f�d�H�f�=���!�U�#��a��&���c�!�f�d�H�f�=�� �H�H�Q��W�q�[�!���D�A�q��!�U�#��a��&���c�!�f�d�H�f�=���!�U�#��a��&������S��1��%��A�s�#��a��x��8������S��1��%�0��A��A� !��%�c�5�1�"+�C��"6�C�"+�C��H�d�H�f�"M�C�����S��3��/� !�-�8�8�8�A�;���]� :��0�1�69� :�� :��D!�*�!�#�a�&�$��&�A�A��H�H�Q��W�q�[�)�*�s)�$G � G'� G$�G�G$�'4H�Hc�P�tjdtd��t|�S)Nz/urllib.parse.to_bytes() is deprecated as of 3.8r�rg)rirjrk� _to_bytes�r�s r@�to_bytesr�"s ���M�M�C�$��4��S�>�r?c��t|t�r" |jd�j�}|S|S#t$rt dt|�zdz��wxYw)zto_bytes(u"URL") --> 'URL'.�ASCIIzURL z contains non-ASCII characters)r[r\rJrQ�UnicodeError�reprr�s r@r�r�(sl�� �#�s�� A��*�*�W�%�,�,�.�C��J�3�J��� A��v��S� �1�?� @�A� A� A�s �5�$Ac��t|�j�}|dddk(r|dddk(r|ddj�}|dddk(r|ddj�}|S)z�Transform a string like '<URL:scheme://host/path>' into 'scheme://host/path'. The string is returned unchanged if it's not a wrapped URL. NrZ�<rrv�zURL:)r\rr�s r@�unwrapr�6sc�� �c�(�.�.� �C� �2�A�w�#�~�#�b�c�(�c�/��!�B�i�o�o��� �2�A�w�&���!�"�g�m�m�o���Jr?c�P�tjdtd��t|�S)NzUurllib.parse.splittype() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splittyper�s r@� splittyper�C�$���M�M�8�$��4��c�?�r?c���t�$tjdtj�atj |�}|r%|j�\}}|j �|fSd|fS)z:splittype('type:opaquestring') --> 'type', 'opaquestring'.Nz ([^/:]+):(.*))� _typeprogr��compile�DOTALLr��groupsr�)r�r�r��datas r@r�r�KsY�����J�J��� � �:� ��O�O�C� �E���|�|�~�����|�|�~�t�#�#���9�r?c�P�tjdtd��t|�S)NzUurllib.parse.splithost() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splithostr�s r@� splithostr�Xr�r?c���t�$tjdtj�atj |�}|r&|j�\}}|r |ddk7rd|z}||fSd|fS)z;splithost('//host[:port]/path') --> 'host[:port]', '/path'.Nz//([^/#?]*)(.*)rr�)� _hostprogr�r�r�r�r�)r�r�� host_portr#s r@r�r�`si�����J�J�0�"�)�)�<� ��O�O�C� �E���,�,�.�� �4��D��G�s�N���:�D��$�����9�r?c�P�tjdtd��t|�S)NzUurllib.parse.splituser() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splituser��hosts r@� splituserr�o�%���M�M�8�$��4��d��r?c�@�|jd�\}}}|r||fSd|fS)zJsplituser('user[:passwd]@host[:port]') --> 'user[:passwd]', 'host[:port]'.r�N�r�)r��userr�s r@r�r�vs.������,��D�%���D�d�*�*�t�d�*�*r?c�P�tjdtd��t|�S)NzWurllib.parse.splitpasswd() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk�_splitpasswd)r�s r@�splitpasswdr�|s%���M�M�8�$��4����r?c�>�|jd�\}}}||r|fSdfS)z/splitpasswd('user:passwd') -> 'user', 'passwd'.r�N�r�)r�r��passwds r@r�r��s,���.�.��-��D�%���E�&�,�,�t�,�,r?c�P�tjdtd��t|�S)NzUurllib.parse.splitport() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splitportr�s r@� splitportr��r�r?c��t�$tjdtj�atj |�}|r|j�\}}|r||fS|dfS)z*splitport('host:port') --> 'host', 'port'.Nz (.*):([0-9]*))� _portprogr�r�r�� fullmatchr�)r�r�r�s r@r�r��sW�����J�J��� � �:� �����%�E���\�\�^� ��d����:����:�r?c�R�tjdtd��t||�S)NzVurllib.parse.splitnport() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk�_splitnport)r��defports r@� splitnportr��s'���M�M�8�$��4��t�W�%�%r?c��|jd�\}}}|s|}||fS|r5|j�r|j�rt|�}||fSd}||fS||fS)z�Split host and port, returning numeric port. Return given default port if no ':' found; defaults to -1. Return numerical port if a valid number is found after ':'. Return None if ':' but not a valid number.r�N)r�r�r�r�)r�r�r�r��nports r@r�r��ss�� ����,��D�%�������=�� ��<�<�>�d�l�l�n���I�E��U�{���E��U�{����=�r?c�P�tjdtd��t|�S)NzVurllib.parse.splitquery() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk�_splitqueryr�s r@� splitqueryr��s%���M�M�8�$��4��s��r?c�@�|jd�\}}}|r||fS|dfS)z/splitquery('/path?query') --> '/path', 'query'.r�Nr�)r�r#r�r�s r@r�r��s.������,��D�%����U�{����9�r?c�P�tjdtd��t|�S)NzTurllib.parse.splittag() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splittagr�s r@�splittagr��s$���M�M�8�$��4��S�>�r?c�@�|jd�\}}}|r||fS|dfS)z)splittag('/path#tag') --> '/path', 'tag'.r�Nr�)r�r#r��tags r@r�r��s.���~�~�c�*��D�%����S�y����9�r?c�P�tjdtd��t|�S)NzUurllib.parse.splitattr() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splitattrr�s r@� splitattrr��r�r?c�8�|jd�}|d|ddfS)zksplitattr('/path;attr1=value1;attr2=value2;...') -> '/path', ['attr1=value1', 'attr2=value2', ...].r�rrZN)r)r��wordss r@r�r��s&�� �I�I�c�N�E���8�U�1�2�Y��r?c�P�tjdtd��t|�S)NzWurllib.parse.splitvalue() is deprecated as of 3.8, use urllib.parse.parse_qsl() insteadr�rg)rirjrk�_splitvalue)�attrs r@� splitvaluer��s%���M�M�9�$��4��t��r?c�>�|jd�\}}}||r|fSdfS)z-splitvalue('attr=value') --> 'attr', 'value'.rTNr�)r�r�rRs r@r�r��s,������,��D�%���5�%�+�+�d�+�+r?)rT)r)T)r6r�)FFr6r�Nr�)r�NN)rNN)r�)r)qro�collectionsr� functoolsr�r�r�rir��__all__rrr��non_hierarchical� uses_query� uses_fragmentrrrrA�_implicit_encoding�_implicit_errorsrGrMrXr`�objectrbrsrxr�r��_DefragResultBase�_SplitResultBase�_ParseResultBaser�r�r�r�r#r�r�� ResultBaserrrrrrr�rr�r�r�r�r�� lru_cacherrr rrr:r9rr1r7r\r2r�rArGrrrr� frozensetrqr�rnrwrlr rr=rr r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r>r?r@�<module>r�s����B#��� ����H��0� � -��-��F��A� �%� � ��\��1��'������"4�/�(�!3�.�H�2�"V�f�V�V��V�#8�V�#8�L�2�O��<�4�6G��<�~�~�>����6�8����=�?�����!K�����&����"����#N�����#�����!�����"�����%����!� ���#3�"9�"9�"A�"A�����"2�"9�"9�"A�"A����� 0� 5� 5� =� =�����#����� "2�!7�!7�!?�!?�����$4�$=�$=�$E�$E����!�#� ��$�o�� �"�$9� � �"�$9� ��)�+<�� �'�)@� � �'�)@� �1����"�<�)�D�$$�&G�����4� �5�!�5�n N��.B:�J6� #�� � �(��%�)�+�c�1��e�i�6G��:�2�:�:�&�'��&�G�.:?�PS�'�T;@�QT�G �R-��!�"���<�(��K��d��*4*�l$�$���%��%��6!�r�D��"�M�`�� �� � � �� � ���+��-�� � ��&��"�������,r?PKS�\i�bp�p�'__pycache__/parse.cpython-312.opt-2.pycnu�[���� T��hl����� ddlmZddlZddlZddlZddlZddlZddlZgd�Zgd�Z gd�Z gd�Zgd�Zgd�Z gd �Zd ZdZgd�Zd �ZdZdZd�Zeefd�Zeefd�Zd�ZGd�de�ZGd�de�ZGd�de�ZGd�dee�ZGd�dee�Zedd�Zed d!�Z ed"d#�Z!d$e_"d%ejF_"d&ejH_"d'e _"d(e jJ_"d)e jL_"d*e jN_"d+e jP_"d,e jH_"d-e!_"e jJjDe!jJ_"e jLjDe!jL_"e jNjDe!jN_"d.e!jR_"e jPjDe!jP_"e jHjDe!jH_"eZ*Gd/�dee�Z+Gd0�d e e�Z,Gd1�d"e!e�Z-Gd2�d3ee�Z.Gd4�d5e e�Z/Gd6�d7e!e�Z0d8�Z1e1�[1dsd;�Z2d<�Z3dtd=�Z4d>�Z5d?�Z6d@�Z7ejpd:�A�dsdB��Z9dC�Z:dD�Z;dudE�Z<dF�Z=dGZ>da?dH�Z@dIeAeBzeCzdJeAeBzfdK�ZDej�dL�ZFdM�ZGdvdN�ZH dwdP�ZI dwdQ�ZJdvdR�ZKeLdS�ZMeAeM�ZNdT�ZOGdU�dVeP�ZQdxdW�ZRdydX�ZSejpdY��ZTdzdZ�ZUdOd9ddeSfd[�ZVd\�ZWd]�ZXd^�ZYd_�ZZda[d`�Z\da�Z]da^db�Z_dc�Z`dd�Zade�Zbdf�Zcdg�Zddaedh�Zfd{di�Zgd{dj�Zhdk�Zidl�Zjdm�Zkdn�Zldo�Zmdp�Zndq�Zodr�Zpy)|�)� namedtupleN)�urlparse� urlunparse�urljoin� urldefrag�urlsplit� urlunsplit� urlencode�parse_qs� parse_qsl�quote� quote_plus�quote_from_bytes�unquote�unquote_plus�unquote_to_bytes�DefragResult�ParseResult�SplitResult�DefragResultBytes�ParseResultBytes�SplitResultBytes)��ftp�http�gopher�nntp�imap�wais�file�https�shttp�mms�prospero�rtsp�rtsps�rtspu�sftp�svn�svn+ssh�ws�wss)rrrrr�telnetrrr r#r!r"�snewsr$r%r&r'�rsyncr)r*r(�nfs�gitzgit+sshr+r,z itms-services)rr�hdlr$rrr!r"r%r&r'�sip�sipsr#r(�tel) rr2�mailto�newsr-rrr.r3r4) rrrrr!r"r#rr%r&r'r3r4) rrr2rrr7rrr!r"r.r r$zAabcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+-.z! )� � � c�V� tj�tj�y�N)r�cache_clear�_byte_quoter_factory���%/usr/lib64/python3.12/urllib/parse.py�clear_cacherB^s��N������$�$�&r@�ascii�strictc��|Sr<r?)�objs rA�_nooprGls���Jr@c�&�|j||�Sr<��encode)rF�encoding�errorss rA�_encode_resultrMos���:�:�h��'�'r@c�0���t��fd�|D��S)Nc3�J�K�|]}|r|j���nd���y�w)rN��decode��.0�xrKrLs ��rA� <genexpr>z_decode_args.<locals>.<genexpr>us$�����G�$�Q�q����(�F�+�b�8�$�s� #)�tuple)�argsrKrLs ``rA�_decode_argsrXss����G�$�G�G�Gr@c��t|dt�}|ddD]#}|s�t|t�|k7s�td��|r |tfzSt |�t fzS)Nr�z$Cannot mix str and non-str arguments)� isinstance�str� TypeErrorrGrXrM)rW� str_input�args rA�_coerce_argsr`wsg���4��7�C�(�I��A�B�x���:�c�3�'�9�4��B�C�C� � ��u�h������� 1�1�1r@c��eZdZ dZdd�Zy)�_ResultMixinStrr?c�8���|j��fd�|D��S)Nc3�B�K�|]}|j������y�wr<rIrRs ��rArUz)_ResultMixinStr.encode.<locals>.<genexpr>�������*T�t�!�1�8�8�H�f�+E�t���)�_encoded_counterpart��selfrKrLs ``rArJz_ResultMixinStr.encode�����(�t�(�(�*T�t�*T�U�Ur@N�rCrD)�__name__� __module__�__qualname__� __slots__rJr?r@rArbrb����H��I�Vr@rbc��eZdZ dZdd�Zy)�_ResultMixinBytesr?c�8���|j��fd�|D��S)Nc3�B�K�|]}|j������y�wr<rPrRs ��rArUz+_ResultMixinBytes.decode.<locals>.<genexpr>�rerf)�_decoded_counterpartrhs ``rArQz_ResultMixinBytes.decode�rjr@Nrk)rlrmrnrorQr?r@rArrrr�rpr@rrc�v�eZdZ dZed��Zed��Zed��Zed��Ze e j�Zy)�_NetlocResultMixinBaser?c� �|jdS)Nr�� _userinfo�ris rA�usernamez_NetlocResultMixinBase.username�����~�~�a� � r@c� �|jdS)NrZryr{s rA�passwordz_NetlocResultMixinBase.password�r}r@c��|jd}|syt|t�rdnd}|j|�\}}}|j �|z|zS)Nr�%�%)� _hostinfor[r\� partition�lower)ri�hostname� separator�percent�zones rAr�z_NetlocResultMixinBase.hostname�sV���>�>�!�$����&�h��4�C�$� �"*�"4�"4�Y�"?���'�4��~�~��'�)�D�0�0r@c���|jd}|�\|j�r|j�rt|�}nt d|����d|cxkrdkst d��t d��|S)NrZz+Port could not be cast to integer value as ri��zPort out of range 0-65535)r��isdigit�isascii�int� ValueError)ri�ports rAr�z_NetlocResultMixinBase.port�sr���~�~�a� �����|�|�~�$�,�,�.��4�y�� �#N�t�h�!W�X�X���&��&� �!<�=�=�'� �!<�=�=��r@N) rlrmrnro�propertyr|rr�r��classmethod�types�GenericAlias�__class_getitem__r?r@rArwrw�sk��R��I� �!��!��!��!��1��1�� �� �$�E�$6�$6�7�r@rwc�0�eZdZdZed��Zed��Zy)�_NetlocResultMixinStrr?c��|j}|jd�\}}}|r|jd�\}}}|sd}||fSdx}}||fS)N�@�:��netloc� rpartitionr��rir��userinfo� have_info�hostinfor|� have_passwordrs rArzz_NetlocResultMixinStr._userinfo�sh������(.�(9�(9�#�(>�%��)�X��08�0B�0B�3�0G�-�H�m�X� �����!�!�#'�&�H�x���!�!r@c��|j}|jd�\}}}|jd�\}}}|r+|jd�\}}}|jd�\}}}n|jd�\}}}|sd}||fS)Nr��[�]r�r��rir��_r��have_open_br� bracketedr�r�s rAr�z_NetlocResultMixinStr._hostinfo�s��������*�*�3�/���1�h�%-�%7�%7��%<�"��<��� )� 3� 3�C� 8��H�a������,�J�A�q�$� (� 2� 2�3� 7��H�a����D���~�r@N�rlrmrnror�rzr�r?r@rAr�r���-���I� � "�� "����r@r�c�0�eZdZdZed��Zed��Zy)�_NetlocResultMixinBytesr?c��|j}|jd�\}}}|r|jd�\}}}|sd}||fSdx}}||fS)N�@�:r�r�s rArzz!_NetlocResultMixinBytes._userinfo�sh������(.�(9�(9�$�(?�%��)�X��08�0B�0B�4�0H�-�H�m�X� �����!�!�#'�&�H�x���!�!r@c��|j}|jd�\}}}|jd�\}}}|r+|jd�\}}}|jd�\}}}n|jd�\}}}|sd}||fS)Nr��[�]r�r�r�s rAr�z!_NetlocResultMixinBytes._hostinfo�s��������*�*�4�0���1�h�%-�%7�%7��%=�"��<��� )� 3� 3�D� 9��H�a������-�J�A�q�$� (� 2� 2�4� 8��H�a����D���~�r@Nr�r?r@rAr�r��r�r@r�rzurl fragmentrz!scheme netloc path query fragmentrz(scheme netloc path params query fragmentz� DefragResult(url, fragment) A 2-tuple that contains the url without fragment identifier and the fragment identifier as a separate argument. z$The URL with no fragment identifier.z� Fragment identifier separated from URL, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. z� SplitResult(scheme, netloc, path, query, fragment) A 5-tuple that contains the different components of a URL. Similar to ParseResult, but does not split params. z%Specifies URL scheme for the request.z0 Network location where the request is made to. z@ The hierarchical path, such as the path to a file to download. z� The query component, that contains non-hierarchical data, that along with data in path component, identifies a resource in the scope of URI's scheme and network location. z� Fragment identifier, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. zq ParseResult(scheme, netloc, path, params, query, fragment) A 6-tuple that contains components of a parsed URL. z� Parameters for last path element used to dereference the URI in order to provide access to perform some operation on the resource. c��eZdZdZd�Zy)rr?c�j�|jr|jdz|jzS|jS�N�#��fragment�urlr{s rA�geturlzDefragResult.geturlEs)���=�=��8�8�c�>�D�M�M�1�1��8�8�Or@N�rlrmrnror�r?r@rArrC����I�r@c��eZdZdZd�Zy)rr?c��t|�Sr<�r r{s rAr�zSplitResult.geturlM� ���$��r@Nr�r?r@rArrK����I� r@c��eZdZdZd�Zy)rr?c��t|�Sr<�rr{s rAr�zParseResult.geturlRr�r@Nr�r?r@rArrPr�r@c��eZdZdZd�Zy)rr?c�j�|jr|jdz|jzS|jS)N�#r�r{s rAr�zDefragResultBytes.geturlXs)���=�=��8�8�d�?�T�]�]�2�2��8�8�Or@Nr�r?r@rArrVr�r@rc��eZdZdZd�Zy)rr?c��t|�Sr<r�r{s rAr�zSplitResultBytes.geturl`r�r@Nr�r?r@rArr^r�r@rc��eZdZdZd�Zy)rr?c��t|�Sr<r�r{s rAr�zParseResultBytes.geturler�r@Nr�r?r@rArrcr�r@rc�z�ttfttftt ff}|D]\}}||_||_�yr<)rrrrrrrgru)� _result_pairs�_decoded�_encodeds rA�_fix_result_transcodingr�isE�� �(�)� �&�'� �&�'��M� ,���(�(0��%�(0��%�,r@rTc�� t||�\}}}t|||�}|\}}}}}|tvrd|vrt|�\}}nd}t ||||||�} || �S)N�;r)r`r�uses_params�_splitparamsr) r��scheme�allow_fragments�_coerce_result�splitresultr��queryr��params�results rArrvsz���&#/�s�F�";��C����3���8�K�+6�(�F�F�C��� ������"�3�'���V��� ����f�e�X� F�F��&�!�!r@c��d|vr*|jd|jd��}|dkr|dfS|jd�}|d|||dzdfS)N�/r�rrrZ)�find�rfind)r��is rAr�r��sY�� �s�{��H�H�S�#�)�)�C�.�)���q�5���7�N��H�H�S�M���r��7�C��!���I��r@c��t|�}dD]&}|j||�}|dk\s�t||�}�(|||||dfS)Nz/?#r)�lenr��min)r��start�delim�c�wdelims rA�_splitnetlocr��sR����H�E� �����!�U�#���Q�;���v�&�E���u�U��S���[�(�(r@c�0�|r|j�ryddl}|jdd�}|jdd�}|jdd�}|jdd�}|jd|�}||k(rydD]}||vs�t d |zd zdz��y)Nrr�rr�r��?�NFKCz/?#@:znetloc 'z' contains invalid z#characters under NFKC normalization)r��unicodedata�replace� normalizer�)r�r��n�netloc2r�s rA�_checknetlocr��s����V�^�^�%������s�B��A� � � �#�r��A� � � �#�r��A� � � �#�r��A��#�#�F�A�.�G��G�|�� ����<��Z�&�0�3H�H�B�C�D� D�r@c��|jd�d}|jd�\}}}|r@|rtd��|jd�\}}}|r1|jd�s td��|jd�\}}}t |�y)Nr��r��Invalid IPv6 URLr�r�)r�r�r�� startswith�_check_bracketed_host)r��hostname_and_port�before_bracketr�r�r�r�r�s rA�_check_bracketed_netlocr��s����)�)�#�.�q�1��.?�.I�.I�#�.N�+�N�L�)����/�0�0�%�/�/��4���!�T������,��/�0�0�-�7�7��<���!�T��(�#r@c���|jd�r"tjd|�std��yt j |�}t |tj�rtd��y)N�vz\Av[a-fA-F0-9]+\..+\ZzIPvFuture address is invalidz%An IPv4 address cannot be in brackets)r��re�matchr�� ipaddress� ip_addressr[�IPv4Address)r��ips rAr�r��sc�����3���x�x�0�(�;��;�=�=�<�� !� !�(� +���b�)�/�/�0��D�F�F�1r@)�typedc�� t||�\}}}|jt�}|jt�}tD]&}|j|d�}|j|d�}�(t |�}dx}x}}|jd�}|dkDrU|dj�rB|dj�r/|d|D]} | tvs�n|d|j�||dzd}}|dddk(r=t|d�\}}d|vrd|vsd|vrd|vrtd ��d|vrd|vrt|�|rd |vr|jd d�\}}d|vr|jdd�\}}t!|�t#|||||�} || �S)Nrr�rrZr��//r�r�r�r�r�)r`�lstrip�_WHATWG_C0_CONTROL_OR_SPACE�strip�_UNSAFE_URL_BYTES_TO_REMOVEr��boolr�r��isalpha�scheme_charsr�r�r�r��splitr�r)r�r�r�r��br�r�r�r�r�r�s rArr�s����(#/�s�F�";��C����*�*�0� 1�C� �\�\�5� 6�F� (���k�k�!�R� ������2�&��)��?�+�O� "�"�F�"�U�X����� �A��1�u��Q����!�c�!�f�n�n�&6��R�a��A���$����b�q�'�-�-�/�3�q��s�t�9�C�F� �2�A�w�$��"�3��*���� �F�]�s�&�0����3�f�#4��/�0�0��&�=�S�F�]�#�F�+��3�#�:�� � �#�q�)� ��X� �c�z��Y�Y�s�A�&� ��U�����F�F�C���9�A��!��r@c �b� t|�\}}}}}}}|r|�d|��}|t|||||f��S)Nr�)r`r )� componentsr�r�r�r�r�r�r�s rArr sM��5� 3?� �2K�A�F�F�C����.� ��f�%���*�f�f�c�5�(�%K�L�M�Mr@c��� t|�\}}}}}}|r|r |dddk7rd|z}d|z|z}n'|dddk(rd|z}n|r|tvr|r|dddk(rd|z}|r|dz|z}|r|dz|z}|r|dz|z}||�S)NrZr�rr�r�r�r�)r`�uses_netloc)rr�r�r�r�r�r�s rAr r s���?�+7� �*C�9�F�F�C���.� ��3�r��7�c�>��s��3��V�m�c�!�� �R�a��D���S�j�� �F�k�)�3�#�b�q�'�S�.��S�j�� ��s�l�S� ����C�i�%�����C�i�(�"���#��r@c �� |s|S|s|St||�\}}}t|d|�\}}}}}} t|||�\} }}} }}| |k7s| tvr||�S| tvr|r|t | ||| ||f��S|}|s!| s|}|} |s|}|t | ||| ||f��S|jd�}|ddk7r|d=|dddk(r|jd�}n&||jd�z}t d|dd�|ddg}|D]0}|dk(r |j��|dk(r� |j|��2|ddvr|jd�|t | |dj|�xsd| ||f��S#t$rY��wxYw)Nrr����rZ�..�.)rr)r`r� uses_relativerrr�filter�pop� IndexError�append�join)�baser�r�r��bscheme�bnetloc�bpath�bparams�bquery� bfragmentr�r��pathr�r�r�� base_parts�segments� resolved_path�segs rArr/s��%��� ���� ,�T�3� 7��D�#�~��T�2��/�8�G�W�e�W�f�i� �S�'�?�3�2�F�F�D�&�%�����F�-�7��c�"�"� ����!�*�f�f�d�.4�e�X�.G�#H�I� I�����������E��j�&�&�$�*0�%��*C�D�E� E����S�!�J��"�~��� �r�N��B�Q�x�3���:�:�c�?���� � �3��/�� ��h�q��n�5���2���M����$�;� ��!�!�#� �C�Z��� � ��%����|�{�"� ���R� ��*�f�f�c�h�h��7�7��v�u�h�&8�9�:�:��� �� �s�E2�2 E>�=E>c�� t|�\}}d|vr$t|�\}}}}}}t|||||df�}nd}|}|t||��S)Nr�r)r`rrr) r�r��sr��p�a�q�frag�defrags rArrtsh���'�s�+��C�� �c�z�&�s�m���1�a��A�t��Q��1�a��B�/�0�������,�v�t�4�5�5r@�0123456789ABCDEFabcdefc�,� tt|��Sr<)�bytes� _unquote_impl)�strings rArr�s��6���v�&�'�'r@r3�returnc ��|s |jyt|t�r|jd�}|jd�}t |�dk(r|St|d�}|j}t�ItD��cic]7}tD],}||zj�tj||z���.�9c}}a|ddD] } |t|dd�||dd��"|Scc}}w#t$r|d�||�Y�FwxYw)Nr@�utf-8r�rZrr�)rr[r\rJr�� bytearray�extend� _hextobyte�_hexdigr1�fromhex�KeyError)r3�bits�resrr+r �items rAr2r2�s ��������&�#�����w�'���<�<���D� �4�y�A�~�� � �D��G� �C� �Z�Z�F���&�9�&�!��1��1�u�n�n�&�� � �a�!�e�(<�<�07�'�&�9� ��Q�R��� ��:�d�2�A�h�'�(��4���8����J��9��� ��4�L��4�L� �s�;<C%�C+�+D�Dz([-]+)c#��K�d}tj|�D]>}|j�\}}|||��t|d�j ||���|}�@||d��y�w)NrrZ)�_asciire�finditer�spanr2rQ)r3rKrL�previous_match_end�ascii_matchr��ends rA�_generate_unquoted_partsrG�st�������(�(��0�� �%�%�'� ��s��'��.�.��K��N�+�2�2�8�V�D�D� ��1��#�$� %�%�s�A!A#c��� t|t�rt|�j||�Sd|vr|j|S|�d}|�d}djt |||��S)Nr�r6r�r)r[r1r2rQrrrG�r3rKrLs rArr�sm����&�%� ��V�$�+�+�H�f�=�=� �&������ ����� �~��� �7�7�+�F�H�f�E�F�Fr@Fc �� i}t|||||||��}|D]$\} } | |vr|| j| ��| g|| <�&|S)N)rKrL�max_num_fieldsr�)rr)�qs�keep_blank_values�strict_parsingrKrLrKr�� parsed_result�pairs�name�values rArr�si���8�M��b�+�^�'��%3�y� J�E����e��=� ��$��&�&�u�-�#(�'�M�$�� � �r@c�Z��� |rt|ttf�std��t|t�r%t|t�st|d�}d}��fd�}n9|sgStt |��}t|t�rt|d�}d}d�}|sgS|�$d|j|�z} || krtd��g} |j |�D]V}|s|s�|j|�\}} }| s|rtd |����|s|s�4||�}||�}| j||f��X| S) Nz*Separator must be of type string or bytes.rC�=c� ��t|����S)N)rKrL)r)r)rKrLs ��rA�_unquotezparse_qsl.<locals>._unquotes�����H�V�D�Dr@�=c�8�t|jdd��S)N�+� )rr�)r)s rArVzparse_qsl.<locals>._unquote$s��#�A�I�I�d�D�$9�:�:r@rZzMax number of fields exceededzbad query field: ) r[r\r1r�� memoryview�countrr�r)rLrMrNrKrLrKr��eqrV� num_fields�r� name_valuerQ�has_eqrRs `` rArr�s:����8�J�y�3��,�?��E�F�F��"�c���)�S�)��I�w�/�I� �� E���I��:�b�>� "���i��%��i��1�I� �� ;��� � �!�����)�,�,� ��J�&��<�=�=� �A��h�h�y�)� ���",�"6�"6�r�":��D�&�%��n� �*�!F�G�G��)���~�� �������$���'�*� �Hr@c�B� |jdd�}t|||�S)N�+� )r�rrIs rArr>s(��� �^�^�C�� %�F��6�8�V�,�,r@sBABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_.-~c�z�|dk(r"tjdtd��tSt dt �d|����)N�QuoterzoDeprecated in 3.11. urllib.parse.Quoter will be removed in Python 3.14. It was not intended to be a public API.r��� stacklevelzmodule z has no attribute )�warnings�warn�DeprecationWarning�_Quoter�AttributeErrorrl)rQs rA�__getattr__rnMsC���x��� � �@�)�Q� 8��� �7�8�,�.@���I� J�Jr@c� �eZdZ d�Zd�Zd�Zy)rlc�:� tj|�|_yr<)�_ALWAYS_SAFE�union�safe)rirss rA�__init__z_Quoter.__init__^s��!� �&�&�t�,�� r@c� �dt|��d�S)Nz<Quoter �>)�dictr{s rA�__repr__z_Quoter.__repr__bs���$�t�*��q�)�)r@c�d�||jvrt|�ndj|�}|||<|S)Nz%{:02X})rs�chr�format)rir r>s rA�__missing__z_Quoter.__missing__es0���T�Y�Y��c�!�f�I�,<�,<�Q�,?����Q��� r@N)rlrmrnrtrxr|r?r@rArlrlVs���-�*�r@rlc�� t|t�r|s|S|�d}|�d}|j||�}n|�td��|�td��t ||�S)Nr6rDz,quote() doesn't support 'encoding' for bytesz*quote() doesn't support 'errors' for bytes)r[r\rJr]r)r3rsrKrLs rAr r ksr��%�L�&�#����M����H��>��F����x��0�����J�K�K����H�I�I��F�D�)�)r@c��� t|t�rd|vst|t�rd|vrt||||�St|t�rd}nd}t|||z||�}|j dd�S)NrdrZrc)r[r\r1r r�)r3rsrKrL�spaces rArr�su��� �F�C� �S��%6� �F�E� "�t�6�'9��V�T�8�V�4�4��$������� �6�4�%�<��6� :�F��>�>�#�s�#�#r@c�,�t|�jSr<)rl�__getitem__)rss rAr>r>�s���4�=�$�$�$r@c�^� t|ttf�std��|syt|t�r|jdd�}nt|D�cgc] }|dks� |��c}�}|j t|z�s|j�St|�}t|�x}dkrdjt||��Stj|�}td||�D�cgc]#}djt|||||z����%}}dj|�Scc}wcc}w)Nz!quote_from_bytes() expected bytesrrC�ignore�i@ r)r[r1r7r]r\rJ�rstrip�_ALWAYS_SAFE_BYTESrQr>r�r�map�math�isqrt�range)�bsrsr��quoter�bs_len� chunk_sizer��chunkss rArr�s����b�5�)�,�-��;�<�<� ���$����{�{�7�H�-����1��A��S��a��1�2�� �9�9�'�$�.�/��y�y�{�� !�$� '�F��b�'���W�$��w�w�s�6�2��'�'��Z�Z��'� � ��F�J�7�9�7�a��'�'�#�f�b��1�Z�<�&8�9�:�7� �9��w�w�v����2��9s� D%�D%�*(D*c�F� t|d�r|j�}n% t|�rt|dt�st �g}|s~|D]x\}} t|t�r |||�}n|t|�|||�}t| t�r || |�} n|t| �|||�} |j|dz| z��zn�|D]�\}} t|t�r |||�}n|t|�|||�}t| t�r!|| |�} |j|dz| z��et| t�r#|| |||�} |j|dz| z��� t| �} | D]G}t|t�r |||�}n|t|�|||�}|j|dz|z��I��dj|�S#t $r}td�|�d}~wwxYw#t $r/|t| �|||�} |j|dz| z�Y��WwxYw)N�itemsrz1not a valid non-string sequence or mapping objectrT�&) �hasattrr�r�r[rVr]r1r\rr)r��doseqrsrKrL� quote_via�err�l�kr�rT�elts rAr r �s�� ��u�g����� �� :��5�z�*�U�1�X�u�"=��� �A���D�A�q��!�U�#��a��&���c�!�f�d�H�f�=���!�U�#��a��&���c�!�f�d�H�f�=�� �H�H�Q��W�q�[�!���D�A�q��!�U�#��a��&���c�!�f�d�H�f�=���!�U�#��a��&������S��1��%��A�s�#��a��x��8������S��1��%�0��A��A� !��%�c�5�1�"+�C��"6�C�"+�C��H�d�H�f�"M�C�����S��3��/� !�-�8�8�8�A�;���]� :��0�1�69� :�� :��D!�*�!�#�a�&�$��&�A�A��H�H�Q��W�q�[�)�*�s)�$G�!G(� G%�G � G%�(4H �H c�P�tjdtd��t|�S)Nz/urllib.parse.to_bytes() is deprecated as of 3.8r�rg)rirjrk� _to_bytes�r�s rA�to_bytesr�"s ���M�M�C�$��4��S�>�r@c�� t|t�r" |jd�j�}|S|S#t$rt dt|�zdz��wxYw)N�ASCIIzURL z contains non-ASCII characters)r[r\rJrQ�UnicodeError�reprr�s rAr�r�(so��%��#�s�� A��*�*�W�%�,�,�.�C��J�3�J��� A��v��S� �1�?� @�A� A� A�s �6�$Ac�� t|�j�}|dddk(r|dddk(r|ddj�}|dddk(r|ddj�}|S)NrZ�<rrv�zURL:)r\rr�s rA�unwrapr�6sh����c�(�.�.� �C� �2�A�w�#�~�#�b�c�(�c�/��!�B�i�o�o��� �2�A�w�&���!�"�g�m�m�o���Jr@c�P�tjdtd��t|�S)NzUurllib.parse.splittype() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splittyper�s rA� splittyper�C�$���M�M�8�$��4��c�?�r@c��� t�$tjdtj�atj |�}|r%|j�\}}|j �|fSd|fS)Nz ([^/:]+):(.*))� _typeprogr��compile�DOTALLr��groupsr�)r�r�r��datas rAr�r�KsZ��D����J�J��� � �:� ��O�O�C� �E���|�|�~�����|�|�~�t�#�#���9�r@c�P�tjdtd��t|�S)NzUurllib.parse.splithost() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splithostr�s rA� splithostr�Xr�r@c��� t�$tjdtj�atj |�}|r&|j�\}}|r |ddk7rd|z}||fSd|fS)Nz//([^/#?]*)(.*)rr�)� _hostprogr�r�r�r�r�)r�r�� host_portr#s rAr�r�`sj��E����J�J�0�"�)�)�<� ��O�O�C� �E���,�,�.�� �4��D��G�s�N���:�D��$�����9�r@c�P�tjdtd��t|�S)NzUurllib.parse.splituser() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splituser��hosts rA� splituserr�o�%���M�M�8�$��4��d��r@c�B� |jd�\}}}|r||fSd|fS)Nr��r�)r��userr�s rAr�r�vs1��T�����,��D�%���D�d�*�*�t�d�*�*r@c�P�tjdtd��t|�S)NzWurllib.parse.splitpasswd() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk�_splitpasswd)r�s rA�splitpasswdr�|s%���M�M�8�$��4����r@c�@� |jd�\}}}||r|fSdfS�Nr��r�)r�r��passwds rAr�r��s/��9��.�.��-��D�%���E�&�,�,�t�,�,r@c�P�tjdtd��t|�S)NzUurllib.parse.splitport() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splitportr�s rA� splitportr��r�r@c�� t�$tjdtj�atj |�}|r|j�\}}|r||fS|dfS)Nz (.*):([0-9]*))� _portprogr�r�r�� fullmatchr�)r�r�r�s rAr�r��sX��4����J�J��� � �:� �����%�E���\�\�^� ��d����:����:�r@c�R�tjdtd��t||�S)NzVurllib.parse.splitnport() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk�_splitnport)r��defports rA� splitnportr��s'���M�M�8�$��4��t�W�%�%r@c�� |jd�\}}}|s|}||fS|r5|j�r|j�rt|�}||fSd}||fS||fSr�)r�r�r�r�)r�r�r�r��nports rAr�r��sx��2�����,��D�%�������=�� ��<�<�>�d�l�l�n���I�E��U�{���E��U�{����=�r@c�P�tjdtd��t|�S)NzVurllib.parse.splitquery() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk�_splitqueryr�s rA� splitqueryr��s%���M�M�8�$��4��s��r@c�B� |jd�\}}}|r||fS|dfS)Nr�r�)r�r#r�r�s rAr�r��s1��9�����,��D�%����U�{����9�r@c�P�tjdtd��t|�S)NzTurllib.parse.splittag() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splittagr�s rA�splittagr��s$���M�M�8�$��4��S�>�r@c�B� |jd�\}}}|r||fS|dfSr�r�)r�r#r��tags rAr�r��s1��3��~�~�c�*��D�%����S�y����9�r@c�P�tjdtd��t|�S)NzUurllib.parse.splitattr() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splitattrr�s rA� splitattrr��r�r@c�:� |jd�}|d|ddfS)Nr�rrZ)r)r��wordss rAr�r��s)��;��I�I�c�N�E���8�U�1�2�Y��r@c�P�tjdtd��t|�S)NzWurllib.parse.splitvalue() is deprecated as of 3.8, use urllib.parse.parse_qsl() insteadr�rg)rirjrk�_splitvalue)�attrs rA� splitvaluer��s%���M�M�9�$��4��t��r@c�@� |jd�\}}}||r|fSdfS)NrTr�)r�r�rRs rAr�r��s/��7�����,��D�%���5�%�+�+�d�+�+r@)rT)r)T)r6r�)FFr6r�Nr�)r�NN)rNN)r�)r)q�collectionsr� functoolsr�r�r�rir��__all__rrr��non_hierarchical� uses_query� uses_fragmentrrrrB�_implicit_encoding�_implicit_errorsrGrMrXr`�objectrbrrrwr�r��_DefragResultBase�_SplitResultBase�_ParseResultBase�__doc__r�r�r�r�r#r�r�� ResultBaserrrrrrr�rr�r�r�r�r�� lru_cacherrr rrr:r9rr1r7r\r2r�rArGrrrr� frozensetrqr�rnrwrlr rr>rr r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r?r@rA�<module>r�s����B#��� ����H��0� � -��-��F��A� �%� � ��\��1��'������"4�/�(�!3�.�H�2�"V�f�V�V��V�#8�V�#8�L�2�O��<�4�6G��<�~�~�>����6�8����=�?�����!K�����&����"����#N�����#�����!�����"�����%����!� ���#3�"9�"9�"A�"A�����"2�"9�"9�"A�"A����� 0� 5� 5� =� =�����#����� "2�!7�!7�!?�!?�����$4�$=�$=�$E�$E����!�#� ��$�o�� �"�$9� � �"�$9� ��)�+<�� �'�)@� � �'�)@� �1����"�<�)�D�$$�&G�����4� �5�!�5�n N��.B:�J6� #�� � �(��%�)�+�c�1��e�i�6G��:�2�:�:�&�'��&�G�.:?�PS�'�T;@�QT�G �R-��!�"���<�(��K��d��*4*�l$�$���%��%��6!�r�D��"�M�`�� �� � � �� � ���+��-�� � ��&��"�������,r@PKS�\���t����!__pycache__/parse.cpython-312.pycnu�[���� T��hl�����dZddlmZddlZddlZddlZddlZddlZddlZgd�Z gd�Z gd�Zgd�Zgd�Z gd �Zgd �ZdZdZgd �Zd�ZdZdZd�Zeefd�Zeefd�Zd�ZGd�de�ZGd�de�ZGd�de�ZGd�dee�ZGd�dee�Zedd �Z ed!d"�Z!ed#d$�Z"d%e _d&e jF_d'e jH_d(e!_d)e!jJ_d*e!jL_d+e!jN_d,e!jP_d-e!jH_d.e"_e!jJje"jJ_e!jLje"jL_e!jNje"jN_d/e"jR_e!jPje"jP_e!jHje"jH_eZ*Gd0�de e�Z+Gd1�d!e!e�Z,Gd2�d#e"e�Z-Gd3�d4e e�Z.Gd5�d6e!e�Z/Gd7�d8e"e�Z0d9�Z1e1�[1dtd<�Z2d=�Z3dud>�Z4d?�Z5d@�Z6dA�Z7ejpd;�B�dtdC��Z9dD�Z:dE�Z;dvdF�Z<dG�Z=dHZ>da?dI�Z@dJeAeBzeCzdKeAeBzfdL�ZDej�dM�ZFdN�ZGdwdO�ZH dxdQ�ZI dxdR�ZJdwdS�ZKeLdT�ZMeAeM�ZNdU�ZOGdV�dWeP�ZQdydX�ZRdzdY�ZSejpdZ��ZTd{d[�ZUdPd:ddeSfd\�ZVd]�ZWd^�ZXd_�ZYd`�ZZda[da�Z\db�Z]da^dc�Z_dd�Z`de�Zadf�Zbdg�Zcdh�Zddaedi�Zfd|dj�Zgd|dk�Zhdl�Zidm�Zjdn�Zkdo�Zldp�Zmdq�Zndr�Zods�Zpy)}a3Parse (absolute and relative) URLs. urlparse module is based upon the following RFC specifications. RFC 3986 (STD66): "Uniform Resource Identifiers" by T. Berners-Lee, R. Fielding and L. Masinter, January 2005. RFC 2732 : "Format for Literal IPv6 Addresses in URL's by R.Hinden, B.Carpenter and L.Masinter, December 1999. RFC 2396: "Uniform Resource Identifiers (URI)": Generic Syntax by T. Berners-Lee, R. Fielding, and L. Masinter, August 1998. RFC 2368: "The mailto URL scheme", by P.Hoffman , L Masinter, J. Zawinski, July 1998. RFC 1808: "Relative Uniform Resource Locators", by R. Fielding, UC Irvine, June 1995. RFC 1738: "Uniform Resource Locators (URL)" by T. Berners-Lee, L. Masinter, M. McCahill, December 1994 RFC 3986 is considered the current standard and any future changes to urlparse module should conform with it. The urlparse module is currently not entirely compliant with this RFC due to defacto scenarios for parsing, and for backward compatibility purposes, some parsing quirks from older RFCs are retained. The testcases in test_urlparse.py provides a good indicator of parsing behavior. The WHATWG URL Parser spec should also be considered. We are not compliant with it either due to existing user code API behavior expectations (Hyrum's Law). It serves as a useful guide when making changes. �)� namedtupleN)�urlparse� urlunparse�urljoin� urldefrag�urlsplit� urlunsplit� urlencode�parse_qs� parse_qsl�quote� quote_plus�quote_from_bytes�unquote�unquote_plus�unquote_to_bytes�DefragResult�ParseResult�SplitResult�DefragResultBytes�ParseResultBytes�SplitResultBytes)��ftp�http�gopher�nntp�imap�wais�file�https�shttp�mms�prospero�rtsp�rtsps�rtspu�sftp�svn�svn+ssh�ws�wss)rrrrr�telnetrrr r#r!r"�snewsr$r%r&r'�rsyncr)r*r(�nfs�gitzgit+sshr+r,z itms-services)rr�hdlr$rrr!r"r%r&r'�sip�sipsr#r(�tel) rr2�mailto�newsr-rrr.r3r4) rrrrr!r"r#rr%r&r'r3r4) rrr2rrr7rrr!r"r.r r$zAabcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+-.z! )� � � c�T�tj�tj�y)zDClear internal performance caches. Undocumented; some tests want it.N)r�cache_clear�_byte_quoter_factory���%/usr/lib64/python3.12/urllib/parse.py�clear_cacherA^s�������$�$�&r?�ascii�strictc��|S�Nr>)�objs r@�_nooprGls���Jr?c�&�|j||�SrE��encode)rF�encoding�errorss r@�_encode_resultrMos���:�:�h��'�'r?c�0���t��fd�|D��S)Nc3�J�K�|]}|r|j���nd���y�w)rN��decode��.0�xrKrLs ��r@� <genexpr>z_decode_args.<locals>.<genexpr>us$�����G�$�Q�q����(�F�+�b�8�$�s� #)�tuple)�argsrKrLs ``r@�_decode_argsrXss����G�$�G�G�Gr?c��t|dt�}|ddD]#}|s�t|t�|k7s�td��|r |tfzSt |�t fzS)Nr�z$Cannot mix str and non-str arguments)� isinstance�str� TypeErrorrGrXrM)rW� str_input�args r@�_coerce_argsr`wsg���4��7�C�(�I��A�B�x���:�c�3�'�9�4��B�C�C� � ��u�h������� 1�1�1r?c��eZdZdZdZdd�Zy)�_ResultMixinStrz>Standard approach to encoding parsed results from str to bytesr>c�8���|j��fd�|D��S)Nc3�B�K�|]}|j������y�wrErIrRs ��r@rUz)_ResultMixinStr.encode.<locals>.<genexpr>�������*T�t�!�1�8�8�H�f�+E�t���)�_encoded_counterpart��selfrKrLs ``r@rJz_ResultMixinStr.encode�����(�t�(�(�*T�t�*T�U�Ur?N�rBrC)�__name__� __module__�__qualname__�__doc__� __slots__rJr>r?r@rbrb����H��I�Vr?rbc��eZdZdZdZdd�Zy)�_ResultMixinBytesz>Standard approach to decoding parsed results from bytes to strr>c�8���|j��fd�|D��S)Nc3�B�K�|]}|j������y�wrErPrRs ��r@rUz+_ResultMixinBytes.decode.<locals>.<genexpr>�rerf)�_decoded_counterpartrhs ``r@rQz_ResultMixinBytes.decode�rjr?Nrk)rlrmrnrorprQr>r?r@rsrs�rqr?rsc�x�eZdZdZdZed��Zed��Zed��Zed��Z e ej�Z y)�_NetlocResultMixinBasezHShared methods for the parsed result objects containing a netloc elementr>c� �|jdS)Nr�� _userinfo�ris r@�usernamez_NetlocResultMixinBase.username�����~�~�a� � r?c� �|jdS)NrZrzr|s r@�passwordz_NetlocResultMixinBase.password�r~r?c��|jd}|syt|t�rdnd}|j|�\}}}|j �|z|zS)Nr�%�%)� _hostinfor[r\� partition�lower)ri�hostname� separator�percent�zones r@r�z_NetlocResultMixinBase.hostname�sV���>�>�!�$����&�h��4�C�$� �"*�"4�"4�Y�"?���'�4��~�~��'�)�D�0�0r?c���|jd}|�\|j�r|j�rt|�}nt d|����d|cxkrdkst d��t d��|S)NrZz+Port could not be cast to integer value as ri��zPort out of range 0-65535)r��isdigit�isascii�int� ValueError)ri�ports r@r�z_NetlocResultMixinBase.port�sr���~�~�a� �����|�|�~�$�,�,�.��4�y�� �#N�t�h�!W�X�X���&��&� �!<�=�=�'� �!<�=�=��r?N)rlrmrnrorp�propertyr}r�r�r��classmethod�types�GenericAlias�__class_getitem__r>r?r@rxrx�sk��R��I� �!��!��!��!��1��1�� �� �$�E�$6�$6�7�r?rxc�0�eZdZdZed��Zed��Zy)�_NetlocResultMixinStrr>c��|j}|jd�\}}}|r|jd�\}}}|sd}||fSdx}}||fS)N�@�:��netloc� rpartitionr��rir��userinfo� have_info�hostinfor}� have_passwordr�s r@r{z_NetlocResultMixinStr._userinfo�sh������(.�(9�(9�#�(>�%��)�X��08�0B�0B�3�0G�-�H�m�X� �����!�!�#'�&�H�x���!�!r?c��|j}|jd�\}}}|jd�\}}}|r+|jd�\}}}|jd�\}}}n|jd�\}}}|sd}||fS)Nr��[�]r�r��rir��_r��have_open_br� bracketedr�r�s r@r�z_NetlocResultMixinStr._hostinfo�s��������*�*�3�/���1�h�%-�%7�%7��%<�"��<��� )� 3� 3�C� 8��H�a������,�J�A�q�$� (� 2� 2�3� 7��H�a����D���~�r?N�rlrmrnrpr�r{r�r>r?r@r�r���-���I� � "�� "����r?r�c�0�eZdZdZed��Zed��Zy)�_NetlocResultMixinBytesr>c��|j}|jd�\}}}|r|jd�\}}}|sd}||fSdx}}||fS)N�@�:r�r�s r@r{z!_NetlocResultMixinBytes._userinfo�sh������(.�(9�(9�$�(?�%��)�X��08�0B�0B�4�0H�-�H�m�X� �����!�!�#'�&�H�x���!�!r?c��|j}|jd�\}}}|jd�\}}}|r+|jd�\}}}|jd�\}}}n|jd�\}}}|sd}||fS)Nr��[�]r�r�r�s r@r�z!_NetlocResultMixinBytes._hostinfo�s��������*�*�4�0���1�h�%-�%7�%7��%=�"��<��� )� 3� 3�D� 9��H�a������-�J�A�q�$� (� 2� 2�4� 8��H�a����D���~�r?Nr�r>r?r@r�r��r�r?r�rzurl fragmentrz!scheme netloc path query fragmentrz(scheme netloc path params query fragmentz� DefragResult(url, fragment) A 2-tuple that contains the url without fragment identifier and the fragment identifier as a separate argument. z$The URL with no fragment identifier.z� Fragment identifier separated from URL, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. z� SplitResult(scheme, netloc, path, query, fragment) A 5-tuple that contains the different components of a URL. Similar to ParseResult, but does not split params. z%Specifies URL scheme for the request.z0 Network location where the request is made to. z@ The hierarchical path, such as the path to a file to download. z� The query component, that contains non-hierarchical data, that along with data in path component, identifies a resource in the scope of URI's scheme and network location. z� Fragment identifier, that allows indirect identification of a secondary resource by reference to a primary resource and additional identifying information. zq ParseResult(scheme, netloc, path, params, query, fragment) A 6-tuple that contains components of a parsed URL. z� Parameters for last path element used to dereference the URI in order to provide access to perform some operation on the resource. c��eZdZdZd�Zy)rr>c�j�|jr|jdz|jzS|jS)N�#��fragment�urlr|s r@�geturlzDefragResult.geturlEs)���=�=��8�8�c�>�D�M�M�1�1��8�8�Or?N�rlrmrnrpr�r>r?r@rrC����I�r?c��eZdZdZd�Zy)rr>c��t|�SrE�r r|s r@r�zSplitResult.geturlM� ���$��r?Nr�r>r?r@rrK����I� r?c��eZdZdZd�Zy)rr>c��t|�SrE�rr|s r@r�zParseResult.geturlRr�r?Nr�r>r?r@rrPr�r?c��eZdZdZd�Zy)rr>c�j�|jr|jdz|jzS|jS)N�#r�r|s r@r�zDefragResultBytes.geturlXs)���=�=��8�8�d�?�T�]�]�2�2��8�8�Or?Nr�r>r?r@rrVr�r?rc��eZdZdZd�Zy)rr>c��t|�SrEr�r|s r@r�zSplitResultBytes.geturl`r�r?Nr�r>r?r@rr^r�r?rc��eZdZdZd�Zy)rr>c��t|�SrEr�r|s r@r�zParseResultBytes.geturler�r?Nr�r>r?r@rrcr�r?rc�z�ttfttftt ff}|D]\}}||_||_�yrE)rrrrrrrgrv)� _result_pairs�_decoded�_encodeds r@�_fix_result_transcodingr�isE�� �(�)� �&�'� �&�'��M� ,���(�(0��%�(0��%�,r?rTc��t||�\}}}t|||�}|\}}}}}|tvrd|vrt|�\}}nd}t ||||||�} || �S)a�Parse a URL into 6 components: <scheme>://<netloc>/<path>;<params>?<query>#<fragment> The result is a named 6-tuple with fields corresponding to the above. It is either a ParseResult or ParseResultBytes object, depending on the type of the url parameter. The username, password, hostname, and port sub-components of netloc can also be accessed as attributes of the returned object. The scheme argument provides the default value of the scheme component when no scheme is found in url. If allow_fragments is False, no attempt is made to separate the fragment component from the previous component, which can be either path or query. Note that % escapes are not expanded. �;r)r`r�uses_params�_splitparamsr) r��scheme�allow_fragments�_coerce_result�splitresultr��queryr��params�results r@rrvsu��(#/�s�F�";��C����3���8�K�+6�(�F�F�C��� ������"�3�'���V��� ����f�e�X� F�F��&�!�!r?c��d|vr*|jd|jd��}|dkr|dfS|jd�}|d|||dzdfS)N�/r�rrrZ)�find�rfind)r��is r@r�r��sY�� �s�{��H�H�S�#�)�)�C�.�)���q�5���7�N��H�H�S�M���r��7�C��!���I��r?c��t|�}dD]&}|j||�}|dk\s�t||�}�(|||||dfS)Nz/?#r)�lenr��min)r��start�delim�c�wdelims r@�_splitnetlocr��sR����H�E� �����!�U�#���Q�;���v�&�E���u�U��S���[�(�(r?c�0�|r|j�ryddl}|jdd�}|jdd�}|jdd�}|jdd�}|jd|�}||k(rydD]}||vs�t d |zd zdz��y)Nrr�rr�r��?�NFKCz/?#@:znetloc 'z' contains invalid z#characters under NFKC normalization)r��unicodedata�replace� normalizer�)r�r��n�netloc2r�s r@�_checknetlocr��s����V�^�^�%������s�B��A� � � �#�r��A� � � �#�r��A� � � �#�r��A��#�#�F�A�.�G��G�|�� ����<��Z�&�0�3H�H�B�C�D� D�r?c��|jd�d}|jd�\}}}|r@|rtd��|jd�\}}}|r1|jd�s td��|jd�\}}}t |�y)Nr��r��Invalid IPv6 URLr�r�)r�r�r�� startswith�_check_bracketed_host)r��hostname_and_port�before_bracketr�r�r�r�r�s r@�_check_bracketed_netlocr��s����)�)�#�.�q�1��.?�.I�.I�#�.N�+�N�L�)����/�0�0�%�/�/��4���!�T������,��/�0�0�-�7�7��<���!�T��(�#r?c���|jd�r"tjd|�std��yt j |�}t |tj�rtd��y)N�vz\Av[a-fA-F0-9]+\..+\ZzIPvFuture address is invalidz%An IPv4 address cannot be in brackets)r��re�matchr�� ipaddress� ip_addressr[�IPv4Address)r��ips r@r�r��sc�����3���x�x�0�(�;��;�=�=�<�� !� !�(� +���b�)�/�/�0��D�F�F�1r?)�typedc��t||�\}}}|jt�}|jt�}tD]&}|j|d�}|j|d�}�(t |�}dx}x}}|jd�}|dkDrU|dj�rB|dj�r/|d|D]} | tvs�n|d|j�||dzd}}|dddk(r=t|d�\}}d|vrd |vsd |vrd|vrtd ��d|vrd |vrt|�|rd|vr|jdd�\}}d|vr|jdd�\}}t!|�t#|||||�} || �S) a�Parse a URL into 5 components: <scheme>://<netloc>/<path>?<query>#<fragment> The result is a named 5-tuple with fields corresponding to the above. It is either a SplitResult or SplitResultBytes object, depending on the type of the url parameter. The username, password, hostname, and port sub-components of netloc can also be accessed as attributes of the returned object. The scheme argument provides the default value of the scheme component when no scheme is found in url. If allow_fragments is False, no attempt is made to separate the fragment component from the previous component, which can be either path or query. Note that % escapes are not expanded. rr�rNrZr��//r�r�r�r�r�)r`�lstrip�_WHATWG_C0_CONTROL_OR_SPACE�strip�_UNSAFE_URL_BYTES_TO_REMOVEr��boolr�r��isalpha�scheme_charsr�r�r�r��splitr�r)r�r�r�r��br�r�r�r�r�r�s r@rr�s���,#/�s�F�";��C����*�*�0� 1�C� �\�\�5� 6�F� (���k�k�!�R� ������2�&��)��?�+�O� "�"�F�"�U�X����� �A��1�u��Q����!�c�!�f�n�n�&6��R�a��A���$����b�q�'�-�-�/�3�q��s�t�9�C�F� �2�A�w�$��"�3��*���� �F�]�s�&�0����3�f�#4��/�0�0��&�=�S�F�]�#�F�+��3�#�:�� � �#�q�)� ��X� �c�z��Y�Y�s�A�&� ��U�����F�F�C���9�A��!��r?c �`�t|�\}}}}}}}|r|�d|��}|t|||||f��S)z�Put a parsed URL back together again. This may result in a slightly different, but equivalent URL, if the URL that was parsed originally had redundant delimiters, e.g. a ? with an empty query (the draft states that these are equivalent).r�)r`r )� componentsr�r�r�r�r�r�r�s r@rr sH��3?� �2K�A�F�F�C����.� ��f�%���*�f�f�c�5�(�%K�L�M�Mr?c���t|�\}}}}}}|r|r |dddk7rd|z}d|z|z}n'|dddk(rd|z}n|r|tvr|r|dddk(rd|z}|r|dz|z}|r|dz|z}|r|dz|z}||�S) akCombine the elements of a tuple as returned by urlsplit() into a complete URL as a string. The data argument can be any five-item iterable. This may result in a slightly different, but equivalent URL, if the URL that was parsed originally had unnecessary delimiters (for example, a ? with an empty query; the RFC states that these are equivalent).NrZr�rr�r�r�r�)r`�uses_netloc)rr�r�r�r�r�r�s r@r r s���+7� �*C�9�F�F�C���.� ��3�r��7�c�>��s��3��V�m�c�!�� �R�a��D���S�j�� �F�k�)�3�#�b�q�'�S�.��S�j�� ��s�l�S� ����C�i�%�����C�i�(�"���#��r?c ��|s|S|s|St||�\}}}t|d|�\}}}}}} t|||�\} }}} }}| |k7s| tvr||�S| tvr|r|t | ||| ||f��S|}|s!| s|}|} |s|}|t | ||| ||f��S|jd�}|ddk7r|d=|dddk(r|jd�}n&||jd�z}t d|dd�|ddg}|D]0}|dk(r |j��|dk(r� |j|��2|ddvr|jd�|t | |dj|�xsd| ||f��S#t$rY��wxYw) zaJoin a base URL and a possibly relative URL to form an absolute interpretation of the latter.rr����NrZ�..�.)rr)r`r� uses_relativerrr�filter�pop� IndexError�append�join)�baser�r�r��bscheme�bnetloc�bpath�bparams�bquery� bfragmentr�r��pathr�r�r�� base_parts�segments� resolved_path�segs r@rr/s���� ���� ,�T�3� 7��D�#�~��T�2��/�8�G�W�e�W�f�i� �S�'�?�3�2�F�F�D�&�%�����F�-�7��c�"�"� ����!�*�f�f�d�.4�e�X�.G�#H�I� I�����������E��j�&�&�$�*0�%��*C�D�E� E����S�!�J��"�~��� �r�N��B�Q�x�3���:�:�c�?���� � �3��/�� ��h�q��n�5���2���M����$�;� ��!�!�#� �C�Z��� � ��%����|�{�"� ���R� ��*�f�f�c�h�h��7�7��v�u�h�&8�9�:�:��� �� �s�E1�1 E=�<E=c��t|�\}}d|vr$t|�\}}}}}}t|||||df�}nd}|}|t||��S)z�Removes any existing fragment from URL. Returns a tuple of the defragmented URL and the fragment. If the URL contained no fragments, the second element is the empty string. r�r)r`rrr) r�r��sr��p�a�q�frag�defrags r@rrtsc��'�s�+��C�� �c�z�&�s�m���1�a��A�t��Q��1�a��B�/�0�������,�v�t�4�5�5r?�0123456789ABCDEFabcdefc�*�tt|��S)z,unquote_to_bytes('abc%20def') -> b'abc def'.)�bytes� _unquote_impl)�strings r@rr�s����v�&�'�'r?r3�returnc ��|s |jyt|t�r|jd�}|jd�}t |�dk(r|St|d�}|j}t�ItD��cic]7}tD],}||zj�tj||z���.�9c}}a|ddD] } |t|dd�||dd��"|Scc}}w#t$r|d�||�Y�FwxYw)Nr?�utf-8r�rZrr�)rr[r\rJr�� bytearray�extend� _hextobyte�_hexdigr1�fromhex�KeyError)r3�bits�resrr+r �items r@r2r2�s ��������&�#�����w�'���<�<���D� �4�y�A�~�� � �D��G� �C� �Z�Z�F���&�9�&�!��1��1�u�n�n�&�� � �a�!�e�(<�<�07�'�&�9� ��Q�R��� ��:�d�2�A�h�'�(��4���8����J��9��� ��4�L��4�L� �s�;<C%�C+�+D�Dz([-]+)c#��K�d}tj|�D]>}|j�\}}|||��t|d�j ||���|}�@||d��y�w)NrrZ)�_asciire�finditer�spanr2rQ)r3rKrL�previous_match_end�ascii_matchr��ends r@�_generate_unquoted_partsrG�st�������(�(��0�� �%�%�'� ��s��'��.�.��K��N�+�2�2�8�V�D�D� ��1��#�$� %�%�s�A!A#c���t|t�rt|�j||�Sd|vr|j|S|�d}|�d}djt |||��S)a�Replace %xx escapes by their single-character equivalent. The optional encoding and errors parameters specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. By default, percent-encoded sequences are decoded with UTF-8, and invalid sequences are replaced by a placeholder character. unquote('abc%20def') -> 'abc def'. r�r6r�r)r[r1r2rQrrrG�r3rKrLs r@rr�sh���&�%� ��V�$�+�+�H�f�=�=� �&������ ����� �~��� �7�7�+�F�H�f�E�F�Fr?Fc ��i}t|||||||��}|D]$\} } | |vr|| j| ��| g|| <�&|S)aXParse a query given as a string argument. Arguments: qs: percent-encoded query string to be parsed keep_blank_values: flag indicating whether blank values in percent-encoded queries should be treated as blank strings. A true value indicates that blanks should be retained as blank strings. The default false value indicates that blank values are to be ignored and treated as if they were not included. strict_parsing: flag indicating what to do with parsing errors. If false (the default), errors are silently ignored. If true, errors raise a ValueError exception. encoding and errors: specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. max_num_fields: int. If set, then throws a ValueError if there are more than n fields read by parse_qsl(). separator: str. The symbol to use for separating the query arguments. Defaults to &. Returns a dictionary. )rKrL�max_num_fieldsr�)rr)�qs�keep_blank_values�strict_parsingrKrLrKr�� parsed_result�pairs�name�values r@rr�sd��<�M��b�+�^�'��%3�y� J�E����e��=� ��$��&�&�u�-�#(�'�M�$�� � �r?c�X���|rt|ttf�std��t|t�r%t|t�st|d�}d}��fd�}n9|sgStt |��}t|t�rt|d�}d}d�}|sgS|�$d|j|�z} || krtd��g} |j |�D]V}|s|s�|j|�\}} }| s|rtd |����|s|s�4||�}||�}| j||f��X| S) aXParse a query given as a string argument. Arguments: qs: percent-encoded query string to be parsed keep_blank_values: flag indicating whether blank values in percent-encoded queries should be treated as blank strings. A true value indicates that blanks should be retained as blank strings. The default false value indicates that blank values are to be ignored and treated as if they were not included. strict_parsing: flag indicating what to do with parsing errors. If false (the default), errors are silently ignored. If true, errors raise a ValueError exception. encoding and errors: specify how to decode percent-encoded sequences into Unicode characters, as accepted by the bytes.decode() method. max_num_fields: int. If set, then throws a ValueError if there are more than n fields read by parse_qsl(). separator: str. The symbol to use for separating the query arguments. Defaults to &. Returns a list, as G-d intended. z*Separator must be of type string or bytes.rB�=c� ��t|����S)N)rKrL)r)r)rKrLs ��r@�_unquotezparse_qsl.<locals>._unquotes�����H�V�D�Dr?�=c�8�t|jdd��S)N�+� )rr�)r)s r@rVzparse_qsl.<locals>._unquote$s��#�A�I�I�d�D�$9�:�:r?rZzMax number of fields exceededzbad query field: ) r[r\r1r�� memoryview�countrr�r)rLrMrNrKrLrKr��eqrV� num_fields�r� name_valuerQ�has_eqrRs `` r@rr�s5���<�J�y�3��,�?��E�F�F��"�c���)�S�)��I�w�/�I� �� E���I��:�b�>� "���i��%��i��1�I� �� ;��� � �!�����)�,�,� ��J�&��<�=�=� �A��h�h�y�)� ���",�"6�"6�r�":��D�&�%��n� �*�!F�G�G��)���~�� �������$���'�*� �Hr?c�@�|jdd�}t|||�S)z�Like unquote(), but also replace plus signs by spaces, as required for unquoting HTML form values. unquote_plus('%7e/abc+def') -> '~/abc def' �+� )r�rrIs r@rr>s#���^�^�C�� %�F��6�8�V�,�,r?sBABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_.-~c�z�|dk(r"tjdtd��tSt dt �d|����)N�QuoterzoDeprecated in 3.11. urllib.parse.Quoter will be removed in Python 3.14. It was not intended to be a public API.r��� stacklevelzmodule z has no attribute )�warnings�warn�DeprecationWarning�_Quoter�AttributeErrorrl)rQs r@�__getattr__rnMsC���x��� � �@�)�Q� 8��� �7�8�,�.@���I� J�Jr?c�"�eZdZdZd�Zd�Zd�Zy)rlz�A mapping from bytes numbers (in range(0,256)) to strings. String values are percent-encoded byte values, unless the key < 128, and in either of the specified safe set, or the always safe set. c�8�tj|�|_y)zsafe: bytes object.N)�_ALWAYS_SAFE�union�safe)rirss r@�__init__z_Quoter.__init__^s�� �&�&�t�,�� r?c� �dt|��d�S)Nz<Quoter �>)�dictr|s r@�__repr__z_Quoter.__repr__bs���$�t�*��q�)�)r?c�d�||jvrt|�ndj|�}|||<|S)Nz%{:02X})rs�chr�format)rir r>s r@�__missing__z_Quoter.__missing__es0���T�Y�Y��c�!�f�I�,<�,<�Q�,?����Q��� r?N)rlrmrnrortrxr|r>r?r@rlrlVs���-�*�r?rlc��t|t�r|s|S|�d}|�d}|j||�}n|�td��|�td��t ||�S)a�quote('abc def') -> 'abc%20def' Each part of a URL, e.g. the path info, the query, etc., has a different set of reserved characters that must be quoted. The quote function offers a cautious (not minimal) way to quote a string for most of these parts. RFC 3986 Uniform Resource Identifier (URI): Generic Syntax lists the following (un)reserved characters. unreserved = ALPHA / DIGIT / "-" / "." / "_" / "~" reserved = gen-delims / sub-delims gen-delims = ":" / "/" / "?" / "#" / "[" / "]" / "@" sub-delims = "!" / "$" / "&" / "'" / "(" / ")" / "*" / "+" / "," / ";" / "=" Each of the reserved characters is reserved in some component of a URL, but not necessarily in all of them. The quote function %-escapes all characters that are neither in the unreserved chars ("always safe") nor the additional chars set via the safe arg. The default for the safe arg is '/'. The character is reserved, but in typical usage the quote function is being called on a path where the existing slash characters are to be preserved. Python 3.7 updates from using RFC 2396 to RFC 3986 to quote URL strings. Now, "~" is included in the set of unreserved characters. string and safe may be either str or bytes objects. encoding and errors must not be specified if string is a bytes object. The optional encoding and errors parameters specify how to deal with non-ASCII characters, as accepted by the str.encode method. By default, encoding='utf-8' (characters are encoded with UTF-8), and errors='strict' (unsupported characters raise a UnicodeEncodeError). r6rCz,quote() doesn't support 'encoding' for bytesz*quote() doesn't support 'errors' for bytes)r[r\rJr]r)r3rsrKrLs r@r r ksm��N�&�#����M����H��>��F����x��0�����J�K�K����H�I�I��F�D�)�)r?c���t|t�rd|vst|t�rd|vrt||||�St|t�rd}nd}t|||z||�}|j dd�S)z�Like quote(), but also replace ' ' with '+', as required for quoting HTML form values. Plus signs in the original string are escaped unless they are included in safe. It also does not have safe default to '/'. rdrZrc)r[r\r1r r�)r3rsrKrL�spaces r@rr�sp�� �F�C� �S��%6� �F�E� "�t�6�'9��V�T�8�V�4�4��$������� �6�4�%�<��6� :�F��>�>�#�s�#�#r?c�,�t|�jSrE)rl�__getitem__)rss r@r=r=�s���4�=�$�$�$r?c�\�t|ttf�std��|syt|t�r|jdd�}nt|D�cgc] }|dks� |��c}�}|j t|z�s|j�St|�}t|�x}dkrdjt||��Stj|�}td||�D�cgc]#}djt|||||z����%}}dj|�Scc}wcc}w)z�Like quote(), but accepts a bytes object rather than a str, and does not perform string-to-bytes encoding. It always returns an ASCII string. quote_from_bytes(b'abc def?') -> 'abc%20def%3f' z!quote_from_bytes() expected bytesrrB�ignore�i@ r)r[r1r7r]r\rJ�rstrip�_ALWAYS_SAFE_BYTESrQr=r�r�map�math�isqrt�range)�bsrsr��quoter�bs_len� chunk_sizer��chunkss r@rr�s�� �b�5�)�,�-��;�<�<� ���$����{�{�7�H�-����1��A��S��a��1�2�� �9�9�'�$�.�/��y�y�{�� !�$� '�F��b�'���W�$��w�w�s�6�2��'�'��Z�Z��'� � ��F�J�7�9�7�a��'�'�#�f�b��1�Z�<�&8�9�:�7� �9��w�w�v����2��9s� D$�D$�)(D)c�D�t|d�r|j�}n% t|�rt|dt�st �g}|s~|D]x\}} t|t�r |||�}n|t|�|||�}t| t�r || |�} n|t| �|||�} |j|dz| z��zn�|D]�\}} t|t�r |||�}n|t|�|||�}t| t�r!|| |�} |j|dz| z��et| t�r#|| |||�} |j|dz| z��� t| �} | D]G}t|t�r |||�}n|t|�|||�}|j|dz|z��I��dj|�S#t $r}td�|�d}~wwxYw#t $r/|t| �|||�} |j|dz| z�Y��WwxYw)a^Encode a dict or sequence of two-element tuples into a URL query string. If any values in the query arg are sequences and doseq is true, each sequence element is converted to a separate parameter. If the query arg is a sequence of two-element tuples, the order of the parameters in the output will match the order of parameters in the input. The components of a query arg may each be either a string or a bytes type. The safe, encoding, and errors parameters are passed down to the function specified by quote_via (encoding and errors only if a component is a str). �itemsrz1not a valid non-string sequence or mapping objectNrT�&) �hasattrr�r�r[rVr]r1r\rr)r��doseqrsrKrL� quote_via�err�l�kr�rT�elts r@r r �s��"�u�g����� �� :��5�z�*�U�1�X�u�"=��� �A���D�A�q��!�U�#��a��&���c�!�f�d�H�f�=���!�U�#��a��&���c�!�f�d�H�f�=�� �H�H�Q��W�q�[�!���D�A�q��!�U�#��a��&���c�!�f�d�H�f�=���!�U�#��a��&������S��1��%��A�s�#��a��x��8������S��1��%�0��A��A� !��%�c�5�1�"+�C��"6�C�"+�C��H�d�H�f�"M�C�����S��3��/� !�-�8�8�8�A�;���]� :��0�1�69� :�� :��D!�*�!�#�a�&�$��&�A�A��H�H�Q��W�q�[�)�*�s)�$G � G'� G$�G�G$�'4H�Hc�P�tjdtd��t|�S)Nz/urllib.parse.to_bytes() is deprecated as of 3.8r�rg)rirjrk� _to_bytes�r�s r@�to_bytesr�"s ���M�M�C�$��4��S�>�r?c��t|t�r" |jd�j�}|S|S#t$rt dt|�zdz��wxYw)zto_bytes(u"URL") --> 'URL'.�ASCIIzURL z contains non-ASCII characters)r[r\rJrQ�UnicodeError�reprr�s r@r�r�(sl�� �#�s�� A��*�*�W�%�,�,�.�C��J�3�J��� A��v��S� �1�?� @�A� A� A�s �5�$Ac��t|�j�}|dddk(r|dddk(r|ddj�}|dddk(r|ddj�}|S)z�Transform a string like '<URL:scheme://host/path>' into 'scheme://host/path'. The string is returned unchanged if it's not a wrapped URL. NrZ�<rrv�zURL:)r\rr�s r@�unwrapr�6sc�� �c�(�.�.� �C� �2�A�w�#�~�#�b�c�(�c�/��!�B�i�o�o��� �2�A�w�&���!�"�g�m�m�o���Jr?c�P�tjdtd��t|�S)NzUurllib.parse.splittype() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splittyper�s r@� splittyper�C�$���M�M�8�$��4��c�?�r?c���t�$tjdtj�atj |�}|r%|j�\}}|j �|fSd|fS)z:splittype('type:opaquestring') --> 'type', 'opaquestring'.Nz ([^/:]+):(.*))� _typeprogr��compile�DOTALLr��groupsr�)r�r�r��datas r@r�r�KsY�����J�J��� � �:� ��O�O�C� �E���|�|�~�����|�|�~�t�#�#���9�r?c�P�tjdtd��t|�S)NzUurllib.parse.splithost() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splithostr�s r@� splithostr�Xr�r?c���t�$tjdtj�atj |�}|r&|j�\}}|r |ddk7rd|z}||fSd|fS)z;splithost('//host[:port]/path') --> 'host[:port]', '/path'.Nz//([^/#?]*)(.*)rr�)� _hostprogr�r�r�r�r�)r�r�� host_portr#s r@r�r�`si�����J�J�0�"�)�)�<� ��O�O�C� �E���,�,�.�� �4��D��G�s�N���:�D��$�����9�r?c�P�tjdtd��t|�S)NzUurllib.parse.splituser() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splituser��hosts r@� splituserr�o�%���M�M�8�$��4��d��r?c�@�|jd�\}}}|r||fSd|fS)zJsplituser('user[:passwd]@host[:port]') --> 'user[:passwd]', 'host[:port]'.r�N�r�)r��userr�s r@r�r�vs.������,��D�%���D�d�*�*�t�d�*�*r?c�P�tjdtd��t|�S)NzWurllib.parse.splitpasswd() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk�_splitpasswd)r�s r@�splitpasswdr�|s%���M�M�8�$��4����r?c�>�|jd�\}}}||r|fSdfS)z/splitpasswd('user:passwd') -> 'user', 'passwd'.r�N�r�)r�r��passwds r@r�r��s,���.�.��-��D�%���E�&�,�,�t�,�,r?c�P�tjdtd��t|�S)NzUurllib.parse.splitport() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splitportr�s r@� splitportr��r�r?c��t�$tjdtj�atj |�}|r|j�\}}|r||fS|dfS)z*splitport('host:port') --> 'host', 'port'.Nz (.*):([0-9]*))� _portprogr�r�r�� fullmatchr�)r�r�r�s r@r�r��sW�����J�J��� � �:� �����%�E���\�\�^� ��d����:����:�r?c�R�tjdtd��t||�S)NzVurllib.parse.splitnport() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk�_splitnport)r��defports r@� splitnportr��s'���M�M�8�$��4��t�W�%�%r?c��|jd�\}}}|s|}||fS|r5|j�r|j�rt|�}||fSd}||fS||fS)z�Split host and port, returning numeric port. Return given default port if no ':' found; defaults to -1. Return numerical port if a valid number is found after ':'. Return None if ':' but not a valid number.r�N)r�r�r�r�)r�r�r�r��nports r@r�r��ss�� ����,��D�%�������=�� ��<�<�>�d�l�l�n���I�E��U�{���E��U�{����=�r?c�P�tjdtd��t|�S)NzVurllib.parse.splitquery() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk�_splitqueryr�s r@� splitqueryr��s%���M�M�8�$��4��s��r?c�@�|jd�\}}}|r||fS|dfS)z/splitquery('/path?query') --> '/path', 'query'.r�Nr�)r�r#r�r�s r@r�r��s.������,��D�%����U�{����9�r?c�P�tjdtd��t|�S)NzTurllib.parse.splittag() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splittagr�s r@�splittagr��s$���M�M�8�$��4��S�>�r?c�@�|jd�\}}}|r||fS|dfS)z)splittag('/path#tag') --> '/path', 'tag'.r�Nr�)r�r#r��tags r@r�r��s.���~�~�c�*��D�%����S�y����9�r?c�P�tjdtd��t|�S)NzUurllib.parse.splitattr() is deprecated as of 3.8, use urllib.parse.urlparse() insteadr�rg)rirjrk� _splitattrr�s r@� splitattrr��r�r?c�8�|jd�}|d|ddfS)zksplitattr('/path;attr1=value1;attr2=value2;...') -> '/path', ['attr1=value1', 'attr2=value2', ...].r�rrZN)r)r��wordss r@r�r��s&�� �I�I�c�N�E���8�U�1�2�Y��r?c�P�tjdtd��t|�S)NzWurllib.parse.splitvalue() is deprecated as of 3.8, use urllib.parse.parse_qsl() insteadr�rg)rirjrk�_splitvalue)�attrs r@� splitvaluer��s%���M�M�9�$��4��t��r?c�>�|jd�\}}}||r|fSdfS)z-splitvalue('attr=value') --> 'attr', 'value'.rTNr�)r�r�rRs r@r�r��s,������,��D�%���5�%�+�+�d�+�+r?)rT)r)T)r6r�)FFr6r�Nr�)r�NN)rNN)r�)r)qro�collectionsr� functoolsr�r�r�rir��__all__rrr��non_hierarchical� uses_query� uses_fragmentrrrrA�_implicit_encoding�_implicit_errorsrGrMrXr`�objectrbrsrxr�r��_DefragResultBase�_SplitResultBase�_ParseResultBaser�r�r�r�r#r�r�� ResultBaserrrrrrr�rr�r�r�r�r�� lru_cacherrr rrr:r9rr1r7r\r2r�rArGrrrr� frozensetrqr�rnrwrlr rr=rr r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r�r>r?r@�<module>r�s����B#��� ����H��0� � -��-��F��A� �%� � ��\��1��'������"4�/�(�!3�.�H�2�"V�f�V�V��V�#8�V�#8�L�2�O��<�4�6G��<�~�~�>����6�8����=�?�����!K�����&����"����#N�����#�����!�����"�����%����!� ���#3�"9�"9�"A�"A�����"2�"9�"9�"A�"A����� 0� 5� 5� =� =�����#����� "2�!7�!7�!?�!?�����$4�$=�$=�$E�$E����!�#� ��$�o�� �"�$9� � �"�$9� ��)�+<�� �'�)@� � �'�)@� �1����"�<�)�D�$$�&G�����4� �5�!�5�n N��.B:�J6� #�� � �(��%�)�+�c�1��e�i�6G��:�2�:�:�&�'��&�G�.:?�PS�'�T;@�QT�G �R-��!�"���<�(��K��d��*4*�l$�$���%��%��6!�r�D��"�M�`�� �� � � �� � ���+��-�� � ��&��"�������,r?PKS�\!�j�j�)__pycache__/request.cpython-312.opt-1.pycnu�[���� T��h[����dZddlZddlZddlZddlZddlZddlZddlZddl Z ddl Z ddlZddlZddl Z ddlZddlZddlZddlmZmZmZddlmZmZmZmZmZmZmZmZmZmZm Z m!Z!m"Z"m#Z#m$Z$m%Z%m&Z&m'Z'ddl(m)Z)m*Z* ddl+Z+dZ,gd�Z.d ej^dd zZ0da1de jdfddddd�d�Z3d �Z4gZ5dgd�Z6d�Z7e jpde jr�Z:d�Z;Gd�d�Z<Gd�d�Z=d�Z>Gd�d�Z?Gd�de?�Z@Gd�de?�ZAGd�de?�ZBd�ZCGd �d!e?�ZDGd"�d#�ZEGd$�d%eE�ZFGd&�d'eF�ZGGd(�d)�ZHGd*�d+eHe?�ZIGd,�d-eHe?�ZJej�ZLGd.�d/�ZMGd0�d1e?eM�ZNGd2�d3e?eM�ZOGd4�d5e?�ZPGd6�d7eP�ZQeRej�d8�rGd9�d:eP�ZTe.j�d:�Gd;�d<e?�ZVGd=�d>e?�ZWd?�ZXd@�ZYGdA�dBe?�ZZdC�Z[GdD�dEe?�Z\GdF�dGe\�Z]GdH�dIe?�Z^dJZ_ej�dKk(r ddLlambZbmcZcndM�ZbdN�ZciZdGdO�dP�ZeGdQ�dRee�ZfdagdS�ZhdaidT�ZjdakdU�ZldamdV�ZnGdW�dX�ZodY�ZpdhdZ�Zqd[�Zrd\�Zsej�d]k(rdd^lumvZvmwZwd_�Zxd`�Zyda�Zzdb�Z{yej�dKk(r dc�Z|dd�Z{de�Z}df�ZzyepZ{eqZzy#e-$rdZ,Y��[wxYw)ia� An extensible library for opening URLs using a variety of protocols The simplest way to use this module is to call the urlopen function, which accepts a string containing a URL or a Request object (described below). It opens the URL and returns the results as file-like object; the returned object has some extra methods described below. The OpenerDirector manages a collection of Handler objects that do all the actual work. Each Handler implements a particular protocol or option. The OpenerDirector is a composite object that invokes the Handlers needed to open the requested URL. For example, the HTTPHandler performs HTTP GET and POST requests and deals with non-error returns. The HTTPRedirectHandler automatically deals with HTTP 301, 302, 303, 307, and 308 redirect errors, and the HTTPDigestAuthHandler deals with digest authentication. urlopen(url, data=None) -- Basic usage is the same as original urllib. pass the url and optionally data to post to an HTTP URL, and get a file-like object back. One difference is that you can also pass a Request instance instead of URL. Raises a URLError (subclass of OSError); for HTTP errors, raises an HTTPError, which can also be treated as a valid response. build_opener -- Function that creates a new OpenerDirector instance. Will install the default handlers. Accepts one or more Handlers as arguments, either instances or Handler classes that it will instantiate. If one of the argument is a subclass of the default handler, the argument will be installed instead of the default. install_opener -- Installs a new opener as the default opener. objects of interest: OpenerDirector -- Sets up the User Agent as the Python-urllib client and manages the Handler classes, while dealing with requests and responses. Request -- An object that encapsulates the state of a request. The state can be as simple as the URL. It can also include extra HTTP headers, e.g. a User-Agent. BaseHandler -- internals: BaseHandler and parent _call_chain conventions Example usage: import urllib.request # set up authentication info authinfo = urllib.request.HTTPBasicAuthHandler() authinfo.add_password(realm='PDQ Application', uri='https://mahler:8092/site-updates.py', user='klem', passwd='geheim$parole') proxy_support = urllib.request.ProxyHandler({"http" : "http://ahad-haam:3128"}) # build a new opener that adds authentication and caching FTP handlers opener = urllib.request.build_opener(proxy_support, authinfo, urllib.request.CacheFTPHandler) # install it urllib.request.install_opener(opener) f = urllib.request.urlopen('https://www.python.org/') �N)�URLError� HTTPError�ContentTooShortError)�urlparse�urlsplit�urljoin�unwrap�quote�unquote� _splittype� _splithost� _splitport� _splituser�_splitpasswd� _splitattr�_splitquery�_splitvalue� _splittag� _to_bytes�unquote_to_bytes� urlunparse)� addinfourl�addclosehookTF)!�Request�OpenerDirector�BaseHandler�HTTPDefaultErrorHandler�HTTPRedirectHandler�HTTPCookieProcessor�ProxyHandler�HTTPPasswordMgr�HTTPPasswordMgrWithDefaultRealm�HTTPPasswordMgrWithPriorAuth�AbstractBasicAuthHandler�HTTPBasicAuthHandler�ProxyBasicAuthHandler�AbstractDigestAuthHandler�HTTPDigestAuthHandler�ProxyDigestAuthHandler�HTTPHandler�FileHandler� FTPHandler�CacheFTPHandler�DataHandler�UnknownHandler�HTTPErrorProcessor�urlopen�install_opener�build_opener�pathname2url�url2pathname� getproxies�urlretrieve� urlcleanup� URLopener�FancyURLopenerz%d.%d�)�cafile�capath� cadefault�contextc���|s|s|r�ddl}|jdtd�|�td��tstd��tjt jj||��}|jdg�t|� �}t|�} n3|rt|� �}t|�} nt� t�xa} nt} | j|||�S) a�Open the URL url, which can be either a string or a Request object. *data* must be an object specifying additional data to be sent to the server, or None if no such data is needed. See Request for details. urllib.request module uses HTTP/1.1 and includes a "Connection:close" header in its HTTP requests. The optional *timeout* parameter specifies a timeout in seconds for blocking operations like the connection attempt (if not specified, the global default timeout setting will be used). This only works for HTTP, HTTPS and FTP connections. If *context* is specified, it must be a ssl.SSLContext instance describing the various SSL options. See HTTPSConnection for more details. The optional *cafile* and *capath* parameters specify a set of trusted CA certificates for HTTPS requests. cafile should point to a single file containing a bundle of CA certificates, whereas capath should point to a directory of hashed certificate files. More information can be found in ssl.SSLContext.load_verify_locations(). The *cadefault* parameter is ignored. This function always returns an object which can work as a context manager and has the properties url, headers, and status. See urllib.response.addinfourl for more detail on these properties. For HTTP and HTTPS URLs, this function returns a http.client.HTTPResponse object slightly modified. In addition to the three new methods above, the msg attribute contains the same information as the reason attribute --- the reason phrase returned by the server --- instead of the response headers as it is specified in the documentation for HTTPResponse. For FTP, file, and data URLs and requests explicitly handled by legacy URLopener and FancyURLopener classes, this function returns a urllib.response.addinfourl object. Note that None may be returned if no handler handles the request (though the default installed global OpenerDirector uses UnknownHandler to ensure this never happens). In addition, if proxy settings are detected (for example, when a *_proxy environment variable like http_proxy is set), ProxyHandler is default installed and makes sure the requests are handled through the proxy. rNzJcafile, capath and cadefault are deprecated, use a custom context instead.r;zDYou can't pass both context and any of cafile, capath, and cadefaultzSSL support not available)r<r=zhttp/1.1�r?)�warnings�warn�DeprecationWarning� ValueError� _have_ssl�ssl�create_default_context�Purpose�SERVER_AUTH�set_alpn_protocols�HTTPSHandlerr3�_opener�open) �url�data�timeoutr<r=r>r?rB� https_handler�openers �'/usr/lib64/python3.12/urllib/request.pyr1r1�s���h��9���� � �0�1C�Q� H������ ���8�9�9��,�,�S�[�[�-D�-D�4:�4:�<�� �"�"�J�<�0�$�W�5� ��m�,�� �$�W�5� ��m�,�� ��'�>�)��&����;�;�s�D�'�*�*�c��|ay�N)rM)rSs rTr2r2�s���GrUc��t|�\}}tjt||��5}|j �}|dk(r,|s*t jj|�|fcddd�S|r t|d�}n7tjd��}|j}tj|�|5||f} d} d}d}d} d |vrt|d �}|r || | |�|j| �x}rD|t!|�z }|j#|�| dz } |r || | |�|j| �x}r�Dddd�ddd�dk\r|krt%d||fz �� S#1swY�.xYw#1swY�2xYw) aW Retrieve a URL into a temporary location on disk. Requires a URL argument. If a filename is passed, it is used as the temporary file location. The reporthook argument should be a callable that accepts a block number, a read size, and the total file size of the URL target. The data argument should be valid URL encoded data. If a filename is passed and the URL points to a local resource, the result is a copy from local file to new file. Returns a tuple containing the path to the newly created data file as well as the resulting HTTPMessage object. �fileN�wbF)�delete� ���r�content-length�Content-Length��1retrieval incomplete: got only %i out of %i bytes)r� contextlib�closingr1�info�os�path�normpathrN�tempfile�NamedTemporaryFile�name�_url_tempfiles�append�int�read�len�writer)rO�filename� reporthookrP�url_typerf�fp�headers�tfp�result�bs�sizern�blocknum�blocks rTr7r7�s��� ��_�N�H�d� � � �G�C��.� /�2��'�'�)���v��h��7�7�#�#�D�)�7�2� 0� /���x��&�C��-�-�U�;�C��x�x�H��!�!�(�+� ��w�&�F��B��D��D��H��7�*��7�#3�4�5����8�R��.��7�7�2�;�&�%�&���E� �"��� � �%� ��A� ����x��T�2��7�7�2�;�&�%�&��! 0�F�q�y�T�D�[�"�?��T�l� �"�$� $��M�1�S��! 0� /�s+�8E3�0AE3�8BE'�:E3�'E0 �,E3�3E<c��tD]} tj|��tdd�=trdayy#t$rY�9wxYw)z0Clean up temporary files from urlretrieve calls.N)rkre�unlink�OSErrorrM)� temp_files rTr8r8sH��#� � ��I�I�i� �$� �q�������� �� �s�5� A�Az:\d+$c��|j}t|�d}|dk(r|jdd�}tj d|d�}|j�S)z�Return request-host, as defined by RFC 2965. Variation from RFC: returned value is lowercased, for convenient comparison. r`��Host)�full_urlr� get_header�_cut_port_re�sub�lower)�requestrO�hosts rT�request_hostr�)sX��� � �C��C�=���D��r�z��!�!�&�"�-�����B��a�(�D��:�:�<�rUc��eZdZdidddfd�Zed��Zejd��Zejd��Zed��Zejd��Zejd ��Zd �Z d�Z d�Zd �Zd�Z d�Zd�Zd�Zdd�Zd�Zd�Zy)rNFc��||_i|_i|_d|_||_d|_|j �D]\}}|j||��|�t|�}||_ ||_ |r||_yyrW)r�ru�unredirected_hdrs�_datarP�_tunnel_host�items� add_headerr��origin_req_host�unverifiable�method) �selfrOrPrur�r�r��key�values rT�__init__zRequest.__init__;s����� ����!#����� ��� � ���!�-�-�/�J�C���O�O�C��'�*��"�*�4�0�O�.���(���� �D�K�rUc�~�|jr&dj|j|j�S|jS)Nz{}#{})�fragment�format� _full_url�r�s rTr�zRequest.full_urlMs,���=�=��>�>�$�.�.�$�-�-�@�@��~�~�rUc��t|�|_t|j�\|_|_|j �yrW)r r�rr��_parse�r�rOs rTr�zRequest.full_urlSs/�� �����(1�$�.�.�(A�%���� ���� rUc�.�d|_d|_d|_y)Nr�)r�r��selectorr�s rTr�zRequest.full_urlZs������� ��� rUc��|jSrW)r�r�s rTrPzRequest.data`s���z�z�rUc�x�||jk7r+||_|jd�r|jd�yyy)N�Content-length)r�� has_header� remove_header)r�rPs rTrPzRequest.datads=���4�:�:���D�J����/�0��"�"�#3�4�1�rUc��d|_yrW)rPr�s rTrPzRequest.datans ���� rUc��t|j�\|_}|j�td|jz��t|�\|_|_|jrt|j�|_yy)Nzunknown url type: %r) rr��typerEr�r r�r�r)r��rests rTr�zRequest._parsersd��$�T�^�^�4��� �4��9�9���3�d�m�m�C�D�D�#-�d�#3� �� �4�=��9�9��� � �*�D�I�rUc�<�|j�dnd}t|d|�S)z3Return a string indicating the HTTP request method.�POST�GETr�)rP�getattr)r��default_methods rT� get_methodzRequest.get_methodzs!��#'�9�9�#8��e���t�X�~�6�6rUc��|jSrW)r�r�s rT�get_full_urlzRequest.get_full_urls���}�}�rUc��|jdk(r%|js|j|_||_y||_|j|_||_y)N�https)r�r�r�r�r�)r�r�r�s rT� set_proxyzRequest.set_proxy�sF���9�9����(9�(9� $� � �D���� ��D�I� �M�M�D�M��� rUc�4�|j|jk(SrW)r�r�r�s rT� has_proxyzRequest.has_proxy�s���}�}�� � �-�-rUc�>�||j|j�<yrW)ru� capitalize�r�r��vals rTr�zRequest.add_header�s��),����S�^�^�%�&rUc�>�||j|j�<yrW)r�r�r�s rT�add_unredirected_headerzRequest.add_unredirected_header�s��36����s�~�~�/�0rUc�>�||jvxs||jvSrW)rur��r��header_names rTr�zRequest.has_header�s&���t�|�|�+�6��t�5�5�5� 7rUc�n�|jj||jj||��SrW)ru�getr�)r�r��defaults rTr�zRequest.get_header�s2���|�|�����"�"�&�&�{�G�<�>� >rUc�t�|jj|d�|jj|d�yrW)ru�popr�r�s rTr�zRequest.remove_header�s,��������d�+����"�"�;��5rUc�h�i|j�|j�}t|j��SrW)r�ru�listr�)r��hdrss rT�header_itemszRequest.header_items�s,��9�$�(�(�9�D�L�L�9���D�J�J�L�!�!rUrW)�__name__� __module__�__qualname__r��propertyr��setter�deleterrPr�r�r�r�r�r�r�r�r�r�r��rUrTrr9s���!%�r�!%�E��!�$���� �_�_���������� ���� �[�[�5��5� �\�\����+�7� ��.�-�7�7�>� 6�"rUrc�R�eZdZd�Zd�Zd�Zd�Zdejfd�Z d d�Z d�Zy) rc�p�dtz}d|fg|_g|_i|_i|_i|_i|_y)N�Python-urllib/%sz User-agent)�__version__� addheaders�handlers�handle_open�handle_error�process_response�process_request)r��client_versions rTr�zOpenerDirector.__init__�sB��+�k�9��(�.�9�:����� ������� "���!��rUc��t|d�stdt|�z��d}t|�D�] }|dvr� |j d�}|d|}||dzd}|jd�rW|j d�|zdz}||dzd} t |�}|jj|i�} | |j|<n=|dk(r|}|j} n)|d k(r|}|j} n|d k(r|}|j} n��| j|g�} | rtj| |�n| j!|�d}��|r2tj|j"|�|j%|�yy#t$rY��wxYw)N� add_parentz%expected BaseHandler instance, got %rF)�redirect_request�do_open� proxy_open�_r`�errorrN�responser�T)�hasattr� TypeErrorr��dir�find� startswithrmrEr�r�r�r�r�� setdefault�bisect�insortrlr�r�)r��handler�added�meth�i�protocol� condition�j�kind�lookupr�s rT�add_handlerzOpenerDirector.add_handler�s����w��-��C� ��M�*�+� +�����L�D��D�D��� � �#��A��B�Q�x�H��Q�q�S�T� �I��#�#�G�,��N�N�3�'�!�+�a�/���A�a�C�D�z����t�9�D��*�*�.�.�x��<��.4��!�!�(�+��f�$����)�)���j�(����.�.���i�'����-�-����(�(��r�2�H��� � �h��0�����(��E�G!�J��M�M�$�-�-��1����t�$���/"����s� E3�3 E?�>E?c��yrWr�r�s rT�closezOpenerDirector.close����rUc�d�|j|d�}|D]}t||�}||�}|��|cSy)Nr�)r�r�) r��chainr�� meth_name�argsr�r��funcrws rT�_call_chainzOpenerDirector._call_chain�s>���9�9�T�2�&���G��7�I�.�D��4�[�F��!�� � rUNc��t|t�r t||�}n|}|�||_||_|j }|dz}|jj|g�D]}t||�}||�}�tjd|j|j|j|j��|j||�} |dz}|jj|g�D]}t||�}||| �} �| S)N�_requestzurllib.Request� _response)� isinstance�strrrPrQr�r�r�r��sys�auditr�rur��_openr�) r��fullurlrPrQ�reqr�r�� processorr�r�s rTrNzOpenerDirector.open�s����g�s�#��'�4�(�C��C����������8�8���Z�'� ��-�-�1�1�(�B�?�I��9�i�0�D��s�)�C�@� � � �"�C�L�L�#�(�(�C�K�K����IY�Z��:�:�c�4�(���[�(� ��.�.�2�2�8�R�@�I��9�i�0�D��C��*�H�A��rUc���|j|jdd|�}|r|S|j}|j|j||dz|�}|r|S|j|jdd|�S)Nr��default_openr�unknown�unknown_open)r�r�r�)r�r rPrwr�s rTrzOpenerDirector._open s����!�!�$�"2�"2�I�"0�#�7����M��8�8���!�!�$�"2�"2�H�h�")�?*�+.�0����M����� 0� 0�)� .��5� 5rUc���|dvr|jd}|d}d|z}d}|}n|j}|dz}d}|||f|z}|j|�}|r|S|r|dd fz}|j|�Sy) N��httpr�rr;z http_error_%sr`�_errorrr��http_error_default)r�r�)r��protor��dictr��http_err� orig_argsrws rTr�zOpenerDirector.errors����%�%��$�$�V�,�D���G�E�'�%�/�I��H��I��$�$�D���(�I��H��e�Y�'�$�.��!��!�!�4�(����M���)�%9�:�Y�F�D�#�4�#�#�T�*�*�rUrW)r�r�r�r�r�r�r��socket�_GLOBAL_DEFAULT_TIMEOUTrNrr�r�rUrTrr�s3�� "�-%�^ � �"&�v�/M�/M��: 5�+rUrc �h�t�}ttttt ttttg }ttjd�r|jt�t�}|D]V}|D]O}t!|t"�rt%||�s� |j'|��2t!||�s�?|j'|��Q�X|D]}|j)|��|D]}|j+|���|D]*}t!|t"�r|�}|j+|��,|S)a*Create an opener object from a list of handlers. The opener will use several default handlers, including support for HTTP, FTP and when applicable HTTPS. If any of the handlers passed as arguments are subclasses of the default handlers, the default handlers will not be used. �HTTPSConnection)rr r/r*rrr,r+r0r.r�r�clientrlrL�setrr�� issubclass�add�remover�)r�rS�default_classes�skip�klass�check�hs rTr3r35s��� �F�#�^�[�.�0C�!�;�0B�"�$�O��t�{�{�-�.����|�,��5�D� ���E��%��&��e�U�+��H�H�U�O��E�5�)�������!������u�%��!�����5�7�#�!����a�����A����1����MrUc�"�eZdZdZd�Zd�Zd�Zy)r��c��||_yrW)�parent)r�r)s rTr�zBaseHandler.add_parent\s ����rUc��yrWr�r�s rTr�zBaseHandler.close_r�rUc�N�t|d�sy|j|jkS)N� handler_orderT)r�r,)r��others rT�__lt__zBaseHandler.__lt__cs(���u�o�.���!�!�E�$7�$7�7�7rUN)r�r�r�r,r�r�r.r�rUrTrrYs���M�� �8rUrc��eZdZdZdZd�ZeZy)r0zProcess HTTP error responses.i�c��|j|j|j�}}}d|cxkrdks"n|jj d|||||�}|S)N���,r)�code�msgrdr)r�)r�r�r�r3r4r�s rT� http_responsez HTTPErrorProcessor.http_responsepsT��"�-�-����x�}�}��4�c���t�!�c�!��{�{�(�(����4��d�<�H��rUN)r�r�r��__doc__r,r5�https_responser�rUrTr0r0ls��'��M� �#�NrUr0c��eZdZd�Zy)rc�4�t|j||||��rW)rr�)r�r rtr3r4r�s rTrz*HTTPDefaultErrorHandler.http_error_default~s������d�C��r�:�:rUN)r�r�r�rr�rUrTrr}s��;rUrc�4�eZdZdZdZd�Zd�ZexZxZxZ Z dZy)r�� c�Z�|j�}|dvr|dvs"|dvr|dk(st|j||||��|jdd�}d}|jj�D� � cic]\} } | j �|vr| | ��}} } t|||jd� �Scc} } w) a�Return a Request or None in response to a redirect. This is called by the http_error_30x methods when a redirection response is received. If a redirection should take place, return a new Request to allow http_error_30x to perform the redirect. Otherwise, raise HTTPError if no-one else should try to handle this url. Return None if you can't but another Handler might. )�-�.�/i3i4)r��HEAD)r>r?r@r�� z%20)r^zcontent-typeT)rur�r�) r�rr��replacerur�r�rr�)r�r rtr3r4ru�newurl�m�CONTENT_HEADERS�k�v� newheaderss rTr�z$HTTPRedirectHandler.redirect_request�s��� �N�N����2�2�q�O�7K���&�1��;��C�L�L�$��W�b�A�A�����U�+��<��'*�{�{�'8�'8�':�;�':�t�q�!�����/�9���d�':� �;��v�)�'*�':�':�$(�*� *��;s�,B'c�Z�d|vr|d}nd|vr|d}nyt|�}|jdvrt|||�d|�d�||��|js|jrt|�}d|d<t |�}t|dtj� �}t|j|�}|j||||||�}|�yt|d �rp|jx} |_| j|d�|j k\st#| �|j$k\r6t|j||j&|z||��ix} x|_|_| j|d�dz| |<|j)�|j+�|j,j/||j0� �S)N�location�uri�rr��ftpr�z - Redirection to url 'z' is not allowed�/r;z iso-8859-1)�encoding�safe� redirect_dictrr`�rQ)r�schemerrf�netlocr�rr �string�punctuationrr�r�r�rRr��max_repeatsro�max_redirections�inf_msgrnr�r)rNrQ) r�r rtr3r4rurD�urlparts�new�visiteds rT�http_error_302z"HTTPRedirectHandler.http_error_302�s����� ��Z�(�F� �g� ��U�^�F���F�#�� �?�?�">�>����AD�f�M���� � �}�}�����H�~�H��H�Q�K��H�%�� ��\��0B�0B�D������v�.�� �#�#�C��T�3���H���;���3��(�*-�*;�*;�;�G�c�'����F�A�&�$�*:�*:�:��G��� 5� 5�5�����d� $���s� 2�G�R�A�A�?A�@�G�@�c�'�#�*;�!�+�+�f�a�0�1�4���� ��� � ��� ��{�{����S�[�[��9�9rUzoThe HTTP server returned a redirect error that would lead to an infinite loop. The last 30x error message was: N)r�r�r�rXrYr�r^�http_error_301�http_error_303�http_error_307�http_error_308rZr�rUrTrr�s<���K��� *�L::�xIW�V�N�V�^�V�n�~�2�GrUrc�f�t|�\}}|jd�sd}|}ne|jd�std|z��d|vr$|jd�}|jd|�}n|jdd�}|dk(rd}|d|}t |�\}}|�t|�\}} ndx}} ||| |fS)aReturn (scheme, user, password, host/port) given a URL or an authority. If a URL is supplied, it must have an authority (host:port) component. According to RFC 3986, having an authority component means the URL must have two slashes after the scheme. rON�//zproxy URL with no authority: %r�@r;r])rr�rEr�rr) �proxyrT�r_scheme� authority�host_separator�end�userinfo�hostport�user�passwords rT�_parse_proxyro�s���"�%�(��F�H����s�#���� ��"�"�4�(��>��F�G�G��(�?�%�]�]�3�/�N��-�-��^�4�C��-�-��Q�'�C��"�9��C��Q�s�O� �#�I�.��H�h���%�h�/���h����x��4��8�+�+rUc��eZdZdZdd�Zd�Zy)r �dNc��|� t�}||_|j�D]4\}}|j�}t |d|z|||j fd���6y)Nz%s_openc��||||�SrWr�)�rrfr�r�s rT�<lambda>z'ProxyHandler.__init__.<locals>.<lambda>s���Q��t�,rU)r6�proxiesr�r��setattrr�)r�rvr�rOs rTr�zProxyHandler.__init__sU���?� �l�G���� ����I�D�#��:�:�<�D��D�)�d�*�$'�d����-� .�)rUc���|j}t|�\}}}}|�|}|jrt|j�ry|rb|r`t |��dt |���} tj| j��jd�} |jdd| z�t |�}|j||�||k(s|dk(ry|jj||j��S)N�:�ascii�Proxy-authorization�Basic r�rS)r�ror��proxy_bypassr�base64� b64encode�encode�decoder�r�r)rNrQ)r�r rfr�� orig_type� proxy_typermrnrl� user_pass�credss rTr�zProxyHandler.proxy_open"s����H�H� �/;�E�/B�,� �D�(�H���"�J��8�8��S�X�X�.���H�#*�4�=�#*�8�#4�6�I��$�$�Y�%5�%5�%7�8�?�?��H�E��N�N�0�(�U�2B�C��8�$��� � �h� �+�� �"�i�7�&:���;�;�#�#�C����#�=�=rUrW)r�r�r�r,r�r�r�rUrTr r s���M� .�>rUr c�,�eZdZd�Zd�Zd�Zdd�Zd�Zy)r!c��i|_yrW)�passwdr�s rTr�zHTTPPasswordMgr.__init__@s ����rUc�����t|t�r|g}|�jvri�j|<dD]+�t��fd�|D��}||f�j||<�-y)N�TFc3�B�K�|]}�j|�����y�wrW)� reduce_uri)�.0�u�default_portr�s ��rT� <genexpr>z/HTTPPasswordMgr.add_password.<locals>.<genexpr>Js ����� ?�:=�Q�����<�0�#�s�)rrr��tuple)r��realmrLrmr��reduced_urir�s` @rT�add_passwordzHTTPPasswordMgr.add_passwordCsf����c�3���%�C�����#�!#�D�K�K���'�L�� ?�:=� ?�?�K�/3�V�n�D�K�K���{�+�(rUc���|jj|i�}dD]M}|j||�}|j�D]&\}}|D]}|j ||�s�|cccS�(�Oy)Nr��NN)r�r�r�r�� is_suburi) r�r��authuri�domainsr��reduced_authuri�uris�authinforLs rT�find_user_passwordz"HTTPPasswordMgr.find_user_passwordNsf���+�+�/�/�%��,��'�L�"�o�o�g�|�D�O�")�-�-�/���h��C��~�~�c�?�;�'�� �#2�(�rUc��t|�}|dr|d}|d}|dxsd}nd}|}d}t|�\}}|r!|�|�ddd�j|�} | �d || fz}||fS) z@Accept authority or URI and extract only the authority and path.r`rr;rON�Pi�rz%s:%d)rrr�) r�rLr��partsrTrhrfr��port�dports rTr�zHTTPPasswordMgr.reduce_uriXs����� ����8��1�X�F��a��I���8�?�s�D��F��I��D�� �*� ��d��D�L�V�-?��!���s�6�{� �� �#�t�U�m�3� ��$��rUc�r�||k(ry|d|dk7ry|d}|dddk7r|dz }|dj|�S)zcCheck if test is below base in a URI tree Both args must be URIs in reduced form. TrFr`r]NrO)r�)r��base�test�prefixs rTr�zHTTPPasswordMgr.is_suburiosV�� �4�<����7�d�1�g����a����"�#�;�#���c�M�F��A�w�!�!�&�)�)rUN)T)r�r�r�r�r�r�r�r�r�rUrTr!r!>s��� =���.*rUr!c��eZdZd�Zy)r"c�p�tj|||�\}}|�||fStj|d|�SrW)r!r�)r�r�r�rmrns rTr�z2HTTPPasswordMgrWithDefaultRealm.find_user_password�sC��(�;�;�D�%�<C�E���h�����>�!��1�1�$��g�F�FrUN)r�r�r�r�r�rUrTr"r"~s��GrUr"c�8��eZdZ�fd�Zd�fd� Zdd�Zd�Z�xZS)r#c�0��i|_t�|� �yrW)� authenticated�superr�)r�� __class__s �rTr�z%HTTPPasswordMgrWithPriorAuth.__init__�s������ ���rUc�v��|j||�|�t�|� d|||�t�|� ||||�yrW)�update_authenticatedr�r�)r�r�rLrmr��is_authenticatedr�s �rTr�z)HTTPPasswordMgrWithPriorAuth.add_password�s@����!�!�#�'7�8����G� ��s�D�&�9� ���U�C��v�6rUc��t|t�r|g}dD]*}|D]#}|j||�}||j|<�%�,y�Nr�)rrr�r�)r�rLr�r�r�r�s rTr�z1HTTPPasswordMgrWithPriorAuth.update_authenticated�sG���c�3���%�C�'�L���"�o�o�a��>��2B��"�"�;�/��(rUc��dD]J}|j||�}|jD]'}|j||�s�|j|ccS�Lyr�)r�r�r�)r�r�r�r�rLs rTr�z-HTTPPasswordMgrWithPriorAuth.is_authenticated�sK��'�L�"�o�o�g�|�D�O��)�)���>�>�#��7��-�-�c�2�2�*�(rU)F)r�r�r�r�r�r�r�� __classcell__)r�s@rTr#r#�s����7�C�3rUr#c�t�eZdZejdej �Zd d�Zd�Zd�Z d�Z d�Zd�ZeZ eZy) r$z1(?:^|,)[ ]*([^ ,]+)[ ]+realm=(["']?)([^"']*)\2Nc�`�|� t�}||_|jj|_yrW)r!r�r�)r��password_mgrs rTr�z!AbstractBasicAuthHandler.__init__�s)����*�,�L�"��� �K�K�4�4��rUc#�K�d}tjj|�D]=}|j�\}}}|dvrt j dtd�||f��d}�?|s|r|j�d}nd}|df��yy�w)NF)�"�'zBasic Auth Realm was unquoted�Trr�)r$�rx�finditer�groupsrBrC�UserWarning�split)r��header�found_challenge�morTr r�s rT�_parse_realmz%AbstractBasicAuthHandler._parse_realm�s�������*�-�-�6�6�v�>�B�#%�9�9�;� �F�E�5��J�&�� � �=�)�1�.��5�/�!�"�O�?��������*�����4�.� ��s�BBc���|j|�}|syd}|D]J}|j|�D]4\}}|j�dk7r|}�|��|j|||�ccS�L|�t d����y)N�basicz@AbstractBasicAuthHandler does not support the following scheme: )�get_allr�r��retry_http_basic_authrE) r��authreqr�r ru�unsupportedr�rTr�s rT�http_error_auth_reqedz.AbstractBasicAuthHandler.http_error_auth_reqed�s����/�/�'�*�������F�!%�!2�!2�6�!:� ����<�<�>�W�,�"(�K���$� �5�5�d�C��G�G�";���"�� &�)�*� *�#rUc��|jj||�\}}|��|�d|��}dtj|j ��jd�z}|j |jd�|k(ry|j|j|�|jj||j��Sy)Nryr|rzrS)r�r�r~rr�r�r��auth_headerr�r)rNrQ)r�r�r r�rm�pw�raw�auths rTr�z.AbstractBasicAuthHandler.retry_http_basic_auth�s����;�;�1�1�%��>���b� �>�!�2�&�C��f�.�.�s�z�z�|�<�C�C�G�L�L�D��~�~�d�.�.��5��=���'�'��(8�(8�$�?��;�;�#�#�C����#�=�=�rUc���t|jd�r%|jj|j�s|S|j d�s�|jjd|j�\}}dj ||�j�}tj|�j�}|jddj |j���|S)Nr�� Authorizationz{0}:{1}zBasic {}) r�r�r�r�r�r�r�r�r~�standard_b64encoder�r��strip)r�r rmr��credentials�auth_strs rT�http_requestz%AbstractBasicAuthHandler.http_request�s�������%7�8��{�{�+�+�C�L�L�9��J��~�~�o�.��;�;�9�9�$����M�L�D�&�#�*�*�4��8�?�?�A�K��0�0��=�D�D�F�H��'�'��(2�(9�(9�(�.�.�:J�(K� M�� rUc��t|jd�rfd|jcxkrdkr+nn(|jj|jd�|S|jj|jd�|S)Nr�r1r2TF)r�r�r3r�r�)r�r r�s rTr5z&AbstractBasicAuthHandler.http_response s`���4�;�;� 2�3��h�m�m�)�c�)����0�0����t�D������0�0����u�E��rUrW)r�r�r��re�compile�Ir�r�r�r�r�r�r5� https_requestr7r�rUrTr$r$�sL�� ����1��D�D� �B�5�!�(*�4 ���!�M�"�NrUr$c��eZdZdZd�Zy)r%r�c�F�|j}|jd|||�}|S)N�www-authenticate)r�r�)r�r rtr3r4rurOr�s rT�http_error_401z#HTTPBasicAuthHandler.http_error_401s*���l�l���-�-�.@�*-�s�G�=���rUN)r�r�r�r�r�r�rUrTr%r%s��!�K�rUr%c��eZdZdZd�Zy)r&r{c�F�|j}|jd|||�}|S�N�proxy-authenticate)r�r�)r�r rtr3r4rurhr�s rT�http_error_407z$ProxyBasicAuthHandler.http_error_407%s-�� �H�H� ��-�-�.B�*3�S�'�C���rUN)r�r�r�r�r�r�rUrTr&r&!s��'�K�rUr&c�>�eZdZd d�Zd�Zd�Zd�Zd�Zd�Zd�Z d �Z y)r'Nc��|� t�}||_|jj|_d|_d|_d|_y�Nr)r!r�r��retried�nonce_count� last_nonce)r�r�s rTr�z"AbstractDigestAuthHandler.__init__?s>���>�$�&�F���� �K�K�4�4�����������rUc��d|_yr�)r�r�s rT�reset_retry_countz+AbstractDigestAuthHandler.reset_retry_countHs ����rUc�Z�|j|d�}|jdkDrt|jdd|d��|xjdz c_|rZ|j �d}|j�dk(r|j ||�S|j�dk7rtd|z��yy) N�i�zdigest auth failedr`r�digestr�zEAbstractDigestAuthHandler does not support the following scheme: '%s')r�r�rr�r�r��retry_http_digest_authrE)r�r�r�r rur�rTs rTr�z/AbstractDigestAuthHandler.http_error_auth_reqedKs����+�+�k�4�0���<�<�!���C�L�L�#�/C�#�T�+� +� �L�L�A��L���]�]�_�Q�'�F��|�|�~��)��2�2�3��@�@�����7�*� �"?�AG�"H�I�I�+� rUc�z�|jdd�\}}ttdt|���}|j ||�}|rtd|z}|j j |jd�|k(ry|j|j|�|jj||j��}|Sy)NrBr`z Digest %srS)r��parse_keqv_list�filter�parse_http_list�get_authorizationrur�r�r�r)rNrQ)r�r r��token� challenge�chal�auth_val�resps rTr�z0AbstractDigestAuthHandler.retry_http_digest_auth_s����:�:�c�1�-���y��v�d�O�I�,F�G�H���%�%�c�4�0���"�T�)�H��{�{���t�/�/��6�(�B���'�'��(8�(8�(�C��;�;�#�#�C����#�=�D��K� rUc���|j�d|�dtj��d�}|jd�t d�z}tj|�j�}|ddS)Nryrz��)r��time�ctimer��_randombytes�hashlib�sha1� hexdigest)r��nonce�s�b�digs rT� get_cnoncez$AbstractDigestAuthHandler.get_cnonceksT�� �+�+�U�D�J�J�L�A�� �H�H�W���Q��/���l�l�1�o�'�'�)���3�B�x�rUc��� |d}|d}|jd�}|jdd�}|jdd�}|j|�\}} |�y|jj ||j �\} }| �y|j�|j|j|�}nd}| �d|�d|��} |j��d|j��}|�| || �|�d||����}n�d|jd �vry||jk(r|xjd z c_nd |_||_d|jz}|j|�}|�d|�d|�dd�d||��� }| || �|�}ntd|z��d | �d|�d|�d|j�d|�d�}|r|d|zz }|r|d|zz }|d|zz }|r|d�d�d�z }|S#t$rYywxYw)Nr�r�qop� algorithm�MD5�opaqueryr��,r`z%08xzqop '%s' is not supported.z username="z ", realm="z ", nonce="z", uri="z ", response="r�z , opaque="%s"z , digest="%s"z, algorithm="%s"z, qop=auth, nc=z , cnonce=")r��KeyError�get_algorithm_implsr�r�r�rP�get_entity_digestr�r�r�r�r�r r)r�r r�r�rrrr�H�KDrmr��entdig�A1�A2�respdig�ncvalue�cnonce�noncebitr�s rTr�z+AbstractDigestAuthHandler.get_authorizationvs�� ���M�E���M�E��(�(�5�/�C�����e�4�I��X�X�h��-�F��(�(��3���2��9���;�;�1�1�%����F���b��<���8�8���+�+�C�H�H�d�;�F��F���� +�����(����&�� �;���2��5�!�B�%� 8�9�G� �s�y�y��~� %�����'�� � �A�%� �#$�� �"'����t�/�/�/�G��_�_�U�+�F�+0�'�6�6�1�R�5�Q�H���2���)�G��7�#�=�>�>�� #'��u�c�l�l�")�+����O�f�,�,�D���O�f�,�,�D��"�Y�.�.������H�H�D����g� �� �s�?G� G#�"G#c�V��|dk(rd��n|dk(rd��ntd|z���fd�}�|fS)Nr c�f�tj|jd��j�S�Nrz)r�md5r�r��xs rTruz?AbstractDigestAuthHandler.get_algorithm_impls.<locals>.<lambda>�s��'�+�+�a�h�h�w�&7�8�B�B�DrU�SHAc�f�tj|jd��j�Sr)rrr�rr s rTruz?AbstractDigestAuthHandler.get_algorithm_impls.<locals>.<lambda>�s��'�,�,�q�x�x��'8�9�C�C�ErUz.Unsupported digest authentication algorithm %rc����|�d|���S)Nryr�)r�drs �rTruz?AbstractDigestAuthHandler.get_algorithm_impls.<locals>.<lambda>�s���!�q�!�,�-rU)rE)r�rrrs @rTrz-AbstractDigestAuthHandler.get_algorithm_impls�sG������D�A� �%� �E�A��,�.7�8�9� 9� -���"�u�rUc��yrWr�)r�rPr�s rTrz+AbstractDigestAuthHandler.get_entity_digest�s��rUrW)r�r�r�r�r�r�r�r r�rrr�rUrTr'r'4s,����I�( � �<�|�rUr'c��eZdZdZdZdZd�Zy)r(z�An authentication protocol defined by RFC 2069 Digest authentication improves on basic authentication because it does not transmit passwords in the clear. r���c�~�t|j�d}|jd|||�}|j�|S)Nr`r�)rr�r�r��r�r rtr3r4rur��retrys rTr�z$HTTPDigestAuthHandler.http_error_401�s@������%�a�(���*�*�+=�+/��g�?����� ��rUN)r�r�r�r6r�r,r�r�rUrTr(r(�s���"�K��M�rUr(c��eZdZdZdZd�Zy)r)�Proxy-Authorizationr(c�f�|j}|jd|||�}|j�|Sr�)r�r�r�r*s rTr�z%ProxyDigestAuthHandler.http_error_407�s6���x�x���*�*�+?�+/��g�?����� ��rUN)r�r�r�r�r,r�r�rUrTr)r)�s��'�K��M�rUr)c�,�eZdZdd�Zd�Zd�Zd�Zd�Zy)�AbstractHTTPHandlerNc�j�|�||_ytjjj|_yrW)rr�HTTPConnection� debuglevel�_debuglevel)r�r3s rTr�zAbstractHTTPHandler.__init__�s&��)3�)?�:���T�[�[�E_�E_�Ej�Ej��rUc��||_yrW)r4)r��levels rT�set_http_debuglevelz'AbstractHTTPHandler.set_http_debuglevel�s �� ��rUc��tjjj|j|j��SrW)rrr2�_get_content_lengthrPr��r�r�s rTr9z'AbstractHTTPHandler._get_content_length�s2���{�{�)�)�=�=��L�L���� �"� "rUc��|j}|std��|j��|j}t|t�r d}t|��|j d�s|jdd�|j d�sR|j d�sA|j|�}|�|jdt |��n|jdd�|}|j�r&t|j�\}}t|�\}} |j d�s|jd|�|jjD]9\} }| j�} |j | �r�(|j| |��;|S) N� no host givenz\POST data should be bytes, an iterable of bytes, or a file object. It cannot be of type str.zContent-type�!application/x-www-form-urlencodedr��Transfer-encoding�chunkedr�)r�rrPrrr�r�r�r9r�rr�r r)r�r�)r�r�r�rPr4�content_length�sel_hostrT�sel�sel_pathrjr�s rT�do_request_zAbstractHTTPHandler.do_request_�sj���|�|����?�+�+��<�<�#��<�<�D��$��$�D����n�$��%�%�n�5��/�/�"�7�9��&�&�'7�8�#�.�.�/B�C�!%�!9�!9�'�!B��!�-��3�3�,�c�.�.A�C��3�3�/��<�������$�W�%5�%5�6�K�F�C�!+�C���H�h��!�!�&�)��+�+�F�H�=��;�;�1�1�K�D�%��?�?�$�D��%�%�d�+��/�/��e�<�2� �rUc ��|j}|std��||fd|ji|��}|j|j�t|j�}|j|jj�D��cic]\}}||vr||��c}}�d|d<|j�D� � cic]\} } | j�| ��}} } |jr0i}d}||vr||||<||=|j|j|�� |j|j�|j|j ||j#d���|j'�}|j*r!|j*j)�d |_|j-�|_|j0|_|Scc}}wcc} } w#t$$r} t| ��d } ~ wwxYw#|j)��xYw) z�Return an HTTPResponse object for the request, using http_class. http_class must implement the HTTPConnection API from http.client. r<rQr�� Connectionr-�rur>)�encode_chunkedN)r�rrQ�set_debuglevelr4rr��updaterur��titler�� set_tunnelr�r�r�rPr�r~�getresponser��sockr�rO�reasonr4)r�� http_classr �http_conn_argsr�r%rurGrHrjr��tunnel_headers�proxy_auth_hdr�errrts rTr�zAbstractHTTPHandler.do_opens��� �x�x����?�+�+� �t�C�S�[�[�C�N�C�� ����)�)�*��s�,�,�-��������):�):�)<�-�)<���A��G�+��1��)<�-� .�!(����6=�m�m�o�F�o���s�4�:�:�<��$�o��F�����N�2�N���(�18��1H��~�.��N�+� �L�L��)�)�>�L�B� � $�� � �#�.�.�*�C�L�L�#�(�(�G�),���8K�)L��N�� � ��A� �6�6� �F�F�L�L�N��A�F�� � �"�����������e-��G�� � $��s�m�#�� $�� � �G�G�I��s7�G �8G�AG�G(� G%�G � G%�%G(�(G;rW)r�r�r�r�r7r9rDr�r�rUrTr0r0�s��k�!�"� $�L@rUr0c�*�eZdZd�Zej Zy)r*c�V�|jtjj|�SrW)r�rrr2�r�r s rT� http_openzHTTPHandler.http_open\s���|�|�D�K�K�6�6��<�<rUN)r�r�r�rXr0rDr�r�rUrTr*r*Zs��=�'�2�2�LrUr*rc�2�eZdZdd�Zd�ZejZy)rLNc�*�|�|n#tjjj}tj||�|�Ctjjj}tjj|�}|�||_||_ yrW) rrrr3r0r�� _http_vsn�_create_https_context�check_hostname�_context)r�r3r?r]�http_versions rTr�zHTTPSHandler.__init__esq��'1�'=��4�;�;�C^�C^�Ci�Ci�J��(�(��z�:���#�{�{�:�:�D�D���+�+�;�;�L�I���)�)7��&�#�D�MrUc�n�|jtjj||j��S)NrA)r�rrrr^rWs rT� https_openzHTTPSHandler.https_openos-���<�<���� ;� ;�S�(,� � � �7� 7rU�NNN)r�r�r�r�rar0rDr�r�rUrTrLrLcs�� $� 7�,�7�7� rUrLc�(�eZdZdd�Zd�Zd�ZeZeZy)rNc�R�ddl}|�|jj�}||_yr�)�http.cookiejar� cookiejar� CookieJar)r�rfrs rTr�zHTTPCookieProcessor.__init__xs$��������0�0�2�I�"��rUc�<�|jj|�|SrW)rf�add_cookie_headerr:s rTr�z HTTPCookieProcessor.http_request~s�����(�(��1��rUc�>�|jj||�|SrW)rf�extract_cookies)r�r�r�s rTr5z!HTTPCookieProcessor.http_response�s�����&�&�x��9��rUrW)r�r�r�r�r�r5r�r7r�rUrTrrws��#���!�M�"�NrUrc��eZdZd�Zy)r/c�6�|j}td|z��)Nzunknown url type: %s)r�r)r�r r�s rTrzUnknownHandler.unknown_open�s���x�x���-��4�5�5rUN)r�r�r�rr�rUrTr/r/�s��6rUr/c�v�i}|D]1}|jdd�\}}|ddk(r |ddk(r|dd}|||<�3|S)z>Parse list of key=value strings where keys are not duplicated.�=r`rr�r])r�)�l�parsed�eltrGrHs rTr�r��sU�� �F����y�y��a� ���1��Q�4�3�;�1�R�5�C�<��!�B��A���q� � � �MrUc��g}d}dx}}|D]H}|r||z }d}� |r|dk(rd}�|dk(rd}||z }�$|dk(r|j|�d}�=|dk(rd}||z }�J|r|j|�|D�cgc]}|j���c}Scc}w)apParse lists as described by RFC 2068 Section 2. In particular, parse comma-separated lists where the elements of the list may include quoted-strings. A quoted-string could contain a comma. A non-quoted string could have quotes in the middle. Neither commas nor quotes count if they are escaped. Only double-quotes count, not single-quotes. r�F�\Tr�r)rlr�)r�res�part�escaper �curs rTr�r��s��� �C� �D���F�U�����C�K�D��F����d�{����������C�K�D���#�:��J�J�t���D���#�:��E�����-�2�� � �4��%(�)�S�T�D�J�J�L�S�)�)��)s�-Bc�"�eZdZd�ZdZd�Zd�Zy)r+c���|j}|dddk(rK|dddk7rC|jr7|jdk7r(|j|j�vrtd��y|j |�S)Nr;rdr�rO� localhost�-file:// scheme is supported only on localhost)r�r�� get_namesr�open_local_file)r�r rOs rT� file_openzFileHandler.file_open�sm���l�l���r��7�d�?�s�1�Q�x�3��C�H�H����K�'��8�8�t�~�~�/�/��N�O�O�0��'�'��,�,rUNc��tj�f ttjd�dtjtj ��dz�t_tjStjS#tj$r1tjd�ft_YtjSwxYw)Nr{r;)r+�namesr�r�gethostbyname_ex�gethostname�gaierror� gethostbynamer�s rTr}zFileHandler.get_names�s������$� I�$)��+�+�K�8��;��+�+�F�,>�,>�,@�A�!�D�E�%F��!� � � � �{� � � ���?�?� I�%+�%9�%9�+�%F�$H��!�� � � � I�s�AB�2C� Cc�D�ddl}ddl}|j}|j}t |�} tj|�}|j}|jj|jd��} |j|�d} |jd| xsd|| fz�}|rt|�\}}|rsBt|�|j�vr'|r d|z|z} nd|z} t!t#|d�|| �St'd��#t$$r}t'|��d}~wwxYw) NrT��usegmtz6Content-type: %s Content-length: %d Last-modified: %s � text/plain�file://�rbzfile not on local host)�email.utils� mimetypesr�r�r5re�stat�st_size�utils� formatdate�st_mtime� guess_type�message_from_stringr�_safe_gethostbynamer}rrNr~r)r�r �emailr�r�rq� localfile�statsry�modified�mtyperur��origurl�exps rTr~zFileHandler.open_local_file�s$�����x�x���<�<�� ��*� � ��G�G�I�&�E��=�=�D��{�{�-�-�e�n�n�T�-�J�H��(�(��2�1�5�E�/�e�/�/�K��&�,��h�7�8�9�G��'��-� ��d���1�$�7�4�>�>�;K�K��'�$�.��9�G�'�(�2�G�!�$�y�$�"7��'�J�J��/�0�0��� ��3�-��� �s�C D� D�D�D)r�r�r�rr�r}r~r�rUrTr+r+�s��-� �E�!�1rUr+c�`� tj|�S#tj$rYywxYwrW)rr�r�)r�s rTr�r��s.����#�#�D�)�)���?�?����s��-�-c��eZdZd�Zd�Zy)r,c�$�ddl}ddl}|j}|std��t |�\}}|� |j }nt |�}t|�\}}|rt|�\}}nd}t|�}|xsd}|xsd} tj|�}t|j�\} } | jd�}t!t#t|��}|dd|d}}|r |ds|dd} |j%||||||j&�} |xrdxsd}| D]9}t)|�\}}|j+�d k(s�%|d vs�*|j-�}�;| j/||�\}}d}|j1|j2�d}|r|d|zz }|� |dk\r|d|zz }t5j6|�}t9|||j2�S#t$r}t|��d}~wwxYw#|j:$r}t|�|�d}~wwxYw) Nr�ftp error: no host givenr�rOr]r`r��Dr���a�Ar�r�r%r�zContent-type: %s zContent-length: %d )�ftplibr�r�rr�FTP_PORTrmrrrrr�r~rr�r�r��map�connect_ftprQrr��upper�retrfiler�r�r�r�r� all_errors)r�r r�r�r�r�rmr�r4rf�attrs�dirsrY�fwr��attrr�rt�retrlenrur�r�s rT�ftp_openzFTPHandler.ftp_open�s$�����x�x����5�6�6���%� ��d��<��?�?�D��t�9�D� ��%� ��d��'��-�L�D�&��F��t�}���z�r����2�� ��'�'��-�D�!����.���e��z�z�#����C���&�'���#�2�Y��R��d����Q�����8�D� )��!�!�$���d�D�#�+�+�N�B��<�C�&�3�D���)�$�/���e��:�:�<�6�)��:�:� �;�;�=�D� � �+�+�d�D�1�K�B���G��(�(����6�q�9�E���/�%�7�7���"�w�!�|��1�G�;�;���/�/��8�G��b�'�3�<�<�8�8��1� ��3�-��� ��2� � � )��3�-�S�(�� )�s>�G�1AG/�G/�BG/� G,�G'�'G,�/H�>H � Hc �&�t||||||d��S)NF)� persistent)� ftpwrapper)r�rmr�r�r�r�rQs rTr�zFTPHandler.connect_ftp1s���$���d�D�'�%*�,� ,rUN)r�r�r�r�r�r�rUrTr,r,�s ��2)�h,rUr,c�0�eZdZd�Zd�Zd�Zd�Zd�Zd�Zy)r-c�J�i|_i|_d|_d|_d|_y)Nr�<r�)�cacherQ�soonest�delay� max_connsr�s rTr�zCacheFTPHandler.__init__8s%���� ��������� ���rUc��||_yrW)r�)r��ts rT� setTimeoutzCacheFTPHandler.setTimeout?s ���� rUc��||_yrW)r�)r�rEs rT�setMaxConnszCacheFTPHandler.setMaxConnsBs ����rUc�|�|||dj|�|f}||jvr/tj�|jz|j|<nKt||||||�|j|<tj�|jz|j|<|j �|j|S)NrO)�joinr�r�r�rQr��check_cache)r�rmr�r�r�r�rQr�s rTr�zCacheFTPHandler.connect_ftpEs����D�$�������7���$�*�*�� $� � ��d�j�j� 8�D�L�L���(��v�t�T�)-�w�8�D�J�J�s�O� $� � ��d�j�j� 8�D�L�L��������z�z�#��rUc���tj�}|j|krht|jj ��D]B\}}||ks�|j |j �|j |=|j|=�Dtt|jj���|_t|j �|jk(r�t|jj ��D]0\}}||jk(s�|j |=|j|=ntt|jj���|_yyrW)r�r�r�rQr�r�r��min�valuesror�)r�r�rGrHs rTr�zCacheFTPHandler.check_cachePs ���I�I�K���<�<�1���T�\�\�/�/�1�2���1��q�5��J�J�q�M�'�'�)�� � �1� ����Q�� 3� �4���� 3� 3� 5�6�7����t�z�z�?�d�n�n�,��T�\�\�/�/�1�2���1�����$�� � �1� ����Q��� 3� �t�D�L�L�$7�$7�$9�:�;�D�L� -rUc���|jj�D]}|j��|jj�|jj�yrW)r�r�r��clearrQ)r��conns rT�clear_cachezCacheFTPHandler.clear_cachedsB���J�J�%�%�'�D��J�J�L�(�� � ���������rUN) r�r�r�r�r�r�r�r�r�r�rUrTr-r-5s ����� �<�(rUr-c��eZdZd�Zy)r.c�h�|j}|jdd�\}}|jdd�\}}t|�}|jd�rt j |�}|dd}|sd}t jd|t|�fz�}ttj|�||�S)Nryr`rz;base64i�����text/plain;charset=US-ASCIIz$Content-type: %s Content-length: %d )r�r�r�endswithr~�decodebytesr�r�ror�io�BytesIO)r�r rOrTrP� mediatyperus rT� data_openzDataHandler.data_openks����l�l���y�y��Q�'�����*�*�S��+�� �4� ��%�����i�(��%�%�d�+�D�!�#�2��I��5�I��+�+�,T� ��D� �"�-#�$���"�*�*�T�*�G�S�9�9rUN)r�r�r�r�r�rUrTr.r.js��:rUr.r<�nt)r5r4c��|dddk(r|dd}n |dddk(r|dd}tj�}tj�}t|||��S) zOS-specific conversion from a relative URL of the 'file' scheme to a file system path; not recommended for general use.Nr�z///r;�z//localhost/��rP�errors)r�getfilesystemencoding�getfilesystemencodeerrorsr��pathnamerPr�s rTr5r5�sc���B�Q�<�5� � ���|�H� �c�r�]�n� ,����}�H��,�,�.���.�.�0���x�(�6�B�BrUc��|dddk(rd|z}tj�}tj�}t|||��S)zOS-specific conversion from a file system path to a relative URL of the 'file' scheme; not recommended for general use.Nr;rdr�)rr�r�r r�s rTr4r4�sH���B�Q�<�4���h��H��,�,�.���.�.�0���X���@�@rUc��eZdZdZdZdezZdd�Zd�Zd�Z d�Z d�Zdd �Zdd �Z dd�Zdd�Zd �Zdd�Zdd�Zd�Zerd�Zdd�Zd�Zd�Zd�Zdd�Zy)r9a,Class to open URLs. This is a class rather than just a subroutine because we may need more than one set of global protocol-specific options. Note -- this is a base class for those who don't want the automatic handling of errors type 302 (relocated) and 401 (authorization needed).Nr�c�v�dd|jjiz}tj|td��|� t�}||_|jd�|_|jd�|_ d|jfdg|_g|_tj|_d|_t"|_y) NzW%(class)s style of invoking requests is deprecated. Use newer urlopen functions/methods�classr�)� stacklevel�key_file� cert_filez User-Agent)�Acceptz*/*)r�r�rBrCrDr6rvr�r�r��versionr��_URLopener__tempfilesrer}�_URLopener__unlink� tempcache�ftpcache)r�rv�x509r4s rTr�zURLopener.__init__�s���4�7>����@W�@W�6X�Y��� � �c�-�!�<��?� �l�G��������,�� ����+�.���(�$�,�,�7�9J�K������� � �� ����!�� rUc�$�|j�yrW)r�r�s rT�__del__zURLopener.__del__�s��� � �rUc�$�|j�yrW)�cleanupr�s rTr�zURLopener.close�s�����rUc���|jr2|jD]} |j|��|jdd�=|jr|jj �yy#t$rY�YwxYwrW)r�r�r~r�r�)r�rYs rTr�zURLopener.cleanup�sn������(�(����M�M�$�'�)� � � ��#��>�>��N�N� � �"�������s�A'�' A3�2A3c�:�|jj|�y)zdAdd a header to be used by the HTTP interface only e.g. u.addheader('Accept', 'sound/basic')N)r�rl)r�r�s rT� addheaderzURLopener.addheader�s�� �����t�$rUc���tt|��}t|d��}|jr9||jvr+|j|\}}t |d�}t|||�St |�\}}|sd}||jvr0|j|}t |�\}} t| �\} }| |f}nd}d|z}||_ |jdd�}t||�r|d k(r'|r|j|||�S|j||�S |�t||�|�St||�||�S#tt f$r�t"$r} t#d | �| �d} ~ wwxYw)z6Use URLopener().open(file) instead of open(file, 'r').z%/:=&?~#+!$,;'@()*[]|�rQr�rYN�open_�-r�r~zsocket error)r rr r�rNrrrvr r�rCr��open_unknown_proxy�open_unknownr�rrr~)r�rrPrqrurt�urltyperOrf� proxyhostr�r�rjr4s rTrNzURLopener.open�sz����7�+�,����&=�>���>�>�g����7� $���w� 7��H�g��h��%�B��b�'�7�3�3�!�'�*������G��d�l�l�"��L�L��)�E�!+�E�!2��G�Y�'� �2�N�D�(���/�C��E��� ���� ��|�|�C��%���t�T�"�d�.?�&?���.�.�u�g�t�D�D��(�(��$�7�7� 8��|�*�w�t�T�*�3�/�/�*�w�t�T�*�3��5�5���8�$� ��� 8��.�#�.�C�7�� 8�s�D7�$D7�7E!� E�E!c�8�t|�\}}tdd|��)�/Overridable interface to open unknown URL type.� url errorzunknown url type�rr~)r�rrPr�rOs rTr�zURLopener.open_unknowns ���w�'� ��c��k�#5�t�<�<rUc�>�t|�\}}tdd|z|��)r�r�zinvalid proxy for %sr�)r�rfrrPr�rOs rTr�zURLopener.open_unknown_proxys%���w�'� ��c��k�#9�D�#@�%�H�HrUc�,�tt|��}|jr||jvr|j|St|�\}}|�R|r|dk(rK |j |�}|j�}|j �tt|�d�|fS|j||�} |j�} |r t|d�} n�t|�\}}t|xsd�\}}t|xsd�\}}t|xsd�\}}tjj|�d} t!j"| �\}}|j$j'|�tj(|d�} || f}|j�||j|<d}d}d}d}d| vrt+| d �}|r ||||�|j-|�x}rD|t/|�z }| j1|�|dz }|r ||||�|j-|�x}r�D| j � |j �|dk\r||krt3d ||fz|��|S#t$rY���wxYw#| j �wxYw#|j �wxYw)ztretrieve(url) returns (filename, headers) for a local object or (tempfilename, headers) for a remote object.rYr`rZr�r\r]rr^r_ra)r rr�rr~rdr�r5r r~rNrrrerf�splitextrh�mkstempr�rl�fdopenrmrnrorpr)r�rOrqrrrPr��url1rtr�rurv�garbagerf�suffix�fdrwrxryrnrzr{s rT�retrievezURLopener.retrieves����Y�s�^�$���>�>�c�T�^�^�3��>�>�#�&�&���_� ��d���T�T�V�^� ��)�)�$�/���w�w�y����� �#�J�t�$4�Q�$7�8�$�>�>��Y�Y�s�D� !��" ��g�g�i�G���8�T�*�� *�3�� ��� *�4�:�2� 6� ��� +�D�J�B� 7� ��g� *�4�:�2� 6� ��g����)�)�$�/��2��!)�!1�!1�&�!9���X�� � �'�'��1��i�i��D�)�� �!�7�*���>�>�-�*0�D�N�N�3�'���������#�w�.��w�'7�8�9�D���x��T�2�!�w�w�r�{�*�e�*��C��J�&�D��I�I�e�$���M�H�!�"�8�R��6� "�w�w�r�{�*�e�*�� � ���H�H�J��1�9����&�C���,�� &�(� (�� ��[� �� ��F� � ����H�H�J�s9�A I�3CJ�BI,�J� I)�(I)�,I>�>J�Jc���d}d}t|t�r,t|�\}}|rt|�\}}t |�}|}nq|\}}t|�\}}t|�\} } | }d}| j �dk7rd}n6t| �\}} |rt|�\}}|r | �d|�| ��}t|�r|}|stdd��|r>t |�}tj|j��jd�}nd}|r>t |�}tj|j��jd�}nd}||�} i}|rd|z|d<|rd|z|d <|r||d <d|d<|jD] \}}|||<�|�d |d<| jd|||�n| jd||�� | j�}d|j(cxkrdkr(nn%t+||j,d|z|j(�S|j/||j0|j(|j2|j,|�S#t j"j$$rt'd��wxYw)a�Make an HTTP connection using connection_class. This is an internal method that should be called from open_http() or open_https(). Arguments: - connection_factory should take a host name and return an HTTPConnection instance. - url is the url to retrieval or a host, relative-path pair. - data is payload for a POST request or None. Nrz://z http errorr<rzzBasic %sr-r�r�r�rFr=zContent-Typer�r�rGz$http protocol error: bad status liner1r2�http:)rrr rrrr�r}r~r~rr�r�r�r�rMrr� BadStatusLiner�statusrr4� http_errorrtrO)r��connection_factoryrOrP�user_passwd�proxy_passwdr�r��realhostr�r�� proxy_authr�� http_connrur�r�r�s rT�_open_generic_httpzURLopener._open_generic_http\s��������c�3��'��_�N�D�(��$.�t�$4�!��T��t�}���H� �N�D�(�!+�D�!1��L�$�&�x�0�M�G�T��C��K��}�}��&�(���!+�D�!1���$��,6�x�,@�)�K���.5�x��F�H���)�#�D��7�<��A�A��"�<�0�L��)�)�,�*=�*=�*?�@�G�G��P�J��J��!�+�.�K��#�#�K�$6�$6�$8�9�@�@��I�D��D�&�t�,� ����-7�*�-D�G�)�*��(2�T�(9�G�O�$��&�G�F�O� !(����!�_�_�M�F�E�#�G�F�O�-���&I�G�N�#����f�h��g�>����e�X�w��?� C� �,�,�.�H��(�/�/�'�C�'��h����g��m�&�o�o�/� /��?�?��X�[�[�������(�,�,��F� F���{�{�(�(� C��A�B�B� C�s�8I�)I,c�X�|jtjj||�S)zUse HTTP protocol.)rrrr2�r�rOrPs rT� open_httpzURLopener.open_http�s!���&�&�t�{�{�'A�'A�3��M�MrUc��d|z}t||�r,t||�}|� ||||||�} n |||||||�} | r| S|j|||||�S)z�Handle http errors. Derived class can override this, or provide specific handlers named http_error_DDD where DDD is the 3-digit error code.z http_error_%d)r�r�r) r�rOrt�errcode�errmsgrurPrjr�rws rTr zURLopener.http_error�so����(���4����T�4�(�F��|���R��&�'�B����R��&�'�4�H���f�}��&�&�s�B����I�IrUc�@�|j�t||||d��)z>Default error handler: close the connection and raise OSError.N)r�r�r�rOrtrrrus rTrzURLopener.http_error_default�s�� ��� ���W�f�g�t�<�<rUc�r�|js|jr}tjjj }tjj |�}|j|j|j�|j� d|_nd}tjj ||��S)NTrA) r�r�rrrr[r\�load_cert_chain�post_handshake_auth)r�r�r_r?s rT�_https_connectionzURLopener._https_connection�s����}�}����#�{�{�:�:�D�D���+�+�;�;�L�I���'�'����� � �F��.�.�:�26�G�/����;�;�.�.�t�W�.�E�ErUc�<�|j|j||�S)zUse HTTPS protocol.)rr rs rT� open_httpszURLopener.open_https�s���*�*�4�+A�+A�3��M�MrUc��t|t�std��|dddk(r)|dddk7r!|ddj�dk7rt d ��|j|�S) z/Use local file or FTP depending on form of URL.zEfile error: proxy support for file protocol currently not implementedNr;rdr�rOr�z localhost/r|)rrrr�rEr~r�s rT� open_filezURLopener.open_file�sb���#�s�#��b�c�c��r��7�d�?�s�1�Q�x�3��3�q��9�?�?�3D��3T��L�M�M��'�'��,�,rUc���ddl}ddl}t|�\}}t|�} t j |�}|j} |jj|jd��} |j|�d}|jd|xsd| | fz�}|s&|} |dddk(rd |z} t!t#|d �|| �St%|�\}}|sht'j(|�t+�ft-�zvr=|} |dddk(rd |z} n|dddk(rt/d |z��t!t#|d �|| �Std��#t$r%}t|j|j��d}~wwxYw)zUse local file.rNTr�z6Content-Type: %s Content-Length: %d Last-modified: %s r�r`rOr�r�r;z./zAlocal file url may start with / or file:. Unknown url of type: %sz#local file error: not on local host)r�r�r r5rer�r~r�strerrorrqr�r�r�r�r�r�rrNrrr�r{�thishostrE)r�rOr�r�r�rY� localnamer��eryr�r�ru�urlfiler�s rTr~zURLopener.open_local_file�s�������_� ��d� ��&� � 3��G�G�I�&�E��}�}���;�;�)�)�%�.�.��)�F���$�$�S�)�!�,��+�%�+�+�G� � "�l�D�(�3� 4�5����G��B�Q�x�3��#�d�*���d�9�d�3�W�g�F�F���%� ��d���#�#�D�)�y�{�n�x�z�.I�J��G��B�Q�x�3��#�d�*���b�q��T�!� �!d�gj�!j�k�k��d�9�d�3�W�g�F�F��<�=�=��-� 3��1�:�:�q�z�z�2�2�� 3�s�E� E4� E/�/E4c�8�t|t�std��ddl}t |�\}}|std��t|�\}}t |�\}}|rt|�\}}nd}t|�}t|xsd�}t|xsd�}tj|�}|sddl}|j}nt|�}t|�\}} t|�}|jd�} | dd| d}} | r | ds| dd} | r | dsd| d<|||dj!| �f}t#|j$�t&kDrLt)|j$�D]4} | |k7s� |j$| }|j$| =|j+��6 ||j$vrt-||||| �|j$|<|sd }nd }| D]9}t/|�\}}|j1�dk(s�%|dvs�*|j3�}�;|j$|j5||�\}}|j7d |z�d}d}|r|d|zz }|� |dk\r|d|zz }t9j:|�}t=||d |z�S#t?�$r}td|���|�d}~wwxYw)zUse FTP protocol.zCftp error: proxy support for ftp protocol currently not implementedrNr�r�rOr]r`r�r�r�r�zftp:zContent-Type: %s zContent-Length: %d �ftp error: ) rrrr�r rrrrrr�r�r�rmrr�r�ror��MAXFTPCACHEr�r�r�rr�r�r�r�r�r�r� ftperrors)r�rOr�r�rfr�rmr�r�r�r�rYr�rGrHr�r�r�rtr�r�rur�s rT�open_ftpzURLopener.open_ftps����#�s�#��`�a�a����_� ��d��8�$>�?�?���%� ��d���%� ��d���T� 2���v��f��t�}���t�z�r�"�����2�&���#�#�D�)�����?�?�D��t�9�D� ��&���e��t�}���z�z�#����#�2�Y��R��d����Q���Q�R�����Q��3��a���D�$������.���t�}�}���+��$�-�-�(����8�� � �a�(�A�� � �a�(��G�G�I� )� 9��$�-�-�'��t�V�T�4��>�� � �c�"�����$���)�$�/���e��:�:�<�6�)��:�:� �;�;�=�D� � !�M�M�#�.�7�7��d�C�M�R���(�(��#��6�q�9�E��G���/�%�7�7���"�w�!�|��1�G�;�;���/�/��8�G��b�'�6�C�<�8�8���{� 9��[���.�/�S�8�� 9�s&�AI8�(I8�-B I8�8 J�J�Jc �T�t|t�std�� |jdd�\}}|sd}|j d�}|dk\rd ||d vr||dzd }|d |}nd}g}|jdtjd tjtj���z�|jd|z�|dk(r4tj|jd��jd�}nt|�}|jdt!|�z�|jd�|j|�dj#|�}t%j&|�}t)j*|�}t-|||�S#t$r tdd��wxYw)zUse "data" URL.zEdata error: proxy support for data protocol currently not implementedrr`z data errorzbad data URLr��;rroNr�zDate: %sz%a, %d %b %Y %H:%M:%S GMTzContent-type: %sr~rzzlatin-1zContent-Length: %d� )rrrr�rEr~�rfindrlr��strftime�gmtimer~r�r�r�rror�r�r�r��StringIOr) r�rOrPr��semirPr4ru�fs rT� open_datazURLopener.open_dataFs����#�s�#��b�c�c� 8��9�9�S�!�,�L�T�4��0�D��z�z�#����1�9��D���K�/��D��F�G�}�H����;�D��H���� � �:�d�m�m�,G�,0�K�K�� � ��,D�F�F� G�� � �%��,�-��x���%�%�d�k�k�'�&:�;�B�B�9�M�D��4�=�D�� � �'�#�d�)�3�4�� � �2��� � �4���i�i��n���+�+�C�0���K�K�����!�W�c�*�*��5� 8��,��7�7� 8�s�F�F'rWrb)r�r�r�r6r�r�r�r�r�r�r�r�rNr�r�rrrr rrFr r"r$r~r/r9r�rUrTr9r9�s�����K� �;�.�G�!�4��#�%�"8�H=� I�:�|ZF�xN�J� =� � F� N�-�>�@89�t'+rUr9c��eZdZdZd�Zd�Zdd�Zd�Zdd�Zdd�Z dd �Z dd �Z dd�Z dd�Z dd �Zdd�Zdd�Zdd�Zdd�Zd�Zy)r:z?Derived class with handlers for errors we can handle (perhaps).c�`�tj|g|��i|��i|_d|_d|_y)Nrr<)r9r�� auth_cache�tries�maxtries)r�r��kwargss rTr�zFancyURLopener.__init__ss/�����4�1�$�1�&�1������ ��� rUc�$�t||d|z|�S)z3Default error handling -- don't raise an exception.r )rrs rTrz!FancyURLopener.http_error_defaultys���"�g�w��}�g�>�>rUNc�>�|xjdz c_ |jrQ|j|jk\r8t|d�r |j}n|j}|||dd|�d|_S|j||||||�}|d|_S#d|_wxYw)z%Error 302 -- relocated (temporarily).r`�http_error_500r'z)Internal Server Error: Redirect Recursionr)r=r>r�rBr�redirect_internal) r�rOrtrrrurPr�rws rTr^zFancyURLopener.http_error_302}s���� � �a�� � ��}�}����t�}�}�!<��4�!1�2��.�.�D��2�2�D��C��S�G�#�%��D�J� �+�+�C��W�f�,3�T�;�F���D�J���D�J�s�AB�4B� Bc��d|vr|d}nd|vr|d}ny|j�t|jdz|z|�}t|�}|jdvrt|||d|zz||��|j |�S)NrKrLryrMz( Redirection to url '%s' is not allowed.)r�rr�rrTrrN) r�rOrtrrrurPrDr[s rTrCz FancyURLopener.redirect_internal�s����� ��Z�(�F� �g� ��U�^�F�� ��� �����S��3�.��7���F�#���?�?�">�>��F�G�"�F��O�P�#�R�)� )� �y�y�� � rUc�.�|j||||||�S)z*Error 301 -- also relocated (permanently).�r^�r�rOrtrrrurPs rTr_zFancyURLopener.http_error_301�����"�"�3��G�V�W�d�K�KrUc�.�|j||||||�S)z;Error 303 -- also relocated (essentially identical to 302).rFrGs rTr`zFancyURLopener.http_error_303�rHrUc�\�|�|j||||||�S|j|||||�S)z1Error 307 -- relocated, but turn POST into error.)r^rrGs rTrazFancyURLopener.http_error_307��;���<��&�&�s�B����$�O�O��*�*�3��G�V�W�M�MrUc�\�|�|j||||||�S|j|||||�S)z1Error 308 -- relocated, but turn POST into error.)r_rrGs rTrbzFancyURLopener.http_error_308�rKrUc���d|vrtj||||||�|d}tjd|�} | stj||||||�| j �\} }| j�dk7rtj||||||�|stj||||||�d|jzdz}|�t||�||�St||�|||�S)z_Error 401 -- authentication required. This function supports Basic authentication only.r��![ ]*([^ ]+)[ ]+realm="([^"]*)"r��retry_�_basic_auth�r9rr��matchr�r�r�r�� r�rOrtrrrurPr+�stuffrRrTr�rjs rTr�zFancyURLopener.http_error_401�s���W�,��(�(��s�B�)0�&�'� C��*�+�����?��G����(�(��s�B�)0�&�'� C����� ����<�<�>�W�$��(�(��s�B�)0�&�'� C���(�(��s�B���� ��$�)�)�#�m�3���<�%�7�4��%�c�5�1�1�%�7�4��%�c�5�$�7�7rUc���d|vrtj||||||�|d}tjd|�} | stj||||||�| j �\} }| j�dk7rtj||||||�|stj||||||�d|jzdz}|�t||�||�St||�|||�S)zeError 407 -- proxy authentication required. This function supports Basic authentication only.r�rNr��retry_proxy_rPrQrSs rTr�zFancyURLopener.http_error_407�s�� �w�.��(�(��s�B�)0�&�'� C��,�-�����?��G����(�(��s�B�)0�&�'� C����� ����<�<�>�W�$��(�(��s�B�)0�&�'� C���(�(��s�B���� ��� � �)�M�9���<�%�7�4��%�c�5�1�1�%�7�4��%�c�5�$�7�7rUc��t|�\}}d|z|z}|jd}t|�\}} t| �\} } | jd�dz}| |d} |j | ||�\}} |s| syt|d���dt| d���d| ��} d| z| z|jd<|�|j |�S|j ||�S)N�http://rrer`r�r�ry�r rvrr��get_user_passwdr rN�r�rOr�rPr�r�rDrfr�r�� proxyselectorr�rmr�s rT�retry_proxy_http_basic_authz*FancyURLopener.retry_proxy_http_basic_auth�s���#�C����h��T�!�H�,�����V�$��'��.����#-�i�#8� � �=��N�N�3��!�#���a�b�M� ��+�+�I�u�a�@���f����"'��2�"6�"'��R�"8�)�E� �(�9�4�}�D����V���<��9�9�V�$�$��9�9�V�T�*�*rUc��t|�\}}d|z|z}|jd}t|�\}} t| �\} } | jd�dz}| |d} |j | ||�\}} |s| syt|d���dt| d���d| ��} d| z| z|jd<|�|j |�S|j ||�S)N�https://r�rer`r�r�ryrYr[s rT�retry_proxy_https_basic_authz+FancyURLopener.retry_proxy_https_basic_auth s���#�C����h��d�"�X�-�����W�%��'��.����#-�i�#8� � �=��N�N�3��!�#���a�b�M� ��+�+�I�u�a�@���f����"'��2�"6�"'��R�"8�)�E� � *�Y� 6�� F����W���<��9�9�V�$�$��9�9�V�T�*�*rUc� �t|�\}}|jd�dz}||d}|j|||�\}}|s|syt|d���dt|d���d|��}d|z|z} |�|j | �S|j | |�S)Nrer`r�r�ryrX�r r�rZr rN� r�rOr�rPr�r�r�rmr�rDs rTr�z$FancyURLopener.retry_http_basic_auth s���#�C����h��I�I�c�N�Q����A�B�x���+�+�D�%��;���f����"�4�b�1�"�6��3�T�;���T�!�H�,���<��9�9�V�$�$��9�9�V�T�*�*rUc� �t|�\}}|jd�dz}||d}|j|||�\}}|s|syt|d���dt|d���d|��}d|z|z} |�|j | �S|j | |�S)Nrer`r�r�ryr_rbrcs rT�retry_https_basic_authz%FancyURLopener.retry_https_basic_auth% s���#�C����h��I�I�c�N�Q����A�B�x���+�+�D�%��;���f����"�4�b�1�"�6��3�T�;���d�"�X�-���<��9�9�V�$�$��9�9�V�T�*�*rUc���|dz|j�z}||jvr|r|j|=n|j|S|j||�\}}|s|r||f|j|<||fS)Nre)r�r<�prompt_user_passwd)r�r�r�r�r�rmr�s rTrZzFancyURLopener.get_user_passwd3 sv���c�k�D�J�J�L�(���$�/�/�!���O�O�C�(����s�+�+��.�.�t�U�;���f��6�4��.�4�?�?�3�/��V�|�rUc ��ddl} td|�d|�d��}|jd|�d|�d|�d��}||fS#t$r t�YywxYw) z#Override this in a GUI environment!rNzEnter username for z at z: zEnter password for z in r�)�getpass�input�KeyboardInterrupt�print)r�r�r�rirmr�s rTrgz!FancyURLopener.prompt_user_passwd> sT��� ��E�4�H�I�D��_�_��u�d�&$�%�F���<��� � ��G�� �s�07�A �A rW)NF)r)r�r�r�r6r�rr^rCr_r`rarbr�r�r]r`r�rerZrgr�rUrTr:r:psm��I��?��$!�8L�L�N�N�FJ��8�2FJ��8�2+�$+�$+�+� � rUr:c�D�t�tjd�atS)z8Return the IP address of the magic hostname 'localhost'.r{)� _localhostrr�r�rUrTr{r{N s �����)�)�+�6� ��rUc��t�: ttjtj��d�atStS#tj $r)ttjd�d�aYtSwxYw)z,Return the IP addresses of the current host.r;r{)� _thishostr�rr�r�r�r�rUrTr'r'V sw���� G��f�5�5�f�6H�6H�6J�K�A�N�O�I���9������ G��f�5�5�k�B�1�E�F�I��� G�s�3A�4B�Bc�:�t�ddl}|jatS)z1Return the set of errors raised by the FTP class.Nr)� _ftperrorsr�r�)r�s rTr.r.a s������&�&� ��rUc�D�t�tjd�atS)z%Return an empty email Message object.r�)� _noheadersr�r�r�rUrT� noheadersruj s �����.�.�r�2� ��rUc�@�eZdZdZ d d�Zd�Zd�Zd�Zd�Zd�Z d �Z y)r�z;Class used by open_ftp() for cache of open FTP connections.Nc���||_||_||_||_||_||_d|_||_ |j�y#|j��xYwr�) rmr�r�r�r�rQ�refcount� keepalive�initr�)r�rmr�r�r�r�rQr�s rTr�zftpwrapper.__init__w s[���� ������ ��� ��� ������ �#��� ��I�I�K�� ��J�J�L��s�A�Ac��ddl}d|_|j�|_|jj |j |j|j�|jj|j|j�dj|j�}|jj|�y)NrrO)r��busy�FTPrN�connectr�r�rQ�loginrmr�r�r��cwd)r�r��_targets rTrzzftpwrapper.init� sw����� ��:�:�<�����������D�I�I�t�|�|�<������t�y�y�$�+�+�.��(�(�4�9�9�%�������W�rUc�8�ddl}|j�|dvrd}d}nd|z}d} |jj|�d}|r&|s$ d|z}|jj |�\}}|s�|jjd�|rY|jj�} |jj|� |jj| �d|z}nd}|jj |�\}}d|_t|jd �|j�} |xj dz c_|j#�| fS#|j$r/|j�|jj|�Y��TwxYw#|j$r+}t|�dddk7rtd |���|�Yd}~��cd}~wwxYw#|j$r}td |z�|�d}~wwxYw#|jj| �wxYw)Nr)r%r�zTYPE Ar`zTYPE zRETR r��550r,z ftp error: %rzLIST �LISTr�)r��endtransferrN�voidcmdr�rz�ntransfercmd� error_permrr�pwdr�r|r�makefile� file_closerxr�)r�rYr�r��cmd�isdirr�r�rOr��ftpobjs rTr�zftpwrapper.retrfile� s���������:��X�s�q�u��d�N�c�A�E� "��H�H���S�!����� G���n�� $��� 5� 5�c� :� ��g���H�H���X�&���h�h�l�l�n��&�M������T�*��H�H�L�L��%���n���� �H�H�1�1�#�6�M�D�'��� ��d�m�m�D�1�4�?�?�C��� � ��� �� � ���� � ��G� � � "��I�I�K��H�H���S�!� "���$�$� G��v�;�r��?�e�+�"�[���#9�:��F�,�� G��"�,�,�M�&���'?�@�f�L��M���H�H�L�L��%�sM�E�#F�&G�:F�F�G�( G�G�G9�%G4�4G9�9G<�<Hc��|jsyd|_ |jj�y#t�$rYywxYwr�)r|rN�voidrespr.r�s rTr�zftpwrapper.endtransfer� s<���y�y���� � ��H�H������{� �� �s�1� A�Ac�R�d|_|jdkr|j�yy)NFr)ryrx� real_closer�s rTr�zftpwrapper.close� s$������=�=�A���O�O��rUc��|j�|xjdzc_|jdkr|js|j�yyy)Nr`r)r�rxryr�r�s rTr�zftpwrapper.file_close� s@������� � ��� ��=�=�A��d�n�n��O�O��'5�rUc��|j� |jj�y#t�$rYywxYwrW)r�rNr�r.r�s rTr�zftpwrapper.real_close� s5������ ��H�H�N�N����{� �� �s�-� =�=)NT)r�r�r�r6r�rzr�r�r�r�r�r�rUrTr�r�t s/��E�?C� �� �*!�X�� �rUr�c���i}g}tjj�D]s}t|�dkDs�|ddk(s�|ddj �dk(s�2tj|}|ddj �}|j|||f�|s�o|||<�udtjvr|j dd�|D])\}}}|ddd k(s�|r|||<�|j |d��+|S) aReturn a dictionary of scheme -> proxy server URL mappings. Scan the environment for variables named <scheme>_proxy; this seems to be the standard convention. If you need a different way, you can pass a proxies dictionary to the [Fancy]URLopener constructor. r�i����r����Nrf�REQUEST_METHODr�_proxy)re�environ�keysror�rlr�)rv�environmentrjr�� proxy_names rT�getproxies_environmentr�� s����G��K�� � ���!���t�9�q�=�T�"�X��_��b�c����1B�g�1M��J�J�t�$�E��c�r����*�J�����e�Z�8�9��&+�� �#�"��2�:�:�%����F�D�!�#.���e�Z����9�� ��&+�� �#����J��-� $/��NrUc��|� t�} |d}|dk(ry|j�}t|�\}}|j d�D]k}|j�}|s�|j d�}|j�}||k(s||k(ryd|z}|j|�s|j|�s�kyy#t$rYywxYw)z�Test if proxies should not be used for a particular host. Checks the proxy dict for the value of no_proxy, which should be a list of comma separated DNS suffixes, or '*' for all hosts. �noF�*Tr�.)r�rr�rr�r��lstripr�)r�rv�no_proxy�hostonlyr�rjs rT�proxy_bypass_environmentr�� s�����(�*����4�=���3����:�:�<�D���%�N�H�d����s�#���z�z�|����;�;�s�#�D��:�:�<�D��4��4�4�<����:�D�� � ��&�$�-�-��*=��$���)����s�B7�7 C�Cc��ddlm}ddlm}m}t |�\}}d�}d|vr|dryd} t||��}|j d d �D]�} | s�tjd| �} | �|�}|| jd��}| jd �}|�'d| jd�jd�dzz}nt|dd�}|dks|dkDr��d|z }||z ||z k(s��y||| �s��yy#|$rY��wxYw)aj Return True iff this host shouldn't be accessed using a proxy This function uses the MacOSX framework SystemConfiguration to fetch the proxy information. proxy_settings come from _scproxy._get_proxy_settings or get mocked ie: { 'exclude_simple': bool, 'exceptions': ['foo.bar', '*.bar.com', '127.0.0.1', '10.1', '10.0/16'] } r��fnmatch)�AddressValueError�IPv4Addressc���|jd�}ttt|��}t |�dk7r |gd�zdd}|ddz|ddzz|dd zz|d zS)Nr�r;)rrrrr�r`r�r;r�r�)r�r�r�rmro)�ipAddrr�s rT�ip2numz,_proxy_bypass_macosx_sysconf.<locals>.ip2num3 sm�����S�!���S��e�_�%���u�:��?��\�)�2�A�.�E��a��B��5��8�r�>�2�e�A�h�!�m�D�u�Q�x�O�OrUr��exclude_simpleTN� exceptionsr�z(\d+(?:\.\d+)*)(/\d+)?r`r;r�� F)r�� ipaddressr�r�rrmr�r�rR�group�count) r��proxy_settingsr�r�r�r�r�r��hostIPr�rEr��masks rT�_proxy_bypass_macosx_sysconfr�" s,�� �8���%�N�H�d�P��$���*�+�� �F� ��[��*�+�� �#�#�L�"�5���h��H�H�.��6���=�V�/��!�'�'�!�*�%�D��7�7�1�:�D��|��A�G�G�A�J�,�,�S�1�A�5�6���4���8�}���a�x�4�"�9����9�D��$��D�D�L�1�� �T�5� !��/6�2��9� �� �s�C;�;D�Dc��ddlm}t|�\}}|jd�}|D])}|j�}|dk(rd|vs�y|||�s�)yy)a Return True if the host should bypass the proxy server. The proxy override list is obtained from the Windows Internet settings proxy override registry value. An example of a proxy override value is: "www.example.com;*.example.net; 192.168.0.1" rr�r1z<local>r�TF)r�rr�r�)r��overrider�r��proxy_overrider�s rT�_proxy_bypass_winreg_overrider�b s\�� ����G�D�!��^�^�C�(�N����z�z�|���9���$��� �T�4� ���rU�darwin)�_get_proxy_settings�_get_proxiesc�.�t�}t||�SrW)r�r�)r�r�s rT�proxy_bypass_macosx_sysconfr�} s��,�.��+�D�.�A�ArUc��t�S)z�Return a dictionary of scheme -> proxy server URL mappings. This function uses the MacOSX framework SystemConfiguration to fetch the proxy information. )r�r�rUrT�getproxies_macosx_sysconfr�� s���~�rUc�H�t�}|rt||�St|�S)z�Return True, if host should be bypassed. Checks proxy settings gathered from the environment, if specified, or from the MacOSX framework SystemConfiguration. )r�r�r��r�rvs rTr}r}� s'��)�*���+�D�'�:�:�.�t�4�4rUc�.�t�xs t�SrW)r�r�r�rUrTr6r6� s��%�'�F�+D�+F�FrUc���i} ddl} |j|jd�}|j |d�d}|r�t|j |d�d�}d|vrd|vrdj |�}|jd�D]F}|jdd �\}}tjd |�s|dvrd|z}n |d k(rd|z}|||<�H|jd �rJtjdd|d �}|jd�xs||d<|jd�xs||d<|j�|S#t$r|cYSwxYw#tttf$rY|SwxYw)zxReturn a dictionary of scheme -> proxy server URL mappings. Win32 uses the registry to store proxies. rN�;Software\Microsoft\Windows\CurrentVersion\Internet Settings�ProxyEnable�ProxyServerror1zhttp={0};https={0};ftp={0}r`z (?:[^/:]+)://)rr�rNrX�sockszsocks://z ^socks://z socks4://rr�)�winreg�ImportError�OpenKey�HKEY_CURRENT_USER�QueryValueExrr�r�r�rRr�r��Closer~rEr�)rvr��internetSettings�proxyEnable�proxyServer�pr��addresss rT�getproxies_registryr�� s����� ��" �%�~�~�f�.F�.F�N� P�� �-�-�.>�/<�>�>?�A�K��!�&�"5�"5�6F�7D�#F�FG�#I�J���k�)�c��.D�">�"E�"E�k�"R�K�$�*�*�3�/�A�()����Q��%�H�g��8�8�O�W�=�#�'?�?�&/�'�&9�G�%��0�&0�7�&:�G�(/�G�H�%�0��;�;�w�'� �f�f�\�;���@P�Q�G�&-�k�k�&�&9�&D�W�G�F�O�'.�{�{�7�';�'F�w�G�G�$��"�"�$����M� ��N� ��B��Y�/� � ��� �s#�D:�D/E�:E�E�E#�"E#c�.�t�xs t�S)z�Return a dictionary of scheme -> proxy server URL mappings. Returns settings gathered from the environment, if specified, or the registry. )r�r�r�rUrTr6r6� s��&�'�@�+>�+@�@rUc� � ddl} |j|jd�}|j |d�d}t|j |d�d�}|r|syt||�S#t$rYywxYw#t$rYywxYw)NrFr�r�� ProxyOverride)r�r�r�r�r�rr~r�)r�r�r�r�� proxyOverrides rT�proxy_bypass_registryr�� s��� �� �%�~�~�f�.F�.F�N� P�� �-�-�.>�/<�>�>?�A�K��� 3� 3�4D�5D�!F�FG�!I�J�M� �-��,�T�=�A�A��� �� ��� �� �s#�A'�AA6�' A3�2A3�6 B�Bc�H�t�}|rt||�St|�S)z�Return True, if host should be bypassed. Checks proxy settings gathered from the environment, if specified, or the registry. )r�r�r�r�s rTr}r}� s'��)�*���+�D�'�:�:�(��.�.rUrbrW)~r6r~r�r�r�http.clientrr�rer�rrVrr�rhrbrB�urllib.errorrrr�urllib.parserrrr r rrr rrrrrrrrrr�urllib.responserrrGrFr��__all__�version_infor�rMrr1r2rkr7r8r��ASCIIr�r�rrr3rr0rrror r!r"r#r$r%r&�urandomrr'r(r)r0r*r�rrLrlrr/r�r�r+r�r,r-r.r-rj� nturl2pathr5r4r�r9r:rnr{rpr'rrr.rtrur�r�r�r�r��platform�_scproxyr�r�r�r�r}r6r�r�r�rUrT�<module>r�s���C�f� ���� � � � � � �����C�B�"�"�"�"�"� 5����I���$��(�(��!�,�,�� ���F�$B�$B�M+��4�5�$�M+�^���:�x��r�z�z�(�B�H�H�-��� k"�k"�ZI+�I+�^"�H8�8�&#��#�";�k�;�n2�+�n2�b,�B)>�;�)>�V=*�=*�@G�o�G�3�#B�3�>k#�k#�^�3�[���4�k�� �z�z��O�O�d�K�)B��$ �[�*C� �s�+�s�l3�%�3��4�;�;�)�*�8�*�8�$�N�N�>�"�#�+�#�$6�[�6� �)*�V11�+�11�f�7,��7,�r3�j�3�j:�+�:�B���7�7�d�?�5�5�C� A���+�+�DX�Y�X�z� �� � ��� ��� ��a�a�H#�J �J<�@�0�<�<�8��:�B��5�G��W�W��_�/�bA�B�(/�(�J�+�L��WT���I��s�:K�K�KPKS�\#F����)__pycache__/request.cpython-312.opt-2.pycnu�[���� T��h[���� ddlZddlZddlZddlZddlZddlZddlZddlZddl Z ddl Z ddlZddlZddl Z ddlZddlZddlmZmZmZddlmZmZmZmZmZmZmZmZmZmZmZm Z m!Z!m"Z"m#Z#m$Z$m%Z%m&Z&ddl'm(Z(m)Z) ddl*Z*dZ+gd�Z-dej\dd zZ/da0de jbfddddd �d�Z2d�Z3gZ4dfd �Z5d�Z6ejndejp�Z9d�Z:Gd�d�Z;Gd�d�Z<d�Z=Gd�d�Z>Gd�de>�Z?Gd�de>�Z@Gd�de>�ZAd�ZBGd�d e>�ZCGd!�d"�ZDGd#�d$eD�ZEGd%�d&eE�ZFGd'�d(�ZGGd)�d*eGe>�ZHGd+�d,eGe>�ZIej�ZKGd-�d.�ZLGd/�d0e>eL�ZMGd1�d2e>eL�ZNGd3�d4e>�ZOGd5�d6eO�ZPeQej�d7�rGd8�d9eO�ZSe-j�d9�Gd:�d;e>�ZUGd<�d=e>�ZVd>�ZWd?�ZXGd@�dAe>�ZYdB�ZZGdC�dDe>�Z[GdE�dFe[�Z\GdG�dHe>�Z]dIZ^ej�dJk(r ddKl`maZambZbndL�ZadM�ZbiZcGdN�dO�ZdGdP�dQed�ZedafdR�ZgdahdS�ZidajdT�ZkdaldU�ZmGdV�dW�ZndX�ZodgdY�ZpdZ�Zqd[�Zrej�d\k(rdd]ltmuZumvZvd^�Zwd_�Zxd`�Zyda�Zzyej�dJk(r db�Z{dc�Zzdd�Z|de�ZyyeoZzepZyy#e,$rdZ+Y��[wxYw)h�N)�URLError� HTTPError�ContentTooShortError)�urlparse�urlsplit�urljoin�unwrap�quote�unquote� _splittype� _splithost� _splitport� _splituser�_splitpasswd� _splitattr�_splitquery�_splitvalue� _splittag� _to_bytes�unquote_to_bytes� urlunparse)� addinfourl�addclosehookTF)!�Request�OpenerDirector�BaseHandler�HTTPDefaultErrorHandler�HTTPRedirectHandler�HTTPCookieProcessor�ProxyHandler�HTTPPasswordMgr�HTTPPasswordMgrWithDefaultRealm�HTTPPasswordMgrWithPriorAuth�AbstractBasicAuthHandler�HTTPBasicAuthHandler�ProxyBasicAuthHandler�AbstractDigestAuthHandler�HTTPDigestAuthHandler�ProxyDigestAuthHandler�HTTPHandler�FileHandler� FTPHandler�CacheFTPHandler�DataHandler�UnknownHandler�HTTPErrorProcessor�urlopen�install_opener�build_opener�pathname2url�url2pathname� getproxies�urlretrieve� urlcleanup� URLopener�FancyURLopenerz%d.%d�)�cafile�capath� cadefault�contextc��� |s|s|r�ddl}|jdtd�|�td��tstd��tjt jj||��}|jdg�t|��}t|�} n3|rt|��}t|�} nt� t�xa} nt} | j|||�S) NrzJcafile, capath and cadefault are deprecated, use a custom context instead.r;zDYou can't pass both context and any of cafile, capath, and cadefaultzSSL support not available)r<r=zhttp/1.1�r?)�warnings�warn�DeprecationWarning� ValueError� _have_ssl�ssl�create_default_context�Purpose�SERVER_AUTH�set_alpn_protocols�HTTPSHandlerr3�_opener�open) �url�data�timeoutr<r=r>r?rB� https_handler�openers �'/usr/lib64/python3.12/urllib/request.pyr1r1�s���0�d��9���� � �0�1C�Q� H������ ���8�9�9��,�,�S�[�[�-D�-D�4:�4:�<�� �"�"�J�<�0�$�W�5� ��m�,�� �$�W�5� ��m�,�� ��'�>�)��&����;�;�s�D�'�*�*�c��|ay�N)rM)rSs rTr2r2�s���GrUc�� t|�\}}tjt||��5}|j �}|dk(r,|s*t jj|�|fcddd�S|r t|d�}n7tjd��}|j}tj|�|5||f} d} d}d}d} d|vrt|d �}|r || | |�|j| �x}rD|t!|�z }|j#|�| d z } |r || | |�|j| �x}r�Dddd�ddd�dk\r|krt%d||fz �� S#1swY�.xYw#1swY�2xYw)N�file�wbF)�delete� ���r�content-length�Content-Length��1retrieval incomplete: got only %i out of %i bytes)r� contextlib�closingr1�info�os�path�normpathrN�tempfile�NamedTemporaryFile�name�_url_tempfiles�append�int�read�len�writer)rO�filename� reporthookrP�url_typerf�fp�headers�tfp�result�bs�sizern�blocknum�blocks rTr7r7�s���� ��_�N�H�d� � � �G�C��.� /�2��'�'�)���v��h��7�7�#�#�D�)�7�2� 0� /���x��&�C��-�-�U�;�C��x�x�H��!�!�(�+� ��w�&�F��B��D��D��H��7�*��7�#3�4�5����8�R��.��7�7�2�;�&�%�&���E� �"��� � �%� ��A� ����x��T�2��7�7�2�;�&�%�&��! 0�F�q�y�T�D�[�"�?��T�l� �"�$� $��M�1�S��! 0� /�s+�8E4�1AE4�9BE(�;E4�(E1 �-E4�4E=c�� tD]} tj|��tdd�=trdayy#t$rY�9wxYwrW)rkre�unlink�OSErrorrM)� temp_files rTr8r8sK��:�#� � ��I�I�i� �$� �q�������� �� �s�6� A�Az:\d+$c�� |j}t|�d}|dk(r|jdd�}tj d|d�}|j�S)Nr`��Host)�full_urlr� get_header�_cut_port_re�sub�lower)�requestrO�hosts rT�request_hostr�)s]���� � �C��C�=���D��r�z��!�!�&�"�-�����B��a�(�D��:�:�<�rUc��eZdZdidddfd�Zed��Zejd��Zejd��Zed��Zejd��Zejd ��Zd �Z d�Z d�Zd �Zd�Z d�Zd�Zd�Zdd�Zd�Zd�Zy)rNFc��||_i|_i|_d|_||_d|_|j �D]\}}|j||��|�t|�}||_ ||_ |r||_yyrW)r�ru�unredirected_hdrs�_datarP�_tunnel_host�items� add_headerr��origin_req_host�unverifiable�method) �selfrOrPrur�r�r��key�values rT�__init__zRequest.__init__;s����� ����!#����� ��� � ���!�-�-�/�J�C���O�O�C��'�*��"�*�4�0�O�.���(���� �D�K�rUc�~�|jr&dj|j|j�S|jS)Nz{}#{})�fragment�format� _full_url�r�s rTr�zRequest.full_urlMs,���=�=��>�>�$�.�.�$�-�-�@�@��~�~�rUc��t|�|_t|j�\|_|_|j �yrW)r r�rr��_parse�r�rOs rTr�zRequest.full_urlSs/�� �����(1�$�.�.�(A�%���� ���� rUc�.�d|_d|_d|_y�Nr�)r�r��selectorr�s rTr�zRequest.full_urlZs������� ��� rUc��|jSrW)r�r�s rTrPzRequest.data`s���z�z�rUc�x�||jk7r+||_|jd�r|jd�yyy)N�Content-length)r�� has_header� remove_header)r�rPs rTrPzRequest.datads=���4�:�:���D�J����/�0��"�"�#3�4�1�rUc��d|_yrW)rPr�s rTrPzRequest.datans ���� rUc��t|j�\|_}|j�td|jz��t|�\|_|_|jrt|j�|_yy)Nzunknown url type: %r) rr��typerEr�r r�r�r)r��rests rTr�zRequest._parsersd��$�T�^�^�4��� �4��9�9���3�d�m�m�C�D�D�#-�d�#3� �� �4�=��9�9��� � �*�D�I�rUc�>� |j�dnd}t|d|�S)N�POST�GETr�)rP�getattr)r��default_methods rT� get_methodzRequest.get_methodzs$��A�#'�9�9�#8��e���t�X�~�6�6rUc��|jSrW)r�r�s rT�get_full_urlzRequest.get_full_urls���}�}�rUc��|jdk(r%|js|j|_||_y||_|j|_||_y)N�https)r�r�r�r�r�)r�r�r�s rT� set_proxyzRequest.set_proxy�sF���9�9����(9�(9� $� � �D���� ��D�I� �M�M�D�M��� rUc�4�|j|jk(SrW)r�r�r�s rT� has_proxyzRequest.has_proxy�s���}�}�� � �-�-rUc�>�||j|j�<yrW)ru� capitalize�r�r��vals rTr�zRequest.add_header�s��),����S�^�^�%�&rUc�>�||j|j�<yrW)r�r�r�s rT�add_unredirected_headerzRequest.add_unredirected_header�s��36����s�~�~�/�0rUc�>�||jvxs||jvSrW)rur��r��header_names rTr�zRequest.has_header�s&���t�|�|�+�6��t�5�5�5� 7rUc�n�|jj||jj||��SrW)ru�getr�)r�r��defaults rTr�zRequest.get_header�s2���|�|�����"�"�&�&�{�G�<�>� >rUc�t�|jj|d�|jj|d�yrW)ru�popr�r�s rTr�zRequest.remove_header�s,��������d�+����"�"�;��5rUc�h�i|j�|j�}t|j��SrW)r�ru�listr�)r��hdrss rT�header_itemszRequest.header_items�s,��9�$�(�(�9�D�L�L�9���D�J�J�L�!�!rUrW)�__name__� __module__�__qualname__r��propertyr��setter�deleterrPr�r�r�r�r�r�r�r�r�r�r��rUrTrr9s���!%�r�!%�E��!�$���� �_�_���������� ���� �[�[�5��5� �\�\����+�7� ��.�-�7�7�>� 6�"rUrc�R�eZdZd�Zd�Zd�Zd�Zdejfd�Z d d�Z d�Zy) rc�p�dtz}d|fg|_g|_i|_i|_i|_i|_y)N�Python-urllib/%sz User-agent)�__version__� addheaders�handlers�handle_open�handle_error�process_response�process_request)r��client_versions rTr�zOpenerDirector.__init__�sB��+�k�9��(�.�9�:����� ������� "���!��rUc��t|d�stdt|�z��d}t|�D�] }|dvr� |j d�}|d|}||dzd}|jd�rW|j d�|zdz}||dzd} t |�}|jj|i�} | |j|<n=|dk(r|}|j} n)|d k(r|}|j} n|d k(r|}|j} n��| j|g�} | rtj| |�n| j!|�d}��|r2tj|j"|�|j%|�yy#t$rY��wxYw)N� add_parentz%expected BaseHandler instance, got %rF)�redirect_request�do_open� proxy_open�_r`�errorrN�responser�T)�hasattr� TypeErrorr��dir�find� startswithrmrEr�r�r�r�r�� setdefault�bisect�insortrlr�r�)r��handler�added�meth�i�protocol� condition�j�kind�lookupr�s rT�add_handlerzOpenerDirector.add_handler�s����w��-��C� ��M�*�+� +�����L�D��D�D��� � �#��A��B�Q�x�H��Q�q�S�T� �I��#�#�G�,��N�N�3�'�!�+�a�/���A�a�C�D�z����t�9�D��*�*�.�.�x��<��.4��!�!�(�+��f�$����)�)���j�(����.�.���i�'����-�-����(�(��r�2�H��� � �h��0�����(��E�G!�J��M�M�$�-�-��1����t�$���/"����s� E3�3 E?�>E?c��yrWr�r�s rT�closezOpenerDirector.close����rUc�d�|j|d�}|D]}t||�}||�}|��|cSy)Nr�)r�r�) r��chainr�� meth_name�argsr�r��funcrws rT�_call_chainzOpenerDirector._call_chain�s>���9�9�T�2�&���G��7�I�.�D��4�[�F��!�� � rUNc��t|t�r t||�}n|}|�||_||_|j }|dz}|jj|g�D]}t||�}||�}�tjd|j|j|j|j��|j||�} |dz}|jj|g�D]}t||�}||| �} �| S)N�_requestzurllib.Request� _response)� isinstance�strrrPrQr�r�r�r��sys�auditr�rur��_openr�) r��fullurlrPrQ�reqr�r�� processorr�r�s rTrNzOpenerDirector.open�s����g�s�#��'�4�(�C��C����������8�8���Z�'� ��-�-�1�1�(�B�?�I��9�i�0�D��s�)�C�@� � � �"�C�L�L�#�(�(�C�K�K����IY�Z��:�:�c�4�(���[�(� ��.�.�2�2�8�R�@�I��9�i�0�D��C��*�H�A��rUc���|j|jdd|�}|r|S|j}|j|j||dz|�}|r|S|j|jdd|�S)Nr��default_openr�unknown�unknown_open)rr�r�)r�r rPrwr�s rTrzOpenerDirector._open s����!�!�$�"2�"2�I�"0�#�7����M��8�8���!�!�$�"2�"2�H�h�")�?*�+.�0����M����� 0� 0�)� .��5� 5rUc���|dvr|jd}|d}d|z}d}|}n|j}|dz}d}|||f|z}|j|�}|r|S|r|dd fz}|j|�Sy) N��httpr�rr;z http_error_%sr`�_errorrr��http_error_default)r�r)r��protor��dictr��http_err� orig_argsrws rTr�zOpenerDirector.errors����%�%��$�$�V�,�D���G�E�'�%�/�I��H��I��$�$�D���(�I��H��e�Y�'�$�.��!��!�!�4�(����M���)�%9�:�Y�F�D�#�4�#�#�T�*�*�rUrW)r�r�r�r�r�r�r�socket�_GLOBAL_DEFAULT_TIMEOUTrNrr�r�rUrTrr�s3�� "�-%�^ � �"&�v�/M�/M��: 5�+rUrc �j� t�}ttttt ttttg }ttjd�r|jt�t�}|D]V}|D]O}t!|t"�rt%||�s� |j'|��2t!||�s�?|j'|��Q�X|D]}|j)|��|D]}|j+|���|D]*}t!|t"�r|�}|j+|��,|S)N�HTTPSConnection)rr r/r*rrr,r+r0r.r�r�clientrlrL�setrr�� issubclass�add�remover�)r�rS�default_classes�skip�klass�check�hs rTr3r35s���� �F�#�^�[�.�0C�!�;�0B�"�$�O��t�{�{�-�.����|�,��5�D� ���E��%��&��e�U�+��H�H�U�O��E�5�)�������!������u�%��!�����5�7�#�!����a�����A����1����MrUc�"�eZdZdZd�Zd�Zd�Zy)r��c��||_yrW)�parent)r�r*s rTr�zBaseHandler.add_parent\s ����rUc��yrWr�r�s rTr�zBaseHandler.close_r�rUc�N�t|d�sy|j|jkS)N� handler_orderT)r�r-)r��others rT�__lt__zBaseHandler.__lt__cs(���u�o�.���!�!�E�$7�$7�7�7rUN)r�r�r�r-r�r�r/r�rUrTrrYs���M�� �8rUrc��eZdZ dZd�ZeZy)r0i�c��|j|j|j�}}}d|cxkrdks"n|jj d|||||�}|S)N���,r)�code�msgrdr*r�)r�r�r�r4r5r�s rT� http_responsez HTTPErrorProcessor.http_responsepsT��"�-�-����x�}�}��4�c���t�!�c�!��{�{�(�(����4��d�<�H��rUN)r�r�r�r-r6�https_responser�rUrTr0r0ls��'��M� �#�NrUr0c��eZdZd�Zy)rc�4�t|j||||��rW)rr�)r�r rtr4r5r�s rTrz*HTTPDefaultErrorHandler.http_error_default~s������d�C��r�:�:rUN)r�r�r�rr�rUrTrr}s��;rUrc�4�eZdZdZdZd�Zd�ZexZxZxZ Z dZy)r�� c�\� |j�}|dvr|dvs"|dvr|dk(st|j||||��|jdd�}d}|jj�D� � cic]\} } | j �|vr| | ��}} } t|||jd� �Scc} } w) N)�-�.�/i3i4)r��HEAD)r>r?r@r�� z%20)r^zcontent-typeT)rur�r�) r�rr��replacerur�r�rr�)r�r rtr4r5ru�newurl�m�CONTENT_HEADERS�k�v� newheaderss rTr�z$HTTPRedirectHandler.redirect_request�s��� � �N�N����2�2�q�O�7K���&�1��;��C�L�L�$��W�b�A�A�����U�+��<��'*�{�{�'8�'8�':�;�':�t�q�!�����/�9���d�':� �;��v�)�'*�':�':�$(�*� *��;s�-B(c�Z�d|vr|d}nd|vr|d}nyt|�}|jdvrt|||�d|�d�||��|js|jrt|�}d|d<t |�}t|dtj� �}t|j|�}|j||||||�}|�yt|d �rp|jx} |_| j|d�|j k\st#| �|j$k\r6t|j||j&|z||��ix} x|_|_| j|d�dz| |<|j)�|j+�|j,j/||j0� �S)N�location�uri�rr��ftpr�z - Redirection to url 'z' is not allowed�/r;z iso-8859-1)�encoding�safe� redirect_dictrr`�rQ)r�schemerrf�netlocr�rr �string�punctuationrr�r�r�rRr��max_repeatsro�max_redirections�inf_msgrnr�r*rNrQ) r�r rtr4r5rurD�urlparts�new�visiteds rT�http_error_302z"HTTPRedirectHandler.http_error_302�s����� ��Z�(�F� �g� ��U�^�F���F�#�� �?�?�">�>����AD�f�M���� � �}�}�����H�~�H��H�Q�K��H�%�� ��\��0B�0B�D������v�.�� �#�#�C��T�3���H���;���3��(�*-�*;�*;�;�G�c�'����F�A�&�$�*:�*:�:��G��� 5� 5�5�����d� $���s� 2�G�R�A�A�?A�@�G�@�c�'�#�*;�!�+�+�f�a�0�1�4���� ��� � ��� ��{�{����S�[�[��9�9rUzoThe HTTP server returned a redirect error that would lead to an infinite loop. The last 30x error message was: N)r�r�r�rXrYr�r^�http_error_301�http_error_303�http_error_307�http_error_308rZr�rUrTrr�s<���K��� *�L::�xIW�V�N�V�^�V�n�~�2�GrUrc�h� t|�\}}|jd�sd}|}ne|jd�std|z��d|vr$|jd�}|jd|�}n|jdd�}|dk(rd}|d|}t |�\}}|�t|�\}} ndx}} ||| |fS)NrO�//zproxy URL with no authority: %r�@r;r])rr�rEr�rr) �proxyrT�r_scheme� authority�host_separator�end�userinfo�hostport�user�passwords rT�_parse_proxyro�s����"�%�(��F�H����s�#���� ��"�"�4�(��>��F�G�G��(�?�%�]�]�3�/�N��-�-��^�4�C��-�-��Q�'�C��"�9��C��Q�s�O� �#�I�.��H�h���%�h�/���h����x��4��8�+�+rUc��eZdZdZdd�Zd�Zy)r �dNc��|� t�}||_|j�D]4\}}|j�}t |d|z|||j fd���6y)Nz%s_openc��||||�SrWr�)�rrfr�r�s rT�<lambda>z'ProxyHandler.__init__.<locals>.<lambda>s���Q��t�,rU)r6�proxiesr�r��setattrr�)r�rvr�rOs rTr�zProxyHandler.__init__sU���?� �l�G���� ����I�D�#��:�:�<�D��D�)�d�*�$'�d����-� .�)rUc���|j}t|�\}}}}|�|}|jrt|j�ry|rb|r`t |��dt |���} tj| j��jd�} |jdd| z�t |�}|j||�||k(s|dk(ry|jj||j��S)N�:�ascii�Proxy-authorization�Basic r�rS)r�ror��proxy_bypassr�base64� b64encode�encode�decoder�r�r*rNrQ)r�r rfr�� orig_type� proxy_typermrnrl� user_pass�credss rTr�zProxyHandler.proxy_open"s����H�H� �/;�E�/B�,� �D�(�H���"�J��8�8��S�X�X�.���H�#*�4�=�#*�8�#4�6�I��$�$�Y�%5�%5�%7�8�?�?��H�E��N�N�0�(�U�2B�C��8�$��� � �h� �+�� �"�i�7�&:���;�;�#�#�C����#�=�=rUrW)r�r�r�r-r�r�r�rUrTr r s���M� .�>rUr c�,�eZdZd�Zd�Zd�Zdd�Zd�Zy)r!c��i|_yrW)�passwdr�s rTr�zHTTPPasswordMgr.__init__@s ����rUc�����t|t�r|g}|�jvri�j|<dD]+�t��fd�|D��}||f�j||<�-y)N�TFc3�B�K�|]}�j|�����y�wrW)� reduce_uri)�.0�u�default_portr�s ��rT� <genexpr>z/HTTPPasswordMgr.add_password.<locals>.<genexpr>Js ����� ?�:=�Q�����<�0�#�s�)rrr��tuple)r��realmrLrmr��reduced_urir�s` @rT�add_passwordzHTTPPasswordMgr.add_passwordCsf����c�3���%�C�����#�!#�D�K�K���'�L�� ?�:=� ?�?�K�/3�V�n�D�K�K���{�+�(rUc���|jj|i�}dD]M}|j||�}|j�D]&\}}|D]}|j ||�s�|cccS�(�Oy)Nr��NN)r�r�r�r�� is_suburi) r�r��authuri�domainsr��reduced_authuri�uris�authinforLs rT�find_user_passwordz"HTTPPasswordMgr.find_user_passwordNsf���+�+�/�/�%��,��'�L�"�o�o�g�|�D�O�")�-�-�/���h��C��~�~�c�?�;�'�� �#2�(�rUc��� t|�}|dr|d}|d}|dxsd}nd}|}d}t|�\}}|r!|�|�ddd�j|�} | �d|| fz}||fS) Nr`rr;rO�Pi�rz%s:%d)rrr�) r�rLr��partsrTrhrfr��port�dports rTr�zHTTPPasswordMgr.reduce_uriXs���N��� ����8��1�X�F��a��I���8�?�s�D��F��I��D�� �*� ��d��D�L�V�-?��!���s�6�{� �� �#�t�U�m�3� ��$��rUc�t� ||k(ry|d|dk7ry|d}|dddk7r|dz }|dj|�S)NTrFr`r]rO)r�)r��base�test�prefixs rTr�zHTTPPasswordMgr.is_suburios[�� ��4�<����7�d�1�g����a����"�#�;�#���c�M�F��A�w�!�!�&�)�)rUN)T)r�r�r�r�r�r�r�r�r�rUrTr!r!>s��� =���.*rUr!c��eZdZd�Zy)r"c�p�tj|||�\}}|�||fStj|d|�SrW)r!r�)r�r�r�rmrns rTr�z2HTTPPasswordMgrWithDefaultRealm.find_user_password�sC��(�;�;�D�%�<C�E���h�����>�!��1�1�$��g�F�FrUN)r�r�r�r�r�rUrTr"r"~s��GrUr"c�8��eZdZ�fd�Zd�fd� Zdd�Zd�Z�xZS)r#c�0��i|_t�|� �yrW)� authenticated�superr�)r�� __class__s �rTr�z%HTTPPasswordMgrWithPriorAuth.__init__�s������ ���rUc�v��|j||�|�t�|� d|||�t�|� ||||�yrW)�update_authenticatedr�r�)r�r�rLrmr��is_authenticatedr�s �rTr�z)HTTPPasswordMgrWithPriorAuth.add_password�s@����!�!�#�'7�8����G� ��s�D�&�9� ���U�C��v�6rUc��t|t�r|g}dD]*}|D]#}|j||�}||j|<�%�,y�Nr�)rrr�r�)r�rLr�r�r�r�s rTr�z1HTTPPasswordMgrWithPriorAuth.update_authenticated�sG���c�3���%�C�'�L���"�o�o�a��>��2B��"�"�;�/��(rUc��dD]J}|j||�}|jD]'}|j||�s�|j|ccS�Lyr�)r�r�r�)r�r�r�r�rLs rTr�z-HTTPPasswordMgrWithPriorAuth.is_authenticated�sK��'�L�"�o�o�g�|�D�O��)�)���>�>�#��7��-�-�c�2�2�*�(rU)F)r�r�r�r�r�r�r�� __classcell__)r�s@rTr#r#�s����7�C�3rUr#c�t�eZdZejdej �Zd d�Zd�Zd�Z d�Z d�Zd�ZeZ eZy) r$z1(?:^|,)[ ]*([^ ,]+)[ ]+realm=(["']?)([^"']*)\2Nc�`�|� t�}||_|jj|_yrW)r!r�r�)r��password_mgrs rTr�z!AbstractBasicAuthHandler.__init__�s)����*�,�L�"��� �K�K�4�4��rUc#�K�d}tjj|�D]=}|j�\}}}|dvrt j dtd�||f��d}�?|s|r|j�d}nd}|df��yy�w)NF)�"�'zBasic Auth Realm was unquoted�Trr�)r$�rx�finditer�groupsrBrC�UserWarning�split)r��header�found_challenge�morTr r�s rT�_parse_realmz%AbstractBasicAuthHandler._parse_realm�s�������*�-�-�6�6�v�>�B�#%�9�9�;� �F�E�5��J�&�� � �=�)�1�.��5�/�!�"�O�?��������*�����4�.� ��s�BBc���|j|�}|syd}|D]J}|j|�D]4\}}|j�dk7r|}�|��|j|||�ccS�L|�t d����y)N�basicz@AbstractBasicAuthHandler does not support the following scheme: )�get_allr�r��retry_http_basic_authrE) r��authreqr�r ru�unsupportedr�rTr�s rT�http_error_auth_reqedz.AbstractBasicAuthHandler.http_error_auth_reqed�s����/�/�'�*�������F�!%�!2�!2�6�!:� ����<�<�>�W�,�"(�K���$� �5�5�d�C��G�G�";���"�� &�)�*� *�#rUc��|jj||�\}}|��|�d|��}dtj|j ��jd�z}|j |jd�|k(ry|j|j|�|jj||j��Sy)Nryr|rzrS)r�r�r~rr�r�r��auth_headerr�r*rNrQ)r�r�r r�rm�pw�raw�auths rTr�z.AbstractBasicAuthHandler.retry_http_basic_auth�s����;�;�1�1�%��>���b� �>�!�2�&�C��f�.�.�s�z�z�|�<�C�C�G�L�L�D��~�~�d�.�.��5��=���'�'��(8�(8�$�?��;�;�#�#�C����#�=�=�rUc���t|jd�r%|jj|j�s|S|j d�s�|jjd|j�\}}dj ||�j�}tj|�j�}|jddj |j���|S)Nr�� Authorizationz{0}:{1}zBasic {}) r�r�r�r�r�r�r�r�r~�standard_b64encoder�r��strip)r�r rmr��credentials�auth_strs rT�http_requestz%AbstractBasicAuthHandler.http_request�s�������%7�8��{�{�+�+�C�L�L�9��J��~�~�o�.��;�;�9�9�$����M�L�D�&�#�*�*�4��8�?�?�A�K��0�0��=�D�D�F�H��'�'��(2�(9�(9�(�.�.�:J�(K� M�� rUc��t|jd�rfd|jcxkrdkr+nn(|jj|jd�|S|jj|jd�|S)Nr�r2r3TF)r�r�r4r�r�)r�r r�s rTr6z&AbstractBasicAuthHandler.http_response s`���4�;�;� 2�3��h�m�m�)�c�)����0�0����t�D������0�0����u�E��rUrW)r�r�r��re�compile�Ir�r�r�r�r�r�r6� https_requestr7r�rUrTr$r$�sL�� ����1��D�D� �B�5�!�(*�4 ���!�M�"�NrUr$c��eZdZdZd�Zy)r%r�c�F�|j}|jd|||�}|S)N�www-authenticate)r�r�)r�r rtr4r5rurOr�s rT�http_error_401z#HTTPBasicAuthHandler.http_error_401s*���l�l���-�-�.@�*-�s�G�=���rUN)r�r�r�r�r�r�rUrTr%r%s��!�K�rUr%c��eZdZdZd�Zy)r&r{c�F�|j}|jd|||�}|S�N�proxy-authenticate)r�r�)r�r rtr4r5rurhr�s rT�http_error_407z$ProxyBasicAuthHandler.http_error_407%s-�� �H�H� ��-�-�.B�*3�S�'�C���rUN)r�r�r�r�r�r�rUrTr&r&!s��'�K�rUr&c�>�eZdZd d�Zd�Zd�Zd�Zd�Zd�Zd�Z d �Z y)r'Nc��|� t�}||_|jj|_d|_d|_d|_y�Nr)r!r�r��retried�nonce_count� last_nonce)r�r�s rTr�z"AbstractDigestAuthHandler.__init__?s>���>�$�&�F���� �K�K�4�4�����������rUc��d|_yr�)r�r�s rT�reset_retry_countz+AbstractDigestAuthHandler.reset_retry_countHs ����rUc�Z�|j|d�}|jdkDrt|jdd|d��|xjdz c_|rZ|j �d}|j�dk(r|j ||�S|j�dk7rtd|z��yy) N�i�zdigest auth failedr`r�digestr�zEAbstractDigestAuthHandler does not support the following scheme: '%s')r�r�rr�r�r��retry_http_digest_authrE)r�r�r�r rur�rTs rTr�z/AbstractDigestAuthHandler.http_error_auth_reqedKs����+�+�k�4�0���<�<�!���C�L�L�#�/C�#�T�+� +� �L�L�A��L���]�]�_�Q�'�F��|�|�~��)��2�2�3��@�@�����7�*� �"?�AG�"H�I�I�+� rUc�z�|jdd�\}}ttdt|���}|j ||�}|rtd|z}|j j |jd�|k(ry|j|j|�|jj||j��}|Sy)NrBr`z Digest %srS)r��parse_keqv_list�filter�parse_http_list�get_authorizationrur�r�r�r*rNrQ)r�r r��token� challenge�chal�auth_val�resps rTr�z0AbstractDigestAuthHandler.retry_http_digest_auth_s����:�:�c�1�-���y��v�d�O�I�,F�G�H���%�%�c�4�0���"�T�)�H��{�{���t�/�/��6�(�B���'�'��(8�(8�(�C��;�;�#�#�C����#�=�D��K� rUc���|j�d|�dtj��d�}|jd�t d�z}tj|�j�}|ddS)Nryrz��)r��time�ctimer��_randombytes�hashlib�sha1� hexdigest)r��nonce�s�b�digs rT� get_cnoncez$AbstractDigestAuthHandler.get_cnonceksT�� �+�+�U�D�J�J�L�A�� �H�H�W���Q��/���l�l�1�o�'�'�)���3�B�x�rUc��� |d}|d}|jd�}|jdd�}|jdd�}|j|�\}} |�y|jj ||j �\} }| �y|j�|j|j|�}nd}| �d|�d|��} |j��d|j��}|�| || �|�d||����}n�d|jd �vry||jk(r|xjd z c_nd |_||_d|jz}|j|�}|�d|�d|�dd�d||��� }| || �|�}ntd|z��d | �d|�d|�d|j�d|�d�}|r|d|zz }|r|d|zz }|d|zz }|r|d�d�d�z }|S#t$rYywxYw)Nr�r�qop� algorithm�MD5�opaqueryr��,r`z%08xzqop '%s' is not supported.z username="z ", realm="z ", nonce="z", uri="z ", response="r�z , opaque="%s"z , digest="%s"z, algorithm="%s"z, qop=auth, nc=z , cnonce=")r��KeyError�get_algorithm_implsr�r�r�rP�get_entity_digestr�r�r�r�r�r r)r�r r�r�rrrr�H�KDrmr��entdig�A1�A2�respdig�ncvalue�cnonce�noncebitr�s rTr�z+AbstractDigestAuthHandler.get_authorizationvs�� ���M�E���M�E��(�(�5�/�C�����e�4�I��X�X�h��-�F��(�(��3���2��9���;�;�1�1�%����F���b��<���8�8���+�+�C�H�H�d�;�F��F���� +�����(����&�� �;���2��5�!�B�%� 8�9�G� �s�y�y��~� %�����'�� � �A�%� �#$�� �"'����t�/�/�/�G��_�_�U�+�F�+0�'�6�6�1�R�5�Q�H���2���)�G��7�#�=�>�>�� #'��u�c�l�l�")�+����O�f�,�,�D���O�f�,�,�D��"�Y�.�.������H�H�D����g� �� �s�?G� G#�"G#c�V��|dk(rd��n|dk(rd��ntd|z���fd�}�|fS)Nr c�f�tj|jd��j�S�Nrz)r�md5r�r��xs rTruz?AbstractDigestAuthHandler.get_algorithm_impls.<locals>.<lambda>�s��'�+�+�a�h�h�w�&7�8�B�B�DrU�SHAc�f�tj|jd��j�Sr)rrr�rr s rTruz?AbstractDigestAuthHandler.get_algorithm_impls.<locals>.<lambda>�s��'�,�,�q�x�x��'8�9�C�C�ErUz.Unsupported digest authentication algorithm %rc����|�d|���S)Nryr�)r�drs �rTruz?AbstractDigestAuthHandler.get_algorithm_impls.<locals>.<lambda>�s���!�q�!�,�-rU)rE)r�rrrs @rTrz-AbstractDigestAuthHandler.get_algorithm_impls�sG������D�A� �%� �E�A��,�.7�8�9� 9� -���"�u�rUc��yrWr�)r�rPr�s rTrz+AbstractDigestAuthHandler.get_entity_digest�s��rUrW)r�r�r�r�r�r�r�r r�rrr�rUrTr'r'4s,����I�( � �<�|�rUr'c��eZdZ dZdZd�Zy)r(r���c�~�t|j�d}|jd|||�}|j�|S)Nr`r�)rr�r�r��r�r rtr4r5rur��retrys rTr�z$HTTPDigestAuthHandler.http_error_401�s@������%�a�(���*�*�+=�+/��g�?����� ��rUN)r�r�r�r�r-r�r�rUrTr(r(�s���"�K��M�rUr(c��eZdZdZdZd�Zy)r)�Proxy-Authorizationr(c�f�|j}|jd|||�}|j�|Sr�)r�r�r�r*s rTr�z%ProxyDigestAuthHandler.http_error_407�s6���x�x���*�*�+?�+/��g�?����� ��rUN)r�r�r�r�r-r�r�rUrTr)r)�s��'�K��M�rUr)c�,�eZdZdd�Zd�Zd�Zd�Zd�Zy)�AbstractHTTPHandlerNc�j�|�||_ytjjj|_yrW)rr�HTTPConnection� debuglevel�_debuglevel)r�r3s rTr�zAbstractHTTPHandler.__init__�s&��)3�)?�:���T�[�[�E_�E_�Ej�Ej��rUc��||_yrW)r4)r��levels rT�set_http_debuglevelz'AbstractHTTPHandler.set_http_debuglevel�s �� ��rUc��tjjj|j|j��SrW)rrr2�_get_content_lengthrPr��r�r�s rTr9z'AbstractHTTPHandler._get_content_length�s2���{�{�)�)�=�=��L�L���� �"� "rUc��|j}|std��|j��|j}t|t�r d}t|��|j d�s|jdd�|j d�sR|j d�sA|j|�}|�|jdt |��n|jdd�|}|j�r&t|j�\}}t|�\}} |j d�s|jd|�|jjD]9\} }| j�} |j | �r�(|j| |��;|S) N� no host givenz\POST data should be bytes, an iterable of bytes, or a file object. It cannot be of type str.zContent-type�!application/x-www-form-urlencodedr��Transfer-encoding�chunkedr�)r�rrPrrr�r�r�r9r�rr�r r*r�r�)r�r�r�rPr5�content_length�sel_hostrT�sel�sel_pathrjr�s rT�do_request_zAbstractHTTPHandler.do_request_�sj���|�|����?�+�+��<�<�#��<�<�D��$��$�D����n�$��%�%�n�5��/�/�"�7�9��&�&�'7�8�#�.�.�/B�C�!%�!9�!9�'�!B��!�-��3�3�,�c�.�.A�C��3�3�/��<�������$�W�%5�%5�6�K�F�C�!+�C���H�h��!�!�&�)��+�+�F�H�=��;�;�1�1�K�D�%��?�?�$�D��%�%�d�+��/�/��e�<�2� �rUc �� |j}|std��||fd|ji|��}|j|j�t|j�}|j|jj�D��cic]\}}||vr||��c}}�d|d<|j�D� � cic]\} } | j�| ��}} } |jr0i}d}||vr||||<||=|j|j|�� |j|j�|j|j ||j#d���|j'�}|j*r!|j*j)�d|_|j-�|_|j0|_|Scc}}wcc} } w#t$$r} t| ��d} ~ wwxYw#|j)��xYw) Nr<rQr�� Connectionr-�rur>)�encode_chunked)r�rrQ�set_debuglevelr4rr��updaterur��titler�� set_tunnelr�r�r�rPr�r~�getresponser��sockr�rO�reasonr5)r�� http_classr �http_conn_argsr�r&rurGrHrjr��tunnel_headers�proxy_auth_hdr�errrts rTr�zAbstractHTTPHandler.do_opens��� ��x�x����?�+�+� �t�C�S�[�[�C�N�C�� ����)�)�*��s�,�,�-��������):�):�)<�-�)<���A��G�+��1��)<�-� .�!(����6=�m�m�o�F�o���s�4�:�:�<��$�o��F�����N�2�N���(�18��1H��~�.��N�+� �L�L��)�)�>�L�B� � $�� � �#�.�.�*�C�L�L�#�(�(�G�),���8K�)L��N�� � ��A� �6�6� �F�F�L�L�N��A�F�� � �"�����������e-��G�� � $��s�m�#�� $�� � �G�G�I��s7�G �9G�AG �G)� G&�G!�!G&�&G)�)G<rW)r�r�r�r�r7r9rDr�r�rUrTr0r0�s��k�!�"� $�L@rUr0c�*�eZdZd�Zej Zy)r*c�V�|jtjj|�SrW)r�rrr2�r�r s rT� http_openzHTTPHandler.http_open\s���|�|�D�K�K�6�6��<�<rUN)r�r�r�rXr0rDr�r�rUrTr*r*Zs��=�'�2�2�LrUr*rc�2�eZdZdd�Zd�ZejZy)rLNc�*�|�|n#tjjj}tj||�|�Ctjjj}tjj|�}|�||_||_ yrW) rrrr3r0r�� _http_vsn�_create_https_context�check_hostname�_context)r�r3r?r]�http_versions rTr�zHTTPSHandler.__init__esq��'1�'=��4�;�;�C^�C^�Ci�Ci�J��(�(��z�:���#�{�{�:�:�D�D���+�+�;�;�L�I���)�)7��&�#�D�MrUc�n�|jtjj||j��S)NrA)r�rrrr^rWs rT� https_openzHTTPSHandler.https_openos-���<�<���� ;� ;�S�(,� � � �7� 7rU�NNN)r�r�r�r�rar0rDr�r�rUrTrLrLcs�� $� 7�,�7�7� rUrLc�(�eZdZdd�Zd�Zd�ZeZeZy)rNc�R�ddl}|�|jj�}||_yr�)�http.cookiejar� cookiejar� CookieJar)r�rfrs rTr�zHTTPCookieProcessor.__init__xs$��������0�0�2�I�"��rUc�<�|jj|�|SrW)rf�add_cookie_headerr:s rTr�z HTTPCookieProcessor.http_request~s�����(�(��1��rUc�>�|jj||�|SrW)rf�extract_cookies)r�r�r�s rTr6z!HTTPCookieProcessor.http_response�s�����&�&�x��9��rUrW)r�r�r�r�r�r6r�r7r�rUrTrrws��#���!�M�"�NrUrc��eZdZd�Zy)r/c�6�|j}td|z��)Nzunknown url type: %s)r�r)r�r r�s rTrzUnknownHandler.unknown_open�s���x�x���-��4�5�5rUN)r�r�r�rr�rUrTr/r/�s��6rUr/c�x� i}|D]1}|jdd�\}}|ddk(r |ddk(r|dd}|||<�3|S)N�=r`rr�r])r�)�l�parsed�eltrGrHs rTr�r��sX��H� �F����y�y��a� ���1��Q�4�3�;�1�R�5�C�<��!�B��A���q� � � �MrUc�� g}d}dx}}|D]H}|r||z }d}� |r|dk(rd}�|dk(rd}||z }�$|dk(r|j|�d}�=|dk(rd}||z }�J|r|j|�|D�cgc]}|j���c}Scc}w)Nr�F�\Tr�r)rlr�)r�res�part�escaper �curs rTr�r��s���� �C� �D���F�U�����C�K�D��F����d�{����������C�K�D���#�:��J�J�t���D���#�:��E�����-�2�� � �4��%(�)�S�T�D�J�J�L�S�)�)��)s�.Bc�"�eZdZd�ZdZd�Zd�Zy)r+c���|j}|dddk(rK|dddk7rC|jr7|jdk7r(|j|j�vrtd��y|j |�S)Nr;rdr�rO� localhost�-file:// scheme is supported only on localhost)r�r�� get_namesr�open_local_file)r�r rOs rT� file_openzFileHandler.file_open�sm���l�l���r��7�d�?�s�1�Q�x�3��C�H�H����K�'��8�8�t�~�~�/�/��N�O�O�0��'�'��,�,rUNc��tj�f ttjd�dtjtj ��dz�t_tjStjS#tj$r1tjd�ft_YtjSwxYw)Nr{r;)r+�namesr�r�gethostbyname_ex�gethostname�gaierror� gethostbynamer�s rTr}zFileHandler.get_names�s������$� I�$)��+�+�K�8��;��+�+�F�,>�,>�,@�A�!�D�E�%F��!� � � � �{� � � ���?�?� I�%+�%9�%9�+�%F�$H��!�� � � � I�s�AB�2C� Cc�D�ddl}ddl}|j}|j}t |�} tj|�}|j}|jj|jd��} |j|�d} |jd| xsd|| fz�}|rt|�\}}|rsBt|�|j�vr'|r d|z|z} nd|z} t!t#|d�|| �St'd��#t$$r}t'|��d}~wwxYw) NrT��usegmtz6Content-type: %s Content-length: %d Last-modified: %s � text/plain�file://�rbzfile not on local host)�email.utils� mimetypesr�r�r5re�stat�st_size�utils� formatdate�st_mtime� guess_type�message_from_stringr�_safe_gethostbynamer}rrNr~r)r�r �emailr�r�rq� localfile�statsry�modified�mtyperur��origurl�exps rTr~zFileHandler.open_local_file�s$�����x�x���<�<�� ��*� � ��G�G�I�&�E��=�=�D��{�{�-�-�e�n�n�T�-�J�H��(�(��2�1�5�E�/�e�/�/�K��&�,��h�7�8�9�G��'��-� ��d���1�$�7�4�>�>�;K�K��'�$�.��9�G�'�(�2�G�!�$�y�$�"7��'�J�J��/�0�0��� ��3�-��� �s�C D� D�D�D)r�r�r�rr�r}r~r�rUrTr+r+�s��-� �E�!�1rUr+c�`� tj|�S#tj$rYywxYwrW)rr�r�)r�s rTr�r��s.����#�#�D�)�)���?�?����s��-�-c��eZdZd�Zd�Zy)r,c�$�ddl}ddl}|j}|std��t |�\}}|� |j }nt |�}t|�\}}|rt|�\}}nd}t|�}|xsd}|xsd} tj|�}t|j�\} } | jd�}t!t#t|��}|dd|d}}|r |ds|dd} |j%||||||j&�} |xrdxsd}| D]9}t)|�\}}|j+�d k(s�%|d vs�*|j-�}�;| j/||�\}}d}|j1|j2�d}|r|d|zz }|� |dk\r|d|zz }t5j6|�}t9|||j2�S#t$r}t|��d}~wwxYw#|j:$r}t|�|�d}~wwxYw) Nr�ftp error: no host givenr�rOr]r`r��Dr���a�Ar�r�r%r�zContent-type: %s zContent-length: %d )�ftplibr�r�rr�FTP_PORTrmrrrrr�r~rr�r�r��map�connect_ftprQrr��upper�retrfiler�r�r�r�r� all_errors)r�r r�r�r�r�rmr�r5rf�attrs�dirsrY�fwr��attrr�rt�retrlenrur�r�s rT�ftp_openzFTPHandler.ftp_open�s$�����x�x����5�6�6���%� ��d��<��?�?�D��t�9�D� ��%� ��d��'��-�L�D�&��F��t�}���z�r����2�� ��'�'��-�D�!����.���e��z�z�#����C���&�'���#�2�Y��R��d����Q�����8�D� )��!�!�$���d�D�#�+�+�N�B��<�C�&�3�D���)�$�/���e��:�:�<�6�)��:�:� �;�;�=�D� � �+�+�d�D�1�K�B���G��(�(����6�q�9�E���/�%�7�7���"�w�!�|��1�G�;�;���/�/��8�G��b�'�3�<�<�8�8��1� ��3�-��� ��2� � � )��3�-�S�(�� )�s>�G�1AG/�G/�BG/� G,�G'�'G,�/H�>H � Hc �&�t||||||d��S)NF)� persistent)� ftpwrapper)r�rmr�r�r�r�rQs rTr�zFTPHandler.connect_ftp1s���$���d�D�'�%*�,� ,rUN)r�r�r�r�r�r�rUrTr,r,�s ��2)�h,rUr,c�0�eZdZd�Zd�Zd�Zd�Zd�Zd�Zy)r-c�J�i|_i|_d|_d|_d|_y)Nr�<r�)�cacherQ�soonest�delay� max_connsr�s rTr�zCacheFTPHandler.__init__8s%���� ��������� ���rUc��||_yrW)r�)r��ts rT� setTimeoutzCacheFTPHandler.setTimeout?s ���� rUc��||_yrW)r�)r�rEs rT�setMaxConnszCacheFTPHandler.setMaxConnsBs ����rUc�|�|||dj|�|f}||jvr/tj�|jz|j|<nKt||||||�|j|<tj�|jz|j|<|j �|j|S)NrO)�joinr�r�r�rQr��check_cache)r�rmr�r�r�r�rQr�s rTr�zCacheFTPHandler.connect_ftpEs����D�$�������7���$�*�*�� $� � ��d�j�j� 8�D�L�L���(��v�t�T�)-�w�8�D�J�J�s�O� $� � ��d�j�j� 8�D�L�L��������z�z�#��rUc���tj�}|j|krht|jj ��D]B\}}||ks�|j |j �|j |=|j|=�Dtt|jj���|_t|j �|jk(r�t|jj ��D]0\}}||jk(s�|j |=|j|=ntt|jj���|_yyrW)r�r�r�rQr�r�r��min�valuesror�)r�r�rGrHs rTr�zCacheFTPHandler.check_cachePs ���I�I�K���<�<�1���T�\�\�/�/�1�2���1��q�5��J�J�q�M�'�'�)�� � �1� ����Q�� 3� �4���� 3� 3� 5�6�7����t�z�z�?�d�n�n�,��T�\�\�/�/�1�2���1�����$�� � �1� ����Q��� 3� �t�D�L�L�$7�$7�$9�:�;�D�L� -rUc���|jj�D]}|j��|jj�|jj�yrW)r�r�r��clearrQ)r��conns rT�clear_cachezCacheFTPHandler.clear_cachedsB���J�J�%�%�'�D��J�J�L�(�� � ���������rUN) r�r�r�r�r�r�r�r�r�r�rUrTr-r-5s ����� �<�(rUr-c��eZdZd�Zy)r.c�h�|j}|jdd�\}}|jdd�\}}t|�}|jd�rt j |�}|dd}|sd}t jd|t|�fz�}ttj|�||�S)Nryr`rz;base64i�����text/plain;charset=US-ASCIIz$Content-type: %s Content-length: %d )r�r�r�endswithr~�decodebytesr�r�ror�io�BytesIO)r�r rOrTrP� mediatyperus rT� data_openzDataHandler.data_openks����l�l���y�y��Q�'�����*�*�S��+�� �4� ��%�����i�(��%�%�d�+�D�!�#�2��I��5�I��+�+�,T� ��D� �"�-#�$���"�*�*�T�*�G�S�9�9rUN)r�r�r�r�r�rUrTr.r.js��:rUr.r<�nt)r5r4c�� |dddk(r|dd}n |dddk(r|dd}tj�}tj�}t|||��S)Nr�z///r;�z//localhost/��rP�errors)r�getfilesystemencoding�getfilesystemencodeerrorsr��pathnamerPr�s rTr5r5�sg�� C��B�Q�<�5� � ���|�H� �c�r�]�n� ,����}�H��,�,�.���.�.�0���x�(�6�B�BrUc�� |dddk(rd|z}tj�}tj�}t|||��S)Nr;rdr�)rr�r�r r�s rTr4r4�sL�� B��B�Q�<�4���h��H��,�,�.���.�.�0���X���@�@rUc��eZdZ dZdezZdd�Zd�Zd�Zd�Z d�Z dd�Zdd �Zdd �Z dd�Zd�Zdd �Zdd�Zd�Zerd�Zdd�Zd�Zd�Zd�Zdd�Zy)r9Nr�c�v�dd|jjiz}tj|td��|� t�}||_|jd�|_|jd�|_ d|jfdg|_g|_tj|_d|_t"|_y) NzW%(class)s style of invoking requests is deprecated. Use newer urlopen functions/methods�classr�)� stacklevel�key_file� cert_filez User-Agent)�Acceptz*/*)r�r�rBrCrDr6rvr�r�r��versionr��_URLopener__tempfilesrer}�_URLopener__unlink� tempcache�ftpcache)r�rv�x509r5s rTr�zURLopener.__init__�s���4�7>����@W�@W�6X�Y��� � �c�-�!�<��?� �l�G��������,�� ����+�.���(�$�,�,�7�9J�K������� � �� ����!�� rUc�$�|j�yrW)r�r�s rT�__del__zURLopener.__del__�s��� � �rUc�$�|j�yrW)�cleanupr�s rTr�zURLopener.close�s�����rUc���|jr2|jD]} |j|��|jdd�=|jr|jj �yy#t$rY�YwxYwrW)r�r�r~r�r�)r�rYs rTr�zURLopener.cleanup�sn������(�(����M�M�$�'�)� � � ��#��>�>��N�N� � �"�������s�A'�' A3�2A3c�<� |jj|�yrW)r�rl)r�r�s rT� addheaderzURLopener.addheader�s�� 5������t�$rUc��� tt|��}t|d��}|jr9||jvr+|j|\}}t |d�}t|||�St |�\}}|sd}||jvr0|j|}t |�\}} t| �\} }| |f}nd}d|z}||_ |jdd�}t||�r|dk(r'|r|j|||�S|j||�S |�t||�|�St||�||�S#tt f$r�t"$r} t#d | �| �d} ~ wwxYw) Nz%/:=&?~#+!$,;'@()*[]|�rQr�rY�open_�-r�r~zsocket error)r rr r�rNrrrvr r�rCr��open_unknown_proxy�open_unknownr�rrr~)r�r rPrqrurt�urltyperOrf� proxyhostr�r�rjr5s rTrNzURLopener.open�s}��D���7�+�,����&=�>���>�>�g����7� $���w� 7��H�g��h��%�B��b�'�7�3�3�!�'�*������G��d�l�l�"��L�L��)�E�!+�E�!2��G�Y�'� �2�N�D�(���/�C��E��� ���� ��|�|�C��%���t�T�"�d�.?�&?���.�.�u�g�t�D�D��(�(��$�7�7� 8��|�*�w�t�T�*�3�/�/�*�w�t�T�*�3��5�5���8�$� ��� 8��.�#�.�C�7�� 8�s�D8�%D8�8E"� E�E"c�:� t|�\}}tdd|��)N� url errorzunknown url type�rr~)r�r rPr�rOs rTr�zURLopener.open_unknowns#��=��w�'� ��c��k�#5�t�<�<rUc�@� t|�\}}tdd|z|��)Nr�zinvalid proxy for %sr�)r�rfr rPr�rOs rTr�zURLopener.open_unknown_proxys(��=��w�'� ��c��k�#9�D�#@�%�H�HrUc�.� tt|��}|jr||jvr|j|St|�\}}|�R|r|dk(rK |j |�}|j�}|j �tt|�d�|fS|j||�} |j�} |r t|d�} n�t|�\}}t|xsd�\}}t|xsd�\}}t|xsd�\}}tjj|�d} t!j"| �\}}|j$j'|�tj(|d�} || f}|j�||j|<d}d}d}d}d| vrt+| d �}|r ||||�|j-|�x}rD|t/|�z }| j1|�|dz }|r ||||�|j-|�x}r�D| j � |j �|dk\r||krt3d ||fz|��|S#t$rY���wxYw#| j �wxYw#|j �wxYw)NrYr`rZr�r\r]rr^r_ra)r rr�rr~rdr�r5r r~rNrrrerf�splitextrh�mkstempr�rl�fdopenrmrnrorpr)r�rOrqrrrPr��url1rtr�rurv�garbagerf�suffix�fdrwrxryrnrzr{s rT�retrievezURLopener.retrieves��� ;��Y�s�^�$���>�>�c�T�^�^�3��>�>�#�&�&���_� ��d���T�T�V�^� ��)�)�$�/���w�w�y����� �#�J�t�$4�Q�$7�8�$�>�>��Y�Y�s�D� !��" ��g�g�i�G���8�T�*�� *�3�� ��� *�4�:�2� 6� ��� +�D�J�B� 7� ��g� *�4�:�2� 6� ��g����)�)�$�/��2��!)�!1�!1�&�!9���X�� � �'�'��1��i�i��D�)�� �!�7�*���>�>�-�*0�D�N�N�3�'���������#�w�.��w�'7�8�9�D���x��T�2�!�w�w�r�{�*�e�*��C��J�&�D��I�I�e�$���M�H�!�"�8�R��6� "�w�w�r�{�*�e�*�� � ���H�H�J��1�9����&�C���,�� &�(� (�� ��[� �� ��F� � ����H�H�J�s9�A I�4CJ�BI-�J� I*�)I*�-I?�?J�Jc��� d}d}t|t�r,t|�\}}|rt|�\}}t |�}|}nq|\}}t|�\}}t|�\} } | }d}| j �dk7rd}n6t| �\}} |rt|�\}}|r | �d|�| ��}t|�r|}|stdd��|r>t |�}tj|j��jd�}nd}|r>t |�}tj|j��jd�}nd}||�} i}|rd|z|d<|rd|z|d<|r||d <d |d<|jD] \}}|||<�|�d|d <| jd|||�n| jd||�� | j�}d|j(cxkrdkr(nn%t+||j,d|z|j(�S|j/||j0|j(|j2|j,|�S#t j"j$$rt'd��wxYw)Nrz://z http errorr<rzzBasic %sr-r�r�r�rFr=zContent-Typer�r�rGz$http protocol error: bad status liner2r3�http:)rrr rrrr�r}r~r~rr�r�r�r�rMrr� BadStatusLiner�statusrr5� http_errorrtrO)r��connection_factoryrOrP�user_passwd�proxy_passwdr�r��realhostr�r�� proxy_authr�� http_connrur�r�r�s rT�_open_generic_httpzURLopener._open_generic_http\s��� ������c�3��'��_�N�D�(��$.�t�$4�!��T��t�}���H� �N�D�(�!+�D�!1��L�$�&�x�0�M�G�T��C��K��}�}��&�(���!+�D�!1���$��,6�x�,@�)�K���.5�x��F�H���)�#�D��7�<��A�A��"�<�0�L��)�)�,�*=�*=�*?�@�G�G��P�J��J��!�+�.�K��#�#�K�$6�$6�$8�9�@�@��I�D��D�&�t�,� ����-7�*�-D�G�)�*��(2�T�(9�G�O�$��&�G�F�O� !(����!�_�_�M�F�E�#�G�F�O�-���&I�G�N�#����f�h��g�>����e�X�w��?� C� �,�,�.�H��(�/�/�'�C�'��h����g��m�&�o�o�/� /��?�?��X�[�[�������(�,�,��F� F���{�{�(�(� C��A�B�B� C�s�9I�)I-c�Z� |jtjj||�SrW)rrrr2�r�rOrPs rT� open_httpzURLopener.open_http�s$�� ��&�&�t�{�{�'A�'A�3��M�MrUc�� d|z}t||�r,t||�}|� ||||||�} n |||||||�} | r| S|j|||||�S)Nz http_error_%d)r�r�r) r�rOrt�errcode�errmsgrurPrjr�rws rTrzURLopener.http_error�su�� E� ��(���4����T�4�(�F��|���R��&�'�B����R��&�'�4�H���f�}��&�&�s�B����I�IrUc�B� |j�t||||d��rW)r�r�r�rOrtrrrus rTrzURLopener.http_error_default�s!��L� ��� ���W�f�g�t�<�<rUc�r�|js|jr}tjjj }tjj |�}|j|j|j�|j� d|_nd}tjj ||��S)NTrA) r�r�rrrr[r\�load_cert_chain�post_handshake_auth)r�r�r_r?s rT�_https_connectionzURLopener._https_connection�s����}�}����#�{�{�:�:�D�D���+�+�;�;�L�I���'�'����� � �F��.�.�:�26�G�/����;�;�.�.�t�W�.�E�ErUc�>� |j|j||�SrW)rrrs rT� open_httpszURLopener.open_https�s ��%��*�*�4�+A�+A�3��M�MrUc�� t|t�std��|dddk(r)|dddk7r!|ddj�dk7rt d��|j|�S) NzEfile error: proxy support for file protocol currently not implementedr;rdr�rOr�z localhost/r|)rrrr�rEr~r�s rT� open_filezURLopener.open_file�se��=��#�s�#��b�c�c��r��7�d�?�s�1�Q�x�3��3�q��9�?�?�3D��3T��L�M�M��'�'��,�,rUc��� ddl}ddl}t|�\}}t|�} t j |�}|j} |jj|jd��} |j|�d}|jd|xsd| | fz�}|s&|} |dddk(rd|z} t!t#|d �|| �St%|�\}}|sht'j(|�t+�ft-�zvr=|} |dddk(rd|z} n|dd dk(rt/d|z��t!t#|d �|| �Std ��#t$r%}t|j|j��d}~wwxYw)NrTr�z6Content-Type: %s Content-Length: %d Last-modified: %s r�r`rOr�r�r;z./zAlocal file url may start with / or file:. Unknown url of type: %sz#local file error: not on local host)r�r�r r5rer�r~r�strerrorrqr�r�r�r�r�r�rrNrrr�r{�thishostrE)r�rOr�r�r�rY� localnamer��eryr�r�ru�urlfiler�s rTr~zURLopener.open_local_file�s��������_� ��d� ��&� � 3��G�G�I�&�E��}�}���;�;�)�)�%�.�.��)�F���$�$�S�)�!�,��+�%�+�+�G� � "�l�D�(�3� 4�5����G��B�Q�x�3��#�d�*���d�9�d�3�W�g�F�F���%� ��d���#�#�D�)�y�{�n�x�z�.I�J��G��B�Q�x�3��#�d�*���b�q��T�!� �!d�gj�!j�k�k��d�9�d�3�W�g�F�F��<�=�=��-� 3��1�:�:�q�z�z�2�2�� 3�s�E� E5� E0�0E5c�:� t|t�std��ddl}t |�\}}|std��t|�\}}t |�\}}|rt|�\}}nd}t|�}t|xsd�}t|xsd�}tj|�}|sddl}|j}nt|�}t|�\}} t|�}|jd�} | dd| d}} | r | ds| dd} | r | dsd| d<|||dj!| �f}t#|j$�t&kDrLt)|j$�D]4} | |k7s� |j$| }|j$| =|j+��6 ||j$vrt-||||| �|j$|<|sd}nd }| D]9}t/|�\}}|j1�d k(s�%|dvs�*|j3�}�;|j$|j5||�\}}|j7d|z�d}d}|r|d |zz }|� |dk\r|d|zz }t9j:|�}t=||d|z�S#t?�$r}td|���|�d}~wwxYw)NzCftp error: proxy support for ftp protocol currently not implementedrr�r�rOr]r`r�r�r�r�zftp:zContent-Type: %s zContent-Length: %d �ftp error: ) rrrr�r rrrrrr�r�r�rmrr�r�ror��MAXFTPCACHEr�r�r�rr�r�r�r�r�r�r� ftperrors)r�rOr�r�rfr�rmr�r�r�r�rYr�rGrHr�r�r�rtr�r�rur�s rT�open_ftpzURLopener.open_ftps�����#�s�#��`�a�a����_� ��d��8�$>�?�?���%� ��d���%� ��d���T� 2���v��f��t�}���t�z�r�"�����2�&���#�#�D�)�����?�?�D��t�9�D� ��&���e��t�}���z�z�#����#�2�Y��R��d����Q���Q�R�����Q��3��a���D�$������.���t�}�}���+��$�-�-�(����8�� � �a�(�A�� � �a�(��G�G�I� )� 9��$�-�-�'��t�V�T�4��>�� � �c�"�����$���)�$�/���e��:�:�<�6�)��:�:� �;�;�=�D� � !�M�M�#�.�7�7��d�C�M�R���(�(��#��6�q�9�E��G���/�%�7�7���"�w�!�|��1�G�;�;���/�/��8�G��b�'�6�C�<�8�8���{� 9��[���.�/�S�8�� 9�s&�AI9�)I9�.B I9�9 J�J�Jc �V� t|t�std�� |jdd�\}}|sd}|j d�}|dk\rd ||dvr||dzd}|d|}nd }g}|jdtjdtjtj���z�|jd |z�|dk(r4tj|jd��jd�}nt|�}|jdt!|�z�|jd �|j|�dj#|�}t%j&|�}t)j*|�}t-|||�S#t$r tdd��wxYw)NzEdata error: proxy support for data protocol currently not implementedrr`z data errorzbad data URLr��;rror�zDate: %sz%a, %d %b %Y %H:%M:%S GMTzContent-type: %sr~rzzlatin-1zContent-Length: %d� )rrrr�rEr~�rfindrlr��strftime�gmtimer~r�r�r�rror�r�r�r��StringIOr) r�rOrPr��semirPr5ru�fs rT� open_datazURLopener.open_dataFs�����#�s�#��b�c�c� 8��9�9�S�!�,�L�T�4��0�D��z�z�#����1�9��D���K�/��D��F�G�}�H����;�D��H���� � �:�d�m�m�,G�,0�K�K�� � ��,D�F�F� G�� � �%��,�-��x���%�%�d�k�k�'�&:�;�B�B�9�M�D��4�=�D�� � �'�#�d�)�3�4�� � �2��� � �4���i�i��n���+�+�C�0���K�K�����!�W�c�*�*��5� 8��,��7�7� 8�s�F�F(rWrb)r�r�r�r�r�r�r�r�r�r�r�rNr�r�rrrrrrFrr!r#r~r.r8r�rUrTr9r9�s�����K� �;�.�G�!�4��#�%�"8�H=� I�:�|ZF�xN�J� =� � F� N�-�>�@89�t'+rUr9c��eZdZ d�Zd�Zdd�Zd�Zdd�Zdd�Zdd�Z dd �Z dd �Z dd�Zdd�Z dd �Zdd�Zdd�Zdd�Zd�Zy)r:c�`�tj|g|��i|��i|_d|_d|_y)Nrr<)r9r�� auth_cache�tries�maxtries)r�r��kwargss rTr�zFancyURLopener.__init__ss/�����4�1�$�1�&�1������ ��� rUc�&� t||d|z|�S)Nr )rrs rTrz!FancyURLopener.http_error_defaultys��A��"�g�w��}�g�>�>rUNc�@� |xjdz c_ |jrQ|j|jk\r8t|d�r |j}n|j}|||dd|�d|_S|j||||||�}|d|_S#d|_wxYw)Nr`�http_error_500r(z)Internal Server Error: Redirect Recursionr)r<r=r�rAr�redirect_internal) r�rOrtrrrurPr�rws rTr^zFancyURLopener.http_error_302}s���3�� � �a�� � ��}�}����t�}�}�!<��4�!1�2��.�.�D��2�2�D��C��S�G�#�%��D�J� �+�+�C��W�f�,3�T�;�F���D�J���D�J�s�AB�5B� Bc��d|vr|d}nd|vr|d}ny|j�t|jdz|z|�}t|�}|jdvrt|||d|zz||��|j |�S)NrKrLryrMz( Redirection to url '%s' is not allowed.)r�rr�rrTrrN) r�rOrtrrrurPrDr[s rTrBz FancyURLopener.redirect_internal�s����� ��Z�(�F� �g� ��U�^�F�� ��� �����S��3�.��7���F�#���?�?�">�>��F�G�"�F��O�P�#�R�)� )� �y�y�� � rUc�0� |j||||||�SrW�r^�r�rOrtrrrurPs rTr_zFancyURLopener.http_error_301�s ��8��"�"�3��G�V�W�d�K�KrUc�0� |j||||||�SrWrErFs rTr`zFancyURLopener.http_error_303�s ��I��"�"�3��G�V�W�d�K�KrUc�^� |�|j||||||�S|j|||||�SrW)r^rrFs rTrazFancyURLopener.http_error_307��>��?��<��&�&�s�B����$�O�O��*�*�3��G�V�W�M�MrUc�^� |�|j||||||�S|j|||||�SrW)r_rrFs rTrbzFancyURLopener.http_error_308�rIrUc��� d|vrtj||||||�|d}tjd|�} | stj||||||�| j �\} }| j�dk7rtj||||||�|stj||||||�d|jzdz}|�t||�||�St||�|||�S)Nr��![ ]*([^ ]+)[ ]+realm="([^"]*)"r��retry_�_basic_auth�r9rr��matchr�r�r�r�� r�rOrtrrrurPr+�stuffrPrTr�rjs rTr�zFancyURLopener.http_error_401�s�� =��W�,��(�(��s�B�)0�&�'� C��*�+�����?��G����(�(��s�B�)0�&�'� C����� ����<�<�>�W�$��(�(��s�B�)0�&�'� C���(�(��s�B���� ��$�)�)�#�m�3���<�%�7�4��%�c�5�1�1�%�7�4��%�c�5�$�7�7rUc��� d|vrtj||||||�|d}tjd|�} | stj||||||�| j �\} }| j�dk7rtj||||||�|stj||||||�d|jzdz}|�t||�||�St||�|||�S)Nr�rLr��retry_proxy_rNrOrQs rTr�zFancyURLopener.http_error_407�s�� =��w�.��(�(��s�B�)0�&�'� C��,�-�����?��G����(�(��s�B�)0�&�'� C����� ����<�<�>�W�$��(�(��s�B�)0�&�'� C���(�(��s�B���� ��� � �)�M�9���<�%�7�4��%�c�5�1�1�%�7�4��%�c�5�$�7�7rUc��t|�\}}d|z|z}|jd}t|�\}} t| �\} } | jd�dz}| |d} |j | ||�\}} |s| syt|d���dt| d���d| ��} d| z| z|jd<|�|j |�S|j ||�S)N�http://rrer`r�r�ry�r rvrr��get_user_passwdr rN�r�rOr�rPr�r�rDrfr�r�� proxyselectorr�rmr�s rT�retry_proxy_http_basic_authz*FancyURLopener.retry_proxy_http_basic_auth�s���#�C����h��T�!�H�,�����V�$��'��.����#-�i�#8� � �=��N�N�3��!�#���a�b�M� ��+�+�I�u�a�@���f����"'��2�"6�"'��R�"8�)�E� �(�9�4�}�D����V���<��9�9�V�$�$��9�9�V�T�*�*rUc��t|�\}}d|z|z}|jd}t|�\}} t| �\} } | jd�dz}| |d} |j | ||�\}} |s| syt|d���dt| d���d| ��} d| z| z|jd<|�|j |�S|j ||�S)N�https://r�rer`r�r�ryrWrYs rT�retry_proxy_https_basic_authz+FancyURLopener.retry_proxy_https_basic_auth s���#�C����h��d�"�X�-�����W�%��'��.����#-�i�#8� � �=��N�N�3��!�#���a�b�M� ��+�+�I�u�a�@���f����"'��2�"6�"'��R�"8�)�E� � *�Y� 6�� F����W���<��9�9�V�$�$��9�9�V�T�*�*rUc� �t|�\}}|jd�dz}||d}|j|||�\}}|s|syt|d���dt|d���d|��}d|z|z} |�|j | �S|j | |�S)Nrer`r�r�ryrV�r r�rXr rN� r�rOr�rPr�r�r�rmr�rDs rTr�z$FancyURLopener.retry_http_basic_auth s���#�C����h��I�I�c�N�Q����A�B�x���+�+�D�%��;���f����"�4�b�1�"�6��3�T�;���T�!�H�,���<��9�9�V�$�$��9�9�V�T�*�*rUc� �t|�\}}|jd�dz}||d}|j|||�\}}|s|syt|d���dt|d���d|��}d|z|z} |�|j | �S|j | |�S)Nrer`r�r�ryr]r`ras rT�retry_https_basic_authz%FancyURLopener.retry_https_basic_auth% s���#�C����h��I�I�c�N�Q����A�B�x���+�+�D�%��;���f����"�4�b�1�"�6��3�T�;���d�"�X�-���<��9�9�V�$�$��9�9�V�T�*�*rUc���|dz|j�z}||jvr|r|j|=n|j|S|j||�\}}|s|r||f|j|<||fS)Nre)r�r;�prompt_user_passwd)r�r�r�r�r�rmr�s rTrXzFancyURLopener.get_user_passwd3 sv���c�k�D�J�J�L�(���$�/�/�!���O�O�C�(����s�+�+��.�.�t�U�;���f��6�4��.�4�?�?�3�/��V�|�rUc �� ddl} td|�d|�d��}|jd|�d|�d|�d��}||fS#t$r t�YywxYw)NrzEnter username for z at z: zEnter password for z in r�)�getpass�input�KeyboardInterrupt�print)r�r�r�rgrmr�s rTrez!FancyURLopener.prompt_user_passwd> sW��1�� ��E�4�H�I�D��_�_��u�d�&$�%�F���<��� � ��G�� �s�08�A� ArW)NF)r)r�r�r�r�rr^rBr_r`rarbr�r�r[r^r�rcrXrer�rUrTr:r:psm��I��?��$!�8L�L�N�N�FJ��8�2FJ��8�2+�$+�$+�+� � rUr:c�F� t�tjd�atS)Nr{)� _localhostrr�r�rUrTr{r{N s!��B����)�)�+�6� ��rUc�� t�: ttjtj��d�atStS#tj $r)ttjd�d�aYtSwxYw)Nr;r{)� _thishostr�rr�r�r�r�rUrTr&r&V sx��6��� G��f�5�5�f�6H�6H�6J�K�A�N�O�I���9������ G��f�5�5�k�B�1�E�F�I��� G�s�3A�4B�Bc�<� t�ddl}|jatSr�)� _ftperrorsr�r�)r�s rTr-r-a s��;�����&�&� ��rUc�F� t�tjd�atSr�)� _noheadersr�r�r�rUrT� noheadersrsj s!��/����.�.�r�2� ��rUc�>�eZdZ d d�Zd�Zd�Zd�Zd�Zd�Zd�Z y) r�Nc���||_||_||_||_||_||_d|_||_ |j�y#|j��xYwr�) rmr�r�r�r�rQ�refcount� keepalive�initr�)r�rmr�r�r�r�rQr�s rTr�zftpwrapper.__init__w s[���� ������ ��� ��� ������ �#��� ��I�I�K�� ��J�J�L��s�A�Ac��ddl}d|_|j�|_|jj |j |j|j�|jj|j|j�dj|j�}|jj|�y)NrrO)r��busy�FTPrN�connectr�r�rQ�loginrmr�r�r��cwd)r�r��_targets rTrxzftpwrapper.init� sw����� ��:�:�<�����������D�I�I�t�|�|�<������t�y�y�$�+�+�.��(�(�4�9�9�%�������W�rUc�8�ddl}|j�|dvrd}d}nd|z}d} |jj|�d}|r&|s$ d|z}|jj |�\}}|s�|jjd�|rY|jj�} |jj|� |jj| �d|z}nd}|jj |�\}}d|_t|jd �|j�} |xj dz c_|j#�| fS#|j$r/|j�|jj|�Y��TwxYw#|j$r+}t|�dddk7rtd |���|�Yd}~��cd}~wwxYw#|j$r}td |z�|�d}~wwxYw#|jj| �wxYw)Nr)r%r�zTYPE Ar`zTYPE zRETR r��550r+z ftp error: %rzLIST �LISTr�)r��endtransferrN�voidcmdr�rx�ntransfercmd� error_permrr�pwdr~rzr�makefile� file_closervr�)r�rYr�r��cmd�isdirr�r�rOr��ftpobjs rTr�zftpwrapper.retrfile� s���������:��X�s�q�u��d�N�c�A�E� "��H�H���S�!����� G���n�� $��� 5� 5�c� :� ��g���H�H���X�&���h�h�l�l�n��&�M������T�*��H�H�L�L��%���n���� �H�H�1�1�#�6�M�D�'��� ��d�m�m�D�1�4�?�?�C��� � ��� �� � ���� � ��G� � � "��I�I�K��H�H���S�!� "���$�$� G��v�;�r��?�e�+�"�[���#9�:��F�,�� G��"�,�,�M�&���'?�@�f�L��M���H�H�L�L��%�sM�E�#F�&G�:F�F�G�( G�G�G9�%G4�4G9�9G<�<Hc��|jsyd|_ |jj�y#t�$rYywxYwr�)rzrN�voidrespr-r�s rTr�zftpwrapper.endtransfer� s<���y�y���� � ��H�H������{� �� �s�1� A�Ac�R�d|_|jdkr|j�yy)NFr)rwrv� real_closer�s rTr�zftpwrapper.close� s$������=�=�A���O�O��rUc��|j�|xjdzc_|jdkr|js|j�yyy)Nr`r)r�rvrwr�r�s rTr�zftpwrapper.file_close� s@������� � ��� ��=�=�A��d�n�n��O�O��'5�rUc��|j� |jj�y#t�$rYywxYwrW)r�rNr�r-r�s rTr�zftpwrapper.real_close� s5������ ��H�H�N�N����{� �� �s�-� =�=)NT) r�r�r�r�rxr�r�r�r�r�r�rUrTr�r�t s/��E�?C� �� �*!�X�� �rUr�c��� i}g}tjj�D]s}t|�dkDs�|ddk(s�|ddj �dk(s�2tj|}|ddj �}|j|||f�|s�o|||<�udtjvr|j dd�|D])\}}}|dddk(s�|r|||<�|j |d��+|S) Nr�i����r����rf�REQUEST_METHODr�_proxy)re�environ�keysror�rlr�)rv�environmentrjr�� proxy_names rT�getproxies_environmentr�� s�����G��K�� � ���!���t�9�q�=�T�"�X��_��b�c����1B�g�1M��J�J�t�$�E��c�r����*�J�����e�Z�8�9��&+�� �#�"��2�:�:�%����F�D�!�#.���e�Z����9�� ��&+�� �#����J��-� $/��NrUc�� |� t�} |d}|dk(ry|j�}t|�\}}|j d�D]k}|j�}|s�|j d�}|j�}||k(s||k(ryd|z}|j|�s|j|�s�kyy#t$rYywxYw)N�noF�*Tr�.)r�rr�rr�r��lstripr�)r�rv�no_proxy�hostonlyr�rjs rT�proxy_bypass_environmentr�� s������(�*����4�=���3����:�:�<�D���%�N�H�d����s�#���z�z�|����;�;�s�#�D��:�:�<�D��4��4�4�<����:�D�� � ��&�$�-�-��*=��$���)����s�B8�8 C�Cc�� ddlm}ddlm}m}t |�\}}d�}d|vr|dryd} t||��}|j dd �D]�} | s�tjd | �} | �|�}|| jd��}| jd�}|�'d | jd�jd�dzz}nt|dd�}|dks|dkDr��d|z }||z ||z k(s��y||| �s��yy#|$rY��wxYw)Nr��fnmatch)�AddressValueError�IPv4Addressc���|jd�}ttt|��}t |�dk7r |gd�zdd}|ddz|ddzz|dd zz|d zS)Nr�r;)rrrrr�r`r�r;r�r�)r�r�r�rmro)�ipAddrr�s rT�ip2numz,_proxy_bypass_macosx_sysconf.<locals>.ip2num3 sm�����S�!���S��e�_�%���u�:��?��\�)�2�A�.�E��a��B��5��8�r�>�2�e�A�h�!�m�D�u�Q�x�O�OrUr��exclude_simpleT� exceptionsr�z(\d+(?:\.\d+)*)(/\d+)?r`r;r�� F)r�� ipaddressr�r�rrmr�r�rP�group�count) r��proxy_settingsr�r�r�r�r�r��hostIPr�rEr��masks rT�_proxy_bypass_macosx_sysconfr�" s1�� � �8���%�N�H�d�P��$���*�+�� �F� ��[��*�+�� �#�#�L�"�5���h��H�H�.��6���=�V�/��!�'�'�!�*�%�D��7�7�1�:�D��|��A�G�G�A�J�,�,�S�1�A�5�6���4���8�}���a�x�4�"�9����9�D��$��D�D�L�1�� �T�5� !��/6�2��9� �� �s�C<�<D�Dc�� ddlm}t|�\}}|jd�}|D])}|j�}|dk(rd|vs�y|||�s�)yy)Nrr�r0z<local>r�TF)r�rr�r�)r��overrider�r��proxy_overrider�s rT�_proxy_bypass_winreg_overrider�b sa��� ����G�D�!��^�^�C�(�N����z�z�|���9���$��� �T�4� ���rU�darwin)�_get_proxy_settings�_get_proxiesc�.�t�}t||�SrW)r�r�)r�r�s rT�proxy_bypass_macosx_sysconfr�} s��,�.��+�D�.�A�ArUc�� t�SrW)r�r�rUrT�getproxies_macosx_sysconfr�� s�� � �~�rUc�J� t�}|rt||�St|�SrW)r�r�r��r�rvs rTr}r}� s,�� �)�*���+�D�'�:�:�.�t�4�4rUc�.�t�xs t�SrW)r�r�r�rUrTr6r6� s��%�'�F�+D�+F�FrUc��� i} ddl} |j|jd�}|j |d�d}|r�t|j |d�d�}d|vrd|vrdj |�}|jd�D]F}|jdd�\}}tjd |�s|d vrd|z}n |dk(rd |z}|||<�H|jd�rJtjdd|d�}|jd�xs||d<|jd�xs||d<|j�|S#t$r|cYSwxYw#tttf$rY|SwxYw)Nr�;Software\Microsoft\Windows\CurrentVersion\Internet Settings�ProxyEnable�ProxyServerror0zhttp={0};https={0};ftp={0}r`z (?:[^/:]+)://)rr�rNrV�sockszsocks://z ^socks://z socks4://rr�)�winreg�ImportError�OpenKey�HKEY_CURRENT_USER�QueryValueExrr�r�r�rPr�r��Closer~rEr�)rvr��internetSettings�proxyEnable�proxyServer�pr��addresss rT�getproxies_registryr�� s��� � �� ��" �%�~�~�f�.F�.F�N� P�� �-�-�.>�/<�>�>?�A�K��!�&�"5�"5�6F�7D�#F�FG�#I�J���k�)�c��.D�">�"E�"E�k�"R�K�$�*�*�3�/�A�()����Q��%�H�g��8�8�O�W�=�#�'?�?�&/�'�&9�G�%��0�&0�7�&:�G�(/�G�H�%�0��;�;�w�'� �f�f�\�;���@P�Q�G�&-�k�k�&�&9�&D�W�G�F�O�'.�{�{�7�';�'F�w�G�G�$��"�"�$����M� ��N� ��B��Y�/� � ��� �s#�D;�D/E�;E �E �E$�#E$c�0� t�xs t�SrW)r�r�r�rUrTr6r6� s�� �&�'�@�+>�+@�@rUc� � ddl} |j|jd�}|j |d�d}t|j |d�d�}|r|syt||�S#t$rYywxYw#t$rYywxYw)NrFr�r�� ProxyOverride)r�r�r�r�r�rr~r�)r�r�r�r�� proxyOverrides rT�proxy_bypass_registryr�� s��� �� �%�~�~�f�.F�.F�N� P�� �-�-�.>�/<�>�>?�A�K��� 3� 3�4D�5D�!F�FG�!I�J�M� �-��,�T�=�A�A��� �� ��� �� �s#�A'�AA6�' A3�2A3�6 B�Bc�J� t�}|rt||�St|�SrW)r�r�r�r�s rTr}r}� s,�� �)�*���+�D�'�:�:�(��.�.rUrbrW)}r~r�r�r�http.clientrr�rer�rrVrr�rhrbrB�urllib.errorrrr�urllib.parserrrr r rrr rrrrrrrrrr�urllib.responserrrGrFr��__all__�version_infor�rMrr1r2rkr7r8r��ASCIIr�r�rrr3rr0rrror r!r"r#r$r%r&�urandomrr'r(r)r0r*r�rrLrlrr/r�r�r+r�r,r-r.r,rj� nturl2pathr5r4r�r9r:rlr{rnr&rpr-rrrsr�r�r�r�r��platform�_scproxyr�r�r�r�r}r6r�r�r�rUrT�<module>r�s���C�f� ���� � � � � � �����C�B�"�"�"�"�"� 5����I���$��(�(��!�,�,�� ���F�$B�$B�M+��4�5�$�M+�^���:�x��r�z�z�(�B�H�H�-��� k"�k"�ZI+�I+�^"�H8�8�&#��#�";�k�;�n2�+�n2�b,�B)>�;�)>�V=*�=*�@G�o�G�3�#B�3�>k#�k#�^�3�[���4�k�� �z�z��O�O�d�K�)B��$ �[�*C� �s�+�s�l3�%�3��4�;�;�)�*�8�*�8�$�N�N�>�"�#�+�#�$6�[�6� �)*�V11�+�11�f�7,��7,�r3�j�3�j:�+�:�B���7�7�d�?�5�5�C� A���+�+�DX�Y�X�z� �� � ��� ��� ��a�a�H#�J �J<�@�0�<�<�8��:�B��5�G��W�W��_�/�bA�B�(/�(�J�+�L��WT���I��s�9K�K�KPKS�\S�Q���#__pycache__/request.cpython-312.pycnu�[���� T��h[����dZddlZddlZddlZddlZddlZddlZddlZddl Z ddl Z ddlZddlZddl Z ddlZddlZddlZddlmZmZmZddlmZmZmZmZmZmZmZmZmZmZm Z m!Z!m"Z"m#Z#m$Z$m%Z%m&Z&m'Z'ddl(m)Z)m*Z* ddl+Z+dZ,gd�Z.d ej^dd zZ0da1de jdfddddd�d�Z3d �Z4gZ5dgd�Z6d�Z7e jpde jr�Z:d�Z;Gd�d�Z<Gd�d�Z=d�Z>Gd�d�Z?Gd�de?�Z@Gd�de?�ZAGd�de?�ZBd�ZCGd �d!e?�ZDGd"�d#�ZEGd$�d%eE�ZFGd&�d'eF�ZGGd(�d)�ZHGd*�d+eHe?�ZIGd,�d-eHe?�ZJej�ZLGd.�d/�ZMGd0�d1e?eM�ZNGd2�d3e?eM�ZOGd4�d5e?�ZPGd6�d7eP�ZQeRej�d8�rGd9�d:eP�ZTe.j�d:�Gd;�d<e?�ZVGd=�d>e?�ZWd?�ZXd@�ZYGdA�dBe?�ZZdC�Z[GdD�dEe?�Z\GdF�dGe\�Z]GdH�dIe?�Z^dJZ_ej�dKk(r ddLlambZbmcZcndM�ZbdN�ZciZdGdO�dP�ZeGdQ�dRee�ZfdagdS�ZhdaidT�ZjdakdU�ZldamdV�ZnGdW�dX�ZodY�ZpdhdZ�Zqd[�Zrd\�Zsej�d]k(rdd^lumvZvmwZwd_�Zxd`�Zyda�Zzdb�Z{yej�dKk(r dc�Z|dd�Z{de�Z}df�ZzyepZ{eqZzy#e-$rdZ,Y��[wxYw)ia� An extensible library for opening URLs using a variety of protocols The simplest way to use this module is to call the urlopen function, which accepts a string containing a URL or a Request object (described below). It opens the URL and returns the results as file-like object; the returned object has some extra methods described below. The OpenerDirector manages a collection of Handler objects that do all the actual work. Each Handler implements a particular protocol or option. The OpenerDirector is a composite object that invokes the Handlers needed to open the requested URL. For example, the HTTPHandler performs HTTP GET and POST requests and deals with non-error returns. The HTTPRedirectHandler automatically deals with HTTP 301, 302, 303, 307, and 308 redirect errors, and the HTTPDigestAuthHandler deals with digest authentication. urlopen(url, data=None) -- Basic usage is the same as original urllib. pass the url and optionally data to post to an HTTP URL, and get a file-like object back. One difference is that you can also pass a Request instance instead of URL. Raises a URLError (subclass of OSError); for HTTP errors, raises an HTTPError, which can also be treated as a valid response. build_opener -- Function that creates a new OpenerDirector instance. Will install the default handlers. Accepts one or more Handlers as arguments, either instances or Handler classes that it will instantiate. If one of the argument is a subclass of the default handler, the argument will be installed instead of the default. install_opener -- Installs a new opener as the default opener. objects of interest: OpenerDirector -- Sets up the User Agent as the Python-urllib client and manages the Handler classes, while dealing with requests and responses. Request -- An object that encapsulates the state of a request. The state can be as simple as the URL. It can also include extra HTTP headers, e.g. a User-Agent. BaseHandler -- internals: BaseHandler and parent _call_chain conventions Example usage: import urllib.request # set up authentication info authinfo = urllib.request.HTTPBasicAuthHandler() authinfo.add_password(realm='PDQ Application', uri='https://mahler:8092/site-updates.py', user='klem', passwd='geheim$parole') proxy_support = urllib.request.ProxyHandler({"http" : "http://ahad-haam:3128"}) # build a new opener that adds authentication and caching FTP handlers opener = urllib.request.build_opener(proxy_support, authinfo, urllib.request.CacheFTPHandler) # install it urllib.request.install_opener(opener) f = urllib.request.urlopen('https://www.python.org/') �N)�URLError� HTTPError�ContentTooShortError)�urlparse�urlsplit�urljoin�unwrap�quote�unquote� _splittype� _splithost� _splitport� _splituser�_splitpasswd� _splitattr�_splitquery�_splitvalue� _splittag� _to_bytes�unquote_to_bytes� urlunparse)� addinfourl�addclosehookTF)!�Request�OpenerDirector�BaseHandler�HTTPDefaultErrorHandler�HTTPRedirectHandler�HTTPCookieProcessor�ProxyHandler�HTTPPasswordMgr�HTTPPasswordMgrWithDefaultRealm�HTTPPasswordMgrWithPriorAuth�AbstractBasicAuthHandler�HTTPBasicAuthHandler�ProxyBasicAuthHandler�AbstractDigestAuthHandler�HTTPDigestAuthHandler�ProxyDigestAuthHandler�HTTPHandler�FileHandler� FTPHandler�CacheFTPHandler�DataHandler�UnknownHandler�HTTPErrorProcessor�urlopen�install_opener�build_opener�pathname2url�url2pathname� getproxies�urlretrieve� urlcleanup� URLopener�FancyURLopenerz%d.%d�)�cafile�capath� cadefault�contextc���|s|s|r�ddl}|jdtd�|�td��tstd��tjt jj||��}|jdg�t|� �}t|�} n3|rt|� �}t|�} nt� t�xa} nt} | j|||�S) a�Open the URL url, which can be either a string or a Request object. *data* must be an object specifying additional data to be sent to the server, or None if no such data is needed. See Request for details. urllib.request module uses HTTP/1.1 and includes a "Connection:close" header in its HTTP requests. The optional *timeout* parameter specifies a timeout in seconds for blocking operations like the connection attempt (if not specified, the global default timeout setting will be used). This only works for HTTP, HTTPS and FTP connections. If *context* is specified, it must be a ssl.SSLContext instance describing the various SSL options. See HTTPSConnection for more details. The optional *cafile* and *capath* parameters specify a set of trusted CA certificates for HTTPS requests. cafile should point to a single file containing a bundle of CA certificates, whereas capath should point to a directory of hashed certificate files. More information can be found in ssl.SSLContext.load_verify_locations(). The *cadefault* parameter is ignored. This function always returns an object which can work as a context manager and has the properties url, headers, and status. See urllib.response.addinfourl for more detail on these properties. For HTTP and HTTPS URLs, this function returns a http.client.HTTPResponse object slightly modified. In addition to the three new methods above, the msg attribute contains the same information as the reason attribute --- the reason phrase returned by the server --- instead of the response headers as it is specified in the documentation for HTTPResponse. For FTP, file, and data URLs and requests explicitly handled by legacy URLopener and FancyURLopener classes, this function returns a urllib.response.addinfourl object. Note that None may be returned if no handler handles the request (though the default installed global OpenerDirector uses UnknownHandler to ensure this never happens). In addition, if proxy settings are detected (for example, when a *_proxy environment variable like http_proxy is set), ProxyHandler is default installed and makes sure the requests are handled through the proxy. rNzJcafile, capath and cadefault are deprecated, use a custom context instead.r;zDYou can't pass both context and any of cafile, capath, and cadefaultzSSL support not available)r<r=zhttp/1.1�r?)�warnings�warn�DeprecationWarning� ValueError� _have_ssl�ssl�create_default_context�Purpose�SERVER_AUTH�set_alpn_protocols�HTTPSHandlerr3�_opener�open) �url�data�timeoutr<r=r>r?rB� https_handler�openers �'/usr/lib64/python3.12/urllib/request.pyr1r1�s���h��9���� � �0�1C�Q� H������ ���8�9�9��,�,�S�[�[�-D�-D�4:�4:�<�� �"�"�J�<�0�$�W�5� ��m�,�� �$�W�5� ��m�,�� ��'�>�)��&����;�;�s�D�'�*�*�c��|ay�N)rM)rSs rTr2r2�s���GrUc��t|�\}}tjt||��5}|j �}|dk(r,|s*t jj|�|fcddd�S|r t|d�}n7tjd��}|j}tj|�|5||f} d} d}d}d} d |vrt|d �}|r || | |�|j| �x}rD|t!|�z }|j#|�| dz } |r || | |�|j| �x}r�Dddd�ddd�dk\r|krt%d||fz �� S#1swY�.xYw#1swY�2xYw) aW Retrieve a URL into a temporary location on disk. Requires a URL argument. If a filename is passed, it is used as the temporary file location. The reporthook argument should be a callable that accepts a block number, a read size, and the total file size of the URL target. The data argument should be valid URL encoded data. If a filename is passed and the URL points to a local resource, the result is a copy from local file to new file. Returns a tuple containing the path to the newly created data file as well as the resulting HTTPMessage object. �fileN�wbF)�delete� ���r�content-length�Content-Length��1retrieval incomplete: got only %i out of %i bytes)r� contextlib�closingr1�info�os�path�normpathrN�tempfile�NamedTemporaryFile�name�_url_tempfiles�append�int�read�len�writer)rO�filename� reporthookrP�url_typerf�fp�headers�tfp�result�bs�sizern�blocknum�blocks rTr7r7�s��� ��_�N�H�d� � � �G�C��.� /�2��'�'�)���v��h��7�7�#�#�D�)�7�2� 0� /���x��&�C��-�-�U�;�C��x�x�H��!�!�(�+� ��w�&�F��B��D��D��H��7�*��7�#3�4�5����8�R��.��7�7�2�;�&�%�&���E� �"��� � �%� ��A� ����x��T�2��7�7�2�;�&�%�&��! 0�F�q�y�T�D�[�"�?��T�l� �"�$� $��M�1�S��! 0� /�s+�8E3�0AE3�8BE'�:E3�'E0 �,E3�3E<c��tD]} tj|��tdd�=trdayy#t$rY�9wxYw)z0Clean up temporary files from urlretrieve calls.N)rkre�unlink�OSErrorrM)� temp_files rTr8r8sH��#� � ��I�I�i� �$� �q�������� �� �s�5� A�Az:\d+$c��|j}t|�d}|dk(r|jdd�}tj d|d�}|j�S)z�Return request-host, as defined by RFC 2965. Variation from RFC: returned value is lowercased, for convenient comparison. r`��Host)�full_urlr� get_header�_cut_port_re�sub�lower)�requestrO�hosts rT�request_hostr�)sX��� � �C��C�=���D��r�z��!�!�&�"�-�����B��a�(�D��:�:�<�rUc��eZdZdidddfd�Zed��Zejd��Zejd��Zed��Zejd��Zejd ��Zd �Z d�Z d�Zd �Zd�Z d�Zd�Zd�Zdd�Zd�Zd�Zy)rNFc��||_i|_i|_d|_||_d|_|j �D]\}}|j||��|�t|�}||_ ||_ |r||_yyrW)r�ru�unredirected_hdrs�_datarP�_tunnel_host�items� add_headerr��origin_req_host�unverifiable�method) �selfrOrPrur�r�r��key�values rT�__init__zRequest.__init__;s����� ����!#����� ��� � ���!�-�-�/�J�C���O�O�C��'�*��"�*�4�0�O�.���(���� �D�K�rUc�~�|jr&dj|j|j�S|jS)Nz{}#{})�fragment�format� _full_url�r�s rTr�zRequest.full_urlMs,���=�=��>�>�$�.�.�$�-�-�@�@��~�~�rUc��t|�|_t|j�\|_|_|j �yrW)r r�rr��_parse�r�rOs rTr�zRequest.full_urlSs/�� �����(1�$�.�.�(A�%���� ���� rUc�.�d|_d|_d|_y)Nr�)r�r��selectorr�s rTr�zRequest.full_urlZs������� ��� rUc��|jSrW)r�r�s rTrPzRequest.data`s���z�z�rUc�x�||jk7r+||_|jd�r|jd�yyy)N�Content-length)r�� has_header� remove_header)r�rPs rTrPzRequest.datads=���4�:�:���D�J����/�0��"�"�#3�4�1�rUc��d|_yrW)rPr�s rTrPzRequest.datans ���� rUc��t|j�\|_}|j�td|jz��t|�\|_|_|jrt|j�|_yy)Nzunknown url type: %r) rr��typerEr�r r�r�r)r��rests rTr�zRequest._parsersd��$�T�^�^�4��� �4��9�9���3�d�m�m�C�D�D�#-�d�#3� �� �4�=��9�9��� � �*�D�I�rUc�<�|j�dnd}t|d|�S)z3Return a string indicating the HTTP request method.�POST�GETr�)rP�getattr)r��default_methods rT� get_methodzRequest.get_methodzs!��#'�9�9�#8��e���t�X�~�6�6rUc��|jSrW)r�r�s rT�get_full_urlzRequest.get_full_urls���}�}�rUc��|jdk(r%|js|j|_||_y||_|j|_||_y)N�https)r�r�r�r�r�)r�r�r�s rT� set_proxyzRequest.set_proxy�sF���9�9����(9�(9� $� � �D���� ��D�I� �M�M�D�M��� rUc�4�|j|jk(SrW)r�r�r�s rT� has_proxyzRequest.has_proxy�s���}�}�� � �-�-rUc�>�||j|j�<yrW)ru� capitalize�r�r��vals rTr�zRequest.add_header�s��),����S�^�^�%�&rUc�>�||j|j�<yrW)r�r�r�s rT�add_unredirected_headerzRequest.add_unredirected_header�s��36����s�~�~�/�0rUc�>�||jvxs||jvSrW)rur��r��header_names rTr�zRequest.has_header�s&���t�|�|�+�6��t�5�5�5� 7rUc�n�|jj||jj||��SrW)ru�getr�)r�r��defaults rTr�zRequest.get_header�s2���|�|�����"�"�&�&�{�G�<�>� >rUc�t�|jj|d�|jj|d�yrW)ru�popr�r�s rTr�zRequest.remove_header�s,��������d�+����"�"�;��5rUc�h�i|j�|j�}t|j��SrW)r�ru�listr�)r��hdrss rT�header_itemszRequest.header_items�s,��9�$�(�(�9�D�L�L�9���D�J�J�L�!�!rUrW)�__name__� __module__�__qualname__r��propertyr��setter�deleterrPr�r�r�r�r�r�r�r�r�r�r��rUrTrr9s���!%�r�!%�E��!�$���� �_�_���������� ���� �[�[�5��5� �\�\����+�7� ��.�-�7�7�>� 6�"rUrc�R�eZdZd�Zd�Zd�Zd�Zdejfd�Z d d�Z d�Zy) rc�p�dtz}d|fg|_g|_i|_i|_i|_i|_y)N�Python-urllib/%sz User-agent)�__version__� addheaders�handlers�handle_open�handle_error�process_response�process_request)r��client_versions rTr�zOpenerDirector.__init__�sB��+�k�9��(�.�9�:����� ������� "���!��rUc��t|d�stdt|�z��d}t|�D�] }|dvr� |j d�}|d|}||dzd}|jd�rW|j d�|zdz}||dzd} t |�}|jj|i�} | |j|<n=|dk(r|}|j} n)|d k(r|}|j} n|d k(r|}|j} n��| j|g�} | rtj| |�n| j!|�d}��|r2tj|j"|�|j%|�yy#t$rY��wxYw)N� add_parentz%expected BaseHandler instance, got %rF)�redirect_request�do_open� proxy_open�_r`�errorrN�responser�T)�hasattr� TypeErrorr��dir�find� startswithrmrEr�r�r�r�r�� setdefault�bisect�insortrlr�r�)r��handler�added�meth�i�protocol� condition�j�kind�lookupr�s rT�add_handlerzOpenerDirector.add_handler�s����w��-��C� ��M�*�+� +�����L�D��D�D��� � �#��A��B�Q�x�H��Q�q�S�T� �I��#�#�G�,��N�N�3�'�!�+�a�/���A�a�C�D�z����t�9�D��*�*�.�.�x��<��.4��!�!�(�+��f�$����)�)���j�(����.�.���i�'����-�-����(�(��r�2�H��� � �h��0�����(��E�G!�J��M�M�$�-�-��1����t�$���/"����s� E3�3 E?�>E?c��yrWr�r�s rT�closezOpenerDirector.close����rUc�d�|j|d�}|D]}t||�}||�}|��|cSy)Nr�)r�r�) r��chainr�� meth_name�argsr�r��funcrws rT�_call_chainzOpenerDirector._call_chain�s>���9�9�T�2�&���G��7�I�.�D��4�[�F��!�� � rUNc��t|t�r t||�}n|}|�||_||_|j }|dz}|jj|g�D]}t||�}||�}�tjd|j|j|j|j��|j||�} |dz}|jj|g�D]}t||�}||| �} �| S)N�_requestzurllib.Request� _response)� isinstance�strrrPrQr�r�r�r��sys�auditr�rur��_openr�) r��fullurlrPrQ�reqr�r�� processorr�r�s rTrNzOpenerDirector.open�s����g�s�#��'�4�(�C��C����������8�8���Z�'� ��-�-�1�1�(�B�?�I��9�i�0�D��s�)�C�@� � � �"�C�L�L�#�(�(�C�K�K����IY�Z��:�:�c�4�(���[�(� ��.�.�2�2�8�R�@�I��9�i�0�D��C��*�H�A��rUc���|j|jdd|�}|r|S|j}|j|j||dz|�}|r|S|j|jdd|�S)Nr��default_openr�unknown�unknown_open)r�r�r�)r�r rPrwr�s rTrzOpenerDirector._open s����!�!�$�"2�"2�I�"0�#�7����M��8�8���!�!�$�"2�"2�H�h�")�?*�+.�0����M����� 0� 0�)� .��5� 5rUc���|dvr|jd}|d}d|z}d}|}n|j}|dz}d}|||f|z}|j|�}|r|S|r|dd fz}|j|�Sy) N��httpr�rr;z http_error_%sr`�_errorrr��http_error_default)r�r�)r��protor��dictr��http_err� orig_argsrws rTr�zOpenerDirector.errors����%�%��$�$�V�,�D���G�E�'�%�/�I��H��I��$�$�D���(�I��H��e�Y�'�$�.��!��!�!�4�(����M���)�%9�:�Y�F�D�#�4�#�#�T�*�*�rUrW)r�r�r�r�r�r�r��socket�_GLOBAL_DEFAULT_TIMEOUTrNrr�r�rUrTrr�s3�� "�-%�^ � �"&�v�/M�/M��: 5�+rUrc �h�t�}ttttt ttttg }ttjd�r|jt�t�}|D]V}|D]O}t!|t"�rt%||�s� |j'|��2t!||�s�?|j'|��Q�X|D]}|j)|��|D]}|j+|���|D]*}t!|t"�r|�}|j+|��,|S)a*Create an opener object from a list of handlers. The opener will use several default handlers, including support for HTTP, FTP and when applicable HTTPS. If any of the handlers passed as arguments are subclasses of the default handlers, the default handlers will not be used. �HTTPSConnection)rr r/r*rrr,r+r0r.r�r�clientrlrL�setrr�� issubclass�add�remover�)r�rS�default_classes�skip�klass�check�hs rTr3r35s��� �F�#�^�[�.�0C�!�;�0B�"�$�O��t�{�{�-�.����|�,��5�D� ���E��%��&��e�U�+��H�H�U�O��E�5�)�������!������u�%��!�����5�7�#�!����a�����A����1����MrUc�"�eZdZdZd�Zd�Zd�Zy)r��c��||_yrW)�parent)r�r)s rTr�zBaseHandler.add_parent\s ����rUc��yrWr�r�s rTr�zBaseHandler.close_r�rUc�N�t|d�sy|j|jkS)N� handler_orderT)r�r,)r��others rT�__lt__zBaseHandler.__lt__cs(���u�o�.���!�!�E�$7�$7�7�7rUN)r�r�r�r,r�r�r.r�rUrTrrYs���M�� �8rUrc��eZdZdZdZd�ZeZy)r0zProcess HTTP error responses.i�c��|j|j|j�}}}d|cxkrdks"n|jj d|||||�}|S)N���,r)�code�msgrdr)r�)r�r�r�r3r4r�s rT� http_responsez HTTPErrorProcessor.http_responsepsT��"�-�-����x�}�}��4�c���t�!�c�!��{�{�(�(����4��d�<�H��rUN)r�r�r��__doc__r,r5�https_responser�rUrTr0r0ls��'��M� �#�NrUr0c��eZdZd�Zy)rc�4�t|j||||��rW)rr�)r�r rtr3r4r�s rTrz*HTTPDefaultErrorHandler.http_error_default~s������d�C��r�:�:rUN)r�r�r�rr�rUrTrr}s��;rUrc�4�eZdZdZdZd�Zd�ZexZxZxZ Z dZy)r�� c�Z�|j�}|dvr|dvs"|dvr|dk(st|j||||��|jdd�}d}|jj�D� � cic]\} } | j �|vr| | ��}} } t|||jd� �Scc} } w) a�Return a Request or None in response to a redirect. This is called by the http_error_30x methods when a redirection response is received. If a redirection should take place, return a new Request to allow http_error_30x to perform the redirect. Otherwise, raise HTTPError if no-one else should try to handle this url. Return None if you can't but another Handler might. )�-�.�/i3i4)r��HEAD)r>r?r@r�� z%20)r^zcontent-typeT)rur�r�) r�rr��replacerur�r�rr�)r�r rtr3r4ru�newurl�m�CONTENT_HEADERS�k�v� newheaderss rTr�z$HTTPRedirectHandler.redirect_request�s��� �N�N����2�2�q�O�7K���&�1��;��C�L�L�$��W�b�A�A�����U�+��<��'*�{�{�'8�'8�':�;�':�t�q�!�����/�9���d�':� �;��v�)�'*�':�':�$(�*� *��;s�,B'c�Z�d|vr|d}nd|vr|d}nyt|�}|jdvrt|||�d|�d�||��|js|jrt|�}d|d<t |�}t|dtj� �}t|j|�}|j||||||�}|�yt|d �rp|jx} |_| j|d�|j k\st#| �|j$k\r6t|j||j&|z||��ix} x|_|_| j|d�dz| |<|j)�|j+�|j,j/||j0� �S)N�location�uri�rr��ftpr�z - Redirection to url 'z' is not allowed�/r;z iso-8859-1)�encoding�safe� redirect_dictrr`�rQ)r�schemerrf�netlocr�rr �string�punctuationrr�r�r�rRr��max_repeatsro�max_redirections�inf_msgrnr�r)rNrQ) r�r rtr3r4rurD�urlparts�new�visiteds rT�http_error_302z"HTTPRedirectHandler.http_error_302�s����� ��Z�(�F� �g� ��U�^�F���F�#�� �?�?�">�>����AD�f�M���� � �}�}�����H�~�H��H�Q�K��H�%�� ��\��0B�0B�D������v�.�� �#�#�C��T�3���H���;���3��(�*-�*;�*;�;�G�c�'����F�A�&�$�*:�*:�:��G��� 5� 5�5�����d� $���s� 2�G�R�A�A�?A�@�G�@�c�'�#�*;�!�+�+�f�a�0�1�4���� ��� � ��� ��{�{����S�[�[��9�9rUzoThe HTTP server returned a redirect error that would lead to an infinite loop. The last 30x error message was: N)r�r�r�rXrYr�r^�http_error_301�http_error_303�http_error_307�http_error_308rZr�rUrTrr�s<���K��� *�L::�xIW�V�N�V�^�V�n�~�2�GrUrc�f�t|�\}}|jd�sd}|}ne|jd�std|z��d|vr$|jd�}|jd|�}n|jdd�}|dk(rd}|d|}t |�\}}|�t|�\}} ndx}} ||| |fS)aReturn (scheme, user, password, host/port) given a URL or an authority. If a URL is supplied, it must have an authority (host:port) component. According to RFC 3986, having an authority component means the URL must have two slashes after the scheme. rON�//zproxy URL with no authority: %r�@r;r])rr�rEr�rr) �proxyrT�r_scheme� authority�host_separator�end�userinfo�hostport�user�passwords rT�_parse_proxyro�s���"�%�(��F�H����s�#���� ��"�"�4�(��>��F�G�G��(�?�%�]�]�3�/�N��-�-��^�4�C��-�-��Q�'�C��"�9��C��Q�s�O� �#�I�.��H�h���%�h�/���h����x��4��8�+�+rUc��eZdZdZdd�Zd�Zy)r �dNc���|� t�}t|d�sJd��||_|j�D]4\}}|j �}t|d|z|||jfd���6y)N�keys�proxies must be a mappingz%s_openc��||||�SrWr�)�rrfr�r�s rT�<lambda>z'ProxyHandler.__init__.<locals>.<lambda>s���Q��t�,rU)r6r��proxiesr�r��setattrr�)r�rxr�rOs rTr�zProxyHandler.__init__sk���?� �l�G��w��'�D�)D�D�'���� ����I�D�#��:�:�<�D��D�)�d�*�$'�d����-� .�)rUc���|j}t|�\}}}}|�|}|jrt|j�ry|rb|r`t |��dt |���} tj| j��jd�} |jdd| z�t |�}|j||�||k(s|dk(ry|jj||j��S)N�:�ascii�Proxy-authorization�Basic r�rS)r�ror��proxy_bypassr�base64� b64encode�encode�decoder�r�r)rNrQ)r�r rfr�� orig_type� proxy_typermrnrl� user_pass�credss rTr�zProxyHandler.proxy_open"s����H�H� �/;�E�/B�,� �D�(�H���"�J��8�8��S�X�X�.���H�#*�4�=�#*�8�#4�6�I��$�$�Y�%5�%5�%7�8�?�?��H�E��N�N�0�(�U�2B�C��8�$��� � �h� �+�� �"�i�7�&:���;�;�#�#�C����#�=�=rUrW)r�r�r�r,r�r�r�rUrTr r s���M� .�>rUr c�,�eZdZd�Zd�Zd�Zdd�Zd�Zy)r!c��i|_yrW)�passwdr�s rTr�zHTTPPasswordMgr.__init__@s ����rUc�����t|t�r|g}|�jvri�j|<dD]+�t��fd�|D��}||f�j||<�-y)N�TFc3�B�K�|]}�j|�����y�wrW)� reduce_uri)�.0�u�default_portr�s ��rT� <genexpr>z/HTTPPasswordMgr.add_password.<locals>.<genexpr>Js ����� ?�:=�Q�����<�0�#�s�)rrr��tuple)r��realmrLrmr��reduced_urir�s` @rT�add_passwordzHTTPPasswordMgr.add_passwordCsf����c�3���%�C�����#�!#�D�K�K���'�L�� ?�:=� ?�?�K�/3�V�n�D�K�K���{�+�(rUc���|jj|i�}dD]M}|j||�}|j�D]&\}}|D]}|j ||�s�|cccS�(�Oy)Nr��NN)r�r�r�r�� is_suburi) r�r��authuri�domainsr��reduced_authuri�uris�authinforLs rT�find_user_passwordz"HTTPPasswordMgr.find_user_passwordNsf���+�+�/�/�%��,��'�L�"�o�o�g�|�D�O�")�-�-�/���h��C��~�~�c�?�;�'�� �#2�(�rUc��t|�}|dr|d}|d}|dxsd}nd}|}d}t|�\}}|r!|�|�ddd�j|�} | �d || fz}||fS) z@Accept authority or URI and extract only the authority and path.r`rr;rON�Pi�rz%s:%d)rrr�) r�rLr��partsrTrhrfr��port�dports rTr�zHTTPPasswordMgr.reduce_uriXs����� ����8��1�X�F��a��I���8�?�s�D��F��I��D�� �*� ��d��D�L�V�-?��!���s�6�{� �� �#�t�U�m�3� ��$��rUc�r�||k(ry|d|dk7ry|d}|dddk7r|dz }|dj|�S)zcCheck if test is below base in a URI tree Both args must be URIs in reduced form. TrFr`r]NrO)r�)r��base�test�prefixs rTr�zHTTPPasswordMgr.is_suburiosV�� �4�<����7�d�1�g����a����"�#�;�#���c�M�F��A�w�!�!�&�)�)rUN)T)r�r�r�r�r�r�r�r�r�rUrTr!r!>s��� =���.*rUr!c��eZdZd�Zy)r"c�p�tj|||�\}}|�||fStj|d|�SrW)r!r�)r�r�r�rmrns rTr�z2HTTPPasswordMgrWithDefaultRealm.find_user_password�sC��(�;�;�D�%�<C�E���h�����>�!��1�1�$��g�F�FrUN)r�r�r�r�r�rUrTr"r"~s��GrUr"c�8��eZdZ�fd�Zd�fd� Zdd�Zd�Z�xZS)r#c�0��i|_t�|� �yrW)� authenticated�superr�)r�� __class__s �rTr�z%HTTPPasswordMgrWithPriorAuth.__init__�s������ ���rUc�v��|j||�|�t�|� d|||�t�|� ||||�yrW)�update_authenticatedr�r�)r�r�rLrmr��is_authenticatedr�s �rTr�z)HTTPPasswordMgrWithPriorAuth.add_password�s@����!�!�#�'7�8����G� ��s�D�&�9� ���U�C��v�6rUc��t|t�r|g}dD]*}|D]#}|j||�}||j|<�%�,y�Nr�)rrr�r�)r�rLr�r�r�r�s rTr�z1HTTPPasswordMgrWithPriorAuth.update_authenticated�sG���c�3���%�C�'�L���"�o�o�a��>��2B��"�"�;�/��(rUc��dD]J}|j||�}|jD]'}|j||�s�|j|ccS�Lyr�)r�r�r�)r�r�r�r�rLs rTr�z-HTTPPasswordMgrWithPriorAuth.is_authenticated�sK��'�L�"�o�o�g�|�D�O��)�)���>�>�#��7��-�-�c�2�2�*�(rU)F)r�r�r�r�r�r�r�� __classcell__)r�s@rTr#r#�s����7�C�3rUr#c�t�eZdZejdej �Zd d�Zd�Zd�Z d�Z d�Zd�ZeZ eZy) r$z1(?:^|,)[ ]*([^ ,]+)[ ]+realm=(["']?)([^"']*)\2Nc�`�|� t�}||_|jj|_yrW)r!r�r�)r��password_mgrs rTr�z!AbstractBasicAuthHandler.__init__�s)����*�,�L�"��� �K�K�4�4��rUc#�K�d}tjj|�D]=}|j�\}}}|dvrt j dtd�||f��d}�?|s|r|j�d}nd}|df��yy�w)NF)�"�'zBasic Auth Realm was unquoted�Trr�)r$�rx�finditer�groupsrBrC�UserWarning�split)r��header�found_challenge�morTr r�s rT�_parse_realmz%AbstractBasicAuthHandler._parse_realm�s�������*�-�-�6�6�v�>�B�#%�9�9�;� �F�E�5��J�&�� � �=�)�1�.��5�/�!�"�O�?��������*�����4�.� ��s�BBc���|j|�}|syd}|D]J}|j|�D]4\}}|j�dk7r|}�|��|j|||�ccS�L|�t d����y)N�basicz@AbstractBasicAuthHandler does not support the following scheme: )�get_allr�r��retry_http_basic_authrE) r��authreqr�r ru�unsupportedr�rTr�s rT�http_error_auth_reqedz.AbstractBasicAuthHandler.http_error_auth_reqed�s����/�/�'�*�������F�!%�!2�!2�6�!:� ����<�<�>�W�,�"(�K���$� �5�5�d�C��G�G�";���"�� &�)�*� *�#rUc��|jj||�\}}|��|�d|��}dtj|j ��jd�z}|j |jd�|k(ry|j|j|�|jj||j��Sy)Nr{r~r|rS)r�r�r�r�r�r�r��auth_headerr�r)rNrQ)r�r�r r�rm�pw�raw�auths rTr�z.AbstractBasicAuthHandler.retry_http_basic_auth�s����;�;�1�1�%��>���b� �>�!�2�&�C��f�.�.�s�z�z�|�<�C�C�G�L�L�D��~�~�d�.�.��5��=���'�'��(8�(8�$�?��;�;�#�#�C����#�=�=�rUc���t|jd�r%|jj|j�s|S|j d�s�|jjd|j�\}}dj ||�j�}tj|�j�}|jddj |j���|S)Nr�� Authorizationz{0}:{1}zBasic {}) r�r�r�r�r�r�r�r�r��standard_b64encoder�r��strip)r�r rmr��credentials�auth_strs rT�http_requestz%AbstractBasicAuthHandler.http_request�s�������%7�8��{�{�+�+�C�L�L�9��J��~�~�o�.��;�;�9�9�$����M�L�D�&�#�*�*�4��8�?�?�A�K��0�0��=�D�D�F�H��'�'��(2�(9�(9�(�.�.�:J�(K� M�� rUc��t|jd�rfd|jcxkrdkr+nn(|jj|jd�|S|jj|jd�|S)Nr�r1r2TF)r�r�r3r�r�)r�r r�s rTr5z&AbstractBasicAuthHandler.http_response s`���4�;�;� 2�3��h�m�m�)�c�)����0�0����t�D������0�0����u�E��rUrW)r�r�r��re�compile�Ir�r�r�r�r�r�r5� https_requestr7r�rUrTr$r$�sL�� ����1��D�D� �B�5�!�(*�4 ���!�M�"�NrUr$c��eZdZdZd�Zy)r%r�c�F�|j}|jd|||�}|S)N�www-authenticate)r�r�)r�r rtr3r4rurOr�s rT�http_error_401z#HTTPBasicAuthHandler.http_error_401s*���l�l���-�-�.@�*-�s�G�=���rUN)r�r�r�r�r�r�rUrTr%r%s��!�K�rUr%c��eZdZdZd�Zy)r&r}c�F�|j}|jd|||�}|S�N�proxy-authenticate)r�r�)r�r rtr3r4rurhr�s rT�http_error_407z$ProxyBasicAuthHandler.http_error_407%s-�� �H�H� ��-�-�.B�*3�S�'�C���rUN)r�r�r�r�r�r�rUrTr&r&!s��'�K�rUr&c�>�eZdZd d�Zd�Zd�Zd�Zd�Zd�Zd�Z d �Z y)r'Nc��|� t�}||_|jj|_d|_d|_d|_y�Nr)r!r�r��retried�nonce_count� last_nonce)r�r�s rTr�z"AbstractDigestAuthHandler.__init__?s>���>�$�&�F���� �K�K�4�4�����������rUc��d|_yr�)r�r�s rT�reset_retry_countz+AbstractDigestAuthHandler.reset_retry_countHs ����rUc�Z�|j|d�}|jdkDrt|jdd|d��|xjdz c_|rZ|j �d}|j�dk(r|j ||�S|j�dk7rtd|z��yy) N�i�zdigest auth failedr`r�digestr�zEAbstractDigestAuthHandler does not support the following scheme: '%s')r�r�rr�r�r��retry_http_digest_authrE)r�r�r�r rur�rTs rTr�z/AbstractDigestAuthHandler.http_error_auth_reqedKs����+�+�k�4�0���<�<�!���C�L�L�#�/C�#�T�+� +� �L�L�A��L���]�]�_�Q�'�F��|�|�~��)��2�2�3��@�@�����7�*� �"?�AG�"H�I�I�+� rUc�z�|jdd�\}}ttdt|���}|j ||�}|rtd|z}|j j |jd�|k(ry|j|j|�|jj||j��}|Sy)NrBr`z Digest %srS)r��parse_keqv_list�filter�parse_http_list�get_authorizationrur�r�r�r)rNrQ)r�r r��token� challenge�chal�auth_val�resps rTr�z0AbstractDigestAuthHandler.retry_http_digest_auth_s����:�:�c�1�-���y��v�d�O�I�,F�G�H���%�%�c�4�0���"�T�)�H��{�{���t�/�/��6�(�B���'�'��(8�(8�(�C��;�;�#�#�C����#�=�D��K� rUc���|j�d|�dtj��d�}|jd�t d�z}tj|�j�}|ddS)Nr{r|��)r��time�ctimer��_randombytes�hashlib�sha1� hexdigest)r��nonce�s�b�digs rT� get_cnoncez$AbstractDigestAuthHandler.get_cnonceksT�� �+�+�U�D�J�J�L�A�� �H�H�W���Q��/���l�l�1�o�'�'�)���3�B�x�rUc��� |d}|d}|jd�}|jdd�}|jdd�}|j|�\}} |�y|jj ||j �\} }| �y|j�|j|j|�}nd}| �d|�d|��} |j��d|j��}|�| || �|�d||����}n�d|jd �vry||jk(r|xjd z c_nd |_||_d|jz}|j|�}|�d|�d|�dd�d||��� }| || �|�}ntd|z��d | �d|�d|�d|j�d|�d�}|r|d|zz }|r|d|zz }|d|zz }|r|d�d�d�z }|S#t$rYywxYw)Nr�r�qop� algorithm�MD5�opaquer{r��,r`z%08xzqop '%s' is not supported.z username="z ", realm="z ", nonce="z", uri="z ", response="r�z , opaque="%s"z , digest="%s"z, algorithm="%s"z, qop=auth, nc=z , cnonce=")r��KeyError�get_algorithm_implsr�r�r�rP�get_entity_digestr�r�r�r�r�rr)r�r r�r�rr rr�H�KDrmr��entdig�A1�A2�respdig�ncvalue�cnonce�noncebitr�s rTr�z+AbstractDigestAuthHandler.get_authorizationvs�� ���M�E���M�E��(�(�5�/�C�����e�4�I��X�X�h��-�F��(�(��3���2��9���;�;�1�1�%����F���b��<���8�8���+�+�C�H�H�d�;�F��F���� +�����(����&�� �;���2��5�!�B�%� 8�9�G� �s�y�y��~� %�����'�� � �A�%� �#$�� �"'����t�/�/�/�G��_�_�U�+�F�+0�'�6�6�1�R�5�Q�H���2���)�G��7�#�=�>�>�� #'��u�c�l�l�")�+����O�f�,�,�D���O�f�,�,�D��"�Y�.�.������H�H�D����g� �� �s�?G� G#�"G#c�V��|dk(rd��n|dk(rd��ntd|z���fd�}�|fS)Nrc�f�tj|jd��j�S�Nr|)r�md5r�r��xs rTrwz?AbstractDigestAuthHandler.get_algorithm_impls.<locals>.<lambda>�s��'�+�+�a�h�h�w�&7�8�B�B�DrU�SHAc�f�tj|jd��j�Sr )rrr�rr"s rTrwz?AbstractDigestAuthHandler.get_algorithm_impls.<locals>.<lambda>�s��'�,�,�q�x�x��'8�9�C�C�ErUz.Unsupported digest authentication algorithm %rc����|�d|���S)Nr{r�)r�drs �rTrwz?AbstractDigestAuthHandler.get_algorithm_impls.<locals>.<lambda>�s���!�q�!�,�-rU)rE)r�rrrs @rTrz-AbstractDigestAuthHandler.get_algorithm_impls�sG������D�A� �%� �E�A��,�.7�8�9� 9� -���"�u�rUc��yrWr�)r�rPr�s rTrz+AbstractDigestAuthHandler.get_entity_digest�s��rUrW)r�r�r�r�r�r�r�rr�rrr�rUrTr'r'4s,����I�( � �<�|�rUr'c��eZdZdZdZdZd�Zy)r(z�An authentication protocol defined by RFC 2069 Digest authentication improves on basic authentication because it does not transmit passwords in the clear. r���c�~�t|j�d}|jd|||�}|j�|S)Nr`r�)rr�r�r��r�r rtr3r4rur��retrys rTr�z$HTTPDigestAuthHandler.http_error_401�s@������%�a�(���*�*�+=�+/��g�?����� ��rUN)r�r�r�r6r�r,r�r�rUrTr(r(�s���"�K��M�rUr(c��eZdZdZdZd�Zy)r)�Proxy-Authorizationr*c�f�|j}|jd|||�}|j�|Sr�)r�r�r�r,s rTr�z%ProxyDigestAuthHandler.http_error_407�s6���x�x���*�*�+?�+/��g�?����� ��rUN)r�r�r�r�r,r�r�rUrTr)r)�s��'�K��M�rUr)c�,�eZdZdd�Zd�Zd�Zd�Zd�Zy)�AbstractHTTPHandlerNc�j�|�||_ytjjj|_yrW)rr�HTTPConnection� debuglevel�_debuglevel)r�r5s rTr�zAbstractHTTPHandler.__init__�s&��)3�)?�:���T�[�[�E_�E_�Ej�Ej��rUc��||_yrW)r6)r��levels rT�set_http_debuglevelz'AbstractHTTPHandler.set_http_debuglevel�s �� ��rUc��tjjj|j|j��SrW)rrr4�_get_content_lengthrPr��r�r�s rTr;z'AbstractHTTPHandler._get_content_length�s2���{�{�)�)�=�=��L�L���� �"� "rUc��|j}|std��|j��|j}t|t�r d}t|��|j d�s|jdd�|j d�sR|j d�sA|j|�}|�|jdt |��n|jdd�|}|j�r&t|j�\}}t|�\}} |j d�s|jd|�|jjD]9\} }| j�} |j | �r�(|j| |��;|S) N� no host givenz\POST data should be bytes, an iterable of bytes, or a file object. It cannot be of type str.zContent-type�!application/x-www-form-urlencodedr��Transfer-encoding�chunkedr�)r�rrPrrr�r�r�r;r�rr�r r)r�r�)r�r�r�rPr4�content_length�sel_hostrT�sel�sel_pathrjr�s rT�do_request_zAbstractHTTPHandler.do_request_�sj���|�|����?�+�+��<�<�#��<�<�D��$��$�D����n�$��%�%�n�5��/�/�"�7�9��&�&�'7�8�#�.�.�/B�C�!%�!9�!9�'�!B��!�-��3�3�,�c�.�.A�C��3�3�/��<�������$�W�%5�%5�6�K�F�C�!+�C���H�h��!�!�&�)��+�+�F�H�=��;�;�1�1�K�D�%��?�?�$�D��%�%�d�+��/�/��e�<�2� �rUc ��|j}|std��||fd|ji|��}|j|j�t|j�}|j|jj�D��cic]\}}||vr||��c}}�d|d<|j�D� � cic]\} } | j�| ��}} } |jr0i}d}||vr||||<||=|j|j|�� |j|j�|j|j ||j#d���|j'�}|j*r!|j*j)�d |_|j-�|_|j0|_|Scc}}wcc} } w#t$$r} t| ��d } ~ wwxYw#|j)��xYw) z�Return an HTTPResponse object for the request, using http_class. http_class must implement the HTTPConnection API from http.client. r>rQr�� Connectionr/�rur@)�encode_chunkedN)r�rrQ�set_debuglevelr6rr��updaterur��titler�� set_tunnelr�r�r�rPr�r~�getresponser��sockr�rO�reasonr4)r�� http_classr �http_conn_argsr�r%rurGrHrjr��tunnel_headers�proxy_auth_hdr�errrvs rTr�zAbstractHTTPHandler.do_opens��� �x�x����?�+�+� �t�C�S�[�[�C�N�C�� ����)�)�*��s�,�,�-��������):�):�)<�-�)<���A��G�+��1��)<�-� .�!(����6=�m�m�o�F�o���s�4�:�:�<��$�o��F�����N�2�N���(�18��1H��~�.��N�+� �L�L��)�)�>�L�B� � $�� � �#�.�.�*�C�L�L�#�(�(�G�),���8K�)L��N�� � ��A� �6�6� �F�F�L�L�N��A�F�� � �"�����������e-��G�� � $��s�m�#�� $�� � �G�G�I��s7�G �8G�AG�G(� G%�G � G%�%G(�(G;rW)r�r�r�r�r9r;rFr�r�rUrTr2r2�s��k�!�"� $�L@rUr2c�*�eZdZd�Zej Zy)r*c�V�|jtjj|�SrW)r�rrr4�r�r s rT� http_openzHTTPHandler.http_open\s���|�|�D�K�K�6�6��<�<rUN)r�r�r�rZr2rFr�r�rUrTr*r*Zs��=�'�2�2�LrUr*rc�2�eZdZdd�Zd�ZejZy)rLNc�*�|�|n#tjjj}tj||�|�Ctjjj}tjj|�}|�||_||_ yrW) rrrr5r2r�� _http_vsn�_create_https_context�check_hostname�_context)r�r5r?r_�http_versions rTr�zHTTPSHandler.__init__esq��'1�'=��4�;�;�C^�C^�Ci�Ci�J��(�(��z�:���#�{�{�:�:�D�D���+�+�;�;�L�I���)�)7��&�#�D�MrUc�n�|jtjj||j��S)NrA)r�rrrr`rYs rT� https_openzHTTPSHandler.https_openos-���<�<���� ;� ;�S�(,� � � �7� 7rU�NNN)r�r�r�r�rcr2rFr�r�rUrTrLrLcs�� $� 7�,�7�7� rUrLc�(�eZdZdd�Zd�Zd�ZeZeZy)rNc�R�ddl}|�|jj�}||_yr�)�http.cookiejar� cookiejar� CookieJar)r�rhrs rTr�zHTTPCookieProcessor.__init__xs$��������0�0�2�I�"��rUc�<�|jj|�|SrW)rh�add_cookie_headerr<s rTr�z HTTPCookieProcessor.http_request~s�����(�(��1��rUc�>�|jj||�|SrW)rh�extract_cookies)r�r�r�s rTr5z!HTTPCookieProcessor.http_response�s�����&�&�x��9��rUrW)r�r�r�r�r�r5r�r7r�rUrTrrws��#���!�M�"�NrUrc��eZdZd�Zy)r/c�6�|j}td|z��)Nzunknown url type: %s)r�r)r�r r�s rTrzUnknownHandler.unknown_open�s���x�x���-��4�5�5rUN)r�r�r�rr�rUrTr/r/�s��6rUr/c�v�i}|D]1}|jdd�\}}|ddk(r |ddk(r|dd}|||<�3|S)z>Parse list of key=value strings where keys are not duplicated.�=r`rr�r])r�)�l�parsed�eltrGrHs rTr�r��sU�� �F����y�y��a� ���1��Q�4�3�;�1�R�5�C�<��!�B��A���q� � � �MrUc��g}d}dx}}|D]H}|r||z }d}� |r|dk(rd}�|dk(rd}||z }�$|dk(r|j|�d}�=|dk(rd}||z }�J|r|j|�|D�cgc]}|j���c}Scc}w)apParse lists as described by RFC 2068 Section 2. In particular, parse comma-separated lists where the elements of the list may include quoted-strings. A quoted-string could contain a comma. A non-quoted string could have quotes in the middle. Neither commas nor quotes count if they are escaped. Only double-quotes count, not single-quotes. r�F�\Tr�r)rlr�)r�res�part�escaper �curs rTr�r��s��� �C� �D���F�U�����C�K�D��F����d�{����������C�K�D���#�:��J�J�t���D���#�:��E�����-�2�� � �4��%(�)�S�T�D�J�J�L�S�)�)��)s�-Bc�"�eZdZd�ZdZd�Zd�Zy)r+c���|j}|dddk(rK|dddk7rC|jr7|jdk7r(|j|j�vrtd��y|j |�S)Nr;rdr�rO� localhost�-file:// scheme is supported only on localhost)r�r�� get_namesr�open_local_file)r�r rOs rT� file_openzFileHandler.file_open�sm���l�l���r��7�d�?�s�1�Q�x�3��C�H�H����K�'��8�8�t�~�~�/�/��N�O�O�0��'�'��,�,rUNc��tj�f ttjd�dtjtj ��dz�t_tjStjS#tj$r1tjd�ft_YtjSwxYw)Nr}r;)r+�namesr�r�gethostbyname_ex�gethostname�gaierror� gethostbynamer�s rTrzFileHandler.get_names�s������$� I�$)��+�+�K�8��;��+�+�F�,>�,>�,@�A�!�D�E�%F��!� � � � �{� � � ���?�?� I�%+�%9�%9�+�%F�$H��!�� � � � I�s�AB�2C� Cc�D�ddl}ddl}|j}|j}t |�} tj|�}|j}|jj|jd��} |j|�d} |jd| xsd|| fz�}|rt|�\}}|rsBt|�|j�vr'|r d|z|z} nd|z} t!t#|d�|| �St'd��#t$$r}t'|��d}~wwxYw) NrT��usegmtz6Content-type: %s Content-length: %d Last-modified: %s � text/plain�file://�rbzfile not on local host)�email.utils� mimetypesr�r�r5re�stat�st_size�utils� formatdate�st_mtime� guess_type�message_from_stringr�_safe_gethostbynamerrrNr~r)r�r �emailr�r�rq� localfile�statsry�modified�mtyperur��origurl�exps rTr�zFileHandler.open_local_file�s$�����x�x���<�<�� ��*� � ��G�G�I�&�E��=�=�D��{�{�-�-�e�n�n�T�-�J�H��(�(��2�1�5�E�/�e�/�/�K��&�,��h�7�8�9�G��'��-� ��d���1�$�7�4�>�>�;K�K��'�$�.��9�G�'�(�2�G�!�$�y�$�"7��'�J�J��/�0�0��� ��3�-��� �s�C D� D�D�D)r�r�r�r�r�rr�r�rUrTr+r+�s��-� �E�!�1rUr+c�`� tj|�S#tj$rYywxYwrW)rr�r�)r�s rTr�r��s.����#�#�D�)�)���?�?����s��-�-c��eZdZd�Zd�Zy)r,c�$�ddl}ddl}|j}|std��t |�\}}|� |j }nt |�}t|�\}}|rt|�\}}nd}t|�}|xsd}|xsd} tj|�}t|j�\} } | jd�}t!t#t|��}|dd|d}}|r |ds|dd} |j%||||||j&�} |xrdxsd}| D]9}t)|�\}}|j+�d k(s�%|d vs�*|j-�}�;| j/||�\}}d}|j1|j2�d}|r|d|zz }|� |dk\r|d|zz }t5j6|�}t9|||j2�S#t$r}t|��d}~wwxYw#|j:$r}t|�|�d}~wwxYw) Nr�ftp error: no host givenr�rOr]r`r��Dr���a�Ar�r�r'r�zContent-type: %s zContent-length: %d )�ftplibr�r�rr�FTP_PORTrmrrrrr�r~rr�r�r��map�connect_ftprQrr��upper�retrfiler�r�r�r�r� all_errors)r�r r�r�r�r�rmr�r4rf�attrs�dirsrY�fwr��attrr�rt�retrlenrur�r�s rT�ftp_openzFTPHandler.ftp_open�s$�����x�x����5�6�6���%� ��d��<��?�?�D��t�9�D� ��%� ��d��'��-�L�D�&��F��t�}���z�r����2�� ��'�'��-�D�!����.���e��z�z�#����C���&�'���#�2�Y��R��d����Q�����8�D� )��!�!�$���d�D�#�+�+�N�B��<�C�&�3�D���)�$�/���e��:�:�<�6�)��:�:� �;�;�=�D� � �+�+�d�D�1�K�B���G��(�(����6�q�9�E���/�%�7�7���"�w�!�|��1�G�;�;���/�/��8�G��b�'�3�<�<�8�8��1� ��3�-��� ��2� � � )��3�-�S�(�� )�s>�G�1AG/�G/�BG/� G,�G'�'G,�/H�>H � Hc �&�t||||||d��S)NF)� persistent)� ftpwrapper)r�rmr�r�r�r�rQs rTr�zFTPHandler.connect_ftp1s���$���d�D�'�%*�,� ,rUN)r�r�r�r�r�r�rUrTr,r,�s ��2)�h,rUr,c�0�eZdZd�Zd�Zd�Zd�Zd�Zd�Zy)r-c�J�i|_i|_d|_d|_d|_y)Nr�<r)�cacherQ�soonest�delay� max_connsr�s rTr�zCacheFTPHandler.__init__8s%���� ��������� ���rUc��||_yrW)r�)r��ts rT� setTimeoutzCacheFTPHandler.setTimeout?s ���� rUc��||_yrW)r�)r�rEs rT�setMaxConnszCacheFTPHandler.setMaxConnsBs ����rUc�|�|||dj|�|f}||jvr/tj�|jz|j|<nKt||||||�|j|<tj�|jz|j|<|j �|j|S)NrO)�joinr�rr�rQr��check_cache)r�rmr�r�r�r�rQr�s rTr�zCacheFTPHandler.connect_ftpEs����D�$�������7���$�*�*�� $� � ��d�j�j� 8�D�L�L���(��v�t�T�)-�w�8�D�J�J�s�O� $� � ��d�j�j� 8�D�L�L��������z�z�#��rUc���tj�}|j|krht|jj ��D]B\}}||ks�|j |j �|j |=|j|=�Dtt|jj���|_t|j �|jk(r�t|jj ��D]0\}}||jk(s�|j |=|j|=ntt|jj���|_yyrW)rr�r�rQr�r�r��min�valuesror�)r�r�rGrHs rTr�zCacheFTPHandler.check_cachePs ���I�I�K���<�<�1���T�\�\�/�/�1�2���1��q�5��J�J�q�M�'�'�)�� � �1� ����Q�� 3� �4���� 3� 3� 5�6�7����t�z�z�?�d�n�n�,��T�\�\�/�/�1�2���1�����$�� � �1� ����Q��� 3� �t�D�L�L�$7�$7�$9�:�;�D�L� -rUc���|jj�D]}|j��|jj�|jj�yrW)r�r�r��clearrQ)r��conns rT�clear_cachezCacheFTPHandler.clear_cachedsB���J�J�%�%�'�D��J�J�L�(�� � ���������rUN) r�r�r�r�r�r�r�r�r�r�rUrTr-r-5s ����� �<�(rUr-c��eZdZd�Zy)r.c�h�|j}|jdd�\}}|jdd�\}}t|�}|jd�rt j |�}|dd}|sd}t jd|t|�fz�}ttj|�||�S)Nr{r`rz;base64i�����text/plain;charset=US-ASCIIz$Content-type: %s Content-length: %d )r�r�r�endswithr��decodebytesr�r�ror�io�BytesIO)r�r rOrTrP� mediatyperus rT� data_openzDataHandler.data_openks����l�l���y�y��Q�'�����*�*�S��+�� �4� ��%�����i�(��%�%�d�+�D�!�#�2��I��5�I��+�+�,T� ��D� �"�-#�$���"�*�*�T�*�G�S�9�9rUN)r�r�r�r�r�rUrTr.r.js��:rUr.r<�nt)r5r4c��|dddk(r|dd}n |dddk(r|dd}tj�}tj�}t|||��S) zOS-specific conversion from a relative URL of the 'file' scheme to a file system path; not recommended for general use.Nr�z///r;�z//localhost/��rP�errors)r�getfilesystemencoding�getfilesystemencodeerrorsr��pathnamerPr�s rTr5r5�sc���B�Q�<�5� � ���|�H� �c�r�]�n� ,����}�H��,�,�.���.�.�0���x�(�6�B�BrUc��|dddk(rd|z}tj�}tj�}t|||��S)zOS-specific conversion from a file system path to a relative URL of the 'file' scheme; not recommended for general use.Nr;rdr�)rr�r�r r�s rTr4r4�sH���B�Q�<�4���h��H��,�,�.���.�.�0���X���@�@rUc��eZdZdZdZdezZdd�Zd�Zd�Z d�Z d�Zdd �Zdd �Z dd�Zdd�Zd �Zdd�Zdd�Zd�Zerd�Zdd�Zd�Zd�Zd�Zdd�Zy)r9a,Class to open URLs. This is a class rather than just a subroutine because we may need more than one set of global protocol-specific options. Note -- this is a base class for those who don't want the automatic handling of errors type 302 (relocated) and 401 (authorization needed).Nr�c��dd|jjiz}tj|td��|� t�}t |d�sJd��||_|jd�|_ |jd�|_ d |jfd g|_g|_ tj|_d|_t$|_y)NzW%(class)s style of invoking requests is deprecated. Use newer urlopen functions/methods�classr�)� stacklevelrsrt�key_file� cert_filez User-Agent)�Acceptz*/*)r�r�rBrCrDr6r�rxr�r�r��versionr��_URLopener__tempfilesrer}�_URLopener__unlink� tempcache�ftpcache)r�rx�x509r4s rTr�zURLopener.__init__�s���4�7>����@W�@W�6X�Y��� � �c�-�!�<��?� �l�G��w��'�D�)D�D�'��������,�� ����+�.���(�$�,�,�7�9J�K������� � �� ����!�� rUc�$�|j�yrW)r�r�s rT�__del__zURLopener.__del__�s��� � �rUc�$�|j�yrW)�cleanupr�s rTr�zURLopener.close�s�����rUc���|jr2|jD]} |j|��|jdd�=|jr|jj �yy#t$rY�YwxYwrW)r�r�r~r�r�)r�rYs rTr�zURLopener.cleanup�sn������(�(����M�M�$�'�)� � � ��#��>�>��N�N� � �"�������s�A'�' A3�2A3c�:�|jj|�y)zdAdd a header to be used by the HTTP interface only e.g. u.addheader('Accept', 'sound/basic')N)r�rl)r�r�s rT� addheaderzURLopener.addheader�s�� �����t�$rUc���tt|��}t|d��}|jr9||jvr+|j|\}}t |d�}t|||�St |�\}}|sd}||jvr0|j|}t |�\}} t| �\} }| |f}nd}d|z}||_ |jdd�}t||�r|d k(r'|r|j|||�S|j||�S |�t||�|�St||�||�S#tt f$r�t"$r} t#d | �| �d} ~ wwxYw)z6Use URLopener().open(file) instead of open(file, 'r').z%/:=&?~#+!$,;'@()*[]|�rQr�rYN�open_�-r�r�zsocket error)r rr r�rNrrrxr r�rCr��open_unknown_proxy�open_unknownr�rrr~)r�rrPrqrurt�urltyperOrf� proxyhostr�r�rjr4s rTrNzURLopener.open�sz����7�+�,����&=�>���>�>�g����7� $���w� 7��H�g��h��%�B��b�'�7�3�3�!�'�*������G��d�l�l�"��L�L��)�E�!+�E�!2��G�Y�'� �2�N�D�(���/�C��E��� ���� ��|�|�C��%���t�T�"�d�.?�&?���.�.�u�g�t�D�D��(�(��$�7�7� 8��|�*�w�t�T�*�3�/�/�*�w�t�T�*�3��5�5���8�$� ��� 8��.�#�.�C�7�� 8�s�D7�$D7�7E!� E�E!c�8�t|�\}}tdd|��)�/Overridable interface to open unknown URL type.� url errorzunknown url type�rr~)r�rrPr�rOs rTr�zURLopener.open_unknowns ���w�'� ��c��k�#5�t�<�<rUc�>�t|�\}}tdd|z|��)r�r�zinvalid proxy for %sr)r�rfrrPr�rOs rTr�zURLopener.open_unknown_proxys%���w�'� ��c��k�#9�D�#@�%�H�HrUc�,�tt|��}|jr||jvr|j|St|�\}}|�R|r|dk(rK |j |�}|j�}|j �tt|�d�|fS|j||�} |j�} |r t|d�} n�t|�\}}t|xsd�\}}t|xsd�\}}t|xsd�\}}tjj|�d} t!j"| �\}}|j$j'|�tj(|d�} || f}|j�||j|<d}d}d}d}d| vrt+| d �}|r ||||�|j-|�x}rD|t/|�z }| j1|�|dz }|r ||||�|j-|�x}r�D| j � |j �|dk\r||krt3d ||fz|��|S#t$rY���wxYw#| j �wxYw#|j �wxYw)ztretrieve(url) returns (filename, headers) for a local object or (tempfilename, headers) for a remote object.rYr`rZr�r\r]rr^r_ra)r rr�rr�rdr�r5r r~rNrrrerf�splitextrh�mkstempr�rl�fdopenrmrnrorpr)r�rOrqrrrPr��url1rtr�rurv�garbagerf�suffix�fdrwrxryrnrzr{s rT�retrievezURLopener.retrieves����Y�s�^�$���>�>�c�T�^�^�3��>�>�#�&�&���_� ��d���T�T�V�^� ��)�)�$�/���w�w�y����� �#�J�t�$4�Q�$7�8�$�>�>��Y�Y�s�D� !��" ��g�g�i�G���8�T�*�� *�3�� ��� *�4�:�2� 6� ��� +�D�J�B� 7� ��g� *�4�:�2� 6� ��g����)�)�$�/��2��!)�!1�!1�&�!9���X�� � �'�'��1��i�i��D�)�� �!�7�*���>�>�-�*0�D�N�N�3�'���������#�w�.��w�'7�8�9�D���x��T�2�!�w�w�r�{�*�e�*��C��J�&�D��I�I�e�$���M�H�!�"�8�R��6� "�w�w�r�{�*�e�*�� � ���H�H�J��1�9����&�C���,�� &�(� (�� ��[� �� ��F� � ����H�H�J�s9�A I�3CJ�BI,�J� I)�(I)�,I>�>J�Jc���d}d}t|t�r,t|�\}}|rt|�\}}t |�}|}nq|\}}t|�\}}t|�\} } | }d}| j �dk7rd}n6t| �\}} |rt|�\}}|r | �d|�| ��}t|�r|}|stdd��|r>t |�}tj|j��jd�}nd}|r>t |�}tj|j��jd�}nd}||�} i}|rd|z|d<|rd|z|d <|r||d <d|d<|jD] \}}|||<�|�d |d<| jd|||�n| jd||�� | j�}d|j(cxkrdkr(nn%t+||j,d|z|j(�S|j/||j0|j(|j2|j,|�S#t j"j$$rt'd��wxYw)a�Make an HTTP connection using connection_class. This is an internal method that should be called from open_http() or open_https(). Arguments: - connection_factory should take a host name and return an HTTPConnection instance. - url is the url to retrieval or a host, relative-path pair. - data is payload for a POST request or None. Nrz://z http errorr>r|zBasic %sr/r�r�r�rHr?zContent-Typer�r�rIz$http protocol error: bad status liner1r2�http:)rrr rrrr�rr~r�r�r�r�r�r�rOrr� BadStatusLiner�statusrr4� http_errorrtrQ)r��connection_factoryrOrP�user_passwd�proxy_passwdr�r��realhostr�r�� proxy_authr�� http_connrur�r�r�s rT�_open_generic_httpzURLopener._open_generic_http\s��������c�3��'��_�N�D�(��$.�t�$4�!��T��t�}���H� �N�D�(�!+�D�!1��L�$�&�x�0�M�G�T��C��K��}�}��&�(���!+�D�!1���$��,6�x�,@�)�K���.5�x��F�H���)�#�D��7�<��A�A��"�<�0�L��)�)�,�*=�*=�*?�@�G�G��P�J��J��!�+�.�K��#�#�K�$6�$6�$8�9�@�@��I�D��D�&�t�,� ����-7�*�-D�G�)�*��(2�T�(9�G�O�$��&�G�F�O� !(����!�_�_�M�F�E�#�G�F�O�-���&I�G�N�#����f�h��g�>����e�X�w��?� C� �,�,�.�H��(�/�/�'�C�'��h����g��m�&�o�o�/� /��?�?��X�[�[�������(�,�,��F� F���{�{�(�(� C��A�B�B� C�s�8I�)I,c�X�|jtjj||�S)zUse HTTP protocol.)rrrr4�r�rOrPs rT� open_httpzURLopener.open_http�s!���&�&�t�{�{�'A�'A�3��M�MrUc��d|z}t||�r,t||�}|� ||||||�} n |||||||�} | r| S|j|||||�S)z�Handle http errors. Derived class can override this, or provide specific handlers named http_error_DDD where DDD is the 3-digit error code.z http_error_%d)r�r�r) r�rOrt�errcode�errmsgrurPrjr�rws rTrzURLopener.http_error�so����(���4����T�4�(�F��|���R��&�'�B����R��&�'�4�H���f�}��&�&�s�B����I�IrUc�@�|j�t||||d��)z>Default error handler: close the connection and raise OSError.N)r�r�r�rOrtrrrus rTrzURLopener.http_error_default�s�� ��� ���W�f�g�t�<�<rUc�r�|js|jr}tjjj }tjj |�}|j|j|j�|j� d|_nd}tjj ||��S)NTrA) r�r�rrrr]r^�load_cert_chain�post_handshake_auth)r�r�rar?s rT�_https_connectionzURLopener._https_connection�s����}�}����#�{�{�:�:�D�D���+�+�;�;�L�I���'�'����� � �F��.�.�:�26�G�/����;�;�.�.�t�W�.�E�ErUc�<�|j|j||�S)zUse HTTPS protocol.)rr"rs rT� open_httpszURLopener.open_https�s���*�*�4�+A�+A�3��M�MrUc��t|t�std��|dddk(r)|dddk7r!|ddj�dk7rt d ��|j|�S) z/Use local file or FTP depending on form of URL.zEfile error: proxy support for file protocol currently not implementedNr;rdr�rOr�z localhost/r~)rrrr�rEr�r�s rT� open_filezURLopener.open_file�sb���#�s�#��b�c�c��r��7�d�?�s�1�Q�x�3��3�q��9�?�?�3D��3T��L�M�M��'�'��,�,rUc���ddl}ddl}t|�\}}t|�} t j |�}|j} |jj|jd��} |j|�d}|jd|xsd| | fz�}|s&|} |dddk(rd |z} t!t#|d �|| �St%|�\}}|sht'j(|�t+�ft-�zvr=|} |dddk(rd |z} n|dddk(rt/d |z��t!t#|d �|| �Std��#t$r%}t|j|j��d}~wwxYw)zUse local file.rNTr�z6Content-Type: %s Content-Length: %d Last-modified: %s r�r`rOr�r�r;z./zAlocal file url may start with / or file:. Unknown url of type: %sz#local file error: not on local host)r�r�r r5rer�r~r�strerrorrqr�r�r�r�r�r�rrNrrr�r}�thishostrE)r�rOr�r�r�rY� localnamer��eryr�r�ru�urlfiler�s rTr�zURLopener.open_local_file�s�������_� ��d� ��&� � 3��G�G�I�&�E��}�}���;�;�)�)�%�.�.��)�F���$�$�S�)�!�,��+�%�+�+�G� � "�l�D�(�3� 4�5����G��B�Q�x�3��#�d�*���d�9�d�3�W�g�F�F���%� ��d���#�#�D�)�y�{�n�x�z�.I�J��G��B�Q�x�3��#�d�*���b�q��T�!� �!d�gj�!j�k�k��d�9�d�3�W�g�F�F��<�=�=��-� 3��1�:�:�q�z�z�2�2�� 3�s�E� E4� E/�/E4c�8�t|t�std��ddl}t |�\}}|std��t|�\}}t |�\}}|rt|�\}}nd}t|�}t|xsd�}t|xsd�}tj|�}|sddl}|j}nt|�}t|�\}} t|�}|jd�} | dd| d}} | r | ds| dd} | r | dsd| d<|||dj!| �f}t#|j$�t&kDrLt)|j$�D]4} | |k7s� |j$| }|j$| =|j+��6 ||j$vrt-||||| �|j$|<|sd }nd }| D]9}t/|�\}}|j1�dk(s�%|dvs�*|j3�}�;|j$|j5||�\}}|j7d |z�d}d}|r|d|zz }|� |dk\r|d|zz }t9j:|�}t=||d |z�S#t?�$r}td|���|�d}~wwxYw)zUse FTP protocol.zCftp error: proxy support for ftp protocol currently not implementedrNr�r�rOr]r`r�r�r�r�zftp:zContent-Type: %s zContent-Length: %d �ftp error: ) rrrr�r rrrrrr�r�r�rmrr�r�ror��MAXFTPCACHEr�r�r�rr�r�r�r�r�r�r� ftperrors)r�rOr�r�rfr�rmr�r�r�r�rYr�rGrHr�r�r�rtr�r�rur�s rT�open_ftpzURLopener.open_ftps����#�s�#��`�a�a����_� ��d��8�$>�?�?���%� ��d���%� ��d���T� 2���v��f��t�}���t�z�r�"�����2�&���#�#�D�)�����?�?�D��t�9�D� ��&���e��t�}���z�z�#����#�2�Y��R��d����Q���Q�R�����Q��3��a���D�$������.���t�}�}���+��$�-�-�(����8�� � �a�(�A�� � �a�(��G�G�I� )� 9��$�-�-�'��t�V�T�4��>�� � �c�"�����$���)�$�/���e��:�:�<�6�)��:�:� �;�;�=�D� � !�M�M�#�.�7�7��d�C�M�R���(�(��#��6�q�9�E��G���/�%�7�7���"�w�!�|��1�G�;�;���/�/��8�G��b�'�6�C�<�8�8���{� 9��[���.�/�S�8�� 9�s&�AI8�(I8�-B I8�8 J�J�Jc �T�t|t�std�� |jdd�\}}|sd}|j d�}|dk\rd ||d vr||dzd }|d |}nd}g}|jdtjd tjtj���z�|jd|z�|dk(r4tj|jd��jd�}nt|�}|jdt!|�z�|jd�|j|�dj#|�}t%j&|�}t)j*|�}t-|||�S#t$r tdd��wxYw)zUse "data" URL.zEdata error: proxy support for data protocol currently not implementedrr`z data errorzbad data URLr��;rrqNr�zDate: %sz%a, %d %b %Y %H:%M:%S GMTzContent-type: %sr�r|zlatin-1zContent-Length: %d� )rrrr�rEr~�rfindrlr�strftime�gmtimer�r�r�r�rror�r�r�r��StringIOr) r�rOrPr��semirPr4ru�fs rT� open_datazURLopener.open_dataFs����#�s�#��b�c�c� 8��9�9�S�!�,�L�T�4��0�D��z�z�#����1�9��D���K�/��D��F�G�}�H����;�D��H���� � �:�d�m�m�,G�,0�K�K�� � ��,D�F�F� G�� � �%��,�-��x���%�%�d�k�k�'�&:�;�B�B�9�M�D��4�=�D�� � �'�#�d�)�3�4�� � �2��� � �4���i�i��n���+�+�C�0���K�K�����!�W�c�*�*��5� 8��,��7�7� 8�s�F�F'rWrd)r�r�r�r6r�r�r�r�r�r�r�r�rNr�r�r rrrrrFr"r$r&r�r1r;r�rUrTr9r9�s�����K� �;�.�G�!�4��#�%�"8�H=� I�:�|ZF�xN�J� =� � F� N�-�>�@89�t'+rUr9c��eZdZdZd�Zd�Zdd�Zd�Zdd�Zdd�Z dd �Z dd �Z dd�Z dd�Z dd �Zdd�Zdd�Zdd�Zdd�Zd�Zy)r:z?Derived class with handlers for errors we can handle (perhaps).c�`�tj|g|��i|��i|_d|_d|_y)Nrr<)r9r�� auth_cache�tries�maxtries)r�r��kwargss rTr�zFancyURLopener.__init__ss/�����4�1�$�1�&�1������ ��� rUc�$�t||d|z|�S)z3Default error handling -- don't raise an exception.r)rrs rTrz!FancyURLopener.http_error_defaultys���"�g�w��}�g�>�>rUNc�>�|xjdz c_ |jrQ|j|jk\r8t|d�r |j}n|j}|||dd|�d|_S|j||||||�}|d|_S#d|_wxYw)z%Error 302 -- relocated (temporarily).r`�http_error_500r'z)Internal Server Error: Redirect Recursionr)r?r@r�rDr�redirect_internal) r�rOrtrrrurPr�rws rTr^zFancyURLopener.http_error_302}s���� � �a�� � ��}�}����t�}�}�!<��4�!1�2��.�.�D��2�2�D��C��S�G�#�%��D�J� �+�+�C��W�f�,3�T�;�F���D�J���D�J�s�AB�4B� Bc��d|vr|d}nd|vr|d}ny|j�t|jdz|z|�}t|�}|jdvrt|||d|zz||��|j |�S)NrKrLr{rMz( Redirection to url '%s' is not allowed.)r�rr�rrTrrN) r�rOrtrrrurPrDr[s rTrEz FancyURLopener.redirect_internal�s����� ��Z�(�F� �g� ��U�^�F�� ��� �����S��3�.��7���F�#���?�?�">�>��F�G�"�F��O�P�#�R�)� )� �y�y�� � rUc�.�|j||||||�S)z*Error 301 -- also relocated (permanently).�r^�r�rOrtrrrurPs rTr_zFancyURLopener.http_error_301�����"�"�3��G�V�W�d�K�KrUc�.�|j||||||�S)z;Error 303 -- also relocated (essentially identical to 302).rHrIs rTr`zFancyURLopener.http_error_303�rJrUc�\�|�|j||||||�S|j|||||�S)z1Error 307 -- relocated, but turn POST into error.)r^rrIs rTrazFancyURLopener.http_error_307��;���<��&�&�s�B����$�O�O��*�*�3��G�V�W�M�MrUc�\�|�|j||||||�S|j|||||�S)z1Error 308 -- relocated, but turn POST into error.)r_rrIs rTrbzFancyURLopener.http_error_308�rMrUc���d|vrtj||||||�|d}tjd|�} | stj||||||�| j �\} }| j�dk7rtj||||||�|stj||||||�d|jzdz}|�t||�||�St||�|||�S)z_Error 401 -- authentication required. This function supports Basic authentication only.r��![ ]*([^ ]+)[ ]+realm="([^"]*)"r��retry_�_basic_auth�r9rr��matchr�r�r�r�� r�rOrtrrrurPr-�stuffrTrTr�rjs rTr�zFancyURLopener.http_error_401�s���W�,��(�(��s�B�)0�&�'� C��*�+�����?��G����(�(��s�B�)0�&�'� C����� ����<�<�>�W�$��(�(��s�B�)0�&�'� C���(�(��s�B���� ��$�)�)�#�m�3���<�%�7�4��%�c�5�1�1�%�7�4��%�c�5�$�7�7rUc���d|vrtj||||||�|d}tjd|�} | stj||||||�| j �\} }| j�dk7rtj||||||�|stj||||||�d|jzdz}|�t||�||�St||�|||�S)zeError 407 -- proxy authentication required. This function supports Basic authentication only.r�rPr��retry_proxy_rRrSrUs rTr�zFancyURLopener.http_error_407�s�� �w�.��(�(��s�B�)0�&�'� C��,�-�����?��G����(�(��s�B�)0�&�'� C����� ����<�<�>�W�$��(�(��s�B�)0�&�'� C���(�(��s�B���� ��� � �)�M�9���<�%�7�4��%�c�5�1�1�%�7�4��%�c�5�$�7�7rUc��t|�\}}d|z|z}|jd}t|�\}} t| �\} } | jd�dz}| |d} |j | ||�\}} |s| syt|d���dt| d���d| ��} d| z| z|jd<|�|j |�S|j ||�S)N�http://rrer`r�r�r{�r rxrr��get_user_passwdr rN�r�rOr�rPr�r�rDrfr�r�� proxyselectorr�rmr�s rT�retry_proxy_http_basic_authz*FancyURLopener.retry_proxy_http_basic_auth�s���#�C����h��T�!�H�,�����V�$��'��.����#-�i�#8� � �=��N�N�3��!�#���a�b�M� ��+�+�I�u�a�@���f����"'��2�"6�"'��R�"8�)�E� �(�9�4�}�D����V���<��9�9�V�$�$��9�9�V�T�*�*rUc��t|�\}}d|z|z}|jd}t|�\}} t| �\} } | jd�dz}| |d} |j | ||�\}} |s| syt|d���dt| d���d| ��} d| z| z|jd<|�|j |�S|j ||�S)N�https://r�rer`r�r�r{r[r]s rT�retry_proxy_https_basic_authz+FancyURLopener.retry_proxy_https_basic_auth s���#�C����h��d�"�X�-�����W�%��'��.����#-�i�#8� � �=��N�N�3��!�#���a�b�M� ��+�+�I�u�a�@���f����"'��2�"6�"'��R�"8�)�E� � *�Y� 6�� F����W���<��9�9�V�$�$��9�9�V�T�*�*rUc� �t|�\}}|jd�dz}||d}|j|||�\}}|s|syt|d���dt|d���d|��}d|z|z} |�|j | �S|j | |�S)Nrer`r�r�r{rZ�r r�r\r rN� r�rOr�rPr�r�r�rmr�rDs rTr�z$FancyURLopener.retry_http_basic_auth s���#�C����h��I�I�c�N�Q����A�B�x���+�+�D�%��;���f����"�4�b�1�"�6��3�T�;���T�!�H�,���<��9�9�V�$�$��9�9�V�T�*�*rUc� �t|�\}}|jd�dz}||d}|j|||�\}}|s|syt|d���dt|d���d|��}d|z|z} |�|j | �S|j | |�S)Nrer`r�r�r{rardres rT�retry_https_basic_authz%FancyURLopener.retry_https_basic_auth% s���#�C����h��I�I�c�N�Q����A�B�x���+�+�D�%��;���f����"�4�b�1�"�6��3�T�;���d�"�X�-���<��9�9�V�$�$��9�9�V�T�*�*rUc���|dz|j�z}||jvr|r|j|=n|j|S|j||�\}}|s|r||f|j|<||fS)Nre)r�r>�prompt_user_passwd)r�r�r�r�r�rmr�s rTr\zFancyURLopener.get_user_passwd3 sv���c�k�D�J�J�L�(���$�/�/�!���O�O�C�(����s�+�+��.�.�t�U�;���f��6�4��.�4�?�?�3�/��V�|�rUc ��ddl} td|�d|�d��}|jd|�d|�d|�d��}||fS#t$r t�YywxYw) z#Override this in a GUI environment!rNzEnter username for z at z: zEnter password for z in r�)�getpass�input�KeyboardInterrupt�print)r�r�r�rkrmr�s rTriz!FancyURLopener.prompt_user_passwd> sT��� ��E�4�H�I�D��_�_��u�d�&$�%�F���<��� � ��G�� �s�07�A �A rW)NF)r)r�r�r�r6r�rr^rEr_r`rarbr�r�r_rbr�rgr\rir�rUrTr:r:psm��I��?��$!�8L�L�N�N�FJ��8�2FJ��8�2+�$+�$+�+� � rUr:c�D�t�tjd�atS)z8Return the IP address of the magic hostname 'localhost'.r})� _localhostrr�r�rUrTr}r}N s �����)�)�+�6� ��rUc��t�: ttjtj��d�atStS#tj $r)ttjd�d�aYtSwxYw)z,Return the IP addresses of the current host.r;r})� _thishostr�rr�r�r�r�rUrTr)r)V sw���� G��f�5�5�f�6H�6H�6J�K�A�N�O�I���9������ G��f�5�5�k�B�1�E�F�I��� G�s�3A�4B�Bc�:�t�ddl}|jatS)z1Return the set of errors raised by the FTP class.Nr)� _ftperrorsr�r�)r�s rTr0r0a s������&�&� ��rUc�D�t�tjd�atS)z%Return an empty email Message object.r�)� _noheadersr�r�r�rUrT� noheadersrwj s �����.�.�r�2� ��rUc�@�eZdZdZ d d�Zd�Zd�Zd�Zd�Zd�Z d �Z y)r�z;Class used by open_ftp() for cache of open FTP connections.Nc���||_||_||_||_||_||_d|_||_ |j�y#|j��xYwr�) rmr�r�r�r�rQ�refcount� keepalive�initr�)r�rmr�r�r�r�rQr�s rTr�zftpwrapper.__init__w s[���� ������ ��� ��� ������ �#��� ��I�I�K�� ��J�J�L��s�A�Ac��ddl}d|_|j�|_|jj |j |j|j�|jj|j|j�dj|j�}|jj|�y)NrrO)r��busy�FTPrN�connectr�r�rQ�loginrmr�r�r��cwd)r�r��_targets rTr|zftpwrapper.init� sw����� ��:�:�<�����������D�I�I�t�|�|�<������t�y�y�$�+�+�.��(�(�4�9�9�%�������W�rUc�8�ddl}|j�|dvrd}d}nd|z}d} |jj|�d}|r&|s$ d|z}|jj |�\}}|s�|jjd�|rY|jj�} |jj|� |jj| �d|z}nd}|jj |�\}}d|_t|jd �|j�} |xj dz c_|j#�| fS#|j$r/|j�|jj|�Y��TwxYw#|j$r+}t|�dddk7rtd |���|�Yd}~��cd}~wwxYw#|j$r}td |z�|�d}~wwxYw#|jj| �wxYw)Nr)r'r�zTYPE Ar`zTYPE zRETR r��550r.z ftp error: %rzLIST �LISTr�)r��endtransferrN�voidcmdr�r|�ntransfercmd� error_permrr�pwdr�r~r�makefile� file_closerzr�)r�rYr�r��cmd�isdirr�r�rQr��ftpobjs rTr�zftpwrapper.retrfile� s���������:��X�s�q�u��d�N�c�A�E� "��H�H���S�!����� G���n�� $��� 5� 5�c� :� ��g���H�H���X�&���h�h�l�l�n��&�M������T�*��H�H�L�L��%���n���� �H�H�1�1�#�6�M�D�'��� ��d�m�m�D�1�4�?�?�C��� � ��� �� � ���� � ��G� � � "��I�I�K��H�H���S�!� "���$�$� G��v�;�r��?�e�+�"�[���#9�:��F�,�� G��"�,�,�M�&���'?�@�f�L��M���H�H�L�L��%�sM�E�#F�&G�:F�F�G�( G�G�G9�%G4�4G9�9G<�<Hc��|jsyd|_ |jj�y#t�$rYywxYwr�)r~rN�voidrespr0r�s rTr�zftpwrapper.endtransfer� s<���y�y���� � ��H�H������{� �� �s�1� A�Ac�R�d|_|jdkr|j�yy)NFr)r{rz� real_closer�s rTr�zftpwrapper.close� s$������=�=�A���O�O��rUc��|j�|xjdzc_|jdkr|js|j�yyy)Nr`r)r�rzr{r�r�s rTr�zftpwrapper.file_close� s@������� � ��� ��=�=�A��d�n�n��O�O��'5�rUc��|j� |jj�y#t�$rYywxYwrW)r�rNr�r0r�s rTr�zftpwrapper.real_close� s5������ ��H�H�N�N����{� �� �s�-� =�=)NT)r�r�r�r6r�r|r�r�r�r�r�r�rUrTr�r�t s/��E�?C� �� �*!�X�� �rUr�c���i}g}tjj�D]s}t|�dkDs�|ddk(s�|ddj �dk(s�2tj|}|ddj �}|j|||f�|s�o|||<�udtjvr|j dd�|D])\}}}|ddd k(s�|r|||<�|j |d��+|S) aReturn a dictionary of scheme -> proxy server URL mappings. Scan the environment for variables named <scheme>_proxy; this seems to be the standard convention. If you need a different way, you can pass a proxies dictionary to the [Fancy]URLopener constructor. r�i����r����Nrf�REQUEST_METHODr�_proxy)re�environrsror�rlr�)rx�environmentrjr�� proxy_names rT�getproxies_environmentr�� s����G��K�� � ���!���t�9�q�=�T�"�X��_��b�c����1B�g�1M��J�J�t�$�E��c�r����*�J�����e�Z�8�9��&+�� �#�"��2�:�:�%����F�D�!�#.���e�Z����9�� ��&+�� �#����J��-� $/��NrUc��|� t�} |d}|dk(ry|j�}t|�\}}|j d�D]k}|j�}|s�|j d�}|j�}||k(s||k(ryd|z}|j|�s|j|�s�kyy#t$rYywxYw)z�Test if proxies should not be used for a particular host. Checks the proxy dict for the value of no_proxy, which should be a list of comma separated DNS suffixes, or '*' for all hosts. �noF�*Tr�.)r�rr�rr�r��lstripr�)r�rx�no_proxy�hostonlyr�rjs rT�proxy_bypass_environmentr�� s�����(�*����4�=���3����:�:�<�D���%�N�H�d����s�#���z�z�|����;�;�s�#�D��:�:�<�D��4��4�4�<����:�D�� � ��&�$�-�-��*=��$���)����s�B7�7 C�Cc��ddlm}ddlm}m}t |�\}}d�}d|vr|dryd} t||��}|j d d �D]�} | s�tjd| �} | �|�}|| jd��}| jd �}|�'d| jd�jd�dzz}nt|dd�}|dks|dkDr��d|z }||z ||z k(s��y||| �s��yy#|$rY��wxYw)aj Return True iff this host shouldn't be accessed using a proxy This function uses the MacOSX framework SystemConfiguration to fetch the proxy information. proxy_settings come from _scproxy._get_proxy_settings or get mocked ie: { 'exclude_simple': bool, 'exceptions': ['foo.bar', '*.bar.com', '127.0.0.1', '10.1', '10.0/16'] } r��fnmatch)�AddressValueError�IPv4Addressc���|jd�}ttt|��}t |�dk7r |gd�zdd}|ddz|ddzz|dd zz|d zS)Nr�r;)rrrrr�r`rr;r�r�)r�r�r�rmro)�ipAddrr�s rT�ip2numz,_proxy_bypass_macosx_sysconf.<locals>.ip2num3 sm�����S�!���S��e�_�%���u�:��?��\�)�2�A�.�E��a��B��5��8�r�>�2�e�A�h�!�m�D�u�Q�x�O�OrUr��exclude_simpleTN� exceptionsr�z(\d+(?:\.\d+)*)(/\d+)?r`r;r�� F)r�� ipaddressr�r�rrmr�r�rT�group�count) r��proxy_settingsr�r�r�r�r�r��hostIPr�rEr��masks rT�_proxy_bypass_macosx_sysconfr�" s,�� �8���%�N�H�d�P��$���*�+�� �F� ��[��*�+�� �#�#�L�"�5���h��H�H�.��6���=�V�/��!�'�'�!�*�%�D��7�7�1�:�D��|��A�G�G�A�J�,�,�S�1�A�5�6���4���8�}���a�x�4�"�9����9�D��$��D�D�L�1�� �T�5� !��/6�2��9� �� �s�C;�;D�Dc��ddlm}t|�\}}|jd�}|D])}|j�}|dk(rd|vs�y|||�s�)yy)a Return True if the host should bypass the proxy server. The proxy override list is obtained from the Windows Internet settings proxy override registry value. An example of a proxy override value is: "www.example.com;*.example.net; 192.168.0.1" rr�r3z<local>r�TF)r�rr�r�)r��overrider�r��proxy_overrider�s rT�_proxy_bypass_winreg_overrider�b s\�� ����G�D�!��^�^�C�(�N����z�z�|���9���$��� �T�4� ���rU�darwin)�_get_proxy_settings�_get_proxiesc�.�t�}t||�SrW)r�r�)r�r�s rT�proxy_bypass_macosx_sysconfr�} s��,�.��+�D�.�A�ArUc��t�S)z�Return a dictionary of scheme -> proxy server URL mappings. This function uses the MacOSX framework SystemConfiguration to fetch the proxy information. )r�r�rUrT�getproxies_macosx_sysconfr�� s���~�rUc�H�t�}|rt||�St|�S)z�Return True, if host should be bypassed. Checks proxy settings gathered from the environment, if specified, or from the MacOSX framework SystemConfiguration. )r�r�r��r�rxs rTrr� s'��)�*���+�D�'�:�:�.�t�4�4rUc�.�t�xs t�SrW)r�r�r�rUrTr6r6� s��%�'�F�+D�+F�FrUc���i} ddl} |j|jd�}|j |d�d}|r�t|j |d�d�}d|vrd|vrdj |�}|jd�D]F}|jdd �\}}tjd |�s|dvrd|z}n |d k(rd|z}|||<�H|jd �rJtjdd|d �}|jd�xs||d<|jd�xs||d<|j�|S#t$r|cYSwxYw#tttf$rY|SwxYw)zxReturn a dictionary of scheme -> proxy server URL mappings. Win32 uses the registry to store proxies. rN�;Software\Microsoft\Windows\CurrentVersion\Internet Settings�ProxyEnable�ProxyServerrqr3zhttp={0};https={0};ftp={0}r`z (?:[^/:]+)://)rr�rNrZ�sockszsocks://z ^socks://z socks4://rr�)�winreg�ImportError�OpenKey�HKEY_CURRENT_USER�QueryValueExrr�r�r�rTr�r��Closer~rEr�)rxr��internetSettings�proxyEnable�proxyServer�pr��addresss rT�getproxies_registryr�� s����� ��" �%�~�~�f�.F�.F�N� P�� �-�-�.>�/<�>�>?�A�K��!�&�"5�"5�6F�7D�#F�FG�#I�J���k�)�c��.D�">�"E�"E�k�"R�K�$�*�*�3�/�A�()����Q��%�H�g��8�8�O�W�=�#�'?�?�&/�'�&9�G�%��0�&0�7�&:�G�(/�G�H�%�0��;�;�w�'� �f�f�\�;���@P�Q�G�&-�k�k�&�&9�&D�W�G�F�O�'.�{�{�7�';�'F�w�G�G�$��"�"�$����M� ��N� ��B��Y�/� � ��� �s#�D:�D/E�:E�E�E#�"E#c�.�t�xs t�S)z�Return a dictionary of scheme -> proxy server URL mappings. Returns settings gathered from the environment, if specified, or the registry. )r�r�r�rUrTr6r6� s��&�'�@�+>�+@�@rUc� � ddl} |j|jd�}|j |d�d}t|j |d�d�}|r|syt||�S#t$rYywxYw#t$rYywxYw)NrFr�r�� ProxyOverride)r�r�r�r�r�rr~r�)r�r�r�r�� proxyOverrides rT�proxy_bypass_registryr�� s��� �� �%�~�~�f�.F�.F�N� P�� �-�-�.>�/<�>�>?�A�K��� 3� 3�4D�5D�!F�FG�!I�J�M� �-��,�T�=�A�A��� �� ��� �� �s#�A'�AA6�' A3�2A3�6 B�Bc�H�t�}|rt||�St|�S)z�Return True, if host should be bypassed. Checks proxy settings gathered from the environment, if specified, or the registry. )r�r�r�r�s rTrr� s'��)�*���+�D�'�:�:�(��.�.rUrdrW)~r6r�r�r�r�http.clientrr�rer�rrVrrrhrbrB�urllib.errorrrr�urllib.parserrrr r rrr rrrrrrrrrr�urllib.responserrrGrFr��__all__�version_infor�rMrr1r2rkr7r8r��ASCIIr�r�rrr3rr0rrror r!r"r#r$r%r&�urandomrr'r(r)r2r*r�rrLrlrr/r�r�r+r�r,r-r.r/rj� nturl2pathr5r4r�r9r:rpr}rrr)rtr0rvrwr�r�r�r�r��platform�_scproxyr�r�r�r�rr6r�r�r�rUrT�<module>r�s���C�f� ���� � � � � � �����C�B�"�"�"�"�"� 5����I���$��(�(��!�,�,�� ���F�$B�$B�M+��4�5�$�M+�^���:�x��r�z�z�(�B�H�H�-��� k"�k"�ZI+�I+�^"�H8�8�&#��#�";�k�;�n2�+�n2�b,�B)>�;�)>�V=*�=*�@G�o�G�3�#B�3�>k#�k#�^�3�[���4�k�� �z�z��O�O�d�K�)B��$ �[�*C� �s�+�s�l3�%�3��4�;�;�)�*�8�*�8�$�N�N�>�"�#�+�#�$6�[�6� �)*�V11�+�11�f�7,��7,�r3�j�3�j:�+�:�B���7�7�d�?�5�5�C� A���+�+�DX�Y�X�z� �� � ��� ��� ��a�a�H#�J �J<�@�0�<�<�8��:�B��5�G��W�W��_�/�bA�B�(/�(�J�+�L��WT���I��s�:K�K�KPKS�\y��@@*__pycache__/response.cpython-312.opt-1.pycnu�[���� T��h9 ���dZddlZgd�ZGd�dej�ZGd�de�ZGd�d e�ZGd �de�Zy)aResponse classes used by urllib. The base class, addbase, defines a minimal file-like interface, including read() and readline(). The typical response object is an addinfourl instance, which defines an info() method that returns headers and a geturl() method that returns the url. �N)�addbase�addclosehook�addinfo� addinfourlc�4��eZdZdZ�fd�Zd�Zd�Zd�Z�xZS)rzOBase class for addinfo and addclosehook. Is a good idea for garbage collection.c�@��tt|�|dd��||_y)Nz<urllib response>F)�delete)�superr�__init__�fp)�selfr� __class__s ��(/usr/lib64/python3.12/urllib/response.pyrzaddbase.__init__s!��� �g��&�r�+>�u�&�M����c�h�d|jj�dt|��d|j�d�S)N�<z at z whose fp = �>)r�__name__�id�file�r s r�__repr__zaddbase.__repr__s(��-1�^�^�-D�-D�-/��X�t�y�y�B� Brc�H�|jjrtd��|S)NzI/O operation on closed file)r�closed� ValueErrorrs r� __enter__zaddbase.__enter__s���7�7�>�>��;�<�<��rc�$�|j�y�N)�close)r �type�value� tracebacks r�__exit__zaddbase.__exit__!s��� � �r) r� __module__�__qualname__�__doc__rrrr#� __classcell__�rs@rrrs���Y�� B�� rrc�,��eZdZdZ�fd�Z�fd�Z�xZS)rz*Class to add a close hook to an open file.c�H��tt|�|�||_||_yr)r rr� closehook�hookargs)r rr+r,rs �rrzaddclosehook.__init__(s ��� �l�D�*�2�.�"��� �� rc��� |j}|j}|rd|_d|_||�tt|��y#tt|��wxYwr)r+r,r rr)r r+r,rs �rrzaddclosehook.close-sN��� .����I��}�}�H��!%��� $�� ��8�$��,��+�-��E�,��+�-�s�-A�A)rr$r%r&rrr'r(s@rrr%s���4�!� .� .rrc�(��eZdZdZ�fd�Zd�Z�xZS)rz.class to add an info() method to an open file.c�:��tt|�|�||_yr)r rr�headers)r rr0rs �rrzaddinfo.__init__<s��� �g�t�%�b�)���rc��|jSr)r0rs r�infozaddinfo.info@s���|�|�r)rr$r%r&rr2r'r(s@rrr9s���8��rrc�@��eZdZdZd�fd� Zed��Zd�Zd�Z�xZ S)rz9class to add info() and geturl() methods to an open file.c�J��tt|�||�||_||_yr)r rr�url�code)r rr0r5r6rs �rrzaddinfourl.__init__Gs"��� �j�$�(��W�5������ rc��|jSr�r6rs r�statuszaddinfourl.statusLs���y�y�rc��|jSrr8rs r�getcodezaddinfourl.getcodePs���y�y�rc��|jSr)r5rs r�geturlzaddinfourl.geturlSs���x�x�rr) rr$r%r&r�propertyr9r;r=r'r(s@rrrDs)���C�� �����rr)r&�tempfile�__all__�_TemporaryFileWrapperrrrr�rr�<module>rCsJ���� >���h�,�,��..�7�.�(�g����rPKS�\y5+1*__pycache__/response.cpython-312.opt-2.pycnu�[���� T��h9 ��� ddlZgd�ZGd�dej�ZGd�de�ZGd�de�ZGd �d e�Zy)�N)�addbase�addclosehook�addinfo� addinfourlc�2��eZdZ �fd�Zd�Zd�Zd�Z�xZS)rc�@��tt|�|dd��||_y)Nz<urllib response>F)�delete)�superr�__init__�fp)�selfr� __class__s ��(/usr/lib64/python3.12/urllib/response.pyrzaddbase.__init__s!��� �g��&�r�+>�u�&�M����c�h�d|jj�dt|��d|j�d�S)N�<z at z whose fp = �>)r�__name__�id�file�r s r�__repr__zaddbase.__repr__s(��-1�^�^�-D�-D�-/��X�t�y�y�B� Brc�H�|jjrtd��|S)NzI/O operation on closed file)r�closed� ValueErrorrs r� __enter__zaddbase.__enter__s���7�7�>�>��;�<�<��rc�$�|j�y�N)�close)r �type�value� tracebacks r�__exit__zaddbase.__exit__!s��� � �r)r� __module__�__qualname__rrrr#� __classcell__�rs@rrrs���Y�� B�� rrc�*��eZdZ �fd�Z�fd�Z�xZS)rc�H��tt|�|�||_||_yr)r rr� closehook�hookargs)r rr*r+rs �rrzaddclosehook.__init__(s ��� �l�D�*�2�.�"��� �� rc��� |j}|j}|rd|_d|_||�tt|��y#tt|��wxYwr)r*r+r rr)r r*r+rs �rrzaddclosehook.close-sN��� .����I��}�}�H��!%��� $�� ��8�$��,��+�-��E�,��+�-�s�-A�A)rr$r%rrr&r's@rrr%s���4�!� .� .rrc�&��eZdZ �fd�Zd�Z�xZS)rc�:��tt|�|�||_yr)r rr�headers)r rr/rs �rrzaddinfo.__init__<s��� �g�t�%�b�)���rc��|jSr)r/rs r�infozaddinfo.info@s���|�|�r)rr$r%rr1r&r's@rrr9s���8��rrc�>��eZdZ d�fd� Zed��Zd�Zd�Z�xZS)rc�J��tt|�||�||_||_yr)r rr�url�code)r rr/r4r5rs �rrzaddinfourl.__init__Gs"��� �j�$�(��W�5������ rc��|jSr�r5rs r�statuszaddinfourl.statusLs���y�y�rc��|jSrr7rs r�getcodezaddinfourl.getcodePs���y�y�rc��|jSr)r4rs r�geturlzaddinfourl.geturlSs���x�x�rr) rr$r%r�propertyr8r:r<r&r's@rrrDs)���C�� �����rr)�tempfile�__all__�_TemporaryFileWrapperrrrr�rr�<module>rBsJ���� >���h�,�,��..�7�.�(�g����rPKS�\y��@@$__pycache__/response.cpython-312.pycnu�[���� T��h9 ���dZddlZgd�ZGd�dej�ZGd�de�ZGd�d e�ZGd �de�Zy)aResponse classes used by urllib. The base class, addbase, defines a minimal file-like interface, including read() and readline(). The typical response object is an addinfourl instance, which defines an info() method that returns headers and a geturl() method that returns the url. �N)�addbase�addclosehook�addinfo� addinfourlc�4��eZdZdZ�fd�Zd�Zd�Zd�Z�xZS)rzOBase class for addinfo and addclosehook. Is a good idea for garbage collection.c�@��tt|�|dd��||_y)Nz<urllib response>F)�delete)�superr�__init__�fp)�selfr� __class__s ��(/usr/lib64/python3.12/urllib/response.pyrzaddbase.__init__s!��� �g��&�r�+>�u�&�M����c�h�d|jj�dt|��d|j�d�S)N�<z at z whose fp = �>)r�__name__�id�file�r s r�__repr__zaddbase.__repr__s(��-1�^�^�-D�-D�-/��X�t�y�y�B� Brc�H�|jjrtd��|S)NzI/O operation on closed file)r�closed� ValueErrorrs r� __enter__zaddbase.__enter__s���7�7�>�>��;�<�<��rc�$�|j�y�N)�close)r �type�value� tracebacks r�__exit__zaddbase.__exit__!s��� � �r) r� __module__�__qualname__�__doc__rrrr#� __classcell__�rs@rrrs���Y�� B�� rrc�,��eZdZdZ�fd�Z�fd�Z�xZS)rz*Class to add a close hook to an open file.c�H��tt|�|�||_||_yr)r rr� closehook�hookargs)r rr+r,rs �rrzaddclosehook.__init__(s ��� �l�D�*�2�.�"��� �� rc��� |j}|j}|rd|_d|_||�tt|��y#tt|��wxYwr)r+r,r rr)r r+r,rs �rrzaddclosehook.close-sN��� .����I��}�}�H��!%��� $�� ��8�$��,��+�-��E�,��+�-�s�-A�A)rr$r%r&rrr'r(s@rrr%s���4�!� .� .rrc�(��eZdZdZ�fd�Zd�Z�xZS)rz.class to add an info() method to an open file.c�:��tt|�|�||_yr)r rr�headers)r rr0rs �rrzaddinfo.__init__<s��� �g�t�%�b�)���rc��|jSr)r0rs r�infozaddinfo.info@s���|�|�r)rr$r%r&rr2r'r(s@rrr9s���8��rrc�@��eZdZdZd�fd� Zed��Zd�Zd�Z�xZ S)rz9class to add info() and geturl() methods to an open file.c�J��tt|�||�||_||_yr)r rr�url�code)r rr0r5r6rs �rrzaddinfourl.__init__Gs"��� �j�$�(��W�5������ rc��|jSr�r6rs r�statuszaddinfourl.statusLs���y�y�rc��|jSrr8rs r�getcodezaddinfourl.getcodePs���y�y�rc��|jSr)r5rs r�geturlzaddinfourl.geturlSs���x�x�rr) rr$r%r&r�propertyr9r;r=r'r(s@rrrDs)���C�� �����rr)r&�tempfile�__all__�_TemporaryFileWrapperrrrr�rr�<module>rCsJ���� >���h�,�,��..�7�.�(�g����rPKS�\y��00-__pycache__/robotparser.cpython-312.opt-1.pycnu�[���� T��h�$���dZddlZddlZddlZddlZdgZejdd�ZGd�d�Z Gd�d�Z Gd �d �Zy)a% robotparser.py Copyright (C) 2000 Bastian Kleineidam You can choose between two licenses when using this package: 1) GNU GPLv2 2) PSF license for Python 2.2 The robots.txt Exclusion Protocol is implemented as specified in http://www.robotstxt.org/norobots-rfc.txt �N�RobotFileParser�RequestRatezrequests secondsc�Z�eZdZdZdd�Zd�Zd�Zd�Zd�Zd�Z d�Z d �Zd �Zd�Z d�Zd �Zy)rzs This class provides a set of methods to read, parse and answer questions about a single robots.txt file. c�z�g|_g|_d|_d|_d|_|j|�d|_y)NFr)�entries�sitemaps� default_entry�disallow_all� allow_all�set_url�last_checked��self�urls �+/usr/lib64/python3.12/urllib/robotparser.py�__init__zRobotFileParser.__init__s;������� �!���!���������S�����c��|jS)z�Returns the time the robots.txt file was last fetched. This is useful for long-running web spiders that need to check for new robots.txt files periodically. )r �rs r�mtimezRobotFileParser.mtime&s��� � � rc�6�ddl}|j�|_y)zYSets the time the robots.txt file was last fetched to the current time. rN)�timer )rrs r�modifiedzRobotFileParser.modified/s�� � �I�I�K��rc�p�||_tjj|�dd\|_|_y)z,Sets the URL referring to a robots.txt file.��N)r�urllib�parse�urlparse�host�pathrs rrzRobotFileParser.set_url7s-�����%�|�|�4�4�S�9�!�A�>��� �4�9rc�� tjj|j�}|j �}|j|j d�j��y#tjj$rU}|jdvrd|_n%|jdk\r|jdkrd|_|j�Yd}~yd}~wwxYw)z4Reads the robots.txt URL and feeds it to the parser.zutf-8)i�i�Ti�i�N)r�request�urlopenr�readr�decode� splitlines�error� HTTPError�coder r�close)r�f�raw�errs rr%zRobotFileParser.read<s��� 9����&�&�t�x�x�0�A��&�&�(�C��J�J�s�z�z�'�*�5�5�7�8���|�|�%�%� ��x�x�:�%�$(��!����S��S�X�X��^�!%����I�I�K�K�� �s�)A*�*C�AC�Cc��d|jvr|j�||_yy|jj|�y�N�*)� useragentsr r�append)r�entrys r� _add_entryzRobotFileParser._add_entryJs=���%�"�"�"��!�!�)�%*��"�*� �L�L����&rc���d}t�}|j�|D�]�}|s4|dk(r t�}d}n"|dk(r|j|�t�}d}|jd�}|dk\r|d|}|j �}|s�h|jdd�}t |�dk(s��|dj �j�|d<tjj|dj ��|d<|ddk(rB|dk(r|j|�t�}|jj|d�d}��*|ddk(r3|dk7s��9|jjt|dd ��d}��e|dd k(r3|dk7s��t|jjt|dd��d}���|ddk(r?|dk7s���|dj �j�rt!|d�|_d}���|dd k(r�|dk7s���|djd�}t |�dk(rk|dj �j�rJ|dj �j�r)t%t!|d�t!|d��|_d}���|ddk(s���|j(j|d����|dk(r|j|�yy)z�Parse the input lines from a robots.txt file. We allow that a user-agent: line is not preceded by one or more blank lines. rr��#N�:z user-agent�disallowF�allowTzcrawl-delayzrequest-rate�/�sitemap)�Entryrr5�find�strip�split�len�lowerrr�unquoter2r3� rulelines�RuleLine�isdigit�int�delayr�req_rater)r�lines�stater4�line�i�numberss rrzRobotFileParser.parseSs��������� � ���D���A�:�!�G�E��E��a�Z��O�O�E�*�!�G�E��E�� � �#��A��A�v��B�Q�x���:�:�<�D����:�:�c�1�%�D��4�y�A�~��q�'�-�-�/�/�/�1��Q�� �,�,�.�.�t�A�w�}�}��?��Q����7�l�*���z�����.� %����$�$�+�+�D��G�4��E��!�W� �*���z����.�.�x��Q���/G�H� !���!�W��'���z����.�.�x��Q���/F�G� !���!�W� �-���z� ��7�=�=�?�2�2�4�*-�d�1�g�,�E�K� !���!�W��.���z�"&�q�'�-�-��"4����L�A�-�'�!�*�2B�2B�2D�2L�2L�2N� '�� � 0� 0� 2� :� :� <�-8��W�Q�Z��#�g�VW�j�/�-Z�E�N� !���!�W� �)� �M�M�(�(��a��1�o�p�A�:��O�O�E�"�rc�b�|jry|jry|jsytjjtjj |��}tjjdd|j|j|j|jf�}tjj|�}|sd}|jD]&}|j|�s�|j|�cS|j r|j j|�Sy)z=using the parsed robots.txt decide if useragent can fetch urlFT�r<)r rr rrrrD� urlunparser!�params�query�fragment�quoter� applies_to� allowancer )r� useragentr� parsed_urlr4s r� can_fetchzRobotFileParser.can_fetch�s��������>�>�� � � ���\�\�*�*�6�<�<�+?�+?��+D�E� ��l�l�%�%�r�"�Z�_�_����j�.�.� �0C�0C�'E�F���l�l� � ��%����C��\�\�E���� �*����s�+�+�"�����%�%�/�/��4�4�rc���|j�sy|jD]!}|j|�s�|jcS|jr|jjSy�N)rrrWrIr �rrYr4s r�crawl_delayzRobotFileParser.crawl_delay�sT���z�z�|���\�\�E���� �*��{�{�"�"�����%�%�+�+�+�rc���|j�sy|jD]!}|j|�s�|jcS|jr|jjSyr])rrrWrJr r^s r�request_ratezRobotFileParser.request_rate�sT���z�z�|���\�\�E���� �*��~�~�%�"�����%�%�.�.�.�rc�4�|jsy|jSr])rrs r� site_mapszRobotFileParser.site_maps�s���}�}���}�}�rc��|j}|j�||jgz}djtt|��S)Nz )rr �join�map�str)rrs r�__str__zRobotFileParser.__str__�s@���,�,�����)���!3�!3� 4�4�G��{�{�3�s�G�,�-�-rN)rQ)�__name__� __module__�__qualname__�__doc__rrrrr%r5rr[r_rarcrh�rrrrsE��� �!�(�?� 9�'�G#�R�:��� .rc�"�eZdZdZd�Zd�Zd�Zy)rFzoA rule line is a single "Allow:" (allowance==True) or "Disallow:" (allowance==False) followed by a path.c���|dk(r|sd}tjjtjj|��}tjj |�|_||_y)NrQT)rrrRrrVr!rX)rr!rXs rrzRuleLine.__init__�sP���2�:�i��I��|�|�&�&�v�|�|�'<�'<�T�'B�C���L�L�&�&�t�,�� �"��rc�Z�|jdk(xs|j|j�Sr0)r!� startswith)r�filenames rrWzRuleLine.applies_to�s%���y�y�C��A�8�#6�#6�t�y�y�#A�Arc�B�|jrdnddz|jzS)N�Allow�Disallowz: )rXr!rs rrhzRuleLine.__str__�s���>�>��z�T�A�D�I�I�M�MrN)rirjrkrlrrWrhrmrrrFrF�s��1�#�B�NrrFc�(�eZdZdZd�Zd�Zd�Zd�Zy)r>z?An entry has one or more user-agents and zero or more rulelinesc�<�g|_g|_d|_d|_yr])r2rErIrJrs rrzEntry.__init__�s���������� ��� rc��g}|jD]}|jd|����|j�|jd|j���|j�7|j}|jd|j�d|j ���|j tt|j��dj|�S)NzUser-agent: z Crawl-delay: zRequest-rate: r<� )r2r3rIrJ�requests�seconds�extendrfrgrEre)r�ret�agent�rates rrhz Entry.__str__�s������_�_�E��J�J��e�W�-�.�%��:�:�!��J�J��t�z�z�l�3�4��=�=�$��=�=�D��J�J��� � ��a����~�F�G�� � �3�s�D�N�N�+�,��y�y��~�rc��|jd�dj�}|jD]}|dk(ry|j�}||vs�yy)z2check if this entry applies to the specified agentr<rr1TF)rArCr2)rrYr~s rrWzEntry.applies_to�sS���O�O�C�(��+�1�1�3� ��_�_�E���|���K�K�M�E�� �!�� %�rc�d�|jD]!}|j|�s�|jcSy)zZPreconditions: - our agent applies to this entry - filename is URL decodedT)rErWrX)rrrrMs rrXzEntry.allowances-���N�N�D����x�(��~�~�%�#�rN)rirjrkrlrrhrWrXrmrrr>r>�s��I�� ��rr>)rl�collections�urllib.errorr�urllib.parse�urllib.request�__all__� namedtuplerrrFr>rmrr�<module>r�sX�� ������ ��$�k�$�$�]�4F�G��.�.�DN�N�$(�(rPKS�\3�9D�*�*-__pycache__/robotparser.cpython-312.opt-2.pycnu�[���� T��h�$��� ddlZddlZddlZddlZdgZejdd�ZGd�d�ZGd�d�Z Gd�d �Z y) �N�RobotFileParser�RequestRatezrequests secondsc�X�eZdZ dd�Zd�Zd�Zd�Zd�Zd�Zd�Z d�Z d �Zd �Zd�Z d�Zy )rc�z�g|_g|_d|_d|_d|_|j|�d|_y)NFr)�entries�sitemaps� default_entry�disallow_all� allow_all�set_url�last_checked��self�urls �+/usr/lib64/python3.12/urllib/robotparser.py�__init__zRobotFileParser.__init__s;������� �!���!���������S�����c�� |jS�N)r �rs r�mtimezRobotFileParser.mtime&s�� �� � � rc�8� ddl}|j�|_y)Nr)�timer )rrs r�modifiedzRobotFileParser.modified/s�� � � �I�I�K��rc�r� ||_tjj|�dd\|_|_y)N��)r�urllib�parse�urlparse�host�pathrs rrzRobotFileParser.set_url7s0��:����%�|�|�4�4�S�9�!�A�>��� �4�9rc�� tjj|j�}|j �}|j|j d�j��y#tjj$rU}|jdvrd|_n%|jdk\r|jdkrd|_|j�Yd}~yd}~wwxYw)Nzutf-8)i�i�Ti�i�)r�request�urlopenr�readr�decode� splitlines�error� HTTPError�coder r�close)r�f�raw�errs rr&zRobotFileParser.read<s���B� 9����&�&�t�x�x�0�A��&�&�(�C��J�J�s�z�z�'�*�5�5�7�8���|�|�%�%� ��x�x�:�%�$(��!����S��S�X�X��^�!%����I�I�K�K�� �s�)A+�+C�AC�Cc��d|jvr|j�||_yy|jj|�y�N�*)� useragentsr r�append)r�entrys r� _add_entryzRobotFileParser._add_entryJs=���%�"�"�"��!�!�)�%*��"�*� �L�L����&rc��� d}t�}|j�|D�]�}|s4|dk(r t�}d}n"|dk(r|j|�t�}d}|jd�}|dk\r|d|}|j �}|s�h|jdd�}t |�dk(s��|dj �j�|d<tjj|dj ��|d<|ddk(rB|dk(r|j|�t�}|jj|d�d}��*|ddk(r3|dk7s��9|jjt|dd��d}��e|dd k(r3|dk7s��t|jjt|dd ��d}���|ddk(r?|dk7s���|dj �j�rt!|d�|_d}���|ddk(r�|dk7s���|djd �}t |�dk(rk|dj �j�rJ|dj �j�r)t%t!|d�t!|d��|_d}���|ddk(s���|j(j|d����|dk(r|j|�yy)Nrr��#�:z user-agent�disallowF�allowTzcrawl-delayzrequest-rate�/�sitemap)�Entryrr6�find�strip�split�len�lowerrr�unquoter3r4� rulelines�RuleLine�isdigit�int�delayr�req_rater)r�lines�stater5�line�i�numberss rrzRobotFileParser.parseSs��� ������� � ���D���A�:�!�G�E��E��a�Z��O�O�E�*�!�G�E��E�� � �#��A��A�v��B�Q�x���:�:�<�D����:�:�c�1�%�D��4�y�A�~��q�'�-�-�/�/�/�1��Q�� �,�,�.�.�t�A�w�}�}��?��Q����7�l�*���z�����.� %����$�$�+�+�D��G�4��E��!�W� �*���z����.�.�x��Q���/G�H� !���!�W��'���z����.�.�x��Q���/F�G� !���!�W� �-���z� ��7�=�=�?�2�2�4�*-�d�1�g�,�E�K� !���!�W��.���z�"&�q�'�-�-��"4����L�A�-�'�!�*�2B�2B�2D�2L�2L�2N� '�� � 0� 0� 2� :� :� <�-8��W�Q�Z��#�g�VW�j�/�-Z�E�N� !���!�W� �)� �M�M�(�(��a��1�o�p�A�:��O�O�E�"�rc�d� |jry|jry|jsytjjtjj |��}tjjdd|j|j|j|jf�}tjj|�}|sd}|jD]&}|j|�s�|j|�cS|j r|j j|�Sy)NFT�r=)r rr rrr rE� urlunparser"�params�query�fragment�quoter� applies_to� allowancer )r� useragentr� parsed_urlr5s r� can_fetchzRobotFileParser.can_fetch�s���K������>�>�� � � ���\�\�*�*�6�<�<�+?�+?��+D�E� ��l�l�%�%�r�"�Z�_�_����j�.�.� �0C�0C�'E�F���l�l� � ��%����C��\�\�E���� �*����s�+�+�"�����%�%�/�/��4�4�rc���|j�sy|jD]!}|j|�s�|jcS|jr|jjSyr)rrrXrJr �rrZr5s r�crawl_delayzRobotFileParser.crawl_delay�sT���z�z�|���\�\�E���� �*��{�{�"�"�����%�%�+�+�+�rc���|j�sy|jD]!}|j|�s�|jcS|jr|jjSyr)rrrXrKr r^s r�request_ratezRobotFileParser.request_rate�sT���z�z�|���\�\�E���� �*��~�~�%�"�����%�%�.�.�.�rc�4�|jsy|jSr)rrs r� site_mapszRobotFileParser.site_maps�s���}�}���}�}�rc��|j}|j�||jgz}djtt|��S)Nz )rr �join�map�str)rrs r�__str__zRobotFileParser.__str__�s@���,�,�����)���!3�!3� 4�4�G��{�{�3�s�G�,�-�-rN)rR)�__name__� __module__�__qualname__rrrrr&r6rr\r_rarcrh�rrrrsE��� �!�(�?� 9�'�G#�R�:��� .rc� �eZdZ d�Zd�Zd�Zy)rGc���|dk(r|sd}tjjtjj|��}tjj |�|_||_y)NrRT)rrrSr rWr"rY)rr"rYs rrzRuleLine.__init__�sP���2�:�i��I��|�|�&�&�v�|�|�'<�'<�T�'B�C���L�L�&�&�t�,�� �"��rc�Z�|jdk(xs|j|j�Sr1)r"� startswith)r�filenames rrXzRuleLine.applies_to�s%���y�y�C��A�8�#6�#6�t�y�y�#A�Arc�B�|jrdnddz|jzS)N�Allow�Disallowz: )rYr"rs rrhzRuleLine.__str__�s���>�>��z�T�A�D�I�I�M�MrN)rirjrkrrXrhrlrrrGrG�s��1�#�B�NrrGc�&�eZdZ d�Zd�Zd�Zd�Zy)r?c�<�g|_g|_d|_d|_yr)r3rFrJrKrs rrzEntry.__init__�s���������� ��� rc��g}|jD]}|jd|����|j�|jd|j���|j�7|j}|jd|j�d|j ���|j tt|j��dj|�S)NzUser-agent: z Crawl-delay: zRequest-rate: r=� )r3r4rJrK�requests�seconds�extendrfrgrFre)r�ret�agent�rates rrhz Entry.__str__�s������_�_�E��J�J��e�W�-�.�%��:�:�!��J�J��t�z�z�l�3�4��=�=�$��=�=�D��J�J��� � ��a����~�F�G�� � �3�s�D�N�N�+�,��y�y��~�rc�� |jd�dj�}|jD]}|dk(ry|j�}||vs�yy)Nr=rr2TF)rBrDr3)rrZr}s rrXzEntry.applies_to�sT��@��O�O�C�(��+�1�1�3� ��_�_�E���|���K�K�M�E�� �!�� %�rc�f� |jD]!}|j|�s�|jcSy)NT)rFrXrY)rrqrNs rrYzEntry.allowances2�� %��N�N�D����x�(��~�~�%�#�rN)rirjrkrrhrXrYrlrrr?r?�s��I�� ��rr?)�collections�urllib.errorr�urllib.parse�urllib.request�__all__� namedtuplerrrGr?rlrr�<module>r�sX�� ������ ��$�k�$�$�]�4F�G��.�.�DN�N�$(�(rPKS�\y��00'__pycache__/robotparser.cpython-312.pycnu�[���� T��h�$���dZddlZddlZddlZddlZdgZejdd�ZGd�d�Z Gd�d�Z Gd �d �Zy)a% robotparser.py Copyright (C) 2000 Bastian Kleineidam You can choose between two licenses when using this package: 1) GNU GPLv2 2) PSF license for Python 2.2 The robots.txt Exclusion Protocol is implemented as specified in http://www.robotstxt.org/norobots-rfc.txt �N�RobotFileParser�RequestRatezrequests secondsc�Z�eZdZdZdd�Zd�Zd�Zd�Zd�Zd�Z d�Z d �Zd �Zd�Z d�Zd �Zy)rzs This class provides a set of methods to read, parse and answer questions about a single robots.txt file. c�z�g|_g|_d|_d|_d|_|j|�d|_y)NFr)�entries�sitemaps� default_entry�disallow_all� allow_all�set_url�last_checked��self�urls �+/usr/lib64/python3.12/urllib/robotparser.py�__init__zRobotFileParser.__init__s;������� �!���!���������S�����c��|jS)z�Returns the time the robots.txt file was last fetched. This is useful for long-running web spiders that need to check for new robots.txt files periodically. )r �rs r�mtimezRobotFileParser.mtime&s��� � � rc�6�ddl}|j�|_y)zYSets the time the robots.txt file was last fetched to the current time. rN)�timer )rrs r�modifiedzRobotFileParser.modified/s�� � �I�I�K��rc�p�||_tjj|�dd\|_|_y)z,Sets the URL referring to a robots.txt file.��N)r�urllib�parse�urlparse�host�pathrs rrzRobotFileParser.set_url7s-�����%�|�|�4�4�S�9�!�A�>��� �4�9rc�� tjj|j�}|j �}|j|j d�j��y#tjj$rU}|jdvrd|_n%|jdk\r|jdkrd|_|j�Yd}~yd}~wwxYw)z4Reads the robots.txt URL and feeds it to the parser.zutf-8)i�i�Ti�i�N)r�request�urlopenr�readr�decode� splitlines�error� HTTPError�coder r�close)r�f�raw�errs rr%zRobotFileParser.read<s��� 9����&�&�t�x�x�0�A��&�&�(�C��J�J�s�z�z�'�*�5�5�7�8���|�|�%�%� ��x�x�:�%�$(��!����S��S�X�X��^�!%����I�I�K�K�� �s�)A*�*C�AC�Cc��d|jvr|j�||_yy|jj|�y�N�*)� useragentsr r�append)r�entrys r� _add_entryzRobotFileParser._add_entryJs=���%�"�"�"��!�!�)�%*��"�*� �L�L����&rc���d}t�}|j�|D�]�}|s4|dk(r t�}d}n"|dk(r|j|�t�}d}|jd�}|dk\r|d|}|j �}|s�h|jdd�}t |�dk(s��|dj �j�|d<tjj|dj ��|d<|ddk(rB|dk(r|j|�t�}|jj|d�d}��*|ddk(r3|dk7s��9|jjt|dd ��d}��e|dd k(r3|dk7s��t|jjt|dd��d}���|ddk(r?|dk7s���|dj �j�rt!|d�|_d}���|dd k(r�|dk7s���|djd�}t |�dk(rk|dj �j�rJ|dj �j�r)t%t!|d�t!|d��|_d}���|ddk(s���|j(j|d����|dk(r|j|�yy)z�Parse the input lines from a robots.txt file. We allow that a user-agent: line is not preceded by one or more blank lines. rr��#N�:z user-agent�disallowF�allowTzcrawl-delayzrequest-rate�/�sitemap)�Entryrr5�find�strip�split�len�lowerrr�unquoter2r3� rulelines�RuleLine�isdigit�int�delayr�req_rater)r�lines�stater4�line�i�numberss rrzRobotFileParser.parseSs��������� � ���D���A�:�!�G�E��E��a�Z��O�O�E�*�!�G�E��E�� � �#��A��A�v��B�Q�x���:�:�<�D����:�:�c�1�%�D��4�y�A�~��q�'�-�-�/�/�/�1��Q�� �,�,�.�.�t�A�w�}�}��?��Q����7�l�*���z�����.� %����$�$�+�+�D��G�4��E��!�W� �*���z����.�.�x��Q���/G�H� !���!�W��'���z����.�.�x��Q���/F�G� !���!�W� �-���z� ��7�=�=�?�2�2�4�*-�d�1�g�,�E�K� !���!�W��.���z�"&�q�'�-�-��"4����L�A�-�'�!�*�2B�2B�2D�2L�2L�2N� '�� � 0� 0� 2� :� :� <�-8��W�Q�Z��#�g�VW�j�/�-Z�E�N� !���!�W� �)� �M�M�(�(��a��1�o�p�A�:��O�O�E�"�rc�b�|jry|jry|jsytjjtjj |��}tjjdd|j|j|j|jf�}tjj|�}|sd}|jD]&}|j|�s�|j|�cS|j r|j j|�Sy)z=using the parsed robots.txt decide if useragent can fetch urlFT�r<)r rr rrrrD� urlunparser!�params�query�fragment�quoter� applies_to� allowancer )r� useragentr� parsed_urlr4s r� can_fetchzRobotFileParser.can_fetch�s��������>�>�� � � ���\�\�*�*�6�<�<�+?�+?��+D�E� ��l�l�%�%�r�"�Z�_�_����j�.�.� �0C�0C�'E�F���l�l� � ��%����C��\�\�E���� �*����s�+�+�"�����%�%�/�/��4�4�rc���|j�sy|jD]!}|j|�s�|jcS|jr|jjSy�N)rrrWrIr �rrYr4s r�crawl_delayzRobotFileParser.crawl_delay�sT���z�z�|���\�\�E���� �*��{�{�"�"�����%�%�+�+�+�rc���|j�sy|jD]!}|j|�s�|jcS|jr|jjSyr])rrrWrJr r^s r�request_ratezRobotFileParser.request_rate�sT���z�z�|���\�\�E���� �*��~�~�%�"�����%�%�.�.�.�rc�4�|jsy|jSr])rrs r� site_mapszRobotFileParser.site_maps�s���}�}���}�}�rc��|j}|j�||jgz}djtt|��S)Nz )rr �join�map�str)rrs r�__str__zRobotFileParser.__str__�s@���,�,�����)���!3�!3� 4�4�G��{�{�3�s�G�,�-�-rN)rQ)�__name__� __module__�__qualname__�__doc__rrrrr%r5rr[r_rarcrh�rrrrsE��� �!�(�?� 9�'�G#�R�:��� .rc�"�eZdZdZd�Zd�Zd�Zy)rFzoA rule line is a single "Allow:" (allowance==True) or "Disallow:" (allowance==False) followed by a path.c���|dk(r|sd}tjjtjj|��}tjj |�|_||_y)NrQT)rrrRrrVr!rX)rr!rXs rrzRuleLine.__init__�sP���2�:�i��I��|�|�&�&�v�|�|�'<�'<�T�'B�C���L�L�&�&�t�,�� �"��rc�Z�|jdk(xs|j|j�Sr0)r!� startswith)r�filenames rrWzRuleLine.applies_to�s%���y�y�C��A�8�#6�#6�t�y�y�#A�Arc�B�|jrdnddz|jzS)N�Allow�Disallowz: )rXr!rs rrhzRuleLine.__str__�s���>�>��z�T�A�D�I�I�M�MrN)rirjrkrlrrWrhrmrrrFrF�s��1�#�B�NrrFc�(�eZdZdZd�Zd�Zd�Zd�Zy)r>z?An entry has one or more user-agents and zero or more rulelinesc�<�g|_g|_d|_d|_yr])r2rErIrJrs rrzEntry.__init__�s���������� ��� rc��g}|jD]}|jd|����|j�|jd|j���|j�7|j}|jd|j�d|j ���|j tt|j��dj|�S)NzUser-agent: z Crawl-delay: zRequest-rate: r<� )r2r3rIrJ�requests�seconds�extendrfrgrEre)r�ret�agent�rates rrhz Entry.__str__�s������_�_�E��J�J��e�W�-�.�%��:�:�!��J�J��t�z�z�l�3�4��=�=�$��=�=�D��J�J��� � ��a����~�F�G�� � �3�s�D�N�N�+�,��y�y��~�rc��|jd�dj�}|jD]}|dk(ry|j�}||vs�yy)z2check if this entry applies to the specified agentr<rr1TF)rArCr2)rrYr~s rrWzEntry.applies_to�sS���O�O�C�(��+�1�1�3� ��_�_�E���|���K�K�M�E�� �!�� %�rc�d�|jD]!}|j|�s�|jcSy)zZPreconditions: - our agent applies to this entry - filename is URL decodedT)rErWrX)rrrrMs rrXzEntry.allowances-���N�N�D����x�(��~�~�%�#�rN)rirjrkrlrrhrWrXrmrrr>r>�s��I�� ��rr>)rl�collections�urllib.errorr�urllib.parse�urllib.request�__all__� namedtuplerrrFr>rmrr�<module>r�sX�� ������ ��$�k�$�$�]�4F�G��.�.�DN�N�$(�(rPK23�\�#�zz)__pycache__/__init__.cpython-36.opt-1.pycnu�[���PK23�\�#�zz)�__pycache__/__init__.cpython-36.opt-2.pycnu�[���PK23�\�#�zz#�__pycache__/__init__.cpython-36.pycnu�[���PK23�\��d�� � &s__pycache__/error.cpython-36.opt-1.pycnu�[���PK23�\���;!!&� __pycache__/error.cpython-36.opt-2.pycnu�[���PK23�\��d�� � __pycache__/error.cpython-36.pycnu�[���PK23�\�I�����&!__pycache__/parse.cpython-36.opt-1.pycnu�[���PK23�\>edd&�__pycache__/parse.cpython-36.opt-2.pycnu�[���PK23�\�I����� u __pycache__/parse.cpython-36.pycnu�[���PK23�\�2`zz(u�__pycache__/request.cpython-36.opt-1.pycnu�[���PK23�\pl� 2�2�(G�__pycache__/request.cpython-36.opt-2.pycnu�[���PK23�\���"ў__pycache__/request.cpython-36.pycnu�[���PK23�\��P2��)%�__pycache__/response.cpython-36.opt-1.pycnu�[���PK23�\3<�Z Z )�__pycache__/response.cpython-36.opt-2.pycnu�[���PK23�\��P2��#��__pycache__/response.cpython-36.pycnu�[���PK23�\� ���,��__pycache__/robotparser.cpython-36.opt-1.pycnu�[���PK23�\w��33,��__pycache__/robotparser.cpython-36.opt-2.pycnu�[���PK23�\� ���&6__pycache__/robotparser.cpython-36.pycnu�[���PK23�\/__init__.pynu�[���PK23�\�Ɇ�Q Q T/error.pynu�[���PK23�\�g��e�e��9parse.pynu�[���PK23�\��~@~�~� z�request.pynu�[���PK23�\��@��2presponse.pynu�[���PK23�\�ኀ"�"hyrobotparser.pynu�[���PKS�\��ԉ�*&�__pycache__/__init__.cpython-312.opt-1.pycnu�[���PKS�\��ԉ�* �__pycache__/__init__.cpython-312.opt-2.pycnu�[���PKS�\��ԉ�$�__pycache__/__init__.cpython-312.pycnu�[���PKS�\�0 �BB'ɞ__pycache__/error.cpython-312.opt-1.pycnu�[���PKS�\��,��'b�__pycache__/error.cpython-312.opt-2.pycnu�[���PKS�\�0 �BB!d�__pycache__/error.cpython-312.pycnu�[���PKS�\���t����'��__pycache__/parse.cpython-312.opt-1.pycnu�[���PKS�\i�bp�p�'�__pycache__/parse.cpython-312.opt-2.pycnu�[���PKS�\���t����!�* __pycache__/parse.cpython-312.pycnu�[���PKS�\!�j�j�)�� __pycache__/request.cpython-312.opt-1.pycnu�[���PKS�\#F����)��__pycache__/request.cpython-312.opt-2.pycnu�[���PKS�\S�Q���#�G __pycache__/request.cpython-312.pycnu�[���PKS�\y��@@*� __pycache__/response.cpython-312.opt-1.pycnu�[���PKS�\y5+1*�__pycache__/response.cpython-312.opt-2.pycnu�[���PKS�\y��@@$,__pycache__/response.cpython-312.pycnu�[���PKS�\y��00-�=__pycache__/robotparser.cpython-312.opt-1.pycnu�[���PKS�\3�9D�*�*-�m__pycache__/robotparser.cpython-312.opt-2.pycnu�[���PKS�\y��00'E�__pycache__/robotparser.cpython-312.pycnu�[���PK**4��
/home/emeraadmin/www/Classes/../node_modules/react/../raf/../function-bind/../../4d695/urllib.zip