@angular/core 16.2.0-next.4 → 16.2.0
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_tokens.mjs +2 -1
- package/esm2022/src/core.mjs +2 -1
- package/esm2022/src/core_private_export.mjs +3 -1
- package/esm2022/src/core_render3_private_export.mjs +3 -2
- package/esm2022/src/di/contextual.mjs +7 -1
- package/esm2022/src/di/create_injector.mjs +1 -3
- package/esm2022/src/di/injector.mjs +1 -1
- package/esm2022/src/di/injector_compatibility.mjs +5 -2
- package/esm2022/src/di/provider_collection.mjs +17 -16
- package/esm2022/src/di/r3_injector.mjs +37 -2
- package/esm2022/src/errors.mjs +1 -1
- package/esm2022/src/hydration/annotate.mjs +60 -15
- package/esm2022/src/hydration/api.mjs +40 -17
- package/esm2022/src/hydration/cleanup.mjs +15 -6
- package/esm2022/src/hydration/utils.mjs +58 -20
- package/esm2022/src/linker/template_ref.mjs +4 -19
- package/esm2022/src/linker/view_container_ref.mjs +5 -13
- package/esm2022/src/metadata/directives.mjs +1 -1
- package/esm2022/src/render3/after_render_hooks.mjs +210 -0
- package/esm2022/src/render3/component_ref.mjs +11 -11
- package/esm2022/src/render3/debug/framework_injector_profiler.mjs +212 -0
- package/esm2022/src/render3/debug/injector_profiler.mjs +103 -0
- package/esm2022/src/render3/di.mjs +32 -2
- package/esm2022/src/render3/di_setup.mjs +11 -4
- package/esm2022/src/render3/hooks.mjs +3 -3
- package/esm2022/src/render3/index.mjs +2 -2
- package/esm2022/src/render3/instructions/all.mjs +2 -1
- package/esm2022/src/render3/instructions/change_detection.mjs +16 -9
- package/esm2022/src/render3/instructions/defer.mjs +19 -0
- package/esm2022/src/render3/instructions/di.mjs +5 -2
- package/esm2022/src/render3/instructions/shared.mjs +1 -1
- package/esm2022/src/render3/interfaces/view.mjs +1 -1
- package/esm2022/src/render3/jit/environment.mjs +2 -1
- package/esm2022/src/render3/node_manipulation.mjs +5 -5
- package/esm2022/src/render3/pipe.mjs +12 -3
- package/esm2022/src/render3/util/global_utils.mjs +7 -1
- package/esm2022/src/render3/util/injector_discovery_utils.mjs +460 -0
- package/esm2022/src/render3/util/misc_utils.mjs +12 -1
- package/esm2022/src/render3/view_manipulation.mjs +65 -0
- package/esm2022/src/render3/view_ref.mjs +3 -3
- package/esm2022/src/util/coercion.mjs +11 -1
- package/esm2022/src/version.mjs +1 -1
- package/esm2022/testing/src/logger.mjs +3 -3
- package/fesm2022/core.mjs +1854 -650
- package/fesm2022/core.mjs.map +1 -1
- package/fesm2022/rxjs-interop.mjs +1 -1
- package/fesm2022/rxjs-interop.mjs.map +1 -1
- package/fesm2022/testing.mjs +4023 -3540
- package/fesm2022/testing.mjs.map +1 -1
- package/index.d.ts +214 -10
- 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 +3664 -2727
- package/schematics/ng-generate/standalone-migration/bundle.js.map +4 -4
- package/testing/index.d.ts +1 -1
|
@@ -0,0 +1,212 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright Google LLC All Rights Reserved.
|
|
4
|
+
*
|
|
5
|
+
* Use of this source code is governed by an MIT-style license that can be
|
|
6
|
+
* found in the LICENSE file at https://angular.io/license
|
|
7
|
+
*/
|
|
8
|
+
import { EnvironmentInjector } from '../../di/r3_injector';
|
|
9
|
+
import { throwError } from '../../util/assert';
|
|
10
|
+
import { getComponentDef } from '../definition';
|
|
11
|
+
import { getNodeInjectorLView, NodeInjector } from '../di';
|
|
12
|
+
import { setInjectorProfiler } from './injector_profiler';
|
|
13
|
+
/**
|
|
14
|
+
* These are the data structures that our framework injector profiler will fill with data in order
|
|
15
|
+
* to support DI debugging APIs.
|
|
16
|
+
*
|
|
17
|
+
* resolverToTokenToDependencies: Maps an injector to a Map of tokens to an Array of
|
|
18
|
+
* dependencies. Injector -> Token -> Dependencies This is used to support the
|
|
19
|
+
* getDependenciesFromInjectable API, which takes in an injector and a token and returns it's
|
|
20
|
+
* dependencies.
|
|
21
|
+
*
|
|
22
|
+
* resolverToProviders: Maps a DI resolver (an Injector or an LView) to the providers configured
|
|
23
|
+
* within it This is used to support the getInjectorProviders API, which takes in an injector and
|
|
24
|
+
* returns the providers that it was configured with.
|
|
25
|
+
*
|
|
26
|
+
* standaloneInjectorToComponent: Maps the injector of a standalone component to the standalone
|
|
27
|
+
* component that it is associated with. Used in the getInjectorProviders API, specificially in the
|
|
28
|
+
* discovery of import paths for each provider. This is necessary because the imports array of a
|
|
29
|
+
* standalone component is processed and configured in its standalone injector, but exists within
|
|
30
|
+
* the component's definition. Because getInjectorProviders takes in an injector, if that injector
|
|
31
|
+
* is the injector of a standalone component, we need to be able to discover the place where the
|
|
32
|
+
* imports array is located (the component) in order to flatten the imports array within it to
|
|
33
|
+
* discover all of it's providers.
|
|
34
|
+
*
|
|
35
|
+
*
|
|
36
|
+
* All of these data structures are instantiated with WeakMaps. This will ensure that the presence
|
|
37
|
+
* of any object in the keys of these maps does not prevent the garbage collector from collecting
|
|
38
|
+
* those objects. Because of this property of WeakMaps, these data structures will never be the
|
|
39
|
+
* source of a memory leak.
|
|
40
|
+
*
|
|
41
|
+
* An example of this advantage: When components are destroyed, we don't need to do
|
|
42
|
+
* any additional work to remove that component from our mappings.
|
|
43
|
+
*
|
|
44
|
+
*/
|
|
45
|
+
class DIDebugData {
|
|
46
|
+
constructor() {
|
|
47
|
+
this.resolverToTokenToDependencies = new WeakMap();
|
|
48
|
+
this.resolverToProviders = new WeakMap();
|
|
49
|
+
this.standaloneInjectorToComponent = new WeakMap();
|
|
50
|
+
}
|
|
51
|
+
reset() {
|
|
52
|
+
this.resolverToTokenToDependencies =
|
|
53
|
+
new WeakMap();
|
|
54
|
+
this.resolverToProviders = new WeakMap();
|
|
55
|
+
this.standaloneInjectorToComponent = new WeakMap();
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
let frameworkDIDebugData = new DIDebugData();
|
|
59
|
+
export function getFrameworkDIDebugData() {
|
|
60
|
+
return frameworkDIDebugData;
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Initalize default handling of injector events. This handling parses events
|
|
64
|
+
* as they are emitted and constructs the data structures necessary to support
|
|
65
|
+
* some of debug APIs.
|
|
66
|
+
*
|
|
67
|
+
* See handleInjectEvent, handleCreateEvent and handleProviderConfiguredEvent
|
|
68
|
+
* for descriptions of each handler
|
|
69
|
+
*
|
|
70
|
+
* Supported APIs:
|
|
71
|
+
* - getDependenciesFromInjectable
|
|
72
|
+
* - getInjectorProviders
|
|
73
|
+
*/
|
|
74
|
+
export function setupFrameworkInjectorProfiler() {
|
|
75
|
+
frameworkDIDebugData.reset();
|
|
76
|
+
setInjectorProfiler((injectorProfilerEvent) => handleInjectorProfilerEvent(injectorProfilerEvent));
|
|
77
|
+
}
|
|
78
|
+
function handleInjectorProfilerEvent(injectorProfilerEvent) {
|
|
79
|
+
const { context, type } = injectorProfilerEvent;
|
|
80
|
+
if (type === 0 /* InjectorProfilerEventType.Inject */) {
|
|
81
|
+
handleInjectEvent(context, injectorProfilerEvent.service);
|
|
82
|
+
}
|
|
83
|
+
else if (type === 1 /* InjectorProfilerEventType.InstanceCreatedByInjector */) {
|
|
84
|
+
handleInstanceCreatedByInjectorEvent(context, injectorProfilerEvent.instance);
|
|
85
|
+
}
|
|
86
|
+
else if (type === 2 /* InjectorProfilerEventType.ProviderConfigured */) {
|
|
87
|
+
handleProviderConfiguredEvent(context, injectorProfilerEvent.providerRecord);
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
/**
|
|
91
|
+
*
|
|
92
|
+
* Stores the injected service in frameworkDIDebugData.resolverToTokenToDependencies
|
|
93
|
+
* based on it's injector and token.
|
|
94
|
+
*
|
|
95
|
+
* @param context InjectorProfilerContext the injection context that this event occurred in.
|
|
96
|
+
* @param data InjectedService the service associated with this inject event.
|
|
97
|
+
*
|
|
98
|
+
*/
|
|
99
|
+
function handleInjectEvent(context, data) {
|
|
100
|
+
const diResolver = getDIResolver(context.injector);
|
|
101
|
+
if (diResolver === null) {
|
|
102
|
+
throwError('An Inject event must be run within an injection context.');
|
|
103
|
+
}
|
|
104
|
+
const diResolverToInstantiatedToken = frameworkDIDebugData.resolverToTokenToDependencies;
|
|
105
|
+
if (!diResolverToInstantiatedToken.has(diResolver)) {
|
|
106
|
+
diResolverToInstantiatedToken.set(diResolver, new WeakMap());
|
|
107
|
+
}
|
|
108
|
+
// if token is a primitive type, ignore this event. We do this because we cannot keep track of
|
|
109
|
+
// non-primitive tokens in WeakMaps since they are not garbage collectable.
|
|
110
|
+
if (!canBeHeldWeakly(context.token)) {
|
|
111
|
+
return;
|
|
112
|
+
}
|
|
113
|
+
const instantiatedTokenToDependencies = diResolverToInstantiatedToken.get(diResolver);
|
|
114
|
+
if (!instantiatedTokenToDependencies.has(context.token)) {
|
|
115
|
+
instantiatedTokenToDependencies.set(context.token, []);
|
|
116
|
+
}
|
|
117
|
+
const { token, value, flags } = data;
|
|
118
|
+
instantiatedTokenToDependencies.get(context.token).push({ token, value, flags });
|
|
119
|
+
}
|
|
120
|
+
/**
|
|
121
|
+
*
|
|
122
|
+
* If the created instance is an instance of a standalone component, maps the injector to that
|
|
123
|
+
* standalone component in frameworkDIDebugData.standaloneInjectorToComponent
|
|
124
|
+
*
|
|
125
|
+
* @param context InjectorProfilerContext the injection context that this event occurred in.
|
|
126
|
+
* @param data InjectorCreatedInstance an object containing the instance that was just created
|
|
127
|
+
*
|
|
128
|
+
*/
|
|
129
|
+
function handleInstanceCreatedByInjectorEvent(context, data) {
|
|
130
|
+
const { value } = data;
|
|
131
|
+
if (getDIResolver(context.injector) === null) {
|
|
132
|
+
throwError('An InjectorCreatedInstance event must be run within an injection context.');
|
|
133
|
+
}
|
|
134
|
+
// if our value is an instance of a standalone component, map the injector of that standalone
|
|
135
|
+
// component to the component class. Otherwise, this event is a noop.
|
|
136
|
+
let standaloneComponent = undefined;
|
|
137
|
+
if (typeof value === 'object') {
|
|
138
|
+
standaloneComponent = value?.constructor;
|
|
139
|
+
}
|
|
140
|
+
if (standaloneComponent === undefined || !isStandaloneComponent(standaloneComponent)) {
|
|
141
|
+
return;
|
|
142
|
+
}
|
|
143
|
+
const environmentInjector = context.injector.get(EnvironmentInjector, null, { optional: true });
|
|
144
|
+
// Standalone components should have an environment injector. If one cannot be
|
|
145
|
+
// found we may be in a test case for low level functionality that did not explictly
|
|
146
|
+
// setup this injector. In those cases, we simply ignore this event.
|
|
147
|
+
if (environmentInjector === null) {
|
|
148
|
+
return;
|
|
149
|
+
}
|
|
150
|
+
const { standaloneInjectorToComponent } = frameworkDIDebugData;
|
|
151
|
+
// If our injector has already been mapped, as is the case
|
|
152
|
+
// when a standalone component imports another standalone component,
|
|
153
|
+
// we consider the original component (the component doing the importing)
|
|
154
|
+
// as the component connected to our injector.
|
|
155
|
+
if (standaloneInjectorToComponent.has(environmentInjector)) {
|
|
156
|
+
return;
|
|
157
|
+
}
|
|
158
|
+
// If our injector hasn't been mapped, then we map it to the standalone component
|
|
159
|
+
standaloneInjectorToComponent.set(environmentInjector, standaloneComponent);
|
|
160
|
+
}
|
|
161
|
+
function isStandaloneComponent(value) {
|
|
162
|
+
const def = getComponentDef(value);
|
|
163
|
+
return !!def?.standalone;
|
|
164
|
+
}
|
|
165
|
+
/**
|
|
166
|
+
*
|
|
167
|
+
* Stores the emitted ProviderRecords from the InjectorProfilerEventType.ProviderConfigured
|
|
168
|
+
* event in frameworkDIDebugData.resolverToProviders
|
|
169
|
+
*
|
|
170
|
+
* @param context InjectorProfilerContext the injection context that this event occurred in.
|
|
171
|
+
* @param data ProviderRecord an object containing the instance that was just created
|
|
172
|
+
*
|
|
173
|
+
*/
|
|
174
|
+
function handleProviderConfiguredEvent(context, data) {
|
|
175
|
+
const { resolverToProviders } = frameworkDIDebugData;
|
|
176
|
+
const diResolver = getDIResolver(context?.injector);
|
|
177
|
+
if (diResolver === null) {
|
|
178
|
+
throwError('A ProviderConfigured event must be run within an injection context.');
|
|
179
|
+
}
|
|
180
|
+
if (!resolverToProviders.has(diResolver)) {
|
|
181
|
+
resolverToProviders.set(diResolver, []);
|
|
182
|
+
}
|
|
183
|
+
resolverToProviders.get(diResolver).push(data);
|
|
184
|
+
}
|
|
185
|
+
function getDIResolver(injector) {
|
|
186
|
+
let diResolver = null;
|
|
187
|
+
if (injector === undefined) {
|
|
188
|
+
return diResolver;
|
|
189
|
+
}
|
|
190
|
+
// We use the LView as the diResolver for NodeInjectors because they
|
|
191
|
+
// do not persist anywhere in the framework. They are simply wrappers around an LView and a TNode
|
|
192
|
+
// that do persist. Because of this, we rely on the LView of the NodeInjector in order to use
|
|
193
|
+
// as a concrete key to represent this injector. If we get the same LView back later, we know
|
|
194
|
+
// we're looking at the same injector.
|
|
195
|
+
if (injector instanceof NodeInjector) {
|
|
196
|
+
diResolver = getNodeInjectorLView(injector);
|
|
197
|
+
}
|
|
198
|
+
// Other injectors can be used a keys for a map because their instances
|
|
199
|
+
// persist
|
|
200
|
+
else {
|
|
201
|
+
diResolver = injector;
|
|
202
|
+
}
|
|
203
|
+
return diResolver;
|
|
204
|
+
}
|
|
205
|
+
// inspired by
|
|
206
|
+
// https://tc39.es/ecma262/multipage/executable-code-and-execution-contexts.html#sec-canbeheldweakly
|
|
207
|
+
function canBeHeldWeakly(value) {
|
|
208
|
+
// we check for value !== null here because typeof null === 'object
|
|
209
|
+
return value !== null &&
|
|
210
|
+
(typeof value === 'object' || typeof value === 'function' || typeof value === 'symbol');
|
|
211
|
+
}
|
|
212
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"framework_injector_profiler.js","sourceRoot":"","sources":["../../../../../../../../packages/core/src/render3/debug/framework_injector_profiler.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 {Injector} from '../../di/injector';\nimport {EnvironmentInjector} from '../../di/r3_injector';\nimport {Type} from '../../interface/type';\nimport {throwError} from '../../util/assert';\nimport {getComponentDef} from '../definition';\nimport {getNodeInjectorLView, NodeInjector} from '../di';\nimport {LView} from '../interfaces/view';\n\nimport {InjectedService, InjectorCreatedInstance, InjectorProfilerContext, InjectorProfilerEvent, InjectorProfilerEventType, ProviderRecord, setInjectorProfiler} from './injector_profiler';\n\n/**\n * These are the data structures that our framework injector profiler will fill with data in order\n * to support DI debugging APIs.\n *\n * resolverToTokenToDependencies: Maps an injector to a Map of tokens to an Array of\n * dependencies. Injector -> Token -> Dependencies This is used to support the\n * getDependenciesFromInjectable API, which takes in an injector and a token and returns it's\n * dependencies.\n *\n * resolverToProviders: Maps a DI resolver (an Injector or an LView) to the providers configured\n * within it This is used to support the getInjectorProviders API, which takes in an injector and\n * returns the providers that it was configured with.\n *\n * standaloneInjectorToComponent: Maps the injector of a standalone component to the standalone\n * component that it is associated with. Used in the getInjectorProviders API, specificially in the\n * discovery of import paths for each provider. This is necessary because the imports array of a\n * standalone component is processed and configured in its standalone injector, but exists within\n * the component's definition. Because getInjectorProviders takes in an injector, if that injector\n * is the injector of a standalone component, we need to be able to discover the place where the\n * imports array is located (the component) in order to flatten the imports array within it to\n * discover all of it's providers.\n *\n *\n * All of these data structures are instantiated with WeakMaps. This will ensure that the presence\n * of any object in the keys of these maps does not prevent the garbage collector from collecting\n * those objects. Because of this property of WeakMaps, these data structures will never be the\n * source of a memory leak.\n *\n * An example of this advantage: When components are destroyed, we don't need to do\n * any additional work to remove that component from our mappings.\n *\n */\nclass DIDebugData {\n  resolverToTokenToDependencies =\n      new WeakMap<Injector|LView, WeakMap<Type<unknown>, InjectedService[]>>();\n  resolverToProviders = new WeakMap<Injector|LView, ProviderRecord[]>();\n  standaloneInjectorToComponent = new WeakMap<Injector, Type<unknown>>();\n\n  reset() {\n    this.resolverToTokenToDependencies =\n        new WeakMap<Injector|LView, WeakMap<Type<unknown>, InjectedService[]>>();\n    this.resolverToProviders = new WeakMap<Injector|LView, ProviderRecord[]>();\n    this.standaloneInjectorToComponent = new WeakMap<Injector, Type<unknown>>();\n  }\n}\n\nlet frameworkDIDebugData = new DIDebugData();\n\nexport function getFrameworkDIDebugData(): DIDebugData {\n  return frameworkDIDebugData;\n}\n\n/**\n * Initalize default handling of injector events. This handling parses events\n * as they are emitted and constructs the data structures necessary to support\n * some of debug APIs.\n *\n * See handleInjectEvent, handleCreateEvent and handleProviderConfiguredEvent\n * for descriptions of each handler\n *\n * Supported APIs:\n *               - getDependenciesFromInjectable\n *               - getInjectorProviders\n */\nexport function setupFrameworkInjectorProfiler(): void {\n  frameworkDIDebugData.reset();\n  setInjectorProfiler(\n      (injectorProfilerEvent) => handleInjectorProfilerEvent(injectorProfilerEvent));\n}\n\nfunction handleInjectorProfilerEvent(injectorProfilerEvent: InjectorProfilerEvent): void {\n  const {context, type} = injectorProfilerEvent;\n\n  if (type === InjectorProfilerEventType.Inject) {\n    handleInjectEvent(context, injectorProfilerEvent.service);\n  } else if (type === InjectorProfilerEventType.InstanceCreatedByInjector) {\n    handleInstanceCreatedByInjectorEvent(context, injectorProfilerEvent.instance);\n  } else if (type === InjectorProfilerEventType.ProviderConfigured) {\n    handleProviderConfiguredEvent(context, injectorProfilerEvent.providerRecord);\n  }\n}\n\n/**\n *\n * Stores the injected service in frameworkDIDebugData.resolverToTokenToDependencies\n * based on it's injector and token.\n *\n * @param context InjectorProfilerContext the injection context that this event occurred in.\n * @param data InjectedService the service associated with this inject event.\n *\n */\nfunction handleInjectEvent(context: InjectorProfilerContext, data: InjectedService) {\n  const diResolver = getDIResolver(context.injector);\n  if (diResolver === null) {\n    throwError('An Inject event must be run within an injection context.');\n  }\n\n  const diResolverToInstantiatedToken = frameworkDIDebugData.resolverToTokenToDependencies;\n\n  if (!diResolverToInstantiatedToken.has(diResolver)) {\n    diResolverToInstantiatedToken.set(diResolver, new WeakMap<Type<unknown>, InjectedService[]>());\n  }\n\n  // if token is a primitive type, ignore this event. We do this because we cannot keep track of\n  // non-primitive tokens in WeakMaps since they are not garbage collectable.\n  if (!canBeHeldWeakly(context.token)) {\n    return;\n  }\n\n  const instantiatedTokenToDependencies = diResolverToInstantiatedToken.get(diResolver)!;\n  if (!instantiatedTokenToDependencies.has(context.token!)) {\n    instantiatedTokenToDependencies.set(context.token!, []);\n  }\n\n  const {token, value, flags} = data;\n  instantiatedTokenToDependencies.get(context.token!)!.push({token, value, flags});\n}\n\n/**\n *\n * If the created instance is an instance of a standalone component, maps the injector to that\n * standalone component in frameworkDIDebugData.standaloneInjectorToComponent\n *\n * @param context InjectorProfilerContext the injection context that this event occurred in.\n * @param data InjectorCreatedInstance an object containing the instance that was just created\n *\n */\nfunction handleInstanceCreatedByInjectorEvent(\n    context: InjectorProfilerContext, data: InjectorCreatedInstance): void {\n  const {value} = data;\n\n  if (getDIResolver(context.injector) === null) {\n    throwError('An InjectorCreatedInstance event must be run within an injection context.');\n  }\n\n  // if our value is an instance of a standalone component, map the injector of that standalone\n  // component to the component class. Otherwise, this event is a noop.\n  let standaloneComponent: Type<unknown>|undefined = undefined;\n  if (typeof value === 'object') {\n    standaloneComponent = value?.constructor as Type<unknown>;\n  }\n  if (standaloneComponent === undefined || !isStandaloneComponent(standaloneComponent)) {\n    return;\n  }\n\n  const environmentInjector: EnvironmentInjector|null =\n      context.injector.get(EnvironmentInjector, null, {optional: true});\n  // Standalone components should have an environment injector. If one cannot be\n  // found we may be in a test case for low level functionality that did not explictly\n  // setup this injector. In those cases, we simply ignore this event.\n  if (environmentInjector === null) {\n    return;\n  }\n\n  const {standaloneInjectorToComponent} = frameworkDIDebugData;\n\n  // If our injector has already been mapped, as is the case\n  // when a standalone component imports another standalone component,\n  // we consider the original component (the component doing the importing)\n  // as the component connected to our injector.\n  if (standaloneInjectorToComponent.has(environmentInjector)) {\n    return;\n  }\n  // If our injector hasn't been mapped, then we map it to the standalone component\n  standaloneInjectorToComponent.set(environmentInjector, standaloneComponent);\n}\n\nfunction isStandaloneComponent(value: Type<unknown>): boolean {\n  const def = getComponentDef(value);\n  return !!def?.standalone;\n}\n\n/**\n *\n * Stores the emitted ProviderRecords from the InjectorProfilerEventType.ProviderConfigured\n * event in frameworkDIDebugData.resolverToProviders\n *\n * @param context InjectorProfilerContext the injection context that this event occurred in.\n * @param data ProviderRecord an object containing the instance that was just created\n *\n */\nfunction handleProviderConfiguredEvent(\n    context: InjectorProfilerContext, data: ProviderRecord): void {\n  const {resolverToProviders} = frameworkDIDebugData;\n\n  const diResolver = getDIResolver(context?.injector);\n  if (diResolver === null) {\n    throwError('A ProviderConfigured event must be run within an injection context.');\n  }\n\n  if (!resolverToProviders.has(diResolver)) {\n    resolverToProviders.set(diResolver, []);\n  }\n\n  resolverToProviders.get(diResolver)!.push(data);\n}\n\nfunction getDIResolver(injector: Injector|undefined): Injector|LView|null {\n  let diResolver: Injector|LView|null = null;\n\n  if (injector === undefined) {\n    return diResolver;\n  }\n\n  // We use the LView as the diResolver for NodeInjectors because they\n  // do not persist anywhere in the framework. They are simply wrappers around an LView and a TNode\n  // that do persist. Because of this, we rely on the LView of the NodeInjector in order to use\n  // as a concrete key to represent this injector. If we get the same LView back later, we know\n  // we're looking at the same injector.\n  if (injector instanceof NodeInjector) {\n    diResolver = getNodeInjectorLView(injector);\n  }\n  // Other injectors can be used a keys for a map because their instances\n  // persist\n  else {\n    diResolver = injector;\n  }\n\n  return diResolver;\n}\n\n// inspired by\n// https://tc39.es/ecma262/multipage/executable-code-and-execution-contexts.html#sec-canbeheldweakly\nfunction canBeHeldWeakly(value: any): boolean {\n  // we check for value !== null here because typeof null === 'object\n  return value !== null &&\n      (typeof value === 'object' || typeof value === 'function' || typeof value === 'symbol');\n}\n"]}
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright Google LLC All Rights Reserved.
|
|
4
|
+
*
|
|
5
|
+
* Use of this source code is governed by an MIT-style license that can be
|
|
6
|
+
* found in the LICENSE file at https://angular.io/license
|
|
7
|
+
*/
|
|
8
|
+
import { resolveForwardRef } from '../../di/forward_ref';
|
|
9
|
+
import { throwError } from '../../util/assert';
|
|
10
|
+
let _injectorProfilerContext;
|
|
11
|
+
export function getInjectorProfilerContext() {
|
|
12
|
+
!ngDevMode && throwError('getInjectorProfilerContext should never be called in production mode');
|
|
13
|
+
return _injectorProfilerContext;
|
|
14
|
+
}
|
|
15
|
+
export function setInjectorProfilerContext(context) {
|
|
16
|
+
!ngDevMode && throwError('setInjectorProfilerContext should never be called in production mode');
|
|
17
|
+
const previous = _injectorProfilerContext;
|
|
18
|
+
_injectorProfilerContext = context;
|
|
19
|
+
return previous;
|
|
20
|
+
}
|
|
21
|
+
let injectorProfilerCallback = null;
|
|
22
|
+
/**
|
|
23
|
+
* Sets the callback function which will be invoked during certain DI events within the
|
|
24
|
+
* runtime (for example: injecting services, creating injectable instances, configuring providers)
|
|
25
|
+
*
|
|
26
|
+
* Warning: this function is *INTERNAL* and should not be relied upon in application's code.
|
|
27
|
+
* The contract of the function might be changed in any release and/or the function can be removed
|
|
28
|
+
* completely.
|
|
29
|
+
*
|
|
30
|
+
* @param profiler function provided by the caller or null value to disable profiling.
|
|
31
|
+
*/
|
|
32
|
+
export const setInjectorProfiler = (injectorProfiler) => {
|
|
33
|
+
!ngDevMode && throwError('setInjectorProfiler should never be called in production mode');
|
|
34
|
+
injectorProfilerCallback = injectorProfiler;
|
|
35
|
+
};
|
|
36
|
+
/**
|
|
37
|
+
* Injector profiler function which emits on DI events executed by the runtime.
|
|
38
|
+
*
|
|
39
|
+
* @param event InjectorProfilerEvent corresponding to the DI event being emitted
|
|
40
|
+
*/
|
|
41
|
+
function injectorProfiler(event) {
|
|
42
|
+
!ngDevMode && throwError('Injector profiler should never be called in production mode');
|
|
43
|
+
if (injectorProfilerCallback != null /* both `null` and `undefined` */) {
|
|
44
|
+
injectorProfilerCallback(event);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Emits an InjectorProfilerEventType.ProviderConfigured to the injector profiler. The data in the
|
|
49
|
+
* emitted event includes the raw provider, as well as the token that provider is providing.
|
|
50
|
+
*
|
|
51
|
+
* @param provider A provider object
|
|
52
|
+
*/
|
|
53
|
+
export function emitProviderConfiguredEvent(provider, isViewProvider = false) {
|
|
54
|
+
!ngDevMode && throwError('Injector profiler should never be called in production mode');
|
|
55
|
+
injectorProfiler({
|
|
56
|
+
type: 2 /* InjectorProfilerEventType.ProviderConfigured */,
|
|
57
|
+
context: getInjectorProfilerContext(),
|
|
58
|
+
providerRecord: {
|
|
59
|
+
token: typeof provider === 'function' ? provider : resolveForwardRef(provider.provide),
|
|
60
|
+
provider,
|
|
61
|
+
isViewProvider
|
|
62
|
+
}
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Emits an event to the injector profiler with the instance that was created. Note that
|
|
67
|
+
* the injector associated with this emission can be accessed by using getDebugInjectContext()
|
|
68
|
+
*
|
|
69
|
+
* @param instance an object created by an injector
|
|
70
|
+
*/
|
|
71
|
+
export function emitInstanceCreatedByInjectorEvent(instance) {
|
|
72
|
+
!ngDevMode && throwError('Injector profiler should never be called in production mode');
|
|
73
|
+
injectorProfiler({
|
|
74
|
+
type: 1 /* InjectorProfilerEventType.InstanceCreatedByInjector */,
|
|
75
|
+
context: getInjectorProfilerContext(),
|
|
76
|
+
instance: { value: instance }
|
|
77
|
+
});
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* @param token DI token associated with injected service
|
|
81
|
+
* @param value the instance of the injected service (i.e the result of `inject(token)`)
|
|
82
|
+
* @param flags the flags that the token was injected with
|
|
83
|
+
*/
|
|
84
|
+
export function emitInjectEvent(token, value, flags) {
|
|
85
|
+
!ngDevMode && throwError('Injector profiler should never be called in production mode');
|
|
86
|
+
injectorProfiler({
|
|
87
|
+
type: 0 /* InjectorProfilerEventType.Inject */,
|
|
88
|
+
context: getInjectorProfilerContext(),
|
|
89
|
+
service: { token, value, flags }
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
export function runInInjectorProfilerContext(injector, token, callback) {
|
|
93
|
+
!ngDevMode &&
|
|
94
|
+
throwError('runInInjectorProfilerContext should never be called in production mode');
|
|
95
|
+
const prevInjectContext = setInjectorProfilerContext({ injector, token });
|
|
96
|
+
try {
|
|
97
|
+
callback();
|
|
98
|
+
}
|
|
99
|
+
finally {
|
|
100
|
+
setInjectorProfilerContext(prevInjectContext);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"injector_profiler.js","sourceRoot":"","sources":["../../../../../../../../packages/core/src/render3/debug/injector_profiler.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAC,iBAAiB,EAAC,MAAM,sBAAsB,CAAC;AAMvD,OAAO,EAAC,UAAU,EAAC,MAAM,mBAAmB,CAAC;AAoI7C,IAAI,wBAAiD,CAAC;AACtD,MAAM,UAAU,0BAA0B;IACxC,CAAC,SAAS,IAAI,UAAU,CAAC,sEAAsE,CAAC,CAAC;IACjG,OAAO,wBAAwB,CAAC;AAClC,CAAC;AAED,MAAM,UAAU,0BAA0B,CAAC,OAAgC;IACzE,CAAC,SAAS,IAAI,UAAU,CAAC,sEAAsE,CAAC,CAAC;IAEjG,MAAM,QAAQ,GAAG,wBAAwB,CAAC;IAC1C,wBAAwB,GAAG,OAAO,CAAC;IACnC,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED,IAAI,wBAAwB,GAA0B,IAAI,CAAC;AAE3D;;;;;;;;;GASG;AACH,MAAM,CAAC,MAAM,mBAAmB,GAAG,CAAC,gBAAuC,EAAE,EAAE;IAC7E,CAAC,SAAS,IAAI,UAAU,CAAC,+DAA+D,CAAC,CAAC;IAC1F,wBAAwB,GAAG,gBAAgB,CAAC;AAC9C,CAAC,CAAC;AAEF;;;;GAIG;AACH,SAAS,gBAAgB,CAAC,KAA4B;IACpD,CAAC,SAAS,IAAI,UAAU,CAAC,6DAA6D,CAAC,CAAC;IAExF,IAAI,wBAAwB,IAAI,IAAI,CAAC,iCAAiC,EAAE;QACtE,wBAAyB,CAAC,KAAK,CAAC,CAAC;KAClC;AACH,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,2BAA2B,CACvC,QAAwB,EAAE,iBAA0B,KAAK;IAC3D,CAAC,SAAS,IAAI,UAAU,CAAC,6DAA6D,CAAC,CAAC;IAExF,gBAAgB,CAAC;QACf,IAAI,sDAA8C;QAClD,OAAO,EAAE,0BAA0B,EAAE;QACrC,cAAc,EAAE;YACd,KAAK,EAAE,OAAO,QAAQ,KAAK,UAAU,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,iBAAiB,CAAC,QAAQ,CAAC,OAAO,CAAC;YACtF,QAAQ;YACR,cAAc;SACf;KACF,CAAC,CAAC;AACL,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,kCAAkC,CAAC,QAAiB;IAClE,CAAC,SAAS,IAAI,UAAU,CAAC,6DAA6D,CAAC,CAAC;IAExF,gBAAgB,CAAC;QACf,IAAI,6DAAqD;QACzD,OAAO,EAAE,0BAA0B,EAAE;QACrC,QAAQ,EAAE,EAAC,KAAK,EAAE,QAAQ,EAAC;KAC5B,CAAC,CAAC;AACL,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,eAAe,CAAC,KAAoB,EAAE,KAAc,EAAE,KAAkB;IACtF,CAAC,SAAS,IAAI,UAAU,CAAC,6DAA6D,CAAC,CAAC;IAExF,gBAAgB,CAAC;QACf,IAAI,0CAAkC;QACtC,OAAO,EAAE,0BAA0B,EAAE;QACrC,OAAO,EAAE,EAAC,KAAK,EAAE,KAAK,EAAE,KAAK,EAAC;KAC/B,CAAC,CAAC;AACL,CAAC;AAED,MAAM,UAAU,4BAA4B,CACxC,QAAkB,EAAE,KAAoB,EAAE,QAAoB;IAChE,CAAC,SAAS;QACN,UAAU,CAAC,wEAAwE,CAAC,CAAC;IAEzF,MAAM,iBAAiB,GAAG,0BAA0B,CAAC,EAAC,QAAQ,EAAE,KAAK,EAAC,CAAC,CAAC;IACxE,IAAI;QACF,QAAQ,EAAE,CAAC;KACZ;YAAS;QACR,0BAA0B,CAAC,iBAAiB,CAAC,CAAC;KAC/C;AACH,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 {resolveForwardRef} from '../../di/forward_ref';\nimport {InjectionToken} from '../../di/injection_token';\nimport type {Injector} from '../../di/injector';\nimport {InjectFlags, InjectOptions, InternalInjectFlags} from '../../di/interface/injector';\nimport type {SingleProvider} from '../../di/provider_collection';\nimport {Type} from '../../interface/type';\nimport {throwError} from '../../util/assert';\n\n/**\n * An enum describing the types of events that can be emitted from the injector profiler\n */\nexport const enum InjectorProfilerEventType {\n  /**\n   * Emits when a service is injected.\n   */\n  Inject,\n\n  /**\n   * Emits when an Angular class instance is created by an injector.\n   */\n  InstanceCreatedByInjector,\n\n  /**\n   * Emits when an injector configures a provider.\n   */\n  ProviderConfigured\n}\n\n/**\n * An object that defines an injection context for the injector profiler.\n */\nexport interface InjectorProfilerContext {\n  /**\n   *  The Injector that service is being injected into.\n   *      - Example: if ModuleA --provides--> ServiceA --injects--> ServiceB\n   *                 then inject(ServiceB) in ServiceA has ModuleA as an injector context\n   */\n  injector: Injector;\n\n  /**\n   *  The class where the constructor that is calling `inject` is located\n   *      - Example: if ModuleA --provides--> ServiceA --injects--> ServiceB\n   *                 then inject(ServiceB) in ServiceA has ServiceA as a construction context\n   */\n  token: Type<unknown>|null;\n}\n\n\nexport interface InjectedServiceEvent {\n  type: InjectorProfilerEventType.Inject;\n  context: InjectorProfilerContext;\n  service: InjectedService;\n}\n\nexport interface InjectorCreatedInstanceEvent {\n  type: InjectorProfilerEventType.InstanceCreatedByInjector;\n  context: InjectorProfilerContext;\n  instance: InjectorCreatedInstance;\n}\n\nexport interface ProviderConfiguredEvent {\n  type: InjectorProfilerEventType.ProviderConfigured;\n  context: InjectorProfilerContext;\n  providerRecord: ProviderRecord;\n}\n\n/**\n * An object representing an event that is emitted through the injector profiler\n */\n\nexport type InjectorProfilerEvent =\n    InjectedServiceEvent|InjectorCreatedInstanceEvent|ProviderConfiguredEvent;\n\n/**\n * An object that contains information about a provider that has been configured\n */\nexport interface ProviderRecord {\n  /**\n   * DI token that this provider is configuring\n   */\n  token: Type<unknown>;\n\n  /**\n   * Determines if provider is configured as view provider.\n   */\n  isViewProvider: boolean;\n\n  /**\n   * The raw provider associated with this ProviderRecord.\n   */\n  provider: SingleProvider;\n\n  /**\n   * The path of DI containers that were followed to import this provider\n   */\n  importPath?: Type<unknown>[];\n}\n\n/**\n * An object that contains information about a value that has been constructed within an injector\n */\nexport interface InjectorCreatedInstance {\n  /**\n   * Value of the created instance\n   */\n  value: unknown;\n}\n\n/**\n * An object that contains information a service that has been injected within an\n * InjectorProfilerContext\n */\nexport interface InjectedService {\n  /**\n   * DI token of the Service that is injected\n   */\n  token?: Type<unknown>|InjectionToken<unknown>;\n\n  /**\n   * Value of the injected service\n   */\n  value: unknown;\n\n  /**\n   * Flags that this service was injected with\n   */\n  flags?: InternalInjectFlags|InjectFlags|InjectOptions;\n\n  /**\n   * Injector that this service was provided in.\n   */\n  providedIn?: Injector;\n}\n\nexport interface InjectorProfiler {\n  (event: InjectorProfilerEvent): void;\n}\n\nlet _injectorProfilerContext: InjectorProfilerContext;\nexport function getInjectorProfilerContext() {\n  !ngDevMode && throwError('getInjectorProfilerContext should never be called in production mode');\n  return _injectorProfilerContext;\n}\n\nexport function setInjectorProfilerContext(context: InjectorProfilerContext) {\n  !ngDevMode && throwError('setInjectorProfilerContext should never be called in production mode');\n\n  const previous = _injectorProfilerContext;\n  _injectorProfilerContext = context;\n  return previous;\n}\n\nlet injectorProfilerCallback: InjectorProfiler|null = null;\n\n/**\n * Sets the callback function which will be invoked during certain DI events within the\n * runtime (for example: injecting services, creating injectable instances, configuring providers)\n *\n * Warning: this function is *INTERNAL* and should not be relied upon in application's code.\n * The contract of the function might be changed in any release and/or the function can be removed\n * completely.\n *\n * @param profiler function provided by the caller or null value to disable profiling.\n */\nexport const setInjectorProfiler = (injectorProfiler: InjectorProfiler|null) => {\n  !ngDevMode && throwError('setInjectorProfiler should never be called in production mode');\n  injectorProfilerCallback = injectorProfiler;\n};\n\n/**\n * Injector profiler function which emits on DI events executed by the runtime.\n *\n * @param event InjectorProfilerEvent corresponding to the DI event being emitted\n */\nfunction injectorProfiler(event: InjectorProfilerEvent): void {\n  !ngDevMode && throwError('Injector profiler should never be called in production mode');\n\n  if (injectorProfilerCallback != null /* both `null` and `undefined` */) {\n    injectorProfilerCallback!(event);\n  }\n}\n\n/**\n * Emits an InjectorProfilerEventType.ProviderConfigured to the injector profiler. The data in the\n * emitted event includes the raw provider, as well as the token that provider is providing.\n *\n * @param provider A provider object\n */\nexport function emitProviderConfiguredEvent(\n    provider: SingleProvider, isViewProvider: boolean = false): void {\n  !ngDevMode && throwError('Injector profiler should never be called in production mode');\n\n  injectorProfiler({\n    type: InjectorProfilerEventType.ProviderConfigured,\n    context: getInjectorProfilerContext(),\n    providerRecord: {\n      token: typeof provider === 'function' ? provider : resolveForwardRef(provider.provide),\n      provider,\n      isViewProvider\n    }\n  });\n}\n\n/**\n * Emits an event to the injector profiler with the instance that was created. Note that\n * the injector associated with this emission can be accessed by using getDebugInjectContext()\n *\n * @param instance an object created by an injector\n */\nexport function emitInstanceCreatedByInjectorEvent(instance: unknown): void {\n  !ngDevMode && throwError('Injector profiler should never be called in production mode');\n\n  injectorProfiler({\n    type: InjectorProfilerEventType.InstanceCreatedByInjector,\n    context: getInjectorProfilerContext(),\n    instance: {value: instance}\n  });\n}\n\n/**\n * @param token DI token associated with injected service\n * @param value the instance of the injected service (i.e the result of `inject(token)`)\n * @param flags the flags that the token was injected with\n */\nexport function emitInjectEvent(token: Type<unknown>, value: unknown, flags: InjectFlags): void {\n  !ngDevMode && throwError('Injector profiler should never be called in production mode');\n\n  injectorProfiler({\n    type: InjectorProfilerEventType.Inject,\n    context: getInjectorProfilerContext(),\n    service: {token, value, flags}\n  });\n}\n\nexport function runInInjectorProfilerContext(\n    injector: Injector, token: Type<unknown>, callback: () => void): void {\n  !ngDevMode &&\n      throwError('runInInjectorProfilerContext should never be called in production mode');\n\n  const prevInjectContext = setInjectorProfilerContext({injector, token});\n  try {\n    callback();\n  } finally {\n    setInjectorProfilerContext(prevInjectContext);\n  }\n}\n"]}
|