
     h>                     J   d Z ddlmZ ddlZddlZ ej
                  e      Z	 ddlmZ	 e	j                  dk(  rej                  d       d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mZmZmZmZmZ ddlmZm Z m!Z!m"Z"m#Z# ddl$m%Z% g dZ&dZ'dZ(g dZ)ddddddddddddddddZ*d Z+i Z,d Z-d Z.	 dGd!Z/e,j`                  e/_1        dHd"Z2 G d# d$e      Z3d a4 e5g d%      Z6dId&Z7 eejp                  js                  d'            r e7         ed(  ed)      D              Z: ed*  ed)      D              Z;dJd+Z<dKd,Z= ed-      j|                  Z?dKd.Zejp                  js                  d/      xs d0Z@e"re@d1v rdd2lAmBZB d3 ZCd4 ZDd5ZEn7e@d6v rddlmZ dd7lmFZF eFd8k\  ZGi ZHd9 ZCd:ZEne@d;v sJ dd<lImJZK dd=lmLZL d> ZCd? ZDd@ZEe@eEk(  rdxZZ	erdAnde	rdBnddCeEz   fD  cg c]  } | r|  c} ZMdAeMv rdDZNydBeMv rdEZNydFZNy# e$ r dZ	Y w xY w# e$ r dZY w xY wc c} w )Lzlpasslib.crypto.digest -- crytographic helpers used by the password hashes in passlib

.. versionadded:: 1.7
    )divisionN)pbkdf2_hmachashlibz*ignoring pure-python hashlib.pbkdf2_hmac())Struct)warn)exc)
join_bytesto_native_strjoin_byte_valuesto_bytesSequenceMixinas_bool)irange	int_typesunicode_or_bytes_typesPY3
error_from)memoized_property)lookup_hashHashInfonorm_hash_namecompile_hmacpbkdf1r   l    l    ))md2r   )md5r   )sha1zsha-1)sha224zsha-224zsha2-224)sha256zsha-256zsha2-256)sha384zsha-384zsha2-384)sha512zsha-512zsha2-512)blake2bzblake-2b)blake2szblake-2s)md4r#   )	ripemd160z
ripemd-160ripemd)@      )    r&   )   r&   )   r&   )   r&   )0   r'   )r+      )r(      )r,   h   )r&   H   )r)      )r!   r"   r#   r   r   r   r   r   sha3_224sha3_256sha3_384sha3_512r    shake128shake256c                      i } t        t        j                  t        dg      z        D ]2  }t	        |      }|j
                  |j                  f| |j                  <   4 | S )z
    internal helper used to generate ``_fallback_info`` dict.
    currently only run manually to update the above list;
    not invoked at runtime.
    r#   )sortedr   algorithms_availablesetr   digest_size
block_sizename)outalginfos      W/var/www/html/Resume-Scraper/venv/lib/python3.12/site-packages/passlib/crypto/digest.py_gen_fallback_inforC   z   s[     Cg22S%\AB =3**DOO<DII= J    c                 h   | }t        | t              st        | dd      } t        j                  dd| j                         j                               } | j                  d      r| dd } | j                  d      r| dd	 } d
 } ||       }|r|S t        j                  d|       }|r^|j                  ddd      \  }}}|r||z  }|}|r|d|z   z  }|r|dz  }||z  } ||      }|r|S t        j                  d|||       ||fS | }| j                  dd      }t        j                  d|||       ||fS )aA  
    internal helper used by :func:`lookup_hash` --
    normalize arbitrary hash name to hashlib format.
    if name not recognized, returns dummy record and issues a warning.

    :arg name:
        unnormalized name

    :returns:
        tuple with 2+ elements: ``(hashlib_name, iana_name|None, ... 0+ aliases)``.
    zutf-8z	hash namez[_ /]-zscram-   Nz-plusc                 *    t         D ]
  }| |v s|c S  y N)_known_hash_names)r>   rows     rB   check_tablez&_get_hash_aliases.<locals>.check_table   s    $ 	Cs{
	rD   z8(?i)^(?P<name>[a-z]+)-?(?P<rev>\d)?-?(?P<size>\d{3,4})?$r>   revsize_z0normalizing unrecognized hash name %r => %r / %rz;normalizing unrecognized hash name and format %r => %r / %r)
isinstancestrr
   resubstriplower
