2023-08-12 22:01:27 +02:00
|
|
|
const { DataTypes, Model, Sequelize } = require('sequelize')
|
2023-07-05 01:14:44 +02:00
|
|
|
|
|
|
|
const oldCollection = require('../objects/Collection')
|
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
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
|
2024-12-15 17:53:31 +01:00
|
|
|
|
|
|
|
// Expanded properties
|
|
|
|
|
|
|
|
/** @type {import('./Book').BookExpandedWithLibraryItem[]} - only set when expanded */
|
|
|
|
this.books
|
2023-08-16 01:03:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get all old collections toJSONExpanded, items filtered for user permissions
|
2024-08-11 00:15:21 +02:00
|
|
|
*
|
|
|
|
* @param {import('./User')} user
|
2024-05-29 00:24:02 +02:00
|
|
|
* @param {string} [libraryId]
|
|
|
|
* @param {string[]} [include]
|
|
|
|
* @returns {Promise<oldCollection[]>} oldCollection.toJSONExpanded
|
2023-08-16 01:03:43 +02:00
|
|
|
*/
|
|
|
|
static async getOldCollectionsJsonExpanded(user, libraryId, include) {
|
|
|
|
let collectionWhere = null
|
|
|
|
if (libraryId) {
|
|
|
|
collectionWhere = {
|
|
|
|
libraryId
|
2023-08-12 00:49:06 +02:00
|
|
|
}
|
2023-08-16 01:03:43 +02:00
|
|
|
}
|
2023-08-12 00:49:06 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
// Optionally include rssfeed for collection
|
|
|
|
const collectionIncludes = []
|
2024-09-06 00:15:38 +02:00
|
|
|
if (include?.includes('rssfeed')) {
|
2023-08-16 01:03:43 +02:00
|
|
|
collectionIncludes.push({
|
|
|
|
model: this.sequelize.models.feed
|
2023-08-12 00:49:06 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
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']
|
|
|
|
}
|
2024-05-29 00:24:02 +02:00
|
|
|
}
|
2023-08-16 01:03:43 +02:00
|
|
|
]
|
|
|
|
},
|
|
|
|
...collectionIncludes
|
|
|
|
],
|
|
|
|
order: [[this.sequelize.models.book, this.sequelize.models.collectionBook, 'order', 'ASC']]
|
|
|
|
})
|
|
|
|
// TODO: Handle user permission restrictions on initial query
|
2024-05-29 00:24:02 +02:00
|
|
|
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 this.sequelize.models.libraryItem.getOldLibraryItem(libraryItem)
|
|
|
|
})
|
|
|
|
|
|
|
|
// Users with restricted permissions will not see this collection
|
|
|
|
if (!books.length && oldCollection.books.length) {
|
|
|
|
return null
|
2023-08-12 22:01:27 +02:00
|
|
|
}
|
|
|
|
|
2024-05-29 00:24:02 +02:00
|
|
|
const collectionExpanded = oldCollection.toJSONExpanded(libraryItems)
|
2023-08-12 22:01:27 +02:00
|
|
|
|
2024-05-29 00:24:02 +02:00
|
|
|
// Map feed if found
|
|
|
|
if (c.feeds?.length) {
|
2024-12-16 00:54:36 +01:00
|
|
|
collectionExpanded.rssFeed = c.feeds[0].toOldJSON()
|
2024-05-29 00:24:02 +02:00
|
|
|
}
|
2023-08-12 22:01:27 +02:00
|
|
|
|
2024-05-29 00:24:02 +02:00
|
|
|
return collectionExpanded
|
|
|
|
})
|
|
|
|
.filter((c) => c)
|
2023-08-16 01:03:43 +02:00
|
|
|
}
|
2023-08-12 22:01:27 +02:00
|
|
|
|
2024-12-15 23:56:59 +01:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} collectionId
|
|
|
|
* @returns {Promise<Collection>}
|
|
|
|
*/
|
|
|
|
static async getExpandedById(collectionId) {
|
|
|
|
return this.findByPk(collectionId, {
|
|
|
|
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: [[this.sequelize.models.book, this.sequelize.models.collectionBook, 'order', 'ASC']]
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
/**
|
|
|
|
* Get old collection from Collection
|
2024-05-29 00:24:02 +02:00
|
|
|
* @param {Collection} collectionExpanded
|
2023-08-16 01:03:43 +02:00
|
|
|
* @returns {oldCollection}
|
|
|
|
*/
|
|
|
|
static getOldCollection(collectionExpanded) {
|
2024-05-29 00:24:02 +02:00
|
|
|
const libraryItemIds = collectionExpanded.books?.map((b) => b.libraryItem?.id || null).filter((lid) => lid) || []
|
2023-08-16 01:03:43 +02:00
|
|
|
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()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-05-29 00:24:02 +02:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {oldCollection} oldCollection
|
|
|
|
* @returns {Promise<Collection>}
|
|
|
|
*/
|
2023-08-16 01:03:43 +02:00
|
|
|
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
|
2023-07-05 01:14:44 +02:00
|
|
|
}
|
2023-08-16 01:03:43 +02:00
|
|
|
}
|
2023-07-22 23:18:55 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
static removeById(collectionId) {
|
|
|
|
return this.destroy({
|
|
|
|
where: {
|
|
|
|
id: collectionId
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get old collection by id
|
2024-05-29 00:24:02 +02:00
|
|
|
* @param {string} collectionId
|
2023-08-16 01:03:43 +02:00
|
|
|
* @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)
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove all collections belonging to library
|
2024-05-29 00:24:02 +02:00
|
|
|
* @param {string} libraryId
|
2023-08-16 01:03:43 +02:00
|
|
|
* @returns {Promise<number>} number of collections destroyed
|
|
|
|
*/
|
|
|
|
static async removeAllForLibrary(libraryId) {
|
|
|
|
if (!libraryId) return 0
|
|
|
|
return this.destroy({
|
|
|
|
where: {
|
|
|
|
libraryId
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2023-07-22 23:18:55 +02:00
|
|
|
|
2023-08-16 01:03:43 +02:00
|
|
|
/**
|
|
|
|
* Initialize model
|
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
|
|
|
|
},
|
|
|
|
name: DataTypes.STRING,
|
|
|
|
description: DataTypes.TEXT
|
2023-08-16 01:03:43 +02:00
|
|
|
},
|
2024-05-29 00:24:02 +02:00
|
|
|
{
|
|
|
|
sequelize,
|
|
|
|
modelName: 'collection'
|
|
|
|
}
|
|
|
|
)
|
2023-08-16 01:03:43 +02:00
|
|
|
|
|
|
|
const { library } = sequelize.models
|
|
|
|
|
|
|
|
library.hasMany(Collection)
|
|
|
|
Collection.belongsTo(library)
|
2023-07-05 01:14:44 +02:00
|
|
|
}
|
2024-09-05 01:00:59 +02:00
|
|
|
|
2024-12-15 17:53:31 +01:00
|
|
|
/**
|
|
|
|
* Get all books in collection expanded with library item
|
|
|
|
*
|
|
|
|
* @returns {Promise<import('./Book').BookExpandedWithLibraryItem[]>}
|
|
|
|
*/
|
|
|
|
getBooksExpandedWithLibraryItem() {
|
|
|
|
return 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')]
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-09-05 01:00:59 +02:00
|
|
|
/**
|
|
|
|
* Get old collection toJSONExpanded, items filtered for user permissions
|
|
|
|
*
|
|
|
|
* @param {import('./User')|null} user
|
|
|
|
* @param {string[]} [include]
|
|
|
|
* @returns {Promise<oldCollection>} 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')]
|
|
|
|
})) || []
|
|
|
|
|
|
|
|
// 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 && this.books.length) {
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
|
|
|
|
const collectionExpanded = this.toOldJSONExpanded(libraryItems)
|
|
|
|
|
|
|
|
if (include?.includes('rssfeed')) {
|
|
|
|
const feeds = await this.getFeeds()
|
|
|
|
if (feeds?.length) {
|
2024-12-16 00:54:36 +01:00
|
|
|
collectionExpanded.rssFeed = feeds[0].toOldJSON()
|
2024-09-05 01:00:59 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return collectionExpanded
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string[]} libraryItemIds
|
|
|
|
* @returns
|
|
|
|
*/
|
|
|
|
toOldJSON(libraryItemIds) {
|
|
|
|
return {
|
|
|
|
id: this.id,
|
|
|
|
libraryId: this.libraryId,
|
|
|
|
name: this.name,
|
|
|
|
description: this.description,
|
|
|
|
books: [...libraryItemIds],
|
|
|
|
lastUpdate: this.updatedAt.valueOf(),
|
|
|
|
createdAt: this.createdAt.valueOf()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {import('../objects/LibraryItem')} oldLibraryItems
|
|
|
|
* @returns
|
|
|
|
*/
|
|
|
|
toOldJSONExpanded(oldLibraryItems) {
|
|
|
|
const json = this.toOldJSON(oldLibraryItems.map((li) => li.id))
|
|
|
|
json.books = json.books
|
|
|
|
.map((libraryItemId) => {
|
|
|
|
const book = oldLibraryItems.find((li) => li.id === libraryItemId)
|
|
|
|
return book ? book.toJSONExpanded() : null
|
|
|
|
})
|
|
|
|
.filter((b) => !!b)
|
|
|
|
return json
|
|
|
|
}
|
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 = Collection
|