@roj-ai/sdk 0.1.3 → 0.1.5

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 (493) hide show
  1. package/dist/bootstrap.js +191 -0
  2. package/dist/bootstrap.js.map +1 -0
  3. package/dist/builtin-events.js +8 -0
  4. package/dist/builtin-events.js.map +1 -0
  5. package/dist/bun-platform/fs.js +39 -0
  6. package/dist/bun-platform/fs.js.map +1 -0
  7. package/dist/bun-platform/index.js +18 -0
  8. package/dist/bun-platform/index.js.map +1 -0
  9. package/dist/bun-platform/process.js +21 -0
  10. package/dist/bun-platform/process.js.map +1 -0
  11. package/dist/config.js +54 -0
  12. package/dist/config.js.map +1 -0
  13. package/dist/config.test.js +155 -0
  14. package/dist/config.test.js.map +1 -0
  15. package/dist/core/agent-loop.integration.test.js +414 -0
  16. package/dist/core/agent-loop.integration.test.js.map +1 -0
  17. package/dist/core/agents/agent-config.test.js +194 -0
  18. package/dist/core/agents/agent-config.test.js.map +1 -0
  19. package/dist/core/agents/agent-roles.js +25 -0
  20. package/dist/core/agents/agent-roles.js.map +1 -0
  21. package/dist/core/agents/agent-shutdown.test.js +180 -0
  22. package/dist/core/agents/agent-shutdown.test.js.map +1 -0
  23. package/dist/core/agents/agent.js +1205 -0
  24. package/dist/core/agents/agent.js.map +1 -0
  25. package/dist/core/agents/agent.test.js +313 -0
  26. package/dist/core/agents/agent.test.js.map +1 -0
  27. package/dist/core/agents/communicator.js +13 -0
  28. package/dist/core/agents/communicator.js.map +1 -0
  29. package/dist/core/agents/config.js +5 -0
  30. package/dist/core/agents/config.js.map +1 -0
  31. package/dist/core/agents/context.js +2 -0
  32. package/dist/core/agents/context.js.map +1 -0
  33. package/dist/core/agents/debounce.js +74 -0
  34. package/dist/core/agents/debounce.js.map +1 -0
  35. package/dist/core/agents/handler-events.test.js +115 -0
  36. package/dist/core/agents/handler-events.test.js.map +1 -0
  37. package/dist/core/agents/index.js +2 -0
  38. package/dist/core/agents/index.js.map +1 -0
  39. package/dist/core/agents/response-sanitizer.js +46 -0
  40. package/dist/core/agents/response-sanitizer.js.map +1 -0
  41. package/dist/core/agents/response-sanitizer.test.js +101 -0
  42. package/dist/core/agents/response-sanitizer.test.js.map +1 -0
  43. package/dist/core/agents/retry.js +105 -0
  44. package/dist/core/agents/retry.js.map +1 -0
  45. package/dist/core/agents/schema.js +39 -0
  46. package/dist/core/agents/schema.js.map +1 -0
  47. package/dist/core/agents/state.js +90 -0
  48. package/dist/core/agents/state.js.map +1 -0
  49. package/dist/core/context/state.js +23 -0
  50. package/dist/core/context/state.js.map +1 -0
  51. package/dist/core/errors.js +38 -0
  52. package/dist/core/errors.js.map +1 -0
  53. package/dist/core/event-sourcing.integration.test.js +154 -0
  54. package/dist/core/event-sourcing.integration.test.js.map +1 -0
  55. package/dist/core/events/base-event-store.js +201 -0
  56. package/dist/core/events/base-event-store.js.map +1 -0
  57. package/dist/core/events/event-store.js +26 -0
  58. package/dist/core/events/event-store.js.map +1 -0
  59. package/dist/core/events/file.js +320 -0
  60. package/dist/core/events/file.js.map +1 -0
  61. package/dist/core/events/file.test.js +284 -0
  62. package/dist/core/events/file.test.js.map +1 -0
  63. package/dist/core/events/index.js +3 -0
  64. package/dist/core/events/index.js.map +1 -0
  65. package/dist/core/events/memory.js +101 -0
  66. package/dist/core/events/memory.js.map +1 -0
  67. package/dist/core/events/memory.test.js +502 -0
  68. package/dist/core/events/memory.test.js.map +1 -0
  69. package/dist/core/events/metadata-utils.js +107 -0
  70. package/dist/core/events/metadata-utils.js.map +1 -0
  71. package/dist/core/events/test-helpers.js +15 -0
  72. package/dist/core/events/test-helpers.js.map +1 -0
  73. package/dist/core/events/types.js +21 -0
  74. package/dist/core/events/types.js.map +1 -0
  75. package/dist/core/file-store/file-store.js +250 -0
  76. package/dist/core/file-store/file-store.js.map +1 -0
  77. package/dist/core/file-store/types.js +7 -0
  78. package/dist/core/file-store/types.js.map +1 -0
  79. package/dist/core/image/image-processor.js +106 -0
  80. package/dist/core/image/image-processor.js.map +1 -0
  81. package/dist/core/image/image-processor.test.js +171 -0
  82. package/dist/core/image/image-processor.test.js.map +1 -0
  83. package/dist/core/image/index.js +4 -0
  84. package/dist/core/image/index.js.map +1 -0
  85. package/dist/core/image/noop-resizer.js +6 -0
  86. package/dist/core/image/noop-resizer.js.map +1 -0
  87. package/dist/core/image/types.js +2 -0
  88. package/dist/core/image/types.js.map +1 -0
  89. package/dist/core/image/vips-resizer.js +100 -0
  90. package/dist/core/image/vips-resizer.js.map +1 -0
  91. package/dist/core/image/vips-resizer.test.js +324 -0
  92. package/dist/core/image/vips-resizer.test.js.map +1 -0
  93. package/dist/core/llm/anthropic.js +396 -0
  94. package/dist/core/llm/anthropic.js.map +1 -0
  95. package/dist/core/llm/anthropic.test.js +434 -0
  96. package/dist/core/llm/anthropic.test.js.map +1 -0
  97. package/dist/core/llm/cache-breakpoints.js +37 -0
  98. package/dist/core/llm/cache-breakpoints.js.map +1 -0
  99. package/dist/core/llm/cache-live.test.js +137 -0
  100. package/dist/core/llm/cache-live.test.js.map +1 -0
  101. package/dist/core/llm/index.js +9 -0
  102. package/dist/core/llm/index.js.map +1 -0
  103. package/dist/core/llm/llm-log-types.js +12 -0
  104. package/dist/core/llm/llm-log-types.js.map +1 -0
  105. package/dist/core/llm/logger.js +241 -0
  106. package/dist/core/llm/logger.js.map +1 -0
  107. package/dist/core/llm/logger.test.js +228 -0
  108. package/dist/core/llm/logger.test.js.map +1 -0
  109. package/dist/core/llm/logging-provider.js +49 -0
  110. package/dist/core/llm/logging-provider.js.map +1 -0
  111. package/dist/core/llm/middleware.js +114 -0
  112. package/dist/core/llm/middleware.js.map +1 -0
  113. package/dist/core/llm/mock.js +186 -0
  114. package/dist/core/llm/mock.js.map +1 -0
  115. package/dist/core/llm/mock.test.js +318 -0
  116. package/dist/core/llm/mock.test.js.map +1 -0
  117. package/dist/core/llm/openrouter-mapping.test.js +125 -0
  118. package/dist/core/llm/openrouter-mapping.test.js.map +1 -0
  119. package/dist/core/llm/openrouter.js +298 -0
  120. package/dist/core/llm/openrouter.js.map +1 -0
  121. package/dist/core/llm/openrouter.test.js +377 -0
  122. package/dist/core/llm/openrouter.test.js.map +1 -0
  123. package/dist/core/llm/provider-integration.test.js +350 -0
  124. package/dist/core/llm/provider-integration.test.js.map +1 -0
  125. package/dist/core/llm/provider.js +18 -0
  126. package/dist/core/llm/provider.js.map +1 -0
  127. package/dist/core/llm/routing-provider.js +52 -0
  128. package/dist/core/llm/routing-provider.js.map +1 -0
  129. package/dist/core/llm/routing-provider.test.js +94 -0
  130. package/dist/core/llm/routing-provider.test.js.map +1 -0
  131. package/dist/core/llm/schema.js +31 -0
  132. package/dist/core/llm/schema.js.map +1 -0
  133. package/dist/core/llm/snapshot-fetch.js +122 -0
  134. package/dist/core/llm/snapshot-fetch.js.map +1 -0
  135. package/dist/core/llm/snapshot-middleware.js +142 -0
  136. package/dist/core/llm/snapshot-middleware.js.map +1 -0
  137. package/dist/core/llm/snapshot-middleware.test.js +144 -0
  138. package/dist/core/llm/snapshot-middleware.test.js.map +1 -0
  139. package/dist/core/llm/state.js +48 -0
  140. package/dist/core/llm/state.js.map +1 -0
  141. package/dist/core/llm/tokens.js +40 -0
  142. package/dist/core/llm/tokens.js.map +1 -0
  143. package/dist/core/multi-agent.integration.test.js +298 -0
  144. package/dist/core/multi-agent.integration.test.js.map +1 -0
  145. package/dist/core/plugin-hooks.integration.test.js +344 -0
  146. package/dist/core/plugin-hooks.integration.test.js.map +1 -0
  147. package/dist/core/plugins/hook-types.js +5 -0
  148. package/dist/core/plugins/hook-types.js.map +1 -0
  149. package/dist/core/plugins/index.js +5 -0
  150. package/dist/core/plugins/index.js.map +1 -0
  151. package/dist/core/plugins/plugin-builder.js +321 -0
  152. package/dist/core/plugins/plugin-builder.js.map +1 -0
  153. package/dist/core/preset/config.js +54 -0
  154. package/dist/core/preset/config.js.map +1 -0
  155. package/dist/core/preset/index.js +6 -0
  156. package/dist/core/preset/index.js.map +1 -0
  157. package/dist/core/preset/preset-builder.js +63 -0
  158. package/dist/core/preset/preset-builder.js.map +1 -0
  159. package/dist/core/session-lifecycle.integration.test.js +159 -0
  160. package/dist/core/session-lifecycle.integration.test.js.map +1 -0
  161. package/dist/core/sessions/apply-event.js +41 -0
  162. package/dist/core/sessions/apply-event.js.map +1 -0
  163. package/dist/core/sessions/context.js +2 -0
  164. package/dist/core/sessions/context.js.map +1 -0
  165. package/dist/core/sessions/fork-utils.js +42 -0
  166. package/dist/core/sessions/fork-utils.js.map +1 -0
  167. package/dist/core/sessions/fork-utils.test.js +129 -0
  168. package/dist/core/sessions/fork-utils.test.js.map +1 -0
  169. package/dist/core/sessions/reducer.js +55 -0
  170. package/dist/core/sessions/reducer.js.map +1 -0
  171. package/dist/core/sessions/schema.js +66 -0
  172. package/dist/core/sessions/schema.js.map +1 -0
  173. package/dist/core/sessions/session-environment.js +2 -0
  174. package/dist/core/sessions/session-environment.js.map +1 -0
  175. package/dist/core/sessions/session-manager.js +650 -0
  176. package/dist/core/sessions/session-manager.js.map +1 -0
  177. package/dist/core/sessions/session-store.js +118 -0
  178. package/dist/core/sessions/session-store.js.map +1 -0
  179. package/dist/core/sessions/session.js +675 -0
  180. package/dist/core/sessions/session.js.map +1 -0
  181. package/dist/core/sessions/session.test.js +1095 -0
  182. package/dist/core/sessions/session.test.js.map +1 -0
  183. package/dist/core/sessions/state.js +377 -0
  184. package/dist/core/sessions/state.js.map +1 -0
  185. package/dist/core/system.js +66 -0
  186. package/dist/core/system.js.map +1 -0
  187. package/dist/core/tools/context.js +2 -0
  188. package/dist/core/tools/context.js.map +1 -0
  189. package/dist/core/tools/definition.js +4 -0
  190. package/dist/core/tools/definition.js.map +1 -0
  191. package/dist/core/tools/executor.js +82 -0
  192. package/dist/core/tools/executor.js.map +1 -0
  193. package/dist/core/tools/executor.test.js +143 -0
  194. package/dist/core/tools/executor.test.js.map +1 -0
  195. package/dist/core/tools/index.js +4 -0
  196. package/dist/core/tools/index.js.map +1 -0
  197. package/dist/core/tools/schema.js +20 -0
  198. package/dist/core/tools/schema.js.map +1 -0
  199. package/dist/core/tools/state.js +29 -0
  200. package/dist/core/tools/state.js.map +1 -0
  201. package/dist/index.js +70 -0
  202. package/dist/index.js.map +1 -0
  203. package/dist/lib/json/index.js +5 -0
  204. package/dist/lib/json/index.js.map +1 -0
  205. package/dist/lib/logger/console.js +147 -0
  206. package/dist/lib/logger/console.js.map +1 -0
  207. package/dist/lib/logger/console.test.js +258 -0
  208. package/dist/lib/logger/console.test.js.map +1 -0
  209. package/dist/lib/logger/file.js +54 -0
  210. package/dist/lib/logger/file.js.map +1 -0
  211. package/dist/lib/logger/index.js +4 -0
  212. package/dist/lib/logger/index.js.map +1 -0
  213. package/dist/lib/logger/logger.js +28 -0
  214. package/dist/lib/logger/logger.js.map +1 -0
  215. package/dist/lib/logger/ring-buffer.js +61 -0
  216. package/dist/lib/logger/ring-buffer.js.map +1 -0
  217. package/dist/lib/logger/tee.js +43 -0
  218. package/dist/lib/logger/tee.js.map +1 -0
  219. package/dist/lib/mime.js +22 -0
  220. package/dist/lib/mime.js.map +1 -0
  221. package/dist/lib/never.js +4 -0
  222. package/dist/lib/never.js.map +1 -0
  223. package/dist/lib/utils/hash.js +35 -0
  224. package/dist/lib/utils/hash.js.map +1 -0
  225. package/dist/lib/utils/result.js +21 -0
  226. package/dist/lib/utils/result.js.map +1 -0
  227. package/dist/platform/fs.js +8 -0
  228. package/dist/platform/fs.js.map +1 -0
  229. package/dist/platform/index.js +9 -0
  230. package/dist/platform/index.js.map +1 -0
  231. package/dist/platform/process.js +8 -0
  232. package/dist/platform/process.js.map +1 -0
  233. package/dist/plugins/agent-status/plugin.js +77 -0
  234. package/dist/plugins/agent-status/plugin.js.map +1 -0
  235. package/dist/plugins/agents/agents.integration.test.js +683 -0
  236. package/dist/plugins/agents/agents.integration.test.js.map +1 -0
  237. package/dist/plugins/agents/index.js +2 -0
  238. package/dist/plugins/agents/index.js.map +1 -0
  239. package/dist/plugins/agents/plugin.js +199 -0
  240. package/dist/plugins/agents/plugin.js.map +1 -0
  241. package/dist/plugins/context-compact/context-compact.integration.test.js +174 -0
  242. package/dist/plugins/context-compact/context-compact.integration.test.js.map +1 -0
  243. package/dist/plugins/context-compact/context-compactor.js +238 -0
  244. package/dist/plugins/context-compact/context-compactor.js.map +1 -0
  245. package/dist/plugins/context-compact/context-compactor.test.js +763 -0
  246. package/dist/plugins/context-compact/context-compactor.test.js.map +1 -0
  247. package/dist/plugins/context-compact/history-offloader.js +42 -0
  248. package/dist/plugins/context-compact/history-offloader.js.map +1 -0
  249. package/dist/plugins/context-compact/history-offloader.test.js +77 -0
  250. package/dist/plugins/context-compact/history-offloader.test.js.map +1 -0
  251. package/dist/plugins/context-compact/index.js +4 -0
  252. package/dist/plugins/context-compact/index.js.map +1 -0
  253. package/dist/plugins/context-compact/plugin.js +37 -0
  254. package/dist/plugins/context-compact/plugin.js.map +1 -0
  255. package/dist/plugins/filesystem/filesystem.integration.test.js +411 -0
  256. package/dist/plugins/filesystem/filesystem.integration.test.js.map +1 -0
  257. package/dist/plugins/filesystem/helpers.js +170 -0
  258. package/dist/plugins/filesystem/helpers.js.map +1 -0
  259. package/dist/plugins/filesystem/index.js +3 -0
  260. package/dist/plugins/filesystem/index.js.map +1 -0
  261. package/dist/plugins/filesystem/listing.js +247 -0
  262. package/dist/plugins/filesystem/listing.js.map +1 -0
  263. package/dist/plugins/filesystem/plugin.js +364 -0
  264. package/dist/plugins/filesystem/plugin.js.map +1 -0
  265. package/dist/plugins/filesystem/schema.js +2 -0
  266. package/dist/plugins/filesystem/schema.js.map +1 -0
  267. package/dist/plugins/git-status/index.js +2 -0
  268. package/dist/plugins/git-status/index.js.map +1 -0
  269. package/dist/plugins/git-status/plugin.js +144 -0
  270. package/dist/plugins/git-status/plugin.js.map +1 -0
  271. package/dist/plugins/limits-guard/config.js +5 -0
  272. package/dist/plugins/limits-guard/config.js.map +1 -0
  273. package/dist/plugins/limits-guard/index.js +3 -0
  274. package/dist/plugins/limits-guard/index.js.map +1 -0
  275. package/dist/plugins/limits-guard/limit-guard.js +125 -0
  276. package/dist/plugins/limits-guard/limit-guard.js.map +1 -0
  277. package/dist/plugins/limits-guard/limit-guard.test.js +121 -0
  278. package/dist/plugins/limits-guard/limit-guard.test.js.map +1 -0
  279. package/dist/plugins/limits-guard/limits-guard.integration.test.js +378 -0
  280. package/dist/plugins/limits-guard/limits-guard.integration.test.js.map +1 -0
  281. package/dist/plugins/limits-guard/plugin.js +240 -0
  282. package/dist/plugins/limits-guard/plugin.js.map +1 -0
  283. package/dist/plugins/llm-debug/index.js +2 -0
  284. package/dist/plugins/llm-debug/index.js.map +1 -0
  285. package/dist/plugins/llm-debug/llm-debug.integration.test.js +157 -0
  286. package/dist/plugins/llm-debug/llm-debug.integration.test.js.map +1 -0
  287. package/dist/plugins/llm-debug/plugin.js +148 -0
  288. package/dist/plugins/llm-debug/plugin.js.map +1 -0
  289. package/dist/plugins/logs/index.js +2 -0
  290. package/dist/plugins/logs/index.js.map +1 -0
  291. package/dist/plugins/logs/plugin.js +38 -0
  292. package/dist/plugins/logs/plugin.js.map +1 -0
  293. package/dist/plugins/mailbox/helpers.js +66 -0
  294. package/dist/plugins/mailbox/helpers.js.map +1 -0
  295. package/dist/plugins/mailbox/index.js +9 -0
  296. package/dist/plugins/mailbox/index.js.map +1 -0
  297. package/dist/plugins/mailbox/mailbox.integration.test.js +605 -0
  298. package/dist/plugins/mailbox/mailbox.integration.test.js.map +1 -0
  299. package/dist/plugins/mailbox/plugin.js +204 -0
  300. package/dist/plugins/mailbox/plugin.js.map +1 -0
  301. package/dist/plugins/mailbox/prompts.js +93 -0
  302. package/dist/plugins/mailbox/prompts.js.map +1 -0
  303. package/dist/plugins/mailbox/query.js +38 -0
  304. package/dist/plugins/mailbox/query.js.map +1 -0
  305. package/dist/plugins/mailbox/schema.js +32 -0
  306. package/dist/plugins/mailbox/schema.js.map +1 -0
  307. package/dist/plugins/mailbox/state.js +41 -0
  308. package/dist/plugins/mailbox/state.js.map +1 -0
  309. package/dist/plugins/resources/index.js +4 -0
  310. package/dist/plugins/resources/index.js.map +1 -0
  311. package/dist/plugins/resources/manifest.js +20 -0
  312. package/dist/plugins/resources/manifest.js.map +1 -0
  313. package/dist/plugins/resources/plugin.js +171 -0
  314. package/dist/plugins/resources/plugin.js.map +1 -0
  315. package/dist/plugins/resources/post-inject.js +32 -0
  316. package/dist/plugins/resources/post-inject.js.map +1 -0
  317. package/dist/plugins/resources/state.js +16 -0
  318. package/dist/plugins/resources/state.js.map +1 -0
  319. package/dist/plugins/result-eviction/index.js +2 -0
  320. package/dist/plugins/result-eviction/index.js.map +1 -0
  321. package/dist/plugins/result-eviction/plugin.js +43 -0
  322. package/dist/plugins/result-eviction/plugin.js.map +1 -0
  323. package/dist/plugins/result-eviction/result-eviction.integration.test.js +217 -0
  324. package/dist/plugins/result-eviction/result-eviction.integration.test.js.map +1 -0
  325. package/dist/plugins/services/plugin.js +453 -0
  326. package/dist/plugins/services/plugin.js.map +1 -0
  327. package/dist/plugins/services/port-pool.js +70 -0
  328. package/dist/plugins/services/port-pool.js.map +1 -0
  329. package/dist/plugins/services/prompt.js +40 -0
  330. package/dist/plugins/services/prompt.js.map +1 -0
  331. package/dist/plugins/services/schema.js +9 -0
  332. package/dist/plugins/services/schema.js.map +1 -0
  333. package/dist/plugins/services/service.js +470 -0
  334. package/dist/plugins/services/service.js.map +1 -0
  335. package/dist/plugins/services/services.integration.test.js +485 -0
  336. package/dist/plugins/services/services.integration.test.js.map +1 -0
  337. package/dist/plugins/session-lifecycle/index.js +2 -0
  338. package/dist/plugins/session-lifecycle/index.js.map +1 -0
  339. package/dist/plugins/session-lifecycle/plugin.js +273 -0
  340. package/dist/plugins/session-lifecycle/plugin.js.map +1 -0
  341. package/dist/plugins/session-lifecycle/session-lifecycle.integration.test.js +498 -0
  342. package/dist/plugins/session-lifecycle/session-lifecycle.integration.test.js.map +1 -0
  343. package/dist/plugins/session-state/plugin.js +159 -0
  344. package/dist/plugins/session-state/plugin.js.map +1 -0
  345. package/dist/plugins/session-stats/index.js +3 -0
  346. package/dist/plugins/session-stats/index.js.map +1 -0
  347. package/dist/plugins/session-stats/plugin.js +81 -0
  348. package/dist/plugins/session-stats/plugin.js.map +1 -0
  349. package/dist/plugins/shell/executor.js +339 -0
  350. package/dist/plugins/shell/executor.js.map +1 -0
  351. package/dist/plugins/shell/index.js +6 -0
  352. package/dist/plugins/shell/index.js.map +1 -0
  353. package/dist/plugins/shell/plugin.js +66 -0
  354. package/dist/plugins/shell/plugin.js.map +1 -0
  355. package/dist/plugins/shell/shell.integration.test.js +234 -0
  356. package/dist/plugins/shell/shell.integration.test.js.map +1 -0
  357. package/dist/plugins/shell/shell.test.js +236 -0
  358. package/dist/plugins/shell/shell.test.js.map +1 -0
  359. package/dist/plugins/skills/discovery.js +205 -0
  360. package/dist/plugins/skills/discovery.js.map +1 -0
  361. package/dist/plugins/skills/discovery.test.js +312 -0
  362. package/dist/plugins/skills/discovery.test.js.map +1 -0
  363. package/dist/plugins/skills/index.js +12 -0
  364. package/dist/plugins/skills/index.js.map +1 -0
  365. package/dist/plugins/skills/plugin.js +293 -0
  366. package/dist/plugins/skills/plugin.js.map +1 -0
  367. package/dist/plugins/skills/prompts.js +70 -0
  368. package/dist/plugins/skills/prompts.js.map +1 -0
  369. package/dist/plugins/skills/schema.js +18 -0
  370. package/dist/plugins/skills/schema.js.map +1 -0
  371. package/dist/plugins/skills/skills.integration.test.js +475 -0
  372. package/dist/plugins/skills/skills.integration.test.js.map +1 -0
  373. package/dist/plugins/snapshotting/index.js +3 -0
  374. package/dist/plugins/snapshotting/index.js.map +1 -0
  375. package/dist/plugins/snapshotting/jj-snapshotter.js +106 -0
  376. package/dist/plugins/snapshotting/jj-snapshotter.js.map +1 -0
  377. package/dist/plugins/snapshotting/plugin.js +28 -0
  378. package/dist/plugins/snapshotting/plugin.js.map +1 -0
  379. package/dist/plugins/snapshotting/snapshotter.js +2 -0
  380. package/dist/plugins/snapshotting/snapshotter.js.map +1 -0
  381. package/dist/plugins/todo/index.js +7 -0
  382. package/dist/plugins/todo/index.js.map +1 -0
  383. package/dist/plugins/todo/plugin.js +319 -0
  384. package/dist/plugins/todo/plugin.js.map +1 -0
  385. package/dist/plugins/todo/prompts.js +54 -0
  386. package/dist/plugins/todo/prompts.js.map +1 -0
  387. package/dist/plugins/todo/schema.js +18 -0
  388. package/dist/plugins/todo/schema.js.map +1 -0
  389. package/dist/plugins/todo/todo.integration.test.js +605 -0
  390. package/dist/plugins/todo/todo.integration.test.js.map +1 -0
  391. package/dist/plugins/uploads/index.js +8 -0
  392. package/dist/plugins/uploads/index.js.map +1 -0
  393. package/dist/plugins/uploads/plugin.js +346 -0
  394. package/dist/plugins/uploads/plugin.js.map +1 -0
  395. package/dist/plugins/uploads/preprocessor.js +44 -0
  396. package/dist/plugins/uploads/preprocessor.js.map +1 -0
  397. package/dist/plugins/uploads/preprocessors/image-classifier.js +127 -0
  398. package/dist/plugins/uploads/preprocessors/image-classifier.js.map +1 -0
  399. package/dist/plugins/uploads/preprocessors/index.js +7 -0
  400. package/dist/plugins/uploads/preprocessors/index.js.map +1 -0
  401. package/dist/plugins/uploads/preprocessors/markitdown-preprocessor.js +204 -0
  402. package/dist/plugins/uploads/preprocessors/markitdown-preprocessor.js.map +1 -0
  403. package/dist/plugins/uploads/preprocessors/zip-preprocessor.js +172 -0
  404. package/dist/plugins/uploads/preprocessors/zip-preprocessor.js.map +1 -0
  405. package/dist/plugins/uploads/schema.js +20 -0
  406. package/dist/plugins/uploads/schema.js.map +1 -0
  407. package/dist/plugins/uploads/state.js +22 -0
  408. package/dist/plugins/uploads/state.js.map +1 -0
  409. package/dist/plugins/uploads/uploads.integration.test.js +496 -0
  410. package/dist/plugins/uploads/uploads.integration.test.js.map +1 -0
  411. package/dist/plugins/user-chat/index.js +5 -0
  412. package/dist/plugins/user-chat/index.js.map +1 -0
  413. package/dist/plugins/user-chat/plugin.js +544 -0
  414. package/dist/plugins/user-chat/plugin.js.map +1 -0
  415. package/dist/plugins/user-chat/prompts.js +29 -0
  416. package/dist/plugins/user-chat/prompts.js.map +1 -0
  417. package/dist/plugins/user-chat/schema.js +46 -0
  418. package/dist/plugins/user-chat/schema.js.map +1 -0
  419. package/dist/plugins/user-chat/user-chat.integration.test.js +668 -0
  420. package/dist/plugins/user-chat/user-chat.integration.test.js.map +1 -0
  421. package/dist/plugins/workers/context.js +143 -0
  422. package/dist/plugins/workers/context.js.map +1 -0
  423. package/dist/plugins/workers/definition.js +30 -0
  424. package/dist/plugins/workers/definition.js.map +1 -0
  425. package/dist/plugins/workers/index.js +7 -0
  426. package/dist/plugins/workers/index.js.map +1 -0
  427. package/dist/plugins/workers/plugin.js +578 -0
  428. package/dist/plugins/workers/plugin.js.map +1 -0
  429. package/dist/plugins/workers/worker.js +18 -0
  430. package/dist/plugins/workers/worker.js.map +1 -0
  431. package/dist/plugins/workers/workers.integration.test.js +629 -0
  432. package/dist/plugins/workers/workers.integration.test.js.map +1 -0
  433. package/dist/prompts/base.js +239 -0
  434. package/dist/prompts/base.js.map +1 -0
  435. package/dist/prompts/builder.js +131 -0
  436. package/dist/prompts/builder.js.map +1 -0
  437. package/dist/prompts/index.js +20 -0
  438. package/dist/prompts/index.js.map +1 -0
  439. package/dist/prompts/macros.js +26 -0
  440. package/dist/prompts/macros.js.map +1 -0
  441. package/dist/prompts/macros.test.js +80 -0
  442. package/dist/prompts/macros.test.js.map +1 -0
  443. package/dist/testing/bootstrap-for-testing.js +28 -0
  444. package/dist/testing/bootstrap-for-testing.js.map +1 -0
  445. package/dist/testing/index.js +7 -0
  446. package/dist/testing/index.js.map +1 -0
  447. package/dist/testing/node-platform.js +65 -0
  448. package/dist/testing/node-platform.js.map +1 -0
  449. package/dist/testing/notification-collector.js +82 -0
  450. package/dist/testing/notification-collector.js.map +1 -0
  451. package/dist/testing/preset-helpers.js +37 -0
  452. package/dist/testing/preset-helpers.js.map +1 -0
  453. package/dist/testing/test-harness.js +226 -0
  454. package/dist/testing/test-harness.js.map +1 -0
  455. package/dist/testing/test-harness.test.js +51 -0
  456. package/dist/testing/test-harness.test.js.map +1 -0
  457. package/dist/testing/wait-helpers.js +64 -0
  458. package/dist/testing/wait-helpers.js.map +1 -0
  459. package/dist/transport/adapter/client-adapter.js +64 -0
  460. package/dist/transport/adapter/client-adapter.js.map +1 -0
  461. package/dist/transport/adapter/index.js +24 -0
  462. package/dist/transport/adapter/index.js.map +1 -0
  463. package/dist/transport/adapter/server-adapter.js +73 -0
  464. package/dist/transport/adapter/server-adapter.js.map +1 -0
  465. package/dist/transport/adapter/types.js +8 -0
  466. package/dist/transport/adapter/types.js.map +1 -0
  467. package/dist/transport/http/app.js +86 -0
  468. package/dist/transport/http/app.js.map +1 -0
  469. package/dist/transport/http/index.js +6 -0
  470. package/dist/transport/http/index.js.map +1 -0
  471. package/dist/transport/http/middleware/bearer-auth.js +33 -0
  472. package/dist/transport/http/middleware/bearer-auth.js.map +1 -0
  473. package/dist/transport/http/middleware/error-handler.js +56 -0
  474. package/dist/transport/http/middleware/error-handler.js.map +1 -0
  475. package/dist/transport/http/routes/files.js +237 -0
  476. package/dist/transport/http/routes/files.js.map +1 -0
  477. package/dist/transport/http/routes/resources.js +77 -0
  478. package/dist/transport/http/routes/resources.js.map +1 -0
  479. package/dist/transport/http/routes/rpc.integration.test.js +189 -0
  480. package/dist/transport/http/routes/rpc.integration.test.js.map +1 -0
  481. package/dist/transport/http/routes/rpc.js +110 -0
  482. package/dist/transport/http/routes/rpc.js.map +1 -0
  483. package/dist/transport/http/routes/rpc.test.js +316 -0
  484. package/dist/transport/http/routes/rpc.test.js.map +1 -0
  485. package/dist/transport/http/routes/upload.js +205 -0
  486. package/dist/transport/http/routes/upload.js.map +1 -0
  487. package/dist/transport/rpc/index.js +7 -0
  488. package/dist/transport/rpc/index.js.map +1 -0
  489. package/dist/transport/rpc/methods.js +8 -0
  490. package/dist/transport/rpc/methods.js.map +1 -0
  491. package/dist/user-config.js +14 -0
  492. package/dist/user-config.js.map +1 -0
  493. package/package.json +47 -57
