@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.
- package/dist/{AWSLambdaFunction-H65WfXLt.mjs → AWSLambdaFunction-DBUENdP0.mjs} +2 -2
- package/dist/{AWSLambdaFunction-H65WfXLt.mjs.map → AWSLambdaFunction-DBUENdP0.mjs.map} +1 -1
- package/dist/{AWSLambdaFunction-C-fuCLA3.cjs → AWSLambdaFunction-vobYqQ0w.cjs} +2 -2
- package/dist/{AWSLambdaFunction-C-fuCLA3.cjs.map → AWSLambdaFunction-vobYqQ0w.cjs.map} +1 -1
- package/dist/{AWSLambdaSubscriberAdaptor-CyFh7MN8.mjs → AWSLambdaSubscriberAdaptor-BLHDyqzQ.mjs} +1 -1
- package/dist/{AWSLambdaSubscriberAdaptor-CyFh7MN8.mjs.map → AWSLambdaSubscriberAdaptor-BLHDyqzQ.mjs.map} +1 -1
- package/dist/{AWSLambdaSubscriberAdaptor-Dum5bkw3.cjs → AWSLambdaSubscriberAdaptor-DVC4VAQR.cjs} +1 -1
- package/dist/{AWSLambdaSubscriberAdaptor-Dum5bkw3.cjs.map → AWSLambdaSubscriberAdaptor-DVC4VAQR.cjs.map} +1 -1
- package/dist/{AmazonApiGatewayEndpointAdaptor-CI9L7Ucn.cjs → AmazonApiGatewayEndpointAdaptor-BLUW--OF.cjs} +4 -4
- package/dist/{AmazonApiGatewayEndpointAdaptor-CI9L7Ucn.cjs.map → AmazonApiGatewayEndpointAdaptor-BLUW--OF.cjs.map} +1 -1
- package/dist/{AmazonApiGatewayEndpointAdaptor-C6Jk5HSy.mjs → AmazonApiGatewayEndpointAdaptor-DBK53gB5.mjs} +4 -4
- package/dist/{AmazonApiGatewayEndpointAdaptor-C6Jk5HSy.mjs.map → AmazonApiGatewayEndpointAdaptor-DBK53gB5.mjs.map} +1 -1
- package/dist/{AmazonApiGatewayV1EndpointAdaptor-DYL1bCBS.cjs → AmazonApiGatewayV1EndpointAdaptor-B-i9_OtQ.cjs} +3 -3
- package/dist/{AmazonApiGatewayV1EndpointAdaptor-DYL1bCBS.cjs.map → AmazonApiGatewayV1EndpointAdaptor-B-i9_OtQ.cjs.map} +1 -1
- package/dist/{AmazonApiGatewayV1EndpointAdaptor-BMy8DdNJ.mjs → AmazonApiGatewayV1EndpointAdaptor-DfU3n5im.mjs} +3 -3
- package/dist/{AmazonApiGatewayV1EndpointAdaptor-BMy8DdNJ.mjs.map → AmazonApiGatewayV1EndpointAdaptor-DfU3n5im.mjs.map} +1 -1
- package/dist/{AmazonApiGatewayV2EndpointAdaptor-BU5wQMOe.mjs → AmazonApiGatewayV2EndpointAdaptor-D-AFyzaQ.mjs} +3 -3
- package/dist/{AmazonApiGatewayV2EndpointAdaptor-BU5wQMOe.mjs.map → AmazonApiGatewayV2EndpointAdaptor-D-AFyzaQ.mjs.map} +1 -1
- package/dist/{AmazonApiGatewayV2EndpointAdaptor-CPLCMeaN.cjs → AmazonApiGatewayV2EndpointAdaptor-D4k_Bg7Q.cjs} +3 -3
- package/dist/{AmazonApiGatewayV2EndpointAdaptor-CPLCMeaN.cjs.map → AmazonApiGatewayV2EndpointAdaptor-D4k_Bg7Q.cjs.map} +1 -1
- package/dist/{Cron-Bi3QOge_.cjs → Cron-CmtKQOmE.cjs} +1 -1
- package/dist/{Cron-Bi3QOge_.cjs.map → Cron-CmtKQOmE.cjs.map} +1 -1
- package/dist/{Cron-Dy_HW2Vv.mjs → Cron-mWi3PQxt.mjs} +1 -1
- package/dist/{Cron-Dy_HW2Vv.mjs.map → Cron-mWi3PQxt.mjs.map} +1 -1
- package/dist/{CronBuilder-Bl3A2Zp4.mjs → CronBuilder-4DxT6wUa.mjs} +2 -2
- package/dist/{CronBuilder-Bl3A2Zp4.mjs.map → CronBuilder-4DxT6wUa.mjs.map} +1 -1
- package/dist/{CronBuilder-Dv_w7Yri.cjs → CronBuilder-CeffP9Rs.cjs} +2 -2
- package/dist/{CronBuilder-Dv_w7Yri.cjs.map → CronBuilder-CeffP9Rs.cjs.map} +1 -1
- package/dist/{Endpoint-DDpF7NO1.cjs → Endpoint-BTvS2vwp.cjs} +1 -1
- package/dist/{Endpoint-DDpF7NO1.cjs.map → Endpoint-BTvS2vwp.cjs.map} +1 -1
- package/dist/{Endpoint-S6Yh2_PN.mjs → Endpoint-D2LVHBEO.mjs} +1 -1
- package/dist/{Endpoint-S6Yh2_PN.mjs.map → Endpoint-D2LVHBEO.mjs.map} +1 -1
- package/dist/{EndpointBuilder-CFnjYXmL.cjs → EndpointBuilder-C4qahFeS.cjs} +2 -2
- package/dist/{EndpointBuilder-CFnjYXmL.cjs.map → EndpointBuilder-C4qahFeS.cjs.map} +1 -1
- package/dist/{EndpointBuilder-DlDft4mJ.mjs → EndpointBuilder-O6B1zJ6v.mjs} +2 -2
- package/dist/{EndpointBuilder-DlDft4mJ.mjs.map → EndpointBuilder-O6B1zJ6v.mjs.map} +1 -1
- package/dist/{EndpointFactory-Ctln6czP.mjs → EndpointFactory-BUYrnjau.mjs} +2 -2
- package/dist/EndpointFactory-BUYrnjau.mjs.map +1 -0
- package/dist/{EndpointFactory-mTfi8x1X.cjs → EndpointFactory-C_neYSiA.cjs} +2 -2
- package/dist/EndpointFactory-C_neYSiA.cjs.map +1 -0
- package/dist/{FunctionExecutionWrapper-DkNycmOh.cjs → FunctionExecutionWrapper-B8agyYHk.cjs} +1 -1
- package/dist/{FunctionExecutionWrapper-DkNycmOh.cjs.map → FunctionExecutionWrapper-B8agyYHk.cjs.map} +1 -1
- package/dist/{FunctionExecutionWrapper-Bubnr0zA.mjs → FunctionExecutionWrapper-BPIdmPe8.mjs} +1 -1
- package/dist/{FunctionExecutionWrapper-Bubnr0zA.mjs.map → FunctionExecutionWrapper-BPIdmPe8.mjs.map} +1 -1
- package/dist/{HonoEndpointAdaptor-DsqGuEIb.d.mts → HonoEndpointAdaptor-Br1vuQ3A.d.mts} +3 -3
- package/dist/{HonoEndpointAdaptor-DajXbh80.d.cts → HonoEndpointAdaptor-C9wC10-w.d.cts} +3 -3
- package/dist/{HonoEndpointAdaptor-DuyE06nH.mjs → HonoEndpointAdaptor-DEFNrIv7.mjs} +5 -5
- package/dist/{HonoEndpointAdaptor-DuyE06nH.mjs.map → HonoEndpointAdaptor-DEFNrIv7.mjs.map} +1 -1
- package/dist/{HonoEndpointAdaptor-CfLRHHFw.cjs → HonoEndpointAdaptor-DbLeXkR6.cjs} +5 -5
- package/dist/{HonoEndpointAdaptor-CfLRHHFw.cjs.map → HonoEndpointAdaptor-DbLeXkR6.cjs.map} +1 -1
- package/dist/{TestEndpointAdaptor-DbwrL-RJ.mjs → TestEndpointAdaptor-BGrZsg5c.mjs} +38 -18
- package/dist/TestEndpointAdaptor-BGrZsg5c.mjs.map +1 -0
- package/dist/{TestEndpointAdaptor-DhRjJHyk.d.mts → TestEndpointAdaptor-Bl2ic-yr.d.mts} +9 -9
- package/dist/{TestEndpointAdaptor-B9tUIlCC.d.cts → TestEndpointAdaptor-ByXqQufk.d.cts} +9 -9
- package/dist/{TestEndpointAdaptor-B9hyZ-mF.cjs → TestEndpointAdaptor-JCvZ3VVi.cjs} +38 -18
- package/dist/TestEndpointAdaptor-JCvZ3VVi.cjs.map +1 -0
- package/dist/adaptors/aws.cjs +9 -9
- package/dist/adaptors/aws.d.cts +1 -1
- package/dist/adaptors/aws.d.mts +1 -1
- package/dist/adaptors/aws.mjs +9 -9
- package/dist/adaptors/hono.cjs +5 -5
- package/dist/adaptors/hono.d.cts +2 -2
- package/dist/adaptors/hono.d.mts +2 -2
- package/dist/adaptors/hono.mjs +5 -5
- package/dist/adaptors/testing.cjs +3 -3
- package/dist/adaptors/testing.d.cts +2 -2
- package/dist/adaptors/testing.d.mts +2 -2
- package/dist/adaptors/testing.mjs +3 -3
- package/dist/crons/Cron.cjs +1 -1
- package/dist/crons/Cron.d.cts +1 -1
- package/dist/crons/Cron.d.mts +1 -1
- package/dist/crons/Cron.mjs +1 -1
- package/dist/crons/CronBuilder.cjs +2 -2
- package/dist/crons/CronBuilder.d.cts +1 -1
- package/dist/crons/CronBuilder.d.mts +1 -1
- package/dist/crons/CronBuilder.mjs +2 -2
- package/dist/crons/index.cjs +2 -2
- package/dist/crons/index.d.cts +5 -5
- package/dist/crons/index.d.mts +5 -5
- package/dist/crons/index.mjs +2 -2
- package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.cjs +3 -3
- package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.d.cts +1 -1
- package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.d.mts +1 -1
- package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.mjs +3 -3
- package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.cjs +5 -5
- package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.d.cts +1 -1
- package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.d.mts +1 -1
- package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.mjs +5 -5
- package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.cjs +5 -5
- package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.d.cts +1 -1
- package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.d.mts +1 -1
- package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.mjs +5 -5
- package/dist/endpoints/Endpoint.cjs +1 -1
- package/dist/endpoints/Endpoint.d.cts +1 -1
- package/dist/endpoints/Endpoint.d.mts +1 -1
- package/dist/endpoints/Endpoint.mjs +1 -1
- package/dist/endpoints/EndpointBuilder.cjs +2 -2
- package/dist/endpoints/EndpointBuilder.d.cts +1 -1
- package/dist/endpoints/EndpointBuilder.d.mts +1 -1
- package/dist/endpoints/EndpointBuilder.mjs +2 -2
- package/dist/endpoints/EndpointFactory.cjs +3 -3
- package/dist/endpoints/EndpointFactory.d.cts +1 -1
- package/dist/endpoints/EndpointFactory.d.mts +1 -1
- package/dist/endpoints/EndpointFactory.mjs +3 -3
- package/dist/endpoints/HonoEndpointAdaptor.cjs +5 -5
- package/dist/endpoints/HonoEndpointAdaptor.d.cts +2 -2
- package/dist/endpoints/HonoEndpointAdaptor.d.mts +2 -2
- package/dist/endpoints/HonoEndpointAdaptor.mjs +5 -5
- package/dist/endpoints/TestEndpointAdaptor.cjs +3 -3
- package/dist/endpoints/TestEndpointAdaptor.d.cts +2 -2
- package/dist/endpoints/TestEndpointAdaptor.d.mts +2 -2
- package/dist/endpoints/TestEndpointAdaptor.mjs +3 -3
- package/dist/endpoints/audit.d.cts +1 -1
- package/dist/endpoints/audit.d.mts +1 -1
- package/dist/endpoints/helpers.cjs +2 -2
- package/dist/endpoints/helpers.d.cts +1 -1
- package/dist/endpoints/helpers.d.mts +1 -1
- package/dist/endpoints/helpers.mjs +2 -2
- package/dist/endpoints/index.cjs +3 -3
- package/dist/endpoints/index.d.cts +3 -3
- package/dist/endpoints/index.d.mts +3 -3
- package/dist/endpoints/index.mjs +3 -3
- package/dist/endpoints/parseHonoQuery.cjs +1 -1
- package/dist/endpoints/parseHonoQuery.mjs +1 -1
- package/dist/endpoints/parseQueryParams.cjs +1 -1
- package/dist/endpoints/parseQueryParams.mjs +1 -1
- package/dist/endpoints/processAudits.cjs +1 -1
- package/dist/endpoints/processAudits.d.cts +1 -1
- package/dist/endpoints/processAudits.d.mts +1 -1
- package/dist/endpoints/processAudits.mjs +1 -1
- package/dist/functions/AWSLambdaFunction.cjs +2 -2
- package/dist/functions/AWSLambdaFunction.mjs +2 -2
- package/dist/functions/FunctionExecutionWrapper.cjs +1 -1
- package/dist/functions/FunctionExecutionWrapper.mjs +1 -1
- package/dist/functions/index.d.cts +1 -1
- package/dist/functions/index.d.mts +1 -1
- package/dist/{helpers-Khuhi_Qx.cjs → helpers-CUYRcimZ.cjs} +2 -2
- package/dist/{helpers-Khuhi_Qx.cjs.map → helpers-CUYRcimZ.cjs.map} +1 -1
- package/dist/{helpers-2CLKTnRm.mjs → helpers-D-OW3LI_.mjs} +2 -2
- package/dist/{helpers-2CLKTnRm.mjs.map → helpers-D-OW3LI_.mjs.map} +1 -1
- package/dist/index-Doa8YPmH.d.cts +10 -0
- package/dist/index-TxufD5Xp.d.mts +10 -0
- package/dist/{parseHonoQuery-CwFKw2ua.mjs → parseHonoQuery-BlwMModJ.mjs} +1 -1
- package/dist/{parseHonoQuery-CwFKw2ua.mjs.map → parseHonoQuery-BlwMModJ.mjs.map} +1 -1
- package/dist/{parseHonoQuery-CT8Cvin-.cjs → parseHonoQuery-D-fMmSbA.cjs} +1 -1
- package/dist/{parseHonoQuery-CT8Cvin-.cjs.map → parseHonoQuery-D-fMmSbA.cjs.map} +1 -1
- package/dist/{parseQueryParams-CwvXXwkW.cjs → parseQueryParams-CbY1zcCU.cjs} +1 -1
- package/dist/{parseQueryParams-CwvXXwkW.cjs.map → parseQueryParams-CbY1zcCU.cjs.map} +1 -1
- package/dist/{parseQueryParams-CHINupbZ.mjs → parseQueryParams-DlbV3_SB.mjs} +1 -1
- package/dist/{parseQueryParams-CHINupbZ.mjs.map → parseQueryParams-DlbV3_SB.mjs.map} +1 -1
- package/dist/{processAudits-DfcB-X-4.mjs → processAudits-CW7z5Kj9.mjs} +1 -1
- package/dist/{processAudits-DfcB-X-4.mjs.map → processAudits-CW7z5Kj9.mjs.map} +1 -1
- package/dist/{processAudits-BFokHhCO.cjs → processAudits-MHp5_fc7.cjs} +1 -1
- package/dist/{processAudits-BFokHhCO.cjs.map → processAudits-MHp5_fc7.cjs.map} +1 -1
- package/dist/subscribers/AWSLambdaSubscriberAdaptor.cjs +1 -1
- package/dist/subscribers/AWSLambdaSubscriberAdaptor.mjs +1 -1
- package/package.json +4 -4
- package/src/endpoints/TestEndpointAdaptor.ts +51 -52
- package/src/endpoints/__tests__/TestEndpointAdaptor.audits.spec.ts +614 -0
- package/dist/EndpointFactory-Ctln6czP.mjs.map +0 -1
- package/dist/EndpointFactory-mTfi8x1X.cjs.map +0 -1
- package/dist/TestEndpointAdaptor-B9hyZ-mF.cjs.map +0 -1
- package/dist/TestEndpointAdaptor-DbwrL-RJ.mjs.map +0 -1
- package/dist/index-Sxtb_Pzw.d.mts +0 -10
- 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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"}
|