@copilotkit/aimock 1.7.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 (368) hide show
  1. package/.claude-plugin/marketplace.json +17 -0
  2. package/.claude-plugin/plugin.json +12 -0
  3. package/LICENSE +21 -0
  4. package/README.md +82 -0
  5. package/dist/_virtual/_rolldown/runtime.cjs +29 -0
  6. package/dist/a2a-handler.cjs +203 -0
  7. package/dist/a2a-handler.cjs.map +1 -0
  8. package/dist/a2a-handler.js +199 -0
  9. package/dist/a2a-handler.js.map +1 -0
  10. package/dist/a2a-mock.cjs +292 -0
  11. package/dist/a2a-mock.cjs.map +1 -0
  12. package/dist/a2a-mock.d.cts +41 -0
  13. package/dist/a2a-mock.d.cts.map +1 -0
  14. package/dist/a2a-mock.d.ts +41 -0
  15. package/dist/a2a-mock.d.ts.map +1 -0
  16. package/dist/a2a-mock.js +290 -0
  17. package/dist/a2a-mock.js.map +1 -0
  18. package/dist/a2a-stub.cjs +4 -0
  19. package/dist/a2a-stub.d.cts +3 -0
  20. package/dist/a2a-stub.d.ts +3 -0
  21. package/dist/a2a-stub.js +3 -0
  22. package/dist/a2a-types.d.cts +68 -0
  23. package/dist/a2a-types.d.cts.map +1 -0
  24. package/dist/a2a-types.d.ts +68 -0
  25. package/dist/a2a-types.d.ts.map +1 -0
  26. package/dist/aimock-cli.cjs +112 -0
  27. package/dist/aimock-cli.cjs.map +1 -0
  28. package/dist/aimock-cli.d.cts +19 -0
  29. package/dist/aimock-cli.d.cts.map +1 -0
  30. package/dist/aimock-cli.d.ts +19 -0
  31. package/dist/aimock-cli.d.ts.map +1 -0
  32. package/dist/aimock-cli.js +110 -0
  33. package/dist/aimock-cli.js.map +1 -0
  34. package/dist/aws-event-stream.cjs +117 -0
  35. package/dist/aws-event-stream.cjs.map +1 -0
  36. package/dist/aws-event-stream.d.cts +38 -0
  37. package/dist/aws-event-stream.d.cts.map +1 -0
  38. package/dist/aws-event-stream.d.ts +38 -0
  39. package/dist/aws-event-stream.d.ts.map +1 -0
  40. package/dist/aws-event-stream.js +114 -0
  41. package/dist/aws-event-stream.js.map +1 -0
  42. package/dist/bedrock-converse.cjs +445 -0
  43. package/dist/bedrock-converse.cjs.map +1 -0
  44. package/dist/bedrock-converse.d.cts +50 -0
  45. package/dist/bedrock-converse.d.cts.map +1 -0
  46. package/dist/bedrock-converse.d.ts +50 -0
  47. package/dist/bedrock-converse.d.ts.map +1 -0
  48. package/dist/bedrock-converse.js +443 -0
  49. package/dist/bedrock-converse.js.map +1 -0
  50. package/dist/bedrock.cjs +557 -0
  51. package/dist/bedrock.cjs.map +1 -0
  52. package/dist/bedrock.d.cts +41 -0
  53. package/dist/bedrock.d.cts.map +1 -0
  54. package/dist/bedrock.d.ts +41 -0
  55. package/dist/bedrock.d.ts.map +1 -0
  56. package/dist/bedrock.js +553 -0
  57. package/dist/bedrock.js.map +1 -0
  58. package/dist/chaos.cjs +114 -0
  59. package/dist/chaos.cjs.map +1 -0
  60. package/dist/chaos.d.cts +27 -0
  61. package/dist/chaos.d.cts.map +1 -0
  62. package/dist/chaos.d.ts +27 -0
  63. package/dist/chaos.d.ts.map +1 -0
  64. package/dist/chaos.js +113 -0
  65. package/dist/chaos.js.map +1 -0
  66. package/dist/cli.cjs +268 -0
  67. package/dist/cli.cjs.map +1 -0
  68. package/dist/cli.d.cts +1 -0
  69. package/dist/cli.d.ts +1 -0
  70. package/dist/cli.js +268 -0
  71. package/dist/cli.js.map +1 -0
  72. package/dist/cohere.cjs +434 -0
  73. package/dist/cohere.cjs.map +1 -0
  74. package/dist/cohere.d.cts +34 -0
  75. package/dist/cohere.d.cts.map +1 -0
  76. package/dist/cohere.d.ts +34 -0
  77. package/dist/cohere.d.ts.map +1 -0
  78. package/dist/cohere.js +433 -0
  79. package/dist/cohere.js.map +1 -0
  80. package/dist/config-loader.cjs +111 -0
  81. package/dist/config-loader.cjs.map +1 -0
  82. package/dist/config-loader.d.cts +100 -0
  83. package/dist/config-loader.d.cts.map +1 -0
  84. package/dist/config-loader.d.ts +100 -0
  85. package/dist/config-loader.d.ts.map +1 -0
  86. package/dist/config-loader.js +107 -0
  87. package/dist/config-loader.js.map +1 -0
  88. package/dist/embeddings.cjs +150 -0
  89. package/dist/embeddings.cjs.map +1 -0
  90. package/dist/embeddings.d.cts +12 -0
  91. package/dist/embeddings.d.cts.map +1 -0
  92. package/dist/embeddings.d.ts +12 -0
  93. package/dist/embeddings.d.ts.map +1 -0
  94. package/dist/embeddings.js +150 -0
  95. package/dist/embeddings.js.map +1 -0
  96. package/dist/fixture-loader.cjs +269 -0
  97. package/dist/fixture-loader.cjs.map +1 -0
  98. package/dist/fixture-loader.d.cts +17 -0
  99. package/dist/fixture-loader.d.cts.map +1 -0
  100. package/dist/fixture-loader.d.ts +17 -0
  101. package/dist/fixture-loader.d.ts.map +1 -0
  102. package/dist/fixture-loader.js +265 -0
  103. package/dist/fixture-loader.js.map +1 -0
  104. package/dist/gemini.cjs +403 -0
  105. package/dist/gemini.cjs.map +1 -0
  106. package/dist/gemini.d.cts +10 -0
  107. package/dist/gemini.d.cts.map +1 -0
  108. package/dist/gemini.d.ts +10 -0
  109. package/dist/gemini.d.ts.map +1 -0
  110. package/dist/gemini.js +403 -0
  111. package/dist/gemini.js.map +1 -0
  112. package/dist/helpers.cjs +276 -0
  113. package/dist/helpers.cjs.map +1 -0
  114. package/dist/helpers.d.cts +39 -0
  115. package/dist/helpers.d.cts.map +1 -0
  116. package/dist/helpers.d.ts +39 -0
  117. package/dist/helpers.d.ts.map +1 -0
  118. package/dist/helpers.js +259 -0
  119. package/dist/helpers.js.map +1 -0
  120. package/dist/index.cjs +113 -0
  121. package/dist/index.d.cts +42 -0
  122. package/dist/index.d.ts +42 -0
  123. package/dist/index.js +39 -0
  124. package/dist/interruption.cjs +40 -0
  125. package/dist/interruption.cjs.map +1 -0
  126. package/dist/interruption.d.cts +15 -0
  127. package/dist/interruption.d.cts.map +1 -0
  128. package/dist/interruption.d.ts +15 -0
  129. package/dist/interruption.d.ts.map +1 -0
  130. package/dist/interruption.js +39 -0
  131. package/dist/interruption.js.map +1 -0
  132. package/dist/journal.cjs +65 -0
  133. package/dist/journal.cjs.map +1 -0
  134. package/dist/journal.d.cts +23 -0
  135. package/dist/journal.d.cts.map +1 -0
  136. package/dist/journal.d.ts +23 -0
  137. package/dist/journal.d.ts.map +1 -0
  138. package/dist/journal.js +65 -0
  139. package/dist/journal.js.map +1 -0
  140. package/dist/jsonrpc.cjs +91 -0
  141. package/dist/jsonrpc.cjs.map +1 -0
  142. package/dist/jsonrpc.d.cts +24 -0
  143. package/dist/jsonrpc.d.cts.map +1 -0
  144. package/dist/jsonrpc.d.ts +24 -0
  145. package/dist/jsonrpc.d.ts.map +1 -0
  146. package/dist/jsonrpc.js +90 -0
  147. package/dist/jsonrpc.js.map +1 -0
  148. package/dist/llmock.cjs +223 -0
  149. package/dist/llmock.cjs.map +1 -0
  150. package/dist/llmock.d.cts +70 -0
  151. package/dist/llmock.d.cts.map +1 -0
  152. package/dist/llmock.d.ts +70 -0
  153. package/dist/llmock.d.ts.map +1 -0
  154. package/dist/llmock.js +223 -0
  155. package/dist/llmock.js.map +1 -0
  156. package/dist/logger.cjs +29 -0
  157. package/dist/logger.cjs.map +1 -0
  158. package/dist/logger.d.cts +14 -0
  159. package/dist/logger.d.cts.map +1 -0
  160. package/dist/logger.d.ts +14 -0
  161. package/dist/logger.d.ts.map +1 -0
  162. package/dist/logger.js +28 -0
  163. package/dist/logger.js.map +1 -0
  164. package/dist/mcp-handler.cjs +189 -0
  165. package/dist/mcp-handler.cjs.map +1 -0
  166. package/dist/mcp-handler.js +188 -0
  167. package/dist/mcp-handler.js.map +1 -0
  168. package/dist/mcp-mock.cjs +169 -0
  169. package/dist/mcp-mock.cjs.map +1 -0
  170. package/dist/mcp-mock.d.cts +40 -0
  171. package/dist/mcp-mock.d.cts.map +1 -0
  172. package/dist/mcp-mock.d.ts +40 -0
  173. package/dist/mcp-mock.d.ts.map +1 -0
  174. package/dist/mcp-mock.js +167 -0
  175. package/dist/mcp-mock.js.map +1 -0
  176. package/dist/mcp-stub.cjs +4 -0
  177. package/dist/mcp-stub.d.cts +3 -0
  178. package/dist/mcp-stub.d.ts +3 -0
  179. package/dist/mcp-stub.js +3 -0
  180. package/dist/mcp-types.d.cts +65 -0
  181. package/dist/mcp-types.d.cts.map +1 -0
  182. package/dist/mcp-types.d.ts +65 -0
  183. package/dist/mcp-types.d.ts.map +1 -0
  184. package/dist/messages.cjs +489 -0
  185. package/dist/messages.cjs.map +1 -0
  186. package/dist/messages.d.cts +10 -0
  187. package/dist/messages.d.cts.map +1 -0
  188. package/dist/messages.d.ts +10 -0
  189. package/dist/messages.d.ts.map +1 -0
  190. package/dist/messages.js +489 -0
  191. package/dist/messages.js.map +1 -0
  192. package/dist/metrics.cjs +160 -0
  193. package/dist/metrics.cjs.map +1 -0
  194. package/dist/metrics.d.cts +24 -0
  195. package/dist/metrics.d.cts.map +1 -0
  196. package/dist/metrics.d.ts +24 -0
  197. package/dist/metrics.d.ts.map +1 -0
  198. package/dist/metrics.js +158 -0
  199. package/dist/metrics.js.map +1 -0
  200. package/dist/moderation.cjs +91 -0
  201. package/dist/moderation.cjs.map +1 -0
  202. package/dist/moderation.d.cts +23 -0
  203. package/dist/moderation.d.cts.map +1 -0
  204. package/dist/moderation.d.ts +23 -0
  205. package/dist/moderation.d.ts.map +1 -0
  206. package/dist/moderation.js +91 -0
  207. package/dist/moderation.js.map +1 -0
  208. package/dist/ndjson-writer.cjs +31 -0
  209. package/dist/ndjson-writer.cjs.map +1 -0
  210. package/dist/ndjson-writer.d.cts +17 -0
  211. package/dist/ndjson-writer.d.cts.map +1 -0
  212. package/dist/ndjson-writer.d.ts +17 -0
  213. package/dist/ndjson-writer.d.ts.map +1 -0
  214. package/dist/ndjson-writer.js +31 -0
  215. package/dist/ndjson-writer.js.map +1 -0
  216. package/dist/ollama.cjs +519 -0
  217. package/dist/ollama.cjs.map +1 -0
  218. package/dist/ollama.d.cts +34 -0
  219. package/dist/ollama.d.cts.map +1 -0
  220. package/dist/ollama.d.ts +34 -0
  221. package/dist/ollama.d.ts.map +1 -0
  222. package/dist/ollama.js +517 -0
  223. package/dist/ollama.js.map +1 -0
  224. package/dist/recorder.cjs +311 -0
  225. package/dist/recorder.cjs.map +1 -0
  226. package/dist/recorder.d.cts +23 -0
  227. package/dist/recorder.d.cts.map +1 -0
  228. package/dist/recorder.d.ts +23 -0
  229. package/dist/recorder.d.ts.map +1 -0
  230. package/dist/recorder.js +305 -0
  231. package/dist/recorder.js.map +1 -0
  232. package/dist/rerank.cjs +71 -0
  233. package/dist/rerank.cjs.map +1 -0
  234. package/dist/rerank.d.cts +22 -0
  235. package/dist/rerank.d.cts.map +1 -0
  236. package/dist/rerank.d.ts +22 -0
  237. package/dist/rerank.d.ts.map +1 -0
  238. package/dist/rerank.js +71 -0
  239. package/dist/rerank.js.map +1 -0
  240. package/dist/responses.cjs +637 -0
  241. package/dist/responses.cjs.map +1 -0
  242. package/dist/responses.d.cts +16 -0
  243. package/dist/responses.d.cts.map +1 -0
  244. package/dist/responses.d.ts +16 -0
  245. package/dist/responses.d.ts.map +1 -0
  246. package/dist/responses.js +634 -0
  247. package/dist/responses.js.map +1 -0
  248. package/dist/router.cjs +68 -0
  249. package/dist/router.cjs.map +1 -0
  250. package/dist/router.d.cts +16 -0
  251. package/dist/router.d.cts.map +1 -0
  252. package/dist/router.d.ts +16 -0
  253. package/dist/router.d.ts.map +1 -0
  254. package/dist/router.js +65 -0
  255. package/dist/router.js.map +1 -0
  256. package/dist/search.cjs +59 -0
  257. package/dist/search.cjs.map +1 -0
  258. package/dist/search.d.cts +23 -0
  259. package/dist/search.d.cts.map +1 -0
  260. package/dist/search.d.ts +23 -0
  261. package/dist/search.d.ts.map +1 -0
  262. package/dist/search.js +59 -0
  263. package/dist/search.js.map +1 -0
  264. package/dist/server.cjs +935 -0
  265. package/dist/server.cjs.map +1 -0
  266. package/dist/server.d.cts +28 -0
  267. package/dist/server.d.cts.map +1 -0
  268. package/dist/server.d.ts +28 -0
  269. package/dist/server.d.ts.map +1 -0
  270. package/dist/server.js +933 -0
  271. package/dist/server.js.map +1 -0
  272. package/dist/sse-writer.cjs +59 -0
  273. package/dist/sse-writer.cjs.map +1 -0
  274. package/dist/sse-writer.d.cts +19 -0
  275. package/dist/sse-writer.d.cts.map +1 -0
  276. package/dist/sse-writer.d.ts +19 -0
  277. package/dist/sse-writer.d.ts.map +1 -0
  278. package/dist/sse-writer.js +55 -0
  279. package/dist/sse-writer.js.map +1 -0
  280. package/dist/stream-collapse.cjs +496 -0
  281. package/dist/stream-collapse.cjs.map +1 -0
  282. package/dist/stream-collapse.d.cts +70 -0
  283. package/dist/stream-collapse.d.cts.map +1 -0
  284. package/dist/stream-collapse.d.ts +70 -0
  285. package/dist/stream-collapse.d.ts.map +1 -0
  286. package/dist/stream-collapse.js +489 -0
  287. package/dist/stream-collapse.js.map +1 -0
  288. package/dist/suite.cjs +46 -0
  289. package/dist/suite.cjs.map +1 -0
  290. package/dist/suite.d.cts +31 -0
  291. package/dist/suite.d.cts.map +1 -0
  292. package/dist/suite.d.ts +31 -0
  293. package/dist/suite.d.ts.map +1 -0
  294. package/dist/suite.js +46 -0
  295. package/dist/suite.js.map +1 -0
  296. package/dist/types.d.cts +243 -0
  297. package/dist/types.d.cts.map +1 -0
  298. package/dist/types.d.ts +243 -0
  299. package/dist/types.d.ts.map +1 -0
  300. package/dist/url.cjs +21 -0
  301. package/dist/url.cjs.map +1 -0
  302. package/dist/url.d.cts +16 -0
  303. package/dist/url.d.cts.map +1 -0
  304. package/dist/url.d.ts +16 -0
  305. package/dist/url.d.ts.map +1 -0
  306. package/dist/url.js +20 -0
  307. package/dist/url.js.map +1 -0
  308. package/dist/vector-handler.cjs +239 -0
  309. package/dist/vector-handler.cjs.map +1 -0
  310. package/dist/vector-handler.js +238 -0
  311. package/dist/vector-handler.js.map +1 -0
  312. package/dist/vector-mock.cjs +229 -0
  313. package/dist/vector-mock.cjs.map +1 -0
  314. package/dist/vector-mock.d.cts +39 -0
  315. package/dist/vector-mock.d.cts.map +1 -0
  316. package/dist/vector-mock.d.ts +39 -0
  317. package/dist/vector-mock.d.ts.map +1 -0
  318. package/dist/vector-mock.js +227 -0
  319. package/dist/vector-mock.js.map +1 -0
  320. package/dist/vector-stub.cjs +4 -0
  321. package/dist/vector-stub.d.cts +3 -0
  322. package/dist/vector-stub.d.ts +3 -0
  323. package/dist/vector-stub.js +3 -0
  324. package/dist/vector-types.d.cts +32 -0
  325. package/dist/vector-types.d.cts.map +1 -0
  326. package/dist/vector-types.d.ts +32 -0
  327. package/dist/vector-types.d.ts.map +1 -0
  328. package/dist/watcher.cjs +59 -0
  329. package/dist/watcher.cjs.map +1 -0
  330. package/dist/watcher.js +58 -0
  331. package/dist/watcher.js.map +1 -0
  332. package/dist/ws-framing.cjs +187 -0
  333. package/dist/ws-framing.cjs.map +1 -0
  334. package/dist/ws-framing.d.cts +26 -0
  335. package/dist/ws-framing.d.cts.map +1 -0
  336. package/dist/ws-framing.d.ts +26 -0
  337. package/dist/ws-framing.d.ts.map +1 -0
  338. package/dist/ws-framing.js +184 -0
  339. package/dist/ws-framing.js.map +1 -0
  340. package/dist/ws-gemini-live.cjs +364 -0
  341. package/dist/ws-gemini-live.cjs.map +1 -0
  342. package/dist/ws-gemini-live.d.cts +18 -0
  343. package/dist/ws-gemini-live.d.cts.map +1 -0
  344. package/dist/ws-gemini-live.d.ts +18 -0
  345. package/dist/ws-gemini-live.d.ts.map +1 -0
  346. package/dist/ws-gemini-live.js +364 -0
  347. package/dist/ws-gemini-live.js.map +1 -0
  348. package/dist/ws-realtime.cjs +435 -0
  349. package/dist/ws-realtime.cjs.map +1 -0
  350. package/dist/ws-realtime.d.cts +17 -0
  351. package/dist/ws-realtime.d.cts.map +1 -0
  352. package/dist/ws-realtime.d.ts +17 -0
  353. package/dist/ws-realtime.d.ts.map +1 -0
  354. package/dist/ws-realtime.js +435 -0
  355. package/dist/ws-realtime.js.map +1 -0
  356. package/dist/ws-responses.cjs +164 -0
  357. package/dist/ws-responses.cjs.map +1 -0
  358. package/dist/ws-responses.d.cts +18 -0
  359. package/dist/ws-responses.d.cts.map +1 -0
  360. package/dist/ws-responses.d.ts +18 -0
  361. package/dist/ws-responses.d.ts.map +1 -0
  362. package/dist/ws-responses.js +164 -0
  363. package/dist/ws-responses.js.map +1 -0
  364. package/fixtures/example-greeting.json +12 -0
  365. package/fixtures/example-multi-turn.json +14 -0
  366. package/fixtures/example-tool-call.json +15 -0
  367. package/package.json +118 -0
  368. package/skills/write-fixtures/SKILL.md +625 -0
