mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-26 00:02:26 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			177 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			177 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| const Path = require('path')
 | |
| const fs = require('../libs/fsExtra')
 | |
| const stream = require('stream')
 | |
| const Logger = require('../Logger')
 | |
| const { resizeImage } = require('../utils/ffmpegHelpers')
 | |
| const { encodeUriPath } = require('../utils/fileUtils')
 | |
| const Database = require('../Database')
 | |
| 
 | |
| class CacheManager {
 | |
|   constructor() {
 | |
|     this.CachePath = null
 | |
|     this.CoverCachePath = null
 | |
|     this.ImageCachePath = null
 | |
|     this.ItemCachePath = null
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Create cache directory paths if they dont exist
 | |
|    */
 | |
|   async ensureCachePaths() {
 | |
|     // Creates cache paths if necessary and sets owner and permissions
 | |
|     this.CachePath = Path.join(global.MetadataPath, 'cache')
 | |
|     this.CoverCachePath = Path.join(this.CachePath, 'covers')
 | |
|     this.ImageCachePath = Path.join(this.CachePath, 'images')
 | |
|     this.ItemCachePath = Path.join(this.CachePath, 'items')
 | |
| 
 | |
|     await fs.ensureDir(this.CachePath)
 | |
|     await fs.ensureDir(this.CoverCachePath)
 | |
|     await fs.ensureDir(this.ImageCachePath)
 | |
|     await fs.ensureDir(this.ItemCachePath)
 | |
|   }
 | |
| 
 | |
|   async handleCoverCache(res, libraryItemId, options = {}) {
 | |
|     const format = options.format || 'webp'
 | |
|     const width = options.width || 400
 | |
|     const height = options.height || null
 | |
| 
 | |
|     res.type(`image/${format}`)
 | |
| 
 | |
|     const cachePath = Path.join(this.CoverCachePath, `${libraryItemId}_${width}${height ? `x${height}` : ''}`) + '.' + format
 | |
| 
 | |
|     // Cache exists
 | |
|     if (await fs.pathExists(cachePath)) {
 | |
|       if (global.XAccel) {
 | |
|         const encodedURI = encodeUriPath(global.XAccel + cachePath)
 | |
|         Logger.debug(`Use X-Accel to serve static file ${encodedURI}`)
 | |
|         return res.status(204).header({ 'X-Accel-Redirect': encodedURI }).send()
 | |
|       }
 | |
| 
 | |
|       const r = fs.createReadStream(cachePath)
 | |
|       const ps = new stream.PassThrough()
 | |
|       stream.pipeline(r, ps, (err) => {
 | |
|         if (err) {
 | |
|           console.log(err)
 | |
|           return res.sendStatus(500)
 | |
|         }
 | |
|       })
 | |
|       return ps.pipe(res)
 | |
|     }
 | |
| 
 | |
|     // Cached cover does not exist, generate it
 | |
|     const coverPath = await Database.libraryItemModel.getCoverPath(libraryItemId)
 | |
|     if (!coverPath || !(await fs.pathExists(coverPath))) {
 | |
|       return res.sendStatus(404)
 | |
|     }
 | |
| 
 | |
|     const writtenFile = await resizeImage(coverPath, cachePath, width, height)
 | |
|     if (!writtenFile) return res.sendStatus(500)
 | |
| 
 | |
|     if (global.XAccel) {
 | |
|       const encodedURI = encodeUriPath(global.XAccel + writtenFile)
 | |
|       Logger.debug(`Use X-Accel to serve static file ${encodedURI}`)
 | |
|       return res.status(204).header({ 'X-Accel-Redirect': encodedURI }).send()
 | |
|     }
 | |
| 
 | |
|     var readStream = fs.createReadStream(writtenFile)
 | |
|     readStream.pipe(res)
 | |
|   }
 | |
| 
 | |
|   purgeCoverCache(libraryItemId) {
 | |
|     return this.purgeEntityCache(libraryItemId, this.CoverCachePath)
 | |
|   }
 | |
| 
 | |
|   purgeImageCache(entityId) {
 | |
|     return this.purgeEntityCache(entityId, this.ImageCachePath)
 | |
|   }
 | |
| 
 | |
|   async purgeEntityCache(entityId, cachePath) {
 | |
|     if (!entityId || !cachePath) return []
 | |
|     return Promise.all(
 | |
|       (await fs.readdir(cachePath)).reduce((promises, file) => {
 | |
|         if (file.startsWith(entityId)) {
 | |
|           Logger.debug(`[CacheManager] Going to purge ${file}`)
 | |
|           promises.push(this.removeCache(Path.join(cachePath, file)))
 | |
|         }
 | |
|         return promises
 | |
|       }, [])
 | |
|     )
 | |
|   }
 | |
| 
 | |
|   removeCache(path) {
 | |
|     if (!path) return false
 | |
|     return fs.pathExists(path).then((exists) => {
 | |
|       if (!exists) return false
 | |
|       return fs
 | |
|         .unlink(path)
 | |
|         .then(() => true)
 | |
|         .catch((err) => {
 | |
|           Logger.error(`[CacheManager] Failed to remove cache "${path}"`, err)
 | |
|           return false
 | |
|         })
 | |
|     })
 | |
|   }
 | |
| 
 | |
|   async purgeAll() {
 | |
|     Logger.info(`[CacheManager] Purging all cache at "${this.CachePath}"`)
 | |
|     if (await fs.pathExists(this.CachePath)) {
 | |
|       await fs.remove(this.CachePath).catch((error) => {
 | |
|         Logger.error(`[CacheManager] Failed to remove cache dir "${this.CachePath}"`, error)
 | |
|       })
 | |
|     }
 | |
|     await this.ensureCachePaths()
 | |
|   }
 | |
| 
 | |
|   async purgeItems() {
 | |
|     Logger.info(`[CacheManager] Purging items cache at "${this.ItemCachePath}"`)
 | |
|     if (await fs.pathExists(this.ItemCachePath)) {
 | |
|       await fs.remove(this.ItemCachePath).catch((error) => {
 | |
|         Logger.error(`[CacheManager] Failed to remove items cache dir "${this.ItemCachePath}"`, error)
 | |
|       })
 | |
|     }
 | |
|     await this.ensureCachePaths()
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    *
 | |
|    * @param {import('express').Response} res
 | |
|    * @param {String} authorId
 | |
|    * @param {{ format?: string, width?: number, height?: number }} options
 | |
|    * @returns
 | |
|    */
 | |
|   async handleAuthorCache(res, authorId, options = {}) {
 | |
|     const format = options.format || 'webp'
 | |
|     const width = options.width || 400
 | |
|     const height = options.height || null
 | |
| 
 | |
|     res.type(`image/${format}`)
 | |
| 
 | |
|     var cachePath = Path.join(this.ImageCachePath, `${authorId}_${width}${height ? `x${height}` : ''}`) + '.' + format
 | |
| 
 | |
|     // Cache exists
 | |
|     if (await fs.pathExists(cachePath)) {
 | |
|       const r = fs.createReadStream(cachePath)
 | |
|       const ps = new stream.PassThrough()
 | |
|       stream.pipeline(r, ps, (err) => {
 | |
|         if (err) {
 | |
|           console.log(err)
 | |
|           return res.sendStatus(500)
 | |
|         }
 | |
|       })
 | |
|       return ps.pipe(res)
 | |
|     }
 | |
| 
 | |
|     const author = await Database.authorModel.findByPk(authorId)
 | |
|     if (!author || !author.imagePath || !(await fs.pathExists(author.imagePath))) {
 | |
|       return res.sendStatus(404)
 | |
|     }
 | |
| 
 | |
|     let writtenFile = await resizeImage(author.imagePath, cachePath, width, height)
 | |
|     if (!writtenFile) return res.sendStatus(500)
 | |
| 
 | |
|     var readStream = fs.createReadStream(writtenFile)
 | |
|     readStream.pipe(res)
 | |
|   }
 | |
| }
 | |
| module.exports = new CacheManager()
 |