2021-08-18 00:01:11 +02:00
|
|
|
const Path = require('path')
|
|
|
|
const njodb = require("njodb")
|
|
|
|
const jwt = require('jsonwebtoken')
|
|
|
|
const Logger = require('./Logger')
|
2021-09-04 21:17:26 +02:00
|
|
|
const Audiobook = require('./objects/Audiobook')
|
|
|
|
const User = require('./objects/User')
|
2021-09-05 02:58:39 +02:00
|
|
|
const ServerSettings = require('./objects/ServerSettings')
|
2021-08-18 00:01:11 +02:00
|
|
|
|
|
|
|
class Db {
|
|
|
|
constructor(CONFIG_PATH) {
|
|
|
|
this.ConfigPath = CONFIG_PATH
|
|
|
|
this.AudiobooksPath = Path.join(CONFIG_PATH, 'audiobooks')
|
|
|
|
this.UsersPath = Path.join(CONFIG_PATH, 'users')
|
|
|
|
this.SettingsPath = Path.join(CONFIG_PATH, 'settings')
|
|
|
|
|
|
|
|
this.audiobooksDb = new njodb.Database(this.AudiobooksPath)
|
|
|
|
this.usersDb = new njodb.Database(this.UsersPath)
|
|
|
|
this.settingsDb = new njodb.Database(this.SettingsPath, { datastores: 2 })
|
|
|
|
|
|
|
|
this.users = []
|
|
|
|
this.audiobooks = []
|
|
|
|
this.settings = []
|
2021-09-05 02:58:39 +02:00
|
|
|
|
|
|
|
this.serverSettings = null
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
getEntityDb(entityName) {
|
|
|
|
if (entityName === 'user') return this.usersDb
|
|
|
|
else if (entityName === 'audiobook') return this.audiobooksDb
|
|
|
|
return this.settingsDb
|
|
|
|
}
|
|
|
|
|
2021-08-21 23:23:35 +02:00
|
|
|
getEntityDbKey(entityName) {
|
|
|
|
if (entityName === 'user') return 'usersDb'
|
|
|
|
else if (entityName === 'audiobook') return 'audiobooksDb'
|
|
|
|
return 'settingsDb'
|
|
|
|
}
|
|
|
|
|
2021-08-18 00:01:11 +02:00
|
|
|
getEntityArrayKey(entityName) {
|
|
|
|
if (entityName === 'user') return 'users'
|
|
|
|
else if (entityName === 'audiobook') return 'audiobooks'
|
|
|
|
return 'settings'
|
|
|
|
}
|
|
|
|
|
|
|
|
getDefaultUser(token) {
|
|
|
|
return new User({
|
|
|
|
id: 'root',
|
|
|
|
type: 'root',
|
|
|
|
|
|
|
|
username: 'root',
|
|
|
|
pash: '',
|
|
|
|
stream: null,
|
|
|
|
token,
|
2021-08-27 14:01:47 +02:00
|
|
|
isActive: true,
|
2021-08-18 00:01:11 +02:00
|
|
|
createdAt: Date.now()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
async init() {
|
|
|
|
await this.load()
|
|
|
|
|
|
|
|
// Insert Defaults
|
|
|
|
if (!this.users.find(u => u.type === 'root')) {
|
|
|
|
var token = await jwt.sign({ userId: 'root' }, process.env.TOKEN_SECRET)
|
|
|
|
Logger.debug('Generated default token', token)
|
|
|
|
await this.insertUser(this.getDefaultUser(token))
|
|
|
|
}
|
2021-09-05 02:58:39 +02:00
|
|
|
|
|
|
|
if (!this.serverSettings) {
|
|
|
|
this.serverSettings = new ServerSettings()
|
|
|
|
await this.insertSettings(this.serverSettings)
|
|
|
|
}
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async load() {
|
|
|
|
var p1 = this.audiobooksDb.select(() => true).then((results) => {
|
|
|
|
this.audiobooks = results.data.map(a => new Audiobook(a))
|
2021-09-07 00:42:15 +02:00
|
|
|
Logger.info(`[DB] Audiobooks Loaded ${this.audiobooks.length}`)
|
2021-08-18 00:01:11 +02:00
|
|
|
})
|
|
|
|
var p2 = this.usersDb.select(() => true).then((results) => {
|
|
|
|
this.users = results.data.map(u => new User(u))
|
2021-09-07 00:42:15 +02:00
|
|
|
Logger.info(`[DB] Users Loaded ${this.users.length}`)
|
2021-08-18 00:01:11 +02:00
|
|
|
})
|
|
|
|
var p3 = this.settingsDb.select(() => true).then((results) => {
|
2021-09-05 02:58:39 +02:00
|
|
|
if (results.data && results.data.length) {
|
|
|
|
this.settings = results.data
|
|
|
|
var serverSettings = this.settings.find(s => s.id === 'server-settings')
|
|
|
|
if (serverSettings) {
|
|
|
|
this.serverSettings = new ServerSettings(serverSettings)
|
|
|
|
}
|
|
|
|
}
|
2021-08-18 00:01:11 +02:00
|
|
|
})
|
|
|
|
await Promise.all([p1, p2, p3])
|
|
|
|
}
|
|
|
|
|
2021-09-05 02:58:39 +02:00
|
|
|
insertSettings(settings) {
|
2021-09-06 01:20:29 +02:00
|
|
|
return this.settingsDb.insert([settings]).then((results) => {
|
2021-09-05 02:58:39 +02:00
|
|
|
Logger.debug(`[DB] Inserted ${results.inserted} settings`)
|
|
|
|
this.settings = this.settings.concat(settings)
|
|
|
|
}).catch((error) => {
|
|
|
|
Logger.error(`[DB] Insert settings Failed ${error}`)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-08-18 00:01:11 +02:00
|
|
|
insertAudiobook(audiobook) {
|
|
|
|
return this.insertAudiobooks([audiobook])
|
|
|
|
}
|
|
|
|
|
|
|
|
insertAudiobooks(audiobooks) {
|
|
|
|
return this.audiobooksDb.insert(audiobooks).then((results) => {
|
|
|
|
Logger.debug(`[DB] Inserted ${results.inserted} audiobooks`)
|
|
|
|
this.audiobooks = this.audiobooks.concat(audiobooks)
|
|
|
|
}).catch((error) => {
|
|
|
|
Logger.error(`[DB] Insert audiobooks Failed ${error}`)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
updateAudiobook(audiobook) {
|
|
|
|
return this.audiobooksDb.update((record) => record.id === audiobook.id, () => audiobook).then((results) => {
|
|
|
|
Logger.debug(`[DB] Audiobook updated ${results.updated}`)
|
2021-08-24 14:15:56 +02:00
|
|
|
return true
|
2021-08-18 00:01:11 +02:00
|
|
|
}).catch((error) => {
|
|
|
|
Logger.error(`[DB] Audiobook update failed ${error}`)
|
2021-08-24 14:15:56 +02:00
|
|
|
return false
|
2021-08-18 00:01:11 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
insertUser(user) {
|
|
|
|
return this.usersDb.insert([user]).then((results) => {
|
|
|
|
Logger.debug(`[DB] Inserted user ${results.inserted}`)
|
|
|
|
this.users.push(user)
|
2021-08-27 14:01:47 +02:00
|
|
|
return true
|
2021-08-18 00:01:11 +02:00
|
|
|
}).catch((error) => {
|
|
|
|
Logger.error(`[DB] Insert user Failed ${error}`)
|
2021-08-27 14:01:47 +02:00
|
|
|
return false
|
2021-08-18 00:01:11 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
updateUserStream(userId, streamId) {
|
|
|
|
return this.usersDb.update((record) => record.id === userId, (user) => {
|
|
|
|
user.stream = streamId
|
|
|
|
return user
|
|
|
|
}).then((results) => {
|
|
|
|
Logger.debug(`[DB] Updated user ${results.updated}`)
|
|
|
|
this.users = this.users.map(u => {
|
|
|
|
if (u.id === userId) {
|
|
|
|
u.stream = streamId
|
|
|
|
}
|
|
|
|
return u
|
|
|
|
})
|
|
|
|
}).catch((error) => {
|
|
|
|
Logger.error(`[DB] Update user Failed ${error}`)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
updateEntity(entityName, entity) {
|
|
|
|
var entityDb = this.getEntityDb(entityName)
|
|
|
|
return entityDb.update((record) => record.id === entity.id, () => entity).then((results) => {
|
|
|
|
Logger.debug(`[DB] Updated entity ${entityName}: ${results.updated}`)
|
|
|
|
var arrayKey = this.getEntityArrayKey(entityName)
|
|
|
|
this[arrayKey] = this[arrayKey].map(e => {
|
|
|
|
return e.id === entity.id ? entity : e
|
|
|
|
})
|
2021-08-22 17:46:04 +02:00
|
|
|
return true
|
2021-08-18 00:01:11 +02:00
|
|
|
}).catch((error) => {
|
|
|
|
Logger.error(`[DB] Update entity ${entityName} Failed: ${error}`)
|
2021-08-22 17:46:04 +02:00
|
|
|
return false
|
2021-08-18 00:01:11 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
removeEntity(entityName, entityId) {
|
|
|
|
var entityDb = this.getEntityDb(entityName)
|
|
|
|
return entityDb.delete((record) => record.id === entityId).then((results) => {
|
|
|
|
Logger.debug(`[DB] Deleted entity ${entityName}: ${results.deleted}`)
|
|
|
|
var arrayKey = this.getEntityArrayKey(entityName)
|
|
|
|
this[arrayKey] = this[arrayKey].filter(e => {
|
|
|
|
return e.id !== entityId
|
|
|
|
})
|
|
|
|
}).catch((error) => {
|
|
|
|
Logger.error(`[DB] Remove entity ${entityName} Failed: ${error}`)
|
|
|
|
})
|
|
|
|
}
|
2021-08-20 00:29:36 +02:00
|
|
|
|
2021-08-21 23:23:35 +02:00
|
|
|
recreateAudiobookDb() {
|
|
|
|
return this.audiobooksDb.drop().then((results) => {
|
|
|
|
Logger.info(`[DB] Dropped audiobook db`, results)
|
|
|
|
this.audiobooksDb = new njodb.Database(this.AudiobooksPath)
|
|
|
|
this.audiobooks = []
|
|
|
|
return true
|
|
|
|
}).catch((error) => {
|
|
|
|
Logger.error(`[DB] Failed to drop audiobook db`, error)
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-08-20 00:29:36 +02:00
|
|
|
getGenres() {
|
|
|
|
var allGenres = []
|
|
|
|
this.db.audiobooks.forEach((audiobook) => {
|
|
|
|
allGenres = allGenres.concat(audiobook.genres)
|
|
|
|
})
|
|
|
|
allGenres = [...new Set(allGenres)] // Removes duplicates
|
|
|
|
return allGenres
|
|
|
|
}
|
|
|
|
|
|
|
|
getTags() {
|
|
|
|
var allTags = []
|
|
|
|
this.db.audiobooks.forEach((audiobook) => {
|
|
|
|
allTags = allTags.concat(audiobook.tags)
|
|
|
|
})
|
|
|
|
allTags = [...new Set(allTags)] // Removes duplicates
|
|
|
|
return allTags
|
|
|
|
}
|
2021-08-18 00:01:11 +02:00
|
|
|
}
|
|
|
|
module.exports = Db
|