@navios/di 0.9.0 → 0.9.1
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/CHANGELOG.md +15 -0
- package/lib/testing/index.cjs +27 -14
- package/lib/testing/index.cjs.map +1 -1
- package/lib/testing/index.d.cts +6 -5
- package/lib/testing/index.d.cts.map +1 -1
- package/lib/testing/index.d.mts +6 -5
- package/lib/testing/index.d.mts.map +1 -1
- package/lib/testing/index.mjs +28 -15
- package/lib/testing/index.mjs.map +1 -1
- package/package.json +1 -1
- package/src/__tests__/unit-test-container.spec.mts +128 -4
- package/src/testing/test-container.mts +23 -5
- package/src/testing/types.mts +8 -2
- package/src/testing/unit-test-container.mts +120 -28
package/lib/testing/index.mjs
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { A as Registry, D as DIError, I as InjectableType, L as InjectableScope, P as InjectionToken, j as globalRegistry, l as defaultInjectors, s as getInjectableToken, t as _Container } from "../container-8-z89TyQ.mjs";
|
|
1
|
+
import { A as Registry, D as DIError, I as InjectableType, L as InjectableScope, M as BoundInjectionToken, P as InjectionToken, j as globalRegistry, l as defaultInjectors, s as getInjectableToken, t as _Container } from "../container-8-z89TyQ.mjs";
|
|
2
2
|
|
|
3
3
|
//#region src/testing/test-container.mts
|
|
4
4
|
/**
|
|
@@ -61,6 +61,7 @@ import { A as Registry, D as DIError, I as InjectableType, L as InjectableScope,
|
|
|
61
61
|
* container.bind(UserService).toValue(mockUserService)
|
|
62
62
|
* container.bind(DatabaseToken).toClass(MockDatabase)
|
|
63
63
|
* container.bind(ConfigToken).toFactory(() => ({ apiKey: 'test' }))
|
|
64
|
+
* container.bind(BOUND_CONFIG_TOKEN).toValue(overrideValue)
|
|
64
65
|
* ```
|
|
65
66
|
*/ bind(token) {
|
|
66
67
|
const realToken = this.resolveToken(token);
|
|
@@ -277,19 +278,22 @@ import { A as Registry, D as DIError, I as InjectableType, L as InjectableScope,
|
|
|
277
278
|
}
|
|
278
279
|
resolveToken(token) {
|
|
279
280
|
if (typeof token === "function") return getInjectableToken(token);
|
|
281
|
+
if (token instanceof BoundInjectionToken) return token.token;
|
|
280
282
|
return token;
|
|
281
283
|
}
|
|
282
284
|
registerValueBinding(token, value) {
|
|
283
285
|
const ValueHolder = class {
|
|
284
|
-
|
|
286
|
+
create() {
|
|
287
|
+
return value;
|
|
288
|
+
}
|
|
285
289
|
};
|
|
286
|
-
this.testRegistry.set(token, InjectableScope.Singleton, ValueHolder, InjectableType.
|
|
290
|
+
this.testRegistry.set(token, InjectableScope.Singleton, ValueHolder, InjectableType.Factory, 1e3);
|
|
287
291
|
const instanceName = this.getNameResolver().generateInstanceName(token, void 0, void 0, InjectableScope.Singleton);
|
|
288
292
|
this.getStorage().storeInstance(instanceName, value);
|
|
289
293
|
this.recordLifecycleEvent(token, "created", instanceName);
|
|
290
294
|
}
|
|
291
295
|
registerClassBinding(token, cls) {
|
|
292
|
-
this.testRegistry.set(token, InjectableScope.Singleton, cls, InjectableType.Class);
|
|
296
|
+
this.testRegistry.set(token, InjectableScope.Singleton, cls, InjectableType.Class, 1e3);
|
|
293
297
|
}
|
|
294
298
|
registerFactoryBinding(token, factory) {
|
|
295
299
|
const FactoryWrapper = class {
|
|
@@ -298,7 +302,7 @@ import { A as Registry, D as DIError, I as InjectableType, L as InjectableScope,
|
|
|
298
302
|
return await factory();
|
|
299
303
|
}
|
|
300
304
|
};
|
|
301
|
-
this.testRegistry.set(token, InjectableScope.Singleton, FactoryWrapper, InjectableType.Factory);
|
|
305
|
+
this.testRegistry.set(token, InjectableScope.Singleton, FactoryWrapper, InjectableType.Factory, 1e3);
|
|
302
306
|
}
|
|
303
307
|
argsMatch(actual, expected) {
|
|
304
308
|
if (actual.length !== expected.length) return false;
|
|
@@ -416,14 +420,17 @@ import { A as Registry, D as DIError, I as InjectableType, L as InjectableScope,
|
|
|
416
420
|
/**
|
|
417
421
|
* Override get to wrap instances in tracking proxies.
|
|
418
422
|
*/ async get(token, args) {
|
|
423
|
+
const tokenId = token instanceof BoundInjectionToken ? token.id : void 0;
|
|
419
424
|
const realToken = this.resolveToken(token);
|
|
420
|
-
if (!this.registeredTokenIds.has(realToken.id)) {
|
|
425
|
+
if (!(tokenId && this.registeredTokenIds.has(tokenId) || this.registeredTokenIds.has(realToken.id))) {
|
|
421
426
|
if (!this.allowUnregistered) throw DIError.factoryNotFound(`${realToken.toString()} is not in the providers list. Add it to providers or enable allowUnregistered.`);
|
|
422
|
-
|
|
423
|
-
|
|
427
|
+
const idToCheck = tokenId || realToken.id;
|
|
428
|
+
if (!this.autoMockedTokenIds.has(idToCheck)) this.autoMockedTokenIds.add(idToCheck);
|
|
429
|
+
return createAutoMockProxy(idToCheck);
|
|
424
430
|
}
|
|
425
431
|
const instance = await super.get(token, args);
|
|
426
|
-
|
|
432
|
+
const trackingId = tokenId || realToken.id;
|
|
433
|
+
if (instance && typeof instance === "object") return createTrackingProxy(instance, trackingId, this.methodCalls);
|
|
427
434
|
return instance;
|
|
428
435
|
}
|
|
429
436
|
/**
|
|
@@ -559,27 +566,33 @@ import { A as Registry, D as DIError, I as InjectableType, L as InjectableScope,
|
|
|
559
566
|
} catch {
|
|
560
567
|
return InjectionToken.create(token);
|
|
561
568
|
}
|
|
569
|
+
if (token instanceof BoundInjectionToken) return token.token;
|
|
562
570
|
return token;
|
|
563
571
|
}
|
|
564
572
|
registerProvider(provider) {
|
|
565
|
-
const
|
|
573
|
+
const providerToken = provider.token;
|
|
574
|
+
const realToken = this.resolveToken(providerToken);
|
|
566
575
|
this.registeredTokenIds.add(realToken.id);
|
|
576
|
+
if (providerToken instanceof BoundInjectionToken) this.registeredTokenIds.add(providerToken.id);
|
|
567
577
|
if (provider.useValue !== void 0) this.registerValueBinding(realToken, provider.useValue);
|
|
568
578
|
else if (provider.useClass) this.registerClassBinding(realToken, provider.useClass);
|
|
569
579
|
else if (provider.useFactory) this.registerFactoryBinding(realToken, provider.useFactory);
|
|
570
|
-
else if (typeof provider.token === "function") this.testRegistry.set(realToken, InjectableScope.Singleton, provider.token, InjectableType.Class);
|
|
580
|
+
else if (typeof provider.token === "function") this.testRegistry.set(realToken, InjectableScope.Singleton, provider.token, InjectableType.Class, 1e3);
|
|
581
|
+
else if (providerToken instanceof BoundInjectionToken) this.registerValueBinding(realToken, providerToken.value);
|
|
571
582
|
}
|
|
572
583
|
registerValueBinding(token, value) {
|
|
573
584
|
const ValueHolder = class {
|
|
574
|
-
|
|
585
|
+
create() {
|
|
586
|
+
return value;
|
|
587
|
+
}
|
|
575
588
|
};
|
|
576
|
-
this.testRegistry.set(token, InjectableScope.Singleton, ValueHolder, InjectableType.
|
|
589
|
+
this.testRegistry.set(token, InjectableScope.Singleton, ValueHolder, InjectableType.Factory, 1e3);
|
|
577
590
|
const instanceName = this.getNameResolver().generateInstanceName(token, void 0, void 0, InjectableScope.Singleton);
|
|
578
591
|
this.getStorage().storeInstance(instanceName, value);
|
|
579
592
|
this.recordLifecycleEvent(token, "created", instanceName);
|
|
580
593
|
}
|
|
581
594
|
registerClassBinding(token, cls) {
|
|
582
|
-
this.testRegistry.set(token, InjectableScope.Singleton, cls, InjectableType.Class);
|
|
595
|
+
this.testRegistry.set(token, InjectableScope.Singleton, cls, InjectableType.Class, 1e3);
|
|
583
596
|
}
|
|
584
597
|
registerFactoryBinding(token, factory) {
|
|
585
598
|
const FactoryWrapper = class {
|
|
@@ -588,7 +601,7 @@ import { A as Registry, D as DIError, I as InjectableType, L as InjectableScope,
|
|
|
588
601
|
return await factory();
|
|
589
602
|
}
|
|
590
603
|
};
|
|
591
|
-
this.testRegistry.set(token, InjectableScope.Singleton, FactoryWrapper, InjectableType.Factory);
|
|
604
|
+
this.testRegistry.set(token, InjectableScope.Singleton, FactoryWrapper, InjectableType.Factory, 1e3);
|
|
592
605
|
}
|
|
593
606
|
argsMatch(actual, expected) {
|
|
594
607
|
if (actual.length !== expected.length) return false;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.mjs","names":["Container","InjectableScope","InjectableType","globalRegistry","Registry","getInjectableToken","defaultInjectors","TestContainer","testRegistry","methodCalls","Map","lifecycleEvents","instanceCounts","boundTokens","Set","options","parentRegistry","logger","bind","token","realToken","resolveToken","tokenId","id","toValue","value","add","registerValueBinding","toClass","cls","registerClassBinding","toFactory","factory","registerFactoryBinding","clear","dispose","expectResolved","storage","getStorage","names","getAllNames","found","some","name","includes","Error","toString","expectNotResolved","expectSingleton","registry","getRegistry","has","record","get","scope","Singleton","expectTransient","Transient","expectRequestScoped","Request","expectSameInstance","instance1","instance2","expectDifferentInstances","expectInitialized","events","initialized","e","event","expectDestroyed","destroyed","expectNotDestroyed","recordMethodCall","method","args","result","error","calls","push","timestamp","Date","now","set","recordLifecycleEvent","instanceName","count","expectCalled","c","expectCalledWith","expectedArgs","argsMatch","JSON","stringify","expectCallCount","actualCount","filter","length","getMethodCalls","getServiceStats","instanceCount","clearMethodCalls","getDependencyGraph","nodes","rootTokens","forEach","holder","tokenMatch","match","dependencies","Array","from","deps","dependents","findDependents","getSimplifiedDependencyGraph","graph","dep","depTokenMatch","depTokenId","key","Object","keys","sort","ValueHolder","instance","Class","nameResolver","getNameResolver","generateInstanceName","undefined","storeInstance","FactoryWrapper","create","Factory","actual","expected","every","arg","index","exp","Container","InjectableScope","InjectableType","DIError","InjectionToken","Registry","getInjectableToken","defaultInjectors","createTrackingProxy","target","tokenId","methodCalls","Proxy","get","obj","prop","value","Reflect","args","calls","record","method","timestamp","Date","now","result","apply","undefined","Promise","then","res","push","set","catch","err","error","createAutoMockProxy","_","Error","UnitTestContainer","testRegistry","Map","lifecycleEvents","instanceCounts","registeredTokenIds","Set","autoMockedTokenIds","allowUnregistered","options","logger","provider","providers","registerProvider","enableAutoMocking","disableAutoMocking","token","realToken","resolveToken","has","id","factoryNotFound","toString","add","instance","clear","dispose","expectResolved","storage","getStorage","names","getAllNames","found","some","name","includes","expectNotResolved","expectAutoMocked","expectNotAutoMocked","recordLifecycleEvent","event","instanceName","events","count","expectInitialized","initialized","e","expectDestroyed","destroyed","expectNotDestroyed","expectCalled","c","expectNotCalled","expectCalledWith","expectedArgs","argsMatch","filter","actualArgs","map","JSON","stringify","join","expectCallCount","actualCount","length","getMethodCalls","getServiceStats","instanceCount","clearMethodCalls","getRegisteredTokenIds","getAutoMockedTokenIds","create","useValue","registerValueBinding","useClass","registerClassBinding","useFactory","registerFactoryBinding","Singleton","Class","ValueHolder","nameResolver","getNameResolver","generateInstanceName","storeInstance","cls","factory","FactoryWrapper","Factory","actual","expected","every","arg","index","exp"],"sources":["../../src/testing/test-container.mts","../../src/testing/unit-test-container.mts"],"sourcesContent":["import type {\n BindingBuilder,\n DependencyGraph,\n DependencyNode,\n LifecycleRecord,\n MethodCallRecord,\n MockServiceStats,\n TestContainerOptions,\n} from './types.mjs'\n\nimport { Container } from '../container/container.mjs'\nimport { InjectableScope, InjectableType } from '../enums/index.mjs'\nimport { InjectionToken } from '../token/injection-token.mjs'\nimport { globalRegistry, Registry } from '../token/registry.mjs'\nimport { getInjectableToken } from '../utils/get-injectable-token.mjs'\nimport { defaultInjectors } from '../utils/index.mjs'\n\ntype AnyToken = InjectionToken<any, any> | (new (...args: any[]) => any)\n\n/**\n * TestContainer extends Container with testing utilities.\n *\n * Provides simple value/class binding for integration/e2e tests,\n * plus assertion helpers and dependency graph inspection.\n *\n * @example\n * ```ts\n * const container = new TestContainer()\n *\n * // Bind mock values\n * container.bind(DatabaseToken).toValue(mockDatabase)\n * container.bind(UserService).toClass(MockUserService)\n *\n * // Use container normally\n * const service = await container.get(MyService)\n *\n * // Assert on container state\n * container.expectResolved(MyService)\n * container.expectSingleton(MyService)\n * ```\n */\nexport class TestContainer extends Container {\n private readonly testRegistry: Registry\n private readonly methodCalls = new Map<string, MethodCallRecord[]>()\n private readonly lifecycleEvents = new Map<string, LifecycleRecord[]>()\n private readonly instanceCounts = new Map<string, number>()\n private readonly boundTokens = new Set<string>()\n\n /**\n * Creates a new TestContainer.\n *\n * @param options - Configuration options\n * @param options.parentRegistry - Parent registry. Defaults to globalRegistry.\n * Pass `null` for a completely isolated container.\n * @param options.logger - Optional logger for debugging.\n *\n * @example\n * ```ts\n * // Uses globalRegistry as parent (default)\n * const container = new TestContainer()\n *\n * // Isolated container (no access to @Injectable classes)\n * const isolated = new TestContainer({ parentRegistry: null })\n *\n * // Custom parent registry\n * const custom = new TestContainer({ parentRegistry: myRegistry })\n * ```\n */\n constructor(options: TestContainerOptions = {}) {\n const { parentRegistry = globalRegistry, logger = null } = options\n const testRegistry = parentRegistry\n ? new Registry(parentRegistry)\n : new Registry()\n super(testRegistry, logger, defaultInjectors)\n this.testRegistry = testRegistry\n }\n\n // ============================================================================\n // BINDING API\n // ============================================================================\n\n /**\n * Creates a binding builder for the given token.\n *\n * @example\n * ```ts\n * container.bind(UserService).toValue(mockUserService)\n * container.bind(DatabaseToken).toClass(MockDatabase)\n * container.bind(ConfigToken).toFactory(() => ({ apiKey: 'test' }))\n * ```\n */\n bind<T>(\n token: InjectionToken<T, any> | (new (...args: any[]) => T),\n ): BindingBuilder<T> {\n const realToken = this.resolveToken(token)\n const tokenId = realToken.id\n\n return {\n toValue: (value: T) => {\n this.boundTokens.add(tokenId)\n this.registerValueBinding(realToken, value)\n },\n toClass: <C extends new (...args: any[]) => T>(cls: C) => {\n this.boundTokens.add(tokenId)\n this.registerClassBinding(realToken, cls)\n },\n toFactory: (factory: () => T | Promise<T>) => {\n this.boundTokens.add(tokenId)\n this.registerFactoryBinding(realToken, factory)\n },\n }\n }\n\n /**\n * Clears all bindings and resets container state.\n */\n async clear(): Promise<void> {\n await this.dispose()\n this.methodCalls.clear()\n this.lifecycleEvents.clear()\n this.instanceCounts.clear()\n this.boundTokens.clear()\n }\n\n // ============================================================================\n // ASSERTION HELPERS\n // ============================================================================\n\n /**\n * Asserts that a service has been resolved at least once.\n */\n expectResolved(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const storage = this.getStorage()\n const names = storage.getAllNames()\n const found = names.some((name) => name.includes(realToken.id))\n\n if (!found) {\n throw new Error(\n `Expected ${realToken.toString()} to be resolved, but it was not`,\n )\n }\n }\n\n /**\n * Asserts that a service has NOT been resolved.\n */\n expectNotResolved(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const storage = this.getStorage()\n const names = storage.getAllNames()\n const found = names.some((name) => name.includes(realToken.id))\n\n if (found) {\n throw new Error(\n `Expected ${realToken.toString()} to NOT be resolved, but it was`,\n )\n }\n }\n\n /**\n * Asserts that a service is registered as singleton scope.\n */\n expectSingleton(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const registry = this.getRegistry()\n\n if (!registry.has(realToken)) {\n throw new Error(\n `Expected ${realToken.toString()} to be registered, but it was not`,\n )\n }\n\n const record = registry.get(realToken)\n if (record.scope !== InjectableScope.Singleton) {\n throw new Error(\n `Expected ${realToken.toString()} to be Singleton scope, but it was ${record.scope}`,\n )\n }\n }\n\n /**\n * Asserts that a service is registered as transient scope.\n */\n expectTransient(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const registry = this.getRegistry()\n\n if (!registry.has(realToken)) {\n throw new Error(\n `Expected ${realToken.toString()} to be registered, but it was not`,\n )\n }\n\n const record = registry.get(realToken)\n if (record.scope !== InjectableScope.Transient) {\n throw new Error(\n `Expected ${realToken.toString()} to be Transient scope, but it was ${record.scope}`,\n )\n }\n }\n\n /**\n * Asserts that a service is registered as request scope.\n */\n expectRequestScoped(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const registry = this.getRegistry()\n\n if (!registry.has(realToken)) {\n throw new Error(\n `Expected ${realToken.toString()} to be registered, but it was not`,\n )\n }\n\n const record = registry.get(realToken)\n if (record.scope !== InjectableScope.Request) {\n throw new Error(\n `Expected ${realToken.toString()} to be Request scope, but it was ${record.scope}`,\n )\n }\n }\n\n /**\n * Asserts that two service resolutions return the same instance.\n */\n async expectSameInstance(token: AnyToken): Promise<void> {\n const instance1 = await this.get(token as any)\n const instance2 = await this.get(token as any)\n\n if (instance1 !== instance2) {\n const realToken = this.resolveToken(token)\n throw new Error(\n `Expected ${realToken.toString()} to return same instance, but got different instances`,\n )\n }\n }\n\n /**\n * Asserts that two service resolutions return different instances.\n */\n async expectDifferentInstances(token: AnyToken): Promise<void> {\n const instance1 = await this.get(token as any)\n const instance2 = await this.get(token as any)\n\n if (instance1 === instance2) {\n const realToken = this.resolveToken(token)\n throw new Error(\n `Expected ${realToken.toString()} to return different instances, but got same instance`,\n )\n }\n }\n\n // ============================================================================\n // LIFECYCLE ASSERTIONS\n // ============================================================================\n\n /**\n * Asserts that a service's onServiceInit was called.\n */\n expectInitialized(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n const initialized = events.some((e) => e.event === 'initialized')\n\n if (!initialized) {\n throw new Error(\n `Expected ${realToken.toString()} to be initialized, but onServiceInit was not called`,\n )\n }\n }\n\n /**\n * Asserts that a service's onServiceDestroy was called.\n */\n expectDestroyed(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n const destroyed = events.some((e) => e.event === 'destroyed')\n\n if (!destroyed) {\n throw new Error(\n `Expected ${realToken.toString()} to be destroyed, but onServiceDestroy was not called`,\n )\n }\n }\n\n /**\n * Asserts that a service has NOT been destroyed.\n */\n expectNotDestroyed(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n const destroyed = events.some((e) => e.event === 'destroyed')\n\n if (destroyed) {\n throw new Error(\n `Expected ${realToken.toString()} to NOT be destroyed, but onServiceDestroy was called`,\n )\n }\n }\n\n // ============================================================================\n // CALL TRACKING\n // ============================================================================\n\n /**\n * Records a method call for tracking.\n * Call this from your mock implementations.\n */\n recordMethodCall(\n token: AnyToken,\n method: string,\n args: unknown[],\n result?: unknown,\n error?: Error,\n ): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n calls.push({\n method,\n args,\n timestamp: Date.now(),\n result,\n error,\n })\n this.methodCalls.set(realToken.id, calls)\n }\n\n /**\n * Records a lifecycle event for tracking.\n */\n recordLifecycleEvent(\n token: AnyToken,\n event: 'created' | 'initialized' | 'destroyed',\n instanceName: string,\n ): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n events.push({\n event,\n timestamp: Date.now(),\n instanceName,\n })\n this.lifecycleEvents.set(realToken.id, events)\n\n if (event === 'created') {\n const count = this.instanceCounts.get(realToken.id) || 0\n this.instanceCounts.set(realToken.id, count + 1)\n }\n }\n\n /**\n * Asserts that a method was called on a service.\n */\n expectCalled(token: AnyToken, method: string): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const found = calls.some((c) => c.method === method)\n\n if (!found) {\n throw new Error(\n `Expected ${realToken.toString()}.${method}() to be called, but it was not`,\n )\n }\n }\n\n /**\n * Asserts that a method was called with specific arguments.\n */\n expectCalledWith(\n token: AnyToken,\n method: string,\n expectedArgs: unknown[],\n ): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const found = calls.some(\n (c) => c.method === method && this.argsMatch(c.args, expectedArgs),\n )\n\n if (!found) {\n throw new Error(\n `Expected ${realToken.toString()}.${method}() to be called with ${JSON.stringify(expectedArgs)}, but it was not`,\n )\n }\n }\n\n /**\n * Asserts that a method was called a specific number of times.\n */\n expectCallCount(token: AnyToken, method: string, count: number): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const actualCount = calls.filter((c) => c.method === method).length\n\n if (actualCount !== count) {\n throw new Error(\n `Expected ${realToken.toString()}.${method}() to be called ${count} times, but was called ${actualCount} times`,\n )\n }\n }\n\n /**\n * Gets all recorded method calls for a service.\n */\n getMethodCalls(token: AnyToken): MethodCallRecord[] {\n const realToken = this.resolveToken(token)\n return this.methodCalls.get(realToken.id) || []\n }\n\n /**\n * Gets statistics about a mocked service.\n */\n getServiceStats(token: AnyToken): MockServiceStats {\n const realToken = this.resolveToken(token)\n return {\n instanceCount: this.instanceCounts.get(realToken.id) || 0,\n methodCalls: this.methodCalls.get(realToken.id) || [],\n lifecycleEvents: this.lifecycleEvents.get(realToken.id) || [],\n }\n }\n\n /**\n * Clears all recorded method calls.\n */\n clearMethodCalls(): void {\n this.methodCalls.clear()\n }\n\n // ============================================================================\n // DEPENDENCY GRAPH\n // ============================================================================\n\n /**\n * Gets the dependency graph for snapshot testing.\n * Returns a serializable structure that can be used with vitest snapshots.\n */\n getDependencyGraph(): DependencyGraph {\n const storage = this.getStorage()\n const nodes: Record<string, DependencyNode> = {}\n const rootTokens: string[] = []\n\n storage.forEach((name, holder) => {\n const tokenMatch = name.match(/^([^:]+)/)\n const tokenId = tokenMatch ? tokenMatch[1] : name\n\n nodes[name] = {\n token: tokenId,\n instanceName: name,\n scope: holder.scope,\n dependencies: Array.from(holder.deps),\n dependents: storage.findDependents(name),\n }\n\n // Root tokens have no dependents\n if (storage.findDependents(name).length === 0) {\n rootTokens.push(name)\n }\n })\n\n return { nodes, rootTokens }\n }\n\n /**\n * Gets a simplified dependency graph showing only token relationships.\n * Useful for cleaner snapshot comparisons.\n */\n getSimplifiedDependencyGraph(): Record<string, string[]> {\n const storage = this.getStorage()\n const graph: Record<string, string[]> = {}\n\n storage.forEach((name, holder) => {\n const tokenMatch = name.match(/^([^:]+)/)\n const tokenId = tokenMatch ? tokenMatch[1] : name\n\n if (!graph[tokenId]) {\n graph[tokenId] = []\n }\n\n for (const dep of holder.deps) {\n const depTokenMatch = dep.match(/^([^:]+)/)\n const depTokenId = depTokenMatch ? depTokenMatch[1] : dep\n if (!graph[tokenId].includes(depTokenId)) {\n graph[tokenId].push(depTokenId)\n }\n }\n })\n\n // Sort for consistent snapshots\n for (const key of Object.keys(graph)) {\n graph[key].sort()\n }\n\n return graph\n }\n\n // ============================================================================\n // INTERNAL HELPERS\n // ============================================================================\n\n private resolveToken(token: AnyToken): InjectionToken<any, any> {\n if (typeof token === 'function') {\n return getInjectableToken(token)\n }\n return token\n }\n\n private registerValueBinding<T>(\n token: InjectionToken<T, any>,\n value: T,\n ): void {\n // Create a simple class that returns the value\n const ValueHolder = class {\n static instance = value\n }\n\n this.testRegistry.set(\n token,\n InjectableScope.Singleton,\n ValueHolder,\n InjectableType.Class,\n )\n\n // Store the instance directly\n const nameResolver = this.getNameResolver()\n const instanceName = nameResolver.generateInstanceName(\n token,\n undefined,\n undefined,\n InjectableScope.Singleton,\n )\n this.getStorage().storeInstance(instanceName, value)\n this.recordLifecycleEvent(token, 'created', instanceName)\n }\n\n private registerClassBinding<T>(\n token: InjectionToken<T, any>,\n cls: new (...args: any[]) => T,\n ): void {\n this.testRegistry.set(\n token,\n InjectableScope.Singleton,\n cls,\n InjectableType.Class,\n )\n }\n\n private registerFactoryBinding<T>(\n token: InjectionToken<T, any>,\n factory: () => T | Promise<T>,\n ): void {\n // Create a factory class wrapper\n const FactoryWrapper = class {\n static factory = factory\n async create(): Promise<T> {\n return await factory()\n }\n }\n\n this.testRegistry.set(\n token,\n InjectableScope.Singleton,\n FactoryWrapper,\n InjectableType.Factory,\n )\n }\n\n private argsMatch(actual: unknown[], expected: unknown[]): boolean {\n if (actual.length !== expected.length) {\n return false\n }\n return actual.every((arg, index) => {\n const exp = expected[index]\n if (typeof exp === 'object' && exp !== null) {\n return JSON.stringify(arg) === JSON.stringify(exp)\n }\n return arg === exp\n })\n }\n}\n","import type { LifecycleRecord, MethodCallRecord, MockServiceStats, ProviderConfig, UnitTestContainerOptions } from './types.mjs'\n\nimport { Container } from '../container/container.mjs'\nimport { InjectableScope, InjectableType } from '../enums/index.mjs'\nimport { DIError } from '../errors/index.mjs'\nimport { InjectionToken } from '../token/injection-token.mjs'\nimport { Registry } from '../token/registry.mjs'\nimport { getInjectableToken } from '../utils/get-injectable-token.mjs'\nimport { defaultInjectors } from '../utils/index.mjs'\n\ntype AnyToken = InjectionToken<any, any> | (new (...args: any[]) => any)\n\n/**\n * Creates a tracking proxy that records method calls.\n */\nfunction createTrackingProxy<T extends object>(\n target: T,\n tokenId: string,\n methodCalls: Map<string, MethodCallRecord[]>,\n): T {\n return new Proxy(target, {\n get(obj, prop) {\n const value = Reflect.get(obj, prop)\n\n if (typeof value === 'function' && typeof prop === 'string') {\n return function (this: unknown, ...args: unknown[]) {\n const calls = methodCalls.get(tokenId) || []\n const record: MethodCallRecord = {\n method: prop,\n args,\n timestamp: Date.now(),\n }\n\n try {\n const result = value.apply(this === undefined ? obj : this, args)\n\n if (result instanceof Promise) {\n return result\n .then((res) => {\n record.result = res\n calls.push(record)\n methodCalls.set(tokenId, calls)\n return res\n })\n .catch((err) => {\n record.error = err\n calls.push(record)\n methodCalls.set(tokenId, calls)\n throw err\n })\n }\n\n record.result = result\n calls.push(record)\n methodCalls.set(tokenId, calls)\n return result\n } catch (err) {\n record.error = err as Error\n calls.push(record)\n methodCalls.set(tokenId, calls)\n throw err\n }\n }\n }\n\n return value\n },\n })\n}\n\n/**\n * Creates an auto-mock proxy that throws on method access.\n */\nfunction createAutoMockProxy(tokenId: string): object {\n return new Proxy(\n {},\n {\n get(_, prop) {\n if (prop === 'then' || prop === 'catch' || prop === 'finally') {\n return undefined\n }\n if (typeof prop === 'symbol') {\n return undefined\n }\n throw new Error(\n `[UnitTestContainer] Attempted to access '${prop}' on auto-mocked service '${tokenId}'. ` +\n `This service was not provided in the providers list. ` +\n `Add it to providers or use allowUnregistered: false to catch this earlier.`,\n )\n },\n },\n )\n}\n\n/**\n * UnitTestContainer for isolated unit testing.\n *\n * Only services explicitly listed in `providers` can be resolved.\n * All method calls are automatically tracked via proxies.\n * Unregistered dependencies throw by default, or can be auto-mocked.\n *\n * @example\n * ```ts\n * const container = new UnitTestContainer({\n * providers: [\n * { token: UserService, useClass: MockUserService },\n * { token: ConfigToken, useValue: { apiUrl: 'test' } },\n * ],\n * })\n *\n * const service = await container.get(UserService)\n *\n * // All method calls are automatically tracked\n * await service.findUser('123')\n *\n * container.expectCalled(UserService, 'findUser')\n * container.expectCalledWith(UserService, 'findUser', ['123'])\n * ```\n */\nexport class UnitTestContainer extends Container {\n private readonly testRegistry: Registry\n private readonly methodCalls = new Map<string, MethodCallRecord[]>()\n private readonly lifecycleEvents = new Map<string, LifecycleRecord[]>()\n private readonly instanceCounts = new Map<string, number>()\n private readonly registeredTokenIds = new Set<string>()\n private readonly autoMockedTokenIds = new Set<string>()\n private allowUnregistered: boolean\n\n constructor(options: UnitTestContainerOptions) {\n const testRegistry = new Registry()\n super(testRegistry, options.logger ?? null, defaultInjectors)\n this.testRegistry = testRegistry\n this.allowUnregistered = options.allowUnregistered ?? false\n\n // Register all providers\n for (const provider of options.providers) {\n this.registerProvider(provider)\n }\n }\n\n /**\n * Enables auto-mocking for unregistered dependencies.\n * Call this to switch from strict mode to auto-mock mode.\n */\n enableAutoMocking(): this {\n this.allowUnregistered = true\n return this\n }\n\n /**\n * Disables auto-mocking (strict mode).\n * Unregistered dependencies will throw.\n */\n disableAutoMocking(): this {\n this.allowUnregistered = false\n return this\n }\n\n /**\n * Override get to wrap instances in tracking proxies.\n */\n override async get(token: any, args?: unknown): Promise<any> {\n const realToken = this.resolveToken(token)\n\n // Check if this is a registered provider\n if (!this.registeredTokenIds.has(realToken.id)) {\n if (!this.allowUnregistered) {\n throw DIError.factoryNotFound(\n `${realToken.toString()} is not in the providers list. ` +\n `Add it to providers or enable allowUnregistered.`,\n )\n }\n\n // Auto-mock unregistered dependency\n if (!this.autoMockedTokenIds.has(realToken.id)) {\n this.autoMockedTokenIds.add(realToken.id)\n }\n\n return createAutoMockProxy(realToken.id)\n }\n\n const instance = await super.get(token, args)\n\n // Wrap in tracking proxy if it's an object\n if (instance && typeof instance === 'object') {\n return createTrackingProxy(instance, realToken.id, this.methodCalls)\n }\n\n return instance\n }\n\n /**\n * Clears all state and disposes the container.\n */\n async clear(): Promise<void> {\n await this.dispose()\n this.methodCalls.clear()\n this.lifecycleEvents.clear()\n this.instanceCounts.clear()\n this.autoMockedTokenIds.clear()\n }\n\n // ============================================================================\n // ASSERTION HELPERS\n // ============================================================================\n\n /**\n * Asserts that a service has been resolved at least once.\n */\n expectResolved(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const storage = this.getStorage()\n const names = storage.getAllNames()\n const found = names.some((name) => name.includes(realToken.id))\n\n if (!found) {\n throw new Error(`Expected ${realToken.toString()} to be resolved, but it was not`)\n }\n }\n\n /**\n * Asserts that a service has NOT been resolved.\n */\n expectNotResolved(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const storage = this.getStorage()\n const names = storage.getAllNames()\n const found = names.some((name) => name.includes(realToken.id))\n\n if (found) {\n throw new Error(`Expected ${realToken.toString()} to NOT be resolved, but it was`)\n }\n }\n\n /**\n * Asserts that a service was auto-mocked (not in providers list).\n */\n expectAutoMocked(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n\n if (!this.autoMockedTokenIds.has(realToken.id)) {\n throw new Error(\n `Expected ${realToken.toString()} to be auto-mocked, but it was not. ` +\n `Either it's in the providers list or hasn't been resolved.`,\n )\n }\n }\n\n /**\n * Asserts that a service was NOT auto-mocked (is in providers list).\n */\n expectNotAutoMocked(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n\n if (this.autoMockedTokenIds.has(realToken.id)) {\n throw new Error(\n `Expected ${realToken.toString()} to NOT be auto-mocked, but it was.`,\n )\n }\n }\n\n // ============================================================================\n // LIFECYCLE ASSERTIONS\n // ============================================================================\n\n /**\n * Records a lifecycle event for tracking.\n */\n recordLifecycleEvent(token: AnyToken, event: 'created' | 'initialized' | 'destroyed', instanceName: string): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n events.push({\n event,\n timestamp: Date.now(),\n instanceName,\n })\n this.lifecycleEvents.set(realToken.id, events)\n\n if (event === 'created') {\n const count = this.instanceCounts.get(realToken.id) || 0\n this.instanceCounts.set(realToken.id, count + 1)\n }\n }\n\n /**\n * Asserts that a service's onServiceInit was called.\n */\n expectInitialized(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n const initialized = events.some((e) => e.event === 'initialized')\n\n if (!initialized) {\n throw new Error(`Expected ${realToken.toString()} to be initialized, but onServiceInit was not called`)\n }\n }\n\n /**\n * Asserts that a service's onServiceDestroy was called.\n */\n expectDestroyed(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n const destroyed = events.some((e) => e.event === 'destroyed')\n\n if (!destroyed) {\n throw new Error(`Expected ${realToken.toString()} to be destroyed, but onServiceDestroy was not called`)\n }\n }\n\n /**\n * Asserts that a service has NOT been destroyed.\n */\n expectNotDestroyed(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n const destroyed = events.some((e) => e.event === 'destroyed')\n\n if (destroyed) {\n throw new Error(`Expected ${realToken.toString()} to NOT be destroyed, but onServiceDestroy was called`)\n }\n }\n\n // ============================================================================\n // CALL TRACKING (AUTO-TRACKED VIA PROXY)\n // ============================================================================\n\n /**\n * Asserts that a method was called on a service.\n */\n expectCalled(token: AnyToken, method: string): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const found = calls.some((c) => c.method === method)\n\n if (!found) {\n throw new Error(`Expected ${realToken.toString()}.${method}() to be called, but it was not`)\n }\n }\n\n /**\n * Asserts that a method was NOT called on a service.\n */\n expectNotCalled(token: AnyToken, method: string): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const found = calls.some((c) => c.method === method)\n\n if (found) {\n throw new Error(`Expected ${realToken.toString()}.${method}() to NOT be called, but it was`)\n }\n }\n\n /**\n * Asserts that a method was called with specific arguments.\n */\n expectCalledWith(token: AnyToken, method: string, expectedArgs: unknown[]): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const found = calls.some(\n (c) => c.method === method && this.argsMatch(c.args, expectedArgs),\n )\n\n if (!found) {\n const methodCalls = calls.filter((c) => c.method === method)\n const actualArgs = methodCalls.map((c) => JSON.stringify(c.args)).join(', ')\n throw new Error(\n `Expected ${realToken.toString()}.${method}() to be called with ${JSON.stringify(expectedArgs)}. ` +\n `Actual calls: [${actualArgs}]`,\n )\n }\n }\n\n /**\n * Asserts that a method was called a specific number of times.\n */\n expectCallCount(token: AnyToken, method: string, count: number): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const actualCount = calls.filter((c) => c.method === method).length\n\n if (actualCount !== count) {\n throw new Error(\n `Expected ${realToken.toString()}.${method}() to be called ${count} times, but was called ${actualCount} times`,\n )\n }\n }\n\n /**\n * Gets all recorded method calls for a service.\n */\n getMethodCalls(token: AnyToken): MethodCallRecord[] {\n const realToken = this.resolveToken(token)\n return this.methodCalls.get(realToken.id) || []\n }\n\n /**\n * Gets statistics about a service.\n */\n getServiceStats(token: AnyToken): MockServiceStats {\n const realToken = this.resolveToken(token)\n return {\n instanceCount: this.instanceCounts.get(realToken.id) || 0,\n methodCalls: this.methodCalls.get(realToken.id) || [],\n lifecycleEvents: this.lifecycleEvents.get(realToken.id) || [],\n }\n }\n\n /**\n * Clears all recorded method calls.\n */\n clearMethodCalls(): void {\n this.methodCalls.clear()\n }\n\n /**\n * Gets list of all registered provider token IDs.\n */\n getRegisteredTokenIds(): ReadonlySet<string> {\n return this.registeredTokenIds\n }\n\n /**\n * Gets list of all auto-mocked token IDs.\n */\n getAutoMockedTokenIds(): ReadonlySet<string> {\n return this.autoMockedTokenIds\n }\n\n // ============================================================================\n // INTERNAL HELPERS\n // ============================================================================\n\n private resolveToken(token: AnyToken): InjectionToken<any, any> {\n if (typeof token === 'function') {\n try {\n return getInjectableToken(token)\n } catch {\n // Class doesn't have @Injectable, create a token for it\n return InjectionToken.create(token)\n }\n }\n return token\n }\n\n private registerProvider<T>(provider: ProviderConfig<T>): void {\n const realToken = this.resolveToken(provider.token as AnyToken)\n this.registeredTokenIds.add(realToken.id)\n\n if (provider.useValue !== undefined) {\n this.registerValueBinding(realToken, provider.useValue)\n } else if (provider.useClass) {\n this.registerClassBinding(realToken, provider.useClass)\n } else if (provider.useFactory) {\n this.registerFactoryBinding(realToken, provider.useFactory)\n } else {\n // Just the token - register as itself\n if (typeof provider.token === 'function') {\n this.testRegistry.set(realToken, InjectableScope.Singleton, provider.token, InjectableType.Class)\n }\n }\n }\n\n private registerValueBinding<T>(token: InjectionToken<T, any>, value: T): void {\n const ValueHolder = class {\n static instance = value\n }\n\n this.testRegistry.set(token, InjectableScope.Singleton, ValueHolder, InjectableType.Class)\n\n const nameResolver = this.getNameResolver()\n const instanceName = nameResolver.generateInstanceName(\n token,\n undefined,\n undefined,\n InjectableScope.Singleton,\n )\n this.getStorage().storeInstance(instanceName, value)\n this.recordLifecycleEvent(token, 'created', instanceName)\n }\n\n private registerClassBinding<T>(token: InjectionToken<T, any>, cls: new (...args: any[]) => T): void {\n this.testRegistry.set(token, InjectableScope.Singleton, cls, InjectableType.Class)\n }\n\n private registerFactoryBinding<T>(token: InjectionToken<T, any>, factory: () => T | Promise<T>): void {\n const FactoryWrapper = class {\n static factory = factory\n async create(): Promise<T> {\n return await factory()\n }\n }\n\n this.testRegistry.set(token, InjectableScope.Singleton, FactoryWrapper, InjectableType.Factory)\n }\n\n private argsMatch(actual: unknown[], expected: unknown[]): boolean {\n if (actual.length !== expected.length) {\n return false\n }\n return actual.every((arg, index) => {\n const exp = expected[index]\n if (typeof exp === 'object' && exp !== null) {\n return JSON.stringify(arg) === JSON.stringify(exp)\n }\n return arg === exp\n })\n }\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.mjs","names":["Container","InjectableScope","InjectableType","BoundInjectionToken","globalRegistry","Registry","getInjectableToken","defaultInjectors","TestContainer","testRegistry","methodCalls","Map","lifecycleEvents","instanceCounts","boundTokens","Set","options","parentRegistry","logger","bind","token","realToken","resolveToken","tokenId","id","toValue","value","add","registerValueBinding","toClass","cls","registerClassBinding","toFactory","factory","registerFactoryBinding","clear","dispose","expectResolved","storage","getStorage","names","getAllNames","found","some","name","includes","Error","toString","expectNotResolved","expectSingleton","registry","getRegistry","has","record","get","scope","Singleton","expectTransient","Transient","expectRequestScoped","Request","expectSameInstance","instance1","instance2","expectDifferentInstances","expectInitialized","events","initialized","e","event","expectDestroyed","destroyed","expectNotDestroyed","recordMethodCall","method","args","result","error","calls","push","timestamp","Date","now","set","recordLifecycleEvent","instanceName","count","expectCalled","c","expectCalledWith","expectedArgs","argsMatch","JSON","stringify","expectCallCount","actualCount","filter","length","getMethodCalls","getServiceStats","instanceCount","clearMethodCalls","getDependencyGraph","nodes","rootTokens","forEach","holder","tokenMatch","match","dependencies","Array","from","deps","dependents","findDependents","getSimplifiedDependencyGraph","graph","dep","depTokenMatch","depTokenId","key","Object","keys","sort","ValueHolder","create","Factory","nameResolver","getNameResolver","generateInstanceName","undefined","storeInstance","Class","FactoryWrapper","actual","expected","every","arg","index","exp","Container","InjectableScope","InjectableType","DIError","BoundInjectionToken","InjectionToken","Registry","getInjectableToken","defaultInjectors","createTrackingProxy","target","tokenId","methodCalls","Proxy","get","obj","prop","value","Reflect","args","calls","record","method","timestamp","Date","now","result","apply","undefined","Promise","then","res","push","set","catch","err","error","createAutoMockProxy","_","Error","UnitTestContainer","testRegistry","Map","lifecycleEvents","instanceCounts","registeredTokenIds","Set","autoMockedTokenIds","allowUnregistered","options","logger","provider","providers","registerProvider","enableAutoMocking","disableAutoMocking","token","isBoundToken","id","realToken","resolveToken","isRegistered","has","factoryNotFound","toString","idToCheck","add","instance","trackingId","clear","dispose","expectResolved","storage","getStorage","names","getAllNames","found","some","name","includes","expectNotResolved","expectAutoMocked","expectNotAutoMocked","recordLifecycleEvent","event","instanceName","events","count","expectInitialized","initialized","e","expectDestroyed","destroyed","expectNotDestroyed","expectCalled","c","expectNotCalled","expectCalledWith","expectedArgs","argsMatch","filter","actualArgs","map","JSON","stringify","join","expectCallCount","actualCount","length","getMethodCalls","getServiceStats","instanceCount","clearMethodCalls","getRegisteredTokenIds","getAutoMockedTokenIds","create","providerToken","useValue","registerValueBinding","useClass","registerClassBinding","useFactory","registerFactoryBinding","Singleton","Class","ValueHolder","Factory","nameResolver","getNameResolver","generateInstanceName","storeInstance","cls","factory","FactoryWrapper","actual","expected","every","arg","index","exp"],"sources":["../../src/testing/test-container.mts","../../src/testing/unit-test-container.mts"],"sourcesContent":["import type {\n BindingBuilder,\n DependencyGraph,\n DependencyNode,\n LifecycleRecord,\n MethodCallRecord,\n MockServiceStats,\n TestContainerOptions,\n} from './types.mjs'\n\nimport { Container } from '../container/container.mjs'\nimport { InjectableScope, InjectableType } from '../enums/index.mjs'\nimport {\n BoundInjectionToken,\n InjectionToken,\n} from '../token/injection-token.mjs'\nimport { globalRegistry, Registry } from '../token/registry.mjs'\nimport { getInjectableToken } from '../utils/get-injectable-token.mjs'\nimport { defaultInjectors } from '../utils/index.mjs'\n\ntype AnyToken =\n | InjectionToken<any, any>\n | BoundInjectionToken<any, any>\n | (new (...args: any[]) => any)\n\n/**\n * TestContainer extends Container with testing utilities.\n *\n * Provides simple value/class binding for integration/e2e tests,\n * plus assertion helpers and dependency graph inspection.\n *\n * @example\n * ```ts\n * const container = new TestContainer()\n *\n * // Bind mock values\n * container.bind(DatabaseToken).toValue(mockDatabase)\n * container.bind(UserService).toClass(MockUserService)\n *\n * // Use container normally\n * const service = await container.get(MyService)\n *\n * // Assert on container state\n * container.expectResolved(MyService)\n * container.expectSingleton(MyService)\n * ```\n */\nexport class TestContainer extends Container {\n private readonly testRegistry: Registry\n private readonly methodCalls = new Map<string, MethodCallRecord[]>()\n private readonly lifecycleEvents = new Map<string, LifecycleRecord[]>()\n private readonly instanceCounts = new Map<string, number>()\n private readonly boundTokens = new Set<string>()\n\n /**\n * Creates a new TestContainer.\n *\n * @param options - Configuration options\n * @param options.parentRegistry - Parent registry. Defaults to globalRegistry.\n * Pass `null` for a completely isolated container.\n * @param options.logger - Optional logger for debugging.\n *\n * @example\n * ```ts\n * // Uses globalRegistry as parent (default)\n * const container = new TestContainer()\n *\n * // Isolated container (no access to @Injectable classes)\n * const isolated = new TestContainer({ parentRegistry: null })\n *\n * // Custom parent registry\n * const custom = new TestContainer({ parentRegistry: myRegistry })\n * ```\n */\n constructor(options: TestContainerOptions = {}) {\n const { parentRegistry = globalRegistry, logger = null } = options\n const testRegistry = parentRegistry\n ? new Registry(parentRegistry)\n : new Registry()\n super(testRegistry, logger, defaultInjectors)\n this.testRegistry = testRegistry\n }\n\n // ============================================================================\n // BINDING API\n // ============================================================================\n\n /**\n * Creates a binding builder for the given token.\n *\n * @example\n * ```ts\n * container.bind(UserService).toValue(mockUserService)\n * container.bind(DatabaseToken).toClass(MockDatabase)\n * container.bind(ConfigToken).toFactory(() => ({ apiKey: 'test' }))\n * container.bind(BOUND_CONFIG_TOKEN).toValue(overrideValue)\n * ```\n */\n bind<T>(\n token:\n | InjectionToken<T, any>\n | BoundInjectionToken<T, any>\n | (new (...args: any[]) => T),\n ): BindingBuilder<T> {\n const realToken = this.resolveToken(token)\n const tokenId = realToken.id\n\n return {\n toValue: (value: T) => {\n this.boundTokens.add(tokenId)\n this.registerValueBinding(realToken, value)\n },\n toClass: <C extends new (...args: any[]) => T>(cls: C) => {\n this.boundTokens.add(tokenId)\n this.registerClassBinding(realToken, cls)\n },\n toFactory: (factory: () => T | Promise<T>) => {\n this.boundTokens.add(tokenId)\n this.registerFactoryBinding(realToken, factory)\n },\n }\n }\n\n /**\n * Clears all bindings and resets container state.\n */\n async clear(): Promise<void> {\n await this.dispose()\n this.methodCalls.clear()\n this.lifecycleEvents.clear()\n this.instanceCounts.clear()\n this.boundTokens.clear()\n }\n\n // ============================================================================\n // ASSERTION HELPERS\n // ============================================================================\n\n /**\n * Asserts that a service has been resolved at least once.\n */\n expectResolved(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const storage = this.getStorage()\n const names = storage.getAllNames()\n const found = names.some((name) => name.includes(realToken.id))\n\n if (!found) {\n throw new Error(\n `Expected ${realToken.toString()} to be resolved, but it was not`,\n )\n }\n }\n\n /**\n * Asserts that a service has NOT been resolved.\n */\n expectNotResolved(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const storage = this.getStorage()\n const names = storage.getAllNames()\n const found = names.some((name) => name.includes(realToken.id))\n\n if (found) {\n throw new Error(\n `Expected ${realToken.toString()} to NOT be resolved, but it was`,\n )\n }\n }\n\n /**\n * Asserts that a service is registered as singleton scope.\n */\n expectSingleton(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const registry = this.getRegistry()\n\n if (!registry.has(realToken)) {\n throw new Error(\n `Expected ${realToken.toString()} to be registered, but it was not`,\n )\n }\n\n const record = registry.get(realToken)\n if (record.scope !== InjectableScope.Singleton) {\n throw new Error(\n `Expected ${realToken.toString()} to be Singleton scope, but it was ${record.scope}`,\n )\n }\n }\n\n /**\n * Asserts that a service is registered as transient scope.\n */\n expectTransient(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const registry = this.getRegistry()\n\n if (!registry.has(realToken)) {\n throw new Error(\n `Expected ${realToken.toString()} to be registered, but it was not`,\n )\n }\n\n const record = registry.get(realToken)\n if (record.scope !== InjectableScope.Transient) {\n throw new Error(\n `Expected ${realToken.toString()} to be Transient scope, but it was ${record.scope}`,\n )\n }\n }\n\n /**\n * Asserts that a service is registered as request scope.\n */\n expectRequestScoped(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const registry = this.getRegistry()\n\n if (!registry.has(realToken)) {\n throw new Error(\n `Expected ${realToken.toString()} to be registered, but it was not`,\n )\n }\n\n const record = registry.get(realToken)\n if (record.scope !== InjectableScope.Request) {\n throw new Error(\n `Expected ${realToken.toString()} to be Request scope, but it was ${record.scope}`,\n )\n }\n }\n\n /**\n * Asserts that two service resolutions return the same instance.\n */\n async expectSameInstance(token: AnyToken): Promise<void> {\n const instance1 = await this.get(token as any)\n const instance2 = await this.get(token as any)\n\n if (instance1 !== instance2) {\n const realToken = this.resolveToken(token)\n throw new Error(\n `Expected ${realToken.toString()} to return same instance, but got different instances`,\n )\n }\n }\n\n /**\n * Asserts that two service resolutions return different instances.\n */\n async expectDifferentInstances(token: AnyToken): Promise<void> {\n const instance1 = await this.get(token as any)\n const instance2 = await this.get(token as any)\n\n if (instance1 === instance2) {\n const realToken = this.resolveToken(token)\n throw new Error(\n `Expected ${realToken.toString()} to return different instances, but got same instance`,\n )\n }\n }\n\n // ============================================================================\n // LIFECYCLE ASSERTIONS\n // ============================================================================\n\n /**\n * Asserts that a service's onServiceInit was called.\n */\n expectInitialized(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n const initialized = events.some((e) => e.event === 'initialized')\n\n if (!initialized) {\n throw new Error(\n `Expected ${realToken.toString()} to be initialized, but onServiceInit was not called`,\n )\n }\n }\n\n /**\n * Asserts that a service's onServiceDestroy was called.\n */\n expectDestroyed(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n const destroyed = events.some((e) => e.event === 'destroyed')\n\n if (!destroyed) {\n throw new Error(\n `Expected ${realToken.toString()} to be destroyed, but onServiceDestroy was not called`,\n )\n }\n }\n\n /**\n * Asserts that a service has NOT been destroyed.\n */\n expectNotDestroyed(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n const destroyed = events.some((e) => e.event === 'destroyed')\n\n if (destroyed) {\n throw new Error(\n `Expected ${realToken.toString()} to NOT be destroyed, but onServiceDestroy was called`,\n )\n }\n }\n\n // ============================================================================\n // CALL TRACKING\n // ============================================================================\n\n /**\n * Records a method call for tracking.\n * Call this from your mock implementations.\n */\n recordMethodCall(\n token: AnyToken,\n method: string,\n args: unknown[],\n result?: unknown,\n error?: Error,\n ): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n calls.push({\n method,\n args,\n timestamp: Date.now(),\n result,\n error,\n })\n this.methodCalls.set(realToken.id, calls)\n }\n\n /**\n * Records a lifecycle event for tracking.\n */\n recordLifecycleEvent(\n token: AnyToken,\n event: 'created' | 'initialized' | 'destroyed',\n instanceName: string,\n ): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n events.push({\n event,\n timestamp: Date.now(),\n instanceName,\n })\n this.lifecycleEvents.set(realToken.id, events)\n\n if (event === 'created') {\n const count = this.instanceCounts.get(realToken.id) || 0\n this.instanceCounts.set(realToken.id, count + 1)\n }\n }\n\n /**\n * Asserts that a method was called on a service.\n */\n expectCalled(token: AnyToken, method: string): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const found = calls.some((c) => c.method === method)\n\n if (!found) {\n throw new Error(\n `Expected ${realToken.toString()}.${method}() to be called, but it was not`,\n )\n }\n }\n\n /**\n * Asserts that a method was called with specific arguments.\n */\n expectCalledWith(\n token: AnyToken,\n method: string,\n expectedArgs: unknown[],\n ): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const found = calls.some(\n (c) => c.method === method && this.argsMatch(c.args, expectedArgs),\n )\n\n if (!found) {\n throw new Error(\n `Expected ${realToken.toString()}.${method}() to be called with ${JSON.stringify(expectedArgs)}, but it was not`,\n )\n }\n }\n\n /**\n * Asserts that a method was called a specific number of times.\n */\n expectCallCount(token: AnyToken, method: string, count: number): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const actualCount = calls.filter((c) => c.method === method).length\n\n if (actualCount !== count) {\n throw new Error(\n `Expected ${realToken.toString()}.${method}() to be called ${count} times, but was called ${actualCount} times`,\n )\n }\n }\n\n /**\n * Gets all recorded method calls for a service.\n */\n getMethodCalls(token: AnyToken): MethodCallRecord[] {\n const realToken = this.resolveToken(token)\n return this.methodCalls.get(realToken.id) || []\n }\n\n /**\n * Gets statistics about a mocked service.\n */\n getServiceStats(token: AnyToken): MockServiceStats {\n const realToken = this.resolveToken(token)\n return {\n instanceCount: this.instanceCounts.get(realToken.id) || 0,\n methodCalls: this.methodCalls.get(realToken.id) || [],\n lifecycleEvents: this.lifecycleEvents.get(realToken.id) || [],\n }\n }\n\n /**\n * Clears all recorded method calls.\n */\n clearMethodCalls(): void {\n this.methodCalls.clear()\n }\n\n // ============================================================================\n // DEPENDENCY GRAPH\n // ============================================================================\n\n /**\n * Gets the dependency graph for snapshot testing.\n * Returns a serializable structure that can be used with vitest snapshots.\n */\n getDependencyGraph(): DependencyGraph {\n const storage = this.getStorage()\n const nodes: Record<string, DependencyNode> = {}\n const rootTokens: string[] = []\n\n storage.forEach((name, holder) => {\n const tokenMatch = name.match(/^([^:]+)/)\n const tokenId = tokenMatch ? tokenMatch[1] : name\n\n nodes[name] = {\n token: tokenId,\n instanceName: name,\n scope: holder.scope,\n dependencies: Array.from(holder.deps),\n dependents: storage.findDependents(name),\n }\n\n // Root tokens have no dependents\n if (storage.findDependents(name).length === 0) {\n rootTokens.push(name)\n }\n })\n\n return { nodes, rootTokens }\n }\n\n /**\n * Gets a simplified dependency graph showing only token relationships.\n * Useful for cleaner snapshot comparisons.\n */\n getSimplifiedDependencyGraph(): Record<string, string[]> {\n const storage = this.getStorage()\n const graph: Record<string, string[]> = {}\n\n storage.forEach((name, holder) => {\n const tokenMatch = name.match(/^([^:]+)/)\n const tokenId = tokenMatch ? tokenMatch[1] : name\n\n if (!graph[tokenId]) {\n graph[tokenId] = []\n }\n\n for (const dep of holder.deps) {\n const depTokenMatch = dep.match(/^([^:]+)/)\n const depTokenId = depTokenMatch ? depTokenMatch[1] : dep\n if (!graph[tokenId].includes(depTokenId)) {\n graph[tokenId].push(depTokenId)\n }\n }\n })\n\n // Sort for consistent snapshots\n for (const key of Object.keys(graph)) {\n graph[key].sort()\n }\n\n return graph\n }\n\n // ============================================================================\n // INTERNAL HELPERS\n // ============================================================================\n\n private resolveToken(token: AnyToken): InjectionToken<any, any> {\n if (typeof token === 'function') {\n return getInjectableToken(token)\n }\n if (token instanceof BoundInjectionToken) {\n return token.token\n }\n return token\n }\n\n private registerValueBinding<T>(\n token: InjectionToken<T, any>,\n value: T,\n ): void {\n // Create a simple class that returns the value\n const ValueHolder = class {\n create(): T {\n return value\n }\n }\n\n this.testRegistry.set(\n token,\n InjectableScope.Singleton,\n ValueHolder,\n InjectableType.Factory,\n 1000, // Higher priority for test overrides\n )\n\n // Store the instance directly\n const nameResolver = this.getNameResolver()\n const instanceName = nameResolver.generateInstanceName(\n token,\n undefined,\n undefined,\n InjectableScope.Singleton,\n )\n this.getStorage().storeInstance(instanceName, value)\n this.recordLifecycleEvent(token, 'created', instanceName)\n }\n\n private registerClassBinding<T>(\n token: InjectionToken<T, any>,\n cls: new (...args: any[]) => T,\n ): void {\n this.testRegistry.set(\n token,\n InjectableScope.Singleton,\n cls,\n InjectableType.Class,\n 1000, // Higher priority for test overrides\n )\n }\n\n private registerFactoryBinding<T>(\n token: InjectionToken<T, any>,\n factory: () => T | Promise<T>,\n ): void {\n // Create a factory class wrapper\n const FactoryWrapper = class {\n static factory = factory\n async create(): Promise<T> {\n return await factory()\n }\n }\n\n this.testRegistry.set(\n token,\n InjectableScope.Singleton,\n FactoryWrapper,\n InjectableType.Factory,\n 1000, // Higher priority for test overrides\n )\n }\n\n private argsMatch(actual: unknown[], expected: unknown[]): boolean {\n if (actual.length !== expected.length) {\n return false\n }\n return actual.every((arg, index) => {\n const exp = expected[index]\n if (typeof exp === 'object' && exp !== null) {\n return JSON.stringify(arg) === JSON.stringify(exp)\n }\n return arg === exp\n })\n }\n}\n","import type {\n LifecycleRecord,\n MethodCallRecord,\n MockServiceStats,\n ProviderConfig,\n UnitTestContainerOptions,\n} from './types.mjs'\n\nimport { Container } from '../container/container.mjs'\nimport { InjectableScope, InjectableType } from '../enums/index.mjs'\nimport { DIError } from '../errors/index.mjs'\nimport {\n BoundInjectionToken,\n InjectionToken,\n} from '../token/injection-token.mjs'\nimport { Registry } from '../token/registry.mjs'\nimport { getInjectableToken } from '../utils/get-injectable-token.mjs'\nimport { defaultInjectors } from '../utils/index.mjs'\n\ntype AnyToken =\n | InjectionToken<any, any>\n | BoundInjectionToken<any, any>\n | (new (...args: any[]) => any)\n\n/**\n * Creates a tracking proxy that records method calls.\n */\nfunction createTrackingProxy<T extends object>(\n target: T,\n tokenId: string,\n methodCalls: Map<string, MethodCallRecord[]>,\n): T {\n return new Proxy(target, {\n get(obj, prop) {\n const value = Reflect.get(obj, prop)\n\n if (typeof value === 'function' && typeof prop === 'string') {\n return function (this: unknown, ...args: unknown[]) {\n const calls = methodCalls.get(tokenId) || []\n const record: MethodCallRecord = {\n method: prop,\n args,\n timestamp: Date.now(),\n }\n\n try {\n const result = value.apply(this === undefined ? obj : this, args)\n\n if (result instanceof Promise) {\n return result\n .then((res) => {\n record.result = res\n calls.push(record)\n methodCalls.set(tokenId, calls)\n return res\n })\n .catch((err) => {\n record.error = err\n calls.push(record)\n methodCalls.set(tokenId, calls)\n throw err\n })\n }\n\n record.result = result\n calls.push(record)\n methodCalls.set(tokenId, calls)\n return result\n } catch (err) {\n record.error = err as Error\n calls.push(record)\n methodCalls.set(tokenId, calls)\n throw err\n }\n }\n }\n\n return value\n },\n })\n}\n\n/**\n * Creates an auto-mock proxy that throws on method access.\n */\nfunction createAutoMockProxy(tokenId: string): object {\n return new Proxy(\n {},\n {\n get(_, prop) {\n if (prop === 'then' || prop === 'catch' || prop === 'finally') {\n return undefined\n }\n if (typeof prop === 'symbol') {\n return undefined\n }\n throw new Error(\n `[UnitTestContainer] Attempted to access '${prop}' on auto-mocked service '${tokenId}'. ` +\n `This service was not provided in the providers list. ` +\n `Add it to providers or use allowUnregistered: false to catch this earlier.`,\n )\n },\n },\n )\n}\n\n/**\n * UnitTestContainer for isolated unit testing.\n *\n * Only services explicitly listed in `providers` can be resolved.\n * All method calls are automatically tracked via proxies.\n * Unregistered dependencies throw by default, or can be auto-mocked.\n *\n * @example\n * ```ts\n * const container = new UnitTestContainer({\n * providers: [\n * { token: UserService, useClass: MockUserService },\n * { token: ConfigToken, useValue: { apiUrl: 'test' } },\n * ],\n * })\n *\n * const service = await container.get(UserService)\n *\n * // All method calls are automatically tracked\n * await service.findUser('123')\n *\n * container.expectCalled(UserService, 'findUser')\n * container.expectCalledWith(UserService, 'findUser', ['123'])\n * ```\n */\nexport class UnitTestContainer extends Container {\n private readonly testRegistry: Registry\n private readonly methodCalls = new Map<string, MethodCallRecord[]>()\n private readonly lifecycleEvents = new Map<string, LifecycleRecord[]>()\n private readonly instanceCounts = new Map<string, number>()\n private readonly registeredTokenIds = new Set<string>()\n private readonly autoMockedTokenIds = new Set<string>()\n private allowUnregistered: boolean\n\n constructor(options: UnitTestContainerOptions) {\n const testRegistry = new Registry()\n super(testRegistry, options.logger ?? null, defaultInjectors)\n this.testRegistry = testRegistry\n this.allowUnregistered = options.allowUnregistered ?? false\n\n // Register all providers\n for (const provider of options.providers) {\n this.registerProvider(provider)\n }\n }\n\n /**\n * Enables auto-mocking for unregistered dependencies.\n * Call this to switch from strict mode to auto-mock mode.\n */\n enableAutoMocking(): this {\n this.allowUnregistered = true\n return this\n }\n\n /**\n * Disables auto-mocking (strict mode).\n * Unregistered dependencies will throw.\n */\n disableAutoMocking(): this {\n this.allowUnregistered = false\n return this\n }\n\n /**\n * Override get to wrap instances in tracking proxies.\n */\n override async get(token: any, args?: unknown): Promise<any> {\n // Check if token is a BoundInjectionToken and if it's registered\n const isBoundToken = token instanceof BoundInjectionToken\n const tokenId = isBoundToken ? token.id : undefined\n const realToken = this.resolveToken(token)\n\n // Check if this is a registered provider (check both bound token ID and real token ID)\n const isRegistered =\n (tokenId && this.registeredTokenIds.has(tokenId)) ||\n this.registeredTokenIds.has(realToken.id)\n\n if (!isRegistered) {\n if (!this.allowUnregistered) {\n throw DIError.factoryNotFound(\n `${realToken.toString()} is not in the providers list. ` +\n `Add it to providers or enable allowUnregistered.`,\n )\n }\n\n // Auto-mock unregistered dependency\n const idToCheck = tokenId || realToken.id\n if (!this.autoMockedTokenIds.has(idToCheck)) {\n this.autoMockedTokenIds.add(idToCheck)\n }\n\n return createAutoMockProxy(idToCheck)\n }\n\n const instance = await super.get(token, args)\n\n // Wrap in tracking proxy if it's an object\n const trackingId = tokenId || realToken.id\n if (instance && typeof instance === 'object') {\n return createTrackingProxy(instance, trackingId, this.methodCalls)\n }\n\n return instance\n }\n\n /**\n * Clears all state and disposes the container.\n */\n async clear(): Promise<void> {\n await this.dispose()\n this.methodCalls.clear()\n this.lifecycleEvents.clear()\n this.instanceCounts.clear()\n this.autoMockedTokenIds.clear()\n }\n\n // ============================================================================\n // ASSERTION HELPERS\n // ============================================================================\n\n /**\n * Asserts that a service has been resolved at least once.\n */\n expectResolved(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const storage = this.getStorage()\n const names = storage.getAllNames()\n const found = names.some((name) => name.includes(realToken.id))\n\n if (!found) {\n throw new Error(\n `Expected ${realToken.toString()} to be resolved, but it was not`,\n )\n }\n }\n\n /**\n * Asserts that a service has NOT been resolved.\n */\n expectNotResolved(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const storage = this.getStorage()\n const names = storage.getAllNames()\n const found = names.some((name) => name.includes(realToken.id))\n\n if (found) {\n throw new Error(\n `Expected ${realToken.toString()} to NOT be resolved, but it was`,\n )\n }\n }\n\n /**\n * Asserts that a service was auto-mocked (not in providers list).\n */\n expectAutoMocked(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n\n if (!this.autoMockedTokenIds.has(realToken.id)) {\n throw new Error(\n `Expected ${realToken.toString()} to be auto-mocked, but it was not. ` +\n `Either it's in the providers list or hasn't been resolved.`,\n )\n }\n }\n\n /**\n * Asserts that a service was NOT auto-mocked (is in providers list).\n */\n expectNotAutoMocked(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n\n if (this.autoMockedTokenIds.has(realToken.id)) {\n throw new Error(\n `Expected ${realToken.toString()} to NOT be auto-mocked, but it was.`,\n )\n }\n }\n\n // ============================================================================\n // LIFECYCLE ASSERTIONS\n // ============================================================================\n\n /**\n * Records a lifecycle event for tracking.\n */\n recordLifecycleEvent(\n token: AnyToken,\n event: 'created' | 'initialized' | 'destroyed',\n instanceName: string,\n ): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n events.push({\n event,\n timestamp: Date.now(),\n instanceName,\n })\n this.lifecycleEvents.set(realToken.id, events)\n\n if (event === 'created') {\n const count = this.instanceCounts.get(realToken.id) || 0\n this.instanceCounts.set(realToken.id, count + 1)\n }\n }\n\n /**\n * Asserts that a service's onServiceInit was called.\n */\n expectInitialized(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n const initialized = events.some((e) => e.event === 'initialized')\n\n if (!initialized) {\n throw new Error(\n `Expected ${realToken.toString()} to be initialized, but onServiceInit was not called`,\n )\n }\n }\n\n /**\n * Asserts that a service's onServiceDestroy was called.\n */\n expectDestroyed(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n const destroyed = events.some((e) => e.event === 'destroyed')\n\n if (!destroyed) {\n throw new Error(\n `Expected ${realToken.toString()} to be destroyed, but onServiceDestroy was not called`,\n )\n }\n }\n\n /**\n * Asserts that a service has NOT been destroyed.\n */\n expectNotDestroyed(token: AnyToken): void {\n const realToken = this.resolveToken(token)\n const events = this.lifecycleEvents.get(realToken.id) || []\n const destroyed = events.some((e) => e.event === 'destroyed')\n\n if (destroyed) {\n throw new Error(\n `Expected ${realToken.toString()} to NOT be destroyed, but onServiceDestroy was called`,\n )\n }\n }\n\n // ============================================================================\n // CALL TRACKING (AUTO-TRACKED VIA PROXY)\n // ============================================================================\n\n /**\n * Asserts that a method was called on a service.\n */\n expectCalled(token: AnyToken, method: string): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const found = calls.some((c) => c.method === method)\n\n if (!found) {\n throw new Error(\n `Expected ${realToken.toString()}.${method}() to be called, but it was not`,\n )\n }\n }\n\n /**\n * Asserts that a method was NOT called on a service.\n */\n expectNotCalled(token: AnyToken, method: string): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const found = calls.some((c) => c.method === method)\n\n if (found) {\n throw new Error(\n `Expected ${realToken.toString()}.${method}() to NOT be called, but it was`,\n )\n }\n }\n\n /**\n * Asserts that a method was called with specific arguments.\n */\n expectCalledWith(\n token: AnyToken,\n method: string,\n expectedArgs: unknown[],\n ): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const found = calls.some(\n (c) => c.method === method && this.argsMatch(c.args, expectedArgs),\n )\n\n if (!found) {\n const methodCalls = calls.filter((c) => c.method === method)\n const actualArgs = methodCalls\n .map((c) => JSON.stringify(c.args))\n .join(', ')\n throw new Error(\n `Expected ${realToken.toString()}.${method}() to be called with ${JSON.stringify(expectedArgs)}. ` +\n `Actual calls: [${actualArgs}]`,\n )\n }\n }\n\n /**\n * Asserts that a method was called a specific number of times.\n */\n expectCallCount(token: AnyToken, method: string, count: number): void {\n const realToken = this.resolveToken(token)\n const calls = this.methodCalls.get(realToken.id) || []\n const actualCount = calls.filter((c) => c.method === method).length\n\n if (actualCount !== count) {\n throw new Error(\n `Expected ${realToken.toString()}.${method}() to be called ${count} times, but was called ${actualCount} times`,\n )\n }\n }\n\n /**\n * Gets all recorded method calls for a service.\n */\n getMethodCalls(token: AnyToken): MethodCallRecord[] {\n const realToken = this.resolveToken(token)\n return this.methodCalls.get(realToken.id) || []\n }\n\n /**\n * Gets statistics about a service.\n */\n getServiceStats(token: AnyToken): MockServiceStats {\n const realToken = this.resolveToken(token)\n return {\n instanceCount: this.instanceCounts.get(realToken.id) || 0,\n methodCalls: this.methodCalls.get(realToken.id) || [],\n lifecycleEvents: this.lifecycleEvents.get(realToken.id) || [],\n }\n }\n\n /**\n * Clears all recorded method calls.\n */\n clearMethodCalls(): void {\n this.methodCalls.clear()\n }\n\n /**\n * Gets list of all registered provider token IDs.\n */\n getRegisteredTokenIds(): ReadonlySet<string> {\n return this.registeredTokenIds\n }\n\n /**\n * Gets list of all auto-mocked token IDs.\n */\n getAutoMockedTokenIds(): ReadonlySet<string> {\n return this.autoMockedTokenIds\n }\n\n // ============================================================================\n // INTERNAL HELPERS\n // ============================================================================\n\n private resolveToken(token: AnyToken): InjectionToken<any, any> {\n if (typeof token === 'function') {\n try {\n return getInjectableToken(token)\n } catch {\n // Class doesn't have @Injectable, create a token for it\n return InjectionToken.create(token)\n }\n }\n if (token instanceof BoundInjectionToken) {\n return token.token\n }\n return token\n }\n\n private registerProvider<T>(provider: ProviderConfig<T>): void {\n const providerToken = provider.token as AnyToken\n const realToken = this.resolveToken(providerToken)\n\n // Track both the real token ID and the bound token ID if it's a bound token\n this.registeredTokenIds.add(realToken.id)\n if (providerToken instanceof BoundInjectionToken) {\n this.registeredTokenIds.add(providerToken.id)\n }\n\n if (provider.useValue !== undefined) {\n this.registerValueBinding(realToken, provider.useValue)\n } else if (provider.useClass) {\n this.registerClassBinding(realToken, provider.useClass)\n } else if (provider.useFactory) {\n this.registerFactoryBinding(realToken, provider.useFactory)\n } else {\n // Just the token - register as itself\n if (typeof provider.token === 'function') {\n this.testRegistry.set(\n realToken,\n InjectableScope.Singleton,\n provider.token,\n InjectableType.Class,\n 1000, // Higher priority for test overrides\n )\n } else if (providerToken instanceof BoundInjectionToken) {\n // If it's a bound token without override, register the bound value\n this.registerValueBinding(realToken, providerToken.value)\n }\n }\n }\n\n private registerValueBinding<T>(\n token: InjectionToken<T, any>,\n value: T,\n ): void {\n const ValueHolder = class {\n create(): T {\n return value\n }\n }\n\n this.testRegistry.set(\n token,\n InjectableScope.Singleton,\n ValueHolder,\n InjectableType.Factory,\n 1000, // Higher priority for test overrides\n )\n\n const nameResolver = this.getNameResolver()\n const instanceName = nameResolver.generateInstanceName(\n token,\n undefined,\n undefined,\n InjectableScope.Singleton,\n )\n this.getStorage().storeInstance(instanceName, value)\n this.recordLifecycleEvent(token, 'created', instanceName)\n }\n\n private registerClassBinding<T>(\n token: InjectionToken<T, any>,\n cls: new (...args: any[]) => T,\n ): void {\n this.testRegistry.set(\n token,\n InjectableScope.Singleton,\n cls,\n InjectableType.Class,\n 1000, // Higher priority for test overrides\n )\n }\n\n private registerFactoryBinding<T>(\n token: InjectionToken<T, any>,\n factory: () => T | Promise<T>,\n ): void {\n const FactoryWrapper = class {\n static factory = factory\n async create(): Promise<T> {\n return await factory()\n }\n }\n\n this.testRegistry.set(\n token,\n InjectableScope.Singleton,\n FactoryWrapper,\n InjectableType.Factory,\n 1000, // Higher priority for test overrides\n )\n }\n\n private argsMatch(actual: unknown[], expected: unknown[]): boolean {\n if (actual.length !== expected.length) {\n return false\n }\n return actual.every((arg, index) => {\n const exp = expected[index]\n if (typeof exp === 'object' && exp !== null) {\n return JSON.stringify(arg) === JSON.stringify(exp)\n }\n return arg === exp\n })\n }\n}\n"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,8 +1,9 @@
|
|
|
1
1
|
import { afterEach, beforeEach, describe, expect, it } from 'vitest'
|
|
2
|
+
import { z } from 'zod'
|
|
2
3
|
|
|
3
4
|
import { Injectable } from '../decorators/injectable.decorator.mjs'
|
|
4
|
-
import { InjectionToken } from '../token/injection-token.mjs'
|
|
5
5
|
import { UnitTestContainer } from '../testing/unit-test-container.mjs'
|
|
6
|
+
import { InjectionToken } from '../token/injection-token.mjs'
|
|
6
7
|
|
|
7
8
|
describe('UnitTestContainer', () => {
|
|
8
9
|
describe('Strict Mode (default)', () => {
|
|
@@ -32,7 +33,9 @@ describe('UnitTestContainer', () => {
|
|
|
32
33
|
providers: [],
|
|
33
34
|
})
|
|
34
35
|
|
|
35
|
-
await expect(container.get(NotProvided)).rejects.toThrow(
|
|
36
|
+
await expect(container.get(NotProvided)).rejects.toThrow(
|
|
37
|
+
/not in the providers list/,
|
|
38
|
+
)
|
|
36
39
|
|
|
37
40
|
await container.dispose()
|
|
38
41
|
})
|
|
@@ -81,6 +84,125 @@ describe('UnitTestContainer', () => {
|
|
|
81
84
|
|
|
82
85
|
await container.dispose()
|
|
83
86
|
})
|
|
87
|
+
|
|
88
|
+
it('should support bound tokens in providers with useValue override', async () => {
|
|
89
|
+
const configSchema = z.object({
|
|
90
|
+
apiUrl: z.string(),
|
|
91
|
+
timeout: z.number(),
|
|
92
|
+
})
|
|
93
|
+
|
|
94
|
+
const CONFIG_TOKEN = InjectionToken.create<
|
|
95
|
+
{ apiUrl: string; timeout: number },
|
|
96
|
+
typeof configSchema
|
|
97
|
+
>('CONFIG', configSchema)
|
|
98
|
+
|
|
99
|
+
const BOUND_CONFIG = InjectionToken.bound(CONFIG_TOKEN, {
|
|
100
|
+
apiUrl: 'https://default.com',
|
|
101
|
+
timeout: 5000,
|
|
102
|
+
})
|
|
103
|
+
|
|
104
|
+
const container = new UnitTestContainer({
|
|
105
|
+
providers: [
|
|
106
|
+
{
|
|
107
|
+
token: BOUND_CONFIG,
|
|
108
|
+
useValue: { apiUrl: 'https://override.com', timeout: 10000 },
|
|
109
|
+
},
|
|
110
|
+
],
|
|
111
|
+
})
|
|
112
|
+
|
|
113
|
+
const config = await container.get(BOUND_CONFIG)
|
|
114
|
+
expect(config.apiUrl).toBe('https://override.com')
|
|
115
|
+
expect(config.timeout).toBe(10000)
|
|
116
|
+
|
|
117
|
+
await container.dispose()
|
|
118
|
+
})
|
|
119
|
+
|
|
120
|
+
it('should support bound tokens in providers with useClass override', async () => {
|
|
121
|
+
const configSchema = z.object({
|
|
122
|
+
apiUrl: z.string(),
|
|
123
|
+
})
|
|
124
|
+
|
|
125
|
+
const CONFIG_TOKEN = InjectionToken.create<
|
|
126
|
+
{ apiUrl: string },
|
|
127
|
+
typeof configSchema
|
|
128
|
+
>('CONFIG', configSchema)
|
|
129
|
+
|
|
130
|
+
const BOUND_CONFIG = InjectionToken.bound(CONFIG_TOKEN, {
|
|
131
|
+
apiUrl: 'https://default.com',
|
|
132
|
+
})
|
|
133
|
+
|
|
134
|
+
class TestConfig {
|
|
135
|
+
apiUrl: string
|
|
136
|
+
constructor() {
|
|
137
|
+
this.apiUrl = 'https://class-override.com'
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
const container = new UnitTestContainer({
|
|
142
|
+
providers: [{ token: BOUND_CONFIG, useClass: TestConfig }],
|
|
143
|
+
})
|
|
144
|
+
|
|
145
|
+
const config = await container.get(BOUND_CONFIG)
|
|
146
|
+
expect(config.apiUrl).toBe('https://class-override.com')
|
|
147
|
+
|
|
148
|
+
await container.dispose()
|
|
149
|
+
})
|
|
150
|
+
|
|
151
|
+
it('should support bound tokens in providers with useFactory override', async () => {
|
|
152
|
+
const configSchema = z.object({
|
|
153
|
+
apiUrl: z.string(),
|
|
154
|
+
})
|
|
155
|
+
|
|
156
|
+
const CONFIG_TOKEN = InjectionToken.create<
|
|
157
|
+
{ apiUrl: string },
|
|
158
|
+
typeof configSchema
|
|
159
|
+
>('CONFIG', configSchema)
|
|
160
|
+
|
|
161
|
+
const BOUND_CONFIG = InjectionToken.bound(CONFIG_TOKEN, {
|
|
162
|
+
apiUrl: 'https://default.com',
|
|
163
|
+
})
|
|
164
|
+
|
|
165
|
+
const container = new UnitTestContainer({
|
|
166
|
+
providers: [
|
|
167
|
+
{
|
|
168
|
+
token: BOUND_CONFIG,
|
|
169
|
+
useFactory: () => ({ apiUrl: 'https://factory-override.com' }),
|
|
170
|
+
},
|
|
171
|
+
],
|
|
172
|
+
})
|
|
173
|
+
|
|
174
|
+
const config = await container.get(BOUND_CONFIG)
|
|
175
|
+
expect(config.apiUrl).toBe('https://factory-override.com')
|
|
176
|
+
|
|
177
|
+
await container.dispose()
|
|
178
|
+
})
|
|
179
|
+
|
|
180
|
+
it('should support bound tokens in providers without override', async () => {
|
|
181
|
+
const configSchema = z.object({
|
|
182
|
+
apiUrl: z.string(),
|
|
183
|
+
timeout: z.number(),
|
|
184
|
+
})
|
|
185
|
+
|
|
186
|
+
const CONFIG_TOKEN = InjectionToken.create<
|
|
187
|
+
{ apiUrl: string; timeout: number },
|
|
188
|
+
typeof configSchema
|
|
189
|
+
>('CONFIG', configSchema)
|
|
190
|
+
|
|
191
|
+
const BOUND_CONFIG = InjectionToken.bound(CONFIG_TOKEN, {
|
|
192
|
+
apiUrl: 'https://bound-value.com',
|
|
193
|
+
timeout: 3000,
|
|
194
|
+
})
|
|
195
|
+
|
|
196
|
+
const container = new UnitTestContainer({
|
|
197
|
+
providers: [{ token: BOUND_CONFIG }],
|
|
198
|
+
})
|
|
199
|
+
|
|
200
|
+
const config = await container.get(BOUND_CONFIG)
|
|
201
|
+
expect(config.apiUrl).toBe('https://bound-value.com')
|
|
202
|
+
expect(config.timeout).toBe(3000)
|
|
203
|
+
|
|
204
|
+
await container.dispose()
|
|
205
|
+
})
|
|
84
206
|
})
|
|
85
207
|
|
|
86
208
|
describe('Auto-Mock Mode', () => {
|
|
@@ -112,7 +234,7 @@ describe('UnitTestContainer', () => {
|
|
|
112
234
|
allowUnregistered: true,
|
|
113
235
|
})
|
|
114
236
|
|
|
115
|
-
const service = await container.get(AutoMocked) as any
|
|
237
|
+
const service = (await container.get(AutoMocked)) as any
|
|
116
238
|
|
|
117
239
|
expect(() => service.doSomething()).toThrow(/auto-mocked service/)
|
|
118
240
|
|
|
@@ -279,7 +401,9 @@ describe('UnitTestContainer', () => {
|
|
|
279
401
|
providers: [{ token: NotResolvedService }],
|
|
280
402
|
})
|
|
281
403
|
|
|
282
|
-
expect(() =>
|
|
404
|
+
expect(() =>
|
|
405
|
+
container.expectNotResolved(NotResolvedService),
|
|
406
|
+
).not.toThrow()
|
|
283
407
|
|
|
284
408
|
await container.dispose()
|
|
285
409
|
})
|
|
@@ -10,12 +10,18 @@ import type {
|
|
|
10
10
|
|
|
11
11
|
import { Container } from '../container/container.mjs'
|
|
12
12
|
import { InjectableScope, InjectableType } from '../enums/index.mjs'
|
|
13
|
-
import {
|
|
13
|
+
import {
|
|
14
|
+
BoundInjectionToken,
|
|
15
|
+
InjectionToken,
|
|
16
|
+
} from '../token/injection-token.mjs'
|
|
14
17
|
import { globalRegistry, Registry } from '../token/registry.mjs'
|
|
15
18
|
import { getInjectableToken } from '../utils/get-injectable-token.mjs'
|
|
16
19
|
import { defaultInjectors } from '../utils/index.mjs'
|
|
17
20
|
|
|
18
|
-
type AnyToken =
|
|
21
|
+
type AnyToken =
|
|
22
|
+
| InjectionToken<any, any>
|
|
23
|
+
| BoundInjectionToken<any, any>
|
|
24
|
+
| (new (...args: any[]) => any)
|
|
19
25
|
|
|
20
26
|
/**
|
|
21
27
|
* TestContainer extends Container with testing utilities.
|
|
@@ -87,10 +93,14 @@ export class TestContainer extends Container {
|
|
|
87
93
|
* container.bind(UserService).toValue(mockUserService)
|
|
88
94
|
* container.bind(DatabaseToken).toClass(MockDatabase)
|
|
89
95
|
* container.bind(ConfigToken).toFactory(() => ({ apiKey: 'test' }))
|
|
96
|
+
* container.bind(BOUND_CONFIG_TOKEN).toValue(overrideValue)
|
|
90
97
|
* ```
|
|
91
98
|
*/
|
|
92
99
|
bind<T>(
|
|
93
|
-
token:
|
|
100
|
+
token:
|
|
101
|
+
| InjectionToken<T, any>
|
|
102
|
+
| BoundInjectionToken<T, any>
|
|
103
|
+
| (new (...args: any[]) => T),
|
|
94
104
|
): BindingBuilder<T> {
|
|
95
105
|
const realToken = this.resolveToken(token)
|
|
96
106
|
const tokenId = realToken.id
|
|
@@ -503,6 +513,9 @@ export class TestContainer extends Container {
|
|
|
503
513
|
if (typeof token === 'function') {
|
|
504
514
|
return getInjectableToken(token)
|
|
505
515
|
}
|
|
516
|
+
if (token instanceof BoundInjectionToken) {
|
|
517
|
+
return token.token
|
|
518
|
+
}
|
|
506
519
|
return token
|
|
507
520
|
}
|
|
508
521
|
|
|
@@ -512,14 +525,17 @@ export class TestContainer extends Container {
|
|
|
512
525
|
): void {
|
|
513
526
|
// Create a simple class that returns the value
|
|
514
527
|
const ValueHolder = class {
|
|
515
|
-
|
|
528
|
+
create(): T {
|
|
529
|
+
return value
|
|
530
|
+
}
|
|
516
531
|
}
|
|
517
532
|
|
|
518
533
|
this.testRegistry.set(
|
|
519
534
|
token,
|
|
520
535
|
InjectableScope.Singleton,
|
|
521
536
|
ValueHolder,
|
|
522
|
-
InjectableType.
|
|
537
|
+
InjectableType.Factory,
|
|
538
|
+
1000, // Higher priority for test overrides
|
|
523
539
|
)
|
|
524
540
|
|
|
525
541
|
// Store the instance directly
|
|
@@ -543,6 +559,7 @@ export class TestContainer extends Container {
|
|
|
543
559
|
InjectableScope.Singleton,
|
|
544
560
|
cls,
|
|
545
561
|
InjectableType.Class,
|
|
562
|
+
1000, // Higher priority for test overrides
|
|
546
563
|
)
|
|
547
564
|
}
|
|
548
565
|
|
|
@@ -563,6 +580,7 @@ export class TestContainer extends Container {
|
|
|
563
580
|
InjectableScope.Singleton,
|
|
564
581
|
FactoryWrapper,
|
|
565
582
|
InjectableType.Factory,
|
|
583
|
+
1000, // Higher priority for test overrides
|
|
566
584
|
)
|
|
567
585
|
}
|
|
568
586
|
|