U
    ʼbX                     @   s  d Z ddlZddlmZmZ ddlmZmZ ddlm	Z	m
Z
mZ ddlmZ ddlmZmZ dd	lmZ dd
lmZmZ eeZdZdZdZdd edD ZzeZW n ek
r   e e!fZY nX dZ"dd Z#dd Z$dd Z%dd Z&dd Z'G dd de(Z)G dd de(Z*dS ) ze
hpack/hpack
~~~~~~~~~~~

Implements the HPACK header compression algorithm as detailed by the IETF.
    N   )HeaderTabletable_entry_size)to_byteto_bytes)HPACKDecodingErrorOversizedHeaderListErrorInvalidTableSizeError)HuffmanEncoder)REQUEST_CODESREQUEST_CODES_LENGTH)decode_huffman)HeaderTupleNeverIndexedHeaderTuple          @c                 C   s   g | ]}d | d qS )   r    ).0ir   r   //tmp/pip-unpacked-wheel-vloce9av/hpack/hpack.py
<listcomp>   s     r   	   i   c                 C   s<   t | d }t | d }|s0|d}|d}| ||S )zj
    Provides a header as a unicode string if raw is False, otherwise returns
    it as a bytestring.
    r   r   utf-8)r   decode	__class__)headerrawnamevaluer   r   r   _unicode_if_needed,   s    

r!   c                 C   s   t d| | | dk r"td|  |dk s2|dkr>td| t| }| |k rXt| gS |g}| |8 } | dkr|| d@ d  | d	L } qf||  t|S d
S )zn
    This encodes an integer according to the wacky integer encoding rules
    defined in the HPACK spec.
    zEncoding %d with %d bitsr   z)Can only encode positive integers, got %sr      +Prefix bits must be between 1 and 8, got %s         N)logdebug
ValueError_PREFIX_BIT_MAX_NUMBERS	bytearrayappend)integerprefix_bits
max_numberelementsr   r   r   encode_integer9   s&    


r1   c                 C   s   |dk s|dkrt d| t| }d}d}dd| ? }zbt| d |@ }||krt| | }|d7 }|dkr||d |> 7 }n|||> 7 }q|d7 }qRW n  tk
r   td|  Y nX td	|| ||fS )
z
    This decodes an integer according to the wacky integer encoding rules
    defined in the HPACK spec. Returns a tuple of the decoded integer and the
    number of bytes that were consumed from ``data`` in order to get that
    integer.
    r   r"   r#   r      r$   r&   z5Unable to decode HPACK integer representation from %rzDecoded %d, consumed %d bytes)r)   r*   r   
IndexErrorr   r'   r(   )datar.   r/   indexshiftmasknumberZ	next_byter   r   r   decode_integer[   s0    
r9   c                 c   s>   t | tstt|  dd d}|D ]}|| | fV  q&dS )z
    This converts a dictionary to an iterable of two-tuples. This is a
    HPACK-specific function becuase it pulls "special-headers" out first and
    then emits them.
    c                 S   s   t | d S )N   :)	_to_bytes
startswith)kr   r   r   <lambda>       z#_dict_to_iterable.<locals>.<lambda>)keyN)
isinstancedictAssertionErrorsortedkeys)Zheader_dictrE   r@   r   r   r   _dict_to_iterable   s    rF   c                 C   s*   t | tst| } t | tr | S | dS )z"
    Convert string to bytes.
    r   )rA   
