U
    luf                     @  sl  d Z ddlmZ ddlZddlZddlZddlZddl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ZddlmZ ddlmZmZmZmZmZmZmZmZm Z m!Z!m"Z"m#Z# dd	l$m%Z%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/m0Z0 ddl1m2Z2m3Z3m4Z4m5Z5 ej6Z7ej8Z8dddddZ9dZ:dZ;dddddZ<dddddZ=ddddd Z>ej?Z@e<ZAe=ZBd!ZCeD ZEd"d#d$d%ZFdd#d&d'ZGd(d)d*d+ZHd,eG d-d.id/fd,d0d-d.id1fd2ZIdd3d4d5d6ZJd7d#d8d9ZKG d:d; d;e&ZLG d<d= d=ZMd"d"d"d>d?d@dAdBZNd?d?dCdDdEZOG dFd, d,e%ZPdS )Ga  Session object for building, serializing, sending, and receiving messages.

The Session object supports serialization, HMAC signatures,
and metadata on messages.

Also defined here are utilities for working with Sessions:
* A SessionFactory to be used as a base class for configurables that work with
Sessions.
* A Message object for convenience that allows attribute-access to the msg dict.
    )annotationsN)b2a_hex)datetimetimezone)compare_digest)IOLoop)AnyBoolCBytesCUnicodeDictDottedObjectNameInstanceIntegerSet
TraitErrorUnicodeobserve)ConfigurableLoggingConfigurable)
get_logger)import_item)	ZMQStream   )protocol_version)adapt)extract_dates
json_cleanjson_defaultsquash_datest.Any)objreturnc                 C  s   t | trJt|  D ]0}t| | | |< t |tr| || t|< qn>t | trtt| D ]\}}t|| |< q\nt | tr| d} | S )z#coerce unicode back to bytestrings.utf8)	
isinstancedictlistkeyssquash_unicodestrpop	enumerateencode)r!   keyiv r0   :/tmp/pip-unpacked-wheel-naub1w99/jupyter_client/session.pyr(   @   s    




r(   @   i   bytesc              
   C  s   zt j| tdddjdddW S  ttfk
r } zBt jt| tdddjddd}tjd| ddd	 | W Y S d
}~X Y nX d
S )z!Convert a json object to a bytes.F)defaultensure_ascii	allow_nanr#   surrogateescape)errorsz#Message serialization failed with:
zk
Supporting this message is deprecated in jupyter-client 7, please make sure your message is JSON-compliant   
stacklevelN)	jsondumpsr   r,   	TypeError
ValueErrorr   warningswarn)r!   epackedr0   r0   r1   json_packer\   s2      
rD   zstr | bytes)sr"   c                 C  s    t | tr| dd} t| S )z,Convert a json bytes or string to an object.r#   replace)r$   r3   decoder<   loadsrE   r0   r0   r1   json_unpackerx   s    
rJ   )or"   c                 C  s   t t| tS )z'Pack an object using the pickle module.)pickler=   r   PICKLE_PROTOCOLrK   r0   r0   r1   pickle_packer   s    rO   s	   <IDS|MSG>r)   r"   c                  C  s2   t d} ddd | dd | dd fD S )zGenerate a new random id.

    Avoids problematic runtime import in stdlib uuid on Python 2.

    Returns
    -------

    id string (16 random bytes as hex-encoded text, chunks separated by '-')
       -c                 s  s   | ]}t |d V  qdS )asciiN)r   rG   ).0xr0   r0   r1   	<genexpr>   s     znew_id.<locals>.<genexpr>N   )osurandomjoin)bufr0   r0   r1   new_id   s    

r\   c                   C  s   t  dS )zReturn new_id as ascii bytesrS   )r\   r,   r0   r0   r0   r1   new_id_bytes   s    r]   zSession.sessionzSession.usernamezSession.keyfile)identuserkeyfileSession )r-   r`   zUse HMAC digests for authentication of messages.
        Setting this flag will generate a new UUID to use as the HMAC key.
            zDon't authenticate messages.)securez	no-secureNone)cfgr"   c                 C  s>   t jdtdd d| kr0d| jks,d| jkr0dS t | j_dS )zSet the default behavior for a config environment to be secure.

    If Session.key/keyfile have not been set, set Session.key to
    a new random UUID.
    zdefault_secure is deprecatedr9   r:   ra   r-   r`   N)r@   rA   DeprecationWarningra   r]   r-   )rf   r0   r0   r1   default_secure   s    rh   r   c                   C  s
   t tS )z#Return timezone-aware UTC timestamp)r   nowutcr0   r0   r0   r1   utcnow   s    rk   c                      s   e Zd ZdZedZeddddddZed	Z	d	d
