const Path = require('path') const express = require('express') const http = require('http') const SocketIO = require('socket.io') const fs = require('fs-extra') const fileUpload = require('express-fileupload') const rateLimit = require('express-rate-limit') const { version } = require('../package.json') // Utils const { ScanResult } = require('./utils/constants') const Logger = require('./Logger') // Classes const Auth = require('./Auth') const Watcher = require('./Watcher') const Scanner = require('./Scanner') const Db = require('./Db') const ApiController = require('./ApiController') const HlsController = require('./HlsController') const StreamManager = require('./StreamManager') const RssFeeds = require('./RssFeeds') const DownloadManager = require('./DownloadManager') const CoverController = require('./CoverController') class Server { constructor(PORT, CONFIG_PATH, METADATA_PATH, AUDIOBOOK_PATH) { this.Port = PORT this.Host = '0.0.0.0' this.ConfigPath = Path.normalize(CONFIG_PATH) this.AudiobookPath = Path.normalize(AUDIOBOOK_PATH) this.MetadataPath = Path.normalize(METADATA_PATH) fs.ensureDirSync(CONFIG_PATH) fs.ensureDirSync(METADATA_PATH) fs.ensureDirSync(AUDIOBOOK_PATH) this.db = new Db(this.ConfigPath, this.AudiobookPath) this.auth = new Auth(this.db) this.watcher = new Watcher(this.AudiobookPath) this.coverController = new CoverController(this.db, this.MetadataPath, this.AudiobookPath) this.scanner = new Scanner(this.AudiobookPath, this.MetadataPath, this.db, this.coverController, this.emitter.bind(this)) this.streamManager = new StreamManager(this.db, this.MetadataPath) this.rssFeeds = new RssFeeds(this.Port, this.db) this.downloadManager = new DownloadManager(this.db, this.MetadataPath, this.AudiobookPath, this.emitter.bind(this)) this.apiController = new ApiController(this.MetadataPath, this.db, this.scanner, this.auth, this.streamManager, this.rssFeeds, this.downloadManager, this.coverController, this.watcher, this.emitter.bind(this), this.clientEmitter.bind(this)) this.hlsController = new HlsController(this.db, this.scanner, this.auth, this.streamManager, this.emitter.bind(this), this.streamManager.StreamsPath) this.expressApp = null this.server = null this.io = null this.clients = {} this.isScanningCovers = false } get audiobooks() { return this.db.audiobooks } get libraries() { return this.db.libraries } get serverSettings() { return this.db.serverSettings } getClientsForUser(userId) { return Object.values(this.clients).filter(c => c.user && c.user.id === userId) } emitter(ev, data) { // Logger.debug('EMITTER', ev) this.io.emit(ev, data) } clientEmitter(userId, ev, data) { var clients = this.getClientsForUser(userId) if (!clients.length) { return Logger.error(`[Server] clientEmitter - no clients found for user ${userId}`) } clients.forEach((client) => { if (client.socket) { client.socket.emit(ev, data) } }) } async init() { Logger.info('[Server] Init v' + version) await this.streamManager.ensureStreamsDir() await this.streamManager.removeOrphanStreams() await this.downloadManager.removeOrphanDownloads() await this.db.init() this.auth.init() await this.purgeMetadata() this.watcher.initWatcher(this.libraries) this.watcher.on('files', this.filesChanged.bind(this)) } async start() { Logger.info('=== Starting Server ===') await this.init() const app = express() this.expressApp = app this.server = http.createServer(app) app.use(this.auth.cors) app.use(fileUpload()) app.use(express.urlencoded({ extended: true })); app.use(express.json()) // Static path to generated nuxt const distPath = Path.join(global.appRoot, '/client/dist') app.use(express.static(distPath)) // Old static path for covers app.use('/local', this.authMiddleware.bind(this), express.static(this.AudiobookPath)) // Metadata folder static path app.use('/metadata', this.authMiddleware.bind(this), express.static(this.MetadataPath)) // Downloads folder static path app.use('/downloads', this.authMiddleware.bind(this), express.static(this.downloadManager.downloadDirPath)) // Static folder app.use(express.static(Path.join(global.appRoot, 'static'))) // Static file routes app.get('/lib/:library/:folder/*', this.authMiddleware.bind(this), (req, res) => { var library = this.libraries.find(lib => lib.id === req.params.library) if (!library) return res.sendStatus(404) var folder = library.folders.find(fol => fol.id === req.params.folder) if (!folder) return res.status(404).send('Folder not found') var remainingPath = decodeURIComponent(req.params['0']) var fullPath = Path.join(folder.fullPath, remainingPath) res.sendFile(fullPath) }) // Book static file routes app.get('/s/book/:id/*', this.authMiddleware.bind(this), (req, res) => { var audiobook = this.audiobooks.find(ab => ab.id === req.params.id) if (!audiobook) return res.status(404).send('Book not found with id ' + req.params.id) var remainingPath = decodeURIComponent(req.params['0']) var fullPath = Path.join(audiobook.fullPath, remainingPath) res.sendFile(fullPath) }) // Client routes app.get('/audiobook/:id', (req, res) => res.sendFile(Path.join(distPath, 'index.html'))) app.get('/audiobook/:id/edit', (req, res) => res.sendFile(Path.join(distPath, 'index.html'))) app.get('/library/:library', (req, res) => res.sendFile(Path.join(distPath, 'index.html'))) app.get('/library/:library/bookshelf/:id?', (req, res) => res.sendFile(Path.join(distPath, 'index.html'))) app.use('/api', this.authMiddleware.bind(this), this.apiController.router) app.use('/hls', this.authMiddleware.bind(this), this.hlsController.router) // Incomplete work in progress // app.use('/feeds', this.rssFeeds.router) app.post('/upload', this.authMiddleware.bind(this), this.handleUpload.bind(this)) var loginRateLimiter = this.getLoginRateLimiter() app.post('/login', loginRateLimiter, (req, res) => this.auth.login(req, res)) app.post('/logout', this.logout.bind(this)) app.get('/ping', (req, res) => { Logger.info('Recieved ping') res.json({ success: true }) }) // Used in development to set-up streams without authentication if (process.env.NODE_ENV !== 'production') { app.use('/test-hls', this.hlsController.router) app.get('/test-stream/:id', async (req, res) => { var uri = await this.streamManager.openTestStream(this.MetadataPath, req.params.id) res.send(uri) }) app.get('/catalog.json', (req, res) => { Logger.error('Catalog request made', req.headers) res.json() }) } this.server.listen(this.Port, this.Host, () => { Logger.info(`Running on http://${this.Host}:${this.Port}`) }) this.io = new SocketIO.Server(this.server, { cors: { origin: '*', methods: ["GET", "POST"] } }) this.io.on('connection', (socket) => { this.clients[socket.id] = { id: socket.id, socket, connected_at: Date.now() } socket.sheepClient = this.clients[socket.id] Logger.info('[SOCKET] Socket Connected', socket.id) socket.on('auth', (token) => this.authenticateSocket(socket, token)) // Scanning socket.on('scan', this.scan.bind(this)) socket.on('scan_covers', this.scanCovers.bind(this)) socket.on('cancel_scan', this.cancelScan.bind(this)) socket.on('scan_audiobook', (audiobookId) => this.scanAudiobook(socket, audiobookId)) socket.on('save_metadata', (audiobookId) => this.saveMetadata(socket, audiobookId)) // Streaming socket.on('open_stream', (audiobookId) => this.streamManager.openStreamSocketRequest(socket, audiobookId)) socket.on('close_stream', () => this.streamManager.closeStreamRequest(socket)) socket.on('stream_update', (payload) => this.streamManager.streamUpdate(socket, payload)) socket.on('progress_update', (payload) => this.audiobookProgressUpdate(socket.sheepClient, payload)) // Downloading socket.on('download', (payload) => this.downloadManager.downloadSocketRequest(socket, payload)) socket.on('remove_download', (downloadId) => this.downloadManager.removeSocketRequest(socket, downloadId)) socket.on('set_log_listener', (level) => Logger.addSocketListener(socket, level)) socket.on('test', () => { socket.emit('test_received', socket.id) }) socket.on('disconnect', () => { Logger.removeSocketListener(socket.id) var _client = this.clients[socket.id] if (!_client) { Logger.warn('[SOCKET] Socket disconnect, no client ' + socket.id) } else if (!_client.user) { Logger.info('[SOCKET] Unauth socket disconnected ' + socket.id) delete this.clients[socket.id] } else { const disconnectTime = Date.now() - _client.connected_at Logger.info(`[SOCKET] Socket ${socket.id} disconnected from client "${_client.user.username}" after ${disconnectTime}ms`) delete this.clients[socket.id] } }) }) } async filesChanged(fileUpdates) { Logger.info('[Server]', fileUpdates.length, 'Files Changed') await this.scanner.filesChanged(fileUpdates) // Logger.debug('[Server] Files changed result', result) } async scan(libraryId, forceAudioFileScan = false) { Logger.info('[Server] Starting Scan') await this.scanner.scan(libraryId, forceAudioFileScan) Logger.info('[Server] Scan complete') } async scanAudiobook(socket, audiobookId) { var result = await this.scanner.scanAudiobookById(audiobookId) var scanResultName = '' for (const key in ScanResult) { if (ScanResult[key] === result) { scanResultName = key } } socket.emit('audiobook_scan_complete', scanResultName) } async scanCovers() { Logger.info('[Server] Start cover scan') this.isScanningCovers = true // this.emitter('scan_start', 'covers') var results = await this.scanner.scanCovers() this.isScanningCovers = false // this.emitter('scan_complete', { scanType: 'covers', results }) Logger.info('[Server] Cover scan complete') } cancelScan(id) { Logger.debug('[Server] Cancel scan', id) this.scanner.cancelLibraryScan[id] = true } // Generates an NFO metadata file, if no audiobookId is passed then all audiobooks are done async saveMetadata(socket, audiobookId = null) { Logger.info('[Server] Starting save metadata files') var response = await this.scanner.saveMetadata(audiobookId) Logger.info(`[Server] Finished saving metadata files Successful: ${response.success}, Failed: ${response.failed}`) socket.emit('save_metadata_complete', response) } // Remove unused /metadata/books/{id} folders async purgeMetadata() { var booksMetadata = Path.join(this.MetadataPath, 'books') var booksMetadataExists = await fs.pathExists(booksMetadata) if (!booksMetadataExists) return var foldersInBooksMetadata = await fs.readdir(booksMetadata) var purged = 0 await Promise.all(foldersInBooksMetadata.map(async foldername => { var hasMatchingAudiobook = this.audiobooks.find(ab => ab.id === foldername) if (!hasMatchingAudiobook) { var folderPath = Path.join(booksMetadata, foldername) Logger.debug(`[Server] Purging unused metadata ${folderPath}`) await fs.remove(folderPath).then(() => { purged++ }).catch((err) => { Logger.error(`[Server] Failed to delete folder path ${folderPath}`, err) }) } })) if (purged > 0) { Logger.info(`[Server] Purged ${purged} unused audiobook metadata`) } return purged } authMiddleware(req, res, next) { this.auth.authMiddleware(req, res, next) } async handleUpload(req, res) { if (!req.user.canUpload) { Logger.warn('User attempted to upload without permission', req.user) return res.sendStatus(403) } var files = Object.values(req.files) var title = req.body.title var author = req.body.author var series = req.body.series var libraryId = req.body.library var folderId = req.body.folder var library = this.db.libraries.find(lib => lib.id === libraryId) if (!library) { return res.status(500).error(`Library not found with id ${libraryId}`) } var folder = library.folders.find(fold => fold.id === folderId) if (!folder) { return res.status(500).error(`Folder not found with id ${folderId} in library ${library.name}`) } if (!files.length || !title || !author) { return res.status(500).error(`Invalid post data`) } var outputDirectory = '' if (series && series.length && series !== 'null') { outputDirectory = Path.join(folder.fullPath, author, series, title) } else { outputDirectory = Path.join(folder.fullPath, author, title) } var exists = await fs.pathExists(outputDirectory) if (exists) { Logger.error(`[Server] Upload directory "${outputDirectory}" already exists`) return res.status(500).error(`Directory "${outputDirectory}" already exists`) } await fs.ensureDir(outputDirectory) Logger.info(`Uploading ${files.length} files to`, outputDirectory) for (let i = 0; i < files.length; i++) { var file = files[i] var path = Path.join(outputDirectory, file.name) await file.mv(path).catch((error) => { Logger.error('Failed to move file', path, error) }) } res.sendStatus(200) } // First time login rate limit is hit loginLimitReached(req, res, options) { Logger.error(`[Server] Login rate limit (${options.max}) was hit for ip ${req.ip}`) options.message = 'Too many attempts. Login temporarily locked.' } getLoginRateLimiter() { return rateLimit({ windowMs: this.db.serverSettings.rateLimitLoginWindow, // 5 minutes max: this.db.serverSettings.rateLimitLoginRequests, skipSuccessfulRequests: true, onLimitReached: this.loginLimitReached }) } logout(req, res) { res.sendStatus(200) } audiobookProgressUpdate(client, progressPayload) { if (!client || !client.user) { Logger.error('[Server] audiobookProgressUpdate invalid socket client') return } client.user.updateAudiobookProgress(progressPayload.audiobookId, progressPayload) } async authenticateSocket(socket, token) { var user = await this.auth.verifyToken(token) if (!user) { Logger.error('Cannot validate socket - invalid token') return socket.emit('invalid_token') } var client = this.clients[socket.id] client.user = user if (!client.user.toJSONForBrowser) { Logger.error('Invalid user...', client.user) return } // Check if user has stream open if (client.user.stream) { Logger.info('User has stream open already', client.user.stream) client.stream = this.streamManager.getStream(client.user.stream) if (!client.stream) { Logger.error('Invalid user stream id', client.user.stream) this.streamManager.removeOrphanStreamFiles(client.user.stream) await this.db.updateUserStream(client.user.id, null) } } const initialPayload = { serverSettings: this.serverSettings.toJSON(), audiobookPath: this.AudiobookPath, metadataPath: this.MetadataPath, configPath: this.ConfigPath, user: client.user.toJSONForBrowser(), stream: client.stream || null, librariesScanning: this.scanner.librariesScanning } client.socket.emit('init', initialPayload) // Setup log listener for root user if (user.type === 'root') { Logger.addSocketListener(socket, this.db.serverSettings.logLevel || 0) } } async stop() { await this.watcher.close() Logger.info('Watcher Closed') return new Promise((resolve) => { this.server.close((err) => { if (err) { Logger.error('Failed to close server', err) } else { Logger.info('Server successfully closed') } resolve() }) }) } } module.exports = Server