o
    O@h(                      @   s   d Z ddlZddlZddlZddlZddlZddlmZ ddlm	Z	m
Z
 ddlmZ ddlmZ ddlmZ edZd	ZG d
d deZG dd deZdd Zdd Zdd Zdd Zd(ddZd)ddZG dd dZddedfd d!Zddedfd"d#ZG d$d% d%ZG d&d' d'eZ dS )*ae  
Functions for creating and restoring url-safe signed JSON objects.

The format used looks like this:

>>> signing.dumps("hello")
'ImhlbGxvIg:1QaUZC:YIye-ze3TTx7gtSv422nZA4sgmk'

There are two components here, separated by a ':'. The first component is a
URLsafe base64 encoded JSON of the object passed to dumps(). The second
component is a base64 encoded hmac/SHA1 hash of "$first_component:$secret"

signing.loads(s) checks the signature and returns the deserialized object.
If the signature fails, a BadSignature exception is raised.

>>> signing.loads("ImhlbGxvIg:1QaUZC:YIye-ze3TTx7gtSv422nZA4sgmk")
'hello'
>>> signing.loads("ImhlbGxvIg:1QaUZC:YIye-ze3TTx7gtSv422nZA4sgmk-modified")
...
BadSignature: Signature failed: ImhlbGxvIg:1QaUZC:YIye-ze3TTx7gtSv422nZA4sgmk-modified

You can optionally compress the JSON prior to base64 encoding it to save
space, using the compress=True argument. This checks if compression actually
helps and only applies compression if the result is a shorter string:

>>> signing.dumps(list(range(1, 20)), compress=True)
'.eJwFwcERACAIwLCF-rCiILN47r-GyZVJsNgkxaFxoDgxcOHGxMKD_T7vhAml:1QaUaL:BA0thEZrp4FQVXIXuOvYJtLJSrQ'

The fact that the string is compressed is signalled by the prefixed '.' at the
start of the base64 JSON.

There are 65 url-safe characters: the 64 used by url-safe base64 and the ':'.
These functions make use of all of them.
    N)settings)constant_time_comparesalted_hmac)force_bytes)import_string)_lazy_re_compilez^[A-z0-9-_=]*$Z>0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyzc                   @      e Zd ZdZdS )BadSignaturezSignature does not match.N__name__
__module____qualname____doc__ r   r   =D:\git\psytec\devenv\Lib\site-packages\django/core/signing.pyr	   4       r	   c                   @   r   )SignatureExpiredz3Signature timestamp is older than required max_age.Nr
   r   r   r   r   r   :   r   r   c                 C   sZ   | dkrdS | dk rdnd}t | } d}| dkr)t| d\} }t| | }| dks|| S )Nr   0- >   )absdivmodBASE62_ALPHABET)ssignencoded	remainderr   r   r   
b62_encode@   s   r   c                 C   sT   | dkrdS d}| d dkr| dd  } d}d}| D ]}|d t | }q|| S )Nr   r      r   r   )r   index)r   r   decodeddigitr   r   r   
b62_decodeL   s   r$   c                 C   s   t | dS )N   =)base64urlsafe_b64encodestrip)r   r   r   r   
b64_encodeY      r)   c                 C   s    dt |  d  }t| | S )Nr%      )lenr&   urlsafe_b64decode)r   padr   r   r   
b64_decode]   s   r/   sha1c                 C   s   t t| |||d  S )N	algorithm)r)   r   digestdecode)saltvaluekeyr2   r   r   r   base64_hmacb   s
   r8   %django.core.signing.get_cookie_signerc                 C   s$   t tj}ttj}|d| | dS )Ns   django.http.cookiesr5   )r   r   ZSIGNING_BACKENDr   
SECRET_KEY)r5   Signerr7   r   r   r   get_cookie_signerh   s   

r=   c                   @   s    e Zd ZdZdd Zdd ZdS )JSONSerializerzW
    Simple wrapper around json to be used in signing.dumps and
    signing.loads.
    c                 C   s   t j|dddS )N),:)