ddZ
edddZedZdd
ddZddd fddZ  ZS )SessionFactoryz[The Base class for configurables that have a Session, Context, logger,
    and IOLoop.
    rb   lognamer    re   changer"   c                 C  s   t |d | _d S )Nnew)logging	getLoggerlogselfro   r0   r0   r1   _logname_changed   s    zSessionFactory._logname_changedzzmq.ContextrP   c                 C  s   t  S N)zmqContextru   r0   r0   r1   _context_default   s    zSessionFactory._context_defaultzjupyter_client.session.SessionT
allow_noneztornado.ioloop.IOLoopr   c                 C  s   t  S rw   )r   currentrz   r0   r0   r1   _loop_default   s    zSessionFactory._loop_defaultkwargsr"   c                   s(   t  jf | | jdkr$tf || _dS )zInitialize a session factory.N)super__init__sessionra   ru   r   	__class__r0   r1   r      s    
zSessionFactory.__init__)__name__
__module____qualname____doc__r   rm   r   rv   r   contextr{   r   Zloopr   r   __classcell__r0   r0   r   r1   rl      s   rl   c                   @  sj   e Zd ZdZdddddZddd	d
ZddddZddddZdddddZdddddZ	dS )MessagezA simple message object that maps dict keys to attributes.

    A Message can be created from a dict and a dict from a Message instance
    simply by calling dict(msg_obj).dict[str, t.Any]re   )msg_dictr"   c                 C  s:   | j }t| D ]"\}}t|tr,t|}|||< qdS )zInitialize a message.N)__dict__r%   itemsr$   r   )ru   r   dctkr/   r0   r0   r1   r      s
    
zMessage.__init__zt.ItemsView[str, t.Any]rP   c                 C  s   t | j S rw   )iterr   r   rz   r0   r0   r1   __iter__   s    zMessage.__iter__r)   c                 C  s
   t | jS rw   )reprr   rz   r0   r0   r1   __repr__  s    zMessage.__repr__c                 C  s   t | jS rw   )pprintpformatr   rz   r0   r0   r1   __str__  s    zMessage.__str__objectbool)r   r"   c                 C  s
   || j kS rw   r   ru   r   r0   r0   r1   __contains__	  s    zMessage.__contains__r    c                 C  s
   | j | S rw   r   r   r0   r0   r1   __getitem__  s    zMessage.__getitem__N)
r   r   r   r   r   r   r   r   r   r   r0   r0   r0   r1   r      s   	r   zSession | strr   )msg_idmsg_typeusernamer   r"   c                 C  s   t  }t}t S )zCreate a new message header)rk   r   locals)r   r   r   r   dateversionr0   r0   r1   
msg_header  s    r   )msg_or_headerr"   c                 C  sh   | si S z| d }W n< t k
rP   z| d }W n t k
rF    Y nX | }Y nX t|tsdt|}|S )z-Given a message or header, return the header.headerr   )KeyErrorr$   r%   )r   hr0   r0   r1   extract_header  s    

