mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-04 21:20:09 +01:00
1209 lines
40 KiB
JavaScript
1209 lines
40 KiB
JavaScript
const express = require('express')
|
|
const Path = require('path')
|
|
const fs = require('fs-extra')
|
|
const date = require('date-and-time')
|
|
|
|
const Logger = require('./Logger')
|
|
const { isObject, getId } = require('./utils/index')
|
|
const audioFileScanner = require('./utils/audioFileScanner')
|
|
|
|
const BookFinder = require('./BookFinder')
|
|
const AuthorController = require('./AuthorController')
|
|
|
|
const Library = require('./objects/Library')
|
|
const User = require('./objects/User')
|
|
const UserCollection = require('./objects/UserCollection')
|
|
|
|
class ApiController {
|
|
constructor(MetadataPath, db, scanner, auth, streamManager, rssFeeds, downloadManager, coverController, backupManager, watcher, emitter, clientEmitter) {
|
|
this.db = db
|
|
this.scanner = scanner
|
|
this.auth = auth
|
|
this.streamManager = streamManager
|
|
this.rssFeeds = rssFeeds
|
|
this.downloadManager = downloadManager
|
|
this.coverController = coverController
|
|
this.backupManager = backupManager
|
|
this.watcher = watcher
|
|
this.emitter = emitter
|
|
this.clientEmitter = clientEmitter
|
|
this.MetadataPath = MetadataPath
|
|
|
|
this.bookFinder = new BookFinder()
|
|
this.authorController = new AuthorController(this.MetadataPath)
|
|
|
|
this.router = express()
|
|
this.init()
|
|
}
|
|
|
|
init() {
|
|
this.router.get('/find/covers', this.findCovers.bind(this))
|
|
this.router.get('/find/:method', this.find.bind(this))
|
|
|
|
this.router.get('/libraries', this.getLibraries.bind(this))
|
|
this.router.patch('/libraries/order', this.reorderLibraries.bind(this))
|
|
this.router.get('/library/:id/search', this.searchLibrary.bind(this))
|
|
this.router.get('/library/:id', this.getLibrary.bind(this))
|
|
this.router.delete('/library/:id', this.deleteLibrary.bind(this))
|
|
this.router.patch('/library/:id', this.updateLibrary.bind(this))
|
|
this.router.get('/library/:id/audiobooks', this.getLibraryAudiobooks.bind(this))
|
|
this.router.post('/library', this.createNewLibrary.bind(this))
|
|
|
|
this.router.get('/audiobooks', this.getAudiobooks.bind(this)) // Old route should pass library id
|
|
this.router.delete('/audiobooks', this.deleteAllAudiobooks.bind(this))
|
|
this.router.post('/audiobooks/delete', this.batchDeleteAudiobooks.bind(this))
|
|
this.router.post('/audiobooks/update', this.batchUpdateAudiobooks.bind(this))
|
|
|
|
this.router.get('/audiobook/:id', this.getAudiobook.bind(this))
|
|
this.router.delete('/audiobook/:id', this.deleteAudiobook.bind(this))
|
|
this.router.patch('/audiobook/:id/tracks', this.updateAudiobookTracks.bind(this))
|
|
this.router.post('/audiobook/:id/cover', this.uploadAudiobookCover.bind(this))
|
|
this.router.patch('/audiobook/:id/coverfile', this.updateAudiobookCoverFromFile.bind(this))
|
|
this.router.get('/audiobook/:id/match', this.matchAudiobookBook.bind(this))
|
|
this.router.get('/audiobook/:id/stream', this.openAudiobookStream.bind(this))
|
|
this.router.patch('/audiobook/:id', this.updateAudiobook.bind(this))
|
|
|
|
this.router.patch('/match/:id', this.match.bind(this))
|
|
|
|
// Old Route : Wait until refactor of mobile app to replace with path /reset-progress
|
|
this.router.delete('/user/audiobook/:id', this.resetUserAudiobookProgress.bind(this))
|
|
|
|
this.router.patch('/user/audiobook/:id/reset-progress', this.resetUserAudiobookProgress.bind(this))
|
|
this.router.patch('/user/audiobook/:id', this.updateUserAudiobookData.bind(this))
|
|
this.router.patch('/user/audiobooks', this.batchUpdateUserAudiobookData.bind(this))
|
|
|
|
this.router.patch('/user/password', this.userChangePassword.bind(this))
|
|
this.router.patch('/user/settings', this.userUpdateSettings.bind(this))
|
|
this.router.get('/users', this.getUsers.bind(this))
|
|
this.router.post('/user', this.createUser.bind(this))
|
|
this.router.get('/user/:id', this.getUser.bind(this))
|
|
this.router.get('/user/:id/listeningSessions', this.getUserListeningSessions.bind(this))
|
|
this.router.get('/user/:id/listeningStats', this.getUserListeningStats.bind(this))
|
|
this.router.patch('/user/:id', this.updateUser.bind(this))
|
|
this.router.delete('/user/:id', this.deleteUser.bind(this))
|
|
|
|
this.router.get('/collections', this.getUserCollections.bind(this))
|
|
this.router.get('/collection/:id', this.getUserCollection.bind(this))
|
|
this.router.post('/collection', this.createUserCollection.bind(this))
|
|
this.router.post('/collection/:id/book', this.addBookToUserCollection.bind(this))
|
|
this.router.delete('/collection/:id/book/:bookId', this.removeBookFromUserCollection.bind(this))
|
|
this.router.patch('/collection/:id', this.updateUserCollection.bind(this))
|
|
this.router.delete('/collection/:id', this.deleteUserCollection.bind(this))
|
|
|
|
this.router.get('/authors', this.getAuthors.bind(this))
|
|
this.router.get('/authors/search', this.searchAuthor.bind(this))
|
|
this.router.get('/authors/:id', this.getAuthor.bind(this))
|
|
this.router.post('/authors', this.createAuthor.bind(this))
|
|
this.router.patch('/authors/:id', this.updateAuthor.bind(this))
|
|
this.router.delete('/authors/:id', this.deleteAuthor.bind(this))
|
|
|
|
this.router.patch('/serverSettings', this.updateServerSettings.bind(this))
|
|
|
|
this.router.delete('/backup/:id', this.deleteBackup.bind(this))
|
|
this.router.post('/backup/upload', this.uploadBackup.bind(this))
|
|
|
|
this.router.post('/authorize', this.authorize.bind(this))
|
|
|
|
this.router.post('/feed', this.openRssFeed.bind(this))
|
|
|
|
this.router.get('/download/:id', this.download.bind(this))
|
|
|
|
this.router.get('/filesystem', this.getFileSystemPaths.bind(this))
|
|
|
|
this.router.get('/scantracks/:id', this.scanAudioTrackNums.bind(this))
|
|
|
|
this.router.get('/listeningSessions', this.getCurrentUserListeningSessions.bind(this))
|
|
this.router.get('/listeningStats', this.getCurrentUserListeningStats.bind(this))
|
|
|
|
this.router.post('/syncUserAudiobookData', this.syncUserAudiobookData.bind(this))
|
|
}
|
|
|
|
async find(req, res) {
|
|
var provider = req.query.provider || 'google'
|
|
var title = req.query.title || ''
|
|
var author = req.query.author || ''
|
|
var results = await this.bookFinder.search(provider, title, author)
|
|
res.json(results)
|
|
}
|
|
|
|
findCovers(req, res) {
|
|
this.scanner.findCovers(req, res)
|
|
}
|
|
|
|
authorize(req, res) {
|
|
if (!req.user) {
|
|
Logger.error('Invalid user in authorize')
|
|
return res.sendStatus(401)
|
|
}
|
|
res.json({ user: req.user })
|
|
}
|
|
|
|
getLibraries(req, res) {
|
|
var libraries = this.db.libraries.map(lib => lib.toJSON())
|
|
res.json(libraries)
|
|
}
|
|
|
|
async reorderLibraries(req, res) {
|
|
if (!req.user || !req.user.isRoot) {
|
|
Logger.error('[ApiController] ReorderLibraries invalid user', req.user)
|
|
return res.sendStatus(401)
|
|
}
|
|
|
|
var orderdata = req.body
|
|
var hasUpdates = false
|
|
for (let i = 0; i < orderdata.length; i++) {
|
|
var library = this.db.libraries.find(lib => lib.id === orderdata[i].id)
|
|
if (!library) {
|
|
Logger.error(`[ApiController] Invalid library not found in reorder ${orderdata[i].id}`)
|
|
return res.sendStatus(500)
|
|
}
|
|
if (library.update({ displayOrder: orderdata[i].newOrder })) {
|
|
hasUpdates = true
|
|
await this.db.updateEntity('library', library)
|
|
}
|
|
}
|
|
|
|
if (hasUpdates) {
|
|
Logger.info(`[ApiController] Updated library display orders`)
|
|
} else {
|
|
Logger.info(`[ApiController] Library orders were up to date`)
|
|
}
|
|
|
|
var libraries = this.db.libraries.map(lib => lib.toJSON())
|
|
res.json(libraries)
|
|
}
|
|
|
|
searchLibrary(req, res) {
|
|
var library = this.db.libraries.find(lib => lib.id === req.params.id)
|
|
if (!library) {
|
|
return res.status(404).send('Library not found')
|
|
}
|
|
if (!req.query.q) {
|
|
return res.status(400).send('No query string')
|
|
}
|
|
var maxResults = req.query.max || 3
|
|
|
|
var bookMatches = []
|
|
var authorMatches = {}
|
|
var seriesMatches = {}
|
|
var tagMatches = {}
|
|
|
|
var audiobooksInLibrary = this.db.audiobooks.filter(ab => ab.libraryId === library.id)
|
|
audiobooksInLibrary.forEach((ab) => {
|
|
var queryResult = ab.searchQuery(req.query.q)
|
|
if (queryResult.book) {
|
|
var bookMatchObj = {
|
|
audiobook: ab,
|
|
matchKey: queryResult.book,
|
|
matchText: queryResult.bookMatchText
|
|
}
|
|
bookMatches.push(bookMatchObj)
|
|
}
|
|
if (queryResult.authors) {
|
|
queryResult.authors.forEach((author) => {
|
|
if (!authorMatches[author]) {
|
|
authorMatches[author] = {
|
|
author: author
|
|
}
|
|
}
|
|
})
|
|
}
|
|
if (queryResult.series) {
|
|
if (!seriesMatches[queryResult.series]) {
|
|
seriesMatches[queryResult.series] = {
|
|
series: queryResult.series,
|
|
audiobooks: [ab]
|
|
}
|
|
} else {
|
|
seriesMatches[queryResult.series].audiobooks.push(ab)
|
|
}
|
|
}
|
|
if (queryResult.tags && queryResult.tags.length) {
|
|
queryResult.tags.forEach((tag) => {
|
|
if (!tagMatches[tag]) {
|
|
tagMatches[tag] = {
|
|
tag,
|
|
audiobooks: [ab]
|
|
}
|
|
} else {
|
|
tagMatches[tag].audiobooks.push(ab)
|
|
}
|
|
})
|
|
}
|
|
})
|
|
|
|
res.json({
|
|
audiobooks: bookMatches.slice(0, maxResults),
|
|
tags: Object.values(tagMatches).slice(0, maxResults),
|
|
authors: Object.values(authorMatches).slice(0, maxResults),
|
|
series: Object.values(seriesMatches).slice(0, maxResults)
|
|
})
|
|
}
|
|
|
|
getLibrary(req, res) {
|
|
var library = this.db.libraries.find(lib => lib.id === req.params.id)
|
|
if (!library) {
|
|
return res.status(404).send('Library not found')
|
|
}
|
|
return res.json(library.toJSON())
|
|
}
|
|
|
|
async deleteLibrary(req, res) {
|
|
var library = this.db.libraries.find(lib => lib.id === req.params.id)
|
|
if (!library) {
|
|
return res.status(404).send('Library not found')
|
|
}
|
|
|
|
// Remove library watcher
|
|
this.watcher.removeLibrary(library)
|
|
|
|
// Remove audiobooks in this library
|
|
var audiobooks = this.db.audiobooks.filter(ab => ab.libraryId === library.id)
|
|
Logger.info(`[Server] deleting library "${library.name}" with ${audiobooks.length} audiobooks"`)
|
|
for (let i = 0; i < audiobooks.length; i++) {
|
|
await this.handleDeleteAudiobook(audiobooks[i])
|
|
}
|
|
|
|
var libraryJson = library.toJSON()
|
|
await this.db.removeEntity('library', library.id)
|
|
this.emitter('library_removed', libraryJson)
|
|
return res.json(libraryJson)
|
|
}
|
|
|
|
async updateLibrary(req, res) {
|
|
var library = this.db.libraries.find(lib => lib.id === req.params.id)
|
|
if (!library) {
|
|
return res.status(404).send('Library not found')
|
|
}
|
|
var hasUpdates = library.update(req.body)
|
|
if (hasUpdates) {
|
|
// Update watcher
|
|
this.watcher.updateLibrary(library)
|
|
|
|
// Remove audiobooks no longer in library
|
|
var audiobooksToRemove = this.db.audiobooks.filter(ab => !library.checkFullPathInLibrary(ab.fullPath))
|
|
if (audiobooksToRemove.length) {
|
|
Logger.info(`[Scanner] Updating library, removing ${audiobooksToRemove.length} audiobooks`)
|
|
for (let i = 0; i < audiobooksToRemove.length; i++) {
|
|
await this.handleDeleteAudiobook(audiobooksToRemove[i])
|
|
}
|
|
}
|
|
await this.db.updateEntity('library', library)
|
|
this.emitter('library_updated', library.toJSON())
|
|
}
|
|
return res.json(library.toJSON())
|
|
}
|
|
|
|
getLibraryAudiobooks(req, res) {
|
|
var libraryId = req.params.id
|
|
var library = this.db.libraries.find(lib => lib.id === libraryId)
|
|
if (!library) {
|
|
return res.status(400).send('Library does not exist')
|
|
}
|
|
|
|
var audiobooks = []
|
|
if (req.query.q) {
|
|
audiobooks = this.db.audiobooks.filter(ab => {
|
|
return ab.libraryId === libraryId && ab.isSearchMatch(req.query.q)
|
|
}).map(ab => ab.toJSONMinified())
|
|
} else {
|
|
audiobooks = this.db.audiobooks.filter(ab => ab.libraryId === libraryId).map(ab => ab.toJSONMinified())
|
|
}
|
|
res.json(audiobooks)
|
|
}
|
|
|
|
async createNewLibrary(req, res) {
|
|
var newLibraryPayload = {
|
|
...req.body
|
|
}
|
|
if (!newLibraryPayload.name || !newLibraryPayload.folders || !newLibraryPayload.folders.length) {
|
|
return res.status(500).send('Invalid request')
|
|
}
|
|
|
|
var library = new Library()
|
|
newLibraryPayload.displayOrder = this.db.libraries.length + 1
|
|
library.setData(newLibraryPayload)
|
|
await this.db.insertEntity('library', library)
|
|
this.emitter('library_added', library.toJSON())
|
|
|
|
// Add library watcher
|
|
this.watcher.addLibrary(library)
|
|
|
|
res.json(library)
|
|
}
|
|
|
|
getAudiobooks(req, res) {
|
|
var audiobooks = []
|
|
if (req.query.q) {
|
|
audiobooks = this.db.audiobooks.filter(ab => {
|
|
return ab.isSearchMatch(req.query.q)
|
|
}).map(ab => ab.toJSONMinified())
|
|
} else {
|
|
audiobooks = this.db.audiobooks.map(ab => ab.toJSONMinified())
|
|
}
|
|
res.json(audiobooks)
|
|
}
|
|
|
|
async deleteAllAudiobooks(req, res) {
|
|
if (!req.user.isRoot) {
|
|
Logger.warn('User other than root attempted to delete all audiobooks', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
Logger.info('Removing all Audiobooks')
|
|
var success = await this.db.recreateAudiobookDb()
|
|
if (success) res.sendStatus(200)
|
|
else res.sendStatus(500)
|
|
}
|
|
|
|
getAudiobook(req, res) {
|
|
if (!req.user) {
|
|
return res.sendStatus(403)
|
|
}
|
|
var audiobook = this.db.audiobooks.find(a => a.id === req.params.id)
|
|
if (!audiobook) return res.sendStatus(404)
|
|
|
|
// Check user can access this audiobooks library
|
|
if (!req.user.checkCanAccessLibrary(audiobook.libraryId)) {
|
|
return res.sendStatus(403)
|
|
}
|
|
|
|
res.json(audiobook.toJSONExpanded())
|
|
}
|
|
|
|
async handleDeleteAudiobook(audiobook) {
|
|
// Remove audiobook from users
|
|
for (let i = 0; i < this.db.users.length; i++) {
|
|
var user = this.db.users[i]
|
|
var madeUpdates = user.deleteAudiobookData(audiobook.id)
|
|
if (madeUpdates) {
|
|
await this.db.updateEntity('user', user)
|
|
}
|
|
}
|
|
|
|
// remove any streams open for this audiobook
|
|
var streams = this.streamManager.streams.filter(stream => stream.audiobookId === audiobook.id)
|
|
for (let i = 0; i < streams.length; i++) {
|
|
var stream = streams[i]
|
|
var client = stream.client
|
|
await stream.close()
|
|
if (client && client.user) {
|
|
client.user.stream = null
|
|
client.stream = null
|
|
this.db.updateUserStream(client.user.id, null)
|
|
}
|
|
}
|
|
|
|
// remove book from collections
|
|
var collectionsWithBook = this.db.collections.filter(c => c.books.includes(audiobook.id))
|
|
for (let i = 0; i < collectionsWithBook.length; i++) {
|
|
var collection = collectionsWithBook[i]
|
|
collection.removeBook(audiobook.id)
|
|
await this.db.updateEntity('collection', collection)
|
|
this.clientEmitter(collection.userId, 'collection_updated', collection.toJSONExpanded(this.db.audiobooks))
|
|
}
|
|
|
|
var audiobookJSON = audiobook.toJSONMinified()
|
|
await this.db.removeEntity('audiobook', audiobook.id)
|
|
this.emitter('audiobook_removed', audiobookJSON)
|
|
}
|
|
|
|
async deleteAudiobook(req, res) {
|
|
if (!req.user.canDelete) {
|
|
Logger.warn('User attempted to delete without permission', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
var audiobook = this.db.audiobooks.find(a => a.id === req.params.id)
|
|
if (!audiobook) return res.sendStatus(404)
|
|
|
|
await this.handleDeleteAudiobook(audiobook)
|
|
res.sendStatus(200)
|
|
}
|
|
|
|
async batchDeleteAudiobooks(req, res) {
|
|
if (!req.user.canDelete) {
|
|
Logger.warn('User attempted to delete without permission', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
var { audiobookIds } = req.body
|
|
if (!audiobookIds || !audiobookIds.length) {
|
|
return res.sendStatus(500)
|
|
}
|
|
|
|
var audiobooksToDelete = this.db.audiobooks.filter(ab => audiobookIds.includes(ab.id))
|
|
if (!audiobooksToDelete.length) {
|
|
return res.sendStatus(404)
|
|
}
|
|
for (let i = 0; i < audiobooksToDelete.length; i++) {
|
|
Logger.info(`[ApiController] Deleting Audiobook "${audiobooksToDelete[i].title}"`)
|
|
await this.handleDeleteAudiobook(audiobooksToDelete[i])
|
|
}
|
|
res.sendStatus(200)
|
|
}
|
|
|
|
async batchUpdateAudiobooks(req, res) {
|
|
if (!req.user.canUpdate) {
|
|
Logger.warn('User attempted to batch update without permission', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
var audiobooks = req.body
|
|
if (!audiobooks || !audiobooks.length) {
|
|
return res.sendStatus(500)
|
|
}
|
|
|
|
var audiobooksUpdated = 0
|
|
audiobooks = audiobooks.map((ab) => {
|
|
var _ab = this.db.audiobooks.find(__ab => __ab.id === ab.id)
|
|
if (!_ab) return null
|
|
var hasUpdated = _ab.update(ab)
|
|
if (!hasUpdated) return null
|
|
audiobooksUpdated++
|
|
return _ab
|
|
}).filter(ab => ab)
|
|
|
|
if (audiobooksUpdated) {
|
|
Logger.info(`[ApiController] ${audiobooksUpdated} Audiobooks have updates`)
|
|
for (let i = 0; i < audiobooks.length; i++) {
|
|
await this.db.updateAudiobook(audiobooks[i])
|
|
this.emitter('audiobook_updated', audiobooks[i].toJSONMinified())
|
|
}
|
|
}
|
|
|
|
res.json({
|
|
success: true,
|
|
updates: audiobooksUpdated
|
|
})
|
|
}
|
|
|
|
async updateAudiobookTracks(req, res) {
|
|
if (!req.user.canUpdate) {
|
|
Logger.warn('User attempted to update audiotracks without permission', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
var audiobook = this.db.audiobooks.find(a => a.id === req.params.id)
|
|
if (!audiobook) return res.sendStatus(404)
|
|
var orderedFileData = req.body.orderedFileData
|
|
Logger.info(`Updating audiobook tracks called ${audiobook.id}`)
|
|
audiobook.updateAudioTracks(orderedFileData)
|
|
await this.db.updateAudiobook(audiobook)
|
|
this.emitter('audiobook_updated', audiobook.toJSONMinified())
|
|
res.json(audiobook.toJSON())
|
|
}
|
|
|
|
async uploadAudiobookCover(req, res) {
|
|
if (!req.user.canUpload || !req.user.canUpdate) {
|
|
Logger.warn('User attempted to upload a cover without permission', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
|
|
var audiobookId = req.params.id
|
|
var audiobook = this.db.audiobooks.find(ab => ab.id === audiobookId)
|
|
if (!audiobook) {
|
|
return res.status(404).send('Audiobook not found')
|
|
}
|
|
|
|
var result = null
|
|
if (req.body && req.body.url) {
|
|
Logger.debug(`[ApiController] Requesting download cover from url "${req.body.url}"`)
|
|
result = await this.coverController.downloadCoverFromUrl(audiobook, req.body.url)
|
|
} else if (req.files && req.files.cover) {
|
|
Logger.debug(`[ApiController] Handling uploaded cover`)
|
|
var coverFile = req.files.cover
|
|
result = await this.coverController.uploadCover(audiobook, coverFile)
|
|
} else {
|
|
return res.status(400).send('Invalid request no file or url')
|
|
}
|
|
|
|
if (result && result.error) {
|
|
return res.status(400).send(result.error)
|
|
} else if (!result || !result.cover) {
|
|
return res.status(500).send('Unknown error occurred')
|
|
}
|
|
|
|
await this.db.updateAudiobook(audiobook)
|
|
this.emitter('audiobook_updated', audiobook.toJSONMinified())
|
|
res.json({
|
|
success: true,
|
|
cover: result.cover
|
|
})
|
|
}
|
|
|
|
async updateAudiobookCoverFromFile(req, res) {
|
|
if (!req.user.canUpdate) {
|
|
Logger.warn('User attempted to update without permission', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
var audiobook = this.db.audiobooks.find(a => a.id === req.params.id)
|
|
if (!audiobook) return res.sendStatus(404)
|
|
|
|
var coverFile = req.body
|
|
var updated = await audiobook.setCoverFromFile(coverFile)
|
|
|
|
if (updated) {
|
|
await this.db.updateAudiobook(audiobook)
|
|
this.emitter('audiobook_updated', audiobook.toJSONMinified())
|
|
}
|
|
|
|
if (updated) res.status(200).send('Cover updated successfully')
|
|
else res.status(200).send('No update was made to cover')
|
|
}
|
|
|
|
async matchAudiobookBook(req, res) {
|
|
var audiobook = this.db.audiobooks.find(a => a.id === req.params.id)
|
|
if (!audiobook) return res.sendStatus(404)
|
|
|
|
var provider = req.query.provider || 'google'
|
|
var excludeAuthor = req.query.excludeAuthor === '1'
|
|
var authorSearch = excludeAuthor ? null : audiobook.authorFL
|
|
|
|
var results = await this.bookFinder.search(provider, audiobook.title, authorSearch)
|
|
res.json(results)
|
|
}
|
|
|
|
async openAudiobookStream(req, res) {
|
|
var audiobook = this.db.audiobooks.find(a => a.id === req.params.id)
|
|
if (!audiobook) return res.sendStatus(404)
|
|
|
|
this.streamManager.openStreamApiRequest(res, req.user, audiobook)
|
|
}
|
|
|
|
async updateAudiobook(req, res) {
|
|
if (!req.user.canUpdate) {
|
|
Logger.warn('User attempted to update without permission', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
var audiobook = this.db.audiobooks.find(a => a.id === req.params.id)
|
|
if (!audiobook) return res.sendStatus(404)
|
|
var hasUpdates = audiobook.update(req.body)
|
|
if (hasUpdates) {
|
|
await this.db.updateAudiobook(audiobook)
|
|
}
|
|
this.emitter('audiobook_updated', audiobook.toJSONMinified())
|
|
res.json(audiobook.toJSON())
|
|
}
|
|
|
|
async match(req, res) {
|
|
var body = req.body
|
|
var audiobookId = req.params.id
|
|
var audiobook = this.db.audiobooks.find(ab => ab.id === audiobookId)
|
|
var bookData = {
|
|
olid: body.id,
|
|
publish_year: body.first_publish_year,
|
|
description: body.description,
|
|
title: body.title,
|
|
author: body.author,
|
|
cover: body.cover
|
|
}
|
|
audiobook.setBook(bookData)
|
|
await this.db.updateAudiobook(audiobook)
|
|
|
|
this.emitter('audiobook_updated', audiobook.toJSONMinified())
|
|
|
|
res.sendStatus(200)
|
|
}
|
|
|
|
async resetUserAudiobookProgress(req, res) {
|
|
var audiobook = this.db.audiobooks.find(ab => ab.id === req.params.id)
|
|
if (!audiobook) {
|
|
return res.status(404).send('Audiobook not found')
|
|
}
|
|
req.user.resetAudiobookProgress(audiobook)
|
|
await this.db.updateEntity('user', req.user)
|
|
|
|
var userAudiobookData = req.user.audiobooks[audiobook.id]
|
|
if (userAudiobookData) {
|
|
this.clientEmitter(req.user.id, 'current_user_audiobook_update', { id: audiobook.id, data: userAudiobookData })
|
|
}
|
|
|
|
this.clientEmitter(req.user.id, 'user_updated', req.user.toJSONForBrowser())
|
|
res.sendStatus(200)
|
|
}
|
|
|
|
async updateUserAudiobookData(req, res) {
|
|
var audiobook = this.db.audiobooks.find(ab => ab.id === req.params.id)
|
|
if (!audiobook) {
|
|
return res.status(404).send('Audiobook not found')
|
|
}
|
|
var wasUpdated = req.user.updateAudiobookData(audiobook, req.body)
|
|
if (wasUpdated) {
|
|
await this.db.updateEntity('user', req.user)
|
|
this.clientEmitter(req.user.id, 'user_updated', req.user.toJSONForBrowser())
|
|
}
|
|
res.sendStatus(200)
|
|
}
|
|
|
|
async batchUpdateUserAudiobookData(req, res) {
|
|
var userAbDataPayloads = req.body
|
|
if (!userAbDataPayloads || !userAbDataPayloads.length) {
|
|
return res.sendStatus(500)
|
|
}
|
|
|
|
var shouldUpdate = false
|
|
userAbDataPayloads.forEach((userAbData) => {
|
|
var audiobook = this.db.audiobooks.find(ab => ab.id === userAbData.audiobookId)
|
|
if (audiobook) {
|
|
var wasUpdated = req.user.updateAudiobookData(audiobook, userAbData)
|
|
if (wasUpdated) shouldUpdate = true
|
|
}
|
|
})
|
|
|
|
if (shouldUpdate) {
|
|
await this.db.updateEntity('user', req.user)
|
|
this.clientEmitter(req.user.id, 'user_updated', req.user.toJSONForBrowser())
|
|
}
|
|
|
|
res.sendStatus(200)
|
|
}
|
|
|
|
userChangePassword(req, res) {
|
|
this.auth.userChangePassword(req, res)
|
|
}
|
|
|
|
async openRssFeed(req, res) {
|
|
var audiobookId = req.body.audiobookId
|
|
var audiobook = this.db.audiobooks.find(ab => ab.id === audiobookId)
|
|
if (!audiobook) return res.sendStatus(404)
|
|
var feed = await this.rssFeeds.openFeed(audiobook)
|
|
console.log('Feed open', feed)
|
|
res.json(feed)
|
|
}
|
|
|
|
async userUpdateSettings(req, res) {
|
|
var settingsUpdate = req.body
|
|
if (!settingsUpdate || !isObject(settingsUpdate)) {
|
|
return res.sendStatus(500)
|
|
}
|
|
var madeUpdates = req.user.updateSettings(settingsUpdate)
|
|
if (madeUpdates) {
|
|
await this.db.updateEntity('user', req.user)
|
|
}
|
|
return res.json({
|
|
success: true,
|
|
settings: req.user.settings
|
|
})
|
|
}
|
|
|
|
userJsonWithBookProgressDetails(user) {
|
|
var json = user.toJSONForBrowser()
|
|
|
|
// User audiobook progress attach book details
|
|
if (json.audiobooks && Object.keys(json.audiobooks).length) {
|
|
for (const audiobookId in json.audiobooks) {
|
|
var audiobook = this.db.audiobooks.find(ab => ab.id === audiobookId)
|
|
if (!audiobook) {
|
|
Logger.error('[ApiController] Audiobook not found for users progress ' + audiobookId)
|
|
} else {
|
|
json.audiobooks[audiobookId].book = audiobook.book.toJSON()
|
|
}
|
|
}
|
|
}
|
|
|
|
return json
|
|
}
|
|
|
|
getUsers(req, res) {
|
|
if (req.user.type !== 'root') return res.sendStatus(403)
|
|
var users = this.db.users.map(u => this.userJsonWithBookProgressDetails(u))
|
|
res.json(users)
|
|
}
|
|
|
|
async createUser(req, res) {
|
|
if (!req.user.isRoot) {
|
|
Logger.warn('Non-root user attempted to create user', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
var account = req.body
|
|
|
|
var username = account.username
|
|
var usernameExists = this.db.users.find(u => u.username.toLowerCase() === username.toLowerCase())
|
|
if (usernameExists) {
|
|
return res.status(500).send('Username already taken')
|
|
}
|
|
|
|
account.id = getId('usr')
|
|
account.pash = await this.auth.hashPass(account.password)
|
|
delete account.password
|
|
account.token = await this.auth.generateAccessToken({ userId: account.id })
|
|
account.createdAt = Date.now()
|
|
var newUser = new User(account)
|
|
var success = await this.db.insertEntity('user', newUser)
|
|
if (success) {
|
|
this.clientEmitter(req.user.id, 'user_added', newUser)
|
|
res.json({
|
|
user: newUser.toJSONForBrowser()
|
|
})
|
|
} else {
|
|
return res.status(500).send('Failed to save new user')
|
|
}
|
|
}
|
|
|
|
async getUser(req, res) {
|
|
if (!req.user.isRoot) {
|
|
Logger.error('User other than root attempting to get user', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
|
|
var user = this.db.users.find(u => u.id === req.params.id)
|
|
if (!user) {
|
|
return res.sendStatus(404)
|
|
}
|
|
|
|
res.json(this.userJsonWithBookProgressDetails(user))
|
|
}
|
|
|
|
async updateUser(req, res) {
|
|
if (!req.user.isRoot) {
|
|
Logger.error('User other than root attempting to update user', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
|
|
var user = this.db.users.find(u => u.id === req.params.id)
|
|
if (!user) {
|
|
return res.sendStatus(404)
|
|
}
|
|
|
|
var account = req.body
|
|
|
|
if (account.username !== undefined && account.username !== user.username) {
|
|
var usernameExists = this.db.users.find(u => u.username.toLowerCase() === account.username.toLowerCase())
|
|
if (usernameExists) {
|
|
return res.status(500).send('Username already taken')
|
|
}
|
|
}
|
|
|
|
// Updating password
|
|
if (account.password) {
|
|
account.pash = await this.auth.hashPass(account.password)
|
|
delete account.password
|
|
}
|
|
|
|
var hasUpdated = user.update(account)
|
|
if (hasUpdated) {
|
|
await this.db.updateEntity('user', user)
|
|
}
|
|
|
|
this.clientEmitter(req.user.id, 'user_updated', user.toJSONForBrowser())
|
|
res.json({
|
|
success: true,
|
|
user: user.toJSONForBrowser()
|
|
})
|
|
}
|
|
|
|
async deleteUser(req, res) {
|
|
if (!req.user.isRoot) {
|
|
Logger.error('User other than root attempting to delete user', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
if (req.params.id === 'root') {
|
|
return res.sendStatus(500)
|
|
}
|
|
if (req.user.id === req.params.id) {
|
|
Logger.error('Attempting to delete themselves...')
|
|
return res.sendStatus(500)
|
|
}
|
|
var user = this.db.users.find(u => u.id === req.params.id)
|
|
if (!user) {
|
|
Logger.error('User not found')
|
|
return res.json({
|
|
error: 'User not found'
|
|
})
|
|
}
|
|
|
|
// delete user collections
|
|
var userCollections = this.db.collections.filter(c => c.userId === user.id)
|
|
var collectionsToRemove = userCollections.map(uc => uc.id)
|
|
for (let i = 0; i < collectionsToRemove.length; i++) {
|
|
await this.db.removeEntity('collection', collectionsToRemove[i])
|
|
}
|
|
|
|
// Todo: check if user is logged in and cancel streams
|
|
|
|
var userJson = user.toJSONForBrowser()
|
|
await this.db.removeEntity('user', user.id)
|
|
this.clientEmitter(req.user.id, 'user_removed', userJson)
|
|
res.json({
|
|
success: true
|
|
})
|
|
}
|
|
|
|
async getUserCollections(req, res) {
|
|
var collections = this.db.collections.filter(c => c.userId === req.user.id)
|
|
var expandedCollections = collections.map(c => c.toJSONExpanded(this.db.audiobooks))
|
|
res.json(expandedCollections)
|
|
}
|
|
|
|
async getUserCollection(req, res) {
|
|
var collection = this.db.collections.find(c => c.id === req.params.id)
|
|
if (!collection) {
|
|
return res.status(404).send('Collection not found')
|
|
}
|
|
res.json(collection.toJSONExpanded(this.db.audiobooks))
|
|
}
|
|
|
|
async createUserCollection(req, res) {
|
|
var newCollection = new UserCollection()
|
|
req.body.userId = req.user.id
|
|
var success = newCollection.setData(req.body)
|
|
if (!success) {
|
|
return res.status(500).send('Invalid collection data')
|
|
}
|
|
var jsonExpanded = newCollection.toJSONExpanded(this.db.audiobooks)
|
|
await this.db.insertEntity('collection', newCollection)
|
|
this.clientEmitter(req.user.id, 'collection_added', jsonExpanded)
|
|
res.json(jsonExpanded)
|
|
}
|
|
|
|
async addBookToUserCollection(req, res) {
|
|
var collection = this.db.collections.find(c => c.id === req.params.id)
|
|
if (!collection) {
|
|
return res.status(404).send('Collection not found')
|
|
}
|
|
var audiobook = this.db.audiobooks.find(ab => ab.id === req.body.id)
|
|
if (!audiobook) {
|
|
return res.status(500).send('Book not found')
|
|
}
|
|
if (audiobook.libraryId !== collection.libraryId) {
|
|
return res.status(500).send('Book in different library')
|
|
}
|
|
if (collection.books.includes(req.body.id)) {
|
|
return res.status(500).send('Book already in collection')
|
|
}
|
|
collection.addBook(req.body.id)
|
|
var jsonExpanded = collection.toJSONExpanded(this.db.audiobooks)
|
|
await this.db.updateEntity('collection', collection)
|
|
this.clientEmitter(req.user.id, 'collection_updated', jsonExpanded)
|
|
res.json(jsonExpanded)
|
|
}
|
|
|
|
async removeBookFromUserCollection(req, res) {
|
|
var collection = this.db.collections.find(c => c.id === req.params.id)
|
|
if (!collection) {
|
|
return res.status(404).send('Collection not found')
|
|
}
|
|
|
|
if (collection.books.includes(req.params.bookId)) {
|
|
collection.removeBook(req.params.bookId)
|
|
var jsonExpanded = collection.toJSONExpanded(this.db.audiobooks)
|
|
await this.db.updateEntity('collection', collection)
|
|
this.clientEmitter(req.user.id, 'collection_updated', jsonExpanded)
|
|
}
|
|
res.json(collection.toJSONExpanded(this.db.audiobooks))
|
|
}
|
|
|
|
async updateUserCollection(req, res) {
|
|
var collection = this.db.collections.find(c => c.id === req.params.id)
|
|
if (!collection) {
|
|
return res.status(404).send('Collection not found')
|
|
}
|
|
var wasUpdated = collection.update(req.body)
|
|
var jsonExpanded = collection.toJSONExpanded(this.db.audiobooks)
|
|
if (wasUpdated) {
|
|
await this.db.updateEntity('collection', collection)
|
|
this.clientEmitter(req.user.id, 'collection_updated', jsonExpanded)
|
|
}
|
|
res.json(jsonExpanded)
|
|
}
|
|
|
|
async deleteUserCollection(req, res) {
|
|
var collection = this.db.collections.find(c => c.id === req.params.id)
|
|
if (!collection) {
|
|
return res.status(404).send('Collection not found')
|
|
}
|
|
var jsonExpanded = collection.toJSONExpanded(this.db.audiobooks)
|
|
await this.db.removeEntity('collection', collection.id)
|
|
this.clientEmitter(req.user.id, 'collection_removed', jsonExpanded)
|
|
res.sendStatus(200)
|
|
}
|
|
|
|
async getAuthors(req, res) {
|
|
var authors = this.db.authors.filter(p => p.isAuthor)
|
|
res.json(authors)
|
|
}
|
|
|
|
async getAuthor(req, res) {
|
|
var author = this.db.authors.find(p => p.id === req.params.id)
|
|
if (!author) {
|
|
return res.status(404).send('Author not found')
|
|
}
|
|
res.json(author.toJSON())
|
|
}
|
|
|
|
async searchAuthor(req, res) {
|
|
var query = req.query.q
|
|
var author = await this.authorController.findAuthorByName(query)
|
|
res.json(author)
|
|
}
|
|
|
|
async createAuthor(req, res) {
|
|
var author = await this.authorController.createAuthor(req.body)
|
|
if (!author) {
|
|
return res.status(500).send('Failed to create author')
|
|
}
|
|
|
|
await this.db.insertEntity('author', author)
|
|
this.emitter('author_added', author.toJSON())
|
|
res.json(author)
|
|
}
|
|
|
|
async updateAuthor(req, res) {
|
|
var author = this.db.authors.find(p => p.id === req.params.id)
|
|
if (!author) {
|
|
return res.status(404).send('Author not found')
|
|
}
|
|
|
|
var wasUpdated = author.update(req.body)
|
|
if (wasUpdated) {
|
|
await this.db.updateEntity('author', author)
|
|
this.emitter('author_updated', author.toJSON())
|
|
}
|
|
res.json(author)
|
|
}
|
|
|
|
async deleteAuthor(req, res) {
|
|
var author = this.db.authors.find(p => p.id === req.params.id)
|
|
if (!author) {
|
|
return res.status(404).send('Author not found')
|
|
}
|
|
|
|
var authorJson = author.toJSON()
|
|
|
|
await this.db.removeEntity('author', author.id)
|
|
this.emitter('author_removed', authorJson)
|
|
res.sendStatus(200)
|
|
}
|
|
|
|
async updateServerSettings(req, res) {
|
|
if (!req.user.isRoot) {
|
|
Logger.error('User other than root attempting to update server settings', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
var settingsUpdate = req.body
|
|
if (!settingsUpdate || !isObject(settingsUpdate)) {
|
|
return res.status(500).send('Invalid settings update object')
|
|
}
|
|
|
|
var madeUpdates = this.db.serverSettings.update(settingsUpdate)
|
|
if (madeUpdates) {
|
|
// If backup schedule is updated - update backup manager
|
|
if (settingsUpdate.backupSchedule !== undefined) {
|
|
this.backupManager.updateCronSchedule()
|
|
}
|
|
|
|
await this.db.updateEntity('settings', this.db.serverSettings)
|
|
}
|
|
return res.json({
|
|
success: true,
|
|
serverSettings: this.db.serverSettings
|
|
})
|
|
}
|
|
|
|
async deleteBackup(req, res) {
|
|
if (!req.user.isRoot) {
|
|
Logger.error(`[ApiController] Non-Root user attempting to delete backup`, req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
var backup = this.backupManager.backups.find(b => b.id === req.params.id)
|
|
if (!backup) {
|
|
return res.sendStatus(404)
|
|
}
|
|
await this.backupManager.removeBackup(backup)
|
|
res.json(this.backupManager.backups.map(b => b.toJSON()))
|
|
}
|
|
|
|
async uploadBackup(req, res) {
|
|
if (!req.user.isRoot) {
|
|
Logger.error(`[ApiController] Non-Root user attempting to upload backup`, req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
if (!req.files.file) {
|
|
Logger.error('[ApiController] Upload backup invalid')
|
|
return res.sendStatus(500)
|
|
}
|
|
this.backupManager.uploadBackup(req, res)
|
|
}
|
|
|
|
async download(req, res) {
|
|
if (!req.user.canDownload) {
|
|
Logger.error('User attempting to download without permission', req.user)
|
|
return res.sendStatus(403)
|
|
}
|
|
var downloadId = req.params.id
|
|
Logger.info('Download Request', downloadId)
|
|
var download = this.downloadManager.getDownload(downloadId)
|
|
if (!download) {
|
|
Logger.error('Download request not found', downloadId)
|
|
return res.sendStatus(404)
|
|
}
|
|
|
|
var options = {
|
|
headers: {
|
|
'Content-Type': download.mimeType
|
|
}
|
|
}
|
|
res.download(download.fullPath, download.filename, options, (err) => {
|
|
if (err) {
|
|
Logger.error('Download Error', err)
|
|
}
|
|
})
|
|
}
|
|
|
|
async getDirectories(dir, relpath, excludedDirs, level = 0) {
|
|
try {
|
|
var paths = await fs.readdir(dir)
|
|
|
|
var dirs = await Promise.all(paths.map(async dirname => {
|
|
var fullPath = Path.join(dir, dirname)
|
|
var path = Path.join(relpath, dirname)
|
|
|
|
var isDir = (await fs.lstat(fullPath)).isDirectory()
|
|
if (isDir && !excludedDirs.includes(path) && dirname !== 'node_modules') {
|
|
return {
|
|
path,
|
|
dirname,
|
|
fullPath,
|
|
level,
|
|
dirs: level < 4 ? (await this.getDirectories(fullPath, path, excludedDirs, level + 1)) : []
|
|
}
|
|
} else {
|
|
return false
|
|
}
|
|
}))
|
|
dirs = dirs.filter(d => d)
|
|
return dirs
|
|
} catch (error) {
|
|
Logger.error('Failed to readdir', dir, error)
|
|
return []
|
|
}
|
|
}
|
|
|
|
async getFileSystemPaths(req, res) {
|
|
var excludedDirs = ['node_modules', 'client', 'server', '.git', 'static', 'build', 'dist', 'metadata', 'config', 'sys', 'proc'].map(dirname => {
|
|
return Path.sep + dirname
|
|
})
|
|
|
|
// Do not include existing mapped library paths in response
|
|
this.db.libraries.forEach(lib => {
|
|
lib.folders.forEach((folder) => {
|
|
var dir = folder.fullPath
|
|
if (dir.includes(global.appRoot)) dir = dir.replace(global.appRoot, '')
|
|
excludedDirs.push(dir)
|
|
})
|
|
})
|
|
|
|
Logger.debug(`[Server] get file system paths, excluded: ${excludedDirs.join(', ')}`)
|
|
var dirs = await this.getDirectories(global.appRoot, '/', excludedDirs)
|
|
res.json(dirs)
|
|
}
|
|
|
|
async scanAudioTrackNums(req, res) {
|
|
if (!req.user || !req.user.isRoot) {
|
|
return res.sendStatus(403)
|
|
}
|
|
var audiobook = this.db.audiobooks.find(ab => ab.id === req.params.id)
|
|
if (!audiobook) {
|
|
return res.status(404).send('Audiobook not found')
|
|
}
|
|
|
|
var scandata = await audioFileScanner.scanTrackNumbers(audiobook)
|
|
res.json(scandata)
|
|
}
|
|
|
|
async getUserListeningSessionsHelper(userId) {
|
|
var userSessions = await this.db.selectUserSessions(userId)
|
|
var listeningSessions = userSessions.filter(us => us.sessionType === 'listeningSession')
|
|
return listeningSessions.sort((a, b) => b.lastUpdate - a.lastUpdate)
|
|
}
|
|
|
|
async getUserListeningSessions(req, res) {
|
|
if (!req.user || !req.user.isRoot) {
|
|
return res.sendStatus(403)
|
|
}
|
|
var listeningSessions = await this.getUserListeningSessionsHelper(req.params.id)
|
|
res.json(listeningSessions.slice(0, 10))
|
|
}
|
|
|
|
async getCurrentUserListeningSessions(req, res) {
|
|
if (!req.user) {
|
|
return res.sendStatus(500)
|
|
}
|
|
var listeningSessions = await this.getUserListeningSessionsHelper(req.user.id)
|
|
res.json(listeningSessions.slice(0, 10))
|
|
}
|
|
|
|
async getUserListeningStatsHelpers(userId) {
|
|
const today = date.format(new Date(), 'YYYY-MM-DD')
|
|
|
|
var listeningSessions = await this.getUserListeningSessionsHelper(userId)
|
|
var listeningStats = {
|
|
totalTime: 0,
|
|
books: {},
|
|
days: {},
|
|
dayOfWeek: {},
|
|
today: 0
|
|
}
|
|
listeningSessions.forEach((s) => {
|
|
if (s.dayOfWeek) {
|
|
if (!listeningStats.dayOfWeek[s.dayOfWeek]) listeningStats.dayOfWeek[s.dayOfWeek] = 0
|
|
listeningStats.dayOfWeek[s.dayOfWeek] += s.timeListening
|
|
}
|
|
if (s.date) {
|
|
if (!listeningStats.days[s.date]) listeningStats.days[s.date] = 0
|
|
listeningStats.days[s.date] += s.timeListening
|
|
|
|
if (s.date === today) {
|
|
listeningStats.today += s.timeListening
|
|
}
|
|
}
|
|
if (!listeningStats.books[s.audiobookId]) listeningStats.books[s.audiobookId] = 0
|
|
listeningStats.books[s.audiobookId] += s.timeListening
|
|
|
|
listeningStats.totalTime += s.timeListening
|
|
})
|
|
return listeningStats
|
|
}
|
|
|
|
async getUserListeningStats(req, res) {
|
|
if (!req.user || !req.user.isRoot) {
|
|
return res.sendStatus(403)
|
|
}
|
|
var listeningStats = await this.getUserListeningStatsHelpers(req.params.id)
|
|
res.json(listeningStats)
|
|
}
|
|
|
|
async getCurrentUserListeningStats(req, res) {
|
|
var listeningStats = await this.getUserListeningStatsHelpers(req.user.id)
|
|
res.json(listeningStats)
|
|
}
|
|
|
|
async syncUserAudiobookData(req, res) {
|
|
if (!req.body.data) {
|
|
return res.status(403).send('Invalid local user audiobook data')
|
|
}
|
|
|
|
var hasUpdates = false
|
|
|
|
// Local user audiobook data use the latest update
|
|
req.body.data.forEach((uab) => {
|
|
if (!uab || !uab.audiobookId) {
|
|
Logger.error('[ApiController] Invalid user audiobook data', uab)
|
|
return
|
|
}
|
|
var audiobook = this.db.audiobooks.find(ab => ab.id === uab.audiobookId)
|
|
if (!audiobook) {
|
|
Logger.info('[ApiController] syncUserAudiobookData local audiobook data audiobook no longer exists', uab.audiobookId)
|
|
return
|
|
}
|
|
if (req.user.syncLocalUserAudiobookData(uab, audiobook)) {
|
|
this.clientEmitter(req.user.id, 'current_user_audiobook_update', { id: uab.audiobookId, data: uab })
|
|
hasUpdates = true
|
|
}
|
|
})
|
|
|
|
if (hasUpdates) {
|
|
await this.db.updateEntity('user', req.user)
|
|
}
|
|
|
|
var allUserAudiobookData = Object.values(req.user.audiobooksToJSON())
|
|
res.json(allUserAudiobookData)
|
|
}
|
|
}
|
|
module.exports = ApiController |