2021-08-27 14:32:44 +02:00
|
|
|
import cors from 'cors'
|
|
|
|
import express from 'express'
|
2021-02-16 16:25:53 +01:00
|
|
|
import { VideosTorrentCache } from '@server/lib/files-cache/videos-torrent-cache'
|
2022-06-20 11:48:40 +02:00
|
|
|
import { MActorImage } from '@server/types/models'
|
2021-07-16 10:42:24 +02:00
|
|
|
import { HttpStatusCode } from '../../shared/models/http/http-error-codes'
|
2021-02-16 16:25:53 +01:00
|
|
|
import { logger } from '../helpers/logger'
|
2022-06-20 11:48:40 +02:00
|
|
|
import { ACTOR_IMAGES_SIZE, LAZY_STATIC_PATHS, STATIC_MAX_AGE } from '../initializers/constants'
|
2023-06-01 14:51:16 +02:00
|
|
|
import { VideosCaptionCache, VideosPreviewCache, VideosStoryboardCache } from '../lib/files-cache'
|
2022-06-24 16:31:32 +02:00
|
|
|
import { actorImagePathUnsafeCache, downloadActorImageFromWorker } from '../lib/local-actor'
|
2022-07-27 14:38:07 +02:00
|
|
|
import { asyncMiddleware, handleStaticError } from '../middlewares'
|
2021-05-11 11:15:29 +02:00
|
|
|
import { ActorImageModel } from '../models/actor/actor-image'
|
2019-08-09 11:32:40 +02:00
|
|
|
|
|
|
|
const lazyStaticRouter = express.Router()
|
|
|
|
|
|
|
|
lazyStaticRouter.use(cors())
|
|
|
|
|
|
|
|
lazyStaticRouter.use(
|
|
|
|
LAZY_STATIC_PATHS.AVATARS + ':filename',
|
2022-07-27 14:38:07 +02:00
|
|
|
asyncMiddleware(getActorImage),
|
|
|
|
handleStaticError
|
2021-04-06 11:35:56 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
lazyStaticRouter.use(
|
|
|
|
LAZY_STATIC_PATHS.BANNERS + ':filename',
|
2022-07-27 14:38:07 +02:00
|
|
|
asyncMiddleware(getActorImage),
|
|
|
|
handleStaticError
|
2019-08-09 11:32:40 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
lazyStaticRouter.use(
|
2021-02-12 16:23:19 +01:00
|
|
|
LAZY_STATIC_PATHS.PREVIEWS + ':filename',
|
2022-07-27 14:38:07 +02:00
|
|
|
asyncMiddleware(getPreview),
|
|
|
|
handleStaticError
|
2019-08-09 11:32:40 +02:00
|
|
|
)
|
|
|
|
|
2023-06-01 14:51:16 +02:00
|
|
|
lazyStaticRouter.use(
|
|
|
|
LAZY_STATIC_PATHS.STORYBOARDS + ':filename',
|
|
|
|
asyncMiddleware(getStoryboard),
|
|
|
|
handleStaticError
|
|
|
|
)
|
|
|
|
|
2019-08-09 11:32:40 +02:00
|
|
|
lazyStaticRouter.use(
|
2021-02-15 14:08:16 +01:00
|
|
|
LAZY_STATIC_PATHS.VIDEO_CAPTIONS + ':filename',
|
2022-07-27 14:38:07 +02:00
|
|
|
asyncMiddleware(getVideoCaption),
|
|
|
|
handleStaticError
|
2019-08-09 11:32:40 +02:00
|
|
|
)
|
|
|
|
|
2021-02-16 16:25:53 +01:00
|
|
|
lazyStaticRouter.use(
|
|
|
|
LAZY_STATIC_PATHS.TORRENTS + ':filename',
|
2022-07-27 14:38:07 +02:00
|
|
|
asyncMiddleware(getTorrent),
|
|
|
|
handleStaticError
|
2021-02-16 16:25:53 +01:00
|
|
|
)
|
|
|
|
|
2019-08-09 11:32:40 +02:00
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
export {
|
|
|
|
lazyStaticRouter,
|
|
|
|
getPreview,
|
|
|
|
getVideoCaption
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
2021-06-14 16:14:45 +02:00
|
|
|
async function getActorImage (req: express.Request, res: express.Response, next: express.NextFunction) {
|
2019-08-09 11:32:40 +02:00
|
|
|
const filename = req.params.filename
|
|
|
|
|
2021-04-06 11:35:56 +02:00
|
|
|
if (actorImagePathUnsafeCache.has(filename)) {
|
|
|
|
return res.sendFile(actorImagePathUnsafeCache.get(filename), { maxAge: STATIC_MAX_AGE.SERVER })
|
2019-08-09 11:32:40 +02:00
|
|
|
}
|
|
|
|
|
2021-04-06 11:35:56 +02:00
|
|
|
const image = await ActorImageModel.loadByName(filename)
|
2021-06-01 01:36:53 +02:00
|
|
|
if (!image) return res.status(HttpStatusCode.NOT_FOUND_404).end()
|
2019-08-22 14:52:41 +02:00
|
|
|
|
2021-04-06 11:35:56 +02:00
|
|
|
if (image.onDisk === false) {
|
2021-06-01 01:36:53 +02:00
|
|
|
if (!image.fileUrl) return res.status(HttpStatusCode.NOT_FOUND_404).end()
|
2019-08-09 11:32:40 +02:00
|
|
|
|
2021-04-06 11:35:56 +02:00
|
|
|
logger.info('Lazy serve remote actor image %s.', image.fileUrl)
|
2019-08-09 11:32:40 +02:00
|
|
|
|
2019-08-09 15:04:36 +02:00
|
|
|
try {
|
2022-06-24 16:31:32 +02:00
|
|
|
await downloadActorImageFromWorker({
|
2022-02-28 08:34:43 +01:00
|
|
|
filename: image.filename,
|
|
|
|
fileUrl: image.fileUrl,
|
2022-06-20 11:48:40 +02:00
|
|
|
size: getActorImageSize(image),
|
2022-02-28 08:34:43 +01:00
|
|
|
type: image.type
|
|
|
|
})
|
2019-08-09 15:04:36 +02:00
|
|
|
} catch (err) {
|
2021-04-06 11:35:56 +02:00
|
|
|
logger.warn('Cannot process remote actor image %s.', image.fileUrl, { err })
|
2021-06-01 01:36:53 +02:00
|
|
|
return res.status(HttpStatusCode.NOT_FOUND_404).end()
|
2019-08-09 15:04:36 +02:00
|
|
|
}
|
2019-08-09 11:32:40 +02:00
|
|
|
|
2021-04-06 11:35:56 +02:00
|
|
|
image.onDisk = true
|
|
|
|
image.save()
|
|
|
|
.catch(err => logger.error('Cannot save new actor image disk state.', { err }))
|
2019-08-09 11:32:40 +02:00
|
|
|
}
|
|
|
|
|
2021-04-06 11:35:56 +02:00
|
|
|
const path = image.getPath()
|
2019-08-09 11:32:40 +02:00
|
|
|
|
2021-04-06 11:35:56 +02:00
|
|
|
actorImagePathUnsafeCache.set(filename, path)
|
2021-06-14 16:14:45 +02:00
|
|
|
|
|
|
|
return res.sendFile(path, { maxAge: STATIC_MAX_AGE.LAZY_SERVER }, (err: any) => {
|
|
|
|
if (!err) return
|
|
|
|
|
|
|
|
// It seems this actor image is not on the disk anymore
|
|
|
|
if (err.status === HttpStatusCode.NOT_FOUND_404 && !image.isOwned()) {
|
|
|
|
logger.error('Cannot lazy serve actor image %s.', filename, { err })
|
|
|
|
|
2022-06-20 11:48:40 +02:00
|
|
|
actorImagePathUnsafeCache.delete(filename)
|
2021-06-14 16:14:45 +02:00
|
|
|
|
|
|
|
image.onDisk = false
|
|
|
|
image.save()
|
|
|
|
.catch(err => logger.error('Cannot save new actor image disk state.', { err }))
|
|
|
|
}
|
|
|
|
|
|
|
|
return next(err)
|
|
|
|
})
|
2019-08-09 11:32:40 +02:00
|
|
|
}
|
|
|
|
|
2022-06-20 11:48:40 +02:00
|
|
|
function getActorImageSize (image: MActorImage): { width: number, height: number } {
|
|
|
|
if (image.width && image.height) {
|
|
|
|
return {
|
|
|
|
height: image.height,
|
|
|
|
width: image.width
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ACTOR_IMAGES_SIZE[image.type][0]
|
|
|
|
}
|
|
|
|
|
2019-08-09 11:32:40 +02:00
|
|
|
async function getPreview (req: express.Request, res: express.Response) {
|
2021-02-12 16:23:19 +01:00
|
|
|
const result = await VideosPreviewCache.Instance.getFilePath(req.params.filename)
|
2021-06-01 01:36:53 +02:00
|
|
|
if (!result) return res.status(HttpStatusCode.NOT_FOUND_404).end()
|
2019-08-09 11:32:40 +02:00
|
|
|
|
2021-02-16 16:25:53 +01:00
|
|
|
return res.sendFile(result.path, { maxAge: STATIC_MAX_AGE.LAZY_SERVER })
|
2019-08-09 11:32:40 +02:00
|
|
|
}
|
|
|
|
|
2023-06-01 14:51:16 +02:00
|
|
|
async function getStoryboard (req: express.Request, res: express.Response) {
|
|
|
|
const result = await VideosStoryboardCache.Instance.getFilePath(req.params.filename)
|
|
|
|
if (!result) return res.status(HttpStatusCode.NOT_FOUND_404).end()
|
|
|
|
|
|
|
|
return res.sendFile(result.path, { maxAge: STATIC_MAX_AGE.LAZY_SERVER })
|
|
|
|
}
|
|
|
|
|
2019-08-09 11:32:40 +02:00
|
|
|
async function getVideoCaption (req: express.Request, res: express.Response) {
|
2021-02-15 14:08:16 +01:00
|
|
|
const result = await VideosCaptionCache.Instance.getFilePath(req.params.filename)
|
2021-06-01 01:36:53 +02:00
|
|
|
if (!result) return res.status(HttpStatusCode.NOT_FOUND_404).end()
|
2019-08-09 11:32:40 +02:00
|
|
|
|
2021-02-16 16:25:53 +01:00
|
|
|
return res.sendFile(result.path, { maxAge: STATIC_MAX_AGE.LAZY_SERVER })
|
|
|
|
}
|
|
|
|
|
|
|
|
async function getTorrent (req: express.Request, res: express.Response) {
|
|
|
|
const result = await VideosTorrentCache.Instance.getFilePath(req.params.filename)
|
2021-06-01 01:36:53 +02:00
|
|
|
if (!result) return res.status(HttpStatusCode.NOT_FOUND_404).end()
|
2021-02-16 16:25:53 +01:00
|
|
|
|
2021-02-18 11:28:00 +01:00
|
|
|
// Torrents still use the old naming convention (video uuid + .torrent)
|
2019-08-09 11:32:40 +02:00
|
|
|
return res.sendFile(result.path, { maxAge: STATIC_MAX_AGE.SERVER })
|
|
|
|
}
|