r   c                      sb  e Zd ZdZeddddZeddddZeddddZe	d	d
ddddZ
eddddZe	dd
ddddZeddddZddddZe	dd
ddddZedZeejddddd Zei dd!dZed"Zedd#dZd$dd%d&Ze	d'd
ddd(d)Zed*dd+dZe	d,d
ddd-d.Z e! Z"d/dd0d1Z#e$e%j&dd2Z'ddd3d4Z(e) Z*ed5dd6dZ+eddd7dZ,e	d8d
ddd9d:Z-e Z.e!e/Z0e	d;d
ddd<d=Z1e!e2Z3e	d>d
ddd?d@Z4ed5ddAdZ5ee6ddBdZ7ee8ddCdZ9d
ddD fdEdFZ:d ddGdHZ;d"Z<e=dddIdJZ>dddKdLZ?ddMdNdOdPZ@dddRdSdSdSdMdTdUdVZAdWd$dXdYdZZBddMd[d\d]d^d_ZCdd`dadSdSdbdcdddSdSdSde
dfdgZDddhdWdidddbddjdkdlZEeFjGddfdhdidddddmdndodpZHddqdddrdsdtduZId$ddvdwdxZJdddydzZKddqdddddMd{d|d}ZLd
d
dMd~ddZM  ZNS )ra   a  Object for handling serialization and sending of messages.

    The Session object handles building messages and sending them
    with ZMQ sockets or ZMQStream objects.  Objects can communicate with each
    other over the network via Session objects, and only need to work with the
    dict-based IPython message spec. The Session will handle
    serialization/deserialization, security, and metadata.

    Sessions support configurable serialization via packer/unpacker traits,
    and signing with HMAC digests via the key/keyfile traits.

    Parameters
    ----------

    debug : bool
        whether to trigger extra debugging statements
    packer/unpacker : str : 'json', 'pickle' or import_string
        importstrings for methods to serialize message parts.  If just
        'json' or 'pickle', predefined JSON and pickle packers will be used.
        Otherwise, the entire importstring must be used.

        The functions must accept at least valid JSON input, and output *bytes*.

        For example, to use msgpack:
        packer = 'msgpack.packb', unpacker='msgpack.unpackb'
    pack/unpack : callables
        You can also set the pack/unpack callables for serialization directly.
    session : bytes
        the ID of this Session object.  The default is to generate a new UUID.
    username : unicode
        username added to message headers.  The default is to ask the OS.
    key : bytes
        The key used to initialize an HMAC signature.  If unset, messages
        will not be signed or checked.
    keyfile : filepath
        The file containing a key.  If this is set, `key` will be initialized
        to the contents of the file.

    FTzDebug output in the Session)confighelpzWhether to check PID to protect against calls after fork.

        This check can be disabled if fork-safety is handled elsewhere.
        r<   zThe name of the packer for serializing messages.
            Should be one of 'json', 'pickle', or an import name
            for a custom callable serializer.packerr    re   rn   c                 C  sZ   |d }|  dkr(t| _t| _|| _n.|  dkrHt| _t| _|| _ntt	|| _d S Nrp   r<   rL   )
lowerrD   packrJ   unpackunpackerrO   pickle_unpackerr   r)   ru   ro   rp   r0   r0   r1   _packer_changedi  s    zSession._packer_changedzjThe name of the unpacker for unserializing messages.
        Only used with custom functions for `packer`.r   c                 C  sZ   |d }|  dkr(t| _t| _|| _n.|  dkrHt| _t| _|| _ntt	|| _d S r   )
r   rD   r   rJ   r   r   rO   r   r   r)   r   r0   r0   r1   _unpacker_changed~  s    zSession._unpacker_changedrb   z"The UUID identifying this session.r)   rP   c                 C  s   t  }|d| _|S NrS   )r\   r,   bsession)ru   ur0   r0   r1   _session_default  s    zSession._session_defaultr   c                 C  s   | j d| _d S r   )r   r,   r   rt   r0   r0   r1   _session_changed  s    zSession._session_changedrc   USERr   z:Username for the Session. Default is your system username.)r   r   zZMetadata dictionary, which serves as the default top-level metadata dict for each message.r   z$execution key, for signing messages.r3   c                 C  s   t  S rw   )r]   rz   r0   r0   r1   _key_default  s    zSession._key_defaultr-   c                 C  s   |    d S rw   )	_new_authrt   r0   r0   r1   _key_changed  s    zSession._key_changedzhmac-sha256zcThe digest scheme used to construct the message signatures.
        Must have the form 'hmac-HASH'.signature_schemec              
   C  s|   |d }| dstd| |ddd }ztt|| _W n0 tk
rn } ztd| |W 5 d }~X Y nX |   d S )Nrp   zhmac-z0signature_scheme must start with 'hmac-', got %rrR   r   z!hashlib has no such attribute: %s)
startswithr   splitgetattrhashlib
digest_modAttributeErrorr   )ru   ro   rp   	hash_namerB   r0   r0   r1   _signature_scheme_changed  s    
 z!Session._signature_scheme_changedz
