o
    Ehi_                     @  s  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	m
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mZmZ ddlm Z  dd	l!m"Z" er|dd
l#m$Z$ ddlm%Z% ddlm&Z& ddlm'Z'm(Z( dZ)dZ*e + re*nej,ej-B Z.ej/Z0ej1ej2B ej3B Z4ej5ej6B ej7B ej8B ej9B ej:B ej;B Z<d(ddZ=d)ddZ>d)ddZ?d)ddZ@d)ddZAG d d! d!ZBG d"d# d#ZCG d$d% d%eZDG d&d' d'eZEdS )*u  :module: watchdog.observers.kqueue
:synopsis: ``kqueue(2)`` based emitter implementation.
:author: yesudeep@google.com (Yesudeep Mangalapilly)
:author: contact@tiger-222.fr (Mickaël Schoentgen)
:platforms: macOS and BSD with kqueue(2).

.. WARNING:: kqueue is a very heavyweight way to monitor file systems.
             Each kqueue-detected directory modification triggers
             a full directory scan. Traversing the entire directory tree
             and opening file descriptors for all files will create
             performance problems. We need to find a way to re-scan
             only those directories which report changes and do a diff
             between two sub-DirectorySnapshots perhaps.

.. ADMONITION:: About OS X performance guidelines

    Quote from the `macOS File System Performance Guidelines`_:

        "When you only want to track changes on a file or directory, be sure to
        open it using the ``O_EVTONLY`` flag. This flag prevents the file or
        directory from being marked as open or in use. This is important
        if you are tracking files on a removable volume and the user tries to
        unmount the volume. With this flag in place, the system knows it can
        dismiss the volume. If you had opened the files or directories without
        this flag, the volume would be marked as busy and would not be
        unmounted."

    ``O_EVTONLY`` is defined as ``0x8000`` in the OS X header files.
    More information here: http://www.mlsite.net/blog/?p=2312

Classes
-------
.. autoclass:: KqueueEmitter
   :members:
   :show-inheritance:

Collections and Utility Classes
-------------------------------
.. autoclass:: KeventDescriptor
   :members:
   :show-inheritance:

.. autoclass:: KeventDescriptorSet
   :members:
   :show-inheritance:

.. _macOS File System Performance Guidelines:
    http://developer.apple.com/library/ios/#documentation/Performance/Conceptual/FileSystem/Articles/TrackingChanges.html#//apple_ref/doc/uid/20001993-CJBJFIDD

    )annotationsN)S_ISDIR)TYPE_CHECKING)EVENT_TYPE_CREATEDEVENT_TYPE_DELETEDEVENT_TYPE_MOVEDDirCreatedEventDirDeletedEventDirModifiedEventDirMovedEventFileCreatedEventFileDeletedEventFileModifiedEventFileMovedEventgenerate_sub_moved_events)DEFAULT_EMITTER_TIMEOUTDEFAULT_OBSERVER_TIMEOUTBaseObserverEventEmitter)platform)DirectorySnapshot)	Generator)Callable)FileSystemEvent)
