@geekmidas/constructs 0.3.2 → 0.6.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 (557) hide show
  1. package/dist/{AWSLambdaFunction-C54a1doJ.mjs → AWSLambdaFunction-BMTaCrG9.mjs} +6 -6
  2. package/dist/AWSLambdaFunction-BMTaCrG9.mjs.map +1 -0
  3. package/dist/{AWSLambdaFunction-BdebRMUh.d.mts → AWSLambdaFunction-BfLF5vNa.d.cts} +6 -6
  4. package/dist/AWSLambdaFunction-BfLF5vNa.d.cts.map +1 -0
  5. package/dist/{AWSLambdaFunction-EPGY4s7i.cjs → AWSLambdaFunction-C1gRqLO7.cjs} +6 -6
  6. package/dist/AWSLambdaFunction-C1gRqLO7.cjs.map +1 -0
  7. package/dist/{AWSLambdaFunction-D0tEOTXQ.d.cts → AWSLambdaFunction-K94K8JU2.d.mts} +6 -6
  8. package/dist/AWSLambdaFunction-K94K8JU2.d.mts.map +1 -0
  9. package/dist/{AWSLambdaSubscriberAdaptor-Dx-Ks1Jp.d.cts → AWSLambdaSubscriberAdaptor-B2Gk3fgx.d.cts} +2 -2
  10. package/dist/AWSLambdaSubscriberAdaptor-B2Gk3fgx.d.cts.map +1 -0
  11. package/dist/{AWSLambdaSubscriberAdaptor-BLfO612H.mjs → AWSLambdaSubscriberAdaptor-BD3FwGUb.mjs} +3 -3
  12. package/dist/AWSLambdaSubscriberAdaptor-BD3FwGUb.mjs.map +1 -0
  13. package/dist/{AWSLambdaSubscriberAdaptor-BNcYYZ-P.cjs → AWSLambdaSubscriberAdaptor-DQDnt1Xk.cjs} +3 -3
  14. package/dist/AWSLambdaSubscriberAdaptor-DQDnt1Xk.cjs.map +1 -0
  15. package/dist/{AWSLambdaSubscriberAdaptor-DrFAvHOp.d.mts → AWSLambdaSubscriberAdaptor-QuMFwltJ.d.mts} +2 -2
  16. package/dist/AWSLambdaSubscriberAdaptor-QuMFwltJ.d.mts.map +1 -0
  17. package/dist/{AmazonApiGatewayEndpointAdaptor-BT9JXihC.mjs → AmazonApiGatewayEndpointAdaptor-B4gLJ2dx.mjs} +56 -14
  18. package/dist/AmazonApiGatewayEndpointAdaptor-B4gLJ2dx.mjs.map +1 -0
  19. package/dist/{AmazonApiGatewayEndpointAdaptor-DLqnYQ4E.d.cts → AmazonApiGatewayEndpointAdaptor-C987ZCsM.d.cts} +51 -5
  20. package/dist/AmazonApiGatewayEndpointAdaptor-C987ZCsM.d.cts.map +1 -0
  21. package/dist/{AmazonApiGatewayEndpointAdaptor-DNFvvdmW.cjs → AmazonApiGatewayEndpointAdaptor-CwRKMRr_.cjs} +55 -13
  22. package/dist/AmazonApiGatewayEndpointAdaptor-CwRKMRr_.cjs.map +1 -0
  23. package/dist/{AmazonApiGatewayEndpointAdaptor-qlKXuZxy.d.mts → AmazonApiGatewayEndpointAdaptor-DpPfO6Vj.d.mts} +52 -6
  24. package/dist/AmazonApiGatewayEndpointAdaptor-DpPfO6Vj.d.mts.map +1 -0
  25. package/dist/{AmazonApiGatewayV1EndpointAdaptor-Ccl8B8kG.cjs → AmazonApiGatewayV1EndpointAdaptor-BeZMu5DZ.cjs} +4 -4
  26. package/dist/AmazonApiGatewayV1EndpointAdaptor-BeZMu5DZ.cjs.map +1 -0
  27. package/dist/{AmazonApiGatewayV1EndpointAdaptor-MRUxs3Xi.d.mts → AmazonApiGatewayV1EndpointAdaptor-DI_5kdqw.d.mts} +5 -5
  28. package/dist/AmazonApiGatewayV1EndpointAdaptor-DI_5kdqw.d.mts.map +1 -0
  29. package/dist/{AmazonApiGatewayV1EndpointAdaptor-DyUaJIhB.d.cts → AmazonApiGatewayV1EndpointAdaptor-DkGMOpoB.d.cts} +4 -4
  30. package/dist/AmazonApiGatewayV1EndpointAdaptor-DkGMOpoB.d.cts.map +1 -0
  31. package/dist/{AmazonApiGatewayV1EndpointAdaptor-CShQI8Gk.mjs → AmazonApiGatewayV1EndpointAdaptor-VccB6FKp.mjs} +4 -4
  32. package/dist/AmazonApiGatewayV1EndpointAdaptor-VccB6FKp.mjs.map +1 -0
  33. package/dist/{AmazonApiGatewayV2EndpointAdaptor-M1-w0U5R.d.cts → AmazonApiGatewayV2EndpointAdaptor-BtBnMJS_.d.cts} +4 -6
  34. package/dist/AmazonApiGatewayV2EndpointAdaptor-BtBnMJS_.d.cts.map +1 -0
  35. package/dist/{AmazonApiGatewayV2EndpointAdaptor-D8-0Aab4.cjs → AmazonApiGatewayV2EndpointAdaptor-BwsL9Gia.cjs} +4 -4
  36. package/dist/AmazonApiGatewayV2EndpointAdaptor-BwsL9Gia.cjs.map +1 -0
  37. package/dist/{AmazonApiGatewayV2EndpointAdaptor-lCRzGE4q.mjs → AmazonApiGatewayV2EndpointAdaptor-CljxmUfz.mjs} +4 -4
  38. package/dist/AmazonApiGatewayV2EndpointAdaptor-CljxmUfz.mjs.map +1 -0
  39. package/dist/{AmazonApiGatewayV2EndpointAdaptor-fuLM6M9k.d.mts → AmazonApiGatewayV2EndpointAdaptor-DnCoqlvw.d.mts} +5 -7
  40. package/dist/AmazonApiGatewayV2EndpointAdaptor-DnCoqlvw.d.mts.map +1 -0
  41. package/dist/{Authorizer-C0ge_tc8.cjs → Authorizer-BXxBee2P.cjs} +1 -1
  42. package/dist/Authorizer-BXxBee2P.cjs.map +1 -0
  43. package/dist/{Authorizer-r9U3y_ms.mjs → Authorizer-BgjU8-z6.mjs} +1 -1
  44. package/dist/Authorizer-BgjU8-z6.mjs.map +1 -0
  45. package/dist/{Authorizer-gWxYsGEp.d.mts → Authorizer-DCcYOx3h.d.mts} +1 -1
  46. package/dist/Authorizer-DCcYOx3h.d.mts.map +1 -0
  47. package/dist/{Authorizer-B-btowNd.d.cts → Authorizer-DWtwC8we.d.cts} +1 -1
  48. package/dist/Authorizer-DWtwC8we.d.cts.map +1 -0
  49. package/dist/{BaseFunctionBuilder-BAtutR6q.d.cts → BaseFunctionBuilder-BqZCqIeU.d.mts} +3 -3
  50. package/dist/BaseFunctionBuilder-BqZCqIeU.d.mts.map +1 -0
  51. package/dist/{BaseFunctionBuilder-DRY419e7.d.mts → BaseFunctionBuilder-CcK691ni.d.cts} +3 -3
  52. package/dist/BaseFunctionBuilder-CcK691ni.d.cts.map +1 -0
  53. package/dist/{BaseFunctionBuilder-Czi1Jwza.mjs → BaseFunctionBuilder-Dsqe6pnn.mjs} +2 -2
  54. package/dist/BaseFunctionBuilder-Dsqe6pnn.mjs.map +1 -0
  55. package/dist/{BaseFunctionBuilder-MYG3C9ug.cjs → BaseFunctionBuilder-Z0XwrKB5.cjs} +2 -2
  56. package/dist/BaseFunctionBuilder-Z0XwrKB5.cjs.map +1 -0
  57. package/dist/Construct-BNDLJJfD.mjs +150 -0
  58. package/dist/Construct-BNDLJJfD.mjs.map +1 -0
  59. package/dist/{Construct-Ba5cMxib.cjs → Construct-CPrCF8NK.cjs} +93 -9
  60. package/dist/Construct-CPrCF8NK.cjs.map +1 -0
  61. package/dist/{Construct-C4rPE67v.d.cts → Construct-Dl0l2d8d.d.cts} +30 -4
  62. package/dist/Construct-Dl0l2d8d.d.cts.map +1 -0
  63. package/dist/{Construct-XrijZFFh.d.mts → Construct-E8QPyHh4.d.mts} +30 -4
  64. package/dist/Construct-E8QPyHh4.d.mts.map +1 -0
  65. package/dist/Construct.cjs +4 -2
  66. package/dist/Construct.d.cts +2 -2
  67. package/dist/Construct.d.mts +2 -2
  68. package/dist/Construct.mjs +2 -2
  69. package/dist/{Cron-CGF4YAfM.cjs → Cron-BmPNTLla.cjs} +3 -3
  70. package/dist/Cron-BmPNTLla.cjs.map +1 -0
  71. package/dist/{Cron-BxhGs5up.mjs → Cron-BpJONaFv.mjs} +3 -3
  72. package/dist/Cron-BpJONaFv.mjs.map +1 -0
  73. package/dist/{Cron-bDLcTvV5.d.cts → Cron-CnmLLh3E.d.mts} +3 -3
  74. package/dist/Cron-CnmLLh3E.d.mts.map +1 -0
  75. package/dist/{Cron-cdjlSKNp.d.mts → Cron-DufAeHry.d.cts} +3 -3
  76. package/dist/Cron-DufAeHry.d.cts.map +1 -0
  77. package/dist/{CronBuilder-d2jh-IB2.mjs → CronBuilder-02HSIHJr.mjs} +4 -4
  78. package/dist/CronBuilder-02HSIHJr.mjs.map +1 -0
  79. package/dist/{CronBuilder-CcxKRtVP.cjs → CronBuilder-DC2zUS8a.cjs} +4 -4
  80. package/dist/CronBuilder-DC2zUS8a.cjs.map +1 -0
  81. package/dist/{CronBuilder-BC4m5-p1.d.mts → CronBuilder-anUSbMxb.d.mts} +4 -4
  82. package/dist/CronBuilder-anUSbMxb.d.mts.map +1 -0
  83. package/dist/{CronBuilder-DKVXyE0Q.d.cts → CronBuilder-fyqf_X8n.d.cts} +4 -4
  84. package/dist/CronBuilder-fyqf_X8n.d.cts.map +1 -0
  85. package/dist/{Endpoint-BVGZXFyV.cjs → Endpoint-BCWVbi8l.cjs} +6 -6
  86. package/dist/Endpoint-BCWVbi8l.cjs.map +1 -0
  87. package/dist/{Endpoint-BPv9_-m_.d.cts → Endpoint-BlcHiGNx.d.cts} +3 -3
  88. package/dist/Endpoint-BlcHiGNx.d.cts.map +1 -0
  89. package/dist/{Endpoint-BdwG75G_.d.mts → Endpoint-DEHuy46S.d.mts} +4 -4
  90. package/dist/Endpoint-DEHuy46S.d.mts.map +1 -0
  91. package/dist/{Endpoint-CuOEswxJ.mjs → Endpoint-ygxv2Ia6.mjs} +6 -6
  92. package/dist/Endpoint-ygxv2Ia6.mjs.map +1 -0
  93. package/dist/{EndpointBuilder-DIy_m1bu.d.cts → EndpointBuilder-DPLAUHqg.d.cts} +4 -4
  94. package/dist/EndpointBuilder-DPLAUHqg.d.cts.map +1 -0
  95. package/dist/{EndpointBuilder-B3az942t.d.mts → EndpointBuilder-DaeCeneF.d.mts} +5 -5
  96. package/dist/EndpointBuilder-DaeCeneF.d.mts.map +1 -0
  97. package/dist/{EndpointBuilder-Cgj1P_ra.cjs → EndpointBuilder-J5D67Y6a.cjs} +6 -6
  98. package/dist/EndpointBuilder-J5D67Y6a.cjs.map +1 -0
  99. package/dist/{EndpointBuilder-DnCB1h1j.mjs → EndpointBuilder-tn4zqoyw.mjs} +6 -6
  100. package/dist/EndpointBuilder-tn4zqoyw.mjs.map +1 -0
  101. package/dist/{EndpointFactory-CbdxPCIH.mjs → EndpointFactory-CIUZTTAF.mjs} +7 -7
  102. package/dist/EndpointFactory-CIUZTTAF.mjs.map +1 -0
  103. package/dist/{EndpointFactory-CyPbm3AD.d.cts → EndpointFactory-CiKcb5PX.d.cts} +4 -4
  104. package/dist/EndpointFactory-CiKcb5PX.d.cts.map +1 -0
  105. package/dist/{EndpointFactory-C-0nE6Jg.d.mts → EndpointFactory-DufM0t2z.d.mts} +4 -4
  106. package/dist/EndpointFactory-DufM0t2z.d.mts.map +1 -0
  107. package/dist/{EndpointFactory-CYj6BYok.cjs → EndpointFactory-d5uwWFkK.cjs} +7 -7
  108. package/dist/EndpointFactory-d5uwWFkK.cjs.map +1 -0
  109. package/dist/{Function-DDZb1525.cjs → Function-2qWSZqYB.cjs} +3 -3
  110. package/dist/Function-2qWSZqYB.cjs.map +1 -0
  111. package/dist/{Function-Cf7f_kCz.d.cts → Function-CbEohg13.d.cts} +3 -3
  112. package/dist/Function-CbEohg13.d.cts.map +1 -0
  113. package/dist/{Function-BVHqIDp9.mjs → Function-CdF1HmFu.mjs} +3 -3
  114. package/dist/Function-CdF1HmFu.mjs.map +1 -0
  115. package/dist/{Function-DN2G6OT5.d.mts → Function-EjIOLxlh.d.mts} +3 -3
  116. package/dist/Function-EjIOLxlh.d.mts.map +1 -0
  117. package/dist/{FunctionBuilder-DswJ-9sD.cjs → FunctionBuilder-B-pbgm09.cjs} +4 -4
  118. package/dist/FunctionBuilder-B-pbgm09.cjs.map +1 -0
  119. package/dist/{FunctionBuilder-CJBzzXL3.d.cts → FunctionBuilder-CX3ooNdq.d.cts} +4 -4
  120. package/dist/FunctionBuilder-CX3ooNdq.d.cts.map +1 -0
  121. package/dist/{FunctionBuilder-Cxx8D2na.d.mts → FunctionBuilder-D11ytDyy.d.mts} +4 -4
  122. package/dist/FunctionBuilder-D11ytDyy.d.mts.map +1 -0
  123. package/dist/{FunctionBuilder-CrDYgfiI.mjs → FunctionBuilder-DaWVthAJ.mjs} +4 -4
  124. package/dist/FunctionBuilder-DaWVthAJ.mjs.map +1 -0
  125. package/dist/{FunctionExecutionWrapper-DF260Aaj.d.mts → FunctionExecutionWrapper-BJcRjH9Z.d.cts} +5 -5
  126. package/dist/FunctionExecutionWrapper-BJcRjH9Z.d.cts.map +1 -0
  127. package/dist/{FunctionExecutionWrapper-BYI2bGTL.cjs → FunctionExecutionWrapper-C6ChBNHs.cjs} +3 -3
  128. package/dist/FunctionExecutionWrapper-C6ChBNHs.cjs.map +1 -0
  129. package/dist/{FunctionExecutionWrapper-Qy8bmgFR.d.cts → FunctionExecutionWrapper-D3RNjGIR.d.mts} +5 -5
  130. package/dist/FunctionExecutionWrapper-D3RNjGIR.d.mts.map +1 -0
  131. package/dist/{FunctionExecutionWrapper-CLDh7Z2_.mjs → FunctionExecutionWrapper-Dj3pmXaN.mjs} +3 -3
  132. package/dist/FunctionExecutionWrapper-Dj3pmXaN.mjs.map +1 -0
  133. package/dist/{HonoEndpointAdaptor-BaPlUhz0.d.mts → HonoEndpointAdaptor-CKP6nrG2.d.mts} +11 -6
  134. package/dist/HonoEndpointAdaptor-CKP6nrG2.d.mts.map +1 -0
  135. package/dist/HonoEndpointAdaptor-Cc8Rnp9G.mjs +251 -0
  136. package/dist/HonoEndpointAdaptor-Cc8Rnp9G.mjs.map +1 -0
  137. package/dist/{HonoEndpointAdaptor-YcRHYALH.d.cts → HonoEndpointAdaptor-hrrvyfw9.d.cts} +10 -5
  138. package/dist/HonoEndpointAdaptor-hrrvyfw9.d.cts.map +1 -0
  139. package/dist/HonoEndpointAdaptor-xGnDZa3a.cjs +257 -0
  140. package/dist/HonoEndpointAdaptor-xGnDZa3a.cjs.map +1 -0
  141. package/dist/{Subscriber-COYMSevD.d.cts → Subscriber-BBcIOkhW.d.cts} +2 -2
  142. package/dist/Subscriber-BBcIOkhW.d.cts.map +1 -0
  143. package/dist/{Subscriber-ikctpU3I.d.mts → Subscriber-BpTAXvbM.d.mts} +2 -2
  144. package/dist/Subscriber-BpTAXvbM.d.mts.map +1 -0
  145. package/dist/{Subscriber-BiHjVXtM.cjs → Subscriber-CL4iYm01.cjs} +2 -2
  146. package/dist/Subscriber-CL4iYm01.cjs.map +1 -0
  147. package/dist/{Subscriber-BmPf9GFb.mjs → Subscriber-CZ8Smwd2.mjs} +2 -2
  148. package/dist/Subscriber-CZ8Smwd2.mjs.map +1 -0
  149. package/dist/{SubscriberBuilder-DJPEeYDJ.mjs → SubscriberBuilder-CxQg3TTm.mjs} +2 -2
  150. package/dist/SubscriberBuilder-CxQg3TTm.mjs.map +1 -0
  151. package/dist/{SubscriberBuilder-Cp1C-xtT.cjs → SubscriberBuilder-D0OS3hd7.cjs} +2 -2
  152. package/dist/SubscriberBuilder-D0OS3hd7.cjs.map +1 -0
  153. package/dist/{SubscriberBuilder-D_9zzllj.d.mts → SubscriberBuilder-D7IgufwB.d.cts} +3 -3
  154. package/dist/SubscriberBuilder-D7IgufwB.d.cts.map +1 -0
  155. package/dist/{SubscriberBuilder-ivHAGIVi.d.cts → SubscriberBuilder-DOFBbWLt.d.mts} +3 -3
  156. package/dist/SubscriberBuilder-DOFBbWLt.d.mts.map +1 -0
  157. package/dist/{TestEndpointAdaptor-DB7bREhS.d.mts → TestEndpointAdaptor-BnJusjf7.d.mts} +4 -4
  158. package/dist/TestEndpointAdaptor-BnJusjf7.d.mts.map +1 -0
  159. package/dist/TestEndpointAdaptor-CPL8ru6q.mjs +170 -0
  160. package/dist/TestEndpointAdaptor-CPL8ru6q.mjs.map +1 -0
  161. package/dist/TestEndpointAdaptor-Ca4K26_d.cjs +176 -0
  162. package/dist/TestEndpointAdaptor-Ca4K26_d.cjs.map +1 -0
  163. package/dist/{TestEndpointAdaptor-C10xBI--.d.cts → TestEndpointAdaptor-EPPTjntv.d.cts} +3 -3
  164. package/dist/TestEndpointAdaptor-EPPTjntv.d.cts.map +1 -0
  165. package/dist/adaptors/aws.cjs +12 -12
  166. package/dist/adaptors/aws.d.cts +15 -14
  167. package/dist/adaptors/aws.d.mts +16 -15
  168. package/dist/adaptors/aws.mjs +12 -12
  169. package/dist/adaptors/hono.cjs +10 -9
  170. package/dist/adaptors/hono.d.cts +8 -8
  171. package/dist/adaptors/hono.d.mts +9 -9
  172. package/dist/adaptors/hono.mjs +10 -9
  173. package/dist/adaptors/testing.cjs +8 -7
  174. package/dist/adaptors/testing.d.cts +8 -8
  175. package/dist/adaptors/testing.d.mts +9 -9
  176. package/dist/adaptors/testing.mjs +8 -7
  177. package/dist/crons/Cron.cjs +6 -6
  178. package/dist/crons/Cron.d.cts +6 -6
  179. package/dist/crons/Cron.d.mts +6 -6
  180. package/dist/crons/Cron.mjs +6 -6
  181. package/dist/crons/CronBuilder.cjs +7 -7
  182. package/dist/crons/CronBuilder.d.cts +7 -7
  183. package/dist/crons/CronBuilder.d.mts +7 -7
  184. package/dist/crons/CronBuilder.mjs +7 -7
  185. package/dist/crons/index.cjs +7 -7
  186. package/dist/crons/index.cjs.map +1 -1
  187. package/dist/crons/index.d.cts +11 -11
  188. package/dist/crons/index.d.mts +7 -7
  189. package/dist/crons/index.mjs +7 -7
  190. package/dist/crons/index.mjs.map +1 -1
  191. package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.cjs +7 -7
  192. package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.d.cts +10 -9
  193. package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.d.mts +11 -10
  194. package/dist/endpoints/AmazonApiGatewayEndpointAdaptor.mjs +7 -7
  195. package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.cjs +8 -8
  196. package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.d.cts +10 -9
  197. package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.d.mts +11 -10
  198. package/dist/endpoints/AmazonApiGatewayV1EndpointAdaptor.mjs +8 -8
  199. package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.cjs +8 -8
  200. package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.d.cts +11 -10
  201. package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.d.mts +12 -11
  202. package/dist/endpoints/AmazonApiGatewayV2EndpointAdaptor.mjs +8 -8
  203. package/dist/endpoints/Authorizer.cjs +1 -1
  204. package/dist/endpoints/Authorizer.d.cts +1 -1
  205. package/dist/endpoints/Authorizer.d.mts +1 -1
  206. package/dist/endpoints/Authorizer.mjs +1 -1
  207. package/dist/endpoints/Endpoint.cjs +6 -6
  208. package/dist/endpoints/Endpoint.d.cts +7 -7
  209. package/dist/endpoints/Endpoint.d.mts +8 -8
  210. package/dist/endpoints/Endpoint.mjs +6 -6
  211. package/dist/endpoints/EndpointBuilder.cjs +9 -9
  212. package/dist/endpoints/EndpointBuilder.d.cts +8 -8
  213. package/dist/endpoints/EndpointBuilder.d.mts +9 -9
  214. package/dist/endpoints/EndpointBuilder.mjs +9 -9
  215. package/dist/endpoints/EndpointFactory.cjs +10 -10
  216. package/dist/endpoints/EndpointFactory.d.cts +9 -9
  217. package/dist/endpoints/EndpointFactory.d.mts +10 -10
  218. package/dist/endpoints/EndpointFactory.mjs +10 -10
  219. package/dist/endpoints/HonoEndpointAdaptor.cjs +10 -9
  220. package/dist/endpoints/HonoEndpointAdaptor.d.cts +8 -8
  221. package/dist/endpoints/HonoEndpointAdaptor.d.mts +9 -9
  222. package/dist/endpoints/HonoEndpointAdaptor.mjs +10 -9
  223. package/dist/endpoints/TestEndpointAdaptor.cjs +8 -7
  224. package/dist/endpoints/TestEndpointAdaptor.d.cts +8 -8
  225. package/dist/endpoints/TestEndpointAdaptor.d.mts +9 -9
  226. package/dist/endpoints/TestEndpointAdaptor.mjs +8 -7
  227. package/dist/endpoints/audit.d.cts +7 -7
  228. package/dist/endpoints/audit.d.mts +8 -8
  229. package/dist/endpoints/helpers.cjs +7 -7
  230. package/dist/endpoints/helpers.d.cts +7 -7
  231. package/dist/endpoints/helpers.d.cts.map +1 -1
  232. package/dist/endpoints/helpers.d.mts +8 -8
  233. package/dist/endpoints/helpers.d.mts.map +1 -1
  234. package/dist/endpoints/helpers.mjs +7 -7
  235. package/dist/endpoints/index.cjs +20 -10
  236. package/dist/endpoints/index.cjs.map +1 -1
  237. package/dist/endpoints/index.d.cts +14 -12
  238. package/dist/endpoints/index.d.cts.map +1 -1
  239. package/dist/endpoints/index.d.mts +15 -13
  240. package/dist/endpoints/index.d.mts.map +1 -1
  241. package/dist/endpoints/index.mjs +13 -11
  242. package/dist/endpoints/index.mjs.map +1 -1
  243. package/dist/endpoints/lazyAccessors.cjs +10 -0
  244. package/dist/endpoints/lazyAccessors.d.cts +10 -0
  245. package/dist/endpoints/lazyAccessors.d.mts +10 -0
  246. package/dist/endpoints/lazyAccessors.mjs +3 -0
  247. package/dist/endpoints/parseHonoQuery.cjs +1 -1
  248. package/dist/endpoints/parseHonoQuery.mjs +1 -1
  249. package/dist/endpoints/parseQueryParams.d.cts.map +1 -1
  250. package/dist/endpoints/parseQueryParams.d.mts.map +1 -1
  251. package/dist/endpoints/processAudits.d.cts +7 -7
  252. package/dist/endpoints/processAudits.d.cts.map +1 -1
  253. package/dist/endpoints/processAudits.d.mts +8 -8
  254. package/dist/endpoints/processAudits.d.mts.map +1 -1
  255. package/dist/endpoints/rls.cjs +1 -1
  256. package/dist/endpoints/rls.d.cts +7 -7
  257. package/dist/endpoints/rls.d.mts +8 -8
  258. package/dist/endpoints/rls.mjs +1 -1
  259. package/dist/functions/AWSLambdaFunction.cjs +6 -6
  260. package/dist/functions/AWSLambdaFunction.d.cts +4 -4
  261. package/dist/functions/AWSLambdaFunction.d.mts +4 -4
  262. package/dist/functions/AWSLambdaFunction.mjs +6 -6
  263. package/dist/functions/BaseFunctionBuilder.cjs +2 -2
  264. package/dist/functions/BaseFunctionBuilder.d.cts +2 -2
  265. package/dist/functions/BaseFunctionBuilder.d.mts +2 -2
  266. package/dist/functions/BaseFunctionBuilder.mjs +2 -2
  267. package/dist/functions/Function.cjs +2 -2
  268. package/dist/functions/Function.d.cts +2 -2
  269. package/dist/functions/Function.d.mts +2 -2
  270. package/dist/functions/Function.mjs +2 -2
  271. package/dist/functions/FunctionBuilder.cjs +4 -4
  272. package/dist/functions/FunctionBuilder.d.cts +4 -4
  273. package/dist/functions/FunctionBuilder.d.mts +4 -4
  274. package/dist/functions/FunctionBuilder.mjs +4 -4
  275. package/dist/functions/FunctionExecutionWrapper.cjs +5 -5
  276. package/dist/functions/FunctionExecutionWrapper.d.cts +3 -3
  277. package/dist/functions/FunctionExecutionWrapper.d.mts +3 -3
  278. package/dist/functions/FunctionExecutionWrapper.mjs +5 -5
  279. package/dist/functions/TestFunctionAdaptor.cjs +7 -7
  280. package/dist/functions/TestFunctionAdaptor.cjs.map +1 -1
  281. package/dist/functions/TestFunctionAdaptor.d.cts +3 -3
  282. package/dist/functions/TestFunctionAdaptor.d.cts.map +1 -1
  283. package/dist/functions/TestFunctionAdaptor.d.mts +3 -3
  284. package/dist/functions/TestFunctionAdaptor.d.mts.map +1 -1
  285. package/dist/functions/TestFunctionAdaptor.mjs +7 -7
  286. package/dist/functions/TestFunctionAdaptor.mjs.map +1 -1
  287. package/dist/functions/index.cjs +5 -5
  288. package/dist/functions/index.d.cts +5 -5
  289. package/dist/functions/index.d.mts +5 -5
  290. package/dist/functions/index.mjs +5 -5
  291. package/dist/{functions-fTid0RMK.cjs → functions-CUEv5NC3.cjs} +2 -2
  292. package/dist/functions-CUEv5NC3.cjs.map +1 -0
  293. package/dist/functions-DOYBrb7n.mjs +8 -0
  294. package/dist/functions-DOYBrb7n.mjs.map +1 -0
  295. package/dist/{helpers-ByRTDO_m.mjs → helpers-Dl1eszfi.mjs} +2 -2
  296. package/dist/helpers-Dl1eszfi.mjs.map +1 -0
  297. package/dist/{helpers-BcP1tXAi.cjs → helpers-QMdfdnvU.cjs} +2 -2
  298. package/dist/helpers-QMdfdnvU.cjs.map +1 -0
  299. package/dist/{index-BWzGIj06.d.mts → index-BrGtMlpJ.d.cts} +2 -2
  300. package/dist/index-BrGtMlpJ.d.cts.map +1 -0
  301. package/dist/index-Dn4py3Db.d.mts +12 -0
  302. package/dist/index-Dn4py3Db.d.mts.map +1 -0
  303. package/dist/index.cjs +4 -2
  304. package/dist/index.d.cts +3 -2
  305. package/dist/index.d.mts +4 -3
  306. package/dist/index.mjs +2 -2
  307. package/dist/lazyAccessors-B-Jgkg2o.mjs +175 -0
  308. package/dist/lazyAccessors-B-Jgkg2o.mjs.map +1 -0
  309. package/dist/lazyAccessors-B8Hhras9.cjs +223 -0
  310. package/dist/lazyAccessors-B8Hhras9.cjs.map +1 -0
  311. package/dist/lazyAccessors-BFAj2-S4.d.mts +49 -0
  312. package/dist/lazyAccessors-BFAj2-S4.d.mts.map +1 -0
  313. package/dist/lazyAccessors-knaNZuTN.d.cts +49 -0
  314. package/dist/lazyAccessors-knaNZuTN.d.cts.map +1 -0
  315. package/dist/{parseHonoQuery-CZC5_97v.cjs → parseHonoQuery-D4MhxTRc.cjs} +1 -1
  316. package/dist/parseHonoQuery-D4MhxTRc.cjs.map +1 -0
  317. package/dist/{parseHonoQuery-DDgIkTO4.mjs → parseHonoQuery-DpK3sGPc.mjs} +1 -1
  318. package/dist/parseHonoQuery-DpK3sGPc.mjs.map +1 -0
  319. package/dist/parseQueryParams-BSNkjmZ9.cjs.map +1 -1
  320. package/dist/parseQueryParams-UMTRnRrW.mjs.map +1 -1
  321. package/dist/processAudits-CzHkPokQ.cjs.map +1 -1
  322. package/dist/processAudits-Dj8UGqcW.mjs.map +1 -1
  323. package/dist/publisher-BXG9YiRi.d.mts +16 -0
  324. package/dist/publisher-BXG9YiRi.d.mts.map +1 -0
  325. package/dist/publisher-Bw4770Hi.mjs.map +1 -1
  326. package/dist/publisher-D9ngDXg3.d.cts +16 -0
  327. package/dist/publisher-D9ngDXg3.d.cts.map +1 -0
  328. package/dist/publisher-lFQleddL.cjs.map +1 -1
  329. package/dist/publisher.d.cts +3 -16
  330. package/dist/publisher.d.mts +3 -16
  331. package/dist/{rls-CmJ7bRsz.cjs → rls-BrywnrQb.cjs} +1 -1
  332. package/dist/{rls-CmJ7bRsz.cjs.map → rls-BrywnrQb.cjs.map} +1 -1
  333. package/dist/{rls-Bf3FRwto.mjs → rls-C0cWOnk4.mjs} +1 -1
  334. package/dist/{rls-Bf3FRwto.mjs.map → rls-C0cWOnk4.mjs.map} +1 -1
  335. package/dist/subscribers/AWSLambdaSubscriberAdaptor.cjs +1 -1
  336. package/dist/subscribers/AWSLambdaSubscriberAdaptor.d.cts +3 -3
  337. package/dist/subscribers/AWSLambdaSubscriberAdaptor.d.mts +3 -3
  338. package/dist/subscribers/AWSLambdaSubscriberAdaptor.mjs +1 -1
  339. package/dist/subscribers/Subscriber.cjs +2 -2
  340. package/dist/subscribers/Subscriber.d.cts +2 -2
  341. package/dist/subscribers/Subscriber.d.mts +2 -2
  342. package/dist/subscribers/Subscriber.mjs +2 -2
  343. package/dist/subscribers/SubscriberBuilder.cjs +3 -3
  344. package/dist/subscribers/SubscriberBuilder.d.cts +3 -3
  345. package/dist/subscribers/SubscriberBuilder.d.mts +3 -3
  346. package/dist/subscribers/SubscriberBuilder.mjs +3 -3
  347. package/dist/subscribers/index.cjs +3 -3
  348. package/dist/subscribers/index.cjs.map +1 -1
  349. package/dist/subscribers/index.d.cts +5 -5
  350. package/dist/subscribers/index.d.cts.map +1 -1
  351. package/dist/subscribers/index.d.mts +5 -5
  352. package/dist/subscribers/index.d.mts.map +1 -1
  353. package/dist/subscribers/index.mjs +3 -3
  354. package/dist/subscribers/index.mjs.map +1 -1
  355. package/dist/telemetry-BTaiRqPo.d.cts +95 -0
  356. package/dist/telemetry-BTaiRqPo.d.cts.map +1 -0
  357. package/dist/telemetry-yAHf5yDs.d.mts +95 -0
  358. package/dist/telemetry-yAHf5yDs.d.mts.map +1 -0
  359. package/dist/telemetry.cjs +0 -0
  360. package/dist/telemetry.d.cts +2 -0
  361. package/dist/telemetry.d.mts +2 -0
  362. package/dist/telemetry.mjs +0 -0
  363. package/dist/types-B5H3piDg.d.cts.map +1 -1
  364. package/dist/{types-DdIlpxAd.d.mts → types-Dw-iLd3Y.d.mts} +1 -1
  365. package/dist/types-Dw-iLd3Y.d.mts.map +1 -0
  366. package/dist/types.d.mts +1 -1
  367. package/package.json +19 -18
  368. package/src/Construct.ts +189 -94
  369. package/src/__benchmarks__/build-time-optimization.bench.ts +274 -0
  370. package/src/__benchmarks__/endpoint.bench.ts +375 -375
  371. package/src/__benchmarks__/fixtures.ts +241 -0
  372. package/src/__benchmarks__/hono-adaptor.bench.ts +533 -0
  373. package/src/__benchmarks__/hono-server.bench.ts +206 -206
  374. package/src/__benchmarks__/response-builder.bench.ts +428 -0
  375. package/src/__benchmarks__/strategies/strategy-a-lazy-services.ts +319 -0
  376. package/src/__benchmarks__/strategies/strategy-c-middleware.ts +530 -0
  377. package/src/__benchmarks__/strategies/strategy-d-opt-in-events.ts +567 -0
  378. package/src/__tests__/Construct.environment.spec.ts +572 -351
  379. package/src/__tests__/publisher.setting.spec.ts +496 -499
  380. package/src/__tests__/publisher.spec.ts +440 -442
  381. package/src/crons/Cron.ts +117 -117
  382. package/src/crons/CronBuilder.ts +237 -238
  383. package/src/crons/__tests__/Cron.spec.ts +448 -448
  384. package/src/crons/__tests__/CronBuilder.state-isolation.spec.ts +214 -214
  385. package/src/crons/index.ts +4 -4
  386. package/src/endpoints/AmazonApiGatewayEndpointAdaptor.ts +571 -463
  387. package/src/endpoints/AmazonApiGatewayV1EndpointAdaptor.ts +86 -86
  388. package/src/endpoints/AmazonApiGatewayV2EndpointAdaptor.ts +85 -81
  389. package/src/endpoints/Authorizer.ts +100 -100
  390. package/src/endpoints/Endpoint.ts +921 -937
  391. package/src/endpoints/EndpointBuilder.ts +703 -703
  392. package/src/endpoints/EndpointFactory.ts +1054 -1056
  393. package/src/endpoints/HonoEndpointAdaptor.ts +621 -548
  394. package/src/endpoints/TestEndpointAdaptor.ts +372 -353
  395. package/src/endpoints/__tests__/AmazonApiGatewayV1EndpointAdaptor.audits.spec.ts +499 -0
  396. package/src/endpoints/__tests__/AmazonApiGatewayV1EndpointAdaptor.events.spec.ts +541 -542
  397. package/src/endpoints/__tests__/AmazonApiGatewayV1EndpointAdaptor.spec.ts +1172 -1174
  398. package/src/endpoints/__tests__/AmazonApiGatewayV2EndpointAdaptor.audits.spec.ts +599 -599
  399. package/src/endpoints/__tests__/AmazonApiGatewayV2EndpointAdaptor.events.spec.ts +710 -710
  400. package/src/endpoints/__tests__/AmazonApiGatewayV2EndpointAdaptor.factory-publisher.spec.ts +280 -280
  401. package/src/endpoints/__tests__/AmazonApiGatewayV2EndpointAdaptor.kysely-audit.integration.spec.ts +924 -730
  402. package/src/endpoints/__tests__/AmazonApiGatewayV2EndpointAdaptor.rls.spec.ts +307 -0
  403. package/src/endpoints/__tests__/AmazonApiGatewayV2EndpointAdaptor.spec.ts +408 -408
  404. package/src/endpoints/__tests__/Authorizer.spec.ts +286 -0
  405. package/src/endpoints/__tests__/Endpoint.cookies.spec.ts +162 -162
  406. package/src/endpoints/__tests__/Endpoint.headers.spec.ts +182 -182
  407. package/src/endpoints/__tests__/Endpoint.manifest.spec.ts +158 -159
  408. package/src/endpoints/__tests__/Endpoint.spec.ts +819 -822
  409. package/src/endpoints/__tests__/EndpointBuilder.audit.spec.ts +427 -427
  410. package/src/endpoints/__tests__/EndpointBuilder.spec.ts +478 -478
  411. package/src/endpoints/__tests__/EndpointFactory.authorizers.spec.ts +426 -426
  412. package/src/endpoints/__tests__/EndpointFactory.database-auditor.spec.ts +501 -501
  413. package/src/endpoints/__tests__/EndpointFactory.reference-audit.spec.ts +177 -177
  414. package/src/endpoints/__tests__/EndpointFactory.spec.ts +471 -471
  415. package/src/endpoints/__tests__/EndpointFactory.state-isolation.spec.ts +143 -143
  416. package/src/endpoints/__tests__/HonoEndpointAdaptor.audit-transactions.spec.ts +663 -663
  417. package/src/endpoints/__tests__/HonoEndpointAdaptor.audits.spec.ts +608 -608
  418. package/src/endpoints/__tests__/HonoEndpointAdaptor.events.spec.ts +561 -561
  419. package/src/endpoints/__tests__/HonoEndpointAdaptor.kysely-audit.integration.spec.ts +1036 -1036
  420. package/src/endpoints/__tests__/HonoEndpointAdaptor.openapi.spec.ts +278 -278
  421. package/src/endpoints/__tests__/HonoEndpointAdaptor.spec.ts +1093 -1093
  422. package/src/endpoints/__tests__/ResponseBuilder.spec.ts +230 -230
  423. package/src/endpoints/__tests__/TestEndpointAdaptor.audits.spec.ts +569 -569
  424. package/src/endpoints/__tests__/TestEndpointAdaptor.spec.ts +841 -841
  425. package/src/endpoints/__tests__/endpoint-types.test.ts +68 -68
  426. package/src/endpoints/__tests__/helpers.spec.ts +37 -0
  427. package/src/endpoints/__tests__/lazyAccessors.spec.ts +330 -0
  428. package/src/endpoints/__tests__/processAudits.spec.ts +631 -0
  429. package/src/endpoints/audit.ts +51 -51
  430. package/src/endpoints/helpers.ts +34 -35
  431. package/src/endpoints/index.ts +33 -23
  432. package/src/endpoints/lazyAccessors.ts +241 -0
  433. package/src/endpoints/parseHonoQuery.ts +41 -41
  434. package/src/endpoints/parseQueryParams.ts +36 -36
  435. package/src/endpoints/processAudits.ts +269 -269
  436. package/src/endpoints/rls.ts +16 -16
  437. package/src/functions/AWSLambdaFunction.ts +294 -295
  438. package/src/functions/BaseFunctionBuilder.ts +185 -185
  439. package/src/functions/Function.ts +190 -190
  440. package/src/functions/FunctionBuilder.ts +353 -353
  441. package/src/functions/FunctionExecutionWrapper.ts +199 -200
  442. package/src/functions/TestFunctionAdaptor.ts +191 -191
  443. package/src/functions/__tests__/AWSLambdaFunctionAdaptor.spec.ts +424 -424
  444. package/src/functions/__tests__/Function.audits.spec.ts +364 -364
  445. package/src/functions/__tests__/Function.spec.ts +453 -454
  446. package/src/functions/__tests__/FunctionBuilder.state-isolation.spec.ts +179 -179
  447. package/src/functions/__tests__/TestFunctionAdaptor.spec.ts +369 -369
  448. package/src/functions/index.ts +6 -4
  449. package/src/index.ts +18 -9
  450. package/src/publisher.ts +71 -72
  451. package/src/subscribers/AWSLambdaSubscriberAdaptor.ts +244 -248
  452. package/src/subscribers/Subscriber.ts +82 -84
  453. package/src/subscribers/SubscriberBuilder.ts +129 -129
  454. package/src/subscribers/__tests__/AWSLambdaSubscriberAdaptor.spec.ts +589 -589
  455. package/src/subscribers/__tests__/Subscriber.spec.ts +402 -404
  456. package/src/subscribers/index.ts +1 -0
  457. package/src/telemetry.ts +103 -0
  458. package/src/types.ts +7 -7
  459. package/test.ts +58 -59
  460. package/dist/AWSLambdaFunction-BdebRMUh.d.mts.map +0 -1
  461. package/dist/AWSLambdaFunction-C54a1doJ.mjs.map +0 -1
  462. package/dist/AWSLambdaFunction-D0tEOTXQ.d.cts.map +0 -1
  463. package/dist/AWSLambdaFunction-EPGY4s7i.cjs.map +0 -1
  464. package/dist/AWSLambdaSubscriberAdaptor-BLfO612H.mjs.map +0 -1
  465. package/dist/AWSLambdaSubscriberAdaptor-BNcYYZ-P.cjs.map +0 -1
  466. package/dist/AWSLambdaSubscriberAdaptor-DrFAvHOp.d.mts.map +0 -1
  467. package/dist/AWSLambdaSubscriberAdaptor-Dx-Ks1Jp.d.cts.map +0 -1
  468. package/dist/AmazonApiGatewayEndpointAdaptor-BT9JXihC.mjs.map +0 -1
  469. package/dist/AmazonApiGatewayEndpointAdaptor-DLqnYQ4E.d.cts.map +0 -1
  470. package/dist/AmazonApiGatewayEndpointAdaptor-DNFvvdmW.cjs.map +0 -1
  471. package/dist/AmazonApiGatewayEndpointAdaptor-qlKXuZxy.d.mts.map +0 -1
  472. package/dist/AmazonApiGatewayV1EndpointAdaptor-CShQI8Gk.mjs.map +0 -1
  473. package/dist/AmazonApiGatewayV1EndpointAdaptor-Ccl8B8kG.cjs.map +0 -1
  474. package/dist/AmazonApiGatewayV1EndpointAdaptor-DyUaJIhB.d.cts.map +0 -1
  475. package/dist/AmazonApiGatewayV1EndpointAdaptor-MRUxs3Xi.d.mts.map +0 -1
  476. package/dist/AmazonApiGatewayV2EndpointAdaptor-D8-0Aab4.cjs.map +0 -1
  477. package/dist/AmazonApiGatewayV2EndpointAdaptor-M1-w0U5R.d.cts.map +0 -1
  478. package/dist/AmazonApiGatewayV2EndpointAdaptor-fuLM6M9k.d.mts.map +0 -1
  479. package/dist/AmazonApiGatewayV2EndpointAdaptor-lCRzGE4q.mjs.map +0 -1
  480. package/dist/Authorizer-B-btowNd.d.cts.map +0 -1
  481. package/dist/Authorizer-C0ge_tc8.cjs.map +0 -1
  482. package/dist/Authorizer-gWxYsGEp.d.mts.map +0 -1
  483. package/dist/Authorizer-r9U3y_ms.mjs.map +0 -1
  484. package/dist/BaseFunctionBuilder-BAtutR6q.d.cts.map +0 -1
  485. package/dist/BaseFunctionBuilder-Czi1Jwza.mjs.map +0 -1
  486. package/dist/BaseFunctionBuilder-DRY419e7.d.mts.map +0 -1
  487. package/dist/BaseFunctionBuilder-MYG3C9ug.cjs.map +0 -1
  488. package/dist/Construct-Ba5cMxib.cjs.map +0 -1
  489. package/dist/Construct-C4rPE67v.d.cts.map +0 -1
  490. package/dist/Construct-DdyGHuag.mjs +0 -78
  491. package/dist/Construct-DdyGHuag.mjs.map +0 -1
  492. package/dist/Construct-XrijZFFh.d.mts.map +0 -1
  493. package/dist/Cron-BxhGs5up.mjs.map +0 -1
  494. package/dist/Cron-CGF4YAfM.cjs.map +0 -1
  495. package/dist/Cron-bDLcTvV5.d.cts.map +0 -1
  496. package/dist/Cron-cdjlSKNp.d.mts.map +0 -1
  497. package/dist/CronBuilder-BC4m5-p1.d.mts.map +0 -1
  498. package/dist/CronBuilder-CcxKRtVP.cjs.map +0 -1
  499. package/dist/CronBuilder-DKVXyE0Q.d.cts.map +0 -1
  500. package/dist/CronBuilder-d2jh-IB2.mjs.map +0 -1
  501. package/dist/Endpoint-BPv9_-m_.d.cts.map +0 -1
  502. package/dist/Endpoint-BVGZXFyV.cjs.map +0 -1
  503. package/dist/Endpoint-BdwG75G_.d.mts.map +0 -1
  504. package/dist/Endpoint-CuOEswxJ.mjs.map +0 -1
  505. package/dist/EndpointBuilder-B3az942t.d.mts.map +0 -1
  506. package/dist/EndpointBuilder-Cgj1P_ra.cjs.map +0 -1
  507. package/dist/EndpointBuilder-DIy_m1bu.d.cts.map +0 -1
  508. package/dist/EndpointBuilder-DnCB1h1j.mjs.map +0 -1
  509. package/dist/EndpointFactory-C-0nE6Jg.d.mts.map +0 -1
  510. package/dist/EndpointFactory-CYj6BYok.cjs.map +0 -1
  511. package/dist/EndpointFactory-CbdxPCIH.mjs.map +0 -1
  512. package/dist/EndpointFactory-CyPbm3AD.d.cts.map +0 -1
  513. package/dist/Function-BVHqIDp9.mjs.map +0 -1
  514. package/dist/Function-Cf7f_kCz.d.cts.map +0 -1
  515. package/dist/Function-DDZb1525.cjs.map +0 -1
  516. package/dist/Function-DN2G6OT5.d.mts.map +0 -1
  517. package/dist/FunctionBuilder-CJBzzXL3.d.cts.map +0 -1
  518. package/dist/FunctionBuilder-CrDYgfiI.mjs.map +0 -1
  519. package/dist/FunctionBuilder-Cxx8D2na.d.mts.map +0 -1
  520. package/dist/FunctionBuilder-DswJ-9sD.cjs.map +0 -1
  521. package/dist/FunctionExecutionWrapper-BYI2bGTL.cjs.map +0 -1
  522. package/dist/FunctionExecutionWrapper-CLDh7Z2_.mjs.map +0 -1
  523. package/dist/FunctionExecutionWrapper-DF260Aaj.d.mts.map +0 -1
  524. package/dist/FunctionExecutionWrapper-Qy8bmgFR.d.cts.map +0 -1
  525. package/dist/HonoEndpointAdaptor-BaPlUhz0.d.mts.map +0 -1
  526. package/dist/HonoEndpointAdaptor-CQe2FqMR.cjs +0 -234
  527. package/dist/HonoEndpointAdaptor-CQe2FqMR.cjs.map +0 -1
  528. package/dist/HonoEndpointAdaptor-Ce-2HBxn.mjs +0 -228
  529. package/dist/HonoEndpointAdaptor-Ce-2HBxn.mjs.map +0 -1
  530. package/dist/HonoEndpointAdaptor-YcRHYALH.d.cts.map +0 -1
  531. package/dist/Subscriber-BiHjVXtM.cjs.map +0 -1
  532. package/dist/Subscriber-BmPf9GFb.mjs.map +0 -1
  533. package/dist/Subscriber-COYMSevD.d.cts.map +0 -1
  534. package/dist/Subscriber-ikctpU3I.d.mts.map +0 -1
  535. package/dist/SubscriberBuilder-Cp1C-xtT.cjs.map +0 -1
  536. package/dist/SubscriberBuilder-DJPEeYDJ.mjs.map +0 -1
  537. package/dist/SubscriberBuilder-D_9zzllj.d.mts.map +0 -1
  538. package/dist/SubscriberBuilder-ivHAGIVi.d.cts.map +0 -1
  539. package/dist/TestEndpointAdaptor-C10xBI--.d.cts.map +0 -1
  540. package/dist/TestEndpointAdaptor-DB7bREhS.d.mts.map +0 -1
  541. package/dist/TestEndpointAdaptor-DJgik2Wj.mjs +0 -160
  542. package/dist/TestEndpointAdaptor-DJgik2Wj.mjs.map +0 -1
  543. package/dist/TestEndpointAdaptor-ZbtQiuHE.cjs +0 -166
  544. package/dist/TestEndpointAdaptor-ZbtQiuHE.cjs.map +0 -1
  545. package/dist/functions-C6EK1xL6.mjs +0 -8
  546. package/dist/functions-C6EK1xL6.mjs.map +0 -1
  547. package/dist/functions-fTid0RMK.cjs.map +0 -1
  548. package/dist/helpers-BcP1tXAi.cjs.map +0 -1
  549. package/dist/helpers-ByRTDO_m.mjs.map +0 -1
  550. package/dist/index-BWzGIj06.d.mts.map +0 -1
  551. package/dist/index-DmFozqLd.d.cts +0 -12
  552. package/dist/index-DmFozqLd.d.cts.map +0 -1
  553. package/dist/parseHonoQuery-CZC5_97v.cjs.map +0 -1
  554. package/dist/parseHonoQuery-DDgIkTO4.mjs.map +0 -1
  555. package/dist/publisher.d.cts.map +0 -1
  556. package/dist/publisher.d.mts.map +0 -1
  557. package/dist/types-DdIlpxAd.d.mts.map +0 -1
