@stratal/testing 0.0.2 → 0.0.5

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 (106) hide show
  1. package/CHANGELOG.md +117 -0
  2. package/dist/auth/acting-as.d.ts +21 -0
  3. package/dist/auth/acting-as.d.ts.map +1 -0
  4. package/dist/auth/acting-as.js +68 -0
  5. package/dist/auth/acting-as.js.map +1 -0
  6. package/dist/auth/index.d.ts +2 -0
  7. package/dist/auth/index.d.ts.map +1 -0
  8. package/dist/auth/index.js +2 -0
  9. package/dist/auth/index.js.map +1 -0
  10. package/dist/core/env/index.d.ts +2 -0
  11. package/dist/core/env/index.d.ts.map +1 -0
  12. package/dist/core/env/index.js +2 -0
  13. package/dist/core/env/index.js.map +1 -0
  14. package/dist/core/env/test-env.d.ts +9 -0
  15. package/dist/core/env/test-env.d.ts.map +1 -0
  16. package/dist/core/env/test-env.js +14 -0
  17. package/dist/core/env/test-env.js.map +1 -0
  18. package/dist/core/http/fetch-mock.d.ts +236 -0
  19. package/dist/core/http/fetch-mock.d.ts.map +1 -0
  20. package/dist/core/http/fetch-mock.js +290 -0
  21. package/dist/core/http/fetch-mock.js.map +1 -0
  22. package/dist/core/http/fetch-mock.types.d.ts +48 -0
  23. package/dist/core/http/fetch-mock.types.d.ts.map +1 -0
  24. package/dist/core/http/fetch-mock.types.js +2 -0
  25. package/dist/core/http/fetch-mock.types.js.map +1 -0
  26. package/dist/core/http/index.d.ts +6 -0
  27. package/dist/core/http/index.d.ts.map +1 -0
  28. package/dist/core/http/index.js +5 -0
  29. package/dist/core/http/index.js.map +1 -0
  30. package/dist/core/http/test-http-client.d.ts +54 -0
  31. package/dist/core/http/test-http-client.d.ts.map +1 -0
  32. package/dist/core/http/test-http-client.js +75 -0
  33. package/dist/core/http/test-http-client.js.map +1 -0
  34. package/dist/core/http/test-http-request.d.ts +60 -0
  35. package/dist/core/http/test-http-request.d.ts.map +1 -0
  36. package/dist/core/http/test-http-request.js +106 -0
  37. package/dist/core/http/test-http-request.js.map +1 -0
  38. package/dist/core/http/test-response.d.ts +161 -0
  39. package/dist/core/http/test-response.d.ts.map +1 -0
  40. package/dist/core/http/test-response.js +309 -0
  41. package/dist/core/http/test-response.js.map +1 -0
  42. package/dist/core/index.d.ts +7 -0
  43. package/dist/core/index.d.ts.map +1 -0
  44. package/dist/core/index.js +7 -0
  45. package/dist/core/index.js.map +1 -0
  46. package/dist/core/override/index.d.ts +2 -0
  47. package/dist/core/override/index.d.ts.map +1 -0
  48. package/dist/core/override/index.js +2 -0
  49. package/dist/core/override/index.js.map +1 -0
  50. package/dist/core/override/provider-override-builder.d.ts +77 -0
  51. package/dist/core/override/provider-override-builder.d.ts.map +1 -0
  52. package/dist/core/override/provider-override-builder.js +94 -0
  53. package/dist/core/override/provider-override-builder.js.map +1 -0
  54. package/dist/core/test.d.ts +48 -0
  55. package/dist/core/test.d.ts.map +1 -0
  56. package/dist/core/test.js +53 -0
  57. package/dist/core/test.js.map +1 -0
  58. package/dist/core/testing-module-builder.d.ts +99 -0
  59. package/dist/core/testing-module-builder.d.ts.map +1 -0
  60. package/dist/core/testing-module-builder.js +156 -0
  61. package/dist/core/testing-module-builder.js.map +1 -0
  62. package/dist/core/testing-module.d.ts +101 -0
  63. package/dist/core/testing-module.d.ts.map +1 -0
  64. package/dist/core/testing-module.js +178 -0
  65. package/dist/core/testing-module.js.map +1 -0
  66. package/dist/errors/index.d.ts +3 -0
  67. package/dist/errors/index.d.ts.map +1 -0
  68. package/dist/errors/index.js +3 -0
  69. package/dist/errors/index.js.map +1 -0
  70. package/dist/errors/setup-error.d.ts +9 -0
  71. package/dist/errors/setup-error.d.ts.map +1 -0
  72. package/dist/errors/setup-error.js +11 -0
  73. package/dist/errors/setup-error.js.map +1 -0
  74. package/dist/errors/test-error.d.ts +9 -0
  75. package/dist/errors/test-error.d.ts.map +1 -0
  76. package/dist/errors/test-error.js +15 -0
  77. package/dist/errors/test-error.js.map +1 -0
  78. package/dist/index.d.ts +15 -0
  79. package/dist/index.d.ts.map +1 -1
  80. package/dist/index.js +6 -3
  81. package/dist/index.js.map +1 -1
  82. package/dist/mocks/index.d.ts +3 -0
  83. package/dist/mocks/index.d.ts.map +1 -0
  84. package/dist/mocks/index.js +3 -0
  85. package/dist/mocks/index.js.map +1 -0
  86. package/dist/mocks/nodemailer.d.ts +10 -0
  87. package/dist/mocks/nodemailer.d.ts.map +1 -0
  88. package/dist/mocks/nodemailer.js +9 -0
  89. package/dist/mocks/nodemailer.js.map +1 -0
  90. package/dist/mocks/zenstack-language.d.ts +46 -0
  91. package/dist/mocks/zenstack-language.d.ts.map +1 -0
  92. package/dist/mocks/zenstack-language.js +47 -0
  93. package/dist/mocks/zenstack-language.js.map +1 -0
  94. package/dist/storage/fake-storage.service.d.ts +114 -0
  95. package/dist/storage/fake-storage.service.d.ts.map +1 -0
  96. package/dist/storage/fake-storage.service.js +233 -0
  97. package/dist/storage/fake-storage.service.js.map +1 -0
  98. package/dist/storage/index.d.ts +2 -0
  99. package/dist/storage/index.d.ts.map +1 -0
  100. package/dist/storage/index.js +2 -0
  101. package/dist/storage/index.js.map +1 -0
  102. package/dist/types.d.ts +5 -0
  103. package/dist/types.d.ts.map +1 -0
  104. package/dist/types.js +3 -0
  105. package/dist/types.js.map +1 -0
  106. package/package.json +27 -10
