@geekmidas/constructs 0.0.18 → 0.0.19

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 (165) hide show
  1. package/dist/{AWSLambdaFunction-H65WfXLt.mjs → AWSLambdaFunction-DBUENdP0.mjs} +2 -2
  2. package/dist/{AWSLambdaFunction-H65WfXLt.mjs.map → AWSLambdaFunction-DBUENdP0.mjs.map} +1 -1
  3. package/dist/{AWSLambdaFunction-C-fuCLA3.cjs → AWSLambdaFunction-vobYqQ0w.cjs} +2 -2
  4. package/dist/{AWSLambdaFunction-C-fuCLA3.cjs.map → AWSLambdaFunction-vobYqQ0w.cjs.map} +1 -1
  5. package/dist/{AWSLambdaSubscriberAdaptor-CyFh7MN8.mjs → AWSLambdaSubscriberAdaptor-BLHDyqzQ.mjs} +1 -1
  6. package/dist/{AWSLambdaSubscriberAdaptor-CyFh7MN8.mjs.map → AWSLambdaSubscriberAdaptor-BLHDyqzQ.mjs.map} +1 -1
  7. package/dist/{AWSLambdaSubscriberAdaptor-Dum5bkw3.cjs → AWSLambdaSubscriberAdaptor-DVC4VAQR.cjs} +1 -1
  8. package/dist/{AWSLambdaSubscriberAdaptor-Dum5bkw3.cjs.map → AWSLambdaSubscriberAdaptor-DVC4VAQR.cjs.map} +1 -1
  9. package/dist/{AmazonApiGatewayEndpointAdaptor-CI9L7Ucn.cjs → AmazonApiGatewayEndpointAdaptor-BLUW--OF.cjs} +4 -4
  10. package/dist/{AmazonApiGatewayEndpointAdaptor-CI9L7Ucn.cjs.map → AmazonApiGatewayEndpointAdaptor-BLUW--OF.cjs.map} +1 -1
  11. package/dist/{AmazonApiGatewayEndpointAdaptor-C6Jk5HSy.mjs → AmazonApiGatewayEndpointAdaptor-DBK53gB5.mjs} +4 -4
  12. package/dist/{AmazonApiGatewayEndpointAdaptor-C6Jk5HSy.mjs.map → AmazonApiGatewayEndpointAdaptor-DBK53gB5.mjs.map} +1 -1
  13. package/dist/{AmazonApiGatewayV1EndpointAdaptor-DYL1bCBS.cjs → AmazonApiGatewayV1EndpointAdaptor-B-i9_OtQ.cjs} +3 -3
  14. package/dist/{AmazonApiGatewayV1EndpointAdaptor-DYL1bCBS.cjs.map → AmazonApiGatewayV1EndpointAdaptor-B-i9_OtQ.cjs.map} +1 -1
  15. package/dist/{AmazonApiGatewayV1EndpointAdaptor-BMy8DdNJ.mjs → AmazonApiGatewayV1EndpointAdaptor-DfU3n5im.mjs} +3 -3
  16. package/dist/{AmazonApiGatewayV1EndpointAdaptor-BMy8DdNJ.mjs.map → AmazonApiGatewayV1EndpointAdaptor-DfU3n5im.mjs.map} +1 -1
  17. package/dist/{AmazonApiGatewayV2EndpointAdaptor-BU5wQMOe.mjs → AmazonApiGatewayV2EndpointAdaptor-D-AFyzaQ.mjs} +3 -3
  18. package/dist/{AmazonApiGatewayV2EndpointAdaptor-BU5wQMOe.mjs.map → AmazonApiGatewayV2EndpointAdaptor-D-AFyzaQ.mjs.map} +1 -1
  19. package/dist/{AmazonApiGatewayV2EndpointAdaptor-CPLCMeaN.cjs → AmazonApiGatewayV2EndpointAdaptor-D4k_Bg7Q.cjs} +3 -3
  20. package/dist/{AmazonApiGatewayV2EndpointAdaptor-CPLCMeaN.cjs.map → AmazonApiGatewayV2EndpointAdaptor-D4k_Bg7Q.cjs.map} +1 -1
  21. package/dist/{Cron-Bi3QOge_.cjs → Cron-CmtKQOmE.cjs} +1 -1
  22. package/dist/{Cron-Bi3QOge_.cjs.map → Cron-CmtKQOmE.cjs.map} +1 -1
  23. package/dist/{Cron-Dy_HW2Vv.mjs → Cron-mWi3PQxt.mjs} +1 -1
  24. package/dist/{Cron-Dy_HW2Vv.mjs.map → Cron-mWi3PQxt.mjs.map} +1 -1
  25. package/dist/{CronBuilder-Bl3A2Zp4.mjs → CronBuilder-4DxT6wUa.mjs} +2 -2
  26. package/dist/{CronBuilder-Bl3A2Zp4.mjs.map → CronBuilder-4DxT6wUa.mjs.map} +1 -1
  27. package/dist/{CronBuilder-Dv_w7Yri.cjs → CronBuilder-CeffP9Rs.cjs} +2 -2
  28. package/dist/{CronBuilder-Dv_w7Yri.cjs.map → CronBuilder-CeffP9Rs.cjs.map} +1 -1
  29. package/dist/{Endpoint-DDpF7NO1.cjs → Endpoint-BTvS2vwp.cjs} +1 -1
  30. package/dist/{Endpoint-DDpF7NO1.cjs.map → Endpoint-BTvS2vwp.cjs.map} +1 -1
  31. package/dist/{Endpoint-S6Yh2_PN.mjs → Endpoint-D2LVHBEO.mjs} +1 -1
  32. package/dist/{Endpoint-S6Yh2_PN.mjs.map → Endpoint-D2LVHBEO.mjs.map} +1 -1
  33. package/dist/{EndpointBuilder-CFnjYXmL.cjs → EndpointBuilder-C4qahFeS.cjs} +2 -2
  34. package/dist/{EndpointBuilder-CFnjYXmL.cjs.map → EndpointBuilder-C4qahFeS.cjs.map} +1 -1
  35. package/dist/{EndpointBuilder-DlDft4mJ.mjs → EndpointBuilder-O6B1zJ6v.mjs} +2 -2
  36. package/dist/{EndpointBuilder-DlDft4mJ.mjs.map → EndpointBuilder-O6B1zJ6v.mjs.map} +1 -1
  37. package/dist/{EndpointFactory-Ctln6czP.mjs → EndpointFactory-BUYrnjau.mjs} +2 -2
  38. package/dist/EndpointFactory-BUYrnjau.mjs.map +1 -0
  39. package/dist/{EndpointFactory-mTfi8x1X.cjs → EndpointFactory-C_neYSiA.cjs} +2 -2
  40. package/dist/EndpointFactory-C_neYSiA.cjs.map +1 -0
  41. package/dist/{FunctionExecutionWrapper-DkNycmOh.cjs → FunctionExecutionWrapper-B8agyYHk.cjs} +1 -1
  42. package/dist/{FunctionExecutionWrapper-DkNycmOh.cjs.map → FunctionExecutionWrapper-B8agyYHk.cjs.map} +1 -1
  43. package/dist/{FunctionExecutionWrapper-Bubnr0zA.mjs → FunctionExecutionWrapper-BPIdmPe8.mjs} +1 -1
  44. package/dist/{FunctionExecutionWrapper-Bubnr0zA.mjs.map → FunctionExecutionWrapper-BPIdmPe8.mjs.map} +1 -1
  45. package/dist/{HonoEndpointAdaptor-DsqGuEIb.d.mts → HonoEndpointAdaptor-Br1vuQ3A.d.mts} +3 -3
  46. package/dist/{HonoEndpointAdaptor-DajXbh80.d.cts → HonoEndpointAdaptor-C9wC10-w.d.cts} +3 -3
  47. package/dist/{HonoEndpointAdaptor-DuyE06nH.mjs → HonoEndpointAdaptor-DEFNrIv7.mjs} +5 -5
  48. package/dist/{HonoEndpointAdaptor-DuyE06nH.mjs.map → HonoEndpointAdaptor-DEFNrIv7.mjs.map} +1 -1
  49. package/dist/{HonoEndpointAdaptor-CfLRHHFw.cjs → HonoEndpointAdaptor-DbLeXkR6.cjs} +5 -5
  50. package/dist/{HonoEndpointAdaptor-CfLRHHFw.cjs.map → HonoEndpointAdaptor-DbLeXkR6.cjs.map} +1 -1
  51. package/dist/{TestEndpointAdaptor-DbwrL-RJ.mjs → TestEndpointAdaptor-BGrZsg5c.mjs} +38 -18
  52. package/dist/TestEndpointAdaptor-BGrZsg5c.mjs.map +1 -0
  53. package/dist/{TestEndpointAdaptor-DhRjJHyk.d.mts → TestEndpointAdaptor-Bl2ic-yr.d.mts} +9 -9
  54. package/dist/{TestEndpointAdaptor-B9tUIlCC.d.cts → TestEndpointAdaptor-ByXqQufk.d.cts} +9 -9
  55. package/dist/{TestEndpointAdaptor-B9hyZ-mF.cjs → TestEndpointAdaptor-JCvZ3VVi.cjs} +38 -18
  56. package/dist/TestEndpointAdaptor-JCvZ3VVi.cjs.map +1 -0
  57. package/dist/adaptors/aws.cjs +9 -9
  58. package/dist/adaptors/aws.d.cts +1 -1
  59. package/dist/adaptors/aws.d.mts +1 -1
  60. package/dist/adaptors/aws.mjs +9 -9
  61. package/dist/adaptors/hono.cjs +5 -5
  62. package/dist/adaptors/hono.d.cts +2 -2
  63. package/dist/adaptors/hono.d.mts +2 -2
  64. package/dist/adaptors/hono.mjs +5 -5
  65. package/dist/adaptors/testing.cjs +3 -3
  66. package/dist/adaptors/testing.d.cts +2 -2
  67. package/dist/adaptors/testing.d.mts +2 -2
  68. package/dist/adaptors/testing.mjs +3 -3
  69. package/dist/crons/Cron.cjs +1 -1
  70. package/dist/crons/Cron.d.cts +1 -1
  71. package/dist/crons/Cron.d.mts +1 -1
  72. package/dist/crons/Cron.mjs +1 -1
  73. package/dist/crons/CronBuilder.cjs +2 -2
  74. package/dist/crons/CronBuilder.d.cts +1 -1
  75. package/dist/crons/CronBuilder.d.mts +1 -1
  76. package/dist/crons/CronBuilder.mjs +2 -2
  77. package/dist/crons/index.cjs +2 -2
  78. package/dist/crons/index.d.cts +5 -5
  79. package/dist/crons/index.d.mts +5 -5
  80. package/dist/crons/index.mjs +2 -2
  81. package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.cjs +3 -3
  82. package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.d.cts +1 -1
  83. package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.d.mts +1 -1
  84. package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.mjs +3 -3
  85. package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.cjs +5 -5
  86. package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.d.cts +1 -1
  87. package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.d.mts +1 -1
  88. package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.mjs +5 -5
  89. package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.cjs +5 -5
  90. package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.d.cts +1 -1
  91. package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.d.mts +1 -1
  92. package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.mjs +5 -5
  93. package/dist/endpoints/Endpoint.cjs +1 -1
  94. package/dist/endpoints/Endpoint.d.cts +1 -1
  95. package/dist/endpoints/Endpoint.d.mts +1 -1
  96. package/dist/endpoints/Endpoint.mjs +1 -1
  97. package/dist/endpoints/EndpointBuilder.cjs +2 -2
  98. package/dist/endpoints/EndpointBuilder.d.cts +1 -1
  99. package/dist/endpoints/EndpointBuilder.d.mts +1 -1
  100. package/dist/endpoints/EndpointBuilder.mjs +2 -2
  101. package/dist/endpoints/EndpointFactory.cjs +3 -3
  102. package/dist/endpoints/EndpointFactory.d.cts +1 -1
  103. package/dist/endpoints/EndpointFactory.d.mts +1 -1
  104. package/dist/endpoints/EndpointFactory.mjs +3 -3
  105. package/dist/endpoints/HonoEndpointAdaptor.cjs +5 -5
  106. package/dist/endpoints/HonoEndpointAdaptor.d.cts +2 -2
  107. package/dist/endpoints/HonoEndpointAdaptor.d.mts +2 -2
  108. package/dist/endpoints/HonoEndpointAdaptor.mjs +5 -5
  109. package/dist/endpoints/TestEndpointAdaptor.cjs +3 -3
  110. package/dist/endpoints/TestEndpointAdaptor.d.cts +2 -2
  111. package/dist/endpoints/TestEndpointAdaptor.d.mts +2 -2
  112. package/dist/endpoints/TestEndpointAdaptor.mjs +3 -3
  113. package/dist/endpoints/audit.d.cts +1 -1
  114. package/dist/endpoints/audit.d.mts +1 -1
  115. package/dist/endpoints/helpers.cjs +2 -2
  116. package/dist/endpoints/helpers.d.cts +1 -1
  117. package/dist/endpoints/helpers.d.mts +1 -1
  118. package/dist/endpoints/helpers.mjs +2 -2
  119. package/dist/endpoints/index.cjs +3 -3
  120. package/dist/endpoints/index.d.cts +3 -3
  121. package/dist/endpoints/index.d.mts +3 -3
  122. package/dist/endpoints/index.mjs +3 -3
  123. package/dist/endpoints/parseHonoQuery.cjs +1 -1
  124. package/dist/endpoints/parseHonoQuery.mjs +1 -1
  125. package/dist/endpoints/parseQueryParams.cjs +1 -1
  126. package/dist/endpoints/parseQueryParams.mjs +1 -1
  127. package/dist/endpoints/processAudits.cjs +1 -1
  128. package/dist/endpoints/processAudits.d.cts +1 -1
  129. package/dist/endpoints/processAudits.d.mts +1 -1
  130. package/dist/endpoints/processAudits.mjs +1 -1
  131. package/dist/functions/AWSLambdaFunction.cjs +2 -2
  132. package/dist/functions/AWSLambdaFunction.mjs +2 -2
  133. package/dist/functions/FunctionExecutionWrapper.cjs +1 -1
  134. package/dist/functions/FunctionExecutionWrapper.mjs +1 -1
  135. package/dist/functions/index.d.cts +1 -1
  136. package/dist/functions/index.d.mts +1 -1
  137. package/dist/{helpers-Khuhi_Qx.cjs → helpers-CUYRcimZ.cjs} +2 -2
  138. package/dist/{helpers-Khuhi_Qx.cjs.map → helpers-CUYRcimZ.cjs.map} +1 -1
  139. package/dist/{helpers-2CLKTnRm.mjs → helpers-D-OW3LI_.mjs} +2 -2
  140. package/dist/{helpers-2CLKTnRm.mjs.map → helpers-D-OW3LI_.mjs.map} +1 -1
  141. package/dist/index-Doa8YPmH.d.cts +10 -0
  142. package/dist/index-TxufD5Xp.d.mts +10 -0
  143. package/dist/{parseHonoQuery-CwFKw2ua.mjs → parseHonoQuery-BlwMModJ.mjs} +1 -1
  144. package/dist/{parseHonoQuery-CwFKw2ua.mjs.map → parseHonoQuery-BlwMModJ.mjs.map} +1 -1
  145. package/dist/{parseHonoQuery-CT8Cvin-.cjs → parseHonoQuery-D-fMmSbA.cjs} +1 -1
  146. package/dist/{parseHonoQuery-CT8Cvin-.cjs.map → parseHonoQuery-D-fMmSbA.cjs.map} +1 -1
  147. package/dist/{parseQueryParams-CwvXXwkW.cjs → parseQueryParams-CbY1zcCU.cjs} +1 -1
  148. package/dist/{parseQueryParams-CwvXXwkW.cjs.map → parseQueryParams-CbY1zcCU.cjs.map} +1 -1
  149. package/dist/{parseQueryParams-CHINupbZ.mjs → parseQueryParams-DlbV3_SB.mjs} +1 -1
  150. package/dist/{parseQueryParams-CHINupbZ.mjs.map → parseQueryParams-DlbV3_SB.mjs.map} +1 -1
  151. package/dist/{processAudits-DfcB-X-4.mjs → processAudits-CW7z5Kj9.mjs} +1 -1
  152. package/dist/{processAudits-DfcB-X-4.mjs.map → processAudits-CW7z5Kj9.mjs.map} +1 -1
  153. package/dist/{processAudits-BFokHhCO.cjs → processAudits-MHp5_fc7.cjs} +1 -1
  154. package/dist/{processAudits-BFokHhCO.cjs.map → processAudits-MHp5_fc7.cjs.map} +1 -1
  155. package/dist/subscribers/AWSLambdaSubscriberAdaptor.cjs +1 -1
  156. package/dist/subscribers/AWSLambdaSubscriberAdaptor.mjs +1 -1
  157. package/package.json +4 -4
  158. package/src/endpoints/TestEndpointAdaptor.ts +51 -52
  159. package/src/endpoints/__tests__/TestEndpointAdaptor.audits.spec.ts +614 -0
  160. package/dist/EndpointFactory-Ctln6czP.mjs.map +0 -1
  161. package/dist/EndpointFactory-mTfi8x1X.cjs.map +0 -1
  162. package/dist/TestEndpointAdaptor-B9hyZ-mF.cjs.map +0 -1
  163. package/dist/TestEndpointAdaptor-DbwrL-RJ.mjs.map +0 -1
  164. package/dist/index-Sxtb_Pzw.d.mts +0 -10
  165. package/dist/index-m7xBtcAW.d.cts +0 -10
