@geekmidas/constructs 0.0.22 → 0.1.0

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 (181) hide show
  1. package/README.md +90 -0
  2. package/dist/{AWSLambdaFunction-qWpalqfr.d.mts → AWSLambdaFunction-DCh1tOhB.d.mts} +3 -3
  3. package/dist/{AWSLambdaSubscriberAdaptor-CWbBNRz3.d.mts → AWSLambdaSubscriberAdaptor-DtNZPMoV.d.mts} +2 -2
  4. package/dist/{AmazonApiGatewayEndpointAdaptor-BQ0IJdaI.d.mts → AmazonApiGatewayEndpointAdaptor-BaTM2TNu.d.mts} +3 -3
  5. package/dist/{AmazonApiGatewayEndpointAdaptor-DXssXsJi.cjs → AmazonApiGatewayEndpointAdaptor-CPqlw2Rx.cjs} +3 -3
  6. package/dist/{AmazonApiGatewayEndpointAdaptor-DXssXsJi.cjs.map → AmazonApiGatewayEndpointAdaptor-CPqlw2Rx.cjs.map} +1 -1
  7. package/dist/{AmazonApiGatewayEndpointAdaptor-CacGag6F.mjs → AmazonApiGatewayEndpointAdaptor-Cm4iD199.mjs} +3 -3
  8. package/dist/{AmazonApiGatewayEndpointAdaptor-CacGag6F.mjs.map → AmazonApiGatewayEndpointAdaptor-Cm4iD199.mjs.map} +1 -1
  9. package/dist/{AmazonApiGatewayEndpointAdaptor-Da9BR5On.d.cts → AmazonApiGatewayEndpointAdaptor-TP_hLNAa.d.cts} +2 -2
  10. package/dist/{AmazonApiGatewayV1EndpointAdaptor-C4_AZ1ek.d.mts → AmazonApiGatewayV1EndpointAdaptor-B2ZpUFiC.d.mts} +4 -4
  11. package/dist/{AmazonApiGatewayV1EndpointAdaptor-CSm3NsWz.d.cts → AmazonApiGatewayV1EndpointAdaptor-CConfCqz.d.cts} +3 -3
  12. package/dist/{AmazonApiGatewayV1EndpointAdaptor-BpnG55R7.mjs → AmazonApiGatewayV1EndpointAdaptor-D38HuQa9.mjs} +2 -2
  13. package/dist/{AmazonApiGatewayV1EndpointAdaptor-BpnG55R7.mjs.map → AmazonApiGatewayV1EndpointAdaptor-D38HuQa9.mjs.map} +1 -1
  14. package/dist/{AmazonApiGatewayV1EndpointAdaptor-Df4kszio.cjs → AmazonApiGatewayV1EndpointAdaptor-tjpvLBRb.cjs} +2 -2
  15. package/dist/{AmazonApiGatewayV1EndpointAdaptor-Df4kszio.cjs.map → AmazonApiGatewayV1EndpointAdaptor-tjpvLBRb.cjs.map} +1 -1
  16. package/dist/{AmazonApiGatewayV2EndpointAdaptor-DdM8Tr1X.d.mts → AmazonApiGatewayV2EndpointAdaptor-BwMQInBi.d.mts} +4 -4
  17. package/dist/{AmazonApiGatewayV2EndpointAdaptor-ZORzMEET.mjs → AmazonApiGatewayV2EndpointAdaptor-ByYtiJtN.mjs} +2 -2
  18. package/dist/{AmazonApiGatewayV2EndpointAdaptor-ZORzMEET.mjs.map → AmazonApiGatewayV2EndpointAdaptor-ByYtiJtN.mjs.map} +1 -1
  19. package/dist/{AmazonApiGatewayV2EndpointAdaptor-5SIvqPby.cjs → AmazonApiGatewayV2EndpointAdaptor-DcJ_w3ro.cjs} +2 -2
  20. package/dist/{AmazonApiGatewayV2EndpointAdaptor-5SIvqPby.cjs.map → AmazonApiGatewayV2EndpointAdaptor-DcJ_w3ro.cjs.map} +1 -1
  21. package/dist/{AmazonApiGatewayV2EndpointAdaptor-6hsBFVLf.d.cts → AmazonApiGatewayV2EndpointAdaptor-DurJvFwa.d.cts} +3 -3
  22. package/dist/Authorizer-C0ge_tc8.cjs +92 -0
  23. package/dist/Authorizer-C0ge_tc8.cjs.map +1 -0
  24. package/dist/Authorizer-CpSUMTIs.d.cts +125 -0
  25. package/dist/Authorizer-D1w7MpK6.d.mts +125 -0
  26. package/dist/Authorizer-r9U3y_ms.mjs +68 -0
  27. package/dist/Authorizer-r9U3y_ms.mjs.map +1 -0
  28. package/dist/{BaseFunctionBuilder-Ct6zY6Jq.d.mts → BaseFunctionBuilder-Oc2xmxmg.d.mts} +2 -2
  29. package/dist/{Construct-DDR0295I.d.mts → Construct-DCPATqec.d.mts} +1 -1
  30. package/dist/Construct.d.mts +1 -1
  31. package/dist/{Cron-DnMRWPFR.d.mts → Cron-DPEcDCDW.d.mts} +2 -2
  32. package/dist/{CronBuilder-RLDitFmP.d.mts → CronBuilder-CxKTiepV.d.mts} +4 -4
  33. package/dist/{Endpoint-D2Imgihs.d.cts → Endpoint-BJPJTGjV.d.cts} +2 -2
  34. package/dist/{Endpoint-PtQ-wLIS.d.mts → Endpoint-DoY1Owv2.d.mts} +4 -4
  35. package/dist/{EndpointBuilder-W5fdXxYQ.mjs → EndpointBuilder-69uVrKZL.mjs} +15 -2
  36. package/dist/EndpointBuilder-69uVrKZL.mjs.map +1 -0
  37. package/dist/{EndpointBuilder-BPHpUekp.d.mts → EndpointBuilder-BJRkivxQ.d.mts} +6 -5
  38. package/dist/{EndpointBuilder-CYkeYpsL.cjs → EndpointBuilder-Bhyft7WY.cjs} +15 -2
  39. package/dist/EndpointBuilder-Bhyft7WY.cjs.map +1 -0
  40. package/dist/{EndpointBuilder-TApJQhtG.d.cts → EndpointBuilder-CQ-jOXsD.d.cts} +4 -3
  41. package/dist/{EndpointFactory-B27nfeiE.mjs → EndpointFactory-ARUlRlyM.mjs} +68 -14
  42. package/dist/EndpointFactory-ARUlRlyM.mjs.map +1 -0
  43. package/dist/{EndpointFactory-CNlfBDuD.d.mts → EndpointFactory-BKEPcQgE.d.mts} +45 -20
  44. package/dist/{EndpointFactory-D5lFZXqY.cjs → EndpointFactory-BcVbkrxf.cjs} +68 -14
  45. package/dist/EndpointFactory-BcVbkrxf.cjs.map +1 -0
  46. package/dist/{EndpointFactory-B5fOINuc.d.cts → EndpointFactory-DgwmAR_8.d.cts} +45 -20
  47. package/dist/{Function-CD3rXWfa.d.mts → Function-Dthlh2Fb.d.mts} +2 -2
  48. package/dist/{FunctionBuilder-j2VkwuGf.d.mts → FunctionBuilder-CZFBLfdV.d.mts} +4 -4
  49. package/dist/{FunctionExecutionWrapper-B0WP-Vec.d.mts → FunctionExecutionWrapper-CfuajSB5.d.mts} +2 -2
  50. package/dist/{HonoEndpointAdaptor-Bg_vTyA5.mjs → HonoEndpointAdaptor-BmB4Sc7L.mjs} +3 -3
  51. package/dist/{HonoEndpointAdaptor-Bg_vTyA5.mjs.map → HonoEndpointAdaptor-BmB4Sc7L.mjs.map} +1 -1
  52. package/dist/{HonoEndpointAdaptor-B_gJPWGD.cjs → HonoEndpointAdaptor-CnqR6PSB.cjs} +3 -3
  53. package/dist/{HonoEndpointAdaptor-B_gJPWGD.cjs.map → HonoEndpointAdaptor-CnqR6PSB.cjs.map} +1 -1
  54. package/dist/{HonoEndpointAdaptor-CLOpobdq.d.cts → HonoEndpointAdaptor-DBjipl6Q.d.mts} +6 -6
  55. package/dist/{HonoEndpointAdaptor-C9gYYBWu.d.mts → HonoEndpointAdaptor-sSG85VER.d.cts} +8 -8
  56. package/dist/{Subscriber-itwm7ugy.d.mts → Subscriber-BfxLwZpX.d.mts} +2 -2
  57. package/dist/{Subscriber-D-FPWts6.cjs → Subscriber-DvOtIyWq.cjs} +1 -1
  58. package/dist/{Subscriber-D-FPWts6.cjs.map → Subscriber-DvOtIyWq.cjs.map} +1 -1
  59. package/dist/{Subscriber-CGb8LjZa.mjs → Subscriber-JzcFFi4p.mjs} +1 -1
  60. package/dist/{Subscriber-CGb8LjZa.mjs.map → Subscriber-JzcFFi4p.mjs.map} +1 -1
  61. package/dist/{SubscriberBuilder-BfE2cL1q.cjs → SubscriberBuilder-DUuV207i.cjs} +2 -2
  62. package/dist/{SubscriberBuilder-BfE2cL1q.cjs.map → SubscriberBuilder-DUuV207i.cjs.map} +1 -1
  63. package/dist/{SubscriberBuilder-9j3JCu8-.d.mts → SubscriberBuilder-lTiTUS1o.d.mts} +2 -2
  64. package/dist/{SubscriberBuilder-BcAspHv9.mjs → SubscriberBuilder-tm4oVOt3.mjs} +2 -2
  65. package/dist/{SubscriberBuilder-BcAspHv9.mjs.map → SubscriberBuilder-tm4oVOt3.mjs.map} +1 -1
  66. package/dist/{TestEndpointAdaptor-Bew9lWsx.cjs → TestEndpointAdaptor-BtLcw4JW.cjs} +3 -3
  67. package/dist/{TestEndpointAdaptor-Bew9lWsx.cjs.map → TestEndpointAdaptor-BtLcw4JW.cjs.map} +1 -1
  68. package/dist/{TestEndpointAdaptor-C-c8v7VI.d.mts → TestEndpointAdaptor-CGQVysE0.d.mts} +3 -3
  69. package/dist/{TestEndpointAdaptor-BYCwwiYk.d.cts → TestEndpointAdaptor-CJO-og9U.d.cts} +2 -2
  70. package/dist/{TestEndpointAdaptor-JONQJeXc.mjs → TestEndpointAdaptor-o7JbNcL-.mjs} +3 -3
  71. package/dist/{TestEndpointAdaptor-JONQJeXc.mjs.map → TestEndpointAdaptor-o7JbNcL-.mjs.map} +1 -1
  72. package/dist/adaptors/aws.cjs +4 -4
  73. package/dist/adaptors/aws.d.cts +5 -5
  74. package/dist/adaptors/aws.d.mts +15 -15
  75. package/dist/adaptors/aws.mjs +4 -4
  76. package/dist/adaptors/hono.cjs +2 -2
  77. package/dist/adaptors/hono.d.cts +3 -3
  78. package/dist/adaptors/hono.d.mts +9 -9
  79. package/dist/adaptors/hono.mjs +2 -2
  80. package/dist/adaptors/testing.cjs +2 -2
  81. package/dist/adaptors/testing.d.cts +3 -3
  82. package/dist/adaptors/testing.d.mts +9 -9
  83. package/dist/adaptors/testing.mjs +2 -2
  84. package/dist/crons/Cron.d.mts +6 -6
  85. package/dist/crons/CronBuilder.d.mts +7 -7
  86. package/dist/crons/index.d.cts +4 -4
  87. package/dist/crons/index.d.mts +7 -7
  88. package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.cjs +2 -2
  89. package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.d.cts +3 -3
  90. package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.d.mts +9 -9
  91. package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.mjs +2 -2
  92. package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.cjs +3 -3
  93. package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.d.cts +4 -4
  94. package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.d.mts +10 -10
  95. package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.mjs +3 -3
  96. package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.cjs +3 -3
  97. package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.d.cts +4 -4
  98. package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.d.mts +10 -10
  99. package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.mjs +3 -3
  100. package/dist/endpoints/Authorizer.cjs +5 -14
  101. package/dist/endpoints/Authorizer.d.cts +2 -2
  102. package/dist/endpoints/Authorizer.d.mts +2 -2
  103. package/dist/endpoints/Authorizer.mjs +2 -13
  104. package/dist/endpoints/Endpoint.d.cts +2 -2
  105. package/dist/endpoints/Endpoint.d.mts +8 -8
  106. package/dist/endpoints/EndpointBuilder.cjs +2 -1
  107. package/dist/endpoints/EndpointBuilder.d.cts +3 -3
  108. package/dist/endpoints/EndpointBuilder.d.mts +9 -9
  109. package/dist/endpoints/EndpointBuilder.mjs +2 -1
  110. package/dist/endpoints/EndpointFactory.cjs +3 -2
  111. package/dist/endpoints/EndpointFactory.d.cts +4 -4
  112. package/dist/endpoints/EndpointFactory.d.mts +10 -10
  113. package/dist/endpoints/EndpointFactory.mjs +3 -2
  114. package/dist/endpoints/HonoEndpointAdaptor.cjs +2 -2
  115. package/dist/endpoints/HonoEndpointAdaptor.d.cts +3 -3
  116. package/dist/endpoints/HonoEndpointAdaptor.d.mts +9 -9
  117. package/dist/endpoints/HonoEndpointAdaptor.mjs +2 -2
  118. package/dist/endpoints/TestEndpointAdaptor.cjs +2 -2
  119. package/dist/endpoints/TestEndpointAdaptor.d.cts +3 -3
  120. package/dist/endpoints/TestEndpointAdaptor.d.mts +9 -9
  121. package/dist/endpoints/TestEndpointAdaptor.mjs +2 -2
  122. package/dist/endpoints/audit.d.cts +2 -2
  123. package/dist/endpoints/audit.d.mts +8 -8
  124. package/dist/endpoints/helpers.d.cts +2 -2
  125. package/dist/endpoints/helpers.d.mts +8 -8
  126. package/dist/endpoints/index.cjs +3 -2
  127. package/dist/endpoints/index.cjs.map +1 -1
  128. package/dist/endpoints/index.d.cts +6 -6
  129. package/dist/endpoints/index.d.mts +12 -12
  130. package/dist/endpoints/index.mjs +3 -2
  131. package/dist/endpoints/index.mjs.map +1 -1
  132. package/dist/endpoints/processAudits.cjs +1 -1
  133. package/dist/endpoints/processAudits.d.cts +24 -5
  134. package/dist/endpoints/processAudits.d.mts +30 -11
  135. package/dist/endpoints/processAudits.mjs +1 -1
  136. package/dist/functions/AWSLambdaFunction.d.mts +4 -4
  137. package/dist/functions/BaseFunctionBuilder.d.mts +2 -2
  138. package/dist/functions/Function.d.mts +2 -2
  139. package/dist/functions/FunctionBuilder.d.mts +4 -4
  140. package/dist/functions/FunctionExecutionWrapper.d.mts +3 -3
  141. package/dist/functions/TestFunctionAdaptor.d.mts +2 -2
  142. package/dist/functions/index.d.mts +5 -5
  143. package/dist/index-9gJrM7Rw.d.mts +10 -0
  144. package/dist/index.d.mts +2 -2
  145. package/dist/{processAudits-BFokHhCO.cjs → processAudits-CzHkPokQ.cjs} +13 -7
  146. package/dist/processAudits-CzHkPokQ.cjs.map +1 -0
  147. package/dist/{processAudits-DfcB-X-4.mjs → processAudits-Dj8UGqcW.mjs} +13 -7
  148. package/dist/processAudits-Dj8UGqcW.mjs.map +1 -0
  149. package/dist/publisher.d.mts +1 -1
  150. package/dist/subscribers/AWSLambdaSubscriberAdaptor.d.mts +3 -3
  151. package/dist/subscribers/Subscriber.cjs +1 -1
  152. package/dist/subscribers/Subscriber.d.mts +2 -2
  153. package/dist/subscribers/Subscriber.mjs +1 -1
  154. package/dist/subscribers/SubscriberBuilder.cjs +2 -2
  155. package/dist/subscribers/SubscriberBuilder.d.mts +3 -3
  156. package/dist/subscribers/SubscriberBuilder.mjs +2 -2
  157. package/dist/subscribers/index.cjs +2 -2
  158. package/dist/subscribers/index.d.cts +2 -2
  159. package/dist/subscribers/index.d.mts +5 -5
  160. package/dist/subscribers/index.mjs +2 -2
  161. package/dist/{types-DKf0juBf.d.mts → types-Cy1IhmUB.d.mts} +1 -1
  162. package/dist/types.d.mts +1 -1
  163. package/package.json +5 -5
  164. package/src/endpoints/AmazonApiGatewayEndpointAdaptor.ts +2 -0
  165. package/src/endpoints/Authorizer.ts +119 -1
  166. package/src/endpoints/EndpointBuilder.ts +23 -6
  167. package/src/endpoints/EndpointFactory.ts +156 -27
  168. package/src/endpoints/HonoEndpointAdaptor.ts +2 -0
  169. package/src/endpoints/TestEndpointAdaptor.ts +2 -0
  170. package/src/endpoints/processAudits.ts +39 -16
  171. package/dist/Authorizer-BTmly8ps.d.cts +0 -29
  172. package/dist/Authorizer-pmPvIVgv.d.mts +0 -29
  173. package/dist/EndpointBuilder-CYkeYpsL.cjs.map +0 -1
  174. package/dist/EndpointBuilder-W5fdXxYQ.mjs.map +0 -1
  175. package/dist/EndpointFactory-B27nfeiE.mjs.map +0 -1
  176. package/dist/EndpointFactory-D5lFZXqY.cjs.map +0 -1
  177. package/dist/endpoints/Authorizer.cjs.map +0 -1
  178. package/dist/endpoints/Authorizer.mjs.map +0 -1
  179. package/dist/index-licEVXjh.d.mts +0 -10
  180. package/dist/processAudits-BFokHhCO.cjs.map +0 -1
  181. package/dist/processAudits-DfcB-X-4.mjs.map +0 -1
