UNPKG

@loopback/repository

Version:

Define and implement a common set of interfaces for interacting with databases

101 lines (96 loc) 3.73 kB
// 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; }