whale-code 6.5.5 → 6.5.6

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 (847) hide show
  1. package/README.md +39 -31
  2. package/bin/{swagmanager-mcp.js → whale-code.js} +17 -2
  3. package/dist/cli/app.js +148 -72
  4. package/dist/cli/app.js.map +1 -0
  5. package/dist/cli/chat/AgentSelector.js +105 -10
  6. package/dist/cli/chat/AgentSelector.js.map +1 -0
  7. package/dist/cli/chat/ChatApp.d.ts +31 -0
  8. package/dist/cli/chat/ChatApp.js +539 -286
  9. package/dist/cli/chat/ChatApp.js.map +1 -0
  10. package/dist/cli/chat/ChatInput.js +1088 -770
  11. package/dist/cli/chat/ChatInput.js.map +1 -0
  12. package/dist/cli/chat/MarkdownText.js +39 -14
  13. package/dist/cli/chat/MarkdownText.js.map +1 -0
  14. package/dist/cli/chat/MemoryManager.js +181 -46
  15. package/dist/cli/chat/MemoryManager.js.map +1 -0
  16. package/dist/cli/chat/MessageList.d.ts +2 -3
  17. package/dist/cli/chat/MessageList.js +186 -45
  18. package/dist/cli/chat/MessageList.js.map +1 -0
  19. package/dist/cli/chat/ModelSelector.js +282 -63
  20. package/dist/cli/chat/ModelSelector.js.map +1 -0
  21. package/dist/cli/chat/NodeManager.js +165 -75
  22. package/dist/cli/chat/NodeManager.js.map +1 -0
  23. package/dist/cli/chat/NodeSelector.js +171 -30
  24. package/dist/cli/chat/NodeSelector.js.map +1 -0
  25. package/dist/cli/chat/PlanApproval.js +281 -57
  26. package/dist/cli/chat/PlanApproval.js.map +1 -0
  27. package/dist/cli/chat/RewindViewer.js +559 -144
  28. package/dist/cli/chat/RewindViewer.js.map +1 -0
  29. package/dist/cli/chat/SessionManager.js +137 -30
  30. package/dist/cli/chat/SessionManager.js.map +1 -0
  31. package/dist/cli/chat/SlashMenu.js +293 -164
  32. package/dist/cli/chat/SlashMenu.js.map +1 -0
  33. package/dist/cli/chat/StatusBar.js +172 -9
  34. package/dist/cli/chat/StatusBar.js.map +1 -0
  35. package/dist/cli/chat/StoreSelector.js +147 -18
  36. package/dist/cli/chat/StoreSelector.js.map +1 -0
  37. package/dist/cli/chat/StreamingText.d.ts +1 -5
  38. package/dist/cli/chat/StreamingText.js +22 -7
  39. package/dist/cli/chat/StreamingText.js.map +1 -0
  40. package/dist/cli/chat/SubagentPanel.d.ts +1 -2
  41. package/dist/cli/chat/SubagentPanel.js +612 -72
  42. package/dist/cli/chat/SubagentPanel.js.map +1 -0
  43. package/dist/cli/chat/TeamPanel.d.ts +1 -0
  44. package/dist/cli/chat/TeamPanel.js +230 -30
  45. package/dist/cli/chat/TeamPanel.js.map +1 -0
  46. package/dist/cli/chat/ThemeSelector.js +84 -24
  47. package/dist/cli/chat/ThemeSelector.js.map +1 -0
  48. package/dist/cli/chat/ToolIndicator.js +1476 -371
  49. package/dist/cli/chat/ToolIndicator.js.map +1 -0
  50. package/dist/cli/chat/hooks/useAgentLoop.d.ts +1 -0
  51. package/dist/cli/chat/hooks/useAgentLoop.js +481 -367
  52. package/dist/cli/chat/hooks/useAgentLoop.js.map +1 -0
  53. package/dist/cli/chat/hooks/useSlashCommands.d.ts +3 -14
  54. package/dist/cli/chat/hooks/useSlashCommands.js +744 -572
  55. package/dist/cli/chat/hooks/useSlashCommands.js.map +1 -0
  56. package/dist/cli/commands/config-cmd.js +56 -57
  57. package/dist/cli/commands/config-cmd.js.map +1 -0
  58. package/dist/cli/commands/db.js +184 -169
  59. package/dist/cli/commands/db.js.map +1 -0
  60. package/dist/cli/commands/doctor.js +212 -122
  61. package/dist/cli/commands/doctor.js.map +1 -0
  62. package/dist/cli/commands/init.js +211 -244
  63. package/dist/cli/commands/init.js.map +1 -0
  64. package/dist/cli/commands/mcp.js +127 -122
  65. package/dist/cli/commands/mcp.js.map +1 -0
  66. package/dist/cli/login/LoginApp.js +355 -141
  67. package/dist/cli/login/LoginApp.js.map +1 -0
  68. package/dist/cli/print-mode.js +196 -177
  69. package/dist/cli/print-mode.js.map +1 -0
  70. package/dist/cli/serve-mode.js +615 -530
  71. package/dist/cli/serve-mode.js.map +1 -0
  72. package/dist/cli/services/agent-config.d.ts +5 -1
  73. package/dist/cli/services/agent-config.js +66 -36
  74. package/dist/cli/services/agent-config.js.map +1 -0
  75. package/dist/cli/services/agent-definitions.d.ts +4 -1
  76. package/dist/cli/services/agent-definitions.js +97 -56
  77. package/dist/cli/services/agent-definitions.js.map +1 -0
  78. package/dist/cli/services/agent-events.js +225 -162
  79. package/dist/cli/services/agent-events.js.map +1 -0
  80. package/dist/cli/services/agent-loop.js +976 -688
  81. package/dist/cli/services/agent-loop.js.map +1 -0
  82. package/dist/cli/services/agent-worker-base.d.ts +35 -5
  83. package/dist/cli/services/agent-worker-base.js +337 -153
  84. package/dist/cli/services/agent-worker-base.js.map +1 -0
  85. package/dist/cli/services/api-retry.js +69 -64
  86. package/dist/cli/services/api-retry.js.map +1 -0
  87. package/dist/cli/services/auth-service.d.ts +3 -3
  88. package/dist/cli/services/auth-service.js +209 -132
  89. package/dist/cli/services/auth-service.js.map +1 -0
  90. package/dist/cli/services/background-processes.js +343 -267
  91. package/dist/cli/services/background-processes.js.map +1 -0
  92. package/dist/cli/services/browser-auth.d.ts +2 -2
  93. package/dist/cli/services/browser-auth.js +159 -118
  94. package/dist/cli/services/browser-auth.js.map +1 -0
  95. package/dist/cli/services/claude-md-loader.js +40 -36
  96. package/dist/cli/services/claude-md-loader.js.map +1 -0
  97. package/dist/cli/services/config-store.d.ts +9 -4
  98. package/dist/cli/services/config-store.js +164 -117
  99. package/dist/cli/services/config-store.js.map +1 -0
  100. package/dist/cli/services/debug-log.d.ts +1 -1
  101. package/dist/cli/services/debug-log.js +34 -35
  102. package/dist/cli/services/debug-log.js.map +1 -0
  103. package/dist/cli/services/env-detect.d.ts +7 -0
  104. package/dist/cli/services/env-detect.js +9 -0
  105. package/dist/cli/services/env-detect.js.map +1 -0
  106. package/dist/cli/services/error-logger.js +187 -169
  107. package/dist/cli/services/error-logger.js.map +1 -0
  108. package/dist/cli/services/file-history.d.ts +1 -1
  109. package/dist/cli/services/file-history.js +50 -54
  110. package/dist/cli/services/file-history.js.map +1 -0
  111. package/dist/cli/services/format-server-response.js +332 -372
  112. package/dist/cli/services/format-server-response.js.map +1 -0
  113. package/dist/cli/services/git-context.js +61 -45
  114. package/dist/cli/services/git-context.js.map +1 -0
  115. package/dist/cli/services/hooks.d.ts +2 -2
  116. package/dist/cli/services/hooks.js +195 -180
  117. package/dist/cli/services/hooks.js.map +1 -0
  118. package/dist/cli/services/ink-incremental.d.ts +19 -0
  119. package/dist/cli/services/ink-incremental.js +59 -0
  120. package/dist/cli/services/ink-incremental.js.map +1 -0
  121. package/dist/cli/services/ink-resize-fix.js +54 -44
  122. package/dist/cli/services/ink-resize-fix.js.map +1 -0
  123. package/dist/cli/services/ink-sync-output.d.ts +12 -0
  124. package/dist/cli/services/ink-sync-output.js +16 -0
  125. package/dist/cli/services/ink-sync-output.js.map +1 -0
  126. package/dist/cli/services/interactive-tools.js +268 -212
  127. package/dist/cli/services/interactive-tools.js.map +1 -0
  128. package/dist/cli/services/keybinding-manager.d.ts +11 -1
  129. package/dist/cli/services/keybinding-manager.js +126 -63
  130. package/dist/cli/services/keybinding-manager.js.map +1 -0
  131. package/dist/cli/services/local-tools.d.ts +1 -1
  132. package/dist/cli/services/local-tools.js +939 -656
  133. package/dist/cli/services/local-tools.js.map +1 -0
  134. package/dist/cli/services/lsp-manager.js +757 -594
  135. package/dist/cli/services/lsp-manager.js.map +1 -0
  136. package/dist/cli/services/mcp-client.d.ts +1 -1
  137. package/dist/cli/services/mcp-client.js +173 -134
  138. package/dist/cli/services/mcp-client.js.map +1 -0
  139. package/dist/cli/services/memory-manager.js +53 -40
  140. package/dist/cli/services/memory-manager.js.map +1 -0
  141. package/dist/cli/services/model-manager.js +55 -40
  142. package/dist/cli/services/model-manager.js.map +1 -0
  143. package/dist/cli/services/model-router.js +115 -85
  144. package/dist/cli/services/model-router.js.map +1 -0
  145. package/dist/cli/services/paths.d.ts +30 -0
  146. package/dist/cli/services/paths.js +81 -0
  147. package/dist/cli/services/paths.js.map +1 -0
  148. package/dist/cli/services/permission-modes.js +32 -25
  149. package/dist/cli/services/permission-modes.js.map +1 -0
  150. package/dist/cli/services/rewind.js +182 -168
  151. package/dist/cli/services/rewind.js.map +1 -0
  152. package/dist/cli/services/ripgrep.js +115 -115
  153. package/dist/cli/services/ripgrep.js.map +1 -0
  154. package/dist/cli/services/sandbox.d.ts +1 -1
  155. package/dist/cli/services/sandbox.js +58 -37
  156. package/dist/cli/services/sandbox.js.map +1 -0
  157. package/dist/cli/services/server-tools.js +738 -565
  158. package/dist/cli/services/server-tools.js.map +1 -0
  159. package/dist/cli/services/session-persistence.js +69 -74
  160. package/dist/cli/services/session-persistence.js.map +1 -0
  161. package/dist/cli/services/subagent-worker.js +42 -27
  162. package/dist/cli/services/subagent-worker.js.map +1 -0
  163. package/dist/cli/services/subagent.d.ts +2 -0
  164. package/dist/cli/services/subagent.js +605 -433
  165. package/dist/cli/services/subagent.js.map +1 -0
  166. package/dist/cli/services/system-prompt.js +86 -78
  167. package/dist/cli/services/system-prompt.js.map +1 -0
  168. package/dist/cli/services/task-decomposer.d.ts +1 -1
  169. package/dist/cli/services/task-decomposer.js +172 -139
  170. package/dist/cli/services/task-decomposer.js.map +1 -0
  171. package/dist/cli/services/team-lead.d.ts +2 -2
  172. package/dist/cli/services/team-lead.js +727 -529
  173. package/dist/cli/services/team-lead.js.map +1 -0
  174. package/dist/cli/services/team-state.js +319 -319
  175. package/dist/cli/services/team-state.js.map +1 -0
  176. package/dist/cli/services/teammate.d.ts +8 -2
  177. package/dist/cli/services/teammate.js +857 -569
  178. package/dist/cli/services/teammate.js.map +1 -0
  179. package/dist/cli/services/telemetry.d.ts +6 -1
  180. package/dist/cli/services/telemetry.js +180 -157
  181. package/dist/cli/services/telemetry.js.map +1 -0
  182. package/dist/cli/services/tools/agent-tools.d.ts +3 -3
  183. package/dist/cli/services/tools/agent-tools.js +480 -322
  184. package/dist/cli/services/tools/agent-tools.js.map +1 -0
  185. package/dist/cli/services/tools/file-ops.js +563 -450
  186. package/dist/cli/services/tools/file-ops.js.map +1 -0
  187. package/dist/cli/services/tools/search-tools.js +231 -162
  188. package/dist/cli/services/tools/search-tools.js.map +1 -0
  189. package/dist/cli/services/tools/shell-exec.js +197 -151
  190. package/dist/cli/services/tools/shell-exec.js.map +1 -0
  191. package/dist/cli/services/tools/task-manager.js +206 -173
  192. package/dist/cli/services/tools/task-manager.js.map +1 -0
  193. package/dist/cli/services/tools/web-tools.js +388 -341
  194. package/dist/cli/services/tools/web-tools.js.map +1 -0
  195. package/dist/cli/setup/SetupApp.d.ts +2 -2
  196. package/dist/cli/setup/SetupApp.js +608 -160
  197. package/dist/cli/setup/SetupApp.js.map +1 -0
  198. package/dist/cli/shared/ErrorBoundary.d.ts +22 -0
  199. package/dist/cli/shared/ErrorBoundary.js +73 -0
  200. package/dist/cli/shared/ErrorBoundary.js.map +1 -0
  201. package/dist/cli/shared/MatrixIntro.js +66 -69
  202. package/dist/cli/shared/MatrixIntro.js.map +1 -0
  203. package/dist/cli/shared/SpinnerSlot.d.ts +14 -0
  204. package/dist/cli/shared/SpinnerSlot.js +63 -0
  205. package/dist/cli/shared/SpinnerSlot.js.map +1 -0
  206. package/dist/cli/shared/Theme.d.ts +1 -1
  207. package/dist/cli/shared/Theme.js +136 -92
  208. package/dist/cli/shared/Theme.js.map +1 -0
  209. package/dist/cli/shared/WhaleBanner.js +99 -11
  210. package/dist/cli/shared/WhaleBanner.js.map +1 -0
  211. package/dist/cli/shared/markdown.d.ts +3 -1
  212. package/dist/cli/shared/markdown.js +736 -674
  213. package/dist/cli/shared/markdown.js.map +1 -0
  214. package/dist/cli/shared/marked-terminal.d.js +2 -0
  215. package/dist/cli/shared/marked-terminal.d.js.map +1 -0
  216. package/dist/cli/shared/theme-manager.js +99 -90
  217. package/dist/cli/shared/theme-manager.js.map +1 -0
  218. package/dist/cli/shared/theme-presets.js +256 -254
  219. package/dist/cli/shared/theme-presets.js.map +1 -0
  220. package/dist/cli/status/StatusApp.js +235 -86
  221. package/dist/cli/status/StatusApp.js.map +1 -0
  222. package/dist/cli/stores/StoreApp.js +275 -65
  223. package/dist/cli/stores/StoreApp.js.map +1 -0
  224. package/dist/index.d.ts +2 -2
  225. package/dist/index.js +509 -396
  226. package/dist/index.js.map +1 -0
  227. package/dist/local-agent/connection.d.ts +2 -2
  228. package/dist/local-agent/connection.js +352 -293
  229. package/dist/local-agent/connection.js.map +1 -0
  230. package/dist/local-agent/discovery.js +259 -122
  231. package/dist/local-agent/discovery.js.map +1 -0
  232. package/dist/local-agent/executor.js +216 -193
  233. package/dist/local-agent/executor.js.map +1 -0
  234. package/dist/local-agent/index.d.ts +2 -2
  235. package/dist/local-agent/index.js +156 -156
  236. package/dist/local-agent/index.js.map +1 -0
  237. package/dist/node/adapters/base.js +18 -8
  238. package/dist/node/adapters/base.js.map +1 -0
  239. package/dist/node/adapters/discord.js +286 -275
  240. package/dist/node/adapters/discord.js.map +1 -0
  241. package/dist/node/adapters/email.js +189 -202
  242. package/dist/node/adapters/email.js.map +1 -0
  243. package/dist/node/adapters/imessage.js +145 -142
  244. package/dist/node/adapters/imessage.js.map +1 -0
  245. package/dist/node/adapters/slack.js +237 -236
  246. package/dist/node/adapters/slack.js.map +1 -0
  247. package/dist/node/adapters/sms.js +149 -151
  248. package/dist/node/adapters/sms.js.map +1 -0
  249. package/dist/node/adapters/telegram.js +88 -92
  250. package/dist/node/adapters/telegram.js.map +1 -0
  251. package/dist/node/adapters/webchat.js +160 -136
  252. package/dist/node/adapters/webchat.js.map +1 -0
  253. package/dist/node/adapters/whatsapp.js +212 -215
  254. package/dist/node/adapters/whatsapp.js.map +1 -0
  255. package/dist/node/cli.js +884 -653
  256. package/dist/node/cli.js.map +1 -0
  257. package/dist/node/config.js +20 -18
  258. package/dist/node/config.js.map +1 -0
  259. package/dist/node/gateway-client.js +191 -181
  260. package/dist/node/gateway-client.js.map +1 -0
  261. package/dist/node/portal/clipboard.js +161 -130
  262. package/dist/node/portal/clipboard.js.map +1 -0
  263. package/dist/node/portal/discovery.js +51 -45
  264. package/dist/node/portal/discovery.js.map +1 -0
  265. package/dist/node/portal/forward.js +64 -58
  266. package/dist/node/portal/forward.js.map +1 -0
  267. package/dist/node/portal/index.js +246 -221
  268. package/dist/node/portal/index.js.map +1 -0
  269. package/dist/node/portal/multiplexer.js +192 -182
  270. package/dist/node/portal/multiplexer.js.map +1 -0
  271. package/dist/node/portal/permissions.js +102 -70
  272. package/dist/node/portal/permissions.js.map +1 -0
  273. package/dist/node/portal/protocol.js +153 -116
  274. package/dist/node/portal/protocol.js.map +1 -0
  275. package/dist/node/portal/screen.js +80 -69
  276. package/dist/node/portal/screen.js.map +1 -0
  277. package/dist/node/portal/session.js +124 -117
  278. package/dist/node/portal/session.js.map +1 -0
  279. package/dist/node/portal/shell.js +140 -113
  280. package/dist/node/portal/shell.js.map +1 -0
  281. package/dist/node/portal/stream.js +77 -75
  282. package/dist/node/portal/stream.js.map +1 -0
  283. package/dist/node/portal/transfer.js +190 -167
  284. package/dist/node/portal/transfer.js.map +1 -0
  285. package/dist/node/portal/ui.js +124 -99
  286. package/dist/node/portal/ui.js.map +1 -0
  287. package/dist/node/remote-desktop/compile-helper.js +50 -45
  288. package/dist/node/remote-desktop/compile-helper.js.map +1 -0
  289. package/dist/node/remote-desktop/index.js +215 -187
  290. package/dist/node/remote-desktop/index.js.map +1 -0
  291. package/dist/node/remote-desktop/protocol.js +45 -29
  292. package/dist/node/remote-desktop/protocol.js.map +1 -0
  293. package/dist/node/runtime.js +493 -410
  294. package/dist/node/runtime.js.map +1 -0
  295. package/dist/server/handlers/__test-utils__/test-db.js +39 -89
  296. package/dist/server/handlers/__test-utils__/test-db.js.map +1 -0
  297. package/dist/server/handlers/analytics.js +467 -261
  298. package/dist/server/handlers/analytics.js.map +1 -0
  299. package/dist/server/handlers/api-docs.js +1030 -895
  300. package/dist/server/handlers/api-docs.js.map +1 -0
  301. package/dist/server/handlers/api-keys.js +291 -242
  302. package/dist/server/handlers/api-keys.js.map +1 -0
  303. package/dist/server/handlers/billing.js +330 -239
  304. package/dist/server/handlers/billing.js.map +1 -0
  305. package/dist/server/handlers/browser.js +468 -395
  306. package/dist/server/handlers/browser.js.map +1 -0
  307. package/dist/server/handlers/catalog.js +1377 -978
  308. package/dist/server/handlers/catalog.js.map +1 -0
  309. package/dist/server/handlers/clickhouse.js +157 -109
  310. package/dist/server/handlers/clickhouse.js.map +1 -0
  311. package/dist/server/handlers/comms.js +1439 -984
  312. package/dist/server/handlers/comms.js.map +1 -0
  313. package/dist/server/handlers/creations.js +461 -394
  314. package/dist/server/handlers/creations.js.map +1 -0
  315. package/dist/server/handlers/crm.js +1082 -791
  316. package/dist/server/handlers/crm.js.map +1 -0
  317. package/dist/server/handlers/discovery.js +251 -232
  318. package/dist/server/handlers/discovery.js.map +1 -0
  319. package/dist/server/handlers/embeddings.js +241 -164
  320. package/dist/server/handlers/embeddings.js.map +1 -0
  321. package/dist/server/handlers/enrichment.js +887 -718
  322. package/dist/server/handlers/enrichment.js.map +1 -0
  323. package/dist/server/handlers/image-gen.js +467 -376
  324. package/dist/server/handlers/image-gen.js.map +1 -0
  325. package/dist/server/handlers/inventory.js +797 -424
  326. package/dist/server/handlers/inventory.js.map +1 -0
  327. package/dist/server/handlers/kali.js +272 -230
  328. package/dist/server/handlers/kali.js.map +1 -0
  329. package/dist/server/handlers/llm-providers.js +803 -580
  330. package/dist/server/handlers/llm-providers.js.map +1 -0
  331. package/dist/server/handlers/local-agent.js +133 -105
  332. package/dist/server/handlers/local-agent.js.map +1 -0
  333. package/dist/server/handlers/media.js +1179 -857
  334. package/dist/server/handlers/media.js.map +1 -0
  335. package/dist/server/handlers/meta-ads.js +2669 -2093
  336. package/dist/server/handlers/meta-ads.js.map +1 -0
  337. package/dist/server/handlers/nodes.js +1321 -913
  338. package/dist/server/handlers/nodes.js.map +1 -0
  339. package/dist/server/handlers/operations.js +183 -157
  340. package/dist/server/handlers/operations.js.map +1 -0
  341. package/dist/server/handlers/platform.js +346 -210
  342. package/dist/server/handlers/platform.js.map +1 -0
  343. package/dist/server/handlers/remove-bg.js +118 -86
  344. package/dist/server/handlers/remove-bg.js.map +1 -0
  345. package/dist/server/handlers/storefront.js +586 -446
  346. package/dist/server/handlers/storefront.js.map +1 -0
  347. package/dist/server/handlers/supply-chain.js +546 -326
  348. package/dist/server/handlers/supply-chain.js.map +1 -0
  349. package/dist/server/handlers/transcription.js +106 -97
  350. package/dist/server/handlers/transcription.js.map +1 -0
  351. package/dist/server/handlers/video-gen.js +593 -424
  352. package/dist/server/handlers/video-gen.js.map +1 -0
  353. package/dist/server/handlers/voice.js +1458 -1039
  354. package/dist/server/handlers/voice.js.map +1 -0
  355. package/dist/server/handlers/workflow-steps.js +2837 -2116
  356. package/dist/server/handlers/workflow-steps.js.map +1 -0
  357. package/dist/server/handlers/workflows.js +1630 -933
  358. package/dist/server/handlers/workflows.js.map +1 -0
  359. package/dist/server/index.js +3167 -2422
  360. package/dist/server/index.js.map +1 -0
  361. package/dist/server/lib/batch-client.js +471 -409
  362. package/dist/server/lib/batch-client.js.map +1 -0
  363. package/dist/server/lib/clickhouse-buffer.js +118 -104
  364. package/dist/server/lib/clickhouse-buffer.js.map +1 -0
  365. package/dist/server/lib/clickhouse-client.js +107 -107
  366. package/dist/server/lib/clickhouse-client.js.map +1 -0
  367. package/dist/server/lib/coa-renderer.js +1786 -356
  368. package/dist/server/lib/coa-renderer.js.map +1 -0
  369. package/dist/server/lib/code-worker-pool.js +227 -177
  370. package/dist/server/lib/code-worker-pool.js.map +1 -0
  371. package/dist/server/lib/code-worker.js +174 -164
  372. package/dist/server/lib/code-worker.js.map +1 -0
  373. package/dist/server/lib/compaction-service.d.ts +2 -12
  374. package/dist/server/lib/compaction-service.js +74 -184
  375. package/dist/server/lib/compaction-service.js.map +1 -0
  376. package/dist/server/lib/logger.js +36 -24
  377. package/dist/server/lib/logger.js.map +1 -0
  378. package/dist/server/lib/otel.js +101 -80
  379. package/dist/server/lib/otel.js.map +1 -0
  380. package/dist/server/lib/pdf-renderer.js +952 -788
  381. package/dist/server/lib/pdf-renderer.js.map +1 -0
  382. package/dist/server/lib/prompt-sanitizer.js +188 -108
  383. package/dist/server/lib/prompt-sanitizer.js.map +1 -0
  384. package/dist/server/lib/provider-capabilities.js +136 -138
  385. package/dist/server/lib/provider-capabilities.js.map +1 -0
  386. package/dist/server/lib/provider-failover.js +190 -168
  387. package/dist/server/lib/provider-failover.js.map +1 -0
  388. package/dist/server/lib/rate-limiter.js +186 -117
  389. package/dist/server/lib/rate-limiter.js.map +1 -0
  390. package/dist/server/lib/react-pdf-layout.js +551 -382
  391. package/dist/server/lib/react-pdf-layout.js.map +1 -0
  392. package/dist/server/lib/server-agent-loop.d.ts +4 -1
  393. package/dist/server/lib/server-agent-loop.js +906 -634
  394. package/dist/server/lib/server-agent-loop.js.map +1 -0
  395. package/dist/server/lib/server-subagent.js +260 -164
  396. package/dist/server/lib/server-subagent.js.map +1 -0
  397. package/dist/server/lib/session-checkpoint.js +105 -96
  398. package/dist/server/lib/session-checkpoint.js.map +1 -0
  399. package/dist/server/lib/ssrf-guard.js +193 -184
  400. package/dist/server/lib/ssrf-guard.js.map +1 -0
  401. package/dist/server/lib/supabase-client.js +94 -82
  402. package/dist/server/lib/supabase-client.js.map +1 -0
  403. package/dist/server/lib/template-resolver.js +154 -176
  404. package/dist/server/lib/template-resolver.js.map +1 -0
  405. package/dist/server/lib/utils.js +242 -133
  406. package/dist/server/lib/utils.js.map +1 -0
  407. package/dist/server/local-agent-gateway.d.ts +2 -2
  408. package/dist/server/local-agent-gateway.js +785 -627
  409. package/dist/server/local-agent-gateway.js.map +1 -0
  410. package/dist/server/providers/anthropic.js +250 -172
  411. package/dist/server/providers/anthropic.js.map +1 -0
  412. package/dist/server/providers/bedrock.js +217 -158
  413. package/dist/server/providers/bedrock.js.map +1 -0
  414. package/dist/server/providers/gemini.js +548 -418
  415. package/dist/server/providers/gemini.js.map +1 -0
  416. package/dist/server/providers/openai.js +571 -437
  417. package/dist/server/providers/openai.js.map +1 -0
  418. package/dist/server/providers/registry.js +23 -18
  419. package/dist/server/providers/registry.js.map +1 -0
  420. package/dist/server/providers/shared.js +123 -95
  421. package/dist/server/providers/shared.js.map +1 -0
  422. package/dist/server/providers/types.js +1 -11
  423. package/dist/server/providers/types.js.map +1 -0
  424. package/dist/server/proxy-handlers.js +209 -165
  425. package/dist/server/proxy-handlers.js.map +1 -0
  426. package/dist/server/tool-router.js +959 -599
  427. package/dist/server/tool-router.js.map +1 -0
  428. package/dist/server/validation.js +248 -188
  429. package/dist/server/validation.js.map +1 -0
  430. package/dist/server/worker.js +202 -133
  431. package/dist/server/worker.js.map +1 -0
  432. package/dist/setup.d.ts +2 -2
  433. package/dist/setup.js +151 -147
  434. package/dist/setup.js.map +1 -0
  435. package/dist/shared/agent-core.d.ts +115 -26
  436. package/dist/shared/agent-core.js +956 -522
  437. package/dist/shared/agent-core.js.map +1 -0
  438. package/dist/shared/anthropic-types.js +1 -6
  439. package/dist/shared/anthropic-types.js.map +1 -0
  440. package/dist/shared/api-client.d.ts +16 -9
  441. package/dist/shared/api-client.js +419 -327
  442. package/dist/shared/api-client.js.map +1 -0
  443. package/dist/shared/compaction.d.ts +36 -0
  444. package/dist/shared/compaction.js +138 -0
  445. package/dist/shared/compaction.js.map +1 -0
  446. package/dist/shared/constants.js +67 -64
  447. package/dist/shared/constants.js.map +1 -0
  448. package/dist/shared/sse-parser.js +221 -219
  449. package/dist/shared/sse-parser.js.map +1 -0
  450. package/dist/shared/tool-dispatch.d.ts +4 -0
  451. package/dist/shared/tool-dispatch.js +226 -165
  452. package/dist/shared/tool-dispatch.js.map +1 -0
  453. package/dist/shared/types.js +1 -6
  454. package/dist/shared/types.js.map +1 -0
  455. package/dist/types/cli-highlight.d.js +2 -0
  456. package/dist/types/cli-highlight.d.js.map +1 -0
  457. package/dist/types/diff.d.js +2 -0
  458. package/dist/types/diff.d.js.map +1 -0
  459. package/dist/types/pdf-parse.d.js +2 -0
  460. package/dist/types/pdf-parse.d.js.map +1 -0
  461. package/dist/updater.d.ts +1 -1
  462. package/dist/updater.js +118 -92
  463. package/dist/updater.js.map +1 -0
  464. package/dist/webchat/widget.js +227 -380
  465. package/dist/webchat/widget.js.map +1 -0
  466. package/package.json +22 -10
  467. package/vendor/ink/build/ansi-tokenizer.d.ts +38 -0
  468. package/vendor/ink/build/ansi-tokenizer.js +316 -0
  469. package/vendor/ink/build/ansi-tokenizer.js.map +1 -0
  470. package/vendor/ink/build/apply-styles.js +175 -0
  471. package/vendor/ink/build/build-layout.js +77 -0
  472. package/vendor/ink/build/calculate-wrapped-text.js +53 -0
  473. package/vendor/ink/build/colorize.d.ts +3 -0
  474. package/vendor/ink/build/colorize.js +48 -0
  475. package/vendor/ink/build/colorize.js.map +1 -0
  476. package/vendor/ink/build/components/AccessibilityContext.d.ts +3 -0
  477. package/vendor/ink/build/components/AccessibilityContext.js +5 -0
  478. package/vendor/ink/build/components/AccessibilityContext.js.map +1 -0
  479. package/vendor/ink/build/components/App.d.ts +18 -0
  480. package/vendor/ink/build/components/App.js +351 -0
  481. package/vendor/ink/build/components/App.js.map +1 -0
  482. package/vendor/ink/build/components/AppContext.d.ts +15 -0
  483. package/vendor/ink/build/components/AppContext.js +11 -0
  484. package/vendor/ink/build/components/AppContext.js.map +1 -0
  485. package/vendor/ink/build/components/BackgroundContext.d.ts +4 -0
  486. package/vendor/ink/build/components/BackgroundContext.js +3 -0
  487. package/vendor/ink/build/components/BackgroundContext.js.map +1 -0
  488. package/vendor/ink/build/components/Box.d.ts +117 -0
  489. package/vendor/ink/build/components/Box.js +34 -0
  490. package/vendor/ink/build/components/Box.js.map +1 -0
  491. package/vendor/ink/build/components/Color.js +62 -0
  492. package/vendor/ink/build/components/Cursor.d.ts +83 -0
  493. package/vendor/ink/build/components/Cursor.js +53 -0
  494. package/vendor/ink/build/components/Cursor.js.map +1 -0
  495. package/vendor/ink/build/components/CursorContext.d.ts +11 -0
  496. package/vendor/ink/build/components/CursorContext.js +8 -0
  497. package/vendor/ink/build/components/CursorContext.js.map +1 -0
  498. package/vendor/ink/build/components/ErrorBoundary.d.ts +18 -0
  499. package/vendor/ink/build/components/ErrorBoundary.js +23 -0
  500. package/vendor/ink/build/components/ErrorBoundary.js.map +1 -0
  501. package/vendor/ink/build/components/ErrorOverview.d.ts +6 -0
  502. package/vendor/ink/build/components/ErrorOverview.js +84 -0
  503. package/vendor/ink/build/components/ErrorOverview.js.map +1 -0
  504. package/vendor/ink/build/components/FocusContext.d.ts +16 -0
  505. package/vendor/ink/build/components/FocusContext.js +17 -0
  506. package/vendor/ink/build/components/FocusContext.js.map +1 -0
  507. package/vendor/ink/build/components/Newline.d.ts +13 -0
  508. package/vendor/ink/build/components/Newline.js +8 -0
  509. package/vendor/ink/build/components/Newline.js.map +1 -0
  510. package/vendor/ink/build/components/Spacer.d.ts +7 -0
  511. package/vendor/ink/build/components/Spacer.js +11 -0
  512. package/vendor/ink/build/components/Spacer.js.map +1 -0
  513. package/vendor/ink/build/components/Static.d.ts +24 -0
  514. package/vendor/ink/build/components/Static.js +28 -0
  515. package/vendor/ink/build/components/Static.js.map +1 -0
  516. package/vendor/ink/build/components/StderrContext.d.ts +15 -0
  517. package/vendor/ink/build/components/StderrContext.js +13 -0
  518. package/vendor/ink/build/components/StderrContext.js.map +1 -0
  519. package/vendor/ink/build/components/StdinContext.d.ts +22 -0
  520. package/vendor/ink/build/components/StdinContext.js +19 -0
  521. package/vendor/ink/build/components/StdinContext.js.map +1 -0
  522. package/vendor/ink/build/components/StdoutContext.d.ts +15 -0
  523. package/vendor/ink/build/components/StdoutContext.js +13 -0
  524. package/vendor/ink/build/components/StdoutContext.js.map +1 -0
  525. package/vendor/ink/build/components/Text.d.ts +55 -0
  526. package/vendor/ink/build/components/Text.js +50 -0
  527. package/vendor/ink/build/components/Text.js.map +1 -0
  528. package/vendor/ink/build/components/Transform.d.ts +16 -0
  529. package/vendor/ink/build/components/Transform.js +15 -0
  530. package/vendor/ink/build/components/Transform.js.map +1 -0
  531. package/vendor/ink/build/cursor-helpers.d.ts +38 -0
  532. package/vendor/ink/build/cursor-helpers.js +56 -0
  533. package/vendor/ink/build/cursor-helpers.js.map +1 -0
  534. package/vendor/ink/build/devtools-window-polyfill.d.ts +1 -0
  535. package/vendor/ink/build/devtools-window-polyfill.js +65 -0
  536. package/vendor/ink/build/devtools-window-polyfill.js.map +1 -0
  537. package/vendor/ink/build/devtools.d.ts +1 -0
  538. package/vendor/ink/build/devtools.js +11 -0
  539. package/vendor/ink/build/devtools.js.map +1 -0
  540. package/vendor/ink/build/dom.d.ts +56 -0
  541. package/vendor/ink/build/dom.js +124 -0
  542. package/vendor/ink/build/dom.js.map +1 -0
  543. package/vendor/ink/build/experimental/apply-style.js +140 -0
  544. package/vendor/ink/build/experimental/dom.js +123 -0
  545. package/vendor/ink/build/experimental/output.js +91 -0
  546. package/vendor/ink/build/experimental/reconciler.js +141 -0
  547. package/vendor/ink/build/experimental/renderer.js +81 -0
  548. package/vendor/ink/build/get-max-width.d.ts +3 -0
  549. package/vendor/ink/build/get-max-width.js +10 -0
  550. package/vendor/ink/build/get-max-width.js.map +1 -0
  551. package/vendor/ink/build/hooks/use-app.d.ts +5 -0
  552. package/vendor/ink/build/hooks/use-app.js +8 -0
  553. package/vendor/ink/build/hooks/use-app.js.map +1 -0
  554. package/vendor/ink/build/hooks/use-cursor.d.ts +12 -0
  555. package/vendor/ink/build/hooks/use-cursor.js +29 -0
  556. package/vendor/ink/build/hooks/use-cursor.js.map +1 -0
  557. package/vendor/ink/build/hooks/use-focus-manager.d.ts +28 -0
  558. package/vendor/ink/build/hooks/use-focus-manager.js +17 -0
  559. package/vendor/ink/build/hooks/use-focus-manager.js.map +1 -0
  560. package/vendor/ink/build/hooks/use-focus.d.ts +29 -0
  561. package/vendor/ink/build/hooks/use-focus.js +42 -0
  562. package/vendor/ink/build/hooks/use-focus.js.map +1 -0
  563. package/vendor/ink/build/hooks/use-input.d.ts +131 -0
  564. package/vendor/ink/build/hooks/use-input.js +124 -0
  565. package/vendor/ink/build/hooks/use-input.js.map +1 -0
  566. package/vendor/ink/build/hooks/use-is-screen-reader-enabled.d.ts +5 -0
  567. package/vendor/ink/build/hooks/use-is-screen-reader-enabled.js +11 -0
  568. package/vendor/ink/build/hooks/use-is-screen-reader-enabled.js.map +1 -0
  569. package/vendor/ink/build/hooks/use-stderr.d.ts +5 -0
  570. package/vendor/ink/build/hooks/use-stderr.js +8 -0
  571. package/vendor/ink/build/hooks/use-stderr.js.map +1 -0
  572. package/vendor/ink/build/hooks/use-stdin.d.ts +5 -0
  573. package/vendor/ink/build/hooks/use-stdin.js +8 -0
  574. package/vendor/ink/build/hooks/use-stdin.js.map +1 -0
  575. package/vendor/ink/build/hooks/use-stdout.d.ts +5 -0
  576. package/vendor/ink/build/hooks/use-stdout.js +8 -0
  577. package/vendor/ink/build/hooks/use-stdout.js.map +1 -0
  578. package/vendor/ink/build/hooks/useInput.js +38 -0
  579. package/vendor/ink/build/index.d.ts +34 -0
  580. package/vendor/ink/build/index.js +20 -0
  581. package/vendor/ink/build/index.js.map +1 -0
  582. package/vendor/ink/build/ink.d.ts +90 -0
  583. package/vendor/ink/build/ink.js +654 -0
  584. package/vendor/ink/build/ink.js.map +1 -0
  585. package/vendor/ink/build/input-parser.d.ts +7 -0
  586. package/vendor/ink/build/input-parser.js +154 -0
  587. package/vendor/ink/build/input-parser.js.map +1 -0
  588. package/vendor/ink/build/instance.js +205 -0
  589. package/vendor/ink/build/instances.d.ts +3 -0
  590. package/vendor/ink/build/instances.js +8 -0
  591. package/vendor/ink/build/instances.js.map +1 -0
  592. package/vendor/ink/build/kitty-keyboard.d.ts +23 -0
  593. package/vendor/ink/build/kitty-keyboard.js +32 -0
  594. package/vendor/ink/build/kitty-keyboard.js.map +1 -0
  595. package/vendor/ink/build/layout.d.ts +7 -0
  596. package/vendor/ink/build/layout.js +33 -0
  597. package/vendor/ink/build/layout.js.map +1 -0
  598. package/vendor/ink/build/log-update.d.ts +19 -0
  599. package/vendor/ink/build/log-update.js +243 -0
  600. package/vendor/ink/build/log-update.js.map +1 -0
  601. package/vendor/ink/build/measure-element.d.ts +16 -0
  602. package/vendor/ink/build/measure-element.js +9 -0
  603. package/vendor/ink/build/measure-element.js.map +1 -0
  604. package/vendor/ink/build/measure-text.d.ts +6 -0
  605. package/vendor/ink/build/measure-text.js +21 -0
  606. package/vendor/ink/build/measure-text.js.map +1 -0
  607. package/vendor/ink/build/options.d.ts +52 -0
  608. package/vendor/ink/build/options.js +2 -0
  609. package/vendor/ink/build/options.js.map +1 -0
  610. package/vendor/ink/build/output.d.ts +35 -0
  611. package/vendor/ink/build/output.js +183 -0
  612. package/vendor/ink/build/output.js.map +1 -0
  613. package/vendor/ink/build/parse-keypress.d.ts +22 -0
  614. package/vendor/ink/build/parse-keypress.js +493 -0
  615. package/vendor/ink/build/parse-keypress.js.map +1 -0
  616. package/vendor/ink/build/reconciler.d.ts +4 -0
  617. package/vendor/ink/build/reconciler.js +274 -0
  618. package/vendor/ink/build/reconciler.js.map +1 -0
  619. package/vendor/ink/build/render-background.d.ts +4 -0
  620. package/vendor/ink/build/render-background.js +25 -0
  621. package/vendor/ink/build/render-background.js.map +1 -0
  622. package/vendor/ink/build/render-border.d.ts +4 -0
  623. package/vendor/ink/build/render-border.js +73 -0
  624. package/vendor/ink/build/render-border.js.map +1 -0
  625. package/vendor/ink/build/render-node-to-output.d.ts +14 -0
  626. package/vendor/ink/build/render-node-to-output.js +147 -0
  627. package/vendor/ink/build/render-node-to-output.js.map +1 -0
  628. package/vendor/ink/build/render-to-string.d.ts +38 -0
  629. package/vendor/ink/build/render-to-string.js +115 -0
  630. package/vendor/ink/build/render-to-string.js.map +1 -0
  631. package/vendor/ink/build/render.d.ts +121 -0
  632. package/vendor/ink/build/render.js +55 -0
  633. package/vendor/ink/build/render.js.map +1 -0
  634. package/vendor/ink/build/renderer.d.ts +8 -0
  635. package/vendor/ink/build/renderer.js +55 -0
  636. package/vendor/ink/build/renderer.js.map +1 -0
  637. package/vendor/ink/build/sanitize-ansi.d.ts +2 -0
  638. package/vendor/ink/build/sanitize-ansi.js +27 -0
  639. package/vendor/ink/build/sanitize-ansi.js.map +1 -0
  640. package/vendor/ink/build/screen-reader-update.d.ts +13 -0
  641. package/vendor/ink/build/screen-reader-update.js +38 -0
  642. package/vendor/ink/build/screen-reader-update.js.map +1 -0
  643. package/vendor/ink/build/squash-text-nodes.d.ts +3 -0
  644. package/vendor/ink/build/squash-text-nodes.js +36 -0
  645. package/vendor/ink/build/squash-text-nodes.js.map +1 -0
  646. package/vendor/ink/build/styles.d.ts +240 -0
  647. package/vendor/ink/build/styles.js +232 -0
  648. package/vendor/ink/build/styles.js.map +1 -0
  649. package/vendor/ink/build/utils.d.ts +2 -0
  650. package/vendor/ink/build/utils.js +4 -0
  651. package/vendor/ink/build/utils.js.map +1 -0
  652. package/vendor/ink/build/wrap-text.d.ts +3 -0
  653. package/vendor/ink/build/wrap-text.js +31 -0
  654. package/vendor/ink/build/wrap-text.js.map +1 -0
  655. package/vendor/ink/build/write-synchronized.d.ts +4 -0
  656. package/vendor/ink/build/write-synchronized.js +7 -0
  657. package/vendor/ink/build/write-synchronized.js.map +1 -0
  658. package/vendor/ink/license +10 -0
  659. package/vendor/ink/node_modules/@types/node/LICENSE +21 -0
  660. package/vendor/ink/node_modules/@types/node/README.md +15 -0
  661. package/vendor/ink/node_modules/@types/node/assert/strict.d.ts +105 -0
  662. package/vendor/ink/node_modules/@types/node/assert.d.ts +955 -0
  663. package/vendor/ink/node_modules/@types/node/async_hooks.d.ts +623 -0
  664. package/vendor/ink/node_modules/@types/node/buffer.buffer.d.ts +466 -0
  665. package/vendor/ink/node_modules/@types/node/buffer.d.ts +1810 -0
  666. package/vendor/ink/node_modules/@types/node/child_process.d.ts +1428 -0
  667. package/vendor/ink/node_modules/@types/node/cluster.d.ts +486 -0
  668. package/vendor/ink/node_modules/@types/node/compatibility/iterators.d.ts +21 -0
  669. package/vendor/ink/node_modules/@types/node/console.d.ts +151 -0
  670. package/vendor/ink/node_modules/@types/node/constants.d.ts +20 -0
  671. package/vendor/ink/node_modules/@types/node/crypto.d.ts +4065 -0
  672. package/vendor/ink/node_modules/@types/node/dgram.d.ts +564 -0
  673. package/vendor/ink/node_modules/@types/node/diagnostics_channel.d.ts +576 -0
  674. package/vendor/ink/node_modules/@types/node/dns/promises.d.ts +503 -0
  675. package/vendor/ink/node_modules/@types/node/dns.d.ts +922 -0
  676. package/vendor/ink/node_modules/@types/node/domain.d.ts +166 -0
  677. package/vendor/ink/node_modules/@types/node/events.d.ts +1054 -0
  678. package/vendor/ink/node_modules/@types/node/fs/promises.d.ts +1329 -0
  679. package/vendor/ink/node_modules/@types/node/fs.d.ts +4676 -0
  680. package/vendor/ink/node_modules/@types/node/globals.d.ts +150 -0
  681. package/vendor/ink/node_modules/@types/node/globals.typedarray.d.ts +101 -0
  682. package/vendor/ink/node_modules/@types/node/http.d.ts +2167 -0
  683. package/vendor/ink/node_modules/@types/node/http2.d.ts +2480 -0
  684. package/vendor/ink/node_modules/@types/node/https.d.ts +405 -0
  685. package/vendor/ink/node_modules/@types/node/index.d.ts +115 -0
  686. package/vendor/ink/node_modules/@types/node/inspector/promises.d.ts +41 -0
  687. package/vendor/ink/node_modules/@types/node/inspector.d.ts +224 -0
  688. package/vendor/ink/node_modules/@types/node/inspector.generated.d.ts +4226 -0
  689. package/vendor/ink/node_modules/@types/node/module.d.ts +819 -0
  690. package/vendor/ink/node_modules/@types/node/net.d.ts +933 -0
  691. package/vendor/ink/node_modules/@types/node/os.d.ts +507 -0
  692. package/vendor/ink/node_modules/@types/node/package.json +155 -0
  693. package/vendor/ink/node_modules/@types/node/path/posix.d.ts +8 -0
  694. package/vendor/ink/node_modules/@types/node/path/win32.d.ts +8 -0
  695. package/vendor/ink/node_modules/@types/node/path.d.ts +187 -0
  696. package/vendor/ink/node_modules/@types/node/perf_hooks.d.ts +643 -0
  697. package/vendor/ink/node_modules/@types/node/process.d.ts +2156 -0
  698. package/vendor/ink/node_modules/@types/node/punycode.d.ts +117 -0
  699. package/vendor/ink/node_modules/@types/node/querystring.d.ts +152 -0
  700. package/vendor/ink/node_modules/@types/node/quic.d.ts +910 -0
  701. package/vendor/ink/node_modules/@types/node/readline/promises.d.ts +161 -0
  702. package/vendor/ink/node_modules/@types/node/readline.d.ts +541 -0
  703. package/vendor/ink/node_modules/@types/node/repl.d.ts +415 -0
  704. package/vendor/ink/node_modules/@types/node/sea.d.ts +162 -0
  705. package/vendor/ink/node_modules/@types/node/sqlite.d.ts +955 -0
  706. package/vendor/ink/node_modules/@types/node/stream/consumers.d.ts +38 -0
  707. package/vendor/ink/node_modules/@types/node/stream/promises.d.ts +211 -0
  708. package/vendor/ink/node_modules/@types/node/stream/web.d.ts +296 -0
  709. package/vendor/ink/node_modules/@types/node/stream.d.ts +1760 -0
  710. package/vendor/ink/node_modules/@types/node/string_decoder.d.ts +67 -0
  711. package/vendor/ink/node_modules/@types/node/test/reporters.d.ts +96 -0
  712. package/vendor/ink/node_modules/@types/node/test.d.ts +2240 -0
  713. package/vendor/ink/node_modules/@types/node/timers/promises.d.ts +108 -0
  714. package/vendor/ink/node_modules/@types/node/timers.d.ts +159 -0
  715. package/vendor/ink/node_modules/@types/node/tls.d.ts +1198 -0
  716. package/vendor/ink/node_modules/@types/node/trace_events.d.ts +197 -0
  717. package/vendor/ink/node_modules/@types/node/ts5.6/buffer.buffer.d.ts +462 -0
  718. package/vendor/ink/node_modules/@types/node/ts5.6/compatibility/float16array.d.ts +71 -0
  719. package/vendor/ink/node_modules/@types/node/ts5.6/globals.typedarray.d.ts +36 -0
  720. package/vendor/ink/node_modules/@types/node/ts5.6/index.d.ts +117 -0
  721. package/vendor/ink/node_modules/@types/node/ts5.7/compatibility/float16array.d.ts +72 -0
  722. package/vendor/ink/node_modules/@types/node/ts5.7/index.d.ts +117 -0
  723. package/vendor/ink/node_modules/@types/node/tty.d.ts +250 -0
  724. package/vendor/ink/node_modules/@types/node/url.d.ts +519 -0
  725. package/vendor/ink/node_modules/@types/node/util/types.d.ts +558 -0
  726. package/vendor/ink/node_modules/@types/node/util.d.ts +1662 -0
  727. package/vendor/ink/node_modules/@types/node/v8.d.ts +983 -0
  728. package/vendor/ink/node_modules/@types/node/vm.d.ts +1208 -0
  729. package/vendor/ink/node_modules/@types/node/wasi.d.ts +202 -0
  730. package/vendor/ink/node_modules/@types/node/web-globals/abortcontroller.d.ts +59 -0
  731. package/vendor/ink/node_modules/@types/node/web-globals/blob.d.ts +23 -0
  732. package/vendor/ink/node_modules/@types/node/web-globals/console.d.ts +9 -0
  733. package/vendor/ink/node_modules/@types/node/web-globals/crypto.d.ts +39 -0
  734. package/vendor/ink/node_modules/@types/node/web-globals/domexception.d.ts +68 -0
  735. package/vendor/ink/node_modules/@types/node/web-globals/encoding.d.ts +11 -0
  736. package/vendor/ink/node_modules/@types/node/web-globals/events.d.ts +106 -0
  737. package/vendor/ink/node_modules/@types/node/web-globals/fetch.d.ts +69 -0
  738. package/vendor/ink/node_modules/@types/node/web-globals/importmeta.d.ts +13 -0
  739. package/vendor/ink/node_modules/@types/node/web-globals/messaging.d.ts +23 -0
  740. package/vendor/ink/node_modules/@types/node/web-globals/navigator.d.ts +25 -0
  741. package/vendor/ink/node_modules/@types/node/web-globals/performance.d.ts +45 -0
  742. package/vendor/ink/node_modules/@types/node/web-globals/storage.d.ts +24 -0
  743. package/vendor/ink/node_modules/@types/node/web-globals/streams.d.ts +115 -0
  744. package/vendor/ink/node_modules/@types/node/web-globals/timers.d.ts +44 -0
  745. package/vendor/ink/node_modules/@types/node/web-globals/url.d.ts +24 -0
  746. package/vendor/ink/node_modules/@types/node/worker_threads.d.ts +717 -0
  747. package/vendor/ink/node_modules/@types/node/zlib.d.ts +618 -0
  748. package/vendor/ink/node_modules/node-pty/LICENSE +69 -0
  749. package/vendor/ink/node_modules/node-pty/README.md +164 -0
  750. package/vendor/ink/node_modules/node-pty/binding.gyp +150 -0
  751. package/vendor/ink/node_modules/node-pty/lib/conpty_console_list_agent.js +25 -0
  752. package/vendor/ink/node_modules/node-pty/lib/eventEmitter2.js +47 -0
  753. package/vendor/ink/node_modules/node-pty/lib/index.js +52 -0
  754. package/vendor/ink/node_modules/node-pty/lib/interfaces.js +7 -0
  755. package/vendor/ink/node_modules/node-pty/lib/shared/conout.js +11 -0
  756. package/vendor/ink/node_modules/node-pty/lib/terminal.js +190 -0
  757. package/vendor/ink/node_modules/node-pty/lib/types.js +7 -0
  758. package/vendor/ink/node_modules/node-pty/lib/unixTerminal.js +349 -0
  759. package/vendor/ink/node_modules/node-pty/lib/utils.js +39 -0
  760. package/vendor/ink/node_modules/node-pty/lib/windowsConoutConnection.js +125 -0
  761. package/vendor/ink/node_modules/node-pty/lib/windowsPtyAgent.js +287 -0
  762. package/vendor/ink/node_modules/node-pty/lib/windowsTerminal.js +201 -0
  763. package/vendor/ink/node_modules/node-pty/lib/worker/conoutSocketWorker.js +22 -0
  764. package/vendor/ink/node_modules/node-pty/package.json +65 -0
  765. package/vendor/ink/node_modules/node-pty/prebuilds/darwin-arm64/pty.node +0 -0
  766. package/vendor/ink/node_modules/node-pty/prebuilds/darwin-arm64/spawn-helper +0 -0
  767. package/vendor/ink/node_modules/node-pty/prebuilds/darwin-x64/pty.node +0 -0
  768. package/vendor/ink/node_modules/node-pty/prebuilds/darwin-x64/spawn-helper +0 -0
  769. package/vendor/ink/node_modules/node-pty/prebuilds/linux-arm64/pty.node +0 -0
  770. package/vendor/ink/node_modules/node-pty/prebuilds/linux-x64/pty.node +0 -0
  771. package/vendor/ink/node_modules/node-pty/prebuilds/win32-arm64/conpty/OpenConsole.exe +0 -0
  772. package/vendor/ink/node_modules/node-pty/prebuilds/win32-arm64/conpty/conpty.dll +0 -0
  773. package/vendor/ink/node_modules/node-pty/prebuilds/win32-arm64/conpty.node +0 -0
  774. package/vendor/ink/node_modules/node-pty/prebuilds/win32-arm64/conpty.pdb +0 -0
  775. package/vendor/ink/node_modules/node-pty/prebuilds/win32-arm64/conpty_console_list.node +0 -0
  776. package/vendor/ink/node_modules/node-pty/prebuilds/win32-arm64/conpty_console_list.pdb +0 -0
  777. package/vendor/ink/node_modules/node-pty/prebuilds/win32-x64/conpty/OpenConsole.exe +0 -0
  778. package/vendor/ink/node_modules/node-pty/prebuilds/win32-x64/conpty/conpty.dll +0 -0
  779. package/vendor/ink/node_modules/node-pty/prebuilds/win32-x64/conpty.node +0 -0
  780. package/vendor/ink/node_modules/node-pty/prebuilds/win32-x64/conpty.pdb +0 -0
  781. package/vendor/ink/node_modules/node-pty/prebuilds/win32-x64/conpty_console_list.node +0 -0
  782. package/vendor/ink/node_modules/node-pty/prebuilds/win32-x64/conpty_console_list.pdb +0 -0
  783. package/vendor/ink/node_modules/node-pty/scripts/post-install.js +76 -0
  784. package/vendor/ink/node_modules/node-pty/scripts/prebuild.js +34 -0
  785. package/vendor/ink/node_modules/node-pty/src/unix/pty.cc +875 -0
  786. package/vendor/ink/node_modules/node-pty/src/unix/spawn-helper.cc +23 -0
  787. package/vendor/ink/node_modules/node-pty/src/win/conpty.cc +582 -0
  788. package/vendor/ink/node_modules/node-pty/src/win/conpty.h +41 -0
  789. package/vendor/ink/node_modules/node-pty/src/win/conpty_console_list.cc +44 -0
  790. package/vendor/ink/node_modules/node-pty/src/win/path_util.cc +95 -0
  791. package/vendor/ink/node_modules/node-pty/src/win/path_util.h +26 -0
  792. package/vendor/ink/node_modules/node-pty/third_party/conpty/1.23.251008001/win10-arm64/OpenConsole.exe +0 -0
  793. package/vendor/ink/node_modules/node-pty/third_party/conpty/1.23.251008001/win10-arm64/conpty.dll +0 -0
  794. package/vendor/ink/node_modules/node-pty/third_party/conpty/1.23.251008001/win10-x64/OpenConsole.exe +0 -0
  795. package/vendor/ink/node_modules/node-pty/third_party/conpty/1.23.251008001/win10-x64/conpty.dll +0 -0
  796. package/vendor/ink/node_modules/node-pty/typings/node-pty.d.ts +215 -0
  797. package/vendor/ink/node_modules/undici-types/LICENSE +21 -0
  798. package/vendor/ink/node_modules/undici-types/README.md +6 -0
  799. package/vendor/ink/node_modules/undici-types/agent.d.ts +32 -0
  800. package/vendor/ink/node_modules/undici-types/api.d.ts +43 -0
  801. package/vendor/ink/node_modules/undici-types/balanced-pool.d.ts +30 -0
  802. package/vendor/ink/node_modules/undici-types/cache-interceptor.d.ts +173 -0
  803. package/vendor/ink/node_modules/undici-types/cache.d.ts +36 -0
  804. package/vendor/ink/node_modules/undici-types/client-stats.d.ts +15 -0
  805. package/vendor/ink/node_modules/undici-types/client.d.ts +108 -0
  806. package/vendor/ink/node_modules/undici-types/connector.d.ts +34 -0
  807. package/vendor/ink/node_modules/undici-types/content-type.d.ts +21 -0
  808. package/vendor/ink/node_modules/undici-types/cookies.d.ts +30 -0
  809. package/vendor/ink/node_modules/undici-types/diagnostics-channel.d.ts +74 -0
  810. package/vendor/ink/node_modules/undici-types/dispatcher.d.ts +276 -0
  811. package/vendor/ink/node_modules/undici-types/env-http-proxy-agent.d.ts +22 -0
  812. package/vendor/ink/node_modules/undici-types/errors.d.ts +161 -0
  813. package/vendor/ink/node_modules/undici-types/eventsource.d.ts +66 -0
  814. package/vendor/ink/node_modules/undici-types/fetch.d.ts +211 -0
  815. package/vendor/ink/node_modules/undici-types/formdata.d.ts +108 -0
  816. package/vendor/ink/node_modules/undici-types/global-dispatcher.d.ts +9 -0
  817. package/vendor/ink/node_modules/undici-types/global-origin.d.ts +7 -0
  818. package/vendor/ink/node_modules/undici-types/h2c-client.d.ts +73 -0
  819. package/vendor/ink/node_modules/undici-types/handlers.d.ts +15 -0
  820. package/vendor/ink/node_modules/undici-types/header.d.ts +160 -0
  821. package/vendor/ink/node_modules/undici-types/index.d.ts +88 -0
  822. package/vendor/ink/node_modules/undici-types/interceptors.d.ts +73 -0
  823. package/vendor/ink/node_modules/undici-types/mock-agent.d.ts +68 -0
  824. package/vendor/ink/node_modules/undici-types/mock-call-history.d.ts +111 -0
  825. package/vendor/ink/node_modules/undici-types/mock-client.d.ts +27 -0
  826. package/vendor/ink/node_modules/undici-types/mock-errors.d.ts +12 -0
  827. package/vendor/ink/node_modules/undici-types/mock-interceptor.d.ts +94 -0
  828. package/vendor/ink/node_modules/undici-types/mock-pool.d.ts +27 -0
  829. package/vendor/ink/node_modules/undici-types/package.json +55 -0
  830. package/vendor/ink/node_modules/undici-types/patch.d.ts +29 -0
  831. package/vendor/ink/node_modules/undici-types/pool-stats.d.ts +19 -0
  832. package/vendor/ink/node_modules/undici-types/pool.d.ts +41 -0
  833. package/vendor/ink/node_modules/undici-types/proxy-agent.d.ts +29 -0
  834. package/vendor/ink/node_modules/undici-types/readable.d.ts +68 -0
  835. package/vendor/ink/node_modules/undici-types/retry-agent.d.ts +8 -0
  836. package/vendor/ink/node_modules/undici-types/retry-handler.d.ts +125 -0
  837. package/vendor/ink/node_modules/undici-types/round-robin-pool.d.ts +41 -0
  838. package/vendor/ink/node_modules/undici-types/snapshot-agent.d.ts +109 -0
  839. package/vendor/ink/node_modules/undici-types/util.d.ts +18 -0
  840. package/vendor/ink/node_modules/undici-types/utility.d.ts +7 -0
  841. package/vendor/ink/node_modules/undici-types/webidl.d.ts +341 -0
  842. package/vendor/ink/node_modules/undici-types/websocket.d.ts +186 -0
  843. package/vendor/ink/package.json +201 -0
  844. package/vendor/ink/readme.md +2636 -0
  845. package/bin/swag-agent.js +0 -9
  846. package/dist/server/lib/pg-rate-limiter.d.ts +0 -21
  847. package/dist/server/lib/pg-rate-limiter.js +0 -86
