mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-26 08:12:25 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			768 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			768 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| const fs = require('fs-extra')
 | |
| const Path = require('path')
 | |
| 
 | |
| // Utils
 | |
| const Logger = require('./Logger')
 | |
| const { version } = require('../package.json')
 | |
| const audioFileScanner = require('./utils/audioFileScanner')
 | |
| const { groupFilesIntoAudiobookPaths, getAudiobookFileData, scanRootDir } = require('./utils/scandir')
 | |
| const { comparePaths, getIno, getId, secondsToTimestamp } = require('./utils/index')
 | |
| const { ScanResult, CoverDestination } = require('./utils/constants')
 | |
| 
 | |
| const BookFinder = require('./BookFinder')
 | |
| const Audiobook = require('./objects/Audiobook')
 | |
| 
 | |
| class Scanner {
 | |
|   constructor(AUDIOBOOK_PATH, METADATA_PATH, db, coverController, emitter) {
 | |
|     this.AudiobookPath = AUDIOBOOK_PATH
 | |
|     this.MetadataPath = METADATA_PATH
 | |
|     this.BookMetadataPath = Path.posix.join(this.MetadataPath.replace(/\\/g, '/'), 'books')
 | |
| 
 | |
|     this.db = db
 | |
|     this.coverController = coverController
 | |
|     this.emitter = emitter
 | |
| 
 | |
|     this.cancelScan = false
 | |
|     this.cancelLibraryScan = {}
 | |
|     this.librariesScanning = []
 | |
| 
 | |
|     this.bookFinder = new BookFinder()
 | |
|   }
 | |
| 
 | |
|   get audiobooks() {
 | |
|     return this.db.audiobooks
 | |
|   }
 | |
| 
 | |
|   getCoverDirectory(audiobook) {
 | |
|     if (this.db.serverSettings.coverDestination === CoverDestination.AUDIOBOOK) {
 | |
|       return {
 | |
|         fullPath: audiobook.fullPath,
 | |
|         relPath: '/s/book/' + audiobook.id
 | |
|       }
 | |
|     } else {
 | |
|       return {
 | |
|         fullPath: Path.posix.join(this.BookMetadataPath, audiobook.id),
 | |
|         relPath: Path.posix.join('/metadata', 'books', audiobook.id)
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   async setAudioFileInos(audiobookDataAudioFiles, audiobookAudioFiles) {
 | |
|     for (let i = 0; i < audiobookDataAudioFiles.length; i++) {
 | |
|       var abdFile = audiobookDataAudioFiles[i]
 | |
|       var matchingFile = audiobookAudioFiles.find(af => comparePaths(af.path, abdFile.path))
 | |
|       if (matchingFile) {
 | |
|         if (!matchingFile.ino) {
 | |
|           matchingFile.ino = await getIno(matchingFile.fullPath)
 | |
|         }
 | |
|         abdFile.ino = matchingFile.ino
 | |
|       } else {
 | |
|         abdFile.ino = await getIno(abdFile.fullPath)
 | |
|         if (!abdFile.ino) {
 | |
|           Logger.error('[Scanner] Invalid abdFile ino - ignoring abd audio file', abdFile.path)
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|     return audiobookDataAudioFiles.filter(abdFile => !!abdFile.ino)
 | |
|   }
 | |
| 
 | |
|   // Only updates audio files with matching paths
 | |
|   syncAudiobookInodeValues(audiobook, { audioFiles, otherFiles }) {
 | |
|     var filesUpdated = 0
 | |
| 
 | |
|     // Sync audio files & audio tracks with updated inodes
 | |
|     audiobook._audioFiles.forEach((audioFile) => {
 | |
|       var matchingAudioFile = audioFiles.find(af => af.ino !== audioFile.ino && af.path === audioFile.path)
 | |
|       if (matchingAudioFile) {
 | |
|         // Audio Track should always have the same ino as the equivalent audio file (not all audio files have a track)
 | |
|         var audioTrack = audiobook.tracks.find(t => t.ino === audioFile.ino)
 | |
|         if (audioTrack) {
 | |
|           Logger.debug(`[Scanner] Found audio file & track with mismatched inode "${audioFile.filename}" - updating it`)
 | |
|           audioTrack.ino = matchingAudioFile.ino
 | |
|           filesUpdated++
 | |
|         } else {
 | |
|           Logger.debug(`[Scanner] Found audio file with mismatched inode "${audioFile.filename}" - updating it`)
 | |
|         }
 | |
| 
 | |
|         audioFile.ino = matchingAudioFile.ino
 | |
|         filesUpdated++
 | |
|       }
 | |
|     })
 | |
| 
 | |
|     // Sync other files with updated inodes
 | |
|     audiobook._otherFiles.forEach((otherFile) => {
 | |
|       var matchingOtherFile = otherFiles.find(of => of.ino !== otherFile.ino && of.path === otherFile.path)
 | |
|       if (matchingOtherFile) {
 | |
|         Logger.debug(`[Scanner] Found other file with mismatched inode "${otherFile.filename}" - updating it`)
 | |
|         otherFile.ino = matchingOtherFile.ino
 | |
|         filesUpdated++
 | |
|       }
 | |
|     })
 | |
| 
 | |
|     return filesUpdated
 | |
|   }
 | |
| 
 | |
|   async searchForCover(audiobook) {
 | |
|     var options = {
 | |
|       titleDistance: 2,
 | |
|       authorDistance: 2
 | |
|     }
 | |
|     var results = await this.bookFinder.findCovers('openlibrary', audiobook.title, audiobook.authorFL, options)
 | |
|     if (results.length) {
 | |
|       Logger.debug(`[Scanner] Found best cover for "${audiobook.title}"`)
 | |
| 
 | |
|       // If the first cover result fails, attempt to download the second
 | |
|       for (let i = 0; i < results.length && i < 2; i++) {
 | |
| 
 | |
|         // Downloads and updates the book cover
 | |
|         var result = await this.coverController.downloadCoverFromUrl(audiobook, results[i])
 | |
| 
 | |
|         if (result.error) {
 | |
|           Logger.error(`[Scanner] Failed to download cover from url "${results[i]}" | Attempt ${i + 1}`, result.error)
 | |
|         } else {
 | |
|           return true
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|     return false
 | |
|   }
 | |
| 
 | |
|   async scanExistingAudiobook(existingAudiobook, audiobookData, hasUpdatedIno, hasUpdatedLibraryOrFolder, forceAudioFileScan) {
 | |
|     // Always sync files and inode values
 | |
|     var filesInodeUpdated = this.syncAudiobookInodeValues(existingAudiobook, audiobookData)
 | |
|     if (hasUpdatedIno || filesInodeUpdated > 0) {
 | |
|       Logger.info(`[Scanner] Updating inode value for "${existingAudiobook.title}" - ${filesInodeUpdated} files updated`)
 | |
|       hasUpdatedIno = true
 | |
|     }
 | |
| 
 | |
|     // TEMP: Check if is older audiobook and needs force rescan
 | |
|     if (!forceAudioFileScan && (!existingAudiobook.scanVersion || existingAudiobook.checkHasOldCoverPath())) {
 | |
|       Logger.info(`[Scanner] Force rescan for "${existingAudiobook.title}" | Last scan v${existingAudiobook.scanVersion} | Old Cover Path ${!!existingAudiobook.checkHasOldCoverPath()}`)
 | |
|       forceAudioFileScan = true
 | |
|     }
 | |
| 
 | |
|     // inode is required
 | |
|     audiobookData.audioFiles = audiobookData.audioFiles.filter(af => af.ino)
 | |
| 
 | |
|     // No valid ebook and audio files found, mark as incomplete
 | |
|     var ebookFiles = audiobookData.otherFiles.filter(f => f.filetype === 'ebook')
 | |
|     if (!audiobookData.audioFiles.length && !ebookFiles.length) {
 | |
|       Logger.error(`[Scanner] "${existingAudiobook.title}" no valid book files found - marking as incomplete`)
 | |
|       existingAudiobook.setLastScan(version)
 | |
|       existingAudiobook.isIncomplete = true
 | |
|       await this.db.updateAudiobook(existingAudiobook)
 | |
|       this.emitter('audiobook_updated', existingAudiobook.toJSONMinified())
 | |
|       return ScanResult.UPDATED
 | |
|     } else if (existingAudiobook.isIncomplete) { // Was incomplete but now is not
 | |
|       Logger.info(`[Scanner] "${existingAudiobook.title}" was incomplete but now has book files`)
 | |
|       existingAudiobook.isIncomplete = false
 | |
|     }
 | |
| 
 | |
|     // Check for audio files that were removed
 | |
|     var abdAudioFileInos = audiobookData.audioFiles.map(af => af.ino)
 | |
|     var removedAudioFiles = existingAudiobook.audioFiles.filter(file => !abdAudioFileInos.includes(file.ino))
 | |
|     if (removedAudioFiles.length) {
 | |
|       Logger.info(`[Scanner] ${removedAudioFiles.length} audio files removed for audiobook "${existingAudiobook.title}"`)
 | |
|       removedAudioFiles.forEach((af) => existingAudiobook.removeAudioFile(af))
 | |
|     }
 | |
| 
 | |
|     // Check for mismatched audio tracks - tracks with no matching audio file
 | |
|     var removedAudioTracks = existingAudiobook.tracks.filter(track => !abdAudioFileInos.includes(track.ino))
 | |
|     if (removedAudioTracks.length) {
 | |
|       Logger.error(`[Scanner] ${removedAudioTracks.length} tracks removed no matching audio file for audiobook "${existingAudiobook.title}"`)
 | |
|       removedAudioTracks.forEach((at) => existingAudiobook.removeAudioTrack(at))
 | |
|     }
 | |
| 
 | |
|     // Check for new audio files and sync existing audio files
 | |
|     var newAudioFiles = []
 | |
|     var hasUpdatedAudioFiles = false
 | |
|     audiobookData.audioFiles.forEach((file) => {
 | |
|       var existingAudioFile = existingAudiobook.getAudioFileByIno(file.ino)
 | |
|       if (existingAudioFile) { // Audio file exists, sync path (path may have been renamed)
 | |
|         if (existingAudiobook.syncAudioFile(existingAudioFile, file)) {
 | |
|           hasUpdatedAudioFiles = true
 | |
|         }
 | |
|       } else {
 | |
|         // New audio file, triple check for matching file path
 | |
|         var audioFileWithMatchingPath = existingAudiobook.getAudioFileByPath(file.fullPath)
 | |
|         if (audioFileWithMatchingPath) {
 | |
|           Logger.warn(`[Scanner] Audio file with path already exists with different inode, New: "${file.filename}" (${file.ino}) | Existing: ${audioFileWithMatchingPath.filename} (${audioFileWithMatchingPath.ino})`)
 | |
|         } else {
 | |
|           newAudioFiles.push(file)
 | |
|         }
 | |
|       }
 | |
|     })
 | |
| 
 | |
| 
 | |
|     // Sync other files (all files that are not audio files) - Updates cover path
 | |
|     var hasOtherFileUpdates = false
 | |
|     var otherFilesUpdated = await existingAudiobook.syncOtherFiles(audiobookData.otherFiles, this.MetadataPath, false, forceAudioFileScan)
 | |
|     if (otherFilesUpdated) {
 | |
|       hasOtherFileUpdates = true
 | |
|     }
 | |
| 
 | |
|     // Rescan audio file metadata
 | |
|     if (forceAudioFileScan) {
 | |
|       Logger.info(`[Scanner] Rescanning ${existingAudiobook.audioFiles.length} audio files for "${existingAudiobook.title}"`)
 | |
|       var numAudioFilesUpdated = await audioFileScanner.rescanAudioFiles(existingAudiobook)
 | |
| 
 | |
|       // Set book details from metadata pulled from audio files
 | |
|       var bookMetadataUpdated = existingAudiobook.setDetailsFromFileMetadata()
 | |
|       if (bookMetadataUpdated) {
 | |
|         Logger.debug(`[Scanner] Book Metadata Updated for "${existingAudiobook.title}"`)
 | |
|         hasUpdatedAudioFiles = true
 | |
|       }
 | |
| 
 | |
|       if (numAudioFilesUpdated > 0) {
 | |
|         Logger.info(`[Scanner] Rescan complete, ${numAudioFilesUpdated} audio files were updated for "${existingAudiobook.title}"`)
 | |
|         hasUpdatedAudioFiles = true
 | |
| 
 | |
|         // Use embedded cover art if audiobook has no cover
 | |
|         if (existingAudiobook.hasEmbeddedCoverArt && !existingAudiobook.cover) {
 | |
|           var outputCoverDirs = this.getCoverDirectory(existingAudiobook)
 | |
|           var relativeDir = await existingAudiobook.saveEmbeddedCoverArt(outputCoverDirs.fullPath, outputCoverDirs.relPath)
 | |
|           if (relativeDir) {
 | |
|             Logger.debug(`[Scanner] Saved embedded cover art "${relativeDir}"`)
 | |
|           }
 | |
|         }
 | |
|       } else {
 | |
|         Logger.info(`[Scanner] Rescan complete, audio files were up to date for "${existingAudiobook.title}"`)
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Scan and add new audio files found and set tracks
 | |
|     if (newAudioFiles.length) {
 | |
|       Logger.info(`[Scanner] ${newAudioFiles.length} new audio files were found for audiobook "${existingAudiobook.title}"`)
 | |
|       await audioFileScanner.scanAudioFiles(existingAudiobook, newAudioFiles)
 | |
|     }
 | |
| 
 | |
|     // After scanning audio files, some may no longer be valid
 | |
|     //   so make sure the directory still has valid book files
 | |
|     if (!existingAudiobook.tracks.length && !ebookFiles.length) {
 | |
|       Logger.error(`[Scanner] "${existingAudiobook.title}" no valid book files found after update - marking as incomplete`)
 | |
|       existingAudiobook.setLastScan(version)
 | |
|       existingAudiobook.isIncomplete = true
 | |
|       await this.db.updateAudiobook(existingAudiobook)
 | |
|       this.emitter('audiobook_updated', existingAudiobook.toJSONMinified())
 | |
|       return ScanResult.UPDATED
 | |
|     }
 | |
| 
 | |
|     var hasUpdates = hasOtherFileUpdates || hasUpdatedIno || hasUpdatedLibraryOrFolder || removedAudioFiles.length || removedAudioTracks.length || newAudioFiles.length || hasUpdatedAudioFiles
 | |
| 
 | |
|     // Check that audio tracks are in sequential order with no gaps
 | |
|     if (existingAudiobook.checkUpdateMissingTracks()) {
 | |
|       Logger.info(`[Scanner] "${existingAudiobook.title}" missing parts updated`)
 | |
|       hasUpdates = true
 | |
|     }
 | |
| 
 | |
|     // Syncs path and fullPath
 | |
|     if (existingAudiobook.syncPaths(audiobookData)) {
 | |
|       hasUpdates = true
 | |
|     }
 | |
| 
 | |
|     // If audiobook was missing before, it is now found
 | |
|     if (existingAudiobook.isMissing) {
 | |
|       existingAudiobook.isMissing = false
 | |
|       hasUpdates = true
 | |
|       Logger.info(`[Scanner] "${existingAudiobook.title}" was missing but now it is found`)
 | |
|     }
 | |
| 
 | |
|     if (hasUpdates || version !== existingAudiobook.scanVersion) {
 | |
|       existingAudiobook.setChapters()
 | |
|       existingAudiobook.setLastScan(version)
 | |
|       await this.db.updateAudiobook(existingAudiobook)
 | |
| 
 | |
|       Logger.info(`[Scanner] "${existingAudiobook.title}" was updated`)
 | |
|       this.emitter('audiobook_updated', existingAudiobook.toJSONMinified())
 | |
|       return ScanResult.UPDATED
 | |
|     }
 | |
| 
 | |
|     return ScanResult.UPTODATE
 | |
|   }
 | |
| 
 | |
|   async scanNewAudiobook(audiobookData) {
 | |
|     var ebookFiles = audiobookData.otherFiles.map(f => f.filetype === 'ebook')
 | |
|     if (!audiobookData.audioFiles.length && !ebookFiles.length) {
 | |
|       Logger.error('[Scanner] No valid audio files and ebooks for Audiobook', audiobookData.path)
 | |
|       return null
 | |
|     }
 | |
| 
 | |
|     var audiobook = new Audiobook()
 | |
|     audiobook.setData(audiobookData)
 | |
| 
 | |
|     // Scan audio files and set tracks, pulls metadata
 | |
|     await audioFileScanner.scanAudioFiles(audiobook, audiobookData.audioFiles)
 | |
| 
 | |
|     if (!audiobook.tracks.length && !audiobook.ebooks.length) {
 | |
|       Logger.warn('[Scanner] Invalid audiobook, no valid audio tracks and ebook files', audiobook.title)
 | |
|       return null
 | |
|     }
 | |
| 
 | |
|     // Look for desc.txt and reader.txt and update
 | |
|     await audiobook.saveDataFromTextFiles(false)
 | |
| 
 | |
|     // Extract embedded cover art if cover is not already in directory
 | |
|     if (audiobook.hasEmbeddedCoverArt && !audiobook.cover) {
 | |
|       var outputCoverDirs = this.getCoverDirectory(audiobook)
 | |
|       var relativeDir = await audiobook.saveEmbeddedCoverArt(outputCoverDirs.fullPath, outputCoverDirs.relPath)
 | |
|       if (relativeDir) {
 | |
|         Logger.debug(`[Scanner] Saved embedded cover art "${relativeDir}"`)
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Set book details from metadata pulled from audio files
 | |
|     audiobook.setDetailsFromFileMetadata()
 | |
| 
 | |
|     // Check for gaps in track numbers
 | |
|     audiobook.checkUpdateMissingTracks()
 | |
| 
 | |
|     // Set chapters from audio files
 | |
|     audiobook.setChapters()
 | |
| 
 | |
|     audiobook.setLastScan(version)
 | |
| 
 | |
|     Logger.info(`[Scanner] Audiobook "${audiobook.title}" Scanned (${audiobook.sizePretty}) [${audiobook.durationPretty}]`)
 | |
|     await this.db.insertEntity('audiobook', audiobook)
 | |
|     this.emitter('audiobook_added', audiobook.toJSONMinified())
 | |
|     return audiobook
 | |
|   }
 | |
| 
 | |
|   async scanAudiobookData(audiobookData, forceAudioFileScan = false) {
 | |
|     var scannerFindCovers = this.db.serverSettings.scannerFindCovers
 | |
|     var libraryId = audiobookData.libraryId
 | |
|     var folderId = audiobookData.folderId
 | |
| 
 | |
|     var hasUpdatedLibraryOrFolder = false
 | |
| 
 | |
|     var existingAudiobook = this.audiobooks.find(ab => ab.ino === audiobookData.ino)
 | |
| 
 | |
|     // Make sure existing audiobook has the same library & folder id
 | |
|     if (existingAudiobook && (existingAudiobook.libraryId !== libraryId || existingAudiobook.folderId !== folderId)) {
 | |
|       var existingAudiobookLibrary = this.db.libraries.find(lib => lib.id === existingAudiobook.libraryId)
 | |
| 
 | |
|       if (!existingAudiobookLibrary) {
 | |
|         Logger.error(`[Scanner] Audiobook "${existingAudiobook.title}" found in different library that no longer exists ${existingAudiobook.libraryId}`)
 | |
|       } else if (existingAudiobook.libraryId !== libraryId) {
 | |
|         Logger.warn(`[Scanner] Audiobook "${existingAudiobook.title}" found in different library "${existingAudiobookLibrary.name}"`)
 | |
|       } else {
 | |
|         Logger.warn(`[Scanner] Audiobook "${existingAudiobook.title}" found in different folder "${existingAudiobook.folderId}" of library "${existingAudiobookLibrary.name}"`)
 | |
|       }
 | |
| 
 | |
|       existingAudiobook.libraryId = libraryId
 | |
|       existingAudiobook.folderId = folderId
 | |
|       hasUpdatedLibraryOrFolder = true
 | |
|       Logger.info(`[Scanner] Updated Audiobook "${existingAudiobook.title}" library and folder to "${libraryId}" "${folderId}"`)
 | |
|     }
 | |
| 
 | |
|     // inode value may change when using shared drives, update inode if matching path is found
 | |
|     // Note: inode will not change on rename
 | |
|     var hasUpdatedIno = false
 | |
|     if (!existingAudiobook) {
 | |
|       // check an audiobook exists with matching path, then update inodes
 | |
|       existingAudiobook = this.audiobooks.find(a => a.path === audiobookData.path)
 | |
|       if (existingAudiobook) {
 | |
|         var oldIno = existingAudiobook.ino
 | |
|         existingAudiobook.ino = audiobookData.ino
 | |
|         Logger.debug(`[Scanner] Scan Audiobook Data: Updated inode from "${oldIno}" to "${existingAudiobook.ino}"`)
 | |
|         hasUpdatedIno = true
 | |
| 
 | |
|         if (existingAudiobook.libraryId !== libraryId || existingAudiobook.folderId !== folderId) {
 | |
|           Logger.warn(`[Scanner] Audiobook found by path is in a different library or folder, ${existingAudiobook.libraryId}/${existingAudiobook.folderId} should be ${libraryId}/${folderId}`)
 | |
| 
 | |
|           existingAudiobook.libraryId = libraryId
 | |
|           existingAudiobook.folderId = folderId
 | |
|           hasUpdatedLibraryOrFolder = true
 | |
|           Logger.info(`[Scanner] Updated Audiobook "${existingAudiobook.title}" library and folder to "${libraryId}" "${folderId}"`)
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     var scanResult = null
 | |
|     var finalAudiobook = null
 | |
| 
 | |
|     if (existingAudiobook) {
 | |
|       finalAudiobook = existingAudiobook
 | |
| 
 | |
|       scanResult = await this.scanExistingAudiobook(existingAudiobook, audiobookData, hasUpdatedIno, hasUpdatedLibraryOrFolder, forceAudioFileScan)
 | |
| 
 | |
|       if (scanResult === ScanResult.REMOVED || scanResult === ScanResult.NOTHING) {
 | |
|         finalAudiobook = null
 | |
|       }
 | |
|     } else {
 | |
|       finalAudiobook = await this.scanNewAudiobook(audiobookData)
 | |
| 
 | |
|       scanResult = finalAudiobook ? ScanResult.ADDED : ScanResult.NOTHING
 | |
| 
 | |
|       if (finalAudiobook === ScanResult.NOTHING) {
 | |
|         finalAudiobook = null
 | |
|         scanResult = ScanResult.NOTHING
 | |
|       } else {
 | |
|         scanResult = ScanResult.ADDED
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Scan for cover if enabled and has no cover
 | |
|     if (finalAudiobook && scannerFindCovers && !finalAudiobook.cover) {
 | |
|       if (finalAudiobook.book.shouldSearchForCover) {
 | |
|         var updatedCover = await this.searchForCover(finalAudiobook)
 | |
| 
 | |
|         finalAudiobook.book.updateLastCoverSearch(updatedCover)
 | |
| 
 | |
|         if (updatedCover && scanResult === ScanResult.UPTODATE) {
 | |
|           scanResult = ScanResult.UPDATED
 | |
|         }
 | |
|         await this.db.updateAudiobook(finalAudiobook)
 | |
|         this.emitter('audiobook_updated', finalAudiobook.toJSONMinified())
 | |
|       } else {
 | |
|         Logger.debug(`[Scanner] Audiobook "${finalAudiobook.title}" cover already scanned - not re-scanning`)
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return scanResult
 | |
|   }
 | |
| 
 | |
|   async scan(libraryId, forceAudioFileScan = false) {
 | |
|     if (this.librariesScanning.includes(libraryId)) {
 | |
|       Logger.error(`[Scanner] Already scanning ${libraryId}`)
 | |
|       return
 | |
|     }
 | |
| 
 | |
|     var library = this.db.libraries.find(lib => lib.id === libraryId)
 | |
|     if (!library) {
 | |
|       Logger.error(`[Scanner] Library not found for scan ${libraryId}`)
 | |
|       return
 | |
|     } else if (!library.folders.length) {
 | |
|       Logger.warn(`[Scanner] Library has no folders to scan "${library.name}"`)
 | |
|       return
 | |
|     }
 | |
| 
 | |
|     var scanPayload = {
 | |
|       id: libraryId,
 | |
|       name: library.name,
 | |
|       folders: library.folders.length
 | |
|     }
 | |
|     this.emitter('scan_start', scanPayload)
 | |
|     Logger.info(`[Scanner] Starting scan of library "${library.name}" with ${library.folders.length} folders`)
 | |
| 
 | |
|     library.lastScan = Date.now()
 | |
|     await this.db.updateEntity('library', library)
 | |
| 
 | |
|     this.librariesScanning.push(scanPayload)
 | |
| 
 | |
|     var audiobooksInLibrary = this.db.audiobooks.filter(ab => ab.libraryId === libraryId)
 | |
| 
 | |
|     // TODO: This temporary fix from pre-release should be removed soon, "checkUpdateInos"
 | |
|     if (audiobooksInLibrary.length) {
 | |
|       for (let i = 0; i < audiobooksInLibrary.length; i++) {
 | |
|         var ab = audiobooksInLibrary[i]
 | |
|         // Update ino if inos are not set
 | |
|         var shouldUpdateIno = ab.hasMissingIno
 | |
|         if (shouldUpdateIno) {
 | |
|           var filesWithMissingIno = ab.getFilesWithMissingIno()
 | |
| 
 | |
|           Logger.debug(`\nUpdating inos for "${ab.title}"`)
 | |
|           Logger.debug(`In Scan, Files with missing inode`, filesWithMissingIno)
 | |
| 
 | |
|           var hasUpdates = await ab.checkUpdateInos()
 | |
|           if (hasUpdates) {
 | |
|             await this.db.updateAudiobook(ab)
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     const scanStart = Date.now()
 | |
|     var audiobookDataFound = []
 | |
|     for (let i = 0; i < library.folders.length; i++) {
 | |
|       var folder = library.folders[i]
 | |
|       var abDataFoundInFolder = await scanRootDir(folder, this.db.serverSettings)
 | |
|       Logger.debug(`[Scanner] ${abDataFoundInFolder.length} ab data found in folder "${folder.fullPath}"`)
 | |
|       audiobookDataFound = audiobookDataFound.concat(abDataFoundInFolder)
 | |
|     }
 | |
| 
 | |
|     // Remove audiobooks with no inode
 | |
|     audiobookDataFound = audiobookDataFound.filter(abd => abd.ino)
 | |
| 
 | |
|     if (this.cancelLibraryScan[libraryId]) {
 | |
|       Logger.info(`[Scanner] Canceling scan ${libraryId}`)
 | |
|       delete this.cancelLibraryScan[libraryId]
 | |
|       this.librariesScanning = this.librariesScanning.filter(l => l.id !== libraryId)
 | |
|       this.emitter('scan_complete', { id: libraryId, name: library.name, results: null })
 | |
|       return null
 | |
|     }
 | |
| 
 | |
|     var scanResults = {
 | |
|       removed: 0,
 | |
|       updated: 0,
 | |
|       added: 0,
 | |
|       missing: 0
 | |
|     }
 | |
| 
 | |
|     // Check for removed audiobooks
 | |
|     for (let i = 0; i < audiobooksInLibrary.length; i++) {
 | |
|       var audiobook = audiobooksInLibrary[i]
 | |
|       var dataFound = audiobookDataFound.find(abd => abd.ino === audiobook.ino || comparePaths(abd.path, audiobook.path))
 | |
|       if (!dataFound) {
 | |
|         Logger.info(`[Scanner] Audiobook "${audiobook.title}" is missing`)
 | |
|         audiobook.isMissing = true
 | |
|         audiobook.lastUpdate = Date.now()
 | |
|         scanResults.missing++
 | |
|         await this.db.updateAudiobook(audiobook)
 | |
|         this.emitter('audiobook_updated', audiobook.toJSONMinified())
 | |
|       }
 | |
|       if (this.cancelLibraryScan[libraryId]) {
 | |
|         Logger.info(`[Scanner] Canceling scan ${libraryId}`)
 | |
|         delete this.cancelLibraryScan[libraryId]
 | |
|         this.librariesScanning = this.librariesScanning.filter(l => l.id !== libraryId)
 | |
|         this.emitter('scan_complete', { id: libraryId, name: library.name, results: scanResults })
 | |
|         return
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Check for new and updated audiobooks
 | |
|     for (let i = 0; i < audiobookDataFound.length; i++) {
 | |
|       var result = await this.scanAudiobookData(audiobookDataFound[i], forceAudioFileScan)
 | |
|       if (result === ScanResult.ADDED) scanResults.added++
 | |
|       if (result === ScanResult.REMOVED) scanResults.removed++
 | |
|       if (result === ScanResult.UPDATED) scanResults.updated++
 | |
| 
 | |
|       var progress = Math.round(100 * (i + 1) / audiobookDataFound.length)
 | |
|       this.emitter('scan_progress', {
 | |
|         id: libraryId,
 | |
|         name: library.name,
 | |
|         progress: {
 | |
|           total: audiobookDataFound.length,
 | |
|           done: i + 1,
 | |
|           progress
 | |
|         }
 | |
|       })
 | |
|       if (this.cancelLibraryScan[libraryId]) {
 | |
|         Logger.info(`[Scanner] Canceling scan ${libraryId}`)
 | |
|         delete this.cancelLibraryScan[libraryId]
 | |
|         break
 | |
|       }
 | |
|     }
 | |
|     const scanElapsed = Math.floor((Date.now() - scanStart) / 1000)
 | |
|     Logger.info(`[Scanned] Finished | ${scanResults.added} added | ${scanResults.updated} updated | ${scanResults.removed} removed | ${scanResults.missing} missing | elapsed: ${secondsToTimestamp(scanElapsed)}`)
 | |
|     this.librariesScanning = this.librariesScanning.filter(l => l.id !== libraryId)
 | |
|     this.emitter('scan_complete', { id: libraryId, name: library.name, results: scanResults })
 | |
|   }
 | |
| 
 | |
|   async scanAudiobookById(audiobookId) {
 | |
|     const audiobook = this.db.audiobooks.find(ab => ab.id === audiobookId)
 | |
|     if (!audiobook) {
 | |
|       Logger.error(`[Scanner] Scan audiobook by id not found ${audiobookId}`)
 | |
|       return ScanResult.NOTHING
 | |
|     }
 | |
|     const library = this.db.libraries.find(lib => lib.id === audiobook.libraryId)
 | |
|     if (!library) {
 | |
|       Logger.error(`[Scanner] Scan audiobook by id library not found "${audiobook.libraryId}"`)
 | |
|       return ScanResult.NOTHING
 | |
|     }
 | |
|     const folder = library.folders.find(f => f.id === audiobook.folderId)
 | |
|     if (!folder) {
 | |
|       Logger.error(`[Scanner] Scan audiobook by id folder not found "${audiobook.folderId}" in library "${library.name}"`)
 | |
|       return ScanResult.NOTHING
 | |
|     }
 | |
|     if (!folder.libraryId) {
 | |
|       Logger.fatal(`[Scanner] Folder does not have a library id set...`, folder)
 | |
|       return ScanResult.NOTHING
 | |
|     }
 | |
| 
 | |
|     Logger.info(`[Scanner] Scanning Audiobook "${audiobook.title}"`)
 | |
|     return this.scanAudiobook(folder, audiobook.fullPath, true)
 | |
|   }
 | |
| 
 | |
|   async scanAudiobook(folder, audiobookFullPath, forceAudioFileScan = false) {
 | |
|     Logger.debug('[Scanner] scanAudiobook', audiobookFullPath)
 | |
|     var audiobookData = await getAudiobookFileData(folder, audiobookFullPath, this.db.serverSettings)
 | |
|     if (!audiobookData) {
 | |
|       return ScanResult.NOTHING
 | |
|     }
 | |
|     return this.scanAudiobookData(audiobookData, forceAudioFileScan)
 | |
|   }
 | |
| 
 | |
|   async scanFolderUpdates(libraryId, folderId, fileUpdateBookGroup) {
 | |
|     var library = this.db.libraries.find(lib => lib.id === libraryId)
 | |
|     if (!library) {
 | |
|       Logger.error(`[Scanner] Library "${libraryId}" not found for scan library updates`)
 | |
|       return null
 | |
|     }
 | |
|     var folder = library.folders.find(f => f.id === folderId)
 | |
|     if (!folder) {
 | |
|       Logger.error(`[Scanner] Folder "${folderId}" not found in library "${library.name}" for scan library updates`)
 | |
|       return null
 | |
|     }
 | |
| 
 | |
|     Logger.debug(`[Scanner] Scanning file update groups in folder "${folder.id}" of library "${library.name}"`)
 | |
| 
 | |
|     var bookGroupingResults = {}
 | |
|     for (const bookDir in fileUpdateBookGroup) {
 | |
|       var fullPath = Path.posix.join(folder.fullPath.replace(/\\/g, '/'), bookDir)
 | |
| 
 | |
|       // Check if book dir group is already an audiobook or in a subdir of an audiobook
 | |
|       var existingAudiobook = this.db.audiobooks.find(ab => fullPath.startsWith(ab.fullPath))
 | |
|       if (existingAudiobook) {
 | |
| 
 | |
|         // Is the audiobook exactly - check if was deleted
 | |
|         if (existingAudiobook.fullPath === fullPath) {
 | |
|           var exists = await fs.pathExists(fullPath)
 | |
|           if (!exists) {
 | |
|             Logger.info(`[Scanner] Scanning file update group and audiobook was deleted "${existingAudiobook.title}" - marking as missing`)
 | |
|             existingAudiobook.isMissing = true
 | |
|             existingAudiobook.lastUpdate = Date.now()
 | |
|             await this.db.updateAudiobook(existingAudiobook)
 | |
|             this.emitter('audiobook_updated', existingAudiobook.toJSONMinified())
 | |
| 
 | |
|             bookGroupingResults[bookDir] = ScanResult.REMOVED
 | |
|             continue;
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         // Scan audiobook for updates
 | |
|         Logger.debug(`[Scanner] Folder update for relative path "${bookDir}" is in audiobook "${existingAudiobook.title}" - scan for updates`)
 | |
|         bookGroupingResults[bookDir] = await this.scanAudiobook(folder, existingAudiobook.fullPath)
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       // Check if an audiobook is a subdirectory of this dir
 | |
|       var childAudiobook = this.db.audiobooks.find(ab => ab.fullPath.startsWith(fullPath))
 | |
|       if (childAudiobook) {
 | |
|         Logger.warn(`[Scanner] Files were modified in a parent directory of an audiobook "${childAudiobook.title}" - ignoring`)
 | |
|         bookGroupingResults[bookDir] = ScanResult.NOTHING
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       Logger.debug(`[Scanner] Folder update group must be a new book "${bookDir}" in library "${library.name}"`)
 | |
|       bookGroupingResults[bookDir] = await this.scanAudiobook(folder, fullPath)
 | |
|     }
 | |
| 
 | |
|     return bookGroupingResults
 | |
|   }
 | |
| 
 | |
|   // Array of file update objects that may have been renamed, removed or added
 | |
|   async filesChanged(fileUpdates) {
 | |
|     if (!fileUpdates.length) return null
 | |
| 
 | |
|     // Group files by folder
 | |
|     var folderGroups = {}
 | |
|     fileUpdates.forEach((file) => {
 | |
|       if (folderGroups[file.folderId]) {
 | |
|         folderGroups[file.folderId].fileUpdates.push(file)
 | |
|       } else {
 | |
|         folderGroups[file.folderId] = {
 | |
|           libraryId: file.libraryId,
 | |
|           folderId: file.folderId,
 | |
|           fileUpdates: [file]
 | |
|         }
 | |
|       }
 | |
|     })
 | |
| 
 | |
|     const libraryScanResults = {}
 | |
| 
 | |
|     // Group files by book
 | |
|     for (const folderId in folderGroups) {
 | |
|       var libraryId = folderGroups[folderId].libraryId
 | |
|       var library = this.db.libraries.find(lib => lib.id === libraryId)
 | |
|       if (!library) {
 | |
|         Logger.error(`[Scanner] Library not found in files changed ${libraryId}`)
 | |
|         continue;
 | |
|       }
 | |
|       var folder = library.getFolderById(folderId)
 | |
|       if (!folder) {
 | |
|         Logger.error(`[Scanner] Folder is not in library in files changed "${folderId}", Library "${library.name}"`)
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       var relFilePaths = folderGroups[folderId].fileUpdates.map(fileUpdate => fileUpdate.relPath)
 | |
|       var fileUpdateBookGroup = groupFilesIntoAudiobookPaths(relFilePaths, true)
 | |
|       var folderScanResults = await this.scanFolderUpdates(libraryId, folderId, fileUpdateBookGroup)
 | |
|       libraryScanResults[libraryId] = folderScanResults
 | |
|     }
 | |
| 
 | |
|     Logger.debug(`[Scanner] Finished scanning file changes, results:`, libraryScanResults)
 | |
|     return libraryScanResults
 | |
|   }
 | |
| 
 | |
|   async saveMetadata(audiobookId) {
 | |
|     if (audiobookId) {
 | |
|       var audiobook = this.db.audiobooks.find(ab => ab.id === audiobookId)
 | |
|       if (!audiobook) {
 | |
|         return {
 | |
|           error: 'Audiobook not found'
 | |
|         }
 | |
|       }
 | |
|       var savedPath = await audiobook.writeNfoFile()
 | |
|       return {
 | |
|         audiobookId,
 | |
|         audiobookTitle: audiobook.title,
 | |
|         savedPath
 | |
|       }
 | |
|     } else {
 | |
|       var response = {
 | |
|         success: 0,
 | |
|         failed: 0
 | |
|       }
 | |
|       for (let i = 0; i < this.db.audiobooks.length; i++) {
 | |
|         var audiobook = this.db.audiobooks[i]
 | |
|         var savedPath = await audiobook.writeNfoFile()
 | |
|         if (savedPath) {
 | |
|           Logger.info(`[Scanner] Saved metadata nfo ${savedPath}`)
 | |
|           response.success++
 | |
|         } else {
 | |
|           response.failed++
 | |
|         }
 | |
|       }
 | |
|       return response
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   async find(req, res) {
 | |
|     var method = req.params.method
 | |
|     var query = req.query
 | |
| 
 | |
|     var result = null
 | |
| 
 | |
|     if (method === 'isbn') {
 | |
|       result = await this.bookFinder.findByISBN(query)
 | |
|     } else if (method === 'search') {
 | |
|       result = await this.bookFinder.search(query.provider, query.title, query.author || null)
 | |
|     }
 | |
| 
 | |
|     res.json(result)
 | |
|   }
 | |
| 
 | |
|   async findCovers(req, res) {
 | |
|     var query = req.query
 | |
|     var options = {
 | |
|       fallbackTitleOnly: !!query.fallbackTitleOnly
 | |
|     }
 | |
|     var result = await this.bookFinder.findCovers(query.provider, query.title, query.author || null, options)
 | |
|     res.json(result)
 | |
|   }
 | |
| 
 | |
|   async fixDuplicateIds() {
 | |
|     var ids = {}
 | |
|     var audiobooksUpdated = 0
 | |
|     for (let i = 0; i < this.db.audiobooks.length; i++) {
 | |
|       var ab = this.db.audiobooks[i]
 | |
|       if (ids[ab.id]) {
 | |
|         var abCopy = new Audiobook(ab.toJSON())
 | |
|         abCopy.id = getId('ab')
 | |
|         if (abCopy.book.cover) {
 | |
|           abCopy.book.cover = abCopy.book.cover.replace(ab.id, abCopy.id)
 | |
|         }
 | |
|         Logger.warn('Found duplicate ID - updating from', ab.id, 'to', abCopy.id)
 | |
|         await this.db.removeEntity('audiobook', ab.id)
 | |
|         await this.db.insertEntity('audiobook', abCopy)
 | |
|         audiobooksUpdated++
 | |
|       } else {
 | |
|         ids[ab.id] = true
 | |
|       }
 | |
|     }
 | |
|     if (audiobooksUpdated) {
 | |
|       Logger.info(`[Scanner] Updated ${audiobooksUpdated} audiobook IDs`)
 | |
|     }
 | |
|   }
 | |
| }
 | |
| module.exports = Scanner |