const { Request, Response, NextFunction } = require('express') const Sequelize = require('sequelize') const Logger = require('../Logger') const SocketAuthority = require('../SocketAuthority') const Database = require('../Database') const RssFeedManager = require('../managers/RssFeedManager') /** * @typedef RequestUserObject * @property {import('../models/User')} user * * @typedef {Request & RequestUserObject} RequestWithUser * * @typedef RequestEntityObject * @property {import('../models/Collection')} collection * * @typedef {RequestWithUser & RequestEntityObject} CollectionControllerRequest */ class CollectionController { constructor() {} /** * POST: /api/collections * Create new collection * * @param {RequestWithUser} req * @param {Response} res */ async create(req, res) { const reqBody = req.body || {} // Validation if (!reqBody.name || !reqBody.libraryId) { return res.status(400).send('Invalid collection data') } if (reqBody.description && typeof reqBody.description !== 'string') { return res.status(400).send('Invalid collection description') } const libraryItemIds = (reqBody.books || []).filter((b) => !!b && typeof b == 'string') if (!libraryItemIds.length) { return res.status(400).send('Invalid collection data. No books') } // Load library items const libraryItems = await Database.libraryItemModel.findAll({ attributes: ['id', 'mediaId', 'mediaType', 'libraryId'], where: { id: libraryItemIds, libraryId: reqBody.libraryId, mediaType: 'book' } }) if (libraryItems.length !== libraryItemIds.length) { return res.status(400).send('Invalid collection data. Invalid books') } /** @type {import('../models/Collection')} */ let newCollection = null const transaction = await Database.sequelize.transaction() try { // Create collection newCollection = await Database.collectionModel.create( { libraryId: reqBody.libraryId, name: reqBody.name, description: reqBody.description || null }, { transaction } ) // Create collectionBooks const collectionBookPayloads = libraryItemIds.map((llid, index) => { const libraryItem = libraryItems.find((li) => li.id === llid) return { collectionId: newCollection.id, bookId: libraryItem.mediaId, order: index + 1 } }) await Database.collectionBookModel.bulkCreate(collectionBookPayloads, { transaction }) await transaction.commit() } catch (error) { await transaction.rollback() Logger.error('[CollectionController] create:', error) return res.status(500).send('Failed to create collection') } // Load books expanded newCollection.books = await newCollection.getBooksExpandedWithLibraryItem() // Note: The old collection model stores expanded libraryItems in the books property const jsonExpanded = newCollection.toOldJSONExpanded() SocketAuthority.emitter('collection_added', jsonExpanded) res.json(jsonExpanded) } /** * GET: /api/collections * * @param {RequestWithUser} req * @param {Response} res */ async findAll(req, res) { const collectionsExpanded = await Database.collectionModel.getOldCollectionsJsonExpanded(req.user) res.json({ collections: collectionsExpanded }) } /** * GET: /api/collections/:id * * @param {CollectionControllerRequest} req * @param {Response} res */ async findOne(req, res) { const includeEntities = (req.query.include || '').split(',') const collectionExpanded = await req.collection.getOldJsonExpanded(req.user, includeEntities) if (!collectionExpanded) { // This may happen if the user is restricted from all books return res.sendStatus(404) } res.json(collectionExpanded) } /** * PATCH: /api/collections/:id * Update collection * * @param {CollectionControllerRequest} req * @param {Response} res */ async update(req, res) { let wasUpdated = false // Update description and name if defined const collectionUpdatePayload = {} if (req.body.description !== undefined && req.body.description !== req.collection.description) { collectionUpdatePayload.description = req.body.description wasUpdated = true } if (req.body.name !== undefined && req.body.name !== req.collection.name) { collectionUpdatePayload.name = req.body.name wasUpdated = true } if (wasUpdated) { await req.collection.update(collectionUpdatePayload) } // If books array is passed in then update order in collection let collectionBooksUpdated = false if (req.body.books?.length) { const collectionBooks = await req.collection.getCollectionBooks({ include: { model: Database.bookModel, include: Database.libraryItemModel }, order: [['order', 'ASC']] }) collectionBooks.sort((a, b) => { const aIndex = req.body.books.findIndex((lid) => lid === a.book.libraryItem.id) const bIndex = req.body.books.findIndex((lid) => lid === b.book.libraryItem.id) return aIndex - bIndex }) for (let i = 0; i < collectionBooks.length; i++) { if (collectionBooks[i].order !== i + 1) { await collectionBooks[i].update({ order: i + 1 }) collectionBooksUpdated = true } } if (collectionBooksUpdated) { req.collection.changed('updatedAt', true) await req.collection.save() wasUpdated = true } } const jsonExpanded = await req.collection.getOldJsonExpanded() if (wasUpdated) { SocketAuthority.emitter('collection_updated', jsonExpanded) } res.json(jsonExpanded) } /** * DELETE: /api/collections/:id * * @this {import('../routers/ApiRouter')} * * @param {CollectionControllerRequest} req * @param {Response} res */ async delete(req, res) { const jsonExpanded = await req.collection.getOldJsonExpanded() // Close rss feed - remove from db and emit socket event await RssFeedManager.closeFeedForEntityId(req.collection.id) await req.collection.destroy() SocketAuthority.emitter('collection_removed', jsonExpanded) res.sendStatus(200) } /** * POST: /api/collections/:id/book * Add a single book to a collection * Req.body { id: } * * @param {CollectionControllerRequest} req * @param {Response} res */ async addBook(req, res) { const libraryItem = await Database.libraryItemModel.getOldById(req.body.id) if (!libraryItem) { return res.status(404).send('Book not found') } if (libraryItem.libraryId !== req.collection.libraryId) { return res.status(400).send('Book in different library') } // Check if book is already in collection const collectionBooks = await req.collection.getCollectionBooks() if (collectionBooks.some((cb) => cb.bookId === libraryItem.media.id)) { return res.status(400).send('Book already in collection') } // Create collectionBook record await Database.collectionBookModel.create({ collectionId: req.collection.id, bookId: libraryItem.media.id, order: collectionBooks.length + 1 }) const jsonExpanded = await req.collection.getOldJsonExpanded() SocketAuthority.emitter('collection_updated', jsonExpanded) res.json(jsonExpanded) } /** * DELETE: /api/collections/:id/book/:bookId * Remove a single book from a collection. Re-order books * TODO: bookId is actually libraryItemId. Clients need updating to use bookId * * @param {CollectionControllerRequest} req * @param {Response} res */ async removeBook(req, res) { const libraryItem = await Database.libraryItemModel.getOldById(req.params.bookId) if (!libraryItem) { return res.sendStatus(404) } // Get books in collection ordered const collectionBooks = await req.collection.getCollectionBooks({ order: [['order', 'ASC']] }) let jsonExpanded = null const collectionBookToRemove = collectionBooks.find((cb) => cb.bookId === libraryItem.media.id) if (collectionBookToRemove) { // Remove collection book record await collectionBookToRemove.destroy() // Update order on collection books let order = 1 for (const collectionBook of collectionBooks) { if (collectionBook.bookId === libraryItem.media.id) continue if (collectionBook.order !== order) { await collectionBook.update({ order }) } order++ } jsonExpanded = await req.collection.getOldJsonExpanded() SocketAuthority.emitter('collection_updated', jsonExpanded) } else { jsonExpanded = await req.collection.getOldJsonExpanded() } res.json(jsonExpanded) } /** * POST: /api/collections/:id/batch/add * Add multiple books to collection * Req.body { books: } * * @param {CollectionControllerRequest} req * @param {Response} res */ async addBatch(req, res) { // filter out invalid libraryItemIds const bookIdsToAdd = (req.body.books || []).filter((b) => !!b && typeof b == 'string') if (!bookIdsToAdd.length) { return res.status(400).send('Invalid request body') } // Get library items associated with ids const libraryItems = await Database.libraryItemModel.findAll({ attributes: ['id', 'mediaId', 'mediaType', 'libraryId'], where: { id: bookIdsToAdd, libraryId: req.collection.libraryId, mediaType: 'book' } }) if (!libraryItems.length) { return res.status(400).send('Invalid request body. No valid books') } // Get collection books already in collection /** @type {import('../models/CollectionBook')[]} */ const collectionBooks = await req.collection.getCollectionBooks() let order = collectionBooks.length + 1 const collectionBooksToAdd = [] let hasUpdated = false // Check and set new collection books to add for (const libraryItem of libraryItems) { if (!collectionBooks.some((cb) => cb.bookId === libraryItem.mediaId)) { collectionBooksToAdd.push({ collectionId: req.collection.id, bookId: libraryItem.mediaId, order: order++ }) hasUpdated = true } else { Logger.warn(`[CollectionController] addBatch: Library item ${libraryItem.id} already in collection`) } } let jsonExpanded = null if (hasUpdated) { await Database.collectionBookModel.bulkCreate(collectionBooksToAdd) jsonExpanded = await req.collection.getOldJsonExpanded() SocketAuthority.emitter('collection_updated', jsonExpanded) } else { jsonExpanded = await req.collection.getOldJsonExpanded() } res.json(jsonExpanded) } /** * POST: /api/collections/:id/batch/remove * Remove multiple books from collection * Req.body { books: } * * @param {CollectionControllerRequest} req * @param {Response} res */ async removeBatch(req, res) { // filter out invalid libraryItemIds const bookIdsToRemove = (req.body.books || []).filter((b) => !!b && typeof b == 'string') if (!bookIdsToRemove.length) { return res.status(500).send('Invalid request body') } // Get library items associated with ids const libraryItems = await Database.libraryItemModel.findAll({ where: { id: bookIdsToRemove }, include: { model: Database.bookModel } }) // Get collection books already in collection /** @type {import('../models/CollectionBook')[]} */ const collectionBooks = await req.collection.getCollectionBooks({ order: [['order', 'ASC']] }) // Remove collection books and update order let order = 1 let hasUpdated = false for (const collectionBook of collectionBooks) { if (libraryItems.some((li) => li.media.id === collectionBook.bookId)) { await collectionBook.destroy() hasUpdated = true continue } else if (collectionBook.order !== order) { await collectionBook.update({ order }) hasUpdated = true } order++ } let jsonExpanded = await req.collection.getOldJsonExpanded() if (hasUpdated) { SocketAuthority.emitter('collection_updated', jsonExpanded) } res.json(jsonExpanded) } /** * * @param {RequestWithUser} req * @param {Response} res * @param {NextFunction} next */ async middleware(req, res, next) { if (req.params.id) { const collection = await Database.collectionModel.findByPk(req.params.id) if (!collection) { return res.status(404).send('Collection not found') } req.collection = collection } if (req.method == 'DELETE' && !req.user.canDelete) { Logger.warn(`[CollectionController] User "${req.user.username}" attempted to delete without permission`) return res.sendStatus(403) } else if ((req.method == 'PATCH' || req.method == 'POST') && !req.user.canUpdate) { Logger.warn(`[CollectionController] User "${req.user.username}" attempted to update without permission`) return res.sendStatus(403) } next() } } module.exports = new CollectionController()