@navios/di 0.4.1 → 0.5.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.
Files changed (128) hide show
  1. package/README.md +211 -1
  2. package/coverage/clover.xml +1912 -1277
  3. package/coverage/coverage-final.json +37 -28
  4. package/coverage/docs/examples/basic-usage.mts.html +1 -1
  5. package/coverage/docs/examples/factory-pattern.mts.html +1 -1
  6. package/coverage/docs/examples/index.html +1 -1
  7. package/coverage/docs/examples/injection-tokens.mts.html +1 -1
  8. package/coverage/docs/examples/request-scope-example.mts.html +1 -1
  9. package/coverage/docs/examples/service-lifecycle.mts.html +1 -1
  10. package/coverage/index.html +71 -41
  11. package/coverage/lib/_tsup-dts-rollup.d.mts.html +682 -43
  12. package/coverage/lib/index.d.mts.html +7 -4
  13. package/coverage/lib/index.html +5 -5
  14. package/coverage/lib/testing/index.d.mts.html +91 -0
  15. package/coverage/lib/testing/index.html +116 -0
  16. package/coverage/src/base-instance-holder-manager.mts.html +589 -0
  17. package/coverage/src/container.mts.html +257 -74
  18. package/coverage/src/decorators/factory.decorator.mts.html +1 -1
  19. package/coverage/src/decorators/index.html +1 -1
  20. package/coverage/src/decorators/index.mts.html +1 -1
  21. package/coverage/src/decorators/injectable.decorator.mts.html +20 -20
  22. package/coverage/src/enums/index.html +1 -1
  23. package/coverage/src/enums/index.mts.html +1 -1
  24. package/coverage/src/enums/injectable-scope.enum.mts.html +1 -1
  25. package/coverage/src/enums/injectable-type.enum.mts.html +1 -1
  26. package/coverage/src/errors/di-error.mts.html +292 -0
  27. package/coverage/src/errors/errors.enum.mts.html +30 -21
  28. package/coverage/src/errors/factory-not-found.mts.html +31 -22
  29. package/coverage/src/errors/factory-token-not-resolved.mts.html +29 -26
  30. package/coverage/src/errors/index.html +56 -41
  31. package/coverage/src/errors/index.mts.html +15 -9
  32. package/coverage/src/errors/instance-destroying.mts.html +31 -22
  33. package/coverage/src/errors/instance-expired.mts.html +31 -22
  34. package/coverage/src/errors/instance-not-found.mts.html +31 -22
  35. package/coverage/src/errors/unknown-error.mts.html +31 -43
  36. package/coverage/src/event-emitter.mts.html +14 -14
  37. package/coverage/src/factory-context.mts.html +1 -1
  38. package/coverage/src/index.html +121 -46
  39. package/coverage/src/index.mts.html +7 -4
  40. package/coverage/src/injection-token.mts.html +28 -28
  41. package/coverage/src/injector.mts.html +1 -1
  42. package/coverage/src/instance-resolver.mts.html +1762 -0
  43. package/coverage/src/interfaces/factory.interface.mts.html +1 -1
  44. package/coverage/src/interfaces/index.html +1 -1
  45. package/coverage/src/interfaces/index.mts.html +1 -1
  46. package/coverage/src/interfaces/on-service-destroy.interface.mts.html +1 -1
  47. package/coverage/src/interfaces/on-service-init.interface.mts.html +1 -1
  48. package/coverage/src/registry.mts.html +28 -28
  49. package/coverage/src/request-context-holder.mts.html +183 -102
  50. package/coverage/src/request-context-manager.mts.html +532 -0
  51. package/coverage/src/service-instantiator.mts.html +49 -49
  52. package/coverage/src/service-invalidator.mts.html +1372 -0
  53. package/coverage/src/service-locator-event-bus.mts.html +48 -48
  54. package/coverage/src/service-locator-instance-holder.mts.html +2 -14
  55. package/coverage/src/service-locator-manager.mts.html +71 -335
  56. package/coverage/src/service-locator.mts.html +240 -2328
  57. package/coverage/src/symbols/index.html +1 -1
  58. package/coverage/src/symbols/index.mts.html +1 -1
  59. package/coverage/src/symbols/injectable-token.mts.html +1 -1
  60. package/coverage/src/testing/index.html +131 -0
  61. package/coverage/src/testing/index.mts.html +88 -0
  62. package/coverage/src/testing/test-container.mts.html +445 -0
  63. package/coverage/src/token-processor.mts.html +607 -0
  64. package/coverage/src/utils/defer.mts.html +28 -214
  65. package/coverage/src/utils/get-injectable-token.mts.html +7 -7
  66. package/coverage/src/utils/get-injectors.mts.html +99 -99
  67. package/coverage/src/utils/index.html +15 -15
  68. package/coverage/src/utils/index.mts.html +4 -7
  69. package/coverage/src/utils/types.mts.html +1 -1
  70. package/docs/injectable.md +51 -11
  71. package/docs/scopes.md +63 -29
  72. package/lib/_tsup-dts-rollup.d.mts +447 -212
  73. package/lib/_tsup-dts-rollup.d.ts +447 -212
  74. package/lib/chunk-44F3LXW5.mjs +2043 -0
  75. package/lib/chunk-44F3LXW5.mjs.map +1 -0
  76. package/lib/index.d.mts +6 -4
  77. package/lib/index.d.ts +6 -4
  78. package/lib/index.js +1199 -773
  79. package/lib/index.js.map +1 -1
  80. package/lib/index.mjs +4 -1599
  81. package/lib/index.mjs.map +1 -1
  82. package/lib/testing/index.d.mts +2 -0
  83. package/lib/testing/index.d.ts +2 -0
  84. package/lib/testing/index.js +2060 -0
  85. package/lib/testing/index.js.map +1 -0
  86. package/lib/testing/index.mjs +73 -0
  87. package/lib/testing/index.mjs.map +1 -0
  88. package/package.json +11 -1
  89. package/src/__tests__/container.spec.mts +47 -13
  90. package/src/__tests__/errors.spec.mts +53 -27
  91. package/src/__tests__/injectable.spec.mts +73 -0
  92. package/src/__tests__/request-scope.spec.mts +0 -2
  93. package/src/__tests__/service-locator-manager.spec.mts +12 -82
  94. package/src/__tests__/service-locator.spec.mts +1009 -1
  95. package/src/__type-tests__/inject.spec-d.mts +30 -7
  96. package/src/__type-tests__/injectable.spec-d.mts +76 -37
  97. package/src/base-instance-holder-manager.mts +2 -9
  98. package/src/container.mts +70 -10
  99. package/src/decorators/injectable.decorator.mts +29 -5
  100. package/src/errors/di-error.mts +69 -0
  101. package/src/errors/index.mts +9 -7
  102. package/src/injection-token.mts +1 -0
  103. package/src/injector.mts +2 -0
  104. package/src/instance-resolver.mts +559 -0
  105. package/src/request-context-holder.mts +0 -2
  106. package/src/request-context-manager.mts +149 -0
  107. package/src/service-invalidator.mts +429 -0
  108. package/src/service-locator-instance-holder.mts +0 -4
  109. package/src/service-locator-manager.mts +10 -40
  110. package/src/service-locator.mts +86 -782
  111. package/src/testing/README.md +80 -0
  112. package/src/testing/__tests__/test-container.spec.mts +173 -0
  113. package/src/testing/index.mts +1 -0
  114. package/src/testing/test-container.mts +120 -0
  115. package/src/token-processor.mts +174 -0
  116. package/src/utils/get-injectors.mts +161 -24
  117. package/src/utils/index.mts +0 -1
  118. package/src/utils/types.mts +12 -8
  119. package/tsup.config.mts +1 -1
  120. package/src/__tests__/defer.spec.mts +0 -166
  121. package/src/errors/errors.enum.mts +0 -8
  122. package/src/errors/factory-not-found.mts +0 -8
  123. package/src/errors/factory-token-not-resolved.mts +0 -10
  124. package/src/errors/instance-destroying.mts +0 -8
  125. package/src/errors/instance-expired.mts +0 -8
  126. package/src/errors/instance-not-found.mts +0 -8
  127. package/src/errors/unknown-error.mts +0 -15
  128. package/src/utils/defer.mts +0 -73