startswithendswithmatchgrouplogrA   replacewarning)	r>   origrM   resultm	iana_namerN   rO   hashlib_names	            rB   _get_hash_aliasesrc      se    DdC T7K866'3

 2 2 45Dx ABx==!9D F 	LdSA  !wwvuf=	3I t#I#D LY'M 	C|Y	0 "" 	||C-Q,		3 ""rD   c                       j                  d      s dvr	 t        t               S t        j                  	   d       d fd	} |_        d|_        d z  |_        |S # t        $ r Y Fw xY w# t        $ r Y nw xY w dk(  rdd	l	m
} |S y
)a  
    internal helper used by :func:`lookup_hash` --
    lookup hash constructor by name

    :arg name:
        name (normalized to hashlib format, e.g. ``"sha256"``)

    :returns:
        hash constructor, e.g. ``hashlib.sha256()``;
        or None if hash can't be located.
    rP   )new
algorithmsrD   c                      |       S rJ    )msgr>   new_ssl_hashs    rB   constz_get_hash_const.<locals>.const   s    c**rD   r   zMwrapper for hashlib.new(%r),
generated by passlib.crypto.digest.lookup_hash()r#   r   )r#   NrD   )rW   getattrr   AttributeErrorre   __name__
__module____doc__
ValueErrorpasslib.crypto._md4r#   )r>   rk   r#   rj   s   `  @rB   _get_hash_constrt      s     ??3D0E$E	7D))
 ;;LT3	+$LOST'  		   u}+

 s"   A$ 	A3 $	A0/A03	A?>A?Fc                 t  	 t         }	 ||    S # t        t        f$ r Y nw xY w|rd}d}t        | t              rOt        |       }|d   		sJ 	| k7  rt        	|      }||| <   |S t        	      }|rt        rz	t        vrrd	fd	}nkt        | t              r| S t        |       r7| }t         |       j                        }|d   	t        	      }|n||u rnd}nt        j                  | dd      t        |||	      }||||<   |r.|D ])  		s|j                  	      d|fv s
J d
	z         ||	<   + |S )af  
    Returns a :class:`HashInfo` record containing information about a given hash function.
    Can be used to look up a hash constructor by name, normalize hash name representation, etc.

    :arg digest:
        This can be any of:

        * A string containing a :mod:`!hashlib` digest name (e.g. ``"sha256"``),
        * A string containing an IANA-assigned hash name,
        * A digest constructor function (e.g. ``hashlib.sha256``).

        Case is ignored, underscores are converted to hyphens,
        and various other cleanups are made.

    :param required:
        By default (True), this function will throw an :exc:`~passlib.exc.UnknownHashError` if no hash constructor
        can be found, or if the hash is not actually available.

        If this flag is False, it will instead return a dummy :class:`!HashInfo` record
        which will defer throwing the error until it's constructor function is called.
        This is mainly used by :func:`norm_hash_name`.

    :param return_unknown:

        .. deprecated:: 1.7.3

            deprecated, and will be removed in passlib 2.0.
            this acts like inverse of **required**.

    :returns HashInfo:
        :class:`HashInfo` instance containing information about specified digest.

        Multiple calls resolving to the same hash should always
        return the same :class:`!HashInfo` instance.
    FTr   requiredc                      t        dz        )Nz4%r disabled for fips by passlib set_mock_fips_mode())rr   )sourcer>   s    rB   rk   zlookup_hash.<locals>.constA  s     !WZ^!^__rD   Nzdigest name or constructordigest)rk   namesrw   z%r already in cacherl   )_hash_info_cacheKeyError	TypeErrorrQ   r   rc   r   rt   mock_fips_mode_fips_algorithmsr   callabler>   r   ExpectedTypeErrorget)
