2023-07-23 16:42:57 +02:00
|
|
|
const { DataTypes, Model, Op } = require('sequelize')
|
2023-07-14 21:50:37 +02:00
|
|
|
const Logger = require('../Logger')
|
2023-07-05 01:14:44 +02:00
|
|
|
|
|
|
|
const oldPlaylist = require('../objects/Playlist')
|
|
|
|
const { areEquivalent } = require('../utils/index')
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
2023-07-10 23:07:22 +02:00
|
|
|
},
|
|
|
|
order: [['playlistMediaItems', 'order', 'ASC']]
|
2023-07-05 01:14:44 +02:00
|
|
|
})
|
|
|
|
return playlists.map(p => this.getOldPlaylist(p))
|
|
|
|
}
|
|
|
|
|
|
|
|
static getOldPlaylist(playlistExpanded) {
|
|
|
|
const items = playlistExpanded.playlistMediaItems.map(pmi => {
|
|
|
|
const libraryItemId = pmi.mediaItem?.podcast?.libraryItem?.id || pmi.mediaItem?.libraryItem?.id || null
|
|
|
|
if (!libraryItemId) {
|
2023-07-14 21:50:37 +02:00
|
|
|
Logger.error(`[Playlist] Invalid playlist media item - No library item id found`, JSON.stringify(pmi, null, 2))
|
|
|
|
return null
|
2023-07-05 01:14:44 +02:00
|
|
|
}
|
|
|
|
return {
|
|
|
|
episodeId: pmi.mediaItemType === 'podcastEpisode' ? pmi.mediaItemId : '',
|
2023-07-14 21:50:37 +02:00
|
|
|
libraryItemId
|
2023-07-05 01:14:44 +02:00
|
|
|
}
|
2023-07-14 21:50:37 +02:00
|
|
|
}).filter(pmi => pmi)
|
|
|
|
|
2023-07-05 01:14:44 +02:00
|
|
|
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()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-08-13 00:29:08 +02:00
|
|
|
/**
|
|
|
|
* Get old playlist toJSONExpanded
|
|
|
|
* @param {[string[]]} include
|
|
|
|
* @returns {Promise<object>} oldPlaylist.toJSONExpanded
|
|
|
|
*/
|
|
|
|
async getOldJsonExpanded(include) {
|
|
|
|
this.playlistMediaItems = await this.getPlaylistMediaItems({
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
model: sequelize.models.book,
|
|
|
|
include: sequelize.models.libraryItem
|
|
|
|
},
|
|
|
|
{
|
|
|
|
model: sequelize.models.podcastEpisode,
|
|
|
|
include: {
|
|
|
|
model: sequelize.models.podcast,
|
|
|
|
include: 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
|
|
|
|
})
|
|
|
|
|
2023-08-13 18:22:38 +02:00
|
|
|
const playlistExpanded = oldPlaylist.toJSONExpanded(libraryItems)
|
2023-08-13 00:29:08 +02:00
|
|
|
|
|
|
|
if (include?.includes('rssfeed')) {
|
|
|
|
const feeds = await this.getFeeds()
|
|
|
|
if (feeds?.length) {
|
|
|
|
playlistExpanded.rssFeed = sequelize.models.feed.getOldFeed(feeds[0])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return playlistExpanded
|
|
|
|
}
|
|
|
|
|
2023-07-05 01:14:44 +02:00
|
|
|
static createFromOld(oldPlaylist) {
|
|
|
|
const playlist = this.getFromOld(oldPlaylist)
|
|
|
|
return this.create(playlist)
|
|
|
|
}
|
|
|
|
|
|
|
|
static async fullUpdateFromOld(oldPlaylist, playlistMediaItems) {
|
|
|
|
const existingPlaylist = await this.findByPk(oldPlaylist.id, {
|
|
|
|
include: sequelize.models.playlistMediaItem
|
|
|
|
})
|
|
|
|
if (!existingPlaylist) return false
|
|
|
|
|
|
|
|
let hasUpdates = false
|
|
|
|
const playlist = this.getFromOld(oldPlaylist)
|
|
|
|
|
|
|
|
for (const pmi of playlistMediaItems) {
|
|
|
|
const existingPmi = existingPlaylist.playlistMediaItems.find(i => i.mediaItemId === pmi.mediaItemId)
|
|
|
|
if (!existingPmi) {
|
|
|
|
await sequelize.models.playlistMediaItem.create(pmi)
|
|
|
|
hasUpdates = true
|
|
|
|
} else if (existingPmi.order != pmi.order) {
|
|
|
|
await existingPmi.update({ order: pmi.order })
|
|
|
|
hasUpdates = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (const pmi of existingPlaylist.playlistMediaItems) {
|
|
|
|
// Pmi was removed
|
|
|
|
if (!playlistMediaItems.some(i => i.mediaItemId === pmi.mediaItemId)) {
|
|
|
|
await pmi.destroy()
|
|
|
|
hasUpdates = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let hasPlaylistUpdates = false
|
|
|
|
for (const key in playlist) {
|
|
|
|
let existingValue = existingPlaylist[key]
|
|
|
|
if (existingValue instanceof Date) existingValue = existingValue.valueOf()
|
|
|
|
|
|
|
|
if (!areEquivalent(playlist[key], existingValue)) {
|
|
|
|
hasPlaylistUpdates = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (hasPlaylistUpdates) {
|
|
|
|
existingPlaylist.update(playlist)
|
|
|
|
hasUpdates = true
|
|
|
|
}
|
|
|
|
return hasUpdates
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2023-07-23 16:42:57 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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<oldPlaylist[]>}
|
|
|
|
*/
|
|
|
|
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: [['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
|
2023-08-13 18:22:38 +02:00
|
|
|
* @returns {Promise<Playlist[]>}
|
2023-07-23 16:42:57 +02:00
|
|
|
*/
|
|
|
|
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']]
|
|
|
|
})
|
2023-08-13 18:22:38 +02:00
|
|
|
|
|
|
|
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 => {
|
2023-07-23 16:42:57 +02:00
|
|
|
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
|
|
|
|
})
|
2023-08-13 18:22:38 +02:00
|
|
|
playlists.push(playlist)
|
|
|
|
}
|
|
|
|
return playlists
|
2023-07-23 16:42:57 +02:00
|
|
|
}
|
2023-07-05 01:14:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
2023-08-13 18:22:38 +02:00
|
|
|
user.hasMany(Playlist, {
|
|
|
|
onDelete: 'CASCADE'
|
|
|
|
})
|
2023-07-05 01:14:44 +02:00
|
|
|
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
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
return Playlist
|
|
|
|
}
|