2023-09-04 00:51:58 +02:00
|
|
|
const Path = require('path')
|
|
|
|
const { LogLevel, ScanResult } = require('../utils/constants')
|
|
|
|
|
|
|
|
const fileUtils = require('../utils/fileUtils')
|
|
|
|
const scanUtils = require('../utils/scandir')
|
|
|
|
const libraryFilters = require('../utils/queries/libraryFilters')
|
2024-11-08 00:26:51 +01:00
|
|
|
const Logger = require('../Logger')
|
2023-09-04 00:51:58 +02:00
|
|
|
const Database = require('../Database')
|
2024-11-08 00:26:51 +01:00
|
|
|
const Watcher = require('../Watcher')
|
2023-09-04 00:51:58 +02:00
|
|
|
const LibraryScan = require('./LibraryScan')
|
|
|
|
const LibraryItemScanData = require('./LibraryItemScanData')
|
|
|
|
const BookScanner = require('./BookScanner')
|
2023-09-04 18:50:55 +02:00
|
|
|
const PodcastScanner = require('./PodcastScanner')
|
2023-09-04 00:51:58 +02:00
|
|
|
const ScanLogger = require('./ScanLogger')
|
|
|
|
const LibraryItem = require('../models/LibraryItem')
|
|
|
|
const LibraryFile = require('../objects/files/LibraryFile')
|
|
|
|
const SocketAuthority = require('../SocketAuthority')
|
|
|
|
|
|
|
|
class LibraryItemScanner {
|
2024-09-22 21:15:17 +02:00
|
|
|
constructor() {}
|
2023-09-04 00:51:58 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Scan single library item
|
2024-09-22 21:15:17 +02:00
|
|
|
*
|
|
|
|
* @param {string} libraryItemId
|
2024-03-20 10:40:50 +01:00
|
|
|
* @param {{relPath:string, path:string}} [updateLibraryItemDetails] used by watcher when item folder was renamed
|
2023-09-04 00:51:58 +02:00
|
|
|
* @returns {number} ScanResult
|
|
|
|
*/
|
2024-03-20 10:40:50 +01:00
|
|
|
async scanLibraryItem(libraryItemId, updateLibraryItemDetails = null) {
|
2023-09-04 00:51:58 +02:00
|
|
|
// TODO: Add task manager
|
|
|
|
const libraryItem = await Database.libraryItemModel.findByPk(libraryItemId)
|
|
|
|
if (!libraryItem) {
|
|
|
|
Logger.error(`[LibraryItemScanner] Library item not found "${libraryItemId}"`)
|
|
|
|
return ScanResult.NOTHING
|
|
|
|
}
|
|
|
|
|
2024-03-20 10:40:50 +01:00
|
|
|
const libraryFolderId = updateLibraryItemDetails?.libraryFolderId || libraryItem.libraryFolderId
|
2023-09-04 00:51:58 +02:00
|
|
|
const library = await Database.libraryModel.findByPk(libraryItem.libraryId, {
|
|
|
|
include: {
|
|
|
|
model: Database.libraryFolderModel,
|
|
|
|
where: {
|
2024-03-20 10:40:50 +01:00
|
|
|
id: libraryFolderId
|
2023-09-04 00:51:58 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
if (!library) {
|
|
|
|
Logger.error(`[LibraryItemScanner] Library "${libraryItem.libraryId}" not found for library item "${libraryItem.id}"`)
|
|
|
|
return ScanResult.NOTHING
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure library filter data is set
|
|
|
|
// this is used to check for existing authors & series
|
|
|
|
await libraryFilters.getFilterData(library.mediaType, library.id)
|
|
|
|
|
|
|
|
const scanLogger = new ScanLogger()
|
|
|
|
scanLogger.verbose = true
|
2024-03-20 10:40:50 +01:00
|
|
|
scanLogger.setData('libraryItem', updateLibraryItemDetails?.relPath || libraryItem.relPath)
|
2023-09-04 00:51:58 +02:00
|
|
|
|
2024-03-20 10:40:50 +01:00
|
|
|
const libraryItemPath = updateLibraryItemDetails?.path || fileUtils.filePathToPOSIX(libraryItem.path)
|
2023-09-04 00:51:58 +02:00
|
|
|
const folder = library.libraryFolders[0]
|
2024-03-23 17:31:52 +01:00
|
|
|
const libraryItemScanData = await this.getLibraryItemScanData(libraryItemPath, library, folder, updateLibraryItemDetails?.isFile || false)
|
2023-09-04 00:51:58 +02:00
|
|
|
|
2023-10-09 23:41:43 +02:00
|
|
|
let libraryItemDataUpdated = await libraryItemScanData.checkLibraryItemData(libraryItem, scanLogger)
|
2023-09-04 00:51:58 +02:00
|
|
|
|
2023-10-09 23:41:43 +02:00
|
|
|
const { libraryItem: expandedLibraryItem, wasUpdated } = await this.rescanLibraryItemMedia(libraryItem, libraryItemScanData, library.settings, scanLogger)
|
|
|
|
if (libraryItemDataUpdated || wasUpdated) {
|
2025-01-04 22:20:41 +01:00
|
|
|
SocketAuthority.emitter('item_updated', expandedLibraryItem.toOldJSONExpanded())
|
2023-10-09 23:41:43 +02:00
|
|
|
|
|
|
|
await this.checkAuthorsAndSeriesRemovedFromBooks(library.id, scanLogger)
|
2023-09-04 00:51:58 +02:00
|
|
|
|
|
|
|
return ScanResult.UPDATED
|
|
|
|
}
|
2023-10-09 23:41:43 +02:00
|
|
|
|
|
|
|
scanLogger.addLog(LogLevel.DEBUG, `Library item is up-to-date`)
|
2023-09-04 00:51:58 +02:00
|
|
|
return ScanResult.UPTODATE
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove empty authors and series
|
2024-09-22 21:15:17 +02:00
|
|
|
* @param {string} libraryId
|
|
|
|
* @param {ScanLogger} scanLogger
|
2023-09-04 00:51:58 +02:00
|
|
|
* @returns {Promise}
|
|
|
|
*/
|
|
|
|
async checkAuthorsAndSeriesRemovedFromBooks(libraryId, scanLogger) {
|
|
|
|
if (scanLogger.authorsRemovedFromBooks.length) {
|
|
|
|
await BookScanner.checkAuthorsRemovedFromBooks(libraryId, scanLogger)
|
|
|
|
}
|
|
|
|
if (scanLogger.seriesRemovedFromBooks.length) {
|
|
|
|
await BookScanner.checkSeriesRemovedFromBooks(libraryId, scanLogger)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-09-22 21:15:17 +02:00
|
|
|
*
|
|
|
|
* @param {string} libraryItemPath
|
|
|
|
* @param {import('../models/Library')} library
|
|
|
|
* @param {import('../models/LibraryFolder')} folder
|
|
|
|
* @param {boolean} isSingleMediaItem
|
2023-09-04 00:51:58 +02:00
|
|
|
* @returns {Promise<LibraryItemScanData>}
|
|
|
|
*/
|
|
|
|
async getLibraryItemScanData(libraryItemPath, library, folder, isSingleMediaItem) {
|
|
|
|
const libraryFolderPath = fileUtils.filePathToPOSIX(folder.path)
|
|
|
|
const libraryItemDir = libraryItemPath.replace(libraryFolderPath, '').slice(1)
|
|
|
|
|
|
|
|
let libraryItemData = {}
|
|
|
|
|
|
|
|
let fileItems = []
|
|
|
|
|
2024-09-22 21:15:17 +02:00
|
|
|
if (isSingleMediaItem) {
|
|
|
|
// Single media item in root of folder
|
2023-09-04 00:51:58 +02:00
|
|
|
fileItems = [
|
|
|
|
{
|
|
|
|
fullpath: libraryItemPath,
|
|
|
|
path: libraryItemDir // actually the relPath (only filename here)
|
|
|
|
}
|
|
|
|
]
|
|
|
|
libraryItemData = {
|
|
|
|
path: libraryItemPath, // full path
|
|
|
|
relPath: libraryItemDir, // only filename
|
|
|
|
mediaMetadata: {
|
|
|
|
title: Path.basename(libraryItemDir, Path.extname(libraryItemDir))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
fileItems = await fileUtils.recurseFiles(libraryItemPath)
|
|
|
|
libraryItemData = scanUtils.getDataFromMediaDir(library.mediaType, libraryFolderPath, libraryItemDir)
|
|
|
|
}
|
|
|
|
|
|
|
|
const libraryFiles = []
|
|
|
|
for (let i = 0; i < fileItems.length; i++) {
|
|
|
|
const fileItem = fileItems[i]
|
2024-11-08 00:26:51 +01:00
|
|
|
|
|
|
|
if (Watcher.checkShouldIgnoreFilePath(fileItem.fullpath)) {
|
|
|
|
// Skip file if it's pending
|
|
|
|
Logger.info(`[LibraryItemScanner] Skipping watcher pending file "${fileItem.fullpath}" during scan of library item path "${libraryItemPath}"`)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-09-04 00:51:58 +02:00
|
|
|
const newLibraryFile = new LibraryFile()
|
|
|
|
// fileItem.path is the relative path
|
|
|
|
await newLibraryFile.setDataFromPath(fileItem.fullpath, fileItem.path)
|
|
|
|
libraryFiles.push(newLibraryFile)
|
|
|
|
}
|
|
|
|
|
|
|
|
const libraryItemStats = await fileUtils.getFileTimestampsWithIno(libraryItemData.path)
|
|
|
|
return new LibraryItemScanData({
|
|
|
|
libraryFolderId: folder.id,
|
|
|
|
libraryId: library.id,
|
|
|
|
mediaType: library.mediaType,
|
|
|
|
ino: libraryItemStats.ino,
|
|
|
|
mtimeMs: libraryItemStats.mtimeMs || 0,
|
|
|
|
ctimeMs: libraryItemStats.ctimeMs || 0,
|
|
|
|
birthtimeMs: libraryItemStats.birthtimeMs || 0,
|
|
|
|
path: libraryItemData.path,
|
|
|
|
relPath: libraryItemData.relPath,
|
|
|
|
isFile: isSingleMediaItem,
|
|
|
|
mediaMetadata: libraryItemData.mediaMetadata || null,
|
|
|
|
libraryFiles
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-09-22 21:15:17 +02:00
|
|
|
*
|
|
|
|
* @param {import('../models/LibraryItem')} existingLibraryItem
|
|
|
|
* @param {LibraryItemScanData} libraryItemData
|
2023-09-04 00:51:58 +02:00
|
|
|
* @param {import('../models/Library').LibrarySettingsObject} librarySettings
|
|
|
|
* @param {LibraryScan} libraryScan
|
2023-10-09 00:10:43 +02:00
|
|
|
* @returns {Promise<{libraryItem:LibraryItem, wasUpdated:boolean}>}
|
2023-09-04 00:51:58 +02:00
|
|
|
*/
|
2023-10-09 00:10:43 +02:00
|
|
|
rescanLibraryItemMedia(existingLibraryItem, libraryItemData, librarySettings, libraryScan) {
|
2023-09-04 00:51:58 +02:00
|
|
|
if (existingLibraryItem.mediaType === 'book') {
|
2023-10-09 00:10:43 +02:00
|
|
|
return BookScanner.rescanExistingBookLibraryItem(existingLibraryItem, libraryItemData, librarySettings, libraryScan)
|
2023-09-04 00:51:58 +02:00
|
|
|
} else {
|
2023-10-09 00:10:43 +02:00
|
|
|
return PodcastScanner.rescanExistingPodcastLibraryItem(existingLibraryItem, libraryItemData, librarySettings, libraryScan)
|
2023-09-04 00:51:58 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-09-22 21:15:17 +02:00
|
|
|
*
|
|
|
|
* @param {LibraryItemScanData} libraryItemData
|
2023-09-04 00:51:58 +02:00
|
|
|
* @param {import('../models/Library').LibrarySettingsObject} librarySettings
|
|
|
|
* @param {LibraryScan} libraryScan
|
|
|
|
* @returns {Promise<LibraryItem>}
|
|
|
|
*/
|
|
|
|
async scanNewLibraryItem(libraryItemData, librarySettings, libraryScan) {
|
2023-09-04 18:50:55 +02:00
|
|
|
let newLibraryItem = null
|
|
|
|
if (libraryItemData.mediaType === 'book') {
|
|
|
|
newLibraryItem = await BookScanner.scanNewBookLibraryItem(libraryItemData, librarySettings, libraryScan)
|
2023-09-04 00:51:58 +02:00
|
|
|
} else {
|
2023-09-04 18:50:55 +02:00
|
|
|
newLibraryItem = await PodcastScanner.scanNewPodcastLibraryItem(libraryItemData, librarySettings, libraryScan)
|
|
|
|
}
|
|
|
|
if (newLibraryItem) {
|
2024-09-22 21:15:17 +02:00
|
|
|
libraryScan.addLog(LogLevel.INFO, `Created new library item "${newLibraryItem.relPath}" with id "${newLibraryItem.id}"`)
|
2023-09-04 00:51:58 +02:00
|
|
|
}
|
2023-09-04 18:50:55 +02:00
|
|
|
return newLibraryItem
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Scan library item folder coming from Watcher
|
2024-09-22 21:15:17 +02:00
|
|
|
* @param {string} libraryItemPath
|
|
|
|
* @param {import('../models/Library')} library
|
|
|
|
* @param {import('../models/LibraryFolder')} folder
|
|
|
|
* @param {boolean} isSingleMediaItem
|
2023-09-04 18:50:55 +02:00
|
|
|
* @returns {Promise<LibraryItem>} ScanResult
|
|
|
|
*/
|
|
|
|
async scanPotentialNewLibraryItem(libraryItemPath, library, folder, isSingleMediaItem) {
|
|
|
|
const libraryItemScanData = await this.getLibraryItemScanData(libraryItemPath, library, folder, isSingleMediaItem)
|
|
|
|
|
|
|
|
const scanLogger = new ScanLogger()
|
|
|
|
scanLogger.verbose = true
|
|
|
|
scanLogger.setData('libraryItem', libraryItemScanData.relPath)
|
|
|
|
|
|
|
|
return this.scanNewLibraryItem(libraryItemScanData, library.settings, scanLogger)
|
2023-09-04 00:51:58 +02:00
|
|
|
}
|
|
|
|
}
|
2024-09-22 21:15:17 +02:00
|
|
|
module.exports = new LibraryItemScanner()
|