rz   return_unknownrw   cachecache_by_name	name_listrA   rk   other_constr>   s
            @rB   r   r      s   L EV}i  
  M&01%f-	| 6>th7D E&MK  % ^4D(D` 
FH	%	&	 %egll3	|%d+ E!  "M ##F,H(SS %y8DD e 	#Dyy4,6T8MPT8TT"d	# Ks      c                     t        | d      }|j                  r't        d|j                  z   t        j
                         |dk(  r|j                  S |dk(  r|j                  S t        d|      )a<  Normalize hash function name (convenience wrapper for :func:`lookup_hash`).

    :arg name:
        Original hash function name.

        This name can be a Python :mod:`~hashlib` digest name,
        a SCRAM mechanism name, IANA assigned hash name, etc.
        Case is ignored, and underscores are converted to hyphens.

    :param format:
        Naming convention to normalize to.
        Possible values are:

        * ``"hashlib"`` (the default) - normalizes name to be compatible
          with Python's :mod:`!hashlib`.

        * ``"iana"`` - normalizes name to IANA-assigned hash function name.
          For hashes which IANA hasn't assigned a name for, this issues a warning,
          and then uses a heuristic to return a "best guess" name.

    :returns:
        Hash name, returned as native :class:`!str`.
    Frv   znorm_hash_name(): r   ianazunknown format: )	r   unknownr   
error_textr   PasslibRuntimeWarningr>   ra   rr   )r>   formatrA   s      rB   r   r   s  sc    0 te,D||!DOO3S5N5NOyy	6	~~9::rD   c                   v    e Zd ZdZdZdZdZdZdZdZ	dZ
dZ	 ddZd Zd Zed        Zed	        Zed
        Zy)r   a  
    Record containing information about a given hash algorithm, as returned :func:`lookup_hash`.

    This class exposes the following attributes:

    .. autoattribute:: const
    .. autoattribute:: digest_size
    .. autoattribute:: block_size
    .. autoattribute:: name
    .. autoattribute:: iana_name
    .. autoattribute:: aliases
    .. autoattribute:: supported

    This object can also be treated a 3-element sequence
    containing ``(const, digest_size, block_size)``.
    Nrh   Fc                     |d   x _         |d    _        |dd  _         fd}|#|t        v rdz  }ndz  }d _         ||       y	  |       }| _
        |j                   _        |j                   _        t        |j                               |j                  k7  rt        d j                   z        |j                    j                   k7  r8t!        d j                   d|d|j                   t"        j$                         yy# t
        $ rR}d	t        |      j                         v rd
z  }nddt        |      j                  d|d} ||       Y d}~yd}~ww xY w)a  
        initialize new instance.
        :arg const:
            hash constructor
        :arg names:
            list of 2+ names. should be list of ``(name, iana_name, ... 0+ aliases)``.
            names must be lower-case. only iana name may be None.
        r         Nc                      d fd	}r |        	  _         |_        	 t           \  _        _        y# t
        $ r Y yw xY w)zc
            helper that installs stub constructor which throws specified error <msg>.
            c                 0    t        j                        rJ   )r   UnknownHashError)ry   ri   r>   s    rB   rk   z8HashInfo.__init__.<locals>.use_stub_const.<locals>.const  s    **355rD   Nrl   )r   rk   _fallback_infor<   r=   r}   )ri   rk   r>   rw   selfs   ` rB   use_stub_constz)HashInfo.__init__.<locals>.use_stub_const  sN    6 +!DODJ4B44H1 $/ s   : 	AAzunsupported hash: %rzunknown hash: %rTzdisabled for fipsz%r hash disabled for fipszinternal error in z constructor
