2024-10-29 05:29:14 -04:00
|
|
|
import immich from './immich'
|
|
|
|
import { Response } from 'express-serve-static-core'
|
2024-11-10 15:00:41 -05:00
|
|
|
import { Asset, AssetType, ImageSize, IncomingShareRequest, SharedLink } from './types'
|
2024-11-04 14:23:34 -05:00
|
|
|
import { getConfigOption } from './functions'
|
2024-10-29 05:29:14 -04:00
|
|
|
|
|
|
|
class Render {
|
2024-11-04 14:23:34 -05:00
|
|
|
lgConfig
|
2024-11-01 10:36:53 -04:00
|
|
|
|
|
|
|
constructor () {
|
2024-11-04 14:23:34 -05:00
|
|
|
this.lgConfig = getConfigOption('lightGallery', {})
|
2024-11-01 10:36:53 -04:00
|
|
|
}
|
|
|
|
|
2024-11-10 15:00:41 -05:00
|
|
|
/**
|
|
|
|
* Stream data from Immich back to the client
|
|
|
|
*/
|
|
|
|
async assetBuffer (req: IncomingShareRequest, res: Response, asset: Asset, size?: ImageSize) {
|
|
|
|
// Prepare the request
|
2024-11-10 15:38:40 -05:00
|
|
|
const headerList = ['content-type', 'content-length', 'last-modified', 'etag']
|
2024-11-12 03:48:03 -05:00
|
|
|
size = immich.validateImageSize(size)
|
|
|
|
let subpath, sizeQueryParam
|
|
|
|
if (asset.type === AssetType.video) {
|
|
|
|
subpath = '/video/playback'
|
|
|
|
} else if (asset.type === AssetType.image) {
|
|
|
|
// For images, there are three combinations of path + query string, depending on image size
|
|
|
|
if (size === ImageSize.original && getConfigOption('ipp.downloadOriginalPhoto', false)) {
|
|
|
|
subpath = '/original'
|
|
|
|
} else if (size === ImageSize.preview || size === ImageSize.original) {
|
|
|
|
// IPP is configured in config.json to send the preview size instead of the original size
|
|
|
|
subpath = '/thumbnail'
|
|
|
|
sizeQueryParam = 'preview'
|
|
|
|
} else {
|
|
|
|
subpath = '/' + size
|
|
|
|
}
|
|
|
|
}
|
2024-11-10 15:00:41 -05:00
|
|
|
const headers = { range: '' }
|
2024-11-11 10:48:35 -05:00
|
|
|
|
2024-11-11 13:34:49 -05:00
|
|
|
// For videos, request them in 2.5MB chunks rather than the entire video
|
2024-11-10 15:38:40 -05:00
|
|
|
if (asset.type === AssetType.video) {
|
2024-11-11 10:48:35 -05:00
|
|
|
const range = (req.range || '').replace(/bytes=/, '').split('-')
|
|
|
|
const start = parseInt(range[0], 10) || 0
|
|
|
|
headers.range = `bytes=${start}-${start + 2499999}`
|
2024-11-11 06:10:53 -05:00
|
|
|
headerList.push('cache-control', 'content-range')
|
|
|
|
res.setHeader('accept-ranges', 'bytes')
|
2024-11-10 15:38:40 -05:00
|
|
|
res.status(206) // Partial Content
|
2024-11-10 15:00:41 -05:00
|
|
|
}
|
2024-11-11 10:48:35 -05:00
|
|
|
|
|
|
|
// Request data from Immich
|
2024-11-10 15:00:41 -05:00
|
|
|
const url = immich.buildUrl(immich.apiUrl() + '/assets/' + encodeURIComponent(asset.id) + subpath, {
|
|
|
|
key: asset.key,
|
2024-11-12 03:48:03 -05:00
|
|
|
size: sizeQueryParam,
|
2024-11-10 15:00:41 -05:00
|
|
|
password: asset.password
|
|
|
|
})
|
|
|
|
const data = await fetch(url, { headers })
|
|
|
|
|
|
|
|
// Return the response to the client
|
|
|
|
if (data.status >= 200 && data.status < 300) {
|
2024-11-11 13:34:49 -05:00
|
|
|
// Populate the whitelisted response headers
|
2024-11-10 15:38:40 -05:00
|
|
|
headerList.forEach(header => {
|
|
|
|
const value = data.headers.get(header)
|
|
|
|
if (value) res.setHeader(header, value)
|
|
|
|
})
|
2024-11-11 13:34:49 -05:00
|
|
|
// Return the Immich asset binary data
|
2024-11-10 15:00:41 -05:00
|
|
|
await data.body?.pipeTo(
|
|
|
|
new WritableStream({
|
2024-11-11 10:48:35 -05:00
|
|
|
write (chunk) { res.write(chunk) }
|
2024-11-10 15:00:41 -05:00
|
|
|
})
|
|
|
|
)
|
|
|
|
res.end()
|
2024-10-29 05:29:14 -04:00
|
|
|
} else {
|
|
|
|
res.status(404).send()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-30 07:45:23 -04:00
|
|
|
/**
|
|
|
|
* Render a gallery page for a given SharedLink, using EJS and lightGallery.
|
|
|
|
*
|
|
|
|
* @param res - ExpressJS Response
|
|
|
|
* @param share - Immich `shared-link` containing the assets to show in the gallery
|
|
|
|
* @param [openItem] - Immediately open a lightbox to the Nth item when the gallery loads
|
|
|
|
*/
|
2024-10-29 10:07:54 -04:00
|
|
|
async gallery (res: Response, share: SharedLink, openItem?: number) {
|
2024-10-29 05:29:14 -04:00
|
|
|
const items = []
|
2024-10-29 10:07:54 -04:00
|
|
|
for (const asset of share.assets) {
|
2024-11-12 03:48:03 -05:00
|
|
|
let video, downloadUrl
|
2024-10-29 05:29:14 -04:00
|
|
|
if (asset.type === AssetType.video) {
|
|
|
|
// Populate the data-video property
|
|
|
|
video = JSON.stringify({
|
|
|
|
source: [
|
|
|
|
{
|
2024-11-01 06:58:27 -04:00
|
|
|
src: immich.videoUrl(share.key, asset.id, asset.password),
|
2024-11-11 10:48:35 -05:00
|
|
|
type: await immich.getVideoContentType(asset)
|
2024-10-29 05:29:14 -04:00
|
|
|
}
|
|
|
|
],
|
|
|
|
attributes: {
|
|
|
|
preload: false,
|
|
|
|
controls: true
|
|
|
|
}
|
|
|
|
})
|
2024-11-12 03:48:03 -05:00
|
|
|
} else if (asset.type === AssetType.image && getConfigOption('ipp.downloadOriginalPhoto', false)) {
|
|
|
|
// Add a download link for the original-size image, if configured in config.json
|
|
|
|
downloadUrl = immich.photoUrl(share.key, asset.id, ImageSize.original, asset.password)
|
2024-10-29 05:29:14 -04:00
|
|
|
}
|
|
|
|
items.push({
|
2024-11-12 03:48:03 -05:00
|
|
|
previewUrl: immich.photoUrl(share.key, asset.id, ImageSize.preview, asset.password),
|
|
|
|
downloadUrl,
|
2024-11-01 06:58:27 -04:00
|
|
|
thumbnailUrl: immich.photoUrl(share.key, asset.id, ImageSize.thumbnail, asset.password),
|
2024-10-29 05:29:14 -04:00
|
|
|
video
|
|
|
|
})
|
|
|
|
}
|
|
|
|
res.render('gallery', {
|
|
|
|
items,
|
2024-11-01 09:19:15 -04:00
|
|
|
openItem,
|
2024-11-01 10:36:53 -04:00
|
|
|
title: this.title(share),
|
2024-11-04 14:23:34 -05:00
|
|
|
lgConfig: getConfigOption('lightGallery', {})
|
2024-10-29 05:29:14 -04:00
|
|
|
})
|
|
|
|
}
|
2024-11-01 09:19:15 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Attempt to get a title from the link description or the album title
|
|
|
|
*/
|
|
|
|
title (share: SharedLink) {
|
|
|
|
return share.description || share?.album?.albumName || ''
|
|
|
|
}
|
2024-10-29 05:29:14 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
const render = new Render()
|
|
|
|
|
|
|
|
export default render
|