ts-procedures 7.3.0 → 8.0.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 (325) hide show
  1. package/README.md +65 -3
  2. package/agent_config/claude-code/agents/ts-procedures-architect.md +6 -8
  3. package/agent_config/claude-code/skills/ts-procedures/SKILL.md +30 -33
  4. package/agent_config/claude-code/skills/ts-procedures/anti-patterns.md +104 -53
  5. package/agent_config/claude-code/skills/ts-procedures/api-reference.md +205 -232
  6. package/agent_config/claude-code/skills/ts-procedures/patterns.md +80 -153
  7. package/agent_config/claude-code/skills/ts-procedures-review/SKILL.md +1 -1
  8. package/agent_config/claude-code/skills/ts-procedures-review/checklist.md +4 -5
  9. package/agent_config/claude-code/skills/ts-procedures-scaffold/SKILL.md +4 -7
  10. package/agent_config/claude-code/skills/ts-procedures-scaffold/templates/hono.md +223 -0
  11. package/agent_config/copilot/copilot-instructions.md +34 -48
  12. package/agent_config/cursor/cursorrules +34 -48
  13. package/build/client/call.js +4 -1
  14. package/build/client/call.js.map +1 -1
  15. package/build/client/call.test.js +23 -0
  16. package/build/client/call.test.js.map +1 -1
  17. package/build/client/fetch-adapter.js +3 -1
  18. package/build/client/fetch-adapter.js.map +1 -1
  19. package/build/client/fetch-adapter.test.js +11 -1
  20. package/build/client/fetch-adapter.test.js.map +1 -1
  21. package/build/client/index.test.js +7 -7
  22. package/build/client/index.test.js.map +1 -1
  23. package/build/client/request-builder.d.ts +1 -1
  24. package/build/client/request-builder.js +2 -2
  25. package/build/client/request-builder.js.map +1 -1
  26. package/build/client/stream.js +13 -2
  27. package/build/client/stream.js.map +1 -1
  28. package/build/client/stream.test.js +32 -7
  29. package/build/client/stream.test.js.map +1 -1
  30. package/build/client/typed-error-dispatch.test.js +8 -92
  31. package/build/client/typed-error-dispatch.test.js.map +1 -1
  32. package/build/client/types.d.ts +21 -3
  33. package/build/codegen/bin/cli.js +0 -0
  34. package/build/codegen/e2e.test.js +87 -23
  35. package/build/codegen/e2e.test.js.map +1 -1
  36. package/build/codegen/emit-errors.integration.test.js +1 -1
  37. package/build/codegen/emit-errors.integration.test.js.map +1 -1
  38. package/build/codegen/emit-scope.js +308 -47
  39. package/build/codegen/emit-scope.js.map +1 -1
  40. package/build/codegen/emit-scope.test.js +363 -110
  41. package/build/codegen/emit-scope.test.js.map +1 -1
  42. package/build/codegen/pipeline.test.js +7 -7
  43. package/build/codegen/pipeline.test.js.map +1 -1
  44. package/build/codegen/resolve-envelope.js +1 -1
  45. package/build/codegen/resolve-envelope.js.map +1 -1
  46. package/build/codegen/resolve-envelope.test.js +5 -5
  47. package/build/codegen/resolve-envelope.test.js.map +1 -1
  48. package/build/codegen/targets/_shared/route-slots.d.ts +8 -3
  49. package/build/codegen/targets/_shared/route-slots.js +49 -8
  50. package/build/codegen/targets/_shared/route-slots.js.map +1 -1
  51. package/build/codegen/targets/_shared/route-slots.test.js +99 -26
  52. package/build/codegen/targets/_shared/route-slots.test.js.map +1 -1
  53. package/build/codegen/targets/kotlin/emit-route-kotlin.test.js +88 -17
  54. package/build/codegen/targets/kotlin/emit-route-kotlin.test.js.map +1 -1
  55. package/build/codegen/targets/kotlin/emit-scope-kotlin.test.js +9 -6
  56. package/build/codegen/targets/kotlin/emit-scope-kotlin.test.js.map +1 -1
  57. package/build/codegen/targets/kotlin/integration.test.js +6 -0
  58. package/build/codegen/targets/kotlin/integration.test.js.map +1 -1
  59. package/build/codegen/targets/swift/access-level.test.js +8 -11
  60. package/build/codegen/targets/swift/access-level.test.js.map +1 -1
  61. package/build/codegen/targets/swift/emit-route-swift.test.js +91 -20
  62. package/build/codegen/targets/swift/emit-route-swift.test.js.map +1 -1
  63. package/build/codegen/targets/swift/emit-scope-swift.test.js +12 -9
  64. package/build/codegen/targets/swift/emit-scope-swift.test.js.map +1 -1
  65. package/build/codegen/targets/swift/integration.test.js +6 -0
  66. package/build/codegen/targets/swift/integration.test.js.map +1 -1
  67. package/build/create-http-stream.d.ts +58 -0
  68. package/build/create-http-stream.js +122 -0
  69. package/build/create-http-stream.js.map +1 -0
  70. package/build/create-http-stream.test.js +88 -0
  71. package/build/create-http-stream.test.js.map +1 -0
  72. package/build/create-http.d.ts +49 -0
  73. package/build/create-http.js +108 -0
  74. package/build/create-http.js.map +1 -0
  75. package/build/create-http.test.js +137 -0
  76. package/build/create-http.test.js.map +1 -0
  77. package/build/create-stream.d.ts +35 -0
  78. package/build/create-stream.js +123 -0
  79. package/build/create-stream.js.map +1 -0
  80. package/build/create-stream.test.js +428 -0
  81. package/build/create-stream.test.js.map +1 -0
  82. package/build/create.d.ts +28 -0
  83. package/build/create.js +82 -0
  84. package/build/create.js.map +1 -0
  85. package/build/create.test.js +483 -0
  86. package/build/create.test.js.map +1 -0
  87. package/build/exports.d.ts +2 -0
  88. package/build/implementations/http/astro/index.test.js +20 -12
  89. package/build/implementations/http/astro/index.test.js.map +1 -1
  90. package/build/implementations/http/doc-registry.js +1 -1
  91. package/build/implementations/http/doc-registry.js.map +1 -1
  92. package/build/implementations/http/doc-registry.test.js +36 -5
  93. package/build/implementations/http/doc-registry.test.js.map +1 -1
  94. package/build/implementations/http/error-dispatch.d.ts +76 -0
  95. package/build/implementations/http/error-dispatch.js +77 -0
  96. package/build/implementations/http/error-dispatch.js.map +1 -0
  97. package/build/implementations/http/error-dispatch.test.js +254 -0
  98. package/build/implementations/http/error-dispatch.test.js.map +1 -0
  99. package/build/implementations/http/error-taxonomy.d.ts +5 -5
  100. package/build/implementations/http/hono/docs/http-doc.d.ts +6 -0
  101. package/build/implementations/http/hono/docs/http-doc.js +42 -0
  102. package/build/implementations/http/hono/docs/http-doc.js.map +1 -0
  103. package/build/implementations/http/hono/docs/http-stream-doc.d.ts +6 -0
  104. package/build/implementations/http/hono/docs/http-stream-doc.js +40 -0
  105. package/build/implementations/http/hono/docs/http-stream-doc.js.map +1 -0
  106. package/build/implementations/http/hono/docs/rpc-doc.d.ts +6 -0
  107. package/build/implementations/http/hono/docs/rpc-doc.js +24 -0
  108. package/build/implementations/http/hono/docs/rpc-doc.js.map +1 -0
  109. package/build/implementations/http/hono/docs/stream-doc.d.ts +6 -0
  110. package/build/implementations/http/hono/docs/stream-doc.js +42 -0
  111. package/build/implementations/http/hono/docs/stream-doc.js.map +1 -0
  112. package/build/implementations/http/hono/handlers/http-stream.d.ts +10 -0
  113. package/build/implementations/http/hono/handlers/http-stream.js +123 -0
  114. package/build/implementations/http/hono/handlers/http-stream.js.map +1 -0
  115. package/build/implementations/http/hono/handlers/http-stream.test.js +128 -0
  116. package/build/implementations/http/hono/handlers/http-stream.test.js.map +1 -0
  117. package/build/implementations/http/hono/handlers/http.d.ts +10 -0
  118. package/build/implementations/http/hono/handlers/http.js +115 -0
  119. package/build/implementations/http/hono/handlers/http.js.map +1 -0
  120. package/build/implementations/http/hono/handlers/http.test.js +118 -0
  121. package/build/implementations/http/hono/handlers/http.test.js.map +1 -0
  122. package/build/implementations/http/hono/handlers/rpc.d.ts +11 -0
  123. package/build/implementations/http/hono/handlers/rpc.js +32 -0
  124. package/build/implementations/http/hono/handlers/rpc.js.map +1 -0
  125. package/build/implementations/http/hono/handlers/rpc.test.js +73 -0
  126. package/build/implementations/http/hono/handlers/rpc.test.js.map +1 -0
  127. package/build/implementations/http/hono/handlers/stream.d.ts +23 -0
  128. package/build/implementations/http/hono/handlers/stream.js +147 -0
  129. package/build/implementations/http/hono/handlers/stream.js.map +1 -0
  130. package/build/implementations/http/hono/handlers/stream.test.d.ts +1 -0
  131. package/build/implementations/http/hono/handlers/stream.test.js +177 -0
  132. package/build/implementations/http/hono/handlers/stream.test.js.map +1 -0
  133. package/build/implementations/http/hono/index.d.ts +57 -0
  134. package/build/implementations/http/hono/index.js +149 -0
  135. package/build/implementations/http/hono/index.js.map +1 -0
  136. package/build/implementations/http/hono/index.test.d.ts +1 -0
  137. package/build/implementations/http/hono/index.test.js +274 -0
  138. package/build/implementations/http/hono/index.test.js.map +1 -0
  139. package/build/implementations/http/hono/path.d.ts +17 -0
  140. package/build/implementations/http/hono/path.js +39 -0
  141. package/build/implementations/http/hono/path.js.map +1 -0
  142. package/build/implementations/http/hono/path.test.d.ts +1 -0
  143. package/build/implementations/http/hono/path.test.js +83 -0
  144. package/build/implementations/http/hono/path.test.js.map +1 -0
  145. package/build/implementations/http/hono/types.d.ts +51 -0
  146. package/build/implementations/http/hono/types.js.map +1 -0
  147. package/build/implementations/http/on-request-error.test.js +6 -96
  148. package/build/implementations/http/on-request-error.test.js.map +1 -1
  149. package/build/implementations/http/route-errors.test.js +11 -59
  150. package/build/implementations/http/route-errors.test.js.map +1 -1
  151. package/build/implementations/types.d.ts +43 -9
  152. package/build/index.d.ts +124 -124
  153. package/build/index.js +10 -221
  154. package/build/index.js.map +1 -1
  155. package/build/index.test.js +20 -919
  156. package/build/index.test.js.map +1 -1
  157. package/build/migration.test.d.ts +1 -0
  158. package/build/migration.test.js +34 -0
  159. package/build/migration.test.js.map +1 -0
  160. package/build/schema/compute-schema.d.ts +11 -3
  161. package/build/schema/compute-schema.js +13 -7
  162. package/build/schema/compute-schema.js.map +1 -1
  163. package/build/schema/parser.d.ts +11 -3
  164. package/build/schema/parser.js +49 -9
  165. package/build/schema/parser.js.map +1 -1
  166. package/build/stack-utils.js +8 -0
  167. package/build/stack-utils.js.map +1 -1
  168. package/build/types.d.ts +142 -0
  169. package/build/types.js.map +1 -0
  170. package/docs/astro-adapter.md +5 -5
  171. package/docs/core.md +15 -17
  172. package/docs/http-integrations.md +83 -170
  173. package/docs/streaming.md +3 -60
  174. package/docs/superpowers/plans/2026-05-07-astro-adapter.md +2 -7
  175. package/docs/superpowers/plans/2026-05-08-create-http.md +3355 -0
  176. package/docs/superpowers/plans/2026-05-08-hono-app-builder-convergence.md +3365 -0
  177. package/docs/superpowers/specs/2026-05-07-astro-adapter-design.md +1 -3
  178. package/docs/superpowers/specs/2026-05-08-create-http-design.md +409 -0
  179. package/docs/superpowers/specs/2026-05-08-hono-app-builder-convergence-design.md +411 -0
  180. package/package.json +4 -22
  181. package/src/client/call.test.ts +26 -0
  182. package/src/client/call.ts +4 -1
  183. package/src/client/fetch-adapter.test.ts +14 -1
  184. package/src/client/fetch-adapter.ts +3 -1
  185. package/src/client/index.test.ts +7 -7
  186. package/src/client/request-builder.ts +2 -2
  187. package/src/client/stream.test.ts +39 -7
  188. package/src/client/stream.ts +16 -2
  189. package/src/client/typed-error-dispatch.test.ts +7 -97
  190. package/src/client/types.ts +21 -3
  191. package/src/codegen/__fixtures__/users-envelope.json +119 -38
  192. package/src/codegen/e2e.test.ts +98 -24
  193. package/src/codegen/emit-errors.integration.test.ts +1 -1
  194. package/src/codegen/emit-scope.test.ts +395 -110
  195. package/src/codegen/emit-scope.ts +350 -55
  196. package/src/codegen/pipeline.test.ts +7 -7
  197. package/src/codegen/resolve-envelope.test.ts +5 -5
  198. package/src/codegen/resolve-envelope.ts +1 -1
  199. package/src/codegen/targets/_shared/route-slots.test.ts +109 -26
  200. package/src/codegen/targets/_shared/route-slots.ts +48 -11
  201. package/src/codegen/targets/kotlin/__fixtures__/users-golden.kt +73 -0
  202. package/src/codegen/targets/kotlin/emit-route-kotlin.test.ts +100 -17
  203. package/src/codegen/targets/kotlin/emit-scope-kotlin.test.ts +9 -6
  204. package/src/codegen/targets/kotlin/integration.test.ts +19 -0
  205. package/src/codegen/targets/swift/__fixtures__/users-golden.swift +79 -0
  206. package/src/codegen/targets/swift/access-level.test.ts +8 -11
  207. package/src/codegen/targets/swift/emit-route-swift.test.ts +103 -20
  208. package/src/codegen/targets/swift/emit-scope-swift.test.ts +12 -9
  209. package/src/codegen/targets/swift/integration.test.ts +17 -0
  210. package/src/create-http-stream.test.ts +97 -0
  211. package/src/create-http-stream.ts +191 -0
  212. package/src/create-http.test.ts +163 -0
  213. package/src/create-http.ts +211 -0
  214. package/src/create-stream.test.ts +565 -0
  215. package/src/create-stream.ts +228 -0
  216. package/src/create.test.ts +658 -0
  217. package/src/create.ts +172 -0
  218. package/src/exports.ts +2 -0
  219. package/src/implementations/http/README.md +135 -95
  220. package/src/implementations/http/astro/README.md +4 -5
  221. package/src/implementations/http/astro/index.test.ts +25 -18
  222. package/src/implementations/http/doc-registry.test.ts +42 -5
  223. package/src/implementations/http/doc-registry.ts +1 -1
  224. package/src/implementations/http/error-dispatch.test.ts +283 -0
  225. package/src/implementations/http/error-dispatch.ts +176 -0
  226. package/src/implementations/http/error-taxonomy.ts +5 -5
  227. package/src/implementations/http/hono/docs/http-doc.ts +43 -0
  228. package/src/implementations/http/hono/docs/http-stream-doc.ts +44 -0
  229. package/src/implementations/http/hono/docs/rpc-doc.ts +34 -0
  230. package/src/implementations/http/hono/docs/stream-doc.ts +53 -0
  231. package/src/implementations/http/hono/handlers/http-stream.test.ts +150 -0
  232. package/src/implementations/http/hono/handlers/http-stream.ts +152 -0
  233. package/src/implementations/http/hono/handlers/http.test.ts +130 -0
  234. package/src/implementations/http/hono/handlers/http.ts +147 -0
  235. package/src/implementations/http/hono/handlers/rpc.test.ts +81 -0
  236. package/src/implementations/http/hono/handlers/rpc.ts +54 -0
  237. package/src/implementations/http/hono/handlers/stream.test.ts +198 -0
  238. package/src/implementations/http/hono/handlers/stream.ts +208 -0
  239. package/src/implementations/http/hono/index.test.ts +329 -0
  240. package/src/implementations/http/hono/index.ts +204 -0
  241. package/src/implementations/http/hono/path.test.ts +96 -0
  242. package/src/implementations/http/hono/path.ts +59 -0
  243. package/src/implementations/http/hono/types.ts +93 -0
  244. package/src/implementations/http/on-request-error.test.ts +10 -116
  245. package/src/implementations/http/route-errors.test.ts +11 -77
  246. package/src/implementations/types.ts +44 -9
  247. package/src/index.test.ts +22 -1249
  248. package/src/index.ts +49 -485
  249. package/src/migration.test.ts +48 -0
  250. package/src/schema/compute-schema.ts +26 -12
  251. package/src/schema/parser.ts +62 -12
  252. package/src/stack-utils.ts +8 -0
  253. package/src/types.ts +133 -0
  254. package/agent_config/claude-code/skills/ts-procedures-scaffold/templates/express-rpc.md +0 -137
  255. package/agent_config/claude-code/skills/ts-procedures-scaffold/templates/hono-api.md +0 -173
  256. package/agent_config/claude-code/skills/ts-procedures-scaffold/templates/hono-rpc.md +0 -142
  257. package/agent_config/claude-code/skills/ts-procedures-scaffold/templates/hono-stream.md +0 -147
  258. package/build/implementations/http/express-rpc/error-taxonomy.test.js +0 -83
  259. package/build/implementations/http/express-rpc/error-taxonomy.test.js.map +0 -1
  260. package/build/implementations/http/express-rpc/index.d.ts +0 -125
  261. package/build/implementations/http/express-rpc/index.js +0 -216
  262. package/build/implementations/http/express-rpc/index.js.map +0 -1
  263. package/build/implementations/http/express-rpc/index.test.js +0 -684
  264. package/build/implementations/http/express-rpc/index.test.js.map +0 -1
  265. package/build/implementations/http/express-rpc/types.d.ts +0 -11
  266. package/build/implementations/http/express-rpc/types.js.map +0 -1
  267. package/build/implementations/http/hono-api/error-taxonomy.test.js +0 -137
  268. package/build/implementations/http/hono-api/error-taxonomy.test.js.map +0 -1
  269. package/build/implementations/http/hono-api/index.d.ts +0 -151
  270. package/build/implementations/http/hono-api/index.js +0 -344
  271. package/build/implementations/http/hono-api/index.js.map +0 -1
  272. package/build/implementations/http/hono-api/index.test.js +0 -992
  273. package/build/implementations/http/hono-api/index.test.js.map +0 -1
  274. package/build/implementations/http/hono-api/types.d.ts +0 -13
  275. package/build/implementations/http/hono-api/types.js.map +0 -1
  276. package/build/implementations/http/hono-rpc/error-taxonomy.test.js +0 -64
  277. package/build/implementations/http/hono-rpc/error-taxonomy.test.js.map +0 -1
  278. package/build/implementations/http/hono-rpc/index.d.ts +0 -130
  279. package/build/implementations/http/hono-rpc/index.js +0 -209
  280. package/build/implementations/http/hono-rpc/index.js.map +0 -1
  281. package/build/implementations/http/hono-rpc/index.test.js +0 -828
  282. package/build/implementations/http/hono-rpc/index.test.js.map +0 -1
  283. package/build/implementations/http/hono-rpc/types.d.ts +0 -11
  284. package/build/implementations/http/hono-rpc/types.js +0 -2
  285. package/build/implementations/http/hono-rpc/types.js.map +0 -1
  286. package/build/implementations/http/hono-stream/error-taxonomy.test.js +0 -159
  287. package/build/implementations/http/hono-stream/error-taxonomy.test.js.map +0 -1
  288. package/build/implementations/http/hono-stream/index.d.ts +0 -171
  289. package/build/implementations/http/hono-stream/index.js +0 -415
  290. package/build/implementations/http/hono-stream/index.js.map +0 -1
  291. package/build/implementations/http/hono-stream/index.test.js +0 -1383
  292. package/build/implementations/http/hono-stream/index.test.js.map +0 -1
  293. package/build/implementations/http/hono-stream/types.d.ts +0 -15
  294. package/build/implementations/http/hono-stream/types.js +0 -2
  295. package/build/implementations/http/hono-stream/types.js.map +0 -1
  296. package/src/implementations/http/express-rpc/README.md +0 -280
  297. package/src/implementations/http/express-rpc/error-taxonomy.test.ts +0 -103
  298. package/src/implementations/http/express-rpc/index.test.ts +0 -957
  299. package/src/implementations/http/express-rpc/index.ts +0 -327
  300. package/src/implementations/http/express-rpc/types.ts +0 -16
  301. package/src/implementations/http/hono-api/README.md +0 -284
  302. package/src/implementations/http/hono-api/error-taxonomy.test.ts +0 -179
  303. package/src/implementations/http/hono-api/index.test.ts +0 -1341
  304. package/src/implementations/http/hono-api/index.ts +0 -519
  305. package/src/implementations/http/hono-api/types.ts +0 -16
  306. package/src/implementations/http/hono-rpc/README.md +0 -357
  307. package/src/implementations/http/hono-rpc/error-taxonomy.test.ts +0 -82
  308. package/src/implementations/http/hono-rpc/index.test.ts +0 -1107
  309. package/src/implementations/http/hono-rpc/index.ts +0 -320
  310. package/src/implementations/http/hono-rpc/types.ts +0 -16
  311. package/src/implementations/http/hono-stream/README.md +0 -559
  312. package/src/implementations/http/hono-stream/error-taxonomy.test.ts +0 -178
  313. package/src/implementations/http/hono-stream/index.test.ts +0 -1804
  314. package/src/implementations/http/hono-stream/index.ts +0 -622
  315. package/src/implementations/http/hono-stream/types.ts +0 -20
  316. /package/build/{implementations/http/express-rpc/error-taxonomy.test.d.ts → create-http-stream.test.d.ts} +0 -0
  317. /package/build/{implementations/http/express-rpc/index.test.d.ts → create-http.test.d.ts} +0 -0
  318. /package/build/{implementations/http/hono-api/error-taxonomy.test.d.ts → create-stream.test.d.ts} +0 -0
  319. /package/build/{implementations/http/hono-api/index.test.d.ts → create.test.d.ts} +0 -0
  320. /package/build/implementations/http/{hono-rpc/error-taxonomy.test.d.ts → error-dispatch.test.d.ts} +0 -0
  321. /package/build/implementations/http/{hono-rpc/index.test.d.ts → hono/handlers/http-stream.test.d.ts} +0 -0
  322. /package/build/implementations/http/{hono-stream/error-taxonomy.test.d.ts → hono/handlers/http.test.d.ts} +0 -0
  323. /package/build/implementations/http/{hono-stream/index.test.d.ts → hono/handlers/rpc.test.d.ts} +0 -0
  324. /package/build/implementations/http/{express-rpc → hono}/types.js +0 -0
  325. /package/build/{implementations/http/hono-api/types.js → types.js} +0 -0
