forked from nestjs/nest
-
Notifications
You must be signed in to change notification settings - Fork 0
/
get-injection-providers.util.ts
50 lines (47 loc) 路 1.55 KB
/
get-injection-providers.util.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
import {
InjectionToken,
Provider,
FactoryProvider,
OptionalFactoryDependency,
} from '../../interfaces';
/**
* check if x is OptionalFactoryDependency, based on prototype presence
* (to avoid classes with a static 'token' field)
* @param x
* @returns x is OptionalFactoryDependency
*/
function isOptionalFactoryDependency(
x: InjectionToken | OptionalFactoryDependency,
): x is OptionalFactoryDependency {
return !!((x as any)?.token && !(x as any)?.prototype);
}
const mapInjectToTokens = (t: InjectionToken | OptionalFactoryDependency) =>
isOptionalFactoryDependency(t) ? t.token : t;
/**
*
* @param providers List of a module's providers
* @param tokens Injection tokens needed for a useFactory function (usually the module's options' token)
* @returns All the providers needed for the tokens' injection (searched recursively)
*/
export function getInjectionProviders(
providers: Provider[],
tokens: FactoryProvider['inject'],
): Provider[] {
const result: Provider[] = [];
let search: InjectionToken[] = tokens.map(mapInjectToTokens);
while (search.length > 0) {
const match = (providers ?? []).filter(
p =>
!result.includes(p) && // this prevents circular loops and duplication
(search.includes(p as any) || search.includes((p as any)?.provide)),
);
result.push(...match);
// get injection tokens of the matched providers, if any
search = match
.filter(p => (p as any)?.inject)
.map(p => (p as FactoryProvider).inject)
.flat()
.map(mapInjectToTokens);
}
return result;
}