/* * 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). */ import { BadRequestException, Body, Controller, Delete, Get, HttpCode, InternalServerErrorException, MethodNotAllowedException, NotFoundException, Param, Patch, Post, Query, UseInterceptors, } from '@nestjs/common'; import { CsrfCheck } from '@tekuconcept/nestjs-csrf'; import { CsrfInterceptor } from '@/interceptors/csrf.interceptor'; import { LoggerService } from '@/logger/logger.service'; 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 { PopulatePipe } from '@/utils/pipes/populate.pipe'; import { SearchFilterPipe } from '@/utils/pipes/search-filter.pipe'; import { TFilterQuery } from '@/utils/types/filter.types'; import { NlpEntityCreateDto } from '../dto/nlp-entity.dto'; import { NlpEntity, NlpEntityFull, NlpEntityPopulate, NlpEntityStub, } from '../schemas/nlp-entity.schema'; import { NlpEntityService } from '../services/nlp-entity.service'; @UseInterceptors(CsrfInterceptor) @Controller('nlpentity') export class NlpEntityController extends BaseController< NlpEntity, NlpEntityStub, NlpEntityPopulate, NlpEntityFull > { constructor( private readonly nlpEntityService: NlpEntityService, private readonly logger: LoggerService, ) { super(nlpEntityService); } /** * Creates a new NLP entity. * * This endpoint receives the NLP entity data in the request body, validates it, and creates a new entry in the database. * * @param createNlpEntityDto - Data transfer object containing the details of the NLP entity to create. * * @returns The newly created NLP entity. */ @CsrfCheck(true) @Post() async create( @Body() createNlpEntityDto: NlpEntityCreateDto, ): Promise { return await this.nlpEntityService.create(createNlpEntityDto); } /** * Counts the number of NLP entities based on provided filters. * * This endpoint allows users to apply filters to count the number of entities in the system that match specific criteria. * * @param filters - Optional filters to apply when counting entities. * * @returns The count of NLP entities matching the filters. */ @Get('count') async filterCount( @Query(new SearchFilterPipe({ allowedFields: ['name', 'doc'] })) filters?: TFilterQuery, ) { return await this.count(filters); } /** * Finds a single NLP entity by ID. * * This endpoint returns an NLP entity if found by its ID, optionally allowing population of specified fields. * * @param id - The ID of the NLP entity to find. * @param populate - Fields to populate, such as 'values'. * * @returns The NLP entity found by the ID. */ @Get(':id') async findOne( @Param('id') id: string, @Query(PopulatePipe) populate: string[], ) { const doc = this.canPopulate(populate) ? await this.nlpEntityService.findOneAndPopulate(id) : await this.nlpEntityService.findOne(id); if (!doc) { this.logger.warn(`Unable to find NLP Entity by id ${id}`); throw new NotFoundException(`NLP Entity with ID ${id} not found`); } return doc; } /** * Retrieves a paginated list of NLP entities with optional filters. * * This endpoint supports pagination and allows users to retrieve a filtered list of NLP entities. * * @param pageQuery - The pagination details such as page number and size. * @param populate - Fields to populate in the retrieved entities. * @param filters - Filters to apply when retrieving entities. * * @returns A paginated list of NLP entities. */ @Get() async findPage( @Query(PageQueryPipe) pageQuery: PageQueryDto, @Query(PopulatePipe) populate: string[], @Query(new SearchFilterPipe({ allowedFields: ['name', 'doc'] })) filters: TFilterQuery, ) { return this.canPopulate(populate) ? await this.nlpEntityService.findAndPopulate(filters, pageQuery) : await this.nlpEntityService.find(filters, pageQuery); } /** * Updates an NLP entity by ID. * * This endpoint allows updating an existing NLP entity. The entity must not be a built-in entity. * * @param id - The ID of the NLP entity to update. * @param updateNlpEntityDto - The new data for the NLP entity. * * @returns The updated NLP entity. */ @CsrfCheck(true) @Patch(':id') async updateOne( @Param('id') id: string, @Body() updateNlpEntityDto: NlpEntityCreateDto, ): Promise { const nlpEntity = await this.nlpEntityService.findOne(id); if (!nlpEntity) { this.logger.warn(`Unable to update NLP Entity by id ${id}`); throw new NotFoundException(`NLP Entity with ID ${id} not found`); } if (nlpEntity.builtin) { throw new MethodNotAllowedException( `Cannot update builtin NLP Entity ${nlpEntity.name}`, ); } const result = await this.nlpEntityService.updateOne( id, updateNlpEntityDto, ); if (!result) { this.logger.warn(`Failed to update NLP Entity by id ${id}`); throw new InternalServerErrorException( `Failed to update NLP Entity with ID ${id}`, ); } return result; } /** * Deletes an NLP entity by ID. * * This endpoint deletes an NLP entity from the system, provided it is not a built-in entity. * * @param id - The ID of the NLP entity to delete. * * @returns The result of the deletion operation. */ @CsrfCheck(true) @Delete(':id') @HttpCode(204) async deleteOne(@Param('id') id: string) { const nlpEntity = await this.nlpEntityService.findOne(id); if (!nlpEntity) { this.logger.warn(`Unable to delete NLP Entity by id ${id}`); throw new NotFoundException(`NLP Entity with ID ${id} not found`); } if (nlpEntity.builtin) { throw new MethodNotAllowedException( `Cannot delete builtin NLP Entity ${nlpEntity.name}`, ); } const result = await this.nlpEntityService.deleteCascadeOne(id); if (result.deletedCount === 0) { this.logger.warn(`Failed to delete NLP Entity by id ${id}`); throw new InternalServerErrorException( `Failed to delete NLP Entity with ID ${id}`, ); } return result; } /** * Deletes multiple NLP entities by their IDs. * @param ids - IDs of NLP entities to be deleted. * @returns A Promise that resolves to the deletion result. */ @CsrfCheck(true) @Delete('') @HttpCode(204) async deleteMany(@Body('ids') ids: string[]): Promise { if (!ids || ids.length === 0) { throw new BadRequestException('No IDs provided for deletion.'); } const deleteResult = await this.nlpEntityService.deleteMany({ _id: { $in: ids }, }); if (deleteResult.deletedCount === 0) { this.logger.warn( `Unable to delete NLP entities with provided IDs: ${ids}`, ); throw new NotFoundException('NLP entities with provided IDs not found'); } this.logger.log(`Successfully deleted NLP entities with IDs: ${ids}`); return deleteResult; } }