@nmtjs/core 0.15.0-beta.2 → 0.15.0-beta.4

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 (57) hide show
  1. package/dist/constants.d.ts +18 -0
  2. package/dist/constants.js +1 -0
  3. package/dist/constants.js.map +1 -0
  4. package/dist/container.d.ts +47 -0
  5. package/dist/container.js +1 -0
  6. package/dist/container.js.map +1 -0
  7. package/dist/enums.d.ts +6 -0
  8. package/dist/enums.js +1 -0
  9. package/dist/enums.js.map +1 -0
  10. package/dist/hooks.d.ts +13 -0
  11. package/dist/hooks.js +1 -0
  12. package/dist/hooks.js.map +1 -0
  13. package/dist/index.d.ts +10 -0
  14. package/dist/index.js +1 -0
  15. package/dist/index.js.map +1 -0
  16. package/dist/injectables.d.ts +88 -0
  17. package/dist/injectables.js +1 -0
  18. package/dist/injectables.js.map +1 -0
  19. package/dist/logger.d.ts +11 -0
  20. package/dist/logger.js +1 -0
  21. package/dist/logger.js.map +1 -0
  22. package/dist/metadata.d.ts +13 -0
  23. package/dist/metadata.js +1 -0
  24. package/dist/metadata.js.map +1 -0
  25. package/dist/plugin.d.ts +9 -0
  26. package/dist/plugin.js +1 -0
  27. package/dist/plugin.js.map +1 -0
  28. package/dist/types.d.ts +2 -0
  29. package/dist/types.js +1 -0
  30. package/dist/types.js.map +1 -0
  31. package/dist/utils/functions.d.ts +7 -0
  32. package/dist/utils/functions.js +1 -0
  33. package/dist/utils/functions.js.map +1 -0
  34. package/dist/utils/index.d.ts +3 -0
  35. package/dist/utils/index.js +1 -0
  36. package/dist/utils/index.js.map +1 -0
  37. package/dist/utils/pool.d.ts +18 -0
  38. package/dist/utils/pool.js +1 -0
  39. package/dist/utils/pool.js.map +1 -0
  40. package/dist/utils/semaphore.d.ts +13 -0
  41. package/dist/utils/semaphore.js +1 -0
  42. package/dist/utils/semaphore.js.map +1 -0
  43. package/package.json +7 -6
  44. package/src/constants.ts +34 -0
  45. package/src/container.ts +509 -0
  46. package/src/enums.ts +6 -0
  47. package/src/hooks.ts +24 -0
  48. package/src/index.ts +14 -0
  49. package/src/injectables.ts +359 -0
  50. package/src/logger.ts +103 -0
  51. package/src/metadata.ts +24 -0
  52. package/src/plugin.ts +16 -0
  53. package/src/types.ts +3 -0
  54. package/src/utils/functions.ts +31 -0
  55. package/src/utils/index.ts +3 -0
  56. package/src/utils/pool.ts +111 -0
  57. package/src/utils/semaphore.ts +63 -0
