From 724307572be57fc3bf4c56f5e3fdaaadb29d2088 Mon Sep 17 00:00:00 2001 From: Agnes Lin Date: Fri, 23 Aug 2019 14:23:19 -0400 Subject: [PATCH] feat(repository): implement inclusionResolver for hasMany Co-authored-by: Nora --- docs/site/HasMany-relation.md | 92 ++++++ .../repository-tests/src/crud-test-suite.ts | 1 + ...-inclusion-resolver.relation.acceptance.ts | 273 ++++++++++++++++++ .../repositories/customer.repository.ts | 2 +- .../src/crud/relations/helpers.ts | 7 +- .../src/types.repository-tests.ts | 8 + ...ts-of-one-to-many-relation-helpers.unit.ts | 37 +++ .../has-many/has-many-repository.factory.ts | 29 +- .../relations/has-many/has-many.helpers.ts | 7 +- .../has-many/has-many.inclusion-resolver.ts | 77 +++++ .../src/relations/has-many/index.ts | 1 + .../src/relations/relation.helpers.ts | 33 +++ 12 files changed, 559 insertions(+), 8 deletions(-) create mode 100644 packages/repository-tests/src/crud/relations/acceptance/has-many-inclusion-resolver.relation.acceptance.ts create mode 100644 packages/repository/src/__tests__/unit/repositories/relations-helpers/flatten-targets-of-one-to-many-relation-helpers.unit.ts create mode 100644 packages/repository/src/relations/has-many/has-many.inclusion-resolver.ts diff --git a/docs/site/HasMany-relation.md b/docs/site/HasMany-relation.md index dcf1353eb6c5..3da5c5b91c30 100644 --- a/docs/site/HasMany-relation.md +++ b/docs/site/HasMany-relation.md @@ -314,3 +314,95 @@ certain properties from the JSON/OpenAPI spec schema built for the `requestBody` payload. See its [GitHub issue](https://github.com/strongloop/loopback-next/issues/1179) to follow the discussion. " %} + +## Querying related models + +LoopBack 4 has the concept of an `inclusion resolver` in relations, which helps +to query data through an `include` filter. An inclusion resolver is a function +that can fetch target models for the given list of source model instances. +LoopBack 4 creates a different inclusion resolver for each relation type. + +Use the relation between `Customer` and `Order` we show above, a `Customer` has +many `Order`s. + +After setting up the relation in the repository class, the inclusion resolver +allows users to retrieve all customers along with their related orders through +the following code: + +```ts +customerRepo.find({include: [{relation: 'orders'}]}); +``` + +### Enable/disable the inclusion resolvers: + +- Base repository classes have a public property `inclusionResolvers`, which + maintains a map containing inclusion resolvers for each relation. +- The `inclusionResolver` of a certain relation is built when the source + repository class calls the `createHasManyRepositoryFactoryFor` function in the + constructor with the relation name. +- Call `registerInclusionResolver` to add the resolver of that relation to the + `inclusionResolvers` map. (As we realized in LB3, not all relations are + allowed to be traversed. Users can decide to which resolvers can be added.) + +The following code snippet shows how to register the inclusion resolver for the +has-many relation 'orders': + +```ts +export class CustomerRepository extends DefaultCrudRepository { + products: HasManyRepositoryFactory; + + constructor( + dataSource: juggler.DataSource, + orderRepositoryGetter: Getter, + ) { + super(Customer, dataSource); + + // we already have this line to create a HasManyRepository factory + this.orders = this.createHasManyRepositoryFactoryFor( + 'orders', + orderRepositoryGetter, + ); + + // add this line to register inclusion resolver + this.registerInclusion('orders', this.orders.inclusionResolver); + } +} +``` + +- We can simply include the relation in queries via `find()`, `findOne()`, and + `findById()` methods. Example: + + ```ts + customerRepository.find({include: [{relation: 'orders'}]}); + ``` + + which returns: + + ```ts + [ + { + id: 1, + name: 'Thor', + orders: [ + {name: 'Mjolnir', customerId: 1}, + {name: 'Rocket Raccoon', customerId: 1}, + ], + }, + { + id: 2, + name: 'Captain', + orders: [{name: 'Shield', customerId: 2}], + }, + ]; + ``` + +- You can delete a relation from `inclusionResolvers` to disable the inclusion + for a certain relation. e.g + `customerRepository.inclusionResolvers.delete('orders')` + +{% include note.html content=" +Inclusion with custom scope: +Besides specifying the relation name to include, it's also possible to specify additional scope constraints. +However, this feature is not supported yet. Check our GitHub issue for more information: +[Include related models with a custom scope](https://github.com/strongloop/loopback-next/issues/3453). +" %} diff --git a/packages/repository-tests/src/crud-test-suite.ts b/packages/repository-tests/src/crud-test-suite.ts index 6ea0f927d579..057edd762fc5 100644 --- a/packages/repository-tests/src/crud-test-suite.ts +++ b/packages/repository-tests/src/crud-test-suite.ts @@ -33,6 +33,7 @@ export function crudRepositoryTestSuite( freeFormProperties: true, emptyValue: undefined, supportsTransactions: true, + supportsInclusionResolvers: true, ...partialFeatures, }; diff --git a/packages/repository-tests/src/crud/relations/acceptance/has-many-inclusion-resolver.relation.acceptance.ts b/packages/repository-tests/src/crud/relations/acceptance/has-many-inclusion-resolver.relation.acceptance.ts new file mode 100644 index 000000000000..6556e7bc8090 --- /dev/null +++ b/packages/repository-tests/src/crud/relations/acceptance/has-many-inclusion-resolver.relation.acceptance.ts @@ -0,0 +1,273 @@ +// Copyright IBM Corp. 2019. All Rights Reserved. +// Node module: @loopback/repository-tests +// This file is licensed under the MIT License. +// License text available at https://opensource.org/licenses/MIT + +import {expect, skipIf, toJSON} from '@loopback/testlab'; +import {Suite} from 'mocha'; +import { + CrudFeatures, + CrudRepositoryCtor, + CrudTestContext, + DataSourceOptions, +} from '../../..'; +import { + deleteAllModelsInDefaultDataSource, + MixedIdType, + withCrudCtx, +} from '../../../helpers.repository-tests'; +import { + Customer, + CustomerRepository, + Order, + OrderRepository, +} from '../fixtures/models'; +import {givenBoundCrudRepositories} from '../helpers'; + +export function hasManyRelationAcceptance( + dataSourceOptions: DataSourceOptions, + repositoryClass: CrudRepositoryCtor, + features: CrudFeatures, +) { + skipIf<[(this: Suite) => void], void>( + !features.supportsInclusionResolvers, + describe, + 'retrieve models including relations', + () => { + describe('HasMany inclusion resolvers - acceptance', () => { + before(deleteAllModelsInDefaultDataSource); + let customerRepo: CustomerRepository; + let orderRepo: OrderRepository; + let existingCustomerId: MixedIdType; + + before( + withCrudCtx(async function setupRepository(ctx: CrudTestContext) { + // when running the test suite on MongoDB, we don't really need to setup + // this config for mongo connector to pass the test. + // however real-world applications might have such config for MongoDB + // setting it up to check if it works fine as well + Order.definition.properties.customerId.type = features.idType; + Order.definition.properties.customerId.mongodb = { + dataType: 'ObjectID', + }; + // this helper should create the inclusion resolvers for us + ({customerRepo, orderRepo} = givenBoundCrudRepositories( + ctx.dataSource, + repositoryClass, + )); + // inclusionResolvers should be defined. And resolver for each + // relation should be created by the hasManyFactory at this point. + expect(customerRepo.inclusionResolvers).to.not.be.undefined(); + expect(orderRepo.inclusionResolvers).to.not.be.undefined(); + expect(customerRepo.orders.inclusionResolver).to.not.be.undefined(); + expect( + customerRepo.customers.inclusionResolver, + ).to.not.be.undefined(); + // inclusionResolvers shouldn't setup yet at this point + expect(customerRepo.inclusionResolvers).to.deepEqual(new Map()); + + await ctx.dataSource.automigrate([Customer.name, Order.name]); + }), + ); + + beforeEach(async () => { + customerRepo.inclusionResolvers.set( + 'orders', + customerRepo.orders.inclusionResolver, + ); + customerRepo.inclusionResolvers.set( + 'customers', + customerRepo.customers.inclusionResolver, + ); + await customerRepo.deleteAll(); + await orderRepo.deleteAll(); + }); + + it("defines a repository's inclusionResolvers property", () => { + expect(customerRepo.inclusionResolvers).to.not.be.undefined(); + expect(orderRepo.inclusionResolvers).to.not.be.undefined(); + }); + + it("throws an error if the repository doesn't have such relation names", async () => { + await orderRepo.create({ + customerId: existingCustomerId, + description: 'Order from Order McForder, the hoarder of Mordor', + }); + await expect( + customerRepo.find({include: [{relation: 'managers'}]}), + ).to.be.rejectedWith( + `Invalid "filter.include" entries: {"relation":"managers"}`, + ); + }); + + it('throws error if the target repository does not have the registered resolver', async () => { + await orderRepo.create({ + customerId: existingCustomerId, + description: 'Order from Order McForder, the hoarder of Mordor', + }); + // unregister the resolver + customerRepo.inclusionResolvers.delete('orders'); + + await expect( + customerRepo.find({include: [{relation: 'orders'}]}), + ).to.be.rejectedWith( + `Invalid "filter.include" entries: {"relation":"orders"}`, + ); + // reset + customerRepo.inclusionResolvers.set( + 'orders', + customerRepo.orders.inclusionResolver, + ); + }); + + it('simple has-many relation retrieve via find() method', async () => { + const c1 = await customerRepo.create({name: 'c1'}); + const o1 = await orderRepo.create({ + customerId: c1.id, + description: 'order from c1', + }); + const result = await customerRepo.find({ + include: [{relation: 'orders'}], + }); + + const expected = { + id: c1.id, + name: 'c1', + orders: [ + { + id: o1.id, + description: 'order from c1', + customerId: c1.id, + isShipped: features.emptyValue, + // eslint-disable-next-line @typescript-eslint/camelcase + shipment_id: features.emptyValue, + }, + ], + parentId: features.emptyValue, + }; + expect(toJSON(result)).to.deepEqual([toJSON(expected)]); + }); + + it('returns related instances to target models via find() method', async () => { + const c1 = await customerRepo.create({name: 'Thor'}); + const c2 = await customerRepo.create({name: 'Hella'}); + const o1 = await orderRepo.create({ + customerId: c1.id, + description: 'Mjolnir', + }); + const o2 = await orderRepo.create({ + customerId: c1.id, + description: 'Pizza', + }); + const o3 = await orderRepo.create({ + customerId: c2.id, + description: 'Blade', + }); + + const result = await customerRepo.find({ + include: [{relation: 'orders'}], + }); + + const expected = [ + { + id: c1.id, + name: 'Thor', + orders: [ + { + id: o1.id, + description: 'Mjolnir', + customerId: c1.id, + isShipped: features.emptyValue, + // eslint-disable-next-line @typescript-eslint/camelcase + shipment_id: features.emptyValue, + }, + { + id: o2.id, + description: 'Pizza', + customerId: c1.id, + isShipped: features.emptyValue, + // eslint-disable-next-line @typescript-eslint/camelcase + shipment_id: features.emptyValue, + }, + ], + parentId: features.emptyValue, + }, + { + id: c2.id, + name: 'Hella', + orders: [ + { + id: o3.id, + description: 'Blade', + customerId: c2.id, + isShipped: features.emptyValue, + // eslint-disable-next-line @typescript-eslint/camelcase + shipment_id: features.emptyValue, + }, + ], + parentId: features.emptyValue, + }, + ]; + expect(toJSON(result)).to.deepEqual(toJSON(expected)); + }); + + it('returns related instances to target models via findById() method', async () => { + const c1 = await customerRepo.create({name: 'Thor'}); + const c2 = await customerRepo.create({name: 'Hella'}); + await orderRepo.create({ + customerId: c1.id, + description: 'Mjolnir', + }); + await orderRepo.create({ + customerId: c1.id, + description: 'Pizza', + }); + const o3 = await orderRepo.create({ + customerId: c2.id, + description: 'Blade', + }); + + const result = await customerRepo.findById(c2.id, { + include: [{relation: 'orders'}], + }); + const expected = { + id: c2.id, + name: 'Hella', + orders: [ + { + id: o3.id, + description: 'Blade', + customerId: c2.id, + isShipped: features.emptyValue, + // eslint-disable-next-line @typescript-eslint/camelcase + shipment_id: features.emptyValue, + }, + ], + parentId: features.emptyValue, + }; + expect(toJSON(result)).to.deepEqual(toJSON(expected)); + }); + + it('throws when navigational properties are present when updating model instance', async () => { + const created = await customerRepo.create({name: 'c1'}); + const customerId = created.id; + + await orderRepo.create({ + description: 'Pen', + customerId, + }); + + const found = await customerRepo.findById(customerId, { + include: [{relation: 'orders'}], + }); + expect(found.orders).to.have.lengthOf(1); + + found.name = 'updated name'; + await expect(customerRepo.save(found)).to.be.rejectedWith( + 'The `Customer` instance is not valid. Details: `orders` is not defined in the model (value: undefined).', + ); + }); + }); + }, + ); +} diff --git a/packages/repository-tests/src/crud/relations/fixtures/repositories/customer.repository.ts b/packages/repository-tests/src/crud/relations/fixtures/repositories/customer.repository.ts index 606d34a75420..9de711d82756 100644 --- a/packages/repository-tests/src/crud/relations/fixtures/repositories/customer.repository.ts +++ b/packages/repository-tests/src/crud/relations/fixtures/repositories/customer.repository.ts @@ -49,8 +49,8 @@ export function createCustomerRepo(repoClass: CrudRepositoryCtor) { addressRepositoryGetter: Getter, ) { super(Customer, db); - // create a has-many relation from this public method const ordersMeta = this.entityClass.definition.relations['orders']; + // create a has-many relation through this public method this.orders = createHasManyRepositoryFactory( ordersMeta as HasManyDefinition, orderRepositoryGetter, diff --git a/packages/repository-tests/src/crud/relations/helpers.ts b/packages/repository-tests/src/crud/relations/helpers.ts index 902ddb16a886..79a989b5fc39 100644 --- a/packages/repository-tests/src/crud/relations/helpers.ts +++ b/packages/repository-tests/src/crud/relations/helpers.ts @@ -49,5 +49,10 @@ export function givenBoundCrudRepositories( async () => customerRepo, ); - return {customerRepo, orderRepo, shipmentRepo, addressRepo}; + return { + customerRepo, + orderRepo, + shipmentRepo, + addressRepo, + }; } diff --git a/packages/repository-tests/src/types.repository-tests.ts b/packages/repository-tests/src/types.repository-tests.ts index a37cc4e7b251..c41e335a53be 100644 --- a/packages/repository-tests/src/types.repository-tests.ts +++ b/packages/repository-tests/src/types.repository-tests.ts @@ -56,6 +56,14 @@ export interface CrudFeatures { * Default: `false` */ supportsTransactions: boolean; + + /** + * Does the repository provide `inclusionResolvers` object where resolvers + * can be registered? + * + * Default: `true` + */ + supportsInclusionResolvers: boolean; } /** diff --git a/packages/repository/src/__tests__/unit/repositories/relations-helpers/flatten-targets-of-one-to-many-relation-helpers.unit.ts b/packages/repository/src/__tests__/unit/repositories/relations-helpers/flatten-targets-of-one-to-many-relation-helpers.unit.ts new file mode 100644 index 000000000000..7030c810faab --- /dev/null +++ b/packages/repository/src/__tests__/unit/repositories/relations-helpers/flatten-targets-of-one-to-many-relation-helpers.unit.ts @@ -0,0 +1,37 @@ +// Copyright IBM Corp. 2019. 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 {expect} from '@loopback/testlab'; +import {flattenTargetsOfOneToManyRelation} from '../../../..'; +import {createProduct} from './relations-helpers-fixtures'; + +describe('flattenTargetsOfOneToManyRelation', () => { + describe('gets the result of using reduceAsArray strategy for hasMany relation', () => { + it('gets the result of passing in a single sourceId', () => { + const pen = createProduct({name: 'pen', categoryId: 1}); + const pencil = createProduct({name: 'pencil', categoryId: 1}); + createProduct({name: 'eraser', categoryId: 2}); + + const result = flattenTargetsOfOneToManyRelation( + [1], + [pen, pencil], + 'categoryId', + ); + expect(result).to.eql([[pen, pencil]]); + }); + it('gets the result of passing in multiple sourceIds', () => { + const pen = createProduct({name: 'pen', categoryId: 1}); + const pencil = createProduct({name: 'pencil', categoryId: 1}); + const eraser = createProduct({name: 'eraser', categoryId: 2}); + // use [2, 1] here to show the order of sourceIds matters + const result = flattenTargetsOfOneToManyRelation( + [2, 1], + [pen, pencil, eraser], + 'categoryId', + ); + expect(result).to.deepEqual([[eraser], [pen, pencil]]); + }); + }); +}); diff --git a/packages/repository/src/relations/has-many/has-many-repository.factory.ts b/packages/repository/src/relations/has-many/has-many-repository.factory.ts index 1e1c6c1717f6..e95972b8cec1 100644 --- a/packages/repository/src/relations/has-many/has-many-repository.factory.ts +++ b/packages/repository/src/relations/has-many/has-many-repository.factory.ts @@ -7,8 +7,9 @@ import * as debugFactory from 'debug'; import {DataObject} from '../../common-types'; import {Entity} from '../../model'; import {EntityCrudRepository} from '../../repositories/repository'; -import {Getter, HasManyDefinition} from '../relation.types'; +import {Getter, HasManyDefinition, InclusionResolver} from '../relation.types'; import {resolveHasManyMetadata} from './has-many.helpers'; +import {createHasManyInclusionResolver} from './has-many.inclusion-resolver'; import { DefaultHasManyRepository, HasManyRepository, @@ -16,9 +17,20 @@ import { const debug = debugFactory('loopback:repository:has-many-repository-factory'); -export type HasManyRepositoryFactory = ( - fkValue: ForeignKeyType, -) => HasManyRepository; +export interface HasManyRepositoryFactory< + Target extends Entity, + ForeignKeyType +> { + /** + * Invoke the function to obtain HasManyRepository. + */ + (fkValue: ForeignKeyType): HasManyRepository; + + /** + * Use `resolver` property to obtain an InclusionResolver for this relation. + */ + inclusionResolver: InclusionResolver; +} /** * Enforces a constraint on a repository based on a relationship contract @@ -43,7 +55,9 @@ export function createHasManyRepositoryFactory< ): HasManyRepositoryFactory { const meta = resolveHasManyMetadata(relationMetadata); debug('Resolved HasMany relation metadata: %o', meta); - return function(fkValue: ForeignKeyType) { + const result: HasManyRepositoryFactory = function( + fkValue: ForeignKeyType, + ) { // eslint-disable-next-line @typescript-eslint/no-explicit-any const constraint: any = {[meta.keyTo]: fkValue}; return new DefaultHasManyRepository< @@ -52,4 +66,9 @@ export function createHasManyRepositoryFactory< EntityCrudRepository >(targetRepositoryGetter, constraint as DataObject); }; + result.inclusionResolver = createHasManyInclusionResolver( + meta, + targetRepositoryGetter, + ); + return result; } diff --git a/packages/repository/src/relations/has-many/has-many.helpers.ts b/packages/repository/src/relations/has-many/has-many.helpers.ts index 702f44721e43..25adeda5ff6d 100644 --- a/packages/repository/src/relations/has-many/has-many.helpers.ts +++ b/packages/repository/src/relations/has-many/has-many.helpers.ts @@ -7,7 +7,7 @@ import * as debugFactory from 'debug'; import {camelCase} from 'lodash'; import {InvalidRelationError} from '../../errors'; import {isTypeResolver} from '../../type-resolver'; -import {HasManyDefinition} from '../relation.types'; +import {HasManyDefinition, RelationType} from '../relation.types'; const debug = debugFactory('loopback:repository:has-many-helpers'); @@ -30,6 +30,11 @@ export type HasManyResolvedDefinition = HasManyDefinition & { export function resolveHasManyMetadata( relationMeta: HasManyDefinition, ): HasManyResolvedDefinition { + if ((relationMeta.type as RelationType) !== RelationType.hasMany) { + const reason = 'relation type must be HasMany'; + throw new InvalidRelationError(reason, relationMeta); + } + if (!isTypeResolver(relationMeta.target)) { const reason = 'target must be a type resolver'; throw new InvalidRelationError(reason, relationMeta); diff --git a/packages/repository/src/relations/has-many/has-many.inclusion-resolver.ts b/packages/repository/src/relations/has-many/has-many.inclusion-resolver.ts new file mode 100644 index 000000000000..65052dbd309b --- /dev/null +++ b/packages/repository/src/relations/has-many/has-many.inclusion-resolver.ts @@ -0,0 +1,77 @@ +// Copyright IBM Corp. 2019. 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 * as debugFactory from 'debug'; +import {AnyObject, Options} from '../../common-types'; +import {Entity} from '../../model'; +import {Filter, Inclusion} from '../../query'; +import {EntityCrudRepository} from '../../repositories/repository'; +import { + findByForeignKeys, + flattenTargetsOfOneToManyRelation, + StringKeyOf, +} from '../relation.helpers'; +import {Getter, HasManyDefinition, InclusionResolver} from '../relation.types'; +import {resolveHasManyMetadata} from './has-many.helpers'; + +const debug = debugFactory('loopback:repository:has-many-inclusion-resolver'); + +/** + * Creates hasMany resolver for the relation. + * + * @param meta - metadata of the hasMany relation + * @param getTargetRepo - Target repo + */ + +export function createHasManyInclusionResolver< + Target extends Entity, + TargetID, + TargetRelations extends object +>( + meta: HasManyDefinition, + getTargetRepo: Getter< + EntityCrudRepository + >, +): InclusionResolver { + const relationMeta = resolveHasManyMetadata(meta); + + return async function fetchHasManyModels( + entities: Entity[], + inclusion: Inclusion, + options?: Options, + ): Promise<((Target & TargetRelations)[] | undefined)[]> { + if (!entities.length) return []; + + debug('Fetching target models for entities:', entities); + debug('Relation metadata:', relationMeta); + + const sourceKey = relationMeta.keyFrom; + const sourceIds = entities.map(e => (e as AnyObject)[sourceKey]); + const targetKey = relationMeta.keyTo as StringKeyOf; + + debug('Parameters:', {sourceKey, sourceIds, targetKey}); + debug('sourceId types', sourceIds.map(i => typeof i)); + + const targetRepo = await getTargetRepo(); + const targetsFound = await findByForeignKeys( + targetRepo, + targetKey, + sourceIds, + inclusion.scope as Filter, + options, + ); + + debug('Targets found:', targetsFound); + + const result = flattenTargetsOfOneToManyRelation( + sourceIds, + targetsFound, + targetKey, + ); + + debug('fetchHasManyModels result', result); + return result; + }; +} diff --git a/packages/repository/src/relations/has-many/index.ts b/packages/repository/src/relations/has-many/index.ts index 0025021d819a..7780e8b4c018 100644 --- a/packages/repository/src/relations/has-many/index.ts +++ b/packages/repository/src/relations/has-many/index.ts @@ -6,3 +6,4 @@ export * from './has-many.decorator'; export * from './has-many.repository'; export * from './has-many-repository.factory'; +export * from './has-many.inclusion-resolver'; diff --git a/packages/repository/src/relations/relation.helpers.ts b/packages/repository/src/relations/relation.helpers.ts index 55b0cd9daa16..416e313fe10d 100644 --- a/packages/repository/src/relations/relation.helpers.ts +++ b/packages/repository/src/relations/relation.helpers.ts @@ -141,6 +141,39 @@ function isInclusionAllowed( return allowed; } +/** + * Returns an array of arrays. Each nested array has one or more instances + * as a result of one to many relation. The order of arrays is based on + * the order of sourceIds + * + * @param sourceIds - One value or array of values of the target key + * @param targetEntities - target entities that satisfy targetKey's value (ids). + * @param targetKey - name of the target key + * + * @return + */ +export function flattenTargetsOfOneToManyRelation( + sourceIds: unknown[], + targetEntities: Target[], + targetKey: StringKeyOf, +): (Target[] | undefined)[] { + debug('flattenTargetsOfOneToManyRelation'); + debug('sourceIds', sourceIds); + debug('sourceId types', sourceIds.map(i => typeof i)); + debug('targetEntities', targetEntities); + debug('targetKey', targetKey); + + const lookup = buildLookupMap( + targetEntities, + targetKey, + reduceAsArray, + ); + + debug('lookup map', lookup); + + return flattenMapByKeys(sourceIds, lookup); +} + /** * Returns an array of instances from the target map. The order of arrays is based on * the order of sourceIds