@copilotkit/aimock 1.16.4 → 1.18.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 (326) hide show
  1. package/.claude-plugin/marketplace.json +1 -1
  2. package/.claude-plugin/plugin.json +1 -1
  3. package/CHANGELOG.md +35 -0
  4. package/README.md +10 -10
  5. package/dist/a2a-mock.d.cts +2 -2
  6. package/dist/a2a-mock.d.cts.map +1 -1
  7. package/dist/a2a-mock.d.ts +2 -2
  8. package/dist/a2a-mock.d.ts.map +1 -1
  9. package/dist/a2a-mock.js +2 -2
  10. package/dist/a2a-mock.js.map +1 -1
  11. package/dist/agui-handler.cjs +120 -5
  12. package/dist/agui-handler.cjs.map +1 -1
  13. package/dist/agui-handler.d.cts +41 -5
  14. package/dist/agui-handler.d.cts.map +1 -1
  15. package/dist/agui-handler.d.ts +41 -5
  16. package/dist/agui-handler.d.ts.map +1 -1
  17. package/dist/agui-handler.js +114 -6
  18. package/dist/agui-handler.js.map +1 -1
  19. package/dist/agui-mock.cjs +18 -7
  20. package/dist/agui-mock.cjs.map +1 -1
  21. package/dist/agui-mock.d.cts +2 -2
  22. package/dist/agui-mock.d.cts.map +1 -1
  23. package/dist/agui-mock.d.ts +2 -2
  24. package/dist/agui-mock.d.ts.map +1 -1
  25. package/dist/agui-mock.js +20 -9
  26. package/dist/agui-mock.js.map +1 -1
  27. package/dist/agui-recorder.cjs +43 -22
  28. package/dist/agui-recorder.cjs.map +1 -1
  29. package/dist/agui-recorder.d.cts +4 -3
  30. package/dist/agui-recorder.d.cts.map +1 -1
  31. package/dist/agui-recorder.d.ts +4 -3
  32. package/dist/agui-recorder.d.ts.map +1 -1
  33. package/dist/agui-recorder.js +45 -24
  34. package/dist/agui-recorder.js.map +1 -1
  35. package/dist/agui-stub.cjs +28 -0
  36. package/dist/agui-stub.d.cts +5 -0
  37. package/dist/agui-stub.d.ts +5 -0
  38. package/dist/agui-stub.js +5 -0
  39. package/dist/agui-types.d.cts +33 -6
  40. package/dist/agui-types.d.cts.map +1 -1
  41. package/dist/agui-types.d.ts +33 -6
  42. package/dist/agui-types.d.ts.map +1 -1
  43. package/dist/aimock-cli.cjs +1 -1
  44. package/dist/aimock-cli.js +1 -1
  45. package/dist/aws-event-stream.d.cts +2 -2
  46. package/dist/aws-event-stream.d.cts.map +1 -1
  47. package/dist/aws-event-stream.d.ts +2 -2
  48. package/dist/aws-event-stream.d.ts.map +1 -1
  49. package/dist/bedrock-converse.cjs +4 -4
  50. package/dist/bedrock-converse.cjs.map +1 -1
  51. package/dist/bedrock-converse.d.cts +3 -3
  52. package/dist/bedrock-converse.d.cts.map +1 -1
  53. package/dist/bedrock-converse.d.ts +3 -3
  54. package/dist/bedrock-converse.d.ts.map +1 -1
  55. package/dist/bedrock-converse.js +4 -4
  56. package/dist/bedrock-converse.js.map +1 -1
  57. package/dist/bedrock.cjs +4 -4
  58. package/dist/bedrock.cjs.map +1 -1
  59. package/dist/bedrock.d.cts +3 -3
  60. package/dist/bedrock.d.cts.map +1 -1
  61. package/dist/bedrock.d.ts +3 -3
  62. package/dist/bedrock.d.ts.map +1 -1
  63. package/dist/bedrock.js +4 -4
  64. package/dist/bedrock.js.map +1 -1
  65. package/dist/chaos.cjs +35 -9
  66. package/dist/chaos.cjs.map +1 -1
  67. package/dist/chaos.d.cts +19 -4
  68. package/dist/chaos.d.cts.map +1 -1
  69. package/dist/chaos.d.ts +19 -4
  70. package/dist/chaos.d.ts.map +1 -1
  71. package/dist/chaos.js +35 -10
  72. package/dist/chaos.js.map +1 -1
  73. package/dist/cli.cjs +6 -5
  74. package/dist/cli.cjs.map +1 -1
  75. package/dist/cli.js +6 -5
  76. package/dist/cli.js.map +1 -1
  77. package/dist/cohere.cjs +2 -2
  78. package/dist/cohere.cjs.map +1 -1
  79. package/dist/cohere.d.cts +2 -2
  80. package/dist/cohere.d.cts.map +1 -1
  81. package/dist/cohere.d.ts +2 -2
  82. package/dist/cohere.d.ts.map +1 -1
  83. package/dist/cohere.js +2 -2
  84. package/dist/cohere.js.map +1 -1
  85. package/dist/config-loader.cjs +3 -3
  86. package/dist/config-loader.d.cts +1 -1
  87. package/dist/config-loader.d.cts.map +1 -1
  88. package/dist/config-loader.d.ts +1 -1
  89. package/dist/config-loader.d.ts.map +1 -1
  90. package/dist/config-loader.js +2 -2
  91. package/dist/convert-vidaimock.cjs +1 -1
  92. package/dist/convert-vidaimock.js +1 -1
  93. package/dist/convert.cjs +1 -1
  94. package/dist/convert.js +1 -1
  95. package/dist/elevenlabs-audio.cjs +212 -0
  96. package/dist/elevenlabs-audio.cjs.map +1 -0
  97. package/dist/elevenlabs-audio.d.cts +11 -0
  98. package/dist/elevenlabs-audio.d.cts.map +1 -0
  99. package/dist/elevenlabs-audio.d.ts +11 -0
  100. package/dist/elevenlabs-audio.d.ts.map +1 -0
  101. package/dist/elevenlabs-audio.js +212 -0
  102. package/dist/elevenlabs-audio.js.map +1 -0
  103. package/dist/embeddings.cjs +2 -2
  104. package/dist/embeddings.cjs.map +1 -1
  105. package/dist/embeddings.d.cts +2 -2
  106. package/dist/embeddings.d.cts.map +1 -1
  107. package/dist/embeddings.d.ts +2 -2
  108. package/dist/embeddings.d.ts.map +1 -1
  109. package/dist/embeddings.js +2 -2
  110. package/dist/embeddings.js.map +1 -1
  111. package/dist/fal-audio.cjs +484 -0
  112. package/dist/fal-audio.cjs.map +1 -0
  113. package/dist/fal-audio.d.cts +10 -0
  114. package/dist/fal-audio.d.cts.map +1 -0
  115. package/dist/fal-audio.d.ts +10 -0
  116. package/dist/fal-audio.d.ts.map +1 -0
  117. package/dist/fal-audio.js +480 -0
  118. package/dist/fal-audio.js.map +1 -0
  119. package/dist/fal.cjs +424 -0
  120. package/dist/fal.cjs.map +1 -0
  121. package/dist/fal.d.cts +39 -0
  122. package/dist/fal.d.cts.map +1 -0
  123. package/dist/fal.d.ts +39 -0
  124. package/dist/fal.d.ts.map +1 -0
  125. package/dist/fal.js +420 -0
  126. package/dist/fal.js.map +1 -0
  127. package/dist/fixture-loader.cjs +16 -3
  128. package/dist/fixture-loader.cjs.map +1 -1
  129. package/dist/fixture-loader.d.cts.map +1 -1
  130. package/dist/fixture-loader.d.ts.map +1 -1
  131. package/dist/fixture-loader.js +17 -4
  132. package/dist/fixture-loader.js.map +1 -1
  133. package/dist/fixtures-remote.cjs +1 -1
  134. package/dist/fixtures-remote.js +1 -1
  135. package/dist/gemini-interactions.cjs +619 -0
  136. package/dist/gemini-interactions.cjs.map +1 -0
  137. package/dist/gemini-interactions.d.cts +46 -0
  138. package/dist/gemini-interactions.d.cts.map +1 -0
  139. package/dist/gemini-interactions.d.ts +46 -0
  140. package/dist/gemini-interactions.d.ts.map +1 -0
  141. package/dist/gemini-interactions.js +618 -0
  142. package/dist/gemini-interactions.js.map +1 -0
  143. package/dist/gemini.cjs +78 -2
  144. package/dist/gemini.cjs.map +1 -1
  145. package/dist/gemini.d.cts +2 -2
  146. package/dist/gemini.d.cts.map +1 -1
  147. package/dist/gemini.d.ts +2 -2
  148. package/dist/gemini.d.ts.map +1 -1
  149. package/dist/gemini.js +79 -3
  150. package/dist/gemini.js.map +1 -1
  151. package/dist/helpers.cjs +28 -1
  152. package/dist/helpers.cjs.map +1 -1
  153. package/dist/helpers.d.cts +13 -3
  154. package/dist/helpers.d.cts.map +1 -1
  155. package/dist/helpers.d.ts +13 -3
  156. package/dist/helpers.d.ts.map +1 -1
  157. package/dist/helpers.js +26 -2
  158. package/dist/helpers.js.map +1 -1
  159. package/dist/images.cjs +2 -2
  160. package/dist/images.cjs.map +1 -1
  161. package/dist/images.d.cts +2 -2
  162. package/dist/images.d.cts.map +1 -1
  163. package/dist/images.d.ts +2 -2
  164. package/dist/images.d.ts.map +1 -1
  165. package/dist/images.js +2 -2
  166. package/dist/images.js.map +1 -1
  167. package/dist/index.cjs +24 -4
  168. package/dist/index.d.cts +12 -8
  169. package/dist/index.d.ts +12 -8
  170. package/dist/index.js +11 -7
  171. package/dist/jest.cjs +1 -1
  172. package/dist/jest.js +1 -1
  173. package/dist/jsonrpc.d.cts +3 -3
  174. package/dist/jsonrpc.d.cts.map +1 -1
  175. package/dist/jsonrpc.d.ts +3 -3
  176. package/dist/jsonrpc.d.ts.map +1 -1
  177. package/dist/llmock.cjs +53 -2
  178. package/dist/llmock.cjs.map +1 -1
  179. package/dist/llmock.d.cts +6 -0
  180. package/dist/llmock.d.cts.map +1 -1
  181. package/dist/llmock.d.ts +6 -0
  182. package/dist/llmock.d.ts.map +1 -1
  183. package/dist/llmock.js +53 -2
  184. package/dist/llmock.js.map +1 -1
  185. package/dist/logger.cjs +5 -4
  186. package/dist/logger.cjs.map +1 -1
  187. package/dist/logger.d.cts +1 -1
  188. package/dist/logger.d.cts.map +1 -1
  189. package/dist/logger.d.ts +1 -1
  190. package/dist/logger.d.ts.map +1 -1
  191. package/dist/logger.js +5 -4
  192. package/dist/logger.js.map +1 -1
  193. package/dist/mcp-mock.d.cts +2 -2
  194. package/dist/mcp-mock.d.cts.map +1 -1
  195. package/dist/mcp-mock.d.ts +2 -2
  196. package/dist/mcp-mock.d.ts.map +1 -1
  197. package/dist/mcp-mock.js +2 -2
  198. package/dist/mcp-mock.js.map +1 -1
  199. package/dist/messages.cjs +2 -2
  200. package/dist/messages.cjs.map +1 -1
  201. package/dist/messages.d.cts +2 -2
  202. package/dist/messages.d.cts.map +1 -1
  203. package/dist/messages.d.ts +2 -2
  204. package/dist/messages.d.ts.map +1 -1
  205. package/dist/messages.js +2 -2
  206. package/dist/messages.js.map +1 -1
  207. package/dist/moderation.d.cts +3 -3
  208. package/dist/moderation.d.cts.map +1 -1
  209. package/dist/moderation.d.ts +3 -3
  210. package/dist/moderation.d.ts.map +1 -1
  211. package/dist/ndjson-writer.d.cts +2 -2
  212. package/dist/ndjson-writer.d.cts.map +1 -1
  213. package/dist/ndjson-writer.d.ts +2 -2
  214. package/dist/ndjson-writer.d.ts.map +1 -1
  215. package/dist/ollama.cjs +4 -4
  216. package/dist/ollama.cjs.map +1 -1
  217. package/dist/ollama.d.cts +3 -3
  218. package/dist/ollama.d.cts.map +1 -1
  219. package/dist/ollama.d.ts +3 -3
  220. package/dist/ollama.d.ts.map +1 -1
  221. package/dist/ollama.js +4 -4
  222. package/dist/ollama.js.map +1 -1
  223. package/dist/recorder.cjs +106 -38
  224. package/dist/recorder.cjs.map +1 -1
  225. package/dist/recorder.d.cts +52 -7
  226. package/dist/recorder.d.cts.map +1 -1
  227. package/dist/recorder.d.ts +52 -7
  228. package/dist/recorder.d.ts.map +1 -1
  229. package/dist/recorder.js +108 -40
  230. package/dist/recorder.js.map +1 -1
  231. package/dist/rerank.d.cts +3 -3
  232. package/dist/rerank.d.cts.map +1 -1
  233. package/dist/rerank.d.ts +3 -3
  234. package/dist/rerank.d.ts.map +1 -1
  235. package/dist/responses.cjs +2 -2
  236. package/dist/responses.cjs.map +1 -1
  237. package/dist/responses.d.cts +2 -2
  238. package/dist/responses.d.cts.map +1 -1
  239. package/dist/responses.d.ts +2 -2
  240. package/dist/responses.d.ts.map +1 -1
  241. package/dist/responses.js +2 -2
  242. package/dist/responses.js.map +1 -1
  243. package/dist/router.cjs +1 -1
  244. package/dist/router.cjs.map +1 -1
  245. package/dist/router.d.cts.map +1 -1
  246. package/dist/router.d.ts.map +1 -1
  247. package/dist/router.js +2 -2
  248. package/dist/router.js.map +1 -1
  249. package/dist/search.d.cts +3 -3
  250. package/dist/search.d.cts.map +1 -1
  251. package/dist/search.d.ts +3 -3
  252. package/dist/search.d.ts.map +1 -1
  253. package/dist/server.cjs +253 -8
  254. package/dist/server.cjs.map +1 -1
  255. package/dist/server.d.cts +2 -2
  256. package/dist/server.d.cts.map +1 -1
  257. package/dist/server.d.ts +2 -2
  258. package/dist/server.d.ts.map +1 -1
  259. package/dist/server.js +257 -12
  260. package/dist/server.js.map +1 -1
  261. package/dist/speech.cjs +20 -11
  262. package/dist/speech.cjs.map +1 -1
  263. package/dist/speech.d.cts +2 -2
  264. package/dist/speech.d.cts.map +1 -1
  265. package/dist/speech.d.ts +2 -2
  266. package/dist/speech.d.ts.map +1 -1
  267. package/dist/speech.js +20 -11
  268. package/dist/speech.js.map +1 -1
  269. package/dist/sse-writer.d.cts +3 -3
  270. package/dist/sse-writer.d.cts.map +1 -1
  271. package/dist/sse-writer.d.ts +3 -3
  272. package/dist/sse-writer.d.ts.map +1 -1
  273. package/dist/stream-collapse.cjs +80 -9
  274. package/dist/stream-collapse.cjs.map +1 -1
  275. package/dist/stream-collapse.d.cts +11 -1
  276. package/dist/stream-collapse.d.cts.map +1 -1
  277. package/dist/stream-collapse.d.ts +11 -1
  278. package/dist/stream-collapse.d.ts.map +1 -1
  279. package/dist/stream-collapse.js +80 -10
  280. package/dist/stream-collapse.js.map +1 -1
  281. package/dist/suite.cjs +1 -1
  282. package/dist/suite.d.cts +2 -2
  283. package/dist/suite.d.ts +2 -2
  284. package/dist/suite.js +1 -1
  285. package/dist/transcription.cjs +2 -2
  286. package/dist/transcription.cjs.map +1 -1
  287. package/dist/transcription.d.cts +2 -2
  288. package/dist/transcription.d.cts.map +1 -1
  289. package/dist/transcription.d.ts +2 -2
  290. package/dist/transcription.d.ts.map +1 -1
  291. package/dist/transcription.js +2 -2
  292. package/dist/transcription.js.map +1 -1
  293. package/dist/types.d.cts +38 -11
  294. package/dist/types.d.cts.map +1 -1
  295. package/dist/types.d.ts +38 -11
  296. package/dist/types.d.ts.map +1 -1
  297. package/dist/vector-mock.d.cts +2 -2
  298. package/dist/vector-mock.d.cts.map +1 -1
  299. package/dist/vector-mock.d.ts +2 -2
  300. package/dist/vector-mock.d.ts.map +1 -1
  301. package/dist/vector-mock.js +2 -2
  302. package/dist/vector-mock.js.map +1 -1
  303. package/dist/vector-types.d.cts.map +1 -1
  304. package/dist/vector-types.d.ts.map +1 -1
  305. package/dist/video.cjs +9 -3
  306. package/dist/video.cjs.map +1 -1
  307. package/dist/video.d.cts +3 -3
  308. package/dist/video.d.cts.map +1 -1
  309. package/dist/video.d.ts +3 -3
  310. package/dist/video.d.ts.map +1 -1
  311. package/dist/video.js +9 -3
  312. package/dist/video.js.map +1 -1
  313. package/dist/vitest.cjs +1 -1
  314. package/dist/vitest.js +1 -1
  315. package/dist/ws-framing.d.cts +2 -2
  316. package/dist/ws-framing.d.cts.map +1 -1
  317. package/dist/ws-framing.d.ts +2 -2
  318. package/dist/ws-framing.d.ts.map +1 -1
  319. package/dist/ws-gemini-live.cjs +145 -2
  320. package/dist/ws-gemini-live.cjs.map +1 -1
  321. package/dist/ws-gemini-live.d.cts.map +1 -1
  322. package/dist/ws-gemini-live.d.ts.map +1 -1
  323. package/dist/ws-gemini-live.js +146 -3
  324. package/dist/ws-gemini-live.js.map +1 -1
  325. package/package.json +16 -2
  326. package/skills/write-fixtures/SKILL.md +10 -10
