2023-12-14 17:55:40 +01:00
|
|
|
import { CallHandler, ExecutionContext, Injectable, NestInterceptor } from '@nestjs/common';
|
2023-07-09 06:37:40 +02:00
|
|
|
import { PATH_METADATA } from '@nestjs/common/constants';
|
|
|
|
import { Reflector } from '@nestjs/core';
|
|
|
|
import { transformException } from '@nestjs/platform-express/multer/multer/multer.utils';
|
|
|
|
import { NextFunction, RequestHandler } from 'express';
|
2023-09-04 21:45:59 +02:00
|
|
|
import multer, { StorageEngine, diskStorage } from 'multer';
|
2024-02-02 04:18:00 +01:00
|
|
|
import { createHash, randomUUID } from 'node:crypto';
|
2023-07-09 06:37:40 +02:00
|
|
|
import { Observable } from 'rxjs';
|
2024-03-20 19:32:04 +01:00
|
|
|
import { AssetService, UploadFieldName, UploadFile } from 'src/domain/asset/asset.service';
|
|
|
|
import { ImmichLogger } from 'src/infra/logger';
|
2024-03-20 21:15:01 +01:00
|
|
|
import { AuthRequest } from 'src/middleware/auth.guard';
|
2023-07-09 06:37:40 +02:00
|
|
|
|
|
|
|
export enum Route {
|
|
|
|
ASSET = 'asset',
|
|
|
|
USER = 'user',
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface ImmichFile extends Express.Multer.File {
|
|
|
|
/** sha1 hash of file */
|
2024-01-04 21:45:16 +01:00
|
|
|
uuid: string;
|
2023-07-09 06:37:40 +02:00
|
|
|
checksum: Buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function mapToUploadFile(file: ImmichFile): UploadFile {
|
|
|
|
return {
|
2024-01-04 21:45:16 +01:00
|
|
|
uuid: file.uuid,
|
2023-07-09 06:37:40 +02:00
|
|
|
checksum: file.checksum,
|
|
|
|
originalPath: file.path,
|
|
|
|
originalName: Buffer.from(file.originalname, 'latin1').toString('utf8'),
|
2024-01-13 01:43:36 +01:00
|
|
|
size: file.size,
|
2023-07-09 06:37:40 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
type DiskStorageCallback = (error: Error | null, result: string) => void;
|
|
|
|
|
2024-01-04 21:45:16 +01:00
|
|
|
type ImmichMulterFile = Express.Multer.File & { uuid: string };
|
|
|
|
|
2023-07-09 06:37:40 +02:00
|
|
|
interface Callback<T> {
|
|
|
|
(error: Error): void;
|
|
|
|
(error: null, result: T): void;
|
|
|
|
}
|
|
|
|
|
2024-03-05 23:23:06 +01:00
|
|
|
const callbackify = <T>(target: (...arguments_: any[]) => T, callback: Callback<T>) => {
|
2023-07-09 06:37:40 +02:00
|
|
|
try {
|
2024-03-05 23:23:06 +01:00
|
|
|
return callback(null, target());
|
2023-07-09 06:37:40 +02:00
|
|
|
} catch (error: Error | any) {
|
|
|
|
return callback(error);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-02-02 04:18:00 +01:00
|
|
|
const asRequest = (request: AuthRequest, file: Express.Multer.File) => {
|
2023-07-09 06:37:40 +02:00
|
|
|
return {
|
2024-02-02 04:18:00 +01:00
|
|
|
auth: request.user || null,
|
2023-07-09 06:37:40 +02:00
|
|
|
fieldName: file.fieldname as UploadFieldName,
|
|
|
|
file: mapToUploadFile(file as ImmichFile),
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class FileUploadInterceptor implements NestInterceptor {
|
2023-12-14 17:55:40 +01:00
|
|
|
private logger = new ImmichLogger(FileUploadInterceptor.name);
|
2023-07-09 06:37:40 +02:00
|
|
|
|
|
|
|
private handlers: {
|
|
|
|
userProfile: RequestHandler;
|
|
|
|
assetUpload: RequestHandler;
|
|
|
|
};
|
|
|
|
private defaultStorage: StorageEngine;
|
|
|
|
|
2023-08-28 21:41:57 +02:00
|
|
|
constructor(
|
|
|
|
private reflect: Reflector,
|
|
|
|
private assetService: AssetService,
|
|
|
|
) {
|
2023-07-09 06:37:40 +02:00
|
|
|
this.defaultStorage = diskStorage({
|
|
|
|
filename: this.filename.bind(this),
|
|
|
|
destination: this.destination.bind(this),
|
|
|
|
});
|
|
|
|
|
|
|
|
const instance = multer({
|
|
|
|
fileFilter: this.fileFilter.bind(this),
|
|
|
|
storage: {
|
|
|
|
_handleFile: this.handleFile.bind(this),
|
|
|
|
_removeFile: this.removeFile.bind(this),
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
this.handlers = {
|
|
|
|
userProfile: instance.single(UploadFieldName.PROFILE_DATA),
|
|
|
|
assetUpload: instance.fields([
|
|
|
|
{ name: UploadFieldName.ASSET_DATA, maxCount: 1 },
|
|
|
|
{ name: UploadFieldName.LIVE_PHOTO_DATA, maxCount: 1 },
|
|
|
|
{ name: UploadFieldName.SIDECAR_DATA, maxCount: 1 },
|
|
|
|
]),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
async intercept(context: ExecutionContext, next: CallHandler<any>): Promise<Observable<any>> {
|
2024-02-02 04:18:00 +01:00
|
|
|
const context_ = context.switchToHttp();
|
2023-07-09 06:37:40 +02:00
|
|
|
const route = this.reflect.get<string>(PATH_METADATA, context.getClass());
|
|
|
|
|
|
|
|
const handler: RequestHandler | null = this.getHandler(route as Route);
|
|
|
|
if (handler) {
|
|
|
|
await new Promise<void>((resolve, reject) => {
|
|
|
|
const next: NextFunction = (error) => (error ? reject(transformException(error)) : resolve());
|
2024-02-02 04:18:00 +01:00
|
|
|
handler(context_.getRequest(), context_.getResponse(), next);
|
2023-07-09 06:37:40 +02:00
|
|
|
});
|
|
|
|
} else {
|
|
|
|
this.logger.warn(`Skipping invalid file upload route: ${route}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
return next.handle();
|
|
|
|
}
|
|
|
|
|
2024-02-02 04:18:00 +01:00
|
|
|
private fileFilter(request: AuthRequest, file: Express.Multer.File, callback: multer.FileFilterCallback) {
|
|
|
|
return callbackify(() => this.assetService.canUploadFile(asRequest(request, file)), callback);
|
2023-07-09 06:37:40 +02:00
|
|
|
}
|
|
|
|
|
2024-02-02 04:18:00 +01:00
|
|
|
private filename(request: AuthRequest, file: Express.Multer.File, callback: DiskStorageCallback) {
|
|
|
|
return callbackify(
|
|
|
|
() => this.assetService.getUploadFilename(asRequest(request, file)),
|
|
|
|
callback as Callback<string>,
|
|
|
|
);
|
2023-07-09 06:37:40 +02:00
|
|
|
}
|
|
|
|
|
2024-02-02 04:18:00 +01:00
|
|
|
private destination(request: AuthRequest, file: Express.Multer.File, callback: DiskStorageCallback) {
|
|
|
|
return callbackify(() => this.assetService.getUploadFolder(asRequest(request, file)), callback as Callback<string>);
|
2023-07-09 06:37:40 +02:00
|
|
|
}
|
|
|
|
|
2024-02-02 04:18:00 +01:00
|
|
|
private handleFile(request: AuthRequest, file: Express.Multer.File, callback: Callback<Partial<ImmichFile>>) {
|
2024-01-04 21:45:16 +01:00
|
|
|
(file as ImmichMulterFile).uuid = randomUUID();
|
2023-07-09 06:37:40 +02:00
|
|
|
if (!this.isAssetUploadFile(file)) {
|
2024-02-02 04:18:00 +01:00
|
|
|
this.defaultStorage._handleFile(request, file, callback);
|
2023-07-09 06:37:40 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const hash = createHash('sha1');
|
|
|
|
file.stream.on('data', (chunk) => hash.update(chunk));
|
2024-02-02 04:18:00 +01:00
|
|
|
this.defaultStorage._handleFile(request, file, (error, info) => {
|
2023-07-09 06:37:40 +02:00
|
|
|
if (error) {
|
|
|
|
hash.destroy();
|
|
|
|
callback(error);
|
|
|
|
} else {
|
|
|
|
callback(null, { ...info, checksum: hash.digest() });
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-02-02 04:18:00 +01:00
|
|
|
private removeFile(request: AuthRequest, file: Express.Multer.File, callback: (error: Error | null) => void) {
|
|
|
|
this.defaultStorage._removeFile(request, file, callback);
|
2023-07-09 06:37:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private isAssetUploadFile(file: Express.Multer.File) {
|
|
|
|
switch (file.fieldname as UploadFieldName) {
|
|
|
|
case UploadFieldName.ASSET_DATA:
|
2024-02-02 04:18:00 +01:00
|
|
|
case UploadFieldName.LIVE_PHOTO_DATA: {
|
2023-07-09 06:37:40 +02:00
|
|
|
return true;
|
2024-02-02 04:18:00 +01:00
|
|
|
}
|
2023-07-09 06:37:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
private getHandler(route: Route) {
|
|
|
|
switch (route) {
|
2024-02-02 04:18:00 +01:00
|
|
|
case Route.ASSET: {
|
2023-07-09 06:37:40 +02:00
|
|
|
return this.handlers.assetUpload;
|
2024-02-02 04:18:00 +01:00
|
|
|
}
|
2023-07-09 06:37:40 +02:00
|
|
|
|
2024-02-02 04:18:00 +01:00
|
|
|
case Route.USER: {
|
2023-07-09 06:37:40 +02:00
|
|
|
return this.handlers.userProfile;
|
2024-02-02 04:18:00 +01:00
|
|
|
}
|
2023-07-09 06:37:40 +02:00
|
|
|
|
2024-02-02 04:18:00 +01:00
|
|
|
default: {
|
2023-07-09 06:37:40 +02:00
|
|
|
return null;
|
2024-02-02 04:18:00 +01:00
|
|
|
}
|
2023-07-09 06:37:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|