mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-24 23:38:56 -04:00 
			
		
		
		
	Merge branch 'advplyr:master' into master
This commit is contained in:
		
						commit
						adafefecd4
					
				| @ -171,7 +171,7 @@ export default { | ||||
|     }, | ||||
|     async fetchCategories() { | ||||
|       const categories = await this.$axios | ||||
|         .$get(`/api/libraries/${this.currentLibraryId}/personalized2?include=rssfeed,numEpisodesIncomplete`) | ||||
|         .$get(`/api/libraries/${this.currentLibraryId}/personalized?include=rssfeed,numEpisodesIncomplete`) | ||||
|         .then((data) => { | ||||
|           return data | ||||
|         }) | ||||
|  | ||||
| @ -628,6 +628,11 @@ export default { | ||||
|       return entitiesPerShelfBefore < this.entitiesPerShelf // Books per shelf has changed | ||||
|     }, | ||||
|     async init(bookshelf) { | ||||
|       if (this.entityName === 'series') { | ||||
|         this.booksPerFetch = 50 | ||||
|       } else { | ||||
|         this.booksPerFetch = 100 | ||||
|       } | ||||
|       this.checkUpdateSearchParams() | ||||
|       this.initSizeData(bookshelf) | ||||
| 
 | ||||
|  | ||||
| @ -36,7 +36,7 @@ export default { | ||||
|       return this.narrator?.name || '' | ||||
|     }, | ||||
|     numBooks() { | ||||
|       return this.narrator?.books?.length || 0 | ||||
|       return this.narrator?.numBooks || this.narrator?.books?.length || 0 | ||||
|     }, | ||||
|     userCanUpdate() { | ||||
|       return this.$store.getters['user/getUserCanUpdate'] | ||||
|  | ||||
| @ -103,7 +103,7 @@ export default { | ||||
|       return this.$store.state.libraries.currentLibraryId | ||||
|     }, | ||||
|     totalResults() { | ||||
|       return this.bookResults.length + this.seriesResults.length + this.authorResults.length + this.tagResults.length + this.podcastResults.length | ||||
|       return this.bookResults.length + this.seriesResults.length + this.authorResults.length + this.tagResults.length + this.podcastResults.length + this.narratorResults.length | ||||
|     } | ||||
|   }, | ||||
|   methods: { | ||||
|  | ||||
| @ -13,8 +13,8 @@ | ||||
| 
 | ||||
|     <div v-if="imageFailed" class="absolute top-0 left-0 right-0 bottom-0 w-full h-full bg-red-100" :style="{ padding: placeholderCoverPadding + 'rem' }"> | ||||
|       <div class="w-full h-full border-2 border-error flex flex-col items-center justify-center"> | ||||
|         <img src="/Logo.png" class="mb-2" :style="{ height: 64 * sizeMultiplier + 'px' }" /> | ||||
|         <p class="text-center text-error" :style="{ fontSize: sizeMultiplier + 'rem' }">Invalid Cover</p> | ||||
|         <img v-if="width > 100" src="/Logo.png" class="mb-2" :style="{ height: 40 * sizeMultiplier + 'px' }" /> | ||||
|         <p class="text-center text-error" :style="{ fontSize: invalidCoverFontSize + 'rem' }">Invalid Cover</p> | ||||
|       </div> | ||||
|     </div> | ||||
| 
 | ||||
| @ -58,6 +58,9 @@ export default { | ||||
|     sizeMultiplier() { | ||||
|       return this.width / 120 | ||||
|     }, | ||||
|     invalidCoverFontSize() { | ||||
|       return Math.max(this.sizeMultiplier * 0.8, 0.5) | ||||
|     }, | ||||
|     placeholderCoverPadding() { | ||||
|       return 0.8 * this.sizeMultiplier | ||||
|     }, | ||||
|  | ||||
| @ -18,7 +18,7 @@ | ||||
|       </div> | ||||
|     </div> | ||||
| 
 | ||||
|     <div class="flex px-4"> | ||||
|     <div v-if="isBookLibrary" class="flex px-4"> | ||||
|       <svg class="h-14 w-14 md:h-18 md:w-18" viewBox="0 0 24 24"> | ||||
|         <path fill="currentColor" d="M12,4A4,4 0 0,1 16,8A4,4 0 0,1 12,12A4,4 0 0,1 8,8A4,4 0 0,1 12,4M12,6A2,2 0 0,0 10,8A2,2 0 0,0 12,10A2,2 0 0,0 14,8A2,2 0 0,0 12,6M12,13C14.67,13 20,14.33 20,17V20H4V17C4,14.33 9.33,13 12,13M12,14.9C9.03,14.9 5.9,16.36 5.9,17V18.1H18.1V17C18.1,16.36 14.97,14.9 12,14.9Z" /> | ||||
|       </svg> | ||||
| @ -58,26 +58,32 @@ export default { | ||||
|     return {} | ||||
|   }, | ||||
|   computed: { | ||||
|     currentLibraryMediaType() { | ||||
|       return this.$store.getters['libraries/getCurrentLibraryMediaType'] | ||||
|     }, | ||||
|     isBookLibrary() { | ||||
|       return this.currentLibraryMediaType === 'book' | ||||
|     }, | ||||
|     user() { | ||||
|       return this.$store.state.user.user | ||||
|     }, | ||||
|     totalItems() { | ||||
|       return this.libraryStats ? this.libraryStats.totalItems : 0 | ||||
|       return this.libraryStats?.totalItems || 0 | ||||
|     }, | ||||
|     totalAuthors() { | ||||
|       return this.libraryStats ? this.libraryStats.totalAuthors : 0 | ||||
|       return this.libraryStats?.totalAuthors || 0 | ||||
|     }, | ||||
|     numAudioTracks() { | ||||
|       return this.libraryStats ? this.libraryStats.numAudioTracks : 0 | ||||
|       return this.libraryStats?.numAudioTracks || 0 | ||||
|     }, | ||||
|     totalDuration() { | ||||
|       return this.libraryStats ? this.libraryStats.totalDuration : 0 | ||||
|       return this.libraryStats?.totalDuration || 0 | ||||
|     }, | ||||
|     totalHours() { | ||||
|       return Math.round(this.totalDuration / (60 * 60)) | ||||
|     }, | ||||
|     totalSizePretty() { | ||||
|       var totalSize = this.libraryStats ? this.libraryStats.totalSize : 0 | ||||
|       var totalSize = this.libraryStats?.totalSize || 0 | ||||
|       return this.$bytesPretty(totalSize, 1) | ||||
|     }, | ||||
|     totalSizeNum() { | ||||
|  | ||||
| @ -343,6 +343,10 @@ export default { | ||||
|       } | ||||
|       this.$store.commit('libraries/removeCollection', collection) | ||||
|     }, | ||||
|     seriesRemoved({ id, libraryId }) { | ||||
|       if (this.currentLibraryId !== libraryId) return | ||||
|       this.$store.commit('libraries/removeSeriesFromFilterData', id) | ||||
|     }, | ||||
|     playlistAdded(playlist) { | ||||
|       if (playlist.userId !== this.user.id || this.currentLibraryId !== playlist.libraryId) return | ||||
|       this.$store.commit('libraries/addUpdateUserPlaylist', playlist) | ||||
| @ -442,6 +446,9 @@ export default { | ||||
|       this.socket.on('collection_updated', this.collectionUpdated) | ||||
|       this.socket.on('collection_removed', this.collectionRemoved) | ||||
| 
 | ||||
|       // Series Listeners | ||||
|       this.socket.on('series_removed', this.seriesRemoved) | ||||
| 
 | ||||
|       // User Playlist Listeners | ||||
|       this.socket.on('playlist_added', this.playlistAdded) | ||||
|       this.socket.on('playlist_updated', this.playlistUpdated) | ||||
|  | ||||
| @ -22,7 +22,7 @@ | ||||
|             </div> | ||||
|           </template> | ||||
|         </div> | ||||
|         <div class="w-80 my-6 mx-auto"> | ||||
|         <div v-if="isBookLibrary" class="w-80 my-6 mx-auto"> | ||||
|           <h1 class="text-2xl mb-4">{{ $strings.HeaderStatsTop10Authors }}</h1> | ||||
|           <p v-if="!top10Authors.length">{{ $strings.MessageNoAuthors }}</p> | ||||
|           <template v-for="(author, index) in top10Authors"> | ||||
| @ -114,43 +114,49 @@ export default { | ||||
|       return this.$store.state.user.user | ||||
|     }, | ||||
|     totalItems() { | ||||
|       return this.libraryStats ? this.libraryStats.totalItems : 0 | ||||
|       return this.libraryStats?.totalItems || 0 | ||||
|     }, | ||||
|     genresWithCount() { | ||||
|       return this.libraryStats ? this.libraryStats.genresWithCount : [] | ||||
|       return this.libraryStats?.genresWithCount || [] | ||||
|     }, | ||||
|     top5Genres() { | ||||
|       return this.genresWithCount.slice(0, 5) | ||||
|       return this.genresWithCount?.slice(0, 5) || [] | ||||
|     }, | ||||
|     top10LongestItems() { | ||||
|       return this.libraryStats ? this.libraryStats.longestItems || [] : [] | ||||
|       return this.libraryStats?.longestItems || [] | ||||
|     }, | ||||
|     longestItemDuration() { | ||||
|       if (!this.top10LongestItems.length) return 0 | ||||
|       return this.top10LongestItems[0].duration | ||||
|     }, | ||||
|     top10LargestItems() { | ||||
|       return this.libraryStats ? this.libraryStats.largestItems || [] : [] | ||||
|       return this.libraryStats?.largestItems || [] | ||||
|     }, | ||||
|     largestItemSize() { | ||||
|       if (!this.top10LargestItems.length) return 0 | ||||
|       return this.top10LargestItems[0].size | ||||
|     }, | ||||
|     authorsWithCount() { | ||||
|       return this.libraryStats ? this.libraryStats.authorsWithCount : [] | ||||
|       return this.libraryStats?.authorsWithCount || [] | ||||
|     }, | ||||
|     mostUsedAuthorCount() { | ||||
|       if (!this.authorsWithCount.length) return 0 | ||||
|       return this.authorsWithCount[0].count | ||||
|     }, | ||||
|     top10Authors() { | ||||
|       return this.authorsWithCount.slice(0, 10) | ||||
|       return this.authorsWithCount?.slice(0, 10) || [] | ||||
|     }, | ||||
|     currentLibraryId() { | ||||
|       return this.$store.state.libraries.currentLibraryId | ||||
|     }, | ||||
|     currentLibraryName() { | ||||
|       return this.$store.getters['libraries/getCurrentLibraryName'] | ||||
|     }, | ||||
|     currentLibraryMediaType() { | ||||
|       return this.$store.getters['libraries/getCurrentLibraryMediaType'] | ||||
|     }, | ||||
|     isBookLibrary() { | ||||
|       return this.currentLibraryMediaType === 'book' | ||||
|     } | ||||
|   }, | ||||
|   methods: { | ||||
|  | ||||
| @ -47,7 +47,7 @@ | ||||
|       <div class="py-2"> | ||||
|         <h1 class="text-lg mb-2 text-white text-opacity-90 px-2 sm:px-0">{{ $strings.HeaderSavedMediaProgress }}</h1> | ||||
| 
 | ||||
|         <table v-if="mediaProgressWithMedia.length" class="userAudiobooksTable"> | ||||
|         <table v-if="mediaProgress.length" class="userAudiobooksTable"> | ||||
|           <tr class="bg-primary bg-opacity-40"> | ||||
|             <th class="w-16 text-left">{{ $strings.LabelItem }}</th> | ||||
|             <th class="text-left"></th> | ||||
| @ -55,19 +55,14 @@ | ||||
|             <th class="w-40 hidden sm:table-cell">{{ $strings.LabelStartedAt }}</th> | ||||
|             <th class="w-40 hidden sm:table-cell">{{ $strings.LabelLastUpdate }}</th> | ||||
|           </tr> | ||||
|           <tr v-for="item in mediaProgressWithMedia" :key="item.id" :class="!item.isFinished ? '' : 'isFinished'"> | ||||
|           <tr v-for="item in mediaProgress" :key="item.id" :class="!item.isFinished ? '' : 'isFinished'"> | ||||
|             <td> | ||||
|               <covers-book-cover :width="50" :library-item="item" :book-cover-aspect-ratio="bookCoverAspectRatio" /> | ||||
|               <covers-preview-cover v-if="item.coverPath" :width="50" :src="$store.getters['globals/getLibraryItemCoverSrcById'](item.libraryItemId, item.mediaUpdatedAt)" :book-cover-aspect-ratio="bookCoverAspectRatio" :show-resolution="false" /> | ||||
|               <div v-else class="bg-primary flex items-center justify-center text-center text-xs text-gray-400 p-1" :style="{ width: '50px', height: 50 * bookCoverAspectRatio + 'px' }">No Cover</div> | ||||
|             </td> | ||||
|             <td> | ||||
|               <template v-if="item.media && item.media.metadata && item.episode"> | ||||
|                 <p>{{ item.episode.title || 'Unknown' }}</p> | ||||
|                 <p class="text-white text-opacity-50 text-sm font-sans">{{ item.media.metadata.title }}</p> | ||||
|               </template> | ||||
|               <template v-else-if="item.media && item.media.metadata"> | ||||
|                 <p>{{ item.media.metadata.title || 'Unknown' }}</p> | ||||
|                 <p v-if="item.media.metadata.authorName" class="text-white text-opacity-50 text-sm font-sans">by {{ item.media.metadata.authorName }}</p> | ||||
|               </template> | ||||
|               <p>{{ item.displayTitle || 'Unknown' }}</p> | ||||
|               <p v-if="item.displaySubtitle" class="text-white text-opacity-50 text-sm font-sans">{{ item.displaySubtitle }}</p> | ||||
|             </td> | ||||
|             <td class="text-center"> | ||||
|               <p class="text-sm">{{ Math.floor(item.progress * 100) }}%</p> | ||||
| @ -124,9 +119,6 @@ export default { | ||||
|     mediaProgress() { | ||||
|       return this.user.mediaProgress.sort((a, b) => b.lastUpdate - a.lastUpdate) | ||||
|     }, | ||||
|     mediaProgressWithMedia() { | ||||
|       return this.mediaProgress.filter((mp) => mp.media) | ||||
|     }, | ||||
|     totalListeningTime() { | ||||
|       return this.listeningStats.totalTime || 0 | ||||
|     }, | ||||
|  | ||||
| @ -234,6 +234,10 @@ export const mutations = { | ||||
|   setNumUserPlaylists(state, numUserPlaylists) { | ||||
|     state.numUserPlaylists = numUserPlaylists | ||||
|   }, | ||||
|   removeSeriesFromFilterData(state, seriesId) { | ||||
|     if (!seriesId || !state.filterData) return | ||||
|     state.filterData.series = state.filterData.series.filter(se => se.id !== seriesId) | ||||
|   }, | ||||
|   updateFilterDataWithItem(state, libraryItem) { | ||||
|     if (!libraryItem || !state.filterData) return | ||||
|     if (state.currentLibraryId !== libraryItem.libraryId) return | ||||
|  | ||||
| @ -32,7 +32,7 @@ class Auth { | ||||
|     await Database.updateServerSettings() | ||||
| 
 | ||||
|     // New token secret creation added in v2.1.0 so generate new API tokens for each user
 | ||||
|     const users = await Database.models.user.getOldUsers() | ||||
|     const users = await Database.userModel.getOldUsers() | ||||
|     if (users.length) { | ||||
|       for (const user of users) { | ||||
|         user.token = await this.generateAccessToken({ userId: user.id, username: user.username }) | ||||
| @ -100,7 +100,7 @@ class Auth { | ||||
|           return resolve(null) | ||||
|         } | ||||
| 
 | ||||
|         const user = await Database.models.user.getUserByIdOrOldId(payload.userId) | ||||
|         const user = await Database.userModel.getUserByIdOrOldId(payload.userId) | ||||
|         if (user && user.username === payload.username) { | ||||
|           resolve(user) | ||||
|         } else { | ||||
| @ -116,7 +116,7 @@ class Auth { | ||||
|    * @returns {object} | ||||
|    */ | ||||
|   async getUserLoginResponsePayload(user) { | ||||
|     const libraryIds = await Database.models.library.getAllLibraryIds() | ||||
|     const libraryIds = await Database.libraryModel.getAllLibraryIds() | ||||
|     return { | ||||
|       user: user.toJSONForBrowser(), | ||||
|       userDefaultLibraryId: user.getDefaultLibraryId(libraryIds), | ||||
| @ -131,7 +131,7 @@ class Auth { | ||||
|     const username = (req.body.username || '').toLowerCase() | ||||
|     const password = req.body.password || '' | ||||
| 
 | ||||
|     const user = await Database.models.user.getUserByUsername(username) | ||||
|     const user = await Database.userModel.getUserByUsername(username) | ||||
| 
 | ||||
|     if (!user?.isActive) { | ||||
|       Logger.warn(`[Auth] Failed login attempt ${req.rateLimit.current} of ${req.rateLimit.limit} from ${ipAddress}`) | ||||
| @ -178,7 +178,7 @@ class Auth { | ||||
|   async userChangePassword(req, res) { | ||||
|     var { password, newPassword } = req.body | ||||
|     newPassword = newPassword || '' | ||||
|     const matchingUser = await Database.models.user.getUserById(req.user.id) | ||||
|     const matchingUser = await Database.userModel.getUserById(req.user.id) | ||||
| 
 | ||||
|     // Only root can have an empty password
 | ||||
|     if (matchingUser.type !== 'root' && !newPassword) { | ||||
|  | ||||
| @ -34,6 +34,100 @@ class Database { | ||||
|     return this.sequelize?.models || {} | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/User')} */ | ||||
|   get userModel() { | ||||
|     return this.models.user | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/Library')} */ | ||||
|   get libraryModel() { | ||||
|     return this.models.library | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/Author')} */ | ||||
|   get authorModel() { | ||||
|     return this.models.author | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/Series')} */ | ||||
|   get seriesModel() { | ||||
|     return this.models.series | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/Book')} */ | ||||
|   get bookModel() { | ||||
|     return this.models.book | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/BookSeries')} */ | ||||
|   get bookSeriesModel() { | ||||
|     return this.models.bookSeries | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/BookAuthor')} */ | ||||
|   get bookAuthorModel() { | ||||
|     return this.models.bookAuthor | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/Podcast')} */ | ||||
|   get podcastModel() { | ||||
|     return this.models.podcast | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/PodcastEpisode')} */ | ||||
|   get podcastEpisodeModel() { | ||||
|     return this.models.podcastEpisode | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/LibraryItem')} */ | ||||
|   get libraryItemModel() { | ||||
|     return this.models.libraryItem | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/PodcastEpisode')} */ | ||||
|   get podcastEpisodeModel() { | ||||
|     return this.models.podcastEpisode | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/MediaProgress')} */ | ||||
|   get mediaProgressModel() { | ||||
|     return this.models.mediaProgress | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/Collection')} */ | ||||
|   get collectionModel() { | ||||
|     return this.models.collection | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/CollectionBook')} */ | ||||
|   get collectionBookModel() { | ||||
|     return this.models.collectionBook | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/Playlist')} */ | ||||
|   get playlistModel() { | ||||
|     return this.models.playlist | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/PlaylistMediaItem')} */ | ||||
|   get playlistMediaItemModel() { | ||||
|     return this.models.playlistMediaItem | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/Feed')} */ | ||||
|   get feedModel() { | ||||
|     return this.models.feed | ||||
|   } | ||||
| 
 | ||||
|   /** @type {typeof import('./models/Feed')} */ | ||||
|   get feedEpisodeModel() { | ||||
|     return this.models.feedEpisode | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Check if db file exists | ||||
|    * @returns {boolean} | ||||
|    */ | ||||
|   async checkHasDb() { | ||||
|     if (!await fs.pathExists(this.dbPath)) { | ||||
|       Logger.info(`[Database] absdatabase.sqlite not found at ${this.dbPath}`) | ||||
| @ -42,6 +136,10 @@ class Database { | ||||
|     return true | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Connect to db, build models and run migrations | ||||
|    * @param {boolean} [force=false] Used for testing, drops & re-creates all tables | ||||
|    */ | ||||
|   async init(force = false) { | ||||
|     this.dbPath = Path.join(global.ConfigPath, 'absdatabase.sqlite') | ||||
| 
 | ||||
| @ -58,6 +156,10 @@ class Database { | ||||
|     await this.loadData() | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Connect to db | ||||
|    * @returns {boolean} | ||||
|    */ | ||||
|   async connect() { | ||||
|     Logger.info(`[Database] Initializing db at "${this.dbPath}"`) | ||||
|     this.sequelize = new Sequelize({ | ||||
| @ -80,39 +182,45 @@ class Database { | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Disconnect from db | ||||
|    */ | ||||
|   async disconnect() { | ||||
|     Logger.info(`[Database] Disconnecting sqlite db`) | ||||
|     await this.sequelize.close() | ||||
|     this.sequelize = null | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Reconnect to db and init | ||||
|    */ | ||||
|   async reconnect() { | ||||
|     Logger.info(`[Database] Reconnecting sqlite db`) | ||||
|     await this.init() | ||||
|   } | ||||
| 
 | ||||
|   buildModels(force = false) { | ||||
|     require('./models/User')(this.sequelize) | ||||
|     require('./models/Library')(this.sequelize) | ||||
|     require('./models/LibraryFolder')(this.sequelize) | ||||
|     require('./models/Book')(this.sequelize) | ||||
|     require('./models/Podcast')(this.sequelize) | ||||
|     require('./models/PodcastEpisode')(this.sequelize) | ||||
|     require('./models/LibraryItem')(this.sequelize) | ||||
|     require('./models/MediaProgress')(this.sequelize) | ||||
|     require('./models/Series')(this.sequelize) | ||||
|     require('./models/BookSeries')(this.sequelize) | ||||
|     require('./models/User').init(this.sequelize) | ||||
|     require('./models/Library').init(this.sequelize) | ||||
|     require('./models/LibraryFolder').init(this.sequelize) | ||||
|     require('./models/Book').init(this.sequelize) | ||||
|     require('./models/Podcast').init(this.sequelize) | ||||
|     require('./models/PodcastEpisode').init(this.sequelize) | ||||
|     require('./models/LibraryItem').init(this.sequelize) | ||||
|     require('./models/MediaProgress').init(this.sequelize) | ||||
|     require('./models/Series').init(this.sequelize) | ||||
|     require('./models/BookSeries').init(this.sequelize) | ||||
|     require('./models/Author').init(this.sequelize) | ||||
|     require('./models/BookAuthor')(this.sequelize) | ||||
|     require('./models/Collection')(this.sequelize) | ||||
|     require('./models/CollectionBook')(this.sequelize) | ||||
|     require('./models/Playlist')(this.sequelize) | ||||
|     require('./models/PlaylistMediaItem')(this.sequelize) | ||||
|     require('./models/Device')(this.sequelize) | ||||
|     require('./models/PlaybackSession')(this.sequelize) | ||||
|     require('./models/Feed')(this.sequelize) | ||||
|     require('./models/FeedEpisode')(this.sequelize) | ||||
|     require('./models/Setting')(this.sequelize) | ||||
|     require('./models/BookAuthor').init(this.sequelize) | ||||
|     require('./models/Collection').init(this.sequelize) | ||||
|     require('./models/CollectionBook').init(this.sequelize) | ||||
|     require('./models/Playlist').init(this.sequelize) | ||||
|     require('./models/PlaylistMediaItem').init(this.sequelize) | ||||
|     require('./models/Device').init(this.sequelize) | ||||
|     require('./models/PlaybackSession').init(this.sequelize) | ||||
|     require('./models/Feed').init(this.sequelize) | ||||
|     require('./models/FeedEpisode').init(this.sequelize) | ||||
|     require('./models/Setting').init(this.sequelize) | ||||
| 
 | ||||
|     return this.sequelize.sync({ force, alter: false }) | ||||
|   } | ||||
| @ -481,6 +589,88 @@ class Database { | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   removeSeriesFromFilterData(libraryId, seriesId) { | ||||
|     if (!this.libraryFilterData[libraryId]) return | ||||
|     this.libraryFilterData[libraryId].series = this.libraryFilterData[libraryId].series.filter(se => se.id !== seriesId) | ||||
|   } | ||||
| 
 | ||||
|   addSeriesToFilterData(libraryId, seriesName, seriesId) { | ||||
|     if (!this.libraryFilterData[libraryId]) return | ||||
|     // Check if series is already added
 | ||||
|     if (this.libraryFilterData[libraryId].series.some(se => se.id === seriesId)) return | ||||
|     this.libraryFilterData[libraryId].series.push({ | ||||
|       id: seriesId, | ||||
|       name: seriesName | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   removeAuthorFromFilterData(libraryId, authorId) { | ||||
|     if (!this.libraryFilterData[libraryId]) return | ||||
|     this.libraryFilterData[libraryId].authors = this.libraryFilterData[libraryId].authors.filter(au => au.id !== authorId) | ||||
|   } | ||||
| 
 | ||||
|   addAuthorToFilterData(libraryId, authorName, authorId) { | ||||
|     if (!this.libraryFilterData[libraryId]) return | ||||
|     // Check if author is already added
 | ||||
|     if (this.libraryFilterData[libraryId].authors.some(au => au.id === authorId)) return | ||||
|     this.libraryFilterData[libraryId].authors.push({ | ||||
|       id: authorId, | ||||
|       name: authorName | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Used when updating items to make sure author id exists | ||||
|    * If library filter data is set then use that for check | ||||
|    * otherwise lookup in db | ||||
|    * @param {string} libraryId  | ||||
|    * @param {string} authorId  | ||||
|    * @returns {Promise<boolean>} | ||||
|    */ | ||||
|   async checkAuthorExists(libraryId, authorId) { | ||||
|     if (!this.libraryFilterData[libraryId]) { | ||||
|       return this.authorModel.checkExistsById(authorId) | ||||
|     } | ||||
|     return this.libraryFilterData[libraryId].authors.some(au => au.id === authorId) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Used when updating items to make sure series id exists | ||||
|    * If library filter data is set then use that for check | ||||
|    * otherwise lookup in db | ||||
|    * @param {string} libraryId  | ||||
|    * @param {string} seriesId  | ||||
|    * @returns {Promise<boolean>} | ||||
|    */ | ||||
|   async checkSeriesExists(libraryId, seriesId) { | ||||
|     if (!this.libraryFilterData[libraryId]) { | ||||
|       return this.seriesModel.checkExistsById(seriesId) | ||||
|     } | ||||
|     return this.libraryFilterData[libraryId].series.some(se => se.id === seriesId) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Reset numIssues for library | ||||
|    * @param {string} libraryId  | ||||
|    */ | ||||
|   async resetLibraryIssuesFilterData(libraryId) { | ||||
|     if (!this.libraryFilterData[libraryId]) return // Do nothing if filter data is not set
 | ||||
| 
 | ||||
|     this.libraryFilterData[libraryId].numIssues = await this.libraryItemModel.count({ | ||||
|       where: { | ||||
|         libraryId, | ||||
|         [Sequelize.Op.or]: [ | ||||
|           { | ||||
|             isMissing: true | ||||
|           }, | ||||
|           { | ||||
|             isInvalid: true | ||||
|           } | ||||
|         ] | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = new Database() | ||||
| @ -114,10 +114,9 @@ class Server { | ||||
| 
 | ||||
|     await this.backupManager.init() | ||||
|     await this.logManager.init() | ||||
|     await this.apiRouter.checkRemoveEmptySeries(Database.series) // Remove empty series
 | ||||
|     await this.rssFeedManager.init() | ||||
| 
 | ||||
|     const libraries = await Database.models.library.getAllOldLibraries() | ||||
|     const libraries = await Database.libraryModel.getAllOldLibraries() | ||||
|     await this.cronManager.init(libraries) | ||||
| 
 | ||||
|     if (Database.serverSettings.scannerDisableWatcher) { | ||||
| @ -254,7 +253,7 @@ class Server { | ||||
|    */ | ||||
|   async cleanUserData() { | ||||
|     // Get all media progress without an associated media item
 | ||||
|     const mediaProgressToRemove = await Database.models.mediaProgress.findAll({ | ||||
|     const mediaProgressToRemove = await Database.mediaProgressModel.findAll({ | ||||
|       where: { | ||||
|         '$podcastEpisode.id$': null, | ||||
|         '$book.id$': null | ||||
| @ -262,18 +261,18 @@ class Server { | ||||
|       attributes: ['id'], | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.book, | ||||
|           model: Database.bookModel, | ||||
|           attributes: ['id'] | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.podcastEpisode, | ||||
|           model: Database.podcastEpisodeModel, | ||||
|           attributes: ['id'] | ||||
|         } | ||||
|       ] | ||||
|     }) | ||||
|     if (mediaProgressToRemove.length) { | ||||
|       // Remove media progress
 | ||||
|       const mediaProgressRemoved = await Database.models.mediaProgress.destroy({ | ||||
|       const mediaProgressRemoved = await Database.mediaProgressModel.destroy({ | ||||
|         where: { | ||||
|           id: { | ||||
|             [Sequelize.Op.in]: mediaProgressToRemove.map(mp => mp.id) | ||||
| @ -286,7 +285,7 @@ class Server { | ||||
|     } | ||||
| 
 | ||||
|     // Remove series from hide from continue listening that no longer exist
 | ||||
|     const users = await Database.models.user.getOldUsers() | ||||
|     const users = await Database.userModel.getOldUsers() | ||||
|     for (const _user of users) { | ||||
|       let hasUpdated = false | ||||
|       if (_user.seriesHideFromContinueListening.length) { | ||||
|  | ||||
| @ -21,7 +21,7 @@ class AuthorController { | ||||
| 
 | ||||
|     // Used on author landing page to include library items and items grouped in series
 | ||||
|     if (include.includes('items')) { | ||||
|       authorJson.libraryItems = await Database.models.libraryItem.getForAuthor(req.author, req.user) | ||||
|       authorJson.libraryItems = await Database.libraryItemModel.getForAuthor(req.author, req.user) | ||||
| 
 | ||||
|       if (include.includes('series')) { | ||||
|         const seriesMap = {} | ||||
| @ -96,7 +96,7 @@ class AuthorController { | ||||
|     const existingAuthor = authorNameUpdate ? Database.authors.find(au => au.id !== req.author.id && payload.name === au.name) : false | ||||
|     if (existingAuthor) { | ||||
|       const bookAuthorsToCreate = [] | ||||
|       const itemsWithAuthor = await Database.models.libraryItem.getForAuthor(req.author) | ||||
|       const itemsWithAuthor = await Database.libraryItemModel.getForAuthor(req.author) | ||||
|       itemsWithAuthor.forEach(libraryItem => { // Replace old author with merging author for each book
 | ||||
|         libraryItem.media.metadata.replaceAuthor(req.author, existingAuthor) | ||||
|         bookAuthorsToCreate.push({ | ||||
| @ -113,9 +113,11 @@ class AuthorController { | ||||
|       // Remove old author
 | ||||
|       await Database.removeAuthor(req.author.id) | ||||
|       SocketAuthority.emitter('author_removed', req.author.toJSON()) | ||||
|       // Update filter data
 | ||||
|       Database.removeAuthorFromFilterData(req.author.libraryId, req.author.id) | ||||
| 
 | ||||
|       // Send updated num books for merged author
 | ||||
|       const numBooks = await Database.models.libraryItem.getForAuthor(existingAuthor).length | ||||
|       const numBooks = await Database.libraryItemModel.getForAuthor(existingAuthor).length | ||||
|       SocketAuthority.emitter('author_updated', existingAuthor.toJSONExpanded(numBooks)) | ||||
| 
 | ||||
|       res.json({ | ||||
| @ -130,7 +132,7 @@ class AuthorController { | ||||
|       if (hasUpdated) { | ||||
|         req.author.updatedAt = Date.now() | ||||
| 
 | ||||
|         const itemsWithAuthor = await Database.models.libraryItem.getForAuthor(req.author) | ||||
|         const itemsWithAuthor = await Database.libraryItemModel.getForAuthor(req.author) | ||||
|         if (authorNameUpdate) { // Update author name on all books
 | ||||
|           itemsWithAuthor.forEach(libraryItem => { | ||||
|             libraryItem.media.metadata.updateAuthor(req.author) | ||||
| @ -202,7 +204,7 @@ class AuthorController { | ||||
| 
 | ||||
|       await Database.updateAuthor(req.author) | ||||
| 
 | ||||
|       const numBooks = await Database.models.libraryItem.getForAuthor(req.author).length | ||||
|       const numBooks = await Database.libraryItemModel.getForAuthor(req.author).length | ||||
|       SocketAuthority.emitter('author_updated', req.author.toJSONExpanded(numBooks)) | ||||
|     } | ||||
| 
 | ||||
|  | ||||
| @ -22,10 +22,10 @@ class CollectionController { | ||||
|     } | ||||
| 
 | ||||
|     // Create collection record
 | ||||
|     await Database.models.collection.createFromOld(newCollection) | ||||
|     await Database.collectionModel.createFromOld(newCollection) | ||||
| 
 | ||||
|     // Get library items in collection
 | ||||
|     const libraryItemsInCollection = await Database.models.libraryItem.getForCollection(newCollection) | ||||
|     const libraryItemsInCollection = await Database.libraryItemModel.getForCollection(newCollection) | ||||
| 
 | ||||
|     // Create collectionBook records
 | ||||
|     let order = 1 | ||||
| @ -50,7 +50,7 @@ class CollectionController { | ||||
|   } | ||||
| 
 | ||||
|   async findAll(req, res) { | ||||
|     const collectionsExpanded = await Database.models.collection.getOldCollectionsJsonExpanded(req.user) | ||||
|     const collectionsExpanded = await Database.collectionModel.getOldCollectionsJsonExpanded(req.user) | ||||
|     res.json({ | ||||
|       collections: collectionsExpanded | ||||
|     }) | ||||
| @ -96,8 +96,8 @@ class CollectionController { | ||||
|     if (req.body.books?.length) { | ||||
|       const collectionBooks = await req.collection.getCollectionBooks({ | ||||
|         include: { | ||||
|           model: Database.models.book, | ||||
|           include: Database.models.libraryItem | ||||
|           model: Database.bookModel, | ||||
|           include: Database.libraryItemModel | ||||
|         }, | ||||
|         order: [['order', 'ASC']] | ||||
|       }) | ||||
| @ -143,7 +143,7 @@ class CollectionController { | ||||
|    * @param {*} res  | ||||
|    */ | ||||
|   async addBook(req, res) { | ||||
|     const libraryItem = await Database.models.libraryItem.getOldById(req.body.id) | ||||
|     const libraryItem = await Database.libraryItemModel.getOldById(req.body.id) | ||||
|     if (!libraryItem) { | ||||
|       return res.status(404).send('Book not found') | ||||
|     } | ||||
| @ -158,7 +158,7 @@ class CollectionController { | ||||
|     } | ||||
| 
 | ||||
|     // Create collectionBook record
 | ||||
|     await Database.models.collectionBook.create({ | ||||
|     await Database.collectionBookModel.create({ | ||||
|       collectionId: req.collection.id, | ||||
|       bookId: libraryItem.media.id, | ||||
|       order: collectionBooks.length + 1 | ||||
| @ -176,7 +176,7 @@ class CollectionController { | ||||
|    * @param {*} res  | ||||
|    */ | ||||
|   async removeBook(req, res) { | ||||
|     const libraryItem = await Database.models.libraryItem.getOldById(req.params.bookId) | ||||
|     const libraryItem = await Database.libraryItemModel.getOldById(req.params.bookId) | ||||
|     if (!libraryItem) { | ||||
|       return res.sendStatus(404) | ||||
|     } | ||||
| @ -227,14 +227,14 @@ class CollectionController { | ||||
|     } | ||||
| 
 | ||||
|     // Get library items associated with ids
 | ||||
|     const libraryItems = await Database.models.libraryItem.findAll({ | ||||
|     const libraryItems = await Database.libraryItemModel.findAll({ | ||||
|       where: { | ||||
|         id: { | ||||
|           [Sequelize.Op.in]: bookIdsToAdd | ||||
|         } | ||||
|       }, | ||||
|       include: { | ||||
|         model: Database.models.book | ||||
|         model: Database.bookModel | ||||
|       } | ||||
|     }) | ||||
| 
 | ||||
| @ -285,14 +285,14 @@ class CollectionController { | ||||
|     } | ||||
| 
 | ||||
|     // Get library items associated with ids
 | ||||
|     const libraryItems = await Database.models.libraryItem.findAll({ | ||||
|     const libraryItems = await Database.libraryItemModel.findAll({ | ||||
|       where: { | ||||
|         id: { | ||||
|           [Sequelize.Op.in]: bookIdsToRemove | ||||
|         } | ||||
|       }, | ||||
|       include: { | ||||
|         model: Database.models.book | ||||
|         model: Database.bookModel | ||||
|       } | ||||
|     }) | ||||
| 
 | ||||
| @ -327,7 +327,7 @@ class CollectionController { | ||||
| 
 | ||||
|   async middleware(req, res, next) { | ||||
|     if (req.params.id) { | ||||
|       const collection = await Database.models.collection.findByPk(req.params.id) | ||||
|       const collection = await Database.collectionModel.findByPk(req.params.id) | ||||
|       if (!collection) { | ||||
|         return res.status(404).send('Collection not found') | ||||
|       } | ||||
|  | ||||
| @ -17,7 +17,7 @@ class FileSystemController { | ||||
|     }) | ||||
| 
 | ||||
|     // Do not include existing mapped library paths in response
 | ||||
|     const libraryFoldersPaths = await Database.models.libraryFolder.getAllLibraryFolderPaths() | ||||
|     const libraryFoldersPaths = await Database.libraryModelFolder.getAllLibraryFolderPaths() | ||||
|     libraryFoldersPaths.forEach((path) => { | ||||
|       let dir = path || '' | ||||
|       if (dir.includes(global.appRoot)) dir = dir.replace(global.appRoot, '') | ||||
|  | ||||
| @ -8,6 +8,7 @@ const Library = require('../objects/Library') | ||||
| const libraryHelpers = require('../utils/libraryHelpers') | ||||
| const libraryItemsBookFilters = require('../utils/queries/libraryItemsBookFilters') | ||||
| const libraryItemFilters = require('../utils/queries/libraryItemFilters') | ||||
| const seriesFilters = require('../utils/queries/seriesFilters') | ||||
| const { sort, createNewSortInstance } = require('../libs/fastSort') | ||||
| const naturalSort = createNewSortInstance({ | ||||
|   comparer: new Intl.Collator(undefined, { numeric: true, sensitivity: 'base' }).compare | ||||
| @ -15,6 +16,8 @@ const naturalSort = createNewSortInstance({ | ||||
| 
 | ||||
| const Database = require('../Database') | ||||
| const libraryFilters = require('../utils/queries/libraryFilters') | ||||
| const libraryItemsPodcastFilters = require('../utils/queries/libraryItemsPodcastFilters') | ||||
| const authorFilters = require('../utils/queries/authorFilters') | ||||
| 
 | ||||
| class LibraryController { | ||||
|   constructor() { } | ||||
| @ -48,7 +51,7 @@ class LibraryController { | ||||
| 
 | ||||
|     const library = new Library() | ||||
| 
 | ||||
|     let currentLargestDisplayOrder = await Database.models.library.getMaxDisplayOrder() | ||||
|     let currentLargestDisplayOrder = await Database.libraryModel.getMaxDisplayOrder() | ||||
|     if (isNaN(currentLargestDisplayOrder)) currentLargestDisplayOrder = 0 | ||||
|     newLibraryPayload.displayOrder = currentLargestDisplayOrder + 1 | ||||
|     library.setData(newLibraryPayload) | ||||
| @ -67,7 +70,7 @@ class LibraryController { | ||||
|   } | ||||
| 
 | ||||
|   async findAll(req, res) { | ||||
|     const libraries = await Database.models.library.getAllOldLibraries() | ||||
|     const libraries = await Database.libraryModel.getAllOldLibraries() | ||||
| 
 | ||||
|     const librariesAccessible = req.user.librariesAccessible || [] | ||||
|     if (librariesAccessible.length) { | ||||
| @ -89,7 +92,7 @@ class LibraryController { | ||||
|       return res.json({ | ||||
|         filterdata, | ||||
|         issues: filterdata.numIssues, | ||||
|         numUserPlaylists: await Database.models.playlist.getNumPlaylistsForUserAndLibrary(req.user.id, req.library.id), | ||||
|         numUserPlaylists: await Database.playlistModel.getNumPlaylistsForUserAndLibrary(req.user.id, req.library.id), | ||||
|         library: req.library | ||||
|       }) | ||||
|     } | ||||
| @ -141,17 +144,17 @@ class LibraryController { | ||||
|       for (const folder of library.folders) { | ||||
|         if (!req.body.folders.some(f => f.id === folder.id)) { | ||||
|           // Remove library items in folder
 | ||||
|           const libraryItemsInFolder = await Database.models.libraryItem.findAll({ | ||||
|           const libraryItemsInFolder = await Database.libraryItemModel.findAll({ | ||||
|             where: { | ||||
|               libraryFolderId: folder.id | ||||
|             }, | ||||
|             attributes: ['id', 'mediaId', 'mediaType'], | ||||
|             include: [ | ||||
|               { | ||||
|                 model: Database.models.podcast, | ||||
|                 model: Database.podcastModel, | ||||
|                 attributes: ['id'], | ||||
|                 include: { | ||||
|                   model: Database.models.podcastEpisode, | ||||
|                   model: Database.podcastEpisodeModel, | ||||
|                   attributes: ['id'] | ||||
|                 } | ||||
|               } | ||||
| @ -188,6 +191,8 @@ class LibraryController { | ||||
|         return user.checkCanAccessLibrary && user.checkCanAccessLibrary(library.id) | ||||
|       } | ||||
|       SocketAuthority.emitter('library_updated', library.toJSON(), userFilter) | ||||
| 
 | ||||
|       await Database.resetLibraryIssuesFilterData(library.id) | ||||
|     } | ||||
|     return res.json(library.toJSON()) | ||||
|   } | ||||
| @ -205,23 +210,23 @@ class LibraryController { | ||||
|     this.watcher.removeLibrary(library) | ||||
| 
 | ||||
|     // Remove collections for library
 | ||||
|     const numCollectionsRemoved = await Database.models.collection.removeAllForLibrary(library.id) | ||||
|     const numCollectionsRemoved = await Database.collectionModel.removeAllForLibrary(library.id) | ||||
|     if (numCollectionsRemoved) { | ||||
|       Logger.info(`[Server] Removed ${numCollectionsRemoved} collections for library "${library.name}"`) | ||||
|     } | ||||
| 
 | ||||
|     // Remove items in this library
 | ||||
|     const libraryItemsInLibrary = await Database.models.libraryItem.findAll({ | ||||
|     const libraryItemsInLibrary = await Database.libraryItemModel.findAll({ | ||||
|       where: { | ||||
|         libraryId: library.id | ||||
|       }, | ||||
|       attributes: ['id', 'mediaId', 'mediaType'], | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.podcast, | ||||
|           model: Database.podcastModel, | ||||
|           attributes: ['id'], | ||||
|           include: { | ||||
|             model: Database.models.podcastEpisode, | ||||
|             model: Database.podcastEpisodeModel, | ||||
|             attributes: ['id'] | ||||
|           } | ||||
|         } | ||||
| @ -243,9 +248,15 @@ class LibraryController { | ||||
|     await Database.removeLibrary(library.id) | ||||
| 
 | ||||
|     // Re-order libraries
 | ||||
|     await Database.models.library.resetDisplayOrder() | ||||
|     await Database.libraryModel.resetDisplayOrder() | ||||
| 
 | ||||
|     SocketAuthority.emitter('library_removed', libraryJson) | ||||
| 
 | ||||
|     // Remove library filter data
 | ||||
|     if (Database.libraryFilterData[library.id]) { | ||||
|       delete Database.libraryFilterData[library.id] | ||||
|     } | ||||
| 
 | ||||
|     return res.json(libraryJson) | ||||
|   } | ||||
| 
 | ||||
| @ -267,7 +278,7 @@ class LibraryController { | ||||
|     } | ||||
|     payload.offset = payload.page * payload.limit | ||||
| 
 | ||||
|     const { libraryItems, count } = await Database.models.libraryItem.getByFilterAndSort(req.library, req.user, payload) | ||||
|     const { libraryItems, count } = await Database.libraryItemModel.getByFilterAndSort(req.library, req.user, payload) | ||||
|     payload.results = libraryItems | ||||
|     payload.total = count | ||||
| 
 | ||||
| @ -471,12 +482,13 @@ class LibraryController { | ||||
|   /** | ||||
|    * DELETE: /libraries/:id/issues | ||||
|    * Remove all library items missing or invalid | ||||
|    * @param {*} req  | ||||
|    * @param {*} res  | ||||
|    * @param {import('express').Request} req  | ||||
|    * @param {import('express').Response} res  | ||||
|    */ | ||||
|   async removeLibraryItemsWithIssues(req, res) { | ||||
|     const libraryItemsWithIssues = await Database.models.libraryItem.findAll({ | ||||
|     const libraryItemsWithIssues = await Database.libraryItemModel.findAll({ | ||||
|       where: { | ||||
|         libraryId: req.library.id, | ||||
|         [Sequelize.Op.or]: [ | ||||
|           { | ||||
|             isMissing: true | ||||
| @ -489,10 +501,10 @@ class LibraryController { | ||||
|       attributes: ['id', 'mediaId', 'mediaType'], | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.podcast, | ||||
|           model: Database.podcastModel, | ||||
|           attributes: ['id'], | ||||
|           include: { | ||||
|             model: Database.models.podcastEpisode, | ||||
|             model: Database.podcastEpisodeModel, | ||||
|             attributes: ['id'] | ||||
|           } | ||||
|         } | ||||
| @ -507,7 +519,7 @@ class LibraryController { | ||||
|     Logger.info(`[LibraryController] Removing ${libraryItemsWithIssues.length} items with issues`) | ||||
|     for (const libraryItem of libraryItemsWithIssues) { | ||||
|       let mediaItemIds = [] | ||||
|       if (library.isPodcast) { | ||||
|       if (req.library.isPodcast) { | ||||
|         mediaItemIds = libraryItem.media.podcastEpisodes.map(pe => pe.id) | ||||
|       } else { | ||||
|         mediaItemIds.push(libraryItem.mediaId) | ||||
| @ -516,19 +528,22 @@ class LibraryController { | ||||
|       await this.handleDeleteLibraryItem(libraryItem.mediaType, libraryItem.id, mediaItemIds) | ||||
|     } | ||||
| 
 | ||||
|     // Set numIssues to 0 for library filter data
 | ||||
|     if (Database.libraryFilterData[req.library.id]) { | ||||
|       Database.libraryFilterData[req.library.id].numIssues = 0 | ||||
|     } | ||||
| 
 | ||||
|     res.sendStatus(200) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * GET: /api/libraries/:id/series | ||||
|    * Optional query string: `?include=rssfeed` that adds `rssFeed` to series if a feed is open | ||||
|    *  | ||||
|    * @param {*} req  | ||||
|    * @param {*} res  | ||||
|    */ | ||||
|  * GET: /api/libraries/:id/series | ||||
|  * Optional query string: `?include=rssfeed` that adds `rssFeed` to series if a feed is open | ||||
|  *  | ||||
|  * @param {import('express').Request} req  | ||||
|  * @param {import('express').Response} res  | ||||
|  */ | ||||
|   async getAllSeriesForLibrary(req, res) { | ||||
|     const libraryItems = req.libraryItems | ||||
| 
 | ||||
|     const include = (req.query.include || '').split(',').map(v => v.trim().toLowerCase()).filter(v => !!v) | ||||
| 
 | ||||
|     const payload = { | ||||
| @ -543,45 +558,10 @@ class LibraryController { | ||||
|       include: include.join(',') | ||||
|     } | ||||
| 
 | ||||
|     let series = libraryHelpers.getSeriesFromBooks(libraryItems, Database.series, null, payload.filterBy, req.user, payload.minified, req.library.settings.hideSingleBookSeries) | ||||
| 
 | ||||
|     const direction = payload.sortDesc ? 'desc' : 'asc' | ||||
|     series = naturalSort(series).by([ | ||||
|       { | ||||
|         [direction]: (se) => { | ||||
|           if (payload.sortBy === 'numBooks') { | ||||
|             return se.books.length | ||||
|           } else if (payload.sortBy === 'totalDuration') { | ||||
|             return se.totalDuration | ||||
|           } else if (payload.sortBy === 'addedAt') { | ||||
|             return se.addedAt | ||||
|           } else if (payload.sortBy === 'lastBookUpdated') { | ||||
|             return Math.max(...(se.books).map(x => x.updatedAt), 0) | ||||
|           } else if (payload.sortBy === 'lastBookAdded') { | ||||
|             return Math.max(...(se.books).map(x => x.addedAt), 0) | ||||
|           } else { // sort by name
 | ||||
|             return Database.serverSettings.sortingIgnorePrefix ? se.nameIgnorePrefixSort : se.name | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     ]) | ||||
| 
 | ||||
|     payload.total = series.length | ||||
| 
 | ||||
|     if (payload.limit) { | ||||
|       const startIndex = payload.page * payload.limit | ||||
|       series = series.slice(startIndex, startIndex + payload.limit) | ||||
|     } | ||||
| 
 | ||||
|     // add rssFeed when "include=rssfeed" is in query string
 | ||||
|     if (include.includes('rssfeed')) { | ||||
|       series = await Promise.all(series.map(async (se) => { | ||||
|         const feedData = await this.rssFeedManager.findFeedForEntityId(se.id) | ||||
|         se.rssFeed = feedData?.toJSONMinified() || null | ||||
|         return se | ||||
|       })) | ||||
|     } | ||||
|     const offset = payload.page * payload.limit | ||||
|     const { series, count } = await seriesFilters.getFilteredSeries(req.library, req.user, payload.filterBy, payload.sortBy, payload.sortDesc, include, payload.limit, offset) | ||||
| 
 | ||||
|     payload.total = count | ||||
|     payload.results = series | ||||
|     res.json(payload) | ||||
|   } | ||||
| @ -644,7 +624,7 @@ class LibraryController { | ||||
|     } | ||||
| 
 | ||||
|     // TODO: Create paginated queries
 | ||||
|     let collections = await Database.models.collection.getOldCollectionsJsonExpanded(req.user, req.library.id, include) | ||||
|     let collections = await Database.collectionModel.getOldCollectionsJsonExpanded(req.user, req.library.id, include) | ||||
| 
 | ||||
|     payload.total = collections.length | ||||
| 
 | ||||
| @ -664,7 +644,7 @@ class LibraryController { | ||||
|    * @param {*} res  | ||||
|    */ | ||||
|   async getUserPlaylistsForLibrary(req, res) { | ||||
|     let playlistsForUser = await Database.models.playlist.getPlaylistsForUserAndLibrary(req.user.id, req.library.id) | ||||
|     let playlistsForUser = await Database.playlistModel.getPlaylistsForUserAndLibrary(req.user.id, req.library.id) | ||||
|     playlistsForUser = await Promise.all(playlistsForUser.map(async p => p.getOldJsonExpanded())) | ||||
| 
 | ||||
|     const payload = { | ||||
| @ -685,8 +665,8 @@ class LibraryController { | ||||
| 
 | ||||
|   /** | ||||
|    * GET: /api/libraries/:id/filterdata | ||||
|    * @param {*} req  | ||||
|    * @param {*} res  | ||||
|    * @param {import('express').Request} req  | ||||
|    * @param {import('express').Response} res  | ||||
|    */ | ||||
|   async getLibraryFilterData(req, res) { | ||||
|     const filterData = await libraryFilters.getFilterData(req.library) | ||||
| @ -694,44 +674,30 @@ class LibraryController { | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * GET: /api/libraries/:id/personalized2 | ||||
|    * TODO: new endpoint | ||||
|    * @param {*} req  | ||||
|    * @param {*} res  | ||||
|    * GET: /api/libraries/:id/personalized | ||||
|    * Home page shelves | ||||
|    * @param {import('express').Request} req  | ||||
|    * @param {import('express').Response} res  | ||||
|    */ | ||||
|   async getUserPersonalizedShelves(req, res) { | ||||
|     const limitPerShelf = req.query.limit && !isNaN(req.query.limit) ? Number(req.query.limit) || 10 : 10 | ||||
|     const include = (req.query.include || '').split(',').map(v => v.trim().toLowerCase()).filter(v => !!v) | ||||
|     const shelves = await Database.models.libraryItem.getPersonalizedShelves(req.library, req.user, include, limitPerShelf) | ||||
|     const shelves = await Database.libraryItemModel.getPersonalizedShelves(req.library, req.user, include, limitPerShelf) | ||||
|     res.json(shelves) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * GET: /api/libraries/:id/personalized | ||||
|    * TODO: remove after personalized2 is ready | ||||
|    * @param {*} req  | ||||
|    * @param {*} res  | ||||
|    */ | ||||
|   async getLibraryUserPersonalizedOptimal(req, res) { | ||||
|     const limitPerShelf = req.query.limit && !isNaN(req.query.limit) ? Number(req.query.limit) || 10 : 10 | ||||
|     const include = (req.query.include || '').split(',').map(v => v.trim().toLowerCase()).filter(v => !!v) | ||||
| 
 | ||||
|     const categories = await libraryHelpers.buildPersonalizedShelves(this, req.user, req.libraryItems, req.library, limitPerShelf, include) | ||||
|     res.json(categories) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * POST: /api/libraries/order | ||||
|    * Change the display order of libraries | ||||
|    * @param {*} req  | ||||
|    * @param {*} res  | ||||
|    * @param {import('express').Request} req  | ||||
|    * @param {import('express').Response} res  | ||||
|    */ | ||||
|   async reorder(req, res) { | ||||
|     if (!req.user.isAdminOrUp) { | ||||
|       Logger.error('[LibraryController] ReorderLibraries invalid user', req.user) | ||||
|       return res.sendStatus(403) | ||||
|     } | ||||
|     const libraries = await Database.models.library.getAllOldLibraries() | ||||
|     const libraries = await Database.libraryModel.getAllOldLibraries() | ||||
| 
 | ||||
|     const orderdata = req.body | ||||
|     let hasUpdates = false | ||||
| @ -759,99 +725,62 @@ class LibraryController { | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   // GET: Global library search
 | ||||
|   search(req, res) { | ||||
|   /** | ||||
|    * GET: /api/libraries/:id/search | ||||
|    * Search library items with query | ||||
|    * ?q=search | ||||
|    * @param {import('express').Request} req  | ||||
|    * @param {import('express').Response} res  | ||||
|    */ | ||||
|   async search(req, res) { | ||||
|     if (!req.query.q) { | ||||
|       return res.status(400).send('No query string') | ||||
|     } | ||||
|     const libraryItems = req.libraryItems | ||||
|     const maxResults = req.query.limit && !isNaN(req.query.limit) ? Number(req.query.limit) : 12 | ||||
|     const limit = req.query.limit && !isNaN(req.query.limit) ? Number(req.query.limit) : 12 | ||||
|     const query = req.query.q.trim().toLowerCase() | ||||
| 
 | ||||
|     const itemMatches = [] | ||||
|     const authorMatches = {} | ||||
|     const narratorMatches = {} | ||||
|     const seriesMatches = {} | ||||
|     const tagMatches = {} | ||||
| 
 | ||||
|     libraryItems.forEach((li) => { | ||||
|       const queryResult = li.searchQuery(req.query.q) | ||||
|       if (queryResult.matchKey) { | ||||
|         itemMatches.push({ | ||||
|           libraryItem: li.toJSONExpanded(), | ||||
|           matchKey: queryResult.matchKey, | ||||
|           matchText: queryResult.matchText | ||||
|         }) | ||||
|       } | ||||
|       if (queryResult.series?.length) { | ||||
|         queryResult.series.forEach((se) => { | ||||
|           if (!seriesMatches[se.id]) { | ||||
|             const _series = Database.series.find(_se => _se.id === se.id) | ||||
|             if (_series) seriesMatches[se.id] = { series: _series.toJSON(), books: [li.toJSON()] } | ||||
|           } else { | ||||
|             seriesMatches[se.id].books.push(li.toJSON()) | ||||
|           } | ||||
|         }) | ||||
|       } | ||||
|       if (queryResult.authors?.length) { | ||||
|         queryResult.authors.forEach((au) => { | ||||
|           if (!authorMatches[au.id]) { | ||||
|             const _author = Database.authors.find(_au => _au.id === au.id) | ||||
|             if (_author) { | ||||
|               authorMatches[au.id] = _author.toJSON() | ||||
|               authorMatches[au.id].numBooks = 1 | ||||
|             } | ||||
|           } else { | ||||
|             authorMatches[au.id].numBooks++ | ||||
|           } | ||||
|         }) | ||||
|       } | ||||
|       if (queryResult.tags?.length) { | ||||
|         queryResult.tags.forEach((tag) => { | ||||
|           if (!tagMatches[tag]) { | ||||
|             tagMatches[tag] = { name: tag, books: [li.toJSON()] } | ||||
|           } else { | ||||
|             tagMatches[tag].books.push(li.toJSON()) | ||||
|           } | ||||
|         }) | ||||
|       } | ||||
|       if (queryResult.narrators?.length) { | ||||
|         queryResult.narrators.forEach((narrator) => { | ||||
|           if (!narratorMatches[narrator]) { | ||||
|             narratorMatches[narrator] = { name: narrator, books: [li.toJSON()] } | ||||
|           } else { | ||||
|             narratorMatches[narrator].books.push(li.toJSON()) | ||||
|           } | ||||
|         }) | ||||
|       } | ||||
|     }) | ||||
|     const itemKey = req.library.mediaType | ||||
|     const results = { | ||||
|       [itemKey]: itemMatches.slice(0, maxResults), | ||||
|       tags: Object.values(tagMatches).slice(0, maxResults), | ||||
|       authors: Object.values(authorMatches).slice(0, maxResults), | ||||
|       series: Object.values(seriesMatches).slice(0, maxResults), | ||||
|       narrators: Object.values(narratorMatches).slice(0, maxResults) | ||||
|     } | ||||
|     res.json(results) | ||||
|     const matches = await libraryItemFilters.search(req.user, req.library, query, limit) | ||||
|     res.json(matches) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * GET: /api/libraries/:id/stats | ||||
|    * Get stats for library | ||||
|    * @param {import('express').Request} req  | ||||
|    * @param {import('express').Response} res  | ||||
|    */ | ||||
|   async stats(req, res) { | ||||
|     var libraryItems = req.libraryItems | ||||
|     var authorsWithCount = libraryHelpers.getAuthorsWithCount(libraryItems) | ||||
|     var genresWithCount = libraryHelpers.getGenresWithCount(libraryItems) | ||||
|     var durationStats = libraryHelpers.getItemDurationStats(libraryItems) | ||||
|     var sizeStats = libraryHelpers.getItemSizeStats(libraryItems) | ||||
|     var stats = { | ||||
|       totalItems: libraryItems.length, | ||||
|       totalAuthors: Object.keys(authorsWithCount).length, | ||||
|       totalGenres: Object.keys(genresWithCount).length, | ||||
|       totalDuration: durationStats.totalDuration, | ||||
|       longestItems: durationStats.longestItems, | ||||
|       numAudioTracks: durationStats.numAudioTracks, | ||||
|       totalSize: libraryHelpers.getLibraryItemsTotalSize(libraryItems), | ||||
|       largestItems: sizeStats.largestItems, | ||||
|       authorsWithCount, | ||||
|       genresWithCount | ||||
|     const stats = { | ||||
|       largestItems: await libraryItemFilters.getLargestItems(req.library.id, 10) | ||||
|     } | ||||
| 
 | ||||
|     if (req.library.isBook) { | ||||
|       const authors = await authorFilters.getAuthorsWithCount(req.library.id) | ||||
|       const genres = await libraryItemsBookFilters.getGenresWithCount(req.library.id) | ||||
|       const bookStats = await libraryItemsBookFilters.getBookLibraryStats(req.library.id) | ||||
|       const longestBooks = await libraryItemsBookFilters.getLongestBooks(req.library.id, 10) | ||||
| 
 | ||||
|       stats.totalAuthors = authors.length | ||||
|       stats.authorsWithCount = authors | ||||
|       stats.totalGenres = genres.length | ||||
|       stats.genresWithCount = genres | ||||
|       stats.totalItems = bookStats.totalItems | ||||
|       stats.longestItems = longestBooks | ||||
|       stats.totalSize = bookStats.totalSize | ||||
|       stats.totalDuration = bookStats.totalDuration | ||||
|       stats.numAudioTracks = bookStats.numAudioFiles | ||||
|     } else { | ||||
|       const genres = await libraryItemsPodcastFilters.getGenresWithCount(req.library.id) | ||||
|       const podcastStats = await libraryItemsPodcastFilters.getPodcastLibraryStats(req.library.id) | ||||
|       const longestPodcasts = await libraryItemsPodcastFilters.getLongestPodcasts(req.library.id, 10) | ||||
| 
 | ||||
|       stats.totalGenres = genres.length | ||||
|       stats.genresWithCount = genres | ||||
|       stats.totalItems = podcastStats.totalItems | ||||
|       stats.longestItems = longestPodcasts | ||||
|       stats.totalSize = podcastStats.totalSize | ||||
|       stats.totalDuration = podcastStats.totalDuration | ||||
|       stats.numAudioTracks = podcastStats.numAudioFiles | ||||
|     } | ||||
|     res.json(stats) | ||||
|   } | ||||
| @ -859,18 +788,18 @@ class LibraryController { | ||||
|   /** | ||||
|    * GET: /api/libraries/:id/authors | ||||
|    * Get authors for library | ||||
|    * @param {*} req  | ||||
|    * @param {*} res  | ||||
|    * @param {import('express').Request} req  | ||||
|    * @param {import('express').Response} res  | ||||
|    */ | ||||
|   async getAuthors(req, res) { | ||||
|     const { bookWhere, replacements } = libraryItemsBookFilters.getUserPermissionBookWhereQuery(req.user) | ||||
|     const authors = await Database.models.author.findAll({ | ||||
|     const authors = await Database.authorModel.findAll({ | ||||
|       where: { | ||||
|         libraryId: req.library.id | ||||
|       }, | ||||
|       replacements, | ||||
|       include: { | ||||
|         model: Database.models.book, | ||||
|         model: Database.bookModel, | ||||
|         attributes: ['id', 'tags', 'explicit'], | ||||
|         where: bookWhere, | ||||
|         required: true, | ||||
| @ -903,12 +832,12 @@ class LibraryController { | ||||
|    */ | ||||
|   async getNarrators(req, res) { | ||||
|     // Get all books with narrators
 | ||||
|     const booksWithNarrators = await Database.models.book.findAll({ | ||||
|     const booksWithNarrators = await Database.bookModel.findAll({ | ||||
|       where: Sequelize.where(Sequelize.fn('json_array_length', Sequelize.col('narrators')), { | ||||
|         [Sequelize.Op.gt]: 0 | ||||
|       }), | ||||
|       include: { | ||||
|         model: Database.models.libraryItem, | ||||
|         model: Database.libraryItemModel, | ||||
|         attributes: ['id', 'libraryId'], | ||||
|         where: { | ||||
|           libraryId: req.library.id | ||||
| @ -975,7 +904,7 @@ class LibraryController { | ||||
|       await libraryItem.media.update({ | ||||
|         narrators: libraryItem.media.narrators | ||||
|       }) | ||||
|       const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(libraryItem) | ||||
|       const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(libraryItem) | ||||
|       itemsUpdated.push(oldLibraryItem) | ||||
|     } | ||||
| 
 | ||||
| @ -1015,7 +944,7 @@ class LibraryController { | ||||
|       await libraryItem.media.update({ | ||||
|         narrators: libraryItem.media.narrators | ||||
|       }) | ||||
|       const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(libraryItem) | ||||
|       const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(libraryItem) | ||||
|       itemsUpdated.push(oldLibraryItem) | ||||
|     } | ||||
| 
 | ||||
| @ -1048,10 +977,16 @@ class LibraryController { | ||||
|     } | ||||
|     res.sendStatus(200) | ||||
|     await this.scanner.scan(req.library, options) | ||||
|     await Database.resetLibraryIssuesFilterData(req.library.id) | ||||
|     Logger.info('[LibraryController] Scan complete') | ||||
|   } | ||||
| 
 | ||||
|   // GET: api/libraries/:id/recent-episode
 | ||||
|   /** | ||||
|    * GET: /api/libraries/:id/recent-episodes | ||||
|    * Used for latest page | ||||
|    * @param {import('express').Request} req  | ||||
|    * @param {import('express').Response} res  | ||||
|    */ | ||||
|   async getRecentEpisodes(req, res) { | ||||
|     if (!req.library.isPodcast) { | ||||
|       return res.sendStatus(404) | ||||
| @ -1059,40 +994,37 @@ class LibraryController { | ||||
| 
 | ||||
|     const payload = { | ||||
|       episodes: [], | ||||
|       total: 0, | ||||
|       limit: req.query.limit && !isNaN(req.query.limit) ? Number(req.query.limit) : 0, | ||||
|       page: req.query.page && !isNaN(req.query.page) ? Number(req.query.page) : 0, | ||||
|     } | ||||
| 
 | ||||
|     var allUnfinishedEpisodes = [] | ||||
|     for (const libraryItem of req.libraryItems) { | ||||
|       const unfinishedEpisodes = libraryItem.media.episodes.filter(ep => { | ||||
|         const userProgress = req.user.getMediaProgress(libraryItem.id, ep.id) | ||||
|         return !userProgress || !userProgress.isFinished | ||||
|       }).map(_ep => { | ||||
|         const ep = _ep.toJSONExpanded() | ||||
|         ep.podcast = libraryItem.media.toJSONMinified() | ||||
|         ep.libraryItemId = libraryItem.id | ||||
|         ep.libraryId = libraryItem.libraryId | ||||
|         return ep | ||||
|       }) | ||||
|       allUnfinishedEpisodes.push(...unfinishedEpisodes) | ||||
|     } | ||||
| 
 | ||||
|     payload.total = allUnfinishedEpisodes.length | ||||
| 
 | ||||
|     allUnfinishedEpisodes = sort(allUnfinishedEpisodes).desc(ep => ep.publishedAt) | ||||
| 
 | ||||
|     if (payload.limit) { | ||||
|       var startIndex = payload.page * payload.limit | ||||
|       allUnfinishedEpisodes = allUnfinishedEpisodes.slice(startIndex, startIndex + payload.limit) | ||||
|     } | ||||
|     payload.episodes = allUnfinishedEpisodes | ||||
|     const offset = payload.page * payload.limit | ||||
|     payload.episodes = await libraryItemsPodcastFilters.getRecentEpisodes(req.user, req.library, payload.limit, offset) | ||||
|     res.json(payload) | ||||
|   } | ||||
| 
 | ||||
|   getOPMLFile(req, res) { | ||||
|     const opmlText = this.podcastManager.generateOPMLFileText(req.libraryItems) | ||||
|   /** | ||||
|    * GET: /api/libraries/:id/opml | ||||
|    * Get OPML file for a podcast library | ||||
|    * @param {import('express').Request} req  | ||||
|    * @param {import('express').Response} res  | ||||
|    */ | ||||
|   async getOPMLFile(req, res) { | ||||
|     const userPermissionPodcastWhere = libraryItemsPodcastFilters.getUserPermissionPodcastWhereQuery(req.user) | ||||
|     const podcasts = await Database.podcastModel.findAll({ | ||||
|       attributes: ['id', 'feedURL', 'title', 'description', 'itunesPageURL', 'language'], | ||||
|       where: userPermissionPodcastWhere.podcastWhere, | ||||
|       replacements: userPermissionPodcastWhere.replacements, | ||||
|       include: { | ||||
|         model: Database.libraryItemModel, | ||||
|         attributes: ['id', 'libraryId'], | ||||
|         where: { | ||||
|           libraryId: req.library.id | ||||
|         } | ||||
|       } | ||||
|     }) | ||||
| 
 | ||||
|     const opmlText = this.podcastManager.generateOPMLFileText(podcasts) | ||||
|     res.type('application/xml') | ||||
|     res.send(opmlText) | ||||
|   } | ||||
| @ -1109,7 +1041,7 @@ class LibraryController { | ||||
|       return res.sendStatus(403) | ||||
|     } | ||||
| 
 | ||||
|     const library = await Database.models.library.getOldById(req.params.id) | ||||
|     const library = await Database.libraryModel.getOldById(req.params.id) | ||||
|     if (!library) { | ||||
|       return res.status(404).send('Library not found') | ||||
|     } | ||||
| @ -1122,9 +1054,9 @@ class LibraryController { | ||||
| 
 | ||||
|   /** | ||||
|    * Middleware that is not using libraryItems from memory | ||||
|    * @param {*} req  | ||||
|    * @param {*} res  | ||||
|    * @param {*} next  | ||||
|    * @param {import('express').Request} req  | ||||
|    * @param {import('express').Response} res  | ||||
|    * @param {import('express').NextFunction} next  | ||||
|    */ | ||||
|   async middlewareNew(req, res, next) { | ||||
|     if (!req.user.checkCanAccessLibrary(req.params.id)) { | ||||
| @ -1132,7 +1064,7 @@ class LibraryController { | ||||
|       return res.sendStatus(403) | ||||
|     } | ||||
| 
 | ||||
|     const library = await Database.models.library.getOldById(req.params.id) | ||||
|     const library = await Database.libraryModel.getOldById(req.params.id) | ||||
|     if (!library) { | ||||
|       return res.status(404).send('Library not found') | ||||
|     } | ||||
|  | ||||
| @ -78,6 +78,7 @@ class LibraryItemController { | ||||
|         Logger.error(`[LibraryItemController] Failed to delete library item from file system at "${libraryItemPath}"`, error) | ||||
|       }) | ||||
|     } | ||||
|     await Database.resetLibraryIssuesFilterData(req.libraryItem.libraryId) | ||||
|     res.sendStatus(200) | ||||
|   } | ||||
| 
 | ||||
| @ -124,7 +125,7 @@ class LibraryItemController { | ||||
|     // Book specific - Get all series being removed from this item
 | ||||
|     let seriesRemoved = [] | ||||
|     if (libraryItem.isBook && mediaPayload.metadata?.series) { | ||||
|       const seriesIdsInUpdate = (mediaPayload.metadata?.series || []).map(se => se.id) | ||||
|       const seriesIdsInUpdate = mediaPayload.metadata.series?.map(se => se.id) || [] | ||||
|       seriesRemoved = libraryItem.media.metadata.series.filter(se => !seriesIdsInUpdate.includes(se.id)) | ||||
|     } | ||||
| 
 | ||||
| @ -135,7 +136,7 @@ class LibraryItemController { | ||||
|       if (seriesRemoved.length) { | ||||
|         // Check remove empty series
 | ||||
|         Logger.debug(`[LibraryItemController] Series was removed from book. Check if series is now empty.`) | ||||
|         await this.checkRemoveEmptySeries(seriesRemoved) | ||||
|         await this.checkRemoveEmptySeries(libraryItem.media.id, seriesRemoved.map(se => se.id)) | ||||
|       } | ||||
| 
 | ||||
|       if (isPodcastAutoDownloadUpdated) { | ||||
| @ -313,7 +314,7 @@ class LibraryItemController { | ||||
|       return res.status(400).send('Invalid request body') | ||||
|     } | ||||
| 
 | ||||
|     const itemsToDelete = await Database.models.libraryItem.getAllOldLibraryItems({ | ||||
|     const itemsToDelete = await Database.libraryItemModel.getAllOldLibraryItems({ | ||||
|       id: libraryItemIds | ||||
|     }) | ||||
| 
 | ||||
| @ -332,6 +333,8 @@ class LibraryItemController { | ||||
|         }) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     await Database.resetLibraryIssuesFilterData(req.libraryItem.libraryId) | ||||
|     res.sendStatus(200) | ||||
|   } | ||||
| 
 | ||||
| @ -346,13 +349,26 @@ class LibraryItemController { | ||||
| 
 | ||||
|     for (const updatePayload of updatePayloads) { | ||||
|       const mediaPayload = updatePayload.mediaPayload | ||||
|       const libraryItem = await Database.models.libraryItem.getOldById(updatePayload.id) | ||||
|       const libraryItem = await Database.libraryItemModel.getOldById(updatePayload.id) | ||||
|       if (!libraryItem) return null | ||||
| 
 | ||||
|       await this.createAuthorsAndSeriesForItemUpdate(mediaPayload, libraryItem.libraryId) | ||||
| 
 | ||||
|       let seriesRemoved = [] | ||||
|       if (libraryItem.isBook && mediaPayload.metadata?.series) { | ||||
|         const seriesIdsInUpdate = (mediaPayload.metadata?.series || []).map(se => se.id) | ||||
|         seriesRemoved = libraryItem.media.metadata.series.filter(se => !seriesIdsInUpdate.includes(se.id)) | ||||
|       } | ||||
| 
 | ||||
|       if (libraryItem.media.update(mediaPayload)) { | ||||
|         Logger.debug(`[LibraryItemController] Updated library item media ${libraryItem.media.metadata.title}`) | ||||
| 
 | ||||
|         if (seriesRemoved.length) { | ||||
|           // Check remove empty series
 | ||||
|           Logger.debug(`[LibraryItemController] Series was removed from book. Check if series is now empty.`) | ||||
|           await this.checkRemoveEmptySeries(libraryItem.media.id, seriesRemoved.map(se => se.id)) | ||||
|         } | ||||
| 
 | ||||
|         await Database.updateLibraryItem(libraryItem) | ||||
|         SocketAuthority.emitter('item_updated', libraryItem.toJSONExpanded()) | ||||
|         itemsUpdated++ | ||||
| @ -371,7 +387,7 @@ class LibraryItemController { | ||||
|     if (!libraryItemIds.length) { | ||||
|       return res.status(403).send('Invalid payload') | ||||
|     } | ||||
|     const libraryItems = await Database.models.libraryItem.getAllOldLibraryItems({ | ||||
|     const libraryItems = await Database.libraryItemModel.getAllOldLibraryItems({ | ||||
|       id: libraryItemIds | ||||
|     }) | ||||
|     res.json({ | ||||
| @ -443,9 +459,11 @@ class LibraryItemController { | ||||
|         await this.scanner.scanLibraryItemByRequest(libraryItem) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     await Database.resetLibraryIssuesFilterData(req.libraryItem.libraryId) | ||||
|   } | ||||
| 
 | ||||
|   // POST: api/items/:id/scan (admin)
 | ||||
|   // POST: api/items/:id/scan
 | ||||
|   async scan(req, res) { | ||||
|     if (!req.user.isAdminOrUp) { | ||||
|       Logger.error(`[LibraryItemController] Non-admin user attempted to scan library item`, req.user) | ||||
| @ -458,6 +476,7 @@ class LibraryItemController { | ||||
|     } | ||||
| 
 | ||||
|     const result = await this.scanner.scanLibraryItemByRequest(req.libraryItem) | ||||
|     await Database.resetLibraryIssuesFilterData(req.libraryItem.libraryId) | ||||
|     res.json({ | ||||
|       result: Object.keys(ScanResult).find(key => ScanResult[key] == result) | ||||
|     }) | ||||
| @ -681,7 +700,7 @@ class LibraryItemController { | ||||
|   } | ||||
| 
 | ||||
|   async middleware(req, res, next) { | ||||
|     req.libraryItem = await Database.models.libraryItem.getOldById(req.params.id) | ||||
|     req.libraryItem = await Database.libraryItemModel.getOldById(req.params.id) | ||||
|     if (!req.libraryItem?.media) return res.sendStatus(404) | ||||
| 
 | ||||
|     // Check user can access this library item
 | ||||
|  | ||||
| @ -59,7 +59,7 @@ class MeController { | ||||
| 
 | ||||
|   // PATCH: api/me/progress/:id
 | ||||
|   async createUpdateMediaProgress(req, res) { | ||||
|     const libraryItem = await Database.models.libraryItem.getOldById(req.params.id) | ||||
|     const libraryItem = await Database.libraryItemModel.getOldById(req.params.id) | ||||
|     if (!libraryItem) { | ||||
|       return res.status(404).send('Item not found') | ||||
|     } | ||||
| @ -75,7 +75,7 @@ class MeController { | ||||
|   // PATCH: api/me/progress/:id/:episodeId
 | ||||
|   async createUpdateEpisodeMediaProgress(req, res) { | ||||
|     const episodeId = req.params.episodeId | ||||
|     const libraryItem = await Database.models.libraryItem.getOldById(req.params.id) | ||||
|     const libraryItem = await Database.libraryItemModel.getOldById(req.params.id) | ||||
|     if (!libraryItem) { | ||||
|       return res.status(404).send('Item not found') | ||||
|     } | ||||
| @ -101,7 +101,7 @@ class MeController { | ||||
| 
 | ||||
|     let shouldUpdate = false | ||||
|     for (const itemProgress of itemProgressPayloads) { | ||||
|       const libraryItem = await Database.models.libraryItem.getOldById(itemProgress.libraryItemId) | ||||
|       const libraryItem = await Database.libraryItemModel.getOldById(itemProgress.libraryItemId) | ||||
|       if (libraryItem) { | ||||
|         if (req.user.createUpdateMediaProgress(libraryItem, itemProgress, itemProgress.episodeId)) { | ||||
|           const mediaProgress = req.user.getMediaProgress(libraryItem.id, itemProgress.episodeId) | ||||
| @ -122,7 +122,7 @@ class MeController { | ||||
| 
 | ||||
|   // POST: api/me/item/:id/bookmark
 | ||||
|   async createBookmark(req, res) { | ||||
|     if (!await Database.models.libraryItem.checkExistsById(req.params.id)) return res.sendStatus(404) | ||||
|     if (!await Database.libraryItemModel.checkExistsById(req.params.id)) return res.sendStatus(404) | ||||
| 
 | ||||
|     const { time, title } = req.body | ||||
|     const bookmark = req.user.createBookmark(req.params.id, time, title) | ||||
| @ -133,7 +133,7 @@ class MeController { | ||||
| 
 | ||||
|   // PATCH: api/me/item/:id/bookmark
 | ||||
|   async updateBookmark(req, res) { | ||||
|     if (!await Database.models.libraryItem.checkExistsById(req.params.id)) return res.sendStatus(404) | ||||
|     if (!await Database.libraryItemModel.checkExistsById(req.params.id)) return res.sendStatus(404) | ||||
| 
 | ||||
|     const { time, title } = req.body | ||||
|     if (!req.user.findBookmark(req.params.id, time)) { | ||||
| @ -151,7 +151,7 @@ class MeController { | ||||
| 
 | ||||
|   // DELETE: api/me/item/:id/bookmark/:time
 | ||||
|   async removeBookmark(req, res) { | ||||
|     if (!await Database.models.libraryItem.checkExistsById(req.params.id)) return res.sendStatus(404) | ||||
|     if (!await Database.libraryItemModel.checkExistsById(req.params.id)) return res.sendStatus(404) | ||||
| 
 | ||||
|     const time = Number(req.params.time) | ||||
|     if (isNaN(time)) return res.sendStatus(500) | ||||
|  | ||||
| @ -38,7 +38,7 @@ class MiscController { | ||||
|     const libraryId = req.body.library | ||||
|     const folderId = req.body.folder | ||||
| 
 | ||||
|     const library = await Database.models.library.getOldById(libraryId) | ||||
|     const library = await Database.libraryModel.getOldById(libraryId) | ||||
|     if (!library) { | ||||
|       return res.status(404).send(`Library not found with id ${libraryId}`) | ||||
|     } | ||||
| @ -177,7 +177,7 @@ class MiscController { | ||||
|     } | ||||
| 
 | ||||
|     const tags = [] | ||||
|     const books = await Database.models.book.findAll({ | ||||
|     const books = await Database.bookModel.findAll({ | ||||
|       attributes: ['tags'], | ||||
|       where: Sequelize.where(Sequelize.fn('json_array_length', Sequelize.col('tags')), { | ||||
|         [Sequelize.Op.gt]: 0 | ||||
| @ -189,7 +189,7 @@ class MiscController { | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     const podcasts = await Database.models.podcast.findAll({ | ||||
|     const podcasts = await Database.podcastModel.findAll({ | ||||
|       attributes: ['tags'], | ||||
|       where: Sequelize.where(Sequelize.fn('json_array_length', Sequelize.col('tags')), { | ||||
|         [Sequelize.Op.gt]: 0 | ||||
| @ -248,7 +248,7 @@ class MiscController { | ||||
|         await libraryItem.media.update({ | ||||
|           tags: libraryItem.media.tags | ||||
|         }) | ||||
|         const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(libraryItem) | ||||
|         const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(libraryItem) | ||||
|         SocketAuthority.emitter('item_updated', oldLibraryItem.toJSONExpanded()) | ||||
|         numItemsUpdated++ | ||||
|       } | ||||
| @ -289,7 +289,7 @@ class MiscController { | ||||
|       await libraryItem.media.update({ | ||||
|         tags: libraryItem.media.tags | ||||
|       }) | ||||
|       const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(libraryItem) | ||||
|       const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(libraryItem) | ||||
|       SocketAuthority.emitter('item_updated', oldLibraryItem.toJSONExpanded()) | ||||
|       numItemsUpdated++ | ||||
|     } | ||||
| @ -311,7 +311,7 @@ class MiscController { | ||||
|       return res.sendStatus(404) | ||||
|     } | ||||
|     const genres = [] | ||||
|     const books = await Database.models.book.findAll({ | ||||
|     const books = await Database.bookModel.findAll({ | ||||
|       attributes: ['genres'], | ||||
|       where: Sequelize.where(Sequelize.fn('json_array_length', Sequelize.col('genres')), { | ||||
|         [Sequelize.Op.gt]: 0 | ||||
| @ -323,7 +323,7 @@ class MiscController { | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     const podcasts = await Database.models.podcast.findAll({ | ||||
|     const podcasts = await Database.podcastModel.findAll({ | ||||
|       attributes: ['genres'], | ||||
|       where: Sequelize.where(Sequelize.fn('json_array_length', Sequelize.col('genres')), { | ||||
|         [Sequelize.Op.gt]: 0 | ||||
| @ -382,7 +382,7 @@ class MiscController { | ||||
|         await libraryItem.media.update({ | ||||
|           genres: libraryItem.media.genres | ||||
|         }) | ||||
|         const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(libraryItem) | ||||
|         const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(libraryItem) | ||||
|         SocketAuthority.emitter('item_updated', oldLibraryItem.toJSONExpanded()) | ||||
|         numItemsUpdated++ | ||||
|       } | ||||
| @ -423,7 +423,7 @@ class MiscController { | ||||
|       await libraryItem.media.update({ | ||||
|         genres: libraryItem.media.genres | ||||
|       }) | ||||
|       const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(libraryItem) | ||||
|       const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(libraryItem) | ||||
|       SocketAuthority.emitter('item_updated', oldLibraryItem.toJSONExpanded()) | ||||
|       numItemsUpdated++ | ||||
|     } | ||||
|  | ||||
| @ -22,11 +22,11 @@ class PlaylistController { | ||||
|     } | ||||
| 
 | ||||
|     // Create Playlist record
 | ||||
|     const newPlaylist = await Database.models.playlist.createFromOld(oldPlaylist) | ||||
|     const newPlaylist = await Database.playlistModel.createFromOld(oldPlaylist) | ||||
| 
 | ||||
|     // Lookup all library items in playlist
 | ||||
|     const libraryItemIds = oldPlaylist.items.map(i => i.libraryItemId).filter(i => i) | ||||
|     const libraryItemsInPlaylist = await Database.models.libraryItem.findAll({ | ||||
|     const libraryItemsInPlaylist = await Database.libraryItemModel.findAll({ | ||||
|       where: { | ||||
|         id: libraryItemIds | ||||
|       } | ||||
| @ -62,7 +62,7 @@ class PlaylistController { | ||||
|    * @param {*} res  | ||||
|    */ | ||||
|   async findAllForUser(req, res) { | ||||
|     const playlistsForUser = await Database.models.playlist.findAll({ | ||||
|     const playlistsForUser = await Database.playlistModel.findAll({ | ||||
|       where: { | ||||
|         userId: req.user.id | ||||
|       } | ||||
| @ -106,7 +106,7 @@ class PlaylistController { | ||||
|     // If array of items is passed in then update order of playlist media items
 | ||||
|     const libraryItemIds = req.body.items?.map(i => i.libraryItemId).filter(i => i) || [] | ||||
|     if (libraryItemIds.length) { | ||||
|       const libraryItems = await Database.models.libraryItem.findAll({ | ||||
|       const libraryItems = await Database.libraryItemModel.findAll({ | ||||
|         where: { | ||||
|           id: libraryItemIds | ||||
|         } | ||||
| @ -173,14 +173,14 @@ class PlaylistController { | ||||
|    * @param {*} res  | ||||
|    */ | ||||
|   async addItem(req, res) { | ||||
|     const oldPlaylist = await Database.models.playlist.getById(req.playlist.id) | ||||
|     const oldPlaylist = await Database.playlistModel.getById(req.playlist.id) | ||||
|     const itemToAdd = req.body | ||||
| 
 | ||||
|     if (!itemToAdd.libraryItemId) { | ||||
|       return res.status(400).send('Request body has no libraryItemId') | ||||
|     } | ||||
| 
 | ||||
|     const libraryItem = await Database.models.libraryItem.getOldById(itemToAdd.libraryItemId) | ||||
|     const libraryItem = await Database.libraryItemModel.getOldById(itemToAdd.libraryItemId) | ||||
|     if (!libraryItem) { | ||||
|       return res.status(400).send('Library item not found') | ||||
|     } | ||||
| @ -217,7 +217,7 @@ class PlaylistController { | ||||
|    * @param {*} res  | ||||
|    */ | ||||
|   async removeItem(req, res) { | ||||
|     const oldLibraryItem = await Database.models.libraryItem.getOldById(req.params.libraryItemId) | ||||
|     const oldLibraryItem = await Database.libraryItemModel.getOldById(req.params.libraryItemId) | ||||
|     if (!oldLibraryItem) { | ||||
|       return res.status(404).send('Library item not found') | ||||
|     } | ||||
| @ -281,7 +281,7 @@ class PlaylistController { | ||||
|     } | ||||
| 
 | ||||
|     // Find all library items
 | ||||
|     const libraryItems = await Database.models.libraryItem.findAll({ | ||||
|     const libraryItems = await Database.libraryItemModel.findAll({ | ||||
|       where: { | ||||
|         id: libraryItemIds | ||||
|       } | ||||
| @ -345,7 +345,7 @@ class PlaylistController { | ||||
|     } | ||||
| 
 | ||||
|     // Find all library items
 | ||||
|     const libraryItems = await Database.models.libraryItem.findAll({ | ||||
|     const libraryItems = await Database.libraryItemModel.findAll({ | ||||
|       where: { | ||||
|         id: libraryItemIds | ||||
|       } | ||||
| @ -391,7 +391,7 @@ class PlaylistController { | ||||
|    * @param {*} res  | ||||
|    */ | ||||
|   async createFromCollection(req, res) { | ||||
|     const collection = await Database.models.collection.findByPk(req.params.collectionId) | ||||
|     const collection = await Database.collectionModel.findByPk(req.params.collectionId) | ||||
|     if (!collection) { | ||||
|       return res.status(404).send('Collection not found') | ||||
|     } | ||||
| @ -416,7 +416,7 @@ class PlaylistController { | ||||
|     }) | ||||
| 
 | ||||
|     // Create Playlist record
 | ||||
|     const newPlaylist = await Database.models.playlist.createFromOld(oldPlaylist) | ||||
|     const newPlaylist = await Database.playlistModel.createFromOld(oldPlaylist) | ||||
| 
 | ||||
|     // Create PlaylistMediaItem records
 | ||||
|     const mediaItemsToAdd = [] | ||||
| @ -438,7 +438,7 @@ class PlaylistController { | ||||
| 
 | ||||
|   async middleware(req, res, next) { | ||||
|     if (req.params.id) { | ||||
|       const playlist = await Database.models.playlist.findByPk(req.params.id) | ||||
|       const playlist = await Database.playlistModel.findByPk(req.params.id) | ||||
|       if (!playlist) { | ||||
|         return res.status(404).send('Playlist not found') | ||||
|       } | ||||
|  | ||||
| @ -19,7 +19,7 @@ class PodcastController { | ||||
|     } | ||||
|     const payload = req.body | ||||
| 
 | ||||
|     const library = await Database.models.library.getOldById(payload.libraryId) | ||||
|     const library = await Database.libraryModel.getOldById(payload.libraryId) | ||||
|     if (!library) { | ||||
|       Logger.error(`[PodcastController] Create: Library not found "${payload.libraryId}"`) | ||||
|       return res.status(404).send('Library not found') | ||||
| @ -34,7 +34,7 @@ class PodcastController { | ||||
|     const podcastPath = filePathToPOSIX(payload.path) | ||||
| 
 | ||||
|     // Check if a library item with this podcast folder exists already
 | ||||
|     const existingLibraryItem = (await Database.models.libraryItem.count({ | ||||
|     const existingLibraryItem = (await Database.libraryItemModel.count({ | ||||
|       where: { | ||||
|         path: podcastPath | ||||
|       } | ||||
| @ -272,13 +272,13 @@ class PodcastController { | ||||
|     } | ||||
| 
 | ||||
|     // Update/remove playlists that had this podcast episode
 | ||||
|     const playlistMediaItems = await Database.models.playlistMediaItem.findAll({ | ||||
|     const playlistMediaItems = await Database.playlistMediaItemModel.findAll({ | ||||
|       where: { | ||||
|         mediaItemId: episodeId | ||||
|       }, | ||||
|       include: { | ||||
|         model: Database.models.playlist, | ||||
|         include: Database.models.playlistMediaItem | ||||
|         model: Database.playlistModel, | ||||
|         include: Database.playlistMediaItemModel | ||||
|       } | ||||
|     }) | ||||
|     for (const pmi of playlistMediaItems) { | ||||
| @ -297,7 +297,7 @@ class PodcastController { | ||||
|     } | ||||
| 
 | ||||
|     // Remove media progress for this episode
 | ||||
|     const mediaProgressRemoved = await Database.models.mediaProgress.destroy({ | ||||
|     const mediaProgressRemoved = await Database.mediaProgressModel.destroy({ | ||||
|       where: { | ||||
|         mediaItemId: episode.id | ||||
|       } | ||||
| @ -312,7 +312,7 @@ class PodcastController { | ||||
|   } | ||||
| 
 | ||||
|   async middleware(req, res, next) { | ||||
|     const item = await Database.models.libraryItem.getOldById(req.params.id) | ||||
|     const item = await Database.libraryItemModel.getOldById(req.params.id) | ||||
|     if (!item?.media) return res.sendStatus(404) | ||||
| 
 | ||||
|     if (!item.isPodcast) { | ||||
|  | ||||
| @ -9,7 +9,7 @@ class RSSFeedController { | ||||
|   async openRSSFeedForItem(req, res) { | ||||
|     const options = req.body || {} | ||||
| 
 | ||||
|     const item = await Database.models.libraryItem.getOldById(req.params.itemId) | ||||
|     const item = await Database.libraryItemModel.getOldById(req.params.itemId) | ||||
|     if (!item) return res.sendStatus(404) | ||||
| 
 | ||||
|     // Check user can access this library item
 | ||||
| @ -46,7 +46,7 @@ class RSSFeedController { | ||||
|   async openRSSFeedForCollection(req, res) { | ||||
|     const options = req.body || {} | ||||
| 
 | ||||
|     const collection = await Database.models.collection.findByPk(req.params.collectionId) | ||||
|     const collection = await Database.collectionModel.findByPk(req.params.collectionId) | ||||
|     if (!collection) return res.sendStatus(404) | ||||
| 
 | ||||
|     // Check request body options exist
 | ||||
|  | ||||
| @ -49,7 +49,7 @@ class SessionController { | ||||
|       return res.sendStatus(404) | ||||
|     } | ||||
| 
 | ||||
|     const minifiedUserObjects = await Database.models.user.getMinifiedUserObjects() | ||||
|     const minifiedUserObjects = await Database.userModel.getMinifiedUserObjects() | ||||
|     const openSessions = this.playbackSessionManager.sessions.map(se => { | ||||
|       return { | ||||
|         ...se.toJSON(), | ||||
|  | ||||
| @ -106,7 +106,7 @@ class ToolsController { | ||||
|     } | ||||
| 
 | ||||
|     if (req.params.id) { | ||||
|       const item = await Database.models.libraryItem.getOldById(req.params.id) | ||||
|       const item = await Database.libraryItemModel.getOldById(req.params.id) | ||||
|       if (!item?.media) return res.sendStatus(404) | ||||
| 
 | ||||
|       // Check user can access this library item
 | ||||
|  | ||||
| @ -17,7 +17,7 @@ class UserController { | ||||
|     const includes = (req.query.include || '').split(',').map(i => i.trim()) | ||||
| 
 | ||||
|     // Minimal toJSONForBrowser does not include mediaProgress and bookmarks
 | ||||
|     const allUsers = await Database.models.user.getOldUsers() | ||||
|     const allUsers = await Database.userModel.getOldUsers() | ||||
|     const users = allUsers.map(u => u.toJSONForBrowser(hideRootToken, true)) | ||||
| 
 | ||||
|     if (includes.includes('latestSession')) { | ||||
| @ -32,20 +32,67 @@ class UserController { | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * GET: /api/users/:id | ||||
|    * Get a single user toJSONForBrowser | ||||
|    * Media progress items include: `displayTitle`, `displaySubtitle` (for podcasts), `coverPath` and `mediaUpdatedAt` | ||||
|    *  | ||||
|    * @param {import("express").Request} req  | ||||
|    * @param {import("express").Response} res  | ||||
|    */ | ||||
|   async findOne(req, res) { | ||||
|     if (!req.user.isAdminOrUp) { | ||||
|       Logger.error('User other than admin attempting to get user', req.user) | ||||
|       return res.sendStatus(403) | ||||
|     } | ||||
| 
 | ||||
|     res.json(this.userJsonWithItemProgressDetails(req.reqUser, !req.user.isRoot)) | ||||
|     // Get user media progress with associated mediaItem
 | ||||
|     const mediaProgresses = await Database.mediaProgressModel.findAll({ | ||||
|       where: { | ||||
|         userId: req.reqUser.id | ||||
|       }, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.bookModel, | ||||
|           attributes: ['id', 'title', 'coverPath', 'updatedAt'] | ||||
|         }, | ||||
|         { | ||||
|           model: Database.podcastEpisodeModel, | ||||
|           attributes: ['id', 'title'], | ||||
|           include: { | ||||
|             model: Database.podcastModel, | ||||
|             attributes: ['id', 'title', 'coverPath', 'updatedAt'] | ||||
|           } | ||||
|         } | ||||
|       ] | ||||
|     }) | ||||
| 
 | ||||
|     const oldMediaProgresses = mediaProgresses.map(mp => { | ||||
|       const oldMediaProgress = mp.getOldMediaProgress() | ||||
|       oldMediaProgress.displayTitle = mp.mediaItem?.title | ||||
|       if (mp.mediaItem?.podcast) { | ||||
|         oldMediaProgress.displaySubtitle = mp.mediaItem.podcast?.title | ||||
|         oldMediaProgress.coverPath = mp.mediaItem.podcast?.coverPath | ||||
|         oldMediaProgress.mediaUpdatedAt = mp.mediaItem.podcast?.updatedAt | ||||
|       } else if (mp.mediaItem) { | ||||
|         oldMediaProgress.coverPath = mp.mediaItem.coverPath | ||||
|         oldMediaProgress.mediaUpdatedAt = mp.mediaItem.updatedAt | ||||
|       } | ||||
|       return oldMediaProgress | ||||
|     }) | ||||
| 
 | ||||
|     const userJson = req.reqUser.toJSONForBrowser(!req.user.isRoot) | ||||
| 
 | ||||
|     userJson.mediaProgress = oldMediaProgresses | ||||
| 
 | ||||
|     res.json(userJson) | ||||
|   } | ||||
| 
 | ||||
|   async create(req, res) { | ||||
|     const account = req.body | ||||
|     const username = account.username | ||||
| 
 | ||||
|     const usernameExists = await Database.models.user.getUserByUsername(username) | ||||
|     const usernameExists = await Database.userModel.getUserByUsername(username) | ||||
|     if (usernameExists) { | ||||
|       return res.status(500).send('Username already taken') | ||||
|     } | ||||
| @ -80,7 +127,7 @@ class UserController { | ||||
|     var shouldUpdateToken = false | ||||
| 
 | ||||
|     if (account.username !== undefined && account.username !== user.username) { | ||||
|       const usernameExists = await Database.models.user.getUserByUsername(account.username) | ||||
|       const usernameExists = await Database.userModel.getUserByUsername(account.username) | ||||
|       if (usernameExists) { | ||||
|         return res.status(500).send('Username already taken') | ||||
|       } | ||||
| @ -122,7 +169,7 @@ class UserController { | ||||
|     // Todo: check if user is logged in and cancel streams
 | ||||
| 
 | ||||
|     // Remove user playlists
 | ||||
|     const userPlaylists = await Database.models.playlist.findAll({ | ||||
|     const userPlaylists = await Database.playlistModel.findAll({ | ||||
|       where: { | ||||
|         userId: user.id | ||||
|       } | ||||
| @ -186,7 +233,7 @@ class UserController { | ||||
|     } | ||||
| 
 | ||||
|     if (req.params.id) { | ||||
|       req.reqUser = await Database.models.user.getUserById(req.params.id) | ||||
|       req.reqUser = await Database.userModel.getUserById(req.params.id) | ||||
|       if (!req.reqUser) { | ||||
|         return res.sendStatus(404) | ||||
|       } | ||||
|  | ||||
| @ -5,23 +5,23 @@ const { Sequelize } = require('sequelize') | ||||
| const Database = require('../Database') | ||||
| 
 | ||||
| const getLibraryItemMinified = (libraryItemId) => { | ||||
|   return Database.models.libraryItem.findByPk(libraryItemId, { | ||||
|   return Database.libraryItemModel.findByPk(libraryItemId, { | ||||
|     include: [ | ||||
|       { | ||||
|         model: Database.models.book, | ||||
|         model: Database.bookModel, | ||||
|         attributes: [ | ||||
|           'id', 'title', 'subtitle', 'publishedYear', 'publishedDate', 'publisher', 'description', 'isbn', 'asin', 'language', 'explicit', 'narrators', 'coverPath', 'genres', 'tags' | ||||
|         ], | ||||
|         include: [ | ||||
|           { | ||||
|             model: Database.models.author, | ||||
|             model: Database.authorModel, | ||||
|             attributes: ['id', 'name'], | ||||
|             through: { | ||||
|               attributes: [] | ||||
|             } | ||||
|           }, | ||||
|           { | ||||
|             model: Database.models.series, | ||||
|             model: Database.seriesModel, | ||||
|             attributes: ['id', 'name'], | ||||
|             through: { | ||||
|               attributes: ['sequence'] | ||||
| @ -30,7 +30,7 @@ const getLibraryItemMinified = (libraryItemId) => { | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         model: Database.models.podcast, | ||||
|         model: Database.podcastModel, | ||||
|         attributes: [ | ||||
|           'id', 'title', 'author', 'releaseDate', 'feedURL', 'imageURL', 'description', 'itunesPageURL', 'itunesId', 'itunesArtistId', 'language', 'podcastType', 'explicit', 'autoDownloadEpisodes', 'genres', 'tags', | ||||
|           [Sequelize.literal('(SELECT COUNT(*) FROM "podcastEpisodes" WHERE "podcastEpisodes"."podcastId" = podcast.id)'), 'numPodcastEpisodes'] | ||||
| @ -41,19 +41,19 @@ const getLibraryItemMinified = (libraryItemId) => { | ||||
| } | ||||
| 
 | ||||
| const getLibraryItemExpanded = (libraryItemId) => { | ||||
|   return Database.models.libraryItem.findByPk(libraryItemId, { | ||||
|   return Database.libraryItemModel.findByPk(libraryItemId, { | ||||
|     include: [ | ||||
|       { | ||||
|         model: Database.models.book, | ||||
|         model: Database.bookModel, | ||||
|         include: [ | ||||
|           { | ||||
|             model: Database.models.author, | ||||
|             model: Database.authorModel, | ||||
|             through: { | ||||
|               attributes: [] | ||||
|             } | ||||
|           }, | ||||
|           { | ||||
|             model: Database.models.series, | ||||
|             model: Database.seriesModel, | ||||
|             through: { | ||||
|               attributes: ['sequence'] | ||||
|             } | ||||
| @ -61,10 +61,10 @@ const getLibraryItemExpanded = (libraryItemId) => { | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
|         model: Database.models.podcast, | ||||
|         model: Database.podcastModel, | ||||
|         include: [ | ||||
|           { | ||||
|             model: Database.models.podcastEpisode | ||||
|             model: Database.podcastEpisodeModel | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|  | ||||
| @ -77,7 +77,7 @@ class CronManager { | ||||
|   async initPodcastCrons() { | ||||
|     const cronExpressionMap = {} | ||||
| 
 | ||||
|     const podcastsWithAutoDownload = await Database.models.podcast.findAll({ | ||||
|     const podcastsWithAutoDownload = await Database.podcastModel.findAll({ | ||||
|       where: { | ||||
|         autoDownloadEpisodes: true, | ||||
|         autoDownloadSchedule: { | ||||
| @ -85,7 +85,7 @@ class CronManager { | ||||
|         } | ||||
|       }, | ||||
|       include: { | ||||
|         model: Database.models.libraryItem | ||||
|         model: Database.libraryItemModel | ||||
|       } | ||||
|     }) | ||||
| 
 | ||||
| @ -139,7 +139,7 @@ class CronManager { | ||||
|     // Get podcast library items to check
 | ||||
|     const libraryItems = [] | ||||
|     for (const libraryItemId of libraryItemIds) { | ||||
|       const libraryItem = await Database.models.libraryItem.getOldById(libraryItemId) | ||||
|       const libraryItem = await Database.libraryItemModel.getOldById(libraryItemId) | ||||
|       if (!libraryItem) { | ||||
|         Logger.error(`[CronManager] Library item ${libraryItemId} not found for episode check cron ${expression}`) | ||||
|         podcastCron.libraryItemIds = podcastCron.libraryItemIds.filter(lid => lid !== libraryItemId) // Filter it out
 | ||||
|  | ||||
| @ -18,7 +18,7 @@ class NotificationManager { | ||||
|     if (!Database.notificationSettings.isUseable) return | ||||
| 
 | ||||
|     Logger.debug(`[NotificationManager] onPodcastEpisodeDownloaded: Episode "${episode.title}" for podcast ${libraryItem.media.metadata.title}`) | ||||
|     const library = await Database.models.library.getOldById(libraryItem.libraryId) | ||||
|     const library = await Database.libraryModel.getOldById(libraryItem.libraryId) | ||||
|     const eventData = { | ||||
|       libraryItemId: libraryItem.id, | ||||
|       libraryId: libraryItem.libraryId, | ||||
|  | ||||
| @ -265,7 +265,7 @@ class PlaybackSessionManager { | ||||
|   } | ||||
| 
 | ||||
|   async syncSession(user, session, syncData) { | ||||
|     const libraryItem = await Database.models.libraryItem.getOldById(session.libraryItemId) | ||||
|     const libraryItem = await Database.libraryItemModel.getOldById(session.libraryItemId) | ||||
|     if (!libraryItem) { | ||||
|       Logger.error(`[PlaybackSessionManager] syncSession Library Item not found "${session.libraryItemId}"`) | ||||
|       return null | ||||
|  | ||||
| @ -150,7 +150,7 @@ class PodcastManager { | ||||
|       return false | ||||
|     } | ||||
| 
 | ||||
|     const libraryItem = await Database.models.libraryItem.getOldById(this.currentDownload.libraryItem.id) | ||||
|     const libraryItem = await Database.libraryItemModel.getOldById(this.currentDownload.libraryItem.id) | ||||
|     if (!libraryItem) { | ||||
|       Logger.error(`[PodcastManager] Podcast Episode finished but library item was not found ${this.currentDownload.libraryItem.id}`) | ||||
|       return false | ||||
| @ -372,8 +372,13 @@ class PodcastManager { | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   generateOPMLFileText(libraryItems) { | ||||
|     return opmlGenerator.generate(libraryItems) | ||||
|   /** | ||||
|    * OPML file string for podcasts in a library | ||||
|    * @param {import('../models/Podcast')[]} podcasts  | ||||
|    * @returns {string} XML string | ||||
|    */ | ||||
|   generateOPMLFileText(podcasts) { | ||||
|     return opmlGenerator.generate(podcasts) | ||||
|   } | ||||
| 
 | ||||
|   getDownloadQueueDetails(libraryId = null) { | ||||
|  | ||||
| @ -13,13 +13,13 @@ class RssFeedManager { | ||||
| 
 | ||||
|   async validateFeedEntity(feedObj) { | ||||
|     if (feedObj.entityType === 'collection') { | ||||
|       const collection = await Database.models.collection.getOldById(feedObj.entityId) | ||||
|       const collection = await Database.collectionModel.getOldById(feedObj.entityId) | ||||
|       if (!collection) { | ||||
|         Logger.error(`[RssFeedManager] Removing feed "${feedObj.id}". Collection "${feedObj.entityId}" not found`) | ||||
|         return false | ||||
|       } | ||||
|     } else if (feedObj.entityType === 'libraryItem') { | ||||
|       const libraryItemExists = await Database.models.libraryItem.checkExistsById(feedObj.entityId) | ||||
|       const libraryItemExists = await Database.libraryItemModel.checkExistsById(feedObj.entityId) | ||||
|       if (!libraryItemExists) { | ||||
|         Logger.error(`[RssFeedManager] Removing feed "${feedObj.id}". Library item "${feedObj.entityId}" not found`) | ||||
|         return false | ||||
| @ -41,7 +41,7 @@ class RssFeedManager { | ||||
|    * Validate all feeds and remove invalid | ||||
|    */ | ||||
|   async init() { | ||||
|     const feeds = await Database.models.feed.getOldFeeds() | ||||
|     const feeds = await Database.feedModel.getOldFeeds() | ||||
|     for (const feed of feeds) { | ||||
|       // Remove invalid feeds
 | ||||
|       if (!await this.validateFeedEntity(feed)) { | ||||
| @ -56,7 +56,7 @@ class RssFeedManager { | ||||
|    * @returns {Promise<objects.Feed>} oldFeed | ||||
|    */ | ||||
|   findFeedForEntityId(entityId) { | ||||
|     return Database.models.feed.findOneOld({ entityId }) | ||||
|     return Database.feedModel.findOneOld({ entityId }) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
| @ -65,7 +65,7 @@ class RssFeedManager { | ||||
|    * @returns {Promise<objects.Feed>} oldFeed | ||||
|    */ | ||||
|   findFeedBySlug(slug) { | ||||
|     return Database.models.feed.findOneOld({ slug }) | ||||
|     return Database.feedModel.findOneOld({ slug }) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
| @ -74,7 +74,7 @@ class RssFeedManager { | ||||
|    * @returns {Promise<objects.Feed>} oldFeed | ||||
|    */ | ||||
|   findFeed(id) { | ||||
|     return Database.models.feed.findByPkOld(id) | ||||
|     return Database.feedModel.findByPkOld(id) | ||||
|   } | ||||
| 
 | ||||
|   async getFeed(req, res) { | ||||
| @ -103,7 +103,7 @@ class RssFeedManager { | ||||
|         await Database.updateFeed(feed) | ||||
|       } | ||||
|     } else if (feed.entityType === 'collection') { | ||||
|       const collection = await Database.models.collection.findByPk(feed.entityId) | ||||
|       const collection = await Database.collectionModel.findByPk(feed.entityId) | ||||
|       if (collection) { | ||||
|         const collectionExpanded = await collection.getOldJsonExpanded() | ||||
| 
 | ||||
|  | ||||
| @ -83,6 +83,15 @@ class Author extends Model { | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Check if author exists | ||||
|    * @param {string} authorId  | ||||
|    * @returns {Promise<boolean>} | ||||
|    */ | ||||
|   static async checkExistsById(authorId) { | ||||
|     return (await this.count({ where: { id: authorId } })) > 0 | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|  | ||||
| @ -1,178 +1,231 @@ | ||||
| const { DataTypes, Model } = require('sequelize') | ||||
| const Logger = require('../Logger') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class Book extends Model { | ||||
|     static getOldBook(libraryItemExpanded) { | ||||
|       const bookExpanded = libraryItemExpanded.media | ||||
|       let authors = [] | ||||
|       if (bookExpanded.authors?.length) { | ||||
|         authors = bookExpanded.authors.map(au => { | ||||
|           return { | ||||
|             id: au.id, | ||||
|             name: au.name | ||||
|           } | ||||
|         }) | ||||
|       } else if (bookExpanded.bookAuthors?.length) { | ||||
|         authors = bookExpanded.bookAuthors.map(ba => { | ||||
|           if (ba.author) { | ||||
|             return { | ||||
|               id: ba.author.id, | ||||
|               name: ba.author.name | ||||
|             } | ||||
|           } else { | ||||
|             Logger.error(`[Book] Invalid bookExpanded bookAuthors: no author`, ba) | ||||
|             return null | ||||
|           } | ||||
|         }).filter(a => a) | ||||
|       } | ||||
| class Book extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|       let series = [] | ||||
|       if (bookExpanded.series?.length) { | ||||
|         series = bookExpanded.series.map(se => { | ||||
|           return { | ||||
|             id: se.id, | ||||
|             name: se.name, | ||||
|             sequence: se.bookSeries.sequence | ||||
|           } | ||||
|         }) | ||||
|       } else if (bookExpanded.bookSeries?.length) { | ||||
|         series = bookExpanded.bookSeries.map(bs => { | ||||
|           if (bs.series) { | ||||
|             return { | ||||
|               id: bs.series.id, | ||||
|               name: bs.series.name, | ||||
|               sequence: bs.sequence | ||||
|             } | ||||
|           } else { | ||||
|             Logger.error(`[Book] Invalid bookExpanded bookSeries: no series`, bs) | ||||
|             return null | ||||
|           } | ||||
|         }).filter(s => s) | ||||
|       } | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {string} */ | ||||
|     this.title | ||||
|     /** @type {string} */ | ||||
|     this.titleIgnorePrefix | ||||
|     /** @type {string} */ | ||||
|     this.publishedYear | ||||
|     /** @type {string} */ | ||||
|     this.publishedDate | ||||
|     /** @type {string} */ | ||||
|     this.publisher | ||||
|     /** @type {string} */ | ||||
|     this.description | ||||
|     /** @type {string} */ | ||||
|     this.isbn | ||||
|     /** @type {string} */ | ||||
|     this.asin | ||||
|     /** @type {string} */ | ||||
|     this.language | ||||
|     /** @type {boolean} */ | ||||
|     this.explicit | ||||
|     /** @type {boolean} */ | ||||
|     this.abridged | ||||
|     /** @type {string} */ | ||||
|     this.coverPath | ||||
|     /** @type {number} */ | ||||
|     this.duration | ||||
|     /** @type {Object} */ | ||||
|     this.narrators | ||||
|     /** @type {Object} */ | ||||
|     this.audioFiles | ||||
|     /** @type {Object} */ | ||||
|     this.ebookFile | ||||
|     /** @type {Object} */ | ||||
|     this.chapters | ||||
|     /** @type {Object} */ | ||||
|     this.tags | ||||
|     /** @type {Object} */ | ||||
|     this.genres | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|   } | ||||
| 
 | ||||
|       return { | ||||
|         id: bookExpanded.id, | ||||
|         libraryItemId: libraryItemExpanded.id, | ||||
|         coverPath: bookExpanded.coverPath, | ||||
|         tags: bookExpanded.tags, | ||||
|         audioFiles: bookExpanded.audioFiles, | ||||
|         chapters: bookExpanded.chapters, | ||||
|         ebookFile: bookExpanded.ebookFile, | ||||
|         metadata: { | ||||
|           title: bookExpanded.title, | ||||
|           subtitle: bookExpanded.subtitle, | ||||
|           authors: authors, | ||||
|           narrators: bookExpanded.narrators, | ||||
|           series: series, | ||||
|           genres: bookExpanded.genres, | ||||
|           publishedYear: bookExpanded.publishedYear, | ||||
|           publishedDate: bookExpanded.publishedDate, | ||||
|           publisher: bookExpanded.publisher, | ||||
|           description: bookExpanded.description, | ||||
|           isbn: bookExpanded.isbn, | ||||
|           asin: bookExpanded.asin, | ||||
|           language: bookExpanded.language, | ||||
|           explicit: bookExpanded.explicit, | ||||
|           abridged: bookExpanded.abridged | ||||
|   static getOldBook(libraryItemExpanded) { | ||||
|     const bookExpanded = libraryItemExpanded.media | ||||
|     let authors = [] | ||||
|     if (bookExpanded.authors?.length) { | ||||
|       authors = bookExpanded.authors.map(au => { | ||||
|         return { | ||||
|           id: au.id, | ||||
|           name: au.name | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * @param {object} oldBook  | ||||
|      * @returns {boolean} true if updated | ||||
|      */ | ||||
|     static saveFromOld(oldBook) { | ||||
|       const book = this.getFromOld(oldBook) | ||||
|       return this.update(book, { | ||||
|         where: { | ||||
|           id: book.id | ||||
|         } | ||||
|       }).then(result => result[0] > 0).catch((error) => { | ||||
|         Logger.error(`[Book] Failed to save book ${book.id}`, error) | ||||
|         return false | ||||
|       }) | ||||
|     } else if (bookExpanded.bookAuthors?.length) { | ||||
|       authors = bookExpanded.bookAuthors.map(ba => { | ||||
|         if (ba.author) { | ||||
|           return { | ||||
|             id: ba.author.id, | ||||
|             name: ba.author.name | ||||
|           } | ||||
|         } else { | ||||
|           Logger.error(`[Book] Invalid bookExpanded bookAuthors: no author`, ba) | ||||
|           return null | ||||
|         } | ||||
|       }).filter(a => a) | ||||
|     } | ||||
| 
 | ||||
|     static getFromOld(oldBook) { | ||||
|       return { | ||||
|         id: oldBook.id, | ||||
|         title: oldBook.metadata.title, | ||||
|         titleIgnorePrefix: oldBook.metadata.titleIgnorePrefix, | ||||
|         subtitle: oldBook.metadata.subtitle, | ||||
|         publishedYear: oldBook.metadata.publishedYear, | ||||
|         publishedDate: oldBook.metadata.publishedDate, | ||||
|         publisher: oldBook.metadata.publisher, | ||||
|         description: oldBook.metadata.description, | ||||
|         isbn: oldBook.metadata.isbn, | ||||
|         asin: oldBook.metadata.asin, | ||||
|         language: oldBook.metadata.language, | ||||
|         explicit: !!oldBook.metadata.explicit, | ||||
|         abridged: !!oldBook.metadata.abridged, | ||||
|         narrators: oldBook.metadata.narrators, | ||||
|         ebookFile: oldBook.ebookFile?.toJSON() || null, | ||||
|         coverPath: oldBook.coverPath, | ||||
|         duration: oldBook.duration, | ||||
|         audioFiles: oldBook.audioFiles?.map(af => af.toJSON()) || [], | ||||
|         chapters: oldBook.chapters, | ||||
|         tags: oldBook.tags, | ||||
|         genres: oldBook.metadata.genres | ||||
|     let series = [] | ||||
|     if (bookExpanded.series?.length) { | ||||
|       series = bookExpanded.series.map(se => { | ||||
|         return { | ||||
|           id: se.id, | ||||
|           name: se.name, | ||||
|           sequence: se.bookSeries.sequence | ||||
|         } | ||||
|       }) | ||||
|     } else if (bookExpanded.bookSeries?.length) { | ||||
|       series = bookExpanded.bookSeries.map(bs => { | ||||
|         if (bs.series) { | ||||
|           return { | ||||
|             id: bs.series.id, | ||||
|             name: bs.series.name, | ||||
|             sequence: bs.sequence | ||||
|           } | ||||
|         } else { | ||||
|           Logger.error(`[Book] Invalid bookExpanded bookSeries: no series`, bs) | ||||
|           return null | ||||
|         } | ||||
|       }).filter(s => s) | ||||
|     } | ||||
| 
 | ||||
|     return { | ||||
|       id: bookExpanded.id, | ||||
|       libraryItemId: libraryItemExpanded.id, | ||||
|       coverPath: bookExpanded.coverPath, | ||||
|       tags: bookExpanded.tags, | ||||
|       audioFiles: bookExpanded.audioFiles, | ||||
|       chapters: bookExpanded.chapters, | ||||
|       ebookFile: bookExpanded.ebookFile, | ||||
|       metadata: { | ||||
|         title: bookExpanded.title, | ||||
|         subtitle: bookExpanded.subtitle, | ||||
|         authors: authors, | ||||
|         narrators: bookExpanded.narrators, | ||||
|         series: series, | ||||
|         genres: bookExpanded.genres, | ||||
|         publishedYear: bookExpanded.publishedYear, | ||||
|         publishedDate: bookExpanded.publishedDate, | ||||
|         publisher: bookExpanded.publisher, | ||||
|         description: bookExpanded.description, | ||||
|         isbn: bookExpanded.isbn, | ||||
|         asin: bookExpanded.asin, | ||||
|         language: bookExpanded.language, | ||||
|         explicit: bookExpanded.explicit, | ||||
|         abridged: bookExpanded.abridged | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   Book.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     title: DataTypes.STRING, | ||||
|     titleIgnorePrefix: DataTypes.STRING, | ||||
|     subtitle: DataTypes.STRING, | ||||
|     publishedYear: DataTypes.STRING, | ||||
|     publishedDate: DataTypes.STRING, | ||||
|     publisher: DataTypes.STRING, | ||||
|     description: DataTypes.TEXT, | ||||
|     isbn: DataTypes.STRING, | ||||
|     asin: DataTypes.STRING, | ||||
|     language: DataTypes.STRING, | ||||
|     explicit: DataTypes.BOOLEAN, | ||||
|     abridged: DataTypes.BOOLEAN, | ||||
|     coverPath: DataTypes.STRING, | ||||
|     duration: DataTypes.FLOAT, | ||||
| 
 | ||||
|     narrators: DataTypes.JSON, | ||||
|     audioFiles: DataTypes.JSON, | ||||
|     ebookFile: DataTypes.JSON, | ||||
|     chapters: DataTypes.JSON, | ||||
|     tags: DataTypes.JSON, | ||||
|     genres: DataTypes.JSON | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'book', | ||||
|     indexes: [ | ||||
|       { | ||||
|         fields: [{ | ||||
|           name: 'title', | ||||
|           collate: 'NOCASE' | ||||
|         }] | ||||
|       }, | ||||
|       { | ||||
|         fields: [{ | ||||
|           name: 'titleIgnorePrefix', | ||||
|           collate: 'NOCASE' | ||||
|         }] | ||||
|       }, | ||||
|       { | ||||
|         fields: ['publishedYear'] | ||||
|       }, | ||||
|       { | ||||
|         fields: ['duration'] | ||||
|   /** | ||||
|    * @param {object} oldBook  | ||||
|    * @returns {boolean} true if updated | ||||
|    */ | ||||
|   static saveFromOld(oldBook) { | ||||
|     const book = this.getFromOld(oldBook) | ||||
|     return this.update(book, { | ||||
|       where: { | ||||
|         id: book.id | ||||
|       } | ||||
|     ] | ||||
|   }) | ||||
|     }).then(result => result[0] > 0).catch((error) => { | ||||
|       Logger.error(`[Book] Failed to save book ${book.id}`, error) | ||||
|       return false | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   return Book | ||||
|   static getFromOld(oldBook) { | ||||
|     return { | ||||
|       id: oldBook.id, | ||||
|       title: oldBook.metadata.title, | ||||
|       titleIgnorePrefix: oldBook.metadata.titleIgnorePrefix, | ||||
|       subtitle: oldBook.metadata.subtitle, | ||||
|       publishedYear: oldBook.metadata.publishedYear, | ||||
|       publishedDate: oldBook.metadata.publishedDate, | ||||
|       publisher: oldBook.metadata.publisher, | ||||
|       description: oldBook.metadata.description, | ||||
|       isbn: oldBook.metadata.isbn, | ||||
|       asin: oldBook.metadata.asin, | ||||
|       language: oldBook.metadata.language, | ||||
|       explicit: !!oldBook.metadata.explicit, | ||||
|       abridged: !!oldBook.metadata.abridged, | ||||
|       narrators: oldBook.metadata.narrators, | ||||
|       ebookFile: oldBook.ebookFile?.toJSON() || null, | ||||
|       coverPath: oldBook.coverPath, | ||||
|       duration: oldBook.duration, | ||||
|       audioFiles: oldBook.audioFiles?.map(af => af.toJSON()) || [], | ||||
|       chapters: oldBook.chapters, | ||||
|       tags: oldBook.tags, | ||||
|       genres: oldBook.metadata.genres | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       title: DataTypes.STRING, | ||||
|       titleIgnorePrefix: DataTypes.STRING, | ||||
|       subtitle: DataTypes.STRING, | ||||
|       publishedYear: DataTypes.STRING, | ||||
|       publishedDate: DataTypes.STRING, | ||||
|       publisher: DataTypes.STRING, | ||||
|       description: DataTypes.TEXT, | ||||
|       isbn: DataTypes.STRING, | ||||
|       asin: DataTypes.STRING, | ||||
|       language: DataTypes.STRING, | ||||
|       explicit: DataTypes.BOOLEAN, | ||||
|       abridged: DataTypes.BOOLEAN, | ||||
|       coverPath: DataTypes.STRING, | ||||
|       duration: DataTypes.FLOAT, | ||||
| 
 | ||||
|       narrators: DataTypes.JSON, | ||||
|       audioFiles: DataTypes.JSON, | ||||
|       ebookFile: DataTypes.JSON, | ||||
|       chapters: DataTypes.JSON, | ||||
|       tags: DataTypes.JSON, | ||||
|       genres: DataTypes.JSON | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'book', | ||||
|       indexes: [ | ||||
|         { | ||||
|           fields: [{ | ||||
|             name: 'title', | ||||
|             collate: 'NOCASE' | ||||
|           }] | ||||
|         }, | ||||
|         { | ||||
|           fields: [{ | ||||
|             name: 'titleIgnorePrefix', | ||||
|             collate: 'NOCASE' | ||||
|           }] | ||||
|         }, | ||||
|         { | ||||
|           fields: ['publishedYear'] | ||||
|         }, | ||||
|         { | ||||
|           fields: ['duration'] | ||||
|         } | ||||
|       ] | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = Book | ||||
| @ -1,41 +1,57 @@ | ||||
| const { DataTypes, Model } = require('sequelize') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class BookAuthor extends Model { | ||||
|     static removeByIds(authorId = null, bookId = null) { | ||||
|       const where = {} | ||||
|       if (authorId) where.authorId = authorId | ||||
|       if (bookId) where.bookId = bookId | ||||
|       return this.destroy({ | ||||
|         where | ||||
|       }) | ||||
|     } | ||||
| class BookAuthor extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.bookId | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.authorId | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|   } | ||||
| 
 | ||||
|   BookAuthor.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     } | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'bookAuthor', | ||||
|     timestamps: true, | ||||
|     updatedAt: false | ||||
|   }) | ||||
|   static removeByIds(authorId = null, bookId = null) { | ||||
|     const where = {} | ||||
|     if (authorId) where.authorId = authorId | ||||
|     if (bookId) where.bookId = bookId | ||||
|     return this.destroy({ | ||||
|       where | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   // Super Many-to-Many
 | ||||
|   // ref: https://sequelize.org/docs/v6/advanced-association-concepts/advanced-many-to-many/#the-best-of-both-worlds-the-super-many-to-many-relationship
 | ||||
|   const { book, author } = sequelize.models | ||||
|   book.belongsToMany(author, { through: BookAuthor }) | ||||
|   author.belongsToMany(book, { through: BookAuthor }) | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       } | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'bookAuthor', | ||||
|       timestamps: true, | ||||
|       updatedAt: false | ||||
|     }) | ||||
| 
 | ||||
|   book.hasMany(BookAuthor) | ||||
|   BookAuthor.belongsTo(book) | ||||
|     // Super Many-to-Many
 | ||||
|     // ref: https://sequelize.org/docs/v6/advanced-association-concepts/advanced-many-to-many/#the-best-of-both-worlds-the-super-many-to-many-relationship
 | ||||
|     const { book, author } = sequelize.models | ||||
|     book.belongsToMany(author, { through: BookAuthor }) | ||||
|     author.belongsToMany(book, { through: BookAuthor }) | ||||
| 
 | ||||
|   author.hasMany(BookAuthor) | ||||
|   BookAuthor.belongsTo(author) | ||||
|     book.hasMany(BookAuthor) | ||||
|     BookAuthor.belongsTo(book) | ||||
| 
 | ||||
|   return BookAuthor | ||||
|     author.hasMany(BookAuthor) | ||||
|     BookAuthor.belongsTo(author) | ||||
|   } | ||||
| } | ||||
| module.exports = BookAuthor | ||||
| @ -1,42 +1,61 @@ | ||||
| const { DataTypes, Model } = require('sequelize') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class BookSeries extends Model { | ||||
|     static removeByIds(seriesId = null, bookId = null) { | ||||
|       const where = {} | ||||
|       if (seriesId) where.seriesId = seriesId | ||||
|       if (bookId) where.bookId = bookId | ||||
|       return this.destroy({ | ||||
|         where | ||||
|       }) | ||||
|     } | ||||
| class BookSeries extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {string} */ | ||||
|     this.sequence | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.bookId | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.seriesId | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|   } | ||||
| 
 | ||||
|   BookSeries.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     sequence: DataTypes.STRING | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'bookSeries', | ||||
|     timestamps: true, | ||||
|     updatedAt: false | ||||
|   }) | ||||
|   static removeByIds(seriesId = null, bookId = null) { | ||||
|     const where = {} | ||||
|     if (seriesId) where.seriesId = seriesId | ||||
|     if (bookId) where.bookId = bookId | ||||
|     return this.destroy({ | ||||
|       where | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   // Super Many-to-Many
 | ||||
|   // ref: https://sequelize.org/docs/v6/advanced-association-concepts/advanced-many-to-many/#the-best-of-both-worlds-the-super-many-to-many-relationship
 | ||||
|   const { book, series } = sequelize.models | ||||
|   book.belongsToMany(series, { through: BookSeries }) | ||||
|   series.belongsToMany(book, { through: BookSeries }) | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       sequence: DataTypes.STRING | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'bookSeries', | ||||
|       timestamps: true, | ||||
|       updatedAt: false | ||||
|     }) | ||||
| 
 | ||||
|   book.hasMany(BookSeries) | ||||
|   BookSeries.belongsTo(book) | ||||
|     // Super Many-to-Many
 | ||||
|     // ref: https://sequelize.org/docs/v6/advanced-association-concepts/advanced-many-to-many/#the-best-of-both-worlds-the-super-many-to-many-relationship
 | ||||
|     const { book, series } = sequelize.models | ||||
|     book.belongsToMany(series, { through: BookSeries }) | ||||
|     series.belongsToMany(book, { through: BookSeries }) | ||||
| 
 | ||||
|   series.hasMany(BookSeries) | ||||
|   BookSeries.belongsTo(series) | ||||
|     book.hasMany(BookSeries) | ||||
|     BookSeries.belongsTo(book) | ||||
| 
 | ||||
|   return BookSeries | ||||
|     series.hasMany(BookSeries) | ||||
|     BookSeries.belongsTo(series) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = BookSeries | ||||
| @ -1,151 +1,97 @@ | ||||
| const { DataTypes, Model, Sequelize } = require('sequelize') | ||||
| 
 | ||||
| const oldCollection = require('../objects/Collection') | ||||
| const { areEquivalent } = require('../utils/index') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class Collection extends Model { | ||||
|     /** | ||||
|      * Get all old collections | ||||
|      * @returns {Promise<oldCollection[]>} | ||||
|      */ | ||||
|     static async getOldCollections() { | ||||
|       const collections = await this.findAll({ | ||||
|         include: { | ||||
|           model: sequelize.models.book, | ||||
|           include: sequelize.models.libraryItem | ||||
| 
 | ||||
| class Collection extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {string} */ | ||||
|     this.name | ||||
|     /** @type {string} */ | ||||
|     this.description | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.libraryId | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|   } | ||||
|   /** | ||||
|    * Get all old collections | ||||
|    * @returns {Promise<oldCollection[]>} | ||||
|    */ | ||||
|   static async getOldCollections() { | ||||
|     const collections = await this.findAll({ | ||||
|       include: { | ||||
|         model: this.sequelize.models.book, | ||||
|         include: this.sequelize.models.libraryItem | ||||
|       }, | ||||
|       order: [[this.sequelize.models.book, this.sequelize.models.collectionBook, 'order', 'ASC']] | ||||
|     }) | ||||
|     return collections.map(c => this.getOldCollection(c)) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get all old collections toJSONExpanded, items filtered for user permissions | ||||
|    * @param {[oldUser]} user | ||||
|    * @param {[string]} libraryId | ||||
|    * @param {[string[]]} include | ||||
|    * @returns {Promise<object[]>} oldCollection.toJSONExpanded | ||||
|    */ | ||||
|   static async getOldCollectionsJsonExpanded(user, libraryId, include) { | ||||
|     let collectionWhere = null | ||||
|     if (libraryId) { | ||||
|       collectionWhere = { | ||||
|         libraryId | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     // Optionally include rssfeed for collection
 | ||||
|     const collectionIncludes = [] | ||||
|     if (include.includes('rssfeed')) { | ||||
|       collectionIncludes.push({ | ||||
|         model: this.sequelize.models.feed | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     const collections = await this.findAll({ | ||||
|       where: collectionWhere, | ||||
|       include: [ | ||||
|         { | ||||
|           model: this.sequelize.models.book, | ||||
|           include: [ | ||||
|             { | ||||
|               model: this.sequelize.models.libraryItem | ||||
|             }, | ||||
|             { | ||||
|               model: this.sequelize.models.author, | ||||
|               through: { | ||||
|                 attributes: [] | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               model: this.sequelize.models.series, | ||||
|               through: { | ||||
|                 attributes: ['sequence'] | ||||
|               } | ||||
|             }, | ||||
| 
 | ||||
|           ] | ||||
|         }, | ||||
|         order: [[sequelize.models.book, sequelize.models.collectionBook, 'order', 'ASC']] | ||||
|       }) | ||||
|       return collections.map(c => this.getOldCollection(c)) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Get all old collections toJSONExpanded, items filtered for user permissions | ||||
|      * @param {[oldUser]} user | ||||
|      * @param {[string]} libraryId | ||||
|      * @param {[string[]]} include | ||||
|      * @returns {Promise<object[]>} oldCollection.toJSONExpanded | ||||
|      */ | ||||
|     static async getOldCollectionsJsonExpanded(user, libraryId, include) { | ||||
|       let collectionWhere = null | ||||
|       if (libraryId) { | ||||
|         collectionWhere = { | ||||
|           libraryId | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       // Optionally include rssfeed for collection
 | ||||
|       const collectionIncludes = [] | ||||
|       if (include.includes('rssfeed')) { | ||||
|         collectionIncludes.push({ | ||||
|           model: sequelize.models.feed | ||||
|         }) | ||||
|       } | ||||
| 
 | ||||
|       const collections = await this.findAll({ | ||||
|         where: collectionWhere, | ||||
|         include: [ | ||||
|           { | ||||
|             model: sequelize.models.book, | ||||
|             include: [ | ||||
|               { | ||||
|                 model: sequelize.models.libraryItem | ||||
|               }, | ||||
|               { | ||||
|                 model: sequelize.models.author, | ||||
|                 through: { | ||||
|                   attributes: [] | ||||
|                 } | ||||
|               }, | ||||
|               { | ||||
|                 model: sequelize.models.series, | ||||
|                 through: { | ||||
|                   attributes: ['sequence'] | ||||
|                 } | ||||
|               }, | ||||
| 
 | ||||
|             ] | ||||
|           }, | ||||
|           ...collectionIncludes | ||||
|         ], | ||||
|         order: [[sequelize.models.book, sequelize.models.collectionBook, 'order', 'ASC']] | ||||
|       }) | ||||
|       // TODO: Handle user permission restrictions on initial query
 | ||||
|       return collections.map(c => { | ||||
|         const oldCollection = this.getOldCollection(c) | ||||
| 
 | ||||
|         // Filter books using user permissions
 | ||||
|         const books = c.books?.filter(b => { | ||||
|           if (user) { | ||||
|             if (b.tags?.length && !user.checkCanAccessLibraryItemWithTags(b.tags)) { | ||||
|               return false | ||||
|             } | ||||
|             if (b.explicit === true && !user.canAccessExplicitContent) { | ||||
|               return false | ||||
|             } | ||||
|           } | ||||
|           return true | ||||
|         }) || [] | ||||
| 
 | ||||
|         // Map to library items
 | ||||
|         const libraryItems = books.map(b => { | ||||
|           const libraryItem = b.libraryItem | ||||
|           delete b.libraryItem | ||||
|           libraryItem.media = b | ||||
|           return sequelize.models.libraryItem.getOldLibraryItem(libraryItem) | ||||
|         }) | ||||
| 
 | ||||
|         // Users with restricted permissions will not see this collection
 | ||||
|         if (!books.length && oldCollection.books.length) { | ||||
|           return null | ||||
|         } | ||||
| 
 | ||||
|         const collectionExpanded = oldCollection.toJSONExpanded(libraryItems) | ||||
| 
 | ||||
|         // Map feed if found
 | ||||
|         if (c.feeds?.length) { | ||||
|           collectionExpanded.rssFeed = sequelize.models.feed.getOldFeed(c.feeds[0]) | ||||
|         } | ||||
| 
 | ||||
|         return collectionExpanded | ||||
|       }).filter(c => c) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Get old collection toJSONExpanded, items filtered for user permissions | ||||
|      * @param {[oldUser]} user | ||||
|      * @param {[string[]]} include | ||||
|      * @returns {Promise<object>} oldCollection.toJSONExpanded | ||||
|      */ | ||||
|     async getOldJsonExpanded(user, include) { | ||||
|       this.books = await this.getBooks({ | ||||
|         include: [ | ||||
|           { | ||||
|             model: sequelize.models.libraryItem | ||||
|           }, | ||||
|           { | ||||
|             model: sequelize.models.author, | ||||
|             through: { | ||||
|               attributes: [] | ||||
|             } | ||||
|           }, | ||||
|           { | ||||
|             model: sequelize.models.series, | ||||
|             through: { | ||||
|               attributes: ['sequence'] | ||||
|             } | ||||
|           }, | ||||
| 
 | ||||
|         ], | ||||
|         order: [Sequelize.literal('`collectionBook.order` ASC')] | ||||
|       }) || [] | ||||
| 
 | ||||
|       const oldCollection = sequelize.models.collection.getOldCollection(this) | ||||
|         ...collectionIncludes | ||||
|       ], | ||||
|       order: [[this.sequelize.models.book, this.sequelize.models.collectionBook, 'order', 'ASC']] | ||||
|     }) | ||||
|     // TODO: Handle user permission restrictions on initial query
 | ||||
|     return collections.map(c => { | ||||
|       const oldCollection = this.getOldCollection(c) | ||||
| 
 | ||||
|       // Filter books using user permissions
 | ||||
|       // TODO: Handle user permission restrictions on initial query
 | ||||
|       const books = this.books?.filter(b => { | ||||
|       const books = c.books?.filter(b => { | ||||
|         if (user) { | ||||
|           if (b.tags?.length && !user.checkCanAccessLibraryItemWithTags(b.tags)) { | ||||
|             return false | ||||
| @ -162,7 +108,7 @@ module.exports = (sequelize) => { | ||||
|         const libraryItem = b.libraryItem | ||||
|         delete b.libraryItem | ||||
|         libraryItem.media = b | ||||
|         return sequelize.models.libraryItem.getOldLibraryItem(libraryItem) | ||||
|         return this.sequelize.models.libraryItem.getOldLibraryItem(libraryItem) | ||||
|       }) | ||||
| 
 | ||||
|       // Users with restricted permissions will not see this collection
 | ||||
| @ -172,151 +118,225 @@ module.exports = (sequelize) => { | ||||
| 
 | ||||
|       const collectionExpanded = oldCollection.toJSONExpanded(libraryItems) | ||||
| 
 | ||||
|       if (include?.includes('rssfeed')) { | ||||
|         const feeds = await this.getFeeds() | ||||
|         if (feeds?.length) { | ||||
|           collectionExpanded.rssFeed = sequelize.models.feed.getOldFeed(feeds[0]) | ||||
|         } | ||||
|       // Map feed if found
 | ||||
|       if (c.feeds?.length) { | ||||
|         collectionExpanded.rssFeed = this.sequelize.models.feed.getOldFeed(c.feeds[0]) | ||||
|       } | ||||
| 
 | ||||
|       return collectionExpanded | ||||
|     }).filter(c => c) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get old collection toJSONExpanded, items filtered for user permissions | ||||
|    * @param {[oldUser]} user | ||||
|    * @param {[string[]]} include | ||||
|    * @returns {Promise<object>} oldCollection.toJSONExpanded | ||||
|    */ | ||||
|   async getOldJsonExpanded(user, include) { | ||||
|     this.books = await this.getBooks({ | ||||
|       include: [ | ||||
|         { | ||||
|           model: this.sequelize.models.libraryItem | ||||
|         }, | ||||
|         { | ||||
|           model: this.sequelize.models.author, | ||||
|           through: { | ||||
|             attributes: [] | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           model: this.sequelize.models.series, | ||||
|           through: { | ||||
|             attributes: ['sequence'] | ||||
|           } | ||||
|         }, | ||||
| 
 | ||||
|       ], | ||||
|       order: [Sequelize.literal('`collectionBook.order` ASC')] | ||||
|     }) || [] | ||||
| 
 | ||||
|     const oldCollection = this.sequelize.models.collection.getOldCollection(this) | ||||
| 
 | ||||
|     // Filter books using user permissions
 | ||||
|     // TODO: Handle user permission restrictions on initial query
 | ||||
|     const books = this.books?.filter(b => { | ||||
|       if (user) { | ||||
|         if (b.tags?.length && !user.checkCanAccessLibraryItemWithTags(b.tags)) { | ||||
|           return false | ||||
|         } | ||||
|         if (b.explicit === true && !user.canAccessExplicitContent) { | ||||
|           return false | ||||
|         } | ||||
|       } | ||||
|       return true | ||||
|     }) || [] | ||||
| 
 | ||||
|     // Map to library items
 | ||||
|     const libraryItems = books.map(b => { | ||||
|       const libraryItem = b.libraryItem | ||||
|       delete b.libraryItem | ||||
|       libraryItem.media = b | ||||
|       return this.sequelize.models.libraryItem.getOldLibraryItem(libraryItem) | ||||
|     }) | ||||
| 
 | ||||
|     // Users with restricted permissions will not see this collection
 | ||||
|     if (!books.length && oldCollection.books.length) { | ||||
|       return null | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Get old collection from Collection | ||||
|      * @param {Collection} collectionExpanded  | ||||
|      * @returns {oldCollection} | ||||
|      */ | ||||
|     static getOldCollection(collectionExpanded) { | ||||
|       const libraryItemIds = collectionExpanded.books?.map(b => b.libraryItem?.id || null).filter(lid => lid) || [] | ||||
|       return new oldCollection({ | ||||
|         id: collectionExpanded.id, | ||||
|         libraryId: collectionExpanded.libraryId, | ||||
|         name: collectionExpanded.name, | ||||
|         description: collectionExpanded.description, | ||||
|         books: libraryItemIds, | ||||
|         lastUpdate: collectionExpanded.updatedAt.valueOf(), | ||||
|         createdAt: collectionExpanded.createdAt.valueOf() | ||||
|       }) | ||||
|     } | ||||
|     const collectionExpanded = oldCollection.toJSONExpanded(libraryItems) | ||||
| 
 | ||||
|     static createFromOld(oldCollection) { | ||||
|       const collection = this.getFromOld(oldCollection) | ||||
|       return this.create(collection) | ||||
|     } | ||||
| 
 | ||||
|     static getFromOld(oldCollection) { | ||||
|       return { | ||||
|         id: oldCollection.id, | ||||
|         name: oldCollection.name, | ||||
|         description: oldCollection.description, | ||||
|         libraryId: oldCollection.libraryId | ||||
|     if (include?.includes('rssfeed')) { | ||||
|       const feeds = await this.getFeeds() | ||||
|       if (feeds?.length) { | ||||
|         collectionExpanded.rssFeed = this.sequelize.models.feed.getOldFeed(feeds[0]) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     static removeById(collectionId) { | ||||
|       return this.destroy({ | ||||
|         where: { | ||||
|           id: collectionId | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
|     return collectionExpanded | ||||
|   } | ||||
| 
 | ||||
|     /** | ||||
|      * Get old collection by id | ||||
|      * @param {string} collectionId  | ||||
|      * @returns {Promise<oldCollection|null>} returns null if not found | ||||
|      */ | ||||
|     static async getOldById(collectionId) { | ||||
|       if (!collectionId) return null | ||||
|       const collection = await this.findByPk(collectionId, { | ||||
|         include: { | ||||
|           model: sequelize.models.book, | ||||
|           include: sequelize.models.libraryItem | ||||
|         }, | ||||
|         order: [[sequelize.models.book, sequelize.models.collectionBook, 'order', 'ASC']] | ||||
|       }) | ||||
|       if (!collection) return null | ||||
|       return this.getOldCollection(collection) | ||||
|     } | ||||
|   /** | ||||
|    * Get old collection from Collection | ||||
|    * @param {Collection} collectionExpanded  | ||||
|    * @returns {oldCollection} | ||||
|    */ | ||||
|   static getOldCollection(collectionExpanded) { | ||||
|     const libraryItemIds = collectionExpanded.books?.map(b => b.libraryItem?.id || null).filter(lid => lid) || [] | ||||
|     return new oldCollection({ | ||||
|       id: collectionExpanded.id, | ||||
|       libraryId: collectionExpanded.libraryId, | ||||
|       name: collectionExpanded.name, | ||||
|       description: collectionExpanded.description, | ||||
|       books: libraryItemIds, | ||||
|       lastUpdate: collectionExpanded.updatedAt.valueOf(), | ||||
|       createdAt: collectionExpanded.createdAt.valueOf() | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|     /** | ||||
|      * Get old collection from current | ||||
|      * @returns {Promise<oldCollection>} | ||||
|      */ | ||||
|     async getOld() { | ||||
|       this.books = await this.getBooks({ | ||||
|         include: [ | ||||
|           { | ||||
|             model: sequelize.models.libraryItem | ||||
|           }, | ||||
|           { | ||||
|             model: sequelize.models.author, | ||||
|             through: { | ||||
|               attributes: [] | ||||
|             } | ||||
|           }, | ||||
|           { | ||||
|             model: sequelize.models.series, | ||||
|             through: { | ||||
|               attributes: ['sequence'] | ||||
|             } | ||||
|           }, | ||||
|   static createFromOld(oldCollection) { | ||||
|     const collection = this.getFromOld(oldCollection) | ||||
|     return this.create(collection) | ||||
|   } | ||||
| 
 | ||||
|         ], | ||||
|         order: [Sequelize.literal('`collectionBook.order` ASC')] | ||||
|       }) || [] | ||||
| 
 | ||||
|       return sequelize.models.collection.getOldCollection(this) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Remove all collections belonging to library | ||||
|      * @param {string} libraryId  | ||||
|      * @returns {Promise<number>} number of collections destroyed | ||||
|      */ | ||||
|     static async removeAllForLibrary(libraryId) { | ||||
|       if (!libraryId) return 0 | ||||
|       return this.destroy({ | ||||
|         where: { | ||||
|           libraryId | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     static async getAllForBook(bookId) { | ||||
|       const collections = await this.findAll({ | ||||
|         include: { | ||||
|           model: sequelize.models.book, | ||||
|           where: { | ||||
|             id: bookId | ||||
|           }, | ||||
|           required: true, | ||||
|           include: sequelize.models.libraryItem | ||||
|         }, | ||||
|         order: [[sequelize.models.book, sequelize.models.collectionBook, 'order', 'ASC']] | ||||
|       }) | ||||
|       return collections.map(c => this.getOldCollection(c)) | ||||
|   static getFromOld(oldCollection) { | ||||
|     return { | ||||
|       id: oldCollection.id, | ||||
|       name: oldCollection.name, | ||||
|       description: oldCollection.description, | ||||
|       libraryId: oldCollection.libraryId | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   Collection.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     name: DataTypes.STRING, | ||||
|     description: DataTypes.TEXT | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'collection' | ||||
|   }) | ||||
|   static removeById(collectionId) { | ||||
|     return this.destroy({ | ||||
|       where: { | ||||
|         id: collectionId | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   const { library } = sequelize.models | ||||
|   /** | ||||
|    * Get old collection by id | ||||
|    * @param {string} collectionId  | ||||
|    * @returns {Promise<oldCollection|null>} returns null if not found | ||||
|    */ | ||||
|   static async getOldById(collectionId) { | ||||
|     if (!collectionId) return null | ||||
|     const collection = await this.findByPk(collectionId, { | ||||
|       include: { | ||||
|         model: this.sequelize.models.book, | ||||
|         include: this.sequelize.models.libraryItem | ||||
|       }, | ||||
|       order: [[this.sequelize.models.book, this.sequelize.models.collectionBook, 'order', 'ASC']] | ||||
|     }) | ||||
|     if (!collection) return null | ||||
|     return this.getOldCollection(collection) | ||||
|   } | ||||
| 
 | ||||
|   library.hasMany(Collection) | ||||
|   Collection.belongsTo(library) | ||||
|   /** | ||||
|    * Get old collection from current | ||||
|    * @returns {Promise<oldCollection>} | ||||
|    */ | ||||
|   async getOld() { | ||||
|     this.books = await this.getBooks({ | ||||
|       include: [ | ||||
|         { | ||||
|           model: this.sequelize.models.libraryItem | ||||
|         }, | ||||
|         { | ||||
|           model: this.sequelize.models.author, | ||||
|           through: { | ||||
|             attributes: [] | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           model: this.sequelize.models.series, | ||||
|           through: { | ||||
|             attributes: ['sequence'] | ||||
|           } | ||||
|         }, | ||||
| 
 | ||||
|   return Collection | ||||
|       ], | ||||
|       order: [Sequelize.literal('`collectionBook.order` ASC')] | ||||
|     }) || [] | ||||
| 
 | ||||
|     return this.sequelize.models.collection.getOldCollection(this) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Remove all collections belonging to library | ||||
|    * @param {string} libraryId  | ||||
|    * @returns {Promise<number>} number of collections destroyed | ||||
|    */ | ||||
|   static async removeAllForLibrary(libraryId) { | ||||
|     if (!libraryId) return 0 | ||||
|     return this.destroy({ | ||||
|       where: { | ||||
|         libraryId | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   static async getAllForBook(bookId) { | ||||
|     const collections = await this.findAll({ | ||||
|       include: { | ||||
|         model: this.sequelize.models.book, | ||||
|         where: { | ||||
|           id: bookId | ||||
|         }, | ||||
|         required: true, | ||||
|         include: this.sequelize.models.libraryItem | ||||
|       }, | ||||
|       order: [[this.sequelize.models.book, this.sequelize.models.collectionBook, 'order', 'ASC']] | ||||
|     }) | ||||
|     return collections.map(c => this.getOldCollection(c)) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       name: DataTypes.STRING, | ||||
|       description: DataTypes.TEXT | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'collection' | ||||
|     }) | ||||
| 
 | ||||
|     const { library } = sequelize.models | ||||
| 
 | ||||
|     library.hasMany(Collection) | ||||
|     Collection.belongsTo(library) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = Collection | ||||
| @ -1,46 +1,61 @@ | ||||
| const { DataTypes, Model } = require('sequelize') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class CollectionBook extends Model { | ||||
|     static removeByIds(collectionId, bookId) { | ||||
|       return this.destroy({ | ||||
|         where: { | ||||
|           bookId, | ||||
|           collectionId | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
| class CollectionBook extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {number} */ | ||||
|     this.order | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.bookId | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.collectionId | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|   } | ||||
| 
 | ||||
|   CollectionBook.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     order: DataTypes.INTEGER | ||||
|   }, { | ||||
|     sequelize, | ||||
|     timestamps: true, | ||||
|     updatedAt: false, | ||||
|     modelName: 'collectionBook' | ||||
|   }) | ||||
|   static removeByIds(collectionId, bookId) { | ||||
|     return this.destroy({ | ||||
|       where: { | ||||
|         bookId, | ||||
|         collectionId | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   // Super Many-to-Many
 | ||||
|   // ref: https://sequelize.org/docs/v6/advanced-association-concepts/advanced-many-to-many/#the-best-of-both-worlds-the-super-many-to-many-relationship
 | ||||
|   const { book, collection } = sequelize.models | ||||
|   book.belongsToMany(collection, { through: CollectionBook }) | ||||
|   collection.belongsToMany(book, { through: CollectionBook }) | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       order: DataTypes.INTEGER | ||||
|     }, { | ||||
|       sequelize, | ||||
|       timestamps: true, | ||||
|       updatedAt: false, | ||||
|       modelName: 'collectionBook' | ||||
|     }) | ||||
| 
 | ||||
|   book.hasMany(CollectionBook, { | ||||
|     onDelete: 'CASCADE' | ||||
|   }) | ||||
|   CollectionBook.belongsTo(book) | ||||
|     // Super Many-to-Many
 | ||||
|     // ref: https://sequelize.org/docs/v6/advanced-association-concepts/advanced-many-to-many/#the-best-of-both-worlds-the-super-many-to-many-relationship
 | ||||
|     const { book, collection } = sequelize.models | ||||
|     book.belongsToMany(collection, { through: CollectionBook }) | ||||
|     collection.belongsToMany(book, { through: CollectionBook }) | ||||
| 
 | ||||
|   collection.hasMany(CollectionBook, { | ||||
|     onDelete: 'CASCADE' | ||||
|   }) | ||||
|   CollectionBook.belongsTo(collection) | ||||
|     book.hasMany(CollectionBook, { | ||||
|       onDelete: 'CASCADE' | ||||
|     }) | ||||
|     CollectionBook.belongsTo(book) | ||||
| 
 | ||||
|   return CollectionBook | ||||
|     collection.hasMany(CollectionBook, { | ||||
|       onDelete: 'CASCADE' | ||||
|     }) | ||||
|     CollectionBook.belongsTo(collection) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = CollectionBook | ||||
| @ -1,116 +1,147 @@ | ||||
| const { DataTypes, Model } = require('sequelize') | ||||
| const oldDevice = require('../objects/DeviceInfo') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class Device extends Model { | ||||
|     getOldDevice() { | ||||
|       let browserVersion = null | ||||
|       let sdkVersion = null | ||||
|       if (this.clientName === 'Abs Android') { | ||||
|         sdkVersion = this.deviceVersion || null | ||||
|       } else { | ||||
|         browserVersion = this.deviceVersion || null | ||||
|       } | ||||
| class Device extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|       return new oldDevice({ | ||||
|         id: this.id, | ||||
|         deviceId: this.deviceId, | ||||
|         userId: this.userId, | ||||
|         ipAddress: this.ipAddress, | ||||
|         browserName: this.extraData.browserName || null, | ||||
|         browserVersion, | ||||
|         osName: this.extraData.osName || null, | ||||
|         osVersion: this.extraData.osVersion || null, | ||||
|         clientVersion: this.clientVersion || null, | ||||
|         manufacturer: this.extraData.manufacturer || null, | ||||
|         model: this.extraData.model || null, | ||||
|         sdkVersion, | ||||
|         deviceName: this.deviceName, | ||||
|         clientName: this.clientName | ||||
|       }) | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {string} */ | ||||
|     this.deviceId | ||||
|     /** @type {string} */ | ||||
|     this.clientName | ||||
|     /** @type {string} */ | ||||
|     this.clientVersion | ||||
|     /** @type {string} */ | ||||
|     this.ipAddress | ||||
|     /** @type {string} */ | ||||
|     this.deviceName | ||||
|     /** @type {string} */ | ||||
|     this.deviceVersion | ||||
|     /** @type {object} */ | ||||
|     this.extraData | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.userId | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|   } | ||||
| 
 | ||||
|   getOldDevice() { | ||||
|     let browserVersion = null | ||||
|     let sdkVersion = null | ||||
|     if (this.clientName === 'Abs Android') { | ||||
|       sdkVersion = this.deviceVersion || null | ||||
|     } else { | ||||
|       browserVersion = this.deviceVersion || null | ||||
|     } | ||||
| 
 | ||||
|     static async getOldDeviceByDeviceId(deviceId) { | ||||
|       const device = await this.findOne({ | ||||
|         where: { | ||||
|           deviceId | ||||
|         } | ||||
|       }) | ||||
|       if (!device) return null | ||||
|       return device.getOldDevice() | ||||
|     return new oldDevice({ | ||||
|       id: this.id, | ||||
|       deviceId: this.deviceId, | ||||
|       userId: this.userId, | ||||
|       ipAddress: this.ipAddress, | ||||
|       browserName: this.extraData.browserName || null, | ||||
|       browserVersion, | ||||
|       osName: this.extraData.osName || null, | ||||
|       osVersion: this.extraData.osVersion || null, | ||||
|       clientVersion: this.clientVersion || null, | ||||
|       manufacturer: this.extraData.manufacturer || null, | ||||
|       model: this.extraData.model || null, | ||||
|       sdkVersion, | ||||
|       deviceName: this.deviceName, | ||||
|       clientName: this.clientName | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   static async getOldDeviceByDeviceId(deviceId) { | ||||
|     const device = await this.findOne({ | ||||
|       where: { | ||||
|         deviceId | ||||
|       } | ||||
|     }) | ||||
|     if (!device) return null | ||||
|     return device.getOldDevice() | ||||
|   } | ||||
| 
 | ||||
|   static createFromOld(oldDevice) { | ||||
|     const device = this.getFromOld(oldDevice) | ||||
|     return this.create(device) | ||||
|   } | ||||
| 
 | ||||
|   static updateFromOld(oldDevice) { | ||||
|     const device = this.getFromOld(oldDevice) | ||||
|     return this.update(device, { | ||||
|       where: { | ||||
|         id: device.id | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   static getFromOld(oldDeviceInfo) { | ||||
|     let extraData = {} | ||||
| 
 | ||||
|     if (oldDeviceInfo.manufacturer) { | ||||
|       extraData.manufacturer = oldDeviceInfo.manufacturer | ||||
|     } | ||||
|     if (oldDeviceInfo.model) { | ||||
|       extraData.model = oldDeviceInfo.model | ||||
|     } | ||||
|     if (oldDeviceInfo.osName) { | ||||
|       extraData.osName = oldDeviceInfo.osName | ||||
|     } | ||||
|     if (oldDeviceInfo.osVersion) { | ||||
|       extraData.osVersion = oldDeviceInfo.osVersion | ||||
|     } | ||||
|     if (oldDeviceInfo.browserName) { | ||||
|       extraData.browserName = oldDeviceInfo.browserName | ||||
|     } | ||||
| 
 | ||||
|     static createFromOld(oldDevice) { | ||||
|       const device = this.getFromOld(oldDevice) | ||||
|       return this.create(device) | ||||
|     } | ||||
| 
 | ||||
|     static updateFromOld(oldDevice) { | ||||
|       const device = this.getFromOld(oldDevice) | ||||
|       return this.update(device, { | ||||
|         where: { | ||||
|           id: device.id | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     static getFromOld(oldDeviceInfo) { | ||||
|       let extraData = {} | ||||
| 
 | ||||
|       if (oldDeviceInfo.manufacturer) { | ||||
|         extraData.manufacturer = oldDeviceInfo.manufacturer | ||||
|       } | ||||
|       if (oldDeviceInfo.model) { | ||||
|         extraData.model = oldDeviceInfo.model | ||||
|       } | ||||
|       if (oldDeviceInfo.osName) { | ||||
|         extraData.osName = oldDeviceInfo.osName | ||||
|       } | ||||
|       if (oldDeviceInfo.osVersion) { | ||||
|         extraData.osVersion = oldDeviceInfo.osVersion | ||||
|       } | ||||
|       if (oldDeviceInfo.browserName) { | ||||
|         extraData.browserName = oldDeviceInfo.browserName | ||||
|       } | ||||
| 
 | ||||
|       return { | ||||
|         id: oldDeviceInfo.id, | ||||
|         deviceId: oldDeviceInfo.deviceId, | ||||
|         clientName: oldDeviceInfo.clientName || null, | ||||
|         clientVersion: oldDeviceInfo.clientVersion || null, | ||||
|         ipAddress: oldDeviceInfo.ipAddress, | ||||
|         deviceName: oldDeviceInfo.deviceName || null, | ||||
|         deviceVersion: oldDeviceInfo.sdkVersion || oldDeviceInfo.browserVersion || null, | ||||
|         userId: oldDeviceInfo.userId, | ||||
|         extraData | ||||
|       } | ||||
|     return { | ||||
|       id: oldDeviceInfo.id, | ||||
|       deviceId: oldDeviceInfo.deviceId, | ||||
|       clientName: oldDeviceInfo.clientName || null, | ||||
|       clientVersion: oldDeviceInfo.clientVersion || null, | ||||
|       ipAddress: oldDeviceInfo.ipAddress, | ||||
|       deviceName: oldDeviceInfo.deviceName || null, | ||||
|       deviceVersion: oldDeviceInfo.sdkVersion || oldDeviceInfo.browserVersion || null, | ||||
|       userId: oldDeviceInfo.userId, | ||||
|       extraData | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   Device.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     deviceId: DataTypes.STRING, | ||||
|     clientName: DataTypes.STRING, // e.g. Abs Web, Abs Android
 | ||||
|     clientVersion: DataTypes.STRING, // e.g. Server version or mobile version
 | ||||
|     ipAddress: DataTypes.STRING, | ||||
|     deviceName: DataTypes.STRING, // e.g. Windows 10 Chrome, Google Pixel 6, Apple iPhone 10,3
 | ||||
|     deviceVersion: DataTypes.STRING, // e.g. Browser version or Android SDK
 | ||||
|     extraData: DataTypes.JSON | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'device' | ||||
|   }) | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       deviceId: DataTypes.STRING, | ||||
|       clientName: DataTypes.STRING, // e.g. Abs Web, Abs Android
 | ||||
|       clientVersion: DataTypes.STRING, // e.g. Server version or mobile version
 | ||||
|       ipAddress: DataTypes.STRING, | ||||
|       deviceName: DataTypes.STRING, // e.g. Windows 10 Chrome, Google Pixel 6, Apple iPhone 10,3
 | ||||
|       deviceVersion: DataTypes.STRING, // e.g. Browser version or Android SDK
 | ||||
|       extraData: DataTypes.JSON | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'device' | ||||
|     }) | ||||
| 
 | ||||
|   const { user } = sequelize.models | ||||
|     const { user } = sequelize.models | ||||
| 
 | ||||
|   user.hasMany(Device, { | ||||
|     onDelete: 'CASCADE' | ||||
|   }) | ||||
|   Device.belongsTo(user) | ||||
| 
 | ||||
|   return Device | ||||
|     user.hasMany(Device, { | ||||
|       onDelete: 'CASCADE' | ||||
|     }) | ||||
|     Device.belongsTo(user) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = Device | ||||
| @ -1,307 +1,361 @@ | ||||
| const { DataTypes, Model } = require('sequelize') | ||||
| const oldFeed = require('../objects/Feed') | ||||
| const areEquivalent = require('../utils/areEquivalent') | ||||
| /* | ||||
|  * Polymorphic association: https://sequelize.org/docs/v6/advanced-association-concepts/polymorphic-associations/
 | ||||
|  * Feeds can be created from LibraryItem, Collection, Playlist or Series | ||||
|  */ | ||||
| module.exports = (sequelize) => { | ||||
|   class Feed extends Model { | ||||
|     static async getOldFeeds() { | ||||
|       const feeds = await this.findAll({ | ||||
|         include: { | ||||
|           model: sequelize.models.feedEpisode | ||||
|         } | ||||
|       }) | ||||
|       return feeds.map(f => this.getOldFeed(f)) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Get old feed from Feed and optionally Feed with FeedEpisodes | ||||
|      * @param {Feed} feedExpanded | ||||
|      * @returns {oldFeed} | ||||
|      */ | ||||
|     static getOldFeed(feedExpanded) { | ||||
|       const episodes = feedExpanded.feedEpisodes?.map((feedEpisode) => feedEpisode.getOldEpisode()) | ||||
|       return new oldFeed({ | ||||
|         id: feedExpanded.id, | ||||
|         slug: feedExpanded.slug, | ||||
|         userId: feedExpanded.userId, | ||||
|         entityType: feedExpanded.entityType, | ||||
|         entityId: feedExpanded.entityId, | ||||
|         entityUpdatedAt: feedExpanded.entityUpdatedAt?.valueOf() || null, | ||||
|         coverPath: feedExpanded.coverPath || null, | ||||
|         meta: { | ||||
|           title: feedExpanded.title, | ||||
|           description: feedExpanded.description, | ||||
|           author: feedExpanded.author, | ||||
|           imageUrl: feedExpanded.imageURL, | ||||
|           feedUrl: feedExpanded.feedURL, | ||||
|           link: feedExpanded.siteURL, | ||||
|           explicit: feedExpanded.explicit, | ||||
|           type: feedExpanded.podcastType, | ||||
|           language: feedExpanded.language, | ||||
|           preventIndexing: feedExpanded.preventIndexing, | ||||
|           ownerName: feedExpanded.ownerName, | ||||
|           ownerEmail: feedExpanded.ownerEmail | ||||
|         }, | ||||
|         serverAddress: feedExpanded.serverAddress, | ||||
| class Feed extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {string} */ | ||||
|     this.slug | ||||
|     /** @type {string} */ | ||||
|     this.entityType | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.entityId | ||||
|     /** @type {Date} */ | ||||
|     this.entityUpdatedAt | ||||
|     /** @type {string} */ | ||||
|     this.serverAddress | ||||
|     /** @type {string} */ | ||||
|     this.feedURL | ||||
|     /** @type {string} */ | ||||
|     this.imageURL | ||||
|     /** @type {string} */ | ||||
|     this.siteURL | ||||
|     /** @type {string} */ | ||||
|     this.title | ||||
|     /** @type {string} */ | ||||
|     this.description | ||||
|     /** @type {string} */ | ||||
|     this.author | ||||
|     /** @type {string} */ | ||||
|     this.podcastType | ||||
|     /** @type {string} */ | ||||
|     this.language | ||||
|     /** @type {string} */ | ||||
|     this.ownerName | ||||
|     /** @type {string} */ | ||||
|     this.ownerEmail | ||||
|     /** @type {boolean} */ | ||||
|     this.explicit | ||||
|     /** @type {boolean} */ | ||||
|     this.preventIndexing | ||||
|     /** @type {string} */ | ||||
|     this.coverPath | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.userId | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|   } | ||||
| 
 | ||||
|   static async getOldFeeds() { | ||||
|     const feeds = await this.findAll({ | ||||
|       include: { | ||||
|         model: this.sequelize.models.feedEpisode | ||||
|       } | ||||
|     }) | ||||
|     return feeds.map(f => this.getOldFeed(f)) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get old feed from Feed and optionally Feed with FeedEpisodes | ||||
|    * @param {Feed} feedExpanded | ||||
|    * @returns {oldFeed} | ||||
|    */ | ||||
|   static getOldFeed(feedExpanded) { | ||||
|     const episodes = feedExpanded.feedEpisodes?.map((feedEpisode) => feedEpisode.getOldEpisode()) | ||||
|     return new oldFeed({ | ||||
|       id: feedExpanded.id, | ||||
|       slug: feedExpanded.slug, | ||||
|       userId: feedExpanded.userId, | ||||
|       entityType: feedExpanded.entityType, | ||||
|       entityId: feedExpanded.entityId, | ||||
|       entityUpdatedAt: feedExpanded.entityUpdatedAt?.valueOf() || null, | ||||
|       coverPath: feedExpanded.coverPath || null, | ||||
|       meta: { | ||||
|         title: feedExpanded.title, | ||||
|         description: feedExpanded.description, | ||||
|         author: feedExpanded.author, | ||||
|         imageUrl: feedExpanded.imageURL, | ||||
|         feedUrl: feedExpanded.feedURL, | ||||
|         episodes: episodes || [], | ||||
|         createdAt: feedExpanded.createdAt.valueOf(), | ||||
|         updatedAt: feedExpanded.updatedAt.valueOf() | ||||
|       }) | ||||
|     } | ||||
|         link: feedExpanded.siteURL, | ||||
|         explicit: feedExpanded.explicit, | ||||
|         type: feedExpanded.podcastType, | ||||
|         language: feedExpanded.language, | ||||
|         preventIndexing: feedExpanded.preventIndexing, | ||||
|         ownerName: feedExpanded.ownerName, | ||||
|         ownerEmail: feedExpanded.ownerEmail | ||||
|       }, | ||||
|       serverAddress: feedExpanded.serverAddress, | ||||
|       feedUrl: feedExpanded.feedURL, | ||||
|       episodes: episodes || [], | ||||
|       createdAt: feedExpanded.createdAt.valueOf(), | ||||
|       updatedAt: feedExpanded.updatedAt.valueOf() | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|     static removeById(feedId) { | ||||
|       return this.destroy({ | ||||
|         where: { | ||||
|           id: feedId | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Find all library item ids that have an open feed (used in library filter) | ||||
|      * @returns {Promise<Array<String>>} array of library item ids | ||||
|      */ | ||||
|     static async findAllLibraryItemIds() { | ||||
|       const feeds = await this.findAll({ | ||||
|         attributes: ['entityId'], | ||||
|         where: { | ||||
|           entityType: 'libraryItem' | ||||
|         } | ||||
|       }) | ||||
|       return feeds.map(f => f.entityId).filter(f => f) || [] | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Find feed where and return oldFeed | ||||
|      * @param {object} where sequelize where object | ||||
|      * @returns {Promise<objects.Feed>} oldFeed | ||||
|      */ | ||||
|     static async findOneOld(where) { | ||||
|       if (!where) return null | ||||
|       const feedExpanded = await this.findOne({ | ||||
|         where, | ||||
|         include: { | ||||
|           model: sequelize.models.feedEpisode | ||||
|         } | ||||
|       }) | ||||
|       if (!feedExpanded) return null | ||||
|       return this.getOldFeed(feedExpanded) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Find feed and return oldFeed | ||||
|      * @param {string} id | ||||
|      * @returns {Promise<objects.Feed>} oldFeed | ||||
|      */ | ||||
|     static async findByPkOld(id) { | ||||
|       if (!id) return null | ||||
|       const feedExpanded = await this.findByPk(id, { | ||||
|         include: { | ||||
|           model: sequelize.models.feedEpisode | ||||
|         } | ||||
|       }) | ||||
|       if (!feedExpanded) return null | ||||
|       return this.getOldFeed(feedExpanded) | ||||
|     } | ||||
| 
 | ||||
|     static async fullCreateFromOld(oldFeed) { | ||||
|       const feedObj = this.getFromOld(oldFeed) | ||||
|       const newFeed = await this.create(feedObj) | ||||
| 
 | ||||
|       if (oldFeed.episodes?.length) { | ||||
|         for (const oldFeedEpisode of oldFeed.episodes) { | ||||
|           const feedEpisode = sequelize.models.feedEpisode.getFromOld(oldFeedEpisode) | ||||
|           feedEpisode.feedId = newFeed.id | ||||
|           await sequelize.models.feedEpisode.create(feedEpisode) | ||||
|         } | ||||
|   static removeById(feedId) { | ||||
|     return this.destroy({ | ||||
|       where: { | ||||
|         id: feedId | ||||
|       } | ||||
|     } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|     static async fullUpdateFromOld(oldFeed) { | ||||
|       const oldFeedEpisodes = oldFeed.episodes || [] | ||||
|       const feedObj = this.getFromOld(oldFeed) | ||||
| 
 | ||||
|       const existingFeed = await this.findByPk(feedObj.id, { | ||||
|         include: sequelize.models.feedEpisode | ||||
|       }) | ||||
|       if (!existingFeed) return false | ||||
| 
 | ||||
|       let hasUpdates = false | ||||
|       for (const feedEpisode of existingFeed.feedEpisodes) { | ||||
|         const oldFeedEpisode = oldFeedEpisodes.find(ep => ep.id === feedEpisode.id) | ||||
|         // Episode removed
 | ||||
|         if (!oldFeedEpisode) { | ||||
|           feedEpisode.destroy() | ||||
|         } else { | ||||
|           let episodeHasUpdates = false | ||||
|           const oldFeedEpisodeCleaned = sequelize.models.feedEpisode.getFromOld(oldFeedEpisode) | ||||
|           for (const key in oldFeedEpisodeCleaned) { | ||||
|             if (!areEquivalent(oldFeedEpisodeCleaned[key], feedEpisode[key])) { | ||||
|               episodeHasUpdates = true | ||||
|             } | ||||
|           } | ||||
|           if (episodeHasUpdates) { | ||||
|             await feedEpisode.update(oldFeedEpisodeCleaned) | ||||
|             hasUpdates = true | ||||
|           } | ||||
|         } | ||||
|   /** | ||||
|    * Find all library item ids that have an open feed (used in library filter) | ||||
|    * @returns {Promise<Array<String>>} array of library item ids | ||||
|    */ | ||||
|   static async findAllLibraryItemIds() { | ||||
|     const feeds = await this.findAll({ | ||||
|       attributes: ['entityId'], | ||||
|       where: { | ||||
|         entityType: 'libraryItem' | ||||
|       } | ||||
|     }) | ||||
|     return feeds.map(f => f.entityId).filter(f => f) || [] | ||||
|   } | ||||
| 
 | ||||
|       let feedHasUpdates = false | ||||
|       for (const key in feedObj) { | ||||
|         let existingValue = existingFeed[key] | ||||
|         if (existingValue instanceof Date) existingValue = existingValue.valueOf() | ||||
| 
 | ||||
|         if (!areEquivalent(existingValue, feedObj[key])) { | ||||
|           feedHasUpdates = true | ||||
|         } | ||||
|   /** | ||||
|    * Find feed where and return oldFeed | ||||
|    * @param {object} where sequelize where object | ||||
|    * @returns {Promise<objects.Feed>} oldFeed | ||||
|    */ | ||||
|   static async findOneOld(where) { | ||||
|     if (!where) return null | ||||
|     const feedExpanded = await this.findOne({ | ||||
|       where, | ||||
|       include: { | ||||
|         model: this.sequelize.models.feedEpisode | ||||
|       } | ||||
|     }) | ||||
|     if (!feedExpanded) return null | ||||
|     return this.getOldFeed(feedExpanded) | ||||
|   } | ||||
| 
 | ||||
|       if (feedHasUpdates) { | ||||
|         await existingFeed.update(feedObj) | ||||
|         hasUpdates = true | ||||
|   /** | ||||
|    * Find feed and return oldFeed | ||||
|    * @param {string} id | ||||
|    * @returns {Promise<objects.Feed>} oldFeed | ||||
|    */ | ||||
|   static async findByPkOld(id) { | ||||
|     if (!id) return null | ||||
|     const feedExpanded = await this.findByPk(id, { | ||||
|       include: { | ||||
|         model: this.sequelize.models.feedEpisode | ||||
|       } | ||||
|     }) | ||||
|     if (!feedExpanded) return null | ||||
|     return this.getOldFeed(feedExpanded) | ||||
|   } | ||||
| 
 | ||||
|       return hasUpdates | ||||
|     } | ||||
|   static async fullCreateFromOld(oldFeed) { | ||||
|     const feedObj = this.getFromOld(oldFeed) | ||||
|     const newFeed = await this.create(feedObj) | ||||
| 
 | ||||
|     static getFromOld(oldFeed) { | ||||
|       const oldFeedMeta = oldFeed.meta || {} | ||||
|       return { | ||||
|         id: oldFeed.id, | ||||
|         slug: oldFeed.slug, | ||||
|         entityType: oldFeed.entityType, | ||||
|         entityId: oldFeed.entityId, | ||||
|         entityUpdatedAt: oldFeed.entityUpdatedAt, | ||||
|         serverAddress: oldFeed.serverAddress, | ||||
|         feedURL: oldFeed.feedUrl, | ||||
|         coverPath: oldFeed.coverPath || null, | ||||
|         imageURL: oldFeedMeta.imageUrl, | ||||
|         siteURL: oldFeedMeta.link, | ||||
|         title: oldFeedMeta.title, | ||||
|         description: oldFeedMeta.description, | ||||
|         author: oldFeedMeta.author, | ||||
|         podcastType: oldFeedMeta.type || null, | ||||
|         language: oldFeedMeta.language || null, | ||||
|         ownerName: oldFeedMeta.ownerName || null, | ||||
|         ownerEmail: oldFeedMeta.ownerEmail || null, | ||||
|         explicit: !!oldFeedMeta.explicit, | ||||
|         preventIndexing: !!oldFeedMeta.preventIndexing, | ||||
|         userId: oldFeed.userId | ||||
|     if (oldFeed.episodes?.length) { | ||||
|       for (const oldFeedEpisode of oldFeed.episodes) { | ||||
|         const feedEpisode = this.sequelize.models.feedEpisode.getFromOld(oldFeedEpisode) | ||||
|         feedEpisode.feedId = newFeed.id | ||||
|         await this.sequelize.models.feedEpisode.create(feedEpisode) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     getEntity(options) { | ||||
|       if (!this.entityType) return Promise.resolve(null) | ||||
|       const mixinMethodName = `get${sequelize.uppercaseFirst(this.entityType)}` | ||||
|       return this[mixinMethodName](options) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   Feed.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     slug: DataTypes.STRING, | ||||
|     entityType: DataTypes.STRING, | ||||
|     entityId: DataTypes.UUIDV4, | ||||
|     entityUpdatedAt: DataTypes.DATE, | ||||
|     serverAddress: DataTypes.STRING, | ||||
|     feedURL: DataTypes.STRING, | ||||
|     imageURL: DataTypes.STRING, | ||||
|     siteURL: DataTypes.STRING, | ||||
|     title: DataTypes.STRING, | ||||
|     description: DataTypes.TEXT, | ||||
|     author: DataTypes.STRING, | ||||
|     podcastType: DataTypes.STRING, | ||||
|     language: DataTypes.STRING, | ||||
|     ownerName: DataTypes.STRING, | ||||
|     ownerEmail: DataTypes.STRING, | ||||
|     explicit: DataTypes.BOOLEAN, | ||||
|     preventIndexing: DataTypes.BOOLEAN, | ||||
|     coverPath: DataTypes.STRING | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'feed' | ||||
|   }) | ||||
|   static async fullUpdateFromOld(oldFeed) { | ||||
|     const oldFeedEpisodes = oldFeed.episodes || [] | ||||
|     const feedObj = this.getFromOld(oldFeed) | ||||
| 
 | ||||
|   const { user, libraryItem, collection, series, playlist } = sequelize.models | ||||
|     const existingFeed = await this.findByPk(feedObj.id, { | ||||
|       include: this.sequelize.models.feedEpisode | ||||
|     }) | ||||
|     if (!existingFeed) return false | ||||
| 
 | ||||
|   user.hasMany(Feed) | ||||
|   Feed.belongsTo(user) | ||||
| 
 | ||||
|   libraryItem.hasMany(Feed, { | ||||
|     foreignKey: 'entityId', | ||||
|     constraints: false, | ||||
|     scope: { | ||||
|       entityType: 'libraryItem' | ||||
|     } | ||||
|   }) | ||||
|   Feed.belongsTo(libraryItem, { foreignKey: 'entityId', constraints: false }) | ||||
| 
 | ||||
|   collection.hasMany(Feed, { | ||||
|     foreignKey: 'entityId', | ||||
|     constraints: false, | ||||
|     scope: { | ||||
|       entityType: 'collection' | ||||
|     } | ||||
|   }) | ||||
|   Feed.belongsTo(collection, { foreignKey: 'entityId', constraints: false }) | ||||
| 
 | ||||
|   series.hasMany(Feed, { | ||||
|     foreignKey: 'entityId', | ||||
|     constraints: false, | ||||
|     scope: { | ||||
|       entityType: 'series' | ||||
|     } | ||||
|   }) | ||||
|   Feed.belongsTo(series, { foreignKey: 'entityId', constraints: false }) | ||||
| 
 | ||||
|   playlist.hasMany(Feed, { | ||||
|     foreignKey: 'entityId', | ||||
|     constraints: false, | ||||
|     scope: { | ||||
|       entityType: 'playlist' | ||||
|     } | ||||
|   }) | ||||
|   Feed.belongsTo(playlist, { foreignKey: 'entityId', constraints: false }) | ||||
| 
 | ||||
|   Feed.addHook('afterFind', findResult => { | ||||
|     if (!findResult) return | ||||
| 
 | ||||
|     if (!Array.isArray(findResult)) findResult = [findResult] | ||||
|     for (const instance of findResult) { | ||||
|       if (instance.entityType === 'libraryItem' && instance.libraryItem !== undefined) { | ||||
|         instance.entity = instance.libraryItem | ||||
|         instance.dataValues.entity = instance.dataValues.libraryItem | ||||
|       } else if (instance.entityType === 'collection' && instance.collection !== undefined) { | ||||
|         instance.entity = instance.collection | ||||
|         instance.dataValues.entity = instance.dataValues.collection | ||||
|       } else if (instance.entityType === 'series' && instance.series !== undefined) { | ||||
|         instance.entity = instance.series | ||||
|         instance.dataValues.entity = instance.dataValues.series | ||||
|       } else if (instance.entityType === 'playlist' && instance.playlist !== undefined) { | ||||
|         instance.entity = instance.playlist | ||||
|         instance.dataValues.entity = instance.dataValues.playlist | ||||
|     let hasUpdates = false | ||||
|     for (const feedEpisode of existingFeed.feedEpisodes) { | ||||
|       const oldFeedEpisode = oldFeedEpisodes.find(ep => ep.id === feedEpisode.id) | ||||
|       // Episode removed
 | ||||
|       if (!oldFeedEpisode) { | ||||
|         feedEpisode.destroy() | ||||
|       } else { | ||||
|         let episodeHasUpdates = false | ||||
|         const oldFeedEpisodeCleaned = this.sequelize.models.feedEpisode.getFromOld(oldFeedEpisode) | ||||
|         for (const key in oldFeedEpisodeCleaned) { | ||||
|           if (!areEquivalent(oldFeedEpisodeCleaned[key], feedEpisode[key])) { | ||||
|             episodeHasUpdates = true | ||||
|           } | ||||
|         } | ||||
|         if (episodeHasUpdates) { | ||||
|           await feedEpisode.update(oldFeedEpisodeCleaned) | ||||
|           hasUpdates = true | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       // To prevent mistakes:
 | ||||
|       delete instance.libraryItem | ||||
|       delete instance.dataValues.libraryItem | ||||
|       delete instance.collection | ||||
|       delete instance.dataValues.collection | ||||
|       delete instance.series | ||||
|       delete instance.dataValues.series | ||||
|       delete instance.playlist | ||||
|       delete instance.dataValues.playlist | ||||
|     } | ||||
|   }) | ||||
| 
 | ||||
|   return Feed | ||||
|     let feedHasUpdates = false | ||||
|     for (const key in feedObj) { | ||||
|       let existingValue = existingFeed[key] | ||||
|       if (existingValue instanceof Date) existingValue = existingValue.valueOf() | ||||
| 
 | ||||
|       if (!areEquivalent(existingValue, feedObj[key])) { | ||||
|         feedHasUpdates = true | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     if (feedHasUpdates) { | ||||
|       await existingFeed.update(feedObj) | ||||
|       hasUpdates = true | ||||
|     } | ||||
| 
 | ||||
|     return hasUpdates | ||||
|   } | ||||
| 
 | ||||
|   static getFromOld(oldFeed) { | ||||
|     const oldFeedMeta = oldFeed.meta || {} | ||||
|     return { | ||||
|       id: oldFeed.id, | ||||
|       slug: oldFeed.slug, | ||||
|       entityType: oldFeed.entityType, | ||||
|       entityId: oldFeed.entityId, | ||||
|       entityUpdatedAt: oldFeed.entityUpdatedAt, | ||||
|       serverAddress: oldFeed.serverAddress, | ||||
|       feedURL: oldFeed.feedUrl, | ||||
|       coverPath: oldFeed.coverPath || null, | ||||
|       imageURL: oldFeedMeta.imageUrl, | ||||
|       siteURL: oldFeedMeta.link, | ||||
|       title: oldFeedMeta.title, | ||||
|       description: oldFeedMeta.description, | ||||
|       author: oldFeedMeta.author, | ||||
|       podcastType: oldFeedMeta.type || null, | ||||
|       language: oldFeedMeta.language || null, | ||||
|       ownerName: oldFeedMeta.ownerName || null, | ||||
|       ownerEmail: oldFeedMeta.ownerEmail || null, | ||||
|       explicit: !!oldFeedMeta.explicit, | ||||
|       preventIndexing: !!oldFeedMeta.preventIndexing, | ||||
|       userId: oldFeed.userId | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   getEntity(options) { | ||||
|     if (!this.entityType) return Promise.resolve(null) | ||||
|     const mixinMethodName = `get${this.sequelize.uppercaseFirst(this.entityType)}` | ||||
|     return this[mixinMethodName](options) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Initialize model | ||||
|    *  | ||||
|    * Polymorphic association: Feeds can be created from LibraryItem, Collection, Playlist or Series | ||||
|    * @see https://sequelize.org/docs/v6/advanced-association-concepts/polymorphic-associations/
 | ||||
|    *  | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       slug: DataTypes.STRING, | ||||
|       entityType: DataTypes.STRING, | ||||
|       entityId: DataTypes.UUIDV4, | ||||
|       entityUpdatedAt: DataTypes.DATE, | ||||
|       serverAddress: DataTypes.STRING, | ||||
|       feedURL: DataTypes.STRING, | ||||
|       imageURL: DataTypes.STRING, | ||||
|       siteURL: DataTypes.STRING, | ||||
|       title: DataTypes.STRING, | ||||
|       description: DataTypes.TEXT, | ||||
|       author: DataTypes.STRING, | ||||
|       podcastType: DataTypes.STRING, | ||||
|       language: DataTypes.STRING, | ||||
|       ownerName: DataTypes.STRING, | ||||
|       ownerEmail: DataTypes.STRING, | ||||
|       explicit: DataTypes.BOOLEAN, | ||||
|       preventIndexing: DataTypes.BOOLEAN, | ||||
|       coverPath: DataTypes.STRING | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'feed' | ||||
|     }) | ||||
| 
 | ||||
|     const { user, libraryItem, collection, series, playlist } = sequelize.models | ||||
| 
 | ||||
|     user.hasMany(Feed) | ||||
|     Feed.belongsTo(user) | ||||
| 
 | ||||
|     libraryItem.hasMany(Feed, { | ||||
|       foreignKey: 'entityId', | ||||
|       constraints: false, | ||||
|       scope: { | ||||
|         entityType: 'libraryItem' | ||||
|       } | ||||
|     }) | ||||
|     Feed.belongsTo(libraryItem, { foreignKey: 'entityId', constraints: false }) | ||||
| 
 | ||||
|     collection.hasMany(Feed, { | ||||
|       foreignKey: 'entityId', | ||||
|       constraints: false, | ||||
|       scope: { | ||||
|         entityType: 'collection' | ||||
|       } | ||||
|     }) | ||||
|     Feed.belongsTo(collection, { foreignKey: 'entityId', constraints: false }) | ||||
| 
 | ||||
|     series.hasMany(Feed, { | ||||
|       foreignKey: 'entityId', | ||||
|       constraints: false, | ||||
|       scope: { | ||||
|         entityType: 'series' | ||||
|       } | ||||
|     }) | ||||
|     Feed.belongsTo(series, { foreignKey: 'entityId', constraints: false }) | ||||
| 
 | ||||
|     playlist.hasMany(Feed, { | ||||
|       foreignKey: 'entityId', | ||||
|       constraints: false, | ||||
|       scope: { | ||||
|         entityType: 'playlist' | ||||
|       } | ||||
|     }) | ||||
|     Feed.belongsTo(playlist, { foreignKey: 'entityId', constraints: false }) | ||||
| 
 | ||||
|     Feed.addHook('afterFind', findResult => { | ||||
|       if (!findResult) return | ||||
| 
 | ||||
|       if (!Array.isArray(findResult)) findResult = [findResult] | ||||
|       for (const instance of findResult) { | ||||
|         if (instance.entityType === 'libraryItem' && instance.libraryItem !== undefined) { | ||||
|           instance.entity = instance.libraryItem | ||||
|           instance.dataValues.entity = instance.dataValues.libraryItem | ||||
|         } else if (instance.entityType === 'collection' && instance.collection !== undefined) { | ||||
|           instance.entity = instance.collection | ||||
|           instance.dataValues.entity = instance.dataValues.collection | ||||
|         } else if (instance.entityType === 'series' && instance.series !== undefined) { | ||||
|           instance.entity = instance.series | ||||
|           instance.dataValues.entity = instance.dataValues.series | ||||
|         } else if (instance.entityType === 'playlist' && instance.playlist !== undefined) { | ||||
|           instance.entity = instance.playlist | ||||
|           instance.dataValues.entity = instance.dataValues.playlist | ||||
|         } | ||||
| 
 | ||||
|         // To prevent mistakes:
 | ||||
|         delete instance.libraryItem | ||||
|         delete instance.dataValues.libraryItem | ||||
|         delete instance.collection | ||||
|         delete instance.dataValues.collection | ||||
|         delete instance.series | ||||
|         delete instance.dataValues.series | ||||
|         delete instance.playlist | ||||
|         delete instance.dataValues.playlist | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = Feed | ||||
| @ -1,82 +1,125 @@ | ||||
| const { DataTypes, Model } = require('sequelize') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class FeedEpisode extends Model { | ||||
|     getOldEpisode() { | ||||
|       const enclosure = { | ||||
|         url: this.enclosureURL, | ||||
|         size: this.enclosureSize, | ||||
|         type: this.enclosureType | ||||
|       } | ||||
|       return { | ||||
|         id: this.id, | ||||
|         title: this.title, | ||||
|         description: this.description, | ||||
|         enclosure, | ||||
|         pubDate: this.pubDate, | ||||
|         link: this.siteURL, | ||||
|         author: this.author, | ||||
|         explicit: this.explicit, | ||||
|         duration: this.duration, | ||||
|         season: this.season, | ||||
|         episode: this.episode, | ||||
|         episodeType: this.episodeType, | ||||
|         fullPath: this.filePath | ||||
|       } | ||||
|     } | ||||
| class FeedEpisode extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     static getFromOld(oldFeedEpisode) { | ||||
|       return { | ||||
|         id: oldFeedEpisode.id, | ||||
|         title: oldFeedEpisode.title, | ||||
|         author: oldFeedEpisode.author, | ||||
|         description: oldFeedEpisode.description, | ||||
|         siteURL: oldFeedEpisode.link, | ||||
|         enclosureURL: oldFeedEpisode.enclosure?.url || null, | ||||
|         enclosureType: oldFeedEpisode.enclosure?.type || null, | ||||
|         enclosureSize: oldFeedEpisode.enclosure?.size || null, | ||||
|         pubDate: oldFeedEpisode.pubDate, | ||||
|         season: oldFeedEpisode.season || null, | ||||
|         episode: oldFeedEpisode.episode || null, | ||||
|         episodeType: oldFeedEpisode.episodeType || null, | ||||
|         duration: oldFeedEpisode.duration, | ||||
|         filePath: oldFeedEpisode.fullPath, | ||||
|         explicit: !!oldFeedEpisode.explicit | ||||
|       } | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {string} */ | ||||
|     this.title | ||||
|     /** @type {string} */ | ||||
|     this.description | ||||
|     /** @type {string} */ | ||||
|     this.siteURL | ||||
|     /** @type {string} */ | ||||
|     this.enclosureURL | ||||
|     /** @type {string} */ | ||||
|     this.enclosureType | ||||
|     /** @type {BigInt} */ | ||||
|     this.enclosureSize | ||||
|     /** @type {string} */ | ||||
|     this.pubDate | ||||
|     /** @type {string} */ | ||||
|     this.season | ||||
|     /** @type {string} */ | ||||
|     this.episode | ||||
|     /** @type {string} */ | ||||
|     this.episodeType | ||||
|     /** @type {number} */ | ||||
|     this.duration | ||||
|     /** @type {string} */ | ||||
|     this.filePath | ||||
|     /** @type {boolean} */ | ||||
|     this.explicit | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.feedId | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|   } | ||||
| 
 | ||||
|   getOldEpisode() { | ||||
|     const enclosure = { | ||||
|       url: this.enclosureURL, | ||||
|       size: this.enclosureSize, | ||||
|       type: this.enclosureType | ||||
|     } | ||||
|     return { | ||||
|       id: this.id, | ||||
|       title: this.title, | ||||
|       description: this.description, | ||||
|       enclosure, | ||||
|       pubDate: this.pubDate, | ||||
|       link: this.siteURL, | ||||
|       author: this.author, | ||||
|       explicit: this.explicit, | ||||
|       duration: this.duration, | ||||
|       season: this.season, | ||||
|       episode: this.episode, | ||||
|       episodeType: this.episodeType, | ||||
|       fullPath: this.filePath | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   FeedEpisode.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     title: DataTypes.STRING, | ||||
|     author: DataTypes.STRING, | ||||
|     description: DataTypes.TEXT, | ||||
|     siteURL: DataTypes.STRING, | ||||
|     enclosureURL: DataTypes.STRING, | ||||
|     enclosureType: DataTypes.STRING, | ||||
|     enclosureSize: DataTypes.BIGINT, | ||||
|     pubDate: DataTypes.STRING, | ||||
|     season: DataTypes.STRING, | ||||
|     episode: DataTypes.STRING, | ||||
|     episodeType: DataTypes.STRING, | ||||
|     duration: DataTypes.FLOAT, | ||||
|     filePath: DataTypes.STRING, | ||||
|     explicit: DataTypes.BOOLEAN | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'feedEpisode' | ||||
|   }) | ||||
|   static getFromOld(oldFeedEpisode) { | ||||
|     return { | ||||
|       id: oldFeedEpisode.id, | ||||
|       title: oldFeedEpisode.title, | ||||
|       author: oldFeedEpisode.author, | ||||
|       description: oldFeedEpisode.description, | ||||
|       siteURL: oldFeedEpisode.link, | ||||
|       enclosureURL: oldFeedEpisode.enclosure?.url || null, | ||||
|       enclosureType: oldFeedEpisode.enclosure?.type || null, | ||||
|       enclosureSize: oldFeedEpisode.enclosure?.size || null, | ||||
|       pubDate: oldFeedEpisode.pubDate, | ||||
|       season: oldFeedEpisode.season || null, | ||||
|       episode: oldFeedEpisode.episode || null, | ||||
|       episodeType: oldFeedEpisode.episodeType || null, | ||||
|       duration: oldFeedEpisode.duration, | ||||
|       filePath: oldFeedEpisode.fullPath, | ||||
|       explicit: !!oldFeedEpisode.explicit | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const { feed } = sequelize.models | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       title: DataTypes.STRING, | ||||
|       author: DataTypes.STRING, | ||||
|       description: DataTypes.TEXT, | ||||
|       siteURL: DataTypes.STRING, | ||||
|       enclosureURL: DataTypes.STRING, | ||||
|       enclosureType: DataTypes.STRING, | ||||
|       enclosureSize: DataTypes.BIGINT, | ||||
|       pubDate: DataTypes.STRING, | ||||
|       season: DataTypes.STRING, | ||||
|       episode: DataTypes.STRING, | ||||
|       episodeType: DataTypes.STRING, | ||||
|       duration: DataTypes.FLOAT, | ||||
|       filePath: DataTypes.STRING, | ||||
|       explicit: DataTypes.BOOLEAN | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'feedEpisode' | ||||
|     }) | ||||
| 
 | ||||
|   feed.hasMany(FeedEpisode, { | ||||
|     onDelete: 'CASCADE' | ||||
|   }) | ||||
|   FeedEpisode.belongsTo(feed) | ||||
|     const { feed } = sequelize.models | ||||
| 
 | ||||
|   return FeedEpisode | ||||
|     feed.hasMany(FeedEpisode, { | ||||
|       onDelete: 'CASCADE' | ||||
|     }) | ||||
|     FeedEpisode.belongsTo(feed) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = FeedEpisode | ||||
| @ -2,217 +2,251 @@ const { DataTypes, Model } = require('sequelize') | ||||
| const Logger = require('../Logger') | ||||
| const oldLibrary = require('../objects/Library') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class Library extends Model { | ||||
|     /** | ||||
|      * Get all old libraries | ||||
|      * @returns {Promise<oldLibrary[]>} | ||||
|      */ | ||||
|     static async getAllOldLibraries() { | ||||
|       const libraries = await this.findAll({ | ||||
|         include: sequelize.models.libraryFolder, | ||||
|         order: [['displayOrder', 'ASC']] | ||||
|       }) | ||||
|       return libraries.map(lib => this.getOldLibrary(lib)) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Convert expanded Library to oldLibrary | ||||
|      * @param {Library} libraryExpanded  | ||||
|      * @returns {Promise<oldLibrary>} | ||||
|      */ | ||||
|     static getOldLibrary(libraryExpanded) { | ||||
|       const folders = libraryExpanded.libraryFolders.map(folder => { | ||||
|         return { | ||||
|           id: folder.id, | ||||
|           fullPath: folder.path, | ||||
|           libraryId: folder.libraryId, | ||||
|           addedAt: folder.createdAt.valueOf() | ||||
|         } | ||||
|       }) | ||||
|       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, | ||||
|         createdAt: libraryExpanded.createdAt.valueOf(), | ||||
|         lastUpdate: libraryExpanded.updatedAt.valueOf() | ||||
|       }) | ||||
|     } | ||||
| class Library extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     /** | ||||
|      * @param {object} oldLibrary  | ||||
|      * @returns {Library|null} | ||||
|      */ | ||||
|     static async createFromOld(oldLibrary) { | ||||
|       const library = this.getFromOld(oldLibrary) | ||||
|     /** @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 | ||||
|     /** @type {Object} */ | ||||
|     this.settings | ||||
|     /** @type {Object} */ | ||||
|     this.extraData | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|   } | ||||
| 
 | ||||
|       library.libraryFolders = oldLibrary.folders.map(folder => { | ||||
|         return { | ||||
|           id: folder.id, | ||||
|           path: folder.fullPath | ||||
|         } | ||||
|       }) | ||||
|   /** | ||||
|    * Get all old libraries | ||||
|    * @returns {Promise<oldLibrary[]>} | ||||
|    */ | ||||
|   static async getAllOldLibraries() { | ||||
|     const libraries = await this.findAll({ | ||||
|       include: this.sequelize.models.libraryFolder, | ||||
|       order: [['displayOrder', 'ASC']] | ||||
|     }) | ||||
|     return libraries.map(lib => this.getOldLibrary(lib)) | ||||
|   } | ||||
| 
 | ||||
|       return this.create(library, { | ||||
|         include: sequelize.models.libraryFolder | ||||
|       }).catch((error) => { | ||||
|         Logger.error(`[Library] Failed to create library ${library.id}`, error) | ||||
|         return null | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Update library and library folders | ||||
|      * @param {object} oldLibrary  | ||||
|      * @returns  | ||||
|      */ | ||||
|     static async updateFromOld(oldLibrary) { | ||||
|       const existingLibrary = await this.findByPk(oldLibrary.id, { | ||||
|         include: sequelize.models.libraryFolder | ||||
|       }) | ||||
|       if (!existingLibrary) { | ||||
|         Logger.error(`[Library] Failed to update library ${oldLibrary.id} - not found`) | ||||
|         return null | ||||
|       } | ||||
| 
 | ||||
|       const library = this.getFromOld(oldLibrary) | ||||
| 
 | ||||
|       const libraryFolders = oldLibrary.folders.map(folder => { | ||||
|         return { | ||||
|           id: folder.id, | ||||
|           path: folder.fullPath, | ||||
|           libraryId: library.id | ||||
|         } | ||||
|       }) | ||||
|       for (const libraryFolder of libraryFolders) { | ||||
|         const existingLibraryFolder = existingLibrary.libraryFolders.find(lf => lf.id === libraryFolder.id) | ||||
|         if (!existingLibraryFolder) { | ||||
|           await sequelize.models.libraryFolder.create(libraryFolder) | ||||
|         } else if (existingLibraryFolder.path !== libraryFolder.path) { | ||||
|           await existingLibraryFolder.update({ path: libraryFolder.path }) | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       const libraryFoldersRemoved = existingLibrary.libraryFolders.filter(lf => !libraryFolders.some(_lf => _lf.id === lf.id)) | ||||
|       for (const existingLibraryFolder of libraryFoldersRemoved) { | ||||
|         await existingLibraryFolder.destroy() | ||||
|       } | ||||
| 
 | ||||
|       return existingLibrary.update(library) | ||||
|     } | ||||
| 
 | ||||
|     static getFromOld(oldLibrary) { | ||||
|       const extraData = {} | ||||
|       if (oldLibrary.oldLibraryId) { | ||||
|         extraData.oldLibraryId = oldLibrary.oldLibraryId | ||||
|       } | ||||
|   /** | ||||
|    * Convert expanded Library to oldLibrary | ||||
|    * @param {Library} libraryExpanded  | ||||
|    * @returns {Promise<oldLibrary>} | ||||
|    */ | ||||
|   static getOldLibrary(libraryExpanded) { | ||||
|     const folders = libraryExpanded.libraryFolders.map(folder => { | ||||
|       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() || {}, | ||||
|         createdAt: oldLibrary.createdAt, | ||||
|         updatedAt: oldLibrary.lastUpdate, | ||||
|         extraData | ||||
|         id: folder.id, | ||||
|         fullPath: folder.path, | ||||
|         libraryId: folder.libraryId, | ||||
|         addedAt: folder.createdAt.valueOf() | ||||
|       } | ||||
|     }) | ||||
|     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, | ||||
|       createdAt: libraryExpanded.createdAt.valueOf(), | ||||
|       lastUpdate: libraryExpanded.updatedAt.valueOf() | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * @param {object} oldLibrary  | ||||
|    * @returns {Library|null} | ||||
|    */ | ||||
|   static async createFromOld(oldLibrary) { | ||||
|     const library = this.getFromOld(oldLibrary) | ||||
| 
 | ||||
|     library.libraryFolders = oldLibrary.folders.map(folder => { | ||||
|       return { | ||||
|         id: folder.id, | ||||
|         path: folder.fullPath | ||||
|       } | ||||
|     }) | ||||
| 
 | ||||
|     return this.create(library, { | ||||
|       include: this.sequelize.models.libraryFolder | ||||
|     }).catch((error) => { | ||||
|       Logger.error(`[Library] Failed to create library ${library.id}`, error) | ||||
|       return null | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Update library and library folders | ||||
|    * @param {object} oldLibrary  | ||||
|    * @returns  | ||||
|    */ | ||||
|   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 | ||||
|     } | ||||
| 
 | ||||
|     const library = this.getFromOld(oldLibrary) | ||||
| 
 | ||||
|     const libraryFolders = oldLibrary.folders.map(folder => { | ||||
|       return { | ||||
|         id: folder.id, | ||||
|         path: folder.fullPath, | ||||
|         libraryId: library.id | ||||
|       } | ||||
|     }) | ||||
|     for (const libraryFolder of libraryFolders) { | ||||
|       const existingLibraryFolder = existingLibrary.libraryFolders.find(lf => lf.id === libraryFolder.id) | ||||
|       if (!existingLibraryFolder) { | ||||
|         await this.sequelize.models.libraryFolder.create(libraryFolder) | ||||
|       } else if (existingLibraryFolder.path !== libraryFolder.path) { | ||||
|         await existingLibraryFolder.update({ path: libraryFolder.path }) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Destroy library by id | ||||
|      * @param {string} libraryId  | ||||
|      * @returns  | ||||
|      */ | ||||
|     static removeById(libraryId) { | ||||
|       return this.destroy({ | ||||
|         where: { | ||||
|           id: libraryId | ||||
|         } | ||||
|       }) | ||||
|     const libraryFoldersRemoved = existingLibrary.libraryFolders.filter(lf => !libraryFolders.some(_lf => _lf.id === lf.id)) | ||||
|     for (const existingLibraryFolder of libraryFoldersRemoved) { | ||||
|       await existingLibraryFolder.destroy() | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 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']] | ||||
|       }) | ||||
|       return libraries.map(l => l.id) | ||||
|     } | ||||
|     return existingLibrary.update(library) | ||||
|   } | ||||
| 
 | ||||
|     /** | ||||
|      * Find Library by primary key & return oldLibrary | ||||
|      * @param {string} libraryId  | ||||
|      * @returns {Promise<oldLibrary|null>} Returns null if not found | ||||
|      */ | ||||
|     static async getOldById(libraryId) { | ||||
|       if (!libraryId) return null | ||||
|       const library = await this.findByPk(libraryId, { | ||||
|         include: sequelize.models.libraryFolder | ||||
|       }) | ||||
|       if (!library) return null | ||||
|       return this.getOldLibrary(library) | ||||
|   static getFromOld(oldLibrary) { | ||||
|     const extraData = {} | ||||
|     if (oldLibrary.oldLibraryId) { | ||||
|       extraData.oldLibraryId = oldLibrary.oldLibraryId | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 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 | ||||
|     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() || {}, | ||||
|       createdAt: oldLibrary.createdAt, | ||||
|       updatedAt: oldLibrary.lastUpdate, | ||||
|       extraData | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|     /** | ||||
|      * 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) { | ||||
|           Logger.dev(`[Library] Updating display order of library from ${library.displayOrder} to ${i + 1}`) | ||||
|           await library.update({ displayOrder: i + 1 }).catch((error) => { | ||||
|             Logger.error(`[Library] Failed to update library display order to ${i + 1}`, error) | ||||
|           }) | ||||
|         } | ||||
|   /** | ||||
|    * Destroy library by id | ||||
|    * @param {string} libraryId  | ||||
|    * @returns  | ||||
|    */ | ||||
|   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']] | ||||
|     }) | ||||
|     return libraries.map(l => l.id) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Find Library by primary key & return oldLibrary | ||||
|    * @param {string} libraryId  | ||||
|    * @returns {Promise<oldLibrary|null>} Returns null if not found | ||||
|    */ | ||||
|   static async getOldById(libraryId) { | ||||
|     if (!libraryId) return null | ||||
|     const library = await this.findByPk(libraryId, { | ||||
|       include: this.sequelize.models.libraryFolder | ||||
|     }) | ||||
|     if (!library) return null | ||||
|     return this.getOldLibrary(library) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * 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) { | ||||
|         Logger.dev(`[Library] Updating display order of library from ${library.displayOrder} to ${i + 1}`) | ||||
|         await library.update({ displayOrder: i + 1 }).catch((error) => { | ||||
|           Logger.error(`[Library] Failed to update library display order to ${i + 1}`, error) | ||||
|         }) | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   Library.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 | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'library' | ||||
|   }) | ||||
| 
 | ||||
|   return Library | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     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 | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'library' | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = Library | ||||
| @ -1,36 +1,55 @@ | ||||
| const { DataTypes, Model } = require('sequelize') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class LibraryFolder extends Model { | ||||
|     /** | ||||
|      * Gets all library folder path strings | ||||
|      * @returns {Promise<string[]>} array of library folder paths | ||||
|      */ | ||||
|     static async getAllLibraryFolderPaths() { | ||||
|       const libraryFolders = await this.findAll({ | ||||
|         attributes: ['path'] | ||||
|       }) | ||||
|       return libraryFolders.map(l => l.path) | ||||
|     } | ||||
| class LibraryFolder extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {string} */ | ||||
|     this.path | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.libraryId | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|   } | ||||
| 
 | ||||
|   LibraryFolder.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     path: DataTypes.STRING | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'libraryFolder' | ||||
|   }) | ||||
|   /** | ||||
|    * Gets all library folder path strings | ||||
|    * @returns {Promise<string[]>} array of library folder paths | ||||
|    */ | ||||
|   static async getAllLibraryFolderPaths() { | ||||
|     const libraryFolders = await this.findAll({ | ||||
|       attributes: ['path'] | ||||
|     }) | ||||
|     return libraryFolders.map(l => l.path) | ||||
|   } | ||||
| 
 | ||||
|   const { library } = sequelize.models | ||||
|   library.hasMany(LibraryFolder, { | ||||
|     onDelete: 'CASCADE' | ||||
|   }) | ||||
|   LibraryFolder.belongsTo(library) | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       path: DataTypes.STRING | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'libraryFolder' | ||||
|     }) | ||||
| 
 | ||||
|   return LibraryFolder | ||||
|     const { library } = sequelize.models | ||||
|     library.hasMany(LibraryFolder, { | ||||
|       onDelete: 'CASCADE' | ||||
|     }) | ||||
|     LibraryFolder.belongsTo(library) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = LibraryFolder | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -1,148 +1,184 @@ | ||||
| const { DataTypes, Model } = require('sequelize') | ||||
| 
 | ||||
| /* | ||||
|  * Polymorphic association: https://sequelize.org/docs/v6/advanced-association-concepts/polymorphic-associations/
 | ||||
|  * Book has many MediaProgress. PodcastEpisode has many MediaProgress. | ||||
|  */ | ||||
| module.exports = (sequelize) => { | ||||
|   class MediaProgress extends Model { | ||||
|     getOldMediaProgress() { | ||||
|       const isPodcastEpisode = this.mediaItemType === 'podcastEpisode' | ||||
| class MediaProgress extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|       return { | ||||
|         id: this.id, | ||||
|         userId: this.userId, | ||||
|         libraryItemId: this.extraData?.libraryItemId || null, | ||||
|         episodeId: isPodcastEpisode ? this.mediaItemId : null, | ||||
|         mediaItemId: this.mediaItemId, | ||||
|         mediaItemType: this.mediaItemType, | ||||
|         duration: this.duration, | ||||
|         progress: this.extraData?.progress || 0, | ||||
|         currentTime: this.currentTime, | ||||
|         isFinished: !!this.isFinished, | ||||
|         hideFromContinueListening: !!this.hideFromContinueListening, | ||||
|         ebookLocation: this.ebookLocation, | ||||
|         ebookProgress: this.ebookProgress, | ||||
|         lastUpdate: this.updatedAt.valueOf(), | ||||
|         startedAt: this.createdAt.valueOf(), | ||||
|         finishedAt: this.finishedAt?.valueOf() || null | ||||
|       } | ||||
|     } | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.mediaItemId | ||||
|     /** @type {string} */ | ||||
|     this.mediaItemType | ||||
|     /** @type {number} */ | ||||
|     this.duration | ||||
|     /** @type {number} */ | ||||
|     this.currentTime | ||||
|     /** @type {boolean} */ | ||||
|     this.isFinished | ||||
|     /** @type {boolean} */ | ||||
|     this.hideFromContinueListening | ||||
|     /** @type {string} */ | ||||
|     this.ebookLocation | ||||
|     /** @type {number} */ | ||||
|     this.ebookProgress | ||||
|     /** @type {Date} */ | ||||
|     this.finishedAt | ||||
|     /** @type {Object} */ | ||||
|     this.extraData | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.userId | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|   } | ||||
| 
 | ||||
|     static upsertFromOld(oldMediaProgress) { | ||||
|       const mediaProgress = this.getFromOld(oldMediaProgress) | ||||
|       return this.upsert(mediaProgress) | ||||
|     } | ||||
|   getOldMediaProgress() { | ||||
|     const isPodcastEpisode = this.mediaItemType === 'podcastEpisode' | ||||
| 
 | ||||
|     static getFromOld(oldMediaProgress) { | ||||
|       return { | ||||
|         id: oldMediaProgress.id, | ||||
|         userId: oldMediaProgress.userId, | ||||
|         mediaItemId: oldMediaProgress.mediaItemId, | ||||
|         mediaItemType: oldMediaProgress.mediaItemType, | ||||
|         duration: oldMediaProgress.duration, | ||||
|         currentTime: oldMediaProgress.currentTime, | ||||
|         ebookLocation: oldMediaProgress.ebookLocation || null, | ||||
|         ebookProgress: oldMediaProgress.ebookProgress || null, | ||||
|         isFinished: !!oldMediaProgress.isFinished, | ||||
|         hideFromContinueListening: !!oldMediaProgress.hideFromContinueListening, | ||||
|         finishedAt: oldMediaProgress.finishedAt, | ||||
|         createdAt: oldMediaProgress.startedAt || oldMediaProgress.lastUpdate, | ||||
|         updatedAt: oldMediaProgress.lastUpdate, | ||||
|         extraData: { | ||||
|           libraryItemId: oldMediaProgress.libraryItemId, | ||||
|           progress: oldMediaProgress.progress | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     static removeById(mediaProgressId) { | ||||
|       return this.destroy({ | ||||
|         where: { | ||||
|           id: mediaProgressId | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     getMediaItem(options) { | ||||
|       if (!this.mediaItemType) return Promise.resolve(null) | ||||
|       const mixinMethodName = `get${sequelize.uppercaseFirst(this.mediaItemType)}` | ||||
|       return this[mixinMethodName](options) | ||||
|     return { | ||||
|       id: this.id, | ||||
|       userId: this.userId, | ||||
|       libraryItemId: this.extraData?.libraryItemId || null, | ||||
|       episodeId: isPodcastEpisode ? this.mediaItemId : null, | ||||
|       mediaItemId: this.mediaItemId, | ||||
|       mediaItemType: this.mediaItemType, | ||||
|       duration: this.duration, | ||||
|       progress: this.extraData?.progress || 0, | ||||
|       currentTime: this.currentTime, | ||||
|       isFinished: !!this.isFinished, | ||||
|       hideFromContinueListening: !!this.hideFromContinueListening, | ||||
|       ebookLocation: this.ebookLocation, | ||||
|       ebookProgress: this.ebookProgress, | ||||
|       lastUpdate: this.updatedAt.valueOf(), | ||||
|       startedAt: this.createdAt.valueOf(), | ||||
|       finishedAt: this.finishedAt?.valueOf() || null | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   static upsertFromOld(oldMediaProgress) { | ||||
|     const mediaProgress = this.getFromOld(oldMediaProgress) | ||||
|     return this.upsert(mediaProgress) | ||||
|   } | ||||
| 
 | ||||
|   MediaProgress.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     mediaItemId: DataTypes.UUIDV4, | ||||
|     mediaItemType: DataTypes.STRING, | ||||
|     duration: DataTypes.FLOAT, | ||||
|     currentTime: DataTypes.FLOAT, | ||||
|     isFinished: DataTypes.BOOLEAN, | ||||
|     hideFromContinueListening: DataTypes.BOOLEAN, | ||||
|     ebookLocation: DataTypes.STRING, | ||||
|     ebookProgress: DataTypes.FLOAT, | ||||
|     finishedAt: DataTypes.DATE, | ||||
|     extraData: DataTypes.JSON | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'mediaProgress', | ||||
|     indexes: [ | ||||
|       { | ||||
|         fields: ['updatedAt'] | ||||
|   static getFromOld(oldMediaProgress) { | ||||
|     return { | ||||
|       id: oldMediaProgress.id, | ||||
|       userId: oldMediaProgress.userId, | ||||
|       mediaItemId: oldMediaProgress.mediaItemId, | ||||
|       mediaItemType: oldMediaProgress.mediaItemType, | ||||
|       duration: oldMediaProgress.duration, | ||||
|       currentTime: oldMediaProgress.currentTime, | ||||
|       ebookLocation: oldMediaProgress.ebookLocation || null, | ||||
|       ebookProgress: oldMediaProgress.ebookProgress || null, | ||||
|       isFinished: !!oldMediaProgress.isFinished, | ||||
|       hideFromContinueListening: !!oldMediaProgress.hideFromContinueListening, | ||||
|       finishedAt: oldMediaProgress.finishedAt, | ||||
|       createdAt: oldMediaProgress.startedAt || oldMediaProgress.lastUpdate, | ||||
|       updatedAt: oldMediaProgress.lastUpdate, | ||||
|       extraData: { | ||||
|         libraryItemId: oldMediaProgress.libraryItemId, | ||||
|         progress: oldMediaProgress.progress | ||||
|       } | ||||
|     ] | ||||
|   }) | ||||
| 
 | ||||
|   const { book, podcastEpisode, user } = sequelize.models | ||||
| 
 | ||||
|   book.hasMany(MediaProgress, { | ||||
|     foreignKey: 'mediaItemId', | ||||
|     constraints: false, | ||||
|     scope: { | ||||
|       mediaItemType: 'book' | ||||
|     } | ||||
|   }) | ||||
|   MediaProgress.belongsTo(book, { foreignKey: 'mediaItemId', constraints: false }) | ||||
|   } | ||||
| 
 | ||||
|   podcastEpisode.hasMany(MediaProgress, { | ||||
|     foreignKey: 'mediaItemId', | ||||
|     constraints: false, | ||||
|     scope: { | ||||
|       mediaItemType: 'podcastEpisode' | ||||
|     } | ||||
|   }) | ||||
|   MediaProgress.belongsTo(podcastEpisode, { foreignKey: 'mediaItemId', constraints: false }) | ||||
| 
 | ||||
|   MediaProgress.addHook('afterFind', findResult => { | ||||
|     if (!findResult) return | ||||
| 
 | ||||
|     if (!Array.isArray(findResult)) findResult = [findResult] | ||||
| 
 | ||||
|     for (const instance of findResult) { | ||||
|       if (instance.mediaItemType === 'book' && instance.book !== undefined) { | ||||
|         instance.mediaItem = instance.book | ||||
|         instance.dataValues.mediaItem = instance.dataValues.book | ||||
|       } else if (instance.mediaItemType === 'podcastEpisode' && instance.podcastEpisode !== undefined) { | ||||
|         instance.mediaItem = instance.podcastEpisode | ||||
|         instance.dataValues.mediaItem = instance.dataValues.podcastEpisode | ||||
|   static removeById(mediaProgressId) { | ||||
|     return this.destroy({ | ||||
|       where: { | ||||
|         id: mediaProgressId | ||||
|       } | ||||
|       // To prevent mistakes:
 | ||||
|       delete instance.book | ||||
|       delete instance.dataValues.book | ||||
|       delete instance.podcastEpisode | ||||
|       delete instance.dataValues.podcastEpisode | ||||
|     } | ||||
|   }) | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   user.hasMany(MediaProgress, { | ||||
|     onDelete: 'CASCADE' | ||||
|   }) | ||||
|   MediaProgress.belongsTo(user) | ||||
|   getMediaItem(options) { | ||||
|     if (!this.mediaItemType) return Promise.resolve(null) | ||||
|     const mixinMethodName = `get${this.sequelize.uppercaseFirst(this.mediaItemType)}` | ||||
|     return this[mixinMethodName](options) | ||||
|   } | ||||
| 
 | ||||
|   return MediaProgress | ||||
|   /** | ||||
|    * Initialize model | ||||
|    *  | ||||
|    * Polymorphic association: Book has many MediaProgress. PodcastEpisode has many MediaProgress. | ||||
|    * @see https://sequelize.org/docs/v6/advanced-association-concepts/polymorphic-associations/
 | ||||
|    *  | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       mediaItemId: DataTypes.UUIDV4, | ||||
|       mediaItemType: DataTypes.STRING, | ||||
|       duration: DataTypes.FLOAT, | ||||
|       currentTime: DataTypes.FLOAT, | ||||
|       isFinished: DataTypes.BOOLEAN, | ||||
|       hideFromContinueListening: DataTypes.BOOLEAN, | ||||
|       ebookLocation: DataTypes.STRING, | ||||
|       ebookProgress: DataTypes.FLOAT, | ||||
|       finishedAt: DataTypes.DATE, | ||||
|       extraData: DataTypes.JSON | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'mediaProgress', | ||||
|       indexes: [ | ||||
|         { | ||||
|           fields: ['updatedAt'] | ||||
|         } | ||||
|       ] | ||||
|     }) | ||||
| 
 | ||||
|     const { book, podcastEpisode, user } = sequelize.models | ||||
| 
 | ||||
|     book.hasMany(MediaProgress, { | ||||
|       foreignKey: 'mediaItemId', | ||||
|       constraints: false, | ||||
|       scope: { | ||||
|         mediaItemType: 'book' | ||||
|       } | ||||
|     }) | ||||
|     MediaProgress.belongsTo(book, { foreignKey: 'mediaItemId', constraints: false }) | ||||
| 
 | ||||
|     podcastEpisode.hasMany(MediaProgress, { | ||||
|       foreignKey: 'mediaItemId', | ||||
|       constraints: false, | ||||
|       scope: { | ||||
|         mediaItemType: 'podcastEpisode' | ||||
|       } | ||||
|     }) | ||||
|     MediaProgress.belongsTo(podcastEpisode, { foreignKey: 'mediaItemId', constraints: false }) | ||||
| 
 | ||||
|     MediaProgress.addHook('afterFind', findResult => { | ||||
|       if (!findResult) return | ||||
| 
 | ||||
|       if (!Array.isArray(findResult)) findResult = [findResult] | ||||
| 
 | ||||
|       for (const instance of findResult) { | ||||
|         if (instance.mediaItemType === 'book' && instance.book !== undefined) { | ||||
|           instance.mediaItem = instance.book | ||||
|           instance.dataValues.mediaItem = instance.dataValues.book | ||||
|         } else if (instance.mediaItemType === 'podcastEpisode' && instance.podcastEpisode !== undefined) { | ||||
|           instance.mediaItem = instance.podcastEpisode | ||||
|           instance.dataValues.mediaItem = instance.dataValues.podcastEpisode | ||||
|         } | ||||
|         // To prevent mistakes:
 | ||||
|         delete instance.book | ||||
|         delete instance.dataValues.book | ||||
|         delete instance.podcastEpisode | ||||
|         delete instance.dataValues.podcastEpisode | ||||
|       } | ||||
|     }) | ||||
| 
 | ||||
|     user.hasMany(MediaProgress, { | ||||
|       onDelete: 'CASCADE' | ||||
|     }) | ||||
|     MediaProgress.belongsTo(user) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = MediaProgress | ||||
| @ -2,197 +2,251 @@ const { DataTypes, Model } = require('sequelize') | ||||
| 
 | ||||
| const oldPlaybackSession = require('../objects/PlaybackSession') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class PlaybackSession extends Model { | ||||
|     static async getOldPlaybackSessions(where = null) { | ||||
|       const playbackSessions = await this.findAll({ | ||||
|         where, | ||||
|         include: [ | ||||
|           { | ||||
|             model: sequelize.models.device | ||||
|           } | ||||
|         ] | ||||
|       }) | ||||
|       return playbackSessions.map(session => this.getOldPlaybackSession(session)) | ||||
|     } | ||||
| 
 | ||||
|     static async getById(sessionId) { | ||||
|       const playbackSession = await this.findByPk(sessionId, { | ||||
|         include: [ | ||||
|           { | ||||
|             model: sequelize.models.device | ||||
|           } | ||||
|         ] | ||||
|       }) | ||||
|       if (!playbackSession) return null | ||||
|       return this.getOldPlaybackSession(playbackSession) | ||||
|     } | ||||
| class PlaybackSession extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     static getOldPlaybackSession(playbackSessionExpanded) { | ||||
|       const isPodcastEpisode = playbackSessionExpanded.mediaItemType === 'podcastEpisode' | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.mediaItemId | ||||
|     /** @type {string} */ | ||||
|     this.mediaItemType | ||||
|     /** @type {string} */ | ||||
|     this.displayTitle | ||||
|     /** @type {string} */ | ||||
|     this.displayAuthor | ||||
|     /** @type {number} */ | ||||
|     this.duration | ||||
|     /** @type {number} */ | ||||
|     this.playMethod | ||||
|     /** @type {string} */ | ||||
|     this.mediaPlayer | ||||
|     /** @type {number} */ | ||||
|     this.startTime | ||||
|     /** @type {number} */ | ||||
|     this.currentTime | ||||
|     /** @type {string} */ | ||||
|     this.serverVersion | ||||
|     /** @type {string} */ | ||||
|     this.coverPath | ||||
|     /** @type {number} */ | ||||
|     this.timeListening | ||||
|     /** @type {Object} */ | ||||
|     this.mediaMetadata | ||||
|     /** @type {string} */ | ||||
|     this.date | ||||
|     /** @type {string} */ | ||||
|     this.dayOfWeek | ||||
|     /** @type {Object} */ | ||||
|     this.extraData | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.userId | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.deviceId | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.libraryId | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|   } | ||||
| 
 | ||||
|       return new oldPlaybackSession({ | ||||
|         id: playbackSessionExpanded.id, | ||||
|         userId: playbackSessionExpanded.userId, | ||||
|         libraryId: playbackSessionExpanded.libraryId, | ||||
|         libraryItemId: playbackSessionExpanded.extraData?.libraryItemId || null, | ||||
|         bookId: isPodcastEpisode ? null : playbackSessionExpanded.mediaItemId, | ||||
|         episodeId: isPodcastEpisode ? playbackSessionExpanded.mediaItemId : null, | ||||
|         mediaType: isPodcastEpisode ? 'podcast' : 'book', | ||||
|         mediaMetadata: playbackSessionExpanded.mediaMetadata, | ||||
|         chapters: null, | ||||
|         displayTitle: playbackSessionExpanded.displayTitle, | ||||
|         displayAuthor: playbackSessionExpanded.displayAuthor, | ||||
|         coverPath: playbackSessionExpanded.coverPath, | ||||
|         duration: playbackSessionExpanded.duration, | ||||
|         playMethod: playbackSessionExpanded.playMethod, | ||||
|         mediaPlayer: playbackSessionExpanded.mediaPlayer, | ||||
|         deviceInfo: playbackSessionExpanded.device?.getOldDevice() || null, | ||||
|         serverVersion: playbackSessionExpanded.serverVersion, | ||||
|         date: playbackSessionExpanded.date, | ||||
|         dayOfWeek: playbackSessionExpanded.dayOfWeek, | ||||
|         timeListening: playbackSessionExpanded.timeListening, | ||||
|         startTime: playbackSessionExpanded.startTime, | ||||
|         currentTime: playbackSessionExpanded.currentTime, | ||||
|         startedAt: playbackSessionExpanded.createdAt.valueOf(), | ||||
|         updatedAt: playbackSessionExpanded.updatedAt.valueOf() | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     static removeById(sessionId) { | ||||
|       return this.destroy({ | ||||
|         where: { | ||||
|           id: sessionId | ||||
|   static async getOldPlaybackSessions(where = null) { | ||||
|     const playbackSessions = await this.findAll({ | ||||
|       where, | ||||
|       include: [ | ||||
|         { | ||||
|           model: this.sequelize.models.device | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
|       ] | ||||
|     }) | ||||
|     return playbackSessions.map(session => this.getOldPlaybackSession(session)) | ||||
|   } | ||||
| 
 | ||||
|     static createFromOld(oldPlaybackSession) { | ||||
|       const playbackSession = this.getFromOld(oldPlaybackSession) | ||||
|       return this.create(playbackSession) | ||||
|     } | ||||
| 
 | ||||
|     static updateFromOld(oldPlaybackSession) { | ||||
|       const playbackSession = this.getFromOld(oldPlaybackSession) | ||||
|       return this.update(playbackSession, { | ||||
|         where: { | ||||
|           id: playbackSession.id | ||||
|   static async getById(sessionId) { | ||||
|     const playbackSession = await this.findByPk(sessionId, { | ||||
|       include: [ | ||||
|         { | ||||
|           model: this.sequelize.models.device | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
|       ] | ||||
|     }) | ||||
|     if (!playbackSession) return null | ||||
|     return this.getOldPlaybackSession(playbackSession) | ||||
|   } | ||||
| 
 | ||||
|     static getFromOld(oldPlaybackSession) { | ||||
|       return { | ||||
|         id: oldPlaybackSession.id, | ||||
|         mediaItemId: oldPlaybackSession.episodeId || oldPlaybackSession.bookId, | ||||
|         mediaItemType: oldPlaybackSession.episodeId ? 'podcastEpisode' : 'book', | ||||
|         libraryId: oldPlaybackSession.libraryId, | ||||
|         displayTitle: oldPlaybackSession.displayTitle, | ||||
|         displayAuthor: oldPlaybackSession.displayAuthor, | ||||
|         duration: oldPlaybackSession.duration, | ||||
|         playMethod: oldPlaybackSession.playMethod, | ||||
|         mediaPlayer: oldPlaybackSession.mediaPlayer, | ||||
|         startTime: oldPlaybackSession.startTime, | ||||
|         currentTime: oldPlaybackSession.currentTime, | ||||
|         serverVersion: oldPlaybackSession.serverVersion || null, | ||||
|         createdAt: oldPlaybackSession.startedAt, | ||||
|         updatedAt: oldPlaybackSession.updatedAt, | ||||
|         userId: oldPlaybackSession.userId, | ||||
|         deviceId: oldPlaybackSession.deviceInfo?.id || null, | ||||
|         timeListening: oldPlaybackSession.timeListening, | ||||
|         coverPath: oldPlaybackSession.coverPath, | ||||
|         mediaMetadata: oldPlaybackSession.mediaMetadata, | ||||
|         date: oldPlaybackSession.date, | ||||
|         dayOfWeek: oldPlaybackSession.dayOfWeek, | ||||
|         extraData: { | ||||
|           libraryItemId: oldPlaybackSession.libraryItemId | ||||
|         } | ||||
|   static getOldPlaybackSession(playbackSessionExpanded) { | ||||
|     const isPodcastEpisode = playbackSessionExpanded.mediaItemType === 'podcastEpisode' | ||||
| 
 | ||||
|     return new oldPlaybackSession({ | ||||
|       id: playbackSessionExpanded.id, | ||||
|       userId: playbackSessionExpanded.userId, | ||||
|       libraryId: playbackSessionExpanded.libraryId, | ||||
|       libraryItemId: playbackSessionExpanded.extraData?.libraryItemId || null, | ||||
|       bookId: isPodcastEpisode ? null : playbackSessionExpanded.mediaItemId, | ||||
|       episodeId: isPodcastEpisode ? playbackSessionExpanded.mediaItemId : null, | ||||
|       mediaType: isPodcastEpisode ? 'podcast' : 'book', | ||||
|       mediaMetadata: playbackSessionExpanded.mediaMetadata, | ||||
|       chapters: null, | ||||
|       displayTitle: playbackSessionExpanded.displayTitle, | ||||
|       displayAuthor: playbackSessionExpanded.displayAuthor, | ||||
|       coverPath: playbackSessionExpanded.coverPath, | ||||
|       duration: playbackSessionExpanded.duration, | ||||
|       playMethod: playbackSessionExpanded.playMethod, | ||||
|       mediaPlayer: playbackSessionExpanded.mediaPlayer, | ||||
|       deviceInfo: playbackSessionExpanded.device?.getOldDevice() || null, | ||||
|       serverVersion: playbackSessionExpanded.serverVersion, | ||||
|       date: playbackSessionExpanded.date, | ||||
|       dayOfWeek: playbackSessionExpanded.dayOfWeek, | ||||
|       timeListening: playbackSessionExpanded.timeListening, | ||||
|       startTime: playbackSessionExpanded.startTime, | ||||
|       currentTime: playbackSessionExpanded.currentTime, | ||||
|       startedAt: playbackSessionExpanded.createdAt.valueOf(), | ||||
|       updatedAt: playbackSessionExpanded.updatedAt.valueOf() | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   static removeById(sessionId) { | ||||
|     return this.destroy({ | ||||
|       where: { | ||||
|         id: sessionId | ||||
|       } | ||||
|     } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|     getMediaItem(options) { | ||||
|       if (!this.mediaItemType) return Promise.resolve(null) | ||||
|       const mixinMethodName = `get${sequelize.uppercaseFirst(this.mediaItemType)}` | ||||
|       return this[mixinMethodName](options) | ||||
|   static createFromOld(oldPlaybackSession) { | ||||
|     const playbackSession = this.getFromOld(oldPlaybackSession) | ||||
|     return this.create(playbackSession) | ||||
|   } | ||||
| 
 | ||||
|   static updateFromOld(oldPlaybackSession) { | ||||
|     const playbackSession = this.getFromOld(oldPlaybackSession) | ||||
|     return this.update(playbackSession, { | ||||
|       where: { | ||||
|         id: playbackSession.id | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   static getFromOld(oldPlaybackSession) { | ||||
|     return { | ||||
|       id: oldPlaybackSession.id, | ||||
|       mediaItemId: oldPlaybackSession.episodeId || oldPlaybackSession.bookId, | ||||
|       mediaItemType: oldPlaybackSession.episodeId ? 'podcastEpisode' : 'book', | ||||
|       libraryId: oldPlaybackSession.libraryId, | ||||
|       displayTitle: oldPlaybackSession.displayTitle, | ||||
|       displayAuthor: oldPlaybackSession.displayAuthor, | ||||
|       duration: oldPlaybackSession.duration, | ||||
|       playMethod: oldPlaybackSession.playMethod, | ||||
|       mediaPlayer: oldPlaybackSession.mediaPlayer, | ||||
|       startTime: oldPlaybackSession.startTime, | ||||
|       currentTime: oldPlaybackSession.currentTime, | ||||
|       serverVersion: oldPlaybackSession.serverVersion || null, | ||||
|       createdAt: oldPlaybackSession.startedAt, | ||||
|       updatedAt: oldPlaybackSession.updatedAt, | ||||
|       userId: oldPlaybackSession.userId, | ||||
|       deviceId: oldPlaybackSession.deviceInfo?.id || null, | ||||
|       timeListening: oldPlaybackSession.timeListening, | ||||
|       coverPath: oldPlaybackSession.coverPath, | ||||
|       mediaMetadata: oldPlaybackSession.mediaMetadata, | ||||
|       date: oldPlaybackSession.date, | ||||
|       dayOfWeek: oldPlaybackSession.dayOfWeek, | ||||
|       extraData: { | ||||
|         libraryItemId: oldPlaybackSession.libraryItemId | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   PlaybackSession.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     mediaItemId: DataTypes.UUIDV4, | ||||
|     mediaItemType: DataTypes.STRING, | ||||
|     displayTitle: DataTypes.STRING, | ||||
|     displayAuthor: DataTypes.STRING, | ||||
|     duration: DataTypes.FLOAT, | ||||
|     playMethod: DataTypes.INTEGER, | ||||
|     mediaPlayer: DataTypes.STRING, | ||||
|     startTime: DataTypes.FLOAT, | ||||
|     currentTime: DataTypes.FLOAT, | ||||
|     serverVersion: DataTypes.STRING, | ||||
|     coverPath: DataTypes.STRING, | ||||
|     timeListening: DataTypes.INTEGER, | ||||
|     mediaMetadata: DataTypes.JSON, | ||||
|     date: DataTypes.STRING, | ||||
|     dayOfWeek: DataTypes.STRING, | ||||
|     extraData: DataTypes.JSON | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'playbackSession' | ||||
|   }) | ||||
|   getMediaItem(options) { | ||||
|     if (!this.mediaItemType) return Promise.resolve(null) | ||||
|     const mixinMethodName = `get${this.sequelize.uppercaseFirst(this.mediaItemType)}` | ||||
|     return this[mixinMethodName](options) | ||||
|   } | ||||
| 
 | ||||
|   const { book, podcastEpisode, user, device, library } = sequelize.models | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       mediaItemId: DataTypes.UUIDV4, | ||||
|       mediaItemType: DataTypes.STRING, | ||||
|       displayTitle: DataTypes.STRING, | ||||
|       displayAuthor: DataTypes.STRING, | ||||
|       duration: DataTypes.FLOAT, | ||||
|       playMethod: DataTypes.INTEGER, | ||||
|       mediaPlayer: DataTypes.STRING, | ||||
|       startTime: DataTypes.FLOAT, | ||||
|       currentTime: DataTypes.FLOAT, | ||||
|       serverVersion: DataTypes.STRING, | ||||
|       coverPath: DataTypes.STRING, | ||||
|       timeListening: DataTypes.INTEGER, | ||||
|       mediaMetadata: DataTypes.JSON, | ||||
|       date: DataTypes.STRING, | ||||
|       dayOfWeek: DataTypes.STRING, | ||||
|       extraData: DataTypes.JSON | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'playbackSession' | ||||
|     }) | ||||
| 
 | ||||
|   user.hasMany(PlaybackSession) | ||||
|   PlaybackSession.belongsTo(user) | ||||
|     const { book, podcastEpisode, user, device, library } = sequelize.models | ||||
| 
 | ||||
|   device.hasMany(PlaybackSession) | ||||
|   PlaybackSession.belongsTo(device) | ||||
|     user.hasMany(PlaybackSession) | ||||
|     PlaybackSession.belongsTo(user) | ||||
| 
 | ||||
|   library.hasMany(PlaybackSession) | ||||
|   PlaybackSession.belongsTo(library) | ||||
|     device.hasMany(PlaybackSession) | ||||
|     PlaybackSession.belongsTo(device) | ||||
| 
 | ||||
|   book.hasMany(PlaybackSession, { | ||||
|     foreignKey: 'mediaItemId', | ||||
|     constraints: false, | ||||
|     scope: { | ||||
|       mediaItemType: 'book' | ||||
|     } | ||||
|   }) | ||||
|   PlaybackSession.belongsTo(book, { foreignKey: 'mediaItemId', constraints: false }) | ||||
|     library.hasMany(PlaybackSession) | ||||
|     PlaybackSession.belongsTo(library) | ||||
| 
 | ||||
|   podcastEpisode.hasOne(PlaybackSession, { | ||||
|     foreignKey: 'mediaItemId', | ||||
|     constraints: false, | ||||
|     scope: { | ||||
|       mediaItemType: 'podcastEpisode' | ||||
|     } | ||||
|   }) | ||||
|   PlaybackSession.belongsTo(podcastEpisode, { foreignKey: 'mediaItemId', constraints: false }) | ||||
| 
 | ||||
|   PlaybackSession.addHook('afterFind', findResult => { | ||||
|     if (!findResult) return | ||||
| 
 | ||||
|     if (!Array.isArray(findResult)) findResult = [findResult] | ||||
| 
 | ||||
|     for (const instance of findResult) { | ||||
|       if (instance.mediaItemType === 'book' && instance.book !== undefined) { | ||||
|         instance.mediaItem = instance.book | ||||
|         instance.dataValues.mediaItem = instance.dataValues.book | ||||
|       } else if (instance.mediaItemType === 'podcastEpisode' && instance.podcastEpisode !== undefined) { | ||||
|         instance.mediaItem = instance.podcastEpisode | ||||
|         instance.dataValues.mediaItem = instance.dataValues.podcastEpisode | ||||
|     book.hasMany(PlaybackSession, { | ||||
|       foreignKey: 'mediaItemId', | ||||
|       constraints: false, | ||||
|       scope: { | ||||
|         mediaItemType: 'book' | ||||
|       } | ||||
|       // To prevent mistakes:
 | ||||
|       delete instance.book | ||||
|       delete instance.dataValues.book | ||||
|       delete instance.podcastEpisode | ||||
|       delete instance.dataValues.podcastEpisode | ||||
|     } | ||||
|   }) | ||||
|     }) | ||||
|     PlaybackSession.belongsTo(book, { foreignKey: 'mediaItemId', constraints: false }) | ||||
| 
 | ||||
|   return PlaybackSession | ||||
|     podcastEpisode.hasOne(PlaybackSession, { | ||||
|       foreignKey: 'mediaItemId', | ||||
|       constraints: false, | ||||
|       scope: { | ||||
|         mediaItemType: 'podcastEpisode' | ||||
|       } | ||||
|     }) | ||||
|     PlaybackSession.belongsTo(podcastEpisode, { foreignKey: 'mediaItemId', constraints: false }) | ||||
| 
 | ||||
|     PlaybackSession.addHook('afterFind', findResult => { | ||||
|       if (!findResult) return | ||||
| 
 | ||||
|       if (!Array.isArray(findResult)) findResult = [findResult] | ||||
| 
 | ||||
|       for (const instance of findResult) { | ||||
|         if (instance.mediaItemType === 'book' && instance.book !== undefined) { | ||||
|           instance.mediaItem = instance.book | ||||
|           instance.dataValues.mediaItem = instance.dataValues.book | ||||
|         } else if (instance.mediaItemType === 'podcastEpisode' && instance.podcastEpisode !== undefined) { | ||||
|           instance.mediaItem = instance.podcastEpisode | ||||
|           instance.dataValues.mediaItem = instance.dataValues.podcastEpisode | ||||
|         } | ||||
|         // To prevent mistakes:
 | ||||
|         delete instance.book | ||||
|         delete instance.dataValues.book | ||||
|         delete instance.podcastEpisode | ||||
|         delete instance.dataValues.podcastEpisode | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = PlaybackSession | ||||
|  | ||||
| @ -3,318 +3,341 @@ const Logger = require('../Logger') | ||||
| 
 | ||||
| const oldPlaylist = require('../objects/Playlist') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class Playlist extends Model { | ||||
|     static async getOldPlaylists() { | ||||
|       const playlists = await this.findAll({ | ||||
|         include: { | ||||
|           model: sequelize.models.playlistMediaItem, | ||||
|           include: [ | ||||
|             { | ||||
|               model: sequelize.models.book, | ||||
|               include: sequelize.models.libraryItem | ||||
|             }, | ||||
|             { | ||||
|               model: sequelize.models.podcastEpisode, | ||||
|               include: { | ||||
|                 model: sequelize.models.podcast, | ||||
|                 include: sequelize.models.libraryItem | ||||
|               } | ||||
|             } | ||||
|           ] | ||||
|         }, | ||||
|         order: [['playlistMediaItems', 'order', 'ASC']] | ||||
|       }) | ||||
|       return playlists.map(p => this.getOldPlaylist(p)) | ||||
|     } | ||||
| class Playlist extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     static getOldPlaylist(playlistExpanded) { | ||||
|       const items = playlistExpanded.playlistMediaItems.map(pmi => { | ||||
|         const libraryItemId = pmi.mediaItem?.podcast?.libraryItem?.id || pmi.mediaItem?.libraryItem?.id || null | ||||
|         if (!libraryItemId) { | ||||
|           Logger.error(`[Playlist] Invalid playlist media item - No library item id found`, JSON.stringify(pmi, null, 2)) | ||||
|           return null | ||||
|         } | ||||
|         return { | ||||
|           episodeId: pmi.mediaItemType === 'podcastEpisode' ? pmi.mediaItemId : '', | ||||
|           libraryItemId | ||||
|         } | ||||
|       }).filter(pmi => pmi) | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {string} */ | ||||
|     this.name | ||||
|     /** @type {string} */ | ||||
|     this.description | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.libraryId | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.userId | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|   } | ||||
| 
 | ||||
|       return new oldPlaylist({ | ||||
|         id: playlistExpanded.id, | ||||
|         libraryId: playlistExpanded.libraryId, | ||||
|         userId: playlistExpanded.userId, | ||||
|         name: playlistExpanded.name, | ||||
|         description: playlistExpanded.description, | ||||
|         items, | ||||
|         lastUpdate: playlistExpanded.updatedAt.valueOf(), | ||||
|         createdAt: playlistExpanded.createdAt.valueOf() | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Get old playlist toJSONExpanded | ||||
|      * @param {[string[]]} include | ||||
|      * @returns {Promise<object>} oldPlaylist.toJSONExpanded | ||||
|      */ | ||||
|     async getOldJsonExpanded(include) { | ||||
|       this.playlistMediaItems = await this.getPlaylistMediaItems({ | ||||
|   static async getOldPlaylists() { | ||||
|     const playlists = await this.findAll({ | ||||
|       include: { | ||||
|         model: this.sequelize.models.playlistMediaItem, | ||||
|         include: [ | ||||
|           { | ||||
|             model: sequelize.models.book, | ||||
|             include: sequelize.models.libraryItem | ||||
|             model: this.sequelize.models.book, | ||||
|             include: this.sequelize.models.libraryItem | ||||
|           }, | ||||
|           { | ||||
|             model: sequelize.models.podcastEpisode, | ||||
|             model: this.sequelize.models.podcastEpisode, | ||||
|             include: { | ||||
|               model: sequelize.models.podcast, | ||||
|               include: sequelize.models.libraryItem | ||||
|               model: this.sequelize.models.podcast, | ||||
|               include: this.sequelize.models.libraryItem | ||||
|             } | ||||
|           } | ||||
|         ], | ||||
|         order: [['order', 'ASC']] | ||||
|       }) || [] | ||||
| 
 | ||||
|       const oldPlaylist = sequelize.models.playlist.getOldPlaylist(this) | ||||
|       const libraryItemIds = oldPlaylist.items.map(i => i.libraryItemId) | ||||
| 
 | ||||
|       let libraryItems = await sequelize.models.libraryItem.getAllOldLibraryItems({ | ||||
|         id: libraryItemIds | ||||
|       }) | ||||
| 
 | ||||
|       const playlistExpanded = oldPlaylist.toJSONExpanded(libraryItems) | ||||
| 
 | ||||
|       if (include?.includes('rssfeed')) { | ||||
|         const feeds = await this.getFeeds() | ||||
|         if (feeds?.length) { | ||||
|           playlistExpanded.rssFeed = sequelize.models.feed.getOldFeed(feeds[0]) | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       return playlistExpanded | ||||
|     } | ||||
| 
 | ||||
|     static createFromOld(oldPlaylist) { | ||||
|       const playlist = this.getFromOld(oldPlaylist) | ||||
|       return this.create(playlist) | ||||
|     } | ||||
| 
 | ||||
|     static getFromOld(oldPlaylist) { | ||||
|       return { | ||||
|         id: oldPlaylist.id, | ||||
|         name: oldPlaylist.name, | ||||
|         description: oldPlaylist.description, | ||||
|         userId: oldPlaylist.userId, | ||||
|         libraryId: oldPlaylist.libraryId | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     static removeById(playlistId) { | ||||
|       return this.destroy({ | ||||
|         where: { | ||||
|           id: playlistId | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Get playlist by id | ||||
|      * @param {string} playlistId  | ||||
|      * @returns {Promise<oldPlaylist|null>} returns null if not found | ||||
|      */ | ||||
|     static async getById(playlistId) { | ||||
|       if (!playlistId) return null | ||||
|       const playlist = await this.findByPk(playlistId, { | ||||
|         include: { | ||||
|           model: sequelize.models.playlistMediaItem, | ||||
|           include: [ | ||||
|             { | ||||
|               model: sequelize.models.book, | ||||
|               include: sequelize.models.libraryItem | ||||
|             }, | ||||
|             { | ||||
|               model: sequelize.models.podcastEpisode, | ||||
|               include: { | ||||
|                 model: sequelize.models.podcast, | ||||
|                 include: sequelize.models.libraryItem | ||||
|               } | ||||
|             } | ||||
|           ] | ||||
|         }, | ||||
|         order: [['playlistMediaItems', 'order', 'ASC']] | ||||
|       }) | ||||
|       if (!playlist) return null | ||||
|       return this.getOldPlaylist(playlist) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Get playlists for user and optionally for library | ||||
|      * @param {string} userId  | ||||
|      * @param {[string]} libraryId optional | ||||
|      * @returns {Promise<Playlist[]>} | ||||
|      */ | ||||
|     static async getPlaylistsForUserAndLibrary(userId, libraryId = null) { | ||||
|       if (!userId && !libraryId) return [] | ||||
|       const whereQuery = {} | ||||
|       if (userId) { | ||||
|         whereQuery.userId = userId | ||||
|       } | ||||
|       if (libraryId) { | ||||
|         whereQuery.libraryId = libraryId | ||||
|       } | ||||
|       const playlists = await this.findAll({ | ||||
|         where: whereQuery, | ||||
|         include: { | ||||
|           model: sequelize.models.playlistMediaItem, | ||||
|           include: [ | ||||
|             { | ||||
|               model: sequelize.models.book, | ||||
|               include: sequelize.models.libraryItem | ||||
|             }, | ||||
|             { | ||||
|               model: sequelize.models.podcastEpisode, | ||||
|               include: { | ||||
|                 model: sequelize.models.podcast, | ||||
|                 include: sequelize.models.libraryItem | ||||
|               } | ||||
|             } | ||||
|           ] | ||||
|         }, | ||||
|         order: [ | ||||
|           [literal('name COLLATE NOCASE'), 'ASC'], | ||||
|           ['playlistMediaItems', 'order', 'ASC'] | ||||
|         ] | ||||
|       }) | ||||
|       return playlists | ||||
|     } | ||||
|       }, | ||||
|       order: [['playlistMediaItems', 'order', 'ASC']] | ||||
|     }) | ||||
|     return playlists.map(p => this.getOldPlaylist(p)) | ||||
|   } | ||||
| 
 | ||||
|     /** | ||||
|      * Get number of playlists for a user and library | ||||
|      * @param {string} userId  | ||||
|      * @param {string} libraryId  | ||||
|      * @returns  | ||||
|      */ | ||||
|     static async getNumPlaylistsForUserAndLibrary(userId, libraryId) { | ||||
|       return this.count({ | ||||
|         where: { | ||||
|           userId, | ||||
|           libraryId | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Get all playlists for mediaItemIds | ||||
|      * @param {string[]} mediaItemIds  | ||||
|      * @returns {Promise<Playlist[]>} | ||||
|      */ | ||||
|     static async getPlaylistsForMediaItemIds(mediaItemIds) { | ||||
|       if (!mediaItemIds?.length) return [] | ||||
| 
 | ||||
|       const playlistMediaItemsExpanded = await sequelize.models.playlistMediaItem.findAll({ | ||||
|         where: { | ||||
|           mediaItemId: { | ||||
|             [Op.in]: mediaItemIds | ||||
|           } | ||||
|         }, | ||||
|         include: [ | ||||
|           { | ||||
|             model: sequelize.models.playlist, | ||||
|             include: { | ||||
|               model: sequelize.models.playlistMediaItem, | ||||
|               include: [ | ||||
|                 { | ||||
|                   model: sequelize.models.book, | ||||
|                   include: sequelize.models.libraryItem | ||||
|                 }, | ||||
|                 { | ||||
|                   model: sequelize.models.podcastEpisode, | ||||
|                   include: { | ||||
|                     model: sequelize.models.podcast, | ||||
|                     include: sequelize.models.libraryItem | ||||
|                   } | ||||
|                 } | ||||
|               ] | ||||
|             } | ||||
|           } | ||||
|         ], | ||||
|         order: [['playlist', 'playlistMediaItems', 'order', 'ASC']] | ||||
|       }) | ||||
| 
 | ||||
|       const playlists = [] | ||||
|       for (const playlistMediaItem of playlistMediaItemsExpanded) { | ||||
|         const playlist = playlistMediaItem.playlist | ||||
|         if (playlists.some(p => p.id === playlist.id)) continue | ||||
| 
 | ||||
|         playlist.playlistMediaItems = playlist.playlistMediaItems.map(pmi => { | ||||
|           if (pmi.mediaItemType === 'book' && pmi.book !== undefined) { | ||||
|             pmi.mediaItem = pmi.book | ||||
|             pmi.dataValues.mediaItem = pmi.dataValues.book | ||||
|           } else if (pmi.mediaItemType === 'podcastEpisode' && pmi.podcastEpisode !== undefined) { | ||||
|             pmi.mediaItem = pmi.podcastEpisode | ||||
|             pmi.dataValues.mediaItem = pmi.dataValues.podcastEpisode | ||||
|           } | ||||
|           delete pmi.book | ||||
|           delete pmi.dataValues.book | ||||
|           delete pmi.podcastEpisode | ||||
|           delete pmi.dataValues.podcastEpisode | ||||
|           return pmi | ||||
|         }) | ||||
|         playlists.push(playlist) | ||||
|   static getOldPlaylist(playlistExpanded) { | ||||
|     const items = playlistExpanded.playlistMediaItems.map(pmi => { | ||||
|       const libraryItemId = pmi.mediaItem?.podcast?.libraryItem?.id || pmi.mediaItem?.libraryItem?.id || null | ||||
|       if (!libraryItemId) { | ||||
|         Logger.error(`[Playlist] Invalid playlist media item - No library item id found`, JSON.stringify(pmi, null, 2)) | ||||
|         return null | ||||
|       } | ||||
|       return playlists | ||||
|       return { | ||||
|         episodeId: pmi.mediaItemType === 'podcastEpisode' ? pmi.mediaItemId : '', | ||||
|         libraryItemId | ||||
|       } | ||||
|     }).filter(pmi => pmi) | ||||
| 
 | ||||
|     return new oldPlaylist({ | ||||
|       id: playlistExpanded.id, | ||||
|       libraryId: playlistExpanded.libraryId, | ||||
|       userId: playlistExpanded.userId, | ||||
|       name: playlistExpanded.name, | ||||
|       description: playlistExpanded.description, | ||||
|       items, | ||||
|       lastUpdate: playlistExpanded.updatedAt.valueOf(), | ||||
|       createdAt: playlistExpanded.createdAt.valueOf() | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get old playlist toJSONExpanded | ||||
|    * @param {[string[]]} include | ||||
|    * @returns {Promise<object>} oldPlaylist.toJSONExpanded | ||||
|    */ | ||||
|   async getOldJsonExpanded(include) { | ||||
|     this.playlistMediaItems = await this.getPlaylistMediaItems({ | ||||
|       include: [ | ||||
|         { | ||||
|           model: this.sequelize.models.book, | ||||
|           include: this.sequelize.models.libraryItem | ||||
|         }, | ||||
|         { | ||||
|           model: this.sequelize.models.podcastEpisode, | ||||
|           include: { | ||||
|             model: this.sequelize.models.podcast, | ||||
|             include: this.sequelize.models.libraryItem | ||||
|           } | ||||
|         } | ||||
|       ], | ||||
|       order: [['order', 'ASC']] | ||||
|     }) || [] | ||||
| 
 | ||||
|     const oldPlaylist = this.sequelize.models.playlist.getOldPlaylist(this) | ||||
|     const libraryItemIds = oldPlaylist.items.map(i => i.libraryItemId) | ||||
| 
 | ||||
|     let libraryItems = await this.sequelize.models.libraryItem.getAllOldLibraryItems({ | ||||
|       id: libraryItemIds | ||||
|     }) | ||||
| 
 | ||||
|     const playlistExpanded = oldPlaylist.toJSONExpanded(libraryItems) | ||||
| 
 | ||||
|     if (include?.includes('rssfeed')) { | ||||
|       const feeds = await this.getFeeds() | ||||
|       if (feeds?.length) { | ||||
|         playlistExpanded.rssFeed = this.sequelize.models.feed.getOldFeed(feeds[0]) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     return playlistExpanded | ||||
|   } | ||||
| 
 | ||||
|   static createFromOld(oldPlaylist) { | ||||
|     const playlist = this.getFromOld(oldPlaylist) | ||||
|     return this.create(playlist) | ||||
|   } | ||||
| 
 | ||||
|   static getFromOld(oldPlaylist) { | ||||
|     return { | ||||
|       id: oldPlaylist.id, | ||||
|       name: oldPlaylist.name, | ||||
|       description: oldPlaylist.description, | ||||
|       userId: oldPlaylist.userId, | ||||
|       libraryId: oldPlaylist.libraryId | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   Playlist.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     name: DataTypes.STRING, | ||||
|     description: DataTypes.TEXT | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'playlist' | ||||
|   }) | ||||
| 
 | ||||
|   const { library, user } = sequelize.models | ||||
|   library.hasMany(Playlist) | ||||
|   Playlist.belongsTo(library) | ||||
| 
 | ||||
|   user.hasMany(Playlist, { | ||||
|     onDelete: 'CASCADE' | ||||
|   }) | ||||
|   Playlist.belongsTo(user) | ||||
| 
 | ||||
|   Playlist.addHook('afterFind', findResult => { | ||||
|     if (!findResult) return | ||||
| 
 | ||||
|     if (!Array.isArray(findResult)) findResult = [findResult] | ||||
| 
 | ||||
|     for (const instance of findResult) { | ||||
|       if (instance.playlistMediaItems?.length) { | ||||
|         instance.playlistMediaItems = instance.playlistMediaItems.map(pmi => { | ||||
|           if (pmi.mediaItemType === 'book' && pmi.book !== undefined) { | ||||
|             pmi.mediaItem = pmi.book | ||||
|             pmi.dataValues.mediaItem = pmi.dataValues.book | ||||
|           } else if (pmi.mediaItemType === 'podcastEpisode' && pmi.podcastEpisode !== undefined) { | ||||
|             pmi.mediaItem = pmi.podcastEpisode | ||||
|             pmi.dataValues.mediaItem = pmi.dataValues.podcastEpisode | ||||
|           } | ||||
|           // To prevent mistakes:
 | ||||
|           delete pmi.book | ||||
|           delete pmi.dataValues.book | ||||
|           delete pmi.podcastEpisode | ||||
|           delete pmi.dataValues.podcastEpisode | ||||
|           return pmi | ||||
|         }) | ||||
|   static removeById(playlistId) { | ||||
|     return this.destroy({ | ||||
|       where: { | ||||
|         id: playlistId | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get playlist by id | ||||
|    * @param {string} playlistId  | ||||
|    * @returns {Promise<oldPlaylist|null>} returns null if not found | ||||
|    */ | ||||
|   static async getById(playlistId) { | ||||
|     if (!playlistId) return null | ||||
|     const playlist = await this.findByPk(playlistId, { | ||||
|       include: { | ||||
|         model: this.sequelize.models.playlistMediaItem, | ||||
|         include: [ | ||||
|           { | ||||
|             model: this.sequelize.models.book, | ||||
|             include: this.sequelize.models.libraryItem | ||||
|           }, | ||||
|           { | ||||
|             model: this.sequelize.models.podcastEpisode, | ||||
|             include: { | ||||
|               model: this.sequelize.models.podcast, | ||||
|               include: this.sequelize.models.libraryItem | ||||
|             } | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       order: [['playlistMediaItems', 'order', 'ASC']] | ||||
|     }) | ||||
|     if (!playlist) return null | ||||
|     return this.getOldPlaylist(playlist) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get playlists for user and optionally for library | ||||
|    * @param {string} userId  | ||||
|    * @param {[string]} libraryId optional | ||||
|    * @returns {Promise<Playlist[]>} | ||||
|    */ | ||||
|   static async getPlaylistsForUserAndLibrary(userId, libraryId = null) { | ||||
|     if (!userId && !libraryId) return [] | ||||
|     const whereQuery = {} | ||||
|     if (userId) { | ||||
|       whereQuery.userId = userId | ||||
|     } | ||||
|   }) | ||||
|     if (libraryId) { | ||||
|       whereQuery.libraryId = libraryId | ||||
|     } | ||||
|     const playlists = await this.findAll({ | ||||
|       where: whereQuery, | ||||
|       include: { | ||||
|         model: this.sequelize.models.playlistMediaItem, | ||||
|         include: [ | ||||
|           { | ||||
|             model: this.sequelize.models.book, | ||||
|             include: this.sequelize.models.libraryItem | ||||
|           }, | ||||
|           { | ||||
|             model: this.sequelize.models.podcastEpisode, | ||||
|             include: { | ||||
|               model: this.sequelize.models.podcast, | ||||
|               include: this.sequelize.models.libraryItem | ||||
|             } | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       order: [ | ||||
|         [literal('name COLLATE NOCASE'), 'ASC'], | ||||
|         ['playlistMediaItems', 'order', 'ASC'] | ||||
|       ] | ||||
|     }) | ||||
|     return playlists | ||||
|   } | ||||
| 
 | ||||
|   return Playlist | ||||
|   /** | ||||
|    * Get number of playlists for a user and library | ||||
|    * @param {string} userId  | ||||
|    * @param {string} libraryId  | ||||
|    * @returns  | ||||
|    */ | ||||
|   static async getNumPlaylistsForUserAndLibrary(userId, libraryId) { | ||||
|     return this.count({ | ||||
|       where: { | ||||
|         userId, | ||||
|         libraryId | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get all playlists for mediaItemIds | ||||
|    * @param {string[]} mediaItemIds  | ||||
|    * @returns {Promise<Playlist[]>} | ||||
|    */ | ||||
|   static async getPlaylistsForMediaItemIds(mediaItemIds) { | ||||
|     if (!mediaItemIds?.length) return [] | ||||
| 
 | ||||
|     const playlistMediaItemsExpanded = await this.sequelize.models.playlistMediaItem.findAll({ | ||||
|       where: { | ||||
|         mediaItemId: { | ||||
|           [Op.in]: mediaItemIds | ||||
|         } | ||||
|       }, | ||||
|       include: [ | ||||
|         { | ||||
|           model: this.sequelize.models.playlist, | ||||
|           include: { | ||||
|             model: this.sequelize.models.playlistMediaItem, | ||||
|             include: [ | ||||
|               { | ||||
|                 model: this.sequelize.models.book, | ||||
|                 include: this.sequelize.models.libraryItem | ||||
|               }, | ||||
|               { | ||||
|                 model: this.sequelize.models.podcastEpisode, | ||||
|                 include: { | ||||
|                   model: this.sequelize.models.podcast, | ||||
|                   include: this.sequelize.models.libraryItem | ||||
|                 } | ||||
|               } | ||||
|             ] | ||||
|           } | ||||
|         } | ||||
|       ], | ||||
|       order: [['playlist', 'playlistMediaItems', 'order', 'ASC']] | ||||
|     }) | ||||
| 
 | ||||
|     const playlists = [] | ||||
|     for (const playlistMediaItem of playlistMediaItemsExpanded) { | ||||
|       const playlist = playlistMediaItem.playlist | ||||
|       if (playlists.some(p => p.id === playlist.id)) continue | ||||
| 
 | ||||
|       playlist.playlistMediaItems = playlist.playlistMediaItems.map(pmi => { | ||||
|         if (pmi.mediaItemType === 'book' && pmi.book !== undefined) { | ||||
|           pmi.mediaItem = pmi.book | ||||
|           pmi.dataValues.mediaItem = pmi.dataValues.book | ||||
|         } else if (pmi.mediaItemType === 'podcastEpisode' && pmi.podcastEpisode !== undefined) { | ||||
|           pmi.mediaItem = pmi.podcastEpisode | ||||
|           pmi.dataValues.mediaItem = pmi.dataValues.podcastEpisode | ||||
|         } | ||||
|         delete pmi.book | ||||
|         delete pmi.dataValues.book | ||||
|         delete pmi.podcastEpisode | ||||
|         delete pmi.dataValues.podcastEpisode | ||||
|         return pmi | ||||
|       }) | ||||
|       playlists.push(playlist) | ||||
|     } | ||||
|     return playlists | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       name: DataTypes.STRING, | ||||
|       description: DataTypes.TEXT | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'playlist' | ||||
|     }) | ||||
| 
 | ||||
|     const { library, user } = sequelize.models | ||||
|     library.hasMany(Playlist) | ||||
|     Playlist.belongsTo(library) | ||||
| 
 | ||||
|     user.hasMany(Playlist, { | ||||
|       onDelete: 'CASCADE' | ||||
|     }) | ||||
|     Playlist.belongsTo(user) | ||||
| 
 | ||||
|     Playlist.addHook('afterFind', findResult => { | ||||
|       if (!findResult) return | ||||
| 
 | ||||
|       if (!Array.isArray(findResult)) findResult = [findResult] | ||||
| 
 | ||||
|       for (const instance of findResult) { | ||||
|         if (instance.playlistMediaItems?.length) { | ||||
|           instance.playlistMediaItems = instance.playlistMediaItems.map(pmi => { | ||||
|             if (pmi.mediaItemType === 'book' && pmi.book !== undefined) { | ||||
|               pmi.mediaItem = pmi.book | ||||
|               pmi.dataValues.mediaItem = pmi.dataValues.book | ||||
|             } else if (pmi.mediaItemType === 'podcastEpisode' && pmi.podcastEpisode !== undefined) { | ||||
|               pmi.mediaItem = pmi.podcastEpisode | ||||
|               pmi.dataValues.mediaItem = pmi.dataValues.podcastEpisode | ||||
|             } | ||||
|             // To prevent mistakes:
 | ||||
|             delete pmi.book | ||||
|             delete pmi.dataValues.book | ||||
|             delete pmi.podcastEpisode | ||||
|             delete pmi.dataValues.podcastEpisode | ||||
|             return pmi | ||||
|           }) | ||||
|         } | ||||
| 
 | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = Playlist | ||||
| @ -1,84 +1,105 @@ | ||||
| const { DataTypes, Model } = require('sequelize') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class PlaylistMediaItem extends Model { | ||||
|     static removeByIds(playlistId, mediaItemId) { | ||||
|       return this.destroy({ | ||||
|         where: { | ||||
|           playlistId, | ||||
|           mediaItemId | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
| class PlaylistMediaItem extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     getMediaItem(options) { | ||||
|       if (!this.mediaItemType) return Promise.resolve(null) | ||||
|       const mixinMethodName = `get${sequelize.uppercaseFirst(this.mediaItemType)}` | ||||
|       return this[mixinMethodName](options) | ||||
|     } | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.mediaItemId | ||||
|     /** @type {string} */ | ||||
|     this.mediaItemType | ||||
|     /** @type {number} */ | ||||
|     this.order | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.playlistId | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|   } | ||||
| 
 | ||||
|   PlaylistMediaItem.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     mediaItemId: DataTypes.UUIDV4, | ||||
|     mediaItemType: DataTypes.STRING, | ||||
|     order: DataTypes.INTEGER | ||||
|   }, { | ||||
|     sequelize, | ||||
|     timestamps: true, | ||||
|     updatedAt: false, | ||||
|     modelName: 'playlistMediaItem' | ||||
|   }) | ||||
| 
 | ||||
|   const { book, podcastEpisode, playlist } = sequelize.models | ||||
| 
 | ||||
|   book.hasMany(PlaylistMediaItem, { | ||||
|     foreignKey: 'mediaItemId', | ||||
|     constraints: false, | ||||
|     scope: { | ||||
|       mediaItemType: 'book' | ||||
|     } | ||||
|   }) | ||||
|   PlaylistMediaItem.belongsTo(book, { foreignKey: 'mediaItemId', constraints: false }) | ||||
| 
 | ||||
|   podcastEpisode.hasOne(PlaylistMediaItem, { | ||||
|     foreignKey: 'mediaItemId', | ||||
|     constraints: false, | ||||
|     scope: { | ||||
|       mediaItemType: 'podcastEpisode' | ||||
|     } | ||||
|   }) | ||||
|   PlaylistMediaItem.belongsTo(podcastEpisode, { foreignKey: 'mediaItemId', constraints: false }) | ||||
| 
 | ||||
|   PlaylistMediaItem.addHook('afterFind', findResult => { | ||||
|     if (!findResult) return | ||||
| 
 | ||||
|     if (!Array.isArray(findResult)) findResult = [findResult] | ||||
| 
 | ||||
|     for (const instance of findResult) { | ||||
|       if (instance.mediaItemType === 'book' && instance.book !== undefined) { | ||||
|         instance.mediaItem = instance.book | ||||
|         instance.dataValues.mediaItem = instance.dataValues.book | ||||
|       } else if (instance.mediaItemType === 'podcastEpisode' && instance.podcastEpisode !== undefined) { | ||||
|         instance.mediaItem = instance.podcastEpisode | ||||
|         instance.dataValues.mediaItem = instance.dataValues.podcastEpisode | ||||
|   static removeByIds(playlistId, mediaItemId) { | ||||
|     return this.destroy({ | ||||
|       where: { | ||||
|         playlistId, | ||||
|         mediaItemId | ||||
|       } | ||||
|       // To prevent mistakes:
 | ||||
|       delete instance.book | ||||
|       delete instance.dataValues.book | ||||
|       delete instance.podcastEpisode | ||||
|       delete instance.dataValues.podcastEpisode | ||||
|     } | ||||
|   }) | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   playlist.hasMany(PlaylistMediaItem, { | ||||
|     onDelete: 'CASCADE' | ||||
|   }) | ||||
|   PlaylistMediaItem.belongsTo(playlist) | ||||
|   getMediaItem(options) { | ||||
|     if (!this.mediaItemType) return Promise.resolve(null) | ||||
|     const mixinMethodName = `get${this.sequelize.uppercaseFirst(this.mediaItemType)}` | ||||
|     return this[mixinMethodName](options) | ||||
|   } | ||||
| 
 | ||||
|   return PlaylistMediaItem | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       mediaItemId: DataTypes.UUIDV4, | ||||
|       mediaItemType: DataTypes.STRING, | ||||
|       order: DataTypes.INTEGER | ||||
|     }, { | ||||
|       sequelize, | ||||
|       timestamps: true, | ||||
|       updatedAt: false, | ||||
|       modelName: 'playlistMediaItem' | ||||
|     }) | ||||
| 
 | ||||
|     const { book, podcastEpisode, playlist } = sequelize.models | ||||
| 
 | ||||
|     book.hasMany(PlaylistMediaItem, { | ||||
|       foreignKey: 'mediaItemId', | ||||
|       constraints: false, | ||||
|       scope: { | ||||
|         mediaItemType: 'book' | ||||
|       } | ||||
|     }) | ||||
|     PlaylistMediaItem.belongsTo(book, { foreignKey: 'mediaItemId', constraints: false }) | ||||
| 
 | ||||
|     podcastEpisode.hasOne(PlaylistMediaItem, { | ||||
|       foreignKey: 'mediaItemId', | ||||
|       constraints: false, | ||||
|       scope: { | ||||
|         mediaItemType: 'podcastEpisode' | ||||
|       } | ||||
|     }) | ||||
|     PlaylistMediaItem.belongsTo(podcastEpisode, { foreignKey: 'mediaItemId', constraints: false }) | ||||
| 
 | ||||
|     PlaylistMediaItem.addHook('afterFind', findResult => { | ||||
|       if (!findResult) return | ||||
| 
 | ||||
|       if (!Array.isArray(findResult)) findResult = [findResult] | ||||
| 
 | ||||
|       for (const instance of findResult) { | ||||
|         if (instance.mediaItemType === 'book' && instance.book !== undefined) { | ||||
|           instance.mediaItem = instance.book | ||||
|           instance.dataValues.mediaItem = instance.dataValues.book | ||||
|         } else if (instance.mediaItemType === 'podcastEpisode' && instance.podcastEpisode !== undefined) { | ||||
|           instance.mediaItem = instance.podcastEpisode | ||||
|           instance.dataValues.mediaItem = instance.dataValues.podcastEpisode | ||||
|         } | ||||
|         // To prevent mistakes:
 | ||||
|         delete instance.book | ||||
|         delete instance.dataValues.book | ||||
|         delete instance.podcastEpisode | ||||
|         delete instance.dataValues.podcastEpisode | ||||
|       } | ||||
|     }) | ||||
| 
 | ||||
|     playlist.hasMany(PlaylistMediaItem, { | ||||
|       onDelete: 'CASCADE' | ||||
|     }) | ||||
|     PlaylistMediaItem.belongsTo(playlist) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = PlaylistMediaItem | ||||
|  | ||||
| @ -1,100 +1,155 @@ | ||||
| const { DataTypes, Model } = require('sequelize') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class Podcast extends Model { | ||||
|     static getOldPodcast(libraryItemExpanded) { | ||||
|       const podcastExpanded = libraryItemExpanded.media | ||||
|       const podcastEpisodes = podcastExpanded.podcastEpisodes?.map(ep => ep.getOldPodcastEpisode(libraryItemExpanded.id)).sort((a, b) => a.index - b.index) | ||||
|       return { | ||||
|         id: podcastExpanded.id, | ||||
|         libraryItemId: libraryItemExpanded.id, | ||||
|         metadata: { | ||||
|           title: podcastExpanded.title, | ||||
|           author: podcastExpanded.author, | ||||
|           description: podcastExpanded.description, | ||||
|           releaseDate: podcastExpanded.releaseDate, | ||||
|           genres: podcastExpanded.genres, | ||||
|           feedUrl: podcastExpanded.feedURL, | ||||
|           imageUrl: podcastExpanded.imageURL, | ||||
|           itunesPageUrl: podcastExpanded.itunesPageURL, | ||||
|           itunesId: podcastExpanded.itunesId, | ||||
|           itunesArtistId: podcastExpanded.itunesArtistId, | ||||
|           explicit: podcastExpanded.explicit, | ||||
|           language: podcastExpanded.language, | ||||
|           type: podcastExpanded.podcastType | ||||
|         }, | ||||
|         coverPath: podcastExpanded.coverPath, | ||||
|         tags: podcastExpanded.tags, | ||||
|         episodes: podcastEpisodes || [], | ||||
|         autoDownloadEpisodes: podcastExpanded.autoDownloadEpisodes, | ||||
|         autoDownloadSchedule: podcastExpanded.autoDownloadSchedule, | ||||
|         lastEpisodeCheck: podcastExpanded.lastEpisodeCheck?.valueOf() || null, | ||||
|         maxEpisodesToKeep: podcastExpanded.maxEpisodesToKeep, | ||||
|         maxNewEpisodesToDownload: podcastExpanded.maxNewEpisodesToDownload | ||||
|       } | ||||
|     } | ||||
| class Podcast extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     static getFromOld(oldPodcast) { | ||||
|       const oldPodcastMetadata = oldPodcast.metadata | ||||
|       return { | ||||
|         id: oldPodcast.id, | ||||
|         title: oldPodcastMetadata.title, | ||||
|         titleIgnorePrefix: oldPodcastMetadata.titleIgnorePrefix, | ||||
|         author: oldPodcastMetadata.author, | ||||
|         releaseDate: oldPodcastMetadata.releaseDate, | ||||
|         feedURL: oldPodcastMetadata.feedUrl, | ||||
|         imageURL: oldPodcastMetadata.imageUrl, | ||||
|         description: oldPodcastMetadata.description, | ||||
|         itunesPageURL: oldPodcastMetadata.itunesPageUrl, | ||||
|         itunesId: oldPodcastMetadata.itunesId, | ||||
|         itunesArtistId: oldPodcastMetadata.itunesArtistId, | ||||
|         language: oldPodcastMetadata.language, | ||||
|         podcastType: oldPodcastMetadata.type, | ||||
|         explicit: !!oldPodcastMetadata.explicit, | ||||
|         autoDownloadEpisodes: !!oldPodcast.autoDownloadEpisodes, | ||||
|         autoDownloadSchedule: oldPodcast.autoDownloadSchedule, | ||||
|         lastEpisodeCheck: oldPodcast.lastEpisodeCheck, | ||||
|         maxEpisodesToKeep: oldPodcast.maxEpisodesToKeep, | ||||
|         maxNewEpisodesToDownload: oldPodcast.maxNewEpisodesToDownload, | ||||
|         coverPath: oldPodcast.coverPath, | ||||
|         tags: oldPodcast.tags, | ||||
|         genres: oldPodcastMetadata.genres | ||||
|       } | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {string} */ | ||||
|     this.title | ||||
|     /** @type {string} */ | ||||
|     this.titleIgnorePrefix | ||||
|     /** @type {string} */ | ||||
|     this.author | ||||
|     /** @type {string} */ | ||||
|     this.releaseDate | ||||
|     /** @type {string} */ | ||||
|     this.feedURL | ||||
|     /** @type {string} */ | ||||
|     this.imageURL | ||||
|     /** @type {string} */ | ||||
|     this.description | ||||
|     /** @type {string} */ | ||||
|     this.itunesPageURL | ||||
|     /** @type {string} */ | ||||
|     this.itunesId | ||||
|     /** @type {string} */ | ||||
|     this.itunesArtistId | ||||
|     /** @type {string} */ | ||||
|     this.language | ||||
|     /** @type {string} */ | ||||
|     this.podcastType | ||||
|     /** @type {boolean} */ | ||||
|     this.explicit | ||||
|     /** @type {boolean} */ | ||||
|     this.autoDownloadEpisodes | ||||
|     /** @type {string} */ | ||||
|     this.autoDownloadSchedule | ||||
|     /** @type {Date} */ | ||||
|     this.lastEpisodeCheck | ||||
|     /** @type {number} */ | ||||
|     this.maxEpisodesToKeep | ||||
|     /** @type {string} */ | ||||
|     this.coverPath | ||||
|     /** @type {Object} */ | ||||
|     this.tags | ||||
|     /** @type {Object} */ | ||||
|     this.genres | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|   } | ||||
| 
 | ||||
|   static getOldPodcast(libraryItemExpanded) { | ||||
|     const podcastExpanded = libraryItemExpanded.media | ||||
|     const podcastEpisodes = podcastExpanded.podcastEpisodes?.map(ep => ep.getOldPodcastEpisode(libraryItemExpanded.id).toJSON()).sort((a, b) => a.index - b.index) | ||||
|     return { | ||||
|       id: podcastExpanded.id, | ||||
|       libraryItemId: libraryItemExpanded.id, | ||||
|       metadata: { | ||||
|         title: podcastExpanded.title, | ||||
|         author: podcastExpanded.author, | ||||
|         description: podcastExpanded.description, | ||||
|         releaseDate: podcastExpanded.releaseDate, | ||||
|         genres: podcastExpanded.genres, | ||||
|         feedUrl: podcastExpanded.feedURL, | ||||
|         imageUrl: podcastExpanded.imageURL, | ||||
|         itunesPageUrl: podcastExpanded.itunesPageURL, | ||||
|         itunesId: podcastExpanded.itunesId, | ||||
|         itunesArtistId: podcastExpanded.itunesArtistId, | ||||
|         explicit: podcastExpanded.explicit, | ||||
|         language: podcastExpanded.language, | ||||
|         type: podcastExpanded.podcastType | ||||
|       }, | ||||
|       coverPath: podcastExpanded.coverPath, | ||||
|       tags: podcastExpanded.tags, | ||||
|       episodes: podcastEpisodes || [], | ||||
|       autoDownloadEpisodes: podcastExpanded.autoDownloadEpisodes, | ||||
|       autoDownloadSchedule: podcastExpanded.autoDownloadSchedule, | ||||
|       lastEpisodeCheck: podcastExpanded.lastEpisodeCheck?.valueOf() || null, | ||||
|       maxEpisodesToKeep: podcastExpanded.maxEpisodesToKeep, | ||||
|       maxNewEpisodesToDownload: podcastExpanded.maxNewEpisodesToDownload | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   Podcast.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     title: DataTypes.STRING, | ||||
|     titleIgnorePrefix: DataTypes.STRING, | ||||
|     author: DataTypes.STRING, | ||||
|     releaseDate: DataTypes.STRING, | ||||
|     feedURL: DataTypes.STRING, | ||||
|     imageURL: DataTypes.STRING, | ||||
|     description: DataTypes.TEXT, | ||||
|     itunesPageURL: DataTypes.STRING, | ||||
|     itunesId: DataTypes.STRING, | ||||
|     itunesArtistId: DataTypes.STRING, | ||||
|     language: DataTypes.STRING, | ||||
|     podcastType: DataTypes.STRING, | ||||
|     explicit: DataTypes.BOOLEAN, | ||||
|   static getFromOld(oldPodcast) { | ||||
|     const oldPodcastMetadata = oldPodcast.metadata | ||||
|     return { | ||||
|       id: oldPodcast.id, | ||||
|       title: oldPodcastMetadata.title, | ||||
|       titleIgnorePrefix: oldPodcastMetadata.titleIgnorePrefix, | ||||
|       author: oldPodcastMetadata.author, | ||||
|       releaseDate: oldPodcastMetadata.releaseDate, | ||||
|       feedURL: oldPodcastMetadata.feedUrl, | ||||
|       imageURL: oldPodcastMetadata.imageUrl, | ||||
|       description: oldPodcastMetadata.description, | ||||
|       itunesPageURL: oldPodcastMetadata.itunesPageUrl, | ||||
|       itunesId: oldPodcastMetadata.itunesId, | ||||
|       itunesArtistId: oldPodcastMetadata.itunesArtistId, | ||||
|       language: oldPodcastMetadata.language, | ||||
|       podcastType: oldPodcastMetadata.type, | ||||
|       explicit: !!oldPodcastMetadata.explicit, | ||||
|       autoDownloadEpisodes: !!oldPodcast.autoDownloadEpisodes, | ||||
|       autoDownloadSchedule: oldPodcast.autoDownloadSchedule, | ||||
|       lastEpisodeCheck: oldPodcast.lastEpisodeCheck, | ||||
|       maxEpisodesToKeep: oldPodcast.maxEpisodesToKeep, | ||||
|       maxNewEpisodesToDownload: oldPodcast.maxNewEpisodesToDownload, | ||||
|       coverPath: oldPodcast.coverPath, | ||||
|       tags: oldPodcast.tags, | ||||
|       genres: oldPodcastMetadata.genres | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|     autoDownloadEpisodes: DataTypes.BOOLEAN, | ||||
|     autoDownloadSchedule: DataTypes.STRING, | ||||
|     lastEpisodeCheck: DataTypes.DATE, | ||||
|     maxEpisodesToKeep: DataTypes.INTEGER, | ||||
|     maxNewEpisodesToDownload: DataTypes.INTEGER, | ||||
|     coverPath: DataTypes.STRING, | ||||
|     tags: DataTypes.JSON, | ||||
|     genres: DataTypes.JSON | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'podcast' | ||||
|   }) | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       title: DataTypes.STRING, | ||||
|       titleIgnorePrefix: DataTypes.STRING, | ||||
|       author: DataTypes.STRING, | ||||
|       releaseDate: DataTypes.STRING, | ||||
|       feedURL: DataTypes.STRING, | ||||
|       imageURL: DataTypes.STRING, | ||||
|       description: DataTypes.TEXT, | ||||
|       itunesPageURL: DataTypes.STRING, | ||||
|       itunesId: DataTypes.STRING, | ||||
|       itunesArtistId: DataTypes.STRING, | ||||
|       language: DataTypes.STRING, | ||||
|       podcastType: DataTypes.STRING, | ||||
|       explicit: DataTypes.BOOLEAN, | ||||
| 
 | ||||
|   return Podcast | ||||
|       autoDownloadEpisodes: DataTypes.BOOLEAN, | ||||
|       autoDownloadSchedule: DataTypes.STRING, | ||||
|       lastEpisodeCheck: DataTypes.DATE, | ||||
|       maxEpisodesToKeep: DataTypes.INTEGER, | ||||
|       maxNewEpisodesToDownload: DataTypes.INTEGER, | ||||
|       coverPath: DataTypes.STRING, | ||||
|       tags: DataTypes.JSON, | ||||
|       genres: DataTypes.JSON | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'podcast' | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = Podcast | ||||
| @ -1,102 +1,154 @@ | ||||
| const { DataTypes, Model } = require('sequelize') | ||||
| const oldPodcastEpisode = require('../objects/entities/PodcastEpisode') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class PodcastEpisode extends Model { | ||||
|     getOldPodcastEpisode(libraryItemId = null) { | ||||
|       let enclosure = null | ||||
|       if (this.enclosureURL) { | ||||
|         enclosure = { | ||||
|           url: this.enclosureURL, | ||||
|           type: this.enclosureType, | ||||
|           length: this.enclosureSize !== null ? String(this.enclosureSize) : null | ||||
|         } | ||||
|       } | ||||
|       return { | ||||
|         libraryItemId: libraryItemId || null, | ||||
|         podcastId: this.podcastId, | ||||
|         id: this.id, | ||||
|         oldEpisodeId: this.extraData?.oldEpisodeId || null, | ||||
|         index: this.index, | ||||
|         season: this.season, | ||||
|         episode: this.episode, | ||||
|         episodeType: this.episodeType, | ||||
|         title: this.title, | ||||
|         subtitle: this.subtitle, | ||||
|         description: this.description, | ||||
|         enclosure, | ||||
|         pubDate: this.pubDate, | ||||
|         chapters: this.chapters, | ||||
|         audioFile: this.audioFile, | ||||
|         publishedAt: this.publishedAt?.valueOf() || null, | ||||
|         addedAt: this.createdAt.valueOf(), | ||||
|         updatedAt: this.updatedAt.valueOf() | ||||
| class PodcastEpisode extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {number} */ | ||||
|     this.index | ||||
|     /** @type {string} */ | ||||
|     this.season | ||||
|     /** @type {string} */ | ||||
|     this.episode | ||||
|     /** @type {string} */ | ||||
|     this.episodeType | ||||
|     /** @type {string} */ | ||||
|     this.title | ||||
|     /** @type {string} */ | ||||
|     this.subtitle | ||||
|     /** @type {string} */ | ||||
|     this.description | ||||
|     /** @type {string} */ | ||||
|     this.pubDate | ||||
|     /** @type {string} */ | ||||
|     this.enclosureURL | ||||
|     /** @type {BigInt} */ | ||||
|     this.enclosureSize | ||||
|     /** @type {string} */ | ||||
|     this.enclosureType | ||||
|     /** @type {Date} */ | ||||
|     this.publishedAt | ||||
|     /** @type {Object} */ | ||||
|     this.audioFile | ||||
|     /** @type {Object} */ | ||||
|     this.chapters | ||||
|     /** @type {Object} */ | ||||
|     this.extraData | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.podcastId | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * @param {string} libraryItemId  | ||||
|    * @returns {oldPodcastEpisode} | ||||
|    */ | ||||
|   getOldPodcastEpisode(libraryItemId = null) { | ||||
|     let enclosure = null | ||||
|     if (this.enclosureURL) { | ||||
|       enclosure = { | ||||
|         url: this.enclosureURL, | ||||
|         type: this.enclosureType, | ||||
|         length: this.enclosureSize !== null ? String(this.enclosureSize) : null | ||||
|       } | ||||
|     } | ||||
|     return new oldPodcastEpisode({ | ||||
|       libraryItemId: libraryItemId || null, | ||||
|       podcastId: this.podcastId, | ||||
|       id: this.id, | ||||
|       oldEpisodeId: this.extraData?.oldEpisodeId || null, | ||||
|       index: this.index, | ||||
|       season: this.season, | ||||
|       episode: this.episode, | ||||
|       episodeType: this.episodeType, | ||||
|       title: this.title, | ||||
|       subtitle: this.subtitle, | ||||
|       description: this.description, | ||||
|       enclosure, | ||||
|       pubDate: this.pubDate, | ||||
|       chapters: this.chapters, | ||||
|       audioFile: this.audioFile, | ||||
|       publishedAt: this.publishedAt?.valueOf() || null, | ||||
|       addedAt: this.createdAt.valueOf(), | ||||
|       updatedAt: this.updatedAt.valueOf() | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|     static createFromOld(oldEpisode) { | ||||
|       const podcastEpisode = this.getFromOld(oldEpisode) | ||||
|       return this.create(podcastEpisode) | ||||
|   static createFromOld(oldEpisode) { | ||||
|     const podcastEpisode = this.getFromOld(oldEpisode) | ||||
|     return this.create(podcastEpisode) | ||||
|   } | ||||
| 
 | ||||
|   static getFromOld(oldEpisode) { | ||||
|     const extraData = {} | ||||
|     if (oldEpisode.oldEpisodeId) { | ||||
|       extraData.oldEpisodeId = oldEpisode.oldEpisodeId | ||||
|     } | ||||
| 
 | ||||
|     static getFromOld(oldEpisode) { | ||||
|       const extraData = {} | ||||
|       if (oldEpisode.oldEpisodeId) { | ||||
|         extraData.oldEpisodeId = oldEpisode.oldEpisodeId | ||||
|       } | ||||
|       return { | ||||
|         id: oldEpisode.id, | ||||
|         index: oldEpisode.index, | ||||
|         season: oldEpisode.season, | ||||
|         episode: oldEpisode.episode, | ||||
|         episodeType: oldEpisode.episodeType, | ||||
|         title: oldEpisode.title, | ||||
|         subtitle: oldEpisode.subtitle, | ||||
|         description: oldEpisode.description, | ||||
|         pubDate: oldEpisode.pubDate, | ||||
|         enclosureURL: oldEpisode.enclosure?.url || null, | ||||
|         enclosureSize: oldEpisode.enclosure?.length || null, | ||||
|         enclosureType: oldEpisode.enclosure?.type || null, | ||||
|         publishedAt: oldEpisode.publishedAt, | ||||
|         podcastId: oldEpisode.podcastId, | ||||
|         audioFile: oldEpisode.audioFile?.toJSON() || null, | ||||
|         chapters: oldEpisode.chapters, | ||||
|         extraData | ||||
|       } | ||||
|     return { | ||||
|       id: oldEpisode.id, | ||||
|       index: oldEpisode.index, | ||||
|       season: oldEpisode.season, | ||||
|       episode: oldEpisode.episode, | ||||
|       episodeType: oldEpisode.episodeType, | ||||
|       title: oldEpisode.title, | ||||
|       subtitle: oldEpisode.subtitle, | ||||
|       description: oldEpisode.description, | ||||
|       pubDate: oldEpisode.pubDate, | ||||
|       enclosureURL: oldEpisode.enclosure?.url || null, | ||||
|       enclosureSize: oldEpisode.enclosure?.length || null, | ||||
|       enclosureType: oldEpisode.enclosure?.type || null, | ||||
|       publishedAt: oldEpisode.publishedAt, | ||||
|       podcastId: oldEpisode.podcastId, | ||||
|       audioFile: oldEpisode.audioFile?.toJSON() || null, | ||||
|       chapters: oldEpisode.chapters, | ||||
|       extraData | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   PodcastEpisode.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     index: DataTypes.INTEGER, | ||||
|     season: DataTypes.STRING, | ||||
|     episode: DataTypes.STRING, | ||||
|     episodeType: DataTypes.STRING, | ||||
|     title: DataTypes.STRING, | ||||
|     subtitle: DataTypes.STRING(1000), | ||||
|     description: DataTypes.TEXT, | ||||
|     pubDate: DataTypes.STRING, | ||||
|     enclosureURL: DataTypes.STRING, | ||||
|     enclosureSize: DataTypes.BIGINT, | ||||
|     enclosureType: DataTypes.STRING, | ||||
|     publishedAt: DataTypes.DATE, | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       index: DataTypes.INTEGER, | ||||
|       season: DataTypes.STRING, | ||||
|       episode: DataTypes.STRING, | ||||
|       episodeType: DataTypes.STRING, | ||||
|       title: DataTypes.STRING, | ||||
|       subtitle: DataTypes.STRING(1000), | ||||
|       description: DataTypes.TEXT, | ||||
|       pubDate: DataTypes.STRING, | ||||
|       enclosureURL: DataTypes.STRING, | ||||
|       enclosureSize: DataTypes.BIGINT, | ||||
|       enclosureType: DataTypes.STRING, | ||||
|       publishedAt: DataTypes.DATE, | ||||
| 
 | ||||
|     audioFile: DataTypes.JSON, | ||||
|     chapters: DataTypes.JSON, | ||||
|     extraData: DataTypes.JSON | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'podcastEpisode' | ||||
|   }) | ||||
|       audioFile: DataTypes.JSON, | ||||
|       chapters: DataTypes.JSON, | ||||
|       extraData: DataTypes.JSON | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'podcastEpisode' | ||||
|     }) | ||||
| 
 | ||||
|   const { podcast } = sequelize.models | ||||
|   podcast.hasMany(PodcastEpisode, { | ||||
|     onDelete: 'CASCADE' | ||||
|   }) | ||||
|   PodcastEpisode.belongsTo(podcast) | ||||
| 
 | ||||
|   return PodcastEpisode | ||||
|     const { podcast } = sequelize.models | ||||
|     podcast.hasMany(PodcastEpisode, { | ||||
|       onDelete: 'CASCADE' | ||||
|     }) | ||||
|     PodcastEpisode.belongsTo(podcast) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = PodcastEpisode | ||||
| @ -2,81 +2,130 @@ const { DataTypes, Model } = require('sequelize') | ||||
| 
 | ||||
| const oldSeries = require('../objects/entities/Series') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class Series extends Model { | ||||
|     static async getAllOldSeries() { | ||||
|       const series = await this.findAll() | ||||
|       return series.map(se => se.getOldSeries()) | ||||
|     } | ||||
| class Series extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     getOldSeries() { | ||||
|       return new oldSeries({ | ||||
|         id: this.id, | ||||
|         name: this.name, | ||||
|         description: this.description, | ||||
|         libraryId: this.libraryId, | ||||
|         addedAt: this.createdAt.valueOf(), | ||||
|         updatedAt: this.updatedAt.valueOf() | ||||
|       }) | ||||
|     } | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {string} */ | ||||
|     this.name | ||||
|     /** @type {string} */ | ||||
|     this.nameIgnorePrefix | ||||
|     /** @type {string} */ | ||||
|     this.description | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.libraryId | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|   } | ||||
| 
 | ||||
|     static updateFromOld(oldSeries) { | ||||
|       const series = this.getFromOld(oldSeries) | ||||
|       return this.update(series, { | ||||
|         where: { | ||||
|           id: series.id | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
|   static async getAllOldSeries() { | ||||
|     const series = await this.findAll() | ||||
|     return series.map(se => se.getOldSeries()) | ||||
|   } | ||||
| 
 | ||||
|     static createFromOld(oldSeries) { | ||||
|       const series = this.getFromOld(oldSeries) | ||||
|       return this.create(series) | ||||
|     } | ||||
|   getOldSeries() { | ||||
|     return new oldSeries({ | ||||
|       id: this.id, | ||||
|       name: this.name, | ||||
|       description: this.description, | ||||
|       libraryId: this.libraryId, | ||||
|       addedAt: this.createdAt.valueOf(), | ||||
|       updatedAt: this.updatedAt.valueOf() | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|     static createBulkFromOld(oldSeriesObjs) { | ||||
|       const series = oldSeriesObjs.map(this.getFromOld) | ||||
|       return this.bulkCreate(series) | ||||
|     } | ||||
| 
 | ||||
|     static getFromOld(oldSeries) { | ||||
|       return { | ||||
|         id: oldSeries.id, | ||||
|         name: oldSeries.name, | ||||
|         nameIgnorePrefix: oldSeries.nameIgnorePrefix, | ||||
|         description: oldSeries.description, | ||||
|         libraryId: oldSeries.libraryId | ||||
|   static updateFromOld(oldSeries) { | ||||
|     const series = this.getFromOld(oldSeries) | ||||
|     return this.update(series, { | ||||
|       where: { | ||||
|         id: series.id | ||||
|       } | ||||
|     } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|     static removeById(seriesId) { | ||||
|       return this.destroy({ | ||||
|         where: { | ||||
|           id: seriesId | ||||
|         } | ||||
|       }) | ||||
|   static createFromOld(oldSeries) { | ||||
|     const series = this.getFromOld(oldSeries) | ||||
|     return this.create(series) | ||||
|   } | ||||
| 
 | ||||
|   static createBulkFromOld(oldSeriesObjs) { | ||||
|     const series = oldSeriesObjs.map(this.getFromOld) | ||||
|     return this.bulkCreate(series) | ||||
|   } | ||||
| 
 | ||||
|   static getFromOld(oldSeries) { | ||||
|     return { | ||||
|       id: oldSeries.id, | ||||
|       name: oldSeries.name, | ||||
|       nameIgnorePrefix: oldSeries.nameIgnorePrefix, | ||||
|       description: oldSeries.description, | ||||
|       libraryId: oldSeries.libraryId | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   Series.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     name: DataTypes.STRING, | ||||
|     nameIgnorePrefix: DataTypes.STRING, | ||||
|     description: DataTypes.TEXT | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'series' | ||||
|   }) | ||||
|   static removeById(seriesId) { | ||||
|     return this.destroy({ | ||||
|       where: { | ||||
|         id: seriesId | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   const { library } = sequelize.models | ||||
|   library.hasMany(Series, { | ||||
|     onDelete: 'CASCADE' | ||||
|   }) | ||||
|   Series.belongsTo(library) | ||||
|   /** | ||||
|    * Check if series exists | ||||
|    * @param {string} seriesId  | ||||
|    * @returns {Promise<boolean>} | ||||
|    */ | ||||
|   static async checkExistsById(seriesId) { | ||||
|     return (await this.count({ where: { id: seriesId } })) > 0 | ||||
|   } | ||||
| 
 | ||||
|   return Series | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       name: DataTypes.STRING, | ||||
|       nameIgnorePrefix: DataTypes.STRING, | ||||
|       description: DataTypes.TEXT | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'series', | ||||
|       indexes: [ | ||||
|         { | ||||
|           fields: [{ | ||||
|             name: 'name', | ||||
|             collate: 'NOCASE' | ||||
|           }] | ||||
|         }, | ||||
|         { | ||||
|           fields: [{ | ||||
|             name: 'nameIgnorePrefix', | ||||
|             collate: 'NOCASE' | ||||
|           }] | ||||
|         }, | ||||
|         { | ||||
|           fields: ['libraryId'] | ||||
|         } | ||||
|       ] | ||||
|     }) | ||||
| 
 | ||||
|     const { library } = sequelize.models | ||||
|     library.hasMany(Series, { | ||||
|       onDelete: 'CASCADE' | ||||
|     }) | ||||
|     Series.belongsTo(library) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = Series | ||||
| @ -4,42 +4,59 @@ const oldEmailSettings = require('../objects/settings/EmailSettings') | ||||
| const oldServerSettings = require('../objects/settings/ServerSettings') | ||||
| const oldNotificationSettings = require('../objects/settings/NotificationSettings') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class Setting extends Model { | ||||
|     static async getOldSettings() { | ||||
|       const settings = (await this.findAll()).map(se => se.value) | ||||
| class Setting extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     /** @type {string} */ | ||||
|     this.key | ||||
|     /** @type {Object} */ | ||||
|     this.value | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|   } | ||||
| 
 | ||||
|   static async getOldSettings() { | ||||
|     const settings = (await this.findAll()).map(se => se.value) | ||||
| 
 | ||||
| 
 | ||||
|       const emailSettingsJson = settings.find(se => se.id === 'email-settings') | ||||
|       const serverSettingsJson = settings.find(se => se.id === 'server-settings') | ||||
|       const notificationSettingsJson = settings.find(se => se.id === 'notification-settings') | ||||
|     const emailSettingsJson = settings.find(se => se.id === 'email-settings') | ||||
|     const serverSettingsJson = settings.find(se => se.id === 'server-settings') | ||||
|     const notificationSettingsJson = settings.find(se => se.id === 'notification-settings') | ||||
| 
 | ||||
|       return { | ||||
|         settings, | ||||
|         emailSettings: new oldEmailSettings(emailSettingsJson), | ||||
|         serverSettings: new oldServerSettings(serverSettingsJson), | ||||
|         notificationSettings: new oldNotificationSettings(notificationSettingsJson) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     static updateSettingObj(setting) { | ||||
|       return this.upsert({ | ||||
|         key: setting.id, | ||||
|         value: setting | ||||
|       }) | ||||
|     return { | ||||
|       settings, | ||||
|       emailSettings: new oldEmailSettings(emailSettingsJson), | ||||
|       serverSettings: new oldServerSettings(serverSettingsJson), | ||||
|       notificationSettings: new oldNotificationSettings(notificationSettingsJson) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   Setting.init({ | ||||
|     key: { | ||||
|       type: DataTypes.STRING, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     value: DataTypes.JSON | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'setting' | ||||
|   }) | ||||
|   static updateSettingObj(setting) { | ||||
|     return this.upsert({ | ||||
|       key: setting.id, | ||||
|       value: setting | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   return Setting | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       key: { | ||||
|         type: DataTypes.STRING, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       value: DataTypes.JSON | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'setting' | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = Setting | ||||
| @ -3,238 +3,273 @@ const { DataTypes, Model, Op } = require('sequelize') | ||||
| const Logger = require('../Logger') | ||||
| const oldUser = require('../objects/user/User') | ||||
| 
 | ||||
| module.exports = (sequelize) => { | ||||
|   class User extends Model { | ||||
|     /** | ||||
|      * Get all oldUsers | ||||
|      * @returns {Promise<oldUser>} | ||||
|      */ | ||||
|     static async getOldUsers() { | ||||
|       const users = await this.findAll({ | ||||
|         include: sequelize.models.mediaProgress | ||||
|       }) | ||||
|       return users.map(u => this.getOldUser(u)) | ||||
|     } | ||||
| class User extends Model { | ||||
|   constructor(values, options) { | ||||
|     super(values, options) | ||||
| 
 | ||||
|     static getOldUser(userExpanded) { | ||||
|       const mediaProgress = userExpanded.mediaProgresses.map(mp => mp.getOldMediaProgress()) | ||||
|     /** @type {UUIDV4} */ | ||||
|     this.id | ||||
|     /** @type {string} */ | ||||
|     this.username | ||||
|     /** @type {string} */ | ||||
|     this.email | ||||
|     /** @type {string} */ | ||||
|     this.pash | ||||
|     /** @type {string} */ | ||||
|     this.type | ||||
|     /** @type {boolean} */ | ||||
|     this.isActive | ||||
|     /** @type {boolean} */ | ||||
|     this.isLocked | ||||
|     /** @type {Date} */ | ||||
|     this.lastSeen | ||||
|     /** @type {Object} */ | ||||
|     this.permissions | ||||
|     /** @type {Object} */ | ||||
|     this.bookmarks | ||||
|     /** @type {Object} */ | ||||
|     this.extraData | ||||
|     /** @type {Date} */ | ||||
|     this.createdAt | ||||
|     /** @type {Date} */ | ||||
|     this.updatedAt | ||||
|   } | ||||
| 
 | ||||
|       const librariesAccessible = userExpanded.permissions?.librariesAccessible || [] | ||||
|       const itemTagsSelected = userExpanded.permissions?.itemTagsSelected || [] | ||||
|       const permissions = userExpanded.permissions || {} | ||||
|       delete permissions.librariesAccessible | ||||
|       delete permissions.itemTagsSelected | ||||
|   /** | ||||
|    * Get all oldUsers | ||||
|    * @returns {Promise<oldUser>} | ||||
|    */ | ||||
|   static async getOldUsers() { | ||||
|     const users = await this.findAll({ | ||||
|       include: this.sequelize.models.mediaProgress | ||||
|     }) | ||||
|     return users.map(u => this.getOldUser(u)) | ||||
|   } | ||||
| 
 | ||||
|       return new oldUser({ | ||||
|         id: userExpanded.id, | ||||
|         oldUserId: userExpanded.extraData?.oldUserId || null, | ||||
|         username: userExpanded.username, | ||||
|         pash: userExpanded.pash, | ||||
|         type: userExpanded.type, | ||||
|         token: userExpanded.token, | ||||
|         mediaProgress, | ||||
|         seriesHideFromContinueListening: userExpanded.extraData?.seriesHideFromContinueListening || [], | ||||
|         bookmarks: userExpanded.bookmarks, | ||||
|         isActive: userExpanded.isActive, | ||||
|         isLocked: userExpanded.isLocked, | ||||
|         lastSeen: userExpanded.lastSeen?.valueOf() || null, | ||||
|         createdAt: userExpanded.createdAt.valueOf(), | ||||
|         permissions, | ||||
|         librariesAccessible, | ||||
|         itemTagsSelected | ||||
|       }) | ||||
|     } | ||||
|   static getOldUser(userExpanded) { | ||||
|     const mediaProgress = userExpanded.mediaProgresses.map(mp => mp.getOldMediaProgress()) | ||||
| 
 | ||||
|     static createFromOld(oldUser) { | ||||
|       const user = this.getFromOld(oldUser) | ||||
|       return this.create(user) | ||||
|     } | ||||
|     const librariesAccessible = userExpanded.permissions?.librariesAccessible || [] | ||||
|     const itemTagsSelected = userExpanded.permissions?.itemTagsSelected || [] | ||||
|     const permissions = userExpanded.permissions || {} | ||||
|     delete permissions.librariesAccessible | ||||
|     delete permissions.itemTagsSelected | ||||
| 
 | ||||
|     static updateFromOld(oldUser) { | ||||
|       const user = this.getFromOld(oldUser) | ||||
|       return this.update(user, { | ||||
|         where: { | ||||
|           id: user.id | ||||
|         } | ||||
|       }).then((result) => result[0] > 0).catch((error) => { | ||||
|         Logger.error(`[User] Failed to save user ${oldUser.id}`, error) | ||||
|         return false | ||||
|       }) | ||||
|     } | ||||
|     return new oldUser({ | ||||
|       id: userExpanded.id, | ||||
|       oldUserId: userExpanded.extraData?.oldUserId || null, | ||||
|       username: userExpanded.username, | ||||
|       pash: userExpanded.pash, | ||||
|       type: userExpanded.type, | ||||
|       token: userExpanded.token, | ||||
|       mediaProgress, | ||||
|       seriesHideFromContinueListening: userExpanded.extraData?.seriesHideFromContinueListening || [], | ||||
|       bookmarks: userExpanded.bookmarks, | ||||
|       isActive: userExpanded.isActive, | ||||
|       isLocked: userExpanded.isLocked, | ||||
|       lastSeen: userExpanded.lastSeen?.valueOf() || null, | ||||
|       createdAt: userExpanded.createdAt.valueOf(), | ||||
|       permissions, | ||||
|       librariesAccessible, | ||||
|       itemTagsSelected | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|     static getFromOld(oldUser) { | ||||
|       return { | ||||
|         id: oldUser.id, | ||||
|         username: oldUser.username, | ||||
|         pash: oldUser.pash || null, | ||||
|         type: oldUser.type || null, | ||||
|         token: oldUser.token || null, | ||||
|         isActive: !!oldUser.isActive, | ||||
|         lastSeen: oldUser.lastSeen || null, | ||||
|         extraData: { | ||||
|           seriesHideFromContinueListening: oldUser.seriesHideFromContinueListening || [], | ||||
|           oldUserId: oldUser.oldUserId | ||||
|         }, | ||||
|         createdAt: oldUser.createdAt || Date.now(), | ||||
|         permissions: { | ||||
|           ...oldUser.permissions, | ||||
|           librariesAccessible: oldUser.librariesAccessible || [], | ||||
|           itemTagsSelected: oldUser.itemTagsSelected || [] | ||||
|         }, | ||||
|         bookmarks: oldUser.bookmarks | ||||
|   static createFromOld(oldUser) { | ||||
|     const user = this.getFromOld(oldUser) | ||||
|     return this.create(user) | ||||
|   } | ||||
| 
 | ||||
|   static updateFromOld(oldUser) { | ||||
|     const user = this.getFromOld(oldUser) | ||||
|     return this.update(user, { | ||||
|       where: { | ||||
|         id: user.id | ||||
|       } | ||||
|     } | ||||
|     }).then((result) => result[0] > 0).catch((error) => { | ||||
|       Logger.error(`[User] Failed to save user ${oldUser.id}`, error) | ||||
|       return false | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|     static removeById(userId) { | ||||
|       return this.destroy({ | ||||
|         where: { | ||||
|           id: userId | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Create root user | ||||
|      * @param {string} username  | ||||
|      * @param {string} pash  | ||||
|      * @param {Auth} auth  | ||||
|      * @returns {oldUser} | ||||
|      */ | ||||
|     static async createRootUser(username, pash, auth) { | ||||
|       const userId = uuidv4() | ||||
| 
 | ||||
|       const token = await auth.generateAccessToken({ userId, username }) | ||||
| 
 | ||||
|       const newRoot = new oldUser({ | ||||
|         id: userId, | ||||
|         type: 'root', | ||||
|         username, | ||||
|         pash, | ||||
|         token, | ||||
|         isActive: true, | ||||
|         createdAt: Date.now() | ||||
|       }) | ||||
|       await this.createFromOld(newRoot) | ||||
|       return newRoot | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Get a user by id or by the old database id | ||||
|      * @temp User ids were updated in v2.3.0 migration and old API tokens may still use that id | ||||
|      * @param {string} userId  | ||||
|      * @returns {Promise<oldUser|null>} null if not found | ||||
|      */ | ||||
|     static async getUserByIdOrOldId(userId) { | ||||
|       if (!userId) return null | ||||
|       const user = await this.findOne({ | ||||
|         where: { | ||||
|           [Op.or]: [ | ||||
|             { | ||||
|               id: userId | ||||
|             }, | ||||
|             { | ||||
|               extraData: { | ||||
|                 [Op.substring]: userId | ||||
|               } | ||||
|             } | ||||
|           ] | ||||
|         }, | ||||
|         include: sequelize.models.mediaProgress | ||||
|       }) | ||||
|       if (!user) return null | ||||
|       return this.getOldUser(user) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Get user by username case insensitive | ||||
|      * @param {string} username  | ||||
|      * @returns {Promise<oldUser|null>} returns null if not found | ||||
|      */ | ||||
|     static async getUserByUsername(username) { | ||||
|       if (!username) return null | ||||
|       const user = await this.findOne({ | ||||
|         where: { | ||||
|           username: { | ||||
|             [Op.like]: username | ||||
|           } | ||||
|         }, | ||||
|         include: sequelize.models.mediaProgress | ||||
|       }) | ||||
|       if (!user) return null | ||||
|       return this.getOldUser(user) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Get user by id | ||||
|      * @param {string} userId  | ||||
|      * @returns {Promise<oldUser|null>} returns null if not found | ||||
|      */ | ||||
|     static async getUserById(userId) { | ||||
|       if (!userId) return null | ||||
|       const user = await this.findByPk(userId, { | ||||
|         include: sequelize.models.mediaProgress | ||||
|       }) | ||||
|       if (!user) return null | ||||
|       return this.getOldUser(user) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Get array of user id and username | ||||
|      * @returns {object[]} { id, username } | ||||
|      */ | ||||
|     static async getMinifiedUserObjects() { | ||||
|       const users = await this.findAll({ | ||||
|         attributes: ['id', 'username'] | ||||
|       }) | ||||
|       return users.map(u => { | ||||
|         return { | ||||
|           id: u.id, | ||||
|           username: u.username | ||||
|         } | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Return true if root user exists | ||||
|      * @returns {boolean} | ||||
|      */ | ||||
|     static async getHasRootUser() { | ||||
|       const count = await this.count({ | ||||
|         where: { | ||||
|           type: 'root' | ||||
|         } | ||||
|       }) | ||||
|       return count > 0 | ||||
|   static getFromOld(oldUser) { | ||||
|     return { | ||||
|       id: oldUser.id, | ||||
|       username: oldUser.username, | ||||
|       pash: oldUser.pash || null, | ||||
|       type: oldUser.type || null, | ||||
|       token: oldUser.token || null, | ||||
|       isActive: !!oldUser.isActive, | ||||
|       lastSeen: oldUser.lastSeen || null, | ||||
|       extraData: { | ||||
|         seriesHideFromContinueListening: oldUser.seriesHideFromContinueListening || [], | ||||
|         oldUserId: oldUser.oldUserId | ||||
|       }, | ||||
|       createdAt: oldUser.createdAt || Date.now(), | ||||
|       permissions: { | ||||
|         ...oldUser.permissions, | ||||
|         librariesAccessible: oldUser.librariesAccessible || [], | ||||
|         itemTagsSelected: oldUser.itemTagsSelected || [] | ||||
|       }, | ||||
|       bookmarks: oldUser.bookmarks | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   User.init({ | ||||
|     id: { | ||||
|       type: DataTypes.UUID, | ||||
|       defaultValue: DataTypes.UUIDV4, | ||||
|       primaryKey: true | ||||
|     }, | ||||
|     username: DataTypes.STRING, | ||||
|     email: DataTypes.STRING, | ||||
|     pash: DataTypes.STRING, | ||||
|     type: DataTypes.STRING, | ||||
|     token: DataTypes.STRING, | ||||
|     isActive: { | ||||
|       type: DataTypes.BOOLEAN, | ||||
|       defaultValue: false | ||||
|     }, | ||||
|     isLocked: { | ||||
|       type: DataTypes.BOOLEAN, | ||||
|       defaultValue: false | ||||
|     }, | ||||
|     lastSeen: DataTypes.DATE, | ||||
|     permissions: DataTypes.JSON, | ||||
|     bookmarks: DataTypes.JSON, | ||||
|     extraData: DataTypes.JSON | ||||
|   }, { | ||||
|     sequelize, | ||||
|     modelName: 'user' | ||||
|   }) | ||||
|   static removeById(userId) { | ||||
|     return this.destroy({ | ||||
|       where: { | ||||
|         id: userId | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   return User | ||||
|   /** | ||||
|    * Create root user | ||||
|    * @param {string} username  | ||||
|    * @param {string} pash  | ||||
|    * @param {Auth} auth  | ||||
|    * @returns {oldUser} | ||||
|    */ | ||||
|   static async createRootUser(username, pash, auth) { | ||||
|     const userId = uuidv4() | ||||
| 
 | ||||
|     const token = await auth.generateAccessToken({ userId, username }) | ||||
| 
 | ||||
|     const newRoot = new oldUser({ | ||||
|       id: userId, | ||||
|       type: 'root', | ||||
|       username, | ||||
|       pash, | ||||
|       token, | ||||
|       isActive: true, | ||||
|       createdAt: Date.now() | ||||
|     }) | ||||
|     await this.createFromOld(newRoot) | ||||
|     return newRoot | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get a user by id or by the old database id | ||||
|    * @temp User ids were updated in v2.3.0 migration and old API tokens may still use that id | ||||
|    * @param {string} userId  | ||||
|    * @returns {Promise<oldUser|null>} null if not found | ||||
|    */ | ||||
|   static async getUserByIdOrOldId(userId) { | ||||
|     if (!userId) return null | ||||
|     const user = await this.findOne({ | ||||
|       where: { | ||||
|         [Op.or]: [ | ||||
|           { | ||||
|             id: userId | ||||
|           }, | ||||
|           { | ||||
|             extraData: { | ||||
|               [Op.substring]: userId | ||||
|             } | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       include: this.sequelize.models.mediaProgress | ||||
|     }) | ||||
|     if (!user) return null | ||||
|     return this.getOldUser(user) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get user by username case insensitive | ||||
|    * @param {string} username  | ||||
|    * @returns {Promise<oldUser|null>} returns null if not found | ||||
|    */ | ||||
|   static async getUserByUsername(username) { | ||||
|     if (!username) return null | ||||
|     const user = await this.findOne({ | ||||
|       where: { | ||||
|         username: { | ||||
|           [Op.like]: username | ||||
|         } | ||||
|       }, | ||||
|       include: this.sequelize.models.mediaProgress | ||||
|     }) | ||||
|     if (!user) return null | ||||
|     return this.getOldUser(user) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get user by id | ||||
|    * @param {string} userId  | ||||
|    * @returns {Promise<oldUser|null>} returns null if not found | ||||
|    */ | ||||
|   static async getUserById(userId) { | ||||
|     if (!userId) return null | ||||
|     const user = await this.findByPk(userId, { | ||||
|       include: this.sequelize.models.mediaProgress | ||||
|     }) | ||||
|     if (!user) return null | ||||
|     return this.getOldUser(user) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get array of user id and username | ||||
|    * @returns {object[]} { id, username } | ||||
|    */ | ||||
|   static async getMinifiedUserObjects() { | ||||
|     const users = await this.findAll({ | ||||
|       attributes: ['id', 'username'] | ||||
|     }) | ||||
|     return users.map(u => { | ||||
|       return { | ||||
|         id: u.id, | ||||
|         username: u.username | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Return true if root user exists | ||||
|    * @returns {boolean} | ||||
|    */ | ||||
|   static async getHasRootUser() { | ||||
|     const count = await this.count({ | ||||
|       where: { | ||||
|         type: 'root' | ||||
|       } | ||||
|     }) | ||||
|     return count > 0 | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Initialize model | ||||
|    * @param {import('../Database').sequelize} sequelize  | ||||
|    */ | ||||
|   static init(sequelize) { | ||||
|     super.init({ | ||||
|       id: { | ||||
|         type: DataTypes.UUID, | ||||
|         defaultValue: DataTypes.UUIDV4, | ||||
|         primaryKey: true | ||||
|       }, | ||||
|       username: DataTypes.STRING, | ||||
|       email: DataTypes.STRING, | ||||
|       pash: DataTypes.STRING, | ||||
|       type: DataTypes.STRING, | ||||
|       token: DataTypes.STRING, | ||||
|       isActive: { | ||||
|         type: DataTypes.BOOLEAN, | ||||
|         defaultValue: false | ||||
|       }, | ||||
|       isLocked: { | ||||
|         type: DataTypes.BOOLEAN, | ||||
|         defaultValue: false | ||||
|       }, | ||||
|       lastSeen: DataTypes.DATE, | ||||
|       permissions: DataTypes.JSON, | ||||
|       bookmarks: DataTypes.JSON, | ||||
|       extraData: DataTypes.JSON | ||||
|     }, { | ||||
|       sequelize, | ||||
|       modelName: 'user' | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| module.exports = User | ||||
| @ -1,5 +1,5 @@ | ||||
| const uuidv4 = require("uuid").v4 | ||||
| const { getTitleIgnorePrefix } = require('../../utils/index') | ||||
| const { getTitleIgnorePrefix, getTitlePrefixAtEnd } = require('../../utils/index') | ||||
| 
 | ||||
| class Series { | ||||
|   constructor(series) { | ||||
| @ -33,6 +33,7 @@ class Series { | ||||
|     return { | ||||
|       id: this.id, | ||||
|       name: this.name, | ||||
|       nameIgnorePrefix: getTitlePrefixAtEnd(this.name), | ||||
|       description: this.description, | ||||
|       addedAt: this.addedAt, | ||||
|       updatedAt: this.updatedAt, | ||||
|  | ||||
| @ -78,23 +78,22 @@ class ApiRouter { | ||||
|     this.router.get('/libraries/:id/items', LibraryController.middleware.bind(this), LibraryController.getLibraryItems.bind(this)) | ||||
|     this.router.delete('/libraries/:id/issues', LibraryController.middlewareNew.bind(this), LibraryController.removeLibraryItemsWithIssues.bind(this)) | ||||
|     this.router.get('/libraries/:id/episode-downloads', LibraryController.middlewareNew.bind(this), LibraryController.getEpisodeDownloadQueue.bind(this)) | ||||
|     this.router.get('/libraries/:id/series', LibraryController.middleware.bind(this), LibraryController.getAllSeriesForLibrary.bind(this)) | ||||
|     this.router.get('/libraries/:id/series', LibraryController.middlewareNew.bind(this), LibraryController.getAllSeriesForLibrary.bind(this)) | ||||
|     this.router.get('/libraries/:id/series/:seriesId', LibraryController.middlewareNew.bind(this), LibraryController.getSeriesForLibrary.bind(this)) | ||||
|     this.router.get('/libraries/:id/collections', LibraryController.middlewareNew.bind(this), LibraryController.getCollectionsForLibrary.bind(this)) | ||||
|     this.router.get('/libraries/:id/playlists', LibraryController.middlewareNew.bind(this), LibraryController.getUserPlaylistsForLibrary.bind(this)) | ||||
|     this.router.get('/libraries/:id/personalized2', LibraryController.middlewareNew.bind(this), LibraryController.getUserPersonalizedShelves.bind(this)) | ||||
|     this.router.get('/libraries/:id/personalized', LibraryController.middleware.bind(this), LibraryController.getLibraryUserPersonalizedOptimal.bind(this)) | ||||
|     this.router.get('/libraries/:id/personalized', LibraryController.middlewareNew.bind(this), LibraryController.getUserPersonalizedShelves.bind(this)) | ||||
|     this.router.get('/libraries/:id/filterdata', LibraryController.middlewareNew.bind(this), LibraryController.getLibraryFilterData.bind(this)) | ||||
|     this.router.get('/libraries/:id/search', LibraryController.middleware.bind(this), LibraryController.search.bind(this)) | ||||
|     this.router.get('/libraries/:id/stats', LibraryController.middleware.bind(this), LibraryController.stats.bind(this)) | ||||
|     this.router.get('/libraries/:id/search', LibraryController.middlewareNew.bind(this), LibraryController.search.bind(this)) | ||||
|     this.router.get('/libraries/:id/stats', LibraryController.middlewareNew.bind(this), LibraryController.stats.bind(this)) | ||||
|     this.router.get('/libraries/:id/authors', LibraryController.middlewareNew.bind(this), LibraryController.getAuthors.bind(this)) | ||||
|     this.router.get('/libraries/:id/narrators', LibraryController.middlewareNew.bind(this), LibraryController.getNarrators.bind(this)) | ||||
|     this.router.patch('/libraries/:id/narrators/:narratorId', LibraryController.middlewareNew.bind(this), LibraryController.updateNarrator.bind(this)) | ||||
|     this.router.delete('/libraries/:id/narrators/:narratorId', LibraryController.middlewareNew.bind(this), LibraryController.removeNarrator.bind(this)) | ||||
|     this.router.get('/libraries/:id/matchall', LibraryController.middleware.bind(this), LibraryController.matchAll.bind(this)) | ||||
|     this.router.post('/libraries/:id/scan', LibraryController.middleware.bind(this), LibraryController.scan.bind(this)) | ||||
|     this.router.get('/libraries/:id/recent-episodes', LibraryController.middleware.bind(this), LibraryController.getRecentEpisodes.bind(this)) | ||||
|     this.router.get('/libraries/:id/opml', LibraryController.middleware.bind(this), LibraryController.getOPMLFile.bind(this)) | ||||
|     this.router.get('/libraries/:id/matchall', LibraryController.middlewareNew.bind(this), LibraryController.matchAll.bind(this)) | ||||
|     this.router.post('/libraries/:id/scan', LibraryController.middlewareNew.bind(this), LibraryController.scan.bind(this)) | ||||
|     this.router.get('/libraries/:id/recent-episodes', LibraryController.middlewareNew.bind(this), LibraryController.getRecentEpisodes.bind(this)) | ||||
|     this.router.get('/libraries/:id/opml', LibraryController.middlewareNew.bind(this), LibraryController.getOPMLFile.bind(this)) | ||||
|     this.router.post('/libraries/order', LibraryController.reorder.bind(this)) | ||||
| 
 | ||||
|     //
 | ||||
| @ -352,34 +351,6 @@ class ApiRouter { | ||||
|   //
 | ||||
|   // Helper Methods
 | ||||
|   //
 | ||||
|   userJsonWithItemProgressDetails(user, hideRootToken = false) { | ||||
|     const json = user.toJSONForBrowser(hideRootToken) | ||||
| 
 | ||||
|     json.mediaProgress = json.mediaProgress.map(lip => { | ||||
|       const libraryItem = Database.libraryItems.find(li => li.id === lip.libraryItemId) | ||||
|       if (!libraryItem) { | ||||
|         Logger.warn('[ApiRouter] Library item not found for users progress ' + lip.libraryItemId) | ||||
|         lip.media = null | ||||
|       } else { | ||||
|         if (lip.episodeId) { | ||||
|           const episode = libraryItem.mediaType === 'podcast' ? libraryItem.media.getEpisode(lip.episodeId) : null | ||||
|           if (!episode) { | ||||
|             Logger.warn(`[ApiRouter] Episode ${lip.episodeId} not found for user media progress, podcast: ${libraryItem.media.metadata.title}`) | ||||
|             lip.media = null | ||||
|           } else { | ||||
|             lip.media = libraryItem.media.toJSONExpanded() | ||||
|             lip.episode = episode.toJSON() | ||||
|           } | ||||
|         } else { | ||||
|           lip.media = libraryItem.media.toJSONExpanded() | ||||
|         } | ||||
|       } | ||||
|       return lip | ||||
|     }).filter(lip => !!lip) | ||||
| 
 | ||||
|     return json | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Remove library item and associated entities | ||||
|    * @param {string} mediaType  | ||||
| @ -388,7 +359,7 @@ class ApiRouter { | ||||
|    */ | ||||
|   async handleDeleteLibraryItem(mediaType, libraryItemId, mediaItemIds) { | ||||
|     // Remove media progress for this library item from all users
 | ||||
|     const users = await Database.models.user.getOldUsers() | ||||
|     const users = await Database.userModel.getOldUsers() | ||||
|     for (const user of users) { | ||||
|       for (const mediaProgress of user.getAllMediaProgressForLibraryItem(libraryItemId)) { | ||||
|         await Database.removeMediaProgress(mediaProgress.id) | ||||
| @ -399,14 +370,15 @@ class ApiRouter { | ||||
| 
 | ||||
|     // Remove series if empty
 | ||||
|     if (mediaType === 'book') { | ||||
|       const bookSeries = await Database.models.bookSeries.findAll({ | ||||
|       // TODO: update filter data
 | ||||
|       const bookSeries = await Database.bookSeriesModel.findAll({ | ||||
|         where: { | ||||
|           bookId: mediaItemIds[0] | ||||
|         }, | ||||
|         include: { | ||||
|           model: Database.models.series, | ||||
|           model: Database.seriesModel, | ||||
|           include: { | ||||
|             model: Database.models.book | ||||
|             model: Database.bookModel | ||||
|           } | ||||
|         } | ||||
|       }) | ||||
| @ -418,7 +390,7 @@ class ApiRouter { | ||||
|     } | ||||
| 
 | ||||
|     // remove item from playlists
 | ||||
|     const playlistsWithItem = await Database.models.playlist.getPlaylistsForMediaItemIds(mediaItemIds) | ||||
|     const playlistsWithItem = await Database.playlistModel.getPlaylistsForMediaItemIds(mediaItemIds) | ||||
|     for (const playlist of playlistsWithItem) { | ||||
|       let numMediaItems = playlist.playlistMediaItems.length | ||||
| 
 | ||||
| @ -468,25 +440,51 @@ class ApiRouter { | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   async checkRemoveEmptySeries(seriesToCheck, excludeLibraryItemId = null) { | ||||
|     if (!seriesToCheck?.length) return | ||||
|   /** | ||||
|    * Used when a series is removed from a book | ||||
|    * Series is removed if it only has 1 book | ||||
|    *  | ||||
|    * @param {string} bookId | ||||
|    * @param {string[]} seriesIds  | ||||
|    */ | ||||
|   async checkRemoveEmptySeries(bookId, seriesIds) { | ||||
|     if (!seriesIds?.length) return | ||||
| 
 | ||||
|     for (const series of seriesToCheck) { | ||||
|       const otherLibraryItemsInSeries = Database.libraryItems.filter(li => li.id !== excludeLibraryItemId && li.isBook && li.media.metadata.hasSeries(series.id)) | ||||
|       if (!otherLibraryItemsInSeries.length) { | ||||
|         // Close open RSS feed for series
 | ||||
|         await this.rssFeedManager.closeFeedForEntityId(series.id) | ||||
|         Logger.info(`[ApiRouter] Series "${series.name}" is now empty. Removing series`) | ||||
|         await Database.removeSeries(series.id) | ||||
|         // TODO: Socket events for series?
 | ||||
|     const bookSeries = await Database.bookSeriesModel.findAll({ | ||||
|       where: { | ||||
|         bookId, | ||||
|         seriesId: seriesIds | ||||
|       }, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.seriesModel, | ||||
|           include: { | ||||
|             model: Database.bookModel | ||||
|           } | ||||
|         } | ||||
|       ] | ||||
|     }) | ||||
|     for (const bs of bookSeries) { | ||||
|       if (bs.series.books.length === 1) { | ||||
|         await this.removeEmptySeries(bs.series) | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Remove an empty series & close an open RSS feed | ||||
|    * @param {import('../models/Series')} series  | ||||
|    */ | ||||
|   async removeEmptySeries(series) { | ||||
|     await this.rssFeedManager.closeFeedForEntityId(series.id) | ||||
|     Logger.info(`[ApiRouter] Series "${series.name}" is now empty. Removing series`) | ||||
|     await Database.removeSeries(series.id) | ||||
|     // Remove series from library filter data
 | ||||
|     Database.removeSeriesFromFilterData(series.libraryId, series.id) | ||||
|     SocketAuthority.emitter('series_removed', { | ||||
|       id: series.id, | ||||
|       libraryId: series.libraryId | ||||
|     }) | ||||
|   } | ||||
| 
 | ||||
|   async getUserListeningSessionsHelper(userId) { | ||||
| @ -497,7 +495,7 @@ class ApiRouter { | ||||
|   async getAllSessionsWithUserData() { | ||||
|     const sessions = await Database.getPlaybackSessions() | ||||
|     sessions.sort((a, b) => b.updatedAt - a.updatedAt) | ||||
|     const minifiedUserObjects = await Database.models.user.getMinifiedUserObjects() | ||||
|     const minifiedUserObjects = await Database.userModel.getMinifiedUserObjects() | ||||
|     return sessions.map(se => { | ||||
|       return { | ||||
|         ...se, | ||||
| @ -557,7 +555,7 @@ class ApiRouter { | ||||
|       const mediaMetadata = mediaPayload.metadata | ||||
| 
 | ||||
|       // Create new authors if in payload
 | ||||
|       if (mediaMetadata.authors && mediaMetadata.authors.length) { | ||||
|       if (mediaMetadata.authors?.length) { | ||||
|         const newAuthors = [] | ||||
|         for (let i = 0; i < mediaMetadata.authors.length; i++) { | ||||
|           const authorName = (mediaMetadata.authors[i].name || '').trim() | ||||
| @ -566,6 +564,12 @@ class ApiRouter { | ||||
|             continue | ||||
|           } | ||||
| 
 | ||||
|           // Ensure the ID for the author exists
 | ||||
|           if (mediaMetadata.authors[i].id && !(await Database.checkAuthorExists(libraryId, mediaMetadata.authors[i].id))) { | ||||
|             Logger.warn(`[ApiRouter] Author id "${mediaMetadata.authors[i].id}" does not exist`) | ||||
|             mediaMetadata.authors[i].id = null | ||||
|           } | ||||
| 
 | ||||
|           if (!mediaMetadata.authors[i].id || mediaMetadata.authors[i].id.startsWith('new')) { | ||||
|             let author = Database.authors.find(au => au.libraryId === libraryId && au.checkNameEquals(authorName)) | ||||
|             if (!author) { | ||||
| @ -573,6 +577,8 @@ class ApiRouter { | ||||
|               author.setData(mediaMetadata.authors[i], libraryId) | ||||
|               Logger.debug(`[ApiRouter] Created new author "${author.name}"`) | ||||
|               newAuthors.push(author) | ||||
|               // Update filter data
 | ||||
|               Database.addAuthorToFilterData(libraryId, author.name, author.id) | ||||
|             } | ||||
| 
 | ||||
|             // Update ID in original payload
 | ||||
| @ -595,6 +601,12 @@ class ApiRouter { | ||||
|             continue | ||||
|           } | ||||
| 
 | ||||
|           // Ensure the ID for the series exists
 | ||||
|           if (mediaMetadata.series[i].id && !(await Database.checkSeriesExists(libraryId, mediaMetadata.series[i].id))) { | ||||
|             Logger.warn(`[ApiRouter] Series id "${mediaMetadata.series[i].id}" does not exist`) | ||||
|             mediaMetadata.series[i].id = null | ||||
|           } | ||||
| 
 | ||||
|           if (!mediaMetadata.series[i].id || mediaMetadata.series[i].id.startsWith('new')) { | ||||
|             let seriesItem = Database.series.find(se => se.libraryId === libraryId && se.checkNameEquals(seriesName)) | ||||
|             if (!seriesItem) { | ||||
| @ -602,6 +614,8 @@ class ApiRouter { | ||||
|               seriesItem.setData(mediaMetadata.series[i], libraryId) | ||||
|               Logger.debug(`[ApiRouter] Created new series "${seriesItem.name}"`) | ||||
|               newSeries.push(seriesItem) | ||||
|               // Update filter data
 | ||||
|               Database.addSeriesToFilterData(libraryId, seriesItem.name, seriesItem.id) | ||||
|             } | ||||
| 
 | ||||
|             // Update ID in original payload
 | ||||
|  | ||||
| @ -1,3 +1,4 @@ | ||||
| const Sequelize = require('sequelize') | ||||
| const fs = require('../libs/fsExtra') | ||||
| const Path = require('path') | ||||
| const Logger = require('../Logger') | ||||
| @ -66,7 +67,7 @@ class Scanner { | ||||
|   } | ||||
| 
 | ||||
|   async scanLibraryItemByRequest(libraryItem) { | ||||
|     const library = await Database.models.library.getOldById(libraryItem.libraryId) | ||||
|     const library = await Database.libraryModel.getOldById(libraryItem.libraryId) | ||||
|     if (!library) { | ||||
|       Logger.error(`[Scanner] Scan libraryItem by id library not found "${libraryItem.libraryId}"`) | ||||
|       return ScanResult.NOTHING | ||||
| @ -485,6 +486,8 @@ class Scanner { | ||||
|           _author = new Author() | ||||
|           _author.setData(tempMinAuthor, libraryItem.libraryId) | ||||
|           newAuthors.push(_author) | ||||
|           // Update filter data
 | ||||
|           Database.addAuthorToFilterData(libraryItem.libraryId, _author.name, _author.id) | ||||
|         } | ||||
| 
 | ||||
|         return { | ||||
| @ -501,11 +504,17 @@ class Scanner { | ||||
|       const newSeries = [] | ||||
|       libraryItem.media.metadata.series = libraryItem.media.metadata.series.map((tempMinSeries) => { | ||||
|         let _series = Database.series.find(se => se.libraryId === libraryItem.libraryId && se.checkNameEquals(tempMinSeries.name)) | ||||
|         if (!_series) _series = newSeries.find(se => se.libraryId === libraryItem.libraryId && se.checkNameEquals(tempMinSeries.name)) // Check new unsaved series
 | ||||
|         if (!_series) { | ||||
|           // Check new unsaved series
 | ||||
|           _series = newSeries.find(se => se.libraryId === libraryItem.libraryId && se.checkNameEquals(tempMinSeries.name)) | ||||
|         } | ||||
| 
 | ||||
|         if (!_series) { // Must create new series
 | ||||
|           _series = new Series() | ||||
|           _series.setData(tempMinSeries, libraryItem.libraryId) | ||||
|           newSeries.push(_series) | ||||
|           // Update filter data
 | ||||
|           Database.addSeriesToFilterData(libraryItem.libraryId, _series.name, _series.id) | ||||
|         } | ||||
|         return { | ||||
|           id: _series.id, | ||||
| @ -552,25 +561,30 @@ class Scanner { | ||||
| 
 | ||||
|     for (const folderId in folderGroups) { | ||||
|       const libraryId = folderGroups[folderId].libraryId | ||||
|       const library = await Database.models.library.getOldById(libraryId) | ||||
|       const library = await Database.libraryModel.getOldById(libraryId) | ||||
|       if (!library) { | ||||
|         Logger.error(`[Scanner] Library not found in files changed ${libraryId}`) | ||||
|         continue; | ||||
|         continue | ||||
|       } | ||||
|       const folder = library.getFolderById(folderId) | ||||
|       if (!folder) { | ||||
|         Logger.error(`[Scanner] Folder is not in library in files changed "${folderId}", Library "${library.name}"`) | ||||
|         continue; | ||||
|         continue | ||||
|       } | ||||
|       const relFilePaths = folderGroups[folderId].fileUpdates.map(fileUpdate => fileUpdate.relPath) | ||||
|       const fileUpdateGroup = groupFilesIntoLibraryItemPaths(library.mediaType, relFilePaths, false) | ||||
| 
 | ||||
|       if (!Object.keys(fileUpdateGroup).length) { | ||||
|         Logger.info(`[Scanner] No important changes to scan for in folder "${folderId}"`) | ||||
|         continue; | ||||
|         continue | ||||
|       } | ||||
|       const folderScanResults = await this.scanFolderUpdates(library, folder, fileUpdateGroup) | ||||
|       Logger.debug(`[Scanner] Folder scan results`, folderScanResults) | ||||
| 
 | ||||
|       // If something was updated then reset numIssues filter data for library
 | ||||
|       if (Object.values(folderScanResults).some(scanResult => scanResult !== ScanResult.NOTHING && scanResult !== ScanResult.UPTODATE)) { | ||||
|         await Database.resetLibraryIssuesFilterData(libraryId) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     this.scanningFilesChanged = false | ||||
| @ -589,28 +603,49 @@ class Scanner { | ||||
|     //    Test Case: Moving audio files from library item folder to author folder should trigger a re-scan of the item
 | ||||
|     const updateGroup = { ...fileUpdateGroup } | ||||
|     for (const itemDir in updateGroup) { | ||||
|       if (itemDir == fileUpdateGroup[itemDir]) continue; // Media in root path
 | ||||
|       if (itemDir == fileUpdateGroup[itemDir]) continue // Media in root path
 | ||||
| 
 | ||||
|       const itemDirNestedFiles = fileUpdateGroup[itemDir].filter(b => b.includes('/')) | ||||
|       if (!itemDirNestedFiles.length) continue; | ||||
|       if (!itemDirNestedFiles.length) continue | ||||
| 
 | ||||
|       const firstNest = itemDirNestedFiles[0].split('/').shift() | ||||
|       const altDir = `${itemDir}/${firstNest}` | ||||
| 
 | ||||
|       const fullPath = Path.posix.join(filePathToPOSIX(folder.fullPath), itemDir) | ||||
|       const childLibraryItem = Database.libraryItems.find(li => li.path !== fullPath && li.path.startsWith(fullPath)) | ||||
|       const childLibraryItem = await Database.libraryItemModel.findOne({ | ||||
|         attributes: ['id', 'path'], | ||||
|         where: { | ||||
|           path: { | ||||
|             [Sequelize.Op.not]: fullPath | ||||
|           }, | ||||
|           path: { | ||||
|             [Sequelize.Op.startsWith]: fullPath | ||||
|           } | ||||
|         } | ||||
|       }) | ||||
|       if (!childLibraryItem) { | ||||
|         continue | ||||
|       } | ||||
| 
 | ||||
|       const altFullPath = Path.posix.join(filePathToPOSIX(folder.fullPath), altDir) | ||||
|       const altChildLibraryItem = Database.libraryItems.find(li => li.path !== altFullPath && li.path.startsWith(altFullPath)) | ||||
|       const altChildLibraryItem = await Database.libraryItemModel.findOne({ | ||||
|         attributes: ['id', 'path'], | ||||
|         where: { | ||||
|           path: { | ||||
|             [Sequelize.Op.not]: altFullPath | ||||
|           }, | ||||
|           path: { | ||||
|             [Sequelize.Op.startsWith]: altFullPath | ||||
|           } | ||||
|         } | ||||
|       }) | ||||
|       if (altChildLibraryItem) { | ||||
|         continue | ||||
|       } | ||||
| 
 | ||||
|       delete fileUpdateGroup[itemDir] | ||||
|       fileUpdateGroup[altDir] = itemDirNestedFiles.map((f) => f.split('/').slice(1).join('/')) | ||||
|       Logger.warn(`[Scanner] Some files were modified in a parent directory of a library item "${childLibraryItem.title}" - ignoring`) | ||||
|       Logger.warn(`[Scanner] Some files were modified in a parent directory of a library item "${childLibraryItem.path}" - ignoring`) | ||||
|     } | ||||
| 
 | ||||
|     // Second pass: Check for new/updated/removed items
 | ||||
| @ -619,10 +654,21 @@ class Scanner { | ||||
|       const fullPath = Path.posix.join(filePathToPOSIX(folder.fullPath), itemDir) | ||||
|       const dirIno = await getIno(fullPath) | ||||
| 
 | ||||
|       const itemDirParts = itemDir.split('/').slice(0, -1) | ||||
|       const potentialChildDirs = [] | ||||
|       for (let i = 0; i < itemDirParts.length; i++) { | ||||
|         potentialChildDirs.push(Path.posix.join(filePathToPOSIX(folder.fullPath), itemDir.split('/').slice(0, -1 - i).join('/'))) | ||||
|       } | ||||
| 
 | ||||
|       // Check if book dir group is already an item
 | ||||
|       let existingLibraryItem = Database.libraryItems.find(li => fullPath.startsWith(li.path)) | ||||
|       let existingLibraryItem = await Database.libraryItemModel.findOneOld({ | ||||
|         path: potentialChildDirs | ||||
|       }) | ||||
| 
 | ||||
|       if (!existingLibraryItem) { | ||||
|         existingLibraryItem = Database.libraryItems.find(li => li.ino === dirIno) | ||||
|         existingLibraryItem = await Database.libraryItemModel.findOneOld({ | ||||
|           ino: dirIno | ||||
|         }) | ||||
|         if (existingLibraryItem) { | ||||
|           Logger.debug(`[Scanner] scanFolderUpdates: Library item found by inode value=${dirIno}. "${existingLibraryItem.relPath} => ${itemDir}"`) | ||||
|           // Update library item paths for scan and all library item paths will get updated in LibraryItem.checkScanData
 | ||||
| @ -655,9 +701,16 @@ class Scanner { | ||||
|       } | ||||
| 
 | ||||
|       // Check if a library item is a subdirectory of this dir
 | ||||
|       var childItem = Database.libraryItems.find(li => (li.path + '/').startsWith(fullPath + '/')) | ||||
|       const childItem = await Database.libraryItemModel.findOne({ | ||||
|         attributes: ['id', 'path'], | ||||
|         where: { | ||||
|           path: { | ||||
|             [Sequelize.Op.startsWith]: fullPath + '/' | ||||
|           } | ||||
|         } | ||||
|       }) | ||||
|       if (childItem) { | ||||
|         Logger.warn(`[Scanner] Files were modified in a parent directory of a library item "${childItem.media.metadata.title}" - ignoring`) | ||||
|         Logger.warn(`[Scanner] Files were modified in a parent directory of a library item "${childItem.path}" - ignoring`) | ||||
|         itemGroupingResults[itemDir] = ScanResult.NOTHING | ||||
|         continue | ||||
|       } | ||||
| @ -884,6 +937,8 @@ class Scanner { | ||||
|           author.setData({ name: authorName }, libraryItem.libraryId) | ||||
|           await Database.createAuthor(author) | ||||
|           SocketAuthority.emitter('author_added', author.toJSON()) | ||||
|           // Update filter data
 | ||||
|           Database.addAuthorToFilterData(libraryItem.libraryId, author.name, author.id) | ||||
|         } | ||||
|         authorPayload.push(author.toJSONMinimal()) | ||||
|       } | ||||
| @ -900,6 +955,8 @@ class Scanner { | ||||
|           seriesItem = new Series() | ||||
|           seriesItem.setData({ name: seriesMatchItem.series }, libraryItem.libraryId) | ||||
|           await Database.createSeries(seriesItem) | ||||
|           // Update filter data
 | ||||
|           Database.addSeriesToFilterData(libraryItem.libraryId, seriesItem.name, seriesItem.id) | ||||
|           SocketAuthority.emitter('series_added', seriesItem.toJSON()) | ||||
|         } | ||||
|         seriesPayload.push(seriesItem.toJSONMinimal(seriesMatchItem.sequence)) | ||||
|  | ||||
| @ -1,23 +1,29 @@ | ||||
| const xml = require('../../libs/xml') | ||||
| 
 | ||||
| module.exports.generate = (libraryItems, indent = true) => { | ||||
| /** | ||||
|  * Generate OPML file string for podcasts in a library | ||||
|  * @param {import('../../models/Podcast')[]} podcasts  | ||||
|  * @param {boolean} [indent=true]  | ||||
|  * @returns {string} | ||||
|  */ | ||||
| module.exports.generate = (podcasts, indent = true) => { | ||||
|   const bodyItems = [] | ||||
|   libraryItems.forEach((item) => { | ||||
|     if (!item.media.metadata.feedUrl) return | ||||
|   podcasts.forEach((podcast) => { | ||||
|     if (!podcast.feedURL) return | ||||
|     const feedAttributes = { | ||||
|       type: 'rss', | ||||
|       text: item.media.metadata.title, | ||||
|       title: item.media.metadata.title, | ||||
|       xmlUrl: item.media.metadata.feedUrl | ||||
|       text: podcast.title, | ||||
|       title: podcast.title, | ||||
|       xmlUrl: podcast.feedURL | ||||
|     } | ||||
|     if (item.media.metadata.description) { | ||||
|       feedAttributes.description = item.media.metadata.description | ||||
|     if (podcast.description) { | ||||
|       feedAttributes.description = podcast.description | ||||
|     } | ||||
|     if (item.media.metadata.itunesPageUrl) { | ||||
|       feedAttributes.htmlUrl = item.media.metadata.itunesPageUrl | ||||
|     if (podcast.itunesPageUrl) { | ||||
|       feedAttributes.htmlUrl = podcast.itunesPageUrl | ||||
|     } | ||||
|     if (item.media.metadata.language) { | ||||
|       feedAttributes.language = item.media.metadata.language | ||||
|     if (podcast.language) { | ||||
|       feedAttributes.language = podcast.language | ||||
|     } | ||||
|     bodyItems.push({ | ||||
|       outline: { | ||||
|  | ||||
| @ -1,5 +1,4 @@ | ||||
| const { sort, createNewSortInstance } = require('../libs/fastSort') | ||||
| const Logger = require('../Logger') | ||||
| const { createNewSortInstance } = require('../libs/fastSort') | ||||
| const Database = require('../Database') | ||||
| const { getTitlePrefixAtEnd, isNullOrNaN, getTitleIgnorePrefix } = require('../utils/index') | ||||
| const naturalSort = createNewSortInstance({ | ||||
| @ -72,7 +71,7 @@ module.exports = { | ||||
|     } else if (filterBy === 'issues') { | ||||
|       filtered = filtered.filter(li => li.hasIssues) | ||||
|     } else if (filterBy === 'feed-open') { | ||||
|       const libraryItemIdsWithFeed = await Database.models.feed.findAllLibraryItemIds() | ||||
|       const libraryItemIdsWithFeed = await Database.feedModel.findAllLibraryItemIds() | ||||
|       filtered = filtered.filter(li => libraryItemIdsWithFeed.includes(li.id)) | ||||
|     } else if (filterBy === 'abridged') { | ||||
|       filtered = filtered.filter(li => !!li.media.metadata?.abridged) | ||||
| @ -126,60 +125,6 @@ module.exports = { | ||||
|     return true | ||||
|   }, | ||||
| 
 | ||||
|   getDistinctFilterDataNew(libraryItems) { | ||||
|     const data = { | ||||
|       authors: [], | ||||
|       genres: [], | ||||
|       tags: [], | ||||
|       series: [], | ||||
|       narrators: [], | ||||
|       languages: [], | ||||
|       publishers: [] | ||||
|     } | ||||
|     libraryItems.forEach((li) => { | ||||
|       const mediaMetadata = li.media.metadata | ||||
|       if (mediaMetadata.authors?.length) { | ||||
|         mediaMetadata.authors.forEach((author) => { | ||||
|           if (author && !data.authors.some(au => au.id === author.id)) data.authors.push({ id: author.id, name: author.name }) | ||||
|         }) | ||||
|       } | ||||
|       if (mediaMetadata.series?.length) { | ||||
|         mediaMetadata.series.forEach((series) => { | ||||
|           if (series && !data.series.some(se => se.id === series.id)) data.series.push({ id: series.id, name: series.name }) | ||||
|         }) | ||||
|       } | ||||
|       if (mediaMetadata.genres?.length) { | ||||
|         mediaMetadata.genres.forEach((genre) => { | ||||
|           if (genre && !data.genres.includes(genre)) data.genres.push(genre) | ||||
|         }) | ||||
|       } | ||||
|       if (li.media.tags.length) { | ||||
|         li.media.tags.forEach((tag) => { | ||||
|           if (tag && !data.tags.includes(tag)) data.tags.push(tag) | ||||
|         }) | ||||
|       } | ||||
|       if (mediaMetadata.narrators?.length) { | ||||
|         mediaMetadata.narrators.forEach((narrator) => { | ||||
|           if (narrator && !data.narrators.includes(narrator)) data.narrators.push(narrator) | ||||
|         }) | ||||
|       } | ||||
|       if (mediaMetadata.publisher && !data.publishers.includes(mediaMetadata.publisher)) { | ||||
|         data.publishers.push(mediaMetadata.publisher) | ||||
|       } | ||||
|       if (mediaMetadata.language && !data.languages.includes(mediaMetadata.language)) { | ||||
|         data.languages.push(mediaMetadata.language) | ||||
|       } | ||||
|     }) | ||||
|     data.authors = naturalSort(data.authors).asc(au => au.name) | ||||
|     data.genres = naturalSort(data.genres).asc() | ||||
|     data.tags = naturalSort(data.tags).asc() | ||||
|     data.series = naturalSort(data.series).asc(se => se.name) | ||||
|     data.narrators = naturalSort(data.narrators).asc() | ||||
|     data.publishers = naturalSort(data.publishers).asc() | ||||
|     data.languages = naturalSort(data.languages).asc() | ||||
|     return data | ||||
|   }, | ||||
| 
 | ||||
|   getSeriesFromBooks(books, allSeries, filterSeries, filterBy, user, minified, hideSingleBookSeries) { | ||||
|     const _series = {} | ||||
|     const seriesToFilterOut = {} | ||||
| @ -246,89 +191,6 @@ module.exports = { | ||||
|     }) | ||||
|   }, | ||||
| 
 | ||||
|   getBooksNextInSeries(seriesWithUserAb, limit, minified = false) { | ||||
|     var incompleteSeires = seriesWithUserAb.filter((series) => series.books.some((book) => !book.userAudiobook || (!book.userAudiobook.isRead && book.userAudiobook.progress == 0))) | ||||
|     var booksNextInSeries = [] | ||||
|     incompleteSeires.forEach((series) => { | ||||
|       var dateLastRead = series.books.filter((data) => data.userAudiobook && data.userAudiobook.isRead).sort((a, b) => { return b.userAudiobook.finishedAt - a.userAudiobook.finishedAt })[0].userAudiobook.finishedAt | ||||
|       var nextUnreadBook = series.books.filter((data) => !data.userAudiobook || (!data.userAudiobook.isRead && data.userAudiobook.progress == 0))[0] | ||||
|       nextUnreadBook.DateLastReadSeries = dateLastRead | ||||
|       booksNextInSeries.push(nextUnreadBook) | ||||
|     }) | ||||
|     return booksNextInSeries.sort((a, b) => { return b.DateLastReadSeries - a.DateLastReadSeries }).map(b => minified ? b.book.toJSONMinified() : b.book.toJSONExpanded()).slice(0, limit) | ||||
|   }, | ||||
| 
 | ||||
|   getGenresWithCount(libraryItems) { | ||||
|     var genresMap = {} | ||||
|     libraryItems.forEach((li) => { | ||||
|       var genres = li.media.metadata.genres || [] | ||||
|       genres.forEach((genre) => { | ||||
|         if (genresMap[genre]) genresMap[genre].count++ | ||||
|         else | ||||
|           genresMap[genre] = { | ||||
|             genre, | ||||
|             count: 1 | ||||
|           } | ||||
|       }) | ||||
|     }) | ||||
|     return Object.values(genresMap).sort((a, b) => b.count - a.count) | ||||
|   }, | ||||
| 
 | ||||
|   getAuthorsWithCount(libraryItems) { | ||||
|     var authorsMap = {} | ||||
|     libraryItems.forEach((li) => { | ||||
|       var authors = li.media.metadata.authors || [] | ||||
|       authors.forEach((author) => { | ||||
|         if (authorsMap[author.id]) authorsMap[author.id].count++ | ||||
|         else | ||||
|           authorsMap[author.id] = { | ||||
|             id: author.id, | ||||
|             name: author.name, | ||||
|             count: 1 | ||||
|           } | ||||
|       }) | ||||
|     }) | ||||
|     return Object.values(authorsMap).sort((a, b) => b.count - a.count) | ||||
|   }, | ||||
| 
 | ||||
|   getItemDurationStats(libraryItems) { | ||||
|     var sorted = sort(libraryItems).desc(li => li.media.duration) | ||||
|     var top10 = sorted.slice(0, 10).map(li => ({ id: li.id, title: li.media.metadata.title, duration: li.media.duration })).filter(i => i.duration > 0) | ||||
|     var totalDuration = 0 | ||||
|     var numAudioTracks = 0 | ||||
|     libraryItems.forEach((li) => { | ||||
|       totalDuration += li.media.duration | ||||
|       numAudioTracks += li.media.numTracks | ||||
|     }) | ||||
|     return { | ||||
|       totalDuration, | ||||
|       numAudioTracks, | ||||
|       longestItems: top10 | ||||
|     } | ||||
|   }, | ||||
| 
 | ||||
|   getItemSizeStats(libraryItems) { | ||||
|     var sorted = sort(libraryItems).desc(li => li.media.size) | ||||
|     var top10 = sorted.slice(0, 10).map(li => ({ id: li.id, title: li.media.metadata.title, size: li.media.size })).filter(i => i.size > 0) | ||||
|     var totalSize = 0 | ||||
|     libraryItems.forEach((li) => { | ||||
|       totalSize += li.media.size | ||||
|     }) | ||||
|     return { | ||||
|       totalSize, | ||||
|       largestItems: top10 | ||||
|     } | ||||
|   }, | ||||
| 
 | ||||
|   getLibraryItemsTotalSize(libraryItems) { | ||||
|     var totalSize = 0 | ||||
|     libraryItems.forEach((li) => { | ||||
|       totalSize += li.media.size | ||||
|     }) | ||||
|     return totalSize | ||||
|   }, | ||||
| 
 | ||||
| 
 | ||||
|   collapseBookSeries(libraryItems, series, filterSeries, hideSingleBookSeries) { | ||||
|     // Get series from the library items. If this list is being collapsed after filtering for a series,
 | ||||
|     // don't collapse that series, only books that are in other series.
 | ||||
| @ -356,550 +218,5 @@ module.exports = { | ||||
|     }) | ||||
| 
 | ||||
|     return filteredLibraryItems | ||||
|   }, | ||||
| 
 | ||||
|   async buildPersonalizedShelves(ctx, user, libraryItems, library, maxEntitiesPerShelf, include) { | ||||
|     const mediaType = library.mediaType | ||||
|     const isPodcastLibrary = mediaType === 'podcast' | ||||
|     const includeRssFeed = include.includes('rssfeed') | ||||
|     const includeNumEpisodesIncomplete = include.includes('numepisodesincomplete') // Podcasts only
 | ||||
|     const hideSingleBookSeries = library.settings.hideSingleBookSeries | ||||
| 
 | ||||
|     const shelves = [ | ||||
|       { | ||||
|         id: 'continue-listening', | ||||
|         label: 'Continue Listening', | ||||
|         labelStringKey: 'LabelContinueListening', | ||||
|         type: isPodcastLibrary ? 'episode' : mediaType, | ||||
|         entities: [] | ||||
|       }, | ||||
|       { | ||||
|         id: 'continue-reading', | ||||
|         label: 'Continue Reading', | ||||
|         labelStringKey: 'LabelContinueReading', | ||||
|         type: 'book', | ||||
|         entities: [] | ||||
|       }, | ||||
|       { | ||||
|         id: 'continue-series', | ||||
|         label: 'Continue Series', | ||||
|         labelStringKey: 'LabelContinueSeries', | ||||
|         type: mediaType, | ||||
|         entities: [] | ||||
|       }, | ||||
|       { | ||||
|         id: 'episodes-recently-added', | ||||
|         label: 'Newest Episodes', | ||||
|         labelStringKey: 'LabelNewestEpisodes', | ||||
|         type: 'episode', | ||||
|         entities: [] | ||||
|       }, | ||||
|       { | ||||
|         id: 'recently-added', | ||||
|         label: 'Recently Added', | ||||
|         labelStringKey: 'LabelRecentlyAdded', | ||||
|         type: mediaType, | ||||
|         entities: [] | ||||
|       }, | ||||
|       { | ||||
|         id: 'recent-series', | ||||
|         label: 'Recent Series', | ||||
|         labelStringKey: 'LabelRecentSeries', | ||||
|         type: 'series', | ||||
|         entities: [] | ||||
|       }, | ||||
|       { | ||||
|         id: 'recommended', | ||||
|         label: 'Recommended', | ||||
|         labelStringKey: 'LabelRecommended', | ||||
|         type: mediaType, | ||||
|         entities: [] | ||||
|       }, | ||||
|       { | ||||
|         id: 'listen-again', | ||||
|         label: 'Listen Again', | ||||
|         labelStringKey: 'LabelListenAgain', | ||||
|         type: isPodcastLibrary ? 'episode' : mediaType, | ||||
|         entities: [] | ||||
|       }, | ||||
|       { | ||||
|         id: 'read-again', | ||||
|         label: 'Read Again', | ||||
|         labelStringKey: 'LabelReadAgain', | ||||
|         type: 'book', | ||||
|         entities: [] | ||||
|       }, | ||||
|       { | ||||
|         id: 'newest-authors', | ||||
|         label: 'Newest Authors', | ||||
|         labelStringKey: 'LabelNewestAuthors', | ||||
|         type: 'authors', | ||||
|         entities: [] | ||||
|       } | ||||
|     ] | ||||
| 
 | ||||
|     const categoryMap = {} | ||||
|     shelves.forEach((shelf) => { | ||||
|       categoryMap[shelf.id] = { | ||||
|         id: shelf.id, | ||||
|         biggest: 0, | ||||
|         smallest: 0, | ||||
|         items: [] | ||||
|       } | ||||
|     }) | ||||
| 
 | ||||
|     const seriesMap = {} | ||||
|     const authorMap = {} | ||||
| 
 | ||||
|     // For use with recommended
 | ||||
|     const topGenresListened = {} | ||||
|     const topAuthorsListened = {} | ||||
|     const topTagsListened = {} | ||||
|     const notStartedBooks = [] | ||||
| 
 | ||||
|     for (const libraryItem of libraryItems) { | ||||
|       if (libraryItem.addedAt > categoryMap['recently-added'].smallest) { | ||||
|         const libraryItemObj = libraryItem.toJSONMinified() | ||||
| 
 | ||||
|         // add numEpisodesIncomplete if "include=numEpisodesIncomplete" was put in query string (only for podcasts)
 | ||||
|         if (includeNumEpisodesIncomplete && libraryItem.isPodcast) { | ||||
|           libraryItemObj.numEpisodesIncomplete = user.getNumEpisodesIncompleteForPodcast(libraryItem) | ||||
|         } | ||||
| 
 | ||||
|         const indexToPut = categoryMap['recently-added'].items.findIndex(i => libraryItem.addedAt > i.addedAt) | ||||
|         if (indexToPut >= 0) { | ||||
|           categoryMap['recently-added'].items.splice(indexToPut, 0, libraryItemObj) | ||||
|         } else { | ||||
|           categoryMap['recently-added'].items.push(libraryItemObj) | ||||
|         } | ||||
| 
 | ||||
|         if (categoryMap['recently-added'].items.length > maxEntitiesPerShelf) { | ||||
|           // Remove last item
 | ||||
|           categoryMap['recently-added'].items.pop() | ||||
|           categoryMap['recently-added'].smallest = categoryMap['recently-added'].items[categoryMap['recently-added'].items.length - 1].addedAt | ||||
|         } | ||||
|         categoryMap['recently-added'].biggest = categoryMap['recently-added'].items[0].addedAt | ||||
|       } | ||||
| 
 | ||||
|       const allItemProgress = user.getAllMediaProgressForLibraryItem(libraryItem.id) | ||||
|       if (libraryItem.isPodcast) { | ||||
|         // Podcast categories
 | ||||
|         const podcastEpisodes = libraryItem.media.episodes || [] | ||||
|         for (const episode of podcastEpisodes) { | ||||
|           const mediaProgress = allItemProgress.find(mp => mp.episodeId === episode.id) | ||||
| 
 | ||||
|           // Newest episodes
 | ||||
|           if (!mediaProgress?.isFinished && episode.addedAt > categoryMap['episodes-recently-added'].smallest) { | ||||
|             const libraryItemWithEpisode = { | ||||
|               ...libraryItem.toJSONMinified(), | ||||
|               recentEpisode: episode.toJSON() | ||||
|             } | ||||
| 
 | ||||
|             const indexToPut = categoryMap['episodes-recently-added'].items.findIndex(i => episode.addedAt > i.recentEpisode.addedAt) | ||||
|             if (indexToPut >= 0) { | ||||
|               categoryMap['episodes-recently-added'].items.splice(indexToPut, 0, libraryItemWithEpisode) | ||||
|             } else { | ||||
|               categoryMap['episodes-recently-added'].items.push(libraryItemWithEpisode) | ||||
|             } | ||||
| 
 | ||||
|             if (categoryMap['episodes-recently-added'].items.length > maxEntitiesPerShelf) { | ||||
|               // Remove last item
 | ||||
|               categoryMap['episodes-recently-added'].items.pop() | ||||
|               categoryMap['episodes-recently-added'].smallest = categoryMap['episodes-recently-added'].items[categoryMap['episodes-recently-added'].items.length - 1].recentEpisode.addedAt | ||||
|             } | ||||
|             categoryMap['episodes-recently-added'].biggest = categoryMap['episodes-recently-added'].items[0].recentEpisode.addedAt | ||||
|           } | ||||
| 
 | ||||
|           // Episode recently listened and finished
 | ||||
|           if (mediaProgress) { | ||||
|             if (mediaProgress.isFinished) { | ||||
|               if (mediaProgress.finishedAt > categoryMap['listen-again'].smallest) { // Item belongs on shelf
 | ||||
|                 const libraryItemWithEpisode = { | ||||
|                   ...libraryItem.toJSONMinified(), | ||||
|                   recentEpisode: episode.toJSON(), | ||||
|                   finishedAt: mediaProgress.finishedAt | ||||
|                 } | ||||
| 
 | ||||
|                 const indexToPut = categoryMap['listen-again'].items.findIndex(i => mediaProgress.finishedAt > i.finishedAt) | ||||
|                 if (indexToPut >= 0) { | ||||
|                   categoryMap['listen-again'].items.splice(indexToPut, 0, libraryItemWithEpisode) | ||||
|                 } else { | ||||
|                   categoryMap['listen-again'].items.push(libraryItemWithEpisode) | ||||
|                 } | ||||
| 
 | ||||
|                 if (categoryMap['listen-again'].items.length > maxEntitiesPerShelf) { | ||||
|                   // Remove last item
 | ||||
|                   categoryMap['listen-again'].items.pop() | ||||
|                   categoryMap['listen-again'].smallest = categoryMap['listen-again'].items[categoryMap['listen-again'].items.length - 1].finishedAt | ||||
|                 } | ||||
|                 categoryMap['listen-again'].biggest = categoryMap['listen-again'].items[0].finishedAt | ||||
|               } | ||||
|             } else if (mediaProgress.inProgress && !mediaProgress.hideFromContinueListening) { // Handle most recently listened
 | ||||
|               if (mediaProgress.lastUpdate > categoryMap['continue-listening'].smallest) { // Item belongs on shelf
 | ||||
|                 const libraryItemWithEpisode = { | ||||
|                   ...libraryItem.toJSONMinified(), | ||||
|                   recentEpisode: episode.toJSON(), | ||||
|                   progressLastUpdate: mediaProgress.lastUpdate | ||||
|                 } | ||||
| 
 | ||||
|                 const indexToPut = categoryMap['continue-listening'].items.findIndex(i => mediaProgress.lastUpdate > i.progressLastUpdate) | ||||
|                 if (indexToPut >= 0) { | ||||
|                   categoryMap['continue-listening'].items.splice(indexToPut, 0, libraryItemWithEpisode) | ||||
|                 } else { | ||||
|                   categoryMap['continue-listening'].items.push(libraryItemWithEpisode) | ||||
|                 } | ||||
| 
 | ||||
|                 if (categoryMap['continue-listening'].items.length > maxEntitiesPerShelf) { | ||||
|                   // Remove last item
 | ||||
|                   categoryMap['continue-listening'].items.pop() | ||||
|                   categoryMap['continue-listening'].smallest = categoryMap['continue-listening'].items[categoryMap['continue-listening'].items.length - 1].progressLastUpdate | ||||
|                 } | ||||
| 
 | ||||
|                 categoryMap['continue-listening'].biggest = categoryMap['continue-listening'].items[0].progressLastUpdate | ||||
|               } | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|       } else if (libraryItem.isBook) { | ||||
|         // Book categories
 | ||||
| 
 | ||||
|         const mediaProgress = allItemProgress.length ? allItemProgress[0] : null | ||||
| 
 | ||||
|         // Used for recommended. Tally up most listened to authors/genres/tags
 | ||||
|         if (mediaProgress && (mediaProgress.inProgress || mediaProgress.isFinished)) { | ||||
|           libraryItem.media.metadata.authors.forEach((author) => { | ||||
|             topAuthorsListened[author.id] = (topAuthorsListened[author.id] || 0) + 1 | ||||
|           }) | ||||
|           libraryItem.media.metadata.genres.forEach((genre) => { | ||||
|             topGenresListened[genre] = (topGenresListened[genre] || 0) + 1 | ||||
|           }) | ||||
|           libraryItem.media.tags.forEach((tag) => { | ||||
|             topTagsListened[tag] = (topTagsListened[tag] || 0) + 1 | ||||
|           }) | ||||
|         } else { | ||||
|           // Insert in random position to add randomization to equal weighted items
 | ||||
|           notStartedBooks.splice(Math.floor(Math.random() * (notStartedBooks.length + 1)), 0, libraryItem) | ||||
|         } | ||||
| 
 | ||||
|         // Newest series
 | ||||
|         if (libraryItem.media.metadata.series.length) { | ||||
|           for (const librarySeries of libraryItem.media.metadata.series) { | ||||
| 
 | ||||
|             const bookInProgress = mediaProgress && (mediaProgress.inProgress || mediaProgress.isFinished) | ||||
|             const bookActive = mediaProgress && mediaProgress.inProgress && !mediaProgress.isFinished | ||||
|             const libraryItemJson = libraryItem.toJSONMinified() | ||||
|             libraryItemJson.seriesSequence = librarySeries.sequence | ||||
| 
 | ||||
|             const hideFromContinueListening = user.checkShouldHideSeriesFromContinueListening(librarySeries.id) | ||||
| 
 | ||||
|             if (!seriesMap[librarySeries.id]) { | ||||
|               const seriesObj = Database.series.find(se => se.id === librarySeries.id) | ||||
|               if (seriesObj) { | ||||
|                 const series = { | ||||
|                   ...seriesObj.toJSON(), | ||||
|                   books: [libraryItemJson], | ||||
|                   inProgress: bookInProgress, | ||||
|                   hasActiveBook: bookActive, | ||||
|                   hideFromContinueListening, | ||||
|                   bookInProgressLastUpdate: bookInProgress ? mediaProgress.lastUpdate : null, | ||||
|                   firstBookUnread: bookInProgress ? null : libraryItemJson | ||||
|                 } | ||||
|                 seriesMap[librarySeries.id] = series | ||||
| 
 | ||||
|                 const indexToPut = categoryMap['recent-series'].items.findIndex(i => series.addedAt > i.addedAt) | ||||
|                 if (indexToPut >= 0) { | ||||
|                   categoryMap['recent-series'].items.splice(indexToPut, 0, series) | ||||
|                 } else { | ||||
|                   categoryMap['recent-series'].items.push(series) | ||||
|                 } | ||||
|               } | ||||
|             } else { | ||||
|               // series already in map - add book
 | ||||
|               seriesMap[librarySeries.id].books.push(libraryItemJson) | ||||
| 
 | ||||
|               if (bookInProgress) { // Update if this series is in progress
 | ||||
|                 seriesMap[librarySeries.id].inProgress = true | ||||
| 
 | ||||
|                 if (seriesMap[librarySeries.id].bookInProgressLastUpdate < mediaProgress.lastUpdate) { | ||||
|                   seriesMap[librarySeries.id].bookInProgressLastUpdate = mediaProgress.lastUpdate | ||||
|                 } | ||||
|               } else if (!seriesMap[librarySeries.id].firstBookUnread) { | ||||
|                 seriesMap[librarySeries.id].firstBookUnread = libraryItemJson | ||||
|               } else if (libraryItemJson.seriesSequence) { | ||||
|                 // If current firstBookUnread has a series sequence greater than this series sequence, then update firstBookUnread
 | ||||
|                 const firstBookUnreadSequence = seriesMap[librarySeries.id].firstBookUnread.seriesSequence | ||||
|                 if (!firstBookUnreadSequence || String(firstBookUnreadSequence).localeCompare(String(librarySeries.sequence), undefined, { sensitivity: 'base', numeric: true }) > 0) { | ||||
|                   seriesMap[librarySeries.id].firstBookUnread = libraryItemJson | ||||
|                 } | ||||
|               } | ||||
| 
 | ||||
|               // Update if series has an active (progress < 100%) book
 | ||||
|               if (bookActive) { | ||||
|                 seriesMap[librarySeries.id].hasActiveBook = true | ||||
|               } | ||||
|             } | ||||
|           } | ||||
|         } | ||||
| 
 | ||||
|         // Newest authors
 | ||||
|         if (libraryItem.media.metadata.authors.length) { | ||||
|           for (const libraryAuthor of libraryItem.media.metadata.authors) { | ||||
|             if (!authorMap[libraryAuthor.id]) { | ||||
|               const authorObj = Database.authors.find(au => au.id === libraryAuthor.id) | ||||
|               if (authorObj) { | ||||
|                 const author = { | ||||
|                   ...authorObj.toJSON(), | ||||
|                   numBooks: 1 | ||||
|                 } | ||||
| 
 | ||||
|                 if (author.addedAt > categoryMap['newest-authors'].smallest) { | ||||
| 
 | ||||
|                   const indexToPut = categoryMap['newest-authors'].items.findIndex(i => author.addedAt > i.addedAt) | ||||
|                   if (indexToPut >= 0) { | ||||
|                     categoryMap['newest-authors'].items.splice(indexToPut, 0, author) | ||||
|                   } else { | ||||
|                     categoryMap['newest-authors'].items.push(author) | ||||
|                   } | ||||
| 
 | ||||
|                   // Max authors is 10
 | ||||
|                   if (categoryMap['newest-authors'].items.length > 10) { | ||||
|                     categoryMap['newest-authors'].items.pop() | ||||
|                     categoryMap['newest-authors'].smallest = categoryMap['newest-authors'].items[categoryMap['newest-authors'].items.length - 1].addedAt | ||||
|                   } | ||||
| 
 | ||||
|                   categoryMap['newest-authors'].biggest = categoryMap['newest-authors'].items[0].addedAt | ||||
|                 } | ||||
| 
 | ||||
|                 authorMap[libraryAuthor.id] = author | ||||
|               } | ||||
|             } else { | ||||
|               authorMap[libraryAuthor.id].numBooks++ | ||||
|             } | ||||
|           } | ||||
|         } | ||||
| 
 | ||||
|         // Book listening and finished
 | ||||
|         if (mediaProgress) { | ||||
|           const categoryId = libraryItem.media.isEBookOnly ? 'read-again' : 'listen-again' | ||||
| 
 | ||||
|           // Handle most recently finished
 | ||||
|           if (mediaProgress.isFinished) { | ||||
|             if (mediaProgress.finishedAt > categoryMap[categoryId].smallest) { // Item belongs on shelf
 | ||||
|               const libraryItemObj = { | ||||
|                 ...libraryItem.toJSONMinified(), | ||||
|                 finishedAt: mediaProgress.finishedAt | ||||
|               } | ||||
| 
 | ||||
|               const indexToPut = categoryMap[categoryId].items.findIndex(i => mediaProgress.finishedAt > i.finishedAt) | ||||
|               if (indexToPut >= 0) { | ||||
|                 categoryMap[categoryId].items.splice(indexToPut, 0, libraryItemObj) | ||||
|               } else { | ||||
|                 categoryMap[categoryId].items.push(libraryItemObj) | ||||
|               } | ||||
|               if (categoryMap[categoryId].items.length > maxEntitiesPerShelf) { | ||||
|                 // Remove last item
 | ||||
|                 categoryMap[categoryId].items.pop() | ||||
|                 categoryMap[categoryId].smallest = categoryMap[categoryId].items[categoryMap[categoryId].items.length - 1].finishedAt | ||||
|               } | ||||
|               categoryMap[categoryId].biggest = categoryMap[categoryId].items[0].finishedAt | ||||
|             } | ||||
|           } else if (mediaProgress.inProgress && !mediaProgress.hideFromContinueListening) { // Handle most recently listened
 | ||||
|             const categoryId = libraryItem.media.isEBookOnly ? 'continue-reading' : 'continue-listening' | ||||
| 
 | ||||
|             if (mediaProgress.lastUpdate > categoryMap[categoryId].smallest) { // Item belongs on shelf
 | ||||
|               const libraryItemObj = { | ||||
|                 ...libraryItem.toJSONMinified(), | ||||
|                 progressLastUpdate: mediaProgress.lastUpdate | ||||
|               } | ||||
| 
 | ||||
|               const indexToPut = categoryMap[categoryId].items.findIndex(i => mediaProgress.lastUpdate > i.progressLastUpdate) | ||||
|               if (indexToPut >= 0) { | ||||
|                 categoryMap[categoryId].items.splice(indexToPut, 0, libraryItemObj) | ||||
|               } else { // Should only happen when array is < max
 | ||||
|                 categoryMap[categoryId].items.push(libraryItemObj) | ||||
|               } | ||||
|               if (categoryMap[categoryId].items.length > maxEntitiesPerShelf) { | ||||
|                 // Remove last item
 | ||||
|                 categoryMap[categoryId].items.pop() | ||||
|                 categoryMap[categoryId].smallest = categoryMap[categoryId].items[categoryMap[categoryId].items.length - 1].progressLastUpdate | ||||
|               } | ||||
|               categoryMap[categoryId].biggest = categoryMap[categoryId].items[0].progressLastUpdate | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     // For Continue Series - Find next book in series for series that are in progress
 | ||||
|     for (const seriesId in seriesMap) { | ||||
|       seriesMap[seriesId].books = naturalSort(seriesMap[seriesId].books).asc(li => li.seriesSequence) | ||||
| 
 | ||||
|       if (seriesMap[seriesId].inProgress && !seriesMap[seriesId].hideFromContinueListening) { | ||||
|         // take the first book unread with the smallest series sequence
 | ||||
|         // unless the user is already listening to a book from this series
 | ||||
|         const hasActiveBook = seriesMap[seriesId].hasActiveBook | ||||
|         const nextBookInSeries = seriesMap[seriesId].firstBookUnread | ||||
| 
 | ||||
|         if (!hasActiveBook && nextBookInSeries) { | ||||
|           const bookForContinueSeries = { | ||||
|             ...nextBookInSeries, | ||||
|             prevBookInProgressLastUpdate: seriesMap[seriesId].bookInProgressLastUpdate | ||||
|           } | ||||
|           bookForContinueSeries.media.metadata.series = { | ||||
|             id: seriesId, | ||||
|             name: seriesMap[seriesId].name, | ||||
|             sequence: nextBookInSeries.seriesSequence | ||||
|           } | ||||
| 
 | ||||
|           const indexToPut = categoryMap['continue-series'].items.findIndex(i => i.prevBookInProgressLastUpdate < bookForContinueSeries.prevBookInProgressLastUpdate) | ||||
|           if (!categoryMap['continue-series'].items.find(book => book.id === bookForContinueSeries.id)) { | ||||
|             if (indexToPut >= 0) { | ||||
|               categoryMap['continue-series'].items.splice(indexToPut, 0, bookForContinueSeries) | ||||
|             } else if (categoryMap['continue-series'].items.length < 10) { // Max 10 books
 | ||||
|               categoryMap['continue-series'].items.push(bookForContinueSeries) | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     // For recommended
 | ||||
|     if (!isPodcastLibrary && notStartedBooks.length) { | ||||
|       const genresCount = Object.values(topGenresListened).reduce((a, b) => a + b, 0) | ||||
|       const authorsCount = Object.values(topAuthorsListened).reduce((a, b) => a + b, 0) | ||||
|       const tagsCount = Object.values(topTagsListened).reduce((a, b) => a + b, 0) | ||||
| 
 | ||||
|       for (const libraryItem of notStartedBooks) { | ||||
|         // dont include books in an unfinished series and books that are not first in an unstarted series
 | ||||
|         let shouldContinue = !libraryItem.media.metadata.series.length | ||||
|         libraryItem.media.metadata.series.forEach((se) => { | ||||
|           if (seriesMap[se.id]) { | ||||
|             if (seriesMap[se.id].inProgress) { | ||||
|               shouldContinue = false | ||||
|               return | ||||
|             } else if (seriesMap[se.id].books[0].id === libraryItem.id) { | ||||
|               shouldContinue = true | ||||
|             } | ||||
|           } | ||||
|         }) | ||||
|         if (!shouldContinue) { | ||||
|           continue; | ||||
|         } | ||||
| 
 | ||||
|         let totalWeight = 0 | ||||
| 
 | ||||
|         if (authorsCount > 0) { | ||||
|           libraryItem.media.metadata.authors.forEach((author) => { | ||||
|             if (topAuthorsListened[author.id]) { | ||||
|               totalWeight += topAuthorsListened[author.id] / authorsCount | ||||
|             } | ||||
|           }) | ||||
|         } | ||||
| 
 | ||||
|         if (genresCount > 0) { | ||||
|           libraryItem.media.metadata.genres.forEach((genre) => { | ||||
|             if (topGenresListened[genre]) { | ||||
|               totalWeight += topGenresListened[genre] / genresCount | ||||
|             } | ||||
|           }) | ||||
|         } | ||||
| 
 | ||||
|         if (tagsCount > 0) { | ||||
|           libraryItem.media.tags.forEach((tag) => { | ||||
|             if (topTagsListened[tag]) { | ||||
|               totalWeight += topTagsListened[tag] / tagsCount | ||||
|             } | ||||
|           }) | ||||
|         } | ||||
| 
 | ||||
|         if (!categoryMap.recommended.smallest || totalWeight > categoryMap.recommended.smallest) { | ||||
|           const libraryItemObj = { | ||||
|             ...libraryItem.toJSONMinified(), | ||||
|             weight: totalWeight | ||||
|           } | ||||
| 
 | ||||
|           const indexToPut = categoryMap.recommended.items.findIndex(i => totalWeight > i.weight) | ||||
|           if (indexToPut >= 0) { | ||||
|             categoryMap.recommended.items.splice(indexToPut, 0, libraryItemObj) | ||||
|           } else { | ||||
|             categoryMap.recommended.items.push(libraryItemObj) | ||||
|           } | ||||
| 
 | ||||
|           if (categoryMap.recommended.items.length > maxEntitiesPerShelf) { | ||||
|             categoryMap.recommended.items.pop() | ||||
|             categoryMap.recommended.smallest = categoryMap.recommended.items[categoryMap.recommended.items.length - 1].weight | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     // Sort series books by sequence
 | ||||
|     if (categoryMap['recent-series'].items.length) { | ||||
|       if (hideSingleBookSeries) { | ||||
|         categoryMap['recent-series'].items = categoryMap['recent-series'].items.filter(seriesItem => seriesItem.books.length > 1) | ||||
|       } | ||||
|       // Limit series shown to 5
 | ||||
|       categoryMap['recent-series'].items = categoryMap['recent-series'].items.slice(0, 5) | ||||
| 
 | ||||
|       for (const seriesItem of categoryMap['recent-series'].items) { | ||||
|         seriesItem.books = naturalSort(seriesItem.books).asc(li => li.seriesSequence) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     const categoriesWithItems = Object.values(categoryMap).filter(cat => cat.items.length) | ||||
| 
 | ||||
|     const finalShelves = [] | ||||
|     for (const categoryWithItems of categoriesWithItems) { | ||||
|       const shelf = shelves.find(s => s.id === categoryWithItems.id) | ||||
|       shelf.entities = categoryWithItems.items | ||||
| 
 | ||||
|       // Add rssFeed to entities if query string "include=rssfeed" was on request
 | ||||
|       if (includeRssFeed) { | ||||
|         if (shelf.type === 'book' || shelf.type === 'podcast') { | ||||
|           shelf.entities = await Promise.all(shelf.entities.map(async (item) => { | ||||
|             const feed = await ctx.rssFeedManager.findFeedForEntityId(item.id) | ||||
|             item.rssFeed = feed?.toJSONMinified() || null | ||||
|             return item | ||||
|           })) | ||||
|         } else if (shelf.type === 'series') { | ||||
|           shelf.entities = await Promise.all(shelf.entities.map(async (series) => { | ||||
|             const feed = await ctx.rssFeedManager.findFeedForEntityId(series.id) | ||||
|             series.rssFeed = feed?.toJSONMinified() || null | ||||
|             return series | ||||
|           })) | ||||
|         } | ||||
|       } | ||||
|       finalShelves.push(shelf) | ||||
|     } | ||||
|     return finalShelves | ||||
|   }, | ||||
| 
 | ||||
|   groupMusicLibraryItemsIntoAlbums(libraryItems) { | ||||
|     const albums = {} | ||||
| 
 | ||||
|     libraryItems.forEach((li) => { | ||||
|       const albumTitle = li.media.metadata.album | ||||
|       const albumArtist = li.media.metadata.albumArtist | ||||
| 
 | ||||
|       if (albumTitle && !albums[albumTitle]) { | ||||
|         albums[albumTitle] = { | ||||
|           title: albumTitle, | ||||
|           artist: albumArtist, | ||||
|           libraryItemId: li.media.coverPath ? li.id : null, | ||||
|           numTracks: 1 | ||||
|         } | ||||
|       } else if (albumTitle && albums[albumTitle].artist === albumArtist) { | ||||
|         if (!albums[albumTitle].libraryItemId && li.media.coverPath) albums[albumTitle].libraryItemId = li.id | ||||
|         albums[albumTitle].numTracks++ | ||||
|       } else { | ||||
|         if (albumTitle) { | ||||
|           Logger.warn(`Music track "${li.media.metadata.title}" with album "${albumTitle}" has a different album artist then another track in the same album.  This track album artist is "${albumArtist}" but the album artist is already set to "${albums[albumTitle].artist}"`) | ||||
|         } | ||||
|         if (!albums['_none_']) albums['_none_'] = { title: 'No Album', artist: 'Various Artists', libraryItemId: null, numTracks: 0 } | ||||
|         albums['_none_'].numTracks++ | ||||
|       } | ||||
|     }) | ||||
| 
 | ||||
|     return Object.values(albums) | ||||
|   } | ||||
| } | ||||
|  | ||||
							
								
								
									
										70
									
								
								server/utils/queries/authorFilters.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								server/utils/queries/authorFilters.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,70 @@ | ||||
| const Sequelize = require('sequelize') | ||||
| const Database = require('../../Database') | ||||
| 
 | ||||
| module.exports = { | ||||
|   /** | ||||
|    * Get authors with count of num books | ||||
|    * @param {string} libraryId  | ||||
|    * @returns {{id:string, name:string, count:number}} | ||||
|    */ | ||||
|   async getAuthorsWithCount(libraryId) { | ||||
|     const authors = await Database.authorModel.findAll({ | ||||
|       where: [ | ||||
|         { | ||||
|           libraryId | ||||
|         }, | ||||
|         Sequelize.where(Sequelize.literal('count'), { | ||||
|           [Sequelize.Op.gt]: 0 | ||||
|         }) | ||||
|       ], | ||||
|       attributes: [ | ||||
|         'id', | ||||
|         'name', | ||||
|         [Sequelize.literal('(SELECT count(*) FROM bookAuthors ba WHERE ba.authorId = author.id)'), 'count'] | ||||
|       ], | ||||
|       order: [ | ||||
|         ['count', 'DESC'] | ||||
|       ] | ||||
|     }) | ||||
|     return authors.map(au => { | ||||
|       return { | ||||
|         id: au.id, | ||||
|         name: au.name, | ||||
|         count: au.dataValues.count | ||||
|       } | ||||
|     }) | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Search authors | ||||
|    * @param {string} libraryId  | ||||
|    * @param {string} query  | ||||
|    * @param {number} limit | ||||
|    * @param {number} offset | ||||
|    * @returns {object[]} oldAuthor with numBooks | ||||
|    */ | ||||
|   async search(libraryId, query, limit, offset) { | ||||
|     const authors = await Database.authorModel.findAll({ | ||||
|       where: { | ||||
|         name: { | ||||
|           [Sequelize.Op.substring]: query | ||||
|         }, | ||||
|         libraryId | ||||
|       }, | ||||
|       attributes: { | ||||
|         include: [ | ||||
|           [Sequelize.literal('(SELECT count(*) FROM bookAuthors ba WHERE ba.authorId = author.id)'), 'numBooks'] | ||||
|         ] | ||||
|       }, | ||||
|       limit, | ||||
|       offset | ||||
|     }) | ||||
|     const authorMatches = [] | ||||
|     for (const author of authors) { | ||||
|       const oldAuthor = author.getOldAuthor().toJSON() | ||||
|       oldAuthor.numBooks = author.dataValues.numBooks | ||||
|       authorMatches.push(oldAuthor) | ||||
|     } | ||||
|     return authorMatches | ||||
|   } | ||||
| } | ||||
| @ -15,8 +15,8 @@ module.exports = { | ||||
| 
 | ||||
|   /** | ||||
|    * Get library items using filter and sort | ||||
|    * @param {oldLibrary} library  | ||||
|    * @param {oldUser} user  | ||||
|    * @param {import('../../objects/Library')} library  | ||||
|    * @param {import('../../objects/user/User')} user  | ||||
|    * @param {object} options  | ||||
|    * @returns {object} { libraryItems:LibraryItem[], count:number } | ||||
|    */ | ||||
| @ -41,20 +41,20 @@ module.exports = { | ||||
| 
 | ||||
|   /** | ||||
|    * Get library items for continue listening & continue reading shelves | ||||
|    * @param {oldLibrary} library  | ||||
|    * @param {oldUser} user  | ||||
|    * @param {import('../../objects/Library')} library  | ||||
|    * @param {import('../../objects/user/User')} user  | ||||
|    * @param {string[]} include  | ||||
|    * @param {number} limit  | ||||
|    * @returns {object} { items:LibraryItem[], count:number } | ||||
|    * @returns {Promise<{ items:import('../../models/LibraryItem')[], count:number }>} | ||||
|    */ | ||||
|   async getMediaItemsInProgress(library, user, include, limit) { | ||||
|     if (library.mediaType === 'book') { | ||||
|       const { libraryItems, count } = await libraryItemsBookFilters.getFilteredLibraryItems(library.id, user, 'progress', 'in-progress', 'progress', true, false, include, limit, 0) | ||||
|       return { | ||||
|         items: libraryItems.map(li => { | ||||
|           const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(li).toJSONMinified() | ||||
|           const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(li).toJSONMinified() | ||||
|           if (li.rssFeed) { | ||||
|             oldLibraryItem.rssFeed = Database.models.feed.getOldFeed(li.rssFeed).toJSONMinified() | ||||
|             oldLibraryItem.rssFeed = Database.feedModel.getOldFeed(li.rssFeed).toJSONMinified() | ||||
|           } | ||||
|           return oldLibraryItem | ||||
|         }), | ||||
| @ -65,7 +65,7 @@ module.exports = { | ||||
|       return { | ||||
|         count, | ||||
|         items: libraryItems.map(li => { | ||||
|           const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(li).toJSONMinified() | ||||
|           const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(li).toJSONMinified() | ||||
|           oldLibraryItem.recentEpisode = li.recentEpisode | ||||
|           return oldLibraryItem | ||||
|         }) | ||||
| @ -86,9 +86,9 @@ module.exports = { | ||||
|       const { libraryItems, count } = await libraryItemsBookFilters.getFilteredLibraryItems(library.id, user, 'recent', null, 'addedAt', true, false, include, limit, 0) | ||||
|       return { | ||||
|         libraryItems: libraryItems.map(li => { | ||||
|           const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(li).toJSONMinified() | ||||
|           const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(li).toJSONMinified() | ||||
|           if (li.rssFeed) { | ||||
|             oldLibraryItem.rssFeed = Database.models.feed.getOldFeed(li.rssFeed).toJSONMinified() | ||||
|             oldLibraryItem.rssFeed = Database.feedModel.getOldFeed(li.rssFeed).toJSONMinified() | ||||
|           } | ||||
|           if (li.size && !oldLibraryItem.media.size) { | ||||
|             oldLibraryItem.media.size = li.size | ||||
| @ -101,9 +101,9 @@ module.exports = { | ||||
|       const { libraryItems, count } = await libraryItemsPodcastFilters.getFilteredLibraryItems(library.id, user, 'recent', null, 'addedAt', true, include, limit, 0) | ||||
|       return { | ||||
|         libraryItems: libraryItems.map(li => { | ||||
|           const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(li).toJSONMinified() | ||||
|           const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(li).toJSONMinified() | ||||
|           if (li.rssFeed) { | ||||
|             oldLibraryItem.rssFeed = Database.models.feed.getOldFeed(li.rssFeed).toJSONMinified() | ||||
|             oldLibraryItem.rssFeed = Database.feedModel.getOldFeed(li.rssFeed).toJSONMinified() | ||||
|           } | ||||
|           if (li.size && !oldLibraryItem.media.size) { | ||||
|             oldLibraryItem.media.size = li.size | ||||
| @ -127,9 +127,9 @@ module.exports = { | ||||
|     const { libraryItems, count } = await libraryItemsBookFilters.getContinueSeriesLibraryItems(library.id, user, include, limit, 0) | ||||
|     return { | ||||
|       libraryItems: libraryItems.map(li => { | ||||
|         const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(li).toJSONMinified() | ||||
|         const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(li).toJSONMinified() | ||||
|         if (li.rssFeed) { | ||||
|           oldLibraryItem.rssFeed = Database.models.feed.getOldFeed(li.rssFeed).toJSONMinified() | ||||
|           oldLibraryItem.rssFeed = Database.feedModel.getOldFeed(li.rssFeed).toJSONMinified() | ||||
|         } | ||||
|         if (li.series) { | ||||
|           oldLibraryItem.media.metadata.series = li.series | ||||
| @ -153,9 +153,9 @@ module.exports = { | ||||
|       const { libraryItems, count } = await libraryItemsBookFilters.getFilteredLibraryItems(library.id, user, 'progress', 'finished', 'progress', true, false, include, limit, 0) | ||||
|       return { | ||||
|         items: libraryItems.map(li => { | ||||
|           const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(li).toJSONMinified() | ||||
|           const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(li).toJSONMinified() | ||||
|           if (li.rssFeed) { | ||||
|             oldLibraryItem.rssFeed = Database.models.feed.getOldFeed(li.rssFeed).toJSONMinified() | ||||
|             oldLibraryItem.rssFeed = Database.feedModel.getOldFeed(li.rssFeed).toJSONMinified() | ||||
|           } | ||||
|           return oldLibraryItem | ||||
|         }), | ||||
| @ -166,7 +166,7 @@ module.exports = { | ||||
|       return { | ||||
|         count, | ||||
|         items: libraryItems.map(li => { | ||||
|           const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(li).toJSONMinified() | ||||
|           const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(li).toJSONMinified() | ||||
|           oldLibraryItem.recentEpisode = li.recentEpisode | ||||
|           return oldLibraryItem | ||||
|         }) | ||||
| @ -176,19 +176,19 @@ module.exports = { | ||||
| 
 | ||||
|   /** | ||||
|    * Get series for recent series shelf | ||||
|    * @param {oldLibrary} library  | ||||
|    * @param {oldUser} user | ||||
|    * @param {import('../../objects/Library')} library  | ||||
|    * @param {import('../../objects/user/User')} user | ||||
|    * @param {string[]} include  | ||||
|    * @param {number} limit  | ||||
|    * @returns {object} { series:oldSeries[], count:number} | ||||
|    * @returns {{ series:import('../../objects/entities/Series')[], count:number}}  | ||||
|    */ | ||||
|   async getSeriesMostRecentlyAdded(library, user, include, limit) { | ||||
|     if (library.mediaType !== 'book') return { series: [], count: 0 } | ||||
|     if (!library.isBook) return { series: [], count: 0 } | ||||
| 
 | ||||
|     const seriesIncludes = [] | ||||
|     if (include.includes('rssfeed')) { | ||||
|       seriesIncludes.push({ | ||||
|         model: Database.models.feed | ||||
|         model: Database.feedModel | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
| @ -221,7 +221,7 @@ module.exports = { | ||||
|       })) | ||||
|     } | ||||
| 
 | ||||
|     const { rows: series, count } = await Database.models.series.findAndCountAll({ | ||||
|     const { rows: series, count } = await Database.seriesModel.findAndCountAll({ | ||||
|       where: seriesWhere, | ||||
|       limit, | ||||
|       offset: 0, | ||||
| @ -230,12 +230,12 @@ module.exports = { | ||||
|       replacements: userPermissionBookWhere.replacements, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.bookSeries, | ||||
|           model: Database.bookSeriesModel, | ||||
|           include: { | ||||
|             model: Database.models.book, | ||||
|             model: Database.bookModel, | ||||
|             where: userPermissionBookWhere.bookWhere, | ||||
|             include: { | ||||
|               model: Database.models.libraryItem | ||||
|               model: Database.libraryItemModel | ||||
|             } | ||||
|           }, | ||||
|           separate: true | ||||
| @ -252,7 +252,7 @@ module.exports = { | ||||
|       const oldSeries = s.getOldSeries().toJSON() | ||||
| 
 | ||||
|       if (s.feeds?.length) { | ||||
|         oldSeries.rssFeed = Database.models.feed.getOldFeed(s.feeds[0]).toJSONMinified() | ||||
|         oldSeries.rssFeed = Database.feedModel.getOldFeed(s.feeds[0]).toJSONMinified() | ||||
|       } | ||||
| 
 | ||||
|       // TODO: Sort books by sequence in query
 | ||||
| @ -268,7 +268,7 @@ module.exports = { | ||||
|         const libraryItem = bs.book.libraryItem.toJSON() | ||||
|         delete bs.book.libraryItem | ||||
|         libraryItem.media = bs.book | ||||
|         const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(libraryItem).toJSONMinified() | ||||
|         const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(libraryItem).toJSONMinified() | ||||
|         return oldLibraryItem | ||||
|       }) | ||||
|       allOldSeries.push(oldSeries) | ||||
| @ -291,7 +291,7 @@ module.exports = { | ||||
|   async getNewestAuthors(library, user, limit) { | ||||
|     if (library.mediaType !== 'book') return { authors: [], count: 0 } | ||||
| 
 | ||||
|     const { rows: authors, count } = await Database.models.author.findAndCountAll({ | ||||
|     const { rows: authors, count } = await Database.authorModel.findAndCountAll({ | ||||
|       where: { | ||||
|         libraryId: library.id, | ||||
|         createdAt: { | ||||
| @ -299,7 +299,7 @@ module.exports = { | ||||
|         } | ||||
|       }, | ||||
|       include: { | ||||
|         model: Database.models.bookAuthor, | ||||
|         model: Database.bookAuthorModel, | ||||
|         required: true // Must belong to a book
 | ||||
|       }, | ||||
|       limit, | ||||
| @ -332,9 +332,9 @@ module.exports = { | ||||
|     const { libraryItems, count } = await libraryItemsBookFilters.getDiscoverLibraryItems(library.id, user, include, limit) | ||||
|     return { | ||||
|       libraryItems: libraryItems.map(li => { | ||||
|         const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(li).toJSONMinified() | ||||
|         const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(li).toJSONMinified() | ||||
|         if (li.rssFeed) { | ||||
|           oldLibraryItem.rssFeed = Database.models.feed.getOldFeed(li.rssFeed).toJSONMinified() | ||||
|           oldLibraryItem.rssFeed = Database.feedModel.getOldFeed(li.rssFeed).toJSONMinified() | ||||
|         } | ||||
|         return oldLibraryItem | ||||
|       }), | ||||
| @ -356,7 +356,7 @@ module.exports = { | ||||
|     return { | ||||
|       count, | ||||
|       libraryItems: libraryItems.map(li => { | ||||
|         const oldLibraryItem = Database.models.libraryItem.getOldLibraryItem(li).toJSONMinified() | ||||
|         const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(li).toJSONMinified() | ||||
|         oldLibraryItem.recentEpisode = li.recentEpisode | ||||
|         return oldLibraryItem | ||||
|       }) | ||||
| @ -390,7 +390,7 @@ module.exports = { | ||||
| 
 | ||||
|   /** | ||||
|    * Get filter data used in filter menus | ||||
|    * @param {oldLibrary} oldLibrary  | ||||
|    * @param {import('../../objects/Library')} oldLibrary  | ||||
|    * @returns {Promise<object>} | ||||
|    */ | ||||
|   async getFilterData(oldLibrary) { | ||||
| @ -417,9 +417,9 @@ module.exports = { | ||||
|     } | ||||
| 
 | ||||
|     if (oldLibrary.isPodcast) { | ||||
|       const podcasts = await Database.models.podcast.findAll({ | ||||
|       const podcasts = await Database.podcastModel.findAll({ | ||||
|         include: { | ||||
|           model: Database.models.libraryItem, | ||||
|           model: Database.libraryItemModel, | ||||
|           attributes: [], | ||||
|           where: { | ||||
|             libraryId: oldLibrary.id | ||||
| @ -436,9 +436,9 @@ module.exports = { | ||||
|         } | ||||
|       } | ||||
|     } else { | ||||
|       const books = await Database.models.book.findAll({ | ||||
|       const books = await Database.bookModel.findAll({ | ||||
|         include: { | ||||
|           model: Database.models.libraryItem, | ||||
|           model: Database.libraryItemModel, | ||||
|           attributes: ['isMissing', 'isInvalid'], | ||||
|           where: { | ||||
|             libraryId: oldLibrary.id | ||||
| @ -461,7 +461,7 @@ module.exports = { | ||||
|         if (book.language) data.languages.add(book.language) | ||||
|       } | ||||
| 
 | ||||
|       const series = await Database.models.series.findAll({ | ||||
|       const series = await Database.seriesModel.findAll({ | ||||
|         where: { | ||||
|           libraryId: oldLibrary.id | ||||
|         }, | ||||
| @ -469,7 +469,7 @@ module.exports = { | ||||
|       }) | ||||
|       series.forEach((s) => data.series.push({ id: s.id, name: s.name })) | ||||
| 
 | ||||
|       const authors = await Database.models.author.findAll({ | ||||
|       const authors = await Database.authorModel.findAll({ | ||||
|         where: { | ||||
|           libraryId: oldLibrary.id | ||||
|         }, | ||||
|  | ||||
| @ -1,15 +1,17 @@ | ||||
| const Sequelize = require('sequelize') | ||||
| const Database = require('../../Database') | ||||
| const libraryItemsBookFilters = require('./libraryItemsBookFilters') | ||||
| const libraryItemsPodcastFilters = require('./libraryItemsPodcastFilters') | ||||
| 
 | ||||
| module.exports = { | ||||
|   /** | ||||
|    * Get all library items that have tags | ||||
|    * @param {string[]} tags  | ||||
|    * @returns {Promise<LibraryItem[]>} | ||||
|    * @returns {Promise<import('../../models/LibraryItem')[]>} | ||||
|    */ | ||||
|   async getAllLibraryItemsWithTags(tags) { | ||||
|     const libraryItems = [] | ||||
|     const booksWithTag = await Database.models.book.findAll({ | ||||
|     const booksWithTag = await Database.bookModel.findAll({ | ||||
|       where: Sequelize.where(Sequelize.literal(`(SELECT count(*) FROM json_each(tags) WHERE json_valid(tags) AND json_each.value IN (:tags))`), { | ||||
|         [Sequelize.Op.gte]: 1 | ||||
|       }), | ||||
| @ -18,16 +20,16 @@ module.exports = { | ||||
|       }, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.libraryItem | ||||
|           model: Database.libraryItemModel | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.author, | ||||
|           model: Database.authorModel, | ||||
|           through: { | ||||
|             attributes: [] | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.series, | ||||
|           model: Database.seriesModel, | ||||
|           through: { | ||||
|             attributes: ['sequence'] | ||||
|           } | ||||
| @ -39,7 +41,7 @@ module.exports = { | ||||
|       libraryItem.media = book | ||||
|       libraryItems.push(libraryItem) | ||||
|     } | ||||
|     const podcastsWithTag = await Database.models.podcast.findAll({ | ||||
|     const podcastsWithTag = await Database.podcastModel.findAll({ | ||||
|       where: Sequelize.where(Sequelize.literal(`(SELECT count(*) FROM json_each(tags) WHERE json_valid(tags) AND json_each.value IN (:tags))`), { | ||||
|         [Sequelize.Op.gte]: 1 | ||||
|       }), | ||||
| @ -48,10 +50,10 @@ module.exports = { | ||||
|       }, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.libraryItem | ||||
|           model: Database.libraryItemModel | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.podcastEpisode | ||||
|           model: Database.podcastEpisodeModel | ||||
|         } | ||||
|       ] | ||||
|     }) | ||||
| @ -70,7 +72,7 @@ module.exports = { | ||||
|    */ | ||||
|   async getAllLibraryItemsWithGenres(genres) { | ||||
|     const libraryItems = [] | ||||
|     const booksWithGenre = await Database.models.book.findAll({ | ||||
|     const booksWithGenre = await Database.bookModel.findAll({ | ||||
|       where: Sequelize.where(Sequelize.literal(`(SELECT count(*) FROM json_each(genres) WHERE json_valid(genres) AND json_each.value IN (:genres))`), { | ||||
|         [Sequelize.Op.gte]: 1 | ||||
|       }), | ||||
| @ -79,16 +81,16 @@ module.exports = { | ||||
|       }, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.libraryItem | ||||
|           model: Database.libraryItemModel | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.author, | ||||
|           model: Database.authorModel, | ||||
|           through: { | ||||
|             attributes: [] | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.series, | ||||
|           model: Database.seriesModel, | ||||
|           through: { | ||||
|             attributes: ['sequence'] | ||||
|           } | ||||
| @ -100,7 +102,7 @@ module.exports = { | ||||
|       libraryItem.media = book | ||||
|       libraryItems.push(libraryItem) | ||||
|     } | ||||
|     const podcastsWithGenre = await Database.models.podcast.findAll({ | ||||
|     const podcastsWithGenre = await Database.podcastModel.findAll({ | ||||
|       where: Sequelize.where(Sequelize.literal(`(SELECT count(*) FROM json_each(genres) WHERE json_valid(genres) AND json_each.value IN (:genres))`), { | ||||
|         [Sequelize.Op.gte]: 1 | ||||
|       }), | ||||
| @ -109,10 +111,10 @@ module.exports = { | ||||
|       }, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.libraryItem | ||||
|           model: Database.libraryItemModel | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.podcastEpisode | ||||
|           model: Database.podcastEpisodeModel | ||||
|         } | ||||
|       ] | ||||
|     }) | ||||
| @ -127,11 +129,11 @@ module.exports = { | ||||
|   /** | ||||
|  * Get all library items that have narrators | ||||
|  * @param {string[]} narrators  | ||||
|  * @returns {Promise<LibraryItem[]>} | ||||
|  * @returns {Promise<import('../../models/LibraryItem')[]>} | ||||
|  */ | ||||
|   async getAllLibraryItemsWithNarrators(narrators) { | ||||
|     const libraryItems = [] | ||||
|     const booksWithGenre = await Database.models.book.findAll({ | ||||
|     const booksWithGenre = await Database.bookModel.findAll({ | ||||
|       where: Sequelize.where(Sequelize.literal(`(SELECT count(*) FROM json_each(narrators) WHERE json_valid(narrators) AND json_each.value IN (:narrators))`), { | ||||
|         [Sequelize.Op.gte]: 1 | ||||
|       }), | ||||
| @ -140,16 +142,16 @@ module.exports = { | ||||
|       }, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.libraryItem | ||||
|           model: Database.libraryItemModel | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.author, | ||||
|           model: Database.authorModel, | ||||
|           through: { | ||||
|             attributes: [] | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.series, | ||||
|           model: Database.seriesModel, | ||||
|           through: { | ||||
|             attributes: ['sequence'] | ||||
|           } | ||||
| @ -162,5 +164,57 @@ module.exports = { | ||||
|       libraryItems.push(libraryItem) | ||||
|     } | ||||
|     return libraryItems | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Search library items | ||||
|    * @param {import('../../objects/user/User')} oldUser  | ||||
|    * @param {import('../../objects/Library')} oldLibrary  | ||||
|    * @param {string} query | ||||
|    * @param {number} limit  | ||||
|    * @returns {{book:object[], narrators:object[], authors:object[], tags:object[], series:object[], podcast:object[]}} | ||||
|    */ | ||||
|   search(oldUser, oldLibrary, query, limit) { | ||||
|     if (oldLibrary.isBook) { | ||||
|       return libraryItemsBookFilters.search(oldUser, oldLibrary, query, limit, 0) | ||||
|     } else { | ||||
|       return libraryItemsPodcastFilters.search(oldUser, oldLibrary, query, limit, 0) | ||||
|     } | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Get largest items in library | ||||
|    * @param {string} libraryId  | ||||
|    * @param {number} limit  | ||||
|    * @returns {Promise<{ id:string, title:string, size:number }[]>} | ||||
|    */ | ||||
|   async getLargestItems(libraryId, limit) { | ||||
|     const libraryItems = await Database.libraryItemModel.findAll({ | ||||
|       attributes: ['id', 'mediaId', 'mediaType', 'size'], | ||||
|       where: { | ||||
|         libraryId | ||||
|       }, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.bookModel, | ||||
|           attributes: ['id', 'title'] | ||||
|         }, | ||||
|         { | ||||
|           model: Database.podcastModel, | ||||
|           attributes: ['id', 'title'] | ||||
|         } | ||||
|       ], | ||||
|       order: [ | ||||
|         ['size', 'DESC'] | ||||
|       ], | ||||
|       limit | ||||
|     }) | ||||
|     return libraryItems.map(libraryItem => { | ||||
|       return { | ||||
|         id: libraryItem.id, | ||||
|         title: libraryItem.media.title, | ||||
|         size: libraryItem.size | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| @ -1,12 +1,13 @@ | ||||
| const Sequelize = require('sequelize') | ||||
| const Database = require('../../Database') | ||||
| const Logger = require('../../Logger') | ||||
| const authorFilters = require('./authorFilters') | ||||
| 
 | ||||
| module.exports = { | ||||
|   /** | ||||
|    * User permissions to restrict books for explicit content & tags | ||||
|    * @param {oldUser} user  | ||||
|    * @returns {object} { bookWhere:Sequelize.WhereOptions, replacements:string[] } | ||||
|    * @param {import('../../objects/user/User')} user  | ||||
|    * @returns {{ bookWhere:Sequelize.WhereOptions, replacements:object }} | ||||
|    */ | ||||
|   getUserPermissionBookWhereQuery(user) { | ||||
|     const bookWhere = [] | ||||
| @ -278,7 +279,7 @@ module.exports = { | ||||
|    * @returns {object} { booksToExclude, bookSeriesToInclude } | ||||
|    */ | ||||
|   async getCollapseSeriesBooksToExclude(bookFindOptions, seriesWhere) { | ||||
|     const allSeries = await Database.models.series.findAll({ | ||||
|     const allSeries = await Database.seriesModel.findAll({ | ||||
|       attributes: [ | ||||
|         'id', | ||||
|         'name', | ||||
| @ -289,7 +290,7 @@ module.exports = { | ||||
|       where: seriesWhere, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.book, | ||||
|           model: Database.bookModel, | ||||
|           attributes: ['id', 'title'], | ||||
|           through: { | ||||
|             attributes: ['id', 'seriesId', 'bookId', 'sequence'] | ||||
| @ -373,10 +374,10 @@ module.exports = { | ||||
|     } | ||||
| 
 | ||||
|     let seriesInclude = { | ||||
|       model: Database.models.bookSeries, | ||||
|       model: Database.bookSeriesModel, | ||||
|       attributes: ['id', 'seriesId', 'sequence', 'createdAt'], | ||||
|       include: { | ||||
|         model: Database.models.series, | ||||
|         model: Database.seriesModel, | ||||
|         attributes: ['id', 'name', 'nameIgnorePrefix'] | ||||
|       }, | ||||
|       order: [ | ||||
| @ -386,10 +387,10 @@ module.exports = { | ||||
|     } | ||||
| 
 | ||||
|     let authorInclude = { | ||||
|       model: Database.models.bookAuthor, | ||||
|       model: Database.bookAuthorModel, | ||||
|       attributes: ['authorId', 'createdAt'], | ||||
|       include: { | ||||
|         model: Database.models.author, | ||||
|         model: Database.authorModel, | ||||
|         attributes: ['id', 'name'] | ||||
|       }, | ||||
|       order: [ | ||||
| @ -404,13 +405,13 @@ module.exports = { | ||||
|     const bookIncludes = [] | ||||
|     if (includeRSSFeed) { | ||||
|       libraryItemIncludes.push({ | ||||
|         model: Database.models.feed, | ||||
|         model: Database.feedModel, | ||||
|         required: filterGroup === 'feed-open' | ||||
|       }) | ||||
|     } | ||||
|     if (filterGroup === 'feed-open' && !includeRSSFeed) { | ||||
|       libraryItemIncludes.push({ | ||||
|         model: Database.models.feed, | ||||
|         model: Database.feedModel, | ||||
|         required: true | ||||
|       }) | ||||
|     } else if (filterGroup === 'ebooks' && filterValue === 'supplementary') { | ||||
| @ -420,7 +421,7 @@ module.exports = { | ||||
|       } | ||||
|     } else if (filterGroup === 'missing' && filterValue === 'authors') { | ||||
|       authorInclude = { | ||||
|         model: Database.models.author, | ||||
|         model: Database.authorModel, | ||||
|         attributes: ['id'], | ||||
|         through: { | ||||
|           attributes: [] | ||||
| @ -428,7 +429,7 @@ module.exports = { | ||||
|       } | ||||
|     } else if ((filterGroup === 'series' && filterValue === 'no-series') || (filterGroup === 'missing' && filterValue === 'series')) { | ||||
|       seriesInclude = { | ||||
|         model: Database.models.series, | ||||
|         model: Database.seriesModel, | ||||
|         attributes: ['id'], | ||||
|         through: { | ||||
|           attributes: [] | ||||
| @ -436,7 +437,7 @@ module.exports = { | ||||
|       } | ||||
|     } else if (filterGroup === 'authors') { | ||||
|       bookIncludes.push({ | ||||
|         model: Database.models.author, | ||||
|         model: Database.authorModel, | ||||
|         attributes: ['id', 'name'], | ||||
|         where: { | ||||
|           id: filterValue | ||||
| @ -447,7 +448,7 @@ module.exports = { | ||||
|       }) | ||||
|     } else if (filterGroup === 'series') { | ||||
|       bookIncludes.push({ | ||||
|         model: Database.models.series, | ||||
|         model: Database.seriesModel, | ||||
|         attributes: ['id', 'name'], | ||||
|         where: { | ||||
|           id: filterValue | ||||
| @ -471,7 +472,7 @@ module.exports = { | ||||
|       ] | ||||
|     } else if (filterGroup === 'progress' && user) { | ||||
|       bookIncludes.push({ | ||||
|         model: Database.models.mediaProgress, | ||||
|         model: Database.mediaProgressModel, | ||||
|         attributes: ['id', 'isFinished', 'currentTime', 'ebookProgress', 'updatedAt'], | ||||
|         where: { | ||||
|           userId: user.id | ||||
| @ -512,7 +513,7 @@ module.exports = { | ||||
|         where: seriesBookWhere, | ||||
|         include: [ | ||||
|           { | ||||
|             model: Database.models.libraryItem, | ||||
|             model: Database.libraryItemModel, | ||||
|             required: true, | ||||
|             where: libraryItemWhere, | ||||
|             include: libraryItemIncludes | ||||
| @ -537,11 +538,11 @@ module.exports = { | ||||
|       if (global.ServerSettings.sortingIgnorePrefix) { | ||||
|         bookAttributes.include.push([Sequelize.literal(`IFNULL((SELECT s.nameIgnorePrefix FROM bookSeries AS bs, series AS s WHERE bs.seriesId = s.id AND bs.bookId = book.id AND bs.id IN (${bookSeriesToInclude.map(v => `"${v.id}"`).join(', ')})), titleIgnorePrefix)`), 'display_title']) | ||||
|       } else { | ||||
|         bookAttributes.include.push([Sequelize.literal(`IFNULL((SELECT s.name FROM bookSeries AS bs, series AS s WHERE bs.seriesId = s.id AND bs.bookId = book.id AND bs.id IN (${bookSeriesToInclude.map(v => `"${v.id}"`).join(', ')})), title)`), 'display_title']) | ||||
|         bookAttributes.include.push([Sequelize.literal(`IFNULL((SELECT s.name FROM bookSeries AS bs, series AS s WHERE bs.seriesId = s.id AND bs.bookId = book.id AND bs.id IN (${bookSeriesToInclude.map(v => `"${v.id}"`).join(', ')})), \`book\`.\`title\`)`), 'display_title']) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     const { rows: books, count } = await Database.models.book.findAndCountAll({ | ||||
|     const { rows: books, count } = await Database.bookModel.findAndCountAll({ | ||||
|       where: bookWhere, | ||||
|       distinct: true, | ||||
|       attributes: bookAttributes, | ||||
| @ -552,7 +553,7 @@ module.exports = { | ||||
|       }, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.libraryItem, | ||||
|           model: Database.libraryItemModel, | ||||
|           required: true, | ||||
|           where: libraryItemWhere, | ||||
|           include: libraryItemIncludes | ||||
| @ -632,7 +633,7 @@ module.exports = { | ||||
|     const libraryItemIncludes = [] | ||||
|     if (include.includes('rssfeed')) { | ||||
|       libraryItemIncludes.push({ | ||||
|         model: Database.models.feed | ||||
|         model: Database.feedModel | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
| @ -642,7 +643,7 @@ module.exports = { | ||||
|     const userPermissionBookWhere = this.getUserPermissionBookWhereQuery(user) | ||||
|     bookWhere.push(...userPermissionBookWhere.bookWhere) | ||||
| 
 | ||||
|     const { rows: series, count } = await Database.models.series.findAndCountAll({ | ||||
|     const { rows: series, count } = await Database.seriesModel.findAndCountAll({ | ||||
|       where: [ | ||||
|         { | ||||
|           libraryId | ||||
| @ -669,7 +670,7 @@ module.exports = { | ||||
|         ...userPermissionBookWhere.replacements | ||||
|       }, | ||||
|       include: { | ||||
|         model: Database.models.bookSeries, | ||||
|         model: Database.bookSeriesModel, | ||||
|         attributes: ['bookId', 'sequence'], | ||||
|         separate: true, | ||||
|         subQuery: false, | ||||
| @ -682,21 +683,21 @@ module.exports = { | ||||
|           } | ||||
|         }, | ||||
|         include: { | ||||
|           model: Database.models.book, | ||||
|           model: Database.bookModel, | ||||
|           where: bookWhere, | ||||
|           include: [ | ||||
|             { | ||||
|               model: Database.models.libraryItem, | ||||
|               model: Database.libraryItemModel, | ||||
|               include: libraryItemIncludes | ||||
|             }, | ||||
|             { | ||||
|               model: Database.models.author, | ||||
|               model: Database.authorModel, | ||||
|               through: { | ||||
|                 attributes: [] | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               model: Database.models.mediaProgress, | ||||
|               model: Database.mediaProgressModel, | ||||
|               where: { | ||||
|                 userId: user.id | ||||
|               }, | ||||
| @ -751,7 +752,7 @@ module.exports = { | ||||
|     const userPermissionBookWhere = this.getUserPermissionBookWhereQuery(user) | ||||
| 
 | ||||
|     // Step 1: Get the first book of every series that hasnt been started yet
 | ||||
|     const seriesNotStarted = await Database.models.series.findAll({ | ||||
|     const seriesNotStarted = await Database.seriesModel.findAll({ | ||||
|       where: [ | ||||
|         { | ||||
|           libraryId | ||||
| @ -764,12 +765,12 @@ module.exports = { | ||||
|       }, | ||||
|       attributes: ['id'], | ||||
|       include: { | ||||
|         model: Database.models.bookSeries, | ||||
|         model: Database.bookSeriesModel, | ||||
|         attributes: ['bookId', 'sequence'], | ||||
|         separate: true, | ||||
|         required: true, | ||||
|         include: { | ||||
|           model: Database.models.book, | ||||
|           model: Database.bookModel, | ||||
|           where: userPermissionBookWhere.bookWhere | ||||
|         }, | ||||
|         order: [ | ||||
| @ -788,12 +789,12 @@ module.exports = { | ||||
|     const libraryItemIncludes = [] | ||||
|     if (include.includes('rssfeed')) { | ||||
|       libraryItemIncludes.push({ | ||||
|         model: Database.models.feed | ||||
|         model: Database.feedModel | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     // Step 2: Get books not started and not in a series OR is the first book of a series not started (ordered randomly)
 | ||||
|     const { rows: books, count } = await Database.models.book.findAndCountAll({ | ||||
|     const { rows: books, count } = await Database.bookModel.findAndCountAll({ | ||||
|       where: [ | ||||
|         { | ||||
|           '$mediaProgresses.isFinished$': { | ||||
| @ -816,32 +817,32 @@ module.exports = { | ||||
|       replacements: userPermissionBookWhere.replacements, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.libraryItem, | ||||
|           model: Database.libraryItemModel, | ||||
|           where: { | ||||
|             libraryId | ||||
|           }, | ||||
|           include: libraryItemIncludes | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.mediaProgress, | ||||
|           model: Database.mediaProgressModel, | ||||
|           where: { | ||||
|             userId: user.id | ||||
|           }, | ||||
|           required: false | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.bookAuthor, | ||||
|           model: Database.bookAuthorModel, | ||||
|           attributes: ['authorId'], | ||||
|           include: { | ||||
|             model: Database.models.author | ||||
|             model: Database.authorModel | ||||
|           }, | ||||
|           separate: true | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.bookSeries, | ||||
|           model: Database.bookSeriesModel, | ||||
|           attributes: ['seriesId', 'sequence'], | ||||
|           include: { | ||||
|             model: Database.models.series | ||||
|             model: Database.seriesModel | ||||
|           }, | ||||
|           separate: true | ||||
|         } | ||||
| @ -883,10 +884,10 @@ module.exports = { | ||||
|       return [] | ||||
|     } | ||||
| 
 | ||||
|     const books = await Database.models.book.findAll({ | ||||
|     const books = await Database.bookModel.findAll({ | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.libraryItem, | ||||
|           model: Database.libraryItemModel, | ||||
|           where: { | ||||
|             id: { | ||||
|               [Sequelize.Op.in]: collection.books | ||||
| @ -894,13 +895,13 @@ module.exports = { | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.author, | ||||
|           model: Database.authorModel, | ||||
|           through: { | ||||
|             attributes: [] | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           model: Database.models.series, | ||||
|           model: Database.seriesModel, | ||||
|           through: { | ||||
|             attributes: ['sequence'] | ||||
|           } | ||||
| @ -918,12 +919,260 @@ module.exports = { | ||||
| 
 | ||||
|   /** | ||||
|    * Get library items for series | ||||
|    * @param {oldSeries} oldSeries  | ||||
|    * @param {[oldUser]} oldUser  | ||||
|    * @returns {Promise<oldLibraryItem[]>} | ||||
|    * @param {import('../../objects/entities/Series')} oldSeries  | ||||
|    * @param {import('../../objects/user/User')} [oldUser]  | ||||
|    * @returns {Promise<import('../../objects/LibraryItem')[]>} | ||||
|    */ | ||||
|   async getLibraryItemsForSeries(oldSeries, oldUser) { | ||||
|     const { libraryItems } = await this.getFilteredLibraryItems(oldSeries.libraryId, oldUser, 'series', oldSeries.id, null, null, false, [], null, null) | ||||
|     return libraryItems.map(li => Database.models.libraryItem.getOldLibraryItem(li)) | ||||
|     return libraryItems.map(li => Database.libraryItemModel.getOldLibraryItem(li)) | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Search books, authors, series | ||||
|    * @param {import('../../objects/user/User')} oldUser | ||||
|    * @param {import('../../objects/Library')} oldLibrary  | ||||
|    * @param {string} query  | ||||
|    * @param {number} limit  | ||||
|    * @param {number} offset  | ||||
|    * @returns {{book:object[], narrators:object[], authors:object[], tags:object[], series:object[]}} | ||||
|    */ | ||||
|   async search(oldUser, oldLibrary, query, limit, offset) { | ||||
|     const userPermissionBookWhere = this.getUserPermissionBookWhereQuery(oldUser) | ||||
| 
 | ||||
|     // Search title, subtitle, asin, isbn
 | ||||
|     const books = await Database.bookModel.findAll({ | ||||
|       where: [ | ||||
|         { | ||||
|           [Sequelize.Op.or]: [ | ||||
|             { | ||||
|               title: { | ||||
|                 [Sequelize.Op.substring]: query | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               subtitle: { | ||||
|                 [Sequelize.Op.substring]: query | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               asin: { | ||||
|                 [Sequelize.Op.substring]: query | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               isbn: { | ||||
|                 [Sequelize.Op.substring]: query | ||||
|               } | ||||
|             } | ||||
|           ] | ||||
|         }, | ||||
|         ...userPermissionBookWhere.bookWhere | ||||
|       ], | ||||
|       replacements: userPermissionBookWhere.replacements, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.libraryItemModel, | ||||
|           where: { | ||||
|             libraryId: oldLibrary.id | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           model: Database.bookSeriesModel, | ||||
|           include: { | ||||
|             model: Database.seriesModel | ||||
|           }, | ||||
|           separate: true | ||||
|         }, | ||||
|         { | ||||
|           model: Database.bookAuthorModel, | ||||
|           include: { | ||||
|             model: Database.authorModel | ||||
|           }, | ||||
|           separate: true | ||||
|         } | ||||
|       ], | ||||
|       subQuery: false, | ||||
|       distinct: true, | ||||
|       limit, | ||||
|       offset | ||||
|     }) | ||||
| 
 | ||||
|     const itemMatches = [] | ||||
| 
 | ||||
|     for (const book of books) { | ||||
|       const libraryItem = book.libraryItem | ||||
|       delete book.libraryItem | ||||
|       libraryItem.media = book | ||||
| 
 | ||||
|       let matchText = null | ||||
|       let matchKey = null | ||||
|       for (const key of ['title', 'subtitle', 'asin', 'isbn']) { | ||||
|         if (book[key]?.toLowerCase().includes(query)) { | ||||
|           matchText = book[key] | ||||
|           matchKey = key | ||||
|           break | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       if (matchKey) { | ||||
|         itemMatches.push({ | ||||
|           matchText, | ||||
|           matchKey, | ||||
|           libraryItem: Database.libraryItemModel.getOldLibraryItem(libraryItem).toJSONExpanded() | ||||
|         }) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     // Search narrators
 | ||||
|     const narratorMatches = [] | ||||
|     const [narratorResults] = await Database.sequelize.query(`SELECT value, count(*) AS numBooks FROM books b, libraryItems li, json_each(b.narrators) WHERE json_valid(b.narrators) AND json_each.value LIKE :query AND b.id = li.mediaId AND li.libraryId = :libraryId GROUP BY value LIMIT :limit OFFSET :offset;`, { | ||||
|       replacements: { | ||||
|         query: `%${query}%`, | ||||
|         libraryId: oldLibrary.id, | ||||
|         limit, | ||||
|         offset | ||||
|       }, | ||||
|       raw: true | ||||
|     }) | ||||
|     for (const row of narratorResults) { | ||||
|       narratorMatches.push({ | ||||
|         name: row.value, | ||||
|         numBooks: row.numBooks | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     // Search tags
 | ||||
|     const tagMatches = [] | ||||
|     const [tagResults] = await Database.sequelize.query(`SELECT value, count(*) AS numItems FROM books b, libraryItems li, json_each(b.tags) WHERE json_valid(b.tags) AND json_each.value LIKE :query AND b.id = li.mediaId AND li.libraryId = :libraryId GROUP BY value LIMIT :limit OFFSET :offset;`, { | ||||
|       replacements: { | ||||
|         query: `%${query}%`, | ||||
|         libraryId: oldLibrary.id, | ||||
|         limit, | ||||
|         offset | ||||
|       }, | ||||
|       raw: true | ||||
|     }) | ||||
|     for (const row of tagResults) { | ||||
|       tagMatches.push({ | ||||
|         name: row.value, | ||||
|         numItems: row.numItems | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     // Search series
 | ||||
|     const allSeries = await Database.seriesModel.findAll({ | ||||
|       where: { | ||||
|         name: { | ||||
|           [Sequelize.Op.substring]: query | ||||
|         }, | ||||
|         libraryId: oldLibrary.id | ||||
|       }, | ||||
|       replacements: userPermissionBookWhere.replacements, | ||||
|       include: { | ||||
|         separate: true, | ||||
|         model: Database.bookSeriesModel, | ||||
|         include: { | ||||
|           model: Database.bookModel, | ||||
|           where: userPermissionBookWhere.bookWhere, | ||||
|           include: { | ||||
|             model: Database.libraryItemModel | ||||
|           } | ||||
|         } | ||||
|       }, | ||||
|       subQuery: false, | ||||
|       distinct: true, | ||||
|       limit, | ||||
|       offset | ||||
|     }) | ||||
|     const seriesMatches = [] | ||||
|     for (const series of allSeries) { | ||||
|       const books = series.bookSeries.map((bs) => { | ||||
|         const libraryItem = bs.book.libraryItem | ||||
|         libraryItem.media = bs.book | ||||
|         return Database.libraryItemModel.getOldLibraryItem(libraryItem).toJSON() | ||||
|       }) | ||||
|       seriesMatches.push({ | ||||
|         series: series.getOldSeries().toJSON(), | ||||
|         books | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     // Search authors
 | ||||
|     const authorMatches = await authorFilters.search(oldLibrary.id, query, limit, offset) | ||||
| 
 | ||||
|     return { | ||||
|       book: itemMatches, | ||||
|       narrators: narratorMatches, | ||||
|       tags: tagMatches, | ||||
|       series: seriesMatches, | ||||
|       authors: authorMatches | ||||
|     } | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Genres with num books | ||||
|    * @param {string} libraryId  | ||||
|    * @returns {{genre:string, count:number}[]} | ||||
|    */ | ||||
|   async getGenresWithCount(libraryId) { | ||||
|     const genres = [] | ||||
|     const [genreResults] = await Database.sequelize.query(`SELECT value, count(*) AS numItems FROM books b, libraryItems li, json_each(b.genres) WHERE json_valid(b.genres) AND b.id = li.mediaId AND li.libraryId = :libraryId GROUP BY value ORDER BY numItems DESC;`, { | ||||
|       replacements: { | ||||
|         libraryId | ||||
|       }, | ||||
|       raw: true | ||||
|     }) | ||||
|     for (const row of genreResults) { | ||||
|       genres.push({ | ||||
|         genre: row.value, | ||||
|         count: row.numItems | ||||
|       }) | ||||
|     } | ||||
|     return genres | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Get stats for book library | ||||
|    * @param {string} libraryId  | ||||
|    * @returns {Promise<{ totalSize:number, totalDuration:number, numAudioFiles:number, totalItems:number}>} | ||||
|    */ | ||||
|   async getBookLibraryStats(libraryId) { | ||||
|     const [statResults] = await Database.sequelize.query(`SELECT SUM(li.size) AS totalSize, SUM(b.duration) AS totalDuration, SUM(json_array_length(b.audioFiles)) AS numAudioFiles, COUNT(*) AS totalItems FROM libraryItems li, books b WHERE b.id = li.mediaId AND li.libraryId = :libraryId;`, { | ||||
|       replacements: { | ||||
|         libraryId | ||||
|       } | ||||
|     }) | ||||
|     return statResults[0] | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Get longest books in library | ||||
|    * @param {string} libraryId  | ||||
|    * @param {number} limit  | ||||
|    * @returns {Promise<{ id:string, title:string, duration:number }[]>} | ||||
|    */ | ||||
|   async getLongestBooks(libraryId, limit) { | ||||
|     const books = await Database.bookModel.findAll({ | ||||
|       attributes: ['id', 'title', 'duration'], | ||||
|       include: { | ||||
|         model: Database.libraryItemModel, | ||||
|         attributes: ['id', 'libraryId'], | ||||
|         where: { | ||||
|           libraryId | ||||
|         } | ||||
|       }, | ||||
|       order: [ | ||||
|         ['duration', 'DESC'] | ||||
|       ], | ||||
|       limit | ||||
|     }) | ||||
|     return books.map(book => { | ||||
|       return { | ||||
|         id: book.libraryItem.id, | ||||
|         title: book.title, | ||||
|         duration: book.duration | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| @ -6,8 +6,8 @@ const Logger = require('../../Logger') | ||||
| module.exports = { | ||||
|   /** | ||||
|    * User permissions to restrict podcasts for explicit content & tags | ||||
|    * @param {oldUser} user  | ||||
|    * @returns {object} { podcastWhere:Sequelize.WhereOptions, replacements:string[] } | ||||
|    * @param {import('../../objects/user/User')} user  | ||||
|    * @returns {{ podcastWhere:Sequelize.WhereOptions, replacements:object }} | ||||
|    */ | ||||
|   getUserPermissionPodcastWhereQuery(user) { | ||||
|     const podcastWhere = [] | ||||
| @ -112,7 +112,7 @@ module.exports = { | ||||
|     const libraryItemIncludes = [] | ||||
|     if (includeRSSFeed) { | ||||
|       libraryItemIncludes.push({ | ||||
|         model: Database.models.feed, | ||||
|         model: Database.feedModel, | ||||
|         required: filterGroup === 'feed-open' | ||||
|       }) | ||||
|     } | ||||
| @ -146,7 +146,7 @@ module.exports = { | ||||
|     replacements = { ...replacements, ...userPermissionPodcastWhere.replacements } | ||||
|     podcastWhere.push(...userPermissionPodcastWhere.podcastWhere) | ||||
| 
 | ||||
|     const { rows: podcasts, count } = await Database.models.podcast.findAndCountAll({ | ||||
|     const { rows: podcasts, count } = await Database.podcastModel.findAndCountAll({ | ||||
|       where: podcastWhere, | ||||
|       replacements, | ||||
|       distinct: true, | ||||
| @ -158,7 +158,7 @@ module.exports = { | ||||
|       }, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.libraryItem, | ||||
|           model: Database.libraryItemModel, | ||||
|           required: true, | ||||
|           where: libraryItemWhere, | ||||
|           include: libraryItemIncludes | ||||
| @ -219,7 +219,7 @@ module.exports = { | ||||
|     } | ||||
|     if (filterGroup === 'progress') { | ||||
|       podcastEpisodeIncludes.push({ | ||||
|         model: Database.models.mediaProgress, | ||||
|         model: Database.mediaProgressModel, | ||||
|         where: { | ||||
|           userId: user.id | ||||
|         }, | ||||
| @ -255,16 +255,16 @@ module.exports = { | ||||
| 
 | ||||
|     const userPermissionPodcastWhere = this.getUserPermissionPodcastWhereQuery(user) | ||||
| 
 | ||||
|     const { rows: podcastEpisodes, count } = await Database.models.podcastEpisode.findAndCountAll({ | ||||
|     const { rows: podcastEpisodes, count } = await Database.podcastEpisodeModel.findAndCountAll({ | ||||
|       where: podcastEpisodeWhere, | ||||
|       replacements: userPermissionPodcastWhere.replacements, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.models.podcast, | ||||
|           model: Database.podcastModel, | ||||
|           where: userPermissionPodcastWhere.podcastWhere, | ||||
|           include: [ | ||||
|             { | ||||
|               model: Database.models.libraryItem, | ||||
|               model: Database.libraryItemModel, | ||||
|               where: libraryItemWhere | ||||
|             } | ||||
|           ] | ||||
| @ -283,7 +283,7 @@ module.exports = { | ||||
|       const podcast = ep.podcast.toJSON() | ||||
|       delete podcast.libraryItem | ||||
|       libraryItem.media = podcast | ||||
|       libraryItem.recentEpisode = ep.getOldPodcastEpisode(libraryItem.id) | ||||
|       libraryItem.recentEpisode = ep.getOldPodcastEpisode(libraryItem.id).toJSON() | ||||
|       return libraryItem | ||||
|     }) | ||||
| 
 | ||||
| @ -291,5 +291,239 @@ module.exports = { | ||||
|       libraryItems, | ||||
|       count | ||||
|     } | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Search podcasts | ||||
|    * @param {import('../../objects/user/User')} oldUser | ||||
|    * @param {import('../../objects/Library')} oldLibrary  | ||||
|    * @param {string} query  | ||||
|    * @param {number} limit  | ||||
|    * @param {number} offset  | ||||
|    * @returns {{podcast:object[], tags:object[]}} | ||||
|    */ | ||||
|   async search(oldUser, oldLibrary, query, limit, offset) { | ||||
|     const userPermissionPodcastWhere = this.getUserPermissionPodcastWhereQuery(user) | ||||
|     // Search title, author, itunesId, itunesArtistId
 | ||||
|     const podcasts = await Database.podcastModel.findAll({ | ||||
|       where: [ | ||||
|         { | ||||
|           [Sequelize.Op.or]: [ | ||||
|             { | ||||
|               title: { | ||||
|                 [Sequelize.Op.substring]: query | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               author: { | ||||
|                 [Sequelize.Op.substring]: query | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               itunesId: { | ||||
|                 [Sequelize.Op.substring]: query | ||||
|               } | ||||
|             }, | ||||
|             { | ||||
|               itunesArtistId: { | ||||
|                 [Sequelize.Op.substring]: query | ||||
|               } | ||||
|             } | ||||
|           ] | ||||
|         }, | ||||
|         ...userPermissionPodcastWhere.podcastWhere | ||||
|       ], | ||||
|       replacements: userPermissionPodcastWhere.replacements, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.libraryItemModel, | ||||
|           where: { | ||||
|             libraryId: oldLibrary.id | ||||
|           } | ||||
|         } | ||||
|       ], | ||||
|       subQuery: false, | ||||
|       distinct: true, | ||||
|       limit, | ||||
|       offset | ||||
|     }) | ||||
| 
 | ||||
|     const itemMatches = [] | ||||
| 
 | ||||
|     for (const podcast of podcasts) { | ||||
|       const libraryItem = podcast.libraryItem | ||||
|       delete podcast.libraryItem | ||||
|       libraryItem.media = podcast | ||||
| 
 | ||||
|       let matchText = null | ||||
|       let matchKey = null | ||||
|       for (const key of ['title', 'author', 'itunesId', 'itunesArtistId']) { | ||||
|         if (podcast[key]?.toLowerCase().includes(query)) { | ||||
|           matchText = podcast[key] | ||||
|           matchKey = key | ||||
|           break | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       if (matchKey) { | ||||
|         itemMatches.push({ | ||||
|           matchText, | ||||
|           matchKey, | ||||
|           libraryItem: Database.libraryItemModel.getOldLibraryItem(libraryItem).toJSONExpanded() | ||||
|         }) | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     // Search tags
 | ||||
|     const tagMatches = [] | ||||
|     const [tagResults] = await Database.sequelize.query(`SELECT value, count(*) AS numItems FROM podcasts p, libraryItems li, json_each(p.tags) WHERE json_valid(p.tags) AND json_each.value LIKE :query AND p.id = li.mediaId AND li.libraryId = :libraryId GROUP BY value LIMIT :limit OFFSET :offset;`, { | ||||
|       replacements: { | ||||
|         query: `%${query}%`, | ||||
|         libraryId: oldLibrary.id, | ||||
|         limit, | ||||
|         offset | ||||
|       }, | ||||
|       raw: true | ||||
|     }) | ||||
|     for (const row of tagResults) { | ||||
|       tagMatches.push({ | ||||
|         name: row.value, | ||||
|         numItems: row.numItems | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     return { | ||||
|       podcast: itemMatches, | ||||
|       tags: tagMatches | ||||
|     } | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Most recent podcast episodes not finished | ||||
|    * @param {import('../../objects/user/User')} oldUser  | ||||
|    * @param {import('../../objects/Library')} oldLibrary  | ||||
|    * @param {number} limit  | ||||
|    * @param {number} offset  | ||||
|    * @returns {Promise<object[]>} | ||||
|    */ | ||||
|   async getRecentEpisodes(oldUser, oldLibrary, limit, offset) { | ||||
|     const userPermissionPodcastWhere = this.getUserPermissionPodcastWhereQuery(oldUser) | ||||
| 
 | ||||
|     const episodes = await Database.podcastEpisodeModel.findAll({ | ||||
|       where: { | ||||
|         '$mediaProgresses.isFinished$': { | ||||
|           [Sequelize.Op.or]: [null, false] | ||||
|         } | ||||
|       }, | ||||
|       replacements: userPermissionPodcastWhere.replacements, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.podcastModel, | ||||
|           where: userPermissionPodcastWhere.podcastWhere, | ||||
|           required: true, | ||||
|           include: { | ||||
|             model: Database.libraryItemModel, | ||||
|             where: { | ||||
|               libraryId: oldLibrary.id | ||||
|             } | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           model: Database.mediaProgressModel, | ||||
|           where: { | ||||
|             userId: oldUser.id | ||||
|           }, | ||||
|           required: false | ||||
|         } | ||||
|       ], | ||||
|       order: [ | ||||
|         ['publishedAt', 'DESC'] | ||||
|       ], | ||||
|       subQuery: false, | ||||
|       limit, | ||||
|       offset | ||||
|     }) | ||||
| 
 | ||||
|     const episodeResults = episodes.map((ep) => { | ||||
|       const libraryItem = ep.podcast.libraryItem | ||||
|       libraryItem.media = ep.podcast | ||||
|       const oldPodcast = Database.podcastModel.getOldPodcast(libraryItem) | ||||
|       const oldPodcastEpisode = ep.getOldPodcastEpisode(libraryItem.id).toJSONExpanded() | ||||
|       oldPodcastEpisode.podcast = oldPodcast | ||||
|       oldPodcastEpisode.libraryId = libraryItem.libraryId | ||||
|       return oldPodcastEpisode | ||||
|     }) | ||||
| 
 | ||||
|     return episodeResults | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Get stats for podcast library | ||||
|    * @param {string} libraryId  | ||||
|    * @returns {Promise<{ totalSize:number, totalDuration:number, numAudioFiles:number, totalItems:number}>} | ||||
|    */ | ||||
|   async getPodcastLibraryStats(libraryId) { | ||||
|     const [statResults] = await Database.sequelize.query(`SELECT SUM(json_extract(pe.audioFile, '$.duration')) AS totalDuration, SUM(li.size) AS totalSize, COUNT(DISTINCT(li.id)) AS totalItems, COUNT(pe.id) AS numAudioFiles FROM libraryItems li, podcasts p LEFT OUTER JOIN podcastEpisodes pe ON pe.podcastId = p.id WHERE p.id = li.mediaId AND li.libraryId = :libraryId;`, { | ||||
|       replacements: { | ||||
|         libraryId | ||||
|       } | ||||
|     }) | ||||
|     return statResults[0] | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Genres with num podcasts | ||||
|    * @param {string} libraryId  | ||||
|    * @returns {{genre:string, count:number}[]} | ||||
|    */ | ||||
|   async getGenresWithCount(libraryId) { | ||||
|     const genres = [] | ||||
|     const [genreResults] = await Database.sequelize.query(`SELECT value, count(*) AS numItems FROM podcasts p, libraryItems li, json_each(p.genres) WHERE json_valid(p.genres) AND p.id = li.mediaId AND li.libraryId = :libraryId GROUP BY value ORDER BY numItems DESC;`, { | ||||
|       replacements: { | ||||
|         libraryId | ||||
|       }, | ||||
|       raw: true | ||||
|     }) | ||||
|     for (const row of genreResults) { | ||||
|       genres.push({ | ||||
|         genre: row.value, | ||||
|         count: row.numItems | ||||
|       }) | ||||
|     } | ||||
|     return genres | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Get longest podcasts in library | ||||
|    * @param {string} libraryId  | ||||
|    * @param {number} limit  | ||||
|    * @returns {Promise<{ id:string, title:string, duration:number }[]>} | ||||
|    */ | ||||
|   async getLongestPodcasts(libraryId, limit) { | ||||
|     const podcasts = await Database.podcastModel.findAll({ | ||||
|       attributes: [ | ||||
|         'id', | ||||
|         'title', | ||||
|         [Sequelize.literal(`(SELECT SUM(json_extract(pe.audioFile, '$.duration')) FROM podcastEpisodes pe WHERE pe.podcastId = podcast.id)`), 'duration'] | ||||
|       ], | ||||
|       include: { | ||||
|         model: Database.libraryItemModel, | ||||
|         attributes: ['id', 'libraryId'], | ||||
|         where: { | ||||
|           libraryId | ||||
|         } | ||||
|       }, | ||||
|       order: [ | ||||
|         ['duration', 'DESC'] | ||||
|       ], | ||||
|       limit | ||||
|     }) | ||||
|     return podcasts.map(podcast => { | ||||
|       return { | ||||
|         id: podcast.libraryItem.id, | ||||
|         title: podcast.title, | ||||
|         duration: podcast.dataValues.duration | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| } | ||||
							
								
								
									
										206
									
								
								server/utils/queries/seriesFilters.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										206
									
								
								server/utils/queries/seriesFilters.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,206 @@ | ||||
| const Sequelize = require('sequelize') | ||||
| const Logger = require('../../Logger') | ||||
| const Database = require('../../Database') | ||||
| const libraryItemsBookFilters = require('./libraryItemsBookFilters') | ||||
| 
 | ||||
| module.exports = { | ||||
|   decode(text) { | ||||
|     return Buffer.from(decodeURIComponent(text), 'base64').toString() | ||||
|   }, | ||||
| 
 | ||||
|   /** | ||||
|    * Get series filtered and sorted | ||||
|    *  | ||||
|    * @param {import('../../objects/Library')} library  | ||||
|    * @param {import('../../objects/user/User')} user  | ||||
|    * @param {string} filterBy  | ||||
|    * @param {string} sortBy  | ||||
|    * @param {boolean} sortDesc  | ||||
|    * @param {string[]} include  | ||||
|    * @param {number} limit  | ||||
|    * @param {number} offset  | ||||
|    * @returns {Promise<{ series:object[], count:number }>} | ||||
|    */ | ||||
|   async getFilteredSeries(library, user, filterBy, sortBy, sortDesc, include, limit, offset) { | ||||
|     let filterValue = null | ||||
|     let filterGroup = null | ||||
|     if (filterBy) { | ||||
|       const searchGroups = ['genres', 'tags', 'authors', 'progress', 'narrators', 'publishers', 'languages'] | ||||
|       const group = searchGroups.find(_group => filterBy.startsWith(_group + '.')) | ||||
|       filterGroup = group || filterBy | ||||
|       filterValue = group ? this.decode(filterBy.replace(`${group}.`, '')) : null | ||||
|     } | ||||
| 
 | ||||
|     const seriesIncludes = [] | ||||
|     if (include.includes('rssfeed')) { | ||||
|       seriesIncludes.push({ | ||||
|         model: Database.feedModel | ||||
|       }) | ||||
|     } | ||||
| 
 | ||||
|     const userPermissionBookWhere = libraryItemsBookFilters.getUserPermissionBookWhereQuery(user) | ||||
| 
 | ||||
|     const seriesWhere = [ | ||||
|       { | ||||
|         libraryId: library.id | ||||
|       } | ||||
|     ] | ||||
| 
 | ||||
|     // Handle library setting to hide single book series
 | ||||
|     // TODO: Merge with existing query
 | ||||
|     if (library.settings.hideSingleBookSeries) { | ||||
|       seriesWhere.push(Sequelize.where(Sequelize.literal(`(SELECT count(*) FROM books b, bookSeries bs WHERE bs.seriesId = series.id AND bs.bookId = b.id)`), { | ||||
|         [Sequelize.Op.gt]: 1 | ||||
|       })) | ||||
|     } | ||||
| 
 | ||||
|     // Handle filters
 | ||||
|     // TODO: Simplify and break-out
 | ||||
|     let attrQuery = null | ||||
|     if (['genres', 'tags', 'narrators'].includes(filterGroup)) { | ||||
|       attrQuery = `SELECT count(*) FROM books b, bookSeries bs WHERE bs.seriesId = series.id AND bs.bookId = b.id AND (SELECT count(*) FROM json_each(b.${filterGroup}) WHERE json_valid(b.${filterGroup}) AND json_each.value = :filterValue) > 0` | ||||
|       userPermissionBookWhere.replacements.filterValue = filterValue | ||||
|     } else if (filterGroup === 'authors') { | ||||
|       attrQuery = 'SELECT count(*) FROM books b, bookSeries bs, bookAuthors ba WHERE bs.seriesId = series.id AND bs.bookId = b.id AND ba.bookId = b.id AND ba.authorId = :filterValue' | ||||
|       userPermissionBookWhere.replacements.filterValue = filterValue | ||||
|     } else if (filterGroup === 'publishers') { | ||||
|       attrQuery = 'SELECT count(*) FROM books b, bookSeries bs WHERE bs.seriesId = series.id AND bs.bookId = b.id AND b.publisher = :filterValue' | ||||
|       userPermissionBookWhere.replacements.filterValue = filterValue | ||||
|     } else if (filterGroup === 'languages') { | ||||
|       attrQuery = 'SELECT count(*) FROM books b, bookSeries bs WHERE bs.seriesId = series.id AND bs.bookId = b.id AND b.language = :filterValue' | ||||
|       userPermissionBookWhere.replacements.filterValue = filterValue | ||||
|     } else if (filterGroup === 'progress') { | ||||
|       if (filterValue === 'not-finished') { | ||||
|         attrQuery = 'SELECT count(*) FROM books b, bookSeries bs LEFT OUTER JOIN mediaProgresses mp ON mp.mediaItemId = b.id WHERE bs.seriesId = series.id AND bs.bookId = b.id AND (mp.isFinished IS NULL OR mp.isFinished = 0)' | ||||
|       } else if (filterValue === 'finished') { | ||||
|         const progQuery = 'SELECT count(*) FROM books b, bookSeries bs LEFT OUTER JOIN mediaProgresses mp ON mp.mediaItemId = b.id WHERE bs.seriesId = series.id AND bs.bookId = b.id AND (mp.isFinished IS NULL OR mp.isFinished = 0)' | ||||
|         seriesWhere.push(Sequelize.where(Sequelize.literal(`(${progQuery})`), 0)) | ||||
|       } else if (filterValue === 'not-started') { | ||||
|         const progQuery = 'SELECT count(*) FROM books b, bookSeries bs LEFT OUTER JOIN mediaProgresses mp ON mp.mediaItemId = b.id WHERE bs.seriesId = series.id AND bs.bookId = b.id AND (mp.isFinished = 1 OR mp.currentTime > 0)' | ||||
|         seriesWhere.push(Sequelize.where(Sequelize.literal(`(${progQuery})`), 0)) | ||||
|       } else if (filterValue === 'in-progress') { | ||||
|         attrQuery = 'SELECT count(*) FROM books b, bookSeries bs LEFT OUTER JOIN mediaProgresses mp ON mp.mediaItemId = b.id WHERE bs.seriesId = series.id AND bs.bookId = b.id AND (mp.currentTime > 0 OR mp.ebookProgress > 0) AND mp.isFinished = 0' | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     // Handle user permissions to only include series with at least 1 book
 | ||||
|     // TODO: Simplify to a single query
 | ||||
|     if (userPermissionBookWhere.bookWhere.length) { | ||||
|       if (!attrQuery) attrQuery = 'SELECT count(*) FROM books b, bookSeries bs WHERE bs.seriesId = series.id AND bs.bookId = b.id' | ||||
| 
 | ||||
|       if (!user.canAccessExplicitContent) { | ||||
|         attrQuery += ' AND b.explicit = 0' | ||||
|       } | ||||
|       if (!user.permissions.accessAllTags && user.itemTagsSelected.length) { | ||||
|         if (user.permissions.selectedTagsNotAccessible) { | ||||
|           attrQuery += ' AND (SELECT count(*) FROM json_each(tags) WHERE json_valid(tags) AND json_each.value IN (:userTagsSelected)) = 0' | ||||
|         } else { | ||||
|           attrQuery += ' AND (SELECT count(*) FROM json_each(tags) WHERE json_valid(tags) AND json_each.value IN (:userTagsSelected)) > 0' | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     if (attrQuery) { | ||||
|       seriesWhere.push(Sequelize.where(Sequelize.literal(`(${attrQuery})`), { | ||||
|         [Sequelize.Op.gt]: 0 | ||||
|       })) | ||||
|     } | ||||
| 
 | ||||
|     const order = [] | ||||
|     let seriesAttributes = { | ||||
|       include: [] | ||||
|     } | ||||
| 
 | ||||
|     // Handle sort order
 | ||||
|     const dir = sortDesc ? 'DESC' : 'ASC' | ||||
|     if (sortBy === 'numBooks') { | ||||
|       seriesAttributes.include.push([Sequelize.literal('(SELECT count(*) FROM bookSeries bs WHERE bs.seriesId = series.id)'), 'numBooks']) | ||||
|       order.push(['numBooks', dir]) | ||||
|     } else if (sortBy === 'addedAt') { | ||||
|       order.push(['createdAt', dir]) | ||||
|     } else if (sortBy === 'name') { | ||||
|       if (global.ServerSettings.sortingIgnorePrefix) { | ||||
|         order.push([Sequelize.literal('nameIgnorePrefix COLLATE NOCASE'), dir]) | ||||
|       } else { | ||||
|         order.push([Sequelize.literal('`series`.`name` COLLATE NOCASE'), dir]) | ||||
|       } | ||||
|     } else if (sortBy === 'totalDuration') { | ||||
|       seriesAttributes.include.push([Sequelize.literal('(SELECT SUM(b.duration) FROM books b, bookSeries bs WHERE bs.seriesId = series.id AND b.id = bs.bookId)'), 'totalDuration']) | ||||
|       order.push(['totalDuration', dir]) | ||||
|     } else if (sortBy === 'lastBookAdded') { | ||||
|       seriesAttributes.include.push([Sequelize.literal('(SELECT MAX(b.createdAt) FROM books b, bookSeries bs WHERE bs.seriesId = series.id AND b.id = bs.bookId)'), 'mostRecentBookAdded']) | ||||
|       order.push(['mostRecentBookAdded', dir]) | ||||
|     } else if (sortBy === 'lastBookUpdated') { | ||||
|       seriesAttributes.include.push([Sequelize.literal('(SELECT MAX(b.updatedAt) FROM books b, bookSeries bs WHERE bs.seriesId = series.id AND b.id = bs.bookId)'), 'mostRecentBookUpdated']) | ||||
|       order.push(['mostRecentBookUpdated', dir]) | ||||
|     } | ||||
| 
 | ||||
|     const { rows: series, count } = await Database.seriesModel.findAndCountAll({ | ||||
|       where: seriesWhere, | ||||
|       limit, | ||||
|       offset, | ||||
|       distinct: true, | ||||
|       subQuery: false, | ||||
|       benchmark: true, | ||||
|       logging: (sql, timeMs) => { | ||||
|         console.log(`[Query] Series filter/sort. Elapsed ${timeMs}ms`) | ||||
|       }, | ||||
|       attributes: seriesAttributes, | ||||
|       replacements: userPermissionBookWhere.replacements, | ||||
|       include: [ | ||||
|         { | ||||
|           model: Database.bookSeriesModel, | ||||
|           include: { | ||||
|             model: Database.bookModel, | ||||
|             where: userPermissionBookWhere.bookWhere, | ||||
|             include: [ | ||||
|               { | ||||
|                 model: Database.libraryItemModel | ||||
|               } | ||||
|             ] | ||||
|           }, | ||||
|           separate: true | ||||
|         }, | ||||
|         ...seriesIncludes | ||||
|       ], | ||||
|       order | ||||
|     }) | ||||
| 
 | ||||
|     // Map series to old series
 | ||||
|     const allOldSeries = [] | ||||
|     for (const s of series) { | ||||
|       const oldSeries = s.getOldSeries().toJSON() | ||||
| 
 | ||||
|       if (s.dataValues.totalDuration) { | ||||
|         oldSeries.totalDuration = s.dataValues.totalDuration | ||||
|       } | ||||
| 
 | ||||
|       if (s.feeds?.length) { | ||||
|         oldSeries.rssFeed = Database.feedModel.getOldFeed(s.feeds[0]).toJSONMinified() | ||||
|       } | ||||
| 
 | ||||
|       // TODO: Sort books by sequence in query
 | ||||
|       s.bookSeries.sort((a, b) => { | ||||
|         if (!a.sequence) return 1 | ||||
|         if (!b.sequence) return -1 | ||||
|         return a.sequence.localeCompare(b.sequence, undefined, { | ||||
|           numeric: true, | ||||
|           sensitivity: 'base' | ||||
|         }) | ||||
|       }) | ||||
|       oldSeries.books = s.bookSeries.map(bs => { | ||||
|         const libraryItem = bs.book.libraryItem.toJSON() | ||||
|         delete bs.book.libraryItem | ||||
|         libraryItem.media = bs.book | ||||
|         const oldLibraryItem = Database.libraryItemModel.getOldLibraryItem(libraryItem).toJSONMinified() | ||||
|         return oldLibraryItem | ||||
|       }) | ||||
|       allOldSeries.push(oldSeries) | ||||
|     } | ||||
| 
 | ||||
|     return { | ||||
|       series: allOldSeries, | ||||
|       count | ||||
|     } | ||||
|   } | ||||
| } | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user