package/dist/fal.cjs ADDED
@@ -0,0 +1,424 @@
1
+ const require_runtime = require('./_virtual/_rolldown/runtime.cjs');
2
+ const require_helpers = require('./helpers.cjs');
3
+ const require_router = require('./router.cjs');
4
+ const require_recorder = require('./recorder.cjs');
5
+ const require_fal_audio = require('./fal-audio.cjs');
6
+ let node_crypto = require("node:crypto");
7
+ node_crypto = require_runtime.__toESM(node_crypto);
8
+
9
+ //#region src/fal.ts
10
+ const FAL_QUEUE_MAX_ENTRIES = 1e4;
11
+ const FAL_QUEUE_TTL_MS = 36e5;
12
+ /**
13
+ * Per-testId queue state for the general fal handler. Mirrors FalJobMap from
14
+ * fal-audio.ts but stores arbitrary JSON payloads instead of audio file
15
+ * objects, so it can serve any fal model (image, video, motion, music, etc.).
16
+ */
17
+ var FalQueueStateMap = class {
18
+ entries = /* @__PURE__ */ new Map();
19
+ get(key) {
20
+ const entry = this.entries.get(key);
21
+ if (!entry) return void 0;
22
+ if (Date.now() - entry.createdAt > FAL_QUEUE_TTL_MS) {
23
+ this.entries.delete(key);
24
+ return;
25
+ }
26
+ return entry.job;
27
+ }
28
+ set(key, job) {
29
+ this.entries.set(key, {
30
+ job,
31
+ createdAt: Date.now()
32
+ });
33
+ if (this.entries.size > FAL_QUEUE_MAX_ENTRIES) {
34
+ const excess = this.entries.size - FAL_QUEUE_MAX_ENTRIES;
35
+ const iter = this.entries.keys();
36
+ for (let i = 0; i < excess; i++) {
37
+ const next = iter.next();
38
+ if (!next.done) this.entries.delete(next.value);
39
+ }
40
+ }
41
+ }
42
+ delete(key) {
43
+ return this.entries.delete(key);
44
+ }
45
+ clear() {
46
+ this.entries.clear();
47
+ }
48
+ get size() {
49
+ return this.entries.size;
50
+ }
51
+ };
52
+ const falQueueStates = new FalQueueStateMap();
53
+ const FAL_HOSTS = {
54
+ queue: "queue.fal.run",
55
+ sync: "fal.run",
56
+ storage: "rest.fal.ai",
57
+ storageAlpha: "rest.alpha.fal.ai",
58
+ gateway: "gateway.fal.ai"
59
+ };
60
+ const QUEUE_REQUESTS_RE = /^(.+)\/requests\/([^/]+)(\/status|\/cancel)?$/;
61
+ const STORAGE_INITIATE_PATH = "/storage/upload/initiate";
62
+ function stripFalPrefix(pathname) {
63
+ const stripped = pathname.replace(/^\/fal/, "");
64
+ return stripped.length > 0 ? stripped : "/";
65
+ }
66
+ function extractPromptFromBody(body) {
67
+ if (!body || typeof body !== "object") return "";
68
+ const obj = body;
69
+ if (typeof obj.prompt === "string") return obj.prompt;
70
+ if (typeof obj.text === "string") return obj.text;
71
+ const input = obj.input;
72
+ if (input && typeof input === "object") {
73
+ const inputObj = input;
74
+ if (typeof inputObj.prompt === "string") return inputObj.prompt;
75
+ if (typeof inputObj.text === "string") return inputObj.text;
76
+ }
77
+ return "";
78
+ }
79
+ function parseFalPath(stripped) {
80
+ if (!stripped.startsWith("/")) return null;
81
+ const trimmed = stripped.replace(/^\/+/, "");
82
+ if (!trimmed) return null;
83
+ const m = QUEUE_REQUESTS_RE.exec(`/${trimmed}`);
84
+ if (m) {
85
+ const modelId = m[1].replace(/^\/+/, "");
86
+ const action = m[3] === "/status" ? "status" : m[3] === "/cancel" ? "cancel" : "result";
87
+ return {
88
+ modelId,
89
+ requestId: m[2],
90
+ action
91
+ };
92
+ }
93
+ return { modelId: trimmed };
94
+ }
95
+ function classifyRoute(req, pathname, targetHost) {
96
+ const stripped = stripFalPrefix(pathname);
97
+ if (targetHost === FAL_HOSTS.storage || targetHost === FAL_HOSTS.storageAlpha) {
98
+ if (req.method === "POST" && stripped === STORAGE_INITIATE_PATH) return {
99
+ kind: "storage",
100
+ targetHost
101
+ };
102
+ return null;
103
+ }
104
+ const parsed = parseFalPath(stripped);
105
+ if (!parsed) return null;
106
+ if (targetHost === FAL_HOSTS.queue) {
107
+ if (parsed.requestId) {
108
+ if (parsed.action === "status" && req.method === "GET") return {
109
+ kind: "queue-status",
110
+ modelId: parsed.modelId,
111
+ requestId: parsed.requestId,
112
+ targetHost
113
+ };
114
+ if (parsed.action === "cancel" && req.method === "PUT") return {
115
+ kind: "queue-cancel",
116
+ modelId: parsed.modelId,
117
+ requestId: parsed.requestId,
118
+ targetHost
119
+ };
120
+ if (parsed.action === "result" && req.method === "GET") return {
121
+ kind: "queue-result",
122
+ modelId: parsed.modelId,
123
+ requestId: parsed.requestId,
124
+ targetHost
125
+ };
126
+ return null;
127
+ }
128
+ if (req.method === "POST") return {
129
+ kind: "queue-submit",
130
+ modelId: parsed.modelId,
131
+ targetHost
132
+ };
133
+ return null;
134
+ }
135
+ if (targetHost === FAL_HOSTS.sync) {
136
+ if (req.method === "POST" && parsed.modelId) return {
137
+ kind: "sync-run",
138
+ modelId: parsed.modelId,
139
+ targetHost
140
+ };
141
+ return null;
142
+ }
143
+ return null;
144
+ }
145
+ /**
146
+ * General fal.ai handler. Routes by `x-fal-target-host` header (the convention
147
+ * used by `@fal-ai/client`'s server-side requestMiddleware workaround for the
148
+ * fact that `proxyUrl` is browser-only).
149
+ *
150
+ * Returns `"passthrough"` when the request does not look like a host-mirrored
151
+ * fal call, so the caller can fall back to the legacy `/fal/queue/...` and
152
+ * `/fal/run/...` audio routes.
153
+ */
154
+ async function handleFal(req, res, body, pathname, fixtures, defaults, journal) {
155
+ const targetHostHeader = req.headers["x-fal-target-host"];
156
+ const targetHost = Array.isArray(targetHostHeader) ? targetHostHeader[0] : targetHostHeader;
157
+ if (!targetHost) return "passthrough";
158
+ const route = classifyRoute(req, pathname, targetHost);
159
+ if (!route) return "passthrough";
160
+ const testId = require_helpers.getTestId(req);
161
+ const stateKey = (id) => `${testId}:${id}`;
162
+ switch (route.kind) {
163
+ case "queue-status": {
164
+ const job = falQueueStates.get(stateKey(route.requestId));
165
+ if (!job) {
166
+ respondNotFound(req, res, pathname, journal, route.requestId);
167
+ return "handled";
168
+ }
169
+ writeJson(req, res, 200, {
170
+ status: job.status,
171
+ request_id: job.requestId,
172
+ response_url: `https://${FAL_HOSTS.queue}/${job.modelId}/requests/${job.requestId}`
173
+ }, pathname, journal);
174
+ return "handled";
175
+ }
176
+ case "queue-result": {
177
+ const job = falQueueStates.get(stateKey(route.requestId));
178
+ if (!job) {
179
+ respondNotFound(req, res, pathname, journal, route.requestId);
180
+ return "handled";
181
+ }
182
+ writeJson(req, res, 200, job.result, pathname, journal);
183
+ return "handled";
184
+ }
185
+ case "queue-cancel":
186
+ if (!falQueueStates.get(stateKey(route.requestId))) {
187
+ journal.add({
188
+ method: req.method ?? "PUT",
189
+ path: pathname,
190
+ headers: require_helpers.flattenHeaders(req.headers),
191
+ body: null,
192
+ response: {
193
+ status: 404,
194
+ fixture: null
195
+ }
196
+ });
197
+ res.writeHead(404, { "Content-Type": "application/json" });
198
+ res.end(JSON.stringify({ status: "NOT_FOUND" }));
199
+ return "handled";
200
+ }
201
+ journal.add({
202
+ method: req.method ?? "PUT",
203
+ path: pathname,
204
+ headers: require_helpers.flattenHeaders(req.headers),
205
+ body: null,
206
+ response: {
207
+ status: 400,
208
+ fixture: null
209
+ }
210
+ });
211
+ res.writeHead(400, { "Content-Type": "application/json" });
212
+ res.end(JSON.stringify({ status: "ALREADY_COMPLETED" }));
213
+ return "handled";
214
+ case "storage": {
215
+ let filename = "upload.bin";
216
+ try {
217
+ const parsed = body ? JSON.parse(body) : {};
218
+ if (typeof parsed.filename === "string") filename = parsed.filename;
219
+ if (typeof parsed.file_name === "string") filename = parsed.file_name;
220
+ } catch {}
221
+ const fileId = node_crypto.default.randomUUID();
222
+ writeJson(req, res, 200, {
223
+ upload_url: `https://${route.targetHost}/storage/upload/${fileId}`,
224
+ file_url: `https://${route.targetHost}/files/${fileId}/${filename}`
225
+ }, pathname, journal);
226
+ return "handled";
227
+ }
228
+ case "queue-submit":
229
+ case "sync-run": {
230
+ const modelId = route.modelId;
231
+ const parsedBody = parseBody(body);
232
+ const syntheticReq = {
233
+ model: modelId,
234
+ messages: [{
235
+ role: "user",
236
+ content: extractPromptFromBody(parsedBody) || JSON.stringify(parsedBody ?? {})
237
+ }],
238
+ _endpointType: "fal"
239
+ };
240
+ const fixture = require_router.matchFixture(fixtures, syntheticReq, journal.getFixtureMatchCountsForTest(testId), defaults.requestTransform);
241
+ if (!fixture) {
242
+ if (defaults.record) {
243
+ const effectiveDefaults = withFalUpstream(defaults, route.targetHost);
244
+ const outcome = await require_recorder.proxyAndRecord(req, res, syntheticReq, "fal", stripFalPrefix(pathname), fixtures, effectiveDefaults, body);
245
+ if (outcome === "handled_by_hook") return "handled";
246
+ if (outcome === "relayed") {
247
+ journal.add({
248
+ method: req.method ?? "POST",
249
+ path: pathname,
250
+ headers: require_helpers.flattenHeaders(req.headers),
251
+ body: syntheticReq,
252
+ response: {
253
+ status: res.statusCode ?? 200,
254
+ fixture: null,
255
+ source: "proxy"
256
+ }
257
+ });
258
+ return "handled";
259
+ }
260
+ }
261
+ const strictStatus = defaults.strict ? 503 : 404;
262
+ const strictMessage = defaults.strict ? "Strict mode: no fixture matched" : "No fixture matched";
263
+ journal.add({
264
+ method: req.method ?? "POST",
265
+ path: pathname,
266
+ headers: require_helpers.flattenHeaders(req.headers),
267
+ body: syntheticReq,
268
+ response: {
269
+ status: strictStatus,
270
+ fixture: null
271
+ }
272
+ });
273
+ res.writeHead(strictStatus, { "Content-Type": "application/json" });
274
+ res.end(JSON.stringify({ error: {
275
+ message: strictMessage,
276
+ type: "invalid_request_error",
277
+ code: "no_fixture_match"
278
+ } }));
279
+ return "handled";
280
+ }
281
+ journal.incrementFixtureMatchCount(fixture, fixtures, testId);
282
+ const response = fixture.response;
283
+ if (require_helpers.isErrorResponse(response)) {
284
+ const status = response.status ?? 500;
285
+ journal.add({
286
+ method: req.method ?? "POST",
287
+ path: pathname,
288
+ headers: require_helpers.flattenHeaders(req.headers),
289
+ body: syntheticReq,
290
+ response: {
291
+ status,
292
+ fixture
293
+ }
294
+ });
295
+ res.writeHead(status, { "Content-Type": "application/json" });
296
+ res.end(JSON.stringify(response));
297
+ return "handled";
298
+ }
299
+ let payload;
300
+ if (require_helpers.isJSONResponse(response)) payload = response.json;
301
+ else if (require_helpers.isAudioResponse(response)) payload = require_fal_audio.audioToFalFile(response);
302
+ else {
303
+ journal.add({
304
+ method: req.method ?? "POST",
305
+ path: pathname,
306
+ headers: require_helpers.flattenHeaders(req.headers),
307
+ body: syntheticReq,
308
+ response: {
309
+ status: 500,
310
+ fixture
311
+ }
312
+ });
313
+ res.writeHead(500, { "Content-Type": "application/json" });
314
+ res.end(JSON.stringify({ error: {
315
+ message: "Fixture response is not JSON or audio for fal endpoint",
316
+ type: "server_error"
317
+ } }));
318
+ return "handled";
319
+ }
320
+ if (route.kind === "sync-run") {
321
+ journal.add({
322
+ method: req.method ?? "POST",
323
+ path: pathname,
324
+ headers: require_helpers.flattenHeaders(req.headers),
325
+ body: syntheticReq,
326
+ response: {
327
+ status: 200,
328
+ fixture
329
+ }
330
+ });
331
+ res.writeHead(200, { "Content-Type": "application/json" });
332
+ res.end(JSON.stringify(payload));
333
+ return "handled";
334
+ }
335
+ const requestId = node_crypto.default.randomUUID();
336
+ falQueueStates.set(stateKey(requestId), {
337
+ requestId,
338
+ modelId,
339
+ status: "COMPLETED",
340
+ result: payload,
341
+ createdAt: Date.now()
342
+ });
343
+ const envelope = {
344
+ request_id: requestId,
345
+ response_url: `https://${FAL_HOSTS.queue}/${modelId}/requests/${requestId}`,
346
+ status_url: `https://${FAL_HOSTS.queue}/${modelId}/requests/${requestId}/status`,
347
+ cancel_url: `https://${FAL_HOSTS.queue}/${modelId}/requests/${requestId}/cancel`,
348
+ queue_position: 0
349
+ };
350
+ journal.add({
351
+ method: req.method ?? "POST",
352
+ path: pathname,
353
+ headers: require_helpers.flattenHeaders(req.headers),
354
+ body: syntheticReq,
355
+ response: {
356
+ status: 200,
357
+ fixture
358
+ }
359
+ });
360
+ res.writeHead(200, { "Content-Type": "application/json" });
361
+ res.end(JSON.stringify(envelope));
362
+ return "handled";
363
+ }
364
+ }
365
+ }
366
+ function parseBody(raw) {
367
+ if (!raw.trim()) return null;
368
+ try {
369
+ return JSON.parse(raw);
370
+ } catch {
371
+ return null;
372
+ }
373
+ }
374
+ function withFalUpstream(defaults, targetHost) {
375
+ if (!defaults.record) return defaults;
376
+ if (defaults.record.providers.fal) return defaults;
377
+ return {
378
+ ...defaults,
379
+ record: {
380
+ ...defaults.record,
381
+ providers: {
382
+ ...defaults.record.providers,
383
+ fal: `https://${targetHost}`
384
+ }
385
+ }
386
+ };
387
+ }
388
+ function writeJson(req, res, status, payload, pathname, journal) {
389
+ journal.add({
390
+ method: req.method ?? "GET",
391
+ path: pathname,
392
+ headers: require_helpers.flattenHeaders(req.headers),
393
+ body: null,
394
+ response: {
395
+ status,
396
+ fixture: null
397
+ }
398
+ });
399
+ res.writeHead(status, { "Content-Type": "application/json" });
400
+ res.end(JSON.stringify(payload));
401
+ }
402
+ function respondNotFound(req, res, pathname, journal, requestId) {
403
+ journal.add({
404
+ method: req.method ?? "GET",
405
+ path: pathname,
406
+ headers: require_helpers.flattenHeaders(req.headers),
407
+ body: null,
408
+ response: {
409
+ status: 404,
410
+ fixture: null
411
+ }
412
+ });
413
+ res.writeHead(404, { "Content-Type": "application/json" });
414
+ res.end(JSON.stringify({ error: {
415
+ message: `Request ${requestId} not found`,
416
+ type: "not_found"
417
+ } }));
418
+ }
419
+
420
+ //#endregion
421
+ exports.FalQueueStateMap = FalQueueStateMap;
422
+ exports.falQueueStates = falQueueStates;
423
+ exports.handleFal = handleFal;
424
+ //# sourceMappingURL=fal.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fal.cjs","names":["getTestId","flattenHeaders","crypto","matchFixture","proxyAndRecord","isErrorResponse","isJSONResponse","isAudioResponse","audioToFalFile"],"sources":["../src/fal.ts"],"sourcesContent":["import type http from \"node:http\";\nimport crypto from \"node:crypto\";\nimport type { ChatCompletionRequest, Fixture, HandlerDefaults, RawJSONResponse } from \"./types.js\";\nimport {\n isAudioResponse,\n isErrorResponse,\n isJSONResponse,\n flattenHeaders,\n getTestId,\n} from \"./helpers.js\";\nimport { matchFixture } from \"./router.js\";\nimport { proxyAndRecord } from \"./recorder.js\";\nimport type { Journal } from \"./journal.js\";\nimport { audioToFalFile } from \"./fal-audio.js\";\n\n// ─── FalQueueState (TTL + bounded) ───────────────────────────────────────\n\nconst FAL_QUEUE_MAX_ENTRIES = 10_000;\nconst FAL_QUEUE_TTL_MS = 3_600_000; // 1 hour\n\ninterface FalQueueJob {\n requestId: string;\n modelId: string;\n status: \"IN_QUEUE\" | \"IN_PROGRESS\" | \"COMPLETED\";\n result: unknown;\n createdAt: number;\n}\n\ninterface FalQueueEntry {\n job: FalQueueJob;\n createdAt: number;\n}\n\n/**\n * Per-testId queue state for the general fal handler. Mirrors FalJobMap from\n * fal-audio.ts but stores arbitrary JSON payloads instead of audio file\n * objects, so it can serve any fal model (image, video, motion, music, etc.).\n */\nexport class FalQueueStateMap {\n private readonly entries = new Map<string, FalQueueEntry>();\n\n get(key: string): FalQueueJob | undefined {\n const entry = this.entries.get(key);\n if (!entry) return undefined;\n if (Date.now() - entry.createdAt > FAL_QUEUE_TTL_MS) {\n this.entries.delete(key);\n return undefined;\n }\n return entry.job;\n }\n\n set(key: string, job: FalQueueJob): void {\n this.entries.set(key, { job, createdAt: Date.now() });\n if (this.entries.size > FAL_QUEUE_MAX_ENTRIES) {\n const excess = this.entries.size - FAL_QUEUE_MAX_ENTRIES;\n const iter = this.entries.keys();\n for (let i = 0; i < excess; i++) {\n const next = iter.next();\n if (!next.done) this.entries.delete(next.value);\n }\n }\n }\n\n delete(key: string): boolean {\n return this.entries.delete(key);\n }\n\n clear(): void {\n this.entries.clear();\n }\n\n get size(): number {\n return this.entries.size;\n }\n}\n\nexport const falQueueStates = new FalQueueStateMap();\n\n// ─── Hosts and routing ──────────────────────────────────────────────────\n\nconst FAL_HOSTS = {\n queue: \"queue.fal.run\",\n sync: \"fal.run\",\n storage: \"rest.fal.ai\",\n storageAlpha: \"rest.alpha.fal.ai\",\n gateway: \"gateway.fal.ai\",\n} as const;\n\nconst QUEUE_REQUESTS_RE = /^(.+)\\/requests\\/([^/]+)(\\/status|\\/cancel)?$/;\nconst STORAGE_INITIATE_PATH = \"/storage/upload/initiate\";\n\nfunction stripFalPrefix(pathname: string): string {\n const stripped = pathname.replace(/^\\/fal/, \"\");\n return stripped.length > 0 ? stripped : \"/\";\n}\n\nfunction extractPromptFromBody(body: unknown): string {\n if (!body || typeof body !== \"object\") return \"\";\n const obj = body as Record<string, unknown>;\n if (typeof obj.prompt === \"string\") return obj.prompt;\n if (typeof obj.text === \"string\") return obj.text;\n const input = obj.input;\n if (input && typeof input === \"object\") {\n const inputObj = input as Record<string, unknown>;\n if (typeof inputObj.prompt === \"string\") return inputObj.prompt;\n if (typeof inputObj.text === \"string\") return inputObj.text;\n }\n return \"\";\n}\n\ninterface ParsedFalPath {\n modelId: string;\n requestId?: string;\n action?: \"status\" | \"cancel\" | \"result\";\n}\n\nfunction parseFalPath(stripped: string): ParsedFalPath | null {\n if (!stripped.startsWith(\"/\")) return null;\n const trimmed = stripped.replace(/^\\/+/, \"\");\n if (!trimmed) return null;\n\n const m = QUEUE_REQUESTS_RE.exec(`/${trimmed}`);\n if (m) {\n const modelId = m[1].replace(/^\\/+/, \"\");\n const action = m[3] === \"/status\" ? \"status\" : m[3] === \"/cancel\" ? \"cancel\" : \"result\";\n return { modelId, requestId: m[2], action };\n }\n return { modelId: trimmed };\n}\n\nexport type HandleFalOutcome = \"handled\" | \"passthrough\";\n\ninterface FalRouteInfo {\n kind: \"queue-submit\" | \"queue-status\" | \"queue-result\" | \"queue-cancel\" | \"sync-run\" | \"storage\";\n modelId?: string;\n requestId?: string;\n targetHost: string;\n}\n\nfunction classifyRoute(\n req: http.IncomingMessage,\n pathname: string,\n targetHost: string,\n): FalRouteInfo | null {\n const stripped = stripFalPrefix(pathname);\n\n if (targetHost === FAL_HOSTS.storage || targetHost === FAL_HOSTS.storageAlpha) {\n if (req.method === \"POST\" && stripped === STORAGE_INITIATE_PATH) {\n return { kind: \"storage\", targetHost };\n }\n return null;\n }\n\n const parsed = parseFalPath(stripped);\n if (!parsed) return null;\n\n if (targetHost === FAL_HOSTS.queue) {\n if (parsed.requestId) {\n if (parsed.action === \"status\" && req.method === \"GET\") {\n return {\n kind: \"queue-status\",\n modelId: parsed.modelId,\n requestId: parsed.requestId,\n targetHost,\n };\n }\n if (parsed.action === \"cancel\" && req.method === \"PUT\") {\n return {\n kind: \"queue-cancel\",\n modelId: parsed.modelId,\n requestId: parsed.requestId,\n targetHost,\n };\n }\n if (parsed.action === \"result\" && req.method === \"GET\") {\n return {\n kind: \"queue-result\",\n modelId: parsed.modelId,\n requestId: parsed.requestId,\n targetHost,\n };\n }\n return null;\n }\n if (req.method === \"POST\") {\n return { kind: \"queue-submit\", modelId: parsed.modelId, targetHost };\n }\n return null;\n }\n\n if (targetHost === FAL_HOSTS.sync) {\n if (req.method === \"POST\" && parsed.modelId) {\n return { kind: \"sync-run\", modelId: parsed.modelId, targetHost };\n }\n return null;\n }\n\n return null;\n}\n\n/**\n * General fal.ai handler. Routes by `x-fal-target-host` header (the convention\n * used by `@fal-ai/client`'s server-side requestMiddleware workaround for the\n * fact that `proxyUrl` is browser-only).\n *\n * Returns `\"passthrough\"` when the request does not look like a host-mirrored\n * fal call, so the caller can fall back to the legacy `/fal/queue/...` and\n * `/fal/run/...` audio routes.\n */\nexport async function handleFal(\n req: http.IncomingMessage,\n res: http.ServerResponse,\n body: string,\n pathname: string,\n fixtures: Fixture[],\n defaults: HandlerDefaults,\n journal: Journal,\n): Promise<HandleFalOutcome> {\n const targetHostHeader = req.headers[\"x-fal-target-host\"];\n const targetHost = Array.isArray(targetHostHeader) ? targetHostHeader[0] : targetHostHeader;\n if (!targetHost) return \"passthrough\";\n\n const route = classifyRoute(req, pathname, targetHost);\n if (!route) return \"passthrough\";\n\n const testId = getTestId(req);\n const stateKey = (id: string) => `${testId}:${id}`;\n\n switch (route.kind) {\n case \"queue-status\": {\n const job = falQueueStates.get(stateKey(route.requestId!));\n if (!job) {\n respondNotFound(req, res, pathname, journal, route.requestId!);\n return \"handled\";\n }\n const responseBody = {\n status: job.status,\n request_id: job.requestId,\n response_url: `https://${FAL_HOSTS.queue}/${job.modelId}/requests/${job.requestId}`,\n };\n writeJson(req, res, 200, responseBody, pathname, journal);\n return \"handled\";\n }\n\n case \"queue-result\": {\n const job = falQueueStates.get(stateKey(route.requestId!));\n if (!job) {\n respondNotFound(req, res, pathname, journal, route.requestId!);\n return \"handled\";\n }\n writeJson(req, res, 200, job.result, pathname, journal);\n return \"handled\";\n }\n\n case \"queue-cancel\": {\n const job = falQueueStates.get(stateKey(route.requestId!));\n if (!job) {\n journal.add({\n method: req.method ?? \"PUT\",\n path: pathname,\n headers: flattenHeaders(req.headers),\n body: null,\n response: { status: 404, fixture: null },\n });\n res.writeHead(404, { \"Content-Type\": \"application/json\" });\n res.end(JSON.stringify({ status: \"NOT_FOUND\" }));\n return \"handled\";\n }\n journal.add({\n method: req.method ?? \"PUT\",\n path: pathname,\n headers: flattenHeaders(req.headers),\n body: null,\n response: { status: 400, fixture: null },\n });\n res.writeHead(400, { \"Content-Type\": \"application/json\" });\n res.end(JSON.stringify({ status: \"ALREADY_COMPLETED\" }));\n return \"handled\";\n }\n\n case \"storage\": {\n let filename = \"upload.bin\";\n try {\n const parsed = body ? (JSON.parse(body) as Record<string, unknown>) : {};\n if (typeof parsed.filename === \"string\") filename = parsed.filename;\n if (typeof parsed.file_name === \"string\") filename = parsed.file_name;\n } catch {\n // ignore — stub doesn't require a structured body\n }\n const fileId = crypto.randomUUID();\n const responseBody = {\n upload_url: `https://${route.targetHost}/storage/upload/${fileId}`,\n file_url: `https://${route.targetHost}/files/${fileId}/${filename}`,\n };\n writeJson(req, res, 200, responseBody, pathname, journal);\n return \"handled\";\n }\n\n case \"queue-submit\":\n case \"sync-run\": {\n const modelId = route.modelId!;\n const parsedBody = parseBody(body);\n const prompt = extractPromptFromBody(parsedBody);\n const syntheticReq: ChatCompletionRequest = {\n model: modelId,\n messages: [{ role: \"user\", content: prompt || JSON.stringify(parsedBody ?? {}) }],\n _endpointType: \"fal\",\n };\n\n const matchCounts = journal.getFixtureMatchCountsForTest(testId);\n const fixture = matchFixture(fixtures, syntheticReq, matchCounts, defaults.requestTransform);\n\n if (!fixture) {\n if (defaults.record) {\n const effectiveDefaults = withFalUpstream(defaults, route.targetHost);\n const outcome = await proxyAndRecord(\n req,\n res,\n syntheticReq,\n \"fal\",\n stripFalPrefix(pathname),\n fixtures,\n effectiveDefaults,\n body,\n );\n if (outcome === \"handled_by_hook\") return \"handled\";\n if (outcome === \"relayed\") {\n journal.add({\n method: req.method ?? \"POST\",\n path: pathname,\n headers: flattenHeaders(req.headers),\n body: syntheticReq,\n response: { status: res.statusCode ?? 200, fixture: null, source: \"proxy\" },\n });\n return \"handled\";\n }\n }\n\n const strictStatus = defaults.strict ? 503 : 404;\n const strictMessage = defaults.strict\n ? \"Strict mode: no fixture matched\"\n : \"No fixture matched\";\n journal.add({\n method: req.method ?? \"POST\",\n path: pathname,\n headers: flattenHeaders(req.headers),\n body: syntheticReq,\n response: { status: strictStatus, fixture: null },\n });\n res.writeHead(strictStatus, { \"Content-Type\": \"application/json\" });\n res.end(\n JSON.stringify({\n error: {\n message: strictMessage,\n type: \"invalid_request_error\",\n code: \"no_fixture_match\",\n },\n }),\n );\n return \"handled\";\n }\n\n journal.incrementFixtureMatchCount(fixture, fixtures, testId);\n const response = fixture.response;\n\n if (isErrorResponse(response)) {\n const status = response.status ?? 500;\n journal.add({\n method: req.method ?? \"POST\",\n path: pathname,\n headers: flattenHeaders(req.headers),\n body: syntheticReq,\n response: { status, fixture },\n });\n res.writeHead(status, { \"Content-Type\": \"application/json\" });\n res.end(JSON.stringify(response));\n return \"handled\";\n }\n\n let payload: unknown;\n if (isJSONResponse(response)) {\n payload = (response as RawJSONResponse).json;\n } else if (isAudioResponse(response)) {\n payload = audioToFalFile(response);\n } else {\n journal.add({\n method: req.method ?? \"POST\",\n path: pathname,\n headers: flattenHeaders(req.headers),\n body: syntheticReq,\n response: { status: 500, fixture },\n });\n res.writeHead(500, { \"Content-Type\": \"application/json\" });\n res.end(\n JSON.stringify({\n error: {\n message: \"Fixture response is not JSON or audio for fal endpoint\",\n type: \"server_error\",\n },\n }),\n );\n return \"handled\";\n }\n\n if (route.kind === \"sync-run\") {\n journal.add({\n method: req.method ?? \"POST\",\n path: pathname,\n headers: flattenHeaders(req.headers),\n body: syntheticReq,\n response: { status: 200, fixture },\n });\n res.writeHead(200, { \"Content-Type\": \"application/json\" });\n res.end(JSON.stringify(payload));\n return \"handled\";\n }\n\n const requestId = crypto.randomUUID();\n falQueueStates.set(stateKey(requestId), {\n requestId,\n modelId,\n status: \"COMPLETED\",\n result: payload,\n createdAt: Date.now(),\n });\n const envelope = {\n request_id: requestId,\n response_url: `https://${FAL_HOSTS.queue}/${modelId}/requests/${requestId}`,\n status_url: `https://${FAL_HOSTS.queue}/${modelId}/requests/${requestId}/status`,\n cancel_url: `https://${FAL_HOSTS.queue}/${modelId}/requests/${requestId}/cancel`,\n queue_position: 0,\n };\n journal.add({\n method: req.method ?? \"POST\",\n path: pathname,\n headers: flattenHeaders(req.headers),\n body: syntheticReq,\n response: { status: 200, fixture },\n });\n res.writeHead(200, { \"Content-Type\": \"application/json\" });\n res.end(JSON.stringify(envelope));\n return \"handled\";\n }\n }\n}\n\nfunction parseBody(raw: string): Record<string, unknown> | null {\n if (!raw.trim()) return null;\n try {\n return JSON.parse(raw) as Record<string, unknown>;\n } catch {\n return null;\n }\n}\n\nfunction withFalUpstream(defaults: HandlerDefaults, targetHost: string): HandlerDefaults {\n if (!defaults.record) return defaults;\n // Respect an explicit record.providers.fal — tests and dev configs need to\n // point at a stub upstream. Only synthesise from the header when the user\n // didn't configure one (the \"or omit upstream URL — it's in the request\n // hostname\" mode from the issue).\n if (defaults.record.providers.fal) return defaults;\n return {\n ...defaults,\n record: {\n ...defaults.record,\n providers: {\n ...defaults.record.providers,\n fal: `https://${targetHost}`,\n },\n },\n };\n}\n\nfunction writeJson(\n req: http.IncomingMessage,\n res: http.ServerResponse,\n status: number,\n payload: unknown,\n pathname: string,\n journal: Journal,\n): void {\n journal.add({\n method: req.method ?? \"GET\",\n path: pathname,\n headers: flattenHeaders(req.headers),\n body: null,\n response: { status, fixture: null },\n });\n res.writeHead(status, { \"Content-Type\": \"application/json\" });\n res.end(JSON.stringify(payload));\n}\n\nfunction respondNotFound(\n req: http.IncomingMessage,\n res: http.ServerResponse,\n pathname: string,\n journal: Journal,\n requestId: string,\n): void {\n journal.add({\n method: req.method ?? \"GET\",\n path: pathname,\n headers: flattenHeaders(req.headers),\n body: null,\n response: { status: 404, fixture: null },\n });\n res.writeHead(404, { \"Content-Type\": \"application/json\" });\n res.end(\n JSON.stringify({\n error: { message: `Request ${requestId} not found`, type: \"not_found\" },\n }),\n );\n}\n"],"mappings":"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"}
package/dist/fal.d.cts ADDED
@@ -0,0 +1,39 @@
1
+ import { Journal } from "./journal.cjs";
2
+ import { Fixture, HandlerDefaults } from "./types.cjs";
3
+ import http from "node:http";
4
+
5
+ //#region src/fal.d.ts
6
+ interface FalQueueJob {
7
+ requestId: string;
8
+ modelId: string;
9
+ status: "IN_QUEUE" | "IN_PROGRESS" | "COMPLETED";
10
+ result: unknown;
11
+ createdAt: number;
12
+ }
13
+ /**
14
+ * Per-testId queue state for the general fal handler. Mirrors FalJobMap from
15
+ * fal-audio.ts but stores arbitrary JSON payloads instead of audio file
16
+ * objects, so it can serve any fal model (image, video, motion, music, etc.).
17
+ */
18
+ declare class FalQueueStateMap {
19
+ private readonly entries;
20
+ get(key: string): FalQueueJob | undefined;
21
+ set(key: string, job: FalQueueJob): void;
22
+ delete(key: string): boolean;
23
+ clear(): void;
24
+ get size(): number;
25
+ }
26
+ type HandleFalOutcome = "handled" | "passthrough";
27
+ /**
28
+ * General fal.ai handler. Routes by `x-fal-target-host` header (the convention
29
+ * used by `@fal-ai/client`'s server-side requestMiddleware workaround for the
30
+ * fact that `proxyUrl` is browser-only).
31
+ *
32
+ * Returns `"passthrough"` when the request does not look like a host-mirrored
33
+ * fal call, so the caller can fall back to the legacy `/fal/queue/...` and
34
+ * `/fal/run/...` audio routes.
35
+ */
36
+ declare function handleFal(req: http.IncomingMessage, res: http.ServerResponse, body: string, pathname: string, fixtures: Fixture[], defaults: HandlerDefaults, journal: Journal): Promise<HandleFalOutcome>;
37
+ //#endregion
38
+ export { FalQueueStateMap, handleFal };
39
+ //# sourceMappingURL=fal.d.cts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fal.d.cts","names":[],"sources":["../src/fal.ts"],"sourcesContent":[],"mappings":";;;;;UAoBU,WAAA;;EAAA,OAAA,EAAA,MAAW;EAkBR,MAAA,EAAA,UAAA,GAAgB,aAAA,GAAA,WAAA;EAAA,MAAA,EAAA,OAAA;WAGT,EAAA,MAAA;;;AAyFpB;AA+EA;;;AAEO,cA7KM,gBAAA,CA6KD;mBAGA,OAAA;KACA,CAAA,GAAA,EAAA,MAAA,CAAA,EA9KQ,WA8KR,GAAA,SAAA;KACD,CAAA,GAAA,EAAA,MAAA,EAAA,GAAA,EArKa,WAqKb,CAAA,EAAA,IAAA;QACA,CAAA,GAAA,EAAA,MAAA,CAAA,EAAA,OAAA;OAAR,CAAA,CAAA,EAAA,IAAA;EAAO,IAAA,IAAA,CAAA,CAAA,EAAA,MAAA;;KAvFE,gBAAA;;;;;;;;;;iBA+EU,SAAA,MACf,IAAA,CAAK,sBACL,IAAA,CAAK,0DAGA,qBACA,0BACD,UACR,QAAQ"}
package/dist/fal.d.ts ADDED
@@ -0,0 +1,39 @@
1
+ import { Journal } from "./journal.js";
2
+ import { Fixture, HandlerDefaults } from "./types.js";
3
+ import http from "node:http";
4
+
5
+ //#region src/fal.d.ts
6
+ interface FalQueueJob {
7
+ requestId: string;
8
+ modelId: string;
9
+ status: "IN_QUEUE" | "IN_PROGRESS" | "COMPLETED";
10
+ result: unknown;
11
+ createdAt: number;
12
+ }
13
+ /**
14
+ * Per-testId queue state for the general fal handler. Mirrors FalJobMap from
15
+ * fal-audio.ts but stores arbitrary JSON payloads instead of audio file
16
+ * objects, so it can serve any fal model (image, video, motion, music, etc.).
17
+ */
18
+ declare class FalQueueStateMap {
19
+ private readonly entries;
20
+ get(key: string): FalQueueJob | undefined;
21
+ set(key: string, job: FalQueueJob): void;
22
+ delete(key: string): boolean;
23
+ clear(): void;
24
+ get size(): number;
25
+ }
26
+ type HandleFalOutcome = "handled" | "passthrough";
27
+ /**
28
+ * General fal.ai handler. Routes by `x-fal-target-host` header (the convention
29
+ * used by `@fal-ai/client`'s server-side requestMiddleware workaround for the
30
+ * fact that `proxyUrl` is browser-only).
31
+ *
32
+ * Returns `"passthrough"` when the request does not look like a host-mirrored
33
+ * fal call, so the caller can fall back to the legacy `/fal/queue/...` and
34
+ * `/fal/run/...` audio routes.
35
+ */
36
+ declare function handleFal(req: http.IncomingMessage, res: http.ServerResponse, body: string, pathname: string, fixtures: Fixture[], defaults: HandlerDefaults, journal: Journal): Promise<HandleFalOutcome>;
37
+ //#endregion
38
+ export { FalQueueStateMap, handleFal };
39
+ //# sourceMappingURL=fal.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fal.d.ts","names":[],"sources":["../src/fal.ts"],"sourcesContent":[],"mappings":";;;;;UAoBU,WAAA;;EAAA,OAAA,EAAA,MAAW;EAkBR,MAAA,EAAA,UAAA,GAAgB,aAAA,GAAA,WAAA;EAAA,MAAA,EAAA,OAAA;WAGT,EAAA,MAAA;;;AAyFpB;AA+EA;;;AAEO,cA7KM,gBAAA,CA6KD;mBAGA,OAAA;KACA,CAAA,GAAA,EAAA,MAAA,CAAA,EA9KQ,WA8KR,GAAA,SAAA;KACD,CAAA,GAAA,EAAA,MAAA,EAAA,GAAA,EArKa,WAqKb,CAAA,EAAA,IAAA;QACA,CAAA,GAAA,EAAA,MAAA,CAAA,EAAA,OAAA;OAAR,CAAA,CAAA,EAAA,IAAA;EAAO,IAAA,IAAA,CAAA,CAAA,EAAA,MAAA;;KAvFE,gBAAA;;;;;;;;;;iBA+EU,SAAA,MACf,IAAA,CAAK,sBACL,IAAA,CAAK,0DAGA,qBACA,0BACD,UACR,QAAQ"}