1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-01 08:31:59 +00:00

refactor(server): immich file responses (#5641)

* refactor(server): immich file response

* chore: open api

* chore: tests

* chore: fix logger import

---------

Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
This commit is contained in:
Jason Rasmussen 2023-12-12 09:58:25 -05:00 committed by GitHub
parent af7c4ae090
commit cbca69841a
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
25 changed files with 178 additions and 146 deletions

View file

@ -17843,7 +17843,7 @@ export const UserApiFp = function(configuration?: Configuration) {
* @param {*} [options] Override http request option. * @param {*} [options] Override http request option.
* @throws {RequiredError} * @throws {RequiredError}
*/ */
async getProfileImage(id: string, options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<object>> { async getProfileImage(id: string, options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<File>> {
const localVarAxiosArgs = await localVarAxiosParamCreator.getProfileImage(id, options); const localVarAxiosArgs = await localVarAxiosParamCreator.getProfileImage(id, options);
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration); return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
}, },
@ -17945,7 +17945,7 @@ export const UserApiFactory = function (configuration?: Configuration, basePath?
* @param {*} [options] Override http request option. * @param {*} [options] Override http request option.
* @throws {RequiredError} * @throws {RequiredError}
*/ */
getProfileImage(requestParameters: UserApiGetProfileImageRequest, options?: AxiosRequestConfig): AxiosPromise<object> { getProfileImage(requestParameters: UserApiGetProfileImageRequest, options?: AxiosRequestConfig): AxiosPromise<File> {
return localVarFp.getProfileImage(requestParameters.id, options).then((request) => request(axios, basePath)); return localVarFp.getProfileImage(requestParameters.id, options).then((request) => request(axios, basePath));
}, },
/** /**

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

View file

@ -1465,6 +1465,15 @@
"get": { "get": {
"operationId": "serveFile", "operationId": "serveFile",
"parameters": [ "parameters": [
{
"name": "id",
"required": true,
"in": "path",
"schema": {
"format": "uuid",
"type": "string"
}
},
{ {
"name": "isThumb", "name": "isThumb",
"required": false, "required": false,
@ -1483,15 +1492,6 @@
"type": "boolean" "type": "boolean"
} }
}, },
{
"name": "id",
"required": true,
"in": "path",
"schema": {
"format": "uuid",
"type": "string"
}
},
{ {
"name": "key", "name": "key",
"required": false, "required": false,
@ -1926,13 +1926,7 @@
"responses": { "responses": {
"200": { "200": {
"content": { "content": {
"image/jpeg": { "application/octet-stream": {
"schema": {
"format": "binary",
"type": "string"
}
},
"image/webp": {
"schema": { "schema": {
"format": "binary", "format": "binary",
"type": "string" "type": "string"
@ -4499,7 +4493,7 @@
"responses": { "responses": {
"200": { "200": {
"content": { "content": {
"image/jpeg": { "application/octet-stream": {
"schema": { "schema": {
"format": "binary", "format": "binary",
"type": "string" "type": "string"
@ -6080,9 +6074,10 @@
"responses": { "responses": {
"200": { "200": {
"content": { "content": {
"application/json": { "application/octet-stream": {
"schema": { "schema": {
"type": "object" "format": "binary",
"type": "string"
} }
} }
}, },

View file

@ -16,6 +16,7 @@ import {
} from '@test'; } from '@test';
import { when } from 'jest-when'; import { when } from 'jest-when';
import { Readable } from 'stream'; import { Readable } from 'stream';
import { ImmichFileResponse } from '../domain.util';
import { JobName } from '../job'; import { JobName } from '../job';
import { import {
AssetStats, AssetStats,
@ -474,15 +475,16 @@ describe(AssetService.name, () => {
}); });
it('should download a file', async () => { it('should download a file', async () => {
const stream = new Readable();
accessMock.asset.checkOwnerAccess.mockResolvedValue(new Set(['asset-1'])); accessMock.asset.checkOwnerAccess.mockResolvedValue(new Set(['asset-1']));
assetMock.getByIds.mockResolvedValue([assetStub.image]); assetMock.getByIds.mockResolvedValue([assetStub.image]);
storageMock.createReadStream.mockResolvedValue({ stream });
await expect(sut.downloadFile(authStub.admin, 'asset-1')).resolves.toEqual({ stream }); await expect(sut.downloadFile(authStub.admin, 'asset-1')).resolves.toEqual(
new ImmichFileResponse({
expect(storageMock.createReadStream).toHaveBeenCalledWith(assetStub.image.originalPath, 'image/jpeg'); path: '/original/path.jpg',
contentType: 'image/jpeg',
cacheControl: false,
}),
);
}); });
it('should download an archive', async () => { it('should download an archive', async () => {

View file

@ -7,7 +7,7 @@ import sanitize from 'sanitize-filename';
import { AccessCore, Permission } from '../access'; import { AccessCore, Permission } from '../access';
import { AuthDto } from '../auth'; import { AuthDto } from '../auth';
import { mimeTypes } from '../domain.constant'; import { mimeTypes } from '../domain.constant';
import { HumanReadableSize, usePagination } from '../domain.util'; import { HumanReadableSize, ImmichFileResponse, usePagination } from '../domain.util';
import { IAssetDeletionJob, ISidecarWriteJob, JOBS_ASSET_PAGINATION_SIZE, JobName } from '../job'; import { IAssetDeletionJob, ISidecarWriteJob, JOBS_ASSET_PAGINATION_SIZE, JobName } from '../job';
import { import {
CommunicationEvent, CommunicationEvent,
@ -274,7 +274,7 @@ export class AssetService {
return { ...options, userIds }; return { ...options, userIds };
} }
async downloadFile(auth: AuthDto, id: string): Promise<ImmichReadStream> { async downloadFile(auth: AuthDto, id: string): Promise<ImmichFileResponse> {
await this.access.requirePermission(auth, Permission.ASSET_DOWNLOAD, id); await this.access.requirePermission(auth, Permission.ASSET_DOWNLOAD, id);
const [asset] = await this.assetRepository.getByIds([id]); const [asset] = await this.assetRepository.getByIds([id]);
@ -286,7 +286,11 @@ export class AssetService {
throw new BadRequestException('Asset is offline'); throw new BadRequestException('Asset is offline');
} }
return this.storageRepository.createReadStream(asset.originalPath, mimeTypes.lookup(asset.originalPath)); return new ImmichFileResponse({
path: asset.originalPath,
contentType: mimeTypes.lookup(asset.originalPath),
cacheControl: false,
});
} }
async getDownloadInfo(auth: AuthDto, dto: DownloadInfoDto): Promise<DownloadResponseDto> { async getDownloadInfo(auth: AuthDto, dto: DownloadInfoDto): Promise<DownloadResponseDto> {

View file

@ -16,6 +16,16 @@ import { CronJob } from 'cron';
import { basename, extname } from 'node:path'; import { basename, extname } from 'node:path';
import sanitize from 'sanitize-filename'; import sanitize from 'sanitize-filename';
export class ImmichFileResponse {
public readonly path!: string;
public readonly contentType!: string;
public readonly cacheControl!: boolean;
constructor(response: ImmichFileResponse) {
Object.assign(this, response);
}
}
export interface OpenGraphTags { export interface OpenGraphTags {
title: string; title: string;
description: string; description: string;

View file

@ -18,6 +18,7 @@ import {
personStub, personStub,
} from '@test'; } from '@test';
import { BulkIdErrorReason } from '../asset'; import { BulkIdErrorReason } from '../asset';
import { ImmichFileResponse } from '../domain.util';
import { JobName } from '../job'; import { JobName } from '../job';
import { import {
IAssetRepository, IAssetRepository,
@ -203,8 +204,13 @@ describe(PersonService.name, () => {
it('should serve the thumbnail', async () => { it('should serve the thumbnail', async () => {
personMock.getById.mockResolvedValue(personStub.noName); personMock.getById.mockResolvedValue(personStub.noName);
accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1'])); accessMock.person.checkOwnerAccess.mockResolvedValue(new Set(['person-1']));
await sut.getThumbnail(authStub.admin, 'person-1'); await expect(sut.getThumbnail(authStub.admin, 'person-1')).resolves.toEqual(
expect(storageMock.createReadStream).toHaveBeenCalledWith('/path/to/thumbnail.jpg', 'image/jpeg'); new ImmichFileResponse({
path: '/path/to/thumbnail.jpg',
contentType: 'image/jpeg',
cacheControl: true,
}),
);
expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.user.id, new Set(['person-1'])); expect(accessMock.person.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.user.id, new Set(['person-1']));
}); });
}); });

View file

@ -5,7 +5,7 @@ import { AccessCore, Permission } from '../access';
import { AssetResponseDto, BulkIdErrorReason, BulkIdResponseDto, mapAsset } from '../asset'; import { AssetResponseDto, BulkIdErrorReason, BulkIdResponseDto, mapAsset } from '../asset';
import { AuthDto } from '../auth'; import { AuthDto } from '../auth';
import { mimeTypes } from '../domain.constant'; import { mimeTypes } from '../domain.constant';
import { usePagination } from '../domain.util'; import { ImmichFileResponse, usePagination } from '../domain.util';
import { IBaseJob, IEntityJob, JOBS_ASSET_PAGINATION_SIZE, JobName } from '../job'; import { IBaseJob, IEntityJob, JOBS_ASSET_PAGINATION_SIZE, JobName } from '../job';
import { FACE_THUMBNAIL_SIZE } from '../media'; import { FACE_THUMBNAIL_SIZE } from '../media';
import { import {
@ -20,7 +20,6 @@ import {
ISmartInfoRepository, ISmartInfoRepository,
IStorageRepository, IStorageRepository,
ISystemConfigRepository, ISystemConfigRepository,
ImmichReadStream,
UpdateFacesData, UpdateFacesData,
WithoutProperty, WithoutProperty,
} from '../repositories'; } from '../repositories';
@ -173,14 +172,18 @@ export class PersonService {
return this.repository.getStatistics(id); return this.repository.getStatistics(id);
} }
async getThumbnail(auth: AuthDto, id: string): Promise<ImmichReadStream> { async getThumbnail(auth: AuthDto, id: string): Promise<ImmichFileResponse> {
await this.access.requirePermission(auth, Permission.PERSON_READ, id); await this.access.requirePermission(auth, Permission.PERSON_READ, id);
const person = await this.repository.getById(id); const person = await this.repository.getById(id);
if (!person || !person.thumbnailPath) { if (!person || !person.thumbnailPath) {
throw new NotFoundException(); throw new NotFoundException();
} }
return this.storageRepository.createReadStream(person.thumbnailPath, mimeTypes.lookup(person.thumbnailPath)); return new ImmichFileResponse({
path: person.thumbnailPath,
contentType: mimeTypes.lookup(person.thumbnailPath),
cacheControl: true,
});
} }
async getAssets(auth: AuthDto, id: string): Promise<AssetResponseDto[]> { async getAssets(auth: AuthDto, id: string): Promise<AssetResponseDto[]> {

View file

@ -17,7 +17,7 @@ import {
userStub, userStub,
} from '@test'; } from '@test';
import { when } from 'jest-when'; import { when } from 'jest-when';
import { Readable } from 'stream'; import { ImmichFileResponse } from '../domain.util';
import { JobName } from '../job'; import { JobName } from '../job';
import { import {
IAlbumRepository, IAlbumRepository,
@ -390,15 +390,17 @@ describe(UserService.name, () => {
}); });
it('should return the profile picture', async () => { it('should return the profile picture', async () => {
const stream = new Readable();
userMock.get.mockResolvedValue(userStub.profilePath); userMock.get.mockResolvedValue(userStub.profilePath);
storageMock.createReadStream.mockResolvedValue({ stream });
await expect(sut.getProfileImage(userStub.profilePath.id)).resolves.toEqual({ stream }); await expect(sut.getProfileImage(userStub.profilePath.id)).resolves.toEqual(
new ImmichFileResponse({
path: '/path/to/profile.jpg',
contentType: 'image/jpeg',
cacheControl: false,
}),
);
expect(userMock.get).toHaveBeenCalledWith(userStub.profilePath.id, {}); expect(userMock.get).toHaveBeenCalledWith(userStub.profilePath.id, {});
expect(storageMock.createReadStream).toHaveBeenCalledWith('/path/to/profile.jpg', 'image/jpeg');
}); });
}); });

View file

@ -2,6 +2,7 @@ import { UserEntity } from '@app/infra/entities';
import { BadRequestException, ForbiddenException, Inject, Injectable, Logger, NotFoundException } from '@nestjs/common'; import { BadRequestException, ForbiddenException, Inject, Injectable, Logger, NotFoundException } from '@nestjs/common';
import { randomBytes } from 'crypto'; import { randomBytes } from 'crypto';
import { AuthDto } from '../auth'; import { AuthDto } from '../auth';
import { ImmichFileResponse } from '../domain.util';
import { IEntityJob, JobName } from '../job'; import { IEntityJob, JobName } from '../job';
import { import {
IAlbumRepository, IAlbumRepository,
@ -11,7 +12,6 @@ import {
ILibraryRepository, ILibraryRepository,
IStorageRepository, IStorageRepository,
IUserRepository, IUserRepository,
ImmichReadStream,
UserFindOptions, UserFindOptions,
} from '../repositories'; } from '../repositories';
import { StorageCore, StorageFolder } from '../storage'; import { StorageCore, StorageFolder } from '../storage';
@ -99,12 +99,17 @@ export class UserService {
await this.jobRepository.queue({ name: JobName.DELETE_FILES, data: { files: [user.profileImagePath] } }); await this.jobRepository.queue({ name: JobName.DELETE_FILES, data: { files: [user.profileImagePath] } });
} }
async getProfileImage(id: string): Promise<ImmichReadStream> { async getProfileImage(id: string): Promise<ImmichFileResponse> {
const user = await this.findOrFail(id, {}); const user = await this.findOrFail(id, {});
if (!user.profileImagePath) { if (!user.profileImagePath) {
throw new NotFoundException('User does not have a profile image'); throw new NotFoundException('User does not have a profile image');
} }
return this.storageRepository.createReadStream(user.profileImagePath, 'image/jpeg');
return new ImmichFileResponse({
path: user.profileImagePath,
contentType: 'image/jpeg',
cacheControl: false,
});
} }
async resetAdminPassword(ask: (admin: UserResponseDto) => Promise<string | undefined>) { async resetAdminPassword(ask: (admin: UserResponseDto) => Promise<string | undefined>) {

View file

@ -14,9 +14,9 @@ import {
UseInterceptors, UseInterceptors,
ValidationPipe, ValidationPipe,
} from '@nestjs/common'; } from '@nestjs/common';
import { ApiBody, ApiConsumes, ApiHeader, ApiOkResponse, ApiOperation, ApiTags } from '@nestjs/swagger'; import { ApiBody, ApiConsumes, ApiHeader, ApiOperation, ApiTags } from '@nestjs/swagger';
import { Response as Res } from 'express'; import { Response as Res } from 'express';
import { Auth, Authenticated, SharedLinkRoute } from '../../app.guard'; import { Auth, Authenticated, FileResponse, SharedLinkRoute } from '../../app.guard';
import { UUIDParamDto } from '../../controllers/dto/uuid-param.dto'; import { UUIDParamDto } from '../../controllers/dto/uuid-param.dto';
import { FileUploadInterceptor, ImmichFile, Route, mapToUploadFile } from '../../interceptors'; import { FileUploadInterceptor, ImmichFile, Route, mapToUploadFile } from '../../interceptors';
import FileNotEmptyValidator from '../validation/file-not-empty-validator'; import FileNotEmptyValidator from '../validation/file-not-empty-validator';
@ -83,35 +83,24 @@ export class AssetController {
@SharedLinkRoute() @SharedLinkRoute()
@Get('/file/:id') @Get('/file/:id')
@ApiOkResponse({ @FileResponse()
content: { serveFile(
'application/octet-stream': { schema: { type: 'string', format: 'binary' } },
},
})
async serveFile(
@Auth() auth: AuthDto, @Auth() auth: AuthDto,
@Response() res: Res,
@Query(new ValidationPipe({ transform: true })) query: ServeFileDto,
@Param() { id }: UUIDParamDto, @Param() { id }: UUIDParamDto,
@Query(new ValidationPipe({ transform: true })) dto: ServeFileDto,
) { ) {
await this.assetService.serveFile(auth, id, query, res); return this.assetService.serveFile(auth, id, dto);
} }
@SharedLinkRoute() @SharedLinkRoute()
@Get('/thumbnail/:id') @Get('/thumbnail/:id')
@ApiOkResponse({ @FileResponse()
content: { getAssetThumbnail(
'image/jpeg': { schema: { type: 'string', format: 'binary' } },
'image/webp': { schema: { type: 'string', format: 'binary' } },
},
})
async getAssetThumbnail(
@Auth() auth: AuthDto, @Auth() auth: AuthDto,
@Response() res: Res,
@Param() { id }: UUIDParamDto, @Param() { id }: UUIDParamDto,
@Query(new ValidationPipe({ transform: true })) query: GetAssetThumbnailDto, @Query(new ValidationPipe({ transform: true })) dto: GetAssetThumbnailDto,
) { ) {
await this.assetService.serveThumbnail(auth, id, query, res); return this.assetService.serveThumbnail(auth, id, dto);
} }
@Get('/curated-objects') @Get('/curated-objects')

View file

@ -6,7 +6,7 @@ import {
IAccessRepository, IAccessRepository,
IJobRepository, IJobRepository,
ILibraryRepository, ILibraryRepository,
isConnectionAborted, ImmichFileResponse,
JobName, JobName,
mapAsset, mapAsset,
mimeTypes, mimeTypes,
@ -16,12 +16,7 @@ import {
} from '@app/domain'; } from '@app/domain';
import { ASSET_CHECKSUM_CONSTRAINT, AssetEntity, AssetType, LibraryType } from '@app/infra/entities'; import { ASSET_CHECKSUM_CONSTRAINT, AssetEntity, AssetType, LibraryType } from '@app/infra/entities';
import { Inject, Injectable, InternalServerErrorException, Logger, NotFoundException } from '@nestjs/common'; import { Inject, Injectable, InternalServerErrorException, Logger, NotFoundException } from '@nestjs/common';
import { Response as Res, Response } from 'express';
import { constants } from 'fs';
import fs from 'fs/promises';
import path from 'path';
import { QueryFailedError } from 'typeorm'; import { QueryFailedError } from 'typeorm';
import { promisify } from 'util';
import { IAssetRepository } from './asset-repository'; import { IAssetRepository } from './asset-repository';
import { AssetCore } from './asset.core'; import { AssetCore } from './asset.core';
import { AssetBulkUploadCheckDto } from './dto/asset-check.dto'; import { AssetBulkUploadCheckDto } from './dto/asset-check.dto';
@ -41,13 +36,6 @@ import { CheckExistingAssetsResponseDto } from './response-dto/check-existing-as
import { CuratedLocationsResponseDto } from './response-dto/curated-locations-response.dto'; import { CuratedLocationsResponseDto } from './response-dto/curated-locations-response.dto';
import { CuratedObjectsResponseDto } from './response-dto/curated-objects-response.dto'; import { CuratedObjectsResponseDto } from './response-dto/curated-objects-response.dto';
type SendFile = Parameters<Response['sendFile']>;
type SendFileOptions = SendFile[1];
// TODO: move file sending logic to an interceptor
const sendFile = (res: Response, path: string, options: SendFileOptions) =>
promisify<string, SendFileOptions>(res.sendFile).bind(res)(path, options);
@Injectable() @Injectable()
export class AssetService { export class AssetService {
readonly logger = new Logger(AssetService.name); readonly logger = new Logger(AssetService.name);
@ -148,7 +136,7 @@ export class AssetService {
} }
} }
async serveThumbnail(auth: AuthDto, assetId: string, query: GetAssetThumbnailDto, res: Res) { async serveThumbnail(auth: AuthDto, assetId: string, dto: GetAssetThumbnailDto): Promise<ImmichFileResponse> {
await this.access.requirePermission(auth, Permission.ASSET_VIEW, assetId); await this.access.requirePermission(auth, Permission.ASSET_VIEW, assetId);
const asset = await this._assetRepository.get(assetId); const asset = await this._assetRepository.get(assetId);
@ -156,19 +144,12 @@ export class AssetService {
throw new NotFoundException('Asset not found'); throw new NotFoundException('Asset not found');
} }
try { const filepath = this.getThumbnailPath(asset, dto.format);
await this.sendFile(res, this.getThumbnailPath(asset, query.format));
} catch (e) { return new ImmichFileResponse({ path: filepath, contentType: mimeTypes.lookup(filepath), cacheControl: true });
res.header('Cache-Control', 'none');
this.logger.error(`Cannot create read stream for asset ${asset.id}`, 'getAssetThumbnail');
throw new InternalServerErrorException(
`Cannot read thumbnail file for asset ${asset.id} - contact your administrator`,
{ cause: e as Error },
);
}
} }
public async serveFile(auth: AuthDto, assetId: string, query: ServeFileDto, res: Res) { public async serveFile(auth: AuthDto, assetId: string, dto: ServeFileDto): Promise<ImmichFileResponse> {
// this is not quite right as sometimes this returns the original still // this is not quite right as sometimes this returns the original still
await this.access.requirePermission(auth, Permission.ASSET_VIEW, assetId); await this.access.requirePermission(auth, Permission.ASSET_VIEW, assetId);
@ -181,10 +162,10 @@ export class AssetService {
const filepath = const filepath =
asset.type === AssetType.IMAGE asset.type === AssetType.IMAGE
? this.getServePath(asset, query, allowOriginalFile) ? this.getServePath(asset, dto, allowOriginalFile)
: asset.encodedVideoPath || asset.originalPath; : asset.encodedVideoPath || asset.originalPath;
await this.sendFile(res, filepath); return new ImmichFileResponse({ path: filepath, contentType: mimeTypes.lookup(filepath), cacheControl: true });
} }
async getAssetSearchTerm(auth: AuthDto): Promise<string[]> { async getAssetSearchTerm(auth: AuthDto): Promise<string[]> {
@ -292,13 +273,13 @@ export class AssetService {
} }
} }
private getServePath(asset: AssetEntity, query: ServeFileDto, allowOriginalFile: boolean): string { private getServePath(asset: AssetEntity, dto: ServeFileDto, allowOriginalFile: boolean): string {
const mimeType = mimeTypes.lookup(asset.originalPath); const mimeType = mimeTypes.lookup(asset.originalPath);
/** /**
* Serve file viewer on the web * Serve file viewer on the web
*/ */
if (query.isWeb && mimeType != 'image/gif') { if (dto.isWeb && mimeType != 'image/gif') {
if (!asset.resizePath) { if (!asset.resizePath) {
this.logger.error('Error serving IMAGE asset for web'); this.logger.error('Error serving IMAGE asset for web');
throw new InternalServerErrorException(`Failed to serve image asset for web`, 'ServeFile'); throw new InternalServerErrorException(`Failed to serve image asset for web`, 'ServeFile');
@ -310,7 +291,7 @@ export class AssetService {
/** /**
* Serve thumbnail image for both web and mobile app * Serve thumbnail image for both web and mobile app
*/ */
if ((!query.isThumb && allowOriginalFile) || (query.isWeb && mimeType === 'image/gif')) { if ((!dto.isThumb && allowOriginalFile) || (dto.isWeb && mimeType === 'image/gif')) {
return asset.originalPath; return asset.originalPath;
} }
@ -325,27 +306,6 @@ export class AssetService {
return asset.resizePath; return asset.resizePath;
} }
private async sendFile(res: Res, filepath: string): Promise<void> {
await fs.access(filepath, constants.R_OK);
const options: SendFileOptions = { dotfiles: 'allow' };
if (!path.isAbsolute(filepath)) {
options.root = process.cwd();
}
res.set('Cache-Control', 'private, max-age=86400, no-transform');
res.header('Content-Type', mimeTypes.lookup(filepath));
try {
await sendFile(res, filepath, options);
} catch (error: Error | any) {
if (!isConnectionAborted(error)) {
this.logger.error(`Unable to send file: ${error.name}`, error.stack);
}
// throwing closes the connection and prevents `Error: write EPIPE`
throw error;
}
}
private async getLibraryId(auth: AuthDto, libraryId?: string) { private async getLibraryId(auth: AuthDto, libraryId?: string) {
if (libraryId) { if (libraryId) {
return libraryId; return libraryId;

View file

@ -9,7 +9,7 @@ import {
createParamDecorator, createParamDecorator,
} from '@nestjs/common'; } from '@nestjs/common';
import { Reflector } from '@nestjs/core'; import { Reflector } from '@nestjs/core';
import { ApiBearerAuth, ApiCookieAuth, ApiQuery, ApiSecurity } from '@nestjs/swagger'; import { ApiBearerAuth, ApiCookieAuth, ApiOkResponse, ApiQuery, ApiSecurity } from '@nestjs/swagger';
import { Request } from 'express'; import { Request } from 'express';
import { UAParser } from 'ua-parser-js'; import { UAParser } from 'ua-parser-js';
@ -54,6 +54,11 @@ export const Auth = createParamDecorator((data, ctx: ExecutionContext): AuthDto
return ctx.switchToHttp().getRequest<{ user: AuthDto }>().user; return ctx.switchToHttp().getRequest<{ user: AuthDto }>().user;
}); });
export const FileResponse = () =>
ApiOkResponse({
content: { 'application/octet-stream': { schema: { type: 'string', format: 'binary' } } },
});
export const GetLoginDetails = createParamDecorator((data, ctx: ExecutionContext): LoginDetails => { export const GetLoginDetails = createParamDecorator((data, ctx: ExecutionContext): LoginDetails => {
const req = ctx.switchToHttp().getRequest<Request>(); const req = ctx.switchToHttp().getRequest<Request>();
const userAgent = UAParser(req.headers['user-agent']); const userAgent = UAParser(req.headers['user-agent']);

View file

@ -32,7 +32,7 @@ import {
TagController, TagController,
UserController, UserController,
} from './controllers'; } from './controllers';
import { ErrorInterceptor, FileUploadInterceptor } from './interceptors'; import { ErrorInterceptor, FileServeInterceptor, FileUploadInterceptor } from './interceptors';
@Module({ @Module({
imports: [ imports: [
@ -66,6 +66,7 @@ import { ErrorInterceptor, FileUploadInterceptor } from './interceptors';
], ],
providers: [ providers: [
{ provide: APP_INTERCEPTOR, useClass: ErrorInterceptor }, { provide: APP_INTERCEPTOR, useClass: ErrorInterceptor },
{ provide: APP_INTERCEPTOR, useClass: FileServeInterceptor },
{ provide: APP_GUARD, useClass: AppGuard }, { provide: APP_GUARD, useClass: AppGuard },
{ provide: IAssetRepository, useClass: AssetRepository }, { provide: IAssetRepository, useClass: AssetRepository },
AppService, AppService,

View file

@ -12,6 +12,7 @@ import {
BulkIdsDto, BulkIdsDto,
DownloadInfoDto, DownloadInfoDto,
DownloadResponseDto, DownloadResponseDto,
ImmichFileResponse,
MapMarkerDto, MapMarkerDto,
MapMarkerResponseDto, MapMarkerResponseDto,
MemoryLaneDto, MemoryLaneDto,
@ -37,9 +38,9 @@ import {
Query, Query,
StreamableFile, StreamableFile,
} from '@nestjs/common'; } from '@nestjs/common';
import { ApiOkResponse, ApiTags } from '@nestjs/swagger'; import { ApiTags } from '@nestjs/swagger';
import { DeviceIdDto } from '../api-v1/asset/dto/device-id.dto'; import { DeviceIdDto } from '../api-v1/asset/dto/device-id.dto';
import { Auth, Authenticated, SharedLinkRoute } from '../app.guard'; import { Auth, Authenticated, FileResponse, SharedLinkRoute } from '../app.guard';
import { UseValidation, asStreamableFile } from '../app.utils'; import { UseValidation, asStreamableFile } from '../app.utils';
import { Route } from '../interceptors'; import { Route } from '../interceptors';
import { UUIDParamDto } from './dto/uuid-param.dto'; import { UUIDParamDto } from './dto/uuid-param.dto';
@ -88,7 +89,7 @@ export class AssetController {
@SharedLinkRoute() @SharedLinkRoute()
@Post('download/archive') @Post('download/archive')
@HttpCode(HttpStatus.OK) @HttpCode(HttpStatus.OK)
@ApiOkResponse({ content: { 'application/octet-stream': { schema: { type: 'string', format: 'binary' } } } }) @FileResponse()
downloadArchive(@Auth() auth: AuthDto, @Body() dto: AssetIdsDto): Promise<StreamableFile> { downloadArchive(@Auth() auth: AuthDto, @Body() dto: AssetIdsDto): Promise<StreamableFile> {
return this.service.downloadArchive(auth, dto).then(asStreamableFile); return this.service.downloadArchive(auth, dto).then(asStreamableFile);
} }
@ -96,9 +97,9 @@ export class AssetController {
@SharedLinkRoute() @SharedLinkRoute()
@Post('download/:id') @Post('download/:id')
@HttpCode(HttpStatus.OK) @HttpCode(HttpStatus.OK)
@ApiOkResponse({ content: { 'application/octet-stream': { schema: { type: 'string', format: 'binary' } } } }) @FileResponse()
downloadFile(@Auth() auth: AuthDto, @Param() { id }: UUIDParamDto) { downloadFile(@Auth() auth: AuthDto, @Param() { id }: UUIDParamDto): Promise<ImmichFileResponse> {
return this.service.downloadFile(auth, id).then(asStreamableFile); return this.service.downloadFile(auth, id);
} }
/** /**

View file

@ -3,7 +3,6 @@ import {
AssetResponseDto, AssetResponseDto,
AuthDto, AuthDto,
BulkIdResponseDto, BulkIdResponseDto,
ImmichReadStream,
MergePersonDto, MergePersonDto,
PeopleResponseDto, PeopleResponseDto,
PeopleUpdateDto, PeopleUpdateDto,
@ -13,16 +12,12 @@ import {
PersonStatisticsResponseDto, PersonStatisticsResponseDto,
PersonUpdateDto, PersonUpdateDto,
} from '@app/domain'; } from '@app/domain';
import { Body, Controller, Get, Param, Post, Put, Query, StreamableFile } from '@nestjs/common'; import { Body, Controller, Get, Param, Post, Put, Query } from '@nestjs/common';
import { ApiOkResponse, ApiTags } from '@nestjs/swagger'; import { ApiTags } from '@nestjs/swagger';
import { Auth, Authenticated } from '../app.guard'; import { Auth, Authenticated, FileResponse } from '../app.guard';
import { UseValidation } from '../app.utils'; import { UseValidation } from '../app.utils';
import { UUIDParamDto } from './dto/uuid-param.dto'; import { UUIDParamDto } from './dto/uuid-param.dto';
function asStreamableFile({ stream, type, length }: ImmichReadStream) {
return new StreamableFile(stream, { type, length });
}
@ApiTags('Person') @ApiTags('Person')
@Controller('person') @Controller('person')
@Authenticated() @Authenticated()
@ -74,13 +69,9 @@ export class PersonController {
} }
@Get(':id/thumbnail') @Get(':id/thumbnail')
@ApiOkResponse({ @FileResponse()
content: {
'image/jpeg': { schema: { type: 'string', format: 'binary' } },
},
})
getPersonThumbnail(@Auth() auth: AuthDto, @Param() { id }: UUIDParamDto) { getPersonThumbnail(@Auth() auth: AuthDto, @Param() { id }: UUIDParamDto) {
return this.service.getThumbnail(auth, id).then(asStreamableFile); return this.service.getThumbnail(auth, id);
} }
@Get(':id/assets') @Get(':id/assets')

View file

@ -3,6 +3,7 @@ import {
CreateUserDto as CreateDto, CreateUserDto as CreateDto,
CreateProfileImageDto, CreateProfileImageDto,
CreateProfileImageResponseDto, CreateProfileImageResponseDto,
ImmichFileResponse,
UpdateUserDto as UpdateDto, UpdateUserDto as UpdateDto,
UserResponseDto, UserResponseDto,
UserService, UserService,
@ -23,8 +24,8 @@ import {
UseInterceptors, UseInterceptors,
} from '@nestjs/common'; } from '@nestjs/common';
import { ApiBody, ApiConsumes, ApiTags } from '@nestjs/swagger'; import { ApiBody, ApiConsumes, ApiTags } from '@nestjs/swagger';
import { AdminRoute, Auth, Authenticated } from '../app.guard'; import { AdminRoute, Auth, Authenticated, FileResponse } from '../app.guard';
import { UseValidation, asStreamableFile } from '../app.utils'; import { UseValidation } from '../app.utils';
import { FileUploadInterceptor, Route } from '../interceptors'; import { FileUploadInterceptor, Route } from '../interceptors';
import { UUIDParamDto } from './dto/uuid-param.dto'; import { UUIDParamDto } from './dto/uuid-param.dto';
@ -93,7 +94,8 @@ export class UserController {
@Get('profile-image/:id') @Get('profile-image/:id')
@Header('Cache-Control', 'private, no-cache, no-transform') @Header('Cache-Control', 'private, no-cache, no-transform')
getProfileImage(@Param() { id }: UUIDParamDto): Promise<any> { @FileResponse()
return this.service.getProfileImage(id).then(asStreamableFile); getProfileImage(@Param() { id }: UUIDParamDto): Promise<ImmichFileResponse> {
return this.service.getProfileImage(id);
} }
} }

View file

@ -0,0 +1,55 @@
import { ImmichFileResponse, isConnectionAborted } from '@app/domain';
import { CallHandler, ExecutionContext, Logger, NestInterceptor } from '@nestjs/common';
import { Response } from 'express';
import { access, constants } from 'fs/promises';
import { isAbsolute } from 'path';
import { Observable, mergeMap } from 'rxjs';
import { promisify } from 'util';
type SendFile = Parameters<Response['sendFile']>;
type SendFileOptions = SendFile[1];
export class FileServeInterceptor implements NestInterceptor {
private logger = new Logger(FileServeInterceptor.name);
intercept(context: ExecutionContext, next: CallHandler<any>): Observable<any> {
const http = context.switchToHttp();
const res = http.getResponse<Response>();
const sendFile = (path: string, options: SendFileOptions) =>
promisify<string, SendFileOptions>(res.sendFile).bind(res)(path, options);
return next.handle().pipe(
mergeMap(async (file) => {
if (file instanceof ImmichFileResponse === false) {
return file;
}
try {
if (file.cacheControl) {
res.set('Cache-Control', 'private, max-age=86400, no-transform');
}
res.header('Content-Type', file.contentType);
const options: SendFileOptions = { dotfiles: 'allow' };
if (!isAbsolute(file.path)) {
options.root = process.cwd();
}
await access(file.path, constants.R_OK);
return sendFile(file.path, options);
} catch (error: Error | any) {
res.header('Cache-Control', 'none');
if (!isConnectionAborted(error)) {
this.logger.error(`Unable to send file: ${error.name}`, error.stack);
}
// throwing closes the connection and prevents `Error: write EPIPE`
throw error;
}
}),
);
}
}

View file

@ -1,2 +1,3 @@
export * from './error.interceptor'; export * from './error.interceptor';
export * from './file.interceptor'; export * from './file-serve.interceptor';
export * from './file-upload.interceptor';

View file

@ -17843,7 +17843,7 @@ export const UserApiFp = function(configuration?: Configuration) {
* @param {*} [options] Override http request option. * @param {*} [options] Override http request option.
* @throws {RequiredError} * @throws {RequiredError}
*/ */
async getProfileImage(id: string, options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<object>> { async getProfileImage(id: string, options?: AxiosRequestConfig): Promise<(axios?: AxiosInstance, basePath?: string) => AxiosPromise<File>> {
const localVarAxiosArgs = await localVarAxiosParamCreator.getProfileImage(id, options); const localVarAxiosArgs = await localVarAxiosParamCreator.getProfileImage(id, options);
return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration); return createRequestFunction(localVarAxiosArgs, globalAxios, BASE_PATH, configuration);
}, },
@ -17945,7 +17945,7 @@ export const UserApiFactory = function (configuration?: Configuration, basePath?
* @param {*} [options] Override http request option. * @param {*} [options] Override http request option.
* @throws {RequiredError} * @throws {RequiredError}
*/ */
getProfileImage(requestParameters: UserApiGetProfileImageRequest, options?: AxiosRequestConfig): AxiosPromise<object> { getProfileImage(requestParameters: UserApiGetProfileImageRequest, options?: AxiosRequestConfig): AxiosPromise<File> {
return localVarFp.getProfileImage(requestParameters.id, options).then((request) => request(axios, basePath)); return localVarFp.getProfileImage(requestParameters.id, options).then((request) => request(axios, basePath));
}, },
/** /**