2024-12-14 23:55:56 +01:00
|
|
|
const Path = require('path')
|
2023-07-05 01:14:44 +02:00
|
|
|
const { DataTypes, Model } = require('sequelize')
|
2024-12-15 23:56:59 +01:00
|
|
|
const Logger = require('../Logger')
|
|
|
|
|
|
|
|
const RSS = require('../libs/rss')
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2024-12-14 23:55:56 +01:00
|
|
|
/**
|
|
|
|
* @typedef FeedOptions
|
|
|
|
* @property {boolean} preventIndexing
|
|
|
|
* @property {string} ownerName
|
|
|
|
* @property {string} ownerEmail
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef FeedExpandedProperties
|
|
|
|
* @property {import('./FeedEpisode')} feedEpisodes
|
|
|
|
*
|
|
|
|
* @typedef {Feed & FeedExpandedProperties} FeedExpanded
|
|
|
|
*/
|
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
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
|
2024-12-14 23:55:56 +01:00
|
|
|
|
2024-12-15 23:56:59 +01:00
|
|
|
// Expanded properties
|
|
|
|
|
2024-12-14 23:55:56 +01:00
|
|
|
/** @type {import('./FeedEpisode')[]} - only set if expanded */
|
|
|
|
this.feedEpisodes
|
2023-08-16 01:03:43 +02:00
|
|
|
}
|
|
|
|
|
2024-12-15 19:37:01 +01:00
|
|
|
/**
|
|
|
|
* @param {string} feedId
|
|
|
|
* @returns {Promise<boolean>} - true if feed was removed
|
|
|
|
*/
|
|
|
|
static async removeById(feedId) {
|
|
|
|
return (
|
|
|
|
(await this.destroy({
|
|
|
|
where: {
|
|
|
|
id: feedId
|
|
|
|
}
|
|
|
|
})) > 0
|
|
|
|
)
|
2023-08-16 01:03:43 +02:00
|
|
|
}
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2024-12-14 23:55:56 +01:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} userId
|
|
|
|
* @param {import('./LibraryItem').LibraryItemExpanded} libraryItem
|
2024-12-15 17:53:31 +01:00
|
|
|
* @param {string} slug
|
2024-12-14 23:55:56 +01:00
|
|
|
* @param {string} serverAddress
|
2024-12-15 23:56:59 +01:00
|
|
|
* @param {FeedOptions} [feedOptions=null]
|
2024-12-14 23:55:56 +01:00
|
|
|
*
|
2024-12-15 23:56:59 +01:00
|
|
|
* @returns {Feed}
|
2024-12-14 23:55:56 +01:00
|
|
|
*/
|
2024-12-15 23:56:59 +01:00
|
|
|
static getFeedObjForLibraryItem(userId, libraryItem, slug, serverAddress, feedOptions = null) {
|
2024-12-14 23:55:56 +01:00
|
|
|
const media = libraryItem.media
|
|
|
|
|
2024-12-15 21:07:46 +01:00
|
|
|
let entityUpdatedAt = libraryItem.updatedAt
|
|
|
|
|
|
|
|
// Podcast feeds should use the most recent episode updatedAt if more recent
|
|
|
|
if (libraryItem.mediaType === 'podcast') {
|
|
|
|
entityUpdatedAt = libraryItem.media.podcastEpisodes.reduce((mostRecent, episode) => {
|
|
|
|
return episode.updatedAt > mostRecent ? episode.updatedAt : mostRecent
|
|
|
|
}, entityUpdatedAt)
|
|
|
|
}
|
|
|
|
|
2024-12-14 23:55:56 +01:00
|
|
|
const feedObj = {
|
|
|
|
slug,
|
|
|
|
entityType: 'libraryItem',
|
|
|
|
entityId: libraryItem.id,
|
2024-12-15 21:07:46 +01:00
|
|
|
entityUpdatedAt,
|
2024-12-14 23:55:56 +01:00
|
|
|
serverAddress,
|
|
|
|
feedURL: `/feed/${slug}`,
|
|
|
|
imageURL: media.coverPath ? `/feed/${slug}/cover${Path.extname(media.coverPath)}` : `/Logo.png`,
|
|
|
|
siteURL: `/item/${libraryItem.id}`,
|
|
|
|
title: media.title,
|
|
|
|
description: media.description,
|
|
|
|
author: libraryItem.mediaType === 'podcast' ? media.author : media.authorName,
|
|
|
|
podcastType: libraryItem.mediaType === 'podcast' ? media.podcastType : 'serial',
|
|
|
|
language: media.language,
|
|
|
|
explicit: media.explicit,
|
|
|
|
coverPath: media.coverPath,
|
|
|
|
userId
|
|
|
|
}
|
|
|
|
|
2024-12-15 23:56:59 +01:00
|
|
|
if (feedOptions) {
|
|
|
|
feedObj.preventIndexing = feedOptions.preventIndexing
|
|
|
|
feedObj.ownerName = feedOptions.ownerName
|
|
|
|
feedObj.ownerEmail = feedOptions.ownerEmail
|
|
|
|
}
|
|
|
|
|
|
|
|
return feedObj
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} userId
|
|
|
|
* @param {import('./LibraryItem').LibraryItemExpanded} libraryItem
|
|
|
|
* @param {string} slug
|
|
|
|
* @param {string} serverAddress
|
|
|
|
* @param {FeedOptions} feedOptions
|
|
|
|
*
|
|
|
|
* @returns {Promise<FeedExpanded>}
|
|
|
|
*/
|
|
|
|
static async createFeedForLibraryItem(userId, libraryItem, slug, serverAddress, feedOptions) {
|
|
|
|
const feedObj = this.getFeedObjForLibraryItem(userId, libraryItem, slug, serverAddress, feedOptions)
|
|
|
|
|
2024-12-14 23:55:56 +01:00
|
|
|
/** @type {typeof import('./FeedEpisode')} */
|
|
|
|
const feedEpisodeModel = this.sequelize.models.feedEpisode
|
|
|
|
|
|
|
|
const transaction = await this.sequelize.transaction()
|
|
|
|
try {
|
|
|
|
const feed = await this.create(feedObj, { transaction })
|
|
|
|
|
|
|
|
if (libraryItem.mediaType === 'podcast') {
|
|
|
|
feed.feedEpisodes = await feedEpisodeModel.createFromPodcastEpisodes(libraryItem, feed, slug, transaction)
|
|
|
|
} else {
|
|
|
|
feed.feedEpisodes = await feedEpisodeModel.createFromAudiobookTracks(libraryItem, feed, slug, transaction)
|
|
|
|
}
|
|
|
|
|
|
|
|
await transaction.commit()
|
|
|
|
|
|
|
|
return feed
|
|
|
|
} catch (error) {
|
|
|
|
Logger.error(`[Feed] Error creating feed for library item ${libraryItem.id}`, error)
|
|
|
|
await transaction.rollback()
|
|
|
|
return null
|
|
|
|
}
|
2023-08-16 01:03:43 +02:00
|
|
|
}
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2024-12-15 17:53:31 +01:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} userId
|
|
|
|
* @param {import('./Collection')} collectionExpanded
|
|
|
|
* @param {string} slug
|
|
|
|
* @param {string} serverAddress
|
2024-12-15 23:56:59 +01:00
|
|
|
* @param {FeedOptions} [feedOptions=null]
|
2024-12-15 17:53:31 +01:00
|
|
|
*
|
2024-12-15 23:56:59 +01:00
|
|
|
* @returns {{ feedObj: Feed, booksWithTracks: import('./Book').BookExpandedWithLibraryItem[] }}
|
2024-12-15 17:53:31 +01:00
|
|
|
*/
|
2024-12-15 23:56:59 +01:00
|
|
|
static getFeedObjForCollection(userId, collectionExpanded, slug, serverAddress, feedOptions = null) {
|
2024-12-15 17:53:31 +01:00
|
|
|
const booksWithTracks = collectionExpanded.books.filter((book) => book.includedAudioFiles.length)
|
2024-12-15 21:07:46 +01:00
|
|
|
|
|
|
|
const entityUpdatedAt = booksWithTracks.reduce((mostRecent, book) => {
|
|
|
|
return book.libraryItem.updatedAt > mostRecent ? book.libraryItem.updatedAt : mostRecent
|
|
|
|
}, collectionExpanded.updatedAt)
|
2024-12-15 17:53:31 +01:00
|
|
|
|
|
|
|
const firstBookWithCover = booksWithTracks.find((book) => book.coverPath)
|
|
|
|
|
|
|
|
const allBookAuthorNames = booksWithTracks.reduce((authorNames, book) => {
|
|
|
|
const bookAuthorsToAdd = book.authors.filter((author) => !authorNames.includes(author.name)).map((author) => author.name)
|
|
|
|
return authorNames.concat(bookAuthorsToAdd)
|
|
|
|
}, [])
|
|
|
|
let author = allBookAuthorNames.slice(0, 3).join(', ')
|
|
|
|
if (allBookAuthorNames.length > 3) {
|
|
|
|
author += ' & more'
|
|
|
|
}
|
|
|
|
|
|
|
|
const feedObj = {
|
|
|
|
slug,
|
|
|
|
entityType: 'collection',
|
|
|
|
entityId: collectionExpanded.id,
|
2024-12-15 21:07:46 +01:00
|
|
|
entityUpdatedAt,
|
2024-12-15 17:53:31 +01:00
|
|
|
serverAddress,
|
|
|
|
feedURL: `/feed/${slug}`,
|
|
|
|
imageURL: firstBookWithCover?.coverPath ? `/feed/${slug}/cover${Path.extname(firstBookWithCover.coverPath)}` : `/Logo.png`,
|
|
|
|
siteURL: `/collection/${collectionExpanded.id}`,
|
|
|
|
title: collectionExpanded.name,
|
|
|
|
description: collectionExpanded.description || '',
|
|
|
|
author,
|
|
|
|
podcastType: 'serial',
|
|
|
|
explicit: booksWithTracks.some((book) => book.explicit), // If any book is explicit, the feed is explicit
|
|
|
|
coverPath: firstBookWithCover?.coverPath || null,
|
|
|
|
userId
|
|
|
|
}
|
|
|
|
|
2024-12-15 23:56:59 +01:00
|
|
|
if (feedOptions) {
|
|
|
|
feedObj.preventIndexing = feedOptions.preventIndexing
|
|
|
|
feedObj.ownerName = feedOptions.ownerName
|
|
|
|
feedObj.ownerEmail = feedOptions.ownerEmail
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
feedObj,
|
|
|
|
booksWithTracks
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} userId
|
|
|
|
* @param {import('./Collection')} collectionExpanded
|
|
|
|
* @param {string} slug
|
|
|
|
* @param {string} serverAddress
|
|
|
|
* @param {FeedOptions} feedOptions
|
|
|
|
*
|
|
|
|
* @returns {Promise<FeedExpanded>}
|
|
|
|
*/
|
|
|
|
static async createFeedForCollection(userId, collectionExpanded, slug, serverAddress, feedOptions) {
|
|
|
|
const { feedObj, booksWithTracks } = this.getFeedObjForCollection(userId, collectionExpanded, slug, serverAddress, feedOptions)
|
|
|
|
|
2024-12-15 17:53:31 +01:00
|
|
|
/** @type {typeof import('./FeedEpisode')} */
|
|
|
|
const feedEpisodeModel = this.sequelize.models.feedEpisode
|
|
|
|
|
|
|
|
const transaction = await this.sequelize.transaction()
|
|
|
|
try {
|
|
|
|
const feed = await this.create(feedObj, { transaction })
|
2024-12-15 18:44:07 +01:00
|
|
|
feed.feedEpisodes = await feedEpisodeModel.createFromBooks(booksWithTracks, feed, slug, transaction)
|
|
|
|
|
|
|
|
await transaction.commit()
|
|
|
|
|
|
|
|
return feed
|
|
|
|
} catch (error) {
|
|
|
|
Logger.error(`[Feed] Error creating feed for collection ${collectionExpanded.id}`, error)
|
|
|
|
await transaction.rollback()
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} userId
|
|
|
|
* @param {import('./Series')} seriesExpanded
|
|
|
|
* @param {string} slug
|
|
|
|
* @param {string} serverAddress
|
2024-12-15 23:56:59 +01:00
|
|
|
* @param {FeedOptions} [feedOptions=null]
|
2024-12-15 18:44:07 +01:00
|
|
|
*
|
2024-12-15 23:56:59 +01:00
|
|
|
* @returns {{ feedObj: Feed, booksWithTracks: import('./Book').BookExpandedWithLibraryItem[] }}
|
2024-12-15 18:44:07 +01:00
|
|
|
*/
|
2024-12-15 23:56:59 +01:00
|
|
|
static getFeedObjForSeries(userId, seriesExpanded, slug, serverAddress, feedOptions = null) {
|
2024-12-15 18:44:07 +01:00
|
|
|
const booksWithTracks = seriesExpanded.books.filter((book) => book.includedAudioFiles.length)
|
2024-12-15 21:07:46 +01:00
|
|
|
const entityUpdatedAt = booksWithTracks.reduce((mostRecent, book) => {
|
|
|
|
return book.libraryItem.updatedAt > mostRecent ? book.libraryItem.updatedAt : mostRecent
|
|
|
|
}, seriesExpanded.updatedAt)
|
2024-12-15 18:44:07 +01:00
|
|
|
|
|
|
|
const firstBookWithCover = booksWithTracks.find((book) => book.coverPath)
|
|
|
|
|
|
|
|
const allBookAuthorNames = booksWithTracks.reduce((authorNames, book) => {
|
|
|
|
const bookAuthorsToAdd = book.authors.filter((author) => !authorNames.includes(author.name)).map((author) => author.name)
|
|
|
|
return authorNames.concat(bookAuthorsToAdd)
|
|
|
|
}, [])
|
|
|
|
let author = allBookAuthorNames.slice(0, 3).join(', ')
|
|
|
|
if (allBookAuthorNames.length > 3) {
|
|
|
|
author += ' & more'
|
|
|
|
}
|
|
|
|
|
|
|
|
const feedObj = {
|
|
|
|
slug,
|
|
|
|
entityType: 'series',
|
|
|
|
entityId: seriesExpanded.id,
|
2024-12-15 21:07:46 +01:00
|
|
|
entityUpdatedAt,
|
2024-12-15 18:44:07 +01:00
|
|
|
serverAddress,
|
|
|
|
feedURL: `/feed/${slug}`,
|
|
|
|
imageURL: firstBookWithCover?.coverPath ? `/feed/${slug}/cover${Path.extname(firstBookWithCover.coverPath)}` : `/Logo.png`,
|
|
|
|
siteURL: `/library/${booksWithTracks[0].libraryItem.libraryId}/series/${seriesExpanded.id}`,
|
|
|
|
title: seriesExpanded.name,
|
|
|
|
description: seriesExpanded.description || '',
|
|
|
|
author,
|
|
|
|
podcastType: 'serial',
|
|
|
|
explicit: booksWithTracks.some((book) => book.explicit), // If any book is explicit, the feed is explicit
|
|
|
|
coverPath: firstBookWithCover?.coverPath || null,
|
|
|
|
userId
|
|
|
|
}
|
|
|
|
|
2024-12-15 23:56:59 +01:00
|
|
|
if (feedOptions) {
|
|
|
|
feedObj.preventIndexing = feedOptions.preventIndexing
|
|
|
|
feedObj.ownerName = feedOptions.ownerName
|
|
|
|
feedObj.ownerEmail = feedOptions.ownerEmail
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
feedObj,
|
|
|
|
booksWithTracks
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} userId
|
|
|
|
* @param {import('./Series')} seriesExpanded
|
|
|
|
* @param {string} slug
|
|
|
|
* @param {string} serverAddress
|
|
|
|
* @param {FeedOptions} feedOptions
|
|
|
|
*
|
|
|
|
* @returns {Promise<FeedExpanded>}
|
|
|
|
*/
|
|
|
|
static async createFeedForSeries(userId, seriesExpanded, slug, serverAddress, feedOptions) {
|
|
|
|
const { feedObj, booksWithTracks } = this.getFeedObjForSeries(userId, seriesExpanded, slug, serverAddress, feedOptions)
|
|
|
|
|
2024-12-15 18:44:07 +01:00
|
|
|
/** @type {typeof import('./FeedEpisode')} */
|
|
|
|
const feedEpisodeModel = this.sequelize.models.feedEpisode
|
|
|
|
|
|
|
|
const transaction = await this.sequelize.transaction()
|
|
|
|
try {
|
|
|
|
const feed = await this.create(feedObj, { transaction })
|
|
|
|
feed.feedEpisodes = await feedEpisodeModel.createFromBooks(booksWithTracks, feed, slug, transaction)
|
2024-12-15 17:53:31 +01:00
|
|
|
|
|
|
|
await transaction.commit()
|
|
|
|
|
|
|
|
return feed
|
|
|
|
} catch (error) {
|
2024-12-15 23:56:59 +01:00
|
|
|
Logger.error(`[Feed] Error creating feed for series ${seriesExpanded.id}`, error)
|
2024-12-15 17:53:31 +01:00
|
|
|
await transaction.rollback()
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
/**
|
|
|
|
* Initialize model
|
2024-05-29 00:24:02 +02:00
|
|
|
*
|
2023-08-16 01:03:43 +02:00
|
|
|
* Polymorphic association: Feeds can be created from LibraryItem, Collection, Playlist or Series
|
|
|
|
* @see https://sequelize.org/docs/v6/advanced-association-concepts/polymorphic-associations/
|
2024-05-29 00:24:02 +02:00
|
|
|
*
|
|
|
|
* @param {import('../Database').sequelize} sequelize
|
2023-08-16 01:03:43 +02:00
|
|
|
*/
|
|
|
|
static init(sequelize) {
|
2024-05-29 00:24:02 +02:00
|
|
|
super.init(
|
|
|
|
{
|
|
|
|
id: {
|
|
|
|
type: DataTypes.UUID,
|
|
|
|
defaultValue: DataTypes.UUIDV4,
|
|
|
|
primaryKey: true
|
|
|
|
},
|
|
|
|
slug: DataTypes.STRING,
|
|
|
|
entityType: DataTypes.STRING,
|
2024-11-17 22:45:21 +01:00
|
|
|
entityId: DataTypes.UUID,
|
2024-05-29 00:24:02 +02:00
|
|
|
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
|
2023-08-16 01:03:43 +02:00
|
|
|
},
|
2024-05-29 00:24:02 +02:00
|
|
|
{
|
|
|
|
sequelize,
|
|
|
|
modelName: 'feed'
|
|
|
|
}
|
|
|
|
)
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
const { user, libraryItem, collection, series, playlist } = sequelize.models
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
user.hasMany(Feed)
|
|
|
|
Feed.belongsTo(user)
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
libraryItem.hasMany(Feed, {
|
|
|
|
foreignKey: 'entityId',
|
|
|
|
constraints: false,
|
|
|
|
scope: {
|
|
|
|
entityType: 'libraryItem'
|
|
|
|
}
|
|
|
|
})
|
|
|
|
Feed.belongsTo(libraryItem, { foreignKey: 'entityId', constraints: false })
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
collection.hasMany(Feed, {
|
|
|
|
foreignKey: 'entityId',
|
|
|
|
constraints: false,
|
|
|
|
scope: {
|
|
|
|
entityType: 'collection'
|
|
|
|
}
|
|
|
|
})
|
|
|
|
Feed.belongsTo(collection, { foreignKey: 'entityId', constraints: false })
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
series.hasMany(Feed, {
|
|
|
|
foreignKey: 'entityId',
|
|
|
|
constraints: false,
|
|
|
|
scope: {
|
|
|
|
entityType: 'series'
|
|
|
|
}
|
|
|
|
})
|
|
|
|
Feed.belongsTo(series, { foreignKey: 'entityId', constraints: false })
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
playlist.hasMany(Feed, {
|
|
|
|
foreignKey: 'entityId',
|
|
|
|
constraints: false,
|
|
|
|
scope: {
|
|
|
|
entityType: 'playlist'
|
2023-07-05 01:14:44 +02:00
|
|
|
}
|
2023-08-16 01:03:43 +02:00
|
|
|
})
|
|
|
|
Feed.belongsTo(playlist, { foreignKey: 'entityId', constraints: false })
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2024-05-29 00:24:02 +02:00
|
|
|
Feed.addHook('afterFind', (findResult) => {
|
2023-08-16 01:03:43 +02:00
|
|
|
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
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2024-12-14 23:55:56 +01:00
|
|
|
|
2024-12-15 23:56:59 +01:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @returns {Promise<FeedExpanded>}
|
|
|
|
*/
|
|
|
|
async updateFeedForEntity() {
|
|
|
|
/** @type {typeof import('./FeedEpisode')} */
|
|
|
|
const feedEpisodeModel = this.sequelize.models.feedEpisode
|
|
|
|
|
|
|
|
let feedObj = null
|
|
|
|
let feedEpisodeCreateFunc = null
|
|
|
|
let feedEpisodeCreateFuncEntity = null
|
|
|
|
|
|
|
|
if (this.entityType === 'libraryItem') {
|
|
|
|
/** @type {typeof import('./LibraryItem')} */
|
|
|
|
const libraryItemModel = this.sequelize.models.libraryItem
|
|
|
|
|
|
|
|
const itemExpanded = await libraryItemModel.getExpandedById(this.entityId)
|
|
|
|
feedObj = Feed.getFeedObjForLibraryItem(this.userId, itemExpanded, this.slug, this.serverAddress)
|
|
|
|
|
|
|
|
feedEpisodeCreateFuncEntity = itemExpanded
|
|
|
|
if (itemExpanded.mediaType === 'podcast') {
|
|
|
|
feedEpisodeCreateFunc = feedEpisodeModel.createFromPodcastEpisodes.bind(feedEpisodeModel)
|
|
|
|
} else {
|
|
|
|
feedEpisodeCreateFunc = feedEpisodeModel.createFromAudiobookTracks.bind(feedEpisodeModel)
|
|
|
|
}
|
|
|
|
} else if (this.entityType === 'collection') {
|
|
|
|
/** @type {typeof import('./Collection')} */
|
|
|
|
const collectionModel = this.sequelize.models.collection
|
|
|
|
|
|
|
|
const collectionExpanded = await collectionModel.getExpandedById(this.entityId)
|
|
|
|
const feedObjData = Feed.getFeedObjForCollection(this.userId, collectionExpanded, this.slug, this.serverAddress)
|
|
|
|
feedObj = feedObjData.feedObj
|
|
|
|
feedEpisodeCreateFuncEntity = feedObjData.booksWithTracks
|
|
|
|
feedEpisodeCreateFunc = feedEpisodeModel.createFromBooks.bind(feedEpisodeModel)
|
|
|
|
} else if (this.entityType === 'series') {
|
|
|
|
/** @type {typeof import('./Series')} */
|
|
|
|
const seriesModel = this.sequelize.models.series
|
|
|
|
|
|
|
|
const seriesExpanded = await seriesModel.getExpandedById(this.entityId)
|
|
|
|
const feedObjData = Feed.getFeedObjForSeries(this.userId, seriesExpanded, this.slug, this.serverAddress)
|
|
|
|
feedObj = feedObjData.feedObj
|
|
|
|
feedEpisodeCreateFuncEntity = feedObjData.booksWithTracks
|
|
|
|
feedEpisodeCreateFunc = feedEpisodeModel.createFromBooks.bind(feedEpisodeModel)
|
|
|
|
} else {
|
|
|
|
Logger.error(`[Feed] Invalid entity type ${this.entityType} for feed ${this.id}`)
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
|
|
|
|
const transaction = await this.sequelize.transaction()
|
|
|
|
try {
|
|
|
|
const updatedFeed = await this.update(feedObj, { transaction })
|
|
|
|
|
|
|
|
// Remove existing feed episodes
|
|
|
|
await feedEpisodeModel.destroy({
|
|
|
|
where: {
|
|
|
|
feedId: this.id
|
|
|
|
},
|
|
|
|
transaction
|
|
|
|
})
|
|
|
|
|
|
|
|
// Create new feed episodes
|
|
|
|
updatedFeed.feedEpisodes = await feedEpisodeCreateFunc(feedEpisodeCreateFuncEntity, updatedFeed, this.slug, transaction)
|
|
|
|
|
|
|
|
await transaction.commit()
|
|
|
|
|
|
|
|
return updatedFeed
|
|
|
|
} catch (error) {
|
|
|
|
Logger.error(`[Feed] Error updating feed ${this.entityId}`, error)
|
|
|
|
await transaction.rollback()
|
|
|
|
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-14 23:55:56 +01:00
|
|
|
getEntity(options) {
|
|
|
|
if (!this.entityType) return Promise.resolve(null)
|
|
|
|
const mixinMethodName = `get${this.sequelize.uppercaseFirst(this.entityType)}`
|
|
|
|
return this[mixinMethodName](options)
|
|
|
|
}
|
|
|
|
|
2024-12-15 23:56:59 +01:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} hostPrefix
|
|
|
|
*/
|
|
|
|
buildXml(hostPrefix) {
|
|
|
|
const blockTags = [{ 'itunes:block': 'yes' }, { 'googleplay:block': 'yes' }]
|
|
|
|
const rssData = {
|
|
|
|
title: this.title,
|
|
|
|
description: this.description || '',
|
|
|
|
generator: 'Audiobookshelf',
|
|
|
|
feed_url: `${hostPrefix}${this.feedURL}`,
|
|
|
|
site_url: `${hostPrefix}${this.siteURL}`,
|
|
|
|
image_url: `${hostPrefix}${this.imageURL}`,
|
|
|
|
custom_namespaces: {
|
|
|
|
itunes: 'http://www.itunes.com/dtds/podcast-1.0.dtd',
|
|
|
|
psc: 'http://podlove.org/simple-chapters',
|
|
|
|
podcast: 'https://podcastindex.org/namespace/1.0',
|
|
|
|
googleplay: 'http://www.google.com/schemas/play-podcasts/1.0'
|
|
|
|
},
|
|
|
|
custom_elements: [
|
|
|
|
{ language: this.language || 'en' },
|
|
|
|
{ author: this.author || 'advplyr' },
|
|
|
|
{ 'itunes:author': this.author || 'advplyr' },
|
|
|
|
{ 'itunes:summary': this.description || '' },
|
|
|
|
{ 'itunes:type': this.podcastType },
|
|
|
|
{
|
|
|
|
'itunes:image': {
|
|
|
|
_attr: {
|
|
|
|
href: `${hostPrefix}${this.imageURL}`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
'itunes:owner': [{ 'itunes:name': this.ownerName || this.author || '' }, { 'itunes:email': this.ownerEmail || '' }]
|
|
|
|
},
|
|
|
|
{ 'itunes:explicit': !!this.explicit },
|
|
|
|
...(this.preventIndexing ? blockTags : [])
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
const rssfeed = new RSS(rssData)
|
|
|
|
this.feedEpisodes.forEach((ep) => {
|
|
|
|
rssfeed.item(ep.getRSSData(hostPrefix))
|
|
|
|
})
|
|
|
|
return rssfeed.xml()
|
|
|
|
}
|
|
|
|
|
2024-12-16 00:54:36 +01:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} id
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
getEpisodePath(id) {
|
|
|
|
const episode = this.feedEpisodes.find((ep) => ep.id === id)
|
|
|
|
if (!episode) return null
|
|
|
|
return episode.filePath
|
|
|
|
}
|
|
|
|
|
2024-12-14 23:55:56 +01:00
|
|
|
toOldJSON() {
|
|
|
|
const episodes = this.feedEpisodes?.map((feedEpisode) => feedEpisode.getOldEpisode())
|
|
|
|
return {
|
|
|
|
id: this.id,
|
|
|
|
slug: this.slug,
|
|
|
|
userId: this.userId,
|
|
|
|
entityType: this.entityType,
|
|
|
|
entityId: this.entityId,
|
|
|
|
entityUpdatedAt: this.entityUpdatedAt?.valueOf() || null,
|
|
|
|
coverPath: this.coverPath || null,
|
|
|
|
meta: {
|
|
|
|
title: this.title,
|
|
|
|
description: this.description,
|
|
|
|
author: this.author,
|
|
|
|
imageUrl: this.imageURL,
|
|
|
|
feedUrl: this.feedURL,
|
|
|
|
link: this.siteURL,
|
|
|
|
explicit: this.explicit,
|
|
|
|
type: this.podcastType,
|
|
|
|
language: this.language,
|
|
|
|
preventIndexing: this.preventIndexing,
|
|
|
|
ownerName: this.ownerName,
|
|
|
|
ownerEmail: this.ownerEmail
|
|
|
|
},
|
|
|
|
serverAddress: this.serverAddress,
|
|
|
|
feedUrl: this.feedURL,
|
|
|
|
episodes: episodes || [],
|
|
|
|
createdAt: this.createdAt.valueOf(),
|
|
|
|
updatedAt: this.updatedAt.valueOf()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
toOldJSONMinified() {
|
|
|
|
return {
|
|
|
|
id: this.id,
|
|
|
|
entityType: this.entityType,
|
|
|
|
entityId: this.entityId,
|
|
|
|
feedUrl: this.feedURL,
|
|
|
|
meta: {
|
|
|
|
title: this.title,
|
|
|
|
description: this.description,
|
|
|
|
preventIndexing: this.preventIndexing,
|
|
|
|
ownerName: this.ownerName,
|
|
|
|
ownerEmail: this.ownerEmail
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-08-16 01:03:43 +02:00
|
|
|
}
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2024-05-29 00:24:02 +02:00
|
|
|
module.exports = Feed
|