(z: )z"%r constructor failed sanity checkzinconsistent digest name: z resolved to z, which reports name as )r>   ra   aliasesrK   r   rr   rR   rV   typero   rk   r<   r=   lenrz   RuntimeErrorr   r   r   )	r   rk   r{   rw   r   ri   hasherrr>   s	   `  `    @rB   __init__zHashInfo.__init__  s`    !8#tyqQRy	& =)),t3(4/#3	7D 
++// t{{}!1!11CdiiOPP 99		!))UDII/030I0IK "-  
	
 #c#hnn&661D8 7 GKDQTIL^L^`cd3
	s   D 	E,AE''E,c                 V    d| j                   d| j                  d| j                  dS )Nz<lookup_hash(z): digest_size=z block_size=r   )r>   r<   r=   r   s    rB   __repr__zHashInfo.__repr__  s!    		4++T__> 	>rD   c                 H    | j                   | j                  | j                  fS rJ   )rk   r<   r=   r   s    rB   	_as_tuplezHashInfo._as_tuple  s    zz4++T__<<rD   c                     | j                   du S )zy
        whether hash is available for use
        (if False, constructor will throw UnknownHashError if called)
        N)r   r   s    rB   	supportedzHashInfo.supported  s     $&&rD   c                 b    t         sy	 t        | j                  ddd       y# t        $ r Y yw xY w)z5helper to detect if hash is supported by fastpbkdf2()N   p   sr   TF)_fast_pbkdf2_hmacr>   rr   r   s    rB   supported_by_fastpbkdf2z HashInfo.supported_by_fastpbkdf2$  s6     !	diitQ7 		   " 	..c                 b    t         sy	 t        | j                  ddd       y# t        $ r Y yw xY w)z>helper to detect if hash is supported by hashlib.pbkdf2_hmac()Nr   r   r   TF)_stdlib_pbkdf2_hmacr>   rr   r   s    rB   supported_by_hashlib_pbkdf2z$HashInfo.supported_by_hashlib_pbkdf20  s6     #			4q9 		r   T)ro   rp   __qualname__rq   r>   ra   r   rk   r<   r=   r   r   r   r   r   r   r   r   r   rh   rD   rB   r   r     s    * D I G E K J J G )-GKX>= ' ' 	 	 	 	rD   r   )r   r   r   r   r    r2   r3   r4   r5   	shake_128	shake_256c                 0    | a t        j                          y)zW
    UT helper which monkeypatches lookup_hash() internals to replicate FIPS mode.
    N)r   r   clear_cache)enables    rB   _set_mock_fips_moder   _  s    
 NrD   PASSLIB_MOCK_FIPS_MODEc              #   &   K   | ]	  }|d z    yw)\   Nrh   .0xs     rB   	<genexpr>r   q       =Aa$h=      c              #   &   K   | ]	  }|d z    yw)6   Nrh   r   s     rB   r   r   r  r   r   c                   	
 t        |       }|\  }}}|dk\  sJ d       t        |t              st        |d      }t	        |      }||kD  r ||      j                         }|}||k  r|d||z
  z  z  } ||j                  t                    j                  	 ||j                  t                    j                  
|r	
fd}n	
fd}||_
        |S )a  
    This function returns an efficient HMAC function, hardcoded with a specific digest & key.
    It can be used via ``hmac = compile_hmac(digest, key)``.

    :arg digest:
        digest name or constructor.

    :arg key:
        secret key as :class:`!bytes` or :class:`!unicode` (unicode will be encoded using utf-8).

    :param multipart:
        request a multipart constructor instead (see return description).

    :returns:
        By default, the returned function has the signature ``hmac(msg) -> digest output``.

        However, if ``multipart=True``, the returned function has the signature
        ``hmac() -> update, finalize``, where ``update(msg)`` may be called multiple times,
        and ``finalize() -> digest_output`` may be repeatedly called at any point to
        calculate the HMAC digest so far.

        The returned object will also have a ``digest_info`` attribute, containing
        a :class:`lookup_hash` instance for the specified digest.

    This function exists, and has the weird signature it does, in order to squeeze as
    provide as much efficiency as possible, by omitting much of the setup cost
    and features of the stdlib :mod:`hmac` module.
    r)   zblock size too smallkeyparam    c                  <            fd} j                   | fS )z)generated by compile_hmac(multipart=True)c                  p            } | j                  j                                | j                         S rJ   updaterz   )outer_outer_copyinners    rB   finalizez,compile_hmac.<locals>.hmac.<locals>.finalize  s(    #U\\^,||~%rD   )r   )r   r   _inner_copyr   s    @rB   hmaczcompile_hmac.<locals>.hmac  s     ME& <<))rD   c                             }|j                  |                }|j                  |j                                |j                         S )zgenerated by compile_hmac()r   )ri   r   r   r   r   s      rB   r   zcompile_hmac.<locals>.hmac  s;    MELLMELL(<<>!rD   )r   rQ   bytesr   r   rz   	translate	_TRANS_36copy	_TRANS_5Cdigest_info)rz   r   	multipartr   rk   r<   r=   klenr   r   r   s            @@rB   r   r   t  s    @ f%K%0"E;
333 c5!s%(s8DjCj!jw*t+,, i0166Ki0166K	*	" #DKrD   c                    t        |       \  }}}t        |d      }t        |d      }t        |t              st	        j
                  |dd      |dk  rt        d      ||}nMt        |t              st	        j
                  |d	d
      |dk  rt        d      ||kD  rt        d|d|      ||z   }t        |      D ]  }	 ||      j                         } |d| S )a;  pkcs#5 password-based key derivation v1.5

    :arg digest:
        digest name or constructor.
        
    :arg secret:
        secret to use when generating the key.
        may be :class:`!bytes` or :class:`unicode` (encoded using UTF-8).
        
    :arg salt:
        salt string to use when generating key.
        may be :class:`!bytes` or :class:`unicode` (encoded using UTF-8).

    :param rounds:
        number of rounds to use to generate key.

    :arg keylen:
        number of bytes to generate (if omitted / ``None``, uses digest's native size)

    :returns:
        raw :class:`bytes` of generated key

    .. note::

        This algorithm has been deprecated, new code should use PBKDF2.
        Among other limitations, ``keylen`` cannot be larger
        than the digest size of the specified hash.
    secretr   saltintroundsr   rounds must be at least 1Nint or Nonekeylenr   zkeylen must be at least 0z keylength too large for digest: z > )	r   r   rQ   r   r   r   rr   r   rz   )
