mirror of
https://github.com/hexastack/hexabot
synced 2024-11-24 13:05:15 +00:00
178 lines
6.0 KiB
TypeScript
178 lines
6.0 KiB
TypeScript
/*
|
|
* Copyright © 2024 Hexastack. All rights reserved.
|
|
*
|
|
* Licensed under the GNU Affero General Public License v3.0 (AGPLv3) with the following additional terms:
|
|
* 1. The name "Hexabot" is a trademark of Hexastack. You may not use this name in derivative works without express written permission.
|
|
* 2. All derivative works must include clear attribution to the original creator and software, Hexastack and Hexabot, in a prominent location (e.g., in the software's "About" section, documentation, and README file).
|
|
* 3. SaaS Restriction: This software, or any derivative of it, may not be used to offer a competing product or service (SaaS) without prior written consent from Hexastack. Offering the software as a service or using it in a commercial cloud environment without express permission is strictly prohibited.
|
|
*/
|
|
|
|
import { extname } from 'path';
|
|
|
|
import {
|
|
BadRequestException,
|
|
Controller,
|
|
Delete,
|
|
Get,
|
|
HttpCode,
|
|
NotFoundException,
|
|
Param,
|
|
Post,
|
|
Query,
|
|
StreamableFile,
|
|
UploadedFiles,
|
|
UseInterceptors,
|
|
} from '@nestjs/common';
|
|
import { FileFieldsInterceptor } from '@nestjs/platform-express';
|
|
import { CsrfCheck } from '@tekuconcept/nestjs-csrf';
|
|
import { TFilterQuery } from 'mongoose';
|
|
import { diskStorage, memoryStorage } from 'multer';
|
|
import { v4 as uuidv4 } from 'uuid';
|
|
|
|
import { config } from '@/config';
|
|
import { CsrfInterceptor } from '@/interceptors/csrf.interceptor';
|
|
import { LoggerService } from '@/logger/logger.service';
|
|
import { Roles } from '@/utils/decorators/roles.decorator';
|
|
import { BaseController } from '@/utils/generics/base-controller';
|
|
import { DeleteResult } from '@/utils/generics/base-repository';
|
|
import { PageQueryDto } from '@/utils/pagination/pagination-query.dto';
|
|
import { PageQueryPipe } from '@/utils/pagination/pagination-query.pipe';
|
|
import { SearchFilterPipe } from '@/utils/pipes/search-filter.pipe';
|
|
|
|
import { AttachmentDownloadDto } from '../dto/attachment.dto';
|
|
import { Attachment } from '../schemas/attachment.schema';
|
|
import { AttachmentService } from '../services/attachment.service';
|
|
|
|
@UseInterceptors(CsrfInterceptor)
|
|
@Controller('attachment')
|
|
export class AttachmentController extends BaseController<Attachment> {
|
|
constructor(
|
|
private readonly attachmentService: AttachmentService,
|
|
private readonly logger: LoggerService,
|
|
) {
|
|
super(attachmentService);
|
|
}
|
|
|
|
/**
|
|
* Counts the filtered number of attachments.
|
|
*
|
|
* @returns A promise that resolves to an object representing the filtered number of attachments.
|
|
*/
|
|
@Get('count')
|
|
async filterCount(
|
|
@Query(
|
|
new SearchFilterPipe<Attachment>({
|
|
allowedFields: ['name', 'type'],
|
|
}),
|
|
)
|
|
filters?: TFilterQuery<Attachment>,
|
|
) {
|
|
return await this.count(filters);
|
|
}
|
|
|
|
@Get(':id')
|
|
async findOne(@Param('id') id: string): Promise<Attachment> {
|
|
const doc = await this.attachmentService.findOne(id);
|
|
if (!doc) {
|
|
this.logger.warn(`Unable to find Attachement by id ${id}`);
|
|
throw new NotFoundException(`Attachement with ID ${id} not found`);
|
|
}
|
|
return doc;
|
|
}
|
|
|
|
/**
|
|
* Retrieves all attachments based on specified filters.
|
|
*
|
|
* @param pageQuery - The pagination to apply when retrieving attachments.
|
|
* @param filters - The filters to apply when retrieving attachments.
|
|
* @returns A promise that resolves to an array of attachments matching the filters.
|
|
*/
|
|
@Get()
|
|
async findPage(
|
|
@Query(PageQueryPipe) pageQuery: PageQueryDto<Attachment>,
|
|
@Query(
|
|
new SearchFilterPipe<Attachment>({ allowedFields: ['name', 'type'] }),
|
|
)
|
|
filters: TFilterQuery<Attachment>,
|
|
) {
|
|
return await this.attachmentService.findPage(filters, pageQuery);
|
|
}
|
|
|
|
/**
|
|
* Uploads files to the server.
|
|
*
|
|
* @param files - An array of files to upload.
|
|
* @returns A promise that resolves to an array of uploaded attachments.
|
|
*/
|
|
@CsrfCheck(true)
|
|
@Post('upload')
|
|
@UseInterceptors(
|
|
FileFieldsInterceptor([{ name: 'file' }], {
|
|
limits: {
|
|
fileSize: config.parameters.maxUploadSize,
|
|
},
|
|
storage: (() => {
|
|
if (config.parameters.storageMode === 'memory') {
|
|
return memoryStorage();
|
|
} else {
|
|
return diskStorage({
|
|
destination: config.parameters.uploadDir,
|
|
filename: (req, file, cb) => {
|
|
const name = file.originalname.split('.')[0];
|
|
const extension = extname(file.originalname);
|
|
cb(null, `${name}-${uuidv4()}${extension}`);
|
|
},
|
|
});
|
|
}
|
|
})(),
|
|
}),
|
|
)
|
|
async uploadFile(
|
|
@UploadedFiles() files: { file: Express.Multer.File[] },
|
|
): Promise<Attachment[]> {
|
|
if (!files || !Array.isArray(files?.file) || files.file.length === 0) {
|
|
throw new BadRequestException('No file was selected');
|
|
}
|
|
|
|
return await this.attachmentService.uploadFiles(files);
|
|
}
|
|
|
|
/**
|
|
* Downloads an attachment identified by the provided parameters.
|
|
*
|
|
* @param params - The parameters identifying the attachment to download.
|
|
* @returns A promise that resolves to a StreamableFile representing the downloaded attachment.
|
|
*/
|
|
@Roles('public')
|
|
@Get('download/:id/:filename?')
|
|
async download(
|
|
@Param() params: AttachmentDownloadDto,
|
|
): Promise<StreamableFile> {
|
|
const attachment = await this.attachmentService.findOne(params.id);
|
|
|
|
if (!attachment) {
|
|
throw new NotFoundException('Attachment not found');
|
|
}
|
|
|
|
return this.attachmentService.download(attachment);
|
|
}
|
|
|
|
/**
|
|
* Deletes an attachment with the specified ID.
|
|
*
|
|
* @param id - The ID of the attachment to delete.
|
|
* @returns A promise that resolves to the result of the deletion operation.
|
|
*/
|
|
@CsrfCheck(true)
|
|
@Delete(':id')
|
|
@HttpCode(204)
|
|
async deleteOne(@Param('id') id: string): Promise<DeleteResult> {
|
|
const result = await this.attachmentService.deleteOne(id);
|
|
if (result.deletedCount === 0) {
|
|
this.logger.warn(`Unable to delete attachment by id ${id}`);
|
|
throw new NotFoundException(`Attachment with ID ${id} not found`);
|
|
}
|
|
return result;
|
|
}
|
|
}
|