const { Request, Response, NextFunction } = require('express') const Logger = require('../Logger') const Database = require('../Database') const { toNumber, isUUID } = require('../utils/index') const ShareManager = require('../managers/ShareManager') /** * @typedef RequestUserObject * @property {import('../models/User')} user * * @typedef {Request & RequestUserObject} RequestWithUser */ class SessionController { constructor() {} /** * GET: /api/sessions * * @this import('../routers/ApiRouter') * * @param {RequestWithUser} req * @param {Response} res */ async getAllWithUserData(req, res) { if (!req.user.isAdminOrUp) { Logger.error(`[SessionController] getAllWithUserData: Non-admin user "${req.user.username}" requested all session data`) return res.sendStatus(404) } // Validate "user" query let userId = req.query.user if (userId && !isUUID(userId)) { Logger.warn(`[SessionController] Invalid "user" query string "${userId}"`) userId = null } // Validate "sort" query const validSortOrders = ['displayTitle', 'duration', 'playMethod', 'startTime', 'currentTime', 'timeListening', 'updatedAt', 'createdAt'] let orderKey = req.query.sort || 'updatedAt' if (!validSortOrders.includes(orderKey)) { Logger.warn(`[SessionController] Invalid "sort" query string "${orderKey}" (Must be one of "${validSortOrders.join('|')}")`) orderKey = 'updatedAt' } let orderDesc = req.query.desc === '1' ? 'DESC' : 'ASC' // Validate "itemsPerPage" and "page" query let itemsPerPage = toNumber(req.query.itemsPerPage, 10) || 10 if (itemsPerPage < 1) { Logger.warn(`[SessionController] Invalid "itemsPerPage" query string "${itemsPerPage}"`) itemsPerPage = 10 } let page = toNumber(req.query.page, 0) if (page < 0) { Logger.warn(`[SessionController] Invalid "page" query string "${page}"`) page = 0 } let where = null const include = [ { model: Database.models.device } ] if (userId) { where = { userId } } else { include.push({ model: Database.userModel, attributes: ['id', 'username'] }) } const { rows, count } = await Database.playbackSessionModel.findAndCountAll({ where, include, order: [[orderKey, orderDesc]], limit: itemsPerPage, offset: itemsPerPage * page }) // Map playback sessions to old playback sessions const sessions = rows.map((session) => { const oldPlaybackSession = Database.playbackSessionModel.getOldPlaybackSession(session) if (session.user) { return { ...oldPlaybackSession, user: { id: session.user.id, username: session.user.username } } } else { return oldPlaybackSession.toJSON() } }) const payload = { total: count, numPages: Math.ceil(count / itemsPerPage), page, itemsPerPage, sessions } if (userId) { payload.userId = userId } res.json(payload) } /** * GET: /api/sessions/open * * @this {import('../routers/ApiRouter')} * * @param {RequestWithUser} req * @param {Response} res */ async getOpenSessions(req, res) { if (!req.user.isAdminOrUp) { Logger.error(`[SessionController] getOpenSessions: Non-admin user "${req.user.username}" requested open session data`) return res.sendStatus(404) } const minifiedUserObjects = await Database.userModel.getMinifiedUserObjects() const openSessions = this.playbackSessionManager.sessions.map((se) => { return { ...se.toJSON(), user: minifiedUserObjects.find((u) => u.id === se.userId) || null } }) const shareSessions = ShareManager.openSharePlaybackSessions.map((se) => se.toJSON()) res.json({ sessions: openSessions, shareSessions }) } /** * GET: /api/session/:id * * @this {import('../routers/ApiRouter')} * * @param {RequestWithUser} req * @param {Response} res */ async getOpenSession(req, res) { const libraryItem = await Database.libraryItemModel.getExpandedById(req.playbackSession.libraryItemId) const sessionForClient = req.playbackSession.toJSONForClient(libraryItem) res.json(sessionForClient) } /** * POST: /api/session/:id/sync * * @this {import('../routers/ApiRouter')} * * @param {RequestWithUser} req * @param {Response} res */ sync(req, res) { this.playbackSessionManager.syncSessionRequest(req.user, req.playbackSession, req.body, res) } /** * POST: /api/session/:id/close * * @this {import('../routers/ApiRouter')} * * @param {RequestWithUser} req * @param {Response} res */ close(req, res) { let syncData = req.body if (syncData && !Object.keys(syncData).length) syncData = null this.playbackSessionManager.closeSessionRequest(req.user, req.playbackSession, syncData, res) } /** * DELETE: /api/session/:id * * @this {import('../routers/ApiRouter')} * * @param {RequestWithUser} req * @param {Response} res */ async delete(req, res) { // if session is open then remove it const openSession = this.playbackSessionManager.getSession(req.playbackSession.id) if (openSession) { await this.playbackSessionManager.removeSession(req.playbackSession.id) } await Database.removePlaybackSession(req.playbackSession.id) res.sendStatus(200) } /** * POST: /api/sessions/batch/delete * @this import('../routers/ApiRouter') * * @typedef batchDeleteReqBody * @property {string[]} sessions * * @param {Request<{}, {}, batchDeleteReqBody, {}> & RequestUserObject} req * @param {Response} res */ async batchDelete(req, res) { if (!req.user.isAdminOrUp) { Logger.error(`[SessionController] Non-admin user "${req.user.username}" attempted to batch delete sessions`) return res.sendStatus(403) } // Validate session ids if (!req.body.sessions?.length || !Array.isArray(req.body.sessions) || req.body.sessions.some((s) => !isUUID(s))) { Logger.error(`[SessionController] Invalid request body. "sessions" array is required`, req.body) return res.status(400).send('Invalid request body. "sessions" array of session id strings is required.') } // Check if any of these sessions are open and close it for (const sessionId of req.body.sessions) { const openSession = this.playbackSessionManager.getSession(sessionId) if (openSession) { await this.playbackSessionManager.removeSession(sessionId) } } try { const sessionsRemoved = await Database.playbackSessionModel.destroy({ where: { id: req.body.sessions } }) Logger.info(`[SessionController] ${sessionsRemoved} playback sessions removed by "${req.user.username}"`) res.sendStatus(200) } catch (error) { Logger.error(`[SessionController] Failed to remove playback sessions`, error) res.status(500).send('Failed to remove sessions') } } /** * POST: /api/session/local * * @this {import('../routers/ApiRouter')} * * @param {RequestWithUser} req * @param {Response} res */ syncLocal(req, res) { this.playbackSessionManager.syncLocalSessionRequest(req, res) } /** * POST: /api/session/local-all * * @this {import('../routers/ApiRouter')} * * @param {RequestWithUser} req * @param {Response} res */ syncLocalSessions(req, res) { this.playbackSessionManager.syncLocalSessionsRequest(req, res) } /** * * @param {RequestWithUser} req * @param {Response} res * @param {NextFunction} next */ openSessionMiddleware(req, res, next) { var playbackSession = this.playbackSessionManager.getSession(req.params.id) if (!playbackSession) return res.sendStatus(404) if (playbackSession.userId !== req.user.id) { Logger.error(`[SessionController] User "${req.user.username}" attempting to access session belonging to another user "${req.params.id}"`) return res.sendStatus(404) } req.playbackSession = playbackSession next() } /** * * @param {RequestWithUser} req * @param {Response} res * @param {NextFunction} next */ async middleware(req, res, next) { const playbackSession = await Database.getPlaybackSession(req.params.id) if (!playbackSession) { Logger.error(`[SessionController] Unable to find playback session with id=${req.params.id}`) return res.sendStatus(404) } if (req.method == 'DELETE' && !req.user.canDelete) { Logger.warn(`[SessionController] 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(`[SessionController] User "${req.user.username}" attempted to update without permission`) return res.sendStatus(403) } req.playbackSession = playbackSession next() } } module.exports = new SessionController()