mirror of
https://github.com/hexastack/hexabot
synced 2025-06-26 18:27:28 +00:00
84 lines
2.7 KiB
TypeScript
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) };
|
|
}
|
|
}
|