Observar Cambios en Archivos
Cómo monitorear cambios en el sistema de archivos en tiempo real.
Nota para desarrolladores hispanohablantes: Esta guía incluye ejemplos y convenciones de nomenclatura adaptadas a equipos que trabajan en español. Cuando existen diferencias significativas en terminología técnica entre el inglés y el español, se indican explícitamente para facilitar la comunicación en equipos multiculturales.
Visión General
Los watchers del sistema de archivos reaccionan a eventos de creación, modificación, eliminación y renombrado en tiempo real. Alimentan servidores de hot-reload, tailers de logs y herramientas de sincronización. Esta receta muestra implementaciones multiplataforma en Python, JavaScript y Java.
Cuándo Usar
Usa este recurso cuando:
- Construyes servidores de desarrollo que recargan ante cambios de código
- Monitoreas directorios de logs para nuevos archivos a procesar
- Disparas pipelines cuando carpetas de upload reciben archivos
Solución
Python
from watchdog.observers import Observer
from watchdog.events import FileSystemEventHandler
class Handler(FileSystemEventHandler):
def on_modified(self, event):
if not event.is_directory:
print(f"Modificado: {event.src_path}")
observer = Observer()
observer.schedule(Handler(), path='./watched', recursive=True)
observer.start()
try:
while True:
pass
except KeyboardInterrupt:
observer.stop()
observer.join()
JavaScript
const fs = require('fs');
// Observar archivo o directorio
const watcher = fs.watch('./watched', { recursive: true }, (eventType, filename) => {
console.log(`${eventType}: ${filename}`);
});
// Limpieza
process.on('SIGINT', () => watcher.close());
Java
import java.nio.file.*;
public class FileWatcher {
public static void watch(Path path) throws Exception {
WatchService watchService = FileSystems.getDefault().newWatchService();
path.register(watchService,
StandardWatchEventKinds.ENTRY_CREATE,
StandardWatchEventKinds.ENTRY_MODIFY,
StandardWatchEventKinds.ENTRY_DELETE);
while (true) {
WatchKey key = watchService.take();
for (WatchEvent<?> event : key.pollEvents()) {
System.out.println(event.kind() + ": " + event.context());
}
key.reset();
}
}
}
Explicación
Los watchers se registran en el kernel del SO, que luego empuja eventos a tu proceso en lugar de requerir polling costoso. Python watchdog abstrae inotify (Linux), FSEvents (macOS) y ReadDirectoryChangesW (Windows). Node.js fs.watch delega a la API nativa más eficiente por plataforma. Java NIO WatchService usa los mismos mecanismos subyacentes del SO a través de una API estandarizada.
Variantes
| Tecnología | Enfoque | Notas |
|---|---|---|
| Python | Biblioteca watchdog | Multiplataforma, maneja casos edge como renombres rápidos |
| JavaScript | Paquete chokidar | Más confiable que fs.watch en macOS y Windows |
| Java | Apache Commons IO FileAlterationMonitor | Fallback por polling para JDKs antiguos |
Mejores Prácticas
- Debounce eventos rápidos (los editores suelen disparar múltiples escrituras)
- Siempre maneja el evento
error/ excepciones deWatchService - Usa watchers recursivos con moderación; consumen recursos del SO
- Filtra por extensión de archivo para ignorar archivos temporales (ej.
.tmp,.swp) - Ejecuta watchers en un hilo o proceso dedicado para evitar bloqueos
Errores Comunes
- Asumir que los eventos
modifyse disparan solo una vez por guardado (los editores pueden disparar muchos) - No limpiar recursos del watcher al cerrar, causando fugas
- Observar unidades de red con APIs nativas que no las soportan
- Ignorar eventos de
rename, que aparecen como create + delete separados en algunos SO - Procesar archivos inmediatamente en
createantes de que el escritor los haya cerrado
Preguntas Frecuentes
¿Puedo observar rutas remotas o de red?
Los watchers nativos generalmente no soportan shares de red. Usa bibliotecas de polling como chokidar con usePolling: true o FileAlterationMonitor como fallback.
¿Por qué recibo eventos duplicados?
Muchos editores escriben archivos atómicamente (crear temp, renombrar), disparando múltiples eventos. Aplica debounce con un pequeño retraso (ej. 100 ms) antes de actuar.
¿Cuántos archivos puedo observar a la vez?
Los límites del SO varían. Linux inotify tiene un límite max_user_watches por usuario (default ~8K). macOS FSEvents escala a millones. Evita watchers recursivos en árboles enormes.
Recursos Relacionados
Read Large Files
How to read large files efficiently without running out of memory.
RecipeWrite Large Files
How to write large files efficiently using buffered and streaming output.
RecipeFile Upload Validation
How to handle file uploads securely with size, type, and content validation.
RecipeGenerate PDFs
How to generate PDF documents programmatically from HTML, templates, or raw data.
RecipeProcess Large Files with Streams
How to read, transform, and write large files efficiently using streams without loading entire files into memory in Python, Node.js, and Java.