@@ -1,7 +1,7 @@
1
1
  import type { AuditableAction } from '@geekmidas/audit';
2
2
  import {
3
- type AuditLogTable,
4
- KyselyAuditStorage,
3
+ type AuditLogTable,
4
+ KyselyAuditStorage,
5
5
  } from '@geekmidas/audit/kysely';
6
6
  import { EnvironmentParser } from '@geekmidas/envkit';
7
7
  import type { Logger } from '@geekmidas/logger';
@@ -9,28 +9,28 @@ import type { Service } from '@geekmidas/services';
9
9
  import { ServiceDiscovery } from '@geekmidas/services';
10
10
  import { Hono } from 'hono';
11
11
  import {
12
- CamelCasePlugin,
13
- type Generated,
14
- Kysely,
15
- PostgresDialect,
16
- sql,
12
+ CamelCasePlugin,
13
+ type Generated,
14
+ Kysely,
15
+ PostgresDialect,
16
+ sql,
17
17
  } from 'kysely';
18
18
  import pg from 'pg';
19
19
  import {
20
- afterAll,
21
- afterEach,
22
- beforeAll,
23
- beforeEach,
24
- describe,
25
- expect,
26
- it,
27
- vi,
20
+ afterAll,
21
+ afterEach,
22
+ beforeAll,
23
+ beforeEach,
24
+ describe,
25
+ expect,
26
+ it,
27
+ vi,
28
28
  } from 'vitest';