@@ -0,0 +1,147 @@
1
+ import { streamSSE, streamText } from 'hono/streaming';
2
+ import { ProcedureValidationError } from '../../../../errors.js';
3
+ import { dispatchMidStreamError, dispatchPreStreamError } from '../../error-dispatch.js';
4
+ import { buildStreamRouteDoc } from '../docs/stream-doc.js';
5
+ const sseMetadata = new WeakMap();
6
+ /**
7
+ * Marks an object yield as an SSE event with custom metadata. Stream handlers
8
+ * read this metadata to set the SSE `event:`, `id:`, and `retry:` fields.
9
+ */
10
+ export function sse(data, options) {
11
+ sseMetadata.set(data, options ?? {});
12
+ return data;
13
+ }
14
+ function getSSEMeta(value) {
15
+ if (typeof value === 'object' && value !== null) {
16
+ return sseMetadata.get(value);
17
+ }
18
+ return undefined;
19
+ }
20
+ export function installRpcStreamRoute(params) {
21
+ const { app, procedure, factoryItem, cfg, docs, streamMode } = params;
22
+ const route = buildStreamRouteDoc(procedure, streamMode, cfg.pathPrefix, factoryItem.extendProcedureDoc);
23
+ docs.push(route);
24
+ const handler = async (c) => {
25
+ try {
26
+ const context = typeof factoryItem.factoryContext === 'function'
27
+ ? await factoryItem.factoryContext(c)
28
+ : factoryItem.factoryContext;
29
+ const reqParams = c.req.method === 'GET'
30
+ ? Object.fromEntries(new URL(c.req.url).searchParams)
31
+ : await c.req.json().catch(() => ({}));
32
+ // Pre-stream validation — throw so the catch routes through dispatchPreStreamError.
33
+ if (procedure.config.validation?.params) {
34
+ const { errors } = procedure.config.validation.params(reqParams);
35
+ if (errors) {
36
+ throw new ProcedureValidationError(procedure.name, `Validation error for ${procedure.name}`, errors);
37
+ }
38
+ }
39
+ cfg.stream?.onStreamStart?.(procedure, c, streamMode);
40
+ return streamMode === 'sse'
41
+ ? handleSSE(procedure, context, reqParams, c, cfg)
42
+ : handleText(procedure, context, reqParams, c, cfg);
43
+ }
44
+ catch (error) {
45
+ return dispatchPreStreamError({
46
+ err: error,
47
+ procedure,
48
+ raw: c,
49
+ cfg: {
50
+ errors: cfg.errors,
51
+ unknownError: cfg.unknownError,
52
+ onError: cfg.onError,
53
+ onRequestError: cfg.onRequestError,
54
+ },
55
+ });
56
+ }
57
+ };
58
+ app.get(route.path, handler);
59
+ app.post(route.path, handler);
60
+ }
61
+ function handleSSE(procedure, context, reqParams, c, cfg) {
62
+ return streamSSE(c, async (stream) => {
63
+ const generator = procedure.handler({ ...context, signal: c.req.raw.signal, isPrevalidated: true }, reqParams);
64
+ stream.onAbort(async () => { await generator.return(undefined); });
65
+ let eventId = 0;
66
+ try {
67
+ const iterator = generator[Symbol.asyncIterator]();
68
+ let it = await iterator.next();
69
+ while (!it.done) {
70
+ const value = it.value;
71
+ const meta = getSSEMeta(value);
72
+ const data = typeof value === 'string' ? value
73
+ : value != null ? JSON.stringify(value)
74
+ : '';
75
+ await stream.writeSSE({
76
+ data,
77
+ event: meta?.event ?? procedure.name,
78
+ id: meta?.id ?? String(eventId++),
79
+ ...(meta?.retry !== undefined && { retry: meta.retry }),
80
+ });
81
+ it = await iterator.next();
82
+ }
83
+ if (it.value !== undefined) {
84
+ const data = typeof it.value === 'string' ? it.value : JSON.stringify(it.value);
85
+ await stream.writeSSE({ data, event: 'return', id: String(eventId++) });
86
+ }
87
+ }
88
+ catch (error) {
89
+ const dispatched = await dispatchMidStreamError({
90
+ err: error,
91
+ procedure,
92
+ raw: c,
93
+ cfg: {
94
+ errors: cfg.errors,
95
+ unknownError: cfg.unknownError,
96
+ onMidStreamError: cfg.stream?.onMidStreamError,
97
+ onRequestError: cfg.onRequestError,
98
+ },
99
+ });
100
+ const meta = getSSEMeta(dispatched.data);
101
+ await stream.writeSSE({
102
+ data: typeof dispatched.data === 'string' ? dispatched.data : JSON.stringify(dispatched.data),
103
+ event: meta?.event ?? dispatched.sseEvent ?? 'error',
104
+ id: meta?.id ?? String(eventId++),
105
+ ...(meta?.retry !== undefined && { retry: meta.retry }),
106
+ });
107
+ if (dispatched.runOnCatch)
108
+ await dispatched.runOnCatch();
109
+ }
110
+ finally {
111
+ cfg.stream?.onStreamEnd?.(procedure, c, 'sse');
112
+ cfg.onRequestEnd?.(c);
113
+ }
114
+ });
115
+ }
116
+ function handleText(procedure, context, reqParams, c, cfg) {
117
+ return streamText(c, async (stream) => {
118
+ const generator = procedure.handler({ ...context, signal: c.req.raw.signal, isPrevalidated: true }, reqParams);
119
+ stream.onAbort(async () => { await generator.return(undefined); });
120
+ try {
121
+ for await (const value of generator) {
122
+ await stream.writeln(JSON.stringify(value));
123
+ }
124
+ }
125
+ catch (error) {
126
+ const dispatched = await dispatchMidStreamError({
127
+ err: error,
128
+ procedure,
129
+ raw: c,
130
+ cfg: {
131
+ errors: cfg.errors,
132
+ unknownError: cfg.unknownError,
133
+ onMidStreamError: cfg.stream?.onMidStreamError,
134
+ onRequestError: cfg.onRequestError,
135
+ },
136
+ });
137
+ await stream.writeln(JSON.stringify(dispatched.data));
138
+ if (dispatched.runOnCatch)
139
+ await dispatched.runOnCatch();
140
+ }
141
+ finally {
142
+ cfg.stream?.onStreamEnd?.(procedure, c, 'text');
143
+ cfg.onRequestEnd?.(c);
144
+ }
145
+ });
146
+ }
147
+ //# sourceMappingURL=stream.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stream.js","sourceRoot":"","sources":["../../../../../src/implementations/http/hono/handlers/stream.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,177 @@
1
+ import { describe, expect, test, vi } from 'vitest';
2
+ import { Hono } from 'hono';
3
+ import { Type } from 'typebox';
4
+ import { Procedures } from '../../../../index.js';
5
+ import { installRpcStreamRoute, sse } from './stream.js';
6
+ function buildApp(setup, cfg = {}) {
7
+ const P = Procedures();
8
+ setup(P);
9
+ const app = new Hono();
10
+ const docs = [];
11
+ for (const proc of P.getProcedures().values()) {
12
+ if (proc.kind !== 'rpc-stream')
13
+ continue;
14
+ installRpcStreamRoute({
15
+ app,
16
+ procedure: proc,
17
+ factoryItem: { factory: P, factoryContext: () => ({ uid: 'u1' }) },
18
+ cfg,
19
+ docs,
20
+ streamMode: cfg.stream?.defaultStreamMode ?? 'sse',
21
+ });
22
+ }
23
+ return { app, docs };
24
+ }
25
+ describe('installRpcStreamRoute', () => {
26
+ test('SSE: yields are written as data events; final return as event:return', async () => {
27
+ const { app, docs } = buildApp((P) => {
28
+ P.CreateStream('Counts', { scope: 'c', version: 1 }, async function* () {
29
+ yield 1;
30
+ yield 2;
31
+ return 'done';
32
+ });
33
+ });
34
+ expect(docs[0].kind).toBe('stream');
35
+ expect(docs[0].path).toBe('/c/counts/1');
36
+ const res = await app.request('/c/counts/1', { method: 'POST', body: '{}', headers: { 'Content-Type': 'application/json' } });
37
+ expect(res.status).toBe(200);
38
+ const text = await res.text();
39
+ expect(text).toContain('data: 1');
40
+ expect(text).toContain('data: 2');
41
+ expect(text).toContain('event: return');
42
+ expect(text).toContain('data: done');
43
+ });
44
+ test('mid-stream throw → dispatchMidStreamError default { error }', async () => {
45
+ const { app } = buildApp((P) => {
46
+ P.CreateStream('Boom', { scope: 'b', version: 1 }, async function* () {
47
+ yield 'first';
48
+ throw new Error('mid-boom');
49
+ });
50
+ });
51
+ const res = await app.request('/b/boom/1', { method: 'POST', body: '{}', headers: { 'Content-Type': 'application/json' } });
52
+ const text = await res.text();
53
+ expect(text).toContain('event: error');
54
+ expect(text).toContain('"error":"mid-boom"');
55
+ });
56
+ test('pre-stream validation error → 400 via dispatchPreStreamError', async () => {
57
+ const { app } = buildApp((P) => {
58
+ P.CreateStream('NeedsParams', {
59
+ scope: 'n', version: 1,
60
+ schema: { params: Type.Object({ id: Type.String() }) },
61
+ }, async function* () { yield 'ok'; });
62
+ });
63
+ const res = await app.request('/n/needs-params/1', { method: 'POST', body: '{}', headers: { 'Content-Type': 'application/json' } });
64
+ expect(res.status).toBe(400);
65
+ const body = await res.json();
66
+ expect(body.name).toBe('ProcedureValidationError');
67
+ });
68
+ test('onStreamStart and onStreamEnd fire', async () => {
69
+ const onStreamStart = vi.fn();
70
+ const onStreamEnd = vi.fn();
71
+ const { app } = buildApp((P) => {
72
+ P.CreateStream('Ev', { scope: 'e', version: 1 }, async function* () { yield 'x'; });
73
+ }, { stream: { onStreamStart, onStreamEnd } });
74
+ const res = await app.request('/e/ev/1', { method: 'POST', body: '{}', headers: { 'Content-Type': 'application/json' } });
75
+ // Consume the stream body so the SSE writer's finally block runs
76
+ await res.text();
77
+ expect(onStreamStart).toHaveBeenCalled();
78
+ expect(onStreamEnd).toHaveBeenCalled();
79
+ });
80
+ test('validateYields: true → invalid yield surfaces ProcedureYieldValidationError on the wire', async () => {
81
+ const { app } = buildApp((P) => {
82
+ P.CreateStream('BadYield', {
83
+ scope: 'y', version: 1,
84
+ schema: { yieldType: Type.Object({ id: Type.Number() }) },
85
+ validateYields: true,
86
+ }, async function* () {
87
+ yield { id: 1 }; // valid
88
+ yield { id: 'not-a-number' }; // invalid → throws ProcedureYieldValidationError
89
+ });
90
+ });
91
+ const res = await app.request('/y/bad-yield/1', {
92
+ method: 'POST',
93
+ body: '{}',
94
+ headers: { 'Content-Type': 'application/json' },
95
+ });
96
+ // Stream is already open by the time validation fails on the second yield;
97
+ // status is committed at 200 and the error shows up as an SSE error event.
98
+ expect(res.status).toBe(200);
99
+ const text = await res.text();
100
+ expect(text).toContain('event: error');
101
+ expect(text).toContain('"name":"ProcedureYieldValidationError"');
102
+ expect(text).toContain('"procedureName":"BadYield"');
103
+ // First valid yield still made it onto the wire.
104
+ expect(text).toContain('data: {"id":1}');
105
+ });
106
+ test('sse() helper propagates event/id/retry into SSE wire output', async () => {
107
+ const { app } = buildApp((P) => {
108
+ P.CreateStream('Tagged', { scope: 'tagged', version: 1 }, async function* () {
109
+ yield sse({ count: 1 }, { event: 'tick', id: 'evt-1', retry: 5000 });
110
+ });
111
+ });
112
+ const res = await app.request('/tagged/tagged/1', {
113
+ method: 'POST',
114
+ body: '{}',
115
+ headers: { 'Content-Type': 'application/json' },
116
+ });
117
+ const text = await res.text();
118
+ expect(text).toContain('event: tick');
119
+ expect(text).toContain('id: evt-1');
120
+ expect(text).toContain('retry: 5000');
121
+ expect(text).toContain('data: {"count":1}');
122
+ });
123
+ test('AbortSignal: natural completion → ctx.signal.reason === "stream-completed"', async () => {
124
+ let capturedSignal;
125
+ const { app } = buildApp((P) => {
126
+ P.CreateStream('Natural', { scope: 'sig', version: 1 }, async function* (ctx) {
127
+ capturedSignal = ctx.signal;
128
+ yield 'a';
129
+ yield 'b';
130
+ return 'done';
131
+ });
132
+ });
133
+ const res = await app.request('/sig/natural/1', {
134
+ method: 'POST',
135
+ body: '{}',
136
+ headers: { 'Content-Type': 'application/json' },
137
+ });
138
+ // Drain the body so the wrappedHandler's finally block runs.
139
+ await res.text();
140
+ expect(capturedSignal).toBeInstanceOf(AbortSignal);
141
+ expect(capturedSignal.aborted).toBe(true);
142
+ expect(capturedSignal.reason).toBe('stream-completed');
143
+ });
144
+ test('AbortSignal: client disconnect (request signal aborted) → ctx.signal.reason !== "stream-completed"', async () => {
145
+ // NOTE: We can't directly simulate a real socket disconnect through
146
+ // `app.request()`. The closest test-friendly proxy: pre-abort the request
147
+ // signal so `c.req.raw.signal` is already aborted by the time the
148
+ // wrappedHandler in create-stream.ts builds the combined signal via
149
+ // `AbortSignal.any([incomingSignal, internalController.signal])`. The
150
+ // pre-existing reason wins over the internal 'stream-completed' that fires
151
+ // later in the wrappedHandler's finally clause.
152
+ let capturedSignal;
153
+ const { app } = buildApp((P) => {
154
+ P.CreateStream('Disconnect', { scope: 'sig', version: 1 }, async function* (ctx) {
155
+ capturedSignal = ctx.signal;
156
+ yield 'first';
157
+ });
158
+ });
159
+ const ac = new AbortController();
160
+ ac.abort('client-gone');
161
+ const req = new Request('http://localhost/sig/disconnect/1', {
162
+ method: 'POST',
163
+ body: '{}',
164
+ headers: { 'Content-Type': 'application/json' },
165
+ signal: ac.signal,
166
+ });
167
+ const res = await app.request(req);
168
+ // Drain the body in case anything was emitted before the abort propagated.
169
+ await res.text().catch(() => '');
170
+ expect(capturedSignal).toBeInstanceOf(AbortSignal);
171
+ expect(capturedSignal.aborted).toBe(true);
172
+ // The disconnect reason wins over the internal 'stream-completed' signal,
173
+ // so handlers can distinguish "client gave up" from "we finished normally".
174
+ expect(capturedSignal.reason).not.toBe('stream-completed');
175
+ });
176
+ });
177
+ //# sourceMappingURL=stream.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stream.test.js","sourceRoot":"","sources":["../../../../../src/implementations/http/hono/handlers/stream.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,57 @@
1
+ import { Context, Hono } from 'hono';
2
+ import type { AnyHttpRouteDoc, ProceduresFactory, ExtractContext, StreamMode, DocEnvelope, ErrorDoc, HeaderDoc } from '../../types.js';
3
+ import type { ErrorTaxonomy } from '../error-taxonomy.js';
4
+ import { makeRoutePath as _makeRoutePath } from './path.js';
5
+ import type { HonoAppBuilderConfig, ExtendProcedureDoc } from './types.js';
6
+ export type { HonoAppBuilderConfig, OnRequestErrorContext, ExtendProcedureDoc } from './types.js';
7
+ export type { AnyProcedureRegistration } from './types.js';
8
+ export { sse } from './handlers/stream.js';
9
+ export type { SSEOptions, MidStreamErrorResult } from './handlers/stream.js';
10
+ export { defineErrorTaxonomy } from '../error-taxonomy.js';
11
+ export type { ErrorTaxonomy, ErrorTaxonomyEntry, UnknownErrorConfig } from '../error-taxonomy.js';
12
+ export type { QueryParser } from './types.js';
13
+ export declare class HonoAppBuilder<TStreamErrorData = unknown> {
14
+ readonly config?: HonoAppBuilderConfig<TStreamErrorData> | undefined;
15
+ private readonly _app;
16
+ private readonly factories;
17
+ private _docs;
18
+ private _skipped;
19
+ private _built;
20
+ constructor(config?: HonoAppBuilderConfig<TStreamErrorData> | undefined);
21
+ static makeRoutePath: typeof _makeRoutePath;
22
+ get app(): Hono;
23
+ /**
24
+ * Lazily computed on first read or `build()`. Computing without `build()`
25
+ * lets tests and tooling introspect routes without spinning up handlers.
26
+ */
27
+ get docs(): AnyHttpRouteDoc[];
28
+ get skippedProcedures(): {
29
+ name: string;
30
+ reason: string;
31
+ }[];
32
+ register<TFactory extends ProceduresFactory>(factory: TFactory, factoryContext: ExtractContext<TFactory> | ((c: Context) => ExtractContext<TFactory> | Promise<ExtractContext<TFactory>>), options?: {
33
+ streamMode?: StreamMode;
34
+ extendProcedureDoc?: ExtendProcedureDoc;
35
+ }): this;
36
+ private computeDocs;
37
+ /**
38
+ * Mounts every registered procedure on the Hono app and returns it.
39
+ * Idempotent: calling twice is a no-op on the second call.
40
+ */
41
+ build(): Hono;
42
+ /**
43
+ * Produces a {@link DocEnvelope} for single-app usage. For multi-app
44
+ * aggregation, use {@link DocRegistry} and `from(builder)` instead.
45
+ *
46
+ * Mirrors `DocRegistry.toJSON()` options so the two paths produce
47
+ * interchangeable envelopes for the codegen pipeline.
48
+ */
49
+ toDocEnvelope<T = DocEnvelope>(options?: {
50
+ basePath?: string;
51
+ errors?: ErrorTaxonomy | ErrorDoc[];
52
+ includeDefaults?: boolean;
53
+ headers?: HeaderDoc[];
54
+ filter?: (route: AnyHttpRouteDoc) => boolean;
55
+ transform?: (envelope: DocEnvelope) => T;
56
+ }): T;
57
+ }
@@ -0,0 +1,149 @@
1
+ import { Hono } from 'hono';
2
+ import { DocRegistry } from '../doc-registry.js';
3
+ import { buildRpcRouteDoc } from './docs/rpc-doc.js';
4
+ import { buildStreamRouteDoc } from './docs/stream-doc.js';
5
+ import { buildHttpRouteDoc } from './docs/http-doc.js';
6
+ import { buildHttpStreamRouteDoc } from './docs/http-stream-doc.js';
7
+ import { installRpcRoute } from './handlers/rpc.js';
8
+ import { installRpcStreamRoute } from './handlers/stream.js';
9
+ import { installHttpRoute } from './handlers/http.js';
10
+ import { installHttpStreamRoute } from './handlers/http-stream.js';
11
+ import { makeRoutePath as _makeRoutePath } from './path.js';
12
+ export { sse } from './handlers/stream.js';
13
+ export { defineErrorTaxonomy } from '../error-taxonomy.js';
14
+ export class HonoAppBuilder {
15
+ config;
16
+ _app;
17
+ factories = [];
18
+ _docs = null;
19
+ _skipped = [];
20
+ _built = false;
21
+ constructor(config) {
22
+ this.config = config;
23
+ this._app = config?.app ?? new Hono();
24
+ if (config?.onRequestStart) {
25
+ this._app.use('*', async (c, next) => {
26
+ config.onRequestStart(c);
27
+ await next();
28
+ });
29
+ }
30
+ if (config?.onRequestEnd) {
31
+ this._app.use('*', async (c, next) => {
32
+ await next();
33
+ config.onRequestEnd(c);
34
+ });
35
+ }
36
+ }
37
+ static makeRoutePath = _makeRoutePath;
38
+ get app() {
39
+ return this._app;
40
+ }
41
+ /**
42
+ * Lazily computed on first read or `build()`. Computing without `build()`
43
+ * lets tests and tooling introspect routes without spinning up handlers.
44
+ */
45
+ get docs() {
46
+ if (this._docs === null)
47
+ this.computeDocs();
48
+ return this._docs;
49
+ }
50
+ get skippedProcedures() {
51
+ return this._skipped;
52
+ }
53
+ register(factory, factoryContext, options) {
54
+ this.factories.push({
55
+ factory,
56
+ factoryContext,
57
+ streamMode: options?.streamMode,
58
+ extendProcedureDoc: options?.extendProcedureDoc,
59
+ });
60
+ // Invalidate docs cache so the next read recomputes.
61
+ this._docs = null;
62
+ return this;
63
+ }
64
+ computeDocs() {
65
+ const docs = [];
66
+ const skipped = [];
67
+ for (const item of this.factories) {
68
+ for (const procedure of item.factory.getProcedures().values()) {
69
+ const prefix = this.config?.pathPrefix;
70
+ switch (procedure.kind) {
71
+ case 'rpc':
72
+ docs.push(buildRpcRouteDoc(procedure, prefix, item.extendProcedureDoc));
73
+ break;
74
+ case 'rpc-stream':
75
+ docs.push(buildStreamRouteDoc(procedure, item.streamMode ?? this.config?.stream?.defaultStreamMode ?? 'sse', prefix, item.extendProcedureDoc));
76
+ break;
77
+ case 'http':
78
+ docs.push(buildHttpRouteDoc(procedure, prefix, item.extendProcedureDoc));
79
+ break;
80
+ case 'http-stream':
81
+ docs.push(buildHttpStreamRouteDoc(procedure, prefix, item.extendProcedureDoc));
82
+ break;
83
+ default: {
84
+ const reason = `Unknown procedure kind "${procedure.kind}"`;
85
+ skipped.push({ name: procedure.name, reason });
86
+ }
87
+ }
88
+ }
89
+ }
90
+ this._docs = docs;
91
+ this._skipped = skipped;
92
+ }
93
+ /**
94
+ * Mounts every registered procedure on the Hono app and returns it.
95
+ * Idempotent: calling twice is a no-op on the second call.
96
+ */
97
+ build() {
98
+ if (this._built)
99
+ return this._app;
100
+ this._built = true;
101
+ const docs = [];
102
+ const skipped = [];
103
+ const cfg = this.config ?? {};
104
+ for (const item of this.factories) {
105
+ for (const procedure of item.factory.getProcedures().values()) {
106
+ switch (procedure.kind) {
107
+ case 'rpc':
108
+ installRpcRoute({ app: this._app, procedure: procedure, factoryItem: item, cfg, docs });
109
+ break;
110
+ case 'rpc-stream': {
111
+ const streamMode = item.streamMode ?? cfg.stream?.defaultStreamMode ?? 'sse';
112
+ installRpcStreamRoute({ app: this._app, procedure: procedure, factoryItem: item, cfg, docs, streamMode });
113
+ break;
114
+ }
115
+ case 'http':
116
+ installHttpRoute({ app: this._app, procedure: procedure, factoryItem: item, cfg, docs });
117
+ break;
118
+ case 'http-stream':
119
+ installHttpStreamRoute({ app: this._app, procedure: procedure, factoryItem: item, cfg, docs });
120
+ break;
121
+ default: {
122
+ const reason = `Unknown procedure kind "${procedure.kind}"`;
123
+ skipped.push({ name: procedure.name, reason });
124
+ console.warn(`[ts-procedures hono] Skipping procedure "${procedure.name}": ${reason}`);
125
+ }
126
+ }
127
+ }
128
+ }
129
+ this._docs = docs;
130
+ this._skipped = skipped;
131
+ return this._app;
132
+ }
133
+ /**
134
+ * Produces a {@link DocEnvelope} for single-app usage. For multi-app
135
+ * aggregation, use {@link DocRegistry} and `from(builder)` instead.
136
+ *
137
+ * Mirrors `DocRegistry.toJSON()` options so the two paths produce
138
+ * interchangeable envelopes for the codegen pipeline.
139
+ */
140
+ toDocEnvelope(options) {
141
+ return new DocRegistry({
142
+ basePath: options?.basePath ?? this.config?.pathPrefix,
143
+ errors: options?.errors ?? this.config?.errors,
144
+ includeDefaults: options?.includeDefaults,
145
+ headers: options?.headers,
146
+ }).from(this).toJSON(options);
147
+ }
148
+ }
149
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/implementations/http/hono/index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};