
    whP                    	   d dl mZ d dlZd dlZd dlZd dlZd dlm	Z	 d dlmZ d dlm
Z
 d dlmZ d dlmZ d dlmZ d d	lmZ d d
l
mZ d dl
mZ d dlmZ d dlmZ d dlmZ ddlmZ ddlmZ ej8                  rd dlmZ  ed      Z  ejB                  d      Z" eg d      Z# eg d      Z$i dddddddddd d!d"d#d$d%d&d'd(d)d*d+d,d-d.d/d0d1d2d3d4d5d6d7d8i d9d:d;d<d=d>d?d@dAdBdCdDdEdFdGdHdIdJdKdLdMdNdOdPdQdRdSdTdUdVdWdXdYdZi d[d\d]d^d_d`dadbdcdddedfdgdhdidjdkdldmdndodpdqdrdsdtdudvdwdxdydzd{d|d}d~ddddddddddddZ% G d de      Z& G d de      Z'dddZ(ddZ)ddZ*ddZ+ddZ,ddZ-ddZ. ejB                  dej^                        Z0 ejB                  dej^                        Z1 ejB                  dej^                  ejd                  z        Z3 ejB                  dej^                        Z4ddZ5 ejB                  dej^                        Z6 ejn                  dd      Z8ejr                  dd       Z:ejr                  dd       Z:	 d	 	 	 	 	 ddZ: ejn                  dd      Z;ejr                  	 d	 	 	 	 	 dd       Z<ejr                  	 	 d	 	 	 	 	 	 	 dd       Z<	 	 d	 	 	 	 	 	 	 ddZ< ejn                  dd      Z=ejr                  	 d	 	 	 	 	 dd       Z>ejr                  	 	 d	 	 	 	 	 	 	 dd       Z>	 	 d	 	 	 	 	 	 	 ddZ>	 d	 	 	 	 	 ddZ?ddZ@	 d	 	 	 	 	 ddZA	 d	 	 	 	 	 ddZBdddZCejr                  dd       ZDejr                  dd       ZD	 	 	 	 ddZDddZEddZFddZG	 d	 	 	 ddZHdddZIdddZJ	 	 	 	 d	 	 	 	 	 	 	 	 	 	 	 ddZK	 d	 	 	 	 	 ddZLddZMddZNddZO	 d	 	 	 	 	 ddZP ejB                  dej                        ZR ejB                  dej                        ZSdddŜZTeTj                  dƄ g  eVdǫ      dȢ eVddʫ      D               	 	 	 	 	 	 	 	 	 	 	 d	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 	 dd˄ZW	 	 	 	 	 	 	 	 dd̄ZXddlYmZZ[ ddl\m]Z^ y)    )annotationsNdate)datetime)time)	timedelta)timezone)Enum)sha1)mktime)struct_time)quote)unquote)parse_http_list   )
_dt_as_utc)
_plain_int)WSGIEnvironmentzM!#$%&'*+-.0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ^_`abcdefghijklmnopqrstuvwxyz|~z&([Ww]/)?(?:"(.*?)"|(.*?))(?:\s*,\s*|$))
allowzcontent-encodingzcontent-languagezcontent-lengthcontent-locationzcontent-md5zcontent-rangezcontent-typeexpireszlast-modified)
connectionz
keep-alivezproxy-authenticatezproxy-authorizationtetrailerztransfer-encodingupgraded   Continuee   zSwitching Protocolsf   
Processingg   zEarly Hints   OK   Created   Accepted   zNon Authoritative Information   z
No Content   zReset Content   zPartial Content   zMulti Status   zAlready Reported   zIM Usedi,  zMultiple Choicesi-  zMoved Permanentlyi.  Foundi/  z	See Otheri0  zNot Modifiedi1  z	Use Proxyi2  zSwitch Proxyi3  zTemporary Redirecti4  zPermanent Redirecti  zBad Requesti  Unauthorizedi  zPayment Requiredi  	Forbiddeni  z	Not Foundi  zMethod Not Allowedi  zNot Acceptablei  zProxy Authentication Requiredi  zRequest Timeouti  Conflicti  Gonei  zLength Requiredi  zPrecondition Failedi  zRequest Entity Too Largei  zRequest URI Too Longi  zUnsupported Media Typei  zRequested Range Not Satisfiablei  zExpectation Failedi  zI'm a teapoti  zMisdirected Requesti  zUnprocessable Entityi  Lockedi  zFailed Dependencyi  z	Too Earlyi  zUpgrade Requiredi  zPrecondition Requiredi  zToo Many Requestsi  zRequest Header Fields Too Largez
Retry WithzUnavailable For Legal ReasonszInternal Server ErrorzNot ImplementedzBad GatewayzService UnavailablezGateway TimeoutzHTTP Version Not SupportedzVariant Also NegotiateszInsufficient StoragezLoop DetectedzNot ExtendedzNetwork Authentication Failed)i  i  i  i  i  i  i  i  i  i  i  i  i  c                      e Zd ZdZdZdZy)COEPzCross Origin Embedder Policiesunsafe-nonezrequire-corpN)__name__
__module____qualname____doc__UNSAFE_NONEREQUIRE_CORP     E/var/www/flask-api/venv/lib/python3.12/site-packages/werkzeug/http.pyr6   r6   |   s    (K!Lr?   r6   c                      e Zd ZdZdZdZdZy)COOPzCross Origin Opener Policiesr7   zsame-origin-allow-popupszsame-originN)r8   r9   r:   r;   r<   SAME_ORIGIN_ALLOW_POPUPSSAME_ORIGINr>   r?   r@   rB   rB      s    &K9Kr?   rB   c                    t        |       }|sy|rt        }|j                  |      r|S |j                  dd      j                  dd      }d| dS )a  Add double quotes around a header value. If the header contains only ASCII token
    characters, it will be returned unchanged. If the header contains ``"`` or ``\``
    characters, they will be escaped with an additional ``\`` character.

    This is the reverse of :func:`unquote_header_value`.

    :param value: The value to quote. Will be converted to a string.
    :param allow_token: Disable to quote the value even if it only has token characters.

    .. versionchanged:: 3.0
        Passing bytes is not supported.

    .. versionchanged:: 3.0
        The ``extra_chars`` parameter is removed.

    .. versionchanged:: 2.3
        The value is quoted if it is the empty string.

    .. versionadded:: 0.5
    z""\\\"\")str_token_chars
