2023-08-13 00:29:08 +02:00
const { DataTypes , Model , WhereOptions } = require ( 'sequelize' )
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' )
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
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
}
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
*
* @ 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
/ * *
* Currently unused because this is too slow and uses too much mem
* @ param { [ WhereOptions ] } where
* @ 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
}
]
}
]
} )
return libraryItems . map ( ti => this . getOldLibraryItem ( ti ) )
}
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
*
* @ param { Model < LibraryItem > } libraryItemExpanded
* @ 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
2023-08-16 23:38:48 +02:00
static async fullUpdateFromOld ( oldLibraryItem ) {
const libraryItemExpanded = await this . findByPk ( oldLibraryItem . id , {
include : [
{
model : this . sequelize . models . book ,
include : [
{
model : this . sequelize . models . author ,
through : {
attributes : [ ]
2023-07-05 01:14:44 +02:00
}
2023-08-16 23:38:48 +02:00
} ,
{
model : this . sequelize . models . series ,
through : {
attributes : [ 'id' , 'sequence' ]
2023-07-05 01:14:44 +02:00
}
}
2023-08-16 23:38:48 +02:00
]
} ,
{
model : this . sequelize . models . podcast ,
include : [
{
model : this . sequelize . models . podcastEpisode
2023-07-05 01:14:44 +02:00
}
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
if ( ! updatedPodcastEpisodes . some ( ep => ep . id === existingPodcastEpisode . id ) ) {
Logger . dev ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " episode " ${ existingPodcastEpisode . title } " was removed ` )
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 ) {
const existingEpisodeMatch = existingPodcastEpisodes . find ( ep => ep . id === updatedPodcastEpisode . id )
if ( ! existingEpisodeMatch ) {
Logger . dev ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " episode " ${ updatedPodcastEpisode . title } " was added ` )
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 ) ) {
Logger . dev ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " episode " ${ existingEpisodeMatch . title } " ${ key } was updated from " ${ existingValue } " to " ${ updatedEpisodeCleaned [ key ] } " ` )
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 || [ ]
const updatedSeriesAll = oldLibraryItem . media . metadata . series || [ ]
for ( const existingAuthor of existingAuthors ) {
// Author was removed from Book
if ( ! updatedAuthors . some ( au => au . id === existingAuthor . id ) ) {
Logger . dev ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " author " ${ existingAuthor . name } " was removed ` )
await this . sequelize . models . bookAuthor . removeByIds ( existingAuthor . id , libraryItemExpanded . media . id )
hasUpdates = true
2023-07-05 01:14:44 +02:00
}
}
2023-08-16 23:38:48 +02:00
for ( const updatedAuthor of updatedAuthors ) {
// Author was added
if ( ! existingAuthors . some ( au => au . id === updatedAuthor . id ) ) {
Logger . dev ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " author " ${ updatedAuthor . name } " was added ` )
await this . sequelize . models . bookAuthor . create ( { authorId : updatedAuthor . id , bookId : libraryItemExpanded . media . id } )
hasUpdates = true
}
}
for ( const existingSeries of existingSeriesAll ) {
// Series was removed
if ( ! updatedSeriesAll . some ( se => se . id === existingSeries . id ) ) {
Logger . dev ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " series " ${ existingSeries . name } " was removed ` )
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
const existingSeriesMatch = existingSeriesAll . find ( se => se . id === updatedSeries . id )
if ( ! existingSeriesMatch ) {
Logger . dev ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " series " ${ updatedSeries . name } " was added ` )
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 ) {
Logger . dev ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " series " ${ updatedSeries . name } " sequence was updated from " ${ existingSeriesMatch . bookSeries . sequence } " to " ${ updatedSeries . sequence } " ` )
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 ) ) {
Logger . dev ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " ${ libraryItemExpanded . mediaType } . ${ key } updated from ${ existingValue } to ${ updatedMedia [ key ] } ` )
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 ) ) {
Logger . dev ( ` [LibraryItem] " ${ libraryItemExpanded . media . title } " ${ key } updated from ${ existingValue } to ${ updatedLibraryItem [ key ] } ` )
hasLibraryItemUpdates = 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 ,
birthtime : oldLibraryItem . birthtimeMs ,
size : oldLibraryItem . size ,
lastScan : oldLibraryItem . lastScan ,
lastScanVersion : oldLibraryItem . scanVersion ,
libraryId : oldLibraryItem . libraryId ,
libraryFolderId : oldLibraryItem . folderId ,
libraryFiles : oldLibraryItem . libraryFiles ? . map ( lf => lf . toJSON ( ) ) || [ ] ,
extraData
}
}
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
static removeById ( libraryItemId ) {
return this . destroy ( {
where : {
id : libraryItemId
} ,
individualHooks : true
} )
}
/ * *
* Get old library item by id
* @ param { string } libraryItemId
* @ 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
* @ param { oldLibrary } library
* @ param { oldUser } user
* @ param { object } options
* @ returns { object } { libraryItems : oldLibraryItem [ ] , count : number }
* /
static async getByFilterAndSort ( library , user , options ) {
let start = Date . now ( )
const { libraryItems , count } = await libraryFilters . getFilteredLibraryItems ( library , user , options )
Logger . debug ( ` Loaded ${ libraryItems . length } of ${ count } items for libary page in ${ ( ( Date . now ( ) - start ) / 1000 ) . toFixed ( 2 ) } s ` )
return {
libraryItems : libraryItems . map ( li => {
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 ) {
oldLibraryItem . rssFeed = this . sequelize . models . feed . getOldFeed ( li . rssFeed ) . toJSONMinified ( )
}
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
}
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
* @ param { oldLibrary } library
* @ param { oldUser } user
* @ param { string [ ] } include
* @ param { number } limit
* @ 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 ) {
const ebookOnlyItemsInProgress = itemsInProgressPayload . items . filter ( li => li . media . isEBookOnly )
const audioOnlyItemsInProgress = itemsInProgressPayload . items . filter ( li => ! li . media . isEBookOnly )
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
}
}
2023-09-17 23:09:21 +02:00
Logger . dev ( ` 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
}
2023-09-17 23:09:21 +02:00
Logger . dev ( ` 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
}
2023-09-17 23:09:21 +02:00
Logger . dev ( ` 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
} )
}
2023-09-17 23:09:21 +02:00
Logger . dev ( ` 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
} )
}
2023-09-17 23:09:21 +02:00
Logger . dev ( ` 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
} )
}
2023-09-17 23:09:21 +02:00
Logger . dev ( ` 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 ) {
const ebookOnlyItemsInProgress = mediaFinishedPayload . items . filter ( li => li . media . isEBookOnly )
const audioOnlyItemsInProgress = mediaFinishedPayload . items . filter ( li => ! li . media . isEBookOnly )
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
}
2023-09-17 23:09:21 +02:00
Logger . dev ( ` 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
}
2023-09-17 23:09:21 +02:00
Logger . dev ( ` 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
* @ param { oldAuthor } author
* @ param { [ oldUser ] } user
* @ returns { Promise < oldLibraryItem [ ] > }
* /
static async getForAuthor ( author , user = null ) {
const { libraryItems } = await libraryFilters . getLibraryItemsForAuthor ( author , user , undefined , undefined )
return libraryItems . map ( li => this . getOldLibraryItem ( li ) )
}
2023-08-06 22:06:45 +02:00
2023-08-16 23:38:48 +02:00
/ * *
* Get book library items in a collection
* @ param { oldCollection } collection
* @ returns { Promise < oldLibraryItem [ ] > }
* /
static async getForCollection ( collection ) {
const libraryItems = await libraryFilters . getLibraryItemsForCollection ( collection )
return libraryItems . map ( li => this . getOldLibraryItem ( li ) )
}
2023-08-12 00:49:06 +02:00
2023-08-16 23:38:48 +02:00
/ * *
* Check if library item exists
* @ param { string } libraryItemId
* @ 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
/ * *
*
* @ param { WhereOptions } where
* @ returns { Object } oldLibraryItem
* /
static async findOneOld ( where ) {
const libraryItem = await this . findOne ( {
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
}
]
}
] ,
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 )
}
2023-08-28 00:19:57 +02:00
/ * *
*
* @ param { import ( 'sequelize' ) . FindOptions } options
* @ returns { Promise < Book | Podcast > }
* /
2023-08-16 23:38:48 +02:00
getMedia ( options ) {
if ( ! this . mediaType ) return Promise . resolve ( null )
const mixinMethodName = ` get ${ this . sequelize . uppercaseFirst ( this . mediaType ) } `
return this [ mixinMethodName ] ( options )
2023-07-05 01:14:44 +02:00
}
2023-08-16 23:38:48 +02:00
/ * *
* Initialize model
* @ param { import ( '../Database' ) . sequelize } sequelize
* /
static init ( sequelize ) {
super . init ( {
id : {
type : DataTypes . UUID ,
defaultValue : DataTypes . UUIDV4 ,
primaryKey : true
2023-08-11 00:46:27 +02:00
} ,
2023-08-16 23:38:48 +02:00
ino : DataTypes . STRING ,
path : DataTypes . STRING ,
relPath : DataTypes . STRING ,
mediaId : DataTypes . UUIDV4 ,
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' ]
} ,
2023-09-30 19:39:16 +02:00
{
fields : [ 'libraryId' , 'mediaId' , 'mediaType' ]
} ,
2023-08-16 23:38:48 +02:00
{
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 } )
LibraryItem . addHook ( 'afterFind' , findResult => {
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
2023-08-16 23:38:48 +02:00
LibraryItem . addHook ( 'afterDestroy' , async instance => {
if ( ! instance ) return
const media = await instance . getMedia ( )
if ( media ) {
media . destroy ( )
}
} )
}
}
2023-07-05 01:14:44 +02:00
2023-08-16 23:38:48 +02:00
module . exports = LibraryItem