@@ -10,893 +10,1215 @@
10
10
  //
11
11
  // Note: file_path is handled client-side in server-tools.ts (converted to base64
12
12
  // before reaching this handler).
13
+
13
14
  import { sanitizeFilterValue } from "../lib/utils.js";
14
15
  const BUCKET = "product-images";
15
- const ALL_ACTIONS = [
16
- "upload", "bulk_upload", "list", "get", "update", "bulk_update",
17
- "delete", "bulk_delete", "search", "move", "archive", "unarchive",
18
- "tag", "folders_list", "folders_create", "folders_update", "folders_delete",
19
- "link_product", "unlink_product", "usage", "analytics", "duplicate", "replace",
20
- ];
16
+ const ALL_ACTIONS = ["upload", "bulk_upload", "list", "get", "update", "bulk_update", "delete", "bulk_delete", "search", "move", "archive", "unarchive", "tag", "folders_list", "folders_create", "folders_update", "folders_delete", "link_product", "unlink_product", "usage", "analytics", "duplicate", "replace"];
21
17
  const LIST_COLUMNS = "id, file_name, file_url, file_size, file_type, category, folder, status, title, alt_text, custom_tags, usage_count, linked_product_ids, created_at, updated_at";
22
18
  const VALID_CATEGORIES = ["product_photos", "marketing", "menus", "brand", "social_media", "print_marketing", "promotional", "brand_assets", "ai_generated"];