issupersetreplace)valueallow_token	value_strtoken_charss       r@   quote_header_valuerR      s]    * E
I"!!),!!$/77UCIykr?   c                    t        |       dk\  r<| d   | d   cxk(  rdk(  r+n | S | dd } | j                  dd      j                  dd      S | S )	a  Remove double quotes and decode slash-escaped ``"`` and ``\`` characters in a
    header value.

    This is the reverse of :func:`quote_header_value`.

    :param value: The header value to unquote.

    .. versionchanged:: 3.0
        The ``is_filename`` parameter is removed.
       r   rH   r   rG   rF   rI   )lenrM   rN   s    r@   unquote_header_valuerX      sZ     5zQ58uRy7C7 L a}}VT*225#>>Lr?   c                    g }| |j                  |        |j                         D ]F  \  }}|	|d   dk(  r|j                  | d|        (|j                  | dt        |              H dj                  |      S )aA  Produce a header value and ``key=value`` parameters separated by semicolons
    ``;``. For example, the ``Content-Type`` header.

    .. code-block:: python

        dump_options_header("text/html", {"charset": "UTF-8"})
        'text/html; charset=UTF-8'

    This is the reverse of :func:`parse_options_header`.

    If a value contains non-token characters, it will be quoted.

    If a value is ``None``, the parameter is skipped.

    In some keys for some headers, a UTF-8 value can be encoded using a special
    ``key*=UTF-8''value`` form, where ``value`` is percent encoded. This function will
    not produce that format automatically, but if a given key ends with an asterisk
    ``*``, the value is assumed to have that form and will not be quoted further.

    :param header: The primary header value.
    :param options: Parameters to encode as ``key=value`` pairs.

    .. versionchanged:: 2.3
        Keys with ``None`` values are skipped rather than treated as a bare key.

    .. versionchanged:: 2.2.3
        If a key ends with ``*``, its value will not be quoted.
    rU   *=; )appenditemsrR   join)headeroptionssegmentskeyrN   s        r@   dump_options_headerrd      s    : Hmmo B
U=r7c>OOse1UG,-OOse1%7%>$?@AB 99Xr?   c                X   t        | t              rmg }| j                         D ]W  \  }}||j                  |       |d   dk(  r|j                  | d|        9|j                  | dt	        |              Y n| D cg c]  }t	        |       }}dj                  |      S c c}w )a'  Produce a header value from a list of items or ``key=value`` pairs, separated by
    commas ``,``.

    This is the reverse of :func:`parse_list_header`, :func:`parse_dict_header`, and
    :func:`parse_set_header`.

    If a value contains non-token characters, it will be quoted.

    If a value is ``None``, the key is output alone.

    In some keys for some headers, a UTF-8 value can be encoded using a special
    ``key*=UTF-8''value`` form, where ``value`` is percent encoded. This function will
    not produce that format automatically, but if a given key ends with an asterisk
    ``*``, the value is assumed to have that form and will not be quoted further.

    .. code-block:: python

        dump_header(["foo", "bar baz"])
        'foo, "bar baz"'

        dump_header({"foo": "bar baz"})
        'foo="bar baz"'

    :param iterable: The items to create a header from.

    .. versionchanged:: 3.0
        The ``allow_token`` parameter is removed.

    .. versionchanged:: 2.2.3
        If a key ends with ``*``, its value will not be quoted.
    rU   rZ   r[   z, )
isinstancedictr^   r]   rR   r_   )iterabler^   rc   rN   xs        r@   dump_headerrj      s    @ (D!"..* 	CJC}S!RCuAeW-.uA&8&?%@AB	C 1991#A&9999U :s   B'ds.ContentSecurityPolicyc                N    dj                  d | j                         D              S )zDump a Content Security Policy header.

    These are structured into policies such as "default-src 'self';
    script-src 'self'".

    .. versionadded:: 1.0.0
       Support for Content Security Policy headers was added.

    r\   c              3  0   K   | ]  \  }}| d |   yw) Nr>   ).0rc   rN   s      r@   	<genexpr>z"dump_csp_header.<locals>.<genexpr>)  s     G*#uuAeW%Gs   )r_   r^   r`   s    r@   dump_csp_headerrr     s     99GGGGr?   c                    g }t        |       D ]:  }t        |      dk\  r|d   |d   cxk(  rdk(  rn n|dd }|j                  |       < |S )a  Parse a header value that consists of a list of comma separated items according
    to `RFC 9110 <https://httpwg.org/specs/rfc9110.html#abnf.extension>`__.

    This extends :func:`urllib.request.parse_http_list` to remove surrounding quotes
    from values.

    .. code-block:: python

        parse_list_header('token, "quoted value"')
        ['token', 'quoted value']

    This is the reverse of :func:`dump_header`.

    :param value: The header value to parse.
    rT   r   rU   rH   r   )_parse_list_headerrV   r]   )rN   resultitems      r@   parse_list_headerrw   ,  sZ      F"5) t9>d1gb8S8":Dd	 Mr?   c                   i }t        |       D ]  }|j                  d      \  }}} |j                         }|s+|sd||<   3| j                         } d}|d   dk(  rP|dd }t        j	                  |       }|r#|j                         \  }} |j                         }|dv rt        | |      } t        |       dk\  r| d   | d   cxk(  rd	k(  rn n| d
