deemixer/server/src/server.ts
RemixDev cb8ab5cfae
Added support for reverse proxy and routing managed by vue
Use --locationbase to specify the path used by the reverse proxy
Updated deezer-js to 1.3.8
2022-07-30 00:23:48 +02:00

126 lines
3.6 KiB
TypeScript

import http, { Server } from 'http'
import path from 'path'
import fs from 'fs'
import express, { Application } from 'express'
import ejs from 'ejs'
import { Server as WsServer, OPEN as WsOpen } from 'ws'
import initDebug from 'debug'
// @ts-expect-error
import deemix from 'deemix'
import { registerMiddlewares } from './middlewares'
import indexRouter from './routes'
import { getErrorCb, getListeningCb } from './helpers/server-callbacks'
import { registerApis } from './routes/api/register'
import { registerWebsocket } from './websocket'
import { logger, removeOldLogs } from './helpers/logger'
import { loadLoginCredentials } from './helpers/loginStorage'
import { Port, Listener } from './types'
import { DeemixApp } from './app'
import { normalizePort } from './helpers/port'
import { WEBUI_DIR } from './helpers/paths'
export class DeemixServer {
host: string
port: Port
locationBase: string
isSingleUser: boolean
wss: WsServer
app: Application
server: Server
deemixApp: DeemixApp
constructor(host: string, port: string, locationBase: string, singleuser: boolean = false) {
this.host = host
this.port = normalizePort(port)
this.locationBase = locationBase
this.isSingleUser = singleuser
this.wss = new WsServer({ noServer: true })
this.app = express()
this.server = http.createServer(this.app)
const listener: Listener = {
send: (key: string, data?: any) => {
const logLine = deemix.utils.formatListener(key, data)
if (logLine) logger.info(logLine)
if (['downloadInfo', 'downloadWarn'].includes(key)) return
this.wss.clients.forEach(client => {
if (client.readyState === WsOpen) {
client.send(JSON.stringify({ key, data }))
}
})
}
}
this.deemixApp = new DeemixApp(listener)
if (this.isSingleUser) loadLoginCredentials()
}
init() {
const debug = initDebug('deemix-gui:server')
this.app.set('deemix', this.deemixApp)
this.app.set('isSingleUser', this.isSingleUser)
/* === Middlewares === */
registerMiddlewares(this.app)
/* === Routes === */
this.app.use('/', indexRouter)
/* === APIs === */
registerApis(this.app)
/* === Fallback === */
this.app.get('*/favicon.ico', (_, res) => {
res.sendFile(path.join(WEBUI_DIR, 'favicon.ico'))
})
this.app.get('*/js/*', (req, res) => {
const link = req.url.substr(req.url.indexOf('/js/'))
res.sendFile(path.join(WEBUI_DIR, link))
})
this.app.get('*/fonts/*', (req, res) => {
const link = req.url.substr(req.url.indexOf('/fonts/'))
res.sendFile(path.join(WEBUI_DIR, link))
})
this.app.get('*/res/*', (req, res) => {
const link = req.url.substr(req.url.indexOf('/res/'))
res.sendFile(path.join(WEBUI_DIR, link))
})
this.app.get('*', (req, res) => {
console.log(req.url)
fs.readFile(path.join(WEBUI_DIR, 'index.ejs'), (_, html) => {
res.send(ejs.render(html.toString(), { locationBase: this.locationBase }))
})
})
/* === Config === */
this.app.set('port', this.port)
/* === Server port === */
if (process.env.NODE_ENV !== 'test') {
this.server.listen({ port: this.port, host: this.host })
}
registerWebsocket(this.wss, this.deemixApp)
/* === Server callbacks === */
this.app.on('mount', a => {
logger.info(a)
})
this.server.on('connect', () => {
logger.info('Server connected')
})
this.server.on('upgrade', (request, socket, head) => {
this.wss.handleUpgrade(request, socket, head, socket => {
this.wss.emit('connection', socket, request)
})
})
this.server.on('error', getErrorCb(this.port))
this.server.on('listening', getListeningCb(this.server, debug))
/* === Remove Old logs files === */
removeOldLogs(5)
}
}