23
19
  const VALID_STATUSES = ["active", "archived", "processing"];
24
20
  const VALID_SORT_COLUMNS = ["created_at", "updated_at", "file_name", "file_size", "usage_count", "title"];
25
21
  const UPDATE_WHITELIST = ["title", "alt_text", "notes", "category", "status", "folder", "custom_tags"];
22
+
26
23
  // Detect mime type from base64 magic bytes
27
24
  function detectMime(base64) {
28
- if (base64.startsWith("iVBORw0KGgo"))
29
- return "image/png";
30
- if (base64.startsWith("/9j/"))
31
- return "image/jpeg";
32
- if (base64.startsWith("R0lGOD"))
33
- return "image/gif";
34
- if (base64.startsWith("UklGR"))
35
- return "image/webp";
36
- if (base64.startsWith("JVBER"))
37
- return "application/pdf";
38
- // Blender files start with "BLENDER" → base64 of "BLENDER" starts with "QkxFTkRFUg"
39
- if (base64.startsWith("QkxFTkRFUg"))
40
- return "application/x-blender";
41
- return "application/octet-stream";
25
+ if (base64.startsWith("iVBORw0KGgo")) return "image/png";
26
+ if (base64.startsWith("/9j/")) return "image/jpeg";
27
+ if (base64.startsWith("R0lGOD")) return "image/gif";
28
+ if (base64.startsWith("UklGR")) return "image/webp";
29
+ if (base64.startsWith("JVBER")) return "application/pdf";
30
+ // Blender files start with "BLENDER" → base64 of "BLENDER" starts with "QkxFTkRFUg"
31
+ if (base64.startsWith("QkxFTkRFUg")) return "application/x-blender";
32
+ return "application/octet-stream";
42
33
  }