@@ -0,0 +1,559 @@
1
+ /* eslint-disable @typescript-eslint/no-explicit-any */
2
+ /* eslint-disable @typescript-eslint/no-empty-object-type */
3
+ import type { z, ZodObject, ZodOptional } from 'zod/v4'
4
+
5
+ import type { FactoryContext } from './factory-context.mjs'
6
+ import type {
7
+ AnyInjectableType,
8
+ BaseInjectionTokenSchemaType,
9
+ InjectionTokenSchemaType,
10
+ InjectionTokenType,
11
+ OptionalInjectionTokenSchemaType,
12
+ } from './injection-token.mjs'
13
+ import type { Registry } from './registry.mjs'
14
+ import type { RequestContextHolder } from './request-context-holder.mjs'
15
+ import type { ServiceInstantiator } from './service-instantiator.mjs'
16
+ import type { ServiceLocatorInstanceHolder } from './service-locator-instance-holder.mjs'
17
+ import type { ServiceLocatorManager } from './service-locator-manager.mjs'
18
+ import type { ServiceLocator } from './service-locator.mjs'
19
+ import type { TokenProcessor } from './token-processor.mjs'
20
+
21
+ import { InjectableScope } from './enums/index.mjs'
22
+ import { DIError, DIErrorCode } from './errors/index.mjs'
23
+ import {
24
+ BoundInjectionToken,
25
+ FactoryInjectionToken,
26
+ InjectionToken,
27
+ } from './injection-token.mjs'
28
+ import { ServiceLocatorInstanceHolderStatus } from './service-locator-instance-holder.mjs'
29
+
30
+ /**
31
+ * InstanceResolver handles instance resolution, creation, and lifecycle management.
32
+ * Extracted from ServiceLocator to improve separation of concerns.
33
+ */
34
+ export class InstanceResolver {
35
+ constructor(
36
+ private readonly registry: Registry,
37
+ private readonly manager: ServiceLocatorManager,
38
+ private readonly serviceInstantiator: ServiceInstantiator,
39
+ private readonly tokenProcessor: TokenProcessor,
40
+ private readonly logger: Console | null = null,
41
+ private readonly serviceLocator: ServiceLocator,
42
+ ) {}
43
+
44
+ /**
45
+ * Resolves an instance for the given token and arguments.
46
+ */
47
+ async resolveInstance(
48
+ token: AnyInjectableType,
49
+ args?: any,
50
+ requestContext?: RequestContextHolder,
51
+ ): Promise<[undefined, any] | [DIError]> {
52
+ const [err, data] = await this.resolveTokenAndPrepareInstanceName(
53
+ token,
54
+ args,
55
+ )
56
+ if (err) {
57
+ return [err]
58
+ }
59
+
60
+ const {
61
+ instanceName,
62
+ validatedArgs,
63
+ actualToken: _actualToken,
64
+ realToken,
65
+ } = data
66
+
67
+ const [error, holder] = await this.retrieveOrCreateInstanceByInstanceName(
68
+ instanceName,
69
+ realToken,
70
+ validatedArgs,
71
+ requestContext,
72
+ )
73
+ if (error) {
74
+ return [error]
75
+ }
76
+ return [undefined, holder.instance]
77
+ }
78
+
79
+ /**
80
+ * Gets a synchronous instance (for sync operations).
81
+ */
82
+ getSyncInstance<
83
+ Instance,
84
+ Schema extends InjectionTokenSchemaType | undefined,
85
+ >(
86
+ token: AnyInjectableType,
87
+ args: Schema extends ZodObject
88
+ ? z.input<Schema>
89
+ : Schema extends ZodOptional<ZodObject>
90
+ ? z.input<Schema> | undefined
91
+ : undefined,
92
+ currentRequestContext: RequestContextHolder | null,
93
+ ): Instance | null {
94
+ const [err, { actualToken, validatedArgs }] =
95
+ this.tokenProcessor.validateAndResolveTokenArgs(token, args)
96
+ if (err) {
97
+ return null
98
+ }
99
+ const instanceName = this.tokenProcessor.generateInstanceName(
100
+ actualToken,
101
+ validatedArgs,
102
+ )
103
+
104
+ // Try request context first
105
+ if (currentRequestContext) {
106
+ const requestHolder = currentRequestContext.get(instanceName)
107
+ if (requestHolder) {
108
+ return requestHolder.instance as Instance
109
+ }
110
+ }
111
+
112
+ // Try singleton manager
113
+ const [error, holder] = this.manager.get(instanceName)
114
+ if (error) {
115
+ return null
116
+ }
117
+ return holder.instance as Instance
118
+ }
119
+
120
+ /**
121
+ * Internal method to resolve token args and create instance name.
122
+ * Handles factory token resolution and validation.
123
+ */
124
+ private async resolveTokenAndPrepareInstanceName(
125
+ token: AnyInjectableType,
126
+ args?: any,
127
+ ): Promise<
128
+ | [
129
+ undefined,
130
+ {
131
+ instanceName: string
132
+ validatedArgs: any
133
+ actualToken: InjectionTokenType
134
+ realToken: InjectionToken<any, any>
135
+ },
136
+ ]
137
+ | [DIError]
138
+ > {
139
+ const [err, { actualToken, validatedArgs }] =
140
+ this.tokenProcessor.validateAndResolveTokenArgs(token, args)
141
+ if (err instanceof DIError && err.code === DIErrorCode.UnknownError) {
142
+ return [err]
143
+ } else if (
144
+ err instanceof DIError &&
145
+ err.code === DIErrorCode.FactoryTokenNotResolved &&
146
+ actualToken instanceof FactoryInjectionToken
147
+ ) {
148
+ this.logger?.log(
149
+ `[InstanceResolver]#resolveTokenAndPrepareInstanceName() Factory token not resolved, resolving it`,
150
+ )
151
+ await actualToken.resolve(this.createFactoryContext())
152
+ return this.resolveTokenAndPrepareInstanceName(token)
153
+ }
154
+ const instanceName = this.tokenProcessor.generateInstanceName(
155
+ actualToken,
156
+ validatedArgs,
157
+ )
158
+ // Determine the real token (the actual InjectionToken that will be used for resolution)
159
+ const realToken =
160
+ actualToken instanceof BoundInjectionToken ||
161
+ actualToken instanceof FactoryInjectionToken
162
+ ? actualToken.token
163
+ : actualToken
164
+ return [undefined, { instanceName, validatedArgs, actualToken, realToken }]
165
+ }
166
+
167
+ /**
168
+ * Gets an instance by its instance name, handling all the logic after instance name creation.
169
+ */
170
+ private async retrieveOrCreateInstanceByInstanceName(
171
+ instanceName: string,
172
+ realToken: InjectionToken<any, any>,
173
+ realArgs: any,
174
+ requestContext?: RequestContextHolder,
175
+ ): Promise<[undefined, ServiceLocatorInstanceHolder<any>] | [DIError]> {
176
+ // Try to get existing instance (handles both request-scoped and singleton)
177
+ const existingHolder = await this.tryGetExistingInstance(
178
+ instanceName,
179
+ realToken,
180
+ requestContext,
181
+ )
182
+ if (existingHolder) {
183
+ return existingHolder
184
+ }
185
+
186
+ // No existing instance found, create a new one
187
+ const result = await this.createNewInstance(
188
+ instanceName,
189
+ realToken,
190
+ realArgs,
191
+ requestContext,
192
+ )
193
+ if (result[0]) {
194
+ return [result[0]]
195
+ }
196
+
197
+ const [, holder] = result
198
+ return this.waitForInstanceReady(holder)
199
+ }
200
+
201
+ /**
202
+ * Attempts to retrieve an existing instance, handling request-scoped and singleton instances.
203
+ * Returns null if no instance exists and a new one should be created.
204
+ */
205
+ private async tryGetExistingInstance(
206
+ instanceName: string,
207
+ realToken: InjectionToken<any, any>,
208
+ requestContext?: RequestContextHolder,
209
+ ): Promise<
210
+ [undefined, ServiceLocatorInstanceHolder<any>] | [DIError] | null
211
+ > {
212
+ // Check request-scoped instances first
213
+ const requestResult = await this.tryGetRequestScopedInstance(
214
+ instanceName,
215
+ realToken,
216
+ requestContext,
217
+ )
218
+ if (requestResult) {
219
+ return requestResult
220
+ }
221
+
222
+ // Check singleton instances
223
+ return this.tryGetSingletonInstance(instanceName)
224
+ }
225
+
226
+ /**
227
+ * Attempts to get a request-scoped instance if applicable.
228
+ */
229
+ private async tryGetRequestScopedInstance(
230
+ instanceName: string,
231
+ realToken: InjectionToken<any, any>,
232
+ requestContext?: RequestContextHolder,
233
+ ): Promise<
234
+ [undefined, ServiceLocatorInstanceHolder<any>] | [DIError] | null
235
+ > {
236
+ if (!this.registry.has(realToken)) {
237
+ return null
238
+ }
239
+
240
+ const record = this.registry.get(realToken)
241
+ if (record.scope !== InjectableScope.Request) {
242
+ return null
243
+ }
244
+
245
+ if (!requestContext) {
246
+ this.logger?.log(
247
+ `[InstanceResolver] No current request context available for request-scoped service ${instanceName}`,
248
+ )
249
+ return [
250
+ DIError.unknown(
251
+ `No current request context available for request-scoped service ${instanceName}`,
252
+ ),
253
+ ]
254
+ }
255
+
256
+ const requestHolder = requestContext.get(instanceName)
257
+ if (!requestHolder) {
258
+ return null
259
+ }
260
+
261
+ return this.waitForInstanceReady(requestHolder)
262
+ }
263
+
264
+ /**
265
+ * Attempts to get a singleton instance from the manager.
266
+ */
267
+ private async tryGetSingletonInstance(
268
+ instanceName: string,
269
+ ): Promise<
270
+ [undefined, ServiceLocatorInstanceHolder<any>] | [DIError] | null
271
+ > {
272
+ const [error, holder] = this.manager.get(instanceName)
273
+
274
+ if (!error) {
275
+ return this.waitForInstanceReady(holder)
276
+ }
277
+
278
+ // Handle recovery scenarios
279
+ switch (error.code) {
280
+ case DIErrorCode.InstanceDestroying:
281
+ this.logger?.log(
282
+ `[InstanceResolver] Instance ${instanceName} is being destroyed, waiting...`,
283
+ )
284
+ await holder?.destroyPromise
285
+ // Retry after destruction is complete
286
+ return this.tryGetSingletonInstance(instanceName)
287
+
288
+ case DIErrorCode.InstanceNotFound:
289
+ return null // Instance doesn't exist, should create new one
290
+
291
+ default:
292
+ return [error]
293
+ }
294
+ }
295
+
296
+ /**
297
+ * Waits for an instance holder to be ready and returns the appropriate result.
298
+ */
299
+ private async waitForInstanceReady<T>(
300
+ holder: ServiceLocatorInstanceHolder<T>,
301
+ ): Promise<[undefined, ServiceLocatorInstanceHolder<T>] | [DIError]> {
302
+ switch (holder.status) {
303
+ case ServiceLocatorInstanceHolderStatus.Creating:
304
+ await holder.creationPromise
305
+ return this.waitForInstanceReady(holder)
306
+
307
+ case ServiceLocatorInstanceHolderStatus.Destroying:
308
+ return [DIError.instanceDestroying(holder.name)]
309
+
310
+ case ServiceLocatorInstanceHolderStatus.Error:
311
+ return [holder.instance as DIError]
312
+
313
+ case ServiceLocatorInstanceHolderStatus.Created:
314
+ return [undefined, holder]
315
+
316
+ default:
317
+ return [DIError.instanceNotFound('unknown')]
318
+ }
319
+ }
320
+
321
+ /**
322
+ * Creates a new instance for the given token and arguments.
323
+ */
324
+ private async createNewInstance<
325
+ Instance,
326
+ Schema extends InjectionTokenSchemaType | undefined,
327
+ >(
328
+ instanceName: string,
329
+ realToken: InjectionToken<Instance, Schema>,
330
+ args: Schema extends ZodObject
331
+ ? z.output<Schema>
332
+ : Schema extends ZodOptional<ZodObject>
333
+ ? z.output<Schema> | undefined
334
+ : undefined,
335
+ requestContext?: RequestContextHolder,
336
+ ): Promise<[undefined, ServiceLocatorInstanceHolder<Instance>] | [DIError]> {
337
+ this.logger?.log(
338
+ `[InstanceResolver]#createNewInstance() Creating instance for ${instanceName}`,
339
+ )
340
+ if (this.registry.has(realToken)) {
341
+ return this.instantiateServiceFromRegistry<Instance, Schema, any>(
342
+ instanceName,
343
+ realToken,
344
+ args,
345
+ requestContext,
346
+ )
347
+ } else {
348
+ return [DIError.factoryNotFound(realToken.name.toString())]
349
+ }
350
+ }
351
+
352
+ /**
353
+ * Instantiates a service from the registry using the service instantiator.
354
+ */
355
+ private instantiateServiceFromRegistry<
356
+ Instance,
357
+ Schema extends InjectionTokenSchemaType | undefined,
358
+ Args extends Schema extends BaseInjectionTokenSchemaType
359
+ ? z.output<Schema>
360
+ : Schema extends OptionalInjectionTokenSchemaType
361
+ ? z.output<Schema> | undefined
362
+ : undefined,
363
+ >(
364
+ instanceName: string,
365
+ token: InjectionToken<Instance, Schema>,
366
+ args: Args,
367
+ requestContext?: RequestContextHolder,
368
+ ): Promise<[undefined, ServiceLocatorInstanceHolder<Instance>]> {
369
+ this.logger?.log(
370
+ `[InstanceResolver]#instantiateServiceFromRegistry(): Creating instance for ${instanceName} from abstract factory`,
371
+ )
372
+ const ctx = this.createFactoryContext()
373
+ let record = this.registry.get<Instance, Schema>(token)
374
+ let { scope, type } = record
375
+
376
+ // Use createCreatingHolder from manager
377
+ const [deferred, holder] = this.manager.createCreatingHolder<Instance>(
378
+ instanceName,
379
+ type,
380
+ scope,
381
+ ctx.deps,
382
+ )
383
+
384
+ // Start the instantiation process
385
+ this.serviceInstantiator
386
+ .instantiateService(ctx, record, args)
387
+ .then(async ([error, instance]) => {
388
+ await this.handleInstantiationResult(
389
+ instanceName,
390
+ holder,
391
+ ctx,
392
+ deferred,
393
+ scope,
394
+ error,
395
+ instance,
396
+ requestContext,
397
+ )
398
+ })
399
+ .catch(async (error) => {
400
+ await this.handleInstantiationError(
401
+ instanceName,
402
+ holder,
403
+ deferred,
404
+ scope,
405
+ error,
406
+ )
407
+ })
408
+
409
+ this.storeInstanceByScope(scope, instanceName, holder, requestContext)
410
+ // @ts-expect-error TS2322 This is correct type
411
+ return [undefined, holder]
412
+ }
413
+
414
+ /**
415
+ * Handles the result of service instantiation.
416
+ */
417
+ private async handleInstantiationResult(
418
+ instanceName: string,
419
+ holder: ServiceLocatorInstanceHolder<any>,
420
+ ctx: FactoryContext & {
421
+ deps: Set<string>
422
+ getDestroyListeners: () => (() => void)[]
423
+ },
424
+ deferred: any,
425
+ scope: InjectableScope,
426
+ error: any,
427
+ instance: any,
428
+ _requestContext?: RequestContextHolder,
429
+ ): Promise<void> {
430
+ holder.destroyListeners = ctx.getDestroyListeners()
431
+ holder.creationPromise = null
432
+
433
+ if (error) {
434
+ await this.handleInstantiationError(
435
+ instanceName,
436
+ holder,
437
+ deferred,
438
+ scope,
439
+ error,
440
+ )
441
+ } else {
442
+ await this.handleInstantiationSuccess(
443
+ instanceName,
444
+ holder,
445
+ ctx,
446
+ deferred,
447
+ instance,
448
+ )
449
+ }
450
+ }
451
+
452
+ /**
453
+ * Handles successful service instantiation.
454
+ */
455
+ private async handleInstantiationSuccess(
456
+ instanceName: string,
457
+ holder: ServiceLocatorInstanceHolder<any>,
458
+ ctx: FactoryContext & {
459
+ deps: Set<string>
460
+ getDestroyListeners: () => (() => void)[]
461
+ },
462
+ deferred: any,
463
+ instance: any,
464
+ ): Promise<void> {
465
+ holder.instance = instance
466
+ holder.status = ServiceLocatorInstanceHolderStatus.Created
467
+
468
+ // Set up dependency invalidation listeners
469
+ if (ctx.deps.size > 0) {
470
+ ctx.deps.forEach((dependency: string) => {
471
+ holder.destroyListeners.push(
472
+ this.serviceLocator.getEventBus().on(dependency, 'destroy', () => {
473
+ this.logger?.log(
474
+ `[InstanceResolver] Dependency ${dependency} destroyed, invalidating ${instanceName}`,
475
+ )
476
+ this.serviceLocator.getServiceInvalidator().invalidate(instanceName)
477
+ }),
478
+ )
479
+ })
480
+ }
481
+
482
+ // Note: Event emission would need access to the event bus
483
+ this.logger?.log(
484
+ `[InstanceResolver] Instance ${instanceName} created successfully`,
485
+ )
486
+ deferred.resolve([undefined, instance])
487
+ }
488
+
489
+ /**
490
+ * Handles service instantiation errors.
491
+ */
492
+ private async handleInstantiationError(
493
+ instanceName: string,
494
+ holder: ServiceLocatorInstanceHolder<any>,
495
+ deferred: any,
496
+ scope: InjectableScope,
497
+ error: any,
498
+ ): Promise<void> {
499
+ this.logger?.error(
500
+ `[InstanceResolver] Error creating instance for ${instanceName}`,
501
+ error,
502
+ )
503
+
504
+ holder.status = ServiceLocatorInstanceHolderStatus.Error
505
+ holder.instance = error
506
+ holder.creationPromise = null
507
+
508
+ if (scope === InjectableScope.Singleton) {
509
+ this.logger?.log(
510
+ `[InstanceResolver] Singleton ${instanceName} failed, will be invalidated`,
511
+ )
512
+ this.serviceLocator.getServiceInvalidator().invalidate(instanceName)
513
+ }
514
+
515
+ deferred.reject(error)
516
+ }
517
+
518
+ /**
519
+ * Stores an instance holder based on its scope.
520
+ */
521
+ private storeInstanceByScope(
522
+ scope: InjectableScope,
523
+ instanceName: string,
524
+ holder: ServiceLocatorInstanceHolder<any>,
525
+ requestContext?: RequestContextHolder,
526
+ ): void {
527
+ switch (scope) {
528
+ case InjectableScope.Singleton:
529
+ this.logger?.debug(
530
+ `[InstanceResolver] Setting singleton instance for ${instanceName}`,
531
+ )
532
+ this.manager.set(instanceName, holder)
533
+ break
534
+
535
+ case InjectableScope.Request:
536
+ if (requestContext) {
537
+ this.logger?.debug(
538
+ `[InstanceResolver] Setting request-scoped instance for ${instanceName}`,
539
+ )
540
+ requestContext.addInstance(instanceName, holder.instance, holder)
541
+ }
542
+ break
543
+
544
+ case InjectableScope.Transient:
545
+ // Transient instances are not stored anywhere
546
+ break
547
+ }
548
+ }
549
+
550
+ /**
551
+ * Creates a factory context for dependency injection during service instantiation.
552
+ */
553
+ private createFactoryContext(): FactoryContext & {
554
+ getDestroyListeners: () => (() => void)[]
555
+ deps: Set<string>
556
+ } {
557
+ return this.tokenProcessor.createFactoryContext(this.serviceLocator)
558
+ }
559
+ }
@@ -3,7 +3,6 @@ import type { ServiceLocatorInstanceHolder } from './service-locator-instance-ho
3
3
  import { BaseInstanceHolderManager } from './base-instance-holder-manager.mjs'
4
4
  import { InjectableScope, InjectableType } from './enums/index.mjs'
5
5
  import { InjectionToken } from './injection-token.mjs'
6
- import { ServiceLocatorInstanceHolderStatus } from './service-locator-instance-holder.mjs'
7
6
 
8
7
  /**
9
8
  * Request context holder that manages pre-prepared instances for a specific request.
@@ -166,7 +165,6 @@ export class DefaultRequestContextHolder
166
165
  InjectableType.Class,
167
166
  InjectableScope.Singleton,
168
167
  new Set(),
169
- Infinity,
170
168
  )
171
169
  this._holders.set(name, createdHolder)
172
170
  } else {