29
29
  import { z } from 'zod';
30
30
  import { TEST_DATABASE_CONFIG } from '../../../../testkit/test/globalSetup';
31
+ import type { MappedAudit } from '../audit';
31
32
  import { Endpoint, type EndpointContext } from '../Endpoint';
32
33
  import { HonoEndpoint } from '../HonoEndpointAdaptor';
33
- import type { MappedAudit } from '../audit';
34
34
 
35
35
  // Use unique table names to avoid conflicts with parallel tests
36
36
  const AUDIT_TABLE = 'hono_audit_logs' as const;
@@ -38,1030 +38,1030 @@ const USERS_TABLE = 'hono_audit_users' as const;
38
38
 
39
39
  // Database schema
40
40
  interface TestDatabase {
41
- [AUDIT_TABLE]: AuditLogTable;
42
- [USERS_TABLE]: {
43
- id: Generated<number>;
44
- name: string;
45
- email: string;
46
- };
41
+ [AUDIT_TABLE]: AuditLogTable;
42
+ [USERS_TABLE]: {
43
+ id: Generated<number>;
44
+ name: string;
45
+ email: string;
46
+ };
47
47
  }
48
48
 
49
49
  // Audit action types
50
50
  type TestAuditAction =
51
- | AuditableAction<'user.created', { userId: number; email: string }>
52
- | AuditableAction<'user.updated', { userId: number; changes: string[] }>
53
- | AuditableAction<'user.deleted', { userId: number }>;
51
+ | AuditableAction<'user.created', { userId: number; email: string }>
52
+ | AuditableAction<'user.updated', { userId: number; changes: string[] }>
53
+ | AuditableAction<'user.deleted', { userId: number }>;
54
54
 