@@ -0,0 +1 @@
1
+ {"version":3,"file":"processAudits-CzHkPokQ.cjs","names":["endpoint: Endpoint<\n any,\n any,\n any,\n OutSchema,\n TServices,\n TLogger,\n TSession,\n any,\n any,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction\n >","response: InferStandardSchema<OutSchema>","serviceDiscovery: ServiceDiscovery<any, any>","logger: TLogger","ctx: {\n session: TSession;\n header: HeaderFn;\n cookie: CookieFn;\n services: Record<string, unknown>;\n }","existingAuditor?: Auditor<TAuditAction>","auditor: Auditor<TAuditAction>","actor: AuditActor","DefaultAuditor","endpoint: Endpoint<\n any,\n any,\n any,\n any,\n TServices,\n TLogger,\n TSession,\n any,\n any,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >","auditContext: AuditExecutionContext<TAuditAction> | undefined","handler: (auditor?: Auditor<TAuditAction>) => Promise<T>","onComplete?: (response: T, auditor: Auditor<TAuditAction>) => Promise<void>","options?: ExecuteWithAuditTransactionOptions","response"],"sources":["../src/endpoints/processAudits.ts"],"sourcesContent":["import type {\n AuditActor,\n AuditStorage,\n AuditableAction,\n Auditor,\n} from '@geekmidas/audit';\nimport { DefaultAuditor } from '@geekmidas/audit';\nimport type { Logger } from '@geekmidas/logger';\nimport type { InferStandardSchema } from '@geekmidas/schema';\nimport type { Service, ServiceDiscovery } from '@geekmidas/services';\nimport type { StandardSchemaV1 } from '@standard-schema/spec';\nimport type { CookieFn, Endpoint, HeaderFn } from './Endpoint';\nimport type { ActorExtractor, MappedAudit } from './audit';\n\n/**\n * Process declarative audit definitions after successful endpoint execution.\n * Similar to publishConstructEvents for events.\n *\n * @param endpoint - The endpoint with audit configuration\n * @param response - The handler response to generate audit payloads from\n * @param serviceDiscovery - Service discovery for registering audit storage\n * @param logger - Logger for debug/error messages\n * @param ctx - Request context (session, headers, cookies, services)\n * @param existingAuditor - Optional existing auditor instance (e.g., from handler context).\n * If provided, uses this auditor (with its stored transaction).\n * If not provided, creates a new auditor.\n */\nexport async function processEndpointAudits<\n TServices extends Service[] = [],\n TSession = unknown,\n TLogger extends Logger = Logger,\n OutSchema extends StandardSchemaV1 | undefined = undefined,\n TAuditStorage extends AuditStorage | undefined = undefined,\n TAuditStorageServiceName extends string = string,\n TAuditAction extends AuditableAction<string, unknown> = AuditableAction<\n string,\n unknown\n >,\n>(\n endpoint: Endpoint<\n any,\n any,\n any,\n OutSchema,\n TServices,\n TLogger,\n TSession,\n any,\n any,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction\n >,\n response: InferStandardSchema<OutSchema>,\n serviceDiscovery: ServiceDiscovery<any, any>,\n logger: TLogger,\n ctx: {\n session: TSession;\n header: HeaderFn;\n cookie: CookieFn;\n services: Record<string, unknown>;\n },\n existingAuditor?: Auditor<TAuditAction>,\n): Promise<void> {\n try {\n const audits = endpoint.audits as MappedAudit<TAuditAction, OutSchema>[];\n\n // If we have an existing auditor (from handler context), we need to flush\n // any manual audits it collected, even if there are no declarative audits\n const hasExistingRecords =\n existingAuditor && existingAuditor.getRecords().length > 0;\n\n // Skip if no declarative audits and no existing records to flush\n if (!audits?.length && !hasExistingRecords) {\n logger.debug('No audits to process');\n return;\n }\n\n // If no auditor storage service and we have things to process, warn\n if (!endpoint.auditorStorageService) {\n if (hasExistingRecords || audits?.length) {\n logger.warn('No auditor storage service available');\n }\n return;\n }\n\n // Get or create auditor\n let auditor: Auditor<TAuditAction>;\n\n if (existingAuditor) {\n // Use existing auditor (preserves stored transaction and manual audits)\n auditor = existingAuditor;\n logger.debug('Using existing auditor from handler context');\n } else {\n // Create new auditor (backward compatibility)\n const services = await serviceDiscovery.register([\n endpoint.auditorStorageService,\n ]);\n const storage = services[\n endpoint.auditorStorageService.serviceName\n ] as AuditStorage;\n\n // Extract actor if configured\n let actor: AuditActor = { id: 'system', type: 'system' };\n if (endpoint.actorExtractor) {\n try {\n actor = await (\n endpoint.actorExtractor as ActorExtractor<\n TServices,\n TSession,\n TLogger\n >\n )({\n services: ctx.services as any,\n session: ctx.session,\n header: ctx.header,\n cookie: ctx.cookie,\n logger,\n });\n } catch (error) {\n logger.error(error as Error, 'Failed to extract actor for audits');\n // Continue with system actor\n }\n }\n\n auditor = new DefaultAuditor<TAuditAction>({\n actor,\n storage,\n metadata: {\n endpoint: endpoint.route,\n method: endpoint.method,\n },\n });\n }\n\n // Process each declarative audit\n if (audits?.length) {\n for (const audit of audits) {\n logger.debug({ audit: audit.type }, 'Processing declarative audit');\n\n // Check when condition\n if (audit.when && !audit.when(response as any)) {\n logger.debug(\n { audit: audit.type },\n 'Audit skipped due to when condition',\n );\n continue;\n }\n\n // Extract payload\n const payload = audit.payload(response as any);\n\n // Extract entityId if configured\n const entityId = audit.entityId?.(response as any);\n\n // Record the audit\n auditor.audit(audit.type as any, payload as any, {\n table: audit.table,\n entityId,\n });\n }\n }\n\n // Flush audits to storage\n // Note: If existingAuditor has a stored transaction (via setTransaction),\n // flush() will use it automatically\n const recordCount = auditor.getRecords().length;\n if (recordCount > 0) {\n // Check if auditor has a stored transaction (for logging purposes)\n const trx =\n 'getTransaction' in auditor\n ? (auditor as { getTransaction(): unknown }).getTransaction()\n : undefined;\n logger.debug(\n { auditCount: recordCount, hasTransaction: !!trx },\n 'Flushing audits',\n );\n await auditor.flush();\n }\n } catch (error) {\n logger.error(error as Error, 'Failed to process audits');\n // Don't rethrow - audit failures shouldn't fail the request\n }\n}\n\n/**\n * Context for audit-aware handler execution.\n */\nexport interface AuditExecutionContext<\n TAuditAction extends AuditableAction<string, unknown> = AuditableAction<\n string,\n unknown\n >,\n> {\n /** The auditor instance for recording audits */\n auditor: Auditor<TAuditAction>;\n /** The audit storage instance */\n storage: AuditStorage;\n}\n\n/**\n * Create audit context for handler execution.\n * Returns the auditor and storage for use in the handler.\n *\n * @param endpoint - The endpoint with audit configuration\n * @param serviceDiscovery - Service discovery for getting audit storage\n * @param logger - Logger for debug/error messages\n * @param ctx - Request context for actor extraction\n * @returns Audit context with auditor and storage, or undefined if not configured\n */\nexport async function createAuditContext<\n TServices extends Service[] = [],\n TSession = unknown,\n TLogger extends Logger = Logger,\n TAuditStorage extends AuditStorage | undefined = undefined,\n TAuditStorageServiceName extends string = string,\n TAuditAction extends AuditableAction<string, unknown> = AuditableAction<\n string,\n unknown\n >,\n TDatabase = undefined,\n TDatabaseServiceName extends string = string,\n>(\n endpoint: Endpoint<\n any,\n any,\n any,\n any,\n TServices,\n TLogger,\n TSession,\n any,\n any,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >,\n serviceDiscovery: ServiceDiscovery<any, any>,\n logger: TLogger,\n ctx: {\n session: TSession;\n header: HeaderFn;\n cookie: CookieFn;\n services: Record<string, unknown>;\n },\n): Promise<AuditExecutionContext<TAuditAction> | undefined> {\n if (!endpoint.auditorStorageService) {\n return undefined;\n }\n\n const services = await serviceDiscovery.register([\n endpoint.auditorStorageService,\n ]);\n const storage = services[\n endpoint.auditorStorageService.serviceName\n ] as AuditStorage;\n\n // Extract actor if configured\n let actor: AuditActor = { id: 'system', type: 'system' };\n if (endpoint.actorExtractor) {\n try {\n actor = await (\n endpoint.actorExtractor as ActorExtractor<TServices, TSession, TLogger>\n )({\n services: ctx.services as any,\n session: ctx.session,\n header: ctx.header,\n cookie: ctx.cookie,\n logger,\n });\n } catch (error) {\n logger.error(error as Error, 'Failed to extract actor for audits');\n }\n }\n\n const auditor = new DefaultAuditor<TAuditAction>({\n actor,\n storage,\n metadata: {\n endpoint: endpoint.route,\n method: endpoint.method,\n },\n });\n\n return { auditor, storage };\n}\n\n/**\n * Options for executeWithAuditTransaction.\n */\nexport interface ExecuteWithAuditTransactionOptions {\n /**\n * Database connection to use for the transaction.\n * If this is already a transaction, it will be reused instead of creating a nested one.\n * If not provided, the storage's internal database is used.\n */\n db?: unknown;\n}\n\n/**\n * Execute a handler with automatic audit transaction support.\n * If the audit storage provides a withTransaction method, wraps execution\n * in a transaction so audits are atomic with handler's database operations.\n *\n * This is database-agnostic - each storage implementation provides its own\n * transaction handling based on the underlying database (Kysely, Drizzle, etc.).\n *\n * If the db parameter is provided and is already a transaction, the storage\n * will reuse it instead of creating a nested transaction (similar to\n * packages/db/src/kysely.ts#withTransaction).\n *\n * @param auditContext - The audit context from createAuditContext\n * @param handler - The handler function to execute (receives auditor)\n * @param onComplete - Called after handler with response, to process declarative audits\n * @param options - Optional configuration including database connection\n * @returns The handler result\n */\nexport async function executeWithAuditTransaction<\n T,\n TAuditAction extends AuditableAction<string, unknown> = AuditableAction<\n string,\n unknown\n >,\n>(\n auditContext: AuditExecutionContext<TAuditAction> | undefined,\n handler: (auditor?: Auditor<TAuditAction>) => Promise<T>,\n onComplete?: (response: T, auditor: Auditor<TAuditAction>) => Promise<void>,\n options?: ExecuteWithAuditTransactionOptions,\n): Promise<T> {\n // No audit context - just run handler\n if (!auditContext) {\n return handler(undefined);\n }\n\n const { auditor, storage } = auditContext;\n\n // Check if storage provides a transaction wrapper\n if (storage.withTransaction) {\n // Wrap in transaction - audits are atomic with handler operations\n // The storage's withTransaction handles setTransaction and flush\n // Pass db so existing transactions are reused\n return storage.withTransaction(\n auditor,\n async () => {\n const response = await handler(auditor);\n\n // Process declarative audits within the transaction\n if (onComplete) {\n await onComplete(response, auditor);\n }\n\n return response;\n },\n options?.db,\n );\n }\n\n // No transaction support - run handler and flush audits after\n const response = await handler(auditor);\n\n if (onComplete) {\n await onComplete(response, auditor);\n }\n\n // Flush audits (no transaction)\n await auditor.flush();\n\n return response;\n}\n"],"mappings":"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"}
@@ -1,5 +1,4 @@
1
1
  import { DefaultAuditor } from "@geekmidas/audit";
