2024-05-24 02:26:22 +02:00
|
|
|
import {
|
|
|
|
Body,
|
|
|
|
Controller,
|
2024-05-31 19:44:04 +02:00
|
|
|
Get,
|
2024-05-25 03:02:22 +02:00
|
|
|
HttpCode,
|
2024-05-24 02:26:22 +02:00
|
|
|
HttpStatus,
|
|
|
|
Inject,
|
2024-05-31 19:44:04 +02:00
|
|
|
Next,
|
2024-05-24 02:26:22 +02:00
|
|
|
Param,
|
|
|
|
ParseFilePipe,
|
2024-05-25 03:02:22 +02:00
|
|
|
Post,
|
2024-05-24 02:26:22 +02:00
|
|
|
Put,
|
2024-05-31 19:44:04 +02:00
|
|
|
Query,
|
2024-05-24 02:26:22 +02:00
|
|
|
Res,
|
|
|
|
UploadedFiles,
|
|
|
|
UseInterceptors,
|
|
|
|
} from '@nestjs/common';
|
2024-05-31 19:44:04 +02:00
|
|
|
import { ApiBody, ApiConsumes, ApiHeader, ApiTags } from '@nestjs/swagger';
|
|
|
|
import { NextFunction, Response } from 'express';
|
2024-05-24 02:26:22 +02:00
|
|
|
import { EndpointLifecycle } from 'src/decorators';
|
2024-05-25 03:02:22 +02:00
|
|
|
import {
|
|
|
|
AssetBulkUploadCheckResponseDto,
|
|
|
|
AssetMediaResponseDto,
|
2024-05-31 19:44:04 +02:00
|
|
|
AssetMediaStatus,
|
2024-05-25 03:02:22 +02:00
|
|
|
CheckExistingAssetsResponseDto,
|
|
|
|
} from 'src/dtos/asset-media-response.dto';
|
|
|
|
import {
|
|
|
|
AssetBulkUploadCheckDto,
|
2024-05-31 19:44:04 +02:00
|
|
|
AssetMediaCreateDto,
|
|
|
|
AssetMediaOptionsDto,
|
2024-05-25 03:02:22 +02:00
|
|
|
AssetMediaReplaceDto,
|
|
|
|
CheckExistingAssetsDto,
|
|
|
|
UploadFieldName,
|
|
|
|
} from 'src/dtos/asset-media.dto';
|
2024-10-17 19:17:32 +02:00
|
|
|
import { AuthDto } from 'src/dtos/auth.dto';
|
|
|
|
import { ImmichHeader, RouteKey } from 'src/enum';
|
2024-05-24 02:26:22 +02:00
|
|
|
import { ILoggerRepository } from 'src/interfaces/logger.interface';
|
2024-05-31 19:44:04 +02:00
|
|
|
import { AssetUploadInterceptor } from 'src/middleware/asset-upload.interceptor';
|
|
|
|
import { Auth, Authenticated, FileResponse } from 'src/middleware/auth.guard';
|
2024-09-27 16:28:42 +02:00
|
|
|
import { FileUploadInterceptor, UploadFiles, getFiles } from 'src/middleware/file-upload.interceptor';
|
2024-05-24 02:26:22 +02:00
|
|
|
import { AssetMediaService } from 'src/services/asset-media.service';
|
2024-05-31 19:44:04 +02:00
|
|
|
import { sendFile } from 'src/utils/file';
|
2024-05-24 02:26:22 +02:00
|
|
|
import { FileNotEmptyValidator, UUIDParamDto } from 'src/validation';
|
|
|
|
|
2024-05-30 00:26:57 +02:00
|
|
|
@ApiTags('Assets')
|
2024-09-27 16:28:42 +02:00
|
|
|
@Controller(RouteKey.ASSET)
|
2024-05-24 02:26:22 +02:00
|
|
|
export class AssetMediaController {
|
|
|
|
constructor(
|
|
|
|
@Inject(ILoggerRepository) private logger: ILoggerRepository,
|
|
|
|
private service: AssetMediaService,
|
|
|
|
) {}
|
|
|
|
|
2024-05-31 19:44:04 +02:00
|
|
|
@Post()
|
|
|
|
@UseInterceptors(AssetUploadInterceptor, FileUploadInterceptor)
|
|
|
|
@ApiConsumes('multipart/form-data')
|
|
|
|
@ApiHeader({
|
|
|
|
name: ImmichHeader.CHECKSUM,
|
|
|
|
description: 'sha1 checksum that can be used for duplicate detection before the file is uploaded',
|
|
|
|
required: false,
|
|
|
|
})
|
|
|
|
@ApiBody({ description: 'Asset Upload Information', type: AssetMediaCreateDto })
|
|
|
|
@Authenticated({ sharedLink: true })
|
|
|
|
async uploadAsset(
|
|
|
|
@Auth() auth: AuthDto,
|
|
|
|
@UploadedFiles(new ParseFilePipe({ validators: [new FileNotEmptyValidator(['assetData'])] })) files: UploadFiles,
|
|
|
|
@Body() dto: AssetMediaCreateDto,
|
|
|
|
@Res({ passthrough: true }) res: Response,
|
|
|
|
): Promise<AssetMediaResponseDto> {
|
|
|
|
const { file, sidecarFile } = getFiles(files);
|
|
|
|
const responseDto = await this.service.uploadAsset(auth, dto, file, sidecarFile);
|
|
|
|
|
|
|
|
if (responseDto.status === AssetMediaStatus.DUPLICATE) {
|
|
|
|
res.status(HttpStatus.OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
return responseDto;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Get(':id/original')
|
|
|
|
@FileResponse()
|
|
|
|
@Authenticated({ sharedLink: true })
|
|
|
|
async downloadAsset(
|
|
|
|
@Auth() auth: AuthDto,
|
|
|
|
@Param() { id }: UUIDParamDto,
|
|
|
|
@Res() res: Response,
|
|
|
|
@Next() next: NextFunction,
|
|
|
|
) {
|
|
|
|
await sendFile(res, next, () => this.service.downloadOriginal(auth, id), this.logger);
|
|
|
|
}
|
|
|
|
|
2024-05-24 02:26:22 +02:00
|
|
|
/**
|
|
|
|
* Replace the asset with new file, without changing its id
|
|
|
|
*/
|
2024-05-31 19:44:04 +02:00
|
|
|
@Put(':id/original')
|
2024-05-24 02:26:22 +02:00
|
|
|
@UseInterceptors(FileUploadInterceptor)
|
|
|
|
@ApiConsumes('multipart/form-data')
|
|
|
|
@EndpointLifecycle({ addedAt: 'v1.106.0' })
|
2024-08-20 14:50:14 +02:00
|
|
|
@Authenticated({ sharedLink: true })
|
2024-05-24 02:26:22 +02:00
|
|
|
async replaceAsset(
|
|
|
|
@Auth() auth: AuthDto,
|
|
|
|
@Param() { id }: UUIDParamDto,
|
|
|
|
@UploadedFiles(new ParseFilePipe({ validators: [new FileNotEmptyValidator([UploadFieldName.ASSET_DATA])] }))
|
|
|
|
files: UploadFiles,
|
|
|
|
@Body() dto: AssetMediaReplaceDto,
|
|
|
|
@Res({ passthrough: true }) res: Response,
|
|
|
|
): Promise<AssetMediaResponseDto> {
|
|
|
|
const { file } = getFiles(files);
|
|
|
|
const responseDto = await this.service.replaceAsset(auth, id, dto, file);
|
2024-05-31 19:44:04 +02:00
|
|
|
if (responseDto.status === AssetMediaStatus.DUPLICATE) {
|
2024-05-24 02:26:22 +02:00
|
|
|
res.status(HttpStatus.OK);
|
|
|
|
}
|
|
|
|
return responseDto;
|
|
|
|
}
|
2024-05-25 03:02:22 +02:00
|
|
|
|
2024-05-31 19:44:04 +02:00
|
|
|
@Get(':id/thumbnail')
|
|
|
|
@FileResponse()
|
|
|
|
@Authenticated({ sharedLink: true })
|
|
|
|
async viewAsset(
|
|
|
|
@Auth() auth: AuthDto,
|
|
|
|
@Param() { id }: UUIDParamDto,
|
|
|
|
@Query() dto: AssetMediaOptionsDto,
|
|
|
|
@Res() res: Response,
|
|
|
|
@Next() next: NextFunction,
|
|
|
|
) {
|
|
|
|
await sendFile(res, next, () => this.service.viewThumbnail(auth, id, dto), this.logger);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Get(':id/video/playback')
|
|
|
|
@FileResponse()
|
|
|
|
@Authenticated({ sharedLink: true })
|
|
|
|
async playAssetVideo(
|
|
|
|
@Auth() auth: AuthDto,
|
|
|
|
@Param() { id }: UUIDParamDto,
|
|
|
|
@Res() res: Response,
|
|
|
|
@Next() next: NextFunction,
|
|
|
|
) {
|
|
|
|
await sendFile(res, next, () => this.service.playbackVideo(auth, id), this.logger);
|
|
|
|
}
|
|
|
|
|
2024-05-25 03:02:22 +02:00
|
|
|
/**
|
|
|
|
* Checks if multiple assets exist on the server and returns all existing - used by background backup
|
|
|
|
*/
|
|
|
|
@Post('exist')
|
|
|
|
@HttpCode(HttpStatus.OK)
|
|
|
|
@Authenticated()
|
|
|
|
checkExistingAssets(
|
|
|
|
@Auth() auth: AuthDto,
|
|
|
|
@Body() dto: CheckExistingAssetsDto,
|
|
|
|
): Promise<CheckExistingAssetsResponseDto> {
|
|
|
|
return this.service.checkExistingAssets(auth, dto);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks if assets exist by checksums
|
|
|
|
*/
|
|
|
|
@Post('bulk-upload-check')
|
|
|
|
@HttpCode(HttpStatus.OK)
|
|
|
|
@Authenticated()
|
|
|
|
checkBulkUpload(
|
|
|
|
@Auth() auth: AuthDto,
|
|
|
|
@Body() dto: AssetBulkUploadCheckDto,
|
|
|
|
): Promise<AssetBulkUploadCheckResponseDto> {
|
|
|
|
return this.service.bulkUploadCheck(auth, dto);
|
|
|
|
}
|
2024-05-24 02:26:22 +02:00
|
|
|
}
|