2024-08-16 00:05:18 +02:00
const util = require ( 'util' )
2024-04-13 00:34:10 +02:00
const Path = require ( 'path' )
2024-03-23 20:56:32 +01:00
const { DataTypes , Model } = require ( 'sequelize' )
2024-04-13 00:34:10 +02:00
const fsExtra = require ( '../libs/fsExtra' )
2023-07-05 01:14:44 +02:00
const Logger = require ( '../Logger' )
const oldLibraryItem = require ( '../objects/LibraryItem' )
2023-07-29 01:03:31 +02:00
const libraryFilters = require ( '../utils/queries/libraryFilters' )
2023-07-05 01:14:44 +02:00
const { areEquivalent } = require ( '../utils/index' )
2024-04-13 00:34:10 +02:00
const { filePathToPOSIX , getFileTimestampsWithIno } = require ( '../utils/fileUtils' )
const LibraryFile = require ( '../objects/files/LibraryFile' )
2023-08-28 00:19:57 +02:00
const Book = require ( './Book' )
const Podcast = require ( './Podcast' )
2023-07-05 01:14:44 +02:00
2023-08-25 00:55:29 +02:00
/ * *
* @ typedef LibraryFileObject
* @ property { string } ino
* @ property { boolean } isSupplementary
* @ property { number } addedAt
* @ property { number } updatedAt
* @ property { { filename : string , ext : string , path : string , relPath : string , size : number , mtimeMs : number , ctimeMs : number , birthtimeMs : number } } metadata
* /
2023-07-19 22:36:18 +02:00
2024-01-16 23:31:16 +01:00
/ * *
* @ typedef LibraryItemExpandedProperties
2024-05-29 00:24:02 +02:00
* @ property { Book . BookExpanded | Podcast . PodcastExpanded } media
*
2024-01-16 23:31:16 +01:00
* @ typedef { LibraryItem & LibraryItemExpandedProperties } LibraryItemExpanded
* /
2023-08-16 23:38:48 +02:00
class LibraryItem extends Model {
constructor ( values , options ) {
super ( values , options )
2023-08-25 00:55:29 +02:00
/** @type {string} */
2023-08-16 23:38:48 +02:00
this . id
/** @type {string} */
this . ino
/** @type {string} */
this . path
/** @type {string} */
this . relPath
2023-08-25 00:55:29 +02:00
/** @type {string} */
2023-08-16 23:38:48 +02:00
this . mediaId
/** @type {string} */
this . mediaType
/** @type {boolean} */
this . isFile
/** @type {boolean} */
this . isMissing
/** @type {boolean} */
this . isInvalid
/** @type {Date} */
this . mtime
/** @type {Date} */
this . ctime
/** @type {Date} */
this . birthtime
/** @type {BigInt} */
this . size
/** @type {Date} */
this . lastScan
/** @type {string} */
this . lastScanVersion
2023-08-25 00:55:29 +02:00
/** @type {LibraryFileObject[]} */
2023-08-16 23:38:48 +02:00
this . libraryFiles
/** @type {Object} */
this . extraData
2023-08-25 00:55:29 +02:00
/** @type {string} */
2023-08-16 23:38:48 +02:00
this . libraryId
2023-08-25 00:55:29 +02:00
/** @type {string} */
2023-08-16 23:38:48 +02:00
this . libraryFolderId
/** @type {Date} */
this . createdAt
/** @type {Date} */
this . updatedAt
2024-12-14 23:55:56 +01:00
/** @type {Book.BookExpanded|Podcast.PodcastExpanded} - only set when expanded */
this . media
2023-08-16 23:38:48 +02:00
}
2023-07-19 22:36:18 +02:00
2023-08-16 23:38:48 +02:00
/ * *
* Gets library items partially expanded , not including podcast episodes
* @ todo temporary solution
2024-05-29 00:24:02 +02:00
*
2023-08-16 23:38:48 +02:00
* @ param { number } offset
* @ param { number } limit
2023-10-21 20:53:00 +02:00
* @ returns { Promise < LibraryItem [ ] > } LibraryItem
2023-08-16 23:38:48 +02:00
* /
2023-09-04 23:33:55 +02:00
static getLibraryItemsIncrement ( offset , limit , where = null ) {
2023-08-16 23:38:48 +02:00
return this . findAll ( {
2023-09-04 23:33:55 +02:00
where ,
2023-08-16 23:38:48 +02:00
include : [
{
model : this . sequelize . models . book ,
include : [
{
model : this . sequelize . models . author ,
through : {
attributes : [ 'createdAt' ]
2023-07-05 01:14:44 +02:00
}
2023-08-16 23:38:48 +02:00
} ,
{
model : this . sequelize . models . series ,
through : {
attributes : [ 'sequence' , 'createdAt' ]
2023-07-05 01:14:44 +02:00
}
2023-08-16 23:38:48 +02:00
}
]
} ,
{
model : this . sequelize . models . podcast
}
] ,
order : [
[ 'createdAt' , 'ASC' ] ,
// Ensure author & series stay in the same order
[ this . sequelize . models . book , this . sequelize . models . author , this . sequelize . models . bookAuthor , 'createdAt' , 'ASC' ] ,
[ this . sequelize . models . book , this . sequelize . models . series , 'bookSeries' , 'createdAt' , 'ASC' ]
] ,
offset ,
limit
} )
}
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
/ * *
2024-12-30 23:54:48 +01:00
*
2024-03-23 20:56:32 +01:00
* @ param { import ( 'sequelize' ) . WhereOptions } [ where ]
2023-08-16 23:38:48 +02:00
* @ returns { Array < objects . LibraryItem > } old library items
* /
static async getAllOldLibraryItems ( where = null ) {
let libraryItems = await this . findAll ( {
where ,
include : [
{
model : this . sequelize . models . book ,
include : [
{
model : this . sequelize . models . author ,
through : {
attributes : [ ]
}
} ,
{
model : this . sequelize . models . series ,
through : {
attributes : [ 'sequence' ]
}
}
]
} ,
{
model : this . sequelize . models . podcast ,
include : [
{
model : this . sequelize . models . podcastEpisode
}
]
}
]
} )
2024-05-29 00:24:02 +02:00
return libraryItems . map ( ( ti ) => this . getOldLibraryItem ( ti ) )
2023-08-16 23:38:48 +02:00
}
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
/ * *
* Convert an expanded LibraryItem into an old library item
2024-05-29 00:24:02 +02:00
*
* @ param { Model < LibraryItem > } libraryItemExpanded
2023-08-16 23:38:48 +02:00
* @ returns { oldLibraryItem }
* /
static getOldLibraryItem ( libraryItemExpanded ) {
let media = null
if ( libraryItemExpanded . mediaType === 'book' ) {
media = this . sequelize . models . book . getOldBook ( libraryItemExpanded )
} else if ( libraryItemExpanded . mediaType === 'podcast' ) {
media = this . sequelize . models . podcast . getOldPodcast ( libraryItemExpanded )
2023-07-05 01:14:44 +02:00
}
2023-08-16 23:38:48 +02:00
return new oldLibraryItem ( {
id : libraryItemExpanded . id ,
ino : libraryItemExpanded . ino ,
oldLibraryItemId : libraryItemExpanded . extraData ? . oldLibraryItemId || null ,
libraryId : libraryItemExpanded . libraryId ,
folderId : libraryItemExpanded . libraryFolderId ,
path : libraryItemExpanded . path ,
relPath : libraryItemExpanded . relPath ,
isFile : libraryItemExpanded . isFile ,
mtimeMs : libraryItemExpanded . mtime ? . valueOf ( ) ,
ctimeMs : libraryItemExpanded . ctime ? . valueOf ( ) ,
birthtimeMs : libraryItemExpanded . birthtime ? . valueOf ( ) ,
addedAt : libraryItemExpanded . createdAt . valueOf ( ) ,
updatedAt : libraryItemExpanded . updatedAt . valueOf ( ) ,
lastScan : libraryItemExpanded . lastScan ? . valueOf ( ) ,
scanVersion : libraryItemExpanded . lastScanVersion ,
isMissing : ! ! libraryItemExpanded . isMissing ,
isInvalid : ! ! libraryItemExpanded . isInvalid ,
mediaType : libraryItemExpanded . mediaType ,
media ,
libraryFiles : libraryItemExpanded . libraryFiles
} )
}
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
static async fullCreateFromOld ( oldLibraryItem ) {
const newLibraryItem = await this . create ( this . getFromOld ( oldLibraryItem ) )
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
if ( oldLibraryItem . mediaType === 'book' ) {
const bookObj = this . sequelize . models . book . getFromOld ( oldLibraryItem . media )
bookObj . libraryItemId = newLibraryItem . id
const newBook = await this . sequelize . models . book . create ( bookObj )
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
const oldBookAuthors = oldLibraryItem . media . metadata . authors || [ ]
const oldBookSeriesAll = oldLibraryItem . media . metadata . series || [ ]
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
for ( const oldBookAuthor of oldBookAuthors ) {
await this . sequelize . models . bookAuthor . create ( { authorId : oldBookAuthor . id , bookId : newBook . id } )
}
for ( const oldSeries of oldBookSeriesAll ) {
await this . sequelize . models . bookSeries . create ( { seriesId : oldSeries . id , bookId : newBook . id , sequence : oldSeries . sequence } )
}
} else if ( oldLibraryItem . mediaType === 'podcast' ) {
const podcastObj = this . sequelize . models . podcast . getFromOld ( oldLibraryItem . media )
podcastObj . libraryItemId = newLibraryItem . id
const newPodcast = await this . sequelize . models . podcast . create ( podcastObj )
const oldEpisodes = oldLibraryItem . media . episodes || [ ]
for ( const oldEpisode of oldEpisodes ) {
const episodeObj = this . sequelize . models . podcastEpisode . getFromOld ( oldEpisode )
episodeObj . libraryItemId = newLibraryItem . id
episodeObj . podcastId = newPodcast . id
await this . sequelize . models . podcastEpisode . create ( episodeObj )
}
2023-07-05 01:14:44 +02:00
}
2023-08-16 23:38:48 +02:00
return newLibraryItem
}
2023-07-05 01:14:44 +02:00
2024-02-18 21:58:46 +01:00
/ * *
* Updates libraryItem , book , authors and series from old library item
2024-05-29 00:24:02 +02:00
*
* @ param { oldLibraryItem } oldLibraryItem
2024-02-18 21:58:46 +01:00
* @ returns { Promise < boolean > } true if updates were made
* /
2023-08-16 23:38:48 +02:00
static async fullUpdateFromOld ( oldLibraryItem ) {
2024-11-16 05:26:32 +01:00
const libraryItemExpanded = await this . getExpandedById ( oldLibraryItem . id )
2023-08-16 23:38:48 +02:00
if ( ! libraryItemExpanded ) return false
let hasUpdates = false
// Check update Book/Podcast
if ( libraryItemExpanded . media ) {
let updatedMedia = null
if ( libraryItemExpanded . mediaType === 'podcast' ) {
updatedMedia = this . sequelize . models . podcast . getFromOld ( oldLibraryItem . media )
const existingPodcastEpisodes = libraryItemExpanded . media . podcastEpisodes || [ ]
const updatedPodcastEpisodes = oldLibraryItem . media . episodes || [ ]
for ( const existingPodcastEpisode of existingPodcastEpisodes ) {
// Episode was removed
2024-05-29 00:24:02 +02:00
if ( ! updatedPodcastEpisodes . some ( ( ep ) => ep . id === existingPodcastEpisode . id ) ) {
2024-01-04 00:19:28 +01:00
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " episode " ${ existingPodcastEpisode . title } " was removed ` )
2023-08-16 23:38:48 +02:00
await existingPodcastEpisode . destroy ( )
hasUpdates = true
2023-07-05 01:14:44 +02:00
}
2023-08-16 23:38:48 +02:00
}
for ( const updatedPodcastEpisode of updatedPodcastEpisodes ) {
2024-05-29 00:24:02 +02:00
const existingEpisodeMatch = existingPodcastEpisodes . find ( ( ep ) => ep . id === updatedPodcastEpisode . id )
2023-08-16 23:38:48 +02:00
if ( ! existingEpisodeMatch ) {
2024-01-04 00:19:28 +01:00
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " episode " ${ updatedPodcastEpisode . title } " was added ` )
2023-08-16 23:38:48 +02:00
await this . sequelize . models . podcastEpisode . createFromOld ( updatedPodcastEpisode )
hasUpdates = true
} else {
const updatedEpisodeCleaned = this . sequelize . models . podcastEpisode . getFromOld ( updatedPodcastEpisode )
let episodeHasUpdates = false
for ( const key in updatedEpisodeCleaned ) {
let existingValue = existingEpisodeMatch [ key ]
if ( existingValue instanceof Date ) existingValue = existingValue . valueOf ( )
if ( ! areEquivalent ( updatedEpisodeCleaned [ key ] , existingValue , true ) ) {
2024-08-16 00:05:18 +02:00
Logger . debug ( util . format ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " episode " ${ existingEpisodeMatch . title } " ${ key } was updated from %j to %j ` , existingValue , updatedEpisodeCleaned [ key ] ) )
2023-08-16 23:38:48 +02:00
episodeHasUpdates = true
}
2023-07-05 01:14:44 +02:00
}
2023-08-16 23:38:48 +02:00
if ( episodeHasUpdates ) {
await existingEpisodeMatch . update ( updatedEpisodeCleaned )
2023-07-05 01:14:44 +02:00
hasUpdates = true
}
}
2023-08-16 23:38:48 +02:00
}
} else if ( libraryItemExpanded . mediaType === 'book' ) {
updatedMedia = this . sequelize . models . book . getFromOld ( oldLibraryItem . media )
const existingAuthors = libraryItemExpanded . media . authors || [ ]
const existingSeriesAll = libraryItemExpanded . media . series || [ ]
const updatedAuthors = oldLibraryItem . media . metadata . authors || [ ]
2024-05-29 00:24:02 +02:00
const uniqueUpdatedAuthors = updatedAuthors . filter ( ( au , idx ) => updatedAuthors . findIndex ( ( a ) => a . id === au . id ) === idx )
2023-08-16 23:38:48 +02:00
const updatedSeriesAll = oldLibraryItem . media . metadata . series || [ ]
for ( const existingAuthor of existingAuthors ) {
// Author was removed from Book
2024-05-29 00:24:02 +02:00
if ( ! uniqueUpdatedAuthors . some ( ( au ) => au . id === existingAuthor . id ) ) {
2024-01-04 00:19:28 +01:00
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " author " ${ existingAuthor . name } " was removed ` )
2023-08-16 23:38:48 +02:00
await this . sequelize . models . bookAuthor . removeByIds ( existingAuthor . id , libraryItemExpanded . media . id )
hasUpdates = true
2023-07-05 01:14:44 +02:00
}
}
2024-02-25 08:01:26 +01:00
for ( const updatedAuthor of uniqueUpdatedAuthors ) {
2023-08-16 23:38:48 +02:00
// Author was added
2024-05-29 00:24:02 +02:00
if ( ! existingAuthors . some ( ( au ) => au . id === updatedAuthor . id ) ) {
2024-01-04 00:19:28 +01:00
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " author " ${ updatedAuthor . name } " was added ` )
2023-08-16 23:38:48 +02:00
await this . sequelize . models . bookAuthor . create ( { authorId : updatedAuthor . id , bookId : libraryItemExpanded . media . id } )
hasUpdates = true
}
}
for ( const existingSeries of existingSeriesAll ) {
// Series was removed
2024-05-29 00:24:02 +02:00
if ( ! updatedSeriesAll . some ( ( se ) => se . id === existingSeries . id ) ) {
2024-01-04 00:19:28 +01:00
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " series " ${ existingSeries . name } " was removed ` )
2023-08-16 23:38:48 +02:00
await this . sequelize . models . bookSeries . removeByIds ( existingSeries . id , libraryItemExpanded . media . id )
hasUpdates = true
2023-07-05 01:14:44 +02:00
}
}
2023-08-16 23:38:48 +02:00
for ( const updatedSeries of updatedSeriesAll ) {
// Series was added/updated
2024-05-29 00:24:02 +02:00
const existingSeriesMatch = existingSeriesAll . find ( ( se ) => se . id === updatedSeries . id )
2023-08-16 23:38:48 +02:00
if ( ! existingSeriesMatch ) {
2024-01-04 00:19:28 +01:00
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " series " ${ updatedSeries . name } " was added ` )
2023-08-16 23:38:48 +02:00
await this . sequelize . models . bookSeries . create ( { seriesId : updatedSeries . id , bookId : libraryItemExpanded . media . id , sequence : updatedSeries . sequence } )
hasUpdates = true
} else if ( existingSeriesMatch . bookSeries . sequence !== updatedSeries . sequence ) {
2024-01-04 00:19:28 +01:00
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " series " ${ updatedSeries . name } " sequence was updated from " ${ existingSeriesMatch . bookSeries . sequence } " to " ${ updatedSeries . sequence } " ` )
2023-08-16 23:38:48 +02:00
await existingSeriesMatch . bookSeries . update ( { id : updatedSeries . id , sequence : updatedSeries . sequence } )
hasUpdates = true
}
2023-07-05 01:14:44 +02:00
}
}
2023-08-16 23:38:48 +02:00
let hasMediaUpdates = false
for ( const key in updatedMedia ) {
let existingValue = libraryItemExpanded . media [ key ]
2023-07-05 01:14:44 +02:00
if ( existingValue instanceof Date ) existingValue = existingValue . valueOf ( )
2023-08-16 23:38:48 +02:00
if ( ! areEquivalent ( updatedMedia [ key ] , existingValue , true ) ) {
2024-09-06 23:58:40 +02:00
if ( key === 'chapters' ) {
// Handle logging of chapters separately because the object is large
const chaptersRemoved = libraryItemExpanded . media . chapters . filter ( ( ch ) => ! updatedMedia . chapters . some ( ( uch ) => uch . id === ch . id ) )
if ( chaptersRemoved . length ) {
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " chapters removed: ${ chaptersRemoved . map ( ( ch ) => ch . title ) . join ( ', ' ) } ` )
}
const chaptersAdded = updatedMedia . chapters . filter ( ( uch ) => ! libraryItemExpanded . media . chapters . some ( ( ch ) => ch . id === uch . id ) )
if ( chaptersAdded . length ) {
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " chapters added: ${ chaptersAdded . map ( ( ch ) => ch . title ) . join ( ', ' ) } ` )
}
if ( ! chaptersRemoved . length && ! chaptersAdded . length ) {
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " chapters updated ` )
}
} else {
Logger . debug ( util . format ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " ${ libraryItemExpanded . mediaType } . ${ key } updated from %j to %j ` , existingValue , updatedMedia [ key ] ) )
}
2023-08-16 23:38:48 +02:00
hasMediaUpdates = true
2023-07-05 01:14:44 +02:00
}
}
2023-08-16 23:38:48 +02:00
if ( hasMediaUpdates && updatedMedia ) {
await libraryItemExpanded . media . update ( updatedMedia )
2023-07-05 01:14:44 +02:00
hasUpdates = true
}
}
2023-08-16 23:38:48 +02:00
const updatedLibraryItem = this . getFromOld ( oldLibraryItem )
let hasLibraryItemUpdates = false
for ( const key in updatedLibraryItem ) {
let existingValue = libraryItemExpanded [ key ]
if ( existingValue instanceof Date ) existingValue = existingValue . valueOf ( )
if ( ! areEquivalent ( updatedLibraryItem [ key ] , existingValue , true ) ) {
2024-08-16 00:05:18 +02:00
if ( key === 'libraryFiles' ) {
// Handle logging of libraryFiles separately because the object is large (should be addressed when migrating off the old library item model)
const libraryFilesRemoved = libraryItemExpanded . libraryFiles . filter ( ( lf ) => ! updatedLibraryItem . libraryFiles . some ( ( ulf ) => ulf . ino === lf . ino ) )
if ( libraryFilesRemoved . length ) {
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " library files removed: ${ libraryFilesRemoved . map ( ( lf ) => lf . metadata . path ) . join ( ', ' ) } ` )
}
const libraryFilesAdded = updatedLibraryItem . libraryFiles . filter ( ( ulf ) => ! libraryItemExpanded . libraryFiles . some ( ( lf ) => lf . ino === ulf . ino ) )
if ( libraryFilesAdded . length ) {
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " library files added: ${ libraryFilesAdded . map ( ( lf ) => lf . metadata . path ) . join ( ', ' ) } ` )
}
if ( ! libraryFilesRemoved . length && ! libraryFilesAdded . length ) {
Logger . debug ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " library files updated ` )
}
} else {
Logger . debug ( util . format ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " ${ key } updated from %j to %j ` , existingValue , updatedLibraryItem [ key ] ) )
}
2023-08-16 23:38:48 +02:00
hasLibraryItemUpdates = true
2024-02-07 19:57:50 +01:00
if ( key === 'updatedAt' ) {
libraryItemExpanded . changed ( 'updatedAt' , true )
}
2023-07-05 01:14:44 +02:00
}
}
2023-08-16 23:38:48 +02:00
if ( hasLibraryItemUpdates ) {
await libraryItemExpanded . update ( updatedLibraryItem )
Logger . info ( ` [LibraryItem] Library item " ${ libraryItemExpanded . id } " updated ` )
hasUpdates = true
}
return hasUpdates
}
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
static getFromOld ( oldLibraryItem ) {
const extraData = { }
if ( oldLibraryItem . oldLibraryItemId ) {
extraData . oldLibraryItemId = oldLibraryItem . oldLibraryItemId
2023-07-05 01:14:44 +02:00
}
2023-08-16 23:38:48 +02:00
return {
id : oldLibraryItem . id ,
ino : oldLibraryItem . ino ,
path : oldLibraryItem . path ,
relPath : oldLibraryItem . relPath ,
mediaId : oldLibraryItem . media . id ,
mediaType : oldLibraryItem . mediaType ,
isFile : ! ! oldLibraryItem . isFile ,
isMissing : ! ! oldLibraryItem . isMissing ,
isInvalid : ! ! oldLibraryItem . isInvalid ,
mtime : oldLibraryItem . mtimeMs ,
ctime : oldLibraryItem . ctimeMs ,
2024-02-07 19:57:50 +01:00
updatedAt : oldLibraryItem . updatedAt ,
2023-08-16 23:38:48 +02:00
birthtime : oldLibraryItem . birthtimeMs ,
size : oldLibraryItem . size ,
lastScan : oldLibraryItem . lastScan ,
lastScanVersion : oldLibraryItem . scanVersion ,
libraryId : oldLibraryItem . libraryId ,
libraryFolderId : oldLibraryItem . folderId ,
2024-05-29 00:24:02 +02:00
libraryFiles : oldLibraryItem . libraryFiles ? . map ( ( lf ) => lf . toJSON ( ) ) || [ ] ,
2023-08-16 23:38:48 +02:00
extraData
}
}
2023-07-05 01:14:44 +02:00
2024-08-04 00:09:17 +02:00
/ * *
* Remove library item by id
*
* @ param { string } libraryItemId
* @ returns { Promise < number > } The number of destroyed rows
* /
2023-08-16 23:38:48 +02:00
static removeById ( libraryItemId ) {
return this . destroy ( {
where : {
id : libraryItemId
} ,
individualHooks : true
} )
}
2024-01-16 23:31:16 +01:00
/ * *
2024-05-29 00:24:02 +02:00
*
* @ param { string } libraryItemId
2024-01-16 23:31:16 +01:00
* @ returns { Promise < LibraryItemExpanded > }
* /
static async getExpandedById ( libraryItemId ) {
if ( ! libraryItemId ) return null
const libraryItem = await this . findByPk ( libraryItemId )
if ( ! libraryItem ) {
Logger . error ( ` [LibraryItem] Library item not found with id " ${ libraryItemId } " ` )
return null
}
if ( libraryItem . mediaType === 'podcast' ) {
libraryItem . media = await libraryItem . getMedia ( {
include : [
{
model : this . sequelize . models . podcastEpisode
}
]
} )
} else {
libraryItem . media = await libraryItem . getMedia ( {
include : [
{
model : this . sequelize . models . author ,
through : {
attributes : [ ]
}
} ,
{
model : this . sequelize . models . series ,
through : {
2024-11-21 21:19:40 +01:00
attributes : [ 'id' , 'sequence' ]
2024-01-16 23:31:16 +01:00
}
}
] ,
order : [
[ this . sequelize . models . author , this . sequelize . models . bookAuthor , 'createdAt' , 'ASC' ] ,
[ this . sequelize . models . series , 'bookSeries' , 'createdAt' , 'ASC' ]
]
} )
}
if ( ! libraryItem . media ) return null
return libraryItem
}
2025-01-03 18:16:03 +01:00
/ * *
*
* @ param { import ( 'sequelize' ) . WhereOptions } where
* @ param { import ( 'sequelize' ) . IncludeOptions } [ include ]
* @ returns { Promise < LibraryItemExpanded > }
* /
static async findOneExpanded ( where , include = null ) {
const libraryItem = await this . findOne ( {
where ,
include
} )
if ( ! libraryItem ) {
Logger . error ( ` [LibraryItem] Library item not found ` )
return null
}
if ( libraryItem . mediaType === 'podcast' ) {
libraryItem . media = await libraryItem . getMedia ( {
include : [
{
model : this . sequelize . models . podcastEpisode
}
]
} )
} else {
libraryItem . media = await libraryItem . getMedia ( {
include : [
{
model : this . sequelize . models . author ,
through : {
attributes : [ ]
}
} ,
{
model : this . sequelize . models . series ,
through : {
attributes : [ 'id' , 'sequence' ]
}
}
] ,
order : [
[ this . sequelize . models . author , this . sequelize . models . bookAuthor , 'createdAt' , 'ASC' ] ,
[ this . sequelize . models . series , 'bookSeries' , 'createdAt' , 'ASC' ]
]
} )
}
if ( ! libraryItem . media ) return null
return libraryItem
}
2023-08-16 23:38:48 +02:00
/ * *
* Get old library item by id
2024-05-29 00:24:02 +02:00
* @ param { string } libraryItemId
2023-08-16 23:38:48 +02:00
* @ returns { oldLibraryItem }
* /
static async getOldById ( libraryItemId ) {
if ( ! libraryItemId ) return null
2023-12-30 19:12:48 +01:00
const libraryItem = await this . findByPk ( libraryItemId )
2023-12-30 23:14:14 +01:00
if ( ! libraryItem ) {
Logger . error ( ` [LibraryItem] Library item not found with id " ${ libraryItemId } " ` )
return null
}
2023-12-30 19:12:48 +01:00
if ( libraryItem . mediaType === 'podcast' ) {
libraryItem . media = await libraryItem . getMedia ( {
include : [
{
model : this . sequelize . models . podcastEpisode
}
]
} )
} else {
libraryItem . media = await libraryItem . getMedia ( {
include : [
{
model : this . sequelize . models . author ,
through : {
attributes : [ ]
2023-08-16 23:38:48 +02:00
}
2023-12-30 19:12:48 +01:00
} ,
{
model : this . sequelize . models . series ,
through : {
attributes : [ 'sequence' ]
2023-08-16 23:38:48 +02:00
}
2023-12-30 19:12:48 +01:00
}
] ,
order : [
[ this . sequelize . models . author , this . sequelize . models . bookAuthor , 'createdAt' , 'ASC' ] ,
[ this . sequelize . models . series , 'bookSeries' , 'createdAt' , 'ASC' ]
]
} )
}
2023-12-30 23:14:14 +01:00
if ( ! libraryItem . media ) return null
2023-08-16 23:38:48 +02:00
return this . getOldLibraryItem ( libraryItem )
}
2023-08-06 22:06:45 +02:00
2023-08-16 23:38:48 +02:00
/ * *
* Get library items using filter and sort
2024-08-24 23:09:54 +02:00
* @ param { import ( './Library' ) } library
2024-08-11 00:15:21 +02:00
* @ param { import ( './User' ) } user
2024-05-29 00:24:02 +02:00
* @ param { object } options
2024-07-02 00:26:13 +02:00
* @ returns { { libraryItems : oldLibraryItem [ ] , count : number } }
2023-08-16 23:38:48 +02:00
* /
static async getByFilterAndSort ( library , user , options ) {
let start = Date . now ( )
2024-08-24 23:09:54 +02:00
const { libraryItems , count } = await libraryFilters . getFilteredLibraryItems ( library . id , user , options )
2023-08-16 23:38:48 +02:00
Logger . debug ( ` Loaded ${ libraryItems . length } of ${ count } items for libary page in ${ ( ( Date . now ( ) - start ) / 1000 ) . toFixed ( 2 ) } s ` )
return {
2024-05-29 00:24:02 +02:00
libraryItems : libraryItems . map ( ( li ) => {
2023-08-16 23:38:48 +02:00
const oldLibraryItem = this . getOldLibraryItem ( li ) . toJSONMinified ( )
if ( li . collapsedSeries ) {
oldLibraryItem . collapsedSeries = li . collapsedSeries
}
if ( li . series ) {
oldLibraryItem . media . metadata . series = li . series
}
if ( li . rssFeed ) {
2024-12-16 00:54:36 +01:00
oldLibraryItem . rssFeed = li . rssFeed . toOldJSONMinified ( )
2023-08-16 23:38:48 +02:00
}
if ( li . media . numEpisodes ) {
oldLibraryItem . media . numEpisodes = li . media . numEpisodes
}
if ( li . size && ! oldLibraryItem . media . size ) {
oldLibraryItem . media . size = li . size
}
if ( li . numEpisodesIncomplete ) {
oldLibraryItem . numEpisodesIncomplete = li . numEpisodesIncomplete
}
2024-07-07 22:51:50 +02:00
if ( li . mediaItemShare ) {
oldLibraryItem . mediaItemShare = li . mediaItemShare
2024-07-02 00:26:13 +02:00
}
2023-08-01 00:59:51 +02:00
2023-08-16 23:38:48 +02:00
return oldLibraryItem
} ) ,
count
2023-07-29 01:03:31 +02:00
}
2023-08-16 23:38:48 +02:00
}
2023-07-29 01:03:31 +02:00
2023-08-16 23:38:48 +02:00
/ * *
* Get home page data personalized shelves
2024-08-24 23:09:54 +02:00
* @ param { import ( './Library' ) } library
2024-08-11 00:15:21 +02:00
* @ param { import ( './User' ) } user
2024-05-29 00:24:02 +02:00
* @ param { string [ ] } include
* @ param { number } limit
2023-08-16 23:38:48 +02:00
* @ returns { object [ ] } array of shelf objects
* /
static async getPersonalizedShelves ( library , user , include , limit ) {
const fullStart = Date . now ( ) // Used for testing load times
const shelves = [ ]
// "Continue Listening" shelf
const itemsInProgressPayload = await libraryFilters . getMediaItemsInProgress ( library , user , include , limit , false )
if ( itemsInProgressPayload . items . length ) {
2024-05-29 00:24:02 +02:00
const ebookOnlyItemsInProgress = itemsInProgressPayload . items . filter ( ( li ) => li . media . isEBookOnly )
const audioOnlyItemsInProgress = itemsInProgressPayload . items . filter ( ( li ) => ! li . media . isEBookOnly )
2023-08-16 23:38:48 +02:00
shelves . push ( {
id : 'continue-listening' ,
label : 'Continue Listening' ,
labelStringKey : 'LabelContinueListening' ,
type : library . isPodcast ? 'episode' : 'book' ,
entities : audioOnlyItemsInProgress ,
total : itemsInProgressPayload . count
} )
2023-08-08 00:59:04 +02:00
2023-08-16 23:38:48 +02:00
if ( ebookOnlyItemsInProgress . length ) {
// "Continue Reading" shelf
2023-08-03 01:29:28 +02:00
shelves . push ( {
2023-08-16 23:38:48 +02:00
id : 'continue-reading' ,
label : 'Continue Reading' ,
labelStringKey : 'LabelContinueReading' ,
type : 'book' ,
entities : ebookOnlyItemsInProgress ,
2023-08-03 01:29:28 +02:00
total : itemsInProgressPayload . count
} )
2023-08-16 23:38:48 +02:00
}
}
2024-01-04 00:19:28 +01:00
Logger . debug ( ` Loaded ${ itemsInProgressPayload . items . length } of ${ itemsInProgressPayload . count } items for "Continue Listening/Reading" in ${ ( ( Date . now ( ) - fullStart ) / 1000 ) . toFixed ( 2 ) } s ` )
2023-08-03 01:29:28 +02:00
2023-08-16 23:38:48 +02:00
let start = Date . now ( )
if ( library . isBook ) {
start = Date . now ( )
// "Continue Series" shelf
const continueSeriesPayload = await libraryFilters . getLibraryItemsContinueSeries ( library , user , include , limit )
if ( continueSeriesPayload . libraryItems . length ) {
shelves . push ( {
id : 'continue-series' ,
label : 'Continue Series' ,
labelStringKey : 'LabelContinueSeries' ,
type : 'book' ,
entities : continueSeriesPayload . libraryItems ,
total : continueSeriesPayload . count
} )
2023-08-08 00:59:04 +02:00
}
2024-01-04 00:19:28 +01:00
Logger . debug ( ` Loaded ${ continueSeriesPayload . libraryItems . length } of ${ continueSeriesPayload . count } items for "Continue Series" in ${ ( ( Date . now ( ) - start ) / 1000 ) . toFixed ( 2 ) } s ` )
2023-08-16 23:38:48 +02:00
} else if ( library . isPodcast ) {
// "Newest Episodes" shelf
const newestEpisodesPayload = await libraryFilters . getNewestPodcastEpisodes ( library , user , limit )
if ( newestEpisodesPayload . libraryItems . length ) {
shelves . push ( {
id : 'newest-episodes' ,
label : 'Newest Episodes' ,
labelStringKey : 'LabelNewestEpisodes' ,
type : 'episode' ,
entities : newestEpisodesPayload . libraryItems ,
total : newestEpisodesPayload . count
} )
2023-08-03 01:29:28 +02:00
}
2024-01-04 00:19:28 +01:00
Logger . debug ( ` Loaded ${ newestEpisodesPayload . libraryItems . length } of ${ newestEpisodesPayload . count } episodes for "Newest Episodes" in ${ ( ( Date . now ( ) - start ) / 1000 ) . toFixed ( 2 ) } s ` )
2023-08-16 23:38:48 +02:00
}
2023-08-03 01:29:28 +02:00
2023-08-16 23:38:48 +02:00
start = Date . now ( )
// "Recently Added" shelf
const mostRecentPayload = await libraryFilters . getLibraryItemsMostRecentlyAdded ( library , user , include , limit )
if ( mostRecentPayload . libraryItems . length ) {
shelves . push ( {
id : 'recently-added' ,
label : 'Recently Added' ,
labelStringKey : 'LabelRecentlyAdded' ,
type : library . mediaType ,
entities : mostRecentPayload . libraryItems ,
total : mostRecentPayload . count
} )
}
2024-01-04 00:19:28 +01:00
Logger . debug ( ` Loaded ${ mostRecentPayload . libraryItems . length } of ${ mostRecentPayload . count } items for "Recently Added" in ${ ( ( Date . now ( ) - start ) / 1000 ) . toFixed ( 2 ) } s ` )
2023-08-16 23:38:48 +02:00
if ( library . isBook ) {
2023-08-05 22:28:16 +02:00
start = Date . now ( )
2023-08-16 23:38:48 +02:00
// "Recent Series" shelf
const seriesMostRecentPayload = await libraryFilters . getSeriesMostRecentlyAdded ( library , user , include , 5 )
if ( seriesMostRecentPayload . series . length ) {
2023-08-03 01:29:28 +02:00
shelves . push ( {
2023-08-16 23:38:48 +02:00
id : 'recent-series' ,
label : 'Recent Series' ,
labelStringKey : 'LabelRecentSeries' ,
type : 'series' ,
entities : seriesMostRecentPayload . series ,
total : seriesMostRecentPayload . count
2023-08-03 01:29:28 +02:00
} )
}
2024-01-04 00:19:28 +01:00
Logger . debug ( ` Loaded ${ seriesMostRecentPayload . series . length } of ${ seriesMostRecentPayload . count } series for "Recent Series" in ${ ( ( Date . now ( ) - start ) / 1000 ) . toFixed ( 2 ) } s ` )
2023-08-05 21:01:16 +02:00
start = Date . now ( )
2023-08-16 23:38:48 +02:00
// "Discover" shelf
const discoverLibraryItemsPayload = await libraryFilters . getLibraryItemsToDiscover ( library , user , include , limit )
if ( discoverLibraryItemsPayload . libraryItems . length ) {
shelves . push ( {
id : 'discover' ,
label : 'Discover' ,
labelStringKey : 'LabelDiscover' ,
type : library . mediaType ,
entities : discoverLibraryItemsPayload . libraryItems ,
total : discoverLibraryItemsPayload . count
} )
}
2024-01-04 00:19:28 +01:00
Logger . debug ( ` Loaded ${ discoverLibraryItemsPayload . libraryItems . length } of ${ discoverLibraryItemsPayload . count } items for "Discover" in ${ ( ( Date . now ( ) - start ) / 1000 ) . toFixed ( 2 ) } s ` )
2023-08-16 23:38:48 +02:00
}
2023-08-08 00:59:04 +02:00
2023-08-16 23:38:48 +02:00
start = Date . now ( )
// "Listen Again" shelf
const mediaFinishedPayload = await libraryFilters . getMediaFinished ( library , user , include , limit )
if ( mediaFinishedPayload . items . length ) {
2024-05-29 00:24:02 +02:00
const ebookOnlyItemsInProgress = mediaFinishedPayload . items . filter ( ( li ) => li . media . isEBookOnly )
const audioOnlyItemsInProgress = mediaFinishedPayload . items . filter ( ( li ) => ! li . media . isEBookOnly )
2023-08-16 23:38:48 +02:00
shelves . push ( {
id : 'listen-again' ,
label : 'Listen Again' ,
labelStringKey : 'LabelListenAgain' ,
type : library . isPodcast ? 'episode' : 'book' ,
entities : audioOnlyItemsInProgress ,
total : mediaFinishedPayload . count
} )
// "Read Again" shelf
if ( ebookOnlyItemsInProgress . length ) {
2023-08-05 21:01:16 +02:00
shelves . push ( {
2023-08-16 23:38:48 +02:00
id : 'read-again' ,
label : 'Read Again' ,
labelStringKey : 'LabelReadAgain' ,
type : 'book' ,
entities : ebookOnlyItemsInProgress ,
2023-08-08 00:59:04 +02:00
total : mediaFinishedPayload . count
2023-08-05 21:01:16 +02:00
} )
2023-08-08 00:59:04 +02:00
}
2023-08-16 23:38:48 +02:00
}
2024-01-04 00:19:28 +01:00
Logger . debug ( ` Loaded ${ mediaFinishedPayload . items . length } of ${ mediaFinishedPayload . count } items for "Listen/Read Again" in ${ ( ( Date . now ( ) - start ) / 1000 ) . toFixed ( 2 ) } s ` )
2023-08-16 23:38:48 +02:00
if ( library . isBook ) {
start = Date . now ( )
// "Newest Authors" shelf
const newestAuthorsPayload = await libraryFilters . getNewestAuthors ( library , user , limit )
if ( newestAuthorsPayload . authors . length ) {
shelves . push ( {
id : 'newest-authors' ,
label : 'Newest Authors' ,
labelStringKey : 'LabelNewestAuthors' ,
type : 'authors' ,
entities : newestAuthorsPayload . authors ,
total : newestAuthorsPayload . count
} )
2023-08-05 21:01:16 +02:00
}
2024-01-04 00:19:28 +01:00
Logger . debug ( ` Loaded ${ newestAuthorsPayload . authors . length } of ${ newestAuthorsPayload . count } authors for "Newest Authors" in ${ ( ( Date . now ( ) - start ) / 1000 ) . toFixed ( 2 ) } s ` )
2023-08-16 23:38:48 +02:00
}
2023-08-05 21:01:16 +02:00
2023-08-16 23:38:48 +02:00
Logger . debug ( ` Loaded ${ shelves . length } personalized shelves in ${ ( ( Date . now ( ) - fullStart ) / 1000 ) . toFixed ( 2 ) } s ` )
2023-08-05 01:07:55 +02:00
2023-08-16 23:38:48 +02:00
return shelves
}
2023-08-03 01:29:28 +02:00
2023-08-16 23:38:48 +02:00
/ * *
* Get book library items for author , optional use user permissions
2024-08-31 20:27:48 +02:00
* @ param { import ( './Author' ) } author
2024-08-11 00:15:21 +02:00
* @ param { import ( './User' ) } user
2023-08-16 23:38:48 +02:00
* @ returns { Promise < oldLibraryItem [ ] > }
* /
static async getForAuthor ( author , user = null ) {
const { libraryItems } = await libraryFilters . getLibraryItemsForAuthor ( author , user , undefined , undefined )
2024-05-29 00:24:02 +02:00
return libraryItems . map ( ( li ) => this . getOldLibraryItem ( li ) )
2023-08-16 23:38:48 +02:00
}
2023-08-06 22:06:45 +02:00
2023-08-16 23:38:48 +02:00
/ * *
* Get book library items in a collection
2024-05-29 00:24:02 +02:00
* @ param { oldCollection } collection
2023-08-16 23:38:48 +02:00
* @ returns { Promise < oldLibraryItem [ ] > }
* /
static async getForCollection ( collection ) {
const libraryItems = await libraryFilters . getLibraryItemsForCollection ( collection )
2024-05-29 00:24:02 +02:00
return libraryItems . map ( ( li ) => this . getOldLibraryItem ( li ) )
2023-08-16 23:38:48 +02:00
}
2023-08-12 00:49:06 +02:00
2023-08-16 23:38:48 +02:00
/ * *
* Check if library item exists
2024-05-29 00:24:02 +02:00
* @ param { string } libraryItemId
2023-08-16 23:38:48 +02:00
* @ returns { Promise < boolean > }
* /
static async checkExistsById ( libraryItemId ) {
return ( await this . count ( { where : { id : libraryItemId } } ) ) > 0
}
2023-08-12 22:52:09 +02:00
2023-08-17 01:08:00 +02:00
/ * *
2024-05-29 00:24:02 +02:00
*
* @ param { import ( 'sequelize' ) . WhereOptions } where
2024-03-23 20:56:32 +01:00
* @ param { import ( 'sequelize' ) . BindOrReplacements } replacements
2023-08-17 01:08:00 +02:00
* @ returns { Object } oldLibraryItem
* /
2024-03-23 20:56:32 +01:00
static async findOneOld ( where , replacements = { } ) {
2023-08-17 01:08:00 +02:00
const libraryItem = await this . findOne ( {
where ,
2024-03-23 20:56:32 +01:00
replacements ,
2023-08-17 01:08:00 +02:00
include : [
{
model : this . sequelize . models . book ,
include : [
{
model : this . sequelize . models . author ,
through : {
attributes : [ ]
}
} ,
{
model : this . sequelize . models . series ,
through : {
attributes : [ 'sequence' ]
}
}
]
} ,
{
model : this . sequelize . models . podcast ,
include : [
{
model : this . sequelize . models . podcastEpisode
}
]
}
] ,
order : [
[ this . sequelize . models . book , this . sequelize . models . author , this . sequelize . models . bookAuthor , 'createdAt' , 'ASC' ] ,
[ this . sequelize . models . book , this . sequelize . models . series , 'bookSeries' , 'createdAt' , 'ASC' ]
]
} )
if ( ! libraryItem ) return null
return this . getOldLibraryItem ( libraryItem )
}
2024-11-02 18:56:40 +01:00
/ * *
*
* @ param { string } libraryItemId
* @ returns { Promise < string > }
* /
static async getCoverPath ( libraryItemId ) {
const libraryItem = await this . findByPk ( libraryItemId , {
attributes : [ 'id' , 'mediaType' , 'mediaId' , 'libraryId' ] ,
include : [
{
2024-11-02 19:02:40 +01:00
model : this . sequelize . models . book ,
2024-11-02 18:56:40 +01:00
attributes : [ 'id' , 'coverPath' ]
} ,
{
2024-11-02 19:02:40 +01:00
model : this . sequelize . models . podcast ,
2024-11-02 18:56:40 +01:00
attributes : [ 'id' , 'coverPath' ]
}
]
} )
if ( ! libraryItem ) {
Logger . warn ( ` [LibraryItem] getCoverPath: Library item " ${ libraryItemId } " does not exist ` )
return null
}
return libraryItem . media . coverPath
}
2024-04-13 00:34:10 +02:00
/ * *
2024-05-29 00:24:02 +02:00
*
2024-04-13 00:34:10 +02:00
* @ returns { Promise }
* /
async saveMetadataFile ( ) {
let metadataPath = Path . join ( global . MetadataPath , 'items' , this . id )
let storeMetadataWithItem = global . ServerSettings . storeMetadataWithItem
if ( storeMetadataWithItem && ! this . isFile ) {
metadataPath = this . path
} else {
// Make sure metadata book dir exists
storeMetadataWithItem = false
await fsExtra . ensureDir ( metadataPath )
}
const metadataFilePath = Path . join ( metadataPath , ` metadata. ${ global . ServerSettings . metadataFileFormat } ` )
// Expanded with series, authors, podcastEpisodes
2024-05-29 00:24:02 +02:00
const mediaExpanded = this . media || ( await this . getMediaExpanded ( ) )
2024-04-13 00:34:10 +02:00
let jsonObject = { }
if ( this . mediaType === 'book' ) {
jsonObject = {
tags : mediaExpanded . tags || [ ] ,
2024-05-29 00:24:02 +02:00
chapters : mediaExpanded . chapters ? . map ( ( c ) => ( { ... c } ) ) || [ ] ,
2024-04-13 00:34:10 +02:00
title : mediaExpanded . title ,
subtitle : mediaExpanded . subtitle ,
2024-05-29 00:24:02 +02:00
authors : mediaExpanded . authors . map ( ( a ) => a . name ) ,
2024-04-13 00:34:10 +02:00
narrators : mediaExpanded . narrators ,
2024-05-29 00:24:02 +02:00
series : mediaExpanded . series . map ( ( se ) => {
2024-04-13 00:34:10 +02:00
const sequence = se . bookSeries ? . sequence || ''
if ( ! sequence ) return se . name
return ` ${ se . name } # ${ sequence } `
} ) ,
genres : mediaExpanded . genres || [ ] ,
publishedYear : mediaExpanded . publishedYear ,
publishedDate : mediaExpanded . publishedDate ,
publisher : mediaExpanded . publisher ,
description : mediaExpanded . description ,
isbn : mediaExpanded . isbn ,
asin : mediaExpanded . asin ,
language : mediaExpanded . language ,
explicit : ! ! mediaExpanded . explicit ,
abridged : ! ! mediaExpanded . abridged
}
} else {
jsonObject = {
tags : mediaExpanded . tags || [ ] ,
title : mediaExpanded . title ,
author : mediaExpanded . author ,
description : mediaExpanded . description ,
releaseDate : mediaExpanded . releaseDate ,
genres : mediaExpanded . genres || [ ] ,
feedURL : mediaExpanded . feedURL ,
imageURL : mediaExpanded . imageURL ,
itunesPageURL : mediaExpanded . itunesPageURL ,
itunesId : mediaExpanded . itunesId ,
itunesArtistId : mediaExpanded . itunesArtistId ,
asin : mediaExpanded . asin ,
language : mediaExpanded . language ,
explicit : ! ! mediaExpanded . explicit ,
podcastType : mediaExpanded . podcastType
}
}
2024-05-29 00:24:02 +02:00
return fsExtra
. writeFile ( metadataFilePath , JSON . stringify ( jsonObject , null , 2 ) )
. then ( async ( ) => {
// Add metadata.json to libraryFiles array if it is new
let metadataLibraryFile = this . libraryFiles . find ( ( lf ) => lf . metadata . path === filePathToPOSIX ( metadataFilePath ) )
if ( storeMetadataWithItem ) {
if ( ! metadataLibraryFile ) {
const newLibraryFile = new LibraryFile ( )
await newLibraryFile . setDataFromPath ( metadataFilePath , ` metadata.json ` )
metadataLibraryFile = newLibraryFile . toJSON ( )
this . libraryFiles . push ( metadataLibraryFile )
} else {
const fileTimestamps = await getFileTimestampsWithIno ( metadataFilePath )
if ( fileTimestamps ) {
metadataLibraryFile . metadata . mtimeMs = fileTimestamps . mtimeMs
metadataLibraryFile . metadata . ctimeMs = fileTimestamps . ctimeMs
metadataLibraryFile . metadata . size = fileTimestamps . size
metadataLibraryFile . ino = fileTimestamps . ino
}
}
const libraryItemDirTimestamps = await getFileTimestampsWithIno ( this . path )
if ( libraryItemDirTimestamps ) {
this . mtime = libraryItemDirTimestamps . mtimeMs
this . ctime = libraryItemDirTimestamps . ctimeMs
let size = 0
this . libraryFiles . forEach ( ( lf ) => ( size += ! isNaN ( lf . metadata . size ) ? Number ( lf . metadata . size ) : 0 ) )
this . size = size
await this . save ( )
2024-04-13 00:34:10 +02:00
}
}
2024-05-29 00:24:02 +02:00
Logger . debug ( ` Success saving abmetadata to " ${ metadataFilePath } " ` )
2024-04-13 00:34:10 +02:00
2024-05-29 00:24:02 +02:00
return metadataLibraryFile
} )
. catch ( ( error ) => {
Logger . error ( ` Failed to save json file at " ${ metadataFilePath } " ` , error )
return null
} )
2024-04-13 00:34:10 +02:00
}
2023-08-16 23:38:48 +02:00
/ * *
* Initialize model
2024-05-29 00:24:02 +02:00
* @ param { import ( '../Database' ) . sequelize } sequelize
2023-08-16 23:38:48 +02:00
* /
static init ( sequelize ) {
2024-05-29 00:24:02 +02:00
super . init (
{
id : {
type : DataTypes . UUID ,
defaultValue : DataTypes . UUIDV4 ,
primaryKey : true
2023-08-16 23:38:48 +02:00
} ,
2024-05-29 00:24:02 +02:00
ino : DataTypes . STRING ,
path : DataTypes . STRING ,
relPath : DataTypes . STRING ,
2024-11-09 21:10:46 +01:00
mediaId : DataTypes . UUID ,
2024-05-29 00:24:02 +02:00
mediaType : DataTypes . STRING ,
isFile : DataTypes . BOOLEAN ,
isMissing : DataTypes . BOOLEAN ,
isInvalid : DataTypes . BOOLEAN ,
mtime : DataTypes . DATE ( 6 ) ,
ctime : DataTypes . DATE ( 6 ) ,
birthtime : DataTypes . DATE ( 6 ) ,
size : DataTypes . BIGINT ,
lastScan : DataTypes . DATE ,
lastScanVersion : DataTypes . STRING ,
libraryFiles : DataTypes . JSON ,
extraData : DataTypes . JSON
} ,
{
sequelize ,
modelName : 'libraryItem' ,
indexes : [
{
fields : [ 'createdAt' ]
} ,
{
fields : [ 'mediaId' ]
} ,
{
fields : [ 'libraryId' , 'mediaType' ]
} ,
2025-01-01 06:34:29 +01:00
{
fields : [ 'libraryId' , 'mediaType' , 'size' ]
} ,
2024-05-29 00:24:02 +02:00
{
fields : [ 'libraryId' , 'mediaId' , 'mediaType' ]
} ,
{
fields : [ 'birthtime' ]
} ,
{
fields : [ 'mtime' ]
}
]
}
)
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
const { library , libraryFolder , book , podcast } = sequelize . models
library . hasMany ( LibraryItem )
LibraryItem . belongsTo ( library )
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
libraryFolder . hasMany ( LibraryItem )
LibraryItem . belongsTo ( libraryFolder )
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
book . hasOne ( LibraryItem , {
foreignKey : 'mediaId' ,
constraints : false ,
scope : {
mediaType : 'book'
2023-07-05 01:14:44 +02:00
}
2023-08-16 23:38:48 +02:00
} )
LibraryItem . belongsTo ( book , { foreignKey : 'mediaId' , constraints : false } )
podcast . hasOne ( LibraryItem , {
foreignKey : 'mediaId' ,
constraints : false ,
scope : {
mediaType : 'podcast'
}
} )
LibraryItem . belongsTo ( podcast , { foreignKey : 'mediaId' , constraints : false } )
2024-05-29 00:24:02 +02:00
LibraryItem . addHook ( 'afterFind' , ( findResult ) => {
2023-08-16 23:38:48 +02:00
if ( ! findResult ) return
if ( ! Array . isArray ( findResult ) ) findResult = [ findResult ]
for ( const instance of findResult ) {
if ( instance . mediaType === 'book' && instance . book !== undefined ) {
instance . media = instance . book
instance . dataValues . media = instance . dataValues . book
} else if ( instance . mediaType === 'podcast' && instance . podcast !== undefined ) {
instance . media = instance . podcast
instance . dataValues . media = instance . dataValues . podcast
}
// To prevent mistakes:
delete instance . book
delete instance . dataValues . book
delete instance . podcast
delete instance . dataValues . podcast
}
} )
2023-07-05 01:14:44 +02:00
2024-05-29 00:24:02 +02:00
LibraryItem . addHook ( 'afterDestroy' , async ( instance ) => {
2023-08-16 23:38:48 +02:00
if ( ! instance ) return
const media = await instance . getMedia ( )
if ( media ) {
media . destroy ( )
}
} )
}
2024-12-14 23:55:56 +01:00
2025-01-02 19:49:58 +01:00
get isBook ( ) {
return this . mediaType === 'book'
}
get isPodcast ( ) {
return this . mediaType === 'podcast'
}
get hasAudioTracks ( ) {
return this . media . hasAudioTracks ( )
}
/ * *
*
* @ param { import ( 'sequelize' ) . FindOptions } options
* @ returns { Promise < Book | Podcast > }
* /
getMedia ( options ) {
if ( ! this . mediaType ) return Promise . resolve ( null )
const mixinMethodName = ` get ${ this . sequelize . uppercaseFirst ( this . mediaType ) } `
return this [ mixinMethodName ] ( options )
}
/ * *
*
* @ returns { Promise < Book | Podcast > }
* /
getMediaExpanded ( ) {
if ( this . mediaType === 'podcast' ) {
return this . getMedia ( {
include : [
{
model : this . sequelize . models . podcastEpisode
}
]
} )
} else {
return this . getMedia ( {
include : [
{
model : this . sequelize . models . author ,
through : {
attributes : [ ]
}
} ,
{
model : this . sequelize . models . series ,
through : {
attributes : [ 'sequence' ]
}
}
] ,
order : [
[ this . sequelize . models . author , this . sequelize . models . bookAuthor , 'createdAt' , 'ASC' ] ,
[ this . sequelize . models . series , 'bookSeries' , 'createdAt' , 'ASC' ]
]
} )
}
}
2024-12-14 23:55:56 +01:00
/ * *
* Check if book or podcast library item has audio tracks
* Requires expanded library item
*
* @ returns { boolean }
* /
hasAudioTracks ( ) {
if ( ! this . media ) {
Logger . error ( ` [LibraryItem] hasAudioTracks: Library item " ${ this . id } " does not have media ` )
return false
}
2025-01-02 22:42:52 +01:00
if ( this . isBook ) {
2024-12-14 23:55:56 +01:00
return this . media . audioFiles ? . length > 0
} else {
return this . media . podcastEpisodes ? . length > 0
}
}
2025-01-02 19:49:58 +01:00
2025-01-02 22:42:52 +01:00
/ * *
*
* @ param { string } ino
* @ returns { import ( './Book' ) . AudioFileObject }
* /
getAudioFileWithIno ( ino ) {
if ( ! this . media ) {
Logger . error ( ` [LibraryItem] getAudioFileWithIno: Library item " ${ this . id } " does not have media ` )
return null
}
if ( this . isBook ) {
return this . media . audioFiles . find ( ( af ) => af . ino === ino )
} else {
return this . media . podcastEpisodes . find ( ( pe ) => pe . audioFile ? . ino === ino ) ? . audioFile
}
}
2025-01-03 18:16:03 +01:00
/ * *
* Get the track list to be used in client audio players
* AudioTrack is the AudioFile with startOffset and contentUrl
* Podcasts must have an episodeId to get the track list
*
* @ param { string } [ episodeId ]
* @ returns { import ( './Book' ) . AudioTrack [ ] }
* /
getTrackList ( episodeId ) {
if ( ! this . media ) {
Logger . error ( ` [LibraryItem] getTrackList: Library item " ${ this . id } " does not have media ` )
return [ ]
}
return this . media . getTracklist ( this . id , episodeId )
}
2025-01-02 22:42:52 +01:00
/ * *
*
* @ param { string } ino
* @ returns { LibraryFile }
* /
getLibraryFileWithIno ( ino ) {
const libraryFile = this . libraryFiles . find ( ( lf ) => lf . ino === ino )
if ( ! libraryFile ) return null
return new LibraryFile ( libraryFile )
}
getLibraryFiles ( ) {
return this . libraryFiles . map ( ( lf ) => new LibraryFile ( lf ) )
}
getLibraryFilesJson ( ) {
return this . libraryFiles . map ( ( lf ) => new LibraryFile ( lf ) . toJSON ( ) )
}
2025-01-02 19:49:58 +01:00
toOldJSON ( ) {
if ( ! this . media ) {
throw new Error ( ` [LibraryItem] Cannot convert to old JSON without media for library item " ${ this . id } " ` )
}
return {
id : this . id ,
ino : this . ino ,
oldLibraryItemId : this . extraData ? . oldLibraryItemId || null ,
libraryId : this . libraryId ,
folderId : this . libraryFolderId ,
path : this . path ,
relPath : this . relPath ,
isFile : this . isFile ,
mtimeMs : this . mtime ? . valueOf ( ) ,
ctimeMs : this . ctime ? . valueOf ( ) ,
birthtimeMs : this . birthtime ? . valueOf ( ) ,
addedAt : this . createdAt . valueOf ( ) ,
updatedAt : this . updatedAt . valueOf ( ) ,
lastScan : this . lastScan ? . valueOf ( ) ,
scanVersion : this . lastScanVersion ,
isMissing : ! ! this . isMissing ,
isInvalid : ! ! this . isInvalid ,
mediaType : this . mediaType ,
media : this . media . toOldJSON ( this . id ) ,
2025-01-02 22:42:52 +01:00
// LibraryFile JSON includes a fileType property that may not be saved in libraryFiles column in the database
libraryFiles : this . getLibraryFilesJson ( )
2025-01-02 19:49:58 +01:00
}
}
toOldJSONMinified ( ) {
if ( ! this . media ) {
throw new Error ( ` [LibraryItem] Cannot convert to old JSON without media for library item " ${ this . id } " ` )
}
return {
id : this . id ,
ino : this . ino ,
oldLibraryItemId : this . extraData ? . oldLibraryItemId || null ,
libraryId : this . libraryId ,
folderId : this . libraryFolderId ,
path : this . path ,
relPath : this . relPath ,
isFile : this . isFile ,
mtimeMs : this . mtime ? . valueOf ( ) ,
ctimeMs : this . ctime ? . valueOf ( ) ,
birthtimeMs : this . birthtime ? . valueOf ( ) ,
addedAt : this . createdAt . valueOf ( ) ,
updatedAt : this . updatedAt . valueOf ( ) ,
isMissing : ! ! this . isMissing ,
isInvalid : ! ! this . isInvalid ,
mediaType : this . mediaType ,
media : this . media . toOldJSONMinified ( ) ,
numFiles : this . libraryFiles . length ,
size : this . size
}
}
toOldJSONExpanded ( ) {
return {
id : this . id ,
ino : this . ino ,
oldLibraryItemId : this . extraData ? . oldLibraryItemId || null ,
libraryId : this . libraryId ,
folderId : this . libraryFolderId ,
path : this . path ,
relPath : this . relPath ,
isFile : this . isFile ,
mtimeMs : this . mtime ? . valueOf ( ) ,
ctimeMs : this . ctime ? . valueOf ( ) ,
birthtimeMs : this . birthtime ? . valueOf ( ) ,
addedAt : this . createdAt . valueOf ( ) ,
updatedAt : this . updatedAt . valueOf ( ) ,
lastScan : this . lastScan ? . valueOf ( ) ,
scanVersion : this . lastScanVersion ,
isMissing : ! ! this . isMissing ,
isInvalid : ! ! this . isInvalid ,
mediaType : this . mediaType ,
media : this . media . toOldJSONExpanded ( this . id ) ,
2025-01-02 22:42:52 +01:00
// LibraryFile JSON includes a fileType property that may not be saved in libraryFiles column in the database
libraryFiles : this . getLibraryFilesJson ( ) ,
2025-01-02 19:49:58 +01:00
size : this . size
}
}
2023-08-16 23:38:48 +02:00
}
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
module . exports = LibraryItem