basestringstrbytesencode)stringr   r   r   r;      s    
r;   c                   @   sj   e Zd ZdZdd Zedd Zejdd Zddd	ZdddZ	dd Z
dddZdddZdd ZdS )Encoderzm
    An HPACK encoder object. This object takes HTTP headers and emits encoded
    HTTP/2 header blocks.
    c                 C   s   t  | _ttt| _g | _d S N)r   header_tabler
   r   r   huffman_codertable_size_changesselfr   r   r   __init__   s     zEncoder.__init__c                 C   s   | j jS z>
        Controls the size of the HPACK header table.
        rN   maxsizerQ   r   r   r   header_table_size   s    zEncoder.header_table_sizec                 C   s    || j _| j jr| j| d S rM   )rN   rV   resizedrP   r,   rR   r    r   r   r   rW      s    Tc                 C   s   t d| g }t|tr"t|}| jjr@||   d| j_|D ]\}d}t|t	r`|j
 }nt|dkrt|d }t|d t|d f}|| ||| qDd|}t d| |S )a	  
        Takes a set of headers and encodes them into a HPACK-encoded header
        block.

        :param headers: The headers to encode. Must be either an iterable of
                        tuples, an iterable of :class:`HeaderTuple
                        <hpack.struct.HeaderTuple>`, or a ``dict``.

                        If an iterable of tuples, the tuples may be either
                        two-tuples or three-tuples. If they are two-tuples, the
                        tuples must be of the format ``(name, value)``. If they
                        are three-tuples, they must be of the format
                        ``(name, value, sensitive)``, where ``sensitive`` is a
                        boolean value indicating whether the header should be
                        added to header tables anywhere. If not present,
                        ``sensitive`` defaults to ``False``.

                        If an iterable of :class:`HeaderTuple
                        <hpack.struct.HeaderTuple>`, the tuples must always be
                        two-tuples. Instead of using ``sensitive`` as a third
                        tuple entry, use :class:`NeverIndexedHeaderTuple
                        <hpack.struct.NeverIndexedHeaderTuple>` to request that
                        the field never be indexed.

                        .. warning:: HTTP/2 requires that all special headers
                            (headers whose names begin with ``:`` characters)
                            appear at the *start* of the header block. While
                            this method will ensure that happens for ``dict``
                            subclasses, callers using any other iterable of
                            tuples **must** ensure they place their special
                            headers at the start of the iterable.

                            For efficiency reasons users should prefer to use
                            iterables of two-tuples: fixing the ordering of
                            dictionary headers is an expensive operation that
                            should be avoided if possible.

        :param huffman: (optional) Whether to Huffman-encode any header sent as
                        a literal value. Except for use when debugging, it is
                        recommended that this be left enabled.

        :returns: A bytestring containing the HPACK-encoded header block.
        zHPACK encoding %sFr   r   r   r?   zEncoded header block to %s)r'   r(   rA   rB   rF   rN   rX   r,   _encode_table_size_changer   Z	indexablelenr;   addjoin)rR   headershuffmanZheader_blockr   	sensitiver   r   r   rJ      s$    2



zEncoder.encodeFc                 C   s   t d| |\}}|stnt}| j||}|dkr\| ||||}|sX| j|| |S |\}	}}
|
rv| |	}n"| 	|	|||}|s| j|| |S )zQ
        This function takes a header key-value tuple and serializes it.
        zAdding %s to the header tableN)
r'   r(   INDEX_INCREMENTALINDEX_NEVERrN   search_encode_literalr\   _encode_indexed_encode_indexed_literal)rR   Zto_addr`   r_   r   r    indexbitmatchencodedr5   Zperfectr   r   r   r\   	  s*    
   zEncoder.addc                 C   s"   t |d}|d  dO  < t|S )zD
        Encodes a header using the indexed representation.
        r&   r   r$   )r1   rI   )rR   r5   fieldr   r   r   re   5  s    
zEncoder._encode_indexedc                 C   sx   |r| j |}| j |}tt|d}tt|d}|r\|d  dO  < |d  dO  < d|t||t||gS )z
        Encodes a header with a literal name and literal value. If ``indexing``
        is True, the header will be added to the header table: otherwise it
        will not.
        r&   r   r$   r?   )rO   rJ   r1   r[   r]   rI   )rR   r   r    rg   r_   name_len	value_lenr   r   r   rd   =  s    zEncoder._encode_literalc                 C   s|   |t krt|d}n
t|d}|d  t|O  < |rB| j|}tt|d}|rd|d  dO  < dt|t||gS )zv
        Encodes a header with an indexed name and a literal value and performs
        incremental indexing.
              r   r&   r$   r?   )ra   r1   ordrO   rJ   r[   r]   rI   )rR   r5   r    rg   r_   prefixrl   r   r   r   rf   R  s    
zEncoder._encode_indexed_literalc                 C   s@   d}| j D ]*}t|d}|d  dO  < |t|7 }q
g | _ |S )zd
        Produces the encoded form of all header table size change context
        updates.
        r?      r       )rP   r1   rI   )rR   blockZ
size_bytesr   r   r   rZ   h  s    

z!Encoder._encode_table_size_changeN)T)F)F)F)__name__
__module____qualname____doc__rS   propertyrW   setterrJ   r\   re   rd   rf   rZ   r   r   r   r   rL      s   


R
,

rL   c                   @   sp   e Zd ZdZefddZedd Zejdd Zddd	Z	d
d Z
dd Zdd Zdd Zdd Zdd ZdS )Decodera  
    An HPACK decoder object.

    .. versionchanged:: 2.3.0
       Added ``max_header_list_size`` argument.

    :param max_header_list_size: The maximum decompressed size we will allow
        for any single header block. This is a protection against DoS attacks
        that attempt to force the application to expand a relatively small
        amount of data into a really large header list, allowing enormous
        amounts of memory to be allocated.

        If this amount of data is exceeded, a `OversizedHeaderListError
        <hpack.OversizedHeaderListError>` exception will be raised. At this
        point the connection should be shut down, as the HPACK state will no
        longer be useable.

        Defaults to 64kB.
    :type max_header_list_size: ``int``
    c                 C   s   t  | _|| _| jj| _d S rM   )r   rN   max_header_list_sizerV   max_allowed_table_size)rR   r{   r   r   r   rS     s    	zDecoder.__init__c                 C   s   | j jS rT   rU   rQ   r   r   r   rW     s    zDecoder.header_table_sizec                 C   s   || j _d S rM   rU   rY   r   r   r   rW     s    Fc                    sd  t d| t|}g }t|}d}d}||k r$t|| }|d@ rJdnd}	|d@ rZdnd}
