2021-08-18 00:01:11 +02:00
|
|
|
const Path = require('path')
|
|
|
|
const express = require('express')
|
|
|
|
const http = require('http')
|
2022-07-06 02:53:01 +02:00
|
|
|
const fs = require('./libs/fsExtra')
|
2022-07-07 02:10:25 +02:00
|
|
|
const fileUpload = require('./libs/expressFileupload')
|
2022-07-07 02:14:47 +02:00
|
|
|
const rateLimit = require('./libs/expressRateLimit')
|
2021-08-18 00:01:11 +02:00
|
|
|
|
2021-10-05 05:11:42 +02:00
|
|
|
const { version } = require('../package.json')
|
|
|
|
|
|
|
|
// Utils
|
2022-03-10 02:23:17 +01:00
|
|
|
const dbMigration = require('./utils/dbMigration')
|
2022-04-25 02:12:00 +02:00
|
|
|
const filePerms = require('./utils/filePerms')
|
2023-01-06 00:45:27 +01:00
|
|
|
const fileUtils = require('./utils/fileUtils')
|
2023-05-28 15:39:41 +02:00
|
|
|
const globals = require('./utils/globals')
|
2021-10-05 05:11:42 +02:00
|
|
|
const Logger = require('./Logger')
|
2021-10-01 01:52:32 +02:00
|
|
|
|
2021-08-18 00:01:11 +02:00
|
|
|
const Auth = require('./Auth')
|
|
|
|
const Watcher = require('./Watcher')
|
2021-12-25 01:06:17 +01:00
|
|
|
const Scanner = require('./scanner/Scanner')
|
2021-08-18 00:01:11 +02:00
|
|
|
const Db = require('./Db')
|
2022-11-24 22:53:58 +01:00
|
|
|
const SocketAuthority = require('./SocketAuthority')
|
2022-03-20 22:41:06 +01:00
|
|
|
|
2022-03-18 01:10:47 +01:00
|
|
|
const ApiRouter = require('./routers/ApiRouter')
|
|
|
|
const HlsRouter = require('./routers/HlsRouter')
|
|
|
|
const StaticRouter = require('./routers/StaticRouter')
|
2022-03-20 22:41:06 +01:00
|
|
|
|
2022-09-21 01:08:41 +02:00
|
|
|
const NotificationManager = require('./managers/NotificationManager')
|
2022-03-20 22:41:06 +01:00
|
|
|
const CoverManager = require('./managers/CoverManager')
|
2022-04-22 01:52:28 +02:00
|
|
|
const AbMergeManager = require('./managers/AbMergeManager')
|
2022-03-20 22:41:06 +01:00
|
|
|
const CacheManager = require('./managers/CacheManager')
|
|
|
|
const LogManager = require('./managers/LogManager')
|
|
|
|
const BackupManager = require('./managers/BackupManager')
|
|
|
|
const PlaybackSessionManager = require('./managers/PlaybackSessionManager')
|
|
|
|
const PodcastManager = require('./managers/PodcastManager')
|
2022-05-02 01:33:46 +02:00
|
|
|
const AudioMetadataMangaer = require('./managers/AudioMetadataManager')
|
2022-05-02 21:41:59 +02:00
|
|
|
const RssFeedManager = require('./managers/RssFeedManager')
|
2022-08-18 01:44:21 +02:00
|
|
|
const CronManager = require('./managers/CronManager')
|
2022-10-02 21:16:17 +02:00
|
|
|
const TaskManager = require('./managers/TaskManager')
|
2021-10-05 05:11:42 +02:00
|
|
|
|
2021-08-18 00:01:11 +02:00
|
|
|
class Server {
|
2022-10-01 23:07:30 +02:00
|
|
|
constructor(SOURCE, PORT, HOST, UID, GID, CONFIG_PATH, METADATA_PATH, ROUTER_BASE_PATH) {
|
2021-08-18 00:01:11 +02:00
|
|
|
this.Port = PORT
|
2022-03-17 12:06:52 +01:00
|
|
|
this.Host = HOST
|
2022-05-21 18:21:03 +02:00
|
|
|
global.Source = SOURCE
|
2023-01-05 23:44:34 +01:00
|
|
|
global.isWin = process.platform === 'win32'
|
2023-01-22 12:30:36 +01:00
|
|
|
global.Uid = isNaN(UID) ? undefined : Number(UID)
|
|
|
|
global.Gid = isNaN(GID) ? undefined : Number(GID)
|
2023-01-06 00:45:27 +01:00
|
|
|
global.ConfigPath = fileUtils.filePathToPOSIX(Path.normalize(CONFIG_PATH))
|
|
|
|
global.MetadataPath = fileUtils.filePathToPOSIX(Path.normalize(METADATA_PATH))
|
2022-10-01 23:07:30 +02:00
|
|
|
global.RouterBasePath = ROUTER_BASE_PATH
|
Implement X-Accel Redirect
This patch implements [X-Accel](https://www.nginx.com/resources/wiki/start/topics/examples/x-accel/)
redirect headers as an optional way for offloading static file delivery
from Express to Nginx, which is far better optimized for static file
delivery.
This provides a really easy to configure way for getting a huge
performance boost over delivering all files through Audiobookshelf.
How it works
------------
The way this works is basically that Audiobookshelf gets an HTTP request
for delivering a static file (let's say an audiobook). It will first
check the user is authorized and then convert the API path to a local
file path.
Now, instead of reading and delivering the file, Audiobookshelf will
return just the HTTP header with an additional `X-Accel-Redirect`
pointing to the file location on the file syste.
This header is picked up by Nginx which will then deliver the file.
Configuration
-------------
The configuration for this is very simple. You need to run Nginx as
reverse proxy and it must have access to your Audiobookshelf data
folder.
You then configure Audiobookshelf to use X-Accel by setting
`USE_X_ACCEL=/protected`. The path is the internal redirect path used by
Nginx.
In the Nginx configuration you then configure this location and map it
to the storage area to serve like this:
```
location /protected/ {
internal;
alias /;
}
```
That's all.
Impact
------
I just did a very simple performance test, downloading a 1170620819
bytes large audiobook file from another machine on the same network
like this, using `time -p` to measure how log the process took:
```sh
URL='https://url to audiobook…'
for i in `seq 1 50`
do
echo "$i"
curl -s -o /dev/null "${URL}"
done
```
This sequential test with 50 iterations and without x-accel resulted in:
```
real 413.42
user 197.11
sys 82.04
```
That is an average download speed of about 1080 MBit/s.
With X-Accel enabled, serving the files through Nginx, the same test
yielded the following results:
```
real 200.37
user 86.95
sys 29.79
```
That is an average download speed of about 2229 MBit/s, more than
doubling the previous speed.
I have also run the same test with 4 parallel processes and 25 downloads
each. Without x-accel, that test resulted in:
```
real 364.89
user 273.09
sys 112.75
```
That is an average speed of about 2448 MBit/s.
With X-Accel enabled, the parallel test also shows a significant
speedup:
```
real 167.19
user 195.62
sys 78.61
```
That is an average speed of about 5342 MBit/s.
While doing that, I also peaked at the system load which was a bit lower
when using X-Accel. Even though the system was delivering far more data.
But I just looked at the `load1` values and did not build a proper test
for that. That means, I cant provide any definitive data.
Supported Media
---------------
The current implementation works for audio files and book covers. There
are other media files which would benefit from this mechanism like feed
covers or author pictures.
But that's something for a future developer ;-)
2022-11-25 23:41:35 +01:00
|
|
|
global.XAccel = process.env.USE_X_ACCEL
|
2022-05-15 00:23:22 +02:00
|
|
|
|
2022-04-25 02:12:00 +02:00
|
|
|
if (!fs.pathExistsSync(global.ConfigPath)) {
|
|
|
|
fs.mkdirSync(global.ConfigPath)
|
|
|
|
filePerms.setDefaultDirSync(global.ConfigPath, false)
|
|
|
|
}
|
|
|
|
if (!fs.pathExistsSync(global.MetadataPath)) {
|
|
|
|
fs.mkdirSync(global.MetadataPath)
|
|
|
|
filePerms.setDefaultDirSync(global.MetadataPath, false)
|
|
|
|
}
|
2021-08-18 00:01:11 +02:00
|
|
|
|
2022-02-27 20:47:52 +01:00
|
|
|
this.db = new Db()
|
2022-03-20 22:41:06 +01:00
|
|
|
this.watcher = new Watcher()
|
2021-08-18 00:01:11 +02:00
|
|
|
this.auth = new Auth(this.db)
|
2022-03-20 22:41:06 +01:00
|
|
|
|
|
|
|
// Managers
|
2022-11-24 22:53:58 +01:00
|
|
|
this.taskManager = new TaskManager()
|
|
|
|
this.notificationManager = new NotificationManager(this.db)
|
|
|
|
this.backupManager = new BackupManager(this.db)
|
2022-02-27 20:47:52 +01:00
|
|
|
this.logManager = new LogManager(this.db)
|
|
|
|
this.cacheManager = new CacheManager()
|
2022-11-24 22:53:58 +01:00
|
|
|
this.abMergeManager = new AbMergeManager(this.db, this.taskManager)
|
|
|
|
this.playbackSessionManager = new PlaybackSessionManager(this.db)
|
2022-03-20 22:41:06 +01:00
|
|
|
this.coverManager = new CoverManager(this.db, this.cacheManager)
|
2023-03-05 12:15:36 +01:00
|
|
|
this.podcastManager = new PodcastManager(this.db, this.watcher, this.notificationManager, this.taskManager)
|
2022-11-24 22:53:58 +01:00
|
|
|
this.audioMetadataManager = new AudioMetadataMangaer(this.db, this.taskManager)
|
|
|
|
this.rssFeedManager = new RssFeedManager(this.db)
|
2022-03-20 22:41:06 +01:00
|
|
|
|
2023-05-27 21:51:03 +02:00
|
|
|
this.scanner = new Scanner(this.db, this.coverManager, this.taskManager)
|
2022-08-20 01:41:58 +02:00
|
|
|
this.cronManager = new CronManager(this.db, this.scanner, this.podcastManager)
|
2022-03-18 01:10:47 +01:00
|
|
|
|
|
|
|
// Routers
|
2022-11-24 22:53:58 +01:00
|
|
|
this.apiRouter = new ApiRouter(this)
|
|
|
|
this.hlsRouter = new HlsRouter(this.db, this.auth, this.playbackSessionManager)
|
2022-03-18 01:10:47 +01:00
|
|
|
this.staticRouter = new StaticRouter(this.db)
|
2021-10-02 01:42:48 +02:00
|
|
|
|
2021-10-31 23:55:28 +01:00
|
|
|
Logger.logManager = this.logManager
|
|
|
|
|
2021-08-18 00:01:11 +02:00
|
|
|
this.server = null
|
|
|
|
this.io = null
|
2021-09-06 01:20:29 +02:00
|
|
|
}
|
|
|
|
|
2021-10-13 03:07:42 +02:00
|
|
|
authMiddleware(req, res, next) {
|
|
|
|
this.auth.authMiddleware(req, res, next)
|
|
|
|
}
|
|
|
|
|
2021-08-18 00:01:11 +02:00
|
|
|
async init() {
|
2021-10-05 05:11:42 +02:00
|
|
|
Logger.info('[Server] Init v' + version)
|
2022-04-16 19:37:10 +02:00
|
|
|
await this.playbackSessionManager.removeOrphanStreams()
|
2021-09-23 03:40:35 +02:00
|
|
|
|
2022-11-24 23:35:26 +01:00
|
|
|
const previousVersion = await this.db.checkPreviousVersion() // Returns null if same server version
|
2022-04-20 15:31:57 +02:00
|
|
|
if (previousVersion) {
|
|
|
|
Logger.debug(`[Server] Upgraded from previous version ${previousVersion}`)
|
|
|
|
}
|
|
|
|
if (previousVersion && previousVersion.localeCompare('2.0.0') < 0) { // Old version data model migration
|
|
|
|
Logger.debug(`[Server] Previous version was < 2.0.0 - migration required`)
|
2022-03-19 16:13:10 +01:00
|
|
|
await dbMigration.migrate(this.db)
|
2022-03-16 00:57:15 +01:00
|
|
|
} else {
|
|
|
|
await this.db.init()
|
2022-03-10 02:23:17 +01:00
|
|
|
}
|
|
|
|
|
2022-07-19 00:19:16 +02:00
|
|
|
// Create token secret if does not exist (Added v2.1.0)
|
|
|
|
if (!this.db.serverSettings.tokenSecret) {
|
|
|
|
await this.auth.initTokenSecret()
|
|
|
|
}
|
|
|
|
|
2022-09-29 00:12:27 +02:00
|
|
|
await this.cleanUserData() // Remove invalid user item progress
|
2022-03-19 16:13:10 +01:00
|
|
|
await this.purgeMetadata() // Remove metadata folders without library item
|
2022-07-30 00:13:46 +02:00
|
|
|
await this.playbackSessionManager.removeInvalidSessions()
|
2022-05-15 18:19:04 +02:00
|
|
|
await this.cacheManager.ensureCachePaths()
|
2022-06-18 20:11:15 +02:00
|
|
|
|
2021-10-09 00:30:20 +02:00
|
|
|
await this.backupManager.init()
|
2021-10-31 23:55:28 +01:00
|
|
|
await this.logManager.init()
|
2022-12-31 23:58:19 +01:00
|
|
|
await this.apiRouter.checkRemoveEmptySeries(this.db.series) // Remove empty series
|
2022-06-08 01:29:43 +02:00
|
|
|
await this.rssFeedManager.init()
|
2022-08-18 01:44:21 +02:00
|
|
|
this.cronManager.init()
|
2021-10-02 01:42:48 +02:00
|
|
|
|
2022-02-24 00:52:21 +01:00
|
|
|
if (this.db.serverSettings.scannerDisableWatcher) {
|
|
|
|
Logger.info(`[Server] Watcher is disabled`)
|
|
|
|
this.watcher.disabled = true
|
|
|
|
} else {
|
2022-03-10 02:23:17 +01:00
|
|
|
this.watcher.initWatcher(this.db.libraries)
|
2022-02-24 00:52:21 +01:00
|
|
|
this.watcher.on('files', this.filesChanged.bind(this))
|
|
|
|
}
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async start() {
|
|
|
|
Logger.info('=== Starting Server ===')
|
|
|
|
await this.init()
|
|
|
|
|
|
|
|
const app = express()
|
2022-10-01 23:07:30 +02:00
|
|
|
const router = express.Router()
|
|
|
|
app.use(global.RouterBasePath, router)
|
2023-01-21 23:18:06 +01:00
|
|
|
app.disable('x-powered-by')
|
2022-10-01 23:07:30 +02:00
|
|
|
|
2021-08-18 00:01:11 +02:00
|
|
|
this.server = http.createServer(app)
|
|
|
|
|
2022-10-01 23:07:30 +02:00
|
|
|
router.use(this.auth.cors)
|
|
|
|
router.use(fileUpload())
|
|
|
|
router.use(express.urlencoded({ extended: true, limit: "5mb" }));
|
|
|
|
router.use(express.json({ limit: "5mb" }))
|
2021-08-18 00:01:11 +02:00
|
|
|
|
|
|
|
// Static path to generated nuxt
|
2021-08-24 02:37:40 +02:00
|
|
|
const distPath = Path.join(global.appRoot, '/client/dist')
|
2022-10-01 23:07:30 +02:00
|
|
|
router.use(express.static(distPath))
|
2021-10-05 05:11:42 +02:00
|
|
|
|
|
|
|
// Metadata folder static path
|
2022-10-01 23:07:30 +02:00
|
|
|
router.use('/metadata', this.authMiddleware.bind(this), express.static(global.MetadataPath))
|
2022-03-18 19:44:29 +01:00
|
|
|
|
2021-10-05 05:11:42 +02:00
|
|
|
// Static folder
|
2022-10-01 23:07:30 +02:00
|
|
|
router.use(express.static(Path.join(global.appRoot, 'static')))
|
2021-08-18 00:01:11 +02:00
|
|
|
|
2022-10-01 23:07:30 +02:00
|
|
|
router.use('/api', this.authMiddleware.bind(this), this.apiRouter.router)
|
|
|
|
router.use('/hls', this.authMiddleware.bind(this), this.hlsRouter.router)
|
|
|
|
router.use('/s', this.authMiddleware.bind(this), this.staticRouter.router)
|
2022-03-11 01:45:02 +01:00
|
|
|
|
2021-10-09 00:30:20 +02:00
|
|
|
// EBook static file routes
|
2022-10-01 23:07:30 +02:00
|
|
|
router.get('/ebook/:library/:folder/*', (req, res) => {
|
2022-11-24 23:35:26 +01:00
|
|
|
const library = this.db.libraries.find(lib => lib.id === req.params.library)
|
2021-10-09 00:30:20 +02:00
|
|
|
if (!library) return res.sendStatus(404)
|
2022-11-24 23:35:26 +01:00
|
|
|
const folder = library.folders.find(fol => fol.id === req.params.folder)
|
2021-10-09 00:30:20 +02:00
|
|
|
if (!folder) return res.status(404).send('Folder not found')
|
|
|
|
|
2023-05-28 15:39:41 +02:00
|
|
|
// Replace backslashes with forward slashes
|
|
|
|
const remainingPath = req.params['0'].replace(/\\/g, '/')
|
|
|
|
|
|
|
|
// Prevent path traversal
|
|
|
|
// e.g. ../../etc/passwd
|
|
|
|
if (/\/?\.?\.\//.test(remainingPath)) {
|
|
|
|
Logger.error(`[Server] Invalid path to get ebook "${remainingPath}"`)
|
|
|
|
return res.sendStatus(403)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check file ext is a valid ebook file
|
|
|
|
const filext = (Path.extname(remainingPath) || '').slice(1).toLowerCase()
|
|
|
|
if (!globals.SupportedEbookTypes.includes(filext)) {
|
|
|
|
Logger.error(`[Server] Invalid ebook file ext requested "${remainingPath}"`)
|
|
|
|
return res.sendStatus(403)
|
|
|
|
}
|
|
|
|
|
2022-11-24 23:35:26 +01:00
|
|
|
const fullPath = Path.join(folder.fullPath, remainingPath)
|
2021-10-09 00:30:20 +02:00
|
|
|
res.sendFile(fullPath)
|
|
|
|
})
|
|
|
|
|
2022-05-02 21:41:59 +02:00
|
|
|
// RSS Feed temp route
|
2022-10-01 23:07:30 +02:00
|
|
|
router.get('/feed/:id', (req, res) => {
|
2022-06-08 01:29:43 +02:00
|
|
|
Logger.info(`[Server] Requesting rss feed ${req.params.id}`)
|
2022-05-02 21:41:59 +02:00
|
|
|
this.rssFeedManager.getFeed(req, res)
|
|
|
|
})
|
2022-10-01 23:07:30 +02:00
|
|
|
router.get('/feed/:id/cover', (req, res) => {
|
2022-05-02 23:42:30 +02:00
|
|
|
this.rssFeedManager.getFeedCover(req, res)
|
|
|
|
})
|
2022-10-01 23:07:30 +02:00
|
|
|
router.get('/feed/:id/item/:episodeId/*', (req, res) => {
|
2022-06-08 01:29:43 +02:00
|
|
|
Logger.debug(`[Server] Requesting rss feed episode ${req.params.id}/${req.params.episodeId}`)
|
2022-05-02 21:41:59 +02:00
|
|
|
this.rssFeedManager.getFeedItem(req, res)
|
|
|
|
})
|
|
|
|
|
2021-11-13 02:43:16 +01:00
|
|
|
// Client dynamic routes
|
2022-05-14 20:08:56 +02:00
|
|
|
const dyanimicRoutes = [
|
|
|
|
'/item/:id',
|
2022-06-01 23:29:29 +02:00
|
|
|
'/author/:id',
|
2022-05-29 19:55:14 +02:00
|
|
|
'/audiobook/:id/chapters',
|
2022-05-14 20:08:56 +02:00
|
|
|
'/audiobook/:id/edit',
|
2022-10-02 18:53:53 +02:00
|
|
|
'/audiobook/:id/manage',
|
2022-05-14 20:08:56 +02:00
|
|
|
'/library/:library',
|
|
|
|
'/library/:library/search',
|
|
|
|
'/library/:library/bookshelf/:id?',
|
|
|
|
'/library/:library/authors',
|
|
|
|
'/library/:library/series/:id?',
|
2022-09-17 22:23:33 +02:00
|
|
|
'/library/:library/podcast/search',
|
|
|
|
'/library/:library/podcast/latest',
|
2022-05-14 20:08:56 +02:00
|
|
|
'/config/users/:id',
|
2022-05-29 19:55:14 +02:00
|
|
|
'/config/users/:id/sessions',
|
2022-12-18 21:17:52 +01:00
|
|
|
'/config/item-metadata-utils/:id',
|
2022-11-27 21:23:28 +01:00
|
|
|
'/collection/:id',
|
|
|
|
'/playlist/:id'
|
2022-05-14 20:08:56 +02:00
|
|
|
]
|
2022-10-01 23:07:30 +02:00
|
|
|
dyanimicRoutes.forEach((route) => router.get(route, (req, res) => res.sendFile(Path.join(distPath, 'index.html'))))
|
2021-08-24 02:37:40 +02:00
|
|
|
|
2022-12-31 17:59:12 +01:00
|
|
|
router.post('/login', this.getLoginRateLimiter(), (req, res) => this.auth.login(req, res))
|
2022-10-01 23:07:30 +02:00
|
|
|
router.post('/logout', this.authMiddleware.bind(this), this.logout.bind(this))
|
|
|
|
router.post('/init', (req, res) => {
|
2022-05-15 00:23:22 +02:00
|
|
|
if (this.db.hasRootUser) {
|
|
|
|
Logger.error(`[Server] attempt to init server when server already has a root user`)
|
|
|
|
return res.sendStatus(500)
|
|
|
|
}
|
|
|
|
this.initializeServer(req, res)
|
|
|
|
})
|
2022-10-01 23:07:30 +02:00
|
|
|
router.get('/status', (req, res) => {
|
2022-05-15 00:23:22 +02:00
|
|
|
// status check for client to see if server has been initialized
|
|
|
|
// server has been initialized if a root user exists
|
|
|
|
const payload = {
|
2022-11-09 01:09:07 +01:00
|
|
|
isInit: this.db.hasRootUser,
|
|
|
|
language: this.db.serverSettings.language
|
2022-05-15 00:23:22 +02:00
|
|
|
}
|
|
|
|
if (!payload.isInit) {
|
|
|
|
payload.ConfigPath = global.ConfigPath
|
|
|
|
payload.MetadataPath = global.MetadataPath
|
|
|
|
}
|
|
|
|
res.json(payload)
|
|
|
|
})
|
2022-10-01 23:07:30 +02:00
|
|
|
router.get('/ping', (req, res) => {
|
2022-07-30 15:37:35 +02:00
|
|
|
Logger.info('Received ping')
|
2021-08-18 00:01:11 +02:00
|
|
|
res.json({ success: true })
|
|
|
|
})
|
2022-07-24 22:46:19 +02:00
|
|
|
app.get('/healthcheck', (req, res) => res.sendStatus(200))
|
2021-08-18 00:01:11 +02:00
|
|
|
|
|
|
|
this.server.listen(this.Port, this.Host, () => {
|
2022-12-17 22:55:53 +01:00
|
|
|
if (this.Host) Logger.info(`Listening on http://${this.Host}:${this.Port}`)
|
|
|
|
else Logger.info(`Listening on port :${this.Port}`)
|
2021-08-18 00:01:11 +02:00
|
|
|
})
|
|
|
|
|
2022-11-24 22:53:58 +01:00
|
|
|
// Start listening for socket connections
|
|
|
|
SocketAuthority.initialize(this)
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
|
|
|
|
2022-05-15 00:23:22 +02:00
|
|
|
async initializeServer(req, res) {
|
|
|
|
Logger.info(`[Server] Initializing new server`)
|
|
|
|
const newRoot = req.body.newRoot
|
|
|
|
let rootPash = newRoot.password ? await this.auth.hashPass(newRoot.password) : ''
|
|
|
|
if (!rootPash) Logger.warn(`[Server] Creating root user with no password`)
|
2022-07-19 00:19:16 +02:00
|
|
|
let rootToken = await this.auth.generateAccessToken({ userId: 'root', username: newRoot.username })
|
2022-05-15 00:23:22 +02:00
|
|
|
await this.db.createRootUser(newRoot.username, rootPash, rootToken)
|
|
|
|
|
|
|
|
res.sendStatus(200)
|
|
|
|
}
|
|
|
|
|
2021-10-05 05:11:42 +02:00
|
|
|
async filesChanged(fileUpdates) {
|
|
|
|
Logger.info('[Server]', fileUpdates.length, 'Files Changed')
|
2021-12-25 01:06:17 +01:00
|
|
|
await this.scanner.scanFilesChanged(fileUpdates)
|
2021-10-05 05:11:42 +02:00
|
|
|
}
|
|
|
|
|
2022-03-19 16:13:10 +01:00
|
|
|
// Remove unused /metadata/items/{id} folders
|
2021-10-05 05:11:42 +02:00
|
|
|
async purgeMetadata() {
|
2022-11-24 23:35:26 +01:00
|
|
|
const itemsMetadata = Path.join(global.MetadataPath, 'items')
|
2022-03-19 16:13:10 +01:00
|
|
|
if (!(await fs.pathExists(itemsMetadata))) return
|
2022-11-24 23:35:26 +01:00
|
|
|
const foldersInItemsMetadata = await fs.readdir(itemsMetadata)
|
2021-10-05 05:11:42 +02:00
|
|
|
|
2022-11-24 23:35:26 +01:00
|
|
|
let purged = 0
|
2022-03-19 16:13:10 +01:00
|
|
|
await Promise.all(foldersInItemsMetadata.map(async foldername => {
|
2022-11-24 23:35:26 +01:00
|
|
|
const hasMatchingItem = this.db.libraryItems.find(ab => ab.id === foldername)
|
2022-03-19 16:13:10 +01:00
|
|
|
if (!hasMatchingItem) {
|
2022-11-24 23:35:26 +01:00
|
|
|
const folderPath = Path.join(itemsMetadata, foldername)
|
2021-10-05 05:11:42 +02:00
|
|
|
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) {
|
2022-03-19 16:13:10 +01:00
|
|
|
Logger.info(`[Server] Purged ${purged} unused library item metadata`)
|
2021-10-05 05:11:42 +02:00
|
|
|
}
|
|
|
|
return purged
|
|
|
|
}
|
|
|
|
|
2022-09-29 00:12:27 +02:00
|
|
|
// Remove user media progress with items that no longer exist & remove seriesHideFrom that no longer exist
|
|
|
|
async cleanUserData() {
|
2021-11-04 13:59:28 +01:00
|
|
|
for (let i = 0; i < this.db.users.length; i++) {
|
2022-11-24 23:35:26 +01:00
|
|
|
const _user = this.db.users[i]
|
|
|
|
let hasUpdated = false
|
2022-09-29 00:12:27 +02:00
|
|
|
if (_user.mediaProgress.length) {
|
|
|
|
const lengthBefore = _user.mediaProgress.length
|
|
|
|
_user.mediaProgress = _user.mediaProgress.filter(mp => {
|
|
|
|
const libraryItem = this.db.libraryItems.find(li => li.id === mp.libraryItemId)
|
|
|
|
if (!libraryItem) return false
|
|
|
|
if (mp.episodeId && (libraryItem.mediaType !== 'podcast' || !libraryItem.media.checkHasEpisode(mp.episodeId))) return false // Episode not found
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
|
|
|
|
if (lengthBefore > _user.mediaProgress.length) {
|
|
|
|
Logger.debug(`[Server] Removing ${_user.mediaProgress.length - lengthBefore} media progress data from user ${_user.username}`)
|
|
|
|
hasUpdated = true
|
2021-11-04 13:59:28 +01:00
|
|
|
}
|
|
|
|
}
|
2022-09-29 00:12:27 +02:00
|
|
|
if (_user.seriesHideFromContinueListening.length) {
|
|
|
|
_user.seriesHideFromContinueListening = _user.seriesHideFromContinueListening.filter(seriesId => {
|
|
|
|
if (!this.db.series.some(se => se.id === seriesId)) { // Series removed
|
|
|
|
hasUpdated = true
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
}
|
|
|
|
if (hasUpdated) {
|
|
|
|
await this.db.updateEntity('user', _user)
|
|
|
|
}
|
2021-11-04 13:59:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-05 05:11:42 +02:00
|
|
|
// 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
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-08-18 00:01:11 +02:00
|
|
|
logout(req, res) {
|
2022-11-24 23:35:26 +01:00
|
|
|
if (req.body.socketId) {
|
|
|
|
Logger.info(`[Server] User ${req.user ? req.user.username : 'Unknown'} is logging out with socket ${req.body.socketId}`)
|
|
|
|
SocketAuthority.logout(req.body.socketId)
|
|
|
|
}
|
2021-10-23 03:08:02 +02:00
|
|
|
|
2021-08-18 00:01:11 +02:00
|
|
|
res.sendStatus(200)
|
|
|
|
}
|
|
|
|
|
|
|
|
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()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2022-03-17 12:06:52 +01:00
|
|
|
module.exports = Server
|