d } | ||<    |S )a~  Parse a list header using :func:`parse_list_header`, then parse each item as a
    ``key=value`` pair.

    .. code-block:: python

        parse_dict_header('a=b, c="d, e", f')
        {"a": "b", "c": "d, e", "f": None}

    This is the reverse of :func:`dump_header`.

    If a key does not have a value, it is ``None``.

    This handles charsets for values as described in
    `RFC 2231 <https://www.rfc-editor.org/rfc/rfc2231#section-3>`__. Only ASCII, UTF-8,
    and ISO-8859-1 charsets are accepted, otherwise the value remains quoted.

    :param value: The header value to parse.

    .. versionchanged:: 3.0
        Passing bytes is not supported.

    .. versionchanged:: 3.0
        The ``cls`` argument is removed.

    .. versionchanged:: 2.3
        Added support for ``key*=charset''value`` encoded items.

    .. versionchanged:: 0.9
       The ``cls`` argument was added.
    r[   NrU   rZ   >   asciiutf-8us-ascii
iso-8859-1encodingrT   r   rH   r   )	rw   	partitionstrip_charset_value_rematchgroupslowerr   rV   )rN   ru   rv   rc   	has_valuer~   r   s          r@   parse_dict_headerr   G  s    > %'F!%( $ $s 3YiikF3K#r7c> cr(C%++E2E"',,.%#>>+
 GG9u:?uQx59;;!BKEsI$L Mr?   z([\w!#$%&'*+\-.^`|~]+)=)flagsz[\w!#$%&'*+\-.^`|~]+z
    ([\w!#$%&*+\-.^`|~]*)'  # charset part, could be empty
    [\w!#$%&*+\-.^`|~]*'  # don't care about language part, usually empty
    ([\w!#$%&'*+\-.^`|~]+)  # one or more token chars with percent encoding
    z\*(\d+)$c                   | di fS | j                  d      \  } }}| j                         } |j                         }| r|s| i fS g }	 t        j                  |      x}|j	                  d      j                         }||j                         d }t        j                  |      x}"|j                  ||j	                         f       n_|dd dk(  rWd}t        |      }||k  rE|||dz    dv r|dz  }n/||   dk(  r"|j                  ||d|dz    f       ||dz   d }n|dz  }||k  rE|j                  d      x}dk(  rn||dz   d j                         }i }	d}
d}|D ]  \  }}|d   d	k(  rV|dd }t        j                  |      }|r#|j                         \  }
}|
j                         }
|
s|}
|
d
v r|
}t        ||
      }|d   |d   cxk(  rdk(  r8n n5|dd j                  dd      j                  dd      j                  dd      }t         j#                  |      }|r,|d|j%                          }|	j'                  |d      |z   |	|<   ||	|<    | |	fS )a  Parse a header that consists of a value with ``key=value`` parameters separated
    by semicolons ``;``. For example, the ``Content-Type`` header.

    .. code-block:: python

        parse_options_header("text/html; charset=UTF-8")
        ('text/html', {'charset': 'UTF-8'})

        parse_options_header("")
        ("", {})

    This is the reverse of :func:`dump_options_header`.

    This parses valid parameter parts as described in
    `RFC 9110 <https://httpwg.org/specs/rfc9110.html#parameter>`__. Invalid parts are
    skipped.

    This handles continuations and charsets as described in
    `RFC 2231 <https://www.rfc-editor.org/rfc/rfc2231#section-3>`__, although not as
    strictly as the RFC. Only ASCII, UTF-8, and ISO-8859-1 charsets are accepted,
    otherwise the value remains quoted.

    Clients may not be consistent in how they handle a quote character within a quoted
    value. The `HTML Standard <https://html.spec.whatwg.org/#multipart-form-data>`__
    replaces it with ``%22`` in multipart form data.
    `RFC 9110 <https://httpwg.org/specs/rfc9110.html#quoted.strings>`__ uses backslash
    escapes in HTTP headers. Both are decoded to the ``"`` character.

    Clients may not be consistent in how they handle non-ASCII characters. HTML
    documents must declare ``<meta charset=UTF-8>``, otherwise browsers may replace with
    HTML character references, which can be decoded using :func:`html.unescape`.

    :param value: The header value to parse.
    :return: ``(value, options)``, where ``options`` is a dict

    .. versionchanged:: 2.3
        Invalid parts, such as keys with no value, quoted keys, and incorrectly quoted
        values, are discarded instead of treating as ``None``.

    .. versionchanged:: 2.3
        Only ASCII, UTF-8, and ISO-8859-1 are accepted for charset values.

    .. versionchanged:: 2.3
        Escaped quotes in quoted values, like ``%22`` and ``\"``, are handled.

    .. versionchanged:: 2.2
        Option names are always converted to lowercase.

    .. versionchanged:: 2.2
        The ``multiple`` parameter was removed.

    .. versionchanged:: 0.15
        :rfc:`2231` parameter continuations are handled.

    .. versionadded:: 0.5
    N ;r   rH   rT   >   rI   rG   rU   rZ   >   ry   rz   r{   r|   r}   r   rG   rF   rI   z%22)r   r   _parameter_key_rer   groupr   end_parameter_token_value_rer]   rV   findlstripr   r   r   rM   _continuation_researchstartget)rN   _restpartsmpkposlengthr   ra   r~   continued_encodingpvr   s                 r@   parse_options_headerr     s   r }2v__S)NE1dKKME::<Dby $&E