@@ -0,0 +1,171 @@
1
+ import { describe, expect, it } from 'bun:test';
2
+ import { tmpdir } from 'node:os';
3
+ import { join } from 'node:path';
4
+ import { Err, Ok } from '../../lib/utils/result.js';
5
+ import { createNodeFileSystem } from '../../testing/node-platform.js';
6
+ import { DefaultImageProcessor } from './image-processor.js';
7
+ import { NoopImageResizer } from './noop-resizer.js';
8
+ describe('DefaultImageProcessor', () => {
9
+ const tempDir = tmpdir();
10
+ function createProcessor(resizer, maxFileSizeBytes = 5 * 1024 * 1024) {
11
+ return new DefaultImageProcessor(resizer ?? new NoopImageResizer(), createNodeFileSystem(), { maxFileSizeBytes });
12
+ }
13
+ it('passes through string content unchanged', async () => {
14
+ const processor = createProcessor();
15
+ const result = await processor.resolveContent('hello world');
16
+ expect(result).toBe('hello world');
17
+ });
18
+ it('passes through non-image content items unchanged', async () => {
19
+ const processor = createProcessor();
20
+ const content = [
21
+ { type: 'text', text: 'some text' },
22
+ ];
23
+ const result = await processor.resolveContent(content);
24
+ expect(result).toEqual([{ type: 'text', text: 'some text' }]);
25
+ });
26
+ it('passes through non-file:// image URLs unchanged', async () => {
27
+ const processor = createProcessor();
28
+ const content = [
29
+ { type: 'image_url', imageUrl: { url: 'https://example.com/img.png' } },
30
+ ];
31
+ const result = await processor.resolveContent(content);
32
+ expect(result).toEqual([{ type: 'image_url', imageUrl: { url: 'https://example.com/img.png' } }]);
33
+ });
34
+ it('resolves file:// image to data URL with correct MIME from resizer', async () => {
35
+ // Create a small test file
36
+ const testPath = join(tempDir, `test-processor-${Date.now()}.png`);
37
+ await Bun.write(testPath, Buffer.from('fake-png-data'));
38
+ const resizer = {
39
+ async resize(filePath, _mimeType, _options) {
40
+ return { path: filePath, mimeType: 'image/jpeg' };
41
+ },
42
+ };
43
+ const processor = createProcessor(resizer);
44
+ const content = [
45
+ { type: 'image_url', imageUrl: { url: `file://${testPath}`, detail: 'auto' } },
46
+ ];
47
+ const result = await processor.resolveContent(content);
48
+ const item = result[0];
49
+ expect(item.type).toBe('image_url');
50
+ if (item.type === 'image_url') {
51
+ expect(item.imageUrl.url).toStartWith('data:image/jpeg;base64,');
52
+ expect(item.imageUrl.detail).toBe('auto');
53
+ }
54
+ await Bun.file(testPath).exists() && await import('node:fs/promises').then(fs => fs.unlink(testPath).catch(() => { }));
55
+ });
56
+ it('returns text placeholder when file exceeds size limit', async () => {
57
+ const testPath = join(tempDir, `test-large-${Date.now()}.jpg`);
58
+ // Write a file larger than the limit
59
+ await Bun.write(testPath, Buffer.alloc(100));
60
+ const processor = createProcessor(new NoopImageResizer(), 50); // 50 byte limit
61
+ const content = [
62
+ { type: 'image_url', imageUrl: { url: `file://${testPath}` } },
63
+ ];
64
+ const result = await processor.resolveContent(content);
65
+ const item = result[0];
66
+ expect(item.type).toBe('text');
67
+ if (item.type === 'text') {
68
+ expect(item.text).toContain('file too large');
69
+ }
70
+ await import('node:fs/promises').then(fs => fs.unlink(testPath).catch(() => { }));
71
+ });
72
+ it('returns text placeholder for unsupported format', async () => {
73
+ const testPath = join(tempDir, `test-${Date.now()}.xyz`);
74
+ await Bun.write(testPath, 'some data');
75
+ const processor = createProcessor();
76
+ const content = [
77
+ { type: 'image_url', imageUrl: { url: `file://${testPath}` } },
78
+ ];
79
+ const result = await processor.resolveContent(content);
80
+ const item = result[0];
81
+ expect(item.type).toBe('text');
82
+ if (item.type === 'text') {
83
+ expect(item.text).toContain('unsupported format');
84
+ }
85
+ await import('node:fs/promises').then(fs => fs.unlink(testPath).catch(() => { }));
86
+ });
87
+ it('returns text placeholder when file not found', async () => {
88
+ const processor = createProcessor();
89
+ const content = [
90
+ { type: 'image_url', imageUrl: { url: 'file:///nonexistent/path/image.png' } },
91
+ ];
92
+ const result = await processor.resolveContent(content);
93
+ const item = result[0];
94
+ expect(item.type).toBe('text');
95
+ if (item.type === 'text') {
96
+ expect(item.text).toContain('file not found');
97
+ }
98
+ });
99
+ it('resolves via FileStore when provided', async () => {
100
+ const testPath = join(tempDir, `test-store-${Date.now()}.png`);
101
+ await Bun.write(testPath, Buffer.from('fake-png-data'));
102
+ const fileStore = {
103
+ realPath: (path) => Ok(testPath),
104
+ };
105
+ const processor = createProcessor();
106
+ const content = [
107
+ { type: 'image_url', imageUrl: { url: 'file://workspace/image.png' } },
108
+ ];
109
+ const result = await processor.resolveContent(content, fileStore);
110
+ const item = result[0];
111
+ expect(item.type).toBe('image_url');
112
+ if (item.type === 'image_url') {
113
+ expect(item.imageUrl.url).toStartWith('data:image/png;base64,');
114
+ }
115
+ await import('node:fs/promises').then(fs => fs.unlink(testPath).catch(() => { }));
116
+ });
117
+ it('returns text placeholder when FileStore.realPath fails', async () => {
118
+ const fileStore = {
119
+ realPath: (_path) => Err('path outside sandbox'),
120
+ };
121
+ const processor = createProcessor();
122
+ const content = [
123
+ { type: 'image_url', imageUrl: { url: 'file://workspace/image.png' } },
124
+ ];
125
+ const result = await processor.resolveContent(content, fileStore);
126
+ const item = result[0];
127
+ expect(item.type).toBe('text');
128
+ if (item.type === 'text') {
129
+ expect(item.text).toContain('path outside sandbox');
130
+ }
131
+ });
132
+ it('passes maxFileSizeBytes to resizer', async () => {
133
+ const testPath = join(tempDir, `test-opts-${Date.now()}.png`);
134
+ await Bun.write(testPath, Buffer.from('fake-png-data'));
135
+ let receivedOptions;
136
+ const resizer = {
137
+ async resize(filePath, _mimeType, options) {
138
+ receivedOptions = options;
139
+ return { path: filePath, mimeType: 'image/png' };
140
+ },
141
+ };
142
+ const processor = createProcessor(resizer, 999);
143
+ const content = [
144
+ { type: 'image_url', imageUrl: { url: `file://${testPath}` } },
145
+ ];
146
+ await processor.resolveContent(content);
147
+ expect(receivedOptions).toEqual({ maxFileSizeBytes: 999 });
148
+ await import('node:fs/promises').then(fs => fs.unlink(testPath).catch(() => { }));
149
+ });
150
+ it('cleans up temp files from resizer', async () => {
151
+ const testPath = join(tempDir, `test-cleanup-${Date.now()}.png`);
152
+ const tempFile = join(tempDir, `test-temp-${Date.now()}.jpg`);
153
+ await Bun.write(testPath, Buffer.from('fake-png-data'));
154
+ await Bun.write(tempFile, Buffer.from('resized-data'));
155
+ const resizer = {
156
+ async resize(_filePath, _mimeType, _options) {
157
+ return { path: tempFile, mimeType: 'image/jpeg', tempFile };
158
+ },
159
+ };
160
+ const processor = createProcessor(resizer);
161
+ const content = [
162
+ { type: 'image_url', imageUrl: { url: `file://${testPath}` } },
163
+ ];
164
+ await processor.resolveContent(content);
165
+ // Temp file should be cleaned up
166
+ const exists = await Bun.file(tempFile).exists();
167
+ expect(exists).toBe(false);
168
+ await import('node:fs/promises').then(fs => fs.unlink(testPath).catch(() => { }));
169
+ });
170
+ });
171
+ //# sourceMappingURL=image-processor.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"image-processor.test.js","sourceRoot":"","sources":["../../../src/core/image/image-processor.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ export { DefaultImageProcessor } from './image-processor.js';
2
+ export { NoopImageResizer } from './noop-resizer.js';
3
+ export { VipsImageResizer } from './vips-resizer.js';
4
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/core/image/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,qBAAqB,EAAE,MAAM,sBAAsB,CAAA;AAC5D,OAAO,EAAE,gBAAgB,EAAE,MAAM,mBAAmB,CAAA;AAEpD,OAAO,EAAE,gBAAgB,EAAE,MAAM,mBAAmB,CAAA"}
@@ -0,0 +1,6 @@
1
+ export class NoopImageResizer {
2
+ async resize(filePath, mimeType, _options) {
3
+ return { path: filePath, mimeType };
4
+ }
5
+ }
6
+ //# sourceMappingURL=noop-resizer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"noop-resizer.js","sourceRoot":"","sources":["../../../src/core/image/noop-resizer.ts"],"names":[],"mappings":"AAEA,MAAM,OAAO,gBAAgB;IAC5B,KAAK,CAAC,MAAM,CAAC,QAAgB,EAAE,QAAgB,EAAE,QAA6B;QAC7E,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,QAAQ,EAAE,CAAA;IACpC,CAAC;CACD"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/core/image/types.ts"],"names":[],"mappings":""}
@@ -0,0 +1,100 @@
1
+ import { join } from 'node:path';
2
+ export class VipsImageResizer {
3
+ fs;
4
+ process;
5
+ tmpDir;
6
+ maxDimension;
7
+ constructor(options) {
8
+ this.fs = options.fs;
9
+ this.process = options.process;
10
+ this.tmpDir = options.tmpDir;
11
+ this.maxDimension = options.maxDimension ?? 8000;
12
+ }
13
+ async resize(filePath, mimeType, options) {
14
+ try {
15
+ // Step 1: Dimension resize if needed
16
+ const result = await this.dimensionResize(filePath, mimeType);
17
+ // Step 2: If no size constraint, done
18
+ if (!options?.maxFileSizeBytes)
19
+ return result;
20
+ // Step 3: Check if result fits
21
+ const fileSize = (await this.fs.stat(result.path)).size;
22
+ if (fileSize <= options.maxFileSizeBytes)
23
+ return result;
24
+ // Step 4: Compress to fit — clean up dimension resize temp first
25
+ if (result.tempFile) {
26
+ await this.fs.unlink(result.tempFile).catch(() => { });
27
+ }
28
+ return await this.compressToFit(filePath, options.maxFileSizeBytes);
29
+ }
30
+ catch (e) {
31
+ console.warn('[image-resize] failed, using original image:', e instanceof Error ? e.message : e);
32
+ return { path: filePath, mimeType };
33
+ }
34
+ }
35
+ async getImageDimensions(filePath) {
36
+ const { stdout } = await this.process.execFile('vipsheader', ['-f', 'width', '-f', 'height', filePath], { timeout: 30_000 });
37
+ const lines = stdout.trim().split('\n');
38
+ if (lines.length < 2)
39
+ return null;
40
+ const width = parseInt(lines[0], 10);
41
+ const height = parseInt(lines[1], 10);
42
+ if (!Number.isFinite(width) || !Number.isFinite(height))
43
+ return null;
44
+ return { width, height };
45
+ }
46
+ async dimensionResize(filePath, mimeType) {
47
+ const dims = await this.getImageDimensions(filePath);
48
+ const needsResize = dims !== null && (dims.width > this.maxDimension || dims.height > this.maxDimension);
49
+ // JPEGs within dimension limits pass through unchanged
50
+ if (mimeType === 'image/jpeg' && !needsResize) {
51
+ return { path: filePath, mimeType };
52
+ }
53
+ // Always convert to JPEG (PNG→JPEG saves significant size for LLM context)
54
+ const id = `${Date.now()}-${Math.random().toString(36).slice(2)}`;
55
+ const outputPath = join(this.tmpDir, `roj-resize-${id}.jpg`);
56
+ await this.process.execFile('vipsthumbnail', [
57
+ filePath,
58
+ '--size',
59
+ `${this.maxDimension}x${this.maxDimension}`,
60
+ '-o',
61
+ outputPath,
62
+ ], { timeout: 30_000 });
63
+ return { path: outputPath, mimeType: 'image/jpeg', tempFile: outputPath };
64
+ }
65
+ async compressToFit(filePath, maxFileSizeBytes) {
66
+ const halfDim = Math.floor(this.maxDimension / 2);
67
+ const attempts = [
68
+ { dimension: this.maxDimension, quality: 85 },
69
+ { dimension: this.maxDimension, quality: 70 },
70
+ { dimension: this.maxDimension, quality: 50 },
71
+ { dimension: this.maxDimension, quality: 30 },
72
+ { dimension: halfDim, quality: 70 },
73
+ { dimension: halfDim, quality: 50 },
74
+ { dimension: halfDim, quality: 30 },
75
+ ];
76
+ let lastResult;
77
+ for (const { dimension, quality } of attempts) {
78
+ const id = `${Date.now()}-${Math.random().toString(36).slice(2)}`;
79
+ const outputPath = join(this.tmpDir, `roj-compress-${id}.jpg`);
80
+ await this.process.execFile('vipsthumbnail', [
81
+ filePath,
82
+ '--size',
83
+ `${dimension}x${dimension}`,
84
+ '-o',
85
+ `${outputPath}[Q=${quality}]`,
86
+ ], { timeout: 30_000 });
87
+ // Clean up previous attempt
88
+ if (lastResult?.tempFile) {
89
+ await this.fs.unlink(lastResult.tempFile).catch(() => { });
90
+ }
91
+ lastResult = { path: outputPath, mimeType: 'image/jpeg', tempFile: outputPath };
92
+ if ((await this.fs.stat(outputPath)).size <= maxFileSizeBytes) {
93
+ return lastResult;
94
+ }
95
+ }
96
+ // Return best effort (most compressed) — caller decides what to do
97
+ return lastResult;
98
+ }
99
+ }
100
+ //# sourceMappingURL=vips-resizer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"vips-resizer.js","sourceRoot":"","sources":["../../../src/core/image/vips-resizer.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,324 @@
1
+ import { afterEach, describe, expect, it, mock, spyOn } from 'bun:test';
2
+ import * as childProcess from 'node:child_process';
3
+ let execFileImpl = () => { };
4
+ mock.module('node:child_process', () => ({
5
+ ...childProcess,
6
+ execFile: (cmd, args, opts, cb) => execFileImpl(cmd, args, opts, cb),
7
+ }));
8
+ const { VipsImageResizer } = await import('./vips-resizer.js');
9
+ const { createNodePlatform } = await import('../../testing/node-platform.js');
10
+ // Test-scoped helper — routes through createNodePlatform so the module-level
11
+ // node:child_process mock still intercepts execFile calls made by ProcessRunner.
12
+ function createResizer(maxDimension) {
13
+ const platform = createNodePlatform();
14
+ return new VipsImageResizer({
15
+ fs: platform.fs,
16
+ process: platform.process,
17
+ tmpDir: platform.tmpDir,
18
+ maxDimension,
19
+ });
20
+ }
21
+ const warnSpy = spyOn(console, 'warn').mockImplementation(() => { });
22
+ afterEach(() => {
23
+ warnSpy.mockClear();
24
+ execFileImpl = () => { };
25
+ });
26
+ describe('VipsImageResizer', () => {
27
+ const testJpegPath = '/tmp/test-image.jpg';
28
+ it('returns original path and mimeType when jpeg is within limits', async () => {
29
+ execFileImpl = (cmd, _args, _opts, cb) => {
30
+ if (cmd === 'vipsheader')
31
+ cb(null, '4000\n3000\n', '');
32
+ };
33
+ const resizer = createResizer();
34
+ const result = await resizer.resize(testJpegPath, 'image/jpeg');
35
+ expect(result).toEqual({ path: testJpegPath, mimeType: 'image/jpeg' });
36
+ });
37
+ it('returns original path when jpeg is exactly at dimension limit', async () => {
38
+ execFileImpl = (cmd, _args, _opts, cb) => {
39
+ if (cmd === 'vipsheader')
40
+ cb(null, '8000\n8000\n', '');
41
+ };
42
+ const resizer = createResizer();
43
+ const result = await resizer.resize(testJpegPath, 'image/jpeg');
44
+ expect(result).toEqual({ path: testJpegPath, mimeType: 'image/jpeg' });
45
+ });
46
+ it('converts png to jpeg even when within dimension limits', async () => {
47
+ execFileImpl = (cmd, _args, _opts, cb) => {
48
+ if (cmd === 'vipsheader')
49
+ cb(null, '4000\n3000\n', '');
50
+ else if (cmd === 'vipsthumbnail')
51
+ cb(null, '', '');
52
+ };
53
+ const resizer = createResizer();
54
+ const result = await resizer.resize('/tmp/test.png', 'image/png');
55
+ expect(result.path).toEndWith('.jpg');
56
+ expect(result.mimeType).toBe('image/jpeg');
57
+ expect(result.tempFile).toBeDefined();
58
+ });
59
+ it('calls vipsthumbnail and returns temp path when width exceeds limit', async () => {
60
+ let thumbnailArgs = [];
61
+ execFileImpl = (cmd, args, _opts, cb) => {
62
+ if (cmd === 'vipsheader')
63
+ cb(null, '10000\n5000\n', '');
64
+ else if (cmd === 'vipsthumbnail') {
65
+ thumbnailArgs = args;
66
+ cb(null, '', '');
67
+ }
68
+ };
69
+ const resizer = createResizer();
70
+ const result = await resizer.resize('/tmp/test.png', 'image/png');
71
+ expect(result.path).not.toBe('/tmp/test.png');
72
+ expect(result.tempFile).toBe(result.path);
73
+ expect(result.path).toContain('roj-resize-');
74
+ expect(result.path).toEndWith('.jpg');
75
+ expect(result.mimeType).toBe('image/jpeg');
76
+ expect(thumbnailArgs).toContain('--size');
77
+ expect(thumbnailArgs).toContain('8000x8000');
78
+ expect(thumbnailArgs).toContain('/tmp/test.png');
79
+ });
80
+ it('calls vipsthumbnail when height exceeds limit', async () => {
81
+ execFileImpl = (cmd, _args, _opts, cb) => {
82
+ if (cmd === 'vipsheader')
83
+ cb(null, '4000\n12000\n', '');
84
+ else if (cmd === 'vipsthumbnail')
85
+ cb(null, '', '');
86
+ };
87
+ const resizer = createResizer();
88
+ const result = await resizer.resize(testJpegPath, 'image/jpeg');
89
+ expect(result.tempFile).toBeDefined();
90
+ expect(result.path).toEndWith('.jpg');
91
+ expect(result.mimeType).toBe('image/jpeg');
92
+ });
93
+ it('converts webp to jpeg on resize', async () => {
94
+ execFileImpl = (cmd, _args, _opts, cb) => {
95
+ if (cmd === 'vipsheader')
96
+ cb(null, '9000\n9000\n', '');
97
+ else if (cmd === 'vipsthumbnail')
98
+ cb(null, '', '');
99
+ };
100
+ const resizer = createResizer();
101
+ const result = await resizer.resize('/tmp/test.webp', 'image/webp');
102
+ expect(result.path).toEndWith('.jpg');
103
+ expect(result.mimeType).toBe('image/jpeg');
104
+ });
105
+ it('converts gif to jpeg on resize', async () => {
106
+ execFileImpl = (cmd, _args, _opts, cb) => {
107
+ if (cmd === 'vipsheader')
108
+ cb(null, '9000\n9000\n', '');
109
+ else if (cmd === 'vipsthumbnail')
110
+ cb(null, '', '');
111
+ };
112
+ const resizer = createResizer();
113
+ const result = await resizer.resize('/tmp/test.gif', 'image/gif');
114
+ expect(result.path).toEndWith('.jpg');
115
+ expect(result.mimeType).toBe('image/jpeg');
116
+ });
117
+ it('converts png to jpeg on resize', async () => {
118
+ execFileImpl = (cmd, _args, _opts, cb) => {
119
+ if (cmd === 'vipsheader')
120
+ cb(null, '9000\n9000\n', '');
121
+ else if (cmd === 'vipsthumbnail')
122
+ cb(null, '', '');
123
+ };
124
+ const resizer = createResizer();
125
+ const result = await resizer.resize('/tmp/test.png', 'image/png');
126
+ expect(result.path).toEndWith('.jpg');
127
+ expect(result.mimeType).toBe('image/jpeg');
128
+ });
129
+ it('returns original path when vipsheader fails (graceful degradation)', async () => {
130
+ execFileImpl = (_cmd, _args, _opts, cb) => {
131
+ cb(new Error('vipsheader: command not found'), '', '');
132
+ };
133
+ const resizer = createResizer();
134
+ const result = await resizer.resize(testJpegPath, 'image/png');
135
+ expect(result).toEqual({ path: testJpegPath, mimeType: 'image/png' });
136
+ expect(warnSpy).toHaveBeenCalled();
137
+ });
138
+ it('returns original path when vipsthumbnail fails (graceful degradation)', async () => {
139
+ execFileImpl = (cmd, _args, _opts, cb) => {
140
+ if (cmd === 'vipsheader')
141
+ cb(null, '10000\n10000\n', '');
142
+ else if (cmd === 'vipsthumbnail')
143
+ cb(new Error('vipsthumbnail: command not found'), '', '');
144
+ };
145
+ const resizer = createResizer();
146
+ const result = await resizer.resize(testJpegPath, 'image/png');
147
+ expect(result).toEqual({ path: testJpegPath, mimeType: 'image/png' });
148
+ expect(warnSpy).toHaveBeenCalled();
149
+ });
150
+ it('returns original path when vipsheader returns unparseable output for jpeg', async () => {
151
+ execFileImpl = (cmd, _args, _opts, cb) => {
152
+ if (cmd === 'vipsheader')
153
+ cb(null, 'not-a-number\n', '');
154
+ };
155
+ const resizer = createResizer();
156
+ const result = await resizer.resize(testJpegPath, 'image/jpeg');
157
+ expect(result).toEqual({ path: testJpegPath, mimeType: 'image/jpeg' });
158
+ });
159
+ it('uses custom maxDimension', async () => {
160
+ let thumbnailArgs = [];
161
+ execFileImpl = (cmd, args, _opts, cb) => {
162
+ if (cmd === 'vipsheader')
163
+ cb(null, '5000\n5000\n', '');
164
+ else if (cmd === 'vipsthumbnail') {
165
+ thumbnailArgs = args;
166
+ cb(null, '', '');
167
+ }
168
+ };
169
+ const resizer = createResizer(4000);
170
+ const result = await resizer.resize(testJpegPath, 'image/jpeg');
171
+ expect(result.tempFile).toBeDefined();
172
+ expect(thumbnailArgs).toContain('4000x4000');
173
+ });
174
+ describe('compression (maxFileSizeBytes)', () => {
175
+ it('skips compression when file fits within limit', async () => {
176
+ const testPath = '/tmp/test-small.jpg';
177
+ // Write a small file for size check
178
+ await Bun.write(testPath, Buffer.alloc(100));
179
+ execFileImpl = (cmd, _args, _opts, cb) => {
180
+ if (cmd === 'vipsheader')
181
+ cb(null, '4000\n3000\n', '');
182
+ };
183
+ const resizer = createResizer();
184
+ const result = await resizer.resize(testPath, 'image/jpeg', { maxFileSizeBytes: 1000 });
185
+ expect(result).toEqual({ path: testPath, mimeType: 'image/jpeg' });
186
+ await import('node:fs/promises').then(fs => fs.unlink(testPath).catch(() => { }));
187
+ });
188
+ it('tries compression with quality steps when file exceeds limit', async () => {
189
+ const testPath = '/tmp/test-large-compress.jpg';
190
+ await Bun.write(testPath, Buffer.alloc(2000));
191
+ const thumbnailCalls = [];
192
+ execFileImpl = (cmd, args, _opts, cb) => {
193
+ if (cmd === 'vipsheader') {
194
+ cb(null, '4000\n3000\n', '');
195
+ }
196
+ else if (cmd === 'vipsthumbnail') {
197
+ thumbnailCalls.push({ args });
198
+ // Write a small file so it fits on first compression attempt
199
+ const outputArg = args[args.indexOf('-o') + 1];
200
+ const outputPath = outputArg.replace(/\[.*\]$/, '');
201
+ Bun.write(outputPath, Buffer.alloc(50)).then(() => cb(null, '', ''));
202
+ }
203
+ };
204
+ const resizer = createResizer();
205
+ const result = await resizer.resize(testPath, 'image/jpeg', { maxFileSizeBytes: 100 });
206
+ expect(result.mimeType).toBe('image/jpeg');
207
+ expect(result.tempFile).toBeDefined();
208
+ expect(thumbnailCalls.length).toBe(1); // First quality step should succeed
209
+ // Check it used quality parameter
210
+ const outputArg = thumbnailCalls[0].args[thumbnailCalls[0].args.indexOf('-o') + 1];
211
+ expect(outputArg).toContain('[Q=85]');
212
+ if (result.tempFile) {
213
+ await import('node:fs/promises').then(fs => fs.unlink(result.tempFile).catch(() => { }));
214
+ }
215
+ await import('node:fs/promises').then(fs => fs.unlink(testPath).catch(() => { }));
216
+ });
217
+ it('tries progressively lower quality until file fits', async () => {
218
+ const testPath = '/tmp/test-progressive.jpg';
219
+ await Bun.write(testPath, Buffer.alloc(2000));
220
+ let compressionAttempts = 0;
221
+ execFileImpl = (cmd, args, _opts, cb) => {
222
+ if (cmd === 'vipsheader') {
223
+ cb(null, '4000\n3000\n', '');
224
+ }
225
+ else if (cmd === 'vipsthumbnail') {
226
+ compressionAttempts++;
227
+ const outputArg = args[args.indexOf('-o') + 1];
228
+ const outputPath = outputArg.replace(/\[.*\]$/, '');
229
+ // First two attempts produce large files, third fits
230
+ const size = compressionAttempts <= 2 ? 2000 : 50;
231
+ Bun.write(outputPath, Buffer.alloc(size)).then(() => cb(null, '', ''));
232
+ }
233
+ };
234
+ const resizer = createResizer();
235
+ const result = await resizer.resize(testPath, 'image/jpeg', { maxFileSizeBytes: 100 });
236
+ expect(compressionAttempts).toBe(3); // Q=85, Q=70 failed, Q=50 succeeded
237
+ expect(result.mimeType).toBe('image/jpeg');
238
+ if (result.tempFile) {
239
+ await import('node:fs/promises').then(fs => fs.unlink(result.tempFile).catch(() => { }));
240
+ }
241
+ await import('node:fs/promises').then(fs => fs.unlink(testPath).catch(() => { }));
242
+ });
243
+ it('halves dimensions when quality reduction is insufficient', async () => {
244
+ const testPath = '/tmp/test-half-dim.jpg';
245
+ await Bun.write(testPath, Buffer.alloc(2000));
246
+ const dimensionArgs = [];
247
+ let compressionAttempts = 0;
248
+ execFileImpl = (cmd, args, _opts, cb) => {
249
+ if (cmd === 'vipsheader') {
250
+ cb(null, '4000\n3000\n', '');
251
+ }
252
+ else if (cmd === 'vipsthumbnail') {
253
+ compressionAttempts++;
254
+ dimensionArgs.push(args[args.indexOf('--size') + 1]);
255
+ const outputArg = args[args.indexOf('-o') + 1];
256
+ const outputPath = outputArg.replace(/\[.*\]$/, '');
257
+ // Only the last attempt (halved dimensions, Q=30) fits
258
+ const size = compressionAttempts < 7 ? 2000 : 50;
259
+ Bun.write(outputPath, Buffer.alloc(size)).then(() => cb(null, '', ''));
260
+ }
261
+ };
262
+ const resizer = createResizer();
263
+ const result = await resizer.resize(testPath, 'image/jpeg', { maxFileSizeBytes: 100 });
264
+ // 4 full-dim attempts + 3 half-dim attempts
265
+ expect(compressionAttempts).toBe(7);
266
+ // First 4 at full dimension, last 3 at half
267
+ expect(dimensionArgs[0]).toBe('8000x8000');
268
+ expect(dimensionArgs[4]).toBe('4000x4000');
269
+ expect(result.mimeType).toBe('image/jpeg');
270
+ if (result.tempFile) {
271
+ await import('node:fs/promises').then(fs => fs.unlink(result.tempFile).catch(() => { }));
272
+ }
273
+ await import('node:fs/promises').then(fs => fs.unlink(testPath).catch(() => { }));
274
+ });
275
+ it('returns best effort when nothing fits', async () => {
276
+ const testPath = '/tmp/test-nothing-fits.jpg';
277
+ await Bun.write(testPath, Buffer.alloc(2000));
278
+ execFileImpl = (cmd, args, _opts, cb) => {
279
+ if (cmd === 'vipsheader') {
280
+ cb(null, '4000\n3000\n', '');
281
+ }
282
+ else if (cmd === 'vipsthumbnail') {
283
+ const outputArg = args[args.indexOf('-o') + 1];
284
+ const outputPath = outputArg.replace(/\[.*\]$/, '');
285
+ // Always too large
286
+ Bun.write(outputPath, Buffer.alloc(2000)).then(() => cb(null, '', ''));
287
+ }
288
+ };
289
+ const resizer = createResizer();
290
+ const result = await resizer.resize(testPath, 'image/jpeg', { maxFileSizeBytes: 100 });
291
+ // Returns the last attempt even though it doesn't fit
292
+ expect(result.mimeType).toBe('image/jpeg');
293
+ expect(result.tempFile).toBeDefined();
294
+ if (result.tempFile) {
295
+ await import('node:fs/promises').then(fs => fs.unlink(result.tempFile).catch(() => { }));
296
+ }
297
+ await import('node:fs/promises').then(fs => fs.unlink(testPath).catch(() => { }));
298
+ });
299
+ it('always outputs jpeg when compressing', async () => {
300
+ const testPath = '/tmp/test-png-compress.png';
301
+ await Bun.write(testPath, Buffer.alloc(2000));
302
+ execFileImpl = (cmd, args, _opts, cb) => {
303
+ if (cmd === 'vipsheader') {
304
+ cb(null, '4000\n3000\n', '');
305
+ }
306
+ else if (cmd === 'vipsthumbnail') {
307
+ const outputArg = args[args.indexOf('-o') + 1];
308
+ const outputPath = outputArg.replace(/\[.*\]$/, '');
309
+ Bun.write(outputPath, Buffer.alloc(50)).then(() => cb(null, '', ''));
310
+ }
311
+ };
312
+ const resizer = createResizer();
313
+ const result = await resizer.resize(testPath, 'image/png', { maxFileSizeBytes: 100 });
314
+ // Even though input was PNG, compression outputs JPEG
315
+ expect(result.mimeType).toBe('image/jpeg');
316
+ expect(result.path).toContain('.jpg');
317
+ if (result.tempFile) {
318
+ await import('node:fs/promises').then(fs => fs.unlink(result.tempFile).catch(() => { }));
319
+ }
320
+ await import('node:fs/promises').then(fs => fs.unlink(testPath).catch(() => { }));
321
+ });
322
+ });
323
+ });
324
+ //# sourceMappingURL=vips-resizer.test.js.map