2
- import { withAuditableTransaction } from "@geekmidas/audit/kysely";
3
2
 
4
3
  //#region src/endpoints/processAudits.ts
5
4
  /**
@@ -128,23 +127,30 @@ async function createAuditContext(endpoint, serviceDiscovery, logger, ctx) {
128
127
  }
129
128
  /**
130
129
  * Execute a handler with automatic audit transaction support.
131
- * If the audit storage has a database (via getDatabase()), wraps execution
130
+ * If the audit storage provides a withTransaction method, wraps execution
132
131
  * in a transaction so audits are atomic with handler's database operations.
133
132
  *
133
+ * This is database-agnostic - each storage implementation provides its own
134
+ * transaction handling based on the underlying database (Kysely, Drizzle, etc.).
135
+ *
136
+ * If the db parameter is provided and is already a transaction, the storage
137
+ * will reuse it instead of creating a nested transaction (similar to
138
+ * packages/db/src/kysely.ts#withTransaction).
139
+ *
134
140
  * @param auditContext - The audit context from createAuditContext
135
141
  * @param handler - The handler function to execute (receives auditor)
136
142
  * @param onComplete - Called after handler with response, to process declarative audits
143
+ * @param options - Optional configuration including database connection
137
144
  * @returns The handler result
138
145
  */