"((..A;!!#B	?D /44T::Gb!'')_- bqSTFlC#'*o=qcc)b$yq/%:;#C!GI q Fl 99S>!Cb(C!GI%%'? B !GH%)  (Bb6S=CRB%++B/E$||~"#>>+ -
 GG
 &."R(3 a5BrF!c!Ab!!&$/77sCKKESVWB ''+Oekkm$B!++b"-2GBKGBKQ(T '>r?   z-?\d+(\.\d+)?_TAnyAccept	ds.Accept)boundc                     y Nr>   rW   s    r@   parse_accept_headerr   @  s    9<r?   c                     y r   r>   )rN   clss     r@   r   r   D  s    SVr?   c                   |/t        j                  t        t           t        j
                        }| s |d      S g }t        |       D ]  }t        |      \  }}d|v rK|j                  d      j                         }t        j                  |      Jt        |      }|dk  s|dkD  r`d}|rt        ||      }|j                  ||f         ||      S )ak  Parse an ``Accept`` header according to
    `RFC 9110 <https://httpwg.org/specs/rfc9110.html#field.accept>`__.

    Returns an :class:`.Accept` instance, which can sort and inspect items based on
    their quality parameter. When parsing ``Accept-Charset``, ``Accept-Encoding``, or
    ``Accept-Language``, pass the appropriate :class:`.Accept` subclass.

    :param value: The header value to parse.
    :param cls: The :class:`.Accept` class to wrap the result in.
    :return: An instance of ``cls``.

    .. versionchanged:: 2.3
        Parse according to RFC 9110. Items with invalid ``q`` values are skipped.
    Nqr   r   )tcasttyper   dsAcceptrw   r   popr   _q_value_re	fullmatchfloatrd   r]   )rN   r   ru   rv   ra   q_strr   s          r@   r   r   H  s    " {ffT+&		24yF!%( !,T2g'>KK$**,E$$U+3eA1uAA&tW5DtQi /!2 v;r?   _TAnyCCzds.cache_control._CacheControlc                     y r   r>   rN   	on_updates     r@   parse_cache_control_headerr     s     !r?   c                     y r   r>   rN   r   r   s      r@   r   r     s    
 r?   c                    |$t        j                  dt        j                        }| s	 |d|      S  |t	        |       |      S )a  Parse a cache control header.  The RFC differs between response and
    request cache control, this method does not.  It's your responsibility
    to not use the wrong control statements.

    .. versionadded:: 0.5
       The `cls` was added.  If not specified an immutable
       :class:`~werkzeug.datastructures.RequestCacheControl` is returned.

    :param value: a cache control header to be parsed.
    :param on_update: an optional callable that is called every time a value
                      on the :class:`~werkzeug.datastructures.CacheControl`
                      object is changed.
    :param cls: the class for the returned object.  By default
                :class:`~werkzeug.datastructures.RequestCacheControl` is used.
    :return: a `cls` object.
    type[_TAnyCC]r>   )r   r   r   RequestCacheControlr   r   s      r@   r   r     sB    * {ff_b&<&<=2y!! '33r?   _TAnyCSPc                     y r   r>   r   s     r@   parse_csp_headerr     s      #r?   c                     y r   r>   r   s      r@   r   r     s    
 r?   c                t   |$t        j                  dt        j                        }| 	 |d|      S g }| j	                  d      D ]i  }|j                         }d|v s|j                         j	                  dd      \  }} |j                  |j                         | j                         f       k  |||      S )a  Parse a Content Security Policy header.

    .. versionadded:: 1.0.0
       Support for Content Security Policy headers was added.

    :param value: a csp header to be parsed.
    :param on_update: an optional callable that is called every time a value
                      on the object is changed.
    :param cls: the class for the returned object.  By default
                :class:`~werkzeug.datastructures.ContentSecurityPolicy` is used.
    :return: a `cls` object.
    type[_TAnyCSP]r>   r   rn   r   )r   r   r   ContentSecurityPolicysplitr   r]   )rN   r   r   r^   policy	directives         r@   r   r     s    " {ff%r'?'?@}2y!!E++c" = &=%||~33C;IuLL)//+U[[];<= ui  r?   c                p    | st        j                  d|      S t        j                  t        |       |      S )a  Parse a set-like header and return a
    :class:`~werkzeug.datastructures.HeaderSet` object:

    >>> hs = parse_set_header('token, "quoted value"')

    The return value is an object that treats the items case-insensitively
    and keeps the order of the items:

    >>> 'TOKEN' in hs
    True
    >>> hs.index('quoted value')
    1
    >>> hs
    HeaderSet(['token', 'quoted value'])

    To create a header from the :class:`HeaderSet` again, use the
    :func:`dump_header` function.

    :param value: a set header to be parsed.
    :param on_update: an optional callable that is called every time a
                      value on the :class:`~werkzeug.datastructures.HeaderSet`
                      object is changed.
    :return: a :class:`~werkzeug.datastructures.HeaderSet`
    N)r   	HeaderSetrw   r   s     r@   parse_set_headerr     s/    8 ||D),,<<)%0)<<r?   c                    | st        j                         S t        |       }|t        j                  |      S t        j                  t        |       d         S )zParses an if-range header which can be an etag or a date.  Returns
    a :class:`~werkzeug.datastructures.IfRange` object.

    .. versionchanged:: 2.0
        If the value represents a datetime, it is timezone-aware.

    .. versionadded:: 0.7
    r   r   )r   IfRange
