mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-31 10:27:01 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			245 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			245 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| const Path = require('path')
 | |
| const SocketAuthority = require('../SocketAuthority')
 | |
| const Logger = require('../Logger')
 | |
| const fs = require('../libs/fsExtra')
 | |
| const ffmpegHelpers = require('../utils/ffmpegHelpers')
 | |
| const TaskManager = require('./TaskManager')
 | |
| const Task = require('../objects/Task')
 | |
| const fileUtils = require('../utils/fileUtils')
 | |
| 
 | |
| /**
 | |
|  * @typedef UpdateMetadataOptions
 | |
|  * @property {boolean} [forceEmbedChapters=false] - Whether to force embed chapters.
 | |
|  * @property {boolean} [backup=false] - Whether to backup the files.
 | |
|  */
 | |
| 
 | |
| class AudioMetadataMangaer {
 | |
|   constructor() {
 | |
|     this.itemsCacheDir = Path.join(global.MetadataPath, 'cache/items')
 | |
| 
 | |
|     this.MAX_CONCURRENT_TASKS = 1
 | |
|     this.tasksRunning = []
 | |
|     this.tasksQueued = []
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Get queued task data
 | |
|    * @return {Array}
 | |
|    */
 | |
|   getQueuedTaskData() {
 | |
|     return this.tasksQueued.map((t) => t.data)
 | |
|   }
 | |
| 
 | |
|   getIsLibraryItemQueuedOrProcessing(libraryItemId) {
 | |
|     return this.tasksQueued.some((t) => t.data.libraryItemId === libraryItemId) || this.tasksRunning.some((t) => t.data.libraryItemId === libraryItemId)
 | |
|   }
 | |
| 
 | |
|   getMetadataObjectForApi(libraryItem) {
 | |
|     return ffmpegHelpers.getFFMetadataObject(libraryItem, libraryItem.media.includedAudioFiles.length)
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    *
 | |
|    * @param {string} userId
 | |
|    * @param {*} libraryItems
 | |
|    * @param {*} options
 | |
|    */
 | |
|   handleBatchEmbed(userId, libraryItems, options = {}) {
 | |
|     libraryItems.forEach((li) => {
 | |
|       this.updateMetadataForItem(userId, li, options)
 | |
|     })
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    *
 | |
|    * @param {string} userId
 | |
|    * @param {import('../objects/LibraryItem')} libraryItem
 | |
|    * @param {UpdateMetadataOptions} [options={}]
 | |
|    */
 | |
|   async updateMetadataForItem(userId, libraryItem, options = {}) {
 | |
|     const forceEmbedChapters = !!options.forceEmbedChapters
 | |
|     const backupFiles = !!options.backup
 | |
| 
 | |
|     const audioFiles = libraryItem.media.includedAudioFiles
 | |
| 
 | |
|     const task = new Task()
 | |
| 
 | |
|     const itemCachePath = Path.join(this.itemsCacheDir, libraryItem.id)
 | |
| 
 | |
|     // Only writing chapters for single file audiobooks
 | |
|     const chapters = audioFiles.length == 1 || forceEmbedChapters ? libraryItem.media.chapters.map((c) => ({ ...c })) : null
 | |
| 
 | |
|     let mimeType = audioFiles[0].mimeType
 | |
|     if (audioFiles.some((a) => a.mimeType !== mimeType)) mimeType = null
 | |
| 
 | |
|     // Create task
 | |
|     const libraryItemDir = libraryItem.isFile ? Path.dirname(libraryItem.path) : libraryItem.path
 | |
|     const taskData = {
 | |
|       libraryItemId: libraryItem.id,
 | |
|       libraryItemDir,
 | |
|       userId,
 | |
|       audioFiles: audioFiles.map((af) => ({
 | |
|         index: af.index,
 | |
|         ino: af.ino,
 | |
|         filename: af.metadata.filename,
 | |
|         path: af.metadata.path,
 | |
|         cachePath: Path.join(itemCachePath, af.metadata.filename),
 | |
|         duration: af.duration
 | |
|       })),
 | |
|       coverPath: libraryItem.media.coverPath,
 | |
|       metadataObject: ffmpegHelpers.getFFMetadataObject(libraryItem, audioFiles.length),
 | |
|       itemCachePath,
 | |
|       chapters,
 | |
|       mimeType,
 | |
|       options: {
 | |
|         forceEmbedChapters,
 | |
|         backupFiles
 | |
|       },
 | |
|       duration: libraryItem.media.duration
 | |
|     }
 | |
|     const taskDescription = `Embedding metadata in audiobook "${libraryItem.media.metadata.title}".`
 | |
|     task.setData('embed-metadata', 'Embedding Metadata', taskDescription, false, taskData)
 | |
| 
 | |
|     if (this.tasksRunning.length >= this.MAX_CONCURRENT_TASKS) {
 | |
|       Logger.info(`[AudioMetadataManager] Queueing embed metadata for audiobook "${libraryItem.media.metadata.title}"`)
 | |
|       SocketAuthority.adminEmitter('metadata_embed_queue_update', {
 | |
|         libraryItemId: libraryItem.id,
 | |
|         queued: true
 | |
|       })
 | |
|       this.tasksQueued.push(task)
 | |
|     } else {
 | |
|       this.runMetadataEmbed(task)
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   async runMetadataEmbed(task) {
 | |
|     this.tasksRunning.push(task)
 | |
|     TaskManager.addTask(task)
 | |
| 
 | |
|     Logger.info(`[AudioMetadataManager] Starting metadata embed task`, task.description)
 | |
| 
 | |
|     // Ensure target directory is writable
 | |
|     const targetDirWritable = await fileUtils.isWritable(task.data.libraryItemDir)
 | |
|     Logger.debug(`[AudioMetadataManager] Target directory ${task.data.libraryItemDir} writable: ${targetDirWritable}`)
 | |
|     if (!targetDirWritable) {
 | |
|       Logger.error(`[AudioMetadataManager] Target directory is not writable: ${task.data.libraryItemDir}`)
 | |
|       task.setFailed('Target directory is not writable')
 | |
|       this.handleTaskFinished(task)
 | |
|       return
 | |
|     }
 | |
| 
 | |
|     // Ensure target audio files are writable
 | |
|     for (const af of task.data.audioFiles) {
 | |
|       try {
 | |
|         await fs.access(af.path, fs.constants.W_OK)
 | |
|       } catch (err) {
 | |
|         Logger.error(`[AudioMetadataManager] Audio file is not writable: ${af.path}`)
 | |
|         task.setFailed(`Audio file "${Path.basename(af.path)}" is not writable`)
 | |
|         this.handleTaskFinished(task)
 | |
|         return
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Ensure item cache dir exists
 | |
|     let cacheDirCreated = false
 | |
|     if (!(await fs.pathExists(task.data.itemCachePath))) {
 | |
|       try {
 | |
|         await fs.mkdir(task.data.itemCachePath)
 | |
|         cacheDirCreated = true
 | |
|       } catch (err) {
 | |
|         Logger.error(`[AudioMetadataManager] Failed to create cache directory ${task.data.itemCachePath}`, err)
 | |
|         task.setFailed('Failed to create cache directory')
 | |
|         this.handleTaskFinished(task)
 | |
|         return
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Create ffmetadata file
 | |
|     const ffmetadataPath = Path.join(task.data.itemCachePath, 'ffmetadata.txt')
 | |
|     const success = await ffmpegHelpers.writeFFMetadataFile(task.data.metadataObject, task.data.chapters, ffmetadataPath)
 | |
|     if (!success) {
 | |
|       Logger.error(`[AudioMetadataManager] Failed to write ffmetadata file for audiobook "${task.data.libraryItemId}"`)
 | |
|       task.setFailed('Failed to write metadata file.')
 | |
|       this.handleTaskFinished(task)
 | |
|       return
 | |
|     }
 | |
| 
 | |
|     // Tag audio files
 | |
|     let cummulativeProgress = 0
 | |
|     for (const af of task.data.audioFiles) {
 | |
|       const audioFileRelativeDuration = af.duration / task.data.duration
 | |
|       SocketAuthority.adminEmitter('track_started', {
 | |
|         libraryItemId: task.data.libraryItemId,
 | |
|         ino: af.ino
 | |
|       })
 | |
| 
 | |
|       // Backup audio file
 | |
|       if (task.data.options.backupFiles) {
 | |
|         try {
 | |
|           const backupFilePath = Path.join(task.data.itemCachePath, af.filename)
 | |
|           await fs.copy(af.path, backupFilePath)
 | |
|           Logger.debug(`[AudioMetadataManager] Backed up audio file at "${backupFilePath}"`)
 | |
|         } catch (err) {
 | |
|           Logger.error(`[AudioMetadataManager] Failed to backup audio file "${af.path}"`, err)
 | |
|           task.setFailed(`Failed to backup audio file "${Path.basename(af.path)}"`)
 | |
|           this.handleTaskFinished(task)
 | |
|           return
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       try {
 | |
|         await ffmpegHelpers.addCoverAndMetadataToFile(af.path, task.data.coverPath, ffmetadataPath, af.index, task.data.mimeType, (progress) => {
 | |
|           SocketAuthority.adminEmitter('task_progress', { libraryItemId: task.data.libraryItemId, progress: cummulativeProgress + progress * audioFileRelativeDuration })
 | |
|           SocketAuthority.adminEmitter('track_progress', { libraryItemId: task.data.libraryItemId, ino: af.ino, progress })
 | |
|         })
 | |
|         Logger.info(`[AudioMetadataManager] Successfully tagged audio file "${af.path}"`)
 | |
|       } catch (err) {
 | |
|         Logger.error(`[AudioMetadataManager] Failed to tag audio file "${af.path}"`, err)
 | |
|         task.setFailed(`Failed to tag audio file "${Path.basename(af.path)}"`)
 | |
|         this.handleTaskFinished(task)
 | |
|         return
 | |
|       }
 | |
| 
 | |
|       SocketAuthority.adminEmitter('track_finished', {
 | |
|         libraryItemId: task.data.libraryItemId,
 | |
|         ino: af.ino
 | |
|       })
 | |
| 
 | |
|       cummulativeProgress += audioFileRelativeDuration * 100
 | |
|     }
 | |
| 
 | |
|     // Remove temp cache file/folder if not backing up
 | |
|     if (!task.data.options.backupFiles) {
 | |
|       // If cache dir was created from this then remove it
 | |
|       if (cacheDirCreated) {
 | |
|         await fs.remove(task.data.itemCachePath)
 | |
|       } else {
 | |
|         await fs.remove(ffmetadataPath)
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     task.setFinished()
 | |
|     this.handleTaskFinished(task)
 | |
|   }
 | |
| 
 | |
|   handleTaskFinished(task) {
 | |
|     TaskManager.taskFinished(task)
 | |
|     this.tasksRunning = this.tasksRunning.filter((t) => t.id !== task.id)
 | |
| 
 | |
|     if (this.tasksRunning.length < this.MAX_CONCURRENT_TASKS && this.tasksQueued.length) {
 | |
|       Logger.info(`[AudioMetadataManager] Task finished and dequeueing next task. ${this.tasksQueued} tasks queued.`)
 | |
|       const nextTask = this.tasksQueued.shift()
 | |
|       SocketAuthority.emitter('metadata_embed_queue_update', {
 | |
|         libraryItemId: nextTask.data.libraryItemId,
 | |
|         queued: false
 | |
|       })
 | |
|       this.runMetadataEmbed(nextTask)
 | |
|     } else if (this.tasksRunning.length > 0) {
 | |
|       Logger.debug(`[AudioMetadataManager] Task finished but not dequeueing. Currently running ${this.tasksRunning.length} tasks. ${this.tasksQueued.length} tasks queued.`)
 | |
|     } else {
 | |
|       Logger.debug(`[AudioMetadataManager] Task finished and no tasks remain in queue`)
 | |
|     }
 | |
|   }
 | |
| }
 | |
| module.exports = AudioMetadataMangaer
 |