139
- async function executeWithAuditTransaction(auditContext, handler, onComplete) {
146
+ async function executeWithAuditTransaction(auditContext, handler, onComplete, options) {
140
147
  if (!auditContext) return handler(void 0);
141
148
  const { auditor, storage } = auditContext;
142
- const db = storage.getDatabase?.();
143
- if (db) return withAuditableTransaction(db, auditor, async () => {
149
+ if (storage.withTransaction) return storage.withTransaction(auditor, async () => {
144
150
  const response$1 = await handler(auditor);
145
151
  if (onComplete) await onComplete(response$1, auditor);
146
152
  return response$1;
147
- });
153
+ }, options?.db);
148
154
  const response = await handler(auditor);
149
155
  if (onComplete) await onComplete(response, auditor);
150
156
  await auditor.flush();
@@ -153,4 +159,4 @@ async function executeWithAuditTransaction(auditContext, handler, onComplete) {
153
159
 
154
160
  //#endregion
155
161
  export { createAuditContext, executeWithAuditTransaction, processEndpointAudits };
156
- //# sourceMappingURL=processAudits-DfcB-X-4.mjs.map
162
+ //# sourceMappingURL=processAudits-Dj8UGqcW.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"processAudits-Dj8UGqcW.mjs","names":["endpoint: Endpoint<\n any,\n any,\n any,\n OutSchema,\n TServices,\n TLogger,\n TSession,\n any,\n any,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction\n >","response: InferStandardSchema<OutSchema>","serviceDiscovery: ServiceDiscovery<any, any>","logger: TLogger","ctx: {\n session: TSession;\n header: HeaderFn;\n cookie: CookieFn;\n services: Record<string, unknown>;\n }","existingAuditor?: Auditor<TAuditAction>","auditor: Auditor<TAuditAction>","actor: AuditActor","endpoint: Endpoint<\n any,\n any,\n any,\n any,\n TServices,\n TLogger,\n TSession,\n any,\n any,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >","auditContext: AuditExecutionContext<TAuditAction> | undefined","handler: (auditor?: Auditor<TAuditAction>) => Promise<T>","onComplete?: (response: T, auditor: Auditor<TAuditAction>) => Promise<void>","options?: ExecuteWithAuditTransactionOptions","response"],"sources":["../src/endpoints/processAudits.ts"],"sourcesContent":["import type {\n AuditActor,\n AuditStorage,\n AuditableAction,\n Auditor,\n} from '@geekmidas/audit';\nimport { DefaultAuditor } from '@geekmidas/audit';\nimport type { Logger } from '@geekmidas/logger';\nimport type { InferStandardSchema } from '@geekmidas/schema';\nimport type { Service, ServiceDiscovery } from '@geekmidas/services';\nimport type { StandardSchemaV1 } from '@standard-schema/spec';\nimport type { CookieFn, Endpoint, HeaderFn } from './Endpoint';\nimport type { ActorExtractor, MappedAudit } from './audit';\n\n/**\n * Process declarative audit definitions after successful endpoint execution.\n * Similar to publishConstructEvents for events.\n *\n * @param endpoint - The endpoint with audit configuration\n * @param response - The handler response to generate audit payloads from\n * @param serviceDiscovery - Service discovery for registering audit storage\n * @param logger - Logger for debug/error messages\n * @param ctx - Request context (session, headers, cookies, services)\n * @param existingAuditor - Optional existing auditor instance (e.g., from handler context).\n * If provided, uses this auditor (with its stored transaction).\n * If not provided, creates a new auditor.\n */\nexport async function processEndpointAudits<\n TServices extends Service[] = [],\n TSession = unknown,\n TLogger extends Logger = Logger,\n OutSchema extends StandardSchemaV1 | undefined = undefined,\n TAuditStorage extends AuditStorage | undefined = undefined,\n TAuditStorageServiceName extends string = string,\n TAuditAction extends AuditableAction<string, unknown> = AuditableAction<\n string,\n unknown\n >,\n>(\n endpoint: Endpoint<\n any,\n any,\n any,\n OutSchema,\n TServices,\n TLogger,\n TSession,\n any,\n any,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction\n >,\n response: InferStandardSchema<OutSchema>,\n serviceDiscovery: ServiceDiscovery<any, any>,\n logger: TLogger,\n ctx: {\n session: TSession;\n header: HeaderFn;\n cookie: CookieFn;\n services: Record<string, unknown>;\n },\n existingAuditor?: Auditor<TAuditAction>,\n): Promise<void> {\n try {\n const audits = endpoint.audits as MappedAudit<TAuditAction, OutSchema>[];\n\n // If we have an existing auditor (from handler context), we need to flush\n // any manual audits it collected, even if there are no declarative audits\n const hasExistingRecords =\n existingAuditor && existingAuditor.getRecords().length > 0;\n\n // Skip if no declarative audits and no existing records to flush\n if (!audits?.length && !hasExistingRecords) {\n logger.debug('No audits to process');\n return;\n }\n\n // If no auditor storage service and we have things to process, warn\n if (!endpoint.auditorStorageService) {\n if (hasExistingRecords || audits?.length) {\n logger.warn('No auditor storage service available');\n }\n return;\n }\n\n // Get or create auditor\n let auditor: Auditor<TAuditAction>;\n\n if (existingAuditor) {\n // Use existing auditor (preserves stored transaction and manual audits)\n auditor = existingAuditor;\n logger.debug('Using existing auditor from handler context');\n } else {\n // Create new auditor (backward compatibility)\n const services = await serviceDiscovery.register([\n endpoint.auditorStorageService,\n ]);\n const storage = services[\n endpoint.auditorStorageService.serviceName\n ] as AuditStorage;\n\n // Extract actor if configured\n let actor: AuditActor = { id: 'system', type: 'system' };\n if (endpoint.actorExtractor) {\n try {\n actor = await (\n endpoint.actorExtractor as ActorExtractor<\n TServices,\n TSession,\n TLogger\n >\n )({\n services: ctx.services as any,\n session: ctx.session,\n header: ctx.header,\n cookie: ctx.cookie,\n logger,\n });\n } catch (error) {\n logger.error(error as Error, 'Failed to extract actor for audits');\n // Continue with system actor\n }\n }\n\n auditor = new DefaultAuditor<TAuditAction>({\n actor,\n storage,\n metadata: {\n endpoint: endpoint.route,\n method: endpoint.method,\n },\n });\n }\n\n // Process each declarative audit\n if (audits?.length) {\n for (const audit of audits) {\n logger.debug({ audit: audit.type }, 'Processing declarative audit');\n\n // Check when condition\n if (audit.when && !audit.when(response as any)) {\n logger.debug(\n { audit: audit.type },\n 'Audit skipped due to when condition',\n );\n continue;\n }\n\n // Extract payload\n const payload = audit.payload(response as any);\n\n // Extract entityId if configured\n const entityId = audit.entityId?.(response as any);\n\n // Record the audit\n auditor.audit(audit.type as any, payload as any, {\n table: audit.table,\n entityId,\n });\n }\n }\n\n // Flush audits to storage\n // Note: If existingAuditor has a stored transaction (via setTransaction),\n // flush() will use it automatically\n const recordCount = auditor.getRecords().length;\n if (recordCount > 0) {\n // Check if auditor has a stored transaction (for logging purposes)\n const trx =\n 'getTransaction' in auditor\n ? (auditor as { getTransaction(): unknown }).getTransaction()\n : undefined;\n logger.debug(\n { auditCount: recordCount, hasTransaction: !!trx },\n 'Flushing audits',\n );\n await auditor.flush();\n }\n } catch (error) {\n logger.error(error as Error, 'Failed to process audits');\n // Don't rethrow - audit failures shouldn't fail the request\n }\n}\n\n/**\n * Context for audit-aware handler execution.\n */\nexport interface AuditExecutionContext<\n TAuditAction extends AuditableAction<string, unknown> = AuditableAction<\n string,\n unknown\n >,\n> {\n /** The auditor instance for recording audits */\n auditor: Auditor<TAuditAction>;\n /** The audit storage instance */\n storage: AuditStorage;\n}\n\n/**\n * Create audit context for handler execution.\n * Returns the auditor and storage for use in the handler.\n *\n * @param endpoint - The endpoint with audit configuration\n * @param serviceDiscovery - Service discovery for getting audit storage\n * @param logger - Logger for debug/error messages\n * @param ctx - Request context for actor extraction\n * @returns Audit context with auditor and storage, or undefined if not configured\n */\nexport async function createAuditContext<\n TServices extends Service[] = [],\n TSession = unknown,\n TLogger extends Logger = Logger,\n TAuditStorage extends AuditStorage | undefined = undefined,\n TAuditStorageServiceName extends string = string,\n TAuditAction extends AuditableAction<string, unknown> = AuditableAction<\n string,\n unknown\n >,\n TDatabase = undefined,\n TDatabaseServiceName extends string = string,\n>(\n endpoint: Endpoint<\n any,\n any,\n any,\n any,\n TServices,\n TLogger,\n TSession,\n any,\n any,\n TAuditStorage,\n TAuditStorageServiceName,\n TAuditAction,\n TDatabase,\n TDatabaseServiceName\n >,\n serviceDiscovery: ServiceDiscovery<any, any>,\n logger: TLogger,\n ctx: {\n session: TSession;\n header: HeaderFn;\n cookie: CookieFn;\n services: Record<string, unknown>;\n },\n): Promise<AuditExecutionContext<TAuditAction> | undefined> {\n if (!endpoint.auditorStorageService) {\n return undefined;\n }\n\n const services = await serviceDiscovery.register([\n endpoint.auditorStorageService,\n ]);\n const storage = services[\n endpoint.auditorStorageService.serviceName\n ] as AuditStorage;\n\n // Extract actor if configured\n let actor: AuditActor = { id: 'system', type: 'system' };\n if (endpoint.actorExtractor) {\n try {\n actor = await (\n endpoint.actorExtractor as ActorExtractor<TServices, TSession, TLogger>\n )({\n services: ctx.services as any,\n session: ctx.session,\n header: ctx.header,\n cookie: ctx.cookie,\n logger,\n });\n } catch (error) {\n logger.error(error as Error, 'Failed to extract actor for audits');\n }\n }\n\n const auditor = new DefaultAuditor<TAuditAction>({\n actor,\n storage,\n metadata: {\n endpoint: endpoint.route,\n method: endpoint.method,\n },\n });\n\n return { auditor, storage };\n}\n\n/**\n * Options for executeWithAuditTransaction.\n */\nexport interface ExecuteWithAuditTransactionOptions {\n /**\n * Database connection to use for the transaction.\n * If this is already a transaction, it will be reused instead of creating a nested one.\n * If not provided, the storage's internal database is used.\n */\n db?: unknown;\n}\n\n/**\n * Execute a handler with automatic audit transaction support.\n * If the audit storage provides a withTransaction method, wraps execution\n * in a transaction so audits are atomic with handler's database operations.\n *\n * This is database-agnostic - each storage implementation provides its own\n * transaction handling based on the underlying database (Kysely, Drizzle, etc.).\n *\n * If the db parameter is provided and is already a transaction, the storage\n * will reuse it instead of creating a nested transaction (similar to\n * packages/db/src/kysely.ts#withTransaction).\n *\n * @param auditContext - The audit context from createAuditContext\n * @param handler - The handler function to execute (receives auditor)\n * @param onComplete - Called after handler with response, to process declarative audits\n * @param options - Optional configuration including database connection\n * @returns The handler result\n */\nexport async function executeWithAuditTransaction<\n T,\n TAuditAction extends AuditableAction<string, unknown> = AuditableAction<\n string,\n unknown\n >,\n>(\n auditContext: AuditExecutionContext<TAuditAction> | undefined,\n handler: (auditor?: Auditor<TAuditAction>) => Promise<T>,\n onComplete?: (response: T, auditor: Auditor<TAuditAction>) => Promise<void>,\n options?: ExecuteWithAuditTransactionOptions,\n): Promise<T> {\n // No audit context - just run handler\n if (!auditContext) {\n return handler(undefined);\n }\n\n const { auditor, storage } = auditContext;\n\n // Check if storage provides a transaction wrapper\n if (storage.withTransaction) {\n // Wrap in transaction - audits are atomic with handler operations\n // The storage's withTransaction handles setTransaction and flush\n // Pass db so existing transactions are reused\n return storage.withTransaction(\n auditor,\n async () => {\n const response = await handler(auditor);\n\n // Process declarative audits within the transaction\n if (onComplete) {\n await onComplete(response, auditor);\n }\n\n return response;\n },\n options?.db,\n );\n }\n\n // No transaction support - run handler and flush audits after\n const response = await handler(auditor);\n\n if (onComplete) {\n await onComplete(response, auditor);\n }\n\n // Flush audits (no transaction)\n await auditor.flush();\n\n return response;\n}\n"],"mappings":"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"}
@@ -1,4 +1,4 @@
1
- import { Construct } from "./Construct-DDR0295I.mjs";
1
+ import { Construct } from "./Construct-DCPATqec.mjs";
2
2
  import { Service, ServiceDiscovery } from "@geekmidas/services";
3
3
  import { AuditStorage } from "@geekmidas/audit";
4
4
  import { EventPublisher, MappedEvent } from "@geekmidas/events";
@@ -1,4 +1,4 @@
1
- import "../Construct-DDR0295I.mjs";
2
- import "../Subscriber-itwm7ugy.mjs";
3
- import { AWSLambdaHandler, AWSLambdaSubscriber } from "../AWSLambdaSubscriberAdaptor-CWbBNRz3.mjs";
1
+ import "../Construct-DCPATqec.mjs";
2
+ import "../Subscriber-BfxLwZpX.mjs";
3
+ import { AWSLambdaHandler, AWSLambdaSubscriber } from "../AWSLambdaSubscriberAdaptor-DtNZPMoV.mjs";
4
4
  export { AWSLambdaHandler, AWSLambdaSubscriber };
@@ -1,4 +1,4 @@
1
1
  require('../Construct-BYSPikVm.cjs');
2
- const require_Subscriber = require('../Subscriber-D-FPWts6.cjs');
2
+ const require_Subscriber = require('../Subscriber-DvOtIyWq.cjs');
3
3
 
4
4
  exports.Subscriber = require_Subscriber.Subscriber;
@@ -1,3 +1,3 @@
1
- import "../Construct-DDR0295I.mjs";
2
- import { Subscriber, SubscriberContext, SubscriberHandler } from "../Subscriber-itwm7ugy.mjs";
1
+ import "../Construct-DCPATqec.mjs";
2
+ import { Subscriber, SubscriberContext, SubscriberHandler } from "../Subscriber-BfxLwZpX.mjs";
3
3
  export { Subscriber, SubscriberContext, SubscriberHandler };
@@ -1,4 +1,4 @@
1
1
  import "../Construct-LWeB1rSQ.mjs";
2
- import { Subscriber } from "../Subscriber-CGb8LjZa.mjs";
2
+ import { Subscriber } from "../Subscriber-JzcFFi4p.mjs";
3
3
 
4
4
  export { Subscriber };
@@ -1,5 +1,5 @@
1
1
  require('../Construct-BYSPikVm.cjs');
2
- require('../Subscriber-D-FPWts6.cjs');
3
- const require_SubscriberBuilder = require('../SubscriberBuilder-BfE2cL1q.cjs');
2
+ require('../Subscriber-DvOtIyWq.cjs');
3
+ const require_SubscriberBuilder = require('../SubscriberBuilder-DUuV207i.cjs');
4
4
 
5
5
  exports.SubscriberBuilder = require_SubscriberBuilder.SubscriberBuilder;
@@ -1,4 +1,4 @@
1
- import "../Construct-DDR0295I.mjs";
2
- import "../Subscriber-itwm7ugy.mjs";
3
- import { SubscriberBuilder } from "../SubscriberBuilder-9j3JCu8-.mjs";
1
+ import "../Construct-DCPATqec.mjs";
2
+ import "../Subscriber-BfxLwZpX.mjs";
3
+ import { SubscriberBuilder } from "../SubscriberBuilder-lTiTUS1o.mjs";
4
4
  export { SubscriberBuilder };
@@ -1,5 +1,5 @@
1
1
  import "../Construct-LWeB1rSQ.mjs";
2
- import "../Subscriber-CGb8LjZa.mjs";
3
- import { SubscriberBuilder } from "../SubscriberBuilder-BcAspHv9.mjs";
2
+ import "../Subscriber-JzcFFi4p.mjs";
3
+ import { SubscriberBuilder } from "../SubscriberBuilder-tm4oVOt3.mjs";
4
4
 
5
5
  export { SubscriberBuilder };
@@ -1,6 +1,6 @@
1
1
  require('../Construct-BYSPikVm.cjs');
2
- const require_Subscriber = require('../Subscriber-D-FPWts6.cjs');
3
- const require_SubscriberBuilder = require('../SubscriberBuilder-BfE2cL1q.cjs');
2
+ const require_Subscriber = require('../Subscriber-DvOtIyWq.cjs');
3
+ const require_SubscriberBuilder = require('../SubscriberBuilder-DUuV207i.cjs');
4
4
 
5
5
  //#region src/subscribers/index.ts
6
6
  const s = new require_SubscriberBuilder.SubscriberBuilder();
@@ -1,10 +1,10 @@
1
1
  import "../Construct-Dkd8Kvc9.cjs";
2
2
  import { Subscriber } from "../Subscriber-DMSzvO_J.cjs";
3
3
  import { SubscriberBuilder } from "../SubscriberBuilder-BxJM3Hz_.cjs";
4
- import * as _geekmidas_logger4 from "@geekmidas/logger";
4
+ import * as _geekmidas_logger10 from "@geekmidas/logger";
5
5
 
6
6
  //#region src/subscribers/index.d.ts
7
- declare const s: SubscriberBuilder<[], _geekmidas_logger4.Logger, undefined, undefined, string, []>;
7
+ declare const s: SubscriberBuilder<[], _geekmidas_logger10.Logger, undefined, undefined, string, []>;
8
8
  //#endregion
9
9
  export { Subscriber, SubscriberBuilder, s };
10
10
  //# sourceMappingURL=index.d.cts.map
@@ -1,10 +1,10 @@
1
- import "../Construct-DDR0295I.mjs";
2
- import { Subscriber } from "../Subscriber-itwm7ugy.mjs";
3
- import { SubscriberBuilder } from "../SubscriberBuilder-9j3JCu8-.mjs";
4
- import * as _geekmidas_logger9 from "@geekmidas/logger";
1
+ import "../Construct-DCPATqec.mjs";
2
+ import { Subscriber } from "../Subscriber-BfxLwZpX.mjs";
3
+ import { SubscriberBuilder } from "../SubscriberBuilder-lTiTUS1o.mjs";
4
+ import * as _geekmidas_logger10 from "@geekmidas/logger";
5
5
 
6
6
  //#region src/subscribers/index.d.ts
7
- declare const s: SubscriberBuilder<[], _geekmidas_logger9.Logger, undefined, undefined, string, []>;
7
+ declare const s: SubscriberBuilder<[], _geekmidas_logger10.Logger, undefined, undefined, string, []>;
8
8
  //#endregion
9
9
  export { Subscriber, SubscriberBuilder, s };
10
10
  //# sourceMappingURL=index.d.mts.map
@@ -1,6 +1,6 @@
1
1
  import "../Construct-LWeB1rSQ.mjs";
2
- import { Subscriber } from "../Subscriber-CGb8LjZa.mjs";
3
- import { SubscriberBuilder } from "../SubscriberBuilder-BcAspHv9.mjs";
2
+ import { Subscriber } from "../Subscriber-JzcFFi4p.mjs";
3
+ import { SubscriberBuilder } from "../SubscriberBuilder-tm4oVOt3.mjs";
4
4
 
5
5
  //#region src/subscribers/index.ts
6
6
  const s = new SubscriberBuilder();
@@ -4,4 +4,4 @@ type HttpMethod = 'GET' | 'POST' | 'PUT' | 'PATCH' | 'DELETE' | 'OPTIONS';
4
4
  type LowerHttpMethod<T extends HttpMethod> = Lowercase<T>;
5
5
  //#endregion
6
6
  export { HttpMethod, LowerHttpMethod, RemoveUndefined };
7
- //# sourceMappingURL=types-DKf0juBf.d.mts.map
7
+ //# sourceMappingURL=types-Cy1IhmUB.d.mts.map
package/dist/types.d.mts CHANGED
@@ -1,2 +1,2 @@
1
- import { HttpMethod, LowerHttpMethod, RemoveUndefined } from "./types-DKf0juBf.mjs";
1
+ import { HttpMethod, LowerHttpMethod, RemoveUndefined } from "./types-Cy1IhmUB.mjs";
2
2
  export { HttpMethod, LowerHttpMethod, RemoveUndefined };
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@geekmidas/constructs",
3
- "version": "0.0.22",
3
+ "version": "0.1.0",
4
4
  "private": false,
5
5
  "type": "module",
6
6
  "exports": {
@@ -67,13 +67,13 @@
67
67
  "lodash.uniqby": "~4.7.0",
68
68
  "openapi-types": "~12.1.3",
69
69
  "@geekmidas/audit": "0.0.8",
70
+ "@geekmidas/errors": "0.0.1",
70
71
  "@geekmidas/cache": "0.0.7",
71
72
  "@geekmidas/events": "0.0.2",
72
- "@geekmidas/errors": "0.0.1",
73
73
  "@geekmidas/logger": "0.0.1",
74
- "@geekmidas/rate-limit": "0.1.0",
74
+ "@geekmidas/services": "0.0.1",
75
75
  "@geekmidas/schema": "0.0.2",
76
- "@geekmidas/services": "0.0.1"
76
+ "@geekmidas/rate-limit": "0.1.0"
77
77
  },
78
78
  "devDependencies": {
79
79
  "@types/lodash.compact": "~3.0.9",
@@ -91,7 +91,7 @@
91
91
  "@middy/core": ">=6.3.1",
92
92
  "@types/aws-lambda": ">=8.10.92",
93
93
  "hono": ">=4.8.2",
94
- "@geekmidas/envkit": "~0.0.8"
94
+ "@geekmidas/envkit": "~0.1.0"
95
95
  },
96
96
  "scripts": {
97
97
  "ts": "tsc --noEmit --skipLibCheck src/**/*.ts"
@@ -356,6 +356,8 @@ export abstract class AmazonApiGatewayEndpoint<
356
356
  });
357
357
  }
358
358
  },