parse_dateunquote_etag)rN   r   s     r@   parse_if_range_headerr     sK     zz|eDzzt$$::l5)!,--r?   c                   | rd| vryg }d}| j                  dd      \  }}|j                         j                         }|j                  d      D ]  }|j                         }d|vr y|j                  d      r|dk  r y	 t	        |      }d}d}nqd|v rm|j                  dd      \  }	}
|	j                         }	|
j                         }
	 t	        |	      }||k  s|dk  r y|
r	 t	        |
      dz   }||k\  r yd}||nd}|j                  f        t        j                  ||      S # t
        $ r Y  yw xY w# t
        $ r Y  yw xY w# t
        $ r Y  yw xY w)a  Parses a range header into a :class:`~werkzeug.datastructures.Range`
    object.  If the header is missing or malformed `None` is returned.
    `ranges` is a list of ``(start, stop)`` tuples where the ranges are
    non-inclusive.

    .. versionadded:: 0.7
    r[   Nr   r   ,-rU   )	r   r   r   
startswithr   
ValueErrorr]   r   Range)rN   make_inclusiverangeslast_endunitsrngrv   beginr   	begin_strend_strs              r@   parse_range_headerr     s    Cu$FHS!$JE3KKM!E		# $$zz|d???3!|"4( CHD[!%C!3Iw!)ImmoG"9- x8a< $W-1C C<!os2Hucl#I$$L 88E6"";     "   s6   D.D>&E.	D;:D;>	E
E	EEc                    | y	 | xs dj                         j                  dd      \  }}d|vry|j                  dd      \  }}|dk(  rd}n	 t        |      }|dk(  r(t	        dd|      syt        j                  |dd||      S d|vry|j                  dd      \  }}	 t        |      }	t        |      dz   }
t	        |	|
|      rt        j                  ||	|
||      S y# t        $ r Y yw xY w# t        $ r Y yw xY w# t        $ r Y yw xY w)a  Parses a range header into a
    :class:`~werkzeug.datastructures.ContentRange` object or `None` if
    parsing is not possible.

    .. versionadded:: 0.7

    :param value: a content range header to be parsed.
    :param on_update: an optional callable that is called every time a value
                      on the :class:`~werkzeug.datastructures.ContentRange`
                      object is changed.
    Nr   r   /rZ   )r   r   )r   r   r   r   is_byte_range_validr   ContentRange)rN   r   r   rangedefr   
length_strr   	start_strstop_strr   stops              r@   parse_content_range_headerr   Q  s@    } ;B--/55dA>x (nnS!,OCS	
+F cz"4v6udD&INN	C))C+Ix9%(#a' 5$/ueT6YOOA    		  s5   'C# C2 "D #	C/.C/2	C>=C>	DDc                >    d| v rt        d      d|  d} |rd|  } | S )ziQuote an etag.

    :param etag: the etag to quote.
    :param weak: set to `True` to tag it "weak".
    rH   zinvalid etagW/)r   etagweaks     r@   
quote_etagr     s6     d{((tfA;DD6{Kr?   c                     y r   r>   r   s    r@   r   r     s    14r?   c                     y r   r>   r   s    r@   r   r     s    36r?   c                    | sy| j                         } d}| j                  d      rd}| dd } | dd | dd cxk(  rd	k(  rn | |fS | dd } | |fS )
zUnquote a single etag:

    >>> unquote_etag('W/"bar"')
    ('bar', True)
    >>> unquote_etag('"bar"')
    ('bar', False)

    :param etag: the etag identifier to unquote.
    :return: a ``(etag, weak)`` tuple.
    NNF)r   zw/TrT   Nr   rU   rH   )r   r   r   s     r@   r   r     sp     ::<DD|$ABxBQx49##: Abz:r?   c                   | st        j                         S g }g }t        |       }d}||k  rt        j	                  | |      }|nn|j                         \  }}}|dk(  rt        j                  d      S |r|}|r|j                  |       n|j                  |       |j                         }||k  rt        j                  ||      S )zParse an etag header.

    :param value: the tag header to parse
    :return: an :class:`~werkzeug.datastructures.ETags` object.
    r   rZ   T)star_tag)r   ETagsrV   _etag_rer   r   r]   r   )	rN   strongr   r   r   r   is_weakquotedraws	            r@   parse_etagsr     s     xxzFD