@@ -0,0 +1,614 @@
1
+ import type {
2
+ AuditRecord,
3
+ AuditStorage,
4
+ AuditableAction,
5
+ } from '@geekmidas/audit';
6
+ import type { Logger } from '@geekmidas/logger';
7
+ import type { Service } from '@geekmidas/services';
8
+ import { beforeEach, describe, expect, it, vi } from 'vitest';
9
+ import { z } from 'zod';
10
+ import { e } from '../EndpointFactory';
11
+ import { TestEndpointAdaptor } from '../TestEndpointAdaptor';
12
+ import type { MappedAudit } from '../audit';
13
+
14
+ // Test audit action types
15
+ type TestAuditAction =
16
+ | AuditableAction<'user.created', { userId: string; email: string }>
17
+ | AuditableAction<'user.updated', { userId: string; changes: string[] }>;
18
+
19
+ // In-memory audit storage for testing - implements AuditStorage<TestAuditAction>
20
+ class InMemoryAuditStorage implements AuditStorage<TestAuditAction> {
21
+ // Type marker for ExtractStorageAuditAction to work
22
+ declare readonly __auditActionType?: TestAuditAction;
23
+
24
+ records: AuditRecord[] = [];
25
+
26
+ async write(records: AuditRecord[]): Promise<void> {
27
+ this.records.push(...records);
28
+ }
29
+
30
+ async query(): Promise<AuditRecord[]> {
31
+ return this.records;
32
+ }
33
+
34
+ clear(): void {
35
+ this.records = [];
36
+ }
37
+ }
38
+
39
+ // Mock database for testing
40
+ interface MockDatabase {
41
+ query: (sql: string) => Promise<any[]>;
42
+ }
43
+
44
+ const createMockDatabase = (): MockDatabase => ({
45
+ query: vi.fn().mockResolvedValue([]),
46
+ });
47
+
48
+ // Mock logger
49
+ const createMockLogger = (): Logger => {
50
+ const logger: Logger = {
51
+ debug: vi.fn(),
52
+ info: vi.fn(),
53
+ warn: vi.fn(),
54
+ error: vi.fn(),
55
+ fatal: vi.fn(),
56
+ trace: vi.fn(),
57
+ child: vi.fn(() => logger),
58
+ };
59
+ return logger;
60
+ };
61
+
62
+ describe('TestEndpointAdaptor with auditorStorage and database', () => {
63
+ let mockLogger: Logger;
64
+
65
+ beforeEach(() => {
66
+ vi.clearAllMocks();
67
+ mockLogger = createMockLogger();
68
+ });
69
+
70
+ describe('auditorStorage', () => {
71
+ it('should process declarative audits when auditorStorage is provided', async () => {
72
+ const auditStorage = new InMemoryAuditStorage();
73
+
74
+ const auditStorageService: Service<
75
+ 'auditStorage',
76
+ InMemoryAuditStorage
77
+ > = {
78
+ serviceName: 'auditStorage' as const,
79
+ register: vi.fn().mockResolvedValue(auditStorage),
80
+ };
81
+
82
+ const outputSchema = z.object({ id: z.string(), email: z.string() });
83
+
84
+ type OutputType = z.infer<typeof outputSchema>;
85
+
86
+ const audits: MappedAudit<TestAuditAction, typeof outputSchema>[] = [
87
+ {
88
+ type: 'user.created',
89
+ payload: (response: OutputType) => ({
90
+ userId: response.id,
91
+ email: response.email,
92
+ }),
93
+ },
94
+ ];
95
+
96
+ const endpoint = e
97
+ .post('/users')
98
+ .logger(mockLogger)
99
+ .auditor(auditStorageService)
100
+ .output(outputSchema)
101
+ .audit(audits)
102
+ .handle(async () => ({ id: '123', email: 'test@example.com' }));
103
+
104
+ const adapter = new TestEndpointAdaptor(endpoint);
105
+
106
+ const result = await adapter.request({
107
+ services: {},
108
+ headers: { host: 'example.com' },
109
+ auditorStorage: auditStorage,
110
+ });
111
+
112
+ expect(result).toEqual({ id: '123', email: 'test@example.com' });
113
+
114
+ // Verify audit was written
115
+ expect(auditStorage.records).toHaveLength(1);
116
+ expect(auditStorage.records[0].type).toBe('user.created');
117
+ expect(auditStorage.records[0].payload).toEqual({
118
+ userId: '123',
119
+ email: 'test@example.com',
120
+ });
121
+ });
122
+
123
+ it('should allow manual auditing via auditor in handler context', async () => {
124
+ const auditStorage = new InMemoryAuditStorage();
125
+
126
+ const auditStorageService: Service<
127
+ 'auditStorage',
128
+ InMemoryAuditStorage
129
+ > = {
130
+ serviceName: 'auditStorage' as const,
131
+ register: vi.fn().mockResolvedValue(auditStorage),
132
+ };
133
+
134
+ const endpoint = e
135
+ .post('/users')
136
+ .logger(mockLogger)
137
+ .auditor(auditStorageService)
138
+ .output(z.object({ id: z.string(), email: z.string() }))
139
+ .handle(async ({ auditor }) => {
140
+ // Manual audit in handler
141
+ auditor.audit('user.created', {
142
+ userId: 'manual-123',
143
+ email: 'manual@example.com',
144
+ });
145
+
146
+ return { id: 'manual-123', email: 'manual@example.com' };
147
+ });
148
+
149
+ const adapter = new TestEndpointAdaptor(endpoint);
150
+
151
+ const result = await adapter.request({
152
+ services: {},
153
+ headers: { host: 'example.com' },
154
+ auditorStorage: auditStorage,
155
+ });
156
+
157
+ expect(result).toEqual({ id: 'manual-123', email: 'manual@example.com' });
158
+
159
+ // Verify manual audit was written
160
+ expect(auditStorage.records).toHaveLength(1);
161
+ expect(auditStorage.records[0].type).toBe('user.created');
162
+ expect(auditStorage.records[0].payload).toEqual({
163
+ userId: 'manual-123',
164
+ email: 'manual@example.com',
165
+ });
166
+ });
167
+
168
+ it('should extract actor from session when actorExtractor is configured', async () => {
169
+ const auditStorage = new InMemoryAuditStorage();
170
+
171
+ const auditStorageService: Service<
172
+ 'auditStorage',
173
+ InMemoryAuditStorage
174
+ > = {
175
+ serviceName: 'auditStorage' as const,
176
+ register: vi.fn().mockResolvedValue(auditStorage),
177
+ };
178
+
179
+ const outputSchema = z.object({ id: z.string(), email: z.string() });
180
+
181
+ type OutputType = z.infer<typeof outputSchema>;
182
+
183
+ const audits: MappedAudit<TestAuditAction, typeof outputSchema>[] = [
184
+ {
185
+ type: 'user.created',
186
+ payload: (response: OutputType) => ({
187
+ userId: response.id,
188
+ email: response.email,
189
+ }),
190
+ },
191
+ ];
192
+
193
+ const endpoint = e
194
+ .post('/users')
195
+ .logger(mockLogger)
196
+ .auditor(auditStorageService)
197
+ .actor(({ header }) => ({
198
+ id: header('x-user-id') ?? 'anonymous',
199
+ type: 'user',
200
+ }))
201
+ .output(outputSchema)
202
+ .audit(audits)
203
+ .handle(async () => ({ id: '123', email: 'test@example.com' }));
204
+
205
+ const adapter = new TestEndpointAdaptor(endpoint);
206
+
207
+ const result = await adapter.request({
208
+ services: {},
209
+ headers: { host: 'example.com', 'x-user-id': 'actor-456' },
210
+ auditorStorage: auditStorage,
211
+ });
212
+
213
+ expect(result).toEqual({ id: '123', email: 'test@example.com' });
214
+
215
+ // Verify actor was extracted
216
+ expect(auditStorage.records).toHaveLength(1);
217
+ expect(auditStorage.records[0].actor).toEqual({
218
+ id: 'actor-456',
219
+ type: 'user',
220
+ });
221
+ });
222
+
223
+ it('should warn when declarative audits are configured but no auditorStorage provided', async () => {
224
+ const outputSchema = z.object({ id: z.string(), email: z.string() });
225
+
226
+ type OutputType = z.infer<typeof outputSchema>;
227
+
228
+ const audits: MappedAudit<TestAuditAction, typeof outputSchema>[] = [
229
+ {
230
+ type: 'user.created',
231
+ payload: (response: OutputType) => ({
232
+ userId: response.id,
233
+ email: response.email,
234
+ }),
235
+ },
236
+ ];
237
+
238
+ // Create endpoint without auditor to test the warning
239
+ const endpoint = e
240
+ .post('/users')
241
+ .logger(mockLogger)
242
+ .output(outputSchema)
243
+ .handle(async () => ({ id: '123', email: 'test@example.com' }));
244
+
245
+ // Manually set audits to simulate a configuration error
246
+ (endpoint as any).audits = audits;
247
+
248
+ const adapter = new TestEndpointAdaptor(endpoint);
249
+
250
+ // Call without auditorStorage - this won't be type-enforced without .auditor()
251
+ const result = await adapter.request({
252
+ services: {},
253
+ headers: { host: 'example.com' },
254
+ });
255
+
256
+ expect(result).toEqual({ id: '123', email: 'test@example.com' });
257
+
258
+ // Should warn about missing audit storage
259
+ expect(mockLogger.warn).toHaveBeenCalledWith(
260
+ 'No auditor storage service available',
261
+ );
262
+ });
263
+ });
264
+
265
+ describe('database', () => {
266
+ it('should provide database instance to handler context', async () => {
267
+ const mockDb = createMockDatabase();
268
+ (mockDb.query as any).mockResolvedValue([{ id: '1', name: 'Test User' }]);
269
+
270
+ const databaseService: Service<'database', MockDatabase> = {
271
+ serviceName: 'database' as const,
272
+ register: vi.fn().mockResolvedValue(mockDb),
273
+ };
274
+
275
+ const endpoint = e
276
+ .get('/users')
277
+ .logger(mockLogger)
278
+ .database(databaseService)
279
+ .output(z.object({ users: z.array(z.any()) }))
280
+ .handle(async ({ db }) => {
281
+ const users = await db.query('SELECT * FROM users');
282
+ return { users };
283
+ });
284
+
285
+ const adapter = new TestEndpointAdaptor(endpoint);
286
+
287
+ const result = await adapter.request({
288
+ services: {},
289
+ headers: { host: 'example.com' },
290
+ database: mockDb,
291
+ });
292
+
293
+ expect(result).toEqual({ users: [{ id: '1', name: 'Test User' }] });
294
+ expect(mockDb.query).toHaveBeenCalledWith('SELECT * FROM users');
295
+ });
296
+
297
+ it('should allow using test doubles for database', async () => {
298
+ // Create a test double that returns specific test data
299
+ const testDb: MockDatabase = {
300
+ query: vi.fn().mockImplementation((sql: string) => {
301
+ if (sql.includes('users')) {
302
+ return Promise.resolve([
303
+ { id: 'test-1', name: 'Test User 1' },
304
+ { id: 'test-2', name: 'Test User 2' },
305
+ ]);
306
+ }
307
+ return Promise.resolve([]);
308
+ }),
309
+ };
310
+
311
+ const databaseService: Service<'database', MockDatabase> = {
312
+ serviceName: 'database' as const,
313
+ register: vi.fn().mockResolvedValue(testDb),
314
+ };
315
+
316
+ const endpoint = e
317
+ .get('/users')
318
+ .logger(mockLogger)
319
+ .database(databaseService)
320
+ .output(z.object({ count: z.number(), users: z.array(z.any()) }))
321
+ .handle(async ({ db }) => {
322
+ const users = await db.query('SELECT * FROM users');
323
+ return { count: users.length, users };
324
+ });
325
+
326
+ const adapter = new TestEndpointAdaptor(endpoint);
327
+
328
+ const result = await adapter.request({
329
+ services: {},
330
+ headers: { host: 'example.com' },
331
+ database: testDb,
332
+ });
333
+
334
+ expect(result).toEqual({
335
+ count: 2,
336
+ users: [
337
+ { id: 'test-1', name: 'Test User 1' },
338
+ { id: 'test-2', name: 'Test User 2' },
339
+ ],
340
+ });
341
+ });
342
+ });
343
+
344
+ describe('auditorStorage and database together', () => {
345
+ it('should support both auditorStorage and database in the same request', async () => {
346
+ const auditStorage = new InMemoryAuditStorage();
347
+ const mockDb = createMockDatabase();
348
+ (mockDb.query as any).mockResolvedValue([
349
+ { id: 'db-user-1', email: 'dbuser@example.com' },
350
+ ]);
351
+
352
+ const auditStorageService: Service<
353
+ 'auditStorage',
354
+ InMemoryAuditStorage
355
+ > = {
356
+ serviceName: 'auditStorage' as const,
357
+ register: vi.fn().mockResolvedValue(auditStorage),
358
+ };
359
+
360
+ const databaseService: Service<'database', MockDatabase> = {
361
+ serviceName: 'database' as const,
362
+ register: vi.fn().mockResolvedValue(mockDb),
363
+ };
364
+
365
+ const outputSchema = z.object({
366
+ id: z.string(),
367
+ email: z.string(),
368
+ source: z.string(),
369
+ });
370
+
371
+ type OutputType = z.infer<typeof outputSchema>;
372
+
373
+ const audits: MappedAudit<TestAuditAction, typeof outputSchema>[] = [
374
+ {
375
+ type: 'user.created',
376
+ payload: (response: OutputType) => ({
377
+ userId: response.id,
378
+ email: response.email,
379
+ }),
380
+ },
381
+ ];
382
+
383
+ const endpoint = e
384
+ .post('/users')
385
+ .logger(mockLogger)
386
+ .database(databaseService)
387
+ .auditor(auditStorageService)
388
+ .output(outputSchema)
389
+ .audit(audits)
390
+ .handle(async ({ db, auditor }) => {
391
+ // Use database
392
+ const users = await db.query('SELECT * FROM users LIMIT 1');
393
+ const user = users[0];
394
+
395
+ // Manual audit
396
+ auditor.audit('user.updated', {
397
+ userId: user.id,
398
+ changes: ['accessed'],
399
+ });
400
+
401
+ return { id: user.id, email: user.email, source: 'database' };
402
+ });
403
+
404
+ const adapter = new TestEndpointAdaptor(endpoint);
405
+
406
+ const result = await adapter.request({
407
+ services: {},
408
+ headers: { host: 'example.com' },
409
+ database: mockDb,
410
+ auditorStorage: auditStorage,
411
+ });
412
+
413
+ expect(result).toEqual({
414
+ id: 'db-user-1',
415
+ email: 'dbuser@example.com',
416
+ source: 'database',
417
+ });
418
+
419
+ // Verify database was queried
420
+ expect(mockDb.query).toHaveBeenCalledWith('SELECT * FROM users LIMIT 1');
421
+
422
+ // Verify both audits were written (manual + declarative)
423
+ expect(auditStorage.records).toHaveLength(2);
424
+ expect(auditStorage.records[0].type).toBe('user.updated');
425
+ expect(auditStorage.records[0].payload).toEqual({
426
+ userId: 'db-user-1',
427
+ changes: ['accessed'],
428
+ });
429
+ expect(auditStorage.records[1].type).toBe('user.created');
430
+ expect(auditStorage.records[1].payload).toEqual({
431
+ userId: 'db-user-1',
432
+ email: 'dbuser@example.com',
433
+ });
434
+ });
435
+
436
+ it('should work with EndpointFactory configured with default auditor and database', async () => {
437
+ const auditStorage = new InMemoryAuditStorage();
438
+ const mockDb = createMockDatabase();
439
+ (mockDb.query as any).mockResolvedValue([]);
440
+
441
+ const auditStorageService: Service<
442
+ 'auditStorage',
443
+ InMemoryAuditStorage
444
+ > = {
445
+ serviceName: 'auditStorage' as const,
446
+ register: vi.fn().mockResolvedValue(auditStorage),
447
+ };
448
+
449
+ const databaseService: Service<'database', MockDatabase> = {
450
+ serviceName: 'database' as const,
451
+ register: vi.fn().mockResolvedValue(mockDb),
452
+ };
453
+
454
+ // Create a router with default auditor and database
455
+ const router = e
456
+ .logger(mockLogger)
457
+ .database(databaseService)
458
+ .auditor(auditStorageService);
459
+
460
+ const outputSchema = z.object({ success: z.boolean() });
461
+
462
+ type OutputType = z.infer<typeof outputSchema>;
463
+
464
+ const endpoint = router
465
+ .get('/health')
466
+ .output(outputSchema)
467
+ .audit([
468
+ {
469
+ type: 'user.created',
470
+ payload: (_response: OutputType) => ({
471
+ userId: 'system',
472
+ email: 'health@check.com',
473
+ }),
474
+ when: (response: OutputType) => response.success,
475
+ },
476
+ ] satisfies MappedAudit<TestAuditAction, typeof outputSchema>[])
477
+ .handle(async ({ db, auditor }) => {
478
+ // Both db and auditor should be available from router defaults
479
+ await db.query('SELECT 1');
480
+
481
+ // Manual audit
482
+ auditor.audit('user.updated', {
483
+ userId: 'system',
484
+ changes: ['health_check'],
485
+ });
486
+
487
+ return { success: true };
488
+ });
489
+
490
+ const adapter = new TestEndpointAdaptor(endpoint);
491
+
492
+ const result = await adapter.request({
493
+ services: {},
494
+ headers: { host: 'example.com' },
495
+ database: mockDb,
496
+ auditorStorage: auditStorage,
497
+ });
498
+
499
+ expect(result).toEqual({ success: true });
500
+ expect(mockDb.query).toHaveBeenCalledWith('SELECT 1');
501
+
502
+ // Both manual and declarative audits
503
+ expect(auditStorage.records).toHaveLength(2);
504
+ });
505
+ });
506
+
507
+ describe('type enforcement', () => {
508
+ it('should require auditorStorage when endpoint uses .auditor()', async () => {
509
+ const auditStorage = new InMemoryAuditStorage();
510
+
511
+ const auditStorageService: Service<
512
+ 'auditStorage',
513
+ InMemoryAuditStorage
514
+ > = {
515
+ serviceName: 'auditStorage' as const,
516
+ register: vi.fn().mockResolvedValue(auditStorage),
517
+ };
518
+
519
+ const endpoint = e
520
+ .post('/users')
521
+ .logger(mockLogger)
522
+ .auditor(auditStorageService)
523
+ .output(z.object({ id: z.string() }))
524
+ .handle(async ({ auditor }) => {
525
+ auditor.audit('user.created', {
526
+ userId: 'test',
527
+ email: 'test@example.com',
528
+ });
529
+ return { id: 'test' };
530
+ });
531
+
532
+ const adapter = new TestEndpointAdaptor(endpoint);
533
+
534
+ // This call demonstrates that auditorStorage is required
535
+ // TypeScript would error if auditorStorage was omitted
536
+ const result = await adapter.request({
537
+ services: {},
538
+ headers: { host: 'example.com' },
539
+ auditorStorage: auditStorage, // Required when .auditor() is used
540
+ });
541
+
542
+ expect(result).toEqual({ id: 'test' });
543
+ });
544
+
545
+ it('should require database when endpoint uses .database()', async () => {
546
+ const mockDb = createMockDatabase();
547
+
548
+ const databaseService: Service<'database', MockDatabase> = {
549
+ serviceName: 'database' as const,
550
+ register: vi.fn().mockResolvedValue(mockDb),
551
+ };
552
+
553
+ const endpoint = e
554
+ .get('/data')
555
+ .logger(mockLogger)
556
+ .database(databaseService)
557
+ .output(z.object({ data: z.array(z.any()) }))
558
+ .handle(async ({ db }) => {
559
+ const data = await db.query('SELECT * FROM data');
560
+ return { data };
561
+ });
562
+
563
+ const adapter = new TestEndpointAdaptor(endpoint);
564
+
565
+ // This call demonstrates that database is required
566
+ // TypeScript would error if database was omitted
567
+ const result = await adapter.request({
568
+ services: {},
569
+ headers: { host: 'example.com' },
570
+ database: mockDb, // Required when .database() is used
571
+ });
572
+
573
+ expect(result).toEqual({ data: [] });
574
+ });
575
+
576
+ it('should not require auditorStorage when endpoint does not use .auditor()', async () => {
577
+ const endpoint = e
578
+ .get('/simple')
579
+ .logger(mockLogger)
580
+ .output(z.object({ message: z.string() }))
581
+ .handle(async () => ({ message: 'Hello' }));
582
+
583
+ const adapter = new TestEndpointAdaptor(endpoint);
584
+
585
+ // auditorStorage is NOT required here because .auditor() was not called
586
+ const result = await adapter.request({
587
+ services: {},
588
+ headers: { host: 'example.com' },
589
+ // No auditorStorage needed
590
+ });
591
+
592
+ expect(result).toEqual({ message: 'Hello' });
593
+ });
594
+
595
+ it('should not require database when endpoint does not use .database()', async () => {
596
+ const endpoint = e
597
+ .get('/simple')
598
+ .logger(mockLogger)
599
+ .output(z.object({ message: z.string() }))
600
+ .handle(async () => ({ message: 'Hello' }));
601
+
602
+ const adapter = new TestEndpointAdaptor(endpoint);
603
+
604
+ // database is NOT required here because .database() was not called
605
+ const result = await adapter.request({
606
+ services: {},
607
+ headers: { host: 'example.com' },
608
+ // No database needed
609
+ });
610
+
611
+ expect(result).toEqual({ message: 'Hello' });
612
+ });
613
+ });
614
+ });
@@ -1 +0,0 @@
1
- {"version":3,"file":"EndpointFactory-Ctln6czP.mjs","names":["DEFAULT_LOGGER","path: P","basePath: TBasePath","authorizers: T","path: TPath","fn: AuthorizeFn<TServices, TLogger, TSession>","services: S","logger: L","publisher: Service<TServiceName, T>","session: SessionFn<TServices, TLogger, T>","service: Service<TName, T>","storage: Service<TName, T>","extractor: ActorExtractor<TServices, TSession, TLogger>","method: TMethod"],"sources":["../src/endpoints/EndpointFactory.ts"],"sourcesContent":["import type {\n AuditStorage,\n AuditableAction,\n ExtractStorageAuditAction,\n} from '@geekmidas/audit';\nimport type { EventPublisher, MappedEvent } from '@geekmidas/events';\nimport type { Logger } from '@geekmidas/logger';\nimport { ConsoleLogger } from '@geekmidas/logger/console';\nimport type { Service } from '@geekmidas/services';\nimport uniqBy from 'lodash.uniqby';\nimport type { HttpMethod } from '../types';\nimport type { ActorExtractor } from './audit';\nimport type { Authorizer } from './Authorizer';\nimport type { AuthorizeFn, SessionFn } from './Endpoint';\nimport { EndpointBuilder } from './EndpointBuilder';\n\nconst DEFAULT_LOGGER = new ConsoleLogger() as any;\n\nexport class EndpointFactory<\n TServices extends Service[] = [],\n TBasePath extends string = '',\n TLogger extends Logger = Logger,\n TSession = unknown,\n TEventPublisher extends EventPublisher<any> | undefined = undefined,\n TEventPublisherServiceName extends string = string,\n TAuthorizers extends readonly string[] = readonly string[],\n TAuditStorage extends AuditStorage<any> | undefined = undefined,\n TAuditStorageServiceName extends string = string,\n TAuditAction extends AuditableAction<string, unknown> = ExtractStorageAuditAction<\n NonNullable<TAuditStorage>\n >,\n TDatabase = undefined,\n TDatabaseServiceName extends string = string,\n> {\n // @ts-ignore\n private defaultServices: TServices;\n private basePath: TBasePath = '' as TBasePath;\n private defaultAuthorizeFn?: AuthorizeFn<TServices, TLogger, TSession>;\n private defaultEventPublisher:\n | Service<TEventPublisherServiceName, TEventPublisher>\n | undefined;\n private defaultSessionExtractor?: SessionFn<TServices, TLogger, TSession>;\n private defaultLogger: TLogger = DEFAULT_LOGGER;\n private availableAuthorizers: Authorizer[] = [];\n private defaultAuthorizerName?: TAuthorizers[number];\n private defaultAuditorStorage:\n | Service<TAuditStorageServiceName, TAuditStorage>\n | undefined;\n private defaultDatabaseService:\n | Service<TDatabaseServiceName, TDatabase>\n | undefined;\n private defaultActorExtractor?: ActorExtractor<TServices, TSession, TLogger>;\n\n constructor({\n basePath,\n defaultAuthorizeFn,\n defaultLogger,\n defaultSessionExtractor,\n // @ts-ignore\n defaultServices = [] as TServices,\n defaultEventPublisher,\n availableAuthorizers = [],\n defaultAuthorizerName,\n defaultAuditorStorage,\n defaultDatabaseService,\n defaultActorExtractor,\n }: EndpointFactoryOptions<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TDatabase,\n TDatabaseServiceName\n > = {}) {\n // Initialize default services\n this.defaultServices = uniqBy(\n defaultServices,\n (s) => s.serviceName,\n ) as TServices;\n\n this.basePath = basePath || ('' as TBasePath);\n this.defaultAuthorizeFn = defaultAuthorizeFn;\n this.defaultLogger = defaultLogger || (DEFAULT_LOGGER as TLogger);\n this.defaultSessionExtractor = defaultSessionExtractor;\n this.defaultEventPublisher = defaultEventPublisher;\n this.availableAuthorizers = availableAuthorizers;\n this.defaultAuthorizerName = defaultAuthorizerName;\n this.defaultAuditorStorage = defaultAuditorStorage;\n this.defaultDatabaseService = defaultDatabaseService;\n this.defaultActorExtractor = defaultActorExtractor;\n }\n\n static joinPaths<TBasePath extends string, P extends string>(\n path: P,\n basePath: TBasePath = '' as TBasePath,\n ): JoinPaths<TBasePath, P> {\n // Handle empty cases\n if (!basePath && !path) return '/' as JoinPaths<TBasePath, P>;\n if (!basePath)\n return (path.startsWith('/') ? path : '/' + path) as JoinPaths<\n TBasePath,\n P\n >;\n if (!path)\n return (\n basePath.startsWith('/') ? basePath : '/' + basePath\n ) as JoinPaths<TBasePath, P>;\n\n const base = basePath.endsWith('/') ? basePath.slice(0, -1) : basePath;\n const segment = path.startsWith('/') ? path : '/' + path;\n\n let result = base + segment;\n\n // Ensure leading slash\n if (!result.startsWith('/')) {\n result = '/' + result;\n }\n\n // Normalize multiple slashes (except in the middle of the path where they might be intentional)\n result = result.replace(/^\\/+/g, '/');\n\n // Remove trailing slash unless it's the root path \"/\"\n if (result.length > 1 && result.endsWith('/')) {\n result = result.slice(0, -1);\n }\n\n return result as JoinPaths<TBasePath, P>;\n }\n\n // Configure available authorizers\n authorizers<const T extends readonly string[]>(\n authorizers: T,\n ): EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n T,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n > {\n const authorizerConfigs = authorizers.map((name) => ({\n name,\n }));\n return new EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n T,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >({\n defaultServices: this.defaultServices,\n basePath: this.basePath,\n defaultAuthorizeFn: this.defaultAuthorizeFn,\n defaultLogger: this.defaultLogger,\n defaultSessionExtractor: this.defaultSessionExtractor,\n defaultEventPublisher: this.defaultEventPublisher,\n availableAuthorizers: authorizerConfigs,\n defaultAuthorizerName: this.defaultAuthorizerName,\n defaultAuditorStorage: this.defaultAuditorStorage,\n defaultDatabaseService: this.defaultDatabaseService,\n defaultActorExtractor: this.defaultActorExtractor,\n });\n }\n\n // Create a sub-router with a path prefix\n route<TPath extends string>(\n path: TPath,\n ): EndpointFactory<\n TServices,\n JoinPaths<TBasePath, TPath>,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n > {\n const newBasePath = EndpointFactory.joinPaths(path, this.basePath);\n return new EndpointFactory<\n TServices,\n JoinPaths<TBasePath, TPath>,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >({\n defaultServices: this.defaultServices,\n basePath: newBasePath,\n defaultAuthorizeFn: this.defaultAuthorizeFn,\n defaultLogger: this.defaultLogger,\n defaultSessionExtractor: this.defaultSessionExtractor,\n defaultEventPublisher: this.defaultEventPublisher,\n availableAuthorizers: this.availableAuthorizers,\n defaultAuthorizerName: this.defaultAuthorizerName,\n defaultAuditorStorage: this.defaultAuditorStorage,\n defaultDatabaseService: this.defaultDatabaseService,\n defaultActorExtractor: this.defaultActorExtractor,\n });\n }\n\n // Create a new factory with authorization\n authorize(\n fn: AuthorizeFn<TServices, TLogger, TSession>,\n ): EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n > {\n return new EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >({\n defaultServices: this.defaultServices,\n basePath: this.basePath,\n defaultAuthorizeFn: fn,\n defaultLogger: this.defaultLogger,\n defaultSessionExtractor: this.defaultSessionExtractor,\n defaultEventPublisher: this.defaultEventPublisher,\n availableAuthorizers: this.availableAuthorizers,\n defaultAuthorizerName: this.defaultAuthorizerName,\n defaultAuditorStorage: this.defaultAuditorStorage,\n defaultDatabaseService: this.defaultDatabaseService,\n defaultActorExtractor: this.defaultActorExtractor,\n });\n }\n\n // Create a new factory with services\n services<S extends Service[]>(\n services: S,\n ): EndpointFactory<\n [...S, ...TServices],\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n > {\n return new EndpointFactory<\n [...S, ...TServices],\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >({\n defaultServices: [...services, ...this.defaultServices],\n basePath: this.basePath,\n defaultAuthorizeFn: this.defaultAuthorizeFn,\n defaultLogger: this.defaultLogger,\n defaultSessionExtractor: this.defaultSessionExtractor,\n defaultEventPublisher: this.defaultEventPublisher,\n availableAuthorizers: this.availableAuthorizers,\n defaultAuthorizerName: this.defaultAuthorizerName,\n defaultAuditorStorage: this.defaultAuditorStorage,\n defaultDatabaseService: this.defaultDatabaseService,\n defaultActorExtractor: this.defaultActorExtractor,\n });\n }\n\n logger<L extends Logger>(\n logger: L,\n ): EndpointFactory<\n TServices,\n TBasePath,\n L,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n > {\n return new EndpointFactory<\n TServices,\n TBasePath,\n L,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >({\n defaultServices: this.defaultServices,\n basePath: this.basePath,\n defaultAuthorizeFn: this.defaultAuthorizeFn as unknown as AuthorizeFn<\n TServices,\n L,\n TSession\n >,\n defaultLogger: logger,\n defaultSessionExtractor: this\n .defaultSessionExtractor as unknown as SessionFn<\n TServices,\n L,\n TSession\n >,\n defaultEventPublisher: this.defaultEventPublisher,\n availableAuthorizers: this.availableAuthorizers,\n defaultAuthorizerName: this.defaultAuthorizerName,\n defaultAuditorStorage: this.defaultAuditorStorage,\n defaultDatabaseService: this.defaultDatabaseService,\n defaultActorExtractor: this\n .defaultActorExtractor as unknown as ActorExtractor<TServices, TSession, L>,\n });\n }\n\n publisher<\n T extends EventPublisher<any>,\n TServiceName extends string = string,\n >(\n publisher: Service<TServiceName, T>,\n ): EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n T,\n TServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n > {\n return new EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n T,\n TServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >({\n defaultServices: this.defaultServices,\n basePath: this.basePath,\n defaultAuthorizeFn: this.defaultAuthorizeFn,\n defaultLogger: this.defaultLogger,\n defaultSessionExtractor: this.defaultSessionExtractor,\n defaultEventPublisher: publisher,\n availableAuthorizers: this.availableAuthorizers,\n defaultAuthorizerName: this.defaultAuthorizerName,\n defaultAuditorStorage: this.defaultAuditorStorage,\n defaultDatabaseService: this.defaultDatabaseService,\n defaultActorExtractor: this.defaultActorExtractor,\n });\n }\n\n session<T>(\n session: SessionFn<TServices, TLogger, T>,\n ): EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n T,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n > {\n return new EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n T,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >({\n defaultServices: this.defaultServices,\n basePath: this.basePath,\n defaultAuthorizeFn: this.defaultAuthorizeFn as unknown as AuthorizeFn<\n TServices,\n TLogger,\n T\n >,\n defaultLogger: this.defaultLogger,\n defaultSessionExtractor: session,\n defaultEventPublisher: this.defaultEventPublisher,\n availableAuthorizers: this.availableAuthorizers,\n defaultAuthorizerName: this.defaultAuthorizerName,\n defaultAuditorStorage: this.defaultAuditorStorage,\n defaultDatabaseService: this.defaultDatabaseService,\n defaultActorExtractor: this\n .defaultActorExtractor as unknown as ActorExtractor<TServices, T, TLogger>,\n });\n }\n\n /**\n * Set the database service for endpoints created from this factory.\n * The database will be available in handler context as `db`.\n */\n database<T, TName extends string>(\n service: Service<TName, T>,\n ): EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n T,\n TName\n > {\n return new EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n T,\n TName\n >({\n defaultServices: this.defaultServices,\n basePath: this.basePath,\n defaultAuthorizeFn: this.defaultAuthorizeFn,\n defaultLogger: this.defaultLogger,\n defaultSessionExtractor: this.defaultSessionExtractor,\n defaultEventPublisher: this.defaultEventPublisher,\n availableAuthorizers: this.availableAuthorizers,\n defaultAuthorizerName: this.defaultAuthorizerName,\n defaultAuditorStorage: this.defaultAuditorStorage,\n defaultDatabaseService: service,\n });\n }\n\n /**\n * Set the auditor storage service for endpoints created from this factory.\n * This enables audit functionality and makes `auditor` available in handler context.\n * The audit action type is automatically inferred from the storage's generic parameter.\n */\n auditor<T extends AuditStorage<any>, TName extends string>(\n storage: Service<TName, T>,\n ): EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n T,\n TName,\n ExtractStorageAuditAction<T>,\n TDatabase,\n TDatabaseServiceName\n > {\n return new EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n T,\n TName,\n ExtractStorageAuditAction<T>,\n TDatabase,\n TDatabaseServiceName\n >({\n defaultServices: this.defaultServices,\n basePath: this.basePath,\n defaultAuthorizeFn: this.defaultAuthorizeFn,\n defaultLogger: this.defaultLogger,\n defaultSessionExtractor: this.defaultSessionExtractor,\n defaultEventPublisher: this.defaultEventPublisher,\n availableAuthorizers: this.availableAuthorizers,\n defaultAuthorizerName: this.defaultAuthorizerName,\n defaultAuditorStorage: storage,\n defaultDatabaseService: this.defaultDatabaseService,\n defaultActorExtractor: this.defaultActorExtractor as unknown as ActorExtractor<\n TServices,\n TSession,\n TLogger\n >,\n });\n }\n\n /**\n * Set the actor extractor function for endpoints created from this factory.\n * The actor is extracted from the request context and attached to all audits.\n */\n actor(\n extractor: ActorExtractor<TServices, TSession, TLogger>,\n ): EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n > {\n return new EndpointFactory<\n TServices,\n TBasePath,\n TLogger,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >({\n defaultServices: this.defaultServices,\n basePath: this.basePath,\n defaultAuthorizeFn: this.defaultAuthorizeFn,\n defaultLogger: this.defaultLogger,\n defaultSessionExtractor: this.defaultSessionExtractor,\n defaultEventPublisher: this.defaultEventPublisher,\n availableAuthorizers: this.availableAuthorizers,\n defaultAuthorizerName: this.defaultAuthorizerName,\n defaultAuditorStorage: this.defaultAuditorStorage,\n defaultDatabaseService: this.defaultDatabaseService,\n defaultActorExtractor: extractor,\n });\n }\n\n private createBuilder<TMethod extends HttpMethod, TPath extends string>(\n method: TMethod,\n path: TPath,\n ): EndpointBuilder<\n JoinPaths<TBasePath, TPath>,\n TMethod,\n {},\n TServices,\n TLogger,\n undefined,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n > {\n const fullPath = EndpointFactory.joinPaths(path, this.basePath);\n const builder = new EndpointBuilder<\n JoinPaths<TBasePath, TPath>,\n TMethod,\n {},\n TServices,\n TLogger,\n undefined,\n TSession,\n TEventPublisher,\n TEventPublisherServiceName,\n TAuthorizers,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >(fullPath, method);\n\n if (this.defaultAuthorizeFn) {\n // @ts-ignore\n builder._authorize = this.defaultAuthorizeFn;\n }\n if (this.defaultServices.length) {\n // Create a copy to avoid sharing references between builders\n builder._services = [...this.defaultServices] as TServices;\n }\n\n if (this.defaultLogger) {\n builder._logger = this.defaultLogger as TLogger;\n }\n\n if (this.defaultSessionExtractor) {\n builder._getSession = this.defaultSessionExtractor as SessionFn<\n TServices,\n TLogger,\n TSession\n >;\n }\n\n if (this.defaultEventPublisher) {\n builder._setPublisher(this.defaultEventPublisher);\n }\n\n // Set available authorizers and default\n builder._availableAuthorizers = this.availableAuthorizers;\n if (this.defaultAuthorizerName) {\n builder._authorizerName = this.defaultAuthorizerName;\n }\n\n // Set auditor storage if configured\n if (this.defaultAuditorStorage) {\n builder._setAuditorStorage(this.defaultAuditorStorage as any);\n }\n\n // Set database service if configured\n if (this.defaultDatabaseService) {\n builder._setDatabaseService(this.defaultDatabaseService as any);\n }\n\n // Set actor extractor if configured\n if (this.defaultActorExtractor) {\n builder._actorExtractor = this.defaultActorExtractor;\n }\n\n return builder;\n }\n\n post<TPath extends string>(path: TPath) {\n return this.createBuilder('POST', path);\n }\n\n get<TPath extends string>(path: TPath) {\n return this.createBuilder('GET', path);\n }\n\n put<TPath extends string>(path: TPath) {\n return this.createBuilder('PUT', path);\n }\n\n delete<TPath extends string>(path: TPath) {\n return this.createBuilder('DELETE', path);\n }\n\n patch<TPath extends string>(path: TPath) {\n return this.createBuilder('PATCH', path);\n }\n\n options<TPath extends string>(path: TPath) {\n return this.createBuilder('OPTIONS', path);\n }\n}\n\nexport type RemoveTrailingSlash<T extends string> = T extends `${infer Rest}/`\n ? Rest extends ''\n ? T // Keep \"/\" as is\n : Rest\n : T;\n\nexport type JoinPaths<\n TBasePath extends string,\n TPath extends string,\n> = RemoveTrailingSlash<\n TBasePath extends ''\n ? TPath\n : TPath extends ''\n ? TBasePath\n : TBasePath extends '/'\n ? TPath extends `/${string}`\n ? TPath\n : `/${TPath}`\n : TBasePath extends `${infer Base}/`\n ? TPath extends `/${infer Rest}`\n ? `${Base}/${Rest}`\n : `${Base}/${TPath}`\n : TPath extends `/${infer Rest}`\n ? `${TBasePath}/${Rest}`\n : `${TBasePath}/${TPath}`\n>;\n\nexport interface EndpointFactoryOptions<\n TServices extends Service[] = [],\n TBasePath extends string = '',\n TLogger extends Logger = Logger,\n TSession = unknown,\n TEventPublisher extends EventPublisher<any> | undefined = undefined,\n TEventPublisherServiceName extends string = string,\n TAuthorizers extends readonly string[] = readonly string[],\n TAuditStorage extends AuditStorage | undefined = undefined,\n TAuditStorageServiceName extends string = string,\n TDatabase = undefined,\n TDatabaseServiceName extends string = string,\n> {\n defaultServices?: TServices;\n basePath?: TBasePath;\n defaultAuthorizeFn?: AuthorizeFn<TServices, TLogger, TSession>;\n defaultLogger?: TLogger;\n defaultSessionExtractor?: SessionFn<TServices, TLogger, TSession>;\n defaultEventPublisher?: Service<TEventPublisherServiceName, TEventPublisher>;\n defaultEvents?: MappedEvent<TEventPublisher, undefined>[];\n availableAuthorizers?: Authorizer[];\n defaultAuthorizerName?: TAuthorizers[number];\n defaultAuditorStorage?: Service<TAuditStorageServiceName, TAuditStorage>;\n defaultDatabaseService?: Service<TDatabaseServiceName, TDatabase>;\n defaultActorExtractor?: ActorExtractor<TServices, TSession, TLogger>;\n}\n\nexport const e = new EndpointFactory();\n"],"mappings":"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"}