Files
hexabot/api/src/utils/generics/base-controller.ts
2024-10-29 14:36:46 +01:00

84 lines
2.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 { NotFoundException } from '@nestjs/common';
import { TFilterQuery } from '@/utils/types/filter.types';
import { TValidateProps } from '../types/filter.types';
import { BaseSchema } from './base-schema';
import { BaseService } from './base-service';
export abstract class BaseController<
T extends BaseSchema,
TStub = never,
P extends string = never,
TFull extends Omit<T, P> = never,
> {
constructor(protected readonly service: BaseService<T, P, TFull>) {}
/**
* Checks if the given populate fields are allowed based on the allowed fields list.
* @param populate - The list of populate fields.
* @param allowedFields - The list of allowed populate fields.
* @return - True if all populate fields are allowed, otherwise false.
*/
protected canPopulate(populate: string[]): boolean {
return populate.some((p) =>
this.service
.getRepository()
.getPopulate()
.includes(p as P),
);
}
/**
* Validates the provided DTO against allowed IDs.
* @param {TValidateProps<T, TStub>} - The validation properties
* @throws {NotFoundException} Throws a NotFoundException if any invalid IDs are found.
*/
protected validate({ dto, allowedIds }: TValidateProps<T, TStub>): void {
const exceptions = [];
Object.entries(dto)
.filter(([key]) => Object.keys(allowedIds).includes(key))
.forEach(([field]) => {
const invalidIds = (
Array.isArray(dto[field]) ? dto[field] : [dto[field]]
).filter(
(id) =>
!(
Array.isArray(allowedIds[field])
? allowedIds[field]
: [allowedIds[field]]
).includes(id),
);
if (invalidIds.length) {
exceptions.push(
`${field} with ID${
invalidIds.length > 1 ? 's' : ''
} '${invalidIds}' not found`,
);
}
});
if (exceptions.length) throw new NotFoundException(exceptions.join('; '));
}
/**
* Counts filtered items.
* @return Object containing the count of items.
*/
async count(filters?: TFilterQuery<T>): Promise<{
count: number;
}> {
return { count: await this.service.count(filters) };
}
}