@loopback/repository
Version:
Define and implement a common set of interfaces for interacting with databases
101 lines (96 loc) • 3.73 kB
text/typescript
// Copyright IBM Corp. and LoopBack contributors 2018,2020. All Rights Reserved.
// Node module: @loopback/repository
// This file is licensed under the MIT License.
// License text available at https://opensource.org/licenses/MIT
import debugFactory from 'debug';
import {DataObject} from '../../common-types';
import {Entity} from '../../model';
import {EntityCrudRepository} from '../../repositories';
import {Getter, HasOneDefinition, InclusionResolver} from '../relation.types';
import {resolveHasOneMetadata} from './has-one.helpers';
import {createHasOneInclusionResolver} from './has-one.inclusion-resolver';
import {DefaultHasOneRepository, HasOneRepository} from './has-one.repository';
const debug = debugFactory(
'loopback:repository:relations:has-one:repository-factory',
);
export interface HasOneRepositoryFactory<
Target extends Entity,
ForeignKeyType,
> {
/**
* Invoke the function to obtain HasOneRepository.
*/
(fkValue: ForeignKeyType): HasOneRepository<Target>;
/**
* Use `resolver` property to obtain an InclusionResolver for this relation.
*/
inclusionResolver: InclusionResolver<Entity, Target>;
}
/**
* Enforces a constraint on a repository based on a relationship contract
* between models. For example, if a Customer model is related to an Address model
* via a HasOne relation, then, the relational repository returned by the
* factory function would be constrained by a Customer model instance's id(s).
*
* If the target model is polymorphic, i.e. stored within different repositories,
* supply the targetRepositoryGetter with a dictionary in the form of {[typeName: string]: repositoryGetter}
*
* @param relationMetadata - The relation metadata used to describe the
* relationship and determine how to apply the constraint.
* @param targetRepositoryGetter - The repository or a dictionary of classname - repository,
* which represents the target model of a relation attached to a datasource.
* For the dictionary, the key is the class name of the concrete class the the polymorphic model.
* @returns The factory function which accepts a foreign key value to constrain
* the given target repository
*/
export function createHasOneRepositoryFactory<
Target extends Entity,
TargetID,
ForeignKeyType,
>(
relationMetadata: HasOneDefinition,
targetRepositoryGetter:
| Getter<EntityCrudRepository<Target, TargetID>>
| {
[repoType: string]: Getter<EntityCrudRepository<Target, TargetID>>;
},
): HasOneRepositoryFactory<Target, ForeignKeyType> {
const meta = resolveHasOneMetadata(relationMetadata);
// resolve the repositoryGetter into a dictionary
if (typeof targetRepositoryGetter === 'function') {
targetRepositoryGetter = {
[meta.target().name]: targetRepositoryGetter as Getter<
EntityCrudRepository<Target, TargetID>
>,
};
}
debug('Resolved HasOne relation metadata: %o', meta);
const result: HasOneRepositoryFactory<Target, ForeignKeyType> = function (
fkValue: ForeignKeyType,
) {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const constraint: any = {[meta.keyTo]: fkValue};
return new DefaultHasOneRepository<
Target,
TargetID,
EntityCrudRepository<Target, TargetID>
>(
targetRepositoryGetter as {
[repoType: string]: Getter<EntityCrudRepository<Target, TargetID>>;
},
constraint as DataObject<Target>,
relationMetadata.target,
);
};
result.inclusionResolver = createHasOneInclusionResolver<
Target,
TargetID,
object
>(
meta,
targetRepositoryGetter as {
[repoType: string]: Getter<EntityCrudRepository<Target, TargetID>>;
},
);
return result;
}