176 lines
5.5 KiB
JavaScript
176 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) {
|
|
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()
|