143 lines
4.8 KiB
Python
143 lines
4.8 KiB
Python
|
""":module: watchdog.observers.polling
|
||
|
:synopsis: Polling emitter implementation.
|
||
|
:author: yesudeep@google.com (Yesudeep Mangalapilly)
|
||
|
:author: contact@tiger-222.fr (Mickaël Schoentgen)
|
||
|
|
||
|
Classes
|
||
|
-------
|
||
|
.. autoclass:: PollingObserver
|
||
|
:members:
|
||
|
:show-inheritance:
|
||
|
|
||
|
.. autoclass:: PollingObserverVFS
|
||
|
:members:
|
||
|
:show-inheritance:
|
||
|
:special-members:
|
||
|
"""
|
||
|
|
||
|
from __future__ import annotations
|
||
|
|
||
|
import os
|
||
|
import threading
|
||
|
from functools import partial
|
||
|
from typing import TYPE_CHECKING
|
||
|
|
||
|
from watchdog.events import (
|
||
|
DirCreatedEvent,
|
||
|
DirDeletedEvent,
|
||
|
DirModifiedEvent,
|
||
|
DirMovedEvent,
|
||
|
FileCreatedEvent,
|
||
|
FileDeletedEvent,
|
||
|
FileModifiedEvent,
|
||
|
FileMovedEvent,
|
||
|
)
|
||
|
from watchdog.observers.api import DEFAULT_EMITTER_TIMEOUT, DEFAULT_OBSERVER_TIMEOUT, BaseObserver, EventEmitter
|
||
|
from watchdog.utils.dirsnapshot import DirectorySnapshot, DirectorySnapshotDiff, EmptyDirectorySnapshot
|
||
|
|
||
|
if TYPE_CHECKING:
|
||
|
from collections.abc import Iterator
|
||
|
from typing import Callable
|
||
|
|
||
|
from watchdog.events import FileSystemEvent
|
||
|
from watchdog.observers.api import EventQueue, ObservedWatch
|
||
|
|
||
|
|
||
|
class PollingEmitter(EventEmitter):
|
||
|
"""Platform-independent emitter that polls a directory to detect file
|
||
|
system changes.
|
||
|
"""
|
||
|
|
||
|
def __init__(
|
||
|
self,
|
||
|
event_queue: EventQueue,
|
||
|
watch: ObservedWatch,
|
||
|
*,
|
||
|
timeout: float = DEFAULT_EMITTER_TIMEOUT,
|
||
|
event_filter: list[type[FileSystemEvent]] | None = None,
|
||
|
stat: Callable[[str], os.stat_result] = os.stat,
|
||
|
listdir: Callable[[str | None], Iterator[os.DirEntry]] = os.scandir,
|
||
|
) -> None:
|
||
|
super().__init__(event_queue, watch, timeout=timeout, event_filter=event_filter)
|
||
|
self._snapshot: DirectorySnapshot = EmptyDirectorySnapshot()
|
||
|
self._lock = threading.Lock()
|
||
|
self._take_snapshot: Callable[[], DirectorySnapshot] = lambda: DirectorySnapshot(
|
||
|
self.watch.path,
|
||
|
recursive=self.watch.is_recursive,
|
||
|
stat=stat,
|
||
|
listdir=listdir,
|
||
|
)
|
||
|
|
||
|
def on_thread_start(self) -> None:
|
||
|
self._snapshot = self._take_snapshot()
|
||
|
|
||
|
def queue_events(self, timeout: float) -> None:
|
||
|
# We don't want to hit the disk continuously.
|
||
|
# timeout behaves like an interval for polling emitters.
|
||
|
if self.stopped_event.wait(timeout):
|
||
|
return
|
||
|
|
||
|
with self._lock:
|
||
|
if not self.should_keep_running():
|
||
|
return
|
||
|
|
||
|
# Get event diff between fresh snapshot and previous snapshot.
|
||
|
# Update snapshot.
|
||
|
try:
|
||
|
new_snapshot = self._take_snapshot()
|
||
|
except OSError:
|
||
|
self.queue_event(DirDeletedEvent(self.watch.path))
|
||
|
self.stop()
|
||
|
return
|
||
|
|
||
|
events = DirectorySnapshotDiff(self._snapshot, new_snapshot)
|
||
|
self._snapshot = new_snapshot
|
||
|
|
||
|
# Files.
|
||
|
for src_path in events.files_deleted:
|
||
|
self.queue_event(FileDeletedEvent(src_path))
|
||
|
for src_path in events.files_modified:
|
||
|
self.queue_event(FileModifiedEvent(src_path))
|
||
|
for src_path in events.files_created:
|
||
|
self.queue_event(FileCreatedEvent(src_path))
|
||
|
for src_path, dest_path in events.files_moved:
|
||
|
self.queue_event(FileMovedEvent(src_path, dest_path))
|
||
|
|
||
|
# Directories.
|
||
|
for src_path in events.dirs_deleted:
|
||
|
self.queue_event(DirDeletedEvent(src_path))
|
||
|
for src_path in events.dirs_modified:
|
||
|
self.queue_event(DirModifiedEvent(src_path))
|
||
|
for src_path in events.dirs_created:
|
||
|
self.queue_event(DirCreatedEvent(src_path))
|
||
|
for src_path, dest_path in events.dirs_moved:
|
||
|
self.queue_event(DirMovedEvent(src_path, dest_path))
|
||
|
|
||
|
|
||
|
class PollingObserver(BaseObserver):
|
||
|
"""Platform-independent observer that polls a directory to detect file
|
||
|
system changes.
|
||
|
"""
|
||
|
|
||
|
def __init__(self, *, timeout: float = DEFAULT_OBSERVER_TIMEOUT) -> None:
|
||
|
super().__init__(PollingEmitter, timeout=timeout)
|
||
|
|
||
|
|
||
|
class PollingObserverVFS(BaseObserver):
|
||
|
"""File system independent observer that polls a directory to detect changes."""
|
||
|
|
||
|
def __init__(
|
||
|
self,
|
||
|
stat: Callable[[str], os.stat_result],
|
||
|
listdir: Callable[[str | None], Iterator[os.DirEntry]],
|
||
|
*,
|
||
|
polling_interval: int = 1,
|
||
|
) -> None:
|
||
|
""":param stat: stat function. See ``os.stat`` for details.
|
||
|
:param listdir: listdir function. See ``os.scandir`` for details.
|
||
|
:type polling_interval: int
|
||
|
:param polling_interval: interval in seconds between polling the file system.
|
||
|
"""
|
||
|
emitter_cls = partial(PollingEmitter, stat=stat, listdir=listdir)
|
||
|
super().__init__(emitter_cls, timeout=polling_interval) # type: ignore[arg-type]
|