EventQueueObservedWatchi   i   pathbytes | strreturnc                 C  s   t jt j| S N)osr   abspathnormpath)r    r#   Z/var/www/html/Persson_Maskin/env/lib/python3.10/site-packages/watchdog/observers/kqueue.pyabsolute_paths      r%   kevselect.keventboolc                 C     | j tj@ dkS )z8Determines whether the given kevent represents deletion.r   )fflagsselectKQ_NOTE_DELETEr'   r#   r#   r$   
is_deletedz      r/   c                 C  s"   | j }|tj@ dkp|tj@ dkS )z<Determines whether the given kevent represents modification.r   )r+   r,   KQ_NOTE_EXTENDKQ_NOTE_WRITE)r'   r+   r#   r#   r$   is_modified   s   r3   c                 C  r*   )zFDetermines whether the given kevent represents attribute modification.r   )r+   r,   KQ_NOTE_ATTRIBr.   r#   r#   r$   is_attrib_modified   r0   r5   c                 C  r*   )z8Determines whether the given kevent represents movement.r   )r+   r,   KQ_NOTE_RENAMEr.   r#   r#   r$   
is_renamed   r0   r7   c                   @  s   e Zd ZdZd)ddZed*ddZed+d
dZd,ddZd-ddZ	d.ddZ
d/ddZd0ddZd)ddZd-dd Zd.d!d"Zd1d$d%Zd1d&d'Zd(S )2KeventDescriptorSetz)Thread-safe kevent descriptor collection.r   Nonec                 C  s(   t  | _i | _i | _g | _t | _d S r   )set_descriptors_descriptor_for_path_descriptor_for_fd_kevents	threadingLock_lockselfr#   r#   r$   __init__   s
   zKeventDescriptorSet.__init__list[select.kevent]c                 C  s0   | j  | jW  d   S 1 sw   Y  dS )zList of kevents monitored.N)rA   r>   rB   r#   r#   r$   kevents   s   $zKeventDescriptorSet.keventslist[bytes | str]c                 C  s8   | j  t| j W  d   S 1 sw   Y  dS )z2List of paths for which kevents have been created.N)rA   listr<   keysrB   r#   r#   r$   paths   s   $zKeventDescriptorSet.pathsfdintKeventDescriptorc                 C  s4   | j  | j| W  d   S 1 sw   Y  dS )zGiven a file descriptor, returns the kevent descriptor object
        for it.

        :param fd:
            OS file descriptor.
        :type fd:
            ``int``
        :returns:
            A :class:`KeventDescriptor` object.
        N)rA   r=   )rC   rK   r#   r#   r$   
get_for_fd   s   $zKeventDescriptorSet.get_for_fdr   r   c                 C  <   | j  t|}| |W  d   S 1 sw   Y  dS )zObtains a :class:`KeventDescriptor` object for the specified path.

        :param path:
            Path for which the descriptor will be obtained.
        N)rA   r%   _getrC   r   r#   r#   r$   get   s   $zKeventDescriptorSet.getr)   c                 C  rO   )zDetermines whether a :class:`KeventDescriptor has been registered
        for the specified path.

        :param path:
            Path for which the descriptor will be obtained.
        N)rA   r%   	_has_pathrQ   r#   r#   r$   __contains__   s   $z KeventDescriptorSet.__contains__is_directoryc                C  s`   | j # t|}| |s| t||d W d   dS W d   dS 1 s)w   Y  dS )ae  Adds a :class:`KeventDescriptor` to the collection for the given
        path.

        :param path:
            The path for which a :class:`KeventDescriptor` object will be
            added.
        :param is_directory:
            ``True`` if the path refers to a directory; ``False`` otherwise.
        :type is_directory:
            ``bool``
        rU   N)rA   r%   rS   _add_descriptorrM   rC   r   rU   r#   r#   r$   add   s   
"zKeventDescriptorSet.addc                 C  s^   | j " t|}| |r| | | W d   dS W d   dS 1 s(w   Y  dS )zRemoves the :class:`KeventDescriptor` object for the given path
        if it already exists.

        :param path:
            Path for which the :class:`KeventDescriptor` object will be
            removed.
        N)rA   r%   rS   _remove_descriptorrP   rQ   r#   r#   r$   remove   s   
"zKeventDescriptorSet.removec                 C  sd   | j % | jD ]}|  q| j  | j  | j  g | _W d   dS 1 s+w   Y  dS )z6Clears the collection and closes all open descriptors.N)rA   r;   closeclearr=   r<   r>   rC   
descriptorr#   r#   r$   r]      s   




"zKeventDescriptorSet.clearc                 C  s
   | j | S )z-Returns a kevent descriptor for a given path.r<   rQ   r#   r#   r$   rP      s   
