@angular/core 16.2.9 → 16.2.11
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/src/application_init.mjs +3 -2
- package/esm2022/src/application_ref.mjs +3 -3
- package/esm2022/src/di/r3_injector.mjs +6 -1
- package/esm2022/src/render3/debug/injector_profiler.mjs +26 -8
- package/esm2022/src/render3/reactive_lview_consumer.mjs +2 -7
- package/esm2022/src/render3/util/injector_discovery_utils.mjs +26 -2
- package/esm2022/src/version.mjs +1 -1
- package/esm2022/testing/src/logger.mjs +3 -3
- package/fesm2022/core.mjs +147 -105
- package/fesm2022/core.mjs.map +1 -1
- package/fesm2022/rxjs-interop.mjs +1 -1
- package/fesm2022/testing.mjs +118 -101
- package/fesm2022/testing.mjs.map +1 -1
- package/index.d.ts +4 -3
- package/package.json +1 -1
- package/rxjs-interop/index.d.ts +1 -1
- package/schematics/migrations/guard-and-resolve-interfaces/bundle.js +13 -13
- package/schematics/migrations/remove-module-id/bundle.js +14 -14
- package/schematics/ng-generate/standalone-migration/bundle.js +376 -376
- package/schematics/ng-generate/standalone-migration/bundle.js.map +1 -1
- package/testing/index.d.ts +1 -1
|
@@ -115,6 +115,11 @@ function getProviderImportsContainer(injector) {
|
|
|
115
115
|
if (defTypeRef === null) {
|
|
116
116
|
return null;
|
|
117
117
|
}
|
|
118
|
+
// In standalone applications, the root environment injector created by bootstrapApplication
|
|
119
|
+
// may have no associated "instance".
|
|
120
|
+
if (defTypeRef.instance === null) {
|
|
121
|
+
return null;
|
|
122
|
+
}
|
|
118
123
|
return defTypeRef.instance.constructor;
|
|
119
124
|
}
|
|
120
125
|
/**
|
|
@@ -296,12 +301,25 @@ function walkProviderTreeToDiscoverImportPaths(providerToPath, visitedContainers
|
|
|
296
301
|
* @returns an array of objects representing the providers of the given injector
|
|
297
302
|
*/
|
|
298
303
|
function getEnvironmentInjectorProviders(injector) {
|
|
304
|
+
const providerRecords = getFrameworkDIDebugData().resolverToProviders.get(injector) ?? [];
|
|
305
|
+
// platform injector has no provider imports container so can we skip trying to
|
|
306
|
+
// find import paths
|
|
307
|
+
if (isPlatformInjector(injector)) {
|
|
308
|
+
return providerRecords;
|
|
309
|
+
}
|
|
299
310
|
const providerImportsContainer = getProviderImportsContainer(injector);
|
|
300
311
|
if (providerImportsContainer === null) {
|
|
312
|
+
// There is a special case where the bootstrapped component does not
|
|
313
|
+
// import any NgModules. In this case the environment injector connected to
|
|
314
|
+
// that component is the root injector, which does not have a provider imports
|
|
315
|
+
// container (and thus no concept of module import paths). Therefore we simply
|
|
316
|
+
// return the provider records as is.
|
|
317
|
+
if (isRootInjector(injector)) {
|
|
318
|
+
return providerRecords;
|
|
319
|
+
}
|
|
301
320
|
throwError('Could not determine where injector providers were configured.');
|
|
302
321
|
}
|
|
303
322
|
const providerToPath = getProviderImportPaths(providerImportsContainer);
|
|
304
|
-
const providerRecords = getFrameworkDIDebugData().resolverToProviders.get(injector) ?? [];
|
|
305
323
|
return providerRecords.map(providerRecord => {
|
|
306
324
|
let importPath = providerToPath.get(providerRecord.provider) ?? [providerImportsContainer];
|
|
307
325
|
const def = getComponentDef(providerImportsContainer);
|
|
@@ -314,6 +332,12 @@ function getEnvironmentInjectorProviders(injector) {
|
|
|
314
332
|
return { ...providerRecord, importPath };
|
|
315
333
|
});
|
|
316
334
|
}
|
|
335
|
+
function isPlatformInjector(injector) {
|
|
336
|
+
return injector instanceof R3Injector && injector.scopes.has('platform');
|
|
337
|
+
}
|
|
338
|
+
function isRootInjector(injector) {
|
|
339
|
+
return injector instanceof R3Injector && injector.scopes.has('root');
|
|
340
|
+
}
|
|
317
341
|
/**
|
|
318
342
|
* Gets the providers configured on an injector.
|
|
319
343
|
*
|
|
@@ -457,4 +481,4 @@ function getModuleInjectorOfNodeInjector(injector) {
|
|
|
457
481
|
}
|
|
458
482
|
return moduleInjector;
|
|
459
483
|
}
|
|
460
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"injector_discovery_utils.js","sourceRoot":"","sources":["../../../../../../../../packages/core/src/render3/util/injector_discovery_utils.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAIH,OAAO,EAAC,cAAc,EAAe,MAAM,yBAAyB,CAAC;AAErE,OAAO,EAAC,YAAY,EAAC,MAAM,wBAAwB,CAAC;AACpD,OAAO,EAAiB,gBAAgB,EAAC,MAAM,8BAA8B,CAAC;AAC9E,OAAO,EAAC,mBAAmB,EAAE,UAAU,EAAC,MAAM,sBAAsB,CAAC;AAErE,OAAO,EAAC,WAAW,IAAI,sBAAsB,EAAC,MAAM,gCAAgC,CAAC;AACrF,OAAO,EAAC,WAAW,EAAC,MAAM,wBAAwB,CAAC;AACnD,OAAO,EAAC,UAAU,EAAC,MAAM,mBAAmB,CAAC;AAE7C,OAAO,EAAC,eAAe,EAAC,MAAM,eAAe,CAAC;AAC9C,OAAO,EAAC,oBAAoB,EAAE,oBAAoB,EAAE,yBAAyB,EAAE,YAAY,EAAC,MAAM,OAAO,CAAC;AAC1G,OAAO,EAAC,uBAAuB,EAAC,MAAM,sCAAsC,CAAC;AAI7E,OAAO,EAAC,QAAQ,EAAS,KAAK,EAAC,MAAM,oBAAoB,CAAC;AAE1D,OAAO,EAAC,sBAAsB,EAAE,qBAAqB,EAAE,iBAAiB,EAAC,MAAM,kBAAkB,CAAC;AAElG;;;;;;;;;GASG;AACH,MAAM,UAAU,6BAA6B,CACzC,QAAkB,EAClB,KAAgC;IAClC,6FAA6F;IAC7F,kFAAkF;IAClF,uFAAuF;IACvF,6CAA6C;IAC7C,MAAM,QAAQ,GAAG,QAAQ,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,EAAE,EAAC,IAAI,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAC,CAAC,CAAC;IACzE,IAAI,QAAQ,KAAK,IAAI,EAAE;QACrB,MAAM,IAAI,KAAK,CAAC,mCAAmC,KAAK,oBAAoB,CAAC,CAAC;KAC/E;IAED,IAAI,UAAU,GAAmB,QAAQ,CAAC;IAC1C,IAAI,QAAQ,YAAY,YAAY,EAAE;QACpC,UAAU,GAAG,oBAAoB,CAAC,QAAQ,CAAC,CAAC;KAC7C;IAED,MAAM,EAAC,6BAA6B,EAAC,GAAG,uBAAuB,EAAE,CAAC;IAElE,IAAI,YAAY,GACZ,6BAA6B,CAAC,GAAG,CAAC,UAAU,CAAC,EAAE,GAAG,EAAE,CAAC,KAAsB,CAAC,IAAI,EAAE,CAAC;IAEvF,MAAM,cAAc,GAAG,yBAAyB,CAAC,QAAQ,CAAC,CAAC;IAC3D,YAAY,GAAG,YAAY,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE;QACpC,MAAM,KAAK,GAAG,GAAG,CAAC,KAA4B,CAAC;QAC/C,GAAG,CAAC,KAAK,GAAG;YACV,QAAQ,EAAE,CAAC,uCAA+B,KAAK,CAAC,yCAAiC;YACjF,IAAI,EAAE,CAAC,mCAA2B,KAAK,CAAC,qCAA6B;YACrE,IAAI,EAAE,CAAC,mCAA2B,KAAK,CAAC,qCAA6B;YACrE,QAAQ,EAAE,CAAC,uCAA+B,KAAK,CAAC,yCAAiC;SAClF,CAAC;QAEF,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,cAAc,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;YAC9C,MAAM,eAAe,GAAG,cAAc,CAAC,CAAC,CAAC,CAAC;YAE1C,iDAAiD;YACjD,IAAI,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,KAAK,CAAC,QAAQ,EAAE;gBACjC,SAAS;aACV;YAED,qCAAqC;YACrC,IAAI,GAAG,CAAC,KAAK,CAAC,IAAI,IAAI,eAAe,YAAY,mBAAmB,EAAE;gBACpE,MAAM;aACP;YAED,MAAM,QAAQ,GACV,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,KAAsB,EAAE,IAAI,EAAE,EAAC,IAAI,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAC,CAAC,CAAC;YAExF,IAAI,QAAQ,KAAK,IAAI,EAAE;gBACrB,0FAA0F;gBAC1F,4FAA4F;gBAC5F,4FAA4F;gBAC5F,mBAAmB;gBACnB,IAAI,GAAG,CAAC,KAAK,CAAC,IAAI,EAAE;oBAClB,MAAM,aAAa,GAAG,cAAc,CAAC,CAAC,CAAC,CAAC;oBACxC,MAAM,uBAAuB,GACzB,aAAa,CAAC,GAAG,CAAC,GAAG,CAAC,KAAsB,EAAE,IAAI,EAAE,EAAC,GAAG,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAC,CAAC,CAAC;oBAExF,IAAI,uBAAuB,KAAK,IAAI,EAAE;wBACpC,GAAG,CAAC,UAAU,GAAG,eAAe,CAAC;qBAClC;oBAED,MAAM;iBACP;gBAED,GAAG,CAAC,UAAU,GAAG,eAAe,CAAC;gBACjC,MAAM;aACP;YAED,mDAAmD;YACnD,IAAI,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,KAAK,CAAC,IAAI,EAAE;gBAC7B,MAAM;aACP;SACF;QAED,OAAO,GAAG,CAAC;IACb,CAAC,CAAC,CAAC;IAEH,OAAO,EAAC,QAAQ,EAAE,YAAY,EAAC,CAAC;AAClC,CAAC;AAED;;;;;;;;;;GAUG;AACH,SAAS,2BAA2B,CAAC,QAAkB;IACrD,MAAM,EAAC,6BAA6B,EAAC,GAAG,uBAAuB,EAAE,CAAC;IAElE,mFAAmF;IACnF,oFAAoF;IACpF,8CAA8C;IAC9C,IAAI,6BAA6B,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE;QAC/C,OAAO,6BAA6B,CAAC,GAAG,CAAC,QAAQ,CAAE,CAAC;KACrD;IAED,iFAAiF;IACjF,wEAAwE;IACxE,MAAM,UAAU,GAAG,QAAQ,CAAC,GAAG,CAAC,sBAAsB,EAAE,IAAI,EAAE,EAAC,IAAI,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAC,CAAE,CAAC;IAE7F,iEAAiE;IACjE,+FAA+F;IAC/F,sCAAsC;IACtC,IAAI,UAAU,KAAK,IAAI,EAAE;QACvB,OAAO,IAAI,CAAC;KACb;IAED,OAAO,UAAU,CAAC,QAAQ,CAAC,WAAW,CAAC;AACzC,CAAC;AAED;;;;;;GAMG;AACH,SAAS,wBAAwB,CAAC,QAAsB;IACtD,MAAM,UAAU,GAAG,oBAAoB,CAAC,QAAQ,CAAC,CAAC;IAClD,MAAM,EAAC,mBAAmB,EAAC,GAAG,uBAAuB,EAAE,CAAC;IACxD,OAAO,mBAAmB,CAAC,GAAG,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;AACnD,CAAC;AAED;;;;;;;;;;;;;;;;;;;GAmBG;AACH,SAAS,sBAAsB,CAAC,wBAAuC;IAErE,MAAM,cAAc,GAAG,IAAI,GAAG,EAA4D,CAAC;IAC3F,MAAM,iBAAiB,GAAG,IAAI,GAAG,EAAiB,CAAC;IACnD,MAAM,OAAO,GAAG,qCAAqC,CAAC,cAAc,EAAE,iBAAiB,CAAC,CAAC;IAEzF,gBAAgB,CAAC,wBAAwB,EAAE,OAAO,EAAE,EAAE,EAAE,IAAI,GAAG,EAAE,CAAC,CAAC;IAEnE,OAAO,cAAc,CAAC;AACxB,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2FG;AACH,SAAS,qCAAqC,CAC1C,cAA6E,EAC7E,iBAAqC;IAEvC,OAAO,CAAC,QAAwB,EAAE,SAA8C,EAAE,EAAE;QAClF,4DAA4D;QAC5D,6FAA6F;QAC7F,YAAY;QACZ,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE;YACjC,cAAc,CAAC,GAAG,CAAC,QAAQ,EAAE,CAAC,SAAS,CAAC,CAAC,CAAC;SAC3C;QAED,0EAA0E;QAC1E,2EAA2E;QAC3E,4EAA4E;QAC5E,wBAAwB;QACxB,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE;YACrC,mDAAmD;YACnD,KAAK,MAAM,IAAI,IAAI,cAAc,CAAC,IAAI,EAAE,EAAE;gBACxC,MAAM,kBAAkB,GAAG,cAAc,CAAC,GAAG,CAAC,IAAI,CAAE,CAAC;gBAErD,IAAI,YAAY,GAAG,cAAc,CAAC,SAAS,CAAC,CAAC;gBAC7C,IAAI,CAAC,YAAY,EAAE;oBACjB,MAAM,QAAQ,GACT,SAAiB,CAAC,QAAoC,CAAC;oBAC5D,YAAY,GAAG,cAAc,CAAC,QAAQ,CAAC,CAAC;iBACzC;gBAED,IAAI,CAAC,YAAY,EAAE;oBACjB,OAAO;iBACR;gBAED,MAAM,wBAAwB,GAAG,kBAAkB,CAAC,CAAC,CAAC,CAAC;gBAEvD,IAAI,gBAAgB,GAAG,KAAK,CAAC;gBAC7B,WAAW,CAAC,YAAY,CAAC,OAAO,EAAE,CAAC,YAAY,EAAE,EAAE;oBACjD,IAAI,gBAAgB,EAAE;wBACpB,OAAO;qBACR;oBAED,gBAAgB,GAAI,YAAoB,CAAC,QAAQ,KAAK,wBAAwB;wBAC1E,YAAY,KAAK,wBAAwB,CAAC;oBAE9C,IAAI,gBAAgB,EAAE;wBACpB,cAAc,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,OAAO,CAAC,SAAS,CAAC,CAAC;qBAC9C;gBACH,CAAC,CAAC,CAAC;aACJ;SACF;QAED,iBAAiB,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IACnC,CAAC,CAAC;AACJ,CAAC;AAED;;;;;GAKG;AACH,SAAS,+BAA+B,CAAC,QAA6B;IACpE,MAAM,wBAAwB,GAAG,2BAA2B,CAAC,QAAQ,CAAC,CAAC;IACvE,IAAI,wBAAwB,KAAK,IAAI,EAAE;QACrC,UAAU,CAAC,+DAA+D,CAAC,CAAC;KAC7E;IAED,MAAM,cAAc,GAAG,sBAAsB,CAAC,wBAAwB,CAAC,CAAC;IACxE,MAAM,eAAe,GAAG,uBAAuB,EAAE,CAAC,mBAAmB,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;IAE1F,OAAO,eAAe,CAAC,GAAG,CAAC,cAAc,CAAC,EAAE;QAC1C,IAAI,UAAU,GAAG,cAAc,CAAC,GAAG,CAAC,cAAc,CAAC,QAAQ,CAAC,IAAI,CAAC,wBAAwB,CAAC,CAAC;QAE3F,MAAM,GAAG,GAAG,eAAe,CAAC,wBAAwB,CAAC,CAAC;QACtD,MAAM,qBAAqB,GAAG,CAAC,CAAC,GAAG,EAAE,UAAU,CAAC;QAChD,8DAA8D;QAC9D,iFAAiF;QACjF,IAAI,qBAAqB,EAAE;YACzB,UAAU,GAAG,CAAC,wBAAwB,EAAE,GAAG,cAAc,CAAC,GAAG,CAAC,cAAc,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC;SAC/F;QAED,OAAO,EAAC,GAAG,cAAc,EAAE,UAAU,EAAC,CAAC;IACzC,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,oBAAoB,CAAC,QAAkB;IACrD,IAAI,QAAQ,YAAY,YAAY,EAAE;QACpC,OAAO,wBAAwB,CAAC,QAAQ,CAAC,CAAC;KAC3C;SAAM,IAAI,QAAQ,YAAY,mBAAmB,EAAE;QAClD,OAAO,+BAA+B,CAAC,QAA+B,CAAC,CAAC;KACzE;IAED,UAAU,CAAC,yEAAyE,CAAC,CAAC;AACxF,CAAC;AAED,MAAM,UAAU,yBAAyB,CAAC,QAAkB;IAC1D,MAAM,cAAc,GAAe,CAAC,QAAQ,CAAC,CAAC;IAC9C,+BAA+B,CAAC,QAAQ,EAAE,cAAc,CAAC,CAAC;IAC1D,OAAO,cAAc,CAAC;AACxB,CAAC;AAED,SAAS,+BAA+B,CACpC,QAAkB,EAAE,cAA0B;IAChD,MAAM,MAAM,GAAG,iBAAiB,CAAC,QAAQ,CAAC,CAAC;IAE3C,8EAA8E;IAC9E,wEAAwE;IACxE,qFAAqF;IACrF,IAAI,MAAM,KAAK,IAAI,EAAE;QACnB,IAAI,QAAQ,YAAY,YAAY,EAAE;YACpC,MAAM,aAAa,GAAG,cAAc,CAAC,CAAC,CAAC,CAAC;YACxC,IAAI,aAAa,YAAY,YAAY,EAAE;gBACzC,MAAM,cAAc,GAAG,+BAA+B,CAAC,aAAa,CAAC,CAAC;gBACtE,IAAI,cAAc,KAAK,IAAI,EAAE;oBAC3B,UAAU,CAAC,oEAAoE,CAAC,CAAC;iBAClF;gBAED,cAAc,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;gBACpC,+BAA+B,CAAC,cAAc,EAAE,cAAc,CAAC,CAAC;aACjE;YAED,OAAO,cAAc,CAAC;SACvB;KACF;SAAM;QACL,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC5B,+BAA+B,CAAC,MAAM,EAAE,cAAc,CAAC,CAAC;KACzD;IAED,OAAO,cAAc,CAAC;AACxB,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqCG;AACH,SAAS,iBAAiB,CAAC,QAAkB;IAC3C,IAAI,QAAQ,YAAY,UAAU,EAAE;QAClC,OAAO,QAAQ,CAAC,MAAM,CAAC;KACxB;IAED,IAAI,KAA6D,CAAC;IAClE,IAAI,KAAqB,CAAC;IAC1B,IAAI,QAAQ,YAAY,YAAY,EAAE;QACpC,KAAK,GAAG,oBAAoB,CAAC,QAAQ,CAAC,CAAC;QACvC,KAAK,GAAG,oBAAoB,CAAC,QAAQ,CAAC,CAAC;KACxC;SAAM,IAAI,QAAQ,YAAY,YAAY,EAAE;QAC3C,OAAO,IAAI,CAAC;KACb;SAAM;QACL,UAAU,CACN,yFAAyF,CAAC,CAAC;KAChG;IAED,MAAM,cAAc,GAAG,yBAAyB,CAC5C,KAA8D,EAAE,KAAK,CAAC,CAAC;IAE3E,IAAI,iBAAiB,CAAC,cAAc,CAAC,EAAE;QACrC,MAAM,mBAAmB,GAAG,sBAAsB,CAAC,cAAc,CAAC,CAAC;QACnE,MAAM,WAAW,GAAG,qBAAqB,CAAC,cAAc,EAAE,KAAK,CAAC,CAAC;QACjE,MAAM,WAAW,GAAG,WAAW,CAAC,KAAK,CAAC,CAAC;QACvC,MAAM,WAAW,GAAG,WAAW,CAAC,IAAI,CAAC,mBAAmB,mCAA2B,CAAU,CAAC;QAC9F,OAAO,IAAI,YAAY,CACnB,WAAoE,EAAE,WAAW,CAAC,CAAC;KACxF;SAAM;QACL,MAAM,eAAe,GAAG,KAAK,CAAC,QAAQ,CAAoB,CAAC;QAE3D,+FAA+F;QAC/F,qBAAqB;QACrB,wEAAwE;QACxE,iFAAiF;QACjF,iDAAiD;QACjD,oEAAoE;QACpE,MAAM,cAAc,GAAI,eAAe,CAAC,QAAgB,EAAE,MAAkB,CAAC;QAE7E,IAAI,cAAc,YAAY,YAAY,EAAE;YAC1C,OAAO,cAAc,CAAC;SACvB;KACF;IAED,OAAO,IAAI,CAAC;AACd,CAAC;AAED;;;;;GAKG;AACH,SAAS,+BAA+B,CAAC,QAAsB;IAC7D,IAAI,KAAqB,CAAC;IAC1B,IAAI,QAAQ,YAAY,YAAY,EAAE;QACpC,KAAK,GAAG,oBAAoB,CAAC,QAAQ,CAAC,CAAC;KACxC;SAAM;QACL,UAAU,CAAC,oEAAoE,CAAC,CAAC;KAClF;IAED,MAAM,eAAe,GAAG,KAAK,CAAC,QAAQ,CAAoB,CAAC;IAC3D,MAAM,cAAc,GAAG,eAAe,CAAC,cAAc,CAAC;IACtD,IAAI,CAAC,cAAc,EAAE;QACnB,UAAU,CAAC,oEAAoE,CAAC,CAAC;KAClF;IAED,OAAO,cAAc,CAAC;AACxB,CAAC","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {InjectionToken} from '../../di/injection_token';\nimport {Injector} from '../../di/injector';\nimport {getInjectorDef, InjectorType} from '../../di/interface/defs';\nimport {InjectFlags, InternalInjectFlags} from '../../di/interface/injector';\nimport {NullInjector} from '../../di/null_injector';\nimport {SingleProvider, walkProviderTree} from '../../di/provider_collection';\nimport {EnvironmentInjector, R3Injector} from '../../di/r3_injector';\nimport {Type} from '../../interface/type';\nimport {NgModuleRef as viewEngine_NgModuleRef} from '../../linker/ng_module_factory';\nimport {deepForEach} from '../../util/array_utils';\nimport {throwError} from '../../util/assert';\nimport type {ChainedInjector} from '../component_ref';\nimport {getComponentDef} from '../definition';\nimport {getNodeInjectorLView, getNodeInjectorTNode, getParentInjectorLocation, NodeInjector} from '../di';\nimport {getFrameworkDIDebugData} from '../debug/framework_injector_profiler';\nimport {InjectedService, ProviderRecord} from '../debug/injector_profiler';\nimport {NodeInjectorOffset} from '../interfaces/injector';\nimport {TContainerNode, TElementContainerNode, TElementNode, TNode} from '../interfaces/node';\nimport {INJECTOR, LView, TVIEW} from '../interfaces/view';\n\nimport {getParentInjectorIndex, getParentInjectorView, hasParentInjector} from './injector_utils';\n\n/**\n * Discovers the dependencies of an injectable instance. Provides DI information about each\n * dependency that the injectable was instantiated with, including where they were provided from.\n *\n * @param injector An injector instance\n * @param token a DI token that was constructed by the given injector instance\n * @returns an object that contains the created instance of token as well as all of the dependencies\n * that it was instantiated with OR undefined if the token was not created within the given\n * injector.\n */\nexport function getDependenciesFromInjectable<T>(\n    injector: Injector,\n    token: Type<T>|InjectionToken<T>): {instance: T; dependencies: InjectedService[]}|undefined {\n  // First we check to see if the token given maps to an actual instance in the injector given.\n  // We use `self: true` because we only want to look at the injector we were given.\n  // We use `optional: true` because it's possible that the token we were given was never\n  // constructed by the injector we were given.\n  const instance = injector.get(token, null, {self: true, optional: true});\n  if (instance === null) {\n    throw new Error(`Unable to determine instance of ${token} in given injector`);\n  }\n\n  let diResolver: Injector|LView = injector;\n  if (injector instanceof NodeInjector) {\n    diResolver = getNodeInjectorLView(injector);\n  }\n\n  const {resolverToTokenToDependencies} = getFrameworkDIDebugData();\n\n  let dependencies =\n      resolverToTokenToDependencies.get(diResolver)?.get?.(token as Type<unknown>) ?? [];\n\n  const resolutionPath = getInjectorResolutionPath(injector);\n  dependencies = dependencies.map(dep => {\n    const flags = dep.flags as InternalInjectFlags;\n    dep.flags = {\n      optional: (InternalInjectFlags.Optional & flags) === InternalInjectFlags.Optional,\n      host: (InternalInjectFlags.Host & flags) === InternalInjectFlags.Host,\n      self: (InternalInjectFlags.Self & flags) === InternalInjectFlags.Self,\n      skipSelf: (InternalInjectFlags.SkipSelf & flags) === InternalInjectFlags.SkipSelf,\n    };\n\n    for (let i = 0; i < resolutionPath.length; i++) {\n      const injectorToCheck = resolutionPath[i];\n\n      // if skipSelf is true we skip the first injector\n      if (i === 0 && dep.flags.skipSelf) {\n        continue;\n      }\n\n      // host only applies to NodeInjectors\n      if (dep.flags.host && injectorToCheck instanceof EnvironmentInjector) {\n        break;\n      }\n\n      const instance =\n          injectorToCheck.get(dep.token as Type<unknown>, null, {self: true, optional: true});\n\n      if (instance !== null) {\n        // if host flag is true we double check that we can get the service from the first element\n        // in the resolution path by using the host flag. This is done to make sure that we've found\n        // the correct providing injector, and not a node injector that is connected to our path via\n        // a router outlet.\n        if (dep.flags.host) {\n          const firstInjector = resolutionPath[0];\n          const lookupFromFirstInjector =\n              firstInjector.get(dep.token as Type<unknown>, null, {...dep.flags, optional: true});\n\n          if (lookupFromFirstInjector !== null) {\n            dep.providedIn = injectorToCheck;\n          }\n\n          break;\n        }\n\n        dep.providedIn = injectorToCheck;\n        break;\n      }\n\n      // if self is true we stop after the first injector\n      if (i === 0 && dep.flags.self) {\n        break;\n      }\n    }\n\n    return dep;\n  });\n\n  return {instance, dependencies};\n}\n\n/**\n * Gets the class associated with an injector that contains a provider `imports` array in it's\n * definition\n *\n * For Module Injectors this returns the NgModule constructor.\n *\n * For Standalone injectors this returns the standalone component constructor.\n *\n * @param injector Injector an injector instance\n * @returns the constructor where the `imports` array that configures this injector is located\n */\nfunction getProviderImportsContainer(injector: Injector): Type<unknown>|null {\n  const {standaloneInjectorToComponent} = getFrameworkDIDebugData();\n\n  // standalone components configure providers through a component def, so we have to\n  // use the standalone component associated with this injector if Injector represents\n  // a standalone components EnvironmentInjector\n  if (standaloneInjectorToComponent.has(injector)) {\n    return standaloneInjectorToComponent.get(injector)!;\n  }\n\n  // Module injectors configure providers through their NgModule def, so we use the\n  // injector to lookup its NgModuleRef and through that grab its instance\n  const defTypeRef = injector.get(viewEngine_NgModuleRef, null, {self: true, optional: true})!;\n\n  // If we can't find an associated imports container, return null.\n  // This could be the case if this function is called with an R3Injector that does not represent\n  // a standalone component or NgModule.\n  if (defTypeRef === null) {\n    return null;\n  }\n\n  return defTypeRef.instance.constructor;\n}\n\n/**\n * Gets the providers configured on a NodeInjector\n *\n * @param injector A NodeInjector instance\n * @returns ProviderRecord[] an array of objects representing the providers configured on this\n *     injector\n */\nfunction getNodeInjectorProviders(injector: NodeInjector): ProviderRecord[] {\n  const diResolver = getNodeInjectorLView(injector);\n  const {resolverToProviders} = getFrameworkDIDebugData();\n  return resolverToProviders.get(diResolver) ?? [];\n}\n\n/**\n * Gets a mapping of providers configured on an injector to their import paths\n *\n * ModuleA -> imports ModuleB\n * ModuleB -> imports ModuleC\n * ModuleB -> provides MyServiceA\n * ModuleC -> provides MyServiceB\n *\n * getProviderImportPaths(ModuleA)\n * > Map(2) {\n *   MyServiceA => [ModuleA, ModuleB]\n *   MyServiceB => [ModuleA, ModuleB, ModuleC]\n *  }\n *\n * @param providerImportsContainer constructor of class that contains an `imports` array in it's\n *     definition\n * @returns A Map object that maps providers to an array of constructors representing it's import\n *     path\n *\n */\nfunction getProviderImportPaths(providerImportsContainer: Type<unknown>):\n    Map<SingleProvider, (Type<unknown>| InjectorType<unknown>)[]> {\n  const providerToPath = new Map<SingleProvider, (Type<unknown>| InjectorType<unknown>)[]>();\n  const visitedContainers = new Set<Type<unknown>>();\n  const visitor = walkProviderTreeToDiscoverImportPaths(providerToPath, visitedContainers);\n\n  walkProviderTree(providerImportsContainer, visitor, [], new Set());\n\n  return providerToPath;\n}\n\n/**\n *\n * Higher order function that returns a visitor for WalkProviderTree\n *\n * Takes in a Map and Set to keep track of the providers and containers\n * visited, so that we can discover the import paths of these providers\n * during the traversal.\n *\n * This visitor takes advantage of the fact that walkProviderTree performs a\n * postorder traversal of the provider tree for the passed in container. Because postorder\n * traversal recursively processes subtrees from leaf nodes until the traversal reaches the root,\n * we write a visitor that constructs provider import paths in reverse.\n *\n *\n * We use the visitedContainers set defined outside this visitor\n * because we want to run some logic only once for\n * each container in the tree. That logic can be described as:\n *\n *\n * 1. for each discovered_provider and discovered_path in the incomplete provider paths we've\n * already discovered\n * 2. get the first container in discovered_path\n * 3. if that first container is in the imports array of the container we're visiting\n *    Then the container we're visiting is also in the import path of discovered_provider, so we\n *    unshift discovered_path with the container we're currently visiting\n *\n *\n * Example Run:\n * ```\n *                 ┌──────────┐\n *                 │containerA│\n *      ┌─imports-─┤          ├──imports─┐\n *      │          │  provA   │          │\n *      │          │  provB   │          │\n *      │          └──────────┘          │\n *      │                                │\n *     ┌▼─────────┐             ┌────────▼─┐\n *     │containerB│             │containerC│\n *     │          │             │          │\n *     │  provD   │             │  provF   │\n *     │  provE   │             │  provG   │\n *     └──────────┘             └──────────┘\n * ```\n *\n * Each step of the traversal,\n *\n * ```\n * visitor(provD, containerB)\n * providerToPath === Map { provD => [containerB] }\n * visitedContainers === Set { containerB }\n *\n * visitor(provE, containerB)\n * providerToPath === Map { provD => [containerB], provE => [containerB] }\n * visitedContainers === Set { containerB }\n *\n * visitor(provF, containerC)\n * providerToPath === Map { provD => [containerB], provE => [containerB], provF => [containerC] }\n * visitedContainers === Set { containerB, containerC }\n *\n * visitor(provG, containerC)\n * providerToPath === Map {\n *   provD => [containerB], provE => [containerB], provF => [containerC], provG => [containerC]\n * }\n * visitedContainers === Set { containerB, containerC }\n *\n * visitor(provA, containerA)\n * providerToPath === Map {\n *   provD => [containerA, containerB],\n *   provE => [containerA, containerB],\n *   provF => [containerA, containerC],\n *   provG => [containerA, containerC],\n *   provA => [containerA]\n * }\n * visitedContainers === Set { containerB, containerC, containerA }\n *\n * visitor(provB, containerA)\n * providerToPath === Map {\n *   provD => [containerA, containerB],\n *   provE => [containerA, containerB],\n *   provF => [containerA, containerC],\n *   provG => [containerA, containerC],\n *   provA => [containerA]\n *   provB => [containerA]\n * }\n * visitedContainers === Set { containerB, containerC, containerA }\n * ```\n *\n * @param providerToPath Map map of providers to paths that this function fills\n * @param visitedContainers Set a set to keep track of the containers we've already visited\n * @return function(provider SingleProvider, container: Type<unknown> | InjectorType<unknown>) =>\n *     void\n */\nfunction walkProviderTreeToDiscoverImportPaths(\n    providerToPath: Map<SingleProvider, (Type<unknown>| InjectorType<unknown>)[]>,\n    visitedContainers: Set<Type<unknown>>):\n    (provider: SingleProvider, container: Type<unknown>|InjectorType<unknown>) => void {\n  return (provider: SingleProvider, container: Type<unknown>|InjectorType<unknown>) => {\n    // If the provider is not already in the providerToPath map,\n    // add an entry with the provider as the key and an array containing the current container as\n    // the value\n    if (!providerToPath.has(provider)) {\n      providerToPath.set(provider, [container]);\n    }\n\n    // This block will run exactly once for each container in the import tree.\n    // This is where we run the logic to check the imports array of the current\n    // container to see if it's the next container in the path for our currently\n    // discovered providers.\n    if (!visitedContainers.has(container)) {\n      // Iterate through the providers we've already seen\n      for (const prov of providerToPath.keys()) {\n        const existingImportPath = providerToPath.get(prov)!;\n\n        let containerDef = getInjectorDef(container);\n        if (!containerDef) {\n          const ngModule: Type<unknown>|undefined =\n              (container as any).ngModule as Type<unknown>| undefined;\n          containerDef = getInjectorDef(ngModule);\n        }\n\n        if (!containerDef) {\n          return;\n        }\n\n        const lastContainerAddedToPath = existingImportPath[0];\n\n        let isNextStepInPath = false;\n        deepForEach(containerDef.imports, (moduleImport) => {\n          if (isNextStepInPath) {\n            return;\n          }\n\n          isNextStepInPath = (moduleImport as any).ngModule === lastContainerAddedToPath ||\n              moduleImport === lastContainerAddedToPath;\n\n          if (isNextStepInPath) {\n            providerToPath.get(prov)?.unshift(container);\n          }\n        });\n      }\n    }\n\n    visitedContainers.add(container);\n  };\n}\n\n/**\n * Gets the providers configured on an EnvironmentInjector\n *\n * @param injector EnvironmentInjector\n * @returns an array of objects representing the providers of the given injector\n */\nfunction getEnvironmentInjectorProviders(injector: EnvironmentInjector): ProviderRecord[] {\n  const providerImportsContainer = getProviderImportsContainer(injector);\n  if (providerImportsContainer === null) {\n    throwError('Could not determine where injector providers were configured.');\n  }\n\n  const providerToPath = getProviderImportPaths(providerImportsContainer);\n  const providerRecords = getFrameworkDIDebugData().resolverToProviders.get(injector) ?? [];\n\n  return providerRecords.map(providerRecord => {\n    let importPath = providerToPath.get(providerRecord.provider) ?? [providerImportsContainer];\n\n    const def = getComponentDef(providerImportsContainer);\n    const isStandaloneComponent = !!def?.standalone;\n    // We prepend the component constructor in the standalone case\n    // because walkProviderTree does not visit this constructor during it's traversal\n    if (isStandaloneComponent) {\n      importPath = [providerImportsContainer, ...providerToPath.get(providerRecord.provider) ?? []];\n    }\n\n    return {...providerRecord, importPath};\n  });\n}\n\n/**\n * Gets the providers configured on an injector.\n *\n * @param injector the injector to lookup the providers of\n * @returns ProviderRecord[] an array of objects representing the providers of the given injector\n */\nexport function getInjectorProviders(injector: Injector): ProviderRecord[] {\n  if (injector instanceof NodeInjector) {\n    return getNodeInjectorProviders(injector);\n  } else if (injector instanceof EnvironmentInjector) {\n    return getEnvironmentInjectorProviders(injector as EnvironmentInjector);\n  }\n\n  throwError('getInjectorProviders only supports NodeInjector and EnvironmentInjector');\n}\n\nexport function getInjectorResolutionPath(injector: Injector): Injector[] {\n  const resolutionPath: Injector[] = [injector];\n  getInjectorResolutionPathHelper(injector, resolutionPath);\n  return resolutionPath;\n}\n\nfunction getInjectorResolutionPathHelper(\n    injector: Injector, resolutionPath: Injector[]): Injector[] {\n  const parent = getInjectorParent(injector);\n\n  // if getInjectorParent can't find a parent, then we've either reached the end\n  // of the path, or we need to move from the Element Injector tree to the\n  // module injector tree using the first injector in our path as the connection point.\n  if (parent === null) {\n    if (injector instanceof NodeInjector) {\n      const firstInjector = resolutionPath[0];\n      if (firstInjector instanceof NodeInjector) {\n        const moduleInjector = getModuleInjectorOfNodeInjector(firstInjector);\n        if (moduleInjector === null) {\n          throwError('NodeInjector must have some connection to the module injector tree');\n        }\n\n        resolutionPath.push(moduleInjector);\n        getInjectorResolutionPathHelper(moduleInjector, resolutionPath);\n      }\n\n      return resolutionPath;\n    }\n  } else {\n    resolutionPath.push(parent);\n    getInjectorResolutionPathHelper(parent, resolutionPath);\n  }\n\n  return resolutionPath;\n}\n\n/**\n * Gets the parent of an injector.\n *\n * This function is not able to make the jump from the Element Injector Tree to the Module\n * injector tree. This is because the \"parent\" (the next step in the reoslution path)\n * of a root NodeInjector is dependent on which NodeInjector ancestor initiated\n * the DI lookup. See getInjectorResolutionPath for a function that can make this jump.\n *\n * In the below diagram:\n * ```ts\n * getInjectorParent(NodeInjectorB)\n *  > NodeInjectorA\n * getInjectorParent(NodeInjectorA) // or getInjectorParent(getInjectorParent(NodeInjectorB))\n *  > null // cannot jump to ModuleInjector tree\n * ```\n *\n * ```\n *                ┌───────┐                ┌───────────────────┐\n *    ┌───────────┤ModuleA├───Injector────►│EnvironmentInjector│\n *    │           └───┬───┘                └───────────────────┘\n *    │               │\n *    │           bootstraps\n *    │               │\n *    │               │\n *    │          ┌────▼─────┐                 ┌─────────────┐\n * declares      │ComponentA├────Injector────►│NodeInjectorA│\n *    │          └────┬─────┘                 └─────▲───────┘\n *    │               │                             │\n *    │            renders                        parent\n *    │               │                             │\n *    │          ┌────▼─────┐                 ┌─────┴───────┐\n *    └─────────►│ComponentB├────Injector────►│NodeInjectorB│\n *               └──────────┘                 └─────────────┘\n *```\n *\n * @param injector an Injector to get the parent of\n * @returns Injector the parent of the given injector\n */\nfunction getInjectorParent(injector: Injector): Injector|null {\n  if (injector instanceof R3Injector) {\n    return injector.parent;\n  }\n\n  let tNode: TElementNode|TContainerNode|TElementContainerNode|null;\n  let lView: LView<unknown>;\n  if (injector instanceof NodeInjector) {\n    tNode = getNodeInjectorTNode(injector);\n    lView = getNodeInjectorLView(injector);\n  } else if (injector instanceof NullInjector) {\n    return null;\n  } else {\n    throwError(\n        'getInjectorParent only support injectors of type R3Injector, NodeInjector, NullInjector');\n  }\n\n  const parentLocation = getParentInjectorLocation(\n      tNode as TElementNode | TContainerNode | TElementContainerNode, lView);\n\n  if (hasParentInjector(parentLocation)) {\n    const parentInjectorIndex = getParentInjectorIndex(parentLocation);\n    const parentLView = getParentInjectorView(parentLocation, lView);\n    const parentTView = parentLView[TVIEW];\n    const parentTNode = parentTView.data[parentInjectorIndex + NodeInjectorOffset.TNODE] as TNode;\n    return new NodeInjector(\n        parentTNode as TElementNode | TContainerNode | TElementContainerNode, parentLView);\n  } else {\n    const chainedInjector = lView[INJECTOR] as ChainedInjector;\n\n    // Case where chainedInjector.injector is an OutletInjector and chainedInjector.injector.parent\n    // is a NodeInjector.\n    // todo(aleksanderbodurri): ideally nothing in packages/core should deal\n    // directly with router concerns. Refactor this so that we can make the jump from\n    // NodeInjector -> OutletInjector -> NodeInjector\n    // without explictly relying on types contracts from packages/router\n    const injectorParent = (chainedInjector.injector as any)?.parent as Injector;\n\n    if (injectorParent instanceof NodeInjector) {\n      return injectorParent;\n    }\n  }\n\n  return null;\n}\n\n/**\n * Gets the module injector of a NodeInjector.\n *\n * @param injector NodeInjector to get module injector of\n * @returns Injector representing module injector of the given NodeInjector\n */\nfunction getModuleInjectorOfNodeInjector(injector: NodeInjector): Injector {\n  let lView: LView<unknown>;\n  if (injector instanceof NodeInjector) {\n    lView = getNodeInjectorLView(injector);\n  } else {\n    throwError('getModuleInjectorOfNodeInjector must be called with a NodeInjector');\n  }\n\n  const chainedInjector = lView[INJECTOR] as ChainedInjector;\n  const moduleInjector = chainedInjector.parentInjector;\n  if (!moduleInjector) {\n    throwError('NodeInjector must have some connection to the module injector tree');\n  }\n\n  return moduleInjector;\n}\n"]}
|
|
484
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"injector_discovery_utils.js","sourceRoot":"","sources":["../../../../../../../../packages/core/src/render3/util/injector_discovery_utils.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {InjectionToken} from '../../di/injection_token';\nimport {Injector} from '../../di/injector';\nimport {getInjectorDef, InjectorType} from '../../di/interface/defs';\nimport {InjectFlags, InternalInjectFlags} from '../../di/interface/injector';\nimport {NullInjector} from '../../di/null_injector';\nimport {SingleProvider, walkProviderTree} from '../../di/provider_collection';\nimport {EnvironmentInjector, R3Injector} from '../../di/r3_injector';\nimport {Type} from '../../interface/type';\nimport {NgModuleRef as viewEngine_NgModuleRef} from '../../linker/ng_module_factory';\nimport {deepForEach} from '../../util/array_utils';\nimport {throwError} from '../../util/assert';\nimport type {ChainedInjector} from '../component_ref';\nimport {getComponentDef} from '../definition';\nimport {getNodeInjectorLView, getNodeInjectorTNode, getParentInjectorLocation, NodeInjector} from '../di';\nimport {getFrameworkDIDebugData} from '../debug/framework_injector_profiler';\nimport {InjectedService, ProviderRecord} from '../debug/injector_profiler';\nimport {NodeInjectorOffset} from '../interfaces/injector';\nimport {TContainerNode, TElementContainerNode, TElementNode, TNode} from '../interfaces/node';\nimport {INJECTOR, LView, TVIEW} from '../interfaces/view';\n\nimport {getParentInjectorIndex, getParentInjectorView, hasParentInjector} from './injector_utils';\n\n/**\n * Discovers the dependencies of an injectable instance. Provides DI information about each\n * dependency that the injectable was instantiated with, including where they were provided from.\n *\n * @param injector An injector instance\n * @param token a DI token that was constructed by the given injector instance\n * @returns an object that contains the created instance of token as well as all of the dependencies\n * that it was instantiated with OR undefined if the token was not created within the given\n * injector.\n */\nexport function getDependenciesFromInjectable<T>(\n    injector: Injector,\n    token: Type<T>|InjectionToken<T>): {instance: T; dependencies: InjectedService[]}|undefined {\n  // First we check to see if the token given maps to an actual instance in the injector given.\n  // We use `self: true` because we only want to look at the injector we were given.\n  // We use `optional: true` because it's possible that the token we were given was never\n  // constructed by the injector we were given.\n  const instance = injector.get(token, null, {self: true, optional: true});\n  if (instance === null) {\n    throw new Error(`Unable to determine instance of ${token} in given injector`);\n  }\n\n  let diResolver: Injector|LView = injector;\n  if (injector instanceof NodeInjector) {\n    diResolver = getNodeInjectorLView(injector);\n  }\n\n  const {resolverToTokenToDependencies} = getFrameworkDIDebugData();\n\n  let dependencies =\n      resolverToTokenToDependencies.get(diResolver)?.get?.(token as Type<unknown>) ?? [];\n\n  const resolutionPath = getInjectorResolutionPath(injector);\n  dependencies = dependencies.map(dep => {\n    const flags = dep.flags as InternalInjectFlags;\n    dep.flags = {\n      optional: (InternalInjectFlags.Optional & flags) === InternalInjectFlags.Optional,\n      host: (InternalInjectFlags.Host & flags) === InternalInjectFlags.Host,\n      self: (InternalInjectFlags.Self & flags) === InternalInjectFlags.Self,\n      skipSelf: (InternalInjectFlags.SkipSelf & flags) === InternalInjectFlags.SkipSelf,\n    };\n\n    for (let i = 0; i < resolutionPath.length; i++) {\n      const injectorToCheck = resolutionPath[i];\n\n      // if skipSelf is true we skip the first injector\n      if (i === 0 && dep.flags.skipSelf) {\n        continue;\n      }\n\n      // host only applies to NodeInjectors\n      if (dep.flags.host && injectorToCheck instanceof EnvironmentInjector) {\n        break;\n      }\n\n      const instance =\n          injectorToCheck.get(dep.token as Type<unknown>, null, {self: true, optional: true});\n\n      if (instance !== null) {\n        // if host flag is true we double check that we can get the service from the first element\n        // in the resolution path by using the host flag. This is done to make sure that we've found\n        // the correct providing injector, and not a node injector that is connected to our path via\n        // a router outlet.\n        if (dep.flags.host) {\n          const firstInjector = resolutionPath[0];\n          const lookupFromFirstInjector =\n              firstInjector.get(dep.token as Type<unknown>, null, {...dep.flags, optional: true});\n\n          if (lookupFromFirstInjector !== null) {\n            dep.providedIn = injectorToCheck;\n          }\n\n          break;\n        }\n\n        dep.providedIn = injectorToCheck;\n        break;\n      }\n\n      // if self is true we stop after the first injector\n      if (i === 0 && dep.flags.self) {\n        break;\n      }\n    }\n\n    return dep;\n  });\n\n  return {instance, dependencies};\n}\n\n/**\n * Gets the class associated with an injector that contains a provider `imports` array in it's\n * definition\n *\n * For Module Injectors this returns the NgModule constructor.\n *\n * For Standalone injectors this returns the standalone component constructor.\n *\n * @param injector Injector an injector instance\n * @returns the constructor where the `imports` array that configures this injector is located\n */\nfunction getProviderImportsContainer(injector: Injector): Type<unknown>|null {\n  const {standaloneInjectorToComponent} = getFrameworkDIDebugData();\n\n  // standalone components configure providers through a component def, so we have to\n  // use the standalone component associated with this injector if Injector represents\n  // a standalone components EnvironmentInjector\n  if (standaloneInjectorToComponent.has(injector)) {\n    return standaloneInjectorToComponent.get(injector)!;\n  }\n\n  // Module injectors configure providers through their NgModule def, so we use the\n  // injector to lookup its NgModuleRef and through that grab its instance\n  const defTypeRef = injector.get(viewEngine_NgModuleRef, null, {self: true, optional: true})!;\n\n  // If we can't find an associated imports container, return null.\n  // This could be the case if this function is called with an R3Injector that does not represent\n  // a standalone component or NgModule.\n  if (defTypeRef === null) {\n    return null;\n  }\n\n  // In standalone applications, the root environment injector created by bootstrapApplication\n  // may have no associated \"instance\".\n  if (defTypeRef.instance === null) {\n    return null;\n  }\n\n  return defTypeRef.instance.constructor;\n}\n\n/**\n * Gets the providers configured on a NodeInjector\n *\n * @param injector A NodeInjector instance\n * @returns ProviderRecord[] an array of objects representing the providers configured on this\n *     injector\n */\nfunction getNodeInjectorProviders(injector: NodeInjector): ProviderRecord[] {\n  const diResolver = getNodeInjectorLView(injector);\n  const {resolverToProviders} = getFrameworkDIDebugData();\n  return resolverToProviders.get(diResolver) ?? [];\n}\n\n/**\n * Gets a mapping of providers configured on an injector to their import paths\n *\n * ModuleA -> imports ModuleB\n * ModuleB -> imports ModuleC\n * ModuleB -> provides MyServiceA\n * ModuleC -> provides MyServiceB\n *\n * getProviderImportPaths(ModuleA)\n * > Map(2) {\n *   MyServiceA => [ModuleA, ModuleB]\n *   MyServiceB => [ModuleA, ModuleB, ModuleC]\n *  }\n *\n * @param providerImportsContainer constructor of class that contains an `imports` array in it's\n *     definition\n * @returns A Map object that maps providers to an array of constructors representing it's import\n *     path\n *\n */\nfunction getProviderImportPaths(providerImportsContainer: Type<unknown>):\n    Map<SingleProvider, (Type<unknown>| InjectorType<unknown>)[]> {\n  const providerToPath = new Map<SingleProvider, (Type<unknown>| InjectorType<unknown>)[]>();\n  const visitedContainers = new Set<Type<unknown>>();\n  const visitor = walkProviderTreeToDiscoverImportPaths(providerToPath, visitedContainers);\n\n  walkProviderTree(providerImportsContainer, visitor, [], new Set());\n\n  return providerToPath;\n}\n\n/**\n *\n * Higher order function that returns a visitor for WalkProviderTree\n *\n * Takes in a Map and Set to keep track of the providers and containers\n * visited, so that we can discover the import paths of these providers\n * during the traversal.\n *\n * This visitor takes advantage of the fact that walkProviderTree performs a\n * postorder traversal of the provider tree for the passed in container. Because postorder\n * traversal recursively processes subtrees from leaf nodes until the traversal reaches the root,\n * we write a visitor that constructs provider import paths in reverse.\n *\n *\n * We use the visitedContainers set defined outside this visitor\n * because we want to run some logic only once for\n * each container in the tree. That logic can be described as:\n *\n *\n * 1. for each discovered_provider and discovered_path in the incomplete provider paths we've\n * already discovered\n * 2. get the first container in discovered_path\n * 3. if that first container is in the imports array of the container we're visiting\n *    Then the container we're visiting is also in the import path of discovered_provider, so we\n *    unshift discovered_path with the container we're currently visiting\n *\n *\n * Example Run:\n * ```\n *                 ┌──────────┐\n *                 │containerA│\n *      ┌─imports-─┤          ├──imports─┐\n *      │          │  provA   │          │\n *      │          │  provB   │          │\n *      │          └──────────┘          │\n *      │                                │\n *     ┌▼─────────┐             ┌────────▼─┐\n *     │containerB│             │containerC│\n *     │          │             │          │\n *     │  provD   │             │  provF   │\n *     │  provE   │             │  provG   │\n *     └──────────┘             └──────────┘\n * ```\n *\n * Each step of the traversal,\n *\n * ```\n * visitor(provD, containerB)\n * providerToPath === Map { provD => [containerB] }\n * visitedContainers === Set { containerB }\n *\n * visitor(provE, containerB)\n * providerToPath === Map { provD => [containerB], provE => [containerB] }\n * visitedContainers === Set { containerB }\n *\n * visitor(provF, containerC)\n * providerToPath === Map { provD => [containerB], provE => [containerB], provF => [containerC] }\n * visitedContainers === Set { containerB, containerC }\n *\n * visitor(provG, containerC)\n * providerToPath === Map {\n *   provD => [containerB], provE => [containerB], provF => [containerC], provG => [containerC]\n * }\n * visitedContainers === Set { containerB, containerC }\n *\n * visitor(provA, containerA)\n * providerToPath === Map {\n *   provD => [containerA, containerB],\n *   provE => [containerA, containerB],\n *   provF => [containerA, containerC],\n *   provG => [containerA, containerC],\n *   provA => [containerA]\n * }\n * visitedContainers === Set { containerB, containerC, containerA }\n *\n * visitor(provB, containerA)\n * providerToPath === Map {\n *   provD => [containerA, containerB],\n *   provE => [containerA, containerB],\n *   provF => [containerA, containerC],\n *   provG => [containerA, containerC],\n *   provA => [containerA]\n *   provB => [containerA]\n * }\n * visitedContainers === Set { containerB, containerC, containerA }\n * ```\n *\n * @param providerToPath Map map of providers to paths that this function fills\n * @param visitedContainers Set a set to keep track of the containers we've already visited\n * @return function(provider SingleProvider, container: Type<unknown> | InjectorType<unknown>) =>\n *     void\n */\nfunction walkProviderTreeToDiscoverImportPaths(\n    providerToPath: Map<SingleProvider, (Type<unknown>| InjectorType<unknown>)[]>,\n    visitedContainers: Set<Type<unknown>>):\n    (provider: SingleProvider, container: Type<unknown>|InjectorType<unknown>) => void {\n  return (provider: SingleProvider, container: Type<unknown>|InjectorType<unknown>) => {\n    // If the provider is not already in the providerToPath map,\n    // add an entry with the provider as the key and an array containing the current container as\n    // the value\n    if (!providerToPath.has(provider)) {\n      providerToPath.set(provider, [container]);\n    }\n\n    // This block will run exactly once for each container in the import tree.\n    // This is where we run the logic to check the imports array of the current\n    // container to see if it's the next container in the path for our currently\n    // discovered providers.\n    if (!visitedContainers.has(container)) {\n      // Iterate through the providers we've already seen\n      for (const prov of providerToPath.keys()) {\n        const existingImportPath = providerToPath.get(prov)!;\n\n        let containerDef = getInjectorDef(container);\n        if (!containerDef) {\n          const ngModule: Type<unknown>|undefined =\n              (container as any).ngModule as Type<unknown>| undefined;\n          containerDef = getInjectorDef(ngModule);\n        }\n\n        if (!containerDef) {\n          return;\n        }\n\n        const lastContainerAddedToPath = existingImportPath[0];\n\n        let isNextStepInPath = false;\n        deepForEach(containerDef.imports, (moduleImport) => {\n          if (isNextStepInPath) {\n            return;\n          }\n\n          isNextStepInPath = (moduleImport as any).ngModule === lastContainerAddedToPath ||\n              moduleImport === lastContainerAddedToPath;\n\n          if (isNextStepInPath) {\n            providerToPath.get(prov)?.unshift(container);\n          }\n        });\n      }\n    }\n\n    visitedContainers.add(container);\n  };\n}\n\n/**\n * Gets the providers configured on an EnvironmentInjector\n *\n * @param injector EnvironmentInjector\n * @returns an array of objects representing the providers of the given injector\n */\nfunction getEnvironmentInjectorProviders(injector: EnvironmentInjector): ProviderRecord[] {\n  const providerRecords = getFrameworkDIDebugData().resolverToProviders.get(injector) ?? [];\n\n  // platform injector has no provider imports container so can we skip trying to\n  // find import paths\n  if (isPlatformInjector(injector)) {\n    return providerRecords;\n  }\n\n  const providerImportsContainer = getProviderImportsContainer(injector);\n  if (providerImportsContainer === null) {\n    // There is a special case where the bootstrapped component does not\n    // import any NgModules. In this case the environment injector connected to\n    // that component is the root injector, which does not have a provider imports\n    // container (and thus no concept of module import paths). Therefore we simply\n    // return the provider records as is.\n    if (isRootInjector(injector)) {\n      return providerRecords;\n    }\n\n    throwError('Could not determine where injector providers were configured.');\n  }\n\n  const providerToPath = getProviderImportPaths(providerImportsContainer);\n\n  return providerRecords.map(providerRecord => {\n    let importPath = providerToPath.get(providerRecord.provider) ?? [providerImportsContainer];\n\n    const def = getComponentDef(providerImportsContainer);\n    const isStandaloneComponent = !!def?.standalone;\n    // We prepend the component constructor in the standalone case\n    // because walkProviderTree does not visit this constructor during it's traversal\n    if (isStandaloneComponent) {\n      importPath = [providerImportsContainer, ...providerToPath.get(providerRecord.provider) ?? []];\n    }\n\n    return {...providerRecord, importPath};\n  });\n}\n\nfunction isPlatformInjector(injector: Injector) {\n  return injector instanceof R3Injector && injector.scopes.has('platform');\n}\n\nfunction isRootInjector(injector: Injector) {\n  return injector instanceof R3Injector && injector.scopes.has('root');\n}\n\n/**\n * Gets the providers configured on an injector.\n *\n * @param injector the injector to lookup the providers of\n * @returns ProviderRecord[] an array of objects representing the providers of the given injector\n */\nexport function getInjectorProviders(injector: Injector): ProviderRecord[] {\n  if (injector instanceof NodeInjector) {\n    return getNodeInjectorProviders(injector);\n  } else if (injector instanceof EnvironmentInjector) {\n    return getEnvironmentInjectorProviders(injector as EnvironmentInjector);\n  }\n\n  throwError('getInjectorProviders only supports NodeInjector and EnvironmentInjector');\n}\n\nexport function getInjectorResolutionPath(injector: Injector): Injector[] {\n  const resolutionPath: Injector[] = [injector];\n  getInjectorResolutionPathHelper(injector, resolutionPath);\n  return resolutionPath;\n}\n\nfunction getInjectorResolutionPathHelper(\n    injector: Injector, resolutionPath: Injector[]): Injector[] {\n  const parent = getInjectorParent(injector);\n\n  // if getInjectorParent can't find a parent, then we've either reached the end\n  // of the path, or we need to move from the Element Injector tree to the\n  // module injector tree using the first injector in our path as the connection point.\n  if (parent === null) {\n    if (injector instanceof NodeInjector) {\n      const firstInjector = resolutionPath[0];\n      if (firstInjector instanceof NodeInjector) {\n        const moduleInjector = getModuleInjectorOfNodeInjector(firstInjector);\n        if (moduleInjector === null) {\n          throwError('NodeInjector must have some connection to the module injector tree');\n        }\n\n        resolutionPath.push(moduleInjector);\n        getInjectorResolutionPathHelper(moduleInjector, resolutionPath);\n      }\n\n      return resolutionPath;\n    }\n  } else {\n    resolutionPath.push(parent);\n    getInjectorResolutionPathHelper(parent, resolutionPath);\n  }\n\n  return resolutionPath;\n}\n\n/**\n * Gets the parent of an injector.\n *\n * This function is not able to make the jump from the Element Injector Tree to the Module\n * injector tree. This is because the \"parent\" (the next step in the reoslution path)\n * of a root NodeInjector is dependent on which NodeInjector ancestor initiated\n * the DI lookup. See getInjectorResolutionPath for a function that can make this jump.\n *\n * In the below diagram:\n * ```ts\n * getInjectorParent(NodeInjectorB)\n *  > NodeInjectorA\n * getInjectorParent(NodeInjectorA) // or getInjectorParent(getInjectorParent(NodeInjectorB))\n *  > null // cannot jump to ModuleInjector tree\n * ```\n *\n * ```\n *                ┌───────┐                ┌───────────────────┐\n *    ┌───────────┤ModuleA├───Injector────►│EnvironmentInjector│\n *    │           └───┬───┘                └───────────────────┘\n *    │               │\n *    │           bootstraps\n *    │               │\n *    │               │\n *    │          ┌────▼─────┐                 ┌─────────────┐\n * declares      │ComponentA├────Injector────►│NodeInjectorA│\n *    │          └────┬─────┘                 └─────▲───────┘\n *    │               │                             │\n *    │            renders                        parent\n *    │               │                             │\n *    │          ┌────▼─────┐                 ┌─────┴───────┐\n *    └─────────►│ComponentB├────Injector────►│NodeInjectorB│\n *               └──────────┘                 └─────────────┘\n *```\n *\n * @param injector an Injector to get the parent of\n * @returns Injector the parent of the given injector\n */\nfunction getInjectorParent(injector: Injector): Injector|null {\n  if (injector instanceof R3Injector) {\n    return injector.parent;\n  }\n\n  let tNode: TElementNode|TContainerNode|TElementContainerNode|null;\n  let lView: LView<unknown>;\n  if (injector instanceof NodeInjector) {\n    tNode = getNodeInjectorTNode(injector);\n    lView = getNodeInjectorLView(injector);\n  } else if (injector instanceof NullInjector) {\n    return null;\n  } else {\n    throwError(\n        'getInjectorParent only support injectors of type R3Injector, NodeInjector, NullInjector');\n  }\n\n  const parentLocation = getParentInjectorLocation(\n      tNode as TElementNode | TContainerNode | TElementContainerNode, lView);\n\n  if (hasParentInjector(parentLocation)) {\n    const parentInjectorIndex = getParentInjectorIndex(parentLocation);\n    const parentLView = getParentInjectorView(parentLocation, lView);\n    const parentTView = parentLView[TVIEW];\n    const parentTNode = parentTView.data[parentInjectorIndex + NodeInjectorOffset.TNODE] as TNode;\n    return new NodeInjector(\n        parentTNode as TElementNode | TContainerNode | TElementContainerNode, parentLView);\n  } else {\n    const chainedInjector = lView[INJECTOR] as ChainedInjector;\n\n    // Case where chainedInjector.injector is an OutletInjector and chainedInjector.injector.parent\n    // is a NodeInjector.\n    // todo(aleksanderbodurri): ideally nothing in packages/core should deal\n    // directly with router concerns. Refactor this so that we can make the jump from\n    // NodeInjector -> OutletInjector -> NodeInjector\n    // without explictly relying on types contracts from packages/router\n    const injectorParent = (chainedInjector.injector as any)?.parent as Injector;\n\n    if (injectorParent instanceof NodeInjector) {\n      return injectorParent;\n    }\n  }\n\n  return null;\n}\n\n/**\n * Gets the module injector of a NodeInjector.\n *\n * @param injector NodeInjector to get module injector of\n * @returns Injector representing module injector of the given NodeInjector\n */\nfunction getModuleInjectorOfNodeInjector(injector: NodeInjector): Injector {\n  let lView: LView<unknown>;\n  if (injector instanceof NodeInjector) {\n    lView = getNodeInjectorLView(injector);\n  } else {\n    throwError('getModuleInjectorOfNodeInjector must be called with a NodeInjector');\n  }\n\n  const chainedInjector = lView[INJECTOR] as ChainedInjector;\n  const moduleInjector = chainedInjector.parentInjector;\n  if (!moduleInjector) {\n    throwError('NodeInjector must have some connection to the module injector tree');\n  }\n\n  return moduleInjector;\n}\n"]}
|
package/esm2022/src/version.mjs
CHANGED
|
@@ -21,5 +21,5 @@ export class Version {
|
|
|
21
21
|
/**
|
|
22
22
|
* @publicApi
|
|
23
23
|
*/
|
|
24
|
-
export const VERSION = new Version('16.2.
|
|
24
|
+
export const VERSION = new Version('16.2.11');
|
|
25
25
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2NvcmUvc3JjL3ZlcnNpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7OztHQU1HO0FBRUg7Ozs7R0FJRztBQUNILE1BQU0sT0FBTyxPQUFPO0lBS2xCLFlBQW1CLElBQVk7UUFBWixTQUFJLEdBQUosSUFBSSxDQUFRO1FBQzdCLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNoQyxJQUFJLENBQUMsS0FBSyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDaEMsSUFBSSxDQUFDLEtBQUssR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDbEQsQ0FBQztDQUNGO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSxPQUFPLEdBQUcsSUFBSSxPQUFPLENBQUMsbUJBQW1CLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQGxpY2Vuc2VcbiAqIENvcHlyaWdodCBHb29nbGUgTExDIEFsbCBSaWdodHMgUmVzZXJ2ZWQuXG4gKlxuICogVXNlIG9mIHRoaXMgc291cmNlIGNvZGUgaXMgZ292ZXJuZWQgYnkgYW4gTUlULXN0eWxlIGxpY2Vuc2UgdGhhdCBjYW4gYmVcbiAqIGZvdW5kIGluIHRoZSBMSUNFTlNFIGZpbGUgYXQgaHR0cHM6Ly9hbmd1bGFyLmlvL2xpY2Vuc2VcbiAqL1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBSZXByZXNlbnRzIHRoZSB2ZXJzaW9uIG9mIEFuZ3VsYXJcbiAqXG4gKiBAcHVibGljQXBpXG4gKi9cbmV4cG9ydCBjbGFzcyBWZXJzaW9uIHtcbiAgcHVibGljIHJlYWRvbmx5IG1ham9yOiBzdHJpbmc7XG4gIHB1YmxpYyByZWFkb25seSBtaW5vcjogc3RyaW5nO1xuICBwdWJsaWMgcmVhZG9ubHkgcGF0Y2g6IHN0cmluZztcblxuICBjb25zdHJ1Y3RvcihwdWJsaWMgZnVsbDogc3RyaW5nKSB7XG4gICAgdGhpcy5tYWpvciA9IGZ1bGwuc3BsaXQoJy4nKVswXTtcbiAgICB0aGlzLm1pbm9yID0gZnVsbC5zcGxpdCgnLicpWzFdO1xuICAgIHRoaXMucGF0Y2ggPSBmdWxsLnNwbGl0KCcuJykuc2xpY2UoMikuam9pbignLicpO1xuICB9XG59XG5cbi8qKlxuICogQHB1YmxpY0FwaVxuICovXG5leHBvcnQgY29uc3QgVkVSU0lPTiA9IG5ldyBWZXJzaW9uKCcwLjAuMC1QTEFDRUhPTERFUicpO1xuIl19
|
|
@@ -25,10 +25,10 @@ export class Log {
|
|
|
25
25
|
result() {
|
|
26
26
|
return this.logItems.join('; ');
|
|
27
27
|
}
|
|
28
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.
|
|
29
|
-
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.
|
|
28
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.11", ngImport: i0, type: Log, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
29
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.11", ngImport: i0, type: Log }); }
|
|
30
30
|
}
|
|
31
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.
|
|
31
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.11", ngImport: i0, type: Log, decorators: [{
|
|
32
32
|
type: Injectable
|
|
33
33
|
}], ctorParameters: function () { return []; } });
|
|
34
34
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nZ2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvY29yZS90ZXN0aW5nL3NyYy9sb2dnZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7OztHQU1HO0FBRUgsT0FBTyxFQUFDLFVBQVUsRUFBQyxNQUFNLGVBQWUsQ0FBQzs7QUFHekMsTUFBTSxPQUFPLEdBQUc7SUFHZDtRQUNFLElBQUksQ0FBQyxRQUFRLEdBQUcsRUFBRSxDQUFDO0lBQ3JCLENBQUM7SUFFRCxHQUFHLENBQUMsS0FBUTtRQUNWLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQzVCLENBQUM7SUFFRCxFQUFFLENBQUMsS0FBUTtRQUNULE9BQU8sR0FBRyxFQUFFO1lBQ1YsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDNUIsQ0FBQyxDQUFDO0lBQ0osQ0FBQztJQUVELEtBQUs7UUFDSCxJQUFJLENBQUMsUUFBUSxHQUFHLEVBQUUsQ0FBQztJQUNyQixDQUFDO0lBRUQsTUFBTTtRQUNKLE9BQU8sSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDbEMsQ0FBQzt5SEF2QlUsR0FBRzs2SEFBSCxHQUFHOztzR0FBSCxHQUFHO2tCQURmLFVBQVUiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuaW1wb3J0IHtJbmplY3RhYmxlfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuQEluamVjdGFibGUoKVxuZXhwb3J0IGNsYXNzIExvZzxUID0gc3RyaW5nPiB7XG4gIGxvZ0l0ZW1zOiBUW107XG5cbiAgY29uc3RydWN0b3IoKSB7XG4gICAgdGhpcy5sb2dJdGVtcyA9IFtdO1xuICB9XG5cbiAgYWRkKHZhbHVlOiBUKTogdm9pZCB7XG4gICAgdGhpcy5sb2dJdGVtcy5wdXNoKHZhbHVlKTtcbiAgfVxuXG4gIGZuKHZhbHVlOiBUKSB7XG4gICAgcmV0dXJuICgpID0+IHtcbiAgICAgIHRoaXMubG9nSXRlbXMucHVzaCh2YWx1ZSk7XG4gICAgfTtcbiAgfVxuXG4gIGNsZWFyKCk6IHZvaWQge1xuICAgIHRoaXMubG9nSXRlbXMgPSBbXTtcbiAgfVxuXG4gIHJlc3VsdCgpOiBzdHJpbmcge1xuICAgIHJldHVybiB0aGlzLmxvZ0l0ZW1zLmpvaW4oJzsgJyk7XG4gIH1cbn1cbiJdfQ==
|
package/fesm2022/core.mjs
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* @license Angular v16.2.
|
|
2
|
+
* @license Angular v16.2.11
|
|
3
3
|
* (c) 2010-2022 Google LLC. https://angular.io/
|
|
4
4
|
* License: MIT
|
|
5
5
|
*/
|
|
@@ -604,6 +604,93 @@ function initNgDevMode() {
|
|
|
604
604
|
return false;
|
|
605
605
|
}
|
|
606
606
|
|
|
607
|
+
/**
|
|
608
|
+
* Creates a token that can be used in a DI Provider.
|
|
609
|
+
*
|
|
610
|
+
* Use an `InjectionToken` whenever the type you are injecting is not reified (does not have a
|
|
611
|
+
* runtime representation) such as when injecting an interface, callable type, array or
|
|
612
|
+
* parameterized type.
|
|
613
|
+
*
|
|
614
|
+
* `InjectionToken` is parameterized on `T` which is the type of object which will be returned by
|
|
615
|
+
* the `Injector`. This provides an additional level of type safety.
|
|
616
|
+
*
|
|
617
|
+
* <div class="alert is-helpful">
|
|
618
|
+
*
|
|
619
|
+
* **Important Note**: Ensure that you use the same instance of the `InjectionToken` in both the
|
|
620
|
+
* provider and the injection call. Creating a new instance of `InjectionToken` in different places,
|
|
621
|
+
* even with the same description, will be treated as different tokens by Angular's DI system,
|
|
622
|
+
* leading to a `NullInjectorError`.
|
|
623
|
+
*
|
|
624
|
+
* </div>
|
|
625
|
+
*
|
|
626
|
+
* <code-example format="typescript" language="typescript" path="injection-token/src/main.ts"
|
|
627
|
+
* region="InjectionToken"></code-example>
|
|
628
|
+
*
|
|
629
|
+
* When creating an `InjectionToken`, you can optionally specify a factory function which returns
|
|
630
|
+
* (possibly by creating) a default value of the parameterized type `T`. This sets up the
|
|
631
|
+
* `InjectionToken` using this factory as a provider as if it was defined explicitly in the
|
|
632
|
+
* application's root injector. If the factory function, which takes zero arguments, needs to inject
|
|
633
|
+
* dependencies, it can do so using the [`inject`](api/core/inject) function.
|
|
634
|
+
* As you can see in the Tree-shakable InjectionToken example below.
|
|
635
|
+
*
|
|
636
|
+
* Additionally, if a `factory` is specified you can also specify the `providedIn` option, which
|
|
637
|
+
* overrides the above behavior and marks the token as belonging to a particular `@NgModule` (note:
|
|
638
|
+
* this option is now deprecated). As mentioned above, `'root'` is the default value for
|
|
639
|
+
* `providedIn`.
|
|
640
|
+
*
|
|
641
|
+
* The `providedIn: NgModule` and `providedIn: 'any'` options are deprecated.
|
|
642
|
+
*
|
|
643
|
+
* @usageNotes
|
|
644
|
+
* ### Basic Examples
|
|
645
|
+
*
|
|
646
|
+
* ### Plain InjectionToken
|
|
647
|
+
*
|
|
648
|
+
* {@example core/di/ts/injector_spec.ts region='InjectionToken'}
|
|
649
|
+
*
|
|
650
|
+
* ### Tree-shakable InjectionToken
|
|
651
|
+
*
|
|
652
|
+
* {@example core/di/ts/injector_spec.ts region='ShakableInjectionToken'}
|
|
653
|
+
*
|
|
654
|
+
* @publicApi
|
|
655
|
+
*/
|
|
656
|
+
class InjectionToken {
|
|
657
|
+
/**
|
|
658
|
+
* @param _desc Description for the token,
|
|
659
|
+
* used only for debugging purposes,
|
|
660
|
+
* it should but does not need to be unique
|
|
661
|
+
* @param options Options for the token's usage, as described above
|
|
662
|
+
*/
|
|
663
|
+
constructor(_desc, options) {
|
|
664
|
+
this._desc = _desc;
|
|
665
|
+
/** @internal */
|
|
666
|
+
this.ngMetadataName = 'InjectionToken';
|
|
667
|
+
this.ɵprov = undefined;
|
|
668
|
+
if (typeof options == 'number') {
|
|
669
|
+
(typeof ngDevMode === 'undefined' || ngDevMode) &&
|
|
670
|
+
assertLessThan(options, 0, 'Only negative numbers are supported here');
|
|
671
|
+
// This is a special hack to assign __NG_ELEMENT_ID__ to this instance.
|
|
672
|
+
// See `InjectorMarkers`
|
|
673
|
+
this.__NG_ELEMENT_ID__ = options;
|
|
674
|
+
}
|
|
675
|
+
else if (options !== undefined) {
|
|
676
|
+
this.ɵprov = ɵɵdefineInjectable({
|
|
677
|
+
token: this,
|
|
678
|
+
providedIn: options.providedIn || 'root',
|
|
679
|
+
factory: options.factory,
|
|
680
|
+
});
|
|
681
|
+
}
|
|
682
|
+
}
|
|
683
|
+
/**
|
|
684
|
+
* @internal
|
|
685
|
+
*/
|
|
686
|
+
get multi() {
|
|
687
|
+
return this;
|
|
688
|
+
}
|
|
689
|
+
toString() {
|
|
690
|
+
return `InjectionToken ${this._desc}`;
|
|
691
|
+
}
|
|
692
|
+
}
|
|
693
|
+
|
|
607
694
|
let _injectorProfilerContext;
|
|
608
695
|
function getInjectorProfilerContext() {
|
|
609
696
|
!ngDevMode && throwError('getInjectorProfilerContext should never be called in production mode');
|
|
@@ -645,18 +732,35 @@ function injectorProfiler(event) {
|
|
|
645
732
|
* Emits an InjectorProfilerEventType.ProviderConfigured to the injector profiler. The data in the
|
|
646
733
|
* emitted event includes the raw provider, as well as the token that provider is providing.
|
|
647
734
|
*
|
|
648
|
-
* @param
|
|
735
|
+
* @param eventProvider A provider object
|
|
649
736
|
*/
|
|
650
|
-
function emitProviderConfiguredEvent(
|
|
737
|
+
function emitProviderConfiguredEvent(eventProvider, isViewProvider = false) {
|
|
651
738
|
!ngDevMode && throwError('Injector profiler should never be called in production mode');
|
|
739
|
+
let token;
|
|
740
|
+
// if the provider is a TypeProvider (typeof provider is function) then the token is the
|
|
741
|
+
// provider itself
|
|
742
|
+
if (typeof eventProvider === 'function') {
|
|
743
|
+
token = eventProvider;
|
|
744
|
+
}
|
|
745
|
+
// if the provider is an injection token, then the token is the injection token.
|
|
746
|
+
else if (eventProvider instanceof InjectionToken) {
|
|
747
|
+
token = eventProvider;
|
|
748
|
+
}
|
|
749
|
+
// in all other cases we can access the token via the `provide` property of the provider
|
|
750
|
+
else {
|
|
751
|
+
token = resolveForwardRef(eventProvider.provide);
|
|
752
|
+
}
|
|
753
|
+
let provider = eventProvider;
|
|
754
|
+
// Injection tokens may define their own default provider which gets attached to the token itself
|
|
755
|
+
// as `ɵprov`. In this case, we want to emit the provider that is attached to the token, not the
|
|
756
|
+
// token itself.
|
|
757
|
+
if (eventProvider instanceof InjectionToken) {
|
|
758
|
+
provider = eventProvider.ɵprov || eventProvider;
|
|
759
|
+
}
|
|
652
760
|
injectorProfiler({
|
|
653
761
|
type: 2 /* InjectorProfilerEventType.ProviderConfigured */,
|
|
654
762
|
context: getInjectorProfilerContext(),
|
|
655
|
-
providerRecord: {
|
|
656
|
-
token: typeof provider === 'function' ? provider : resolveForwardRef(provider.provide),
|
|
657
|
-
provider,
|
|
658
|
-
isViewProvider
|
|
659
|
-
}
|
|
763
|
+
providerRecord: { token, provider, isViewProvider }
|
|
660
764
|
});
|
|
661
765
|
}
|
|
662
766
|
/**
|
|
@@ -8757,93 +8861,6 @@ function getSanitizer() {
|
|
|
8757
8861
|
return lView && lView[ENVIRONMENT].sanitizer;
|
|
8758
8862
|
}
|
|
8759
8863
|
|
|
8760
|
-
/**
|
|
8761
|
-
* Creates a token that can be used in a DI Provider.
|
|
8762
|
-
*
|
|
8763
|
-
* Use an `InjectionToken` whenever the type you are injecting is not reified (does not have a
|
|
8764
|
-
* runtime representation) such as when injecting an interface, callable type, array or
|
|
8765
|
-
* parameterized type.
|
|
8766
|
-
*
|
|
8767
|
-
* `InjectionToken` is parameterized on `T` which is the type of object which will be returned by
|
|
8768
|
-
* the `Injector`. This provides an additional level of type safety.
|
|
8769
|
-
*
|
|
8770
|
-
* <div class="alert is-helpful">
|
|
8771
|
-
*
|
|
8772
|
-
* **Important Note**: Ensure that you use the same instance of the `InjectionToken` in both the
|
|
8773
|
-
* provider and the injection call. Creating a new instance of `InjectionToken` in different places,
|
|
8774
|
-
* even with the same description, will be treated as different tokens by Angular's DI system,
|
|
8775
|
-
* leading to a `NullInjectorError`.
|
|
8776
|
-
*
|
|
8777
|
-
* </div>
|
|
8778
|
-
*
|
|
8779
|
-
* <code-example format="typescript" language="typescript" path="injection-token/src/main.ts"
|
|
8780
|
-
* region="InjectionToken"></code-example>
|
|
8781
|
-
*
|
|
8782
|
-
* When creating an `InjectionToken`, you can optionally specify a factory function which returns
|
|
8783
|
-
* (possibly by creating) a default value of the parameterized type `T`. This sets up the
|
|
8784
|
-
* `InjectionToken` using this factory as a provider as if it was defined explicitly in the
|
|
8785
|
-
* application's root injector. If the factory function, which takes zero arguments, needs to inject
|
|
8786
|
-
* dependencies, it can do so using the [`inject`](api/core/inject) function.
|
|
8787
|
-
* As you can see in the Tree-shakable InjectionToken example below.
|
|
8788
|
-
*
|
|
8789
|
-
* Additionally, if a `factory` is specified you can also specify the `providedIn` option, which
|
|
8790
|
-
* overrides the above behavior and marks the token as belonging to a particular `@NgModule` (note:
|
|
8791
|
-
* this option is now deprecated). As mentioned above, `'root'` is the default value for
|
|
8792
|
-
* `providedIn`.
|
|
8793
|
-
*
|
|
8794
|
-
* The `providedIn: NgModule` and `providedIn: 'any'` options are deprecated.
|
|
8795
|
-
*
|
|
8796
|
-
* @usageNotes
|
|
8797
|
-
* ### Basic Examples
|
|
8798
|
-
*
|
|
8799
|
-
* ### Plain InjectionToken
|
|
8800
|
-
*
|
|
8801
|
-
* {@example core/di/ts/injector_spec.ts region='InjectionToken'}
|
|
8802
|
-
*
|
|
8803
|
-
* ### Tree-shakable InjectionToken
|
|
8804
|
-
*
|
|
8805
|
-
* {@example core/di/ts/injector_spec.ts region='ShakableInjectionToken'}
|
|
8806
|
-
*
|
|
8807
|
-
* @publicApi
|
|
8808
|
-
*/
|
|
8809
|
-
class InjectionToken {
|
|
8810
|
-
/**
|
|
8811
|
-
* @param _desc Description for the token,
|
|
8812
|
-
* used only for debugging purposes,
|
|
8813
|
-
* it should but does not need to be unique
|
|
8814
|
-
* @param options Options for the token's usage, as described above
|
|
8815
|
-
*/
|
|
8816
|
-
constructor(_desc, options) {
|
|
8817
|
-
this._desc = _desc;
|
|
8818
|
-
/** @internal */
|
|
8819
|
-
this.ngMetadataName = 'InjectionToken';
|
|
8820
|
-
this.ɵprov = undefined;
|
|
8821
|
-
if (typeof options == 'number') {
|
|
8822
|
-
(typeof ngDevMode === 'undefined' || ngDevMode) &&
|
|
8823
|
-
assertLessThan(options, 0, 'Only negative numbers are supported here');
|
|
8824
|
-
// This is a special hack to assign __NG_ELEMENT_ID__ to this instance.
|
|
8825
|
-
// See `InjectorMarkers`
|
|
8826
|
-
this.__NG_ELEMENT_ID__ = options;
|
|
8827
|
-
}
|
|
8828
|
-
else if (options !== undefined) {
|
|
8829
|
-
this.ɵprov = ɵɵdefineInjectable({
|
|
8830
|
-
token: this,
|
|
8831
|
-
providedIn: options.providedIn || 'root',
|
|
8832
|
-
factory: options.factory,
|
|
8833
|
-
});
|
|
8834
|
-
}
|
|
8835
|
-
}
|
|
8836
|
-
/**
|
|
8837
|
-
* @internal
|
|
8838
|
-
*/
|
|
8839
|
-
get multi() {
|
|
8840
|
-
return this;
|
|
8841
|
-
}
|
|
8842
|
-
toString() {
|
|
8843
|
-
return `InjectionToken ${this._desc}`;
|
|
8844
|
-
}
|
|
8845
|
-
}
|
|
8846
|
-
|
|
8847
8864
|
/**
|
|
8848
8865
|
* A multi-provider token for initialization functions that will run upon construction of an
|
|
8849
8866
|
* environment injector.
|
|
@@ -9289,6 +9306,11 @@ class R3Injector extends EnvironmentInjector {
|
|
|
9289
9306
|
if (def && this.injectableDefInScope(def)) {
|
|
9290
9307
|
// Found an injectable def and it's scoped to this injector. Pretend as if it was here
|
|
9291
9308
|
// all along.
|
|
9309
|
+
if (ngDevMode) {
|
|
9310
|
+
runInInjectorProfilerContext(this, token, () => {
|
|
9311
|
+
emitProviderConfiguredEvent(token);
|
|
9312
|
+
});
|
|
9313
|
+
}
|
|
9292
9314
|
record = makeRecord(injectableDefOrInjectorDefFactory(token), NOT_YET);
|
|
9293
9315
|
}
|
|
9294
9316
|
else {
|
|
@@ -10276,7 +10298,7 @@ class Version {
|
|
|
10276
10298
|
/**
|
|
10277
10299
|
* @publicApi
|
|
10278
10300
|
*/
|
|
10279
|
-
const VERSION = new Version('16.2.
|
|
10301
|
+
const VERSION = new Version('16.2.11');
|
|
10280
10302
|
|
|
10281
10303
|
// This default value is when checking the hierarchy for a token.
|
|
10282
10304
|
//
|
|
@@ -11665,11 +11687,6 @@ function getExpressionChangedErrorDetails(lView, bindingIndex, oldValue, newValu
|
|
|
11665
11687
|
}
|
|
11666
11688
|
|
|
11667
11689
|
let currentConsumer = null;
|
|
11668
|
-
function setLViewForConsumer(node, lView) {
|
|
11669
|
-
(typeof ngDevMode === 'undefined' || ngDevMode) &&
|
|
11670
|
-
assertEqual(node.lView, null, 'Consumer already associated with a view.');
|
|
11671
|
-
node.lView = lView;
|
|
11672
|
-
}
|
|
11673
11690
|
/**
|
|
11674
11691
|
* Create a new template consumer pointing at the specified LView.
|
|
11675
11692
|
* Sometimes, a previously created consumer may be reused, in order to save on allocations. In that
|
|
@@ -26545,7 +26562,7 @@ const ITS_JUST_ANGULAR = true;
|
|
|
26545
26562
|
* ```
|
|
26546
26563
|
*
|
|
26547
26564
|
* ### Example with standalone application
|
|
26548
|
-
*
|
|
26565
|
+
* ```
|
|
26549
26566
|
* function initializeAppFactory(httpClient: HttpClient): () => Observable<any> {
|
|
26550
26567
|
* return () => httpClient.get("https://someUrl.com/api/user")
|
|
26551
26568
|
* .pipe(
|
|
@@ -26564,6 +26581,7 @@ const ITS_JUST_ANGULAR = true;
|
|
|
26564
26581
|
* },
|
|
26565
26582
|
* ],
|
|
26566
26583
|
* });
|
|
26584
|
+
* ```
|
|
26567
26585
|
*
|
|
26568
26586
|
* @publicApi
|
|
26569
26587
|
*/
|
|
@@ -27280,6 +27298,11 @@ function getProviderImportsContainer(injector) {
|
|
|
27280
27298
|
if (defTypeRef === null) {
|
|
27281
27299
|
return null;
|
|
27282
27300
|
}
|
|
27301
|
+
// In standalone applications, the root environment injector created by bootstrapApplication
|
|
27302
|
+
// may have no associated "instance".
|
|
27303
|
+
if (defTypeRef.instance === null) {
|
|
27304
|
+
return null;
|
|
27305
|
+
}
|
|
27283
27306
|
return defTypeRef.instance.constructor;
|
|
27284
27307
|
}
|
|
27285
27308
|
/**
|
|
@@ -27461,12 +27484,25 @@ function walkProviderTreeToDiscoverImportPaths(providerToPath, visitedContainers
|
|
|
27461
27484
|
* @returns an array of objects representing the providers of the given injector
|
|
27462
27485
|
*/
|
|
27463
27486
|
function getEnvironmentInjectorProviders(injector) {
|
|
27487
|
+
const providerRecords = getFrameworkDIDebugData().resolverToProviders.get(injector) ?? [];
|
|
27488
|
+
// platform injector has no provider imports container so can we skip trying to
|
|
27489
|
+
// find import paths
|
|
27490
|
+
if (isPlatformInjector(injector)) {
|
|
27491
|
+
return providerRecords;
|
|
27492
|
+
}
|
|
27464
27493
|
const providerImportsContainer = getProviderImportsContainer(injector);
|
|
27465
27494
|
if (providerImportsContainer === null) {
|
|
27495
|
+
// There is a special case where the bootstrapped component does not
|
|
27496
|
+
// import any NgModules. In this case the environment injector connected to
|
|
27497
|
+
// that component is the root injector, which does not have a provider imports
|
|
27498
|
+
// container (and thus no concept of module import paths). Therefore we simply
|
|
27499
|
+
// return the provider records as is.
|
|
27500
|
+
if (isRootInjector(injector)) {
|
|
27501
|
+
return providerRecords;
|
|
27502
|
+
}
|
|
27466
27503
|
throwError('Could not determine where injector providers were configured.');
|
|
27467
27504
|
}
|
|
27468
27505
|
const providerToPath = getProviderImportPaths(providerImportsContainer);
|
|
27469
|
-
const providerRecords = getFrameworkDIDebugData().resolverToProviders.get(injector) ?? [];
|
|
27470
27506
|
return providerRecords.map(providerRecord => {
|
|
27471
27507
|
let importPath = providerToPath.get(providerRecord.provider) ?? [providerImportsContainer];
|
|
27472
27508
|
const def = getComponentDef(providerImportsContainer);
|
|
@@ -27479,6 +27515,12 @@ function getEnvironmentInjectorProviders(injector) {
|
|
|
27479
27515
|
return { ...providerRecord, importPath };
|
|
27480
27516
|
});
|
|
27481
27517
|
}
|
|
27518
|
+
function isPlatformInjector(injector) {
|
|
27519
|
+
return injector instanceof R3Injector && injector.scopes.has('platform');
|
|
27520
|
+
}
|
|
27521
|
+
function isRootInjector(injector) {
|
|
27522
|
+
return injector instanceof R3Injector && injector.scopes.has('root');
|
|
27523
|
+
}
|
|
27482
27524
|
/**
|
|
27483
27525
|
* Gets the providers configured on an injector.
|
|
27484
27526
|
*
|
|
@@ -28105,10 +28147,10 @@ function createOrReusePlatformInjector(providers = []) {
|
|
|
28105
28147
|
// is already bootstrapped and no additional actions are required.
|
|
28106
28148
|
if (_platformInjector)
|
|
28107
28149
|
return _platformInjector;
|
|
28150
|
+
publishDefaultGlobalUtils();
|
|
28108
28151
|
// Otherwise, setup a new platform injector and run platform initializers.
|
|
28109
28152
|
const injector = createPlatformInjector(providers);
|
|
28110
28153
|
_platformInjector = injector;
|
|
28111
|
-
publishDefaultGlobalUtils();
|
|
28112
28154
|
publishSignalConfiguration();
|
|
28113
28155
|
runPlatformInitializers(injector);
|
|
28114
28156
|
return injector;
|
|
@@ -28900,7 +28942,7 @@ function internalProvideZoneChangeDetection(ngZoneFactory) {
|
|
|
28900
28942
|
* `BootstrapOptions` instead.
|
|
28901
28943
|
*
|
|
28902
28944
|
* @usageNotes
|
|
28903
|
-
* ```typescript
|
|
28945
|
+
* ```typescript
|
|
28904
28946
|
* bootstrapApplication(MyApp, {providers: [
|
|
28905
28947
|
* provideZoneChangeDetection({eventCoalescing: true}),
|
|
28906
28948
|
* ]});
|