2022-07-06 02:53:01 +02:00
|
|
|
const fs = require('../libs/fsExtra')
|
2021-10-02 01:42:48 +02:00
|
|
|
const Path = require('path')
|
2022-03-20 22:41:06 +01:00
|
|
|
const Logger = require('../Logger')
|
2022-06-08 02:44:38 +02:00
|
|
|
const readChunk = require('../libs/readChunk')
|
2022-06-08 02:53:05 +02:00
|
|
|
const imageType = require('../libs/imageType')
|
2021-10-02 01:42:48 +02:00
|
|
|
|
2022-03-20 22:41:06 +01:00
|
|
|
const globals = require('../utils/globals')
|
2023-10-14 17:52:56 +02:00
|
|
|
const { downloadImageFile, filePathToPOSIX, checkPathIsFile } = require('../utils/fileUtils')
|
2022-03-20 22:41:06 +01:00
|
|
|
const { extractCoverArt } = require('../utils/ffmpegHelpers')
|
2024-01-08 00:51:07 +01:00
|
|
|
const parseEbookMetadata = require('../utils/parsers/parseEbookMetadata')
|
|
|
|
|
2023-09-07 00:48:50 +02:00
|
|
|
const CacheManager = require('../managers/CacheManager')
|
2021-10-02 01:42:48 +02:00
|
|
|
|
2022-03-20 22:41:06 +01:00
|
|
|
class CoverManager {
|
2024-12-06 23:59:34 +01:00
|
|
|
constructor() {}
|
2021-10-02 01:42:48 +02:00
|
|
|
|
2022-03-13 00:45:32 +01:00
|
|
|
getCoverDirectory(libraryItem) {
|
2023-09-07 00:48:50 +02:00
|
|
|
if (global.ServerSettings.storeCoverWithItem && !libraryItem.isFile) {
|
2022-03-13 00:45:32 +01:00
|
|
|
return libraryItem.path
|
2021-10-02 01:42:48 +02:00
|
|
|
} else {
|
2023-09-07 00:48:50 +02:00
|
|
|
return Path.posix.join(Path.posix.join(global.MetadataPath, 'items'), libraryItem.id)
|
2021-10-02 01:42:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
getFilesInDirectory(dir) {
|
|
|
|
try {
|
|
|
|
return fs.readdir(dir)
|
|
|
|
} catch (error) {
|
2022-03-20 22:41:06 +01:00
|
|
|
Logger.error(`[CoverManager] Failed to get files in dir ${dir}`, error)
|
2021-10-02 01:42:48 +02:00
|
|
|
return []
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
removeFile(filepath) {
|
|
|
|
try {
|
|
|
|
return fs.pathExists(filepath).then((exists) => {
|
2022-03-20 22:41:06 +01:00
|
|
|
if (!exists) Logger.warn(`[CoverManager] Attempting to remove file that does not exist ${filepath}`)
|
2021-10-02 01:42:48 +02:00
|
|
|
return exists ? fs.unlink(filepath) : false
|
|
|
|
})
|
|
|
|
} catch (error) {
|
2022-03-20 22:41:06 +01:00
|
|
|
Logger.error(`[CoverManager] Failed to remove file "${filepath}"`, error)
|
2021-10-02 01:42:48 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-02 03:29:00 +02:00
|
|
|
// Remove covers that dont have the same filename as the new cover
|
|
|
|
async removeOldCovers(dirpath, newCoverExt) {
|
2021-10-02 01:42:48 +02:00
|
|
|
var filesInDir = await this.getFilesInDirectory(dirpath)
|
|
|
|
|
2022-04-21 00:34:20 +02:00
|
|
|
const imageExtensions = ['.jpeg', '.jpg', '.png', '.webp', '.jiff']
|
2021-10-02 01:42:48 +02:00
|
|
|
for (let i = 0; i < filesInDir.length; i++) {
|
|
|
|
var file = filesInDir[i]
|
2022-04-21 00:34:20 +02:00
|
|
|
var _extname = Path.extname(file).toLowerCase()
|
2023-02-11 22:56:18 +01:00
|
|
|
var _filename = Path.basename(file, _extname).toLowerCase()
|
2022-04-21 00:34:20 +02:00
|
|
|
if (_filename === 'cover' && _extname !== newCoverExt && imageExtensions.includes(_extname)) {
|
2021-10-02 01:42:48 +02:00
|
|
|
var filepath = Path.join(dirpath, file)
|
2022-03-20 22:41:06 +01:00
|
|
|
Logger.debug(`[CoverManager] Removing old cover from metadata "${filepath}"`)
|
2021-10-02 01:42:48 +02:00
|
|
|
await this.removeFile(filepath)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-13 00:45:32 +01:00
|
|
|
async checkFileIsValidImage(imagepath, removeOnInvalid = false) {
|
2021-10-02 01:42:48 +02:00
|
|
|
const buffer = await readChunk(imagepath, 0, 12)
|
|
|
|
const imgType = imageType(buffer)
|
|
|
|
if (!imgType) {
|
2022-03-13 00:45:32 +01:00
|
|
|
if (removeOnInvalid) await this.removeFile(imagepath)
|
2021-10-02 01:42:48 +02:00
|
|
|
return {
|
|
|
|
error: 'Invalid image'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!globals.SupportedImageTypes.includes(imgType.ext)) {
|
2022-03-13 00:45:32 +01:00
|
|
|
if (removeOnInvalid) await this.removeFile(imagepath)
|
2021-10-02 01:42:48 +02:00
|
|
|
return {
|
|
|
|
error: `Invalid image type ${imgType.ext} (Supported: ${globals.SupportedImageTypes.join(',')})`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return imgType
|
|
|
|
}
|
|
|
|
|
2025-01-02 22:42:52 +01:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {import('../models/LibraryItem')} libraryItem
|
|
|
|
* @param {*} coverFile - file object from req.files
|
|
|
|
* @returns {Promise<{error:string}|{cover:string}>}
|
|
|
|
*/
|
2022-03-13 00:45:32 +01:00
|
|
|
async uploadCover(libraryItem, coverFile) {
|
2023-02-11 22:56:18 +01:00
|
|
|
const extname = Path.extname(coverFile.name.toLowerCase())
|
2021-10-02 01:42:48 +02:00
|
|
|
if (!extname || !globals.SupportedImageTypes.includes(extname.slice(1))) {
|
|
|
|
return {
|
|
|
|
error: `Invalid image type ${extname} (Supported: ${globals.SupportedImageTypes.join(',')})`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-11 22:56:18 +01:00
|
|
|
const coverDirPath = this.getCoverDirectory(libraryItem)
|
2022-03-13 00:45:32 +01:00
|
|
|
await fs.ensureDir(coverDirPath)
|
2021-10-02 01:42:48 +02:00
|
|
|
|
2023-02-11 22:56:18 +01:00
|
|
|
const coverFullPath = Path.posix.join(coverDirPath, `cover${extname}`)
|
2021-10-02 01:42:48 +02:00
|
|
|
|
|
|
|
// Move cover from temp upload dir to destination
|
2024-12-06 23:59:34 +01:00
|
|
|
const success = await coverFile
|
|
|
|
.mv(coverFullPath)
|
|
|
|
.then(() => true)
|
|
|
|
.catch((error) => {
|
|
|
|
Logger.error('[CoverManager] Failed to move cover file', coverFullPath, error)
|
|
|
|
return false
|
|
|
|
})
|
2021-10-02 01:42:48 +02:00
|
|
|
|
|
|
|
if (!success) {
|
|
|
|
return {
|
|
|
|
error: 'Failed to move cover into destination'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-13 00:45:32 +01:00
|
|
|
await this.removeOldCovers(coverDirPath, extname)
|
2023-09-07 00:48:50 +02:00
|
|
|
await CacheManager.purgeCoverCache(libraryItem.id)
|
2021-10-02 03:29:00 +02:00
|
|
|
|
2025-01-02 22:42:52 +01:00
|
|
|
Logger.info(`[CoverManager] Uploaded libraryItem cover "${coverFullPath}" for "${libraryItem.media.title}"`)
|
2021-10-02 01:42:48 +02:00
|
|
|
|
|
|
|
return {
|
2022-03-13 00:45:32 +01:00
|
|
|
cover: coverFullPath
|
2021-10-02 01:42:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-02 22:42:52 +01:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} coverPath
|
|
|
|
* @param {import('../models/LibraryItem')} libraryItem
|
|
|
|
* @returns {Promise<{error:string}|{cover:string,updated:boolean}>}
|
|
|
|
*/
|
2022-03-13 00:45:32 +01:00
|
|
|
async validateCoverPath(coverPath, libraryItem) {
|
|
|
|
// Invalid cover path
|
|
|
|
if (!coverPath || coverPath.startsWith('http:') || coverPath.startsWith('https:')) {
|
2022-03-20 22:41:06 +01:00
|
|
|
Logger.error(`[CoverManager] validate cover path invalid http url "${coverPath}"`)
|
2022-03-13 00:45:32 +01:00
|
|
|
return {
|
|
|
|
error: 'Invalid cover path'
|
|
|
|
}
|
|
|
|
}
|
2023-01-06 00:45:27 +01:00
|
|
|
coverPath = filePathToPOSIX(coverPath)
|
2022-03-13 00:45:32 +01:00
|
|
|
// Cover path already set on media
|
|
|
|
if (libraryItem.media.coverPath == coverPath) {
|
2022-03-20 22:41:06 +01:00
|
|
|
Logger.debug(`[CoverManager] validate cover path already set "${coverPath}"`)
|
2022-03-13 00:45:32 +01:00
|
|
|
return {
|
|
|
|
cover: coverPath,
|
|
|
|
updated: false
|
|
|
|
}
|
|
|
|
}
|
2023-08-31 01:05:52 +02:00
|
|
|
|
2022-03-13 00:45:32 +01:00
|
|
|
// Cover path does not exist
|
2024-12-06 23:59:34 +01:00
|
|
|
if (!(await fs.pathExists(coverPath))) {
|
2022-03-20 22:41:06 +01:00
|
|
|
Logger.error(`[CoverManager] validate cover path does not exist "${coverPath}"`)
|
2022-03-13 00:45:32 +01:00
|
|
|
return {
|
|
|
|
error: 'Cover path does not exist'
|
|
|
|
}
|
|
|
|
}
|
2023-08-31 01:05:52 +02:00
|
|
|
|
|
|
|
// Cover path is not a file
|
2024-12-06 23:59:34 +01:00
|
|
|
if (!(await checkPathIsFile(coverPath))) {
|
2023-08-31 01:05:52 +02:00
|
|
|
Logger.error(`[CoverManager] validate cover path is not a file "${coverPath}"`)
|
|
|
|
return {
|
|
|
|
error: 'Cover path is not a file'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-13 00:45:32 +01:00
|
|
|
// Check valid image at path
|
2023-08-31 01:05:52 +02:00
|
|
|
var imgtype = await this.checkFileIsValidImage(coverPath, false)
|
2022-03-13 00:45:32 +01:00
|
|
|
if (imgtype.error) {
|
|
|
|
return imgtype
|
|
|
|
}
|
|
|
|
|
|
|
|
var coverDirPath = this.getCoverDirectory(libraryItem)
|
|
|
|
|
|
|
|
// Cover path is not in correct directory - make a copy
|
|
|
|
if (!coverPath.startsWith(coverDirPath)) {
|
|
|
|
await fs.ensureDir(coverDirPath)
|
|
|
|
|
|
|
|
var coverFilename = `cover.${imgtype.ext}`
|
|
|
|
var newCoverPath = Path.posix.join(coverDirPath, coverFilename)
|
2022-03-20 22:41:06 +01:00
|
|
|
Logger.debug(`[CoverManager] validate cover path copy cover from "${coverPath}" to "${newCoverPath}"`)
|
2022-03-13 00:45:32 +01:00
|
|
|
|
2024-12-06 23:59:34 +01:00
|
|
|
var copySuccess = await fs
|
|
|
|
.copy(coverPath, newCoverPath, { overwrite: true })
|
|
|
|
.then(() => true)
|
|
|
|
.catch((error) => {
|
|
|
|
Logger.error(`[CoverManager] validate cover path failed to copy cover`, error)
|
|
|
|
return false
|
|
|
|
})
|
2022-03-13 00:45:32 +01:00
|
|
|
if (!copySuccess) {
|
|
|
|
return {
|
|
|
|
error: 'Failed to copy cover to dir'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
await this.removeOldCovers(coverDirPath, '.' + imgtype.ext)
|
2022-03-20 22:41:06 +01:00
|
|
|
Logger.debug(`[CoverManager] cover copy success`)
|
2022-03-13 00:45:32 +01:00
|
|
|
coverPath = newCoverPath
|
|
|
|
}
|
|
|
|
|
2023-09-07 00:48:50 +02:00
|
|
|
await CacheManager.purgeCoverCache(libraryItem.id)
|
2022-03-13 00:45:32 +01:00
|
|
|
|
|
|
|
return {
|
|
|
|
cover: coverPath,
|
|
|
|
updated: true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-07 00:48:50 +02:00
|
|
|
/**
|
|
|
|
* Extract cover art from audio file and save for library item
|
2024-12-06 23:59:34 +01:00
|
|
|
*
|
|
|
|
* @param {import('../models/Book').AudioFileObject[]} audioFiles
|
|
|
|
* @param {string} libraryItemId
|
|
|
|
* @param {string} [libraryItemPath] null for isFile library items
|
2023-09-07 00:48:50 +02:00
|
|
|
* @returns {Promise<string>} returns cover path
|
|
|
|
*/
|
2023-09-17 21:53:25 +02:00
|
|
|
async saveEmbeddedCoverArt(audioFiles, libraryItemId, libraryItemPath) {
|
2024-12-06 23:59:34 +01:00
|
|
|
let audioFileWithCover = audioFiles.find((af) => af.embeddedCoverArt)
|
2023-09-02 01:01:17 +02:00
|
|
|
if (!audioFileWithCover) return null
|
|
|
|
|
|
|
|
let coverDirPath = null
|
|
|
|
if (global.ServerSettings.storeCoverWithItem && libraryItemPath) {
|
|
|
|
coverDirPath = libraryItemPath
|
|
|
|
} else {
|
2023-09-03 00:49:28 +02:00
|
|
|
coverDirPath = Path.posix.join(global.MetadataPath, 'items', libraryItemId)
|
2023-09-02 01:01:17 +02:00
|
|
|
}
|
|
|
|
await fs.ensureDir(coverDirPath)
|
|
|
|
|
|
|
|
const coverFilename = audioFileWithCover.embeddedCoverArt === 'png' ? 'cover.png' : 'cover.jpg'
|
|
|
|
const coverFilePath = Path.join(coverDirPath, coverFilename)
|
|
|
|
|
|
|
|
const coverAlreadyExists = await fs.pathExists(coverFilePath)
|
|
|
|
if (coverAlreadyExists) {
|
2023-09-18 23:45:30 +02:00
|
|
|
Logger.warn(`[CoverManager] Extract embedded cover art but cover already exists for "${coverFilePath}" - bail`)
|
2023-09-02 01:01:17 +02:00
|
|
|
return null
|
|
|
|
}
|
|
|
|
|
|
|
|
const success = await extractCoverArt(audioFileWithCover.metadata.path, coverFilePath)
|
|
|
|
if (success) {
|
2023-09-17 21:53:25 +02:00
|
|
|
await CacheManager.purgeCoverCache(libraryItemId)
|
2023-09-02 01:01:17 +02:00
|
|
|
return coverFilePath
|
|
|
|
}
|
|
|
|
return null
|
|
|
|
}
|
2023-09-07 00:48:50 +02:00
|
|
|
|
2024-01-08 00:51:07 +01:00
|
|
|
/**
|
|
|
|
* Extract cover art from ebook and save for library item
|
2024-12-06 23:59:34 +01:00
|
|
|
*
|
|
|
|
* @param {import('../utils/parsers/parseEbookMetadata').EBookFileScanData} ebookFileScanData
|
|
|
|
* @param {string} libraryItemId
|
|
|
|
* @param {string} [libraryItemPath] null for isFile library items
|
2024-01-08 00:51:07 +01:00
|
|
|
* @returns {Promise<string>} returns cover path
|
|
|
|
*/
|
|
|
|
async saveEbookCoverArt(ebookFileScanData, libraryItemId, libraryItemPath) {
|
|
|
|
if (!ebookFileScanData?.ebookCoverPath) return null
|
|
|
|
|
|
|
|
let coverDirPath = null
|
|
|
|
if (global.ServerSettings.storeCoverWithItem && libraryItemPath) {
|
|
|
|
coverDirPath = libraryItemPath
|
|
|
|
} else {
|
|
|
|
coverDirPath = Path.posix.join(global.MetadataPath, 'items', libraryItemId)
|
|
|
|
}
|
|
|
|
await fs.ensureDir(coverDirPath)
|
|
|
|
|
|
|
|
let extname = Path.extname(ebookFileScanData.ebookCoverPath) || '.jpg'
|
|
|
|
if (extname === '.jpeg') extname = '.jpg'
|
|
|
|
const coverFilename = `cover${extname}`
|
|
|
|
const coverFilePath = Path.join(coverDirPath, coverFilename)
|
|
|
|
|
|
|
|
// TODO: Overwrite if exists?
|
|
|
|
const coverAlreadyExists = await fs.pathExists(coverFilePath)
|
|
|
|
if (coverAlreadyExists) {
|
|
|
|
Logger.warn(`[CoverManager] Extract embedded cover art but cover already exists for "${coverFilePath}" - overwriting`)
|
|
|
|
}
|
|
|
|
|
|
|
|
const success = await parseEbookMetadata.extractCoverImage(ebookFileScanData, coverFilePath)
|
|
|
|
if (success) {
|
|
|
|
await CacheManager.purgeCoverCache(libraryItemId)
|
|
|
|
return coverFilePath
|
|
|
|
}
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
|
2023-09-07 00:48:50 +02:00
|
|
|
/**
|
2024-12-06 23:59:34 +01:00
|
|
|
*
|
|
|
|
* @param {string} url
|
|
|
|
* @param {string} libraryItemId
|
2025-01-04 19:41:09 +01:00
|
|
|
* @param {string} [libraryItemPath] - null if library item isFile
|
|
|
|
* @param {boolean} [forceLibraryItemFolder=false] - force save cover with library item (used for adding new podcasts)
|
2023-09-07 00:48:50 +02:00
|
|
|
* @returns {Promise<{error:string}|{cover:string}>}
|
|
|
|
*/
|
2025-01-04 19:41:09 +01:00
|
|
|
async downloadCoverFromUrlNew(url, libraryItemId, libraryItemPath, forceLibraryItemFolder = false) {
|
2023-09-07 00:48:50 +02:00
|
|
|
try {
|
|
|
|
let coverDirPath = null
|
2025-01-04 19:41:09 +01:00
|
|
|
if ((global.ServerSettings.storeCoverWithItem || forceLibraryItemFolder) && libraryItemPath) {
|
2023-09-07 00:48:50 +02:00
|
|
|
coverDirPath = libraryItemPath
|
|
|
|
} else {
|
|
|
|
coverDirPath = Path.posix.join(global.MetadataPath, 'items', libraryItemId)
|
|
|
|
}
|
|
|
|
|
|
|
|
await fs.ensureDir(coverDirPath)
|
|
|
|
|
|
|
|
const temppath = Path.posix.join(coverDirPath, 'cover')
|
2024-12-06 23:59:34 +01:00
|
|
|
const success = await downloadImageFile(url, temppath)
|
|
|
|
.then(() => true)
|
|
|
|
.catch((err) => {
|
|
|
|
Logger.error(`[CoverManager] Download image file failed for "${url}"`, err)
|
|
|
|
return false
|
|
|
|
})
|
2023-09-07 00:48:50 +02:00
|
|
|
if (!success) {
|
|
|
|
return {
|
|
|
|
error: 'Failed to download image from url'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const imgtype = await this.checkFileIsValidImage(temppath, true)
|
|
|
|
if (imgtype.error) {
|
|
|
|
return imgtype
|
|
|
|
}
|
|
|
|
|
|
|
|
const coverFullPath = Path.posix.join(coverDirPath, `cover.${imgtype.ext}`)
|
|
|
|
await fs.rename(temppath, coverFullPath)
|
|
|
|
|
|
|
|
await this.removeOldCovers(coverDirPath, '.' + imgtype.ext)
|
|
|
|
await CacheManager.purgeCoverCache(libraryItemId)
|
|
|
|
|
|
|
|
Logger.info(`[CoverManager] Downloaded libraryItem cover "${coverFullPath}" from url "${url}"`)
|
|
|
|
return {
|
|
|
|
cover: coverFullPath
|
|
|
|
}
|
|
|
|
} catch (error) {
|
|
|
|
Logger.error(`[CoverManager] Fetch cover image from url "${url}" failed`, error)
|
|
|
|
return {
|
|
|
|
error: 'Failed to fetch image from url'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-10-02 01:42:48 +02:00
|
|
|
}
|
2024-12-06 23:59:34 +01:00
|
|
|
module.exports = new CoverManager()
|