From 5201625d38aa2dcda85fb3df3b270ae9adb3c021 Mon Sep 17 00:00:00 2001 From: advplyr Date: Wed, 1 Jan 2025 11:32:39 -0600 Subject: [PATCH] Fix FeedEpisodes using a new ID when updating #3757 --- server/managers/RssFeedManager.js | 20 ++++++++++--- server/models/Feed.js | 26 +++++++++++----- server/models/FeedEpisode.js | 49 +++++++++++++++++++++++-------- 3 files changed, 71 insertions(+), 24 deletions(-) diff --git a/server/managers/RssFeedManager.js b/server/managers/RssFeedManager.js index cedf0dfb..abfa445f 100644 --- a/server/managers/RssFeedManager.js +++ b/server/managers/RssFeedManager.js @@ -98,11 +98,22 @@ class RssFeedManager { podcastId: feed.entity.mediaId }, attributes: ['id', 'updatedAt'], - order: [['createdAt', 'DESC']] + order: [['updatedAt', 'DESC']] }) + if (mostRecentPodcastEpisode && mostRecentPodcastEpisode.updatedAt > newEntityUpdatedAt) { newEntityUpdatedAt = mostRecentPodcastEpisode.updatedAt } + } else { + const book = await Database.bookModel.findOne({ + where: { + id: feed.entity.mediaId + }, + attributes: ['id', 'updatedAt'] + }) + if (book && book.updatedAt > newEntityUpdatedAt) { + newEntityUpdatedAt = book.updatedAt + } } return newEntityUpdatedAt > feed.entityUpdatedAt @@ -111,7 +122,7 @@ class RssFeedManager { attributes: ['id', 'updatedAt'], include: { model: Database.bookModel, - attributes: ['id'], + attributes: ['id', 'updatedAt'], through: { attributes: [] }, @@ -125,8 +136,9 @@ class RssFeedManager { let newEntityUpdatedAt = feed.entity.updatedAt const mostRecentItemUpdatedAt = feed.entity.books.reduce((mostRecent, book) => { - if (book.libraryItem.updatedAt > mostRecent) { - return book.libraryItem.updatedAt + let updatedAt = book.libraryItem.updatedAt > book.updatedAt ? book.libraryItem.updatedAt : book.updatedAt + if (updatedAt > mostRecent) { + return updatedAt } return mostRecent }, 0) diff --git a/server/models/Feed.js b/server/models/Feed.js index d8f8553c..a6ccff22 100644 --- a/server/models/Feed.js +++ b/server/models/Feed.js @@ -107,6 +107,9 @@ class Feed extends Model { entityUpdatedAt = libraryItem.media.podcastEpisodes.reduce((mostRecent, episode) => { return episode.updatedAt > mostRecent ? episode.updatedAt : mostRecent }, entityUpdatedAt) + } else if (libraryItem.media.updatedAt > entityUpdatedAt) { + // Book feeds will use Book.updatedAt if more recent + entityUpdatedAt = libraryItem.media.updatedAt } const feedObj = { @@ -472,6 +475,8 @@ class Feed extends Model { /** @type {typeof import('./FeedEpisode')} */ const feedEpisodeModel = this.sequelize.models.feedEpisode + this.feedEpisodes = await this.getFeedEpisodes() + let feedObj = null let feedEpisodeCreateFunc = null let feedEpisodeCreateFuncEntity = null @@ -516,17 +521,24 @@ class Feed extends Model { try { const updatedFeed = await this.update(feedObj, { transaction }) - // Remove existing feed episodes - await feedEpisodeModel.destroy({ - where: { - feedId: this.id - }, - transaction - }) + const existingFeedEpisodeIds = this.feedEpisodes.map((ep) => ep.id) // Create new feed episodes updatedFeed.feedEpisodes = await feedEpisodeCreateFunc(feedEpisodeCreateFuncEntity, updatedFeed, this.slug, transaction) + const newFeedEpisodeIds = updatedFeed.feedEpisodes.map((ep) => ep.id) + const feedEpisodeIdsToRemove = existingFeedEpisodeIds.filter((epid) => !newFeedEpisodeIds.includes(epid)) + + if (feedEpisodeIdsToRemove.length) { + Logger.info(`[Feed] Removing ${feedEpisodeIdsToRemove.length} episodes from feed ${this.id}`) + await feedEpisodeModel.destroy({ + where: { + id: feedEpisodeIdsToRemove + }, + transaction + }) + } + await transaction.commit() return updatedFeed diff --git a/server/models/FeedEpisode.js b/server/models/FeedEpisode.js index 0d1a3a48..5825dd4e 100644 --- a/server/models/FeedEpisode.js +++ b/server/models/FeedEpisode.js @@ -53,9 +53,10 @@ class FeedEpisode extends Model { * @param {import('./Feed')} feed * @param {string} slug * @param {import('./PodcastEpisode')} episode + * @param {string} [existingEpisodeId] */ - static getFeedEpisodeObjFromPodcastEpisode(libraryItemExpanded, feed, slug, episode) { - const episodeId = uuidv4() + static getFeedEpisodeObjFromPodcastEpisode(libraryItemExpanded, feed, slug, episode, existingEpisodeId = null) { + const episodeId = existingEpisodeId || uuidv4() return { id: episodeId, title: episode.title, @@ -94,11 +95,18 @@ class FeedEpisode extends Model { libraryItemExpanded.media.podcastEpisodes.sort((a, b) => new Date(a.pubDate) - new Date(b.pubDate)) } + let numExisting = 0 for (const episode of libraryItemExpanded.media.podcastEpisodes) { - feedEpisodeObjs.push(this.getFeedEpisodeObjFromPodcastEpisode(libraryItemExpanded, feed, slug, episode)) + // Check for existing episode by filepath + const existingEpisode = feed.feedEpisodes?.find((feedEpisode) => { + return feedEpisode.filePath === episode.audioFile.metadata.path + }) + numExisting = existingEpisode ? numExisting + 1 : numExisting + + feedEpisodeObjs.push(this.getFeedEpisodeObjFromPodcastEpisode(libraryItemExpanded, feed, slug, episode, existingEpisode?.id)) } - Logger.info(`[FeedEpisode] Creating ${feedEpisodeObjs.length} episodes for feed ${feed.id}`) - return this.bulkCreate(feedEpisodeObjs, { transaction }) + Logger.info(`[FeedEpisode] Upserting ${feedEpisodeObjs.length} episodes for feed ${feed.id} (${numExisting} existing)`) + return this.bulkCreate(feedEpisodeObjs, { transaction, updateOnDuplicate: ['title', 'author', 'description', 'siteURL', 'enclosureURL', 'enclosureType', 'enclosureSize', 'pubDate', 'season', 'episode', 'episodeType', 'duration', 'filePath', 'explicit'] }) } /** @@ -127,11 +135,12 @@ class FeedEpisode extends Model { * @param {string} slug * @param {import('./Book').AudioFileObject} audioTrack * @param {boolean} useChapterTitles + * @param {string} [existingEpisodeId] */ - static getFeedEpisodeObjFromAudiobookTrack(book, pubDateStart, feed, slug, audioTrack, useChapterTitles) { + static getFeedEpisodeObjFromAudiobookTrack(book, pubDateStart, feed, slug, audioTrack, useChapterTitles, existingEpisodeId = null) { // Example: Fri, 04 Feb 2015 00:00:00 GMT let timeOffset = isNaN(audioTrack.index) ? 0 : Number(audioTrack.index) * 1000 // Offset pubdate to ensure correct order - let episodeId = uuidv4() + let episodeId = existingEpisodeId || uuidv4() // e.g. Track 1 will have a pub date before Track 2 const audiobookPubDate = date.format(new Date(pubDateStart.valueOf() + timeOffset), 'ddd, DD MMM YYYY HH:mm:ss [GMT]') @@ -179,11 +188,18 @@ class FeedEpisode extends Model { const useChapterTitles = this.checkUseChapterTitlesForEpisodes(libraryItemExpanded.media) const feedEpisodeObjs = [] + let numExisting = 0 for (const track of libraryItemExpanded.media.trackList) { - feedEpisodeObjs.push(this.getFeedEpisodeObjFromAudiobookTrack(libraryItemExpanded.media, libraryItemExpanded.createdAt, feed, slug, track, useChapterTitles)) + // Check for existing episode by filepath + const existingEpisode = feed.feedEpisodes?.find((episode) => { + return episode.filePath === track.metadata.path + }) + numExisting = existingEpisode ? numExisting + 1 : numExisting + + feedEpisodeObjs.push(this.getFeedEpisodeObjFromAudiobookTrack(libraryItemExpanded.media, libraryItemExpanded.createdAt, feed, slug, track, useChapterTitles, existingEpisode?.id)) } - Logger.info(`[FeedEpisode] Creating ${feedEpisodeObjs.length} episodes for feed ${feed.id}`) - return this.bulkCreate(feedEpisodeObjs, { transaction }) + Logger.info(`[FeedEpisode] Upserting ${feedEpisodeObjs.length} episodes for feed ${feed.id} (${numExisting} existing)`) + return this.bulkCreate(feedEpisodeObjs, { transaction, updateOnDuplicate: ['title', 'author', 'description', 'siteURL', 'enclosureURL', 'enclosureType', 'enclosureSize', 'pubDate', 'season', 'episode', 'episodeType', 'duration', 'filePath', 'explicit'] }) } /** @@ -200,14 +216,21 @@ class FeedEpisode extends Model { }).libraryItem.createdAt const feedEpisodeObjs = [] + let numExisting = 0 for (const book of books) { const useChapterTitles = this.checkUseChapterTitlesForEpisodes(book) for (const track of book.trackList) { - feedEpisodeObjs.push(this.getFeedEpisodeObjFromAudiobookTrack(book, earliestLibraryItemCreatedAt, feed, slug, track, useChapterTitles)) + // Check for existing episode by filepath + const existingEpisode = feed.feedEpisodes?.find((episode) => { + return episode.filePath === track.metadata.path + }) + numExisting = existingEpisode ? numExisting + 1 : numExisting + + feedEpisodeObjs.push(this.getFeedEpisodeObjFromAudiobookTrack(book, earliestLibraryItemCreatedAt, feed, slug, track, useChapterTitles, existingEpisode?.id)) } } - Logger.info(`[FeedEpisode] Creating ${feedEpisodeObjs.length} episodes for feed ${feed.id}`) - return this.bulkCreate(feedEpisodeObjs, { transaction }) + Logger.info(`[FeedEpisode] Upserting ${feedEpisodeObjs.length} episodes for feed ${feed.id} (${numExisting} existing)`) + return this.bulkCreate(feedEpisodeObjs, { transaction, updateOnDuplicate: ['title', 'author', 'description', 'siteURL', 'enclosureURL', 'enclosureType', 'enclosureSize', 'pubDate', 'season', 'episode', 'episodeType', 'duration', 'filePath', 'explicit'] }) } /**