const { Request, Response, NextFunction } = require('express') const Path = require('path') const fs = require('../libs/fsExtra') const uaParserJs = require('../libs/uaParser') const Logger = require('../Logger') const SocketAuthority = require('../SocketAuthority') const Database = require('../Database') const zipHelpers = require('../utils/zipHelpers') const { reqSupportsWebp } = require('../utils/index') const { ScanResult, AudioMimeType } = require('../utils/constants') const { getAudioMimeTypeFromExtname, encodeUriPath } = require('../utils/fileUtils') const LibraryItemScanner = require('../scanner/LibraryItemScanner') const AudioFileScanner = require('../scanner/AudioFileScanner') const Scanner = require('../scanner/Scanner') const RssFeedManager = require('../managers/RssFeedManager') const CacheManager = require('../managers/CacheManager') const CoverManager = require('../managers/CoverManager') const ShareManager = require('../managers/ShareManager') /** * @typedef RequestUserObject * @property {import('../models/User')} user * * @typedef {Request & RequestUserObject} RequestWithUser * * @typedef RequestEntityObject * @property {import('../models/LibraryItem')} libraryItem * * @typedef {RequestWithUser & RequestEntityObject} LibraryItemControllerRequest * * @typedef RequestLibraryFileObject * @property {import('../objects/files/LibraryFile')} libraryFile * * @typedef {RequestWithUser & RequestEntityObject & RequestLibraryFileObject} LibraryItemControllerRequestWithFile */ class LibraryItemController { constructor() {} /** * GET: /api/items/:id * Optional query params: * ?include=progress,rssfeed,downloads,share * ?expanded=1 * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async findOne(req, res) { const includeEntities = (req.query.include || '').split(',') if (req.query.expanded == 1) { const item = req.libraryItem.toOldJSONExpanded() // Include users media progress if (includeEntities.includes('progress')) { const episodeId = req.query.episode || null item.userMediaProgress = req.user.getOldMediaProgress(item.id, episodeId) } if (includeEntities.includes('rssfeed')) { const feedData = await RssFeedManager.findFeedForEntityId(item.id) item.rssFeed = feedData?.toOldJSONMinified() || null } if (item.mediaType === 'book' && req.user.isAdminOrUp && includeEntities.includes('share')) { item.mediaItemShare = ShareManager.findByMediaItemId(item.media.id) } if (item.mediaType === 'podcast' && includeEntities.includes('downloads')) { const downloadsInQueue = this.podcastManager.getEpisodeDownloadsInQueue(req.libraryItem.id) item.episodeDownloadsQueued = downloadsInQueue.map((d) => d.toJSONForClient()) if (this.podcastManager.currentDownload?.libraryItemId === req.libraryItem.id) { item.episodesDownloading = [this.podcastManager.currentDownload.toJSONForClient()] } } return res.json(item) } res.json(req.libraryItem.toOldJSON()) } /** * DELETE: /api/items/:id * Delete library item. Will delete from database and file system if hard delete is requested. * Optional query params: * ?hard=1 * * @this {import('../routers/ApiRouter')} * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async delete(req, res) { const hardDelete = req.query.hard == 1 // Delete from file system const libraryItemPath = req.libraryItem.path const mediaItemIds = [] const authorIds = [] const seriesIds = [] if (req.libraryItem.isPodcast) { mediaItemIds.push(...req.libraryItem.media.podcastEpisodes.map((ep) => ep.id)) } else { mediaItemIds.push(req.libraryItem.media.id) if (req.libraryItem.media.authors?.length) { authorIds.push(...req.libraryItem.media.authors.map((au) => au.id)) } if (req.libraryItem.media.series?.length) { seriesIds.push(...req.libraryItem.media.series.map((se) => se.id)) } } await this.handleDeleteLibraryItem(req.libraryItem.id, mediaItemIds) if (hardDelete) { Logger.info(`[LibraryItemController] Deleting library item from file system at "${libraryItemPath}"`) await fs.remove(libraryItemPath).catch((error) => { Logger.error(`[LibraryItemController] Failed to delete library item from file system at "${libraryItemPath}"`, error) }) } if (authorIds.length) { await this.checkRemoveAuthorsWithNoBooks(authorIds) } if (seriesIds.length) { await this.checkRemoveEmptySeries(seriesIds) } await Database.resetLibraryIssuesFilterData(req.libraryItem.libraryId) res.sendStatus(200) } static handleDownloadError(error, res) { if (!res.headersSent) { if (error.code === 'ENOENT') { return res.status(404).send('File not found') } else { return res.status(500).send('Download failed') } } } /** * GET: /api/items/:id/download * Download library item. Zip file if multiple files. * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async download(req, res) { if (!req.user.canDownload) { Logger.warn(`User "${req.user.username}" attempted to download without permission`) return res.sendStatus(403) } const libraryItemPath = req.libraryItem.path const itemTitle = req.libraryItem.media.title Logger.info(`[LibraryItemController] User "${req.user.username}" requested download for item "${itemTitle}" at "${libraryItemPath}"`) try { // If library item is a single file in root dir then no need to zip if (req.libraryItem.isFile) { // Express does not set the correct mimetype for m4b files so use our defined mimetypes if available const audioMimeType = getAudioMimeTypeFromExtname(Path.extname(libraryItemPath)) if (audioMimeType) { res.setHeader('Content-Type', audioMimeType) } await new Promise((resolve, reject) => res.download(libraryItemPath, req.libraryItem.relPath, (error) => (error ? reject(error) : resolve()))) } else { const filename = `${itemTitle}.zip` await zipHelpers.zipDirectoryPipe(libraryItemPath, filename, res) } Logger.info(`[LibraryItemController] Downloaded item "${itemTitle}" at "${libraryItemPath}"`) } catch (error) { Logger.error(`[LibraryItemController] Download failed for item "${itemTitle}" at "${libraryItemPath}"`, error) LibraryItemController.handleDownloadError(error, res) } } /** * PATCH: /items/:id/media * Update media for a library item. Will create new authors & series when necessary * * @this {import('../routers/ApiRouter')} * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async updateMedia(req, res) { const mediaPayload = req.body if (mediaPayload.url) { await LibraryItemController.prototype.uploadCover.bind(this)(req, res, false) if (res.writableEnded || res.headersSent) return } // Podcast specific let isPodcastAutoDownloadUpdated = false if (req.libraryItem.isPodcast) { if (mediaPayload.autoDownloadEpisodes !== undefined && req.libraryItem.media.autoDownloadEpisodes !== mediaPayload.autoDownloadEpisodes) { isPodcastAutoDownloadUpdated = true } else if (mediaPayload.autoDownloadSchedule !== undefined && req.libraryItem.media.autoDownloadSchedule !== mediaPayload.autoDownloadSchedule) { isPodcastAutoDownloadUpdated = true } } let hasUpdates = (await req.libraryItem.media.updateFromRequest(mediaPayload)) || mediaPayload.url if (req.libraryItem.isBook && Array.isArray(mediaPayload.metadata?.series)) { const seriesUpdateData = await req.libraryItem.media.updateSeriesFromRequest(mediaPayload.metadata.series, req.libraryItem.libraryId) if (seriesUpdateData?.seriesRemoved.length) { // Check remove empty series Logger.debug(`[LibraryItemController] Series were removed from book. Check if series are now empty.`) await this.checkRemoveEmptySeries(seriesUpdateData.seriesRemoved.map((se) => se.id)) } if (seriesUpdateData?.seriesAdded.length) { // Add series to filter data seriesUpdateData.seriesAdded.forEach((se) => { Database.addSeriesToFilterData(req.libraryItem.libraryId, se.name, se.id) }) } if (seriesUpdateData?.hasUpdates) { hasUpdates = true } } if (req.libraryItem.isBook && Array.isArray(mediaPayload.metadata?.authors)) { const authorNames = mediaPayload.metadata.authors.map((au) => (typeof au.name === 'string' ? au.name.trim() : null)).filter((au) => au) const authorUpdateData = await req.libraryItem.media.updateAuthorsFromRequest(authorNames, req.libraryItem.libraryId) if (authorUpdateData?.authorsRemoved.length) { // Check remove empty authors Logger.debug(`[LibraryItemController] Authors were removed from book. Check if authors are now empty.`) await this.checkRemoveAuthorsWithNoBooks(authorUpdateData.authorsRemoved.map((au) => au.id)) hasUpdates = true } if (authorUpdateData?.authorsAdded.length) { // Add authors to filter data authorUpdateData.authorsAdded.forEach((au) => { Database.addAuthorToFilterData(req.libraryItem.libraryId, au.name, au.id) }) hasUpdates = true } } if (hasUpdates) { req.libraryItem.changed('updatedAt', true) await req.libraryItem.save() await req.libraryItem.saveMetadataFile() if (isPodcastAutoDownloadUpdated) { this.cronManager.checkUpdatePodcastCron(req.libraryItem) } Logger.debug(`[LibraryItemController] Updated library item media ${req.libraryItem.media.title}`) SocketAuthority.emitter('item_updated', req.libraryItem.toOldJSONExpanded()) } res.json({ updated: hasUpdates, libraryItem: req.libraryItem.toOldJSON() }) } /** * POST: /api/items/:id/cover * * @param {LibraryItemControllerRequest} req * @param {Response} res * @param {boolean} [updateAndReturnJson=true] - Allows the function to be used for both direct API calls and internally */ async uploadCover(req, res, updateAndReturnJson = true) { if (!req.user.canUpload) { Logger.warn(`User "${req.user.username}" attempted to upload a cover without permission`) return res.sendStatus(403) } let result = null if (req.body?.url) { Logger.debug(`[LibraryItemController] Requesting download cover from url "${req.body.url}"`) result = await CoverManager.downloadCoverFromUrlNew(req.body.url, req.libraryItem.id, req.libraryItem.isFile ? null : req.libraryItem.path) } else if (req.files?.cover) { Logger.debug(`[LibraryItemController] Handling uploaded cover`) result = await CoverManager.uploadCover(req.libraryItem, req.files.cover) } else { return res.status(400).send('Invalid request no file or url') } if (result?.error) { return res.status(400).send(result.error) } else if (!result?.cover) { return res.status(500).send('Unknown error occurred') } req.libraryItem.media.coverPath = result.cover req.libraryItem.media.changed('coverPath', true) await req.libraryItem.media.save() if (updateAndReturnJson) { // client uses updatedAt timestamp in URL to force refresh cover req.libraryItem.changed('updatedAt', true) await req.libraryItem.save() SocketAuthority.emitter('item_updated', req.libraryItem.toOldJSONExpanded()) res.json({ success: true, cover: result.cover }) } } /** * PATCH: /api/items/:id/cover * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async updateCover(req, res) { if (!req.body.cover) { return res.status(400).send('Invalid request no cover path') } const validationResult = await CoverManager.validateCoverPath(req.body.cover, req.libraryItem) if (validationResult.error) { return res.status(500).send(validationResult.error) } if (validationResult.updated) { req.libraryItem.media.coverPath = validationResult.cover req.libraryItem.media.changed('coverPath', true) await req.libraryItem.media.save() // client uses updatedAt timestamp in URL to force refresh cover req.libraryItem.changed('updatedAt', true) await req.libraryItem.save() SocketAuthority.emitter('item_updated', req.libraryItem.toOldJSONExpanded()) } res.json({ success: true, cover: validationResult.cover }) } /** * DELETE: /api/items/:id/cover * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async removeCover(req, res) { if (req.libraryItem.media.coverPath) { req.libraryItem.media.coverPath = null req.libraryItem.media.changed('coverPath', true) await req.libraryItem.media.save() // client uses updatedAt timestamp in URL to force refresh cover req.libraryItem.changed('updatedAt', true) await req.libraryItem.save() await CacheManager.purgeCoverCache(req.libraryItem.id) SocketAuthority.emitter('item_updated', req.libraryItem.toOldJSONExpanded()) } res.sendStatus(200) } /** * GET: /api/items/:id/cover * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async getCover(req, res) { const { query: { width, height, format, raw } } = req if (req.query.ts) res.set('Cache-Control', 'private, max-age=86400') const libraryItemId = req.params.id if (!libraryItemId) { return res.sendStatus(400) } if (raw) { const coverPath = await Database.libraryItemModel.getCoverPath(libraryItemId) if (!coverPath || !(await fs.pathExists(coverPath))) { return res.sendStatus(404) } // any value if (global.XAccel) { const encodedURI = encodeUriPath(global.XAccel + coverPath) Logger.debug(`Use X-Accel to serve static file ${encodedURI}`) return res.status(204).header({ 'X-Accel-Redirect': encodedURI }).send() } return res.sendFile(coverPath) } const options = { format: format || (reqSupportsWebp(req) ? 'webp' : 'jpeg'), height: height ? parseInt(height) : null, width: width ? parseInt(width) : null } return CacheManager.handleCoverCache(res, libraryItemId, options) } /** * POST: /api/items/:id/play * * @this {import('../routers/ApiRouter')} * * @param {LibraryItemControllerRequest} req * @param {Response} res */ startPlaybackSession(req, res) { if (!req.libraryItem.hasAudioTracks) { Logger.error(`[LibraryItemController] startPlaybackSession cannot playback ${req.libraryItem.id}`) return res.sendStatus(404) } this.playbackSessionManager.startSessionRequest(req, res, null) } /** * POST: /api/items/:id/play/:episodeId * * @this {import('../routers/ApiRouter')} * * @param {LibraryItemControllerRequest} req * @param {Response} res */ startEpisodePlaybackSession(req, res) { if (!req.libraryItem.isPodcast) { Logger.error(`[LibraryItemController] startEpisodePlaybackSession invalid media type ${req.libraryItem.id}`) return res.sendStatus(400) } const episodeId = req.params.episodeId if (!req.libraryItem.media.podcastEpisodes.some((ep) => ep.id === episodeId)) { Logger.error(`[LibraryItemController] startPlaybackSession episode ${episodeId} not found for item ${req.libraryItem.id}`) return res.sendStatus(404) } this.playbackSessionManager.startSessionRequest(req, res, episodeId) } /** * PATCH: /api/items/:id/tracks * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async updateTracks(req, res) { const orderedFileData = req.body?.orderedFileData if (!req.libraryItem.isBook) { Logger.error(`[LibraryItemController] updateTracks invalid media type ${req.libraryItem.id}`) return res.sendStatus(400) } if (!Array.isArray(orderedFileData) || !orderedFileData.length) { Logger.error(`[LibraryItemController] updateTracks invalid orderedFileData ${req.libraryItem.id}`) return res.sendStatus(400) } // Ensure that each orderedFileData has a valid ino and is in the book audioFiles if (orderedFileData.some((fileData) => !fileData?.ino || !req.libraryItem.media.audioFiles.some((af) => af.ino === fileData.ino))) { Logger.error(`[LibraryItemController] updateTracks invalid orderedFileData ${req.libraryItem.id}`) return res.sendStatus(400) } let index = 1 const updatedAudioFiles = orderedFileData.map((fileData) => { const audioFile = req.libraryItem.media.audioFiles.find((af) => af.ino === fileData.ino) audioFile.manuallyVerified = true audioFile.exclude = !!fileData.exclude if (audioFile.exclude) { audioFile.index = -1 } else { audioFile.index = index++ } return audioFile }) updatedAudioFiles.sort((a, b) => a.index - b.index) req.libraryItem.media.audioFiles = updatedAudioFiles req.libraryItem.media.changed('audioFiles', true) await req.libraryItem.media.save() SocketAuthority.emitter('item_updated', req.libraryItem.toOldJSONExpanded()) res.json(req.libraryItem.toOldJSON()) } /** * POST /api/items/:id/match * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async match(req, res) { const reqBody = req.body || {} const options = {} const matchOptions = ['provider', 'title', 'author', 'isbn', 'asin'] for (const key of matchOptions) { if (reqBody[key] && typeof reqBody[key] === 'string') { options[key] = reqBody[key] } } if (reqBody.overrideCover !== undefined) { options.overrideCover = !!reqBody.overrideCover } if (reqBody.overrideDetails !== undefined) { options.overrideDetails = !!reqBody.overrideDetails } const matchResult = await Scanner.quickMatchLibraryItem(this, req.libraryItem, options) res.json(matchResult) } /** * POST: /api/items/batch/delete * Batch delete library items. Will delete from database and file system if hard delete is requested. * Optional query params: * ?hard=1 * * @this {import('../routers/ApiRouter')} * * @param {RequestWithUser} req * @param {Response} res */ async batchDelete(req, res) { if (!req.user.canDelete) { Logger.warn(`[LibraryItemController] User "${req.user.username}" attempted to delete without permission`) return res.sendStatus(403) } const hardDelete = req.query.hard == 1 // Delete files from filesystem const { libraryItemIds } = req.body if (!libraryItemIds?.length || !Array.isArray(libraryItemIds)) { return res.status(400).send('Invalid request body') } const itemsToDelete = await Database.libraryItemModel.findAllExpandedWhere({ id: libraryItemIds }) if (!itemsToDelete.length) { return res.sendStatus(404) } const libraryId = itemsToDelete[0].libraryId for (const libraryItem of itemsToDelete) { const libraryItemPath = libraryItem.path Logger.info(`[LibraryItemController] (${hardDelete ? 'Hard' : 'Soft'}) deleting Library Item "${libraryItem.media.title}" with id "${libraryItem.id}"`) const mediaItemIds = [] const seriesIds = [] const authorIds = [] if (libraryItem.isPodcast) { mediaItemIds.push(...libraryItem.media.podcastEpisodes.map((ep) => ep.id)) } else { mediaItemIds.push(libraryItem.media.id) if (libraryItem.media.series?.length) { seriesIds.push(...libraryItem.media.series.map((se) => se.id)) } if (libraryItem.media.authors?.length) { authorIds.push(...libraryItem.media.authors.map((au) => au.id)) } } await this.handleDeleteLibraryItem(libraryItem.id, mediaItemIds) if (hardDelete) { Logger.info(`[LibraryItemController] Deleting library item from file system at "${libraryItemPath}"`) await fs.remove(libraryItemPath).catch((error) => { Logger.error(`[LibraryItemController] Failed to delete library item from file system at "${libraryItemPath}"`, error) }) } if (seriesIds.length) { await this.checkRemoveEmptySeries(seriesIds) } if (authorIds.length) { await this.checkRemoveAuthorsWithNoBooks(authorIds) } } await Database.resetLibraryIssuesFilterData(libraryId) res.sendStatus(200) } /** * POST: /api/items/batch/update * * @this {import('../routers/ApiRouter')} * * @param {RequestWithUser} req * @param {Response} res */ async batchUpdate(req, res) { const updatePayloads = req.body if (!Array.isArray(updatePayloads) || !updatePayloads.length) { Logger.error(`[LibraryItemController] Batch update failed. Invalid payload`) return res.sendStatus(400) } // Ensure that each update payload has a unique library item id const libraryItemIds = [...new Set(updatePayloads.map((up) => up?.id).filter((id) => id))] if (!libraryItemIds.length || libraryItemIds.length !== updatePayloads.length) { Logger.error(`[LibraryItemController] Batch update failed. Each update payload must have a unique library item id`) return res.sendStatus(400) } // Get all library items to update const libraryItems = await Database.libraryItemModel.findAllExpandedWhere({ id: libraryItemIds }) if (updatePayloads.length !== libraryItems.length) { Logger.error(`[LibraryItemController] Batch update failed. Not all library items found`) return res.sendStatus(404) } let itemsUpdated = 0 const seriesIdsRemoved = [] const authorIdsRemoved = [] for (const updatePayload of updatePayloads) { const mediaPayload = updatePayload.mediaPayload const libraryItem = libraryItems.find((li) => li.id === updatePayload.id) let hasUpdates = await libraryItem.media.updateFromRequest(mediaPayload) if (libraryItem.isBook && Array.isArray(mediaPayload.metadata?.series)) { const seriesUpdateData = await libraryItem.media.updateSeriesFromRequest(mediaPayload.metadata.series, libraryItem.libraryId) if (seriesUpdateData?.seriesRemoved.length) { seriesIdsRemoved.push(...seriesUpdateData.seriesRemoved.map((se) => se.id)) } if (seriesUpdateData?.seriesAdded.length) { seriesUpdateData.seriesAdded.forEach((se) => { Database.addSeriesToFilterData(libraryItem.libraryId, se.name, se.id) }) } if (seriesUpdateData?.hasUpdates) { hasUpdates = true } } if (libraryItem.isBook && Array.isArray(mediaPayload.metadata?.authors)) { const authorNames = mediaPayload.metadata.authors.map((au) => (typeof au.name === 'string' ? au.name.trim() : null)).filter((au) => au) const authorUpdateData = await libraryItem.media.updateAuthorsFromRequest(authorNames, libraryItem.libraryId) if (authorUpdateData?.authorsRemoved.length) { authorIdsRemoved.push(...authorUpdateData.authorsRemoved.map((au) => au.id)) hasUpdates = true } if (authorUpdateData?.authorsAdded.length) { authorUpdateData.authorsAdded.forEach((au) => { Database.addAuthorToFilterData(libraryItem.libraryId, au.name, au.id) }) hasUpdates = true } } if (hasUpdates) { libraryItem.changed('updatedAt', true) await libraryItem.save() await libraryItem.saveMetadataFile() Logger.debug(`[LibraryItemController] Updated library item media "${libraryItem.media.title}"`) SocketAuthority.emitter('item_updated', libraryItem.toOldJSONExpanded()) itemsUpdated++ } } if (seriesIdsRemoved.length) { await this.checkRemoveEmptySeries(seriesIdsRemoved) } if (authorIdsRemoved.length) { await this.checkRemoveAuthorsWithNoBooks(authorIdsRemoved) } res.json({ success: true, updates: itemsUpdated }) } /** * POST: /api/items/batch/get * * @param {RequestWithUser} req * @param {Response} res */ async batchGet(req, res) { const libraryItemIds = req.body.libraryItemIds || [] if (!libraryItemIds.length) { return res.status(403).send('Invalid payload') } const libraryItems = await Database.libraryItemModel.findAllExpandedWhere({ id: libraryItemIds }) res.json({ libraryItems: libraryItems.map((li) => li.toOldJSONExpanded()) }) } /** * POST: /api/items/batch/quickmatch * * @param {RequestWithUser} req * @param {Response} res */ async batchQuickMatch(req, res) { if (!req.user.isAdminOrUp) { Logger.warn(`Non-admin user "${req.user.username}" other than admin attempted to batch quick match library items`) return res.sendStatus(403) } let itemsUpdated = 0 let itemsUnmatched = 0 if (!req.body.libraryItemIds?.length) { return res.sendStatus(400) } const libraryItems = await Database.libraryItemModel.findAllExpandedWhere({ id: req.body.libraryItemIds }) if (!libraryItems?.length) { return res.sendStatus(400) } res.sendStatus(200) const reqBodyOptions = req.body.options || {} const options = {} if (reqBodyOptions.provider && typeof reqBodyOptions.provider === 'string') { options.provider = reqBodyOptions.provider } if (reqBodyOptions.overrideCover !== undefined) { options.overrideCover = !!reqBodyOptions.overrideCover } if (reqBodyOptions.overrideDetails !== undefined) { options.overrideDetails = !!reqBodyOptions.overrideDetails } for (const libraryItem of libraryItems) { const matchResult = await Scanner.quickMatchLibraryItem(this, libraryItem, options) if (matchResult.updated) { itemsUpdated++ } else if (matchResult.warning) { itemsUnmatched++ } } const result = { success: itemsUpdated > 0, updates: itemsUpdated, unmatched: itemsUnmatched } SocketAuthority.clientEmitter(req.user.id, 'batch_quickmatch_complete', result) } /** * POST: /api/items/batch/scan * * @param {RequestWithUser} req * @param {Response} res */ async batchScan(req, res) { if (!req.user.isAdminOrUp) { Logger.warn(`Non-admin user "${req.user.username}" other than admin attempted to batch scan library items`) return res.sendStatus(403) } if (!req.body.libraryItemIds?.length) { return res.sendStatus(400) } const libraryItems = await Database.libraryItemModel.findAll({ where: { id: req.body.libraryItemIds }, attributes: ['id', 'libraryId', 'isFile'] }) if (!libraryItems?.length) { return res.sendStatus(400) } res.sendStatus(200) const libraryId = libraryItems[0].libraryId for (const libraryItem of libraryItems) { if (libraryItem.isFile) { Logger.warn(`[LibraryItemController] Re-scanning file library items not yet supported`) } else { await LibraryItemScanner.scanLibraryItem(libraryItem.id) } } await Database.resetLibraryIssuesFilterData(libraryId) } /** * POST: /api/items/:id/scan * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async scan(req, res) { if (!req.user.isAdminOrUp) { Logger.error(`[LibraryItemController] Non-admin user "${req.user.username}" attempted to scan library item`) return res.sendStatus(403) } if (req.libraryItem.isFile) { Logger.error(`[LibraryItemController] Re-scanning file library items not yet supported`) return res.sendStatus(500) } const result = await LibraryItemScanner.scanLibraryItem(req.libraryItem.id) await Database.resetLibraryIssuesFilterData(req.libraryItem.libraryId) res.json({ result: Object.keys(ScanResult).find((key) => ScanResult[key] == result) }) } /** * GET: /api/items/:id/metadata-object * * @param {LibraryItemControllerRequest} req * @param {Response} res */ getMetadataObject(req, res) { if (!req.user.isAdminOrUp) { Logger.error(`[LibraryItemController] Non-admin user "${req.user.username}" attempted to get metadata object`) return res.sendStatus(403) } if (req.libraryItem.isMissing || !req.libraryItem.isBook || !req.libraryItem.media.includedAudioFiles.length) { Logger.error(`[LibraryItemController] Invalid library item`) return res.sendStatus(500) } res.json(this.audioMetadataManager.getMetadataObjectForApi(req.libraryItem)) } /** * POST: /api/items/:id/chapters * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async updateMediaChapters(req, res) { if (!req.user.canUpdate) { Logger.error(`[LibraryItemController] User "${req.user.username}" attempted to update chapters with invalid permissions`) return res.sendStatus(403) } if (req.libraryItem.isMissing || !req.libraryItem.isBook || !req.libraryItem.media.hasAudioTracks) { Logger.error(`[LibraryItemController] Invalid library item`) return res.sendStatus(500) } if (!Array.isArray(req.body.chapters) || req.body.chapters.some((c) => !c.title || typeof c.title !== 'string' || c.start === undefined || typeof c.start !== 'number' || c.end === undefined || typeof c.end !== 'number')) { Logger.error(`[LibraryItemController] Invalid payload`) return res.sendStatus(400) } const chapters = req.body.chapters || [] let hasUpdates = false if (chapters.length !== req.libraryItem.media.chapters.length) { req.libraryItem.media.chapters = chapters.map((c, index) => { return { id: index, title: c.title, start: c.start, end: c.end } }) hasUpdates = true } else { for (const [index, chapter] of chapters.entries()) { const currentChapter = req.libraryItem.media.chapters[index] if (currentChapter.title !== chapter.title || currentChapter.start !== chapter.start || currentChapter.end !== chapter.end) { currentChapter.title = chapter.title currentChapter.start = chapter.start currentChapter.end = chapter.end hasUpdates = true } } } if (hasUpdates) { req.libraryItem.media.changed('chapters', true) await req.libraryItem.media.save() await req.libraryItem.saveMetadataFile() SocketAuthority.emitter('item_updated', req.libraryItem.toOldJSONExpanded()) } res.json({ success: true, updated: hasUpdates }) } /** * GET: /api/items/:id/ffprobe/:fileid * FFProbe JSON result from audio file * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async getFFprobeData(req, res) { if (!req.user.isAdminOrUp) { Logger.error(`[LibraryItemController] Non-admin user "${req.user.username}" attempted to get ffprobe data`) return res.sendStatus(403) } const audioFile = req.libraryItem.getAudioFileWithIno(req.params.fileid) if (!audioFile) { Logger.error(`[LibraryItemController] Audio file not found with inode value ${req.params.fileid}`) return res.sendStatus(404) } const ffprobeData = await AudioFileScanner.probeAudioFile(audioFile.metadata.path) res.json(ffprobeData) } /** * GET api/items/:id/file/:fileid * * @param {LibraryItemControllerRequestWithFile} req * @param {Response} res */ async getLibraryFile(req, res) { const libraryFile = req.libraryFile if (global.XAccel) { const encodedURI = encodeUriPath(global.XAccel + libraryFile.metadata.path) Logger.debug(`Use X-Accel to serve static file ${encodedURI}`) return res.status(204).header({ 'X-Accel-Redirect': encodedURI }).send() } // Express does not set the correct mimetype for m4b files so use our defined mimetypes if available const audioMimeType = getAudioMimeTypeFromExtname(Path.extname(libraryFile.metadata.path)) if (audioMimeType) { res.setHeader('Content-Type', audioMimeType) } res.sendFile(libraryFile.metadata.path) } /** * DELETE api/items/:id/file/:fileid * * @param {LibraryItemControllerRequestWithFile} req * @param {Response} res */ async deleteLibraryFile(req, res) { const libraryFile = req.libraryFile Logger.info(`[LibraryItemController] User "${req.user.username}" requested file delete at "${libraryFile.metadata.path}"`) await fs.remove(libraryFile.metadata.path).catch((error) => { Logger.error(`[LibraryItemController] Failed to delete library file at "${libraryFile.metadata.path}"`, error) }) req.libraryItem.libraryFiles = req.libraryItem.libraryFiles.filter((lf) => lf.ino !== req.params.fileid) req.libraryItem.changed('libraryFiles', true) if (req.libraryItem.isBook) { if (req.libraryItem.media.audioFiles.some((af) => af.ino === req.params.fileid)) { req.libraryItem.media.audioFiles = req.libraryItem.media.audioFiles.filter((af) => af.ino !== req.params.fileid) req.libraryItem.media.changed('audioFiles', true) } else if (req.libraryItem.media.ebookFile?.ino === req.params.fileid) { req.libraryItem.media.ebookFile = null req.libraryItem.media.changed('ebookFile', true) } if (!req.libraryItem.media.hasMediaFiles) { req.libraryItem.isMissing = true } } else if (req.libraryItem.media.podcastEpisodes.some((ep) => ep.audioFile.ino === req.params.fileid)) { const episodeToRemove = req.libraryItem.media.podcastEpisodes.find((ep) => ep.audioFile.ino === req.params.fileid) // Remove episode from all playlists await Database.playlistModel.removeMediaItemsFromPlaylists([episodeToRemove.id]) // Remove episode media progress const numProgressRemoved = await Database.mediaProgressModel.destroy({ where: { mediaItemId: episodeToRemove.id } }) if (numProgressRemoved > 0) { Logger.info(`[LibraryItemController] Removed media progress for episode ${episodeToRemove.id}`) } // Remove episode await episodeToRemove.destroy() req.libraryItem.media.podcastEpisodes = req.libraryItem.media.podcastEpisodes.filter((ep) => ep.audioFile.ino !== req.params.fileid) } if (req.libraryItem.media.changed()) { await req.libraryItem.media.save() } await req.libraryItem.save() SocketAuthority.emitter('item_updated', req.libraryItem.toOldJSONExpanded()) res.sendStatus(200) } /** * GET api/items/:id/file/:fileid/download * Same as GET api/items/:id/file/:fileid but allows logging and restricting downloads * * @param {LibraryItemControllerRequestWithFile} req * @param {Response} res */ async downloadLibraryFile(req, res) { const libraryFile = req.libraryFile const ua = uaParserJs(req.headers['user-agent']) if (!req.user.canDownload) { Logger.error(`[LibraryItemController] User "${req.user.username}" without download permission attempted to download file "${libraryFile.metadata.path}"`) return res.sendStatus(403) } Logger.info(`[LibraryItemController] User "${req.user.username}" requested download for item "${req.libraryItem.media.title}" file at "${libraryFile.metadata.path}"`) if (global.XAccel) { const encodedURI = encodeUriPath(global.XAccel + libraryFile.metadata.path) Logger.debug(`Use X-Accel to serve static file ${encodedURI}`) return res.status(204).header({ 'X-Accel-Redirect': encodedURI }).send() } // Express does not set the correct mimetype for m4b files so use our defined mimetypes if available let audioMimeType = getAudioMimeTypeFromExtname(Path.extname(libraryFile.metadata.path)) if (audioMimeType) { // Work-around for Apple devices mishandling Content-Type on mobile browsers: // https://github.com/advplyr/audiobookshelf/issues/3310 // We actually need to check for Webkit on Apple mobile devices because this issue impacts all browsers on iOS/iPadOS/etc, not just Safari. const isAppleMobileBrowser = ua.device.vendor === 'Apple' && ua.device.type === 'mobile' && ua.engine.name === 'WebKit' if (isAppleMobileBrowser && audioMimeType === AudioMimeType.M4B) { audioMimeType = 'audio/m4b' } res.setHeader('Content-Type', audioMimeType) } try { await new Promise((resolve, reject) => res.download(libraryFile.metadata.path, libraryFile.metadata.filename, (error) => (error ? reject(error) : resolve()))) Logger.info(`[LibraryItemController] Downloaded file "${libraryFile.metadata.path}"`) } catch (error) { Logger.error(`[LibraryItemController] Failed to download file "${libraryFile.metadata.path}"`, error) LibraryItemController.handleDownloadError(error, res) } } /** * GET api/items/:id/ebook/:fileid? * fileid is the inode value stored in LibraryFile.ino or EBookFile.ino * fileid is only required when reading a supplementary ebook * when no fileid is passed in the primary ebook will be returned * * @param {LibraryItemControllerRequest} req * @param {Response} res */ async getEBookFile(req, res) { let ebookFile = null if (req.params.fileid) { ebookFile = req.libraryItem.getLibraryFileWithIno(req.params.fileid) if (!ebookFile?.isEBookFile) { Logger.error(`[LibraryItemController] Invalid ebook file id "${req.params.fileid}"`) return res.status(400).send('Invalid ebook file id') } } else { ebookFile = req.libraryItem.media.ebookFile } if (!ebookFile) { Logger.error(`[LibraryItemController] No ebookFile for library item "${req.libraryItem.media.title}"`) return res.sendStatus(404) } const ebookFilePath = ebookFile.metadata.path Logger.info(`[LibraryItemController] User "${req.user.username}" requested download for item "${req.libraryItem.media.title}" ebook at "${ebookFilePath}"`) if (global.XAccel) { const encodedURI = encodeUriPath(global.XAccel + ebookFilePath) Logger.debug(`Use X-Accel to serve static file ${encodedURI}`) return res.status(204).header({ 'X-Accel-Redirect': encodedURI }).send() } try { await new Promise((resolve, reject) => res.sendFile(ebookFilePath, (error) => (error ? reject(error) : resolve()))) Logger.info(`[LibraryItemController] Downloaded ebook file "${ebookFilePath}"`) } catch (error) { Logger.error(`[LibraryItemController] Failed to download ebook file "${ebookFilePath}"`, error) LibraryItemController.handleDownloadError(error, res) } } /** * PATCH api/items/:id/ebook/:fileid/status * toggle the status of an ebook file. * if an ebook file is the primary ebook, then it will be changed to supplementary * if an ebook file is supplementary, then it will be changed to primary * * @param {LibraryItemControllerRequestWithFile} req * @param {Response} res */ async updateEbookFileStatus(req, res) { if (!req.libraryItem.isBook) { Logger.error(`[LibraryItemController] Invalid media type for ebook file status update`) return res.sendStatus(400) } if (!req.libraryFile?.isEBookFile) { Logger.error(`[LibraryItemController] Invalid ebook file id "${req.params.fileid}"`) return res.status(400).send('Invalid ebook file id') } const ebookLibraryFile = req.libraryFile let primaryEbookFile = null const ebookLibraryFileInos = req.libraryItem .getLibraryFiles() .filter((lf) => lf.isEBookFile) .map((lf) => lf.ino) if (ebookLibraryFile.isSupplementary) { Logger.info(`[LibraryItemController] Updating ebook file "${ebookLibraryFile.metadata.filename}" to primary`) primaryEbookFile = ebookLibraryFile.toJSON() delete primaryEbookFile.isSupplementary delete primaryEbookFile.fileType primaryEbookFile.ebookFormat = ebookLibraryFile.metadata.format } else { Logger.info(`[LibraryItemController] Updating ebook file "${ebookLibraryFile.metadata.filename}" to supplementary`) } req.libraryItem.media.ebookFile = primaryEbookFile req.libraryItem.media.changed('ebookFile', true) await req.libraryItem.media.save() req.libraryItem.libraryFiles = req.libraryItem.libraryFiles.map((lf) => { if (ebookLibraryFileInos.includes(lf.ino)) { lf.isSupplementary = lf.ino !== primaryEbookFile?.ino } return lf }) req.libraryItem.changed('libraryFiles', true) req.libraryItem.isMissing = !req.libraryItem.media.hasMediaFiles await req.libraryItem.save() SocketAuthority.emitter('item_updated', req.libraryItem.toOldJSONExpanded()) res.sendStatus(200) } /** * * @param {RequestWithUser} req * @param {Response} res * @param {NextFunction} next */ async middleware(req, res, next) { req.libraryItem = await Database.libraryItemModel.getExpandedById(req.params.id) if (!req.libraryItem?.media) return res.sendStatus(404) // Check user can access this library item if (!req.user.checkCanAccessLibraryItem(req.libraryItem)) { return res.sendStatus(403) } // For library file routes, get the library file if (req.params.fileid) { req.libraryFile = req.libraryItem.getLibraryFileWithIno(req.params.fileid) if (!req.libraryFile) { Logger.error(`[LibraryItemController] Library file "${req.params.fileid}" does not exist for library item`) return res.sendStatus(404) } } if (req.path.includes('/play')) { // allow POST requests using /play and /play/:episodeId } else if (req.method == 'DELETE' && !req.user.canDelete) { Logger.warn(`[LibraryItemController] User "${req.user.username}" attempted to delete without permission`) return res.sendStatus(403) } else if ((req.method == 'PATCH' || req.method == 'POST') && !req.user.canUpdate) { Logger.warn(`[LibraryItemController] User "${req.user.username}" attempted to update without permission`) return res.sendStatus(403) } next() } } module.exports = new LibraryItemController()