zKeventDescriptorSet._getc                 C  s
   || j v S )zxDetermines whether a :class:`KeventDescriptor` for the specified
        path exists already in the collection.
        r`   rQ   r#   r#   r$   rS      s   
zKeventDescriptorSet._has_pathr_   c                 C  s6   | j | | j|j || j|j< || j|j< dS )zAdds a descriptor to the collection.

        :param descriptor:
            An instance of :class:`KeventDescriptor` to be added.
        N)	r;   rY   r>   appendkeventr<   r   r=   rK   r^   r#   r#   r$   rW      s   z#KeventDescriptorSet._add_descriptorc                 C  s:   | j | | j|j= | j|j= | j|j |  dS )zRemoves a descriptor from the collection.

        :param descriptor:
            An instance of :class:`KeventDescriptor` to be removed.
        N)	r;   r[   r=   rK   r<   r   r>   rb   r\   r^   r#   r#   r$   rZ     s
   

z&KeventDescriptorSet._remove_descriptorNr   r9   )r   rE   )r   rG   )rK   rL   r   rM   )r   r   r   rM   )r   r   r   r)   r   r   rU   r)   r   r9   r   r   r   r9   )r_   rM   r   r9   )__name__
__module____qualname____doc__rD   propertyrF   rJ   rN   rR   rT   rY   r[   r]   rP   rS   rW   rZ   r#   r#   r#   r$   r8      s"    










r8   c                   @  s   e Zd ZdZd%dd	Zed&ddZed'ddZed(ddZed)ddZ	d*ddZ
ed+ddZd,ddZd,ddZd&dd Zd-d"d#Zd$S ).rM   a{  A kevent descriptor convenience data structure to keep together:

        * kevent
        * directory status
        * path
        * file descriptor

    :param path:
        Path string for which a kevent descriptor will be created.
    :param is_directory:
        ``True`` if the path refers to a directory; ``False`` otherwise.
    :type is_directory:
        ``bool``
    r   r   rU   r)   r   r9   c                C  s8   t || _|| _t|t| _tj| jt	t
td| _d S )N)filterflagsr+   )r%   _path_is_directoryr    openWATCHDOG_OS_OPEN_FLAGS_fdr,   rb   WATCHDOG_KQ_FILTERWATCHDOG_KQ_EV_FLAGSWATCHDOG_KQ_FFLAGS_kevrX   r#   r#   r$   rD   #  s   
zKeventDescriptor.__init__rL   c                 C     | j S )z-OS file descriptor for the kevent descriptor.)rq   rB   r#   r#   r$   rK   .     zKeventDescriptor.fdc                 C  rv   )z/The path associated with the kevent descriptor.)rm   rB   r#   r#   r$   r   3  rw   zKeventDescriptor.pathr(   c                 C  rv   )z8The kevent object associated with the kevent descriptor.)ru   rB   r#   r#   r$   rb   8  rw   zKeventDescriptor.keventc                 C  rv   )z}Determines whether the kevent descriptor refers to a directory.

        :returns:
            ``True`` or ``False``
        )rn   rB   r#   r#   r$   rU   =  s   zKeventDescriptor.is_directoryc                 C  s<   t t t| j W d   dS 1 sw   Y  dS )z?Closes the file descriptor associated with a kevent descriptor.N)
contextlibsuppressOSErrorr    r\   rK   rB   r#   r#   r$   r\   F  s   "zKeventDescriptor.closetuple[bytes | str, bool]c                 C  s   | j | jfS r   )r   rU   rB   r#   r#   r$   keyK  s   zKeventDescriptor.keyr_   objectc                 C  s   t |tstS | j|jkS r   
isinstancerM   NotImplementedr|   r^   r#   r#   r$   __eq__O     
zKeventDescriptor.__eq__c                 C  s   t |tstS | j|jkS r   r~   r^   r#   r#   r$   __ne__T  r   zKeventDescriptor.__ne__c                 C  s
   t | jS r   )hashr|   rB   r#   r#   r$   __hash__Y  s   
zKeventDescriptor.__hash__strc                 C  s"   dt | j d| jd| j dS )N<z: path=z, is_directory=>)typerf   r   rU   rB   r#   r#   r$   __repr__\  s   "zKeventDescriptor.__repr__Nrd   )r   rL   )r   r   )r   r(   )r   r)   rc   )r   r{   )r_   r}   r   r)   )r   r   )rf   rg   rh   ri   rD   rj   rK   r   rb   rU   r\   r|   r   r   r   r   r#   r#   r#   r$   rM     s$    




rM   c                      s   e Zd ZdZedejdd3 fddZd4ddZd5ddZ	d6ddZ
d7d$d%Zd8d(d)Zd9d*d+Zd:d-d.Zd;d/d0Zd<d1d2Z  ZS )=KqueueEmittera}  kqueue(2)-based event emitter.

    .. ADMONITION:: About ``kqueue(2)`` behavior and this implementation

              ``kqueue(2)`` monitors file system events only for
              open descriptors, which means, this emitter does a lot of
              book-keeping behind the scenes to keep track of open
              descriptors for every entry in the monitored directory tree.

              This also means the number of maximum open file descriptors
              on your system must be increased **manually**.
              Usually, issuing a call to ``ulimit`` should suffice::

                  ulimit -n 1024

              Ensure that you pick a number that is larger than the
              number of files you expect to be monitored.

              ``kqueue(2)`` does not provide enough information about the
              following things:

              * The destination path of a file or directory that is renamed.
              * Creation of a file or directory within a directory; in this
                case, ``kqueue(2)`` only indicates a modified event on the
                parent directory.

              Therefore, this emitter takes a snapshot of the directory
              tree when ``kqueue(2)`` detects a change on the file system
              to be able to determine the above information.

    :param event_queue:
        The event queue to fill with events.
    :param watch:
        A watch object representing the directory to monitor.
    :type watch:
        :class:`watchdog.observers.api.ObservedWatch`
    :param timeout:
        Read events blocking timeout (in seconds).
    :type timeout:
        ``float``
    :param event_filter:
        Collection of event types to emit, or None for no filtering (default).
    :type event_filter:
        Iterable[:class:`watchdog.events.FileSystemEvent`] | None
    :param stat: stat function. See ``os.stat`` for details.
    N)timeoutevent_filterstatevent_queuer   watchr   r   floatr   "list[type[FileSystemEvent]] | Noner   Callable[[str], os.stat_result]r   r9   c                  sZ   t  j||||d t | _t | _t | _	| fd fdd	}t
|j|j|d
| _d S )N)r   r   r   r   clsr   r   os.stat_resultc                   s     | }|j | t|jd |S )NrV   )_register_keventr   st_mode)r   r   	stat_infor   r#   r$   custom_stat  s   z+KqueueEmitter.__init__.<locals>.custom_stat)	recursiver   )r   r   r   r   r   r   )superrD   r,   kqueue_kqr?   RLockrA   r8   r;   r   r   is_recursive	_snapshot)rC   r   r   r   r   r   r   	__class__r   r$   rD     s   	

zKqueueEmitter.__init__r   r   rU   r)   c             
   C  sj   z| j j||d W dS  ty4 } z|jtjkrn|jtjkr!n W Y d}~dS W Y d}~dS d}~ww )a+  Registers a kevent descriptor for the given path.

        :param path:
            Path for which a kevent descriptor will be created.
        :param is_directory:
            ``True`` if the path refers to a directory; ``False`` otherwise.
        :type is_directory:
            ``bool``
        rV   N)r;   rY   rz   errnoENOENT
EOPNOTSUPP)rC   r   rU   er#   r#   r$   r     s   
zKqueueEmitter._register_keventc                 C  s   | j | dS )zConvenience function to close the kevent descriptor for a
        specified kqueue-monitored path.

        :param path:
            Path for which the kevent descriptor will be closed.
        N)r;   r[   rQ   r#   r#   r$   _unregister_kevent  s   z KqueueEmitter._unregister_keventeventr   c                 C  sv   t | | |jtkr| j|j|jd dS |jtkr,| |j | j|j	|jd dS |jt
kr9| |j dS dS )zHandles queueing a single event object.

        :param event:
            An instance of :class:`watchdog.events.FileSystemEvent`
            or a subclass.
        rV   N)r   queue_event
event_typer   r   src_pathrU   r   r   	dest_pathr   )rC   r   r#   r#   r$   r     s   


zKqueueEmitter.queue_eventr'   r(   ref_snapshotr   new_snapshotGenerator[FileSystemEvent]c                 c  s    | j |j}|j}t|r| j||||jdE dH  dS t|r3|jr,t|V  dS t	|V  dS t
|rT|jrM| jjsD| jj|krKt|V  dS dS t	|V  dS t|ri|jrbt|V  dS t|V  dS dS )a  Generate events from the kevent list returned from the call to
        :meth:`select.kqueue.control`.

        .. NOTE:: kqueue only tells us about deletions, file modifications,
                  attribute modifications. The other events, namely,
                  file creation, directory modification, file rename,
                  directory rename, directory creation, etc. are
                  determined by comparing directory snapshots.
        rV   N)r;   rN   identr   r7   _gen_renamed_eventsrU   r5   r
   r   r3   r   r   r/   r	   r   )rC   r'   r   r   r_   r   r#   r#   r$   _gen_kqueue_events  s2   z KqueueEmitter._gen_kqueue_eventsr   r
   c                 C  s   t tj|S )z@Helper to generate a DirModifiedEvent on the parent of src_path.)r
   r    r   dirname)rC   r   r#   r#   r$   _parent_dir_modified  r0   z"KqueueEmitter._parent_dir_modifiedc                c  s    z| |}W n# ty+   |rt|V  t|V  Y dS t|V  t|V  Y dS w ||}|durht|}|rBt||V  nt	||V  | 
|V  | 
|V  |rd| jjrft||E dH  dS dS dS |rpt|V  nt|V  | 
|V  dS )a  Compares information from two directory snapshots (one taken before
        the rename operation and another taken right after) to determine the
        destination path of the file system object renamed, and yields
        the appropriate events to be queued.
        N)inodeKeyErrorr   r	   r   r   r   r%   r   r   r   r   r   r   )rC   r   r   r   rU   f_inoder   r#   r#   r$   r     s6   





z!KqueueEmitter._gen_renamed_eventsrE   c                 C  s   | j | jjt|S )zReads events from a call to the blocking
        :meth:`select.kqueue.control()` method.

        :param timeout:
            Blocking timeout for reading events.
        :type timeout:
            ``float`` (seconds)
        )r   controlr;   rF   
MAX_EVENTSrC   r   r#   r#   r$   _read_eventsQ  s   	zKqueueEmitter._read_eventsc                 C  s$  | j  zZ| |}|  t| jj| jjd}| j}|| _|| }|jD ]	}| 	t
| q%|jD ]	}| 	t| q2|jD ]	}| 	t| q?|D ]}	| |	||D ]}
| 	|
 qTqKW n tyw } z|jtjkrm W Y d}~nd}~ww W d   dS W d   dS 1 sw   Y  dS )zQueues events by reading them from a call to the blocking
        :meth:`select.kqueue.control()` method.

        :param timeout:
            Blocking timeout for reading events.
        :type timeout:
            ``float`` (seconds)
        )r   N)rA   r   reverser   r   r   r   r   dirs_createdr   r   files_createdr   files_modifiedr   r   rz   r   EBADF)rC   r   
event_listr   r   diff_eventsdirectory_createdfile_createdfile_modifiedr'   r   r   r#   r#   r$   queue_events\  s:   	



"zKqueueEmitter.queue_eventsc                 C  s@   | j  | j  | j  W d    d S 1 sw   Y  d S r   )rA   r;   r]   r   r\   rB   r#   r#   r$   on_thread_stop  s   
"zKqueueEmitter.on_thread_stop)r   r   r   r   r   r   r   r   r   r   r   r9   rd   re   )r   r   r   r9   )r'   r(   r   r   r   r   r   r   )r   r   r   r
   )
r   r   r   r   r   r   rU   r)   r   r   )r   r   r   rE   r   r   r   r9   rc   )rf   rg   rh   ri   r   r    r   rD   r   r   r   r   r   r   r   r   r   __classcell__r#   r#   r   r$   r   `  s    4