@@ -0,0 +1,509 @@
1
+ import type { PerformanceMeasure } from 'node:perf_hooks'
2
+ import assert from 'node:assert'
3
+
4
+ import { tryCaptureStackTrace } from '@nmtjs/common'
5
+
6
+ import type {
7
+ AnyInjectable,
8
+ Dependencies,
9
+ DependencyContext,
10
+ Injection,
11
+ ResolveInjectableType,
12
+ } from './injectables.ts'
13
+ import type { Logger } from './logger.ts'
14
+ import { Scope } from './enums.ts'
15
+ import {
16
+ CoreInjectables,
17
+ compareScope,
18
+ createValueInjectable,
19
+ getDepedencencyInjectable,
20
+ isFactoryInjectable,
21
+ isInjectable,
22
+ isLazyInjectable,
23
+ isOptionalInjectable,
24
+ isValueInjectable,
25
+ provide,
26
+ } from './injectables.ts'
27
+
28
+ type InstanceWrapper = { private: any; public: any; context: any }
29
+
30
+ type ContainerOptions = { logger: Logger }
31
+
32
+ export class Container {
33
+ readonly instances = new Map<AnyInjectable, InstanceWrapper[]>()
34
+ private readonly resolvers = new Map<AnyInjectable, Promise<any>>()
35
+ private readonly injectables = new Set<AnyInjectable>()
36
+ private readonly dependants = new Map<AnyInjectable, Set<AnyInjectable>>()
37
+ private disposing = false
38
+
39
+ constructor(
40
+ private readonly runtime: ContainerOptions,
41
+ public readonly scope: Exclude<Scope, Scope.Transient> = Scope.Global,
42
+ private readonly parent?: Container,
43
+ ) {
44
+ if ((scope as any) === Scope.Transient) {
45
+ throw new Error('Invalid scope')
46
+ }
47
+ this.provide(CoreInjectables.inject, this.createInjectFunction())
48
+ this.provide(CoreInjectables.dispose, this.createDisposeFunction())
49
+ }
50
+
51
+ async initialize(injectables: Iterable<AnyInjectable>) {
52
+ const measurements: PerformanceMeasure[] = []
53
+
54
+ const traverse = (dependencies: Dependencies) => {
55
+ for (const key in dependencies) {
56
+ const dependency = dependencies[key]
57
+ const injectable = getDepedencencyInjectable(dependency)
58
+ if (injectable.scope === this.scope) {
59
+ this.injectables.add(injectable)
60
+ }
61
+ traverse(injectable.dependencies)
62
+ }
63
+ }
64
+
65
+ for (const dependant of injectables) {
66
+ traverse(dependant.dependencies)
67
+ }
68
+
69
+ await Promise.all(
70
+ [...this.injectables].map((injectable) =>
71
+ this.resolve(injectable, measurements),
72
+ ),
73
+ )
74
+
75
+ return measurements
76
+ }
77
+
78
+ fork(scope: Exclude<Scope, Scope.Transient>) {
79
+ return new Container(this.runtime, scope, this)
80
+ }
81
+
82
+ find(scope: Exclude<Scope, Scope.Transient>): Container | undefined {
83
+ if (this.scope === scope) {
84
+ return this
85
+ } else {
86
+ return this.parent?.find(scope)
87
+ }
88
+ }
89
+
90
+ async [Symbol.asyncDispose]() {
91
+ await this.dispose()
92
+ }
93
+
94
+ async dispose() {
95
+ this.runtime.logger.trace('Disposing [%s] scope context...', this.scope)
96
+
97
+ // Prevent new resolutions during disposal
98
+ this.disposing = true
99
+
100
+ // Get proper disposal order using topological sort
101
+ const disposalOrder = this.getDisposalOrder()
102
+
103
+ try {
104
+ // Dispose in the correct order
105
+ for (const injectable of disposalOrder) {
106
+ if (this.instances.has(injectable)) {
107
+ await this.disposeInjectableInstances(injectable)
108
+ }
109
+ }
110
+ } catch (error) {
111
+ this.runtime.logger.fatal(
112
+ { error },
113
+ 'Potential memory leak: error during container disposal',
114
+ )
115
+ }
116
+
117
+ this.instances.clear()
118
+ this.injectables.clear()
119
+ this.resolvers.clear()
120
+ this.dependants.clear()
121
+
122
+ this.disposing = false
123
+ }
124
+
125
+ containsWithinSelf(injectable: AnyInjectable) {
126
+ return this.instances.has(injectable) || this.resolvers.has(injectable)
127
+ }
128
+
129
+ contains(injectable: AnyInjectable): boolean {
130
+ return (
131
+ this.containsWithinSelf(injectable) ||
132
+ (this.parent?.contains(injectable) ?? false)
133
+ )
134
+ }
135
+
136
+ get<T extends AnyInjectable>(injectable: T): ResolveInjectableType<T> {
137
+ if (injectable.scope === Scope.Transient) {
138
+ throw new Error('Cannot get transient injectable directly')
139
+ }
140
+
141
+ if (this.instances.has(injectable)) {
142
+ return this.instances.get(injectable)!.at(0)!.public
143
+ }
144
+
145
+ if (this.parent?.contains(injectable)) {
146
+ return this.parent.get(injectable)
147
+ }
148
+
149
+ throw new Error('No instance found')
150
+ }
151
+
152
+ resolve<T extends AnyInjectable>(
153
+ injectable: T,
154
+ measurements?: PerformanceMeasure[],
155
+ ) {
156
+ return this.resolveInjectable(
157
+ injectable,
158
+ undefined,
159
+ undefined,
160
+ measurements,
161
+ )
162
+ }
163
+
164
+ async createContext<T extends Dependencies>(dependencies: T) {
165
+ return this.createInjectableContext(dependencies)
166
+ }
167
+
168
+ async provide<T extends Injection[]>(injections: T)
169
+ async provide<T extends AnyInjectable>(
170
+ injectable: T,
171
+ value: ResolveInjectableType<T> | AnyInjectable<ResolveInjectableType<T>>,
172
+ )
173
+ async provide<T extends AnyInjectable | Injection[]>(
174
+ injectable: T,
175
+ ...[value]: T extends AnyInjectable
176
+ ? [
177
+ value:
178
+ | ResolveInjectableType<T>
179
+ | AnyInjectable<ResolveInjectableType<T>>,
180
+ ]
181
+ : []
182
+ ) {
183
+ const injections = Array.isArray(injectable)
184
+ ? injectable
185
+ : [provide(injectable, value)]
186
+ await Promise.all(
187
+ injections.map(async ({ token, value }) => {
188
+ if (compareScope(token.scope, '>', this.scope)) {
189
+ // TODO: more informative error
190
+ throw new Error('Invalid scope')
191
+ }
192
+ const _value = isInjectable(value) ? await this.resolve(value) : value
193
+ this.instances.set(token, [
194
+ { private: _value, public: _value, context: undefined },
195
+ ])
196
+ }),
197
+ )
198
+ }
199
+
200
+ satisfies(injectable: AnyInjectable) {
201
+ return compareScope(injectable.scope, '<=', this.scope)
202
+ }
203
+
204
+ async disposeInjectableInstances(injectable: AnyInjectable) {
205
+ try {
206
+ if (this.instances.has(injectable)) {
207
+ const wrappers = this.instances.get(injectable)!
208
+ await Promise.all(
209
+ wrappers.map((wrapper) =>
210
+ this.disposeInjectableInstance(
211
+ injectable,
212
+ wrapper.private,
213
+ wrapper.context,
214
+ ),
215
+ ),
216
+ )
217
+ }
218
+ } catch (cause) {
219
+ const error = new Error(
220
+ 'Injectable disposal error. Potential memory leak',
221
+ { cause },
222
+ )
223
+ this.runtime.logger.error(error)
224
+ } finally {
225
+ this.instances.delete(injectable)
226
+ }
227
+ }
228
+
229
+ async disposeInjectableInstance(
230
+ injectable: AnyInjectable,
231
+ instance: any,
232
+ context: any,
233
+ ) {
234
+ if (isFactoryInjectable(injectable)) {
235
+ const { dispose } = injectable
236
+ if (dispose) await dispose(instance, context)
237
+ }
238
+ }
239
+
240
+ private async createInjectableContext<T extends Dependencies>(
241
+ dependencies: T,
242
+ dependant?: AnyInjectable,
243
+ measurements?: PerformanceMeasure[],
244
+ ) {
245
+ const injections: Record<string, any> = {}
246
+ const deps = Object.entries(dependencies)
247
+ const resolvers: Promise<any>[] = Array(deps.length)
248
+ for (let i = 0; i < deps.length; i++) {
249
+ const [key, dependency] = deps[i]
250
+ const isOptional = isOptionalInjectable(dependency)
251
+ const injectable = getDepedencencyInjectable(dependency)
252
+ const resolver = this.resolveInjectable(
253
+ injectable,
254
+ dependant,
255
+ isOptional,
256
+ measurements,
257
+ )
258
+ resolvers[i] = resolver.then((value) => (injections[key] = value))
259
+ }
260
+ await Promise.all(resolvers)
261
+ return Object.freeze(injections) as DependencyContext<T>
262
+ }
263
+
264
+ private resolveInjectable<T extends AnyInjectable>(
265
+ injectable: T,
266
+ dependant?: AnyInjectable,
267
+ isOptional?: boolean,
268
+ measurements?: PerformanceMeasure[],
269
+ ): Promise<ResolveInjectableType<T>> {
270
+ if (this.disposing) {
271
+ return Promise.reject(new Error('Cannot resolve during disposal'))
272
+ }
273
+
274
+ if (dependant && compareScope(dependant.scope, '<', injectable.scope)) {
275
+ // TODO: more informative error
276
+ return Promise.reject(
277
+ new Error('Invalid scope: dependant is looser than injectable'),
278
+ )
279
+ }
280
+
281
+ if (isValueInjectable(injectable)) {
282
+ return Promise.resolve(injectable.value)
283
+ } else if (
284
+ this.parent?.contains(injectable) ||
285
+ (this.parent?.satisfies(injectable) &&
286
+ compareScope(this.parent.scope, '<', this.scope))
287
+ ) {
288
+ return this.parent.resolveInjectable(
289
+ injectable,
290
+ dependant,
291
+ undefined,
292
+ measurements,
293
+ )
294
+ } else {
295
+ const { stack, label } = injectable
296
+
297
+ if (dependant) {
298
+ let dependants = this.dependants.get(injectable)
299
+ if (!dependants) {
300
+ this.dependants.set(injectable, (dependants = new Set()))
301
+ }
302
+ dependants.add(dependant)
303
+ }
304
+
305
+ const isTransient = injectable.scope === Scope.Transient
306
+
307
+ if (!isTransient && this.instances.has(injectable)) {
308
+ return Promise.resolve(this.instances.get(injectable)!.at(0)!.public)
309
+ } else if (!isTransient && this.resolvers.has(injectable)) {
310
+ return this.resolvers.get(injectable)!
311
+ } else {
312
+ const isLazy = isLazyInjectable(injectable)
313
+ if (isLazy) {
314
+ if (isOptional) return Promise.resolve(undefined as any)
315
+ return Promise.reject(
316
+ new Error(
317
+ `No instance provided for ${label || 'an'} injectable:\n${stack}`,
318
+ ),
319
+ )
320
+ } else {
321
+ const measure = measurements
322
+ ? performance.measure(injectable.label || injectable.stack || '')
323
+ : null
324
+ const resolution = this.createResolution(
325
+ injectable,
326
+ measurements,
327
+ ).finally(() => {
328
+ this.resolvers.delete(injectable)
329
+
330
+ // biome-ignore lint: false
331
+ // @ts-ignore
332
+ if (measurements && measure) measurements.push(measure)
333
+ })
334
+ if (injectable.scope !== Scope.Transient) {
335
+ this.resolvers.set(injectable, resolution)
336
+ }
337
+ return resolution
338
+ }
339
+ }
340
+ }
341
+ }
342
+
343
+ private async createResolution<T extends AnyInjectable>(
344
+ injectable: T,
345
+ measurements?: PerformanceMeasure[],
346
+ ): Promise<ResolveInjectableType<T>> {
347
+ const { dependencies } = injectable
348
+ const context = await this.createInjectableContext(
349
+ dependencies,
350
+ injectable,
351
+ measurements,
352
+ )
353
+ const wrapper = {
354
+ private: null as any,
355
+ public: null as ResolveInjectableType<T>,
356
+ context,
357
+ }
358
+ if (isFactoryInjectable(injectable)) {
359
+ wrapper.private = await Promise.resolve(
360
+ injectable.factory(wrapper.context),
361
+ )
362
+ wrapper.public = injectable.pick(wrapper.private)
363
+ } else {
364
+ throw new Error('Invalid injectable type')
365
+ }
366
+
367
+ let instances = this.instances.get(injectable)
368
+
369
+ if (!instances) {
370
+ instances = []
371
+ this.instances.set(injectable, instances)
372
+ }
373
+ instances.push(wrapper)
374
+
375
+ return wrapper.public
376
+ }
377
+
378
+ private createInjectFunction() {
379
+ const inject = <T extends AnyInjectable>(
380
+ injectable: T,
381
+ context: InlineInjectionDependencies<T>,
382
+ scope: Exclude<Scope, Scope.Transient> = this.scope,
383
+ ) => {
384
+ const container = this.find(scope)
385
+ if (!container)
386
+ throw new Error('No container found for the specified scope')
387
+
388
+ const dependencies: Dependencies = { ...injectable.dependencies }
389
+
390
+ for (const key in context) {
391
+ const dep = context[key]
392
+ if (isInjectable(dep) || isOptionalInjectable(dep)) {
393
+ dependencies[key] = dep
394
+ } else {
395
+ dependencies[key] = createValueInjectable(dep)
396
+ }
397
+ }
398
+
399
+ const newInjectable = {
400
+ ...injectable,
401
+ dependencies,
402
+ scope: Scope.Transient,
403
+ stack: tryCaptureStackTrace(1),
404
+ }
405
+
406
+ return container.resolve(newInjectable) as Promise<
407
+ ResolveInjectableType<T>
408
+ >
409
+ }
410
+
411
+ const explicit = async <T extends AnyInjectable>(
412
+ injectable: T,
413
+ context: InlineInjectionDependencies<T>,
414
+ scope: Exclude<Scope, Scope.Transient> = this.scope,
415
+ ) => {
416
+ if ('asyncDispose' in Symbol === false) {
417
+ throw new Error(
418
+ 'Symbol.asyncDispose is not supported in this environment',
419
+ )
420
+ }
421
+
422
+ const container = this.find(scope)
423
+ if (!container)
424
+ throw new Error('No container found for the specified scope')
425
+
426
+ const instance = await inject(injectable, context)
427
+ const dispose = container.createDisposeFunction()
428
+
429
+ return {
430
+ instance,
431
+ [Symbol.asyncDispose]: async () => {
432
+ await dispose(injectable, instance)
433
+ },
434
+ }
435
+ }
436
+
437
+ return Object.assign(inject, { explicit })
438
+ }
439
+
440
+ private createDisposeFunction() {
441
+ return async <T extends AnyInjectable>(injectable: T, instance?: any) => {
442
+ if (injectable.scope === Scope.Transient) {
443
+ assert(
444
+ instance,
445
+ 'Instance is required for transient injectable disposal',
446
+ )
447
+ const wrappers = this.instances.get(injectable)
448
+ if (wrappers) {
449
+ for (const wrapper of wrappers) {
450
+ if (wrapper.public === instance) {
451
+ await this.disposeInjectableInstance(
452
+ injectable,
453
+ wrapper.private,
454
+ wrapper.context,
455
+ )
456
+ const index = wrappers.indexOf(wrapper)
457
+ wrappers.splice(index, 1)
458
+ }
459
+ }
460
+
461
+ if (wrappers.length === 0) {
462
+ this.instances.delete(injectable)
463
+ }
464
+ }
465
+ } else {
466
+ await this.disposeInjectableInstances(injectable)
467
+ }
468
+ }
469
+ }
470
+
471
+ private getDisposalOrder(): AnyInjectable[] {
472
+ const visited = new Set<AnyInjectable>()
473
+ const result: AnyInjectable[] = []
474
+
475
+ const visit = (injectable: AnyInjectable) => {
476
+ if (visited.has(injectable)) return
477
+ visited.add(injectable)
478
+
479
+ const dependants = this.dependants.get(injectable)
480
+ if (dependants) {
481
+ for (const dependant of dependants) {
482
+ if (this.instances.has(dependant)) {
483
+ visit(dependant)
484
+ }
485
+ }
486
+ }
487
+
488
+ // Only add to result if this container owns the instance
489
+ if (this.instances.has(injectable)) {
490
+ result.push(injectable)
491
+ }
492
+ }
493
+
494
+ for (const injectable of this.instances.keys()) {
495
+ visit(injectable)
496
+ }
497
+
498
+ return result
499
+ }
500
+ }
501
+
502
+ type InlineInjectionDependencies<T extends AnyInjectable> = {
503
+ [K in keyof T['dependencies']]?:
504
+ | ResolveInjectableType<T['dependencies'][K]>
505
+ | AnyInjectable<ResolveInjectableType<T['dependencies'][K]>>
506
+ }
507
+
508
+ export type InjectFn = ReturnType<Container['createInjectFunction']>
509
+ export type DisposeFn = ReturnType<Container['createDisposeFunction']>
package/src/enums.ts ADDED
@@ -0,0 +1,6 @@
1
+ export enum Scope {
2
+ Global = 'Global',
3
+ Connection = 'Connection',
4
+ Call = 'Call',
5
+ Transient = 'Transient',
6
+ }
package/src/hooks.ts ADDED
@@ -0,0 +1,24 @@
1
+ import type { NestedHooks } from 'hookable'
2
+ import { Hookable } from 'hookable'
3
+
4
+ import type { HookTypes } from './types.ts'
5
+
6
+ export class Hooks<T extends HookTypes = HookTypes> extends Hookable<T> {
7
+ _!: { config: NestedHooks<T> }
8
+
9
+ createSignal<T extends keyof T>(
10
+ hook: T,
11
+ ): {
12
+ controller: AbortController
13
+ signal: AbortSignal
14
+ unregister: () => void
15
+ } {
16
+ const controller = new AbortController()
17
+ const unregister = this.hookOnce(
18
+ String(hook),
19
+ //@ts-expect-error
20
+ () => controller.abort(),
21
+ )
22
+ return { controller, signal: controller.signal, unregister }
23
+ }
24
+ }
package/src/index.ts ADDED
@@ -0,0 +1,14 @@
1
+ // // biome-ignore lint/correctness/noUnusedImports: TSC wants it
2
+ // // biome-ignore assist/source/organizeImports: TSC wants it
3
+ // import {} from 'pino'
4
+
5
+ export * from './constants.ts'
6
+ export * from './container.ts'
7
+ export * from './enums.ts'
8
+ export * from './hooks.ts'
9
+ export * from './injectables.ts'
10
+ export * from './logger.ts'
11
+ export * from './metadata.ts'
12
+ export * from './plugin.ts'
13
+ export * from './types.ts'
14
+ export * from './utils/index.ts'