|d@ rjdnd}|	r| ||d \}}nZ|
r| ||d \}}n>|r|rtd	| ||d }d}n| 	||d \}}|r|
| |t| 7 }|| jkrtd
| j ||7 }q(|   z fdd|D W S  tk
r^   tdY nX dS )a  
        Takes an HPACK-encoded header block and decodes it into a header set.

        :param data: A bytestring representing a complete HPACK-encoded header
                     block.
        :param raw: (optional) Whether to return the headers as tuples of raw
                    byte strings or to decode them as UTF-8 before returning
                    them. The default value is False, which returns tuples of
                    Unicode strings
        :returns: A list of two-tuples of ``(name, value)`` representing the
                  HPACK-encoded headers, in the order they were decoded.
        :raises HPACKDecodingError: If an error is encountered while decoding
                                    the header block.
        zDecoding %sr   r$   TF@   rr   Nz/Table size update not at the start of the blockz.A header list larger than %d has been receivedc                    s   g | ]}t | qS r   )r!   )r   hr   r   r   r     s     z"Decoder.decode.<locals>.<listcomp>z"Unable to decode headers as UTF-8.)r'   r(   
memoryviewr[   r   _decode_indexed_decode_literal_indexr   _update_encoding_context_decode_literal_no_indexr,   r   r{   r   _assert_valid_table_sizeUnicodeDecodeError)rR   r4   r   Zdata_memr^   Zdata_lenZinflated_sizeZcurrent_indexcurrentZindexedZliteral_indexZencoding_updater   consumedr   r   r   r     s\    








zDecoder.decodec                 C   s   | j | jkrtddS )zs
        Check that the table size set by the encoder is lower than the maximum
        we expect to have.
        z3Encoder did not shrink table size to within the maxN)rW   r|   r	   rQ   r   r   r   r     s    z Decoder._assert_valid_table_sizec                 C   s*   t |d\}}|| jkr td|| _|S )zC
        Handles a byte that updates the encoding context.
        rq   z)Encoder exceeded max allowable table size)r9   r|   r	   rW   )rR   r4   Znew_sizer   r   r   r   r     s    
z Decoder._update_encoding_contextc                 C   s4   t |d\}}t| j| }td|| ||fS )zP
        Decodes a header represented using the indexed representation.
        r&   zDecoded %s, consumed %d)r9   r   rN   get_by_indexr'   r(   )rR   r4   r5   r   r   r   r   r   r     s    zDecoder._decode_indexedc                 C   s   |  |dS )NF_decode_literalrR   r4   r   r   r   r   '  s    z Decoder._decode_literal_no_indexc                 C   s   |  |dS )NTr   r   r   r   r   r   *  s    zDecoder._decode_literal_indexc                 C   s  d}|r"t |d d@ }d}d}n t |d }|d@ }d}|d@ }|rnt||\}}	| j|d }
|	}d}nb|dd	 }t|d
\}}	||	|	|  }
t|
|krtdt |d d@ rt|
}
|	| d }||	| d	 }t|d
\}}	||	|	|  }t||krtdt |d d@ r.t|}|||	 7 }|rLt|
|}n
t|
|}|rj| j	|
| t
d||| ||fS )z>
        Decodes a header represented with a literal.
        r   ?   rn   F   rm      r   Nr&   zTruncated header blockr$   z/Decoded %s, total consumed %d bytes, indexed %s)r   r9   rN   r   r[   r   r   r   r   r\   r'   r(   )rR   r4   Zshould_indexZtotal_consumedZindexed_namerk   Znot_indexableZ	high_byter5   r   r   lengthr    r   r   r   r   r   -  sT    
zDecoder._decode_literalN)F)rt   ru   rv   rw   DEFAULT_MAX_HEADER_LIST_SIZErS   rx   rW   ry   r   r   r   r   r   r   r   r   r   r   r   rz   v  s   


U
	rz   )+rw   loggingtabler   r   compatr   r   
exceptionsr   r   r	   r_   r
   Zhuffman_constantsr   r   Zhuffman_tabler   structr   r   	getLoggerrt   r'   Z
INDEX_NONErb   ra   ranger*   rG   	NameErrorrH   rI   r   r!   r1   r9   rF   r;   objectrL   rz   r   r   r   r   <module>   s4   
")
 Z