359
+ // Pass rawDb so storage can reuse existing transactions
360
+ { db: rawDb },
359
361
  );
360
362
 
361
363
  const { output, metadata } = result;
@@ -1,3 +1,89 @@
1
+ /**
2
+ * OpenAPI 3.1 compliant security scheme definition.
3
+ * @see https://spec.openapis.org/oas/v3.1.0#security-scheme-object
4
+ */
5
+ export interface SecurityScheme {
6
+ /** The type of the security scheme */
7
+ type: 'apiKey' | 'http' | 'mutualTLS' | 'oauth2' | 'openIdConnect';
8
+ /** A description for security scheme */
9
+ description?: string;
10
+ /** Required for apiKey. The name of the header, query or cookie parameter */
11
+ name?: string;
12
+ /** Required for apiKey. The location of the API key */
13
+ in?: 'query' | 'header' | 'cookie';
14
+ /** Required for http. The name of the HTTP Authorization scheme (e.g., 'bearer') */
15
+ scheme?: string;
16
+ /** Optional for http bearer. A hint to the format of the bearer token */
17
+ bearerFormat?: string;
18
+ /** Required for oauth2. An object containing configuration for the flow types */
19
+ flows?: OAuthFlows;
20
+ /** Required for openIdConnect. The URL to discover OAuth2 configuration */
21
+ openIdConnectUrl?: string;
22
+ /** Vendor extensions (e.g., x-amazon-apigateway-authtype) */
23
+ [key: `x-${string}`]: unknown;
24
+ }
25
+
26
+ /**
27
+ * OAuth2 flow configuration
28
+ */
29
+ export interface OAuthFlows {
30
+ implicit?: OAuthFlow;
31
+ password?: OAuthFlow;
32
+ clientCredentials?: OAuthFlow;
33
+ authorizationCode?: OAuthFlow;
34
+ }
35
+
36
+ export interface OAuthFlow {
37
+ authorizationUrl?: string;
38
+ tokenUrl?: string;
39
+ refreshUrl?: string;
40
+ scopes: Record<string, string>;
41
+ }
42
+
43
+ /**
44
+ * Built-in security schemes available by default.
45
+ * Users can use these without defining them via .securitySchemes().
46
+ */
47
+ export const BUILT_IN_SECURITY_SCHEMES = {
48
+ jwt: {
49
+ type: 'http',
50
+ scheme: 'bearer',
51
+ bearerFormat: 'JWT',
52
+ description: 'JWT Bearer token authentication',
53
+ },
54
+ bearer: {
55
+ type: 'http',
56
+ scheme: 'bearer',
57
+ description: 'Bearer token authentication',
58
+ },
59
+ apiKey: {
60
+ type: 'apiKey',
61
+ in: 'header',
62
+ name: 'X-API-Key',
63
+ description: 'API key authentication via header',
64
+ },
65
+ oauth2: {
66
+ type: 'oauth2',
67
+ flows: {},
68
+ description: 'OAuth 2.0 authentication',
69
+ },
70
+ oidc: {
71
+ type: 'openIdConnect',
72
+ openIdConnectUrl: '',
73
+ description: 'OpenID Connect authentication',
74
+ },
75
+ iam: {
76
+ type: 'apiKey',
77
+ in: 'header',
78
+ name: 'Authorization',
79
+ description: 'AWS IAM Signature Version 4 authentication',
80
+ 'x-amazon-apigateway-authtype': 'awsSigv4',
81
+ },
82
+ } as const satisfies Record<string, SecurityScheme>;
83
+
84
+ /** Names of built-in security schemes */
85
+ export type BuiltInSecuritySchemeId = keyof typeof BUILT_IN_SECURITY_SCHEMES;
86
+
1
87
  /**
2
88
  * Represents an authorizer configuration for endpoints
3
89
  */
