o
    XRh')                     @   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ZddlZejZ	G dd de
ZG dd de
ZG dd de	ZG d	d
 d
eZdddZdS )zb
Fast cryptographic hash of Python objects, with a special case for fast
hashing of numpy arrays.
    Nc                   @      e Zd ZdZdd ZdS )_ConsistentSetza Class used to ensure the hash of Sets is preserved
        whatever the order of its items.
    c              	   C   s@   zt || _W d S  ttjfy   t dd |D | _Y d S w )Nc                 s   s    | ]}t |V  qd S Nhash).0e r	   P/home/air/sanwanet/backup_V2/venv/lib/python3.10/site-packages/joblib/hashing.py	<genexpr>%   s    z*_ConsistentSet.__init__.<locals>.<genexpr>)sorted	_sequence	TypeErrordecimalInvalidOperation)selfset_sequencer	   r	   r
   __init__   s
   z_ConsistentSet.__init__N__name__
__module____qualname____doc__r   r	   r	   r	   r
   r      s    r   c                   @   r   )_MyHashz7 Class used to hash objects that won't normally pickle c                 G   s
   || _ d S r   )args)r   r   r	   r	   r
   r   +   s   
z_MyHash.__init__Nr   r	   r	   r	   r
   r   (   s    r   c                   @   s   e Zd ZdZdddZdddZdd	 Zd
d Zdej	fddZ
ej Ze
eee< e
eee< e
eee< e
eeej< dd Zdd Zeeee < dS )HasherzW A subclass of pickler, to do cryptographic hashing, rather than
        pickling.
    md5c                 C   s0   t  | _d}tj| | j|d t|| _d S )N   )protocol)ioBytesIOstreamPicklerr   hashlibnew_hash)r   	hash_namer   r	   r	   r
   r   4   s   
zHasher.__init__Tc              
   C   sl   z|  | W n tjy! } z| jd||f f7  _ d }~ww | j }| j| |r4| j S d S )Nz"PicklingError while hashing %r: %r)	dumppicklePicklingErrorr   r!   getvaluer%   update	hexdigest)r   objreturn_digestr   dumpsr	   r	   r
   r   =   s   

zHasher.hashc                 C   s   t |tjti jfr>t|dr|jj}n|j}|j}t|tt	u r*t
||j}n|d u r4t
||}n
|jj}t
|||}t| | d S )N__func__)
isinstancetypes
MethodTypetypepophasattrr0   r   __self__r(   r   	__class__r"   save)r   r-   	func_nameinstclsr	   r	   r
   r9   H   s   

zHasher.savec                 C   s"   t |ttfr	d S t| | d S r   )r1   bytesstrr"   memoize)r   r-   r	   r	   r
   r?   [   s   zHasher.memoizeNc                 C   s   t ||d}|d= ztj| |fi | W d S  tjyT   tj| |fi | t|dd }|dkrN|}|d u r;|j}tj| }t	||sQt
||| Y d S Y d S Y d S w )N)namepackrA   r   __main__)dictr"   save_globalr(   r)   getattrr   sysmodulesr6   setattr)r   r-   r@   rA   kwargsmodulemy_namemodr	   r	   r
   rD   f   s"   

zHasher.save_globalc              
   C   sN   zt | tt| W d S  ty&   t | ttdd |D  Y d S w )Nc                 s   s     | ]\}}t ||fV  qd S r   r   )r   kvr	   r	   r
   r      s    z)Hasher._batch_setitems.<locals>.<genexpr>)r"   _batch_setitemsiterr   r   )r   itemsr	   r	   r
   rO      s   zHasher._batch_setitemsc                 C   s   t | t| d S r   )r"   r9   r   )r   	set_itemsr	   r	   r
   save_set   s   zHasher.save_set)r   )T)r   r   r   r   r   r   r9   r?   structrA   rD   r"   dispatchcopyr4   lenobjectr(   r'   rO   rS   setr	   r	   r	   r
   r   /   s    

	
r   c                   @   s"   e Zd ZdZd	ddZdd ZdS )
NumpyHasherz7 Special case the hasher for when numpy is loaded.
    r   Fc                 C   sB   || _ tj| |d ddl}|| _t|dr|j| _dS t| _dS )a  
            Parameters
            ----------
            hash_name: string
                The hash algorithm to be used
            coerce_mmap: boolean
                Make no difference between np.memmap and np.ndarray
                objects.
        r&   r   N	getbuffer)	coerce_mmapr   r   numpynpr6   r\   
_getbuffer
memoryview)r   r&   r]   r_   r	   r	   r
   r      s   


zNumpyHasher.__init__c                 C   s   t || jjrT|jjsT|jdkr| }n|jjr|}n|jj	r$|j
}n| }| j| || jj | jrEt || jjrE| jj}n|j}|d|j|j|jff}nt || jjro| jdd | jt| dS t| | dS )z Subclass the save method, to hash ndarray subclass, rather
            than pickling them. Off course, this is a total abuse of
            the Pickler class.
        r	   HASHED_HASHED_DTYPEzutf-8N)r1   r_   ndarraydtype	hasobjectshapeflattenflagsc_contiguousf_contiguousTr%   r+   r`   viewuint8r]   memmapr8   stridesencoder(   r/   r   r9   )r   r-   obj_c_contiguousklassr	   r	   r
   r9      s(   


zNumpyHasher.saveNr   F)r   r   r   r   r   r9   r	   r	   r	   r
   rZ      s    
rZ   r   Fc                 C   sH   d}||vrt d||dtjv rt||d}nt|d}|| S )aj   Quick calculation of a hash to identify uniquely Python objects
        containing numpy arrays.

        Parameters
        ----------
        hash_name: 'md5' or 'sha1'
            Hashing algorithm used. sha1 is supposedly safer, but md5 is
            faster.
        coerce_mmap: boolean
            Make no difference between np.memmap and np.ndarray
    )r   sha1zAValid options for 'hash_name' are {}. Got hash_name={!r} instead.r^   )r&   r]   r[   )
ValueErrorformatrF   rG   rZ   r   r   )r-   r&   r]   valid_hash_nameshasherr	   r	   r
   r      s   


r   rt   )r   r(   r#   rF   r2   rT   r   r   _Picklerr"   rX   r   r   r   rZ   r   r	   r	   r	   r
   <module>   s    	kZ