rz   r   r   r   r   rk   r<   r=   blockrP   s
             rB   r   r     s    < &1%8"E;
 fH-FD'D fi(##FE8<<z455 ~	*##FM8DD	!455	+	 +/ 0 	0 TMEF^ &e##%&&>rD   z>Lc                   	 t        |d      }t        d      t        |       }|j                  }t        t              st        j                  dd      dk  rt        d      ||}n7t        |t              st        j                  |d	d
      |dk  rt        d      ||z   dz
  |z  }|t        kD  rt        d      |j                  rt        |j                  ||      S |j                  rt        |j                  ||      S t        | |      	t!        |      t#        	fdt%        d|dz         D              d| S )a  pkcs#5 password-based key derivation v2.0 using HMAC + arbitrary digest.

    :arg digest:
        digest name or constructor.

    :arg secret:
        passphrase to use to generate key.
        may be :class:`!bytes` or :class:`unicode` (encoded using UTF-8).

    :arg salt:
        salt string to use when generating key.
        may be :class:`!bytes` or :class:`unicode` (encoded using UTF-8).

    :param rounds:
        number of rounds to use to generate key.

    :arg keylen:
        number of bytes to generate.
        if omitted / ``None``, will use digest's native output size.

    :returns:
        raw bytes of generated key

    .. versionchanged:: 1.7

        This function will use the first available of the following backends:

        * `fastpbk2 <https://pypi.python.org/pypi/fastpbkdf2>`_
        * :func:`hashlib.pbkdf2_hmac` (only available in py2 >= 2.7.8, and py3 >= 3.4)
        * builtin pure-python backend

        See :data:`passlib.crypto.digest.PBKDF2_BACKENDS` to determine
        which backend(s) are in use.
    r   r   r   r   r   r   r   Nr   r   zkeylen must be at least 1zkeylen too long for digestc           
   3   V   K   | ]   }  t        |      z                " y wrJ   )_pack_uint32)r   i
calc_block
keyed_hmacr   r   s     rB   r   zpbkdf2_hmac.<locals>.<genexpr>c  s0       	:z$a*@A6Js   &))r   r   r<   rQ   r   r   r   rr   
MAX_UINT32OverflowErrorr   r   r>   r   r   r   _get_pbkdf2_looperr	   r   )
rz   r   r   r   r   r   r<   block_countr   r   s
     ``    @@rB   r   r     sf   H fH-FD'D f%K))K fi(##FE8<<z455 ~	*##FM8DD	!455 K'!+;KZ899 ** !1!164PP
 ..";#3#3VT66RR ff-J $K0J  ;?+  v rD   PASSLIB_PBKDF2_BACKENDany)r   
from-bytes)partialc                 "    t        t        |       S rJ   )r   _pbkdf2_looperr<   s    rB   r   r   r  s    ~{33rD   c                     t         j                  }d} |||      }t        |dz
        D ]  } ||      }| |||      z  } |j                  | |      S )zk
        py3-only implementation of pbkdf2 inner loop;
        uses 'int.from_bytes' + integer XOR
        bigr   )r   
from_bytesr   r   )r<   r   rz   r   r   BIGaccumrP   s           rB   r   r   u  sd    
 ^^