@@ -6,18 +92,25 @@ export interface Authorizer {
6
92
  * Unique identifier for the authorizer
7
93
  */
8
94
  name: string;
95
+ /**
96
+ * The OpenAPI security scheme definition for this authorizer
97
+ */
98
+ securityScheme?: SecurityScheme;
9
99
  /**
10
100
  * Type of authorizer (e.g., 'iam', 'jwt', 'custom')
101
+ * @deprecated Use securityScheme.type instead
11
102
  */
12
103
  type?: string;
13
104
  /**
14
105
  * Description of what this authorizer does
106
+ * @deprecated Use securityScheme.description instead
15
107
  */
16
108
  description?: string;
17
109
  /**
18
110
  * Additional metadata specific to the authorizer type
111
+ * @deprecated Use securityScheme with x-* extensions instead
19
112
  */
20
- metadata?: Record<string, any>;
113
+ metadata?: Record<string, unknown>;
21
114
  }
22
115
 
23
116
  /**
@@ -32,3 +125,28 @@ export function createAuthorizer(
32
125
  ...options,
33
126
  };
34
127
  }
128
+
129
+ /**
130
+ * Check if a name is a built-in security scheme
131
+ */
132
+ export function isBuiltInSecurityScheme(
133
+ name: string,
134
+ ): name is BuiltInSecuritySchemeId {
135
+ return name in BUILT_IN_SECURITY_SCHEMES;
136
+ }
137
+
138
+ /**
139
+ * Get a security scheme by name (built-in or custom)
140
+ */
141
+ export function getSecurityScheme(
142
+ name: string,
143
+ customSchemes?: Record<string, SecurityScheme>,
144
+ ): SecurityScheme | undefined {
145
+ if (customSchemes && name in customSchemes) {
146
+ return customSchemes[name];
147
+ }
148
+ if (isBuiltInSecurityScheme(name)) {
149
+ return BUILT_IN_SECURITY_SCHEMES[name];
150
+ }
151
+ return undefined;
152
+ }
@@ -12,7 +12,8 @@ import uniqBy from 'lodash.uniqby';
12
12
  import { ConstructType } from '../Construct';