e*C
C
)uc*=$||~#:88T**CKKMM#iik ) 88FD!!r?   c                4    t        |       j                         S )zGenerate an etag for some data.

    .. versionchanged:: 2.0
        Use SHA-1. MD5 may not be available in some environments.
    )r   	hexdigest)datas    r@   generate_etagr     s     :!!r?   c                    | y	 t         j                  j                  |       }|j
                   |j                  t        j                        S |S # t        t        f$ r Y yw xY w)aK  Parse an :rfc:`2822` date into a timezone-aware
    :class:`datetime.datetime` object, or ``None`` if parsing fails.

    This is a wrapper for :func:`email.utils.parsedate_to_datetime`. It
    returns ``None`` if parsing fails instead of raising an exception,
    and always returns a timezone-aware datetime object. If the string
    doesn't have timezone information, it is assumed to be UTC.

    :param value: A string with a supported date format.

    .. versionchanged:: 2.0
        Return a timezone-aware datetime object. Use
        ``email.utils.parsedate_to_datetime``.
    Ntzinfo)	emailutilsparsedate_to_datetime	TypeErrorr   r  rM   r	   utc)rN   dts     r@   r   r     sd     }[[..u5 
yyzzz..I z" s   A A$#A$c                p   t        | t              rkt        | t              s/t        j                  | t	               t
        j                        } nt        |       } t        j                  j                  | d      S t        | t              rt        |       } t        j                  j                  | d      S )a  Format a datetime object or timestamp into an :rfc:`2822` date
    string.

    This is a wrapper for :func:`email.utils.format_datetime`. It
    assumes naive datetime objects are in UTC instead of raising an
    exception.

    :param timestamp: The datetime or timestamp to format. Defaults to
        the current time.

    .. versionchanged:: 2.0
        Use ``email.utils.format_datetime``. Accept ``date`` objects.
    r  T)usegmt)rf   r   r   combiner   r	   r  r   r  r  format_datetimer   r   
formatdate)	timestamps    r@   	http_dater    s      )T")X. ((DF8<<PI #9-I{{**9T*BB)[)9%	;;!!)D!99r?   c                    | sy	 t        |       }|dk  ry	 t        |      S # t        $ r Y yw xY w# t        $ r Y yw xY w)zParses a base-10 integer count of seconds into a timedelta.

    If parsing fails, the return value is `None`.

    :param value: a string consisting of an integer represented in base-10
    :return: a :class:`datetime.timedelta` object or `None`.
    Nr   )seconds)intr   r   OverflowError)rN   r  s     r@   	parse_ager    s[     e* {))    s   # 2 	//	>>c                    | yt        | t              rt        | j                               } nt        |       } | dk  rt	        d      t        |       S )zFormats the duration as a base-10 integer.

    :param age: should be an integer number of seconds,
                a :class:`datetime.timedelta` object, or,
                if the age is unknown, `None` (default).
    Nr   zage cannot be negative)rf   r   r  total_secondsr   rJ   )ages    r@   dump_ager  +  sN     {#y!###%&#h
Qw122s8Or?   c                    t        j                  | j                  d      | j                  d      | j                  d      | j                  d      | j                  d      ||||	      S )a2  Convenience method for conditional requests.

    :param environ: the WSGI environment of the request to be checked.
    :param etag: the etag for the response for comparison.
    :param data: or alternatively the data of the response to automatically
                 generate an etag using :func:`generate_etag`.
    :param last_modified: an optional date of the last modification.
    :param ignore_if_range: If `False`, `If-Range` header will be taken into
                            account.
    :return: `True` if the resource was modified, otherwise `False`.

    .. versionchanged:: 2.0
        SHA-1 is used to generate an etag value for the data. MD5 may
        not be available in some environments.

    .. versionchanged:: 1.0.0
        The check is run for methods other than ``GET`` and ``HEAD``.
    
HTTP_RANGEHTTP_IF_RANGEHTTP_IF_MODIFIED_SINCEHTTP_IF_NONE_MATCHHTTP_IF_MATCH)	
http_rangehttp_if_rangehttp_if_modified_sincehttp_if_none_matchhttp_if_matchr   r   last_modifiedignore_if_range)_sansio_httpis_resource_modifiedr   )environr   r   r$  r%  s        r@   r'  r'  ?  sa    2 ,,;;|,kk/2&{{+CD";;';<kk/2#'
 
r?   c                    |D ch c]  }|j                          }}| D cg c]&  \  }}t        |      r|j                         |v r||f( c}}| dd yc c}w c c}}w )a  Remove all entity headers from a list or :class:`Headers` object.  This
    operation works in-place.  `Expires` and `Content-Location` headers are
    by default not removed.  The reason for this is :rfc:`2616` section
    10.3.5 which specifies some entity headers that should be sent.

    .. versionchanged:: 0.5
       added `allowed` parameter.

    :param headers: a list or :class:`Headers` object.
    :param allowed: a list of headers that should still be allowed even though
                    they are entity headers.
    N)r   is_entity_header)headersallowedri   rc   rN   s        r@   remove_entity_headersr-  e  sb      #**Qqwwy*G* "C$		w(> 
