mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-11-04 03:17:00 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			279 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			279 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
const Logger = require('../Logger')
 | 
						|
const { reqSupportsWebp } = require('../utils/index')
 | 
						|
 | 
						|
class BookController {
 | 
						|
  constructor() { }
 | 
						|
 | 
						|
  findAll(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)
 | 
						|
  }
 | 
						|
 | 
						|
  findOne(req, res) {
 | 
						|
    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 update(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)
 | 
						|
 | 
						|
    // Book has cover and update is removing cover then purge cache
 | 
						|
    if (audiobook.cover && req.body.book && (req.body.book.cover === '' || req.body.book.cover === null)) {
 | 
						|
      await this.cacheManager.purgeCoverCache(audiobook.id)
 | 
						|
    }
 | 
						|
 | 
						|
    var hasUpdates = audiobook.update(req.body)
 | 
						|
    if (hasUpdates) {
 | 
						|
      await this.db.updateAudiobook(audiobook)
 | 
						|
      this.emitter('audiobook_updated', audiobook.toJSONExpanded())
 | 
						|
    }
 | 
						|
    res.json(audiobook.toJSON())
 | 
						|
  }
 | 
						|
 | 
						|
  async delete(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)
 | 
						|
  }
 | 
						|
 | 
						|
  // DELETE: api/books/all
 | 
						|
  async deleteAll(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)
 | 
						|
  }
 | 
						|
 | 
						|
 | 
						|
  // POST: api/books/batch/delete
 | 
						|
  async batchDelete(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)
 | 
						|
  }
 | 
						|
 | 
						|
  // POST: api/books/batch/update
 | 
						|
  async batchUpdate(req, res) {
 | 
						|
    if (!req.user.canUpdate) {
 | 
						|
      Logger.warn('User attempted to batch update without permission', req.user)
 | 
						|
      return res.sendStatus(403)
 | 
						|
    }
 | 
						|
    var updatePayloads = req.body
 | 
						|
    if (!updatePayloads || !updatePayloads.length) {
 | 
						|
      return res.sendStatus(500)
 | 
						|
    }
 | 
						|
 | 
						|
    var audiobooksUpdated = 0
 | 
						|
    var audiobooks = updatePayloads.map((up) => {
 | 
						|
      var audiobookUpdates = up.updates
 | 
						|
      var ab = this.db.audiobooks.find(_ab => _ab.id === up.id)
 | 
						|
      if (!ab) return null
 | 
						|
      var hasUpdated = ab.update(audiobookUpdates)
 | 
						|
      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].toJSONExpanded())
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    res.json({
 | 
						|
      success: true,
 | 
						|
      updates: audiobooksUpdated
 | 
						|
    })
 | 
						|
  }
 | 
						|
 | 
						|
  // POST: api/books/batch/get
 | 
						|
  async batchGet(req, res) {
 | 
						|
    var bookIds = req.body.books || []
 | 
						|
    if (!bookIds.length) {
 | 
						|
      return res.status(403).send('Invalid payload')
 | 
						|
    }
 | 
						|
    var audiobooks = this.db.audiobooks.filter(ab => bookIds.includes(ab.id)).map((ab) => ab.toJSONExpanded())
 | 
						|
    res.json(audiobooks)
 | 
						|
  }
 | 
						|
 | 
						|
  // PATCH: api/books/:id/tracks
 | 
						|
  async updateTracks(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.toJSONExpanded())
 | 
						|
    res.json(audiobook.toJSON())
 | 
						|
  }
 | 
						|
 | 
						|
  // GET: api/books/:id/stream
 | 
						|
  openStream(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)
 | 
						|
  }
 | 
						|
 | 
						|
  // POST: api/books/:id/cover
 | 
						|
  async uploadCover(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.toJSONExpanded())
 | 
						|
    res.json({
 | 
						|
      success: true,
 | 
						|
      cover: result.cover
 | 
						|
    })
 | 
						|
  }
 | 
						|
 | 
						|
  // PATCH api/books/:id/coverfile
 | 
						|
  async updateCoverFromFile(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)
 | 
						|
      await this.cacheManager.purgeCoverCache(audiobook.id)
 | 
						|
      this.emitter('audiobook_updated', audiobook.toJSONExpanded())
 | 
						|
    }
 | 
						|
 | 
						|
    if (updated) res.status(200).send('Cover updated successfully')
 | 
						|
    else res.status(200).send('No update was made to cover')
 | 
						|
  }
 | 
						|
 | 
						|
  // GET api/books/:id/cover
 | 
						|
  async getCover(req, res) {
 | 
						|
    let { query: { width, height, format }, params: { id } } = req
 | 
						|
    var audiobook = this.db.audiobooks.find(a => a.id === id)
 | 
						|
    if (!audiobook || !audiobook.book.cover) return res.sendStatus(404)
 | 
						|
 | 
						|
    // Check user can access this audiobooks library
 | 
						|
    if (!req.user.checkCanAccessLibrary(audiobook.libraryId)) {
 | 
						|
      return res.sendStatus(403)
 | 
						|
    }
 | 
						|
 | 
						|
    // Temp fix for books without a full cover path
 | 
						|
    if (audiobook.book.cover && !audiobook.book.coverFullPath) {
 | 
						|
      var isFixed = audiobook.fixFullCoverPath()
 | 
						|
      if (!isFixed) {
 | 
						|
        Logger.warn(`[BookController] Failed to fix full cover path "${audiobook.book.cover}" for "${audiobook.book.title}"`)
 | 
						|
        return res.sendStatus(404)
 | 
						|
      }
 | 
						|
      await this.db.updateEntity('audiobook', audiobook)
 | 
						|
    }
 | 
						|
 | 
						|
    const options = {
 | 
						|
      format: format || (reqSupportsWebp(req) ? 'webp' : 'jpeg'),
 | 
						|
      height: height ? parseInt(height) : null,
 | 
						|
      width: width ? parseInt(width) : null
 | 
						|
    }
 | 
						|
    return this.cacheManager.handleCoverCache(res, audiobook, options)
 | 
						|
  }
 | 
						|
 | 
						|
  // POST api/books/:id/match
 | 
						|
  async match(req, res) {
 | 
						|
    if (!req.user.canUpdate) {
 | 
						|
      Logger.warn('User attempted to match without permission', req.user)
 | 
						|
      return res.sendStatus(403)
 | 
						|
    }
 | 
						|
    var audiobook = this.db.audiobooks.find(a => a.id === req.params.id)
 | 
						|
    if (!audiobook || !audiobook.book.cover) return res.sendStatus(404)
 | 
						|
 | 
						|
    // Check user can access this audiobooks library
 | 
						|
    if (!req.user.checkCanAccessLibrary(audiobook.libraryId)) {
 | 
						|
      return res.sendStatus(403)
 | 
						|
    }
 | 
						|
 | 
						|
    var options = req.body || {}
 | 
						|
    var matchResult = await this.scanner.quickMatchBook(audiobook, options)
 | 
						|
    res.json(matchResult)
 | 
						|
  }
 | 
						|
}
 | 
						|
module.exports = new BookController() |