hexabot/api/src/nlp/controllers/nlp-entity.controller.ts

244 lines
7.7 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).
*/
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<NlpEntity> {
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<NlpEntity>({ allowedFields: ['name', 'doc'] }))
filters?: TFilterQuery<NlpEntity>,
) {
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<NlpEntity>,
@Query(PopulatePipe) populate: string[],
@Query(new SearchFilterPipe<NlpEntity>({ allowedFields: ['name', 'doc'] }))
filters: TFilterQuery<NlpEntity>,
) {
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<NlpEntity> {
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<DeleteResult> {
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;
}
}