63'
# 	-A'FZ,,E	- ~~k3//rD   r   )r   unpackr   )sys_bitsr&   c           
         	 t         |    S # t        $ r Y nw xY wt        r| dz  s| dz	  }d|z  }n4| dz  s!t        r| dz	  }d|z  }|dz  }n| dz	  }d|z  }nt        d| z        t	        |      }t        | d	j                  d
 t        |      D              d	j                  d t        |      D                    } dj                  di |}t        |      D ]  }|d||fz  z  } | dj                  di |z  }t        |dd      }t        t        |      }t               }	t        |||	       |	d   }
	 ||
_        |
t         | <   |
S )a)  
        We want a helper function which performs equivalent of the following::

          def helper(keyed_hmac, digest, rounds):
              accum = digest
              for _ in irange(rounds - 1):
                  digest = keyed_hmac(digest)
                  accum ^= digest
              return accum

        However, no efficient way to implement "bytes ^ bytes" in python.
        Instead, using approach where we dynamically compile a helper function based
        on digest size.  Instead of a single `accum` var, this helper breaks the digest
        into a series of integers.

        It stores these in a series of`accum_<i>` vars, and performs `accum ^= digest`
        by unpacking digest and perform xor for each "accum_<i> ^= digest_<i>".
        this keeps everything in locals, avoiding excessive list creation, encoding or decoding,
        etc.

        :param digest_size:
            digest size to compile for, in bytes. (must be multiple of 4).

        :return:
            helper function with call signature outlined above.
              z=%dQz=%dQIr   r   z=%dIzunsupported digest size: %dz, c              3   &   K   | ]	  }d |z    yw)zacc_%dNrh   r   r   s     rB   r   z%_get_pbkdf2_looper.<locals>.<genexpr>  s      E!A Er   c              3   &   K   | ]	  }d |z    yw)zdig_%dNrh   r  s     rB   r   z%_get_pbkdf2_looper.<locals>.<genexpr>  s     !F1(Q,!Fr   )r<   
accum_varsdigest_varsa   def helper(keyed_hmac, digest, rounds):
    '''pbkdf2 loop helper for digest_size={digest_size}'''
    unpack_digest = struct.unpack
    {accum_vars} = unpack_digest(digest)
    for _ in irange(1, rounds):
        digest = keyed_hmac(digest)
        {digest_vars} = unpack_digest(digest)
z        acc_%d ^= dig_%d
z%    return struct.pack({accum_vars})
z9<generated by passlib.crypto.digest._get_pbkdf2_looper()>exec)r   structhelperrh   )_looper_cacher}   _have_64_bitNotImplementedErrorr   dictjoinr   r   compileeval
__source__)r<   countfmtr  tdictry   r   codegdictldictr  s              rB   r   r     s   <	 -- 		 c 1 A%E5.Cs"$)o
 %)un &&Ck&QRR
 #yy Eve} EE		!Fu!FF
J &   	>A41v==F	> 	G@GGP%PP
 vZ\bcF62T5% x &F
 &,k"s    	r   )r   hexlify)r  )int_to_bytesc                     t         S rJ   )r   r   s    rB   r   r     s    rD   c                     t         }t         ||      d      }t        |dz
        D ]  } | |      }|t         ||      d      z  }! t        |t	        |            S )Nr)   r   )_hexlifyr   r   r  r   )r   rz   r   r  r   rP   s         rB   r   r     sa    GFOR(
# 	.A'FS"--E	. E3v;//rD   r  
fastpbkdf2zhashlib-sslzbuiltin-r   gffffff?r   )FT)r   r   )FrJ   )Orq   
__future__r   r   logging	getLoggerro   r[   r   r   rp   debugImportErrorrS   osr  r   warningsr   r  r   passlibr   passlib.utilsr	   r
   r   r   r   r   passlib.utils.compatr   r   r   r   r   passlib.utils.decorr   __all__r   
MAX_UINT64rK   r   rC   r|   rc   rt   r   clearr   r   r   r   r;   r   r   environr   r   r   r   r   packr   _force_backend	functoolsr   r   r   _builtin_backendr   r
  r	  binasciir  r  r  PBKDF2_BACKENDSPBKDF2_SPEED_FACTOR)bs   0rB   <module>r4     s      'g''1	:%%2 			>?" 
 	  ; 1 1 [ [ 1& 
 
 H !(
  =#@-b 04sl +00  ;Fd} d` 
    ( 2::>>234 ===	===	K`9~ d|  aP  89BU>22!40 $66&NL MdL   ///
 -*0 ! %%.22+ &L4(Md!!  	   ?"o% S     ds)   (H !H %H HHHH