2023-07-05 01:14:44 +02:00
|
|
|
const { DataTypes, Model } = require('sequelize')
|
2023-07-14 21:50:37 +02:00
|
|
|
const Logger = require('../Logger')
|
2023-07-05 01:14:44 +02:00
|
|
|
const oldLibrary = require('../objects/Library')
|
|
|
|
|
2023-09-04 00:51:58 +02:00
|
|
|
/**
|
|
|
|
* @typedef LibrarySettingsObject
|
|
|
|
* @property {number} coverAspectRatio BookCoverAspectRatio
|
|
|
|
* @property {boolean} disableWatcher
|
|
|
|
* @property {boolean} skipMatchingMediaWithAsin
|
|
|
|
* @property {boolean} skipMatchingMediaWithIsbn
|
|
|
|
* @property {string} autoScanCronExpression
|
|
|
|
* @property {boolean} audiobooksOnly
|
2024-05-29 00:24:02 +02:00
|
|
|
* @property {boolean} hideSingleBookSeries Do not show series that only have 1 book
|
2024-03-12 17:04:26 +01:00
|
|
|
* @property {boolean} onlyShowLaterBooksInContinueSeries Skip showing books that are earlier than the max sequence read
|
2023-10-09 00:10:43 +02:00
|
|
|
* @property {string[]} metadataPrecedence
|
2023-09-04 00:51:58 +02:00
|
|
|
*/
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
class Library extends Model {
|
|
|
|
constructor(values, options) {
|
|
|
|
super(values, options)
|
|
|
|
|
|
|
|
/** @type {UUIDV4} */
|
|
|
|
this.id
|
|
|
|
/** @type {string} */
|
|
|
|
this.name
|
|
|
|
/** @type {number} */
|
|
|
|
this.displayOrder
|
|
|
|
/** @type {string} */
|
|
|
|
this.icon
|
|
|
|
/** @type {string} */
|
|
|
|
this.mediaType
|
|
|
|
/** @type {string} */
|
|
|
|
this.provider
|
|
|
|
/** @type {Date} */
|
|
|
|
this.lastScan
|
|
|
|
/** @type {string} */
|
|
|
|
this.lastScanVersion
|
2023-09-04 00:51:58 +02:00
|
|
|
/** @type {LibrarySettingsObject} */
|
2023-08-16 01:03:43 +02:00
|
|
|
this.settings
|
|
|
|
/** @type {Object} */
|
|
|
|
this.extraData
|
|
|
|
/** @type {Date} */
|
|
|
|
this.createdAt
|
|
|
|
/** @type {Date} */
|
|
|
|
this.updatedAt
|
|
|
|
}
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2024-08-23 00:39:28 +02:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} mediaType
|
|
|
|
* @returns
|
|
|
|
*/
|
|
|
|
static getDefaultLibrarySettingsForMediaType(mediaType) {
|
|
|
|
if (mediaType === 'podcast') {
|
|
|
|
return {
|
|
|
|
coverAspectRatio: 1, // Square
|
|
|
|
disableWatcher: false,
|
|
|
|
autoScanCronExpression: null,
|
|
|
|
podcastSearchRegion: 'us'
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return {
|
|
|
|
coverAspectRatio: 1, // Square
|
|
|
|
disableWatcher: false,
|
|
|
|
autoScanCronExpression: null,
|
|
|
|
skipMatchingMediaWithAsin: false,
|
|
|
|
skipMatchingMediaWithIsbn: false,
|
|
|
|
audiobooksOnly: false,
|
|
|
|
epubsAllowScriptedContent: false,
|
|
|
|
hideSingleBookSeries: false,
|
|
|
|
onlyShowLaterBooksInContinueSeries: false,
|
|
|
|
metadataPrecedence: ['folderStructure', 'audioMetatags', 'nfoFile', 'txtFiles', 'opfFile', 'absMetadata']
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
/**
|
|
|
|
* Get all old libraries
|
|
|
|
* @returns {Promise<oldLibrary[]>}
|
|
|
|
*/
|
|
|
|
static async getAllOldLibraries() {
|
|
|
|
const libraries = await this.findAll({
|
|
|
|
include: this.sequelize.models.libraryFolder,
|
|
|
|
order: [['displayOrder', 'ASC']]
|
|
|
|
})
|
2024-05-29 00:24:02 +02:00
|
|
|
return libraries.map((lib) => this.getOldLibrary(lib))
|
2023-08-16 01:03:43 +02:00
|
|
|
}
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
/**
|
|
|
|
* Convert expanded Library to oldLibrary
|
2024-05-29 00:24:02 +02:00
|
|
|
* @param {Library} libraryExpanded
|
2024-07-17 00:05:52 +02:00
|
|
|
* @returns {oldLibrary}
|
2023-08-16 01:03:43 +02:00
|
|
|
*/
|
|
|
|
static getOldLibrary(libraryExpanded) {
|
2024-05-29 00:24:02 +02:00
|
|
|
const folders = libraryExpanded.libraryFolders.map((folder) => {
|
2023-08-16 01:03:43 +02:00
|
|
|
return {
|
|
|
|
id: folder.id,
|
|
|
|
fullPath: folder.path,
|
|
|
|
libraryId: folder.libraryId,
|
|
|
|
addedAt: folder.createdAt.valueOf()
|
2023-07-05 01:14:44 +02:00
|
|
|
}
|
2023-08-16 01:03:43 +02:00
|
|
|
})
|
|
|
|
return new oldLibrary({
|
|
|
|
id: libraryExpanded.id,
|
|
|
|
oldLibraryId: libraryExpanded.extraData?.oldLibraryId || null,
|
|
|
|
name: libraryExpanded.name,
|
|
|
|
folders,
|
|
|
|
displayOrder: libraryExpanded.displayOrder,
|
|
|
|
icon: libraryExpanded.icon,
|
|
|
|
mediaType: libraryExpanded.mediaType,
|
|
|
|
provider: libraryExpanded.provider,
|
|
|
|
settings: libraryExpanded.settings,
|
2023-10-10 00:48:21 +02:00
|
|
|
lastScan: libraryExpanded.lastScan?.valueOf() || null,
|
|
|
|
lastScanVersion: libraryExpanded.lastScanVersion || null,
|
|
|
|
lastScanMetadataPrecedence: libraryExpanded.extraData?.lastScanMetadataPrecedence || null,
|
2023-08-16 01:03:43 +02:00
|
|
|
createdAt: libraryExpanded.createdAt.valueOf(),
|
|
|
|
lastUpdate: libraryExpanded.updatedAt.valueOf()
|
|
|
|
})
|
|
|
|
}
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
/**
|
|
|
|
* Update library and library folders
|
2024-05-29 00:24:02 +02:00
|
|
|
* @param {object} oldLibrary
|
|
|
|
* @returns
|
2023-08-16 01:03:43 +02:00
|
|
|
*/
|
|
|
|
static async updateFromOld(oldLibrary) {
|
|
|
|
const existingLibrary = await this.findByPk(oldLibrary.id, {
|
|
|
|
include: this.sequelize.models.libraryFolder
|
|
|
|
})
|
|
|
|
if (!existingLibrary) {
|
|
|
|
Logger.error(`[Library] Failed to update library ${oldLibrary.id} - not found`)
|
|
|
|
return null
|
2023-07-05 01:14:44 +02:00
|
|
|
}
|
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
const library = this.getFromOld(oldLibrary)
|
|
|
|
|
2024-05-29 00:24:02 +02:00
|
|
|
const libraryFolders = oldLibrary.folders.map((folder) => {
|
2023-07-05 01:14:44 +02:00
|
|
|
return {
|
2023-08-16 01:03:43 +02:00
|
|
|
id: folder.id,
|
|
|
|
path: folder.fullPath,
|
|
|
|
libraryId: library.id
|
|
|
|
}
|
|
|
|
})
|
|
|
|
for (const libraryFolder of libraryFolders) {
|
2024-05-29 00:24:02 +02:00
|
|
|
const existingLibraryFolder = existingLibrary.libraryFolders.find((lf) => lf.id === libraryFolder.id)
|
2023-08-16 01:03:43 +02:00
|
|
|
if (!existingLibraryFolder) {
|
|
|
|
await this.sequelize.models.libraryFolder.create(libraryFolder)
|
|
|
|
} else if (existingLibraryFolder.path !== libraryFolder.path) {
|
|
|
|
await existingLibraryFolder.update({ path: libraryFolder.path })
|
2023-07-05 01:14:44 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-29 00:24:02 +02:00
|
|
|
const libraryFoldersRemoved = existingLibrary.libraryFolders.filter((lf) => !libraryFolders.some((_lf) => _lf.id === lf.id))
|
2023-08-16 01:03:43 +02:00
|
|
|
for (const existingLibraryFolder of libraryFoldersRemoved) {
|
|
|
|
await existingLibraryFolder.destroy()
|
2023-07-05 01:14:44 +02:00
|
|
|
}
|
2023-07-22 21:25:20 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
return existingLibrary.update(library)
|
|
|
|
}
|
2023-07-22 21:25:20 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
static getFromOld(oldLibrary) {
|
|
|
|
const extraData = {}
|
|
|
|
if (oldLibrary.oldLibraryId) {
|
|
|
|
extraData.oldLibraryId = oldLibrary.oldLibraryId
|
2023-07-22 21:25:20 +02:00
|
|
|
}
|
2023-10-10 00:48:21 +02:00
|
|
|
if (oldLibrary.lastScanMetadataPrecedence) {
|
|
|
|
extraData.lastScanMetadataPrecedence = oldLibrary.lastScanMetadataPrecedence
|
|
|
|
}
|
2023-08-16 01:03:43 +02:00
|
|
|
return {
|
|
|
|
id: oldLibrary.id,
|
|
|
|
name: oldLibrary.name,
|
|
|
|
displayOrder: oldLibrary.displayOrder,
|
|
|
|
icon: oldLibrary.icon || null,
|
|
|
|
mediaType: oldLibrary.mediaType || null,
|
|
|
|
provider: oldLibrary.provider,
|
|
|
|
settings: oldLibrary.settings?.toJSON() || {},
|
2023-10-10 00:48:21 +02:00
|
|
|
lastScan: oldLibrary.lastScan || null,
|
|
|
|
lastScanVersion: oldLibrary.lastScanVersion || null,
|
2023-08-16 01:03:43 +02:00
|
|
|
createdAt: oldLibrary.createdAt,
|
|
|
|
updatedAt: oldLibrary.lastUpdate,
|
|
|
|
extraData
|
2023-07-22 21:25:20 +02:00
|
|
|
}
|
2023-08-16 01:03:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Destroy library by id
|
2024-05-29 00:24:02 +02:00
|
|
|
* @param {string} libraryId
|
|
|
|
* @returns
|
2023-08-16 01:03:43 +02:00
|
|
|
*/
|
|
|
|
static removeById(libraryId) {
|
|
|
|
return this.destroy({
|
|
|
|
where: {
|
|
|
|
id: libraryId
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get all library ids
|
|
|
|
* @returns {Promise<string[]>} array of library ids
|
|
|
|
*/
|
|
|
|
static async getAllLibraryIds() {
|
|
|
|
const libraries = await this.findAll({
|
|
|
|
attributes: ['id', 'displayOrder'],
|
|
|
|
order: [['displayOrder', 'ASC']]
|
|
|
|
})
|
2024-05-29 00:24:02 +02:00
|
|
|
return libraries.map((l) => l.id)
|
2023-08-16 01:03:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find Library by primary key & return oldLibrary
|
2024-05-29 00:24:02 +02:00
|
|
|
* @param {string} libraryId
|
2023-08-16 01:03:43 +02:00
|
|
|
* @returns {Promise<oldLibrary|null>} Returns null if not found
|
|
|
|
*/
|
|
|
|
static async getOldById(libraryId) {
|
|
|
|
if (!libraryId) return null
|
|
|
|
const library = await this.findByPk(libraryId, {
|
2023-08-16 21:49:06 +02:00
|
|
|
include: this.sequelize.models.libraryFolder
|
2023-08-16 01:03:43 +02:00
|
|
|
})
|
|
|
|
if (!library) return null
|
|
|
|
return this.getOldLibrary(library)
|
|
|
|
}
|
2023-07-22 21:25:20 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
/**
|
|
|
|
* Get the largest value in the displayOrder column
|
|
|
|
* Used for setting a new libraries display order
|
|
|
|
* @returns {Promise<number>}
|
|
|
|
*/
|
|
|
|
static getMaxDisplayOrder() {
|
|
|
|
return this.max('displayOrder') || 0
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates displayOrder to be sequential
|
|
|
|
* Used after removing a library
|
|
|
|
*/
|
|
|
|
static async resetDisplayOrder() {
|
|
|
|
const libraries = await this.findAll({
|
|
|
|
order: [['displayOrder', 'ASC']]
|
|
|
|
})
|
|
|
|
for (let i = 0; i < libraries.length; i++) {
|
|
|
|
const library = libraries[i]
|
|
|
|
if (library.displayOrder !== i + 1) {
|
2024-01-04 00:19:28 +01:00
|
|
|
Logger.debug(`[Library] Updating display order of library from ${library.displayOrder} to ${i + 1}`)
|
2023-08-16 01:03:43 +02:00
|
|
|
await library.update({ displayOrder: i + 1 }).catch((error) => {
|
|
|
|
Logger.error(`[Library] Failed to update library display order to ${i + 1}`, error)
|
|
|
|
})
|
2023-07-22 21:25:20 +02:00
|
|
|
}
|
|
|
|
}
|
2023-07-05 01:14:44 +02:00
|
|
|
}
|
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
/**
|
|
|
|
* Initialize model
|
2024-05-29 00:24:02 +02:00
|
|
|
* @param {import('../Database').sequelize} sequelize
|
2023-08-16 01:03:43 +02:00
|
|
|
*/
|
|
|
|
static init(sequelize) {
|
2024-05-29 00:24:02 +02:00
|
|
|
super.init(
|
|
|
|
{
|
|
|
|
id: {
|
|
|
|
type: DataTypes.UUID,
|
|
|
|
defaultValue: DataTypes.UUIDV4,
|
|
|
|
primaryKey: true
|
|
|
|
},
|
|
|
|
name: DataTypes.STRING,
|
|
|
|
displayOrder: DataTypes.INTEGER,
|
|
|
|
icon: DataTypes.STRING,
|
|
|
|
mediaType: DataTypes.STRING,
|
|
|
|
provider: DataTypes.STRING,
|
|
|
|
lastScan: DataTypes.DATE,
|
|
|
|
lastScanVersion: DataTypes.STRING,
|
|
|
|
settings: DataTypes.JSON,
|
|
|
|
extraData: DataTypes.JSON
|
2023-08-16 01:03:43 +02:00
|
|
|
},
|
2024-05-29 00:24:02 +02:00
|
|
|
{
|
|
|
|
sequelize,
|
|
|
|
modelName: 'library'
|
|
|
|
}
|
|
|
|
)
|
2023-08-16 01:03:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-29 00:24:02 +02:00
|
|
|
module.exports = Library
|