eGAJ +s
   A+Ac                \    | D cg c]  \  }}t        |      r||f c}}| dd yc c}}w )zRemove all HTTP/1.1 "Hop-by-Hop" headers from a list or
    :class:`Headers` object.  This operation works in-place.

    .. versionadded:: 0.5

    :param headers: a list or :class:`Headers` object.
    N)is_hop_by_hop_header)r+  rc   rN   s      r@   remove_hop_by_hop_headersr0  }  s2     (/#e6J36OeGAJ s   ((c                .    | j                         t        v S )zCheck if a header is an entity header.

    .. versionadded:: 0.5

    :param header: the header to test.
    :return: `True` if it's an entity header, `False` otherwise.
    )r   _entity_headersrq   s    r@   r*  r*    s     <<>_,,r?   c                .    | j                         t        v S )zCheck if a header is an HTTP/1.1 "Hop-by-Hop" header.

    .. versionadded:: 0.5

    :param header: the header to test.
    :return: `True` if it's an HTTP/1.1 "Hop-by-Hop" header, `False` otherwise.
    )r   _hop_by_hop_headersrq   s    r@   r/  r/    s     <<>000r?   c                    t        | t              r| j                  d      }n| }|r|j                  d      j	                         }t        j                  ||      S )aO  Parse a cookie from a string or WSGI environ.

    The same key can be provided multiple times, the values are stored
    in-order. The default :class:`MultiDict` will have the first value
    first, and all values can be retrieved with
    :meth:`MultiDict.getlist`.

    :param header: The cookie header as a string, or a WSGI environ dict
        with a ``HTTP_COOKIE`` key.
    :param cls: A dict-like class to store the parsed cookies in.
        Defaults to :class:`MultiDict`.

    .. versionchanged:: 3.0
        Passing bytes, and the ``charset`` and ``errors`` parameters, were removed.

    .. versionchanged:: 1.0
        Returns a :class:`MultiDict` instead of a ``TypeConversionDict``.

    .. versionchanged:: 0.5
        Returns a :class:`TypeConversionDict` instead of a regular dict. The ``cls``
        parameter was added.
    HTTP_COOKIElatin1)cookier   )rf   rg   r   encodedecoder&  parse_cookie)r`   r   r8  s      r@   r;  r;    sN    4 &$M*x(//1$$F<<r?   z#[\w!#$%&'()*+\-./:<=>?@\[\]^`{|}~]*s   [\x00-\x19\",;\\\x7f-\xff]s   \"s   \\)   "   \c              #  J   K   | ]  }|j                  d d      d|z  f  yw)r   bigs   \%03oN)to_bytes)ro   vs     r@   rp   rp     s,      	 ZZ59q=)s   !#    s   ,;      c                H   |t        |d      }|rA|j                  d      d   j                  d      j                  d      j	                  d      }t        |t              rt        |j                               }|t        |t              sNt        |      }nB|@|r>t        t        j                  t        j                        j                         |z         }|
|
j!                         }
|
d	vrt#        d
      |rd}t$        j'                  |      s:t(        j+                  d |j                               j	                  d      }d| d}| j                         j	                  d       d| g}d|fd|fd|fd|fd|fd|fd|
fd|ffD ]8  \  }}||du r|du r|j-                  |       #|j-                  | d|        : dj/                  |      }t1        |      }|	r:||	kD  r5t1        |      }t3        j4                  d|  d| d||z
   d| d|	 dd !       |S )"a:  Create a Set-Cookie header without the ``Set-Cookie`` prefix.

    The return value is usually restricted to ascii as the vast majority
    of values are properly escaped, but that is no guarantee. It's
    tunneled through latin1 as required by :pep:`3333`.

    The return value is not ASCII safe if the key contains unicode
    characters.  This is technically against the specification but
    happens in the wild.  It's strongly recommended to not use
    non-ASCII values for the keys.

    :param max_age: should be a number of seconds, or `None` (default) if
                    the cookie should last only as long as the client's
                    browser session.  Additionally `timedelta` objects
                    are accepted, too.
    :param expires: should be a `datetime` object or unix timestamp.
    :param path: limits the cookie to a given path, per default it will
                 span the whole domain.
    :param domain: Use this if you want to set a cross-domain cookie. For
                   example, ``domain="example.com"`` will set a cookie
                   that is readable by the domain ``www.example.com``,
                   ``foo.example.com`` etc. Otherwise, a cookie will only
                   be readable by the domain that set it.
    :param secure: The cookie will only be available via HTTPS
    :param httponly: disallow JavaScript to access the cookie.  This is an
                     extension to the cookie standard and probably not
                     supported by all browsers.
    :param charset: the encoding for string values.
    :param sync_expires: automatically set expires if max_age is defined
                         but expires not.
    :param max_size: Warn if the final header value exceeds this size. The
        default, 4093, should be safely `supported by most browsers
        <cookie_>`_. Set to 0 to disable this check.
    :param samesite: Limits the scope of the cookie such that it will
        only be attached to requests if those requests are same-site.
    :param partitioned: Opts the cookie into partitioned storage. This
        will also set secure to True

    .. _`cookie`: http://browsercookielimits.squawky.net/

    .. versionchanged:: 3.1
        The ``partitioned`` parameter was added.

    .. versionchanged:: 3.0
        Passing bytes, and the ``charset`` parameter, were removed.

    .. versionchanged:: 2.3.3
        The ``path`` parameter is ``/`` by default.

    .. versionchanged:: 2.3.1
        The value allows more characters without quoting.

    .. versionchanged:: 2.3
        ``localhost`` and other names without a dot are allowed for the domain. A
        leading dot is ignored.

    .. versionchanged:: 2.3
        The ``path`` parameter is ``None`` by default.

    .. versionchanged:: 1.0.0
        The string ``'None'`` is accepted for ``samesite``.
    z%!$&'()*+,/:=@)safe:r   .idnary   )tz>   LaxNoneStrictz,SameSite must be 'Strict', 'Lax', or 'None'.Tc                0    t         | j                            S r   )_cookie_slash_mapr   )r   s    r@   <lambda>zdump_cookie.<locals>.<lambda>:  s    '	2 r?   rH   r7  r[   DomainExpireszMax-AgeSecureHttpOnlyPathSameSitePartitionedFr\   zThe 'z%' cookie is too large: the value was z bytes but the header required z! extra bytes. The final size was z bytes but the limit is z> bytes. Browsers may silently ignore cookies larger than this.rT   )
stacklevel)r   r   r   r9  r:  rf   r   r  r  rJ   r  r   nowr	   r  r  titler   _cookie_no_quote_rer   _cookie_slash_resubr]   r_   rV   warningswarn)rc   rN   max_ager   pathdomainsecurehttponlysync_expiresmax_sizesamesitepartitionedbufkrA  rvcookie_size
value_sizes                     r@   dump_cookiern    sd   X  T 01!!#&q)005<<VDKKGT'9%g++-.'3'(G		HLLHLL9CCEOP>>#44KLL ((/ $$2ELLN