43
34
  function mimeToExt(mime) {
44
- const map = {
45
- "image/png": "png", "image/jpeg": "jpg", "image/gif": "gif",
46
- "image/webp": "webp", "image/svg+xml": "svg", "application/pdf": "pdf",
47
- "application/x-blender": "blend",
48
- };
49
- return map[mime] || "bin";
35
+ const map = {
36
+ "image/png": "png",
37
+ "image/jpeg": "jpg",
38
+ "image/gif": "gif",
39
+ "image/webp": "webp",
40
+ "image/svg+xml": "svg",
41
+ "application/pdf": "pdf",
42
+ "application/x-blender": "blend"
43
+ };
44
+ return map[mime] || "bin";
50
45
  }
51
46
  export async function handleMedia(sb, args, storeId) {
52
- const action = args.action || "upload";
53
- if (!storeId)
54
- return { success: false, error: "store_id is required" };
55
- switch (action) {
56
- // ── UPLOAD ─────────────────────────────────────────────────────────────────
57
- case "upload": {
58
- let base64 = args.base64;
59
- const fileUrl = args.file_url;
60
- if (!base64 && !fileUrl) {
61
- return { success: false, error: "Provide base64 (file contents), file_url (public URL), or file_path (local — handled by CLI)" };
62
- }
63
- if (base64)
64
- base64 = base64.replace(/^data:[^;]+;base64,/, "").replace(/\s/g, "");
65
- if (!base64 && fileUrl) {
66
- if (fileUrl.startsWith("/") || /^[A-Z]:\\/i.test(fileUrl)) {
67
- return {
68
- success: false,
69
- error: `Local file paths cannot be used as file_url. Use file_path instead: media(action="upload", file_path="${fileUrl}"). The CLI reads the file locally and uploads it.`,
70
- };
71
- }
72
- let fetchUrl = fileUrl;
73
- if (!fetchUrl.startsWith("http://") && !fetchUrl.startsWith("https://")) {
74
- fetchUrl = `https://${fetchUrl}`;
75
- }
76
- const resp = await fetch(fetchUrl);
77
- if (!resp.ok)
78
- return { success: false, error: `Failed to fetch file_url: ${resp.status}` };
79
- const buf = await resp.arrayBuffer();
80
- base64 = Buffer.from(buf).toString("base64");
81
- }
82
- const mime = args.mime_type || detectMime(base64);
83
- const ext = mimeToExt(mime);
84
- const originalName = args.file_name || "";
85
- const folder = args.folder || "uploads";
86
- const id = crypto.randomUUID();
87
- const fileName = originalName
88
- ? `${id}-${originalName.replace(/[^a-zA-Z0-9._-]/g, "_")}`
89
- : `${id}.${ext}`;
90
- const storagePath = `${folder}/${storeId.toUpperCase()}/${fileName}`;
91
- const buffer = Buffer.from(base64, "base64");
92
- const { error: uploadErr } = await sb.storage
93
- .from(BUCKET)
94
- .upload(storagePath, buffer, { contentType: mime, upsert: true });
95
- if (uploadErr)
96
- return { success: false, error: `Upload failed: ${uploadErr.message}` };
97
- const { data: urlData } = sb.storage.from(BUCKET).getPublicUrl(storagePath);
98
- const cdnUrl = urlData.publicUrl;
99
- const insertData = {
100
- store_id: storeId,
101
- file_name: fileName,
102
- file_path: storagePath,
103
- file_url: cdnUrl,
104
- file_size: buffer.length,
105
- file_type: mime,
106
- category: args.category || "product_photos",
107
- source: "media-upload",
108
- folder,
109
- };
110
- if (args.title)
111
- insertData.title = args.title;
112
- if (args.alt_text)
113
- insertData.alt_text = args.alt_text;
114
- if (args.notes)
115
- insertData.notes = args.notes;
116
- if (args.tags)
117
- insertData.custom_tags = args.tags;
118
- const { data: mediaRow, error: mediaErr } = await sb
119
- .from("store_media")
120
- .insert(insertData)
121
- .select("id")
122
- .single();
123
- if (mediaErr)
124
- console.error("[media] store_media insert error:", mediaErr.message);
47
+ const action = args.action || "upload";
48
+ if (!storeId) return {
49
+ success: false,
50
+ error: "store_id is required"
51
+ };
52
+ switch (action) {
53
+ // ── UPLOAD ─────────────────────────────────────────────────────────────────
54
+ case "upload":
55
+ {
56
+ let base64 = args.base64;
57
+ const fileUrl = args.file_url;
58
+ if (!base64 && !fileUrl) {
59
+ return {
60
+ success: false,
61
+ error: "Provide base64 (file contents), file_url (public URL), or file_path (local — handled by CLI)"
62
+ };
63
+ }
64
+ if (base64) base64 = base64.replace(/^data:[^;]+;base64,/, "").replace(/\s/g, "");
65
+ if (!base64 && fileUrl) {
66
+ if (fileUrl.startsWith("/") || /^[A-Z]:\\/i.test(fileUrl)) {
125
67
  return {
126
- success: true,
127
- data: {
128
- media_id: mediaRow?.id || id,
129
- file_url: cdnUrl,
130
- file_name: fileName,
131
- file_size: buffer.length,
132
- mime_type: mime,
133
- storage_path: storagePath,
134
- },
68
+ success: false,
69
+ error: `Local file paths cannot be used as file_url. Use file_path instead: media(action="upload", file_path="${fileUrl}"). The CLI reads the file locally and uploads it.`
135
70
  };
71
+ }
72
+ let fetchUrl = fileUrl;
73
+ if (!fetchUrl.startsWith("http://") && !fetchUrl.startsWith("https://")) {
74
+ fetchUrl = `https://${fetchUrl}`;
75
+ }
76
+ const resp = await fetch(fetchUrl);
77
+ if (!resp.ok) return {
78
+ success: false,
79
+ error: `Failed to fetch file_url: ${resp.status}`
80
+ };
81
+ const buf = await resp.arrayBuffer();
82
+ base64 = Buffer.from(buf).toString("base64");
136
83
  }
137
- // ── BULK UPLOAD ────────────────────────────────────────────────────────────
138
- case "bulk_upload": {
139
- const files = args.files;
140
- if (!files || !Array.isArray(files) || files.length === 0) {
141
- return { success: false, error: "bulk_upload requires files: [{base64, file_name}]" };
142
- }
143
- const folder = args.folder || "uploads";
144
- const category = args.category || "product_photos";
145
- const readErrors = args._read_errors || [];
146
- const results = await Promise.allSettled(files.map(async (file) => {
147
- const b64 = file.base64.replace(/^data:[^;]+;base64,/, "").replace(/\s/g, "");
148
- const mime = args.mime_type || detectMime(b64);
149
- const ext = mimeToExt(mime);
150
- const fid = crypto.randomUUID();
151
- const originalName = file.file_name || "";
152
- const fName = originalName
153
- ? `${fid}-${originalName.replace(/[^a-zA-Z0-9._-]/g, "_")}`
154
- : `${fid}.${ext}`;
155
- const storagePath = `${folder}/${storeId.toUpperCase()}/${fName}`;
156
- const buffer = Buffer.from(b64, "base64");
157
- const { error: uploadErr } = await sb.storage
158
- .from(BUCKET)
159
- .upload(storagePath, buffer, { contentType: mime, upsert: true });
160
- if (uploadErr)
161
- throw new Error(`Storage: ${uploadErr.message}`);
162
- const { data: urlData } = sb.storage.from(BUCKET).getPublicUrl(storagePath);
163
- const cdnUrl = urlData.publicUrl;
164
- const insertData = {
165
- store_id: storeId, file_name: fName, file_path: storagePath,
166
- file_url: cdnUrl, file_size: buffer.length, file_type: mime,
167
- category, source: "media-upload", folder,
168
- };
169
- if (args.title)
170
- insertData.title = args.title;
171
- if (args.alt_text)
172
- insertData.alt_text = args.alt_text;
173
- if (args.notes)
174
- insertData.notes = args.notes;
175
- if (args.tags)
176
- insertData.custom_tags = args.tags;
177
- const { data: mediaRow, error: mediaErr } = await sb
178
- .from("store_media")
179
- .insert(insertData)
180
- .select("id")
181
- .single();
182
- if (mediaErr)
183
- console.error("[media] bulk insert error:", mediaErr.message);
184
- return {
185
- media_id: mediaRow?.id || fid, file_url: cdnUrl, file_name: fName,
186
- file_size: buffer.length, mime_type: mime, storage_path: storagePath,
187
- };
188
- }));
189
- const uploaded = [];
190
- const failed = [];
191
- results.forEach((r, i) => {
192
- if (r.status === "fulfilled") {
193
- uploaded.push(r.value);
194
- }
195
- else {
196
- failed.push({ file_name: files[i].file_name || `file_${i}`, error: r.reason?.message || String(r.reason) });
197
- }
84
+ const mime = args.mime_type || detectMime(base64);
85
+ const ext = mimeToExt(mime);
86
+ const originalName = args.file_name || "";
87
+ const folder = args.folder || "uploads";
88
+ const id = crypto.randomUUID();
89
+ const fileName = originalName ? `${id}-${originalName.replace(/[^a-zA-Z0-9._-]/g, "_")}` : `${id}.${ext}`;
90
+ const storagePath = `${folder}/${storeId.toUpperCase()}/${fileName}`;
91
+ const buffer = Buffer.from(base64, "base64");
92
+ const {
93
+ error: uploadErr
94
+ } = await sb.storage.from(BUCKET).upload(storagePath, buffer, {
95
+ contentType: mime,
96
+ upsert: true
97
+ });
98
+ if (uploadErr) return {
99
+ success: false,
100
+ error: `Upload failed: ${uploadErr.message}`
101
+ };
102
+ const {
103
+ data: urlData
104
+ } = sb.storage.from(BUCKET).getPublicUrl(storagePath);
105
+ const cdnUrl = urlData.publicUrl;
106
+ const insertData = {
107
+ store_id: storeId,
108
+ file_name: fileName,
109
+ file_path: storagePath,
110
+ file_url: cdnUrl,
111
+ file_size: buffer.length,
112
+ file_type: mime,
113
+ category: args.category || "product_photos",
114
+ source: "media-upload",
115
+ folder
116
+ };
117
+ if (args.title) insertData.title = args.title;
118
+ if (args.alt_text) insertData.alt_text = args.alt_text;
119
+ if (args.notes) insertData.notes = args.notes;
120
+ if (args.tags) insertData.custom_tags = args.tags;
121
+ const {
122
+ data: mediaRow,
123
+ error: mediaErr
124
+ } = await sb.from("store_media").insert(insertData).select("id").single();
125
+ if (mediaErr) console.error("[media] store_media insert error:", mediaErr.message);
126
+ return {
127
+ success: true,
128
+ data: {
129
+ media_id: mediaRow?.id || id,
130
+ file_url: cdnUrl,
131
+ file_name: fileName,
132
+ file_size: buffer.length,
133
+ mime_type: mime,
134
+ storage_path: storagePath
135
+ }
136
+ };
137
+ }
138
+
139
+ // ── BULK UPLOAD ────────────────────────────────────────────────────────────
140
+ case "bulk_upload":
141
+ {
142
+ const files = args.files;
143
+ if (!files || !Array.isArray(files) || files.length === 0) {
144
+ return {
145
+ success: false,
146
+ error: "bulk_upload requires files: [{base64, file_name}]"
147
+ };
148
+ }
149
+ const folder = args.folder || "uploads";
150
+ const category = args.category || "product_photos";
151
+ const readErrors = args._read_errors || [];
152
+ const results = await Promise.allSettled(files.map(async file => {
153
+ const b64 = file.base64.replace(/^data:[^;]+;base64,/, "").replace(/\s/g, "");
154
+ const mime = args.mime_type || detectMime(b64);
155
+ const ext = mimeToExt(mime);
156
+ const fid = crypto.randomUUID();
157
+ const originalName = file.file_name || "";
158
+ const fName = originalName ? `${fid}-${originalName.replace(/[^a-zA-Z0-9._-]/g, "_")}` : `${fid}.${ext}`;
159
+ const storagePath = `${folder}/${storeId.toUpperCase()}/${fName}`;
160
+ const buffer = Buffer.from(b64, "base64");
161
+ const {
162
+ error: uploadErr
163
+ } = await sb.storage.from(BUCKET).upload(storagePath, buffer, {
164
+ contentType: mime,
165
+ upsert: true
166
+ });
167
+ if (uploadErr) throw new Error(`Storage: ${uploadErr.message}`);
168
+ const {
169
+ data: urlData
170
+ } = sb.storage.from(BUCKET).getPublicUrl(storagePath);
171
+ const cdnUrl = urlData.publicUrl;
172
+ const insertData = {
173
+ store_id: storeId,
174
+ file_name: fName,
175
+ file_path: storagePath,
176
+ file_url: cdnUrl,
177
+ file_size: buffer.length,
178
+ file_type: mime,
179
+ category,
180
+ source: "media-upload",
181
+ folder
182
+ };
183
+ if (args.title) insertData.title = args.title;
184
+ if (args.alt_text) insertData.alt_text = args.alt_text;
185
+ if (args.notes) insertData.notes = args.notes;
186
+ if (args.tags) insertData.custom_tags = args.tags;
187
+ const {
188
+ data: mediaRow,
189
+ error: mediaErr
190
+ } = await sb.from("store_media").insert(insertData).select("id").single();
191
+ if (mediaErr) console.error("[media] bulk insert error:", mediaErr.message);
192
+ return {
193
+ media_id: mediaRow?.id || fid,
194
+ file_url: cdnUrl,
195
+ file_name: fName,
196
+ file_size: buffer.length,
197
+ mime_type: mime,
198
+ storage_path: storagePath
199
+ };
200
+ }));
201
+ const uploaded = [];
202
+ const failed = [];
203
+ results.forEach((r, i) => {
204
+ if (r.status === "fulfilled") {
205
+ uploaded.push(r.value);
206
+ } else {
207
+ failed.push({
208
+ file_name: files[i].file_name || `file_${i}`,
209
+ error: r.reason?.message || String(r.reason)
198
210
  });
199
- for (const e of readErrors)
200
- failed.push({ file_name: "(read error)", error: e });
201
- return {
202
- success: uploaded.length > 0,
203
- data: {
204
- uploaded_count: uploaded.length,
205
- failed_count: failed.length,
206
- total_size: uploaded.reduce((sum, u) => sum + u.file_size, 0),
207
- files: uploaded,
208
- ...(failed.length > 0 ? { errors: failed } : {}),
209
- },
210
- };
211
+ }
212
+ });
213
+ for (const e of readErrors) failed.push({
214
+ file_name: "(read error)",
215
+ error: e
216
+ });
217
+ return {
218
+ success: uploaded.length > 0,
219
+ data: {
220
+ uploaded_count: uploaded.length,
221
+ failed_count: failed.length,
222
+ total_size: uploaded.reduce((sum, u) => sum + u.file_size, 0),
223
+ files: uploaded,
224
+ ...(failed.length > 0 ? {
225
+ errors: failed
226
+ } : {})
227
+ }
228
+ };
229
+ }
230
+
231
+ // ── LIST ───────────────────────────────────────────────────────────────────
232
+ case "list":
233
+ {
234
+ const limit = Math.min(Number(args.limit) || 50, 200);
235
+ const offset = Number(args.offset) || 0;
236
+ const sortBy = VALID_SORT_COLUMNS.includes(args.sort_by) ? args.sort_by : "created_at";
237
+ const sortOrder = args.sort_order === "asc";
238
+ let query = sb.from("store_media").select(LIST_COLUMNS, {
239
+ count: "exact"
240
+ }).eq("store_id", storeId).order(sortBy, {
241
+ ascending: sortOrder
242
+ }).range(offset, offset + limit - 1);
243
+
244
+ // Default: exclude archived unless explicitly requested
245
+ if (args.status) {
246
+ query = query.eq("status", args.status);
247
+ } else if (!args.include_archived) {
248
+ query = query.neq("status", "archived");
211
249
  }
212
- // ── LIST ───────────────────────────────────────────────────────────────────
213
- case "list": {
214
- const limit = Math.min(Number(args.limit) || 50, 200);
215
- const offset = Number(args.offset) || 0;
216
- const sortBy = VALID_SORT_COLUMNS.includes(args.sort_by) ? args.sort_by : "created_at";
217
- const sortOrder = args.sort_order === "asc";
218
- let query = sb.from("store_media")
219
- .select(LIST_COLUMNS, { count: "exact" })
220
- .eq("store_id", storeId)
221
- .order(sortBy, { ascending: sortOrder })
222
- .range(offset, offset + limit - 1);
223
- // Default: exclude archived unless explicitly requested
224
- if (args.status) {
225
- query = query.eq("status", args.status);
226
- }
227
- else if (!args.include_archived) {
228
- query = query.neq("status", "archived");
229
- }
230
- if (args.folder)
231
- query = query.eq("folder", args.folder);
232
- if (args.category)
233
- query = query.eq("category", args.category);
234
- if (args.file_type) {
235
- const ft = sanitizeFilterValue(args.file_type);
236
- query = query.ilike("file_type", `${ft}%`);
237
- }
238
- if (args.tags && Array.isArray(args.tags)) {
239
- query = query.overlaps("custom_tags", args.tags);
240
- }
241
- const { data, error, count } = await query;
242
- if (error)
243
- return { success: false, error: error.message };
244
- return {
245
- success: true,
246
- data: { total: count || 0, count: data?.length || 0, offset, files: data },
247
- };
250
+ if (args.folder) query = query.eq("folder", args.folder);
251
+ if (args.category) query = query.eq("category", args.category);
252
+ if (args.file_type) {
253
+ const ft = sanitizeFilterValue(args.file_type);
254
+ query = query.ilike("file_type", `${ft}%`);
248
255
  }
249
- // ── GET ────────────────────────────────────────────────────────────────────
250
- case "get": {
251
- const mediaId = args.media_id;
252
- if (!mediaId)
253
- return { success: false, error: "media_id required" };
254
- const { data, error } = await sb.from("store_media").select("*").eq("id", mediaId).eq("store_id", storeId).single();
255
- if (error)
256
- return { success: false, error: error.message };
257
- return { success: true, data };
258
- }
259
- // ── UPDATE ─────────────────────────────────────────────────────────────────
260
- case "update": {
261
- const mediaId = args.media_id;
262
- if (!mediaId)
263
- return { success: false, error: "media_id required" };
264
- const updates = {};
265
- for (const key of UPDATE_WHITELIST) {
266
- if (args[key] !== undefined)
267
- updates[key] = args[key];
268
- }
269
- // Accept description as alias for notes
270
- if (args.description !== undefined && updates.notes === undefined) {
271
- updates.notes = args.description;
272
- }
273
- // Accept tags as alias for custom_tags
274
- if (args.tags !== undefined && updates.custom_tags === undefined) {
275
- updates.custom_tags = args.tags;
276
- }
277
- if (Object.keys(updates).length === 0) {
278
- return { success: false, error: `Nothing to update. Updatable fields: ${UPDATE_WHITELIST.join(", ")}` };
279
- }
280
- // Validate enums
281
- if (updates.category && !VALID_CATEGORIES.includes(updates.category)) {
282
- return { success: false, error: `Invalid category. Valid: ${VALID_CATEGORIES.join(", ")}` };
283
- }
284
- if (updates.status && !VALID_STATUSES.includes(updates.status)) {
285
- return { success: false, error: `Invalid status. Valid: ${VALID_STATUSES.join(", ")}` };
286
- }
287
- const { data, error } = await sb.from("store_media")
288
- .update(updates)
289
- .eq("id", mediaId)
290
- .eq("store_id", storeId)
291
- .select("*")
292
- .single();
293
- if (error)
294
- return { success: false, error: error.message };
295
- return { success: true, data };
296
- }
297
- // ── BULK UPDATE ────────────────────────────────────────────────────────────
298
- // Two modes:
299
- // 1. Uniform: media_ids[] + shared fields → applies same update to all
300
- // 2. Per-item: items[] → each item has {id, title?, alt_text?, tags?, ...}
301
- case "bulk_update": {
302
- const items = args.items;
303
- // ── Per-item mode ──
304
- if (items && Array.isArray(items) && items.length > 0) {
305
- if (items.length > 500) {
306
- return { success: false, error: `Too many items (${items.length}). Max 500 per call.` };
307
- }
308
- // Validate all items up-front
309
- for (let i = 0; i < items.length; i++) {
310
- const item = items[i];
311
- if (!item.id && !item.media_id) {
312
- return { success: false, error: `items[${i}] missing id` };
313
- }
314
- if (item.category && !VALID_CATEGORIES.includes(item.category)) {
315
- return { success: false, error: `items[${i}] invalid category: ${item.category}` };
316
- }
317
- if (item.status && !VALID_STATUSES.includes(item.status)) {
318
- return { success: false, error: `items[${i}] invalid status: ${item.status}` };
319
- }
320
- }
321
- // Process in parallel batches of 25
322
- const BATCH_SIZE = 25;
323
- let updatedCount = 0;
324
- let failedCount = 0;
325
- const errors = [];
326
- for (let i = 0; i < items.length; i += BATCH_SIZE) {
327
- const batch = items.slice(i, i + BATCH_SIZE);
328
- const results = await Promise.allSettled(batch.map(async (item) => {
329
- const itemId = (item.id || item.media_id);
330
- const updates = {};
331
- for (const key of UPDATE_WHITELIST) {
332
- if (item[key] !== undefined)
333
- updates[key] = item[key];
334
- }
335
- if (item.description !== undefined && updates.notes === undefined) {
336
- updates.notes = item.description;
337
- }
338
- if (item.tags !== undefined && updates.custom_tags === undefined) {
339
- updates.custom_tags = item.tags;
340
- }
341
- if (Object.keys(updates).length === 0)
342
- return null; // skip no-ops
343
- const { error } = await sb.from("store_media")
344
- .update(updates)
345
- .eq("id", itemId)
346
- .eq("store_id", storeId);
347
- if (error)
348
- throw new Error(`${itemId}: ${error.message}`);
349
- return itemId;
350
- }));
351
- for (const r of results) {
352
- if (r.status === "fulfilled" && r.value !== null)
353
- updatedCount++;
354
- else if (r.status === "rejected") {
355
- failedCount++;
356
- errors.push(r.reason?.message || String(r.reason));
357
- }
358
- }
359
- }
360
- return {
361
- success: updatedCount > 0,
362
- data: {
363
- updated_count: updatedCount,
364
- failed_count: failedCount,
365
- ...(errors.length > 0 ? { errors: errors.slice(0, 10) } : {}),
366
- },
367
- };
368
- }
369
- // ── Uniform mode ──
370
- const ids = args.media_ids;
371
- if (!ids || !Array.isArray(ids) || ids.length === 0) {
372
- return { success: false, error: "Provide items[] (per-item updates) or media_ids[] + shared fields (uniform update)" };
373
- }
374
- const updates = {};
375
- for (const key of UPDATE_WHITELIST) {
376
- if (args[key] !== undefined)
377
- updates[key] = args[key];
378
- }
379
- if (args.description !== undefined && updates.notes === undefined) {
380
- updates.notes = args.description;
381
- }
382
- if (args.tags !== undefined && updates.custom_tags === undefined) {
383
- updates.custom_tags = args.tags;
384
- }
385
- if (Object.keys(updates).length === 0) {
386
- return { success: false, error: `Nothing to update. Updatable fields: ${UPDATE_WHITELIST.join(", ")}` };
387
- }
388
- if (updates.category && !VALID_CATEGORIES.includes(updates.category)) {
389
- return { success: false, error: `Invalid category. Valid: ${VALID_CATEGORIES.join(", ")}` };
390
- }
391
- if (updates.status && !VALID_STATUSES.includes(updates.status)) {
392
- return { success: false, error: `Invalid status. Valid: ${VALID_STATUSES.join(", ")}` };
393
- }
394
- const { data, error } = await sb.from("store_media")
395
- .update(updates)
396
- .in("id", ids)
397
- .eq("store_id", storeId)
398
- .select("id");
399
- if (error)
400
- return { success: false, error: error.message };
401
- return { success: true, data: { updated_count: data?.length || 0 } };
402
- }
403
- // ── DELETE ─────────────────────────────────────────────────────────────────
404
- case "delete": {
405
- const mediaId = args.media_id;
406
- if (!mediaId)
407
- return { success: false, error: "media_id required" };
408
- const { data: row } = await sb.from("store_media").select("file_path").eq("id", mediaId).eq("store_id", storeId).single();
409
- if (row?.file_path) {
410
- await sb.storage.from(BUCKET).remove([row.file_path]);
411
- }
412
- const { error } = await sb.from("store_media").delete().eq("id", mediaId).eq("store_id", storeId);
413
- if (error)
414
- return { success: false, error: error.message };
415
- return { success: true, data: { deleted: mediaId } };
416
- }
417
- // ── BULK DELETE ────────────────────────────────────────────────────────────
418
- case "bulk_delete": {
419
- const ids = args.media_ids;
420
- if (!ids || !Array.isArray(ids) || ids.length === 0) {
421
- return { success: false, error: "media_ids[] required" };
422
- }
423
- // Fetch file paths for storage cleanup
424
- const { data: rows } = await sb.from("store_media")
425
- .select("id, file_path")
426
- .in("id", ids)
427
- .eq("store_id", storeId);
428
- const filePaths = (rows || []).map(r => r.file_path).filter(Boolean);
429
- if (filePaths.length > 0) {
430
- await sb.storage.from(BUCKET).remove(filePaths);
431
- }
432
- const { error } = await sb.from("store_media").delete().in("id", ids).eq("store_id", storeId);
433
- if (error)
434
- return { success: false, error: error.message };
435
- return { success: true, data: { deleted_count: rows?.length || 0 } };
436
- }
437
- // ── SEARCH ─────────────────────────────────────────────────────────────────
438
- case "search": {
439
- const query = args.query;
440
- if (!query)
441
- return { success: false, error: "query required for search" };
442
- const sq = sanitizeFilterValue(query);
443
- const limit = Math.min(Number(args.limit) || 50, 200);
444
- const offset = Number(args.offset) || 0;
445
- const sortBy = VALID_SORT_COLUMNS.includes(args.sort_by) ? args.sort_by : "created_at";
446
- const sortOrder = args.sort_order === "asc";
447
- let q = sb.from("store_media")
448
- .select(LIST_COLUMNS, { count: "exact" })
449
- .eq("store_id", storeId)
450
- .or(`file_name.ilike.%${sq}%,title.ilike.%${sq}%,alt_text.ilike.%${sq}%,notes.ilike.%${sq}%`)
451
- .order(sortBy, { ascending: sortOrder })
452
- .range(offset, offset + limit - 1);
453
- // Filters
454
- if (args.status) {
455
- q = q.eq("status", args.status);
456
- }
457
- else if (!args.include_archived) {
458
- q = q.neq("status", "archived");
459
- }
460
- if (args.folder)
461
- q = q.eq("folder", args.folder);
462
- if (args.category)
463
- q = q.eq("category", args.category);
464
- if (args.file_type) {
465
- const ft = sanitizeFilterValue(args.file_type);
466
- q = q.ilike("file_type", `${ft}%`);
467
- }
468
- if (args.tags && Array.isArray(args.tags)) {
469
- q = q.overlaps("custom_tags", args.tags);
470
- }
471
- if (args.date_from)
472
- q = q.gte("created_at", args.date_from);
473
- if (args.date_to)
474
- q = q.lte("created_at", args.date_to);
475
- if (args.min_usage !== undefined)
476
- q = q.gte("usage_count", Number(args.min_usage));
477
- if (args.max_usage !== undefined)
478
- q = q.lte("usage_count", Number(args.max_usage));
479
- const { data, error, count } = await q;
480
- if (error)
481
- return { success: false, error: error.message };
256
+ if (args.tags && Array.isArray(args.tags)) {
257
+ query = query.overlaps("custom_tags", args.tags);
258
+ }
259
+ const {
260
+ data,
261
+ error,
262
+ count
263
+ } = await query;
264
+ if (error) return {
265
+ success: false,
266
+ error: error.message
267
+ };
268
+ return {
269
+ success: true,
270
+ data: {
271
+ total: count || 0,
272
+ count: data?.length || 0,
273
+ offset,
274
+ files: data
275
+ }
276
+ };
277
+ }
278
+
279
+ // ── GET ────────────────────────────────────────────────────────────────────
280
+ case "get":
281
+ {
282
+ const mediaId = args.media_id;
283
+ if (!mediaId) return {
284
+ success: false,
285
+ error: "media_id required"
286
+ };
287
+ const {
288
+ data,
289
+ error
290
+ } = await sb.from("store_media").select("*").eq("id", mediaId).eq("store_id", storeId).single();
291
+ if (error) return {
292
+ success: false,
293
+ error: error.message
294
+ };
295
+ return {
296
+ success: true,
297
+ data
298
+ };
299
+ }
300
+
301
+ // ── UPDATE ─────────────────────────────────────────────────────────────────
302
+ case "update":
303
+ {
304
+ const mediaId = args.media_id;
305
+ if (!mediaId) return {
306
+ success: false,
307
+ error: "media_id required"
308
+ };
309
+ const updates = {};
310
+ for (const key of UPDATE_WHITELIST) {
311
+ if (args[key] !== undefined) updates[key] = args[key];
312
+ }
313
+ // Accept description as alias for notes
314
+ if (args.description !== undefined && updates.notes === undefined) {
315
+ updates.notes = args.description;
316
+ }
317
+ // Accept tags as alias for custom_tags
318
+ if (args.tags !== undefined && updates.custom_tags === undefined) {
319
+ updates.custom_tags = args.tags;
320
+ }
321
+ if (Object.keys(updates).length === 0) {
322
+ return {
323
+ success: false,
324
+ error: `Nothing to update. Updatable fields: ${UPDATE_WHITELIST.join(", ")}`
325
+ };
326
+ }
327
+
328
+ // Validate enums
329
+ if (updates.category && !VALID_CATEGORIES.includes(updates.category)) {
330
+ return {
331
+ success: false,
332
+ error: `Invalid category. Valid: ${VALID_CATEGORIES.join(", ")}`
333
+ };
334
+ }
335
+ if (updates.status && !VALID_STATUSES.includes(updates.status)) {
336
+ return {
337
+ success: false,
338
+ error: `Invalid status. Valid: ${VALID_STATUSES.join(", ")}`
339
+ };
340
+ }
341
+ const {
342
+ data,
343
+ error
344
+ } = await sb.from("store_media").update(updates).eq("id", mediaId).eq("store_id", storeId).select("*").single();
345
+ if (error) return {
346
+ success: false,
347
+ error: error.message
348
+ };
349
+ return {
350
+ success: true,
351
+ data
352
+ };
353
+ }
354
+
355
+ // ── BULK UPDATE ────────────────────────────────────────────────────────────
356
+ // Two modes:
357
+ // 1. Uniform: media_ids[] + shared fields → applies same update to all
358
+ // 2. Per-item: items[] each item has {id, title?, alt_text?, tags?, ...}
359
+ case "bulk_update":
360
+ {
361
+ const items = args.items;
362
+
363
+ // ── Per-item mode ──
364
+ if (items && Array.isArray(items) && items.length > 0) {
365
+ if (items.length > 500) {
482
366
  return {
483
- success: true,
484
- data: { total: count || 0, count: data?.length || 0, offset, query, files: data },
367
+ success: false,
368
+ error: `Too many items (${items.length}). Max 500 per call.`
485
369
  };
486
- }
487
- // ── MOVE ───────────────────────────────────────────────────────────────────
488
- case "move": {
489
- const folder = args.folder;
490
- if (!folder)
491
- return { success: false, error: "folder required" };
492
- const ids = args.media_ids || (args.media_id ? [args.media_id] : []);
493
- if (ids.length === 0)
494
- return { success: false, error: "media_id or media_ids[] required" };
495
- const { data, error } = await sb.from("store_media")
496
- .update({ folder })
497
- .in("id", ids)
498
- .eq("store_id", storeId)
499
- .select("id");
500
- if (error)
501
- return { success: false, error: error.message };
502
- return { success: true, data: { moved_count: data?.length || 0, folder } };
503
- }
504
- // ── ARCHIVE ────────────────────────────────────────────────────────────────
505
- case "archive": {
506
- const ids = args.media_ids || (args.media_id ? [args.media_id] : []);
507
- if (ids.length === 0)
508
- return { success: false, error: "media_id or media_ids[] required" };
509
- const { data, error } = await sb.from("store_media")
510
- .update({ status: "archived" })
511
- .in("id", ids)
512
- .eq("store_id", storeId)
513
- .select("id");
514
- if (error)
515
- return { success: false, error: error.message };
516
- return { success: true, data: { archived_count: data?.length || 0 } };
517
- }
518
- // ── UNARCHIVE ──────────────────────────────────────────────────────────────
519
- case "unarchive": {
520
- const ids = args.media_ids || (args.media_id ? [args.media_id] : []);
521
- if (ids.length === 0)
522
- return { success: false, error: "media_id or media_ids[] required" };
523
- const { data, error } = await sb.from("store_media")
524
- .update({ status: "active" })
525
- .in("id", ids)
526
- .eq("store_id", storeId)
527
- .select("id");
528
- if (error)
529
- return { success: false, error: error.message };
530
- return { success: true, data: { unarchived_count: data?.length || 0 } };
531
- }
532
- // ── TAG ────────────────────────────────────────────────────────────────────
533
- case "tag": {
534
- const ids = args.media_ids || (args.media_id ? [args.media_id] : []);
535
- if (ids.length === 0)
536
- return { success: false, error: "media_id or media_ids[] required" };
537
- const setTags = args.set;
538
- const addTags = args.add;
539
- const removeTags = args.remove;
540
- if (!setTags && !addTags && !removeTags) {
541
- return { success: false, error: "Provide set (replace all), add (append), or remove (delete specific tags)" };
370
+ }
371
+
372
+ // Validate all items up-front
373
+ for (let i = 0; i < items.length; i++) {
374
+ const item = items[i];
375
+ if (!item.id && !item.media_id) {
376
+ return {
377
+ success: false,
378
+ error: `items[${i}] missing id`
379
+ };
542
380
  }
543
- if (setTags) {
544
- // Direct replacement
545
- const { data, error } = await sb.from("store_media")
546
- .update({ custom_tags: setTags })
547
- .in("id", ids)
548
- .eq("store_id", storeId)
549
- .select("id, custom_tags");
550
- if (error)
551
- return { success: false, error: error.message };
552
- return { success: true, data: { updated_count: data?.length || 0, tags: setTags } };
381
+ if (item.category && !VALID_CATEGORIES.includes(item.category)) {
382
+ return {
383
+ success: false,
384
+ error: `items[${i}] invalid category: ${item.category}`
385
+ };
553
386
  }
554
- // Incremental add/remove read-modify-write per item
555
- const { data: rows, error: fetchErr } = await sb.from("store_media")
556
- .select("id, custom_tags")
557
- .in("id", ids)
558
- .eq("store_id", storeId);
559
- if (fetchErr)
560
- return { success: false, error: fetchErr.message };
561
- let updatedCount = 0;
562
- for (const row of rows || []) {
563
- let tags = row.custom_tags || [];
564
- if (addTags) {
565
- const newTags = addTags.filter(t => !tags.includes(t));
566
- tags = [...tags, ...newTags];
567
- }
568
- if (removeTags) {
569
- tags = tags.filter(t => !removeTags.includes(t));
570
- }
571
- const { error: upErr } = await sb.from("store_media")
572
- .update({ custom_tags: tags })
573
- .eq("id", row.id)
574
- .eq("store_id", storeId);
575
- if (!upErr)
576
- updatedCount++;
387
+ if (item.status && !VALID_STATUSES.includes(item.status)) {
388
+ return {
389
+ success: false,
390
+ error: `items[${i}] invalid status: ${item.status}`
391
+ };
577
392
  }
578
- return { success: true, data: { updated_count: updatedCount } };
579
- }
580
- // ── FOLDERS LIST ───────────────────────────────────────────────────────────
581
- case "folders_list": {
582
- const { data, error } = await sb.from("media_folders")
583
- .select("*")
584
- .eq("store_id", storeId)
585
- .order("name", { ascending: true });
586
- if (error)
587
- return { success: false, error: error.message };
588
- return { success: true, data: { count: data?.length || 0, folders: data } };
589
- }
590
- // ── FOLDERS CREATE ─────────────────────────────────────────────────────────
591
- case "folders_create": {
592
- const name = args.name;
593
- if (!name)
594
- return { success: false, error: "name required" };
595
- const insertData = { store_id: storeId, name };
596
- if (args.parent_folder_id)
597
- insertData.parent_folder_id = args.parent_folder_id;
598
- if (args.color)
599
- insertData.color = args.color;
600
- if (args.icon)
601
- insertData.icon = args.icon;
602
- const { data, error } = await sb.from("media_folders")
603
- .insert(insertData)
604
- .select("*")
605
- .single();
606
- if (error)
607
- return { success: false, error: error.message };
608
- return { success: true, data };
609
- }
610
- // ── FOLDERS UPDATE ─────────────────────────────────────────────────────────
611
- case "folders_update": {
612
- const folderId = args.folder_id;
613
- if (!folderId)
614
- return { success: false, error: "folder_id required" };
615
- const updates = {};
616
- if (args.name !== undefined)
617
- updates.name = args.name;
618
- if (args.parent_folder_id !== undefined)
619
- updates.parent_folder_id = args.parent_folder_id;
620
- if (args.color !== undefined)
621
- updates.color = args.color;
622
- if (args.icon !== undefined)
623
- updates.icon = args.icon;
624
- if (Object.keys(updates).length === 0) {
625
- return { success: false, error: "Nothing to update. Fields: name, parent_folder_id, color, icon" };
393
+ }
394
+
395
+ // Process in parallel batches of 25
396
+ const BATCH_SIZE = 25;
397
+ let updatedCount = 0;
398
+ let failedCount = 0;
399
+ const errors = [];
400
+ for (let i = 0; i < items.length; i += BATCH_SIZE) {
401
+ const batch = items.slice(i, i + BATCH_SIZE);
402
+ const results = await Promise.allSettled(batch.map(async item => {
403
+ const itemId = item.id || item.media_id;
404
+ const updates = {};
405
+ for (const key of UPDATE_WHITELIST) {
406
+ if (item[key] !== undefined) updates[key] = item[key];
407
+ }
408
+ if (item.description !== undefined && updates.notes === undefined) {
409
+ updates.notes = item.description;
410
+ }
411
+ if (item.tags !== undefined && updates.custom_tags === undefined) {
412
+ updates.custom_tags = item.tags;
413
+ }
414
+ if (Object.keys(updates).length === 0) return null; // skip no-ops
415
+
416
+ const {
417
+ error
418
+ } = await sb.from("store_media").update(updates).eq("id", itemId).eq("store_id", storeId);
419
+ if (error) throw new Error(`${itemId}: ${error.message}`);
420
+ return itemId;
421
+ }));
422
+ for (const r of results) {
423
+ if (r.status === "fulfilled" && r.value !== null) updatedCount++;else if (r.status === "rejected") {
424
+ failedCount++;
425
+ errors.push(r.reason?.message || String(r.reason));
426
+ }
626
427
  }
627
- const { data, error } = await sb.from("media_folders")
628
- .update(updates)
629
- .eq("id", folderId)
630
- .eq("store_id", storeId)
631
- .select("*")
632
- .single();
633
- if (error)
634
- return { success: false, error: error.message };
635
- return { success: true, data };
636
- }
637
- // ── FOLDERS DELETE ─────────────────────────────────────────────────────────
638
- case "folders_delete": {
639
- const folderId = args.folder_id;
640
- if (!folderId)
641
- return { success: false, error: "folder_id required" };
642
- // Optionally move orphaned media to another folder
643
- if (args.move_contents_to) {
644
- await sb.from("store_media")
645
- .update({ folder: args.move_contents_to })
646
- .eq("folder_id", folderId)
647
- .eq("store_id", storeId);
428
+ }
429
+ return {
430
+ success: updatedCount > 0,
431
+ data: {
432
+ updated_count: updatedCount,
433
+ failed_count: failedCount,
434
+ ...(errors.length > 0 ? {
435
+ errors: errors.slice(0, 10)
436
+ } : {})
648
437
  }
649
- // FK is ON DELETE SET NULL — safe to delete
650
- const { error } = await sb.from("media_folders")
651
- .delete()
652
- .eq("id", folderId)
653
- .eq("store_id", storeId);
654
- if (error)
655
- return { success: false, error: error.message };
656
- return { success: true, data: { deleted: folderId } };
657
- }
658
- // ── LINK PRODUCT ───────────────────────────────────────────────────────────
659
- case "link_product": {
660
- const mediaId = args.media_id;
661
- const productId = args.product_id;
662
- if (!mediaId || !productId)
663
- return { success: false, error: "media_id and product_id required" };
664
- const { error } = await sb.rpc("link_media_to_product", {
665
- p_media_id: mediaId,
666
- p_product_id: productId,
667
- });
668
- if (error)
669
- return { success: false, error: error.message };
670
- return { success: true, data: { linked: true, media_id: mediaId, product_id: productId } };
671
- }
672
- // ── UNLINK PRODUCT ─────────────────────────────────────────────────────────
673
- case "unlink_product": {
674
- const mediaId = args.media_id;
675
- const productId = args.product_id;
676
- if (!mediaId || !productId)
677
- return { success: false, error: "media_id and product_id required" };
678
- const { error } = await sb.rpc("unlink_media_from_product", {
679
- p_media_id: mediaId,
680
- p_product_id: productId,
681
- });
682
- if (error)
683
- return { success: false, error: error.message };
684
- return { success: true, data: { unlinked: true, media_id: mediaId, product_id: productId } };
685
- }
686
- // ── USAGE ──────────────────────────────────────────────────────────────────
687
- case "usage": {
688
- const mediaId = args.media_id;
689
- if (!mediaId)
690
- return { success: false, error: "media_id required" };
691
- // Get the media's file_url
692
- const { data: media, error: mediaErr } = await sb.from("store_media")
693
- .select("file_url")
694
- .eq("id", mediaId)
695
- .eq("store_id", storeId)
696
- .single();
697
- if (mediaErr)
698
- return { success: false, error: mediaErr.message };
699
- if (!media?.file_url)
700
- return { success: false, error: "Media not found or has no file_url" };
701
- // Query media_references for this URL
702
- const { data: refs, error: refErr } = await sb.from("media_references")
703
- .select("entity_type, entity_id, entity_name, field_name, created_at")
704
- .eq("media_url", media.file_url)
705
- .eq("store_id", storeId)
706
- .order("created_at", { ascending: false });
707
- if (refErr)
708
- return { success: false, error: refErr.message };
709
- return {
710
- success: true,
711
- data: {
712
- media_id: mediaId,
713
- reference_count: refs?.length || 0,
714
- references: refs,
715
- },
716
- };
438
+ };
717
439
  }
718
- // ── ANALYTICS ──────────────────────────────────────────────────────────────
719
- case "analytics": {
720
- // Run parallel aggregate queries
721
- const [activeRes, archivedRes, allRes, orphanRes, typeRes, catRes, topRes] = await Promise.all([
722
- sb.from("store_media").select("id", { count: "exact", head: true }).eq("store_id", storeId).eq("status", "active"),
723
- sb.from("store_media").select("id", { count: "exact", head: true }).eq("store_id", storeId).eq("status", "archived"),
724
- sb.from("store_media").select("file_size, file_type, category").eq("store_id", storeId),
725
- sb.from("store_media").select("id", { count: "exact", head: true }).eq("store_id", storeId).eq("usage_count", 0).neq("status", "archived"),
726
- sb.from("store_media").select("file_type").eq("store_id", storeId),
727
- sb.from("store_media").select("category").eq("store_id", storeId),
728
- sb.from("store_media").select("id, file_name, title, file_url, usage_count").eq("store_id", storeId).order("usage_count", { ascending: false }).limit(10),
729
- ]);
730
- // Compute total storage
731
- const allRows = allRes.data || [];
732
- const totalBytes = allRows.reduce((sum, r) => sum + (r.file_size || 0), 0);
733
- // Type breakdown (group by prefix before /)
734
- const typeBreakdown = {};
735
- for (const row of typeRes.data || []) {
736
- const prefix = (row.file_type || "unknown").split("/")[0];
737
- typeBreakdown[prefix] = (typeBreakdown[prefix] || 0) + 1;
738
- }
739
- // Category breakdown
740
- const catBreakdown = {};
741
- for (const row of catRes.data || []) {
742
- const cat = row.category || "unknown";
743
- catBreakdown[cat] = (catBreakdown[cat] || 0) + 1;
744
- }
745
- return {
746
- success: true,
747
- data: {
748
- total_items: (activeRes.count || 0) + (archivedRes.count || 0),
749
- active_count: activeRes.count || 0,
750
- archived_count: archivedRes.count || 0,
751
- orphan_count: orphanRes.count || 0,
752
- total_storage_bytes: totalBytes,
753
- total_storage_mb: Math.round(totalBytes / 1_048_576 * 100) / 100,
754
- type_breakdown: typeBreakdown,
755
- category_breakdown: catBreakdown,
756
- top_used: topRes.data || [],
757
- },
758
- };
440
+
441
+ // ── Uniform mode ──
442
+ const ids = args.media_ids;
443
+ if (!ids || !Array.isArray(ids) || ids.length === 0) {
444
+ return {
445
+ success: false,
446
+ error: "Provide items[] (per-item updates) or media_ids[] + shared fields (uniform update)"
447
+ };
759
448
  }
760
- // ── DUPLICATE ──────────────────────────────────────────────────────────────
761
- case "duplicate": {
762
- const mediaId = args.media_id;
763
- if (!mediaId)
764
- return { success: false, error: "media_id required" };
765
- // Fetch source row
766
- const { data: source, error: srcErr } = await sb.from("store_media")
767
- .select("*")
768
- .eq("id", mediaId)
769
- .eq("store_id", storeId)
770
- .single();
771
- if (srcErr)
772
- return { success: false, error: srcErr.message };
773
- // Download source file from storage
774
- const { data: fileData, error: dlErr } = await sb.storage
775
- .from(BUCKET)
776
- .download(source.file_path);
777
- if (dlErr)
778
- return { success: false, error: `Download source failed: ${dlErr.message}` };
779
- const buffer = Buffer.from(await fileData.arrayBuffer());
780
- // Upload with new UUID path
781
- const newId = crypto.randomUUID();
782
- const ext = source.file_name.split(".").pop() || "bin";
783
- const newFileName = `${newId}.${ext}`;
784
- const newPath = `${source.folder || "uploads"}/${storeId.toUpperCase()}/${newFileName}`;
785
- const { error: upErr } = await sb.storage
786
- .from(BUCKET)
787
- .upload(newPath, buffer, { contentType: source.file_type, upsert: true });
788
- if (upErr)
789
- return { success: false, error: `Upload duplicate failed: ${upErr.message}` };
790
- const { data: urlData } = sb.storage.from(BUCKET).getPublicUrl(newPath);
791
- const cdnUrl = urlData.publicUrl;
792
- const newTitle = args.new_title || (source.title ? `${source.title} (copy)` : null);
793
- const { data: newRow, error: insertErr } = await sb.from("store_media")
794
- .insert({
795
- store_id: storeId,
796
- file_name: newFileName,
797
- file_path: newPath,
798
- file_url: cdnUrl,
799
- file_size: buffer.length,
800
- file_type: source.file_type,
801
- category: source.category,
802
- source: "media-duplicate",
803
- folder: source.folder,
804
- title: newTitle,
805
- alt_text: source.alt_text,
806
- notes: source.notes,
807
- custom_tags: source.custom_tags,
808
- })
809
- .select("id, file_name, file_url, title")
810
- .single();
811
- if (insertErr)
812
- return { success: false, error: insertErr.message };
813
- return {
814
- success: true,
815
- data: {
816
- original_id: mediaId,
817
- id: newRow.id,
818
- file_name: newRow.file_name,
819
- file_url: newRow.file_url,
820
- title: newRow.title,
821
- },
822
- };
449
+ const updates = {};
450
+ for (const key of UPDATE_WHITELIST) {
451
+ if (args[key] !== undefined) updates[key] = args[key];
823
452
  }
824
- // ── REPLACE ────────────────────────────────────────────────────────────────
825
- case "replace": {
826
- const mediaId = args.media_id;
827
- if (!mediaId)
828
- return { success: false, error: "media_id required" };
829
- let base64 = args.base64;
830
- const fileUrl = args.file_url;
831
- if (!base64 && !fileUrl) {
832
- return { success: false, error: "Provide base64, file_url, or file_path (local — handled by CLI)" };
833
- }
834
- if (base64)
835
- base64 = base64.replace(/^data:[^;]+;base64,/, "").replace(/\s/g, "");
836
- if (!base64 && fileUrl) {
837
- if (fileUrl.startsWith("/") || /^[A-Z]:\\/i.test(fileUrl)) {
838
- return { success: false, error: `Use file_path for local files: media(action="replace", media_id="...", file_path="${fileUrl}")` };
839
- }
840
- let fetchUrl = fileUrl;
841
- if (!fetchUrl.startsWith("http://") && !fetchUrl.startsWith("https://")) {
842
- fetchUrl = `https://${fetchUrl}`;
843
- }
844
- const resp = await fetch(fetchUrl);
845
- if (!resp.ok)
846
- return { success: false, error: `Failed to fetch file_url: ${resp.status}` };
847
- const buf = await resp.arrayBuffer();
848
- base64 = Buffer.from(buf).toString("base64");
849
- }
850
- // Fetch existing row
851
- const { data: existing, error: getErr } = await sb.from("store_media")
852
- .select("file_path")
853
- .eq("id", mediaId)
854
- .eq("store_id", storeId)
855
- .single();
856
- if (getErr)
857
- return { success: false, error: getErr.message };
858
- // Delete old storage file
859
- if (existing.file_path) {
860
- await sb.storage.from(BUCKET).remove([existing.file_path]);
453
+ if (args.description !== undefined && updates.notes === undefined) {
454
+ updates.notes = args.description;
455
+ }
456
+ if (args.tags !== undefined && updates.custom_tags === undefined) {
457
+ updates.custom_tags = args.tags;
458
+ }
459
+ if (Object.keys(updates).length === 0) {
460
+ return {
461
+ success: false,
462
+ error: `Nothing to update. Updatable fields: ${UPDATE_WHITELIST.join(", ")}`
463
+ };
464
+ }
465
+ if (updates.category && !VALID_CATEGORIES.includes(updates.category)) {
466
+ return {
467
+ success: false,
468
+ error: `Invalid category. Valid: ${VALID_CATEGORIES.join(", ")}`
469
+ };
470
+ }
471
+ if (updates.status && !VALID_STATUSES.includes(updates.status)) {
472
+ return {
473
+ success: false,
474
+ error: `Invalid status. Valid: ${VALID_STATUSES.join(", ")}`
475
+ };
476
+ }
477
+ const {
478
+ data,
479
+ error
480
+ } = await sb.from("store_media").update(updates).in("id", ids).eq("store_id", storeId).select("id");
481
+ if (error) return {
482
+ success: false,
483
+ error: error.message
484
+ };
485
+ return {
486
+ success: true,
487
+ data: {
488
+ updated_count: data?.length || 0
489
+ }
490
+ };
491
+ }
492
+
493
+ // ── DELETE ─────────────────────────────────────────────────────────────────
494
+ case "delete":
495
+ {
496
+ const mediaId = args.media_id;
497
+ if (!mediaId) return {
498
+ success: false,
499
+ error: "media_id required"
500
+ };
501
+ const {
502
+ data: row
503
+ } = await sb.from("store_media").select("file_path").eq("id", mediaId).eq("store_id", storeId).single();
504
+ if (row?.file_path) {
505
+ await sb.storage.from(BUCKET).remove([row.file_path]);
506
+ }
507
+ const {
508
+ error
509
+ } = await sb.from("store_media").delete().eq("id", mediaId).eq("store_id", storeId);
510
+ if (error) return {
511
+ success: false,
512
+ error: error.message
513
+ };
514
+ return {
515
+ success: true,
516
+ data: {
517
+ deleted: mediaId
518
+ }
519
+ };
520
+ }
521
+
522
+ // ── BULK DELETE ────────────────────────────────────────────────────────────
523
+ case "bulk_delete":
524
+ {
525
+ const ids = args.media_ids;
526
+ if (!ids || !Array.isArray(ids) || ids.length === 0) {
527
+ return {
528
+ success: false,
529
+ error: "media_ids[] required"
530
+ };
531
+ }
532
+
533
+ // Fetch file paths for storage cleanup
534
+ const {
535
+ data: rows
536
+ } = await sb.from("store_media").select("id, file_path").in("id", ids).eq("store_id", storeId);
537
+ const filePaths = (rows || []).map(r => r.file_path).filter(Boolean);
538
+ if (filePaths.length > 0) {
539
+ await sb.storage.from(BUCKET).remove(filePaths);
540
+ }
541
+ const {
542
+ error
543
+ } = await sb.from("store_media").delete().in("id", ids).eq("store_id", storeId);
544
+ if (error) return {
545
+ success: false,
546
+ error: error.message
547
+ };
548
+ return {
549
+ success: true,
550
+ data: {
551
+ deleted_count: rows?.length || 0
552
+ }
553
+ };
554
+ }
555
+
556
+ // ── SEARCH ─────────────────────────────────────────────────────────────────
557
+ case "search":
558
+ {
559
+ const query = args.query;
560
+ if (!query) return {
561
+ success: false,
562
+ error: "query required for search"
563
+ };
564
+ const sq = sanitizeFilterValue(query);
565
+ const limit = Math.min(Number(args.limit) || 50, 200);
566
+ const offset = Number(args.offset) || 0;
567
+ const sortBy = VALID_SORT_COLUMNS.includes(args.sort_by) ? args.sort_by : "created_at";
568
+ const sortOrder = args.sort_order === "asc";
569
+ let q = sb.from("store_media").select(LIST_COLUMNS, {
570
+ count: "exact"
571
+ }).eq("store_id", storeId).or(`file_name.ilike.%${sq}%,title.ilike.%${sq}%,alt_text.ilike.%${sq}%,notes.ilike.%${sq}%`).order(sortBy, {
572
+ ascending: sortOrder
573
+ }).range(offset, offset + limit - 1);
574
+
575
+ // Filters
576
+ if (args.status) {
577
+ q = q.eq("status", args.status);
578
+ } else if (!args.include_archived) {
579
+ q = q.neq("status", "archived");
580
+ }
581
+ if (args.folder) q = q.eq("folder", args.folder);
582
+ if (args.category) q = q.eq("category", args.category);
583
+ if (args.file_type) {
584
+ const ft = sanitizeFilterValue(args.file_type);
585
+ q = q.ilike("file_type", `${ft}%`);
586
+ }
587
+ if (args.tags && Array.isArray(args.tags)) {
588
+ q = q.overlaps("custom_tags", args.tags);
589
+ }
590
+ if (args.date_from) q = q.gte("created_at", args.date_from);
591
+ if (args.date_to) q = q.lte("created_at", args.date_to);
592
+ if (args.min_usage !== undefined) q = q.gte("usage_count", Number(args.min_usage));
593
+ if (args.max_usage !== undefined) q = q.lte("usage_count", Number(args.max_usage));
594
+ const {
595
+ data,
596
+ error,
597
+ count
598
+ } = await q;
599
+ if (error) return {
600
+ success: false,
601
+ error: error.message
602
+ };
603
+ return {
604
+ success: true,
605
+ data: {
606
+ total: count || 0,
607
+ count: data?.length || 0,
608
+ offset,
609
+ query,
610
+ files: data
611
+ }
612
+ };
613
+ }
614
+
615
+ // ── MOVE ───────────────────────────────────────────────────────────────────
616
+ case "move":
617
+ {
618
+ const folder = args.folder;
619
+ if (!folder) return {
620
+ success: false,
621
+ error: "folder required"
622
+ };
623
+ const ids = args.media_ids || (args.media_id ? [args.media_id] : []);
624
+ if (ids.length === 0) return {
625
+ success: false,
626
+ error: "media_id or media_ids[] required"
627
+ };
628
+ const {
629
+ data,
630
+ error
631
+ } = await sb.from("store_media").update({
632
+ folder
633
+ }).in("id", ids).eq("store_id", storeId).select("id");
634
+ if (error) return {
635
+ success: false,
636
+ error: error.message
637
+ };
638
+ return {
639
+ success: true,
640
+ data: {
641
+ moved_count: data?.length || 0,
642
+ folder
643
+ }
644
+ };
645
+ }
646
+
647
+ // ── ARCHIVE ────────────────────────────────────────────────────────────────
648
+ case "archive":
649
+ {
650
+ const ids = args.media_ids || (args.media_id ? [args.media_id] : []);
651
+ if (ids.length === 0) return {
652
+ success: false,
653
+ error: "media_id or media_ids[] required"
654
+ };
655
+ const {
656
+ data,
657
+ error
658
+ } = await sb.from("store_media").update({
659
+ status: "archived"
660
+ }).in("id", ids).eq("store_id", storeId).select("id");
661
+ if (error) return {
662
+ success: false,
663
+ error: error.message
664
+ };
665
+ return {
666
+ success: true,
667
+ data: {
668
+ archived_count: data?.length || 0
669
+ }
670
+ };
671
+ }
672
+
673
+ // ── UNARCHIVE ──────────────────────────────────────────────────────────────
674
+ case "unarchive":
675
+ {
676
+ const ids = args.media_ids || (args.media_id ? [args.media_id] : []);
677
+ if (ids.length === 0) return {
678
+ success: false,
679
+ error: "media_id or media_ids[] required"
680
+ };
681
+ const {
682
+ data,
683
+ error
684
+ } = await sb.from("store_media").update({
685
+ status: "active"
686
+ }).in("id", ids).eq("store_id", storeId).select("id");
687
+ if (error) return {
688
+ success: false,
689
+ error: error.message
690
+ };
691
+ return {
692
+ success: true,
693
+ data: {
694
+ unarchived_count: data?.length || 0
695
+ }
696
+ };
697
+ }
698
+
699
+ // ── TAG ────────────────────────────────────────────────────────────────────
700
+ case "tag":
701
+ {
702
+ const ids = args.media_ids || (args.media_id ? [args.media_id] : []);
703
+ if (ids.length === 0) return {
704
+ success: false,
705
+ error: "media_id or media_ids[] required"
706
+ };
707
+ const setTags = args.set;
708
+ const addTags = args.add;
709
+ const removeTags = args.remove;
710
+ if (!setTags && !addTags && !removeTags) {
711
+ return {
712
+ success: false,
713
+ error: "Provide set (replace all), add (append), or remove (delete specific tags)"
714
+ };
715
+ }
716
+ if (setTags) {
717
+ // Direct replacement
718
+ const {
719
+ data,
720
+ error
721
+ } = await sb.from("store_media").update({
722
+ custom_tags: setTags
723
+ }).in("id", ids).eq("store_id", storeId).select("id, custom_tags");
724
+ if (error) return {
725
+ success: false,
726
+ error: error.message
727
+ };
728
+ return {
729
+ success: true,
730
+ data: {
731
+ updated_count: data?.length || 0,
732
+ tags: setTags
861
733
  }
862
- const mime = args.mime_type || detectMime(base64);
863
- const ext = mimeToExt(mime);
864
- const buffer = Buffer.from(base64, "base64");
865
- const newId = crypto.randomUUID();
866
- const originalName = args.file_name || "";
867
- const newFileName = originalName
868
- ? `${newId}-${originalName.replace(/[^a-zA-Z0-9._-]/g, "_")}`
869
- : `${newId}.${ext}`;
870
- const folder = args.folder || "uploads";
871
- const newPath = `${folder}/${storeId.toUpperCase()}/${newFileName}`;
872
- const { error: upErr } = await sb.storage
873
- .from(BUCKET)
874
- .upload(newPath, buffer, { contentType: mime, upsert: true });
875
- if (upErr)
876
- return { success: false, error: `Upload replacement failed: ${upErr.message}` };
877
- const { data: urlData } = sb.storage.from(BUCKET).getPublicUrl(newPath);
878
- const cdnUrl = urlData.publicUrl;
879
- const { data, error } = await sb.from("store_media")
880
- .update({
881
- file_name: newFileName,
882
- file_path: newPath,
883
- file_url: cdnUrl,
884
- file_size: buffer.length,
885
- file_type: mime,
886
- })
887
- .eq("id", mediaId)
888
- .eq("store_id", storeId)
889
- .select("*")
890
- .single();
891
- if (error)
892
- return { success: false, error: error.message };
893
- return { success: true, data };
894
- }
895
- // ── DEFAULT ────────────────────────────────────────────────────────────────
896
- default:
734
+ };
735
+ }
736
+
737
+ // Incremental add/remove — read-modify-write per item
738
+ const {
739
+ data: rows,
740
+ error: fetchErr
741
+ } = await sb.from("store_media").select("id, custom_tags").in("id", ids).eq("store_id", storeId);
742
+ if (fetchErr) return {
743
+ success: false,
744
+ error: fetchErr.message
745
+ };
746
+ let updatedCount = 0;
747
+ for (const row of rows || []) {
748
+ let tags = row.custom_tags || [];
749
+ if (addTags) {
750
+ const newTags = addTags.filter(t => !tags.includes(t));
751
+ tags = [...tags, ...newTags];
752
+ }
753
+ if (removeTags) {
754
+ tags = tags.filter(t => !removeTags.includes(t));
755
+ }
756
+ const {
757
+ error: upErr
758
+ } = await sb.from("store_media").update({
759
+ custom_tags: tags
760
+ }).eq("id", row.id).eq("store_id", storeId);
761
+ if (!upErr) updatedCount++;
762
+ }
763
+ return {
764
+ success: true,
765
+ data: {
766
+ updated_count: updatedCount
767
+ }
768
+ };
769
+ }
770
+
771
+ // ── FOLDERS LIST ───────────────────────────────────────────────────────────
772
+ case "folders_list":
773
+ {
774
+ const {
775
+ data,
776
+ error
777
+ } = await sb.from("media_folders").select("*").eq("store_id", storeId).order("name", {
778
+ ascending: true
779
+ });
780
+ if (error) return {
781
+ success: false,
782
+ error: error.message
783
+ };
784
+ return {
785
+ success: true,
786
+ data: {
787
+ count: data?.length || 0,
788
+ folders: data
789
+ }
790
+ };
791
+ }
792
+
793
+ // ── FOLDERS CREATE ─────────────────────────────────────────────────────────
794
+ case "folders_create":
795
+ {
796
+ const name = args.name;
797
+ if (!name) return {
798
+ success: false,
799
+ error: "name required"
800
+ };
801
+ const insertData = {
802
+ store_id: storeId,
803
+ name
804
+ };
805
+ if (args.parent_folder_id) insertData.parent_folder_id = args.parent_folder_id;
806
+ if (args.color) insertData.color = args.color;
807
+ if (args.icon) insertData.icon = args.icon;
808
+ const {
809
+ data,
810
+ error
811
+ } = await sb.from("media_folders").insert(insertData).select("*").single();
812
+ if (error) return {
813
+ success: false,
814
+ error: error.message
815
+ };
816
+ return {
817
+ success: true,
818
+ data
819
+ };
820
+ }
821
+
822
+ // ── FOLDERS UPDATE ─────────────────────────────────────────────────────────
823
+ case "folders_update":
824
+ {
825
+ const folderId = args.folder_id;
826
+ if (!folderId) return {
827
+ success: false,
828
+ error: "folder_id required"
829
+ };
830
+ const updates = {};
831
+ if (args.name !== undefined) updates.name = args.name;
832
+ if (args.parent_folder_id !== undefined) updates.parent_folder_id = args.parent_folder_id;
833
+ if (args.color !== undefined) updates.color = args.color;
834
+ if (args.icon !== undefined) updates.icon = args.icon;
835
+ if (Object.keys(updates).length === 0) {
836
+ return {
837
+ success: false,
838
+ error: "Nothing to update. Fields: name, parent_folder_id, color, icon"
839
+ };
840
+ }
841
+ const {
842
+ data,
843
+ error
844
+ } = await sb.from("media_folders").update(updates).eq("id", folderId).eq("store_id", storeId).select("*").single();
845
+ if (error) return {
846
+ success: false,
847
+ error: error.message
848
+ };
849
+ return {
850
+ success: true,
851
+ data
852
+ };
853
+ }
854
+
855
+ // ── FOLDERS DELETE ─────────────────────────────────────────────────────────
856
+ case "folders_delete":
857
+ {
858
+ const folderId = args.folder_id;
859
+ if (!folderId) return {
860
+ success: false,
861
+ error: "folder_id required"
862
+ };
863
+
864
+ // Optionally move orphaned media to another folder
865
+ if (args.move_contents_to) {
866
+ await sb.from("store_media").update({
867
+ folder: args.move_contents_to
868
+ }).eq("folder_id", folderId).eq("store_id", storeId);
869
+ }
870
+
871
+ // FK is ON DELETE SET NULL — safe to delete
872
+ const {
873
+ error
874
+ } = await sb.from("media_folders").delete().eq("id", folderId).eq("store_id", storeId);
875
+ if (error) return {
876
+ success: false,
877
+ error: error.message
878
+ };
879
+ return {
880
+ success: true,
881
+ data: {
882
+ deleted: folderId
883
+ }
884
+ };
885
+ }
886
+
887
+ // ── LINK PRODUCT ───────────────────────────────────────────────────────────
888
+ case "link_product":
889
+ {
890
+ const mediaId = args.media_id;
891
+ const productId = args.product_id;
892
+ if (!mediaId || !productId) return {
893
+ success: false,
894
+ error: "media_id and product_id required"
895
+ };
896
+ const {
897
+ error
898
+ } = await sb.rpc("link_media_to_product", {
899
+ p_media_id: mediaId,
900
+ p_product_id: productId
901
+ });
902
+ if (error) return {
903
+ success: false,
904
+ error: error.message
905
+ };
906
+ return {
907
+ success: true,
908
+ data: {
909
+ linked: true,
910
+ media_id: mediaId,
911
+ product_id: productId
912
+ }
913
+ };
914
+ }
915
+
916
+ // ── UNLINK PRODUCT ─────────────────────────────────────────────────────────
917
+ case "unlink_product":
918
+ {
919
+ const mediaId = args.media_id;
920
+ const productId = args.product_id;
921
+ if (!mediaId || !productId) return {
922
+ success: false,
923
+ error: "media_id and product_id required"
924
+ };
925
+ const {
926
+ error
927
+ } = await sb.rpc("unlink_media_from_product", {
928
+ p_media_id: mediaId,
929
+ p_product_id: productId
930
+ });
931
+ if (error) return {
932
+ success: false,
933
+ error: error.message
934
+ };
935
+ return {
936
+ success: true,
937
+ data: {
938
+ unlinked: true,
939
+ media_id: mediaId,
940
+ product_id: productId
941
+ }
942
+ };
943
+ }
944
+
945
+ // ── USAGE ──────────────────────────────────────────────────────────────────
946
+ case "usage":
947
+ {
948
+ const mediaId = args.media_id;
949
+ if (!mediaId) return {
950
+ success: false,
951
+ error: "media_id required"
952
+ };
953
+
954
+ // Get the media's file_url
955
+ const {
956
+ data: media,
957
+ error: mediaErr
958
+ } = await sb.from("store_media").select("file_url").eq("id", mediaId).eq("store_id", storeId).single();
959
+ if (mediaErr) return {
960
+ success: false,
961
+ error: mediaErr.message
962
+ };
963
+ if (!media?.file_url) return {
964
+ success: false,
965
+ error: "Media not found or has no file_url"
966
+ };
967
+
968
+ // Query media_references for this URL
969
+ const {
970
+ data: refs,
971
+ error: refErr
972
+ } = await sb.from("media_references").select("entity_type, entity_id, entity_name, field_name, created_at").eq("media_url", media.file_url).eq("store_id", storeId).order("created_at", {
973
+ ascending: false
974
+ });
975
+ if (refErr) return {
976
+ success: false,
977
+ error: refErr.message
978
+ };
979
+ return {
980
+ success: true,
981
+ data: {
982
+ media_id: mediaId,
983
+ reference_count: refs?.length || 0,
984
+ references: refs
985
+ }
986
+ };
987
+ }
988
+
989
+ // ── ANALYTICS ──────────────────────────────────────────────────────────────
990
+ case "analytics":
991
+ {
992
+ // Run parallel aggregate queries
993
+ const [activeRes, archivedRes, allRes, orphanRes, typeRes, catRes, topRes] = await Promise.all([sb.from("store_media").select("id", {
994
+ count: "exact",
995
+ head: true
996
+ }).eq("store_id", storeId).eq("status", "active"), sb.from("store_media").select("id", {
997
+ count: "exact",
998
+ head: true
999
+ }).eq("store_id", storeId).eq("status", "archived"), sb.from("store_media").select("file_size, file_type, category").eq("store_id", storeId), sb.from("store_media").select("id", {
1000
+ count: "exact",
1001
+ head: true
1002
+ }).eq("store_id", storeId).eq("usage_count", 0).neq("status", "archived"), sb.from("store_media").select("file_type").eq("store_id", storeId), sb.from("store_media").select("category").eq("store_id", storeId), sb.from("store_media").select("id, file_name, title, file_url, usage_count").eq("store_id", storeId).order("usage_count", {
1003
+ ascending: false
1004
+ }).limit(10)]);
1005
+
1006
+ // Compute total storage
1007
+ const allRows = allRes.data || [];
1008
+ const totalBytes = allRows.reduce((sum, r) => sum + (r.file_size || 0), 0);
1009
+
1010
+ // Type breakdown (group by prefix before /)
1011
+ const typeBreakdown = {};
1012
+ for (const row of typeRes.data || []) {
1013
+ const prefix = (row.file_type || "unknown").split("/")[0];
1014
+ typeBreakdown[prefix] = (typeBreakdown[prefix] || 0) + 1;
1015
+ }
1016
+
1017
+ // Category breakdown
1018
+ const catBreakdown = {};
1019
+ for (const row of catRes.data || []) {
1020
+ const cat = row.category || "unknown";
1021
+ catBreakdown[cat] = (catBreakdown[cat] || 0) + 1;
1022
+ }
1023
+ return {
1024
+ success: true,
1025
+ data: {
1026
+ total_items: (activeRes.count || 0) + (archivedRes.count || 0),
1027
+ active_count: activeRes.count || 0,
1028
+ archived_count: archivedRes.count || 0,
1029
+ orphan_count: orphanRes.count || 0,
1030
+ total_storage_bytes: totalBytes,
1031
+ total_storage_mb: Math.round(totalBytes / 1_048_576 * 100) / 100,
1032
+ type_breakdown: typeBreakdown,
1033
+ category_breakdown: catBreakdown,
1034
+ top_used: topRes.data || []
1035
+ }
1036
+ };
1037
+ }
1038
+
1039
+ // ── DUPLICATE ──────────────────────────────────────────────────────────────
1040
+ case "duplicate":
1041
+ {
1042
+ const mediaId = args.media_id;
1043
+ if (!mediaId) return {
1044
+ success: false,
1045
+ error: "media_id required"
1046
+ };
1047
+
1048
+ // Fetch source row
1049
+ const {
1050
+ data: source,
1051
+ error: srcErr
1052
+ } = await sb.from("store_media").select("*").eq("id", mediaId).eq("store_id", storeId).single();
1053
+ if (srcErr) return {
1054
+ success: false,
1055
+ error: srcErr.message
1056
+ };
1057
+
1058
+ // Download source file from storage
1059
+ const {
1060
+ data: fileData,
1061
+ error: dlErr
1062
+ } = await sb.storage.from(BUCKET).download(source.file_path);
1063
+ if (dlErr) return {
1064
+ success: false,
1065
+ error: `Download source failed: ${dlErr.message}`
1066
+ };
1067
+ const buffer = Buffer.from(await fileData.arrayBuffer());
1068
+
1069
+ // Upload with new UUID path
1070
+ const newId = crypto.randomUUID();
1071
+ const ext = source.file_name.split(".").pop() || "bin";
1072
+ const newFileName = `${newId}.${ext}`;
1073
+ const newPath = `${source.folder || "uploads"}/${storeId.toUpperCase()}/${newFileName}`;
1074
+ const {
1075
+ error: upErr
1076
+ } = await sb.storage.from(BUCKET).upload(newPath, buffer, {
1077
+ contentType: source.file_type,
1078
+ upsert: true
1079
+ });
1080
+ if (upErr) return {
1081
+ success: false,
1082
+ error: `Upload duplicate failed: ${upErr.message}`
1083
+ };
1084
+ const {
1085
+ data: urlData
1086
+ } = sb.storage.from(BUCKET).getPublicUrl(newPath);
1087
+ const cdnUrl = urlData.publicUrl;
1088
+ const newTitle = args.new_title || (source.title ? `${source.title} (copy)` : null);
1089
+ const {
1090
+ data: newRow,
1091
+ error: insertErr
1092
+ } = await sb.from("store_media").insert({
1093
+ store_id: storeId,
1094
+ file_name: newFileName,
1095
+ file_path: newPath,
1096
+ file_url: cdnUrl,
1097
+ file_size: buffer.length,
1098
+ file_type: source.file_type,
1099
+ category: source.category,
1100
+ source: "media-duplicate",
1101
+ folder: source.folder,
1102
+ title: newTitle,
1103
+ alt_text: source.alt_text,
1104
+ notes: source.notes,
1105
+ custom_tags: source.custom_tags
1106
+ }).select("id, file_name, file_url, title").single();
1107
+ if (insertErr) return {
1108
+ success: false,
1109
+ error: insertErr.message
1110
+ };
1111
+ return {
1112
+ success: true,
1113
+ data: {
1114
+ original_id: mediaId,
1115
+ id: newRow.id,
1116
+ file_name: newRow.file_name,
1117
+ file_url: newRow.file_url,
1118
+ title: newRow.title
1119
+ }
1120
+ };
1121
+ }
1122
+
1123
+ // ── REPLACE ────────────────────────────────────────────────────────────────
1124
+ case "replace":
1125
+ {
1126
+ const mediaId = args.media_id;
1127
+ if (!mediaId) return {
1128
+ success: false,
1129
+ error: "media_id required"
1130
+ };
1131
+ let base64 = args.base64;
1132
+ const fileUrl = args.file_url;
1133
+ if (!base64 && !fileUrl) {
1134
+ return {
1135
+ success: false,
1136
+ error: "Provide base64, file_url, or file_path (local — handled by CLI)"
1137
+ };
1138
+ }
1139
+ if (base64) base64 = base64.replace(/^data:[^;]+;base64,/, "").replace(/\s/g, "");
1140
+ if (!base64 && fileUrl) {
1141
+ if (fileUrl.startsWith("/") || /^[A-Z]:\\/i.test(fileUrl)) {
897
1142
  return {
898
- success: false,
899
- error: `Unknown action: ${action}. Available actions: ${ALL_ACTIONS.join(", ")}`,
1143
+ success: false,
1144
+ error: `Use file_path for local files: media(action="replace", media_id="...", file_path="${fileUrl}")`
900
1145
  };
901
- }
1146
+ }
1147
+ let fetchUrl = fileUrl;
1148
+ if (!fetchUrl.startsWith("http://") && !fetchUrl.startsWith("https://")) {
1149
+ fetchUrl = `https://${fetchUrl}`;
1150
+ }
1151
+ const resp = await fetch(fetchUrl);
1152
+ if (!resp.ok) return {
1153
+ success: false,
1154
+ error: `Failed to fetch file_url: ${resp.status}`
1155
+ };
1156
+ const buf = await resp.arrayBuffer();
1157
+ base64 = Buffer.from(buf).toString("base64");
1158
+ }
1159
+
1160
+ // Fetch existing row
1161
+ const {
1162
+ data: existing,
1163
+ error: getErr
1164
+ } = await sb.from("store_media").select("file_path").eq("id", mediaId).eq("store_id", storeId).single();
1165
+ if (getErr) return {
1166
+ success: false,
1167
+ error: getErr.message
1168
+ };
1169
+
1170
+ // Delete old storage file
1171
+ if (existing.file_path) {
1172
+ await sb.storage.from(BUCKET).remove([existing.file_path]);
1173
+ }
1174
+ const mime = args.mime_type || detectMime(base64);
1175
+ const ext = mimeToExt(mime);
1176
+ const buffer = Buffer.from(base64, "base64");
1177
+ const newId = crypto.randomUUID();
1178
+ const originalName = args.file_name || "";
1179
+ const newFileName = originalName ? `${newId}-${originalName.replace(/[^a-zA-Z0-9._-]/g, "_")}` : `${newId}.${ext}`;
1180
+ const folder = args.folder || "uploads";
1181
+ const newPath = `${folder}/${storeId.toUpperCase()}/${newFileName}`;
1182
+ const {
1183
+ error: upErr
1184
+ } = await sb.storage.from(BUCKET).upload(newPath, buffer, {
1185
+ contentType: mime,
1186
+ upsert: true
1187
+ });
1188
+ if (upErr) return {
1189
+ success: false,
1190
+ error: `Upload replacement failed: ${upErr.message}`
1191
+ };
1192
+ const {
1193
+ data: urlData
1194
+ } = sb.storage.from(BUCKET).getPublicUrl(newPath);
1195
+ const cdnUrl = urlData.publicUrl;
1196
+ const {
1197
+ data,
1198
+ error
1199
+ } = await sb.from("store_media").update({
1200
+ file_name: newFileName,
1201
+ file_path: newPath,
1202
+ file_url: cdnUrl,
1203
+ file_size: buffer.length,
1204
+ file_type: mime
1205
+ }).eq("id", mediaId).eq("store_id", storeId).select("*").single();
1206
+ if (error) return {
1207
+ success: false,
1208
+ error: error.message
1209
+ };
1210
+ return {
1211
+ success: true,
1212
+ data
1213
+ };
1214
+ }
1215
+
1216
+ // ── DEFAULT ────────────────────────────────────────────────────────────────
1217
+ default:
1218
+ return {
1219
+ success: false,
1220
+ error: `Unknown action: ${action}. Available actions: ${ALL_ACTIONS.join(", ")}`
1221
+ };
1222
+ }
902
1223
  }
1224
+ //# sourceMappingURL=media.js.map