| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111 |
- import {Service} from '../service/index.js';
- import {cloneDeep} from '../utils/index.js';
- import {singularize} from '../utils/index.js';
- import {InvalidArgumentError} from '../errors/index.js';
- import {RepositoryRegistry} from '../repository/index.js';
- import {ModelDefinitionUtils} from '../definition/index.js';
- /**
- * References many resolver.
- */
- export class ReferencesManyResolver extends Service {
- /**
- * Include to.
- *
- * @param {Record<string, unknown>[]} entities
- * @param {string} sourceName
- * @param {string} targetName
- * @param {string} relationName
- * @param {string|undefined} foreignKey
- * @param {Record<string, unknown>|undefined} scope
- * @return {Promise<void>}
- */
- async includeTo(
- entities,
- sourceName,
- targetName,
- relationName,
- foreignKey = undefined,
- scope = undefined,
- ) {
- if (!entities || !Array.isArray(entities))
- throw new InvalidArgumentError(
- 'The parameter "entities" of ReferencesManyResolver.includeTo requires ' +
- 'an Array of Object, but %s given.',
- entities,
- );
- if (!sourceName || typeof sourceName !== 'string')
- throw new InvalidArgumentError(
- 'The parameter "sourceName" of ReferencesManyResolver.includeTo requires ' +
- 'a non-empty String, but %s given.',
- sourceName,
- );
- if (!targetName || typeof targetName !== 'string')
- throw new InvalidArgumentError(
- 'The parameter "targetName" of ReferencesManyResolver.includeTo requires ' +
- 'a non-empty String, but %s given.',
- targetName,
- );
- if (!relationName || typeof relationName !== 'string')
- throw new InvalidArgumentError(
- 'The parameter "relationName" of ReferencesManyResolver.includeTo requires ' +
- 'a non-empty String, but %s given.',
- relationName,
- );
- if (foreignKey && typeof foreignKey !== 'string')
- throw new InvalidArgumentError(
- 'The provided parameter "foreignKey" of ReferencesManyResolver.includeTo ' +
- 'should be a String, but %s given.',
- foreignKey,
- );
- if (scope && (typeof scope !== 'object' || Array.isArray(scope)))
- throw new InvalidArgumentError(
- 'The provided parameter "scope" of ReferencesManyResolver.includeTo ' +
- 'should be an Object, but %s given.',
- scope,
- );
- if (foreignKey == null) {
- const singularRelationName = singularize(relationName);
- foreignKey = `${singularRelationName}Ids`;
- }
- const targetIds = entities.reduce((acc, entity) => {
- if (!entity || typeof entity !== 'object' || Array.isArray(entity))
- throw new InvalidArgumentError(
- 'The parameter "entities" of ReferencesManyResolver.includeTo requires ' +
- 'an Array of Object, but %s given.',
- entity,
- );
- const ids = entity[foreignKey];
- if (Array.isArray(ids))
- ids.forEach(id => {
- if (id == null || acc.includes(id)) return;
- acc.push(id);
- });
- return acc;
- }, []);
- const targetRepository =
- this.get(RepositoryRegistry).getRepository(targetName);
- const targetPkPropName =
- this.get(ModelDefinitionUtils).getPrimaryKeyAsPropertyName(targetName);
- scope = scope ? cloneDeep(scope) : {};
- const filter = cloneDeep(scope);
- filter.where = {
- and: [
- {[targetPkPropName]: {inq: targetIds}},
- ...(scope.where ? [scope.where] : []),
- ],
- };
- const targets = await targetRepository.find(filter);
- entities.forEach(entity => {
- const ids = entity[foreignKey];
- entity[relationName] = [];
- if (Array.isArray(ids))
- targets.forEach(target => {
- const targetId = target[targetPkPropName];
- if (ids.includes(targetId)) entity[relationName].push(target);
- });
- });
- }
- }
|