&/ 	 E7! jjl!!(+,AeW5
6C 
6	G	G	6	X		X	$	 1 9U
9JJqM

aS!:#& 
3B
 b'KK(*Z
C5=j\ J  +j 89 := 8
 C99 	
 Ir?   c                    | du |du k7  ry| |du xs |dk\  S |d| cxk  xr |k  S c S | |k\  ryd| cxk  xr |k  S c S )zcChecks if a given byte content range is valid for the given length.

    .. versionadded:: 0.7
    NFr   r>   )r   r   r   s      r@   r   r   i  sj     	44<(	~,1,	E D    	$r?   )datastructures)http)T)rN   zt.AnyrO   boolreturnrJ   )rN   rJ   rs  rJ   )r`   
str | Nonera   zt.Mapping[str, t.Any]rs  rJ   )rh   z$dict[str, t.Any] | t.Iterable[t.Any]rs  rJ   )r`   rk   rs  rJ   )rN   rJ   rs  z	list[str])rN   rJ   rs  zdict[str, str | None])rN   rt  rs  ztuple[str, dict[str, str]])rN   rt  rs  r   )rN   rt  r   ztype[_TAnyAccept]rs  r   r   )rN   rt  r   ztype[_TAnyAccept] | Noners  r   )rN   rt  r   9t.Callable[[ds.cache_control._CacheControl], None] | Noners  zds.RequestCacheControl)N.)rN   rt  r   ru  r   r   rs  r   r   )rN   rt  r   ru  r   ztype[_TAnyCC] | Noners  r   )rN   rt  r   3t.Callable[[ds.ContentSecurityPolicy], None] | Noners  rk   )rN   rt  r   rv  r   r   rs  r   )rN   rt  r   rv  r   ztype[_TAnyCSP] | Noners  r   )rN   rt  r   z't.Callable[[ds.HeaderSet], None] | Noners  zds.HeaderSet)rN   rt  rs  z
ds.IfRange)rN   rt  r   rr  rs  zds.Range | None)rN   rt  r   z*t.Callable[[ds.ContentRange], None] | Noners  zds.ContentRange | None)F)r   rJ   r   rr  rs  rJ   )r   rJ   rs  ztuple[str, bool])r   rL  rs  ztuple[None, None])r   rt  rs  z$tuple[str, bool] | tuple[None, None])rN   rt  rs  zds.ETags)r   bytesrs  rJ   )rN   rt  rs  zdatetime | None)r  z2datetime | date | int | float | struct_time | Noners  rJ   )rN   rt  rs  ztimedelta | None)r  timedelta | int | Noners  rt  )NNNT)r(  r   r   rt  r   zbytes | Noner$  zdatetime | str | Noner%  rr  rs  rr  ))r   r   )r+  "ds.Headers | list[tuple[str, str]]r,  zt.Iterable[str]rs  rL  )r+  ry  rs  rL  )r`   rJ   rs  rr  )r`   zWSGIEnvironment | str | Noner   z#type[ds.MultiDict[str, str]] | Noners  zds.MultiDict[str, str])r   NNr   NFFTi  NF)rc   rJ   rN   rJ   r`  rx  r   z#str | datetime | int | float | Nonera  rt  rb  rt  rc  rr  rd  rr  re  rr  rf  r  rg  rt  rh  rr  rs  rJ   )r   
int | Noner   rz  r   rz  rs  rr  )_
__future__r   email.utilsr  retypingr   r^  r   r   r   r   r	   enumr
   hashlibr   r   r   urllib.parser   r   urllib.requestr   rt   	_internalr   r   TYPE_CHECKING_typeshed.wsgir   	frozensetrK   compiler   r2  r4  HTTP_STATUS_CODESr6   rB   rR   rX   rd   rj   rr   rw   r   ASCIIr   r   VERBOSEr   r   r   r   TypeVarr   overloadr   r   r   r   r   r   r   r   r   r   r   r   r   r   r  r  r  r'  r-  r0  r*  r/  r;  Ar[  r\  rO  updaterangern  r   r   rp  r   sansiorq  r&  r>   r?   r@   <module>r     s;	   "  	               @ ! !??.S 2::?@  	 AA	A A 	A
 A A A 	(A A A 	A A 	A A 	A  	!A" #A$ %A& 'A( )A* +A, 	-A. 	/A0 1A2 3A4 	5A6 7A8 9A: 	;A< 	=A> 	(?A@ 	AAB CAD EAF 	GAH 	IAJ 	#KAL 	MAN 	!OAP 	*QAR 	SAT UAV 	WAX 	YAZ [A\ 	]A^ _A` 	aAb 	 cAd 	eAf 	*gAh 
	(	 					%	"				(AA H"4 " 4  !H$+\-`
H6GV BJJ9J &BJJ'>bhhO BJJ
 HHrzz  2::k2884 Xv bjj)2884aii[9  <  <  V  V 8<22422j !))I%E
F  LP!!H! ! !  LPH 
 	  LP $44H4 
4 	4< 199Z'AB  FJ##B# # #  FJB 
 	  FJ!%!!!!B!! 
!! 	!!L :>==6= =B.& /38#8#'+8#8#z =A3393 3l  4  4 6  6
)2":"< EI:A::@,, +/ ##
# # )	#
 # 
#P  ?/ 
0
-1 04"=("=	,"= "=J !bjj!GN 2::<bddC !'2     6uT{6U6U4%56  &*37X	XX $X 1	X
 X X X X X X X X 	Xv'1;	& # (r?   