@@ -0,0 +1,94 @@
1
+ /**
2
+ * Fluent builder for provider overrides
3
+ *
4
+ * Provides a NestJS-style API for overriding providers in tests.
5
+ *
6
+ * @example
7
+ * ```typescript
8
+ * const module = await Test.createTestingModule({
9
+ * imports: [RegistrationModule],
10
+ * })
11
+ * .overrideProvider(EMAIL_TOKENS.EmailService)
12
+ * .useValue(mockEmailService)
13
+ * .compile()
14
+ * ```
15
+ */
16
+ export class ProviderOverrideBuilder {
17
+ parent;
18
+ token;
19
+ constructor(parent, token) {
20
+ this.parent = parent;
21
+ this.token = token;
22
+ }
23
+ /**
24
+ * Use a static value as the provider
25
+ *
26
+ * The value will be registered directly in the container.
27
+ *
28
+ * @param value - The value to use as the provider
29
+ * @returns The parent TestingModuleBuilder for chaining
30
+ */
31
+ useValue(value) {
32
+ return this.parent.addProviderOverride({
33
+ token: this.token,
34
+ type: 'value',
35
+ implementation: value,
36
+ });
37
+ }
38
+ /**
39
+ * Use a class as the provider
40
+ *
41
+ * The class will be registered as a singleton.
42
+ *
43
+ * @param cls - The class constructor to use as the provider
44
+ * @returns The parent TestingModuleBuilder for chaining
45
+ */
46
+ useClass(cls) {
47
+ return this.parent.addProviderOverride({
48
+ token: this.token,
49
+ type: 'class',
50
+ implementation: cls,
51
+ });
52
+ }
53
+ /**
54
+ * Use a factory function as the provider
55
+ *
56
+ * The factory receives the container and should return the provider instance.
57
+ *
58
+ * @param factory - Factory function that creates the provider
59
+ * @returns The parent TestingModuleBuilder for chaining
60
+ */
61
+ useFactory(factory) {
62
+ return this.parent.addProviderOverride({
63
+ token: this.token,
64
+ type: 'factory',
65
+ implementation: factory,
66
+ });
67
+ }
68
+ /**
69
+ * Use an existing token as the provider (alias)
70
+ *
71
+ * The override token will resolve to the same instance as the target token.
72
+ *
73
+ * @param existingToken - The token to alias
74
+ * @returns The parent TestingModuleBuilder for chaining
75
+ *
76
+ * @example
77
+ * ```typescript
78
+ * const module = await Test.createTestingModule({
79
+ * imports: [MyModule],
80
+ * })
81
+ * .overrideProvider(ABSTRACT_TOKEN)
82
+ * .useExisting(ConcreteService)
83
+ * .compile()
84
+ * ```
85
+ */
86
+ useExisting(existingToken) {
87
+ return this.parent.addProviderOverride({
88
+ token: this.token,
89
+ type: 'existing',
90
+ implementation: existingToken,
91
+ });
92
+ }
93
+ }
94
+ //# sourceMappingURL=provider-override-builder.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"provider-override-builder.js","sourceRoot":"","sources":["../../../src/core/override/provider-override-builder.ts"],"names":[],"mappings":"AAYA;;;;;;;;;;;;;;GAcG;AACH,MAAM,OAAO,uBAAuB;IAEf;IACA;IAFnB,YACmB,MAA4B,EAC5B,KAAwB;QADxB,WAAM,GAAN,MAAM,CAAsB;QAC5B,UAAK,GAAL,KAAK,CAAmB;IACvC,CAAC;IAEL;;;;;;;OAOG;IACH,QAAQ,CAAC,KAAQ;QACf,OAAO,IAAI,CAAC,MAAM,CAAC,mBAAmB,CAAC;YACrC,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,IAAI,EAAE,OAAO;YACb,cAAc,EAAE,KAAK;SACtB,CAAC,CAAA;IACJ,CAAC;IAED;;;;;;;OAOG;IACH,QAAQ,CAAC,GAAkC;QACzC,OAAO,IAAI,CAAC,MAAM,CAAC,mBAAmB,CAAC;YACrC,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,IAAI,EAAE,OAAO;YACb,cAAc,EAAE,GAAG;SACpB,CAAC,CAAA;IACJ,CAAC;IAED;;;;;;;OAOG;IACH,UAAU,CAAC,OAAoC;QAC7C,OAAO,IAAI,CAAC,MAAM,CAAC,mBAAmB,CAAC;YACrC,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,IAAI,EAAE,SAAS;YACf,cAAc,EAAE,OAAO;SACxB,CAAC,CAAA;IACJ,CAAC;IAED;;;;;;;;;;;;;;;;;OAiBG;IACH,WAAW,CAAC,aAAgC;QAC1C,OAAO,IAAI,CAAC,MAAM,CAAC,mBAAmB,CAAC;YACrC,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,IAAI,EAAE,UAAU;YAChB,cAAc,EAAE,aAAa;SAC9B,CAAC,CAAA;IACJ,CAAC;CACF"}
@@ -0,0 +1,48 @@
1
+ import { type DynamicModule, type ModuleClass } from 'stratal/module';
2
+ import { TestingModuleBuilder, type TestingModuleConfig } from './testing-module-builder';
3
+ /**
4
+ * Test
5
+ *
6
+ * Static class for creating testing modules.
7
+ * Provides a NestJS-style API for configuring test modules.
8
+ *
9
+ * @example
10
+ * ```typescript
11
+ * // In vitest.setup.ts:
12
+ * Test.setBaseModules([CoreModule])
13
+ *
14
+ * // In test files:
15
+ * const module = await Test.createTestingModule({
16
+ * imports: [RegistrationModule, GeoModule],
17
+ * })
18
+ * .overrideProvider(EMAIL_TOKENS.EmailService)
19
+ * .useValue(mockEmailService)
20
+ * .compile()
21
+ * ```
22
+ */
23
+ export declare class Test {
24
+ /**
25
+ * Base modules to include in all test modules
26
+ * Set once in vitest.setup.ts
27
+ */
28
+ private static baseModules;
29
+ /**
30
+ * Set base modules to include in all test modules
31
+ * Should be called once in vitest.setup.ts
32
+ *
33
+ * @param modules - Modules to include before test-specific modules (e.g., CoreModule)
34
+ */
35
+ static setBaseModules(modules: (ModuleClass | DynamicModule)[]): void;
36
+ /**
37
+ * Get base modules
38
+ */
39
+ static getBaseModules(): (ModuleClass | DynamicModule)[];
40
+ /**
41
+ * Create a testing module builder
42
+ *
43
+ * @param config - Configuration with modules and optional env overrides
44
+ * @returns TestingModuleBuilder for configuring and compiling the module
45
+ */
46
+ static createTestingModule(config: TestingModuleConfig): TestingModuleBuilder;
47
+ }
48
+ //# sourceMappingURL=test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"test.d.ts","sourceRoot":"","sources":["../../src/core/test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,aAAa,EAAE,KAAK,WAAW,EAAE,MAAM,gBAAgB,CAAA;AACrE,OAAO,EAAE,oBAAoB,EAAE,KAAK,mBAAmB,EAAE,MAAM,0BAA0B,CAAA;AAEzF;;;;;;;;;;;;;;;;;;;GAmBG;AACH,qBAAa,IAAI;IACf;;;OAGG;IACH,OAAO,CAAC,MAAM,CAAC,WAAW,CAAsC;IAEhE;;;;;OAKG;IACH,MAAM,CAAC,cAAc,CAAC,OAAO,EAAE,CAAC,WAAW,GAAG,aAAa,CAAC,EAAE,GAAG,IAAI;IAIrE;;OAEG;IACH,MAAM,CAAC,cAAc,IAAI,CAAC,WAAW,GAAG,aAAa,CAAC,EAAE;IAIxD;;;;;OAKG;IACH,MAAM,CAAC,mBAAmB,CAAC,MAAM,EAAE,mBAAmB,GAAG,oBAAoB;CAG9E"}
@@ -0,0 +1,53 @@
1
+ import { TestingModuleBuilder } from './testing-module-builder';
2
+ /**
3
+ * Test
4
+ *
5
+ * Static class for creating testing modules.
6
+ * Provides a NestJS-style API for configuring test modules.
7
+ *
8
+ * @example
9
+ * ```typescript
10
+ * // In vitest.setup.ts:
11
+ * Test.setBaseModules([CoreModule])
12
+ *
13
+ * // In test files:
14
+ * const module = await Test.createTestingModule({
15
+ * imports: [RegistrationModule, GeoModule],
16
+ * })
17
+ * .overrideProvider(EMAIL_TOKENS.EmailService)
18
+ * .useValue(mockEmailService)
19
+ * .compile()
20
+ * ```
21
+ */
22
+ export class Test {
23
+ /**
24
+ * Base modules to include in all test modules
25
+ * Set once in vitest.setup.ts
26
+ */
27
+ static baseModules = [];
28
+ /**
29
+ * Set base modules to include in all test modules
30
+ * Should be called once in vitest.setup.ts
31
+ *
32
+ * @param modules - Modules to include before test-specific modules (e.g., CoreModule)
33
+ */
34
+ static setBaseModules(modules) {
35
+ this.baseModules = modules;
36
+ }
37
+ /**
38
+ * Get base modules
39
+ */
40
+ static getBaseModules() {
41
+ return this.baseModules;
42
+ }
43
+ /**
44
+ * Create a testing module builder
45
+ *
46
+ * @param config - Configuration with modules and optional env overrides
47
+ * @returns TestingModuleBuilder for configuring and compiling the module
48
+ */
49
+ static createTestingModule(config) {
50
+ return new TestingModuleBuilder(config);
51
+ }
52
+ }
53
+ //# sourceMappingURL=test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"test.js","sourceRoot":"","sources":["../../src/core/test.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,oBAAoB,EAA4B,MAAM,0BAA0B,CAAA;AAEzF;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAM,OAAO,IAAI;IACf;;;OAGG;IACK,MAAM,CAAC,WAAW,GAAoC,EAAE,CAAA;IAEhE;;;;;OAKG;IACH,MAAM,CAAC,cAAc,CAAC,OAAwC;QAC5D,IAAI,CAAC,WAAW,GAAG,OAAO,CAAA;IAC5B,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,cAAc;QACnB,OAAO,IAAI,CAAC,WAAW,CAAA;IACzB,CAAC;IAED;;;;;OAKG;IACH,MAAM,CAAC,mBAAmB,CAAC,MAA2B;QACpD,OAAO,IAAI,oBAAoB,CAAC,MAAM,CAAC,CAAA;IACzC,CAAC"}
@@ -0,0 +1,99 @@
1
+ import { ApplicationConfig, type StratalEnv } from 'stratal';
2
+ import { InjectionToken, ModuleOptions } from 'stratal/module';
3
+ import { ProviderOverrideBuilder, type ProviderOverrideConfig } from './override';
4
+ import { TestingModule } from './testing-module';
5
+ /**
6
+ * Configuration for creating a testing module
7
+ *
8
+ * Extends ModuleOptions to support all module properties like NestJS.
9
+ *
10
+ * @example
11
+ * ```typescript
12
+ * const module = await Test.createTestingModule({
13
+ * imports: [RegistrationModule, GeoModule],
14
+ * providers: [{ provide: MOCK_TOKEN, useValue: mockValue }],
15
+ * controllers: [TestController],
16
+ * }).compile()
17
+ * ```
18
+ */
19
+ export interface TestingModuleConfig extends ModuleOptions {
20
+ /** Optional environment overrides */
21
+ env?: Partial<StratalEnv>;
22
+ /** Logging configuration. Defaults: level=ERROR, formatter='json' */
23
+ logging?: ApplicationConfig['logging'];
24
+ }
25
+ /**
26
+ * Builder for creating test modules with provider overrides
27
+ *
28
+ * Provides a NestJS-style fluent API for configuring test modules.
29
+ * Supports all ModuleOptions properties (imports, providers, controllers, consumers, jobs).
30
+ *
31
+ * @example Basic usage
32
+ * ```typescript
33
+ * const module = await Test.createTestingModule({
34
+ * imports: [RegistrationModule],
35
+ * }).compile()
36
+ * ```
37
+ *
38
+ * @example Provider override
39
+ * ```typescript
40
+ * const module = await Test.createTestingModule({
41
+ * imports: [RegistrationModule, GeoModule],
42
+ * })
43
+ * .overrideProvider(EMAIL_TOKENS.EmailService)
44
+ * .useValue(mockEmailService)
45
+ * .compile()
46
+ * ```
47
+ *
48
+ * @example Full ModuleOptions
49
+ * ```typescript
50
+ * const module = await Test.createTestingModule({
51
+ * imports: [RegistrationModule],
52
+ * providers: [{ provide: MOCK_TOKEN, useValue: mockValue }],
53
+ * controllers: [TestController],
54
+ * consumers: [TestConsumer],
55
+ * jobs: [TestJob],
56
+ * }).compile()
57
+ * ```
58
+ */
59
+ export declare class TestingModuleBuilder {
60
+ private config;
61
+ private overrides;
62
+ constructor(config: TestingModuleConfig);
63
+ /**
64
+ * Override a provider with a custom implementation
65
+ *
66
+ * Returns a ProviderOverrideBuilder for specifying the override type.
67
+ *
68
+ * @param token - The injection token to override
69
+ * @returns ProviderOverrideBuilder for chaining useValue/useClass/useFactory
70
+ */
71
+ overrideProvider<T>(token: InjectionToken<T>): ProviderOverrideBuilder<T>;
72
+ /**
73
+ * Add a provider override (internal use by ProviderOverrideBuilder)
74
+ *
75
+ * @internal
76
+ */
77
+ addProviderOverride<T>(override: ProviderOverrideConfig<T>): this;
78
+ /**
79
+ * Merge additional environment bindings
80
+ *
81
+ * @param env - Partial environment to merge
82
+ * @returns This builder for chaining
83
+ */
84
+ withEnv(env: Partial<StratalEnv>): this;
85
+ /**
86
+ * Compile the testing module
87
+ *
88
+ * Creates the Application, applies overrides, initializes, and returns TestingModule.
89
+ * The ModuleRegistry handles all module registrations automatically.
90
+ *
91
+ * @returns Promise<TestingModule> - The compiled testing module
92
+ */
93
+ compile(): Promise<TestingModule>;
94
+ /**
95
+ * Create a test root module with the given options
96
+ */
97
+ private createTestRootModule;
98
+ }
99
+ //# sourceMappingURL=testing-module-builder.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testing-module-builder.d.ts","sourceRoot":"","sources":["../../src/core/testing-module-builder.ts"],"names":[],"mappings":"AACA,OAAO,EAEL,iBAAiB,EAEjB,KAAK,UAAU,EAChB,MAAM,SAAS,CAAA;AAGhB,OAAO,EAAE,cAAc,EAAuB,aAAa,EAAE,MAAM,gBAAgB,CAAA;AAInF,OAAO,EAAE,uBAAuB,EAAE,KAAK,sBAAsB,EAAE,MAAM,YAAY,CAAA;AAEjF,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAA;AAEhD;;;;;;;;;;;;;GAaG;AACH,MAAM,WAAW,mBAAoB,SAAQ,aAAa;IACxD,qCAAqC;IACrC,GAAG,CAAC,EAAE,OAAO,CAAC,UAAU,CAAC,CAAA;IACzB,qEAAqE;IACrE,OAAO,CAAC,EAAE,iBAAiB,CAAC,SAAS,CAAC,CAAA;CACvC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAiCG;AACH,qBAAa,oBAAoB;IAGnB,OAAO,CAAC,MAAM;IAF1B,OAAO,CAAC,SAAS,CAAuC;gBAEpC,MAAM,EAAE,mBAAmB;IAE/C;;;;;;;OAOG;IACH,gBAAgB,CAAC,CAAC,EAAE,KAAK,EAAE,cAAc,CAAC,CAAC,CAAC,GAAG,uBAAuB,CAAC,CAAC,CAAC;IAIzE;;;;OAIG;IACH,mBAAmB,CAAC,CAAC,EAAE,QAAQ,EAAE,sBAAsB,CAAC,CAAC,CAAC,GAAG,IAAI;IAKjE;;;;;OAKG;IACH,OAAO,CAAC,GAAG,EAAE,OAAO,CAAC,UAAU,CAAC,GAAG,IAAI;IAKvC;;;;;;;OAOG;IACG,OAAO,IAAI,OAAO,CAAC,aAAa,CAAC;IAmEvC;;OAEG;IACH,OAAO,CAAC,oBAAoB;CAK7B"}
@@ -0,0 +1,156 @@
1
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
2
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
3
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
4
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
5
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
6
+ };
7
+ import { createExecutionContext } from 'cloudflare:test';
8
+ import { Application, } from 'stratal';
9
+ import { LogLevel } from 'stratal/logger';
10
+ import { Module } from 'stratal/module';
11
+ import { STORAGE_TOKENS } from 'stratal/storage';
12
+ import { FakeStorageService } from '../storage';
13
+ import { getTestEnv } from './env';
14
+ import { ProviderOverrideBuilder } from './override';
15
+ import { Test } from './test';
16
+ import { TestingModule } from './testing-module';
17
+ /**
18
+ * Builder for creating test modules with provider overrides
19
+ *
20
+ * Provides a NestJS-style fluent API for configuring test modules.
21
+ * Supports all ModuleOptions properties (imports, providers, controllers, consumers, jobs).
22
+ *
23
+ * @example Basic usage
24
+ * ```typescript
25
+ * const module = await Test.createTestingModule({
26
+ * imports: [RegistrationModule],
27
+ * }).compile()
28
+ * ```
29
+ *
30
+ * @example Provider override
31
+ * ```typescript
32
+ * const module = await Test.createTestingModule({
33
+ * imports: [RegistrationModule, GeoModule],
34
+ * })
35
+ * .overrideProvider(EMAIL_TOKENS.EmailService)
36
+ * .useValue(mockEmailService)
37
+ * .compile()
38
+ * ```
39
+ *
40
+ * @example Full ModuleOptions
41
+ * ```typescript
42
+ * const module = await Test.createTestingModule({
43
+ * imports: [RegistrationModule],
44
+ * providers: [{ provide: MOCK_TOKEN, useValue: mockValue }],
45
+ * controllers: [TestController],
46
+ * consumers: [TestConsumer],
47
+ * jobs: [TestJob],
48
+ * }).compile()
49
+ * ```
50
+ */
51
+ export class TestingModuleBuilder {
52
+ config;
53
+ overrides = [];
54
+ constructor(config) {
55
+ this.config = config;
56
+ }
57
+ /**
58
+ * Override a provider with a custom implementation
59
+ *
60
+ * Returns a ProviderOverrideBuilder for specifying the override type.
61
+ *
62
+ * @param token - The injection token to override
63
+ * @returns ProviderOverrideBuilder for chaining useValue/useClass/useFactory
64
+ */
65
+ overrideProvider(token) {
66
+ return new ProviderOverrideBuilder(this, token);
67
+ }
68
+ /**
69
+ * Add a provider override (internal use by ProviderOverrideBuilder)
70
+ *
71
+ * @internal
72
+ */
73
+ addProviderOverride(override) {
74
+ this.overrides.push(override);
75
+ return this;
76
+ }
77
+ /**
78
+ * Merge additional environment bindings
79
+ *
80
+ * @param env - Partial environment to merge
81
+ * @returns This builder for chaining
82
+ */
83
+ withEnv(env) {
84
+ this.config.env = { ...this.config.env, ...env };
85
+ return this;
86
+ }
87
+ /**
88
+ * Compile the testing module
89
+ *
90
+ * Creates the Application, applies overrides, initializes, and returns TestingModule.
91
+ * The ModuleRegistry handles all module registrations automatically.
92
+ *
93
+ * @returns Promise<TestingModule> - The compiled testing module
94
+ */
95
+ async compile() {
96
+ // 1. Create environment (cloudflare:test base + overrides)
97
+ const env = getTestEnv(this.config.env);
98
+ // 2. Create ExecutionContext
99
+ const ctx = createExecutionContext();
100
+ // 3. Build root module from config (supports all ModuleOptions)
101
+ const baseModules = Test.getBaseModules();
102
+ const configImports = this.config.imports ?? [];
103
+ const allImports = [...baseModules, ...configImports];
104
+ // Create root module with all config properties
105
+ const rootModule = this.createTestRootModule({
106
+ imports: allImports,
107
+ providers: this.config.providers,
108
+ controllers: this.config.controllers,
109
+ consumers: this.config.consumers,
110
+ jobs: this.config.jobs,
111
+ });
112
+ // 4. Create Application
113
+ const app = new Application(env, ctx, {
114
+ module: rootModule,
115
+ logging: {
116
+ level: this.config.logging?.level ?? LogLevel.ERROR,
117
+ formatter: this.config.logging?.formatter ?? 'pretty',
118
+ },
119
+ });
120
+ // 7. Initialize application - ModuleRegistry handles all module registrations
121
+ await app.initialize();
122
+ // 5. Auto-register FakeStorageService as singleton (can be overridden by user)
123
+ // Must be singleton so files uploaded in setup are available when consumers run
124
+ app.container.registerSingleton(STORAGE_TOKENS.StorageService, FakeStorageService);
125
+ // 6. Apply user overrides BEFORE initialize
126
+ for (const override of this.overrides) {
127
+ switch (override.type) {
128
+ case 'value':
129
+ app.container.registerValue(override.token, override.implementation);
130
+ break;
131
+ case 'class':
132
+ app.container.registerSingleton(override.token, override.implementation);
133
+ break;
134
+ case 'factory':
135
+ app.container.registerFactory(override.token, override.implementation);
136
+ break;
137
+ case 'existing':
138
+ app.container.registerExisting(override.token, override.implementation);
139
+ break;
140
+ }
141
+ }
142
+ return new TestingModule(app, env);
143
+ }
144
+ /**
145
+ * Create a test root module with the given options
146
+ */
147
+ createTestRootModule(options) {
148
+ let TestRootModule = class TestRootModule {
149
+ };
150
+ TestRootModule = __decorate([
151
+ Module(options)
152
+ ], TestRootModule);
153
+ return TestRootModule;
154
+ }
155
+ }
156
+ //# sourceMappingURL=testing-module-builder.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testing-module-builder.js","sourceRoot":"","sources":["../../src/core/testing-module-builder.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,101 @@
1
+ import type { ConnectionName, DatabaseService } from '@stratal/framework/database';
2
+ import type { Application, StratalEnv } from 'stratal';
3
+ import { type Container } from 'stratal/di';
4
+ import { type InjectionToken } from 'stratal/module';
5
+ import type { FakeStorageService } from '../storage';
6
+ import type { Seeder } from '../types';
7
+ import { TestHttpClient } from './http/test-http-client';
8
+ /**
9
+ * TestingModule
10
+ *
11
+ * Provides access to the test application, container, HTTP client, and utilities.
12
+ *
13
+ * @example
14
+ * ```typescript
15
+ * const module = await Test.createTestingModule({
16
+ * modules: [RegistrationModule],
17
+ * }).compile()
18
+ *
19
+ * // Make HTTP requests
20
+ * const response = await module.http
21
+ * .post('/api/v1/register')
22
+ * .withBody({ ... })
23
+ * .send()
24
+ *
25
+ * // Access services
26
+ * const service = module.get(REGISTRATION_TOKENS.RegistrationService)
27
+ *
28
+ * // Database utilities
29
+ * await module.truncateDb()
30
+ * await module.seed(new UserSeeder())
31
+ * await module.assertDatabaseHas('user', { email: 'test@example.com' })
32
+ *
33
+ * // Cleanup
34
+ * await module.close()
35
+ * ```
36
+ */
37
+ export declare class TestingModule {
38
+ private readonly app;
39
+ private readonly env;
40
+ private _http;
41
+ constructor(app: Application, env: StratalEnv);
42
+ /**
43
+ * Resolve a service from the container
44
+ */
45
+ get<T>(token: InjectionToken<T>): T;
46
+ /**
47
+ * Get HTTP test client for making requests
48
+ */
49
+ get http(): TestHttpClient;
50
+ /**
51
+ * Get fake storage service for assertions
52
+ */
53
+ get storage(): FakeStorageService;
54
+ /**
55
+ * Get Application instance
56
+ */
57
+ get application(): Application;
58
+ /**
59
+ * Get DI Container
60
+ */
61
+ get container(): Container;
62
+ /**
63
+ * Execute an HTTP request through RouterService
64
+ */
65
+ fetch(request: Request): Promise<Response>;
66
+ /**
67
+ * Run callback in request scope (for DB operations, service access)
68
+ */
69
+ runInRequestScope<T>(callback: () => T | Promise<T>): Promise<T>;
70
+ /**
71
+ * Get database service instance (resolved in request scope)
72
+ */
73
+ getDb(): Promise<DatabaseService>;
74
+ getDb<K extends ConnectionName>(name: K): Promise<DatabaseService<K>>;
75
+ /**
76
+ * Truncate all non-prisma tables in the database
77
+ */
78
+ truncateDb(name?: ConnectionName): Promise<void>;
79
+ /**
80
+ * Run seeders in a database transaction
81
+ */
82
+ seed(...seeders: Seeder[]): Promise<void>;
83
+ seed(name: ConnectionName, ...seeders: Seeder[]): Promise<void>;
84
+ /**
85
+ * Assert that a record exists in the database
86
+ */
87
+ assertDatabaseHas(table: string, data: Record<string, unknown>, name?: ConnectionName): Promise<void>;
88
+ /**
89
+ * Assert that a record does not exist in the database
90
+ */
91
+ assertDatabaseMissing(table: string, data: Record<string, unknown>, name?: ConnectionName): Promise<void>;
92
+ /**
93
+ * Assert the number of records in a table
94
+ */
95
+ assertDatabaseCount(table: string, expected: number, name?: ConnectionName): Promise<void>;
96
+ /**
97
+ * Cleanup - call in afterAll
98
+ */
99
+ close(): Promise<void>;
100
+ }
101
+ //# sourceMappingURL=testing-module.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testing-module.d.ts","sourceRoot":"","sources":["../../src/core/testing-module.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,6BAA6B,CAAA;AAElF,OAAO,KAAK,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,SAAS,CAAA;AACtD,OAAO,EAAa,KAAK,SAAS,EAAE,MAAM,YAAY,CAAA;AACtD,OAAO,EAAE,KAAK,cAAc,EAAE,MAAM,gBAAgB,CAAA;AAGpD,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAA;AACpD,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,UAAU,CAAA;AACtC,OAAO,EAAE,cAAc,EAAE,MAAM,yBAAyB,CAAA;AAExD;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA4BG;AACH,qBAAa,aAAa;IAItB,OAAO,CAAC,QAAQ,CAAC,GAAG;IACpB,OAAO,CAAC,QAAQ,CAAC,GAAG;IAJtB,OAAO,CAAC,KAAK,CAA8B;gBAGxB,GAAG,EAAE,WAAW,EAChB,GAAG,EAAE,UAAU;IAGlC;;OAEG;IACH,GAAG,CAAC,CAAC,EAAE,KAAK,EAAE,cAAc,CAAC,CAAC,CAAC,GAAG,CAAC;IAInC;;OAEG;IACH,IAAI,IAAI,IAAI,cAAc,CAGzB;IAED;;OAEG;IACH,IAAI,OAAO,IAAI,kBAAkB,CAEhC;IAED;;OAEG;IACH,IAAI,WAAW,IAAI,WAAW,CAE7B;IAED;;OAEG;IACH,IAAI,SAAS,IAAI,SAAS,CAEzB;IAED;;OAEG;IACG,KAAK,CAAC,OAAO,EAAE,OAAO,GAAG,OAAO,CAAC,QAAQ,CAAC;IAKhD;;OAEG;IACG,iBAAiB,CAAC,CAAC,EAAE,QAAQ,EAAE,MAAM,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;IAKtE;;OAEG;IACG,KAAK,IAAI,OAAO,CAAC,eAAe,CAAC;IACjC,KAAK,CAAC,CAAC,SAAS,cAAc,EAAE,IAAI,EAAE,CAAC,GAAG,OAAO,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC;IAQ3E;;OAEG;IACG,UAAU,CAAC,IAAI,CAAC,EAAE,cAAc,GAAG,OAAO,CAAC,IAAI,CAAC;IAetD;;OAEG;IACG,IAAI,CAAC,GAAG,OAAO,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;IACzC,IAAI,CAAC,IAAI,EAAE,cAAc,EAAE,GAAG,OAAO,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;IAuBrE;;OAEG;IACG,iBAAiB,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,IAAI,CAAC,EAAE,cAAc,GAAG,OAAO,CAAC,IAAI,CAAC;IAW3G;;OAEG;IACG,qBAAqB,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,IAAI,CAAC,EAAE,cAAc,GAAG,OAAO,CAAC,IAAI,CAAC;IAW/G;;OAEG;IACG,mBAAmB,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,cAAc,GAAG,OAAO,CAAC,IAAI,CAAC;IAWhG;;OAEG;IACG,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;CAG7B"}