55
55
  describe('HonoEndpoint Kysely Audit Integration', () => {
56
- let db: Kysely<TestDatabase>;
57
- let auditStorage: KyselyAuditStorage<TestDatabase>;
58
- let mockLogger: Logger;
59
-
60
- const createMockLogger = (): Logger => ({
61
- debug: vi.fn(),
62
- info: vi.fn(),
63
- warn: vi.fn(),
64
- error: vi.fn(),
65
- fatal: vi.fn(),
66
- trace: vi.fn(),
67
- child: vi.fn(function (this: Logger) {
68
- return this;
69
- }),
70
- });
71
-
72
- const createServiceDiscovery = (logger: Logger) => {
73
- const envParser = new EnvironmentParser({});
74
- ServiceDiscovery.reset();
75
- return ServiceDiscovery.getInstance(logger, envParser);
76
- };
77
-
78
- beforeAll(async () => {
79
- db = new Kysely<TestDatabase>({
80
- dialect: new PostgresDialect({
81
- pool: new pg.Pool({
82
- ...TEST_DATABASE_CONFIG,
83
- database: 'postgres',
84
- }),
85
- }),
86
- plugins: [new CamelCasePlugin()],
87
- });
88
-
89
- // Create audit_logs table
90
- await db.schema
91
- .createTable(AUDIT_TABLE)
92
- .ifNotExists()
93
- .addColumn('id', 'varchar(32)', (col) => col.primaryKey())
94
- .addColumn('type', 'varchar', (col) => col.notNull())
95
- .addColumn('operation', 'varchar', (col) => col.notNull())
96
- .addColumn('table', 'varchar')
97
- .addColumn('entityId', 'varchar')
98
- .addColumn('oldValues', 'jsonb')
99
- .addColumn('newValues', 'jsonb')
100
- .addColumn('payload', 'jsonb')
101
- .addColumn('timestamp', 'timestamp', (col) =>
102
- col.defaultTo(sql`now()`).notNull(),
103
- )
104
- .addColumn('actorId', 'varchar')
105
- .addColumn('actorType', 'varchar')
106
- .addColumn('actorData', 'jsonb')
107
- .addColumn('metadata', 'jsonb')
108
- .execute();
109
-
110
- // Create users table
111
- await db.schema
112
- .createTable(USERS_TABLE)
113
- .ifNotExists()
114
- .addColumn('id', 'serial', (col) => col.primaryKey())
115
- .addColumn('name', 'varchar', (col) => col.notNull())
116
- .addColumn('email', 'varchar', (col) => col.notNull().unique())
117
- .execute();
118
-
119
- auditStorage = new KyselyAuditStorage({
120
- db,
121
- tableName: AUDIT_TABLE,
122
- });
123
- });
124
-
125
- beforeEach(() => {
126
- mockLogger = createMockLogger();
127
- });
128
-
129
- afterEach(async () => {
130
- await db.deleteFrom(AUDIT_TABLE).execute();
131
- await db.deleteFrom(USERS_TABLE).execute();
132
- });
133
-
134
- afterAll(async () => {
135
- await db.schema.dropTable(AUDIT_TABLE).ifExists().execute();
136
- await db.schema.dropTable(USERS_TABLE).ifExists().execute();
137
- await db.destroy();
138
- });
139
-
140
- describe('declarative audits with real database', () => {
141
- it('should write declarative audit to database on successful request', async () => {
142
- const serviceDiscovery = createServiceDiscovery(mockLogger);
143
-
144
- const auditStorageService: Service<
145
- 'auditStorage',
146
- KyselyAuditStorage<TestDatabase>
147
- > = {
148
- serviceName: 'auditStorage' as const,
149
- register: vi.fn().mockResolvedValue(auditStorage),
150
- };
151
-
152
- const outputSchema = z.object({ id: z.number(), email: z.string() });
153
-
154
- const audits: MappedAudit<TestAuditAction, typeof outputSchema>[] = [
155
- {
156
- type: 'user.created',
157
- payload: (response) => ({
158
- userId: response.id,
159
- email: response.email,
160
- }),
161
- },
162
- ];
163
-
164
- const endpoint = new Endpoint({
165
- route: '/users',
166
- method: 'POST',
167
- fn: async () => {
168
- return { id: 1, email: 'test@example.com' };
169
- },
170
- input: undefined,
171
- output: outputSchema,
172
- services: [],
173
- logger: mockLogger,
174
- timeout: undefined,
175
- memorySize: undefined,
176
- status: 201,
177
- getSession: undefined,
178
- authorize: undefined,
179
- description: undefined,
180
- events: [],
181
- publisherService: undefined,
182
- auditorStorageService: auditStorageService,
183
- audits,
184
- });
185
-
186
- const adaptor = new HonoEndpoint(endpoint);
187
- const app = new Hono();
188
- HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
189
- adaptor.addRoute(serviceDiscovery, app);
190
-
191
- const response = await app.request('/users', {
192
- method: 'POST',
193
- body: JSON.stringify({}),
194
- headers: { 'Content-Type': 'application/json' },
195
- });
196
-
197
- expect(response.status).toBe(201);
198
-
199
- // Verify audit was written to the real database
200
- const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
201
-
202
- expect(auditsInDb).toHaveLength(1);
203
- expect(auditsInDb[0].type).toBe('user.created');
204
- expect(auditsInDb[0].payload).toEqual({
205
- userId: 1,
206
- email: 'test@example.com',
207
- });
208
- });
209
-
210
- it('should not write audit when handler fails', async () => {
211
- const serviceDiscovery = createServiceDiscovery(mockLogger);
212
-
213
- const auditStorageService: Service<
214
- 'auditStorage',
215
- KyselyAuditStorage<TestDatabase>
216
- > = {
217
- serviceName: 'auditStorage' as const,
218
- register: vi.fn().mockResolvedValue(auditStorage),
219
- };
220
-
221
- const outputSchema = z.object({ id: z.number(), email: z.string() });
222
-
223
- const audits: MappedAudit<TestAuditAction, typeof outputSchema>[] = [
224
- {
225
- type: 'user.created',
226
- payload: (response) => ({
227
- userId: response.id,
228
- email: response.email,
229
- }),
230
- },
231
- ];
232
-
233
- const endpoint = new Endpoint({
234
- route: '/users',
235
- method: 'POST',
236
- fn: async () => {
237
- throw new Error('Handler failed');
238
- },
239
- input: undefined,
240
- output: outputSchema,
241
- services: [],
242
- logger: mockLogger,
243
- timeout: undefined,
244
- memorySize: undefined,
245
- status: 201,
246
- getSession: undefined,
247
- authorize: undefined,
248
- description: undefined,
249
- events: [],
250
- publisherService: undefined,
251
- auditorStorageService: auditStorageService,
252
- audits,
253
- });
254
-
255
- const adaptor = new HonoEndpoint(endpoint);
256
- const app = new Hono();
257
- HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
258
- adaptor.addRoute(serviceDiscovery, app);
259
-
260
- const response = await app.request('/users', {
261
- method: 'POST',
262
- body: JSON.stringify({}),
263
- headers: { 'Content-Type': 'application/json' },
264
- });
265
-
266
- expect(response.status).toBe(500);
267
-
268
- // Verify no audit was written
269
- const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
270
-
271
- expect(auditsInDb).toHaveLength(0);
272
- });
273
- });
274
-
275
- describe('manual audits with real database', () => {
276
- it('should write manual audits from handler to database', async () => {
277
- const serviceDiscovery = createServiceDiscovery(mockLogger);
278
-
279
- const auditStorageService: Service<
280
- 'auditStorage',
281
- KyselyAuditStorage<TestDatabase>
282
- > = {
283
- serviceName: 'auditStorage' as const,
284
- register: vi.fn().mockResolvedValue(auditStorage),
285
- };
286
-
287
- const outputSchema = z.object({ id: z.number(), email: z.string() });
288
-
289
- const endpoint = new Endpoint({
290
- route: '/users',
291
- method: 'POST',
292
- fn: async (
293
- ctx: EndpointContext<
294
- undefined,
295
- [],
296
- Logger,
297
- unknown,
298
- TestAuditAction,
299
- undefined,
300
- KyselyAuditStorage<TestDatabase>
301
- >,
302
- ) => {
303
- // Manual audit in handler - auditor is guaranteed to exist when TAuditStorage is configured
304
- ctx.auditor.audit('user.created', {
305
- userId: 42,
306
- email: 'manual@example.com',
307
- });
308
-
309
- return { id: 42, email: 'manual@example.com' };
310
- },
311
- input: undefined,
312
- output: outputSchema,
313
- services: [],
314
- logger: mockLogger,
315
- timeout: undefined,
316
- memorySize: undefined,
317
- status: 201,
318
- getSession: undefined,
319
- authorize: undefined,
320
- description: undefined,
321
- events: [],
322
- publisherService: undefined,
323
- auditorStorageService: auditStorageService,
324
- audits: [],
325
- });
326
-
327
- const adaptor = new HonoEndpoint(endpoint);
328
- const app = new Hono();
329
- HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
330
- adaptor.addRoute(serviceDiscovery, app);
331
-
332
- const response = await app.request('/users', {
333
- method: 'POST',
334
- body: JSON.stringify({}),
335
- headers: { 'Content-Type': 'application/json' },
336
- });
337
-
338
- expect(response.status).toBe(201);
339
-
340
- // Verify manual audit was written
341
- const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
342
-
343
- expect(auditsInDb).toHaveLength(1);
344
- expect(auditsInDb[0].type).toBe('user.created');
345
- expect(auditsInDb[0].payload).toEqual({
346
- userId: 42,
347
- email: 'manual@example.com',
348
- });
349
- });
350
-
351
- it('should not write manual audit when handler fails after audit call', async () => {
352
- const serviceDiscovery = createServiceDiscovery(mockLogger);
353
-
354
- const auditStorageService: Service<
355
- 'auditStorage',
356
- KyselyAuditStorage<TestDatabase>
357
- > = {
358
- serviceName: 'auditStorage' as const,
359
- register: vi.fn().mockResolvedValue(auditStorage),
360
- };
361
-
362
- const outputSchema = z.object({ id: z.number(), email: z.string() });
363
-
364
- const endpoint = new Endpoint({
365
- route: '/users',
366
- method: 'POST',
367
- fn: async (
368
- ctx: EndpointContext<
369
- undefined,
370
- [],
371
- Logger,
372
- unknown,
373
- TestAuditAction,
374
- undefined,
375
- KyselyAuditStorage<TestDatabase>
376
- >,
377
- ) => {
378
- // Manual audit before failure - auditor is guaranteed to exist
379
- ctx.auditor.audit('user.created', {
380
- userId: 99,
381
- email: 'shouldnotexist@example.com',
382
- });
383
-
384
- // Fail after audit
385
- throw new Error('Handler failed after audit');
386
- },
387
- input: undefined,
388
- output: outputSchema,
389
- services: [],
390
- logger: mockLogger,
391
- timeout: undefined,
392
- memorySize: undefined,
393
- status: 201,
394
- getSession: undefined,
395
- authorize: undefined,
396
- description: undefined,
397
- events: [],
398
- publisherService: undefined,
399
- auditorStorageService: auditStorageService,
400
- audits: [],
401
- });
402
-
403
- const adaptor = new HonoEndpoint(endpoint);
404
- const app = new Hono();
405
- HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
406
- adaptor.addRoute(serviceDiscovery, app);
407
-
408
- const response = await app.request('/users', {
409
- method: 'POST',
410
- body: JSON.stringify({}),
411
- headers: { 'Content-Type': 'application/json' },
412
- });
413
-
414
- expect(response.status).toBe(500);
415
-
416
- // Verify no audit was written (transaction rolled back)
417
- const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
418
-
419
- expect(auditsInDb).toHaveLength(0);
420
- });
421
- });
422
-
423
- describe('transactional consistency with real database', () => {
424
- it('should commit both user insert and audit on success', async () => {
425
- const serviceDiscovery = createServiceDiscovery(mockLogger);
426
-
427
- const databaseService: Service<'database', Kysely<TestDatabase>> = {
428
- serviceName: 'database' as const,
429
- register: vi.fn().mockResolvedValue(db),
430
- };
431
-
432
- const auditStorageService: Service<
433
- 'auditStorage',
434
- KyselyAuditStorage<TestDatabase>
435
- > = {
436
- serviceName: 'auditStorage' as const,
437
- register: vi.fn().mockResolvedValue(auditStorage),
438
- };
439
-
440
- const outputSchema = z.object({ id: z.number(), email: z.string() });
441
-
442
- const endpoint = new Endpoint({
443
- route: '/users',
444
- method: 'POST',
445
- fn: async (
446
- ctx: EndpointContext<
447
- undefined,
448
- [typeof databaseService],
449
- Logger,
450
- unknown,
451
- TestAuditAction,
452
- undefined,
453
- KyselyAuditStorage<TestDatabase>
454
- >,
455
- ) => {
456
- const database = ctx.services.database;
457
-
458
- // Insert user
459
- const user = await database
460
- .insertInto(USERS_TABLE)
461
- .values({ name: 'Success User', email: 'success@example.com' })
462
- .returningAll()
463
- .executeTakeFirstOrThrow();
464
-
465
- // Record audit - auditor is guaranteed to exist
466
- ctx.auditor.audit('user.created', {
467
- userId: user.id,
468
- email: user.email,
469
- });
470
-
471
- return { id: user.id, email: user.email };
472
- },
473
- input: undefined,
474
- output: outputSchema,
475
- services: [databaseService],
476
- logger: mockLogger,
477
- timeout: undefined,
478
- memorySize: undefined,
479
- status: 201,
480
- getSession: undefined,
481
- authorize: undefined,
482
- description: undefined,
483
- events: [],
484
- publisherService: undefined,
485
- auditorStorageService: auditStorageService,
486
- audits: [],
487
- });
488
-
489
- const adaptor = new HonoEndpoint(endpoint);
490
- const app = new Hono();
491
- HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
492
- adaptor.addRoute(serviceDiscovery, app);
493
-
494
- const response = await app.request('/users', {
495
- method: 'POST',
496
- body: JSON.stringify({}),
497
- headers: { 'Content-Type': 'application/json' },
498
- });
499
-
500
- expect(response.status).toBe(201);
501
-
502
- // Verify user was created
503
- const usersInDb = await db.selectFrom(USERS_TABLE).selectAll().execute();
504
- expect(usersInDb).toHaveLength(1);
505
- expect(usersInDb[0].email).toBe('success@example.com');
506
-
507
- // Verify audit was written
508
- const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
509
- expect(auditsInDb).toHaveLength(1);
510
- expect(auditsInDb[0].type).toBe('user.created');
511
- expect(auditsInDb[0].payload).toEqual({
512
- userId: usersInDb[0].id,
513
- email: 'success@example.com',
514
- });
515
- });
516
-
517
- it('should handle combined declarative and manual audits', async () => {
518
- const serviceDiscovery = createServiceDiscovery(mockLogger);
519
-
520
- const auditStorageService: Service<
521
- 'auditStorage',
522
- KyselyAuditStorage<TestDatabase>
523
- > = {
524
- serviceName: 'auditStorage' as const,
525
- register: vi.fn().mockResolvedValue(auditStorage),
526
- };
527
-
528
- const outputSchema = z.object({ id: z.number(), email: z.string() });
529
-
530
- // Declarative audit
531
- const audits: MappedAudit<TestAuditAction, typeof outputSchema>[] = [
532
- {
533
- type: 'user.created',
534
- payload: (response) => ({
535
- userId: response.id,
536
- email: response.email,
537
- }),
538
- },
539
- ];
540
-
541
- const endpoint = new Endpoint({
542
- route: '/users',
543
- method: 'POST',
544
- fn: async (
545
- ctx: EndpointContext<
546
- undefined,
547
- [],
548
- Logger,
549
- unknown,
550
- TestAuditAction,
551
- undefined,
552
- KyselyAuditStorage<TestDatabase>
553
- >,
554
- ) => {
555
- // Manual audit - auditor is guaranteed to exist
556
- ctx.auditor.audit('user.updated', {
557
- userId: 100,
558
- changes: ['verified'],
559
- });
560
-
561
- return { id: 100, email: 'combined@example.com' };
562
- },
563
- input: undefined,
564
- output: outputSchema,
565
- services: [],
566
- logger: mockLogger,
567
- timeout: undefined,
568
- memorySize: undefined,
569
- status: 201,
570
- getSession: undefined,
571
- authorize: undefined,
572
- description: undefined,
573
- events: [],
574
- publisherService: undefined,
575
- auditorStorageService: auditStorageService,
576
- audits,
577
- });
578
-
579
- const adaptor = new HonoEndpoint(endpoint);
580
- const app = new Hono();
581
- HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
582
- adaptor.addRoute(serviceDiscovery, app);
583
-
584
- const response = await app.request('/users', {
585
- method: 'POST',
586
- body: JSON.stringify({}),
587
- headers: { 'Content-Type': 'application/json' },
588
- });
589
-
590
- expect(response.status).toBe(201);
591
-
592
- // Verify both audits were written
593
- const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
594
-
595
- expect(auditsInDb).toHaveLength(2);
596
-
597
- const auditTypes = auditsInDb.map((a) => a.type).sort();
598
- expect(auditTypes).toEqual(['user.created', 'user.updated']);
599
- });
600
- });
601
-
602
- describe('actor extraction with real database', () => {
603
- it('should include actor information in audit records', async () => {
604
- const serviceDiscovery = createServiceDiscovery(mockLogger);
605
-
606
- const auditStorageService: Service<
607
- 'auditStorage',
608
- KyselyAuditStorage<TestDatabase>
609
- > = {
610
- serviceName: 'auditStorage' as const,
611
- register: vi.fn().mockResolvedValue(auditStorage),
612
- };
613
-
614
- const outputSchema = z.object({ id: z.number(), email: z.string() });
615
-
616
- const audits: MappedAudit<TestAuditAction, typeof outputSchema>[] = [
617
- {
618
- type: 'user.created',
619
- payload: (response) => ({
620
- userId: response.id,
621
- email: response.email,
622
- }),
623
- },
624
- ];
625
-
626
- const endpoint = new Endpoint({
627
- route: '/users',
628
- method: 'POST',
629
- fn: async () => {
630
- return { id: 1, email: 'actor-test@example.com' };
631
- },
632
- input: undefined,
633
- output: outputSchema,
634
- services: [],
635
- logger: mockLogger,
636
- timeout: undefined,
637
- memorySize: undefined,
638
- status: 201,
639
- getSession: undefined,
640
- authorize: undefined,
641
- description: undefined,
642
- events: [],
643
- publisherService: undefined,
644
- auditorStorageService: auditStorageService,
645
- audits,
646
- actorExtractor: async ({ header }) => {
647
- const userId = header('x-user-id');
648
- return {
649
- id: userId ?? 'anonymous',
650
- type: userId ? 'user' : 'anonymous',
651
- };
652
- },
653
- });
654
-
655
- const adaptor = new HonoEndpoint(endpoint);
656
- const app = new Hono();
657
- HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
658
- adaptor.addRoute(serviceDiscovery, app);
659
-
660
- const response = await app.request('/users', {
661
- method: 'POST',
662
- body: JSON.stringify({}),
663
- headers: {
664
- 'Content-Type': 'application/json',
665
- 'x-user-id': 'user-123',
666
- },
667
- });
668
-
669
- expect(response.status).toBe(201);
670
-
671
- // Verify actor was included in audit
672
- const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
673
-
674
- expect(auditsInDb).toHaveLength(1);
675
- expect(auditsInDb[0].actorId).toBe('user-123');
676
- expect(auditsInDb[0].actorType).toBe('user');
677
- });
678
- });
679
-
680
- describe('database service name matching', () => {
681
- it('should use audit transaction as db when databaseServiceName matches', async () => {
682
- const serviceDiscovery = createServiceDiscovery(mockLogger);
683
-
684
- // Create audit storage WITH databaseServiceName
685
- const auditStorageWithServiceName = new KyselyAuditStorage({
686
- db,
687
- tableName: AUDIT_TABLE,
688
- databaseServiceName: 'database', // Matches the database service
689
- });
690
-
691
- const databaseService: Service<'database', Kysely<TestDatabase>> = {
692
- serviceName: 'database' as const,
693
- register: vi.fn().mockResolvedValue(db),
694
- };
695
-
696
- const auditStorageService: Service<
697
- 'auditStorage',
698
- KyselyAuditStorage<TestDatabase>
699
- > = {
700
- serviceName: 'auditStorage' as const,
701
- register: vi.fn().mockResolvedValue(auditStorageWithServiceName),
702
- };
703
-
704
- const outputSchema = z.object({ id: z.number(), email: z.string() });
705
-
706
- let receivedDbIsTransaction = false;
707
-
708
- const endpoint = new Endpoint({
709
- route: '/users',
710
- method: 'POST',
711
- fn: async (
712
- ctx: EndpointContext<
713
- undefined,
714
- [typeof databaseService],
715
- Logger,
716
- unknown,
717
- TestAuditAction,
718
- Kysely<TestDatabase>,
719
- KyselyAuditStorage<TestDatabase>
720
- >,
721
- ) => {
722
- // Check if db is a transaction (has isTransaction property from Kysely)
723
- // When databaseServiceName matches, db should be the transaction
724
- receivedDbIsTransaction = (ctx.db as any)?.isTransaction === true;
725
-
726
- // Insert user using ctx.db (should be the transaction)
727
- const user = await ctx.db
728
- .insertInto(USERS_TABLE)
729
- .values({ name: 'Transaction User', email: 'trx@example.com' })
730
- .returningAll()
731
- .executeTakeFirstOrThrow();
732
-
733
- // Record audit
734
- ctx.auditor.audit('user.created', {
735
- userId: user.id,
736
- email: user.email,
737
- });
738
-
739
- return { id: user.id, email: user.email };
740
- },
741
- input: undefined,
742
- output: outputSchema,
743
- services: [databaseService],
744
- logger: mockLogger,
745
- timeout: undefined,
746
- memorySize: undefined,
747
- status: 201,
748
- getSession: undefined,
749
- authorize: undefined,
750
- description: undefined,
751
- events: [],
752
- publisherService: undefined,
753
- auditorStorageService: auditStorageService,
754
- audits: [],
755
- databaseService,
756
- });
757
-
758
- const adaptor = new HonoEndpoint(endpoint);
759
- const app = new Hono();
760
- HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
761
- adaptor.addRoute(serviceDiscovery, app);
762
-
763
- const response = await app.request('/users', {
764
- method: 'POST',
765
- body: JSON.stringify({}),
766
- headers: { 'Content-Type': 'application/json' },
767
- });
768
-
769
- expect(response.status).toBe(201);
770
- expect(receivedDbIsTransaction).toBe(true);
771
-
772
- // Verify both user and audit were committed
773
- const usersInDb = await db.selectFrom(USERS_TABLE).selectAll().execute();
774
- expect(usersInDb).toHaveLength(1);
775
- expect(usersInDb[0].email).toBe('trx@example.com');
776
-
777
- const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
778
- expect(auditsInDb).toHaveLength(1);
779
- });
780
-
781
- it('should use raw db when databaseServiceName does not match', async () => {
782
- const serviceDiscovery = createServiceDiscovery(mockLogger);
783
-
784
- // Create audit storage with DIFFERENT databaseServiceName
785
- const auditStorageWithDifferentServiceName = new KyselyAuditStorage({
786
- db,
787
- tableName: AUDIT_TABLE,
788
- databaseServiceName: 'auditDatabase', // Different from 'database'
789
- });
790
-
791
- const databaseService: Service<'database', Kysely<TestDatabase>> = {
792
- serviceName: 'database' as const,
793
- register: vi.fn().mockResolvedValue(db),
794
- };
795
-
796
- const auditStorageService: Service<
797
- 'auditStorage',
798
- KyselyAuditStorage<TestDatabase>
799
- > = {
800
- serviceName: 'auditStorage' as const,
801
- register: vi
802
- .fn()
803
- .mockResolvedValue(auditStorageWithDifferentServiceName),
804
- };
805
-
806
- const outputSchema = z.object({ id: z.number(), email: z.string() });
807
-
808
- let receivedDbIsTransaction = false;
809
-
810
- const endpoint = new Endpoint({
811
- route: '/users',
812
- method: 'POST',
813
- fn: async (
814
- ctx: EndpointContext<
815
- undefined,
816
- [typeof databaseService],
817
- Logger,
818
- unknown,
819
- TestAuditAction,
820
- Kysely<TestDatabase>,
821
- KyselyAuditStorage<TestDatabase>
822
- >,
823
- ) => {
824
- // When databaseServiceName doesn't match, db should be raw (not a transaction)
825
- receivedDbIsTransaction = (ctx.db as any)?.isTransaction === true;
826
-
827
- // Insert user using ctx.db (should be raw db, not transaction)
828
- const user = await ctx.db
829
- .insertInto(USERS_TABLE)
830
- .values({ name: 'Raw DB User', email: 'raw@example.com' })
831
- .returningAll()
832
- .executeTakeFirstOrThrow();
833
-
834
- // Record audit
835
- ctx.auditor.audit('user.created', {
836
- userId: user.id,
837
- email: user.email,
838
- });
839
-
840
- return { id: user.id, email: user.email };
841
- },
842
- input: undefined,
843
- output: outputSchema,
844
- services: [databaseService],
845
- logger: mockLogger,
846
- timeout: undefined,
847
- memorySize: undefined,
848
- status: 201,
849
- getSession: undefined,
850
- authorize: undefined,
851
- description: undefined,
852
- events: [],
853
- publisherService: undefined,
854
- auditorStorageService: auditStorageService,
855
- audits: [],
856
- databaseService,
857
- });
858
-
859
- const adaptor = new HonoEndpoint(endpoint);
860
- const app = new Hono();
861
- HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
862
- adaptor.addRoute(serviceDiscovery, app);
863
-
864
- const response = await app.request('/users', {
865
- method: 'POST',
866
- body: JSON.stringify({}),
867
- headers: { 'Content-Type': 'application/json' },
868
- });
869
-
870
- expect(response.status).toBe(201);
871
- // db should NOT be a transaction since service names don't match
872
- expect(receivedDbIsTransaction).toBe(false);
873
-
874
- // Both should still be committed (but not in the same transaction)
875
- const usersInDb = await db.selectFrom(USERS_TABLE).selectAll().execute();
876
- expect(usersInDb).toHaveLength(1);
877
-
878
- const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
879
- expect(auditsInDb).toHaveLength(1);
880
- });
881
-
882
- it('should use raw db when databaseServiceName is not set on audit storage', async () => {
883
- const serviceDiscovery = createServiceDiscovery(mockLogger);
884
-
885
- // Create audit storage WITHOUT databaseServiceName (uses default auditStorage from beforeAll)
886
- const databaseService: Service<'database', Kysely<TestDatabase>> = {
887
- serviceName: 'database' as const,
888
- register: vi.fn().mockResolvedValue(db),
889
- };
890
-
891
- const auditStorageService: Service<
892
- 'auditStorage',
893
- KyselyAuditStorage<TestDatabase>
894
- > = {
895
- serviceName: 'auditStorage' as const,
896
- register: vi.fn().mockResolvedValue(auditStorage), // No databaseServiceName set
897
- };
898
-
899
- const outputSchema = z.object({ id: z.number(), email: z.string() });
900
-
901
- let receivedDbIsTransaction = false;
902
-
903
- const endpoint = new Endpoint({
904
- route: '/users',
905
- method: 'POST',
906
- fn: async (
907
- ctx: EndpointContext<
908
- undefined,
909
- [typeof databaseService],
910
- Logger,
911
- unknown,
912
- TestAuditAction,
913
- Kysely<TestDatabase>,
914
- KyselyAuditStorage<TestDatabase>
915
- >,
916
- ) => {
917
- // When databaseServiceName is not set, db should be raw
918
- receivedDbIsTransaction = (ctx.db as any)?.isTransaction === true;
919
-
920
- const user = await ctx.db
921
- .insertInto(USERS_TABLE)
922
- .values({
923
- name: 'No ServiceName User',
924
- email: 'noname@example.com',
925
- })
926
- .returningAll()
927
- .executeTakeFirstOrThrow();
928
-
929
- ctx.auditor.audit('user.created', {
930
- userId: user.id,
931
- email: user.email,
932
- });
933
-
934
- return { id: user.id, email: user.email };
935
- },
936
- input: undefined,
937
- output: outputSchema,
938
- services: [databaseService],
939
- logger: mockLogger,
940
- timeout: undefined,
941
- memorySize: undefined,
942
- status: 201,
943
- getSession: undefined,
944
- authorize: undefined,
945
- description: undefined,
946
- events: [],
947
- publisherService: undefined,
948
- auditorStorageService: auditStorageService,
949
- audits: [],
950
- databaseService,
951
- });
952
-
953
- const adaptor = new HonoEndpoint(endpoint);
954
- const app = new Hono();
955
- HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
956
- adaptor.addRoute(serviceDiscovery, app);
957
-
958
- const response = await app.request('/users', {
959
- method: 'POST',
960
- body: JSON.stringify({}),
961
- headers: { 'Content-Type': 'application/json' },
962
- });
963
-
964
- expect(response.status).toBe(201);
965
- expect(receivedDbIsTransaction).toBe(false);
966
-
967
- const usersInDb = await db.selectFrom(USERS_TABLE).selectAll().execute();
968
- expect(usersInDb).toHaveLength(1);
969
-
970
- const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
971
- expect(auditsInDb).toHaveLength(1);
972
- });
973
-
974
- it('should rollback both user insert and audit when handler fails with matching databaseServiceName', async () => {
975
- const serviceDiscovery = createServiceDiscovery(mockLogger);
976
-
977
- // Create audit storage WITH databaseServiceName
978
- const auditStorageWithServiceName = new KyselyAuditStorage({
979
- db,
980
- tableName: AUDIT_TABLE,
981
- databaseServiceName: 'database',
982
- });
983
-
984
- const databaseService: Service<'database', Kysely<TestDatabase>> = {
985
- serviceName: 'database' as const,
986
- register: vi.fn().mockResolvedValue(db),
987
- };
988
-
989
- const auditStorageService: Service<
990
- 'auditStorage',
991
- KyselyAuditStorage<TestDatabase>
992
- > = {
993
- serviceName: 'auditStorage' as const,
994
- register: vi.fn().mockResolvedValue(auditStorageWithServiceName),
995
- };
996
-
997
- const outputSchema = z.object({ id: z.number(), email: z.string() });
998
-
999
- const endpoint = new Endpoint({
1000
- route: '/users',
1001
- method: 'POST',
1002
- fn: async (
1003
- ctx: EndpointContext<
1004
- undefined,
1005
- [typeof databaseService],
1006
- Logger,
1007
- unknown,
1008
- TestAuditAction,
1009
- Kysely<TestDatabase>,
1010
- KyselyAuditStorage<TestDatabase>
1011
- >,
1012
- ) => {
1013
- // Insert user (should be rolled back)
1014
- const user = await ctx.db
1015
- .insertInto(USERS_TABLE)
1016
- .values({ name: 'Rollback User', email: 'rollback@example.com' })
1017
- .returningAll()
1018
- .executeTakeFirstOrThrow();
1019
-
1020
- // Record audit (should also be rolled back)
1021
- ctx.auditor.audit('user.created', {
1022
- userId: user.id,
1023
- email: user.email,
1024
- });
1025
-
1026
- // Fail after both operations
1027
- throw new Error('Simulated failure');
1028
- },
1029
- input: undefined,
1030
- output: outputSchema,
1031
- services: [databaseService],
1032
- logger: mockLogger,
1033
- timeout: undefined,
1034
- memorySize: undefined,
1035
- status: 201,
1036
- getSession: undefined,
1037
- authorize: undefined,
1038
- description: undefined,
1039
- events: [],
1040
- publisherService: undefined,
1041
- auditorStorageService: auditStorageService,
1042
- audits: [],
1043
- databaseService,
1044
- });
1045
-
1046
- const adaptor = new HonoEndpoint(endpoint);
1047
- const app = new Hono();
1048
- HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
1049
- adaptor.addRoute(serviceDiscovery, app);
1050
-
1051
- const response = await app.request('/users', {
1052
- method: 'POST',
1053
- body: JSON.stringify({}),
1054
- headers: { 'Content-Type': 'application/json' },
1055
- });
1056
-
1057
- expect(response.status).toBe(500);
1058
-
1059
- // Both should be rolled back since they're in the same transaction
1060
- const usersInDb = await db.selectFrom(USERS_TABLE).selectAll().execute();
1061
- expect(usersInDb).toHaveLength(0);
1062
-
1063
- const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
1064
- expect(auditsInDb).toHaveLength(0);
1065
- });
1066
- });
56
+ let db: Kysely<TestDatabase>;
57
+ let auditStorage: KyselyAuditStorage<TestDatabase>;
58
+ let mockLogger: Logger;
59
+
60
+ const createMockLogger = (): Logger => ({
61
+ debug: vi.fn(),
62
+ info: vi.fn(),
63
+ warn: vi.fn(),
64
+ error: vi.fn(),
65
+ fatal: vi.fn(),
66
+ trace: vi.fn(),
67
+ child: vi.fn(function (this: Logger) {
68
+ return this;
69
+ }),
70
+ });
71
+
72
+ const createServiceDiscovery = () => {
73
+ const envParser = new EnvironmentParser({});
74
+ ServiceDiscovery.reset();
75
+ return ServiceDiscovery.getInstance(envParser);
76
+ };
77
+
78
+ beforeAll(async () => {
79
+ db = new Kysely<TestDatabase>({
80
+ dialect: new PostgresDialect({
81
+ pool: new pg.Pool({
82
+ ...TEST_DATABASE_CONFIG,
83
+ database: 'postgres',
84
+ }),
85
+ }),
86
+ plugins: [new CamelCasePlugin()],
87
+ });
88
+
89
+ // Create audit_logs table
90
+ await db.schema
91
+ .createTable(AUDIT_TABLE)
92
+ .ifNotExists()
93
+ .addColumn('id', 'varchar(32)', (col) => col.primaryKey())
94
+ .addColumn('type', 'varchar', (col) => col.notNull())
95
+ .addColumn('operation', 'varchar', (col) => col.notNull())
96
+ .addColumn('table', 'varchar')
97
+ .addColumn('entityId', 'varchar')
98
+ .addColumn('oldValues', 'jsonb')
99
+ .addColumn('newValues', 'jsonb')
100
+ .addColumn('payload', 'jsonb')
101
+ .addColumn('timestamp', 'timestamp', (col) =>
102
+ col.defaultTo(sql`now()`).notNull(),
103
+ )
104
+ .addColumn('actorId', 'varchar')
105
+ .addColumn('actorType', 'varchar')
106
+ .addColumn('actorData', 'jsonb')
107
+ .addColumn('metadata', 'jsonb')
108
+ .execute();
109
+
110
+ // Create users table
111
+ await db.schema
112
+ .createTable(USERS_TABLE)
113
+ .ifNotExists()
114
+ .addColumn('id', 'serial', (col) => col.primaryKey())
115
+ .addColumn('name', 'varchar', (col) => col.notNull())
116
+ .addColumn('email', 'varchar', (col) => col.notNull().unique())
117
+ .execute();
118
+
119
+ auditStorage = new KyselyAuditStorage({
120
+ db,
121
+ tableName: AUDIT_TABLE,
122
+ });
123
+ });
124
+
125
+ beforeEach(() => {
126
+ mockLogger = createMockLogger();
127
+ });
128
+
129
+ afterEach(async () => {
130
+ await db.deleteFrom(AUDIT_TABLE).execute();
131
+ await db.deleteFrom(USERS_TABLE).execute();
132
+ });
133
+
134
+ afterAll(async () => {
135
+ await db.schema.dropTable(AUDIT_TABLE).ifExists().execute();
136
+ await db.schema.dropTable(USERS_TABLE).ifExists().execute();
137
+ await db.destroy();
138
+ });
139
+
140
+ describe('declarative audits with real database', () => {
141
+ it('should write declarative audit to database on successful request', async () => {
142
+ const serviceDiscovery = createServiceDiscovery();
143
+
144
+ const auditStorageService: Service<
145
+ 'auditStorage',
146
+ KyselyAuditStorage<TestDatabase>
147
+ > = {
148
+ serviceName: 'auditStorage' as const,
149
+ register: vi.fn().mockResolvedValue(auditStorage),
150
+ };
151
+
152
+ const outputSchema = z.object({ id: z.number(), email: z.string() });
153
+
154
+ const audits: MappedAudit<TestAuditAction, typeof outputSchema>[] = [
155
+ {
156
+ type: 'user.created',
157
+ payload: (response) => ({
158
+ userId: response.id,
159
+ email: response.email,
160
+ }),
161
+ },
162
+ ];
163
+
164
+ const endpoint = new Endpoint({
165
+ route: '/users',
166
+ method: 'POST',
167
+ fn: async () => {
168
+ return { id: 1, email: 'test@example.com' };
169
+ },
170
+ input: undefined,
171
+ output: outputSchema,
172
+ services: [],
173
+ logger: mockLogger,
174
+ timeout: undefined,
175
+ memorySize: undefined,
176
+ status: 201,
177
+ getSession: undefined,
178
+ authorize: undefined,
179
+ description: undefined,
180
+ events: [],
181
+ publisherService: undefined,
182
+ auditorStorageService: auditStorageService,
183
+ audits,
184
+ });
185
+
186
+ const adaptor = new HonoEndpoint(endpoint);
187
+ const app = new Hono();
188
+ HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
189
+ adaptor.addRoute(serviceDiscovery, app);
190
+
191
+ const response = await app.request('/users', {
192
+ method: 'POST',
193
+ body: JSON.stringify({}),
194
+ headers: { 'Content-Type': 'application/json' },
195
+ });
196
+
197
+ expect(response.status).toBe(201);
198
+
199
+ // Verify audit was written to the real database
200
+ const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
201
+
202
+ expect(auditsInDb).toHaveLength(1);
203
+ expect(auditsInDb[0].type).toBe('user.created');
204
+ expect(auditsInDb[0].payload).toEqual({
205
+ userId: 1,
206
+ email: 'test@example.com',
207
+ });
208
+ });
209
+
210
+ it('should not write audit when handler fails', async () => {
211
+ const serviceDiscovery = createServiceDiscovery();
212
+
213
+ const auditStorageService: Service<
214
+ 'auditStorage',
215
+ KyselyAuditStorage<TestDatabase>
216
+ > = {
217
+ serviceName: 'auditStorage' as const,
218
+ register: vi.fn().mockResolvedValue(auditStorage),
219
+ };
220
+
221
+ const outputSchema = z.object({ id: z.number(), email: z.string() });
222
+
223
+ const audits: MappedAudit<TestAuditAction, typeof outputSchema>[] = [
224
+ {
225
+ type: 'user.created',
226
+ payload: (response) => ({
227
+ userId: response.id,
228
+ email: response.email,
229
+ }),
230
+ },
231
+ ];
232
+
233
+ const endpoint = new Endpoint({
234
+ route: '/users',
235
+ method: 'POST',
236
+ fn: async () => {
237
+ throw new Error('Handler failed');
238
+ },
239
+ input: undefined,
240
+ output: outputSchema,
241
+ services: [],
242
+ logger: mockLogger,
243
+ timeout: undefined,
244
+ memorySize: undefined,
245
+ status: 201,
246
+ getSession: undefined,
247
+ authorize: undefined,
248
+ description: undefined,
249
+ events: [],
250
+ publisherService: undefined,
251
+ auditorStorageService: auditStorageService,
252
+ audits,
253
+ });
254
+
255
+ const adaptor = new HonoEndpoint(endpoint);
256
+ const app = new Hono();
257
+ HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
258
+ adaptor.addRoute(serviceDiscovery, app);
259
+
260
+ const response = await app.request('/users', {
261
+ method: 'POST',
262
+ body: JSON.stringify({}),
263
+ headers: { 'Content-Type': 'application/json' },
264
+ });
265
+
266
+ expect(response.status).toBe(500);
267
+
268
+ // Verify no audit was written
269
+ const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
270
+
271
+ expect(auditsInDb).toHaveLength(0);
272
+ });
273
+ });
274
+
275
+ describe('manual audits with real database', () => {
276
+ it('should write manual audits from handler to database', async () => {
277
+ const serviceDiscovery = createServiceDiscovery();
278
+
279
+ const auditStorageService: Service<
280
+ 'auditStorage',
281
+ KyselyAuditStorage<TestDatabase>
282
+ > = {
283
+ serviceName: 'auditStorage' as const,
284
+ register: vi.fn().mockResolvedValue(auditStorage),
285
+ };
286
+
287
+ const outputSchema = z.object({ id: z.number(), email: z.string() });
288
+
289
+ const endpoint = new Endpoint({
290
+ route: '/users',
291
+ method: 'POST',
292
+ fn: async (
293
+ ctx: EndpointContext<
294
+ undefined,
295
+ [],
296
+ Logger,
297
+ unknown,
298
+ TestAuditAction,
299
+ undefined,
300
+ KyselyAuditStorage<TestDatabase>
301
+ >,
302
+ ) => {
303
+ // Manual audit in handler - auditor is guaranteed to exist when TAuditStorage is configured
304
+ ctx.auditor.audit('user.created', {
305
+ userId: 42,
306
+ email: 'manual@example.com',
307
+ });
308
+
309
+ return { id: 42, email: 'manual@example.com' };
310
+ },
311
+ input: undefined,
312
+ output: outputSchema,
313
+ services: [],
314
+ logger: mockLogger,
315
+ timeout: undefined,
316
+ memorySize: undefined,
317
+ status: 201,
318
+ getSession: undefined,
319
+ authorize: undefined,
320
+ description: undefined,
321
+ events: [],
322
+ publisherService: undefined,
323
+ auditorStorageService: auditStorageService,
324
+ audits: [],
325
+ });
326
+
327
+ const adaptor = new HonoEndpoint(endpoint);
328
+ const app = new Hono();
329
+ HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
330
+ adaptor.addRoute(serviceDiscovery, app);
331
+
332
+ const response = await app.request('/users', {
333
+ method: 'POST',
334
+ body: JSON.stringify({}),
335
+ headers: { 'Content-Type': 'application/json' },
336
+ });
337
+
338
+ expect(response.status).toBe(201);
339
+
340
+ // Verify manual audit was written
341
+ const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
342
+
343
+ expect(auditsInDb).toHaveLength(1);
344
+ expect(auditsInDb[0].type).toBe('user.created');
345
+ expect(auditsInDb[0].payload).toEqual({
346
+ userId: 42,
347
+ email: 'manual@example.com',
348
+ });
349
+ });
350
+
351
+ it('should not write manual audit when handler fails after audit call', async () => {
352
+ const serviceDiscovery = createServiceDiscovery();
353
+
354
+ const auditStorageService: Service<
355
+ 'auditStorage',
356
+ KyselyAuditStorage<TestDatabase>
357
+ > = {
358
+ serviceName: 'auditStorage' as const,
359
+ register: vi.fn().mockResolvedValue(auditStorage),
360
+ };
361
+
362
+ const outputSchema = z.object({ id: z.number(), email: z.string() });
363
+
364
+ const endpoint = new Endpoint({
365
+ route: '/users',
366
+ method: 'POST',
367
+ fn: async (
368
+ ctx: EndpointContext<
369
+ undefined,
370
+ [],
371
+ Logger,
372
+ unknown,
373
+ TestAuditAction,
374
+ undefined,
375
+ KyselyAuditStorage<TestDatabase>
376
+ >,
377
+ ) => {
378
+ // Manual audit before failure - auditor is guaranteed to exist
379
+ ctx.auditor.audit('user.created', {
380
+ userId: 99,
381
+ email: 'shouldnotexist@example.com',
382
+ });
383
+
384
+ // Fail after audit
385
+ throw new Error('Handler failed after audit');
386
+ },
387
+ input: undefined,
388
+ output: outputSchema,
389
+ services: [],
390
+ logger: mockLogger,
391
+ timeout: undefined,
392
+ memorySize: undefined,
393
+ status: 201,
394
+ getSession: undefined,
395
+ authorize: undefined,
396
+ description: undefined,
397
+ events: [],
398
+ publisherService: undefined,
399
+ auditorStorageService: auditStorageService,
400
+ audits: [],
401
+ });
402
+
403
+ const adaptor = new HonoEndpoint(endpoint);
404
+ const app = new Hono();
405
+ HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
406
+ adaptor.addRoute(serviceDiscovery, app);
407
+
408
+ const response = await app.request('/users', {
409
+ method: 'POST',
410
+ body: JSON.stringify({}),
411
+ headers: { 'Content-Type': 'application/json' },
412
+ });
413
+
414
+ expect(response.status).toBe(500);
415
+
416
+ // Verify no audit was written (transaction rolled back)
417
+ const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
418
+
419
+ expect(auditsInDb).toHaveLength(0);
420
+ });
421
+ });
422
+
423
+ describe('transactional consistency with real database', () => {
424
+ it('should commit both user insert and audit on success', async () => {
425
+ const serviceDiscovery = createServiceDiscovery();
426
+
427
+ const databaseService: Service<'database', Kysely<TestDatabase>> = {
428
+ serviceName: 'database' as const,
429
+ register: vi.fn().mockResolvedValue(db),
430
+ };
431
+
432
+ const auditStorageService: Service<
433
+ 'auditStorage',
434
+ KyselyAuditStorage<TestDatabase>
435
+ > = {
436
+ serviceName: 'auditStorage' as const,
437
+ register: vi.fn().mockResolvedValue(auditStorage),
438
+ };
439
+
440
+ const outputSchema = z.object({ id: z.number(), email: z.string() });
441
+
442
+ const endpoint = new Endpoint({
443
+ route: '/users',
444
+ method: 'POST',
445
+ fn: async (
446
+ ctx: EndpointContext<
447
+ undefined,
448
+ [typeof databaseService],
449
+ Logger,
450
+ unknown,
451
+ TestAuditAction,
452
+ undefined,
453
+ KyselyAuditStorage<TestDatabase>
454
+ >,
455
+ ) => {
456
+ const database = ctx.services.database;
457
+
458
+ // Insert user
459
+ const user = await database
460
+ .insertInto(USERS_TABLE)
461
+ .values({ name: 'Success User', email: 'success@example.com' })
462
+ .returningAll()
463
+ .executeTakeFirstOrThrow();
464
+
465
+ // Record audit - auditor is guaranteed to exist
466
+ ctx.auditor.audit('user.created', {
467
+ userId: user.id,
468
+ email: user.email,
469
+ });
470
+
471
+ return { id: user.id, email: user.email };
472
+ },
473
+ input: undefined,
474
+ output: outputSchema,
475
+ services: [databaseService],
476
+ logger: mockLogger,
477
+ timeout: undefined,
478
+ memorySize: undefined,
479
+ status: 201,
480
+ getSession: undefined,
481
+ authorize: undefined,
482
+ description: undefined,
483
+ events: [],
484
+ publisherService: undefined,
485
+ auditorStorageService: auditStorageService,
486
+ audits: [],
487
+ });
488
+
489
+ const adaptor = new HonoEndpoint(endpoint);
490
+ const app = new Hono();
491
+ HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
492
+ adaptor.addRoute(serviceDiscovery, app);
493
+
494
+ const response = await app.request('/users', {
495
+ method: 'POST',
496
+ body: JSON.stringify({}),
497
+ headers: { 'Content-Type': 'application/json' },
498
+ });
499
+
500
+ expect(response.status).toBe(201);
501
+
502
+ // Verify user was created
503
+ const usersInDb = await db.selectFrom(USERS_TABLE).selectAll().execute();
504
+ expect(usersInDb).toHaveLength(1);
505
+ expect(usersInDb[0].email).toBe('success@example.com');
506
+
507
+ // Verify audit was written
508
+ const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
509
+ expect(auditsInDb).toHaveLength(1);
510
+ expect(auditsInDb[0].type).toBe('user.created');
511
+ expect(auditsInDb[0].payload).toEqual({
512
+ userId: usersInDb[0].id,
513
+ email: 'success@example.com',
514
+ });
515
+ });
516
+
517
+ it('should handle combined declarative and manual audits', async () => {
518
+ const serviceDiscovery = createServiceDiscovery();
519
+
520
+ const auditStorageService: Service<
521
+ 'auditStorage',
522
+ KyselyAuditStorage<TestDatabase>
523
+ > = {
524
+ serviceName: 'auditStorage' as const,
525
+ register: vi.fn().mockResolvedValue(auditStorage),
526
+ };
527
+
528
+ const outputSchema = z.object({ id: z.number(), email: z.string() });
529
+
530
+ // Declarative audit
531
+ const audits: MappedAudit<TestAuditAction, typeof outputSchema>[] = [
532
+ {
533
+ type: 'user.created',
534
+ payload: (response) => ({
535
+ userId: response.id,
536
+ email: response.email,
537
+ }),
538
+ },
539
+ ];
540
+
541
+ const endpoint = new Endpoint({
542
+ route: '/users',
543
+ method: 'POST',
544
+ fn: async (
545
+ ctx: EndpointContext<
546
+ undefined,
547
+ [],
548
+ Logger,
549
+ unknown,
550
+ TestAuditAction,
551
+ undefined,
552
+ KyselyAuditStorage<TestDatabase>
553
+ >,
554
+ ) => {
555
+ // Manual audit - auditor is guaranteed to exist
556
+ ctx.auditor.audit('user.updated', {
557
+ userId: 100,
558
+ changes: ['verified'],
559
+ });
560
+
561
+ return { id: 100, email: 'combined@example.com' };
562
+ },
563
+ input: undefined,
564
+ output: outputSchema,
565
+ services: [],
566
+ logger: mockLogger,
567
+ timeout: undefined,
568
+ memorySize: undefined,
569
+ status: 201,
570
+ getSession: undefined,
571
+ authorize: undefined,
572
+ description: undefined,
573
+ events: [],
574
+ publisherService: undefined,
575
+ auditorStorageService: auditStorageService,
576
+ audits,
577
+ });
578
+
579
+ const adaptor = new HonoEndpoint(endpoint);
580
+ const app = new Hono();
581
+ HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
582
+ adaptor.addRoute(serviceDiscovery, app);
583
+
584
+ const response = await app.request('/users', {
585
+ method: 'POST',
586
+ body: JSON.stringify({}),
587
+ headers: { 'Content-Type': 'application/json' },
588
+ });
589
+
590
+ expect(response.status).toBe(201);
591
+
592
+ // Verify both audits were written
593
+ const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
594
+
595
+ expect(auditsInDb).toHaveLength(2);
596
+
597
+ const auditTypes = auditsInDb.map((a) => a.type).sort();
598
+ expect(auditTypes).toEqual(['user.created', 'user.updated']);
599
+ });
600
+ });
601
+
602
+ describe('actor extraction with real database', () => {
603
+ it('should include actor information in audit records', async () => {
604
+ const serviceDiscovery = createServiceDiscovery();
605
+
606
+ const auditStorageService: Service<
607
+ 'auditStorage',
608
+ KyselyAuditStorage<TestDatabase>
609
+ > = {
610
+ serviceName: 'auditStorage' as const,
611
+ register: vi.fn().mockResolvedValue(auditStorage),
612
+ };
613
+
614
+ const outputSchema = z.object({ id: z.number(), email: z.string() });
615
+
616
+ const audits: MappedAudit<TestAuditAction, typeof outputSchema>[] = [
617
+ {
618
+ type: 'user.created',
619
+ payload: (response) => ({
620
+ userId: response.id,
621
+ email: response.email,
622
+ }),
623
+ },
624
+ ];
625
+
626
+ const endpoint = new Endpoint({
627
+ route: '/users',
628
+ method: 'POST',
629
+ fn: async () => {
630
+ return { id: 1, email: 'actor-test@example.com' };
631
+ },
632
+ input: undefined,
633
+ output: outputSchema,
634
+ services: [],
635
+ logger: mockLogger,
636
+ timeout: undefined,
637
+ memorySize: undefined,
638
+ status: 201,
639
+ getSession: undefined,
640
+ authorize: undefined,
641
+ description: undefined,
642
+ events: [],
643
+ publisherService: undefined,
644
+ auditorStorageService: auditStorageService,
645
+ audits,
646
+ actorExtractor: async ({ header }) => {
647
+ const userId = header('x-user-id');
648
+ return {
649
+ id: userId ?? 'anonymous',
650
+ type: userId ? 'user' : 'anonymous',
651
+ };
652
+ },
653
+ });
654
+
655
+ const adaptor = new HonoEndpoint(endpoint);
656
+ const app = new Hono();
657
+ HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
658
+ adaptor.addRoute(serviceDiscovery, app);
659
+
660
+ const response = await app.request('/users', {
661
+ method: 'POST',
662
+ body: JSON.stringify({}),
663
+ headers: {
664
+ 'Content-Type': 'application/json',
665
+ 'x-user-id': 'user-123',
666
+ },
667
+ });
668
+
669
+ expect(response.status).toBe(201);
670
+
671
+ // Verify actor was included in audit
672
+ const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
673
+
674
+ expect(auditsInDb).toHaveLength(1);
675
+ expect(auditsInDb[0].actorId).toBe('user-123');
676
+ expect(auditsInDb[0].actorType).toBe('user');
677
+ });
678
+ });
679
+
680
+ describe('database service name matching', () => {
681
+ it('should use audit transaction as db when databaseServiceName matches', async () => {
682
+ const serviceDiscovery = createServiceDiscovery();
683
+
684
+ // Create audit storage WITH databaseServiceName
685
+ const auditStorageWithServiceName = new KyselyAuditStorage({
686
+ db,
687
+ tableName: AUDIT_TABLE,
688
+ databaseServiceName: 'database', // Matches the database service
689
+ });
690
+
691
+ const databaseService: Service<'database', Kysely<TestDatabase>> = {
692
+ serviceName: 'database' as const,
693
+ register: vi.fn().mockResolvedValue(db),
694
+ };
695
+
696
+ const auditStorageService: Service<
697
+ 'auditStorage',
698
+ KyselyAuditStorage<TestDatabase>
699
+ > = {
700
+ serviceName: 'auditStorage' as const,
701
+ register: vi.fn().mockResolvedValue(auditStorageWithServiceName),
702
+ };
703
+
704
+ const outputSchema = z.object({ id: z.number(), email: z.string() });
705
+
706
+ let receivedDbIsTransaction = false;
707
+
708
+ const endpoint = new Endpoint({
709
+ route: '/users',
710
+ method: 'POST',
711
+ fn: async (
712
+ ctx: EndpointContext<
713
+ undefined,
714
+ [typeof databaseService],
715
+ Logger,
716
+ unknown,
717
+ TestAuditAction,
718
+ Kysely<TestDatabase>,
719
+ KyselyAuditStorage<TestDatabase>
720
+ >,
721
+ ) => {
722
+ // Check if db is a transaction (has isTransaction property from Kysely)
723
+ // When databaseServiceName matches, db should be the transaction
724
+ receivedDbIsTransaction = (ctx.db as any)?.isTransaction === true;
725
+
726
+ // Insert user using ctx.db (should be the transaction)
727
+ const user = await ctx.db
728
+ .insertInto(USERS_TABLE)
729
+ .values({ name: 'Transaction User', email: 'trx@example.com' })
730
+ .returningAll()
731
+ .executeTakeFirstOrThrow();
732
+
733
+ // Record audit
734
+ ctx.auditor.audit('user.created', {
735
+ userId: user.id,
736
+ email: user.email,
737
+ });
738
+
739
+ return { id: user.id, email: user.email };
740
+ },
741
+ input: undefined,
742
+ output: outputSchema,
743
+ services: [databaseService],
744
+ logger: mockLogger,
745
+ timeout: undefined,
746
+ memorySize: undefined,
747
+ status: 201,
748
+ getSession: undefined,
749
+ authorize: undefined,
750
+ description: undefined,
751
+ events: [],
752
+ publisherService: undefined,
753
+ auditorStorageService: auditStorageService,
754
+ audits: [],
755
+ databaseService,
756
+ });
757
+
758
+ const adaptor = new HonoEndpoint(endpoint);
759
+ const app = new Hono();
760
+ HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
761
+ adaptor.addRoute(serviceDiscovery, app);
762
+
763
+ const response = await app.request('/users', {
764
+ method: 'POST',
765
+ body: JSON.stringify({}),
766
+ headers: { 'Content-Type': 'application/json' },
767
+ });
768
+
769
+ expect(response.status).toBe(201);
770
+ expect(receivedDbIsTransaction).toBe(true);
771
+
772
+ // Verify both user and audit were committed
773
+ const usersInDb = await db.selectFrom(USERS_TABLE).selectAll().execute();
774
+ expect(usersInDb).toHaveLength(1);
775
+ expect(usersInDb[0].email).toBe('trx@example.com');
776
+
777
+ const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
778
+ expect(auditsInDb).toHaveLength(1);
779
+ });
780
+
781
+ it('should use raw db when databaseServiceName does not match', async () => {
782
+ const serviceDiscovery = createServiceDiscovery();
783
+
784
+ // Create audit storage with DIFFERENT databaseServiceName
785
+ const auditStorageWithDifferentServiceName = new KyselyAuditStorage({
786
+ db,
787
+ tableName: AUDIT_TABLE,
788
+ databaseServiceName: 'auditDatabase', // Different from 'database'
789
+ });
790
+
791
+ const databaseService: Service<'database', Kysely<TestDatabase>> = {
792
+ serviceName: 'database' as const,
793
+ register: vi.fn().mockResolvedValue(db),
794
+ };
795
+
796
+ const auditStorageService: Service<
797
+ 'auditStorage',
798
+ KyselyAuditStorage<TestDatabase>
799
+ > = {
800
+ serviceName: 'auditStorage' as const,
801
+ register: vi
802
+ .fn()
803
+ .mockResolvedValue(auditStorageWithDifferentServiceName),
804
+ };
805
+
806
+ const outputSchema = z.object({ id: z.number(), email: z.string() });
807
+
808
+ let receivedDbIsTransaction = false;
809
+
810
+ const endpoint = new Endpoint({
811
+ route: '/users',
812
+ method: 'POST',
813
+ fn: async (
814
+ ctx: EndpointContext<
815
+ undefined,
816
+ [typeof databaseService],
817
+ Logger,
818
+ unknown,
819
+ TestAuditAction,
820
+ Kysely<TestDatabase>,
821
+ KyselyAuditStorage<TestDatabase>
822
+ >,
823
+ ) => {
824
+ // When databaseServiceName doesn't match, db should be raw (not a transaction)
825
+ receivedDbIsTransaction = (ctx.db as any)?.isTransaction === true;
826
+
827
+ // Insert user using ctx.db (should be raw db, not transaction)
828
+ const user = await ctx.db
829
+ .insertInto(USERS_TABLE)
830
+ .values({ name: 'Raw DB User', email: 'raw@example.com' })
831
+ .returningAll()
832
+ .executeTakeFirstOrThrow();
833
+
834
+ // Record audit
835
+ ctx.auditor.audit('user.created', {
836
+ userId: user.id,
837
+ email: user.email,
838
+ });
839
+
840
+ return { id: user.id, email: user.email };
841
+ },
842
+ input: undefined,
843
+ output: outputSchema,
844
+ services: [databaseService],
845
+ logger: mockLogger,
846
+ timeout: undefined,
847
+ memorySize: undefined,
848
+ status: 201,
849
+ getSession: undefined,
850
+ authorize: undefined,
851
+ description: undefined,
852
+ events: [],
853
+ publisherService: undefined,
854
+ auditorStorageService: auditStorageService,
855
+ audits: [],
856
+ databaseService,
857
+ });
858
+
859
+ const adaptor = new HonoEndpoint(endpoint);
860
+ const app = new Hono();
861
+ HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
862
+ adaptor.addRoute(serviceDiscovery, app);
863
+
864
+ const response = await app.request('/users', {
865
+ method: 'POST',
866
+ body: JSON.stringify({}),
867
+ headers: { 'Content-Type': 'application/json' },
868
+ });
869
+
870
+ expect(response.status).toBe(201);
871
+ // db should NOT be a transaction since service names don't match
872
+ expect(receivedDbIsTransaction).toBe(false);
873
+
874
+ // Both should still be committed (but not in the same transaction)
875
+ const usersInDb = await db.selectFrom(USERS_TABLE).selectAll().execute();
876
+ expect(usersInDb).toHaveLength(1);
877
+
878
+ const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
879
+ expect(auditsInDb).toHaveLength(1);
880
+ });
881
+
882
+ it('should use raw db when databaseServiceName is not set on audit storage', async () => {
883
+ const serviceDiscovery = createServiceDiscovery();
884
+
885
+ // Create audit storage WITHOUT databaseServiceName (uses default auditStorage from beforeAll)
886
+ const databaseService: Service<'database', Kysely<TestDatabase>> = {
887
+ serviceName: 'database' as const,
888
+ register: vi.fn().mockResolvedValue(db),
889
+ };
890
+
891
+ const auditStorageService: Service<
892
+ 'auditStorage',
893
+ KyselyAuditStorage<TestDatabase>
894
+ > = {
895
+ serviceName: 'auditStorage' as const,
896
+ register: vi.fn().mockResolvedValue(auditStorage), // No databaseServiceName set
897
+ };
898
+
899
+ const outputSchema = z.object({ id: z.number(), email: z.string() });
900
+
901
+ let receivedDbIsTransaction = false;
902
+
903
+ const endpoint = new Endpoint({
904
+ route: '/users',
905
+ method: 'POST',
906
+ fn: async (
907
+ ctx: EndpointContext<
908
+ undefined,
909
+ [typeof databaseService],
910
+ Logger,
911
+ unknown,
912
+ TestAuditAction,
913
+ Kysely<TestDatabase>,
914
+ KyselyAuditStorage<TestDatabase>
915
+ >,
916
+ ) => {
917
+ // When databaseServiceName is not set, db should be raw
918
+ receivedDbIsTransaction = (ctx.db as any)?.isTransaction === true;
919
+
920
+ const user = await ctx.db
921
+ .insertInto(USERS_TABLE)
922
+ .values({
923
+ name: 'No ServiceName User',
924
+ email: 'noname@example.com',
925
+ })
926
+ .returningAll()
927
+ .executeTakeFirstOrThrow();
928
+
929
+ ctx.auditor.audit('user.created', {
930
+ userId: user.id,
931
+ email: user.email,
932
+ });
933
+
934
+ return { id: user.id, email: user.email };
935
+ },
936
+ input: undefined,
937
+ output: outputSchema,
938
+ services: [databaseService],
939
+ logger: mockLogger,
940
+ timeout: undefined,
941
+ memorySize: undefined,
942
+ status: 201,
943
+ getSession: undefined,
944
+ authorize: undefined,
945
+ description: undefined,
946
+ events: [],
947
+ publisherService: undefined,
948
+ auditorStorageService: auditStorageService,
949
+ audits: [],
950
+ databaseService,
951
+ });
952
+
953
+ const adaptor = new HonoEndpoint(endpoint);
954
+ const app = new Hono();
955
+ HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
956
+ adaptor.addRoute(serviceDiscovery, app);
957
+
958
+ const response = await app.request('/users', {
959
+ method: 'POST',
960
+ body: JSON.stringify({}),
961
+ headers: { 'Content-Type': 'application/json' },
962
+ });
963
+
964
+ expect(response.status).toBe(201);
965
+ expect(receivedDbIsTransaction).toBe(false);
966
+
967
+ const usersInDb = await db.selectFrom(USERS_TABLE).selectAll().execute();
968
+ expect(usersInDb).toHaveLength(1);
969
+
970
+ const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
971
+ expect(auditsInDb).toHaveLength(1);
972
+ });
973
+
974
+ it('should rollback both user insert and audit when handler fails with matching databaseServiceName', async () => {
975
+ const serviceDiscovery = createServiceDiscovery();
976
+
977
+ // Create audit storage WITH databaseServiceName
978
+ const auditStorageWithServiceName = new KyselyAuditStorage({
979
+ db,
980
+ tableName: AUDIT_TABLE,
981
+ databaseServiceName: 'database',
982
+ });
983
+
984
+ const databaseService: Service<'database', Kysely<TestDatabase>> = {
985
+ serviceName: 'database' as const,
986
+ register: vi.fn().mockResolvedValue(db),
987
+ };
988
+
989
+ const auditStorageService: Service<
990
+ 'auditStorage',
991
+ KyselyAuditStorage<TestDatabase>
992
+ > = {
993
+ serviceName: 'auditStorage' as const,
994
+ register: vi.fn().mockResolvedValue(auditStorageWithServiceName),
995
+ };
996
+
997
+ const outputSchema = z.object({ id: z.number(), email: z.string() });
998
+
999
+ const endpoint = new Endpoint({
1000
+ route: '/users',
1001
+ method: 'POST',
1002
+ fn: async (
1003
+ ctx: EndpointContext<
1004
+ undefined,
1005
+ [typeof databaseService],
1006
+ Logger,
1007
+ unknown,
1008
+ TestAuditAction,
1009
+ Kysely<TestDatabase>,
1010
+ KyselyAuditStorage<TestDatabase>
1011
+ >,
1012
+ ) => {
1013
+ // Insert user (should be rolled back)
1014
+ const user = await ctx.db
1015
+ .insertInto(USERS_TABLE)
1016
+ .values({ name: 'Rollback User', email: 'rollback@example.com' })
1017
+ .returningAll()
1018
+ .executeTakeFirstOrThrow();
1019
+
1020
+ // Record audit (should also be rolled back)
1021
+ ctx.auditor.audit('user.created', {
1022
+ userId: user.id,
1023
+ email: user.email,
1024
+ });
1025
+
1026
+ // Fail after both operations
1027
+ throw new Error('Simulated failure');
1028
+ },
1029
+ input: undefined,
1030
+ output: outputSchema,
1031
+ services: [databaseService],
1032
+ logger: mockLogger,
1033
+ timeout: undefined,
1034
+ memorySize: undefined,
1035
+ status: 201,
1036
+ getSession: undefined,
1037
+ authorize: undefined,
1038
+ description: undefined,
1039
+ events: [],
1040
+ publisherService: undefined,
1041
+ auditorStorageService: auditStorageService,
1042
+ audits: [],
1043
+ databaseService,
1044
+ });
1045
+
1046
+ const adaptor = new HonoEndpoint(endpoint);
1047
+ const app = new Hono();
1048
+ HonoEndpoint.applyEventMiddleware(app, serviceDiscovery);
1049
+ adaptor.addRoute(serviceDiscovery, app);
1050
+
1051
+ const response = await app.request('/users', {
1052
+ method: 'POST',
1053
+ body: JSON.stringify({}),
1054
+ headers: { 'Content-Type': 'application/json' },
1055
+ });
1056
+
1057
+ expect(response.status).toBe(500);
1058
+
1059
+ // Both should be rolled back since they're in the same transaction
1060
+ const usersInDb = await db.selectFrom(USERS_TABLE).selectAll().execute();
1061
+ expect(usersInDb).toHaveLength(0);
1062
+
1063
+ const auditsInDb = await db.selectFrom(AUDIT_TABLE).selectAll().execute();
1064
+ expect(auditsInDb).toHaveLength(0);
1065
+ });
1066
+ });
1067
1067
  });