t.Callablec                 C  s   t jS rw   )r   sha256rz   r0   r0   r1   _digest_mod_default  s    zSession._digest_mod_defaultr|   c                 C  s&   | j rtj| j | jd| _nd | _d S )N)	digestmod)r-   hmacHMACr   authrz   r0   r0   r1   r     s    zSession._new_authi   zzThe maximum number of digests to remember.

        The digest history will be culled when it exceeds this value.
        z&path to file containing execution key.r`   c              	   C  s,   t |d d}|  | _W 5 Q R X d S )Nrp   rb)openreadstripr-   )ru   ro   fr0   r0   r1   _keyfile_changed  s    zSession._keyfile_changedr   c                 C  s$   |d }t |s tdt| d S )Nrp   zpacker must be callable, not %scallabler>   typer   r0   r0   r1   _pack_changed  s    zSession._pack_changedr   c                 C  s$   |d }t |s tdt| d S )Nrp   z!unpacker must be callable, not %sr   r   r0   r0   r1   _unpack_changed  s    zSession._unpack_changedzJThreshold (in bytes) beyond which a buffer should be sent without copying.z[Threshold (in bytes) beyond which an object's buffer should be extracted to avoid pickling.zThe maximum number of items for a container to be introspected for custom serialization.
        Containers larger than this are pickled outright.
        r   c                   sP   t  jf | |   | i | _| j t | _| 	  | j
sLt d dS )ae  create a Session object

        Parameters
        ----------

        debug : bool
            whether to trigger extra debugging statements
        packer/unpacker : str : 'json', 'pickle' or import_string
            importstrings for methods to serialize message parts.  If just
            'json' or 'pickle', predefined JSON and pickle packers will be used.
            Otherwise, the entire importstring must be used.

            The functions must accept at least valid JSON input, and output
            *bytes*.

            For example, to use msgpack:
            packer = 'msgpack.packb', unpacker='msgpack.unpackb'
        pack/unpack : callables
            You can also set the pack/unpack callables for serialization
            directly.
        session : unicode (must be ascii)
            the ID of this Session object.  The default is to generate a new
            UUID.
        bsession : bytes
            The session as bytes
        username : unicode
            username added to message headers.  The default is to ask the OS.
        key : bytes
            The key used to initialize an HMAC signature.  If unset, messages
            will not be signed or checked.
        signature_scheme : str
            The message digest scheme. Currently must be of the form 'hmac-HASH',
            where 'HASH' is a hashing function available in Python's hashlib.
            The default is 'hmac-sha256'.
            This is ignored if 'key' is empty.
        keyfile : filepath
            The file containing a key.  If this is set, `key` will be
            initialized to the contents of the file.
        zCMessage signing is disabled.  This is insecure and not recommended!N)r   r   _check_packersr   noner   rX   getpidpidr   r-   r   warningr   r   r0   r1   r     s    (
zSession.__init__c                 C  sD   t |  }|  D ]}t||t| | qt |_|j| j |S )a  Create a copy of this Session

        Useful when connecting multiple times to a given kernel.
        This prevents a shared digest_history warning about duplicate digests
        due to multiple connections to IOPub in the same process.

        .. versionadded:: 5.1
        )r   Ztraitssetattrr   setdigest_historyupdate)ru   Znew_sessionnamer0   r0   r1   cloneD  s    

zSession.clonec                 C  s.   | j }|  j d7  _ | j dt  d| S )Nr   _)message_countr   rX   r   )ru   Zmessage_numberr0   r0   r1   r   X  s    zSession.msg_idc              
     sL  | j  | jdddgi}z |}W n> tk
rb } z d| j d| }t||W 5 d}~X Y nX t|ts~tdt| z|}||kstW nF tk
r } z(d| j	 d	| j d
| }t||W 5 d}~X Y nX dt
 i}z, |}t|d trd}t|W n2 tk
rF    fdd| _ fdd| _Y nX dS )z#check packers for datetime support.ar   hizpacker 'z(' could not serialize a simple message: Nz,message packed to %r, but bytes are requiredz
unpacker 'z'' could not handle output from packer 'z': tz!Shouldn't deserialize to datetimec                   s    t | S rw   )r   rN   )r   r0   r1   <lambda>  rc   z(Session._check_packers.<locals>.<lambda>c                   s    | S rw   r0   rI   )r   r0   r1   r     rc   )r   r   	Exceptionr   r?   r$   r3   r   AssertionErrorr   rk   r   )ru   msg_listrC   rB   msgunpackedZmsg_datetimer0   )r   r   r1   r   ^  s4    

zSession._check_packersr   )r   r"   c                 C  s   t | j|| j| jS )z#Create a header for a message type.)r   r   r   r   )ru   r   r0   r0   r1   r     s    zSession.msg_headerNzdict | Nonezdict[str, t.Any] | None)r   contentparentr   metadatar"   c                 C  s   i }|dkr|  |n|}||d< |d |d< |d |d< |dkrFi nt||d< |dkr^i n||d< | j |d< |dk	r|d | |S )zReturn the nested message dict.

        This format is different from what is sent over the wire. The
        serialize/deserialize methods converts this nested message dict to the wire
        format, which is a list of message parts.
        Nr   r   r   parent_headerr   r   )r   r   r   copyr   )ru   r   r   r   r   r   r   r0   r0   r1   r     s    zSession.msgr&   )r   r"   c                 C  s8   | j dkrdS | j  }|D ]}|| q|  S )zSign a message with HMAC digest. If no auth, return b''.

        Parameters
        ----------
        msg_list : list
            The [p_header,p_parent,p_content] part of the message list.
        Nrc   )r   r   r   	hexdigestr,   )ru   r   r   mr0   r0   r1   sign  s    

zSession.signzlist[bytes] | bytes | Nonezlist[bytes])r   r^   r"   c                 C  s   | di }|dkr| j}nHt|tr2| |}n2t|tr>n&t|trT|d}ntdt	| | |d | |d | |d |g}g }t|t
r|| n|dk	r|| |t | |}|| || |S )a2  Serialize the message components to bytes.

        This is roughly the inverse of deserialize. The serialize/deserialize
        methods work with full message lists, whereas pack/unpack work with
        the individual message parts in the message list.

        Parameters
        ----------
        msg : dict or Message
            The next message dict as returned by the self.msg method.

        Returns
        -------
        msg_list : list
            The list of bytes objects to be sent with the format::

                [ident1, ident2, ..., DELIM, HMAC, p_header, p_parent,
                 p_metadata, p_content, buffer1, buffer2, ...]

            In this list, the ``p_*`` entities are the packed or serialized
            versions, so if JSON is used, these are utf8 encoded JSON strings.
        r   Nr#   zContent incorrect type: %sr   r   r   )getr   r$   r%   r   r3   r)   r,   r>   r   r&   extendappendDELIMr   )ru   r   r^   r   Zreal_messageto_send	signaturer0   r0   r1   	serialize  s2    








zSession.serializez*zmq.sugar.socket.Socket | ZMQStream | Nonezdict[str, t.Any] | strzbytes | list[bytes] | Nonezlist[bytes] | Noner   )
streammsg_or_typer   r   r^   bufferstrackr   r   r"   c
                 C  s  t |tjsd}t |tjjr8|dk	s*ttj|j}t |ttfr\|}
|pX|
	dg }n| j
|||||	d}
| jrt | jkrt d|
 dS |dkrg n|}t|D ]x\}}t |tr|}n@zt|}W n2 tk
r } zd}t||W 5 d}~X Y nX t|dr|jstd||f q| jr<t|
| j}
| |
|}|| td	d
 |D }|| jk }|r|r|r|s|j|ddd}n|rt}|j||d nt}| jrt  |
 t  | t  | ||
d< |
S )aW  Build and send a message via stream or socket.

        The message format used by this function internally is as follows:

        [ident1,ident2,...,DELIM,HMAC,p_header,p_parent,p_content,
         buffer1,buffer2,...]

        The serialize/deserialize methods convert the nested message dict into this
        format.

        Parameters
        ----------

        stream : zmq.Socket or ZMQStream
            The socket-like object used to send the data.
        msg_or_type : str or Message/dict
            Normally, msg_or_type will be a msg_type unless a message is being
            sent more than once. If a header is supplied, this can be set to
            None and the msg_type will be pulled from the header.

        content : dict or None
            The content of the message (ignored if msg_or_type is a message).
        header : dict or None
            The header dict for the message (ignored if msg_to_type is a message).
        parent : Message or dict or None
            The parent or parent header describing the parent of this message
            (ignored if msg_or_type is a message).
        ident : bytes or list of bytes
            The zmq.IDENTITY routing path.
        metadata : dict or None
            The metadata describing the message
        buffers : list or None
            The already-serialized buffers to be appended to the message.
        track : bool
            Whether to track.  Only for use with Sockets, because ZMQStream
            objects cannot track messages.


        Returns
        -------
        msg : dict
            The constructed message.
        FNr  )r   r   r   r   z/WARNING: attempted to send message from fork
%sz0Buffer objects must support the buffer protocol.
contiguousz Buffer %i (%r) is not contiguousc                 S  s   g | ]}t |qS r0   )len)rT   rE   r0   r0   r1   
<listcomp>V  s     z Session.send.<locals>.<listcomp>T)r   r  r   tracker)!r$   rx   Socketasyncior   shadow
underlyingr   r%   r   r   	check_pidrX   r   r   r   r   r+   
memoryviewr>   hasattrr  r?   adapt_versionr   r  r   maxcopy_thresholdsend_multipartDONEdebugr   )ru   r  r  r   r   r^   r  r  r   r   r   idxr[   viewrB   emsgr   longestr   r
  r0   r0   r1   send  s^    7





zSession.sendzzmq.sugar.socket.Socketint)r  r   flagsr   r^   r"   c                 C  s   g }t |tr|g}|dk	r&|| |t || |dd  || t |tjjrntj	|j
}|j|||d dS )a0  Send a raw message via ident path.

        This method is used to send a already serialized message.

        Parameters
        ----------
        stream : ZMQStream or Socket
            The ZMQ stream or socket to use for sending the message.
        msg_list : list
            The serialized list of messages to send. This only includes the
            [p_header,p_parent,p_metadata,p_content,buffer1,buffer2,...] portion of
            the message.
        ident : ident or list
            A single ident or a list of idents to use in sending.
        Nr   rW   r	  )r$   r3   r   r   r   r   rx   r  r  r  r  r  )ru   r  r   r  r   r^   r   r0   r0   r1   send_rawl  s    



zSession.send_rawz2tuple[list[bytes] | None, dict[str, t.Any] | None])socketmoder   r   r"   c              
   C  s   t |tr|j}t |tjjr,tj|j}z|j||d}W n< tj	k
rz } z|j
tjkrhW Y 
dS  W 5 d}~X Y nX | ||\}}z|| j|||dfW S  tk
r } z|W 5 d}~X Y nX dS )aY  Receive and unpack a message.

        Parameters
        ----------
        socket : ZMQStream or Socket
            The socket or stream to use in receiving.

        Returns
        -------
        [idents], msg
            [idents] is a list of idents and msg is a nested message dict of
            same format as self.msg returns.
        r	  )NNN)r   r   )r$   r   r   rx   r  r  r  r  Zrecv_multipartZZMQErrorerrnoEAGAINfeed_identitiesdeserializer   )ru   r   r!  r   r   r   rB   identsr0   r0   r1   recv  s    

zSession.recvzlist[bytes] | list[zmq.Message]z3tuple[list[bytes], list[bytes] | list[zmq.Message]])r   r   r"   c                 C  s   |r<t t jt |}|t}|d| ||d d fS t t jtj |}d}t|D ]\}}|jtkr\d} qxq\|rd}t	||d| ||d d  }}dd |D |fS dS )ae  Split the identities from the rest of the message.

        Feed until DELIM is reached, then return the prefix as idents and
        remainder as msg_list. This is easily broken by setting an IDENT to DELIM,
        but that would be silly.

        Parameters
        ----------
        msg_list : a list of Message or bytes objects
            The message to be split.
        copy : bool
            flag determining whether the arguments are bytes or Messages

        Returns
        -------
        (idents, msg_list) : two lists
            idents will always be a list of bytes, each of which is a ZMQ
            identity. msg_list will be a list of bytes or zmq.Messages of the
            form [HMAC,p_header,p_parent,p_content,buffer1,buffer2,...] and
            should be unpackable/unserializable via self.deserialize at this
            point.
        Nr   TFzDELIM not in msg_listc                 S  s   g | ]}t |j qS r0   r3   )rT   r   r0   r0   r1   r    s     z+Session.feed_identities.<locals>.<listcomp>)
r   castListr3   indexr   rx   r   r+   r?   )ru   r   r   r  failedr   r   r&  r0   r0   r1   r$    s    

zSession.feed_identities)r   r"   c                 C  s6   | j dkrdS | j| t| j| j kr2|   dS )z9add a digest to history to protect against replay attacksr   N)digest_history_sizer   addr  _cull_digest_history)ru   r   r0   r0   r1   _add_digest  s
    
zSession._add_digestc                 C  s\   t | j}tt|d || j }||kr6t | _dS ttt	| j|}| j
| dS )z_cull the digest history

        Removes a randomly selected 10% of the digest history
        
   N)r  r   r  r  r-  r   randomsampletuplesorteddifference_update)ru   r~   Z	n_to_cullZto_cullr0   r0   r1   r/    s    
zSession._cull_digest_history)r   r   r   r"   c                 C  s  d}i }|sXt t jtj |}dd |d| D }t t jt |}|||d  }t t jt |}| jdk	r|d }|sd}t||| jkrtd| |r| 	| | 
|dd }	t||	sd	| }t|t||ksd
| }t|| |d }
t|
|d< |
d |d< |
d |d< t| |d |d< | |d |d< |rl| |d |d< n|d |d< dd |dd D }|r|d jdkrt t jtj |}dd |dd D }||d< | jrt| t|S )a  Unserialize a msg_list to a nested message dict.

        This is roughly the inverse of serialize. The serialize/deserialize
        methods work with full message lists, whereas pack/unpack work with
        the individual message parts in the message list.

        Parameters
        ----------
        msg_list : list of bytes or Message objects
            The list of message parts of the form [HMAC,p_header,p_parent,
            p_metadata,p_content,buffer1,buffer2,...].
        content : bool (True)
            Whether to unpack the content dict (True), or leave it packed
            (False).
        copy : bool (True)
            Whether msg_list contains bytes (True) or the non-copying Message
            objects in each place (False).

        Returns
        -------
        msg : dict
            The nested message dict with top-level keys [header, parent_header,
            content, buffers].  The buffers are returned as memoryviews.
           c                 S  s   g | ]}t |j qS r0   r(  )rT   r   r0   r0   r1   r  !  s     z'Session.deserialize.<locals>.<listcomp>Nr   zUnsigned MessagezDuplicate Signature: %rr   zInvalid Signature: %rz1malformed message, must have at least %i elementsr   r   r   r9   r      r   rW   r   c                 S  s   g | ]}t |qS r0   )r  rT   br0   r0   r1   r  @  s     c                 S  s   g | ]}t t|jqS r0   )r  r3   r9  r0   r0   r1   r  D  s     r  )r   r)  r*  rx   r   r3   r   r?   r   r0  r   r   r  r>   r   r   shaper  r   r   )ru   r   r   r   ZminlenmessageZmsg_list_beginningr   r   checkr   r  r0   r0   r1   r%    sR    




zSession.deserialize)argsr   r"   c                 O  s   t jdtdd | j||S )z'**DEPRECATED** Use deserialize instead.z;Session.unserialize is deprecated. Use Session.deserialize.r9   r:   )r@   rA   rg   r%  )ru   r>  r   r0   r0   r1   unserializeK  s    zSession.unserialize)NNNN)N)NNNNFNN)r   TN)T)TT)Or   r   r   r   r	   r  r  r   r   r   r   r   r   r   r   r   r   r
   r   r   rX   environr   r   r   r   r   r  r-   r   r   r   r   r   r   r   r   r   r   r   r   r   r   r-  r`   r   r   default_packerr   r   default_unpackerr   r   r  	MAX_BYTESZbuffer_threshold	MAX_ITEMSZitem_thresholdr   r   r   propertyr   r   r   r   r   r  r  r  rx   ZNOBLOCKr'  r$  r0  r/  r%  r?  r   r0   r0   r   r1   ra   -  s   (		4'     D       "~   (, *  M)Qr   
__future__r   r   r   r<   rq   rX   rL   r   r2  typingr   r@   binasciir   r   r   r   Zzmq.asynciorx   Ztornado.ioloopr   Z	traitletsr   r	   r
   r   r   r   r   r   r   r   r   r   Ztraitlets.config.configurabler   r   Ztraitlets.logr   Ztraitlets.utils.importstringr   Zzmq.eventloop.zmqstreamr   _versionr   adapterr   Zjsonutilr   r   r   r   DEFAULT_PROTOCOLrM   rj   r(   rD  rC  rD   rJ   rO   rH   r   rA  rB  r   ZMessageTrackerr  r\   r]   Zsession_aliasesZsession_flagsrh   rk   rl   r   r   r   ra   r0   r0   r0   r1   <module>   sr   8
!	