@@ -0,0 +1,110 @@
1
+ #!/usr/bin/env node
2
+ import { loadConfig, startFromConfig } from "./config-loader.js";
3
+ import { parseArgs } from "node:util";
4
+ import { resolve } from "node:path";
5
+
6
+ //#region src/aimock-cli.ts
7
+ const HELP = `
8
+ Usage: aimock [options]
9
+
10
+ Options:
11
+ -c, --config <path> Path to aimock config JSON file (required)
12
+ -p, --port <number> Port override (default: from config or 0)
13
+ -h, --host <string> Host override (default: from config or 127.0.0.1)
14
+ --help Show this help message
15
+ `.trim();
16
+ function runAimockCli(deps = {}) {
17
+ /* v8 ignore next 6 -- defaults used only when called from CLI entry point */
18
+ const argv = deps.argv ?? process.argv.slice(2);
19
+ const log = deps.log ?? console.log.bind(console);
20
+ const logError = deps.logError ?? console.error.bind(console);
21
+ const exit = deps.exit ?? process.exit.bind(process);
22
+ const loadConfigFn = deps.loadConfigFn ?? loadConfig;
23
+ const startFromConfigFn = deps.startFromConfigFn ?? startFromConfig;
24
+ let values;
25
+ try {
26
+ ({values} = parseArgs({
27
+ args: argv,
28
+ options: {
29
+ config: {
30
+ type: "string",
31
+ short: "c"
32
+ },
33
+ port: {
34
+ type: "string",
35
+ short: "p"
36
+ },
37
+ host: {
38
+ type: "string",
39
+ short: "h"
40
+ },
41
+ help: {
42
+ type: "boolean",
43
+ default: false
44
+ }
45
+ },
46
+ strict: true
47
+ }));
48
+ } catch (err) {
49
+ logError(`Error: ${err instanceof Error ? err.message : String(err)}\n\n${HELP}`);
50
+ exit(1);
51
+ return;
52
+ }
53
+ if (values.help) {
54
+ log(HELP);
55
+ exit(0);
56
+ return;
57
+ }
58
+ if (!values.config) {
59
+ logError("Error: --config is required.\n\n" + HELP);
60
+ exit(1);
61
+ return;
62
+ }
63
+ const configPath = resolve(values.config);
64
+ let config;
65
+ try {
66
+ config = loadConfigFn(configPath);
67
+ } catch (err) {
68
+ logError(`Failed to load config from ${configPath}: ${err instanceof Error ? err.message : String(err)}`);
69
+ exit(1);
70
+ return;
71
+ }
72
+ const port = values.port ? Number(values.port) : void 0;
73
+ if (port !== void 0 && (Number.isNaN(port) || !Number.isInteger(port) || port < 0 || port > 65535)) {
74
+ logError(`Error: invalid port "${values.port}".\n\n${HELP}`);
75
+ exit(1);
76
+ return;
77
+ }
78
+ const host = values.host;
79
+ async function main() {
80
+ const { llmock, url } = await startFromConfigFn(config, {
81
+ port,
82
+ host
83
+ });
84
+ log(`aimock server listening on ${url}`);
85
+ function shutdown() {
86
+ log("Shutting down...");
87
+ process.removeListener("SIGINT", shutdown);
88
+ process.removeListener("SIGTERM", shutdown);
89
+ llmock.stop().then(() => exit(0), (err) => {
90
+ logError(`Shutdown error: ${err instanceof Error ? err.message : String(err)}`);
91
+ exit(1);
92
+ });
93
+ }
94
+ process.on("SIGINT", shutdown);
95
+ process.on("SIGTERM", shutdown);
96
+ if (deps.onReady) deps.onReady({ shutdown });
97
+ }
98
+ main().catch((err) => {
99
+ logError(err instanceof Error ? err.message : String(err));
100
+ exit(1);
101
+ });
102
+ }
103
+ /* v8 ignore start -- entry-point guard, exercised by integration tests */
104
+ const scriptName = process.argv[1] ?? "";
105
+ if (scriptName.endsWith("aimock-cli.js") || scriptName.endsWith("aimock-cli.ts")) runAimockCli();
106
+ /* v8 ignore stop */
107
+
108
+ //#endregion
109
+ export { runAimockCli };
110
+ //# sourceMappingURL=aimock-cli.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"aimock-cli.js","names":[],"sources":["../src/aimock-cli.ts"],"sourcesContent":["#!/usr/bin/env node\nimport { parseArgs } from \"node:util\";\nimport { resolve } from \"node:path\";\nimport { loadConfig, startFromConfig } from \"./config-loader.js\";\n\nconst HELP = `\nUsage: aimock [options]\n\nOptions:\n -c, --config <path> Path to aimock config JSON file (required)\n -p, --port <number> Port override (default: from config or 0)\n -h, --host <string> Host override (default: from config or 127.0.0.1)\n --help Show this help message\n`.trim();\n\nexport interface AimockCliDeps {\n argv?: string[];\n log?: (msg: string) => void;\n logError?: (msg: string) => void;\n exit?: (code: number) => void;\n loadConfigFn?: typeof loadConfig;\n startFromConfigFn?: typeof startFromConfig;\n onReady?: (ctx: { shutdown: () => void }) => void;\n}\n\nexport function runAimockCli(deps: AimockCliDeps = {}): void {\n /* v8 ignore next 6 -- defaults used only when called from CLI entry point */\n const argv = deps.argv ?? process.argv.slice(2);\n const log = deps.log ?? console.log.bind(console);\n const logError = deps.logError ?? console.error.bind(console);\n const exit = deps.exit ?? process.exit.bind(process);\n const loadConfigFn = deps.loadConfigFn ?? loadConfig;\n const startFromConfigFn = deps.startFromConfigFn ?? startFromConfig;\n\n let values;\n try {\n ({ values } = parseArgs({\n args: argv,\n options: {\n config: { type: \"string\", short: \"c\" },\n port: { type: \"string\", short: \"p\" },\n host: { type: \"string\", short: \"h\" },\n help: { type: \"boolean\", default: false },\n },\n strict: true,\n }));\n } catch (err) {\n /* v8 ignore next -- parseArgs always throws Error subclasses */\n const msg = err instanceof Error ? err.message : String(err);\n logError(`Error: ${msg}\\n\\n${HELP}`);\n exit(1);\n return;\n }\n\n if (values.help) {\n log(HELP);\n exit(0);\n return;\n }\n if (!values.config) {\n logError(\"Error: --config is required.\\n\\n\" + HELP);\n exit(1);\n return;\n }\n\n const configPath = resolve(values.config);\n let config;\n try {\n config = loadConfigFn(configPath);\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n logError(`Failed to load config from ${configPath}: ${msg}`);\n exit(1);\n return;\n }\n\n const port = values.port ? Number(values.port) : undefined;\n if (\n port !== undefined &&\n (Number.isNaN(port) || !Number.isInteger(port) || port < 0 || port > 65535)\n ) {\n logError(`Error: invalid port \"${values.port}\".\\n\\n${HELP}`);\n exit(1);\n return;\n }\n const host = values.host;\n\n async function main() {\n const { llmock, url } = await startFromConfigFn(config!, { port, host });\n log(`aimock server listening on ${url}`);\n\n function shutdown() {\n log(\"Shutting down...\");\n process.removeListener(\"SIGINT\", shutdown);\n process.removeListener(\"SIGTERM\", shutdown);\n llmock.stop().then(\n () => exit(0),\n (err) => {\n logError(`Shutdown error: ${err instanceof Error ? err.message : String(err)}`);\n exit(1);\n },\n );\n }\n process.on(\"SIGINT\", shutdown);\n process.on(\"SIGTERM\", shutdown);\n\n if (deps.onReady) {\n deps.onReady({ shutdown });\n }\n }\n\n main().catch((err) => {\n logError(err instanceof Error ? err.message : String(err));\n exit(1);\n });\n}\n\n// Run when executed as a script (not when imported for testing).\n/* v8 ignore start -- entry-point guard, exercised by integration tests */\nconst scriptName = process.argv[1] ?? \"\";\nif (scriptName.endsWith(\"aimock-cli.js\") || scriptName.endsWith(\"aimock-cli.ts\")) {\n runAimockCli();\n}\n/* v8 ignore stop */\n"],"mappings":";;;;;;AAKA,MAAM,OAAO;;;;;;;;EAQX,MAAM;AAYR,SAAgB,aAAa,OAAsB,EAAE,EAAQ;;CAE3D,MAAM,OAAO,KAAK,QAAQ,QAAQ,KAAK,MAAM,EAAE;CAC/C,MAAM,MAAM,KAAK,OAAO,QAAQ,IAAI,KAAK,QAAQ;CACjD,MAAM,WAAW,KAAK,YAAY,QAAQ,MAAM,KAAK,QAAQ;CAC7D,MAAM,OAAO,KAAK,QAAQ,QAAQ,KAAK,KAAK,QAAQ;CACpD,MAAM,eAAe,KAAK,gBAAgB;CAC1C,MAAM,oBAAoB,KAAK,qBAAqB;CAEpD,IAAI;AACJ,KAAI;AACF,GAAC,CAAE,UAAW,UAAU;GACtB,MAAM;GACN,SAAS;IACP,QAAQ;KAAE,MAAM;KAAU,OAAO;KAAK;IACtC,MAAM;KAAE,MAAM;KAAU,OAAO;KAAK;IACpC,MAAM;KAAE,MAAM;KAAU,OAAO;KAAK;IACpC,MAAM;KAAE,MAAM;KAAW,SAAS;KAAO;IAC1C;GACD,QAAQ;GACT,CAAC;UACK,KAAK;AAGZ,WAAS,UADG,eAAe,QAAQ,IAAI,UAAU,OAAO,IAAI,CACrC,MAAM,OAAO;AACpC,OAAK,EAAE;AACP;;AAGF,KAAI,OAAO,MAAM;AACf,MAAI,KAAK;AACT,OAAK,EAAE;AACP;;AAEF,KAAI,CAAC,OAAO,QAAQ;AAClB,WAAS,qCAAqC,KAAK;AACnD,OAAK,EAAE;AACP;;CAGF,MAAM,aAAa,QAAQ,OAAO,OAAO;CACzC,IAAI;AACJ,KAAI;AACF,WAAS,aAAa,WAAW;UAC1B,KAAK;AAEZ,WAAS,8BAA8B,WAAW,IADtC,eAAe,QAAQ,IAAI,UAAU,OAAO,IAAI,GACA;AAC5D,OAAK,EAAE;AACP;;CAGF,MAAM,OAAO,OAAO,OAAO,OAAO,OAAO,KAAK,GAAG;AACjD,KACE,SAAS,WACR,OAAO,MAAM,KAAK,IAAI,CAAC,OAAO,UAAU,KAAK,IAAI,OAAO,KAAK,OAAO,QACrE;AACA,WAAS,wBAAwB,OAAO,KAAK,QAAQ,OAAO;AAC5D,OAAK,EAAE;AACP;;CAEF,MAAM,OAAO,OAAO;CAEpB,eAAe,OAAO;EACpB,MAAM,EAAE,QAAQ,QAAQ,MAAM,kBAAkB,QAAS;GAAE;GAAM;GAAM,CAAC;AACxE,MAAI,8BAA8B,MAAM;EAExC,SAAS,WAAW;AAClB,OAAI,mBAAmB;AACvB,WAAQ,eAAe,UAAU,SAAS;AAC1C,WAAQ,eAAe,WAAW,SAAS;AAC3C,UAAO,MAAM,CAAC,WACN,KAAK,EAAE,GACZ,QAAQ;AACP,aAAS,mBAAmB,eAAe,QAAQ,IAAI,UAAU,OAAO,IAAI,GAAG;AAC/E,SAAK,EAAE;KAEV;;AAEH,UAAQ,GAAG,UAAU,SAAS;AAC9B,UAAQ,GAAG,WAAW,SAAS;AAE/B,MAAI,KAAK,QACP,MAAK,QAAQ,EAAE,UAAU,CAAC;;AAI9B,OAAM,CAAC,OAAO,QAAQ;AACpB,WAAS,eAAe,QAAQ,IAAI,UAAU,OAAO,IAAI,CAAC;AAC1D,OAAK,EAAE;GACP;;;AAKJ,MAAM,aAAa,QAAQ,KAAK,MAAM;AACtC,IAAI,WAAW,SAAS,gBAAgB,IAAI,WAAW,SAAS,gBAAgB,CAC9E,eAAc"}
@@ -0,0 +1,117 @@
1
+ const require_runtime = require('./_virtual/_rolldown/runtime.cjs');
2
+ const require_sse_writer = require('./sse-writer.cjs');
3
+ let node_zlib = require("node:zlib");
4
+
5
+ //#region src/aws-event-stream.ts
6
+ /**
7
+ * AWS Event Stream binary frame encoder.
8
+ *
9
+ * Implements the AWS binary event stream framing protocol used by Bedrock's
10
+ * streaming (invoke-with-response-stream) endpoint. Each frame carries a set of
11
+ * string headers and a raw-bytes payload, wrapped in a prelude with CRC32
12
+ * checksums for integrity.
13
+ *
14
+ * Binary frame layout:
15
+ * [total_length: 4B uint32-BE]
16
+ * [headers_length: 4B uint32-BE]
17
+ * [prelude_crc32: 4B CRC32 of first 8 bytes]
18
+ * [headers: variable]
19
+ * [payload: variable, raw JSON bytes]
20
+ * [message_crc32: 4B CRC32 of entire frame minus last 4 bytes]
21
+ */
22
+ function encodeHeaders(headers) {
23
+ const parts = [];
24
+ for (const [name, value] of Object.entries(headers)) {
25
+ const nameBytes = Buffer.from(name, "utf8");
26
+ const valueBytes = Buffer.from(value, "utf8");
27
+ const header = Buffer.alloc(1 + nameBytes.length + 1 + 2 + valueBytes.length);
28
+ let offset = 0;
29
+ header.writeUInt8(nameBytes.length, offset);
30
+ offset += 1;
31
+ nameBytes.copy(header, offset);
32
+ offset += nameBytes.length;
33
+ header.writeUInt8(7, offset);
34
+ offset += 1;
35
+ header.writeUInt16BE(valueBytes.length, offset);
36
+ offset += 2;
37
+ valueBytes.copy(header, offset);
38
+ parts.push(header);
39
+ }
40
+ return Buffer.concat(parts);
41
+ }
42
+ /**
43
+ * Encode a single AWS Event Stream binary frame with the given headers and
44
+ * payload buffer.
45
+ */
46
+ function encodeEventStreamFrame(headers, payload) {
47
+ const headersBuffer = encodeHeaders(headers);
48
+ const headersLength = headersBuffer.length;
49
+ const totalLength = 12 + headersLength + payload.length + 4;
50
+ const frame = Buffer.alloc(totalLength);
51
+ let offset = 0;
52
+ frame.writeUInt32BE(totalLength, offset);
53
+ offset += 4;
54
+ frame.writeUInt32BE(headersLength, offset);
55
+ offset += 4;
56
+ const preludeCrc = (0, node_zlib.crc32)(frame.subarray(0, 8));
57
+ frame.writeUInt32BE(preludeCrc >>> 0, offset);
58
+ offset += 4;
59
+ headersBuffer.copy(frame, offset);
60
+ offset += headersLength;
61
+ payload.copy(frame, offset);
62
+ offset += payload.length;
63
+ const messageCrc = (0, node_zlib.crc32)(frame.subarray(0, totalLength - 4));
64
+ frame.writeUInt32BE(messageCrc >>> 0, offset);
65
+ return frame;
66
+ }
67
+ /**
68
+ * Encode an event-stream message with standard AWS headers for a JSON event.
69
+ *
70
+ * Sets `:content-type` = `application/json`, `:event-type` = eventType,
71
+ * `:message-type` = `event`.
72
+ */
73
+ function encodeEventStreamMessage(eventType, jsonPayload) {
74
+ return encodeEventStreamFrame({
75
+ ":content-type": "application/json",
76
+ ":event-type": eventType,
77
+ ":message-type": "event"
78
+ }, Buffer.from(JSON.stringify(jsonPayload), "utf8"));
79
+ }
80
+ /**
81
+ * Write a sequence of event-stream frames to an HTTP response with optional
82
+ * timing control. Mirrors the writeSSEStream pattern from sse-writer.ts.
83
+ *
84
+ * Returns `true` when all events are written (including when the response
85
+ * was already ended before writing began), or `false` if interrupted by
86
+ * the provided abort signal.
87
+ */
88
+ async function writeEventStream(res, events, options) {
89
+ const opts = options ?? {};
90
+ const latency = opts.latency ?? 0;
91
+ const profile = opts.streamingProfile;
92
+ const signal = opts.signal;
93
+ const onChunkSent = opts.onChunkSent;
94
+ if (res.writableEnded) return true;
95
+ res.setHeader("Content-Type", "application/vnd.amazon.eventstream");
96
+ res.setHeader("Transfer-Encoding", "chunked");
97
+ let chunkIndex = 0;
98
+ for (const event of events) {
99
+ const chunkDelay = require_sse_writer.calculateDelay(chunkIndex, profile, latency);
100
+ if (chunkDelay > 0) await require_sse_writer.delay(chunkDelay, signal);
101
+ if (signal?.aborted) return false;
102
+ if (res.writableEnded) return true;
103
+ const frame = encodeEventStreamMessage(event.eventType, event.payload);
104
+ res.write(frame);
105
+ onChunkSent?.();
106
+ if (signal?.aborted) return false;
107
+ chunkIndex++;
108
+ }
109
+ if (!res.writableEnded) res.end();
110
+ return true;
111
+ }
112
+
113
+ //#endregion
114
+ exports.encodeEventStreamFrame = encodeEventStreamFrame;
115
+ exports.encodeEventStreamMessage = encodeEventStreamMessage;
116
+ exports.writeEventStream = writeEventStream;
117
+ //# sourceMappingURL=aws-event-stream.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"aws-event-stream.cjs","names":["calculateDelay","delay"],"sources":["../src/aws-event-stream.ts"],"sourcesContent":["/**\n * AWS Event Stream binary frame encoder.\n *\n * Implements the AWS binary event stream framing protocol used by Bedrock's\n * streaming (invoke-with-response-stream) endpoint. Each frame carries a set of\n * string headers and a raw-bytes payload, wrapped in a prelude with CRC32\n * checksums for integrity.\n *\n * Binary frame layout:\n * [total_length: 4B uint32-BE]\n * [headers_length: 4B uint32-BE]\n * [prelude_crc32: 4B CRC32 of first 8 bytes]\n * [headers: variable]\n * [payload: variable, raw JSON bytes]\n * [message_crc32: 4B CRC32 of entire frame minus last 4 bytes]\n */\n\nimport { crc32 } from \"node:zlib\";\nimport type * as http from \"node:http\";\nimport type { StreamingProfile } from \"./types.js\";\nimport { delay, calculateDelay } from \"./sse-writer.js\";\n\n// ─── Header encoding ────────────────────────────────────────────────────────\n\nfunction encodeHeaders(headers: Record<string, string>): Buffer {\n const parts: Buffer[] = [];\n for (const [name, value] of Object.entries(headers)) {\n const nameBytes = Buffer.from(name, \"utf8\");\n const valueBytes = Buffer.from(value, \"utf8\");\n\n // name_length (1 byte) + name + type (1 byte, 7 = STRING) +\n // value_length (2 bytes BE) + value\n const header = Buffer.alloc(1 + nameBytes.length + 1 + 2 + valueBytes.length);\n let offset = 0;\n header.writeUInt8(nameBytes.length, offset);\n offset += 1;\n nameBytes.copy(header, offset);\n offset += nameBytes.length;\n header.writeUInt8(7, offset); // STRING type\n offset += 1;\n header.writeUInt16BE(valueBytes.length, offset);\n offset += 2;\n valueBytes.copy(header, offset);\n\n parts.push(header);\n }\n return Buffer.concat(parts);\n}\n\n// ─── Frame encoding ─────────────────────────────────────────────────────────\n\n/**\n * Encode a single AWS Event Stream binary frame with the given headers and\n * payload buffer.\n */\nexport function encodeEventStreamFrame(headers: Record<string, string>, payload: Buffer): Buffer {\n const headersBuffer = encodeHeaders(headers);\n const headersLength = headersBuffer.length;\n\n // prelude (8) + prelude_crc (4) + headers + payload + message_crc (4)\n const totalLength = 4 + 4 + 4 + headersLength + payload.length + 4;\n\n const frame = Buffer.alloc(totalLength);\n let offset = 0;\n\n // Prelude\n frame.writeUInt32BE(totalLength, offset);\n offset += 4;\n frame.writeUInt32BE(headersLength, offset);\n offset += 4;\n\n // Prelude CRC32 (covers first 8 bytes)\n const preludeCrc = crc32(frame.subarray(0, 8));\n frame.writeUInt32BE(preludeCrc >>> 0, offset);\n offset += 4;\n\n // Headers\n headersBuffer.copy(frame, offset);\n offset += headersLength;\n\n // Payload\n payload.copy(frame, offset);\n offset += payload.length;\n\n // Message CRC32 (covers entire frame minus last 4 bytes)\n const messageCrc = crc32(frame.subarray(0, totalLength - 4));\n frame.writeUInt32BE(messageCrc >>> 0, offset);\n\n return frame;\n}\n\n// ─── Convenience wrappers ───────────────────────────────────────────────────\n\n/**\n * Encode an event-stream message with standard AWS headers for a JSON event.\n *\n * Sets `:content-type` = `application/json`, `:event-type` = eventType,\n * `:message-type` = `event`.\n */\nexport function encodeEventStreamMessage(eventType: string, jsonPayload: object): Buffer {\n const headers: Record<string, string> = {\n \":content-type\": \"application/json\",\n \":event-type\": eventType,\n \":message-type\": \"event\",\n };\n const payload = Buffer.from(JSON.stringify(jsonPayload), \"utf8\");\n return encodeEventStreamFrame(headers, payload);\n}\n\n/**\n * Write a sequence of event-stream frames to an HTTP response with optional\n * timing control. Mirrors the writeSSEStream pattern from sse-writer.ts.\n *\n * Returns `true` when all events are written (including when the response\n * was already ended before writing began), or `false` if interrupted by\n * the provided abort signal.\n */\nexport async function writeEventStream(\n res: http.ServerResponse,\n events: Array<{ eventType: string; payload: object }>,\n options?: {\n latency?: number;\n streamingProfile?: StreamingProfile;\n signal?: AbortSignal;\n onChunkSent?: () => void;\n },\n): Promise<boolean> {\n const opts = options ?? {};\n const latency = opts.latency ?? 0;\n const profile = opts.streamingProfile;\n const signal = opts.signal;\n const onChunkSent = opts.onChunkSent;\n\n if (res.writableEnded) return true;\n res.setHeader(\"Content-Type\", \"application/vnd.amazon.eventstream\");\n res.setHeader(\"Transfer-Encoding\", \"chunked\");\n\n let chunkIndex = 0;\n for (const event of events) {\n const chunkDelay = calculateDelay(chunkIndex, profile, latency);\n if (chunkDelay > 0) {\n await delay(chunkDelay, signal);\n }\n if (signal?.aborted) return false;\n if (res.writableEnded) return true;\n\n const frame = encodeEventStreamMessage(event.eventType, event.payload);\n res.write(frame);\n onChunkSent?.();\n if (signal?.aborted) return false;\n chunkIndex++;\n }\n\n if (!res.writableEnded) {\n res.end();\n }\n return true;\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;AAwBA,SAAS,cAAc,SAAyC;CAC9D,MAAM,QAAkB,EAAE;AAC1B,MAAK,MAAM,CAAC,MAAM,UAAU,OAAO,QAAQ,QAAQ,EAAE;EACnD,MAAM,YAAY,OAAO,KAAK,MAAM,OAAO;EAC3C,MAAM,aAAa,OAAO,KAAK,OAAO,OAAO;EAI7C,MAAM,SAAS,OAAO,MAAM,IAAI,UAAU,SAAS,IAAI,IAAI,WAAW,OAAO;EAC7E,IAAI,SAAS;AACb,SAAO,WAAW,UAAU,QAAQ,OAAO;AAC3C,YAAU;AACV,YAAU,KAAK,QAAQ,OAAO;AAC9B,YAAU,UAAU;AACpB,SAAO,WAAW,GAAG,OAAO;AAC5B,YAAU;AACV,SAAO,cAAc,WAAW,QAAQ,OAAO;AAC/C,YAAU;AACV,aAAW,KAAK,QAAQ,OAAO;AAE/B,QAAM,KAAK,OAAO;;AAEpB,QAAO,OAAO,OAAO,MAAM;;;;;;AAS7B,SAAgB,uBAAuB,SAAiC,SAAyB;CAC/F,MAAM,gBAAgB,cAAc,QAAQ;CAC5C,MAAM,gBAAgB,cAAc;CAGpC,MAAM,cAAc,KAAY,gBAAgB,QAAQ,SAAS;CAEjE,MAAM,QAAQ,OAAO,MAAM,YAAY;CACvC,IAAI,SAAS;AAGb,OAAM,cAAc,aAAa,OAAO;AACxC,WAAU;AACV,OAAM,cAAc,eAAe,OAAO;AAC1C,WAAU;CAGV,MAAM,kCAAmB,MAAM,SAAS,GAAG,EAAE,CAAC;AAC9C,OAAM,cAAc,eAAe,GAAG,OAAO;AAC7C,WAAU;AAGV,eAAc,KAAK,OAAO,OAAO;AACjC,WAAU;AAGV,SAAQ,KAAK,OAAO,OAAO;AAC3B,WAAU,QAAQ;CAGlB,MAAM,kCAAmB,MAAM,SAAS,GAAG,cAAc,EAAE,CAAC;AAC5D,OAAM,cAAc,eAAe,GAAG,OAAO;AAE7C,QAAO;;;;;;;;AAWT,SAAgB,yBAAyB,WAAmB,aAA6B;AAOvF,QAAO,uBANiC;EACtC,iBAAiB;EACjB,eAAe;EACf,iBAAiB;EAClB,EACe,OAAO,KAAK,KAAK,UAAU,YAAY,EAAE,OAAO,CACjB;;;;;;;;;;AAWjD,eAAsB,iBACpB,KACA,QACA,SAMkB;CAClB,MAAM,OAAO,WAAW,EAAE;CAC1B,MAAM,UAAU,KAAK,WAAW;CAChC,MAAM,UAAU,KAAK;CACrB,MAAM,SAAS,KAAK;CACpB,MAAM,cAAc,KAAK;AAEzB,KAAI,IAAI,cAAe,QAAO;AAC9B,KAAI,UAAU,gBAAgB,qCAAqC;AACnE,KAAI,UAAU,qBAAqB,UAAU;CAE7C,IAAI,aAAa;AACjB,MAAK,MAAM,SAAS,QAAQ;EAC1B,MAAM,aAAaA,kCAAe,YAAY,SAAS,QAAQ;AAC/D,MAAI,aAAa,EACf,OAAMC,yBAAM,YAAY,OAAO;AAEjC,MAAI,QAAQ,QAAS,QAAO;AAC5B,MAAI,IAAI,cAAe,QAAO;EAE9B,MAAM,QAAQ,yBAAyB,MAAM,WAAW,MAAM,QAAQ;AACtE,MAAI,MAAM,MAAM;AAChB,iBAAe;AACf,MAAI,QAAQ,QAAS,QAAO;AAC5B;;AAGF,KAAI,CAAC,IAAI,cACP,KAAI,KAAK;AAEX,QAAO"}
@@ -0,0 +1,38 @@
1
+ import { StreamingProfile } from "./types.cjs";
2
+ import * as http from "node:http";
3
+
4
+ //#region src/aws-event-stream.d.ts
5
+
6
+ /**
7
+ * Encode a single AWS Event Stream binary frame with the given headers and
8
+ * payload buffer.
9
+ */
10
+ declare function encodeEventStreamFrame(headers: Record<string, string>, payload: Buffer): Buffer;
11
+ /**
12
+ * Encode an event-stream message with standard AWS headers for a JSON event.
13
+ *
14
+ * Sets `:content-type` = `application/json`, `:event-type` = eventType,
15
+ * `:message-type` = `event`.
16
+ */
17
+ declare function encodeEventStreamMessage(eventType: string, jsonPayload: object): Buffer;
18
+ /**
19
+ * Write a sequence of event-stream frames to an HTTP response with optional
20
+ * timing control. Mirrors the writeSSEStream pattern from sse-writer.ts.
21
+ *
22
+ * Returns `true` when all events are written (including when the response
23
+ * was already ended before writing began), or `false` if interrupted by
24
+ * the provided abort signal.
25
+ */
26
+ declare function writeEventStream(res: http.ServerResponse, events: Array<{
27
+ eventType: string;
28
+ payload: object;
29
+ }>, options?: {
30
+ latency?: number;
31
+ streamingProfile?: StreamingProfile;
32
+ signal?: AbortSignal;
33
+ onChunkSent?: () => void;
34
+ }): Promise<boolean>;
35
+ //# sourceMappingURL=aws-event-stream.d.ts.map
36
+ //#endregion
37
+ export { encodeEventStreamFrame, encodeEventStreamMessage, writeEventStream };
38
+ //# sourceMappingURL=aws-event-stream.d.cts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"aws-event-stream.d.cts","names":[],"sources":["../src/aws-event-stream.ts"],"sourcesContent":[],"mappings":";;;;;;;;;iBAuDgB,sBAAA,UAAgC,iCAAiC,SAAS;;;;;;;iBA4C1E,wBAAA,0CAAkE;;;;;;;;;iBAkB5D,gBAAA,MACf,IAAA,CAAK,wBACF;;;;;qBAGa;WACV;;IAGV"}
@@ -0,0 +1,38 @@
1
+ import { StreamingProfile } from "./types.js";
2
+ import * as http from "node:http";
3
+
4
+ //#region src/aws-event-stream.d.ts
5
+
6
+ /**
7
+ * Encode a single AWS Event Stream binary frame with the given headers and
8
+ * payload buffer.
9
+ */
10
+ declare function encodeEventStreamFrame(headers: Record<string, string>, payload: Buffer): Buffer;
11
+ /**
12
+ * Encode an event-stream message with standard AWS headers for a JSON event.
13
+ *
14
+ * Sets `:content-type` = `application/json`, `:event-type` = eventType,
15
+ * `:message-type` = `event`.
16
+ */
17
+ declare function encodeEventStreamMessage(eventType: string, jsonPayload: object): Buffer;
18
+ /**
19
+ * Write a sequence of event-stream frames to an HTTP response with optional
20
+ * timing control. Mirrors the writeSSEStream pattern from sse-writer.ts.
21
+ *
22
+ * Returns `true` when all events are written (including when the response
23
+ * was already ended before writing began), or `false` if interrupted by
24
+ * the provided abort signal.
25
+ */
26
+ declare function writeEventStream(res: http.ServerResponse, events: Array<{
27
+ eventType: string;
28
+ payload: object;
29
+ }>, options?: {
30
+ latency?: number;
31
+ streamingProfile?: StreamingProfile;
32
+ signal?: AbortSignal;
33
+ onChunkSent?: () => void;
34
+ }): Promise<boolean>;
35
+ //# sourceMappingURL=aws-event-stream.d.ts.map
36
+ //#endregion
37
+ export { encodeEventStreamFrame, encodeEventStreamMessage, writeEventStream };
38
+ //# sourceMappingURL=aws-event-stream.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"aws-event-stream.d.ts","names":[],"sources":["../src/aws-event-stream.ts"],"sourcesContent":[],"mappings":";;;;;;;;;iBAuDgB,sBAAA,UAAgC,iCAAiC,SAAS;;;;;;;iBA4C1E,wBAAA,0CAAkE;;;;;;;;;iBAkB5D,gBAAA,MACf,IAAA,CAAK,wBACF;;;;;qBAGa;WACV;;IAGV"}
@@ -0,0 +1,114 @@
1
+ import { calculateDelay, delay } from "./sse-writer.js";
2
+ import { crc32 } from "node:zlib";
3
+
4
+ //#region src/aws-event-stream.ts
5
+ /**
6
+ * AWS Event Stream binary frame encoder.
7
+ *
8
+ * Implements the AWS binary event stream framing protocol used by Bedrock's
9
+ * streaming (invoke-with-response-stream) endpoint. Each frame carries a set of
10
+ * string headers and a raw-bytes payload, wrapped in a prelude with CRC32
11
+ * checksums for integrity.
12
+ *
13
+ * Binary frame layout:
14
+ * [total_length: 4B uint32-BE]
15
+ * [headers_length: 4B uint32-BE]
16
+ * [prelude_crc32: 4B CRC32 of first 8 bytes]
17
+ * [headers: variable]
18
+ * [payload: variable, raw JSON bytes]
19
+ * [message_crc32: 4B CRC32 of entire frame minus last 4 bytes]
20
+ */
21
+ function encodeHeaders(headers) {
22
+ const parts = [];
23
+ for (const [name, value] of Object.entries(headers)) {
24
+ const nameBytes = Buffer.from(name, "utf8");
25
+ const valueBytes = Buffer.from(value, "utf8");
26
+ const header = Buffer.alloc(1 + nameBytes.length + 1 + 2 + valueBytes.length);
27
+ let offset = 0;
28
+ header.writeUInt8(nameBytes.length, offset);
29
+ offset += 1;
30
+ nameBytes.copy(header, offset);
31
+ offset += nameBytes.length;
32
+ header.writeUInt8(7, offset);
33
+ offset += 1;
34
+ header.writeUInt16BE(valueBytes.length, offset);
35
+ offset += 2;
36
+ valueBytes.copy(header, offset);
37
+ parts.push(header);
38
+ }
39
+ return Buffer.concat(parts);
40
+ }
41
+ /**
42
+ * Encode a single AWS Event Stream binary frame with the given headers and
43
+ * payload buffer.
44
+ */
45
+ function encodeEventStreamFrame(headers, payload) {
46
+ const headersBuffer = encodeHeaders(headers);
47
+ const headersLength = headersBuffer.length;
48
+ const totalLength = 12 + headersLength + payload.length + 4;
49
+ const frame = Buffer.alloc(totalLength);
50
+ let offset = 0;
51
+ frame.writeUInt32BE(totalLength, offset);
52
+ offset += 4;
53
+ frame.writeUInt32BE(headersLength, offset);
54
+ offset += 4;
55
+ const preludeCrc = crc32(frame.subarray(0, 8));
56
+ frame.writeUInt32BE(preludeCrc >>> 0, offset);
57
+ offset += 4;
58
+ headersBuffer.copy(frame, offset);
59
+ offset += headersLength;
60
+ payload.copy(frame, offset);
61
+ offset += payload.length;
62
+ const messageCrc = crc32(frame.subarray(0, totalLength - 4));
63
+ frame.writeUInt32BE(messageCrc >>> 0, offset);
64
+ return frame;
65
+ }
66
+ /**
67
+ * Encode an event-stream message with standard AWS headers for a JSON event.
68
+ *
69
+ * Sets `:content-type` = `application/json`, `:event-type` = eventType,
70
+ * `:message-type` = `event`.
71
+ */
72
+ function encodeEventStreamMessage(eventType, jsonPayload) {
73
+ return encodeEventStreamFrame({
74
+ ":content-type": "application/json",
75
+ ":event-type": eventType,
76
+ ":message-type": "event"
77
+ }, Buffer.from(JSON.stringify(jsonPayload), "utf8"));
78
+ }
79
+ /**
80
+ * Write a sequence of event-stream frames to an HTTP response with optional
81
+ * timing control. Mirrors the writeSSEStream pattern from sse-writer.ts.
82
+ *
83
+ * Returns `true` when all events are written (including when the response
84
+ * was already ended before writing began), or `false` if interrupted by
85
+ * the provided abort signal.
86
+ */
87
+ async function writeEventStream(res, events, options) {
88
+ const opts = options ?? {};
89
+ const latency = opts.latency ?? 0;
90
+ const profile = opts.streamingProfile;
91
+ const signal = opts.signal;
92
+ const onChunkSent = opts.onChunkSent;
93
+ if (res.writableEnded) return true;
94
+ res.setHeader("Content-Type", "application/vnd.amazon.eventstream");
95
+ res.setHeader("Transfer-Encoding", "chunked");
96
+ let chunkIndex = 0;
97
+ for (const event of events) {
98
+ const chunkDelay = calculateDelay(chunkIndex, profile, latency);
99
+ if (chunkDelay > 0) await delay(chunkDelay, signal);
100
+ if (signal?.aborted) return false;
101
+ if (res.writableEnded) return true;
102
+ const frame = encodeEventStreamMessage(event.eventType, event.payload);
103
+ res.write(frame);
104
+ onChunkSent?.();
105
+ if (signal?.aborted) return false;
106
+ chunkIndex++;
107
+ }
108
+ if (!res.writableEnded) res.end();
109
+ return true;
110
+ }
111
+
112
+ //#endregion
113
+ export { encodeEventStreamFrame, encodeEventStreamMessage, writeEventStream };
114
+ //# sourceMappingURL=aws-event-stream.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"aws-event-stream.js","names":[],"sources":["../src/aws-event-stream.ts"],"sourcesContent":["/**\n * AWS Event Stream binary frame encoder.\n *\n * Implements the AWS binary event stream framing protocol used by Bedrock's\n * streaming (invoke-with-response-stream) endpoint. Each frame carries a set of\n * string headers and a raw-bytes payload, wrapped in a prelude with CRC32\n * checksums for integrity.\n *\n * Binary frame layout:\n * [total_length: 4B uint32-BE]\n * [headers_length: 4B uint32-BE]\n * [prelude_crc32: 4B CRC32 of first 8 bytes]\n * [headers: variable]\n * [payload: variable, raw JSON bytes]\n * [message_crc32: 4B CRC32 of entire frame minus last 4 bytes]\n */\n\nimport { crc32 } from \"node:zlib\";\nimport type * as http from \"node:http\";\nimport type { StreamingProfile } from \"./types.js\";\nimport { delay, calculateDelay } from \"./sse-writer.js\";\n\n// ─── Header encoding ────────────────────────────────────────────────────────\n\nfunction encodeHeaders(headers: Record<string, string>): Buffer {\n const parts: Buffer[] = [];\n for (const [name, value] of Object.entries(headers)) {\n const nameBytes = Buffer.from(name, \"utf8\");\n const valueBytes = Buffer.from(value, \"utf8\");\n\n // name_length (1 byte) + name + type (1 byte, 7 = STRING) +\n // value_length (2 bytes BE) + value\n const header = Buffer.alloc(1 + nameBytes.length + 1 + 2 + valueBytes.length);\n let offset = 0;\n header.writeUInt8(nameBytes.length, offset);\n offset += 1;\n nameBytes.copy(header, offset);\n offset += nameBytes.length;\n header.writeUInt8(7, offset); // STRING type\n offset += 1;\n header.writeUInt16BE(valueBytes.length, offset);\n offset += 2;\n valueBytes.copy(header, offset);\n\n parts.push(header);\n }\n return Buffer.concat(parts);\n}\n\n// ─── Frame encoding ─────────────────────────────────────────────────────────\n\n/**\n * Encode a single AWS Event Stream binary frame with the given headers and\n * payload buffer.\n */\nexport function encodeEventStreamFrame(headers: Record<string, string>, payload: Buffer): Buffer {\n const headersBuffer = encodeHeaders(headers);\n const headersLength = headersBuffer.length;\n\n // prelude (8) + prelude_crc (4) + headers + payload + message_crc (4)\n const totalLength = 4 + 4 + 4 + headersLength + payload.length + 4;\n\n const frame = Buffer.alloc(totalLength);\n let offset = 0;\n\n // Prelude\n frame.writeUInt32BE(totalLength, offset);\n offset += 4;\n frame.writeUInt32BE(headersLength, offset);\n offset += 4;\n\n // Prelude CRC32 (covers first 8 bytes)\n const preludeCrc = crc32(frame.subarray(0, 8));\n frame.writeUInt32BE(preludeCrc >>> 0, offset);\n offset += 4;\n\n // Headers\n headersBuffer.copy(frame, offset);\n offset += headersLength;\n\n // Payload\n payload.copy(frame, offset);\n offset += payload.length;\n\n // Message CRC32 (covers entire frame minus last 4 bytes)\n const messageCrc = crc32(frame.subarray(0, totalLength - 4));\n frame.writeUInt32BE(messageCrc >>> 0, offset);\n\n return frame;\n}\n\n// ─── Convenience wrappers ───────────────────────────────────────────────────\n\n/**\n * Encode an event-stream message with standard AWS headers for a JSON event.\n *\n * Sets `:content-type` = `application/json`, `:event-type` = eventType,\n * `:message-type` = `event`.\n */\nexport function encodeEventStreamMessage(eventType: string, jsonPayload: object): Buffer {\n const headers: Record<string, string> = {\n \":content-type\": \"application/json\",\n \":event-type\": eventType,\n \":message-type\": \"event\",\n };\n const payload = Buffer.from(JSON.stringify(jsonPayload), \"utf8\");\n return encodeEventStreamFrame(headers, payload);\n}\n\n/**\n * Write a sequence of event-stream frames to an HTTP response with optional\n * timing control. Mirrors the writeSSEStream pattern from sse-writer.ts.\n *\n * Returns `true` when all events are written (including when the response\n * was already ended before writing began), or `false` if interrupted by\n * the provided abort signal.\n */\nexport async function writeEventStream(\n res: http.ServerResponse,\n events: Array<{ eventType: string; payload: object }>,\n options?: {\n latency?: number;\n streamingProfile?: StreamingProfile;\n signal?: AbortSignal;\n onChunkSent?: () => void;\n },\n): Promise<boolean> {\n const opts = options ?? {};\n const latency = opts.latency ?? 0;\n const profile = opts.streamingProfile;\n const signal = opts.signal;\n const onChunkSent = opts.onChunkSent;\n\n if (res.writableEnded) return true;\n res.setHeader(\"Content-Type\", \"application/vnd.amazon.eventstream\");\n res.setHeader(\"Transfer-Encoding\", \"chunked\");\n\n let chunkIndex = 0;\n for (const event of events) {\n const chunkDelay = calculateDelay(chunkIndex, profile, latency);\n if (chunkDelay > 0) {\n await delay(chunkDelay, signal);\n }\n if (signal?.aborted) return false;\n if (res.writableEnded) return true;\n\n const frame = encodeEventStreamMessage(event.eventType, event.payload);\n res.write(frame);\n onChunkSent?.();\n if (signal?.aborted) return false;\n chunkIndex++;\n }\n\n if (!res.writableEnded) {\n res.end();\n }\n return true;\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAwBA,SAAS,cAAc,SAAyC;CAC9D,MAAM,QAAkB,EAAE;AAC1B,MAAK,MAAM,CAAC,MAAM,UAAU,OAAO,QAAQ,QAAQ,EAAE;EACnD,MAAM,YAAY,OAAO,KAAK,MAAM,OAAO;EAC3C,MAAM,aAAa,OAAO,KAAK,OAAO,OAAO;EAI7C,MAAM,SAAS,OAAO,MAAM,IAAI,UAAU,SAAS,IAAI,IAAI,WAAW,OAAO;EAC7E,IAAI,SAAS;AACb,SAAO,WAAW,UAAU,QAAQ,OAAO;AAC3C,YAAU;AACV,YAAU,KAAK,QAAQ,OAAO;AAC9B,YAAU,UAAU;AACpB,SAAO,WAAW,GAAG,OAAO;AAC5B,YAAU;AACV,SAAO,cAAc,WAAW,QAAQ,OAAO;AAC/C,YAAU;AACV,aAAW,KAAK,QAAQ,OAAO;AAE/B,QAAM,KAAK,OAAO;;AAEpB,QAAO,OAAO,OAAO,MAAM;;;;;;AAS7B,SAAgB,uBAAuB,SAAiC,SAAyB;CAC/F,MAAM,gBAAgB,cAAc,QAAQ;CAC5C,MAAM,gBAAgB,cAAc;CAGpC,MAAM,cAAc,KAAY,gBAAgB,QAAQ,SAAS;CAEjE,MAAM,QAAQ,OAAO,MAAM,YAAY;CACvC,IAAI,SAAS;AAGb,OAAM,cAAc,aAAa,OAAO;AACxC,WAAU;AACV,OAAM,cAAc,eAAe,OAAO;AAC1C,WAAU;CAGV,MAAM,aAAa,MAAM,MAAM,SAAS,GAAG,EAAE,CAAC;AAC9C,OAAM,cAAc,eAAe,GAAG,OAAO;AAC7C,WAAU;AAGV,eAAc,KAAK,OAAO,OAAO;AACjC,WAAU;AAGV,SAAQ,KAAK,OAAO,OAAO;AAC3B,WAAU,QAAQ;CAGlB,MAAM,aAAa,MAAM,MAAM,SAAS,GAAG,cAAc,EAAE,CAAC;AAC5D,OAAM,cAAc,eAAe,GAAG,OAAO;AAE7C,QAAO;;;;;;;;AAWT,SAAgB,yBAAyB,WAAmB,aAA6B;AAOvF,QAAO,uBANiC;EACtC,iBAAiB;EACjB,eAAe;EACf,iBAAiB;EAClB,EACe,OAAO,KAAK,KAAK,UAAU,YAAY,EAAE,OAAO,CACjB;;;;;;;;;;AAWjD,eAAsB,iBACpB,KACA,QACA,SAMkB;CAClB,MAAM,OAAO,WAAW,EAAE;CAC1B,MAAM,UAAU,KAAK,WAAW;CAChC,MAAM,UAAU,KAAK;CACrB,MAAM,SAAS,KAAK;CACpB,MAAM,cAAc,KAAK;AAEzB,KAAI,IAAI,cAAe,QAAO;AAC9B,KAAI,UAAU,gBAAgB,qCAAqC;AACnE,KAAI,UAAU,qBAAqB,UAAU;CAE7C,IAAI,aAAa;AACjB,MAAK,MAAM,SAAS,QAAQ;EAC1B,MAAM,aAAa,eAAe,YAAY,SAAS,QAAQ;AAC/D,MAAI,aAAa,EACf,OAAM,MAAM,YAAY,OAAO;AAEjC,MAAI,QAAQ,QAAS,QAAO;AAC5B,MAAI,IAAI,cAAe,QAAO;EAE9B,MAAM,QAAQ,yBAAyB,MAAM,WAAW,MAAM,QAAQ;AACtE,MAAI,MAAM,MAAM;AAChB,iBAAe;AACf,MAAI,QAAQ,QAAS,QAAO;AAC5B;;AAGF,KAAI,CAAC,IAAI,cACP,KAAI,KAAK;AAEX,QAAO"}