mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2024-12-20 19:06:06 +01:00
300 lines
9.7 KiB
JavaScript
300 lines
9.7 KiB
JavaScript
const Path = require('path')
|
|
const express = require('express')
|
|
const http = require('http')
|
|
const SocketIO = require('socket.io')
|
|
const fs = require('fs-extra')
|
|
|
|
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 Logger = require('./Logger')
|
|
const { ScanResult } = require('./utils/constants')
|
|
|
|
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.auth = new Auth(this.db)
|
|
this.watcher = new Watcher(this.AudiobookPath)
|
|
this.scanner = new Scanner(this.AudiobookPath, this.MetadataPath, this.db, 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.emitter.bind(this))
|
|
this.apiController = new ApiController(this.db, this.scanner, this.auth, this.streamManager, this.rssFeeds, this.downloadManager, 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.MetadataPath)
|
|
|
|
this.server = null
|
|
this.io = null
|
|
|
|
this.clients = {}
|
|
|
|
this.isScanning = false
|
|
this.isScanningCovers = false
|
|
this.isInitialized = false
|
|
}
|
|
|
|
get audiobooks() {
|
|
return this.db.audiobooks
|
|
}
|
|
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 newFilesAdded({ dir, files }) {
|
|
Logger.info(files.length, 'New Files Added in dir', dir)
|
|
var result = await this.scanner.scanAudiobook(dir)
|
|
Logger.info('New Files Added result', result)
|
|
}
|
|
async filesRemoved({ dir, files }) {
|
|
Logger.info(files.length, 'Files Removed in dir', dir)
|
|
var result = await this.scanner.scanAudiobook(dir)
|
|
Logger.info('Files Removed result', result)
|
|
}
|
|
async filesRenamed({ dir, files }) {
|
|
Logger.info(files.length, 'Files Renamed in dir', dir)
|
|
var result = await this.scanner.scanAudiobook(dir)
|
|
Logger.info('Files Renamed result', result)
|
|
}
|
|
|
|
async scan() {
|
|
Logger.info('[Server] Starting Scan')
|
|
this.isScanning = true
|
|
this.isInitialized = true
|
|
this.emitter('scan_start', 'files')
|
|
var results = await this.scanner.scan()
|
|
this.isScanning = false
|
|
this.emitter('scan_complete', { scanType: 'files', results })
|
|
Logger.info('[Server] Scan complete')
|
|
}
|
|
|
|
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() {
|
|
if (!this.isScanningCovers && !this.isScanning) return
|
|
this.scanner.cancelScan = true
|
|
}
|
|
|
|
async init() {
|
|
Logger.info('[Server] Init')
|
|
await this.streamManager.removeOrphanStreams()
|
|
await this.downloadManager.removeOrphanDownloads()
|
|
await this.db.init()
|
|
this.auth.init()
|
|
|
|
this.watcher.initWatcher()
|
|
this.watcher.on('new_files', this.newFilesAdded.bind(this))
|
|
this.watcher.on('removed_files', this.filesRemoved.bind(this))
|
|
this.watcher.on('renamed_files', this.filesRenamed.bind(this))
|
|
}
|
|
|
|
authMiddleware(req, res, next) {
|
|
this.auth.authMiddleware(req, res, next)
|
|
}
|
|
|
|
async start() {
|
|
Logger.info('=== Starting Server ===')
|
|
|
|
await this.init()
|
|
|
|
const app = express()
|
|
|
|
this.server = http.createServer(app)
|
|
|
|
app.use(this.auth.cors)
|
|
|
|
// Static path to generated nuxt
|
|
const distPath = Path.join(global.appRoot, '/client/dist')
|
|
if (process.env.NODE_ENV === 'production') {
|
|
app.use(express.static(distPath))
|
|
app.use('/local', express.static(this.AudiobookPath))
|
|
} else {
|
|
app.use(express.static(this.AudiobookPath))
|
|
}
|
|
|
|
app.use(express.static(this.MetadataPath))
|
|
app.use(express.static(Path.join(global.appRoot, 'static')))
|
|
app.use(express.urlencoded({ extended: true }));
|
|
app.use(express.json())
|
|
|
|
// Dynamic routes are not generated on client
|
|
app.get('/audiobook/: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)
|
|
// app.use('/hls', this.hlsController.router)
|
|
app.use('/feeds', this.rssFeeds.router)
|
|
|
|
app.post('/login', (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))
|
|
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('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('download', (payload) => this.downloadManager.downloadSocketRequest(socket, payload))
|
|
socket.on('test', () => {
|
|
socket.emit('test_received', socket.id)
|
|
})
|
|
|
|
socket.on('disconnect', () => {
|
|
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]
|
|
}
|
|
})
|
|
})
|
|
}
|
|
|
|
logout(req, res) {
|
|
res.sendStatus(200)
|
|
}
|
|
|
|
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(),
|
|
isScanning: this.isScanning,
|
|
isInitialized: this.isInitialized,
|
|
audiobookPath: this.AudiobookPath,
|
|
metadataPath: this.MetadataPath,
|
|
configPath: this.ConfigPath,
|
|
user: client.user.toJSONForBrowser(),
|
|
stream: client.stream || null
|
|
}
|
|
client.socket.emit('init', initialPayload)
|
|
}
|
|
|
|
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 |