13
13
  import { BaseFunctionBuilder } from '../functions';
14
14
  import type { HttpMethod } from '../types';
15
- import type { Authorizer } from './Authorizer';
15
+ import type { Authorizer, SecurityScheme } from './Authorizer';
16
+ import { getSecurityScheme } from './Authorizer';
16
17
  import { Endpoint, type EndpointSchemas } from './Endpoint';
17
18
  import type {
18
19
  AuthorizeFn,
@@ -66,6 +67,7 @@ export class EndpointBuilder<
66
67
  _authorizerName?: TAuthorizers[number];
67
68
  _actorExtractor?: ActorExtractor<TServices, TSession, TLogger>;
68
69
  _audits: MappedAudit<TAuditAction, OutSchema>[] = [];
70
+ _customSecuritySchemes: Record<string, SecurityScheme> = {};
69
71
 
70
72
  constructor(
71
73
  readonly route: TRoute,
@@ -617,11 +619,26 @@ export class EndpointBuilder<
617
619
  > {
618
620
  // Find authorizer metadata if name is set
619
621
  // If the authorizer name is set but not in availableAuthorizers, create a simple authorizer object
620
- const authorizer = this._authorizerName
621
- ? (this._availableAuthorizers.find(
622
- (a) => a.name === this._authorizerName,
623
- ) ?? { name: this._authorizerName })
624
- : undefined;
622
+ let authorizer: Authorizer | undefined;
623
+ if (this._authorizerName) {
624
+ const existingAuthorizer = this._availableAuthorizers.find(
625
+ (a) => a.name === this._authorizerName,
626
+ );
627
+
628
+ if (existingAuthorizer) {
629
+ authorizer = existingAuthorizer;
630
+ } else {
631
+ // Create authorizer with security scheme if available (built-in or custom)
632
+ const securityScheme = getSecurityScheme(
633
+ this._authorizerName as string,
634
+ this._customSecuritySchemes,
635
+ );
636
+ authorizer = {
637
+ name: this._authorizerName as string,
638
+ securityScheme,
639
+ };
640
+ }
641
+ }
625
642
 
626
643
  return new Endpoint({
627
644
  fn,