mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-26 16:22:24 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			430 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			430 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| const Path = require('path')
 | |
| const EventEmitter = require('events')
 | |
| const Watcher = require('./libs/watcher/watcher')
 | |
| const Logger = require('./Logger')
 | |
| const Task = require('./objects/Task')
 | |
| const TaskManager = require('./managers/TaskManager')
 | |
| 
 | |
| const { filePathToPOSIX, isSameOrSubPath, getFileMTimeMs, shouldIgnoreFile } = require('./utils/fileUtils')
 | |
| 
 | |
| /**
 | |
|  * @typedef PendingFileUpdate
 | |
|  * @property {string} path
 | |
|  * @property {string} relPath
 | |
|  * @property {string} folderId
 | |
|  * @property {string} type
 | |
|  */
 | |
| class FolderWatcher extends EventEmitter {
 | |
|   constructor() {
 | |
|     super()
 | |
| 
 | |
|     /** @type {{id:string, name:string, libraryFolders:import('./models/Folder')[], paths:string[], watcher:Watcher[]}[]} */
 | |
|     this.libraryWatchers = []
 | |
|     /** @type {PendingFileUpdate[]} */
 | |
|     this.pendingFileUpdates = []
 | |
|     this.pendingDelay = 10000
 | |
|     /** @type {NodeJS.Timeout} */
 | |
|     this.pendingTimeout = null
 | |
|     /** @type {Task} */
 | |
|     this.pendingTask = null
 | |
| 
 | |
|     this.filesBeingAdded = new Set()
 | |
| 
 | |
|     /** @type {Set<string>} */
 | |
|     this.ignoreFilePathsDownloading = new Set()
 | |
|     /** @type {string[]} */
 | |
|     this.ignoreDirs = []
 | |
|     /** @type {string[]} */
 | |
|     this.pendingDirsToRemoveFromIgnore = []
 | |
|     /** @type {NodeJS.Timeout} */
 | |
|     this.removeFromIgnoreTimer = null
 | |
| 
 | |
|     this.disabled = false
 | |
|   }
 | |
| 
 | |
|   get pendingFilePaths() {
 | |
|     return this.pendingFileUpdates.map((f) => f.path)
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    *
 | |
|    * @param {import('./models/Library')} library
 | |
|    */
 | |
|   buildLibraryWatcher(library) {
 | |
|     if (this.libraryWatchers.find((w) => w.id === library.id)) {
 | |
|       Logger.warn('[Watcher] Already watching library', library.name)
 | |
|       return
 | |
|     }
 | |
|     Logger.info(`[Watcher] Initializing watcher for "${library.name}".`)
 | |
| 
 | |
|     const folderPaths = library.libraryFolders.map((f) => f.path)
 | |
|     folderPaths.forEach((fp) => {
 | |
|       Logger.debug(`[Watcher] Init watcher for library folder path "${fp}"`)
 | |
|     })
 | |
|     const watcher = new Watcher(folderPaths, {
 | |
|       ignored: /(^|[\/\\])\../, // ignore dotfiles
 | |
|       renameDetection: true,
 | |
|       renameTimeout: 2000,
 | |
|       recursive: true,
 | |
|       ignoreInitial: true,
 | |
|       persistent: true
 | |
|     })
 | |
|     watcher
 | |
|       .on('add', (path) => {
 | |
|         this.onFileAdded(library.id, filePathToPOSIX(path))
 | |
|       })
 | |
|       .on('change', (path) => {
 | |
|         // This is triggered from metadata changes, not what we want
 | |
|       })
 | |
|       .on('unlink', (path) => {
 | |
|         this.onFileRemoved(library.id, filePathToPOSIX(path))
 | |
|       })
 | |
|       .on('rename', (path, pathNext) => {
 | |
|         this.onFileRename(library.id, filePathToPOSIX(path), filePathToPOSIX(pathNext))
 | |
|       })
 | |
|       .on('error', (error) => {
 | |
|         Logger.error(`[Watcher] ${error}`)
 | |
|       })
 | |
|       .on('ready', () => {
 | |
|         Logger.info(`[Watcher] "${library.name}" Ready`)
 | |
|       })
 | |
|       .on('close', () => {
 | |
|         Logger.debug(`[Watcher] "${library.name}" Closed`)
 | |
|       })
 | |
| 
 | |
|     this.libraryWatchers.push({
 | |
|       id: library.id,
 | |
|       name: library.name,
 | |
|       libraryFolders: library.libraryFolders,
 | |
|       paths: folderPaths,
 | |
|       watcher
 | |
|     })
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    *
 | |
|    * @param {import('./models/Library')[]} libraries
 | |
|    */
 | |
|   initWatcher(libraries) {
 | |
|     libraries.forEach((lib) => {
 | |
|       if (!lib.settings.disableWatcher) {
 | |
|         this.buildLibraryWatcher(lib)
 | |
|       }
 | |
|     })
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    *
 | |
|    * @param {import('./models/Library')} library
 | |
|    */
 | |
|   addLibrary(library) {
 | |
|     if (this.disabled || library.settings.disableWatcher) return
 | |
|     this.buildLibraryWatcher(library)
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    *
 | |
|    * @param {import('./models/Library')} library
 | |
|    */
 | |
|   updateLibrary(library) {
 | |
|     if (this.disabled) return
 | |
| 
 | |
|     const libwatcher = this.libraryWatchers.find((lib) => lib.id === library.id)
 | |
|     if (libwatcher) {
 | |
|       // Library watcher was disabled
 | |
|       if (library.settings.disableWatcher) {
 | |
|         Logger.info(`[Watcher] updateLibrary: Library "${library.name}" watcher disabled`)
 | |
|         libwatcher.watcher.close()
 | |
|         this.libraryWatchers = this.libraryWatchers.filter((lw) => lw.id !== libwatcher.id)
 | |
|         return
 | |
|       }
 | |
| 
 | |
|       libwatcher.name = library.name
 | |
| 
 | |
|       // If any folder paths were added or removed then re-init watcher
 | |
|       const folderPaths = library.libraryFolders.map((f) => f.path)
 | |
|       const pathsToAdd = folderPaths.filter((path) => !libwatcher.paths.includes(path))
 | |
|       const pathsRemoved = libwatcher.paths.filter((path) => !folderPaths.includes(path))
 | |
|       if (pathsToAdd.length || pathsRemoved.length) {
 | |
|         Logger.info(`[Watcher] Re-Initializing watcher for "${library.name}".`)
 | |
| 
 | |
|         libwatcher.watcher.close()
 | |
|         this.libraryWatchers = this.libraryWatchers.filter((lw) => lw.id !== libwatcher.id)
 | |
|         this.buildLibraryWatcher(library)
 | |
|       }
 | |
|     } else if (!library.settings.disableWatcher) {
 | |
|       // Library watcher was enabled
 | |
|       Logger.info(`[Watcher] updateLibrary: Library "${library.name}" watcher enabled - initializing`)
 | |
|       this.buildLibraryWatcher(library)
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    *
 | |
|    * @param {import('./models/Library')} library
 | |
|    */
 | |
|   removeLibrary(library) {
 | |
|     if (this.disabled) return
 | |
|     var libwatcher = this.libraryWatchers.find((lib) => lib.id === library.id)
 | |
|     if (libwatcher) {
 | |
|       Logger.info(`[Watcher] Removed watcher for "${library.name}"`)
 | |
|       libwatcher.watcher.close()
 | |
|       this.libraryWatchers = this.libraryWatchers.filter((lib) => lib.id !== library.id)
 | |
|     } else {
 | |
|       Logger.error(`[Watcher] Library watcher not found for "${library.name}"`)
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   close() {
 | |
|     return this.libraryWatchers.map((lib) => lib.watcher.close())
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Watcher detected file added
 | |
|    *
 | |
|    * @param {string} libraryId
 | |
|    * @param {string} path
 | |
|    */
 | |
|   onFileAdded(libraryId, path) {
 | |
|     if (this.checkShouldIgnorePath(path)) {
 | |
|       return
 | |
|     }
 | |
|     Logger.debug('[Watcher] File Added', path)
 | |
|     if (!this.addFileUpdate(libraryId, path, 'added')) {
 | |
|       return
 | |
|     }
 | |
| 
 | |
|     if (!this.filesBeingAdded.has(path)) {
 | |
|       this.filesBeingAdded.add(path)
 | |
|       this.waitForFileToAdd(path)
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Watcher detected file removed
 | |
|    *
 | |
|    * @param {string} libraryId
 | |
|    * @param {string} path
 | |
|    */
 | |
|   onFileRemoved(libraryId, path) {
 | |
|     if (this.checkShouldIgnorePath(path)) {
 | |
|       return
 | |
|     }
 | |
|     Logger.debug('[Watcher] File Removed', path)
 | |
|     this.addFileUpdate(libraryId, path, 'deleted')
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Watcher detected file renamed
 | |
|    *
 | |
|    * @param {string} libraryId
 | |
|    * @param {string} path
 | |
|    */
 | |
|   onFileRename(libraryId, pathFrom, pathTo) {
 | |
|     if (this.checkShouldIgnorePath(pathTo)) {
 | |
|       return
 | |
|     }
 | |
|     Logger.debug(`[Watcher] Rename ${pathFrom} => ${pathTo}`)
 | |
|     this.addFileUpdate(libraryId, pathTo, 'renamed')
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Get mtimeMs from an added file every 3 seconds until it is no longer changing
 | |
|    * Times out after 600s
 | |
|    *
 | |
|    * @param {string} path
 | |
|    * @param {number} [lastMTimeMs=0]
 | |
|    * @param {number} [loop=0]
 | |
|    */
 | |
|   async waitForFileToAdd(path, lastMTimeMs = 0, loop = 0) {
 | |
|     // Safety to catch infinite loop (600s)
 | |
|     if (loop >= 200) {
 | |
|       Logger.warn(`[Watcher] Waiting to add file at "${path}" timeout (loop ${loop}) - bailing`)
 | |
|       this.pendingFileUpdates = this.pendingFileUpdates.filter((pfu) => pfu.path !== path)
 | |
|       return this.filesBeingAdded.delete(path)
 | |
|     }
 | |
| 
 | |
|     const mtimeMs = await getFileMTimeMs(path)
 | |
|     if (mtimeMs === lastMTimeMs) {
 | |
|       if (lastMTimeMs) Logger.debug(`[Watcher] File finished adding at "${path}"`)
 | |
|       return this.filesBeingAdded.delete(path)
 | |
|     }
 | |
|     if (loop % 5 === 0) {
 | |
|       Logger.debug(`[Watcher] Waiting to add file at "${path}". mtimeMs=${mtimeMs} lastMTimeMs=${lastMTimeMs} (loop ${loop})`)
 | |
|     }
 | |
|     // Wait 3 seconds
 | |
|     await new Promise((resolve) => setTimeout(resolve, 3000))
 | |
|     this.waitForFileToAdd(path, mtimeMs, ++loop)
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Queue file update
 | |
|    *
 | |
|    * @param {string} libraryId
 | |
|    * @param {string} path
 | |
|    * @param {string} type
 | |
|    * @returns {boolean} - If file was added to pending updates
 | |
|    */
 | |
|   addFileUpdate(libraryId, path, type) {
 | |
|     if (this.pendingFilePaths.includes(path)) return false
 | |
| 
 | |
|     // Get file library
 | |
|     const libwatcher = this.libraryWatchers.find((lw) => lw.id === libraryId)
 | |
|     if (!libwatcher) {
 | |
|       Logger.error(`[Watcher] Invalid library id from watcher ${libraryId}`)
 | |
|       return false
 | |
|     }
 | |
| 
 | |
|     // Get file folder
 | |
|     const folder = libwatcher.libraryFolders.find((fold) => isSameOrSubPath(fold.path, path))
 | |
|     if (!folder) {
 | |
|       Logger.error(`[Watcher] New file folder not found in library "${libwatcher.name}" with path "${path}"`)
 | |
|       return false
 | |
|     }
 | |
| 
 | |
|     const folderPath = filePathToPOSIX(folder.path)
 | |
| 
 | |
|     const relPath = path.replace(folderPath, '')
 | |
| 
 | |
|     // Check for ignored extensions or directories, such as dotfiles and hidden directories
 | |
|     const shouldIgnore = shouldIgnoreFile(relPath)
 | |
|     if (shouldIgnore) {
 | |
|       Logger.debug(`[Watcher] Ignoring ${shouldIgnore} - "${relPath}"`)
 | |
|       return false
 | |
|     }
 | |
| 
 | |
|     Logger.debug(`[Watcher] Modified file in library "${libwatcher.name}" and folder "${folder.id}" with relPath "${relPath}"`)
 | |
| 
 | |
|     if (!this.pendingTask) {
 | |
|       const taskData = {
 | |
|         libraryId,
 | |
|         libraryName: libwatcher.name
 | |
|       }
 | |
|       const taskTitleString = {
 | |
|         text: `Scanning file changes in "${libwatcher.name}"`,
 | |
|         key: 'MessageTaskScanningFileChanges',
 | |
|         subs: [libwatcher.name]
 | |
|       }
 | |
|       this.pendingTask = TaskManager.createAndAddTask('watcher-scan', taskTitleString, null, true, taskData)
 | |
|     }
 | |
|     this.pendingFileUpdates.push({
 | |
|       path,
 | |
|       relPath,
 | |
|       folderId: folder.id,
 | |
|       libraryId,
 | |
|       type
 | |
|     })
 | |
| 
 | |
|     this.handlePendingFileUpdatesTimeout()
 | |
|     return true
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Wait X seconds before notifying scanner that files changed
 | |
|    * reset timer if files are still copying
 | |
|    */
 | |
|   handlePendingFileUpdatesTimeout() {
 | |
|     clearTimeout(this.pendingTimeout)
 | |
|     this.pendingTimeout = setTimeout(() => {
 | |
|       // Check that files are not still being added
 | |
|       if (this.pendingFileUpdates.some((pfu) => this.filesBeingAdded.has(pfu.path))) {
 | |
|         Logger.debug(`[Watcher] Still waiting for pending files "${[...this.filesBeingAdded].join(', ')}"`)
 | |
|         return this.handlePendingFileUpdatesTimeout()
 | |
|       }
 | |
| 
 | |
|       if (this.pendingFileUpdates.length) {
 | |
|         this.emit('scanFilesChanged', this.pendingFileUpdates, this.pendingTask)
 | |
|       } else {
 | |
|         const taskFinishedString = {
 | |
|           text: 'No files to scan',
 | |
|           key: 'MessageTaskNoFilesToScan'
 | |
|         }
 | |
|         this.pendingTask.setFinished(taskFinishedString)
 | |
|         TaskManager.taskFinished(this.pendingTask)
 | |
|       }
 | |
|       this.pendingTask = null
 | |
|       this.pendingFileUpdates = []
 | |
|       this.filesBeingAdded.clear()
 | |
|     }, this.pendingDelay)
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    *
 | |
|    * @param {string} path
 | |
|    * @returns {boolean}
 | |
|    */
 | |
|   checkShouldIgnorePath(path) {
 | |
|     return !!this.ignoreDirs.find((dirpath) => {
 | |
|       return isSameOrSubPath(dirpath, path)
 | |
|     })
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * When scanning a library item folder these files should be ignored
 | |
|    * Either a podcast episode downloading or a file that is pending by the watcher
 | |
|    *
 | |
|    * @param {string} path
 | |
|    * @returns {boolean}
 | |
|    */
 | |
|   checkShouldIgnoreFilePath(path) {
 | |
|     if (this.pendingFilePaths.includes(path)) return true
 | |
|     return this.ignoreFilePathsDownloading.has(path)
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Convert to POSIX and remove trailing slash
 | |
|    * @param {string} path
 | |
|    * @returns {string}
 | |
|    */
 | |
|   cleanDirPath(path) {
 | |
|     path = filePathToPOSIX(path)
 | |
|     if (path.endsWith('/')) path = path.slice(0, -1)
 | |
|     return path
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Ignore this directory if files are picked up by watcher
 | |
|    * @param {string} path
 | |
|    */
 | |
|   addIgnoreDir(path) {
 | |
|     path = this.cleanDirPath(path)
 | |
|     this.pendingDirsToRemoveFromIgnore = this.pendingDirsToRemoveFromIgnore.filter((p) => p !== path)
 | |
|     if (this.ignoreDirs.includes(path)) {
 | |
|       // Already ignoring dir
 | |
|       return
 | |
|     }
 | |
|     Logger.debug(`[Watcher] addIgnoreDir: Ignoring directory "${path}"`)
 | |
|     this.ignoreDirs.push(path)
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * When downloading a podcast episode we dont want the scanner triggering for that podcast
 | |
|    * when the episode finishes the watcher may have a delayed response so a timeout is added
 | |
|    * to prevent the watcher from picking up the episode
 | |
|    *
 | |
|    * @param {string} path
 | |
|    */
 | |
|   removeIgnoreDir(path) {
 | |
|     path = this.cleanDirPath(path)
 | |
|     if (!this.ignoreDirs.includes(path)) {
 | |
|       Logger.debug(`[Watcher] removeIgnoreDir: Path is not being ignored "${path}"`)
 | |
|       return
 | |
|     }
 | |
| 
 | |
|     // Add a 5 second delay before removing the ignore from this dir
 | |
|     if (!this.pendingDirsToRemoveFromIgnore.includes(path)) {
 | |
|       this.pendingDirsToRemoveFromIgnore.push(path)
 | |
|     }
 | |
| 
 | |
|     clearTimeout(this.removeFromIgnoreTimer)
 | |
|     this.removeFromIgnoreTimer = setTimeout(() => {
 | |
|       if (this.pendingDirsToRemoveFromIgnore.includes(path)) {
 | |
|         this.pendingDirsToRemoveFromIgnore = this.pendingDirsToRemoveFromIgnore.filter((p) => p !== path)
 | |
|         Logger.debug(`[Watcher] removeIgnoreDir: No longer ignoring directory "${path}"`)
 | |
|         this.ignoreDirs = this.ignoreDirs.filter((p) => p !== path)
 | |
|       }
 | |
|     }, 5000)
 | |
|   }
 | |
| }
 | |
| module.exports = new FolderWatcher()
 |