separatorslatin-1)jsondumpsencode)selfobjr   r   r   rD   t   s   zJSONSerializer.dumpsc                 C   s   t |dS )NrB   )rC   loadsr4   )rF   datar   r   r   rH   w   r*   zJSONSerializer.loadsN)r   r   r   r   rD   rH   r   r   r   r   r>   n   s    r>   zdjango.core.signingFc                 C      t ||dj| ||dS )a  
    Return URL-safe, hmac signed base64 compressed JSON string. If key is
    None, use settings.SECRET_KEY instead. The hmac algorithm is the default
    Signer algorithm.

    If compress is True (not the default), check if compressing using zlib can
    save some space. Prepend a '.' to signify compression. This is included
    in the signature, to protect against zip bombs.

    Salt can be used to namespace the hash, so that a signed string is
    only valid for a given namespace. Leaving this at the default
    value or re-using a salt value across different parts of your
    application without good cause is a security risk.

    The serializer is expected to return a bytestring.
    r:   )
serializercompress)TimestampSignersign_object)rG   r7   r5   rK   rL   r   r   r   rD   {   s   rD   c                 C   rJ   )z|
    Reverse of dumps(), raise BadSignature if signature fails.

    The serializer is expected to accept a bytestring.
    r:   )rK   max_age)rM   unsign_object)r   r7   r5   rK   rO   r   r   r   rH      s   rH   c                   @   sH   e Zd ZdddZdd Zdd Zd	d
 ZedfddZefddZ	dS )r<   Nr@   c                 C   sR   |pt j| _|| _t| jrtd| |p d| jj| jj	f | _
|p%d| _d S )NzJUnsafe Signer separator: %r (cannot be empty or consist of only A-z0-9-_=)z%s.%ssha256)r   r;   r7   sep_SEP_UNSAFEmatch
ValueError	__class__r   r   r5   r2   )rF   r7   rR   r5   r2   r   r   r   __init__   s   zSigner.__init__c                 C   s   t | jd || j| jdS )NZsignerr1   )r8   r5   r7   r2   rF   r6   r   r   r   	signature   s   zSigner.signaturec                 C   s   d|| j | |f S Nz%s%s%s)rR   rY   rX   r   r   r   r      s   zSigner.signc                 C   sJ   | j |vrtd| j  || j d\}}t|| |r|S td| )NzNo "%s" found in valuer   zSignature "%s" does not match)rR   r	   rsplitr   rY   )rF   Zsigned_valuer6   sigr   r   r   unsign   s   
zSigner.unsignFc                 C   s\   |  |}d}|rt|}t|t|d k r|}d}t| }|r)d| }| |S )ae  
        Return URL-safe, hmac signed base64 compressed JSON string.

        If compress is True (not the default), check if compressing using zlib
        can save some space. Prepend a '.' to signify compression. This is
        included in the signature, to protect against zip bombs.

        The serializer is expected to return a bytestring.
        Fr   T.)rD   zlibrL   r,   r)   r4   r   )rF   rG   rK   rL   rI   Zis_compressed
compressedbase64dr   r   r   rN      s   


zSigner.sign_objectc                 K   sX   | j |fi | }|d d dk}|r|dd  }t|}|r&t|}| |S )Nr      .)r]   rE   r/   r_   
decompressrH   )rF   Z
signed_objrK   kwargsra   rc   rI   r   r   r   rP      s   
zSigner.unsign_object)Nr@   NN)
r   r   r   rW   rY   r   r]   r>   rN   rP   r   r   r   r   r<      s    
r<   c                       s2   e Zd Zdd Z fddZd fdd	Z  ZS )	rM   c                 C   s   t tt S N)r   inttime)rF   r   r   r   	timestamp   r*   zTimestampSigner.timestampc                    s    d|| j |  f }t |S rZ   )rR   rh   superr   rX   rV   r   r   r      s   zTimestampSigner.signNc                    sj   t  |}|| jd\}}t|}|dur3t|tjr!| }t		 | }||kr3t
d||f |S )zk
        Retrieve original value and check it wasn't signed more
        than max_age seconds ago.
        r   NzSignature age %s > %s seconds)ri   r]   r[   rR   r$   
isinstancedatetime	timedeltatotal_secondsrg   r   )rF   r6   rO   resultrh   Zagerj   r   r   r]      s   zTimestampSigner.unsignre   )r   r   r   rh   r   r]   __classcell__r   r   rj   r   rM      s    rM   )r0   )r9   )!r   r&   rl   rC   rg   r_   Zdjango.confr   Zdjango.utils.cryptor   r   Zdjango.utils.encodingr   Zdjango.utils.module_loadingr   Zdjango.utils.regex_helperr   rS   r   	Exceptionr	   r   r   r$   r)   r/   r8   r=   r>   rD   rH   r<   rM   r   r   r   r   <module>   s8    #



F