@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,359 @@
1
+ import type { Async } from '@nmtjs/common'
2
+ import { tryCaptureStackTrace } from '@nmtjs/common'
3
+
4
+ import type { DisposeFn, InjectFn } from './container.ts'
5
+ import type { Logger } from './logger.ts'
6
+ import {
7
+ kFactoryInjectable,
8
+ kInjectable,
9
+ kLazyInjectable,
10
+ kOptionalDependency,
11
+ kValueInjectable,
12
+ } from './constants.ts'
13
+ import { Scope } from './enums.ts'
14
+
15
+ const ScopeStrictness = {
16
+ [Scope.Transient]: Number.NaN, // this should make it always fail to compare with other scopes
17
+ [Scope.Global]: 1,
18
+ [Scope.Connection]: 2,
19
+ [Scope.Call]: 3,
20
+ }
21
+
22
+ export type DependencyOptional<T extends AnyInjectable = AnyInjectable> = {
23
+ [kOptionalDependency]: any
24
+ injectable: T
25
+ }
26
+
27
+ export type Depedency = DependencyOptional | AnyInjectable
28
+
29
+ export type Dependencies = Record<string, Depedency>
30
+
31
+ export type ResolveInjectableType<T extends AnyInjectable> =
32
+ T extends Injectable<infer Type, any, any> ? Type : never
33
+
34
+ export interface Dependant<Deps extends Dependencies = Dependencies> {
35
+ dependencies: Deps
36
+ label?: string
37
+ stack?: string
38
+ }
39
+
40
+ export type DependencyInjectable<T extends Depedency> = T extends AnyInjectable
41
+ ? T
42
+ : T extends DependencyOptional
43
+ ? T['injectable']
44
+ : never
45
+
46
+ export type DependencyContext<Deps extends Dependencies> = {
47
+ readonly [K in keyof Deps as Deps[K] extends AnyInjectable
48
+ ? K
49
+ : never]: Deps[K] extends AnyInjectable
50
+ ? ResolveInjectableType<Deps[K]>
51
+ : never
52
+ } & {
53
+ readonly [K in keyof Deps as Deps[K] extends DependencyOptional
54
+ ? K
55
+ : never]?: Deps[K] extends DependencyOptional
56
+ ? ResolveInjectableType<Deps[K]['injectable']>
57
+ : never
58
+ }
59
+
60
+ export type InjectableFactoryType<
61
+ InjectableType,
62
+ InjectableDeps extends Dependencies,
63
+ > = (context: DependencyContext<InjectableDeps>) => Async<InjectableType>
64
+
65
+ export type InjectablePickType<Input, Output> = (injectable: Input) => Output
66
+
67
+ export type InjectableDisposeType<
68
+ InjectableType,
69
+ InjectableDeps extends Dependencies,
70
+ > = (
71
+ instance: InjectableType,
72
+ context: DependencyContext<InjectableDeps>,
73
+ ) => any
74
+
75
+ export interface LazyInjectable<T, S extends Scope = Scope.Global>
76
+ extends Dependant<{}> {
77
+ scope: S
78
+ $withType<O extends T>(): LazyInjectable<O, S>
79
+ optional(): DependencyOptional<LazyInjectable<T, S>>
80
+ [kInjectable]: any
81
+ [kLazyInjectable]: T
82
+ }
83
+
84
+ export interface ValueInjectable<T> extends Dependant<{}> {
85
+ scope: Scope.Global
86
+ value: T
87
+ [kInjectable]: any
88
+ [kValueInjectable]: any
89
+ }
90
+
91
+ export interface FactoryInjectable<
92
+ T,
93
+ D extends Dependencies = {},
94
+ S extends Scope = Scope.Global,
95
+ P = T,
96
+ > extends Dependant<D> {
97
+ scope: S
98
+ factory: InjectableFactoryType<P, D>
99
+ pick: InjectablePickType<P, T>
100
+ dispose?: InjectableDisposeType<P, D>
101
+ optional(): DependencyOptional<FactoryInjectable<T, D, S, P>>
102
+ [kInjectable]: any
103
+ [kFactoryInjectable]: any
104
+ }
105
+
106
+ export type Injectable<
107
+ V = any,
108
+ D extends Dependencies = {},
109
+ S extends Scope = Scope,
110
+ > = LazyInjectable<V, S> | ValueInjectable<V> | FactoryInjectable<V, D, S, any>
111
+
112
+ export type AnyInjectable<T = any, S extends Scope = Scope> = Injectable<
113
+ T,
114
+ any,
115
+ S
116
+ >
117
+
118
+ export const isLazyInjectable = (
119
+ injectable: any,
120
+ ): injectable is LazyInjectable<any> => injectable[kLazyInjectable]
121
+
122
+ export const isFactoryInjectable = (
123
+ injectable: any,
124
+ ): injectable is FactoryInjectable<any> => injectable[kFactoryInjectable]
125
+
126
+ export const isValueInjectable = (
127
+ injectable: any,
128
+ ): injectable is ValueInjectable<any> => injectable[kValueInjectable]
129
+
130
+ export const isInjectable = (
131
+ injectable: any,
132
+ ): injectable is AnyInjectable<any> => injectable[kInjectable]
133
+
134
+ export const isOptionalInjectable = (
135
+ injectable: any,
136
+ ): injectable is DependencyOptional<any> => injectable[kOptionalDependency]
137
+
138
+ export function getInjectableScope(injectable: AnyInjectable) {
139
+ let scope = injectable.scope
140
+ const deps = injectable.dependencies as Dependencies
141
+ for (const key in deps) {
142
+ const dependency = deps[key]
143
+ const injectable = getDepedencencyInjectable(dependency)
144
+ const dependencyScope = getInjectableScope(injectable)
145
+ if (
146
+ dependencyScope !== Scope.Transient &&
147
+ scope !== Scope.Transient &&
148
+ compareScope(dependencyScope, '>', scope)
149
+ ) {
150
+ scope = dependencyScope
151
+ }
152
+ }
153
+ return scope
154
+ }
155
+
156
+ export function getDepedencencyInjectable(
157
+ dependency: Depedency,
158
+ ): AnyInjectable {
159
+ if (kOptionalDependency in dependency) {
160
+ return dependency.injectable
161
+ }
162
+ return dependency
163
+ }
164
+
165
+ export function createOptionalInjectable<T extends AnyInjectable>(
166
+ injectable: T,
167
+ ) {
168
+ return Object.freeze({
169
+ [kOptionalDependency]: true,
170
+ injectable,
171
+ }) as DependencyOptional<T>
172
+ }
173
+
174
+ export function createLazyInjectable<T, S extends Scope = Scope.Global>(
175
+ scope = Scope.Global as S,
176
+ label?: string,
177
+ stackTraceDepth = 0,
178
+ ): LazyInjectable<T, S> {
179
+ const injectable = Object.freeze({
180
+ scope,
181
+ dependencies: {},
182
+ label,
183
+ stack: tryCaptureStackTrace(stackTraceDepth),
184
+ optional: () => createOptionalInjectable(injectable),
185
+ $withType: () => injectable as any,
186
+ [kInjectable]: true,
187
+ [kLazyInjectable]: true as unknown as T,
188
+ })
189
+ return injectable
190
+ }
191
+
192
+ export function createValueInjectable<T>(
193
+ value: T,
194
+ label?: string,
195
+ stackTraceDepth = 0,
196
+ ): ValueInjectable<T> {
197
+ return Object.freeze({
198
+ value,
199
+ scope: Scope.Global,
200
+ dependencies: {},
201
+ label,
202
+ stack: tryCaptureStackTrace(stackTraceDepth),
203
+ [kInjectable]: true,
204
+ [kValueInjectable]: true,
205
+ })
206
+ }
207
+
208
+ export function createFactoryInjectable<
209
+ T,
210
+ D extends Dependencies = {},
211
+ S extends Scope = Scope.Global,
212
+ P = T,
213
+ >(
214
+ paramsOrFactory:
215
+ | {
216
+ dependencies?: D
217
+ scope?: S
218
+ pick?: InjectablePickType<P, T>
219
+ factory: InjectableFactoryType<P, D>
220
+ dispose?: InjectableDisposeType<P, D>
221
+ }
222
+ | InjectableFactoryType<P, D>,
223
+ label?: string,
224
+ stackTraceDepth = 0,
225
+ ): FactoryInjectable<null extends T ? P : T, D, S, P> {
226
+ const isFactory = typeof paramsOrFactory === 'function'
227
+ const params = isFactory ? { factory: paramsOrFactory } : paramsOrFactory
228
+ const injectable = {
229
+ dependencies: (params.dependencies ?? {}) as D,
230
+ scope: (params.scope ?? Scope.Global) as S,
231
+ factory: params.factory,
232
+ dispose: params.dispose,
233
+ pick: params.pick ?? ((instance: P) => instance as unknown as T),
234
+ label,
235
+ stack: tryCaptureStackTrace(stackTraceDepth),
236
+ optional: () => createOptionalInjectable(injectable),
237
+ [kInjectable]: true,
238
+ [kFactoryInjectable]: true,
239
+ }
240
+ injectable.scope = resolveInjectableScope(
241
+ typeof params.scope === 'undefined',
242
+ injectable,
243
+ ) as S
244
+ return Object.freeze(injectable) as any
245
+ }
246
+
247
+ export type DependenciesSubstitution<T extends Dependencies> = {
248
+ [K in keyof T]?: T[K] extends AnyInjectable<infer Type>
249
+ ? AnyInjectable<Type> | DependenciesSubstitution<T[K]['dependencies']>
250
+ : never
251
+ }
252
+
253
+ export function substitute<T extends FactoryInjectable<any, any, Scope>>(
254
+ injectable: T,
255
+ substitution: DependenciesSubstitution<T['dependencies']>,
256
+ stackTraceDepth = 0,
257
+ ): T {
258
+ const dependencies = { ...injectable.dependencies }
259
+ const depth = stackTraceDepth + 1
260
+ for (const key in substitution) {
261
+ const value = substitution[key]!
262
+ if (key in dependencies) {
263
+ const original = dependencies[key]
264
+ if (isInjectable(value)) {
265
+ dependencies[key] = value
266
+ } else if (isFactoryInjectable(original)) {
267
+ dependencies[key] = substitute(original, value, depth)
268
+ }
269
+ }
270
+ }
271
+
272
+ if (isFactoryInjectable(injectable)) {
273
+ // @ts-expect-error
274
+ return createFactoryInjectable(
275
+ { ...injectable, dependencies },
276
+ injectable.label,
277
+ depth,
278
+ )
279
+ }
280
+
281
+ throw new Error('Invalid injectable type')
282
+ }
283
+
284
+ export function compareScope(
285
+ left: Scope,
286
+ operator: '>' | '<' | '>=' | '<=' | '=' | '!=',
287
+ right: Scope,
288
+ ) {
289
+ const leftScope = ScopeStrictness[left]
290
+ const rightScope = ScopeStrictness[right]
291
+ switch (operator) {
292
+ case '=':
293
+ return leftScope === rightScope
294
+ case '!=':
295
+ return leftScope !== rightScope
296
+ case '>':
297
+ return leftScope > rightScope
298
+ case '<':
299
+ return leftScope < rightScope
300
+ case '>=':
301
+ return leftScope >= rightScope
302
+ case '<=':
303
+ return leftScope <= rightScope
304
+ default:
305
+ throw new Error('Invalid operator')
306
+ }
307
+ }
308
+
309
+ const logger = Object.assign(
310
+ (label: string) =>
311
+ createFactoryInjectable({
312
+ dependencies: { logger },
313
+ scope: Scope.Global,
314
+ factory: ({ logger }) => logger.child({ $label: label }),
315
+ }),
316
+ createLazyInjectable<Logger>(Scope.Global, 'Logger'),
317
+ ) as unknown as ((
318
+ label: string,
319
+ ) => FactoryInjectable<Logger, { logger: LazyInjectable<Logger> }>) &
320
+ LazyInjectable<Logger>
321
+
322
+ const inject = createLazyInjectable<InjectFn>(Scope.Global, 'Inject function')
323
+ const dispose = createLazyInjectable<DisposeFn>(
324
+ Scope.Global,
325
+ 'Dispose function',
326
+ )
327
+
328
+ function resolveInjectableScope(
329
+ isDefaultScope: boolean,
330
+ injectable: AnyInjectable,
331
+ ) {
332
+ const actualScope = getInjectableScope(injectable)
333
+ if (!isDefaultScope && compareScope(actualScope, '>', injectable.scope))
334
+ throw new Error(
335
+ `Invalid scope ${injectable.scope} for an injectable: dependencies have stricter scope - ${actualScope}`,
336
+ )
337
+ return actualScope
338
+ }
339
+
340
+ export const CoreInjectables = { logger, inject, dispose }
341
+
342
+ export type Injection<
343
+ T extends AnyInjectable<any, any> = AnyInjectable<any, any>,
344
+ > = {
345
+ token: T
346
+ value: T extends AnyInjectable<infer R, Scope> ? R | AnyInjectable<R> : never
347
+ }
348
+
349
+ export const provide = <
350
+ T extends AnyInjectable<any, any>,
351
+ V extends T extends AnyInjectable<infer R, Scope>
352
+ ? R | AnyInjectable<R>
353
+ : never,
354
+ >(
355
+ token: T,
356
+ value: V,
357
+ ): Injection<T> => {
358
+ return { token, value }
359
+ }
package/src/logger.ts ADDED
@@ -0,0 +1,103 @@
1
+ import { threadId } from 'node:worker_threads'
2
+
3
+ import type * as pinoType from 'pino'
4
+ import { pino, stdTimeFunctions } from 'pino'
5
+ import { build as pretty } from 'pino-pretty'
6
+
7
+ export type { StreamEntry } from 'pino'
8
+ export type Logger = pinoType.Logger
9
+ export type LoggerOptions = pinoType.LoggerOptions
10
+ export type LoggerChildOptions = pinoType.ChildLoggerOptions
11
+ export type LoggingOptions = {
12
+ destinations?: Array<
13
+ pinoType.DestinationStream | pinoType.StreamEntry<pinoType.Level>
14
+ >
15
+ pinoOptions?: LoggerOptions
16
+ }
17
+
18
+ import { errWithCause } from 'pino-std-serializers'
19
+
20
+ // TODO: use node:util inspect
21
+ const bg = (value, color) => `\x1b[${color}m${value}\x1b[0m`
22
+ const fg = (value, color) => `\x1b[38;5;${color}m${value}\x1b[0m`
23
+
24
+ const levelColors = {
25
+ 10: 100,
26
+ 20: 102,
27
+ 30: 106,
28
+ 40: 104,
29
+ 50: 101,
30
+ 60: 105,
31
+ [Number.POSITIVE_INFINITY]: 0,
32
+ }
33
+ const messageColors = {
34
+ 10: 0,
35
+ 20: 2,
36
+ 30: 6,
37
+ 40: 4,
38
+ 50: 1,
39
+ 60: 5,
40
+ [Number.POSITIVE_INFINITY]: 0,
41
+ }
42
+
43
+ const levelLabels = {
44
+ 10: ' TRACE ',
45
+ 20: ' DEBUG ',
46
+ 30: ' INFO ',
47
+ 40: ' WARN ',
48
+ 50: ' ERROR ',
49
+ 60: ' FATAL ',
50
+ [Number.POSITIVE_INFINITY]: 'SILENT',
51
+ }
52
+
53
+ export const createLogger = (options: LoggingOptions = {}, $lable: string) => {
54
+ let { destinations, pinoOptions } = options
55
+
56
+ if (!destinations || !destinations?.length) {
57
+ destinations = [
58
+ createConsolePrettyDestination(
59
+ (options.pinoOptions?.level || 'info') as pinoType.Level,
60
+ ),
61
+ ]
62
+ }
63
+
64
+ const lowestLevelValue = destinations!.reduce(
65
+ (acc, destination) =>
66
+ Math.min(
67
+ acc,
68
+ 'stream' in destination
69
+ ? pino.levels.values[destination.level!]
70
+ : Number.POSITIVE_INFINITY,
71
+ ),
72
+ Number.POSITIVE_INFINITY,
73
+ )
74
+ const level = pino.levels.labels[lowestLevelValue]
75
+ const serializers = { ...pinoOptions?.serializers, err: errWithCause }
76
+
77
+ return pino(
78
+ { timestamp: stdTimeFunctions.isoTime, ...pinoOptions, level, serializers },
79
+ pino.multistream(destinations!),
80
+ ).child({ $lable, $threadId: threadId })
81
+ }
82
+
83
+ export const createConsolePrettyDestination = (
84
+ level: pinoType.Level,
85
+ sync = true,
86
+ ): pinoType.StreamEntry => ({
87
+ level,
88
+ stream: pretty({
89
+ colorize: true,
90
+ ignore: 'hostname,$lable,$threadId',
91
+ errorLikeObjectKeys: ['err', 'error', 'cause'],
92
+ messageFormat: (log, messageKey) => {
93
+ const group = fg(`[${log.$lable}]`, 11)
94
+ const msg = fg(log[messageKey], messageColors[log.level as number])
95
+ const thread = fg(`(Thread-${log.$threadId})`, 89)
96
+ return `\x1b[0m${thread} ${group} ${msg}`
97
+ },
98
+ customPrettifiers: {
99
+ level: (level: any) => bg(levelLabels[level], levelColors[level]),
100
+ },
101
+ sync,
102
+ }),
103
+ })
@@ -0,0 +1,24 @@
1
+ import { kMetadata } from './constants.ts'
2
+
3
+ export type Metadata<T = any> = { key: MetadataKey<T>; value: T }
4
+
5
+ export type MetadataKey<T = any> = {
6
+ [kMetadata]: string
7
+ as(value: T): Metadata<T>
8
+ }
9
+
10
+ export const createMetadataKey = <T>(key: string): MetadataKey<T> => {
11
+ const metadataKey = {
12
+ [kMetadata]: key,
13
+ as(value: T) {
14
+ return { key: metadataKey, value }
15
+ },
16
+ }
17
+ return metadataKey
18
+ }
19
+
20
+ export class MetadataStore extends Map<MetadataKey, Metadata> {
21
+ get<T>(key: MetadataKey<T>): T | undefined {
22
+ return super.get(key) as T | undefined
23
+ }
24
+ }
package/src/plugin.ts ADDED
@@ -0,0 +1,16 @@
1
+ import type { Async } from '@nmtjs/common'
2
+
3
+ import { kPlugin } from './constants.ts'
4
+
5
+ export interface Plugin<Type = void, Context = unknown> {
6
+ name: string
7
+ factory: (context: Context) => Async<Type>
8
+ [kPlugin]: any
9
+ }
10
+
11
+ export const createPlugin = <Type = void, Context = unknown>(
12
+ name: string,
13
+ factory: Plugin<Type, Context>['factory'],
14
+ ): Plugin<Type, Context> => ({ name, factory, [kPlugin]: true })
15
+
16
+ export const isPlugin = (value: any): value is Plugin => kPlugin in value
package/src/types.ts ADDED
@@ -0,0 +1,3 @@
1
+ export type { Pattern } from '@nmtjs/common'
2
+
3
+ export type HookTypes = Record<string | symbol, any>
@@ -0,0 +1,31 @@
1
+ export { match } from '@nmtjs/common'
2
+
3
+ export function isJsFile(name: string) {
4
+ if (name.endsWith('.d.ts')) return false
5
+ const leading = name.split('.').slice(1)
6
+ const ext = leading.join('.')
7
+ return ['js', 'mjs', 'cjs', 'ts', 'mts', 'cts'].includes(ext)
8
+ }
9
+
10
+ export function pick<
11
+ T extends object,
12
+ K extends {
13
+ [KK in keyof T as T[KK] extends (...args: any[]) => any ? never : KK]?: true
14
+ },
15
+ >(
16
+ obj: T,
17
+ keys: K,
18
+ ): Pick<
19
+ T,
20
+ keyof {
21
+ [KK in keyof K as K[KK] extends true ? KK : never]: K[KK]
22
+ }
23
+ > {
24
+ const result = {} as any
25
+ for (const key in keys) {
26
+ if (key in obj) {
27
+ result[key] = obj[key as unknown as keyof typeof obj]
28
+ }
29
+ }
30
+ return result
31
+ }
@@ -0,0 +1,3 @@
1
+ export * from './functions.ts'
2
+ export * from './pool.ts'
3
+ export * from './semaphore.ts'
@@ -0,0 +1,111 @@
1
+ import type { Callback } from '@nmtjs/common'
2
+
3
+ interface PoolOptions {
4
+ timeout?: number
5
+ }
6
+
7
+ interface PoolQueueItem {
8
+ resolve?: Callback
9
+ timer?: ReturnType<typeof setTimeout>
10
+ }
11
+
12
+ export class PoolError extends Error {}
13
+
14
+ // Fixed pool from https://github.com/metarhia/metautil
15
+ export class Pool<T = unknown> {
16
+ #items: Array<T> = []
17
+ #free: Array<boolean> = []
18
+ #queue: PoolQueueItem[] = []
19
+ #current: number = 0
20
+ #size: number = 0
21
+ #available: number = 0
22
+
23
+ constructor(private readonly options: PoolOptions = {}) {}
24
+
25
+ add(item: T) {
26
+ if (this.#items.includes(item)) throw new PoolError('Item already exists')
27
+ this.#size++
28
+ this.#available++
29
+ this.#items.push(item)
30
+ this.#free.push(true)
31
+ }
32
+
33
+ remove(item: T) {
34
+ if (this.#size === 0) throw new PoolError('Pool is empty')
35
+ const index = this.#items.indexOf(item)
36
+ if (index < 0) throw new PoolError('Item is not in the pool')
37
+ const isCaptured = this.isFree(item)
38
+ if (isCaptured) this.#available--
39
+ this.#size--
40
+ this.#current--
41
+ this.#items.splice(index, 1)
42
+ this.#free.splice(index, 1)
43
+ }
44
+
45
+ capture(timeout = this.options.timeout) {
46
+ return this.next(true, timeout)
47
+ }
48
+
49
+ async next(exclusive = false, timeout = this.options.timeout): Promise<T> {
50
+ if (this.#size === 0) throw new PoolError('Pool is empty')
51
+ if (this.#available === 0) {
52
+ return new Promise((resolve, reject) => {
53
+ const waiting: PoolQueueItem = {
54
+ resolve: (item: T) => {
55
+ if (exclusive) this.#capture(item)
56
+ resolve(item)
57
+ },
58
+ timer: undefined,
59
+ }
60
+ if (timeout) {
61
+ waiting.timer = setTimeout(() => {
62
+ waiting.resolve = undefined
63
+ this.#queue.shift()
64
+ reject(new PoolError('Next item timeout'))
65
+ }, timeout)
66
+ }
67
+ this.#queue.push(waiting)
68
+ })
69
+ }
70
+ let item: T | undefined
71
+ let free = false
72
+ do {
73
+ item = this.#items[this.#current]
74
+ free = this.#free[this.#current]
75
+ this.#current++
76
+ if (this.#current >= this.#size) this.#current = 0
77
+ } while (typeof item === 'undefined' || !free)
78
+ if (exclusive) this.#capture(item)
79
+ return item
80
+ }
81
+
82
+ release(item: T) {
83
+ const index = this.#items.indexOf(item)
84
+ if (index < 0) throw new PoolError('Unexpected item')
85
+ if (this.#free[index])
86
+ throw new PoolError('Unable to release not captured item')
87
+ this.#free[index] = true
88
+ this.#available++
89
+ if (this.#queue.length > 0) {
90
+ const { resolve, timer } = this.#queue.shift()!
91
+ clearTimeout(timer)
92
+ if (resolve) setTimeout(resolve, 0, item)
93
+ }
94
+ }
95
+
96
+ isFree(item: T) {
97
+ const index = this.#items.indexOf(item)
98
+ if (index < 0) return false
99
+ return this.#free[index]
100
+ }
101
+
102
+ get items() {
103
+ return [...this.#items]
104
+ }
105
+
106
+ #capture(item: T) {
107
+ const index = this.#items.indexOf(item)
108
+ this.#free[index] = false
109
+ this.#available--
110
+ }
111
+ }