#
	

.

7
&r   c                      s(   e Zd ZdZedd	 fddZ  ZS )
KqueueObserverzdObserver thread that schedules watching directories and dispatches
    calls to event handlers.
    r   r   r   r   r9   c                  s   t  jt|d d S )Nr   )r   rD   r   r   r   r#   r$   rD     r&   zKqueueObserver.__init__r   )rf   rg   rh   ri   r   rD   r   r#   r#   r   r$   r     s    r   )r   r   r   r   )r'   r(   r   r)   )Fri   
__future__r   rx   r   r    os.pathr,   r?   r   r   typingr   watchdog.eventsr   r   r   r   r	   r
   r   r   r   r   r   r   watchdog.observers.apir   r   r   r   watchdog.utilsr   watchdog.utils.dirsnapshotr   collections.abcr   r   r   r   r   r   	O_EVTONLY	is_darwinO_RDONLY
O_NONBLOCKrp   KQ_FILTER_VNODErr   	KQ_EV_ADDKQ_EV_ENABLEKQ_EV_CLEARrs   r-   r2   r1   r4   KQ_NOTE_LINKr6   KQ_NOTE_REVOKErt   r%   r/   r3   r5   r7   r8   rM   r   r   r#   r#   r#   r$   <module>   sd    98




 M  +