@ebowwa/coder 0.2.1 → 0.7.64

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 (401) hide show
  1. package/README.md +31 -32
  2. package/dist/core/__tests__/permissions.test.d.ts +12 -0
  3. package/dist/core/__tests__/permissions.test.d.ts.map +1 -0
  4. package/dist/core/__tests__/permissions.test.js +851 -0
  5. package/dist/core/agent-loop/__tests__/compaction.test.d.ts +5 -0
  6. package/dist/core/agent-loop/__tests__/compaction.test.d.ts.map +1 -0
  7. package/dist/core/agent-loop/__tests__/compaction.test.js +209 -0
  8. package/dist/core/agent-loop/__tests__/formatters.test.d.ts +5 -0
  9. package/dist/core/agent-loop/__tests__/formatters.test.d.ts.map +1 -0
  10. package/dist/core/agent-loop/__tests__/formatters.test.js +195 -0
  11. package/dist/core/agent-loop/__tests__/index.test.d.ts +5 -0
  12. package/dist/core/agent-loop/__tests__/index.test.d.ts.map +1 -0
  13. package/dist/core/agent-loop/__tests__/index.test.js +121 -0
  14. package/dist/core/agent-loop/__tests__/loop-state.test.d.ts +5 -0
  15. package/dist/core/agent-loop/__tests__/loop-state.test.d.ts.map +1 -0
  16. package/dist/core/agent-loop/__tests__/loop-state.test.js +340 -0
  17. package/dist/core/agent-loop/__tests__/message-builder.test.d.ts +5 -0
  18. package/dist/core/agent-loop/__tests__/message-builder.test.d.ts.map +1 -0
  19. package/dist/core/agent-loop/__tests__/message-builder.test.js +178 -0
  20. package/dist/core/agent-loop/__tests__/tool-executor.test.d.ts +5 -0
  21. package/dist/core/agent-loop/__tests__/tool-executor.test.d.ts.map +1 -0
  22. package/dist/core/agent-loop/__tests__/tool-executor.test.js +331 -0
  23. package/dist/core/agent-loop/compaction.d.ts +39 -0
  24. package/dist/core/agent-loop/compaction.d.ts.map +1 -0
  25. package/dist/core/agent-loop/compaction.js +51 -0
  26. package/dist/core/agent-loop/formatters.d.ts +21 -0
  27. package/dist/core/agent-loop/formatters.d.ts.map +1 -0
  28. package/dist/core/agent-loop/formatters.js +42 -0
  29. package/dist/core/agent-loop/index.d.ts +25 -0
  30. package/dist/core/agent-loop/index.d.ts.map +1 -0
  31. package/dist/core/agent-loop/index.js +83 -0
  32. package/dist/core/agent-loop/loop-state.d.ts +74 -0
  33. package/dist/core/agent-loop/loop-state.d.ts.map +1 -0
  34. package/dist/core/agent-loop/loop-state.js +147 -0
  35. package/dist/core/agent-loop/message-builder.d.ts +13 -0
  36. package/dist/core/agent-loop/message-builder.d.ts.map +1 -0
  37. package/dist/core/agent-loop/message-builder.js +49 -0
  38. package/dist/core/agent-loop/tool-executor.d.ts +23 -0
  39. package/dist/core/agent-loop/tool-executor.d.ts.map +1 -0
  40. package/dist/core/agent-loop/tool-executor.js +152 -0
  41. package/dist/core/agent-loop/turn-executor.d.ts +57 -0
  42. package/dist/core/agent-loop/turn-executor.d.ts.map +1 -0
  43. package/dist/core/agent-loop/turn-executor.js +124 -0
  44. package/dist/core/agent-loop/types.d.ts +141 -0
  45. package/dist/core/agent-loop/types.d.ts.map +1 -0
  46. package/dist/core/agent-loop/types.js +4 -0
  47. package/dist/core/agent-loop.d.ts +17 -0
  48. package/dist/core/agent-loop.d.ts.map +1 -0
  49. package/dist/core/agent-loop.js +16 -0
  50. package/dist/core/api-client-impl.d.ts +62 -0
  51. package/dist/core/api-client-impl.d.ts.map +1 -0
  52. package/dist/core/api-client-impl.js +479 -0
  53. package/dist/core/api-client.d.ts +6 -0
  54. package/dist/core/api-client.d.ts.map +1 -0
  55. package/dist/core/api-client.js +5 -0
  56. package/dist/core/checkpoints.d.ts +128 -0
  57. package/dist/core/checkpoints.d.ts.map +1 -0
  58. package/dist/core/checkpoints.js +438 -0
  59. package/dist/core/claude-md.d.ts +71 -0
  60. package/dist/core/claude-md.d.ts.map +1 -0
  61. package/dist/core/claude-md.js +198 -0
  62. package/dist/core/cognitive-security/hooks.d.ts +138 -0
  63. package/dist/core/cognitive-security/hooks.d.ts.map +1 -0
  64. package/dist/core/cognitive-security/hooks.js +389 -0
  65. package/dist/core/cognitive-security/index.d.ts +751 -0
  66. package/dist/core/cognitive-security/index.d.ts.map +1 -0
  67. package/dist/core/cognitive-security/index.js +1123 -0
  68. package/dist/core/cognitive-security/middleware.d.ts +136 -0
  69. package/dist/core/cognitive-security/middleware.d.ts.map +1 -0
  70. package/dist/core/cognitive-security/middleware.js +376 -0
  71. package/dist/core/config-loader.d.ts +127 -0
  72. package/dist/core/config-loader.d.ts.map +1 -0
  73. package/dist/core/config-loader.js +219 -0
  74. package/dist/core/context-compaction.d.ts +87 -0
  75. package/dist/core/context-compaction.d.ts.map +1 -0
  76. package/dist/core/context-compaction.js +428 -0
  77. package/dist/core/git-status.d.ts +25 -0
  78. package/dist/core/git-status.d.ts.map +1 -0
  79. package/dist/core/git-status.js +204 -0
  80. package/dist/core/image.d.ts +69 -0
  81. package/dist/core/image.d.ts.map +1 -0
  82. package/dist/core/image.js +290 -0
  83. package/dist/core/image.test.d.ts +2 -0
  84. package/dist/core/image.test.d.ts.map +1 -0
  85. package/dist/core/image.test.js +149 -0
  86. package/dist/core/models.d.ts +123 -0
  87. package/dist/core/models.d.ts.map +1 -0
  88. package/dist/core/models.js +325 -0
  89. package/dist/core/permissions.d.ts +81 -0
  90. package/dist/core/permissions.d.ts.map +1 -0
  91. package/dist/core/permissions.js +327 -0
  92. package/dist/core/retry.d.ts +25 -0
  93. package/dist/core/retry.d.ts.map +1 -0
  94. package/dist/core/retry.js +121 -0
  95. package/dist/core/session-store.d.ts +9 -0
  96. package/dist/core/session-store.d.ts.map +1 -0
  97. package/dist/core/session-store.js +10 -0
  98. package/dist/core/sessions/export.d.ts +47 -0
  99. package/dist/core/sessions/export.d.ts.map +1 -0
  100. package/dist/core/sessions/export.js +256 -0
  101. package/dist/core/sessions/index.d.ts +132 -0
  102. package/dist/core/sessions/index.d.ts.map +1 -0
  103. package/dist/core/sessions/index.js +442 -0
  104. package/dist/core/sessions/metadata.d.ts +77 -0
  105. package/dist/core/sessions/metadata.d.ts.map +1 -0
  106. package/dist/core/sessions/metadata.js +233 -0
  107. package/dist/core/sessions/persistence.d.ts +72 -0
  108. package/dist/core/sessions/persistence.d.ts.map +1 -0
  109. package/dist/core/sessions/persistence.js +201 -0
  110. package/dist/core/sessions/types.d.ts +110 -0
  111. package/dist/core/sessions/types.d.ts.map +1 -0
  112. package/dist/core/sessions/types.js +4 -0
  113. package/dist/core/stream-highlighter.d.ts +18 -0
  114. package/dist/core/stream-highlighter.d.ts.map +1 -0
  115. package/dist/core/stream-highlighter.js +916 -0
  116. package/dist/core/system-reminders.d.ts +89 -0
  117. package/dist/core/system-reminders.d.ts.map +1 -0
  118. package/dist/core/system-reminders.js +285 -0
  119. package/dist/ecosystem/hooks/__tests__/index.test.d.ts +5 -0
  120. package/dist/ecosystem/hooks/__tests__/index.test.d.ts.map +1 -0
  121. package/dist/ecosystem/hooks/__tests__/index.test.js +458 -0
  122. package/dist/ecosystem/hooks/index.d.ts +59 -0
  123. package/dist/ecosystem/hooks/index.d.ts.map +1 -0
  124. package/dist/ecosystem/hooks/index.js +294 -0
  125. package/dist/ecosystem/hooks/prompt-evaluator.d.ts +32 -0
  126. package/dist/ecosystem/hooks/prompt-evaluator.d.ts.map +1 -0
  127. package/dist/ecosystem/hooks/prompt-evaluator.js +229 -0
  128. package/dist/ecosystem/skills/index.d.ts +55 -0
  129. package/dist/ecosystem/skills/index.d.ts.map +1 -0
  130. package/dist/ecosystem/skills/index.js +258 -0
  131. package/dist/ecosystem/tools/__tests__/index.test.d.ts +7 -0
  132. package/dist/ecosystem/tools/__tests__/index.test.d.ts.map +1 -0
  133. package/dist/ecosystem/tools/__tests__/index.test.js +856 -0
  134. package/dist/ecosystem/tools/index.d.ts +24 -0
  135. package/dist/ecosystem/tools/index.d.ts.map +1 -0
  136. package/dist/ecosystem/tools/index.js +1709 -0
  137. package/dist/index.d.ts +24 -0
  138. package/dist/index.d.ts.map +1 -0
  139. package/dist/index.js +32 -2
  140. package/dist/interfaces/mcp/client.d.ts +40 -0
  141. package/dist/interfaces/mcp/client.d.ts.map +1 -0
  142. package/dist/interfaces/mcp/client.js +309 -0
  143. package/dist/interfaces/ui/index.d.ts +36 -0
  144. package/dist/interfaces/ui/index.d.ts.map +1 -0
  145. package/dist/interfaces/ui/index.js +61 -0
  146. package/dist/interfaces/ui/spinner.d.ts +140 -0
  147. package/dist/interfaces/ui/spinner.d.ts.map +1 -0
  148. package/dist/interfaces/ui/spinner.js +342 -0
  149. package/dist/interfaces/ui/terminal/cli/index.d.ts +12 -0
  150. package/dist/interfaces/ui/terminal/cli/index.d.ts.map +1 -0
  151. package/dist/interfaces/ui/terminal/cli/index.js +167 -0
  152. package/dist/interfaces/ui/terminal/shared/args.d.ts +39 -0
  153. package/dist/interfaces/ui/terminal/shared/args.d.ts.map +1 -0
  154. package/dist/interfaces/ui/terminal/shared/args.js +176 -0
  155. package/dist/interfaces/ui/terminal/shared/index.d.ts +11 -0
  156. package/dist/interfaces/ui/terminal/shared/index.d.ts.map +1 -0
  157. package/dist/interfaces/ui/terminal/shared/index.js +16 -0
  158. package/dist/interfaces/ui/terminal/shared/loading-state.d.ts +124 -0
  159. package/dist/interfaces/ui/terminal/shared/loading-state.d.ts.map +1 -0
  160. package/dist/interfaces/ui/terminal/shared/loading-state.js +246 -0
  161. package/dist/interfaces/ui/terminal/shared/query.d.ts +22 -0
  162. package/dist/interfaces/ui/terminal/shared/query.d.ts.map +1 -0
  163. package/dist/interfaces/ui/terminal/shared/query.js +100 -0
  164. package/dist/interfaces/ui/terminal/shared/setup.d.ts +33 -0
  165. package/dist/interfaces/ui/terminal/shared/setup.d.ts.map +1 -0
  166. package/dist/interfaces/ui/terminal/shared/setup.js +226 -0
  167. package/dist/interfaces/ui/terminal/shared/status-line.d.ts +117 -0
  168. package/dist/interfaces/ui/terminal/shared/status-line.d.ts.map +1 -0
  169. package/dist/interfaces/ui/terminal/shared/status-line.js +267 -0
  170. package/dist/interfaces/ui/terminal/shared/system-prompt.d.ts +38 -0
  171. package/dist/interfaces/ui/terminal/shared/system-prompt.d.ts.map +1 -0
  172. package/dist/interfaces/ui/terminal/shared/system-prompt.js +102 -0
  173. package/dist/interfaces/ui/terminal/tui/HelpPanel.d.ts +39 -0
  174. package/dist/interfaces/ui/terminal/tui/HelpPanel.d.ts.map +1 -0
  175. package/dist/interfaces/ui/terminal/tui/HelpPanel.js +215 -0
  176. package/dist/interfaces/ui/terminal/tui/InputContext.d.ts +91 -0
  177. package/dist/interfaces/ui/terminal/tui/InputContext.d.ts.map +1 -0
  178. package/dist/interfaces/ui/terminal/tui/InputContext.js +154 -0
  179. package/dist/interfaces/ui/terminal/tui/InputField.d.ts +18 -0
  180. package/dist/interfaces/ui/terminal/tui/InputField.d.ts.map +1 -0
  181. package/dist/interfaces/ui/terminal/tui/InputField.js +41 -0
  182. package/dist/interfaces/ui/terminal/tui/InteractiveTUI.d.ts +16 -0
  183. package/dist/interfaces/ui/terminal/tui/InteractiveTUI.d.ts.map +1 -0
  184. package/dist/interfaces/ui/terminal/tui/InteractiveTUI.js +451 -0
  185. package/dist/interfaces/ui/terminal/tui/MessageArea.d.ts +10 -0
  186. package/dist/interfaces/ui/terminal/tui/MessageArea.d.ts.map +1 -0
  187. package/dist/interfaces/ui/terminal/tui/MessageArea.js +91 -0
  188. package/dist/interfaces/ui/terminal/tui/MessageStore.d.ts +48 -0
  189. package/dist/interfaces/ui/terminal/tui/MessageStore.d.ts.map +1 -0
  190. package/dist/interfaces/ui/terminal/tui/MessageStore.js +151 -0
  191. package/dist/interfaces/ui/terminal/tui/StatusBar.d.ts +9 -0
  192. package/dist/interfaces/ui/terminal/tui/StatusBar.d.ts.map +1 -0
  193. package/dist/interfaces/ui/terminal/tui/StatusBar.js +36 -0
  194. package/dist/interfaces/ui/terminal/tui/commands.d.ts +21 -0
  195. package/dist/interfaces/ui/terminal/tui/commands.d.ts.map +1 -0
  196. package/dist/interfaces/ui/terminal/tui/commands.js +359 -0
  197. package/dist/interfaces/ui/terminal/tui/components/InteractiveElements.d.ts +115 -0
  198. package/dist/interfaces/ui/terminal/tui/components/InteractiveElements.d.ts.map +1 -0
  199. package/dist/interfaces/ui/terminal/tui/components/InteractiveElements.js +306 -0
  200. package/dist/interfaces/ui/terminal/tui/components/MultilineInput.d.ts +92 -0
  201. package/dist/interfaces/ui/terminal/tui/components/MultilineInput.d.ts.map +1 -0
  202. package/dist/interfaces/ui/terminal/tui/components/MultilineInput.js +399 -0
  203. package/dist/interfaces/ui/terminal/tui/components/PaneManager.d.ts +59 -0
  204. package/dist/interfaces/ui/terminal/tui/components/PaneManager.d.ts.map +1 -0
  205. package/dist/interfaces/ui/terminal/tui/components/PaneManager.js +139 -0
  206. package/dist/interfaces/ui/terminal/tui/components/Sidebar.d.ts +68 -0
  207. package/dist/interfaces/ui/terminal/tui/components/Sidebar.d.ts.map +1 -0
  208. package/dist/interfaces/ui/terminal/tui/components/Sidebar.js +340 -0
  209. package/dist/interfaces/ui/terminal/tui/components/index.d.ts +23 -0
  210. package/dist/interfaces/ui/terminal/tui/components/index.d.ts.map +1 -0
  211. package/dist/interfaces/ui/terminal/tui/components/index.js +51 -0
  212. package/dist/interfaces/ui/terminal/tui/console.d.ts +20 -0
  213. package/dist/interfaces/ui/terminal/tui/console.d.ts.map +1 -0
  214. package/dist/interfaces/ui/terminal/tui/console.js +46 -0
  215. package/dist/interfaces/ui/terminal/tui/index.d.ts +20 -0
  216. package/dist/interfaces/ui/terminal/tui/index.d.ts.map +1 -0
  217. package/dist/interfaces/ui/terminal/tui/index.js +28 -0
  218. package/dist/interfaces/ui/terminal/tui/run.d.ts +13 -0
  219. package/dist/interfaces/ui/terminal/tui/run.d.ts.map +1 -0
  220. package/dist/interfaces/ui/terminal/tui/run.js +31 -0
  221. package/dist/interfaces/ui/terminal/tui/spinner.d.ts +44 -0
  222. package/dist/interfaces/ui/terminal/tui/spinner.d.ts.map +1 -0
  223. package/dist/interfaces/ui/terminal/tui/spinner.js +59 -0
  224. package/dist/interfaces/ui/terminal/tui/tui-app.d.ts +39 -0
  225. package/dist/interfaces/ui/terminal/tui/tui-app.d.ts.map +1 -0
  226. package/dist/interfaces/ui/terminal/tui/tui-app.js +198 -0
  227. package/dist/interfaces/ui/terminal/tui/tui-footer.d.ts +167 -0
  228. package/dist/interfaces/ui/terminal/tui/tui-footer.d.ts.map +1 -0
  229. package/dist/interfaces/ui/terminal/tui/tui-footer.js +330 -0
  230. package/dist/interfaces/ui/terminal/tui/types.d.ts +165 -0
  231. package/dist/interfaces/ui/terminal/tui/types.d.ts.map +1 -0
  232. package/dist/interfaces/ui/terminal/tui/types.js +5 -0
  233. package/dist/interfaces/ui/terminal/tui/useInputHandler.d.ts +23 -0
  234. package/dist/interfaces/ui/terminal/tui/useInputHandler.d.ts.map +1 -0
  235. package/dist/interfaces/ui/terminal/tui/useInputHandler.js +72 -0
  236. package/dist/interfaces/ui/terminal/tui/useNativeInput.d.ts +90 -0
  237. package/dist/interfaces/ui/terminal/tui/useNativeInput.d.ts.map +1 -0
  238. package/dist/interfaces/ui/terminal/tui/useNativeInput.js +188 -0
  239. package/dist/native/index.d.ts +480 -0
  240. package/dist/native/index.d.ts.map +1 -0
  241. package/dist/native/index.js +1625 -0
  242. package/dist/teammates/index.d.ts +161 -0
  243. package/dist/teammates/index.d.ts.map +1 -0
  244. package/dist/teammates/index.js +827 -0
  245. package/dist/types/index.d.ts +482 -0
  246. package/dist/types/index.d.ts.map +1 -0
  247. package/dist/types/index.js +52 -0
  248. package/native/README.md +5 -5
  249. package/native/index.darwin-arm64.node +0 -0
  250. package/native/index.node +0 -0
  251. package/native/package.json +4 -4
  252. package/package.json +33 -16
  253. package/packages/src/core/__tests__/permissions.test.ts +1091 -0
  254. package/packages/src/core/agent-loop/__tests__/compaction.test.ts +280 -0
  255. package/packages/src/core/agent-loop/__tests__/formatters.test.ts +234 -0
  256. package/packages/src/core/agent-loop/__tests__/index.test.ts +162 -0
  257. package/packages/src/core/agent-loop/__tests__/loop-state.test.ts +413 -0
  258. package/packages/src/core/agent-loop/__tests__/message-builder.test.ts +229 -0
  259. package/packages/src/core/agent-loop/__tests__/tool-executor.test.ts +457 -0
  260. package/packages/src/core/agent-loop/compaction.ts +88 -0
  261. package/packages/src/core/agent-loop/formatters.ts +50 -0
  262. package/packages/src/core/agent-loop/index.ts +135 -0
  263. package/packages/src/core/agent-loop/loop-state.ts +187 -0
  264. package/packages/src/core/agent-loop/message-builder.ts +62 -0
  265. package/packages/src/core/agent-loop/tool-executor.ts +211 -0
  266. package/packages/src/core/agent-loop/turn-executor.ts +222 -0
  267. package/packages/src/core/agent-loop/types.ts +148 -0
  268. package/packages/src/core/agent-loop.ts +18 -0
  269. package/packages/src/core/api-client-impl.ts +619 -0
  270. package/packages/src/core/api-client.ts +6 -0
  271. package/packages/src/core/checkpoints.ts +606 -0
  272. package/packages/src/core/claude-md.ts +272 -0
  273. package/packages/src/core/cognitive-security/hooks.ts +590 -0
  274. package/packages/src/core/cognitive-security/index.ts +2041 -0
  275. package/packages/src/core/cognitive-security/middleware.ts +536 -0
  276. package/packages/src/core/config-loader.ts +324 -0
  277. package/packages/src/core/context-compaction.ts +578 -0
  278. package/packages/src/core/git-status.ts +262 -0
  279. package/packages/src/core/image.test.ts +180 -0
  280. package/packages/src/core/image.ts +350 -0
  281. package/packages/src/core/lmdb.db +0 -0
  282. package/packages/src/core/lmdb.db-lock +0 -0
  283. package/packages/src/core/models.ts +430 -0
  284. package/packages/src/core/normalizers/todo +4 -0
  285. package/packages/src/core/permissions.ts +431 -0
  286. package/packages/src/core/retry.ts +170 -0
  287. package/packages/src/core/session-store.ts +36 -0
  288. package/packages/src/core/sessions/export.ts +329 -0
  289. package/packages/src/core/sessions/index.ts +587 -0
  290. package/packages/src/core/sessions/metadata.ts +309 -0
  291. package/packages/src/core/sessions/persistence.ts +244 -0
  292. package/packages/src/core/sessions/types.ts +169 -0
  293. package/packages/src/core/stream-highlighter.ts +1123 -0
  294. package/packages/src/core/system-reminders.ts +402 -0
  295. package/packages/src/core/todo +8 -0
  296. package/packages/src/ecosystem/hooks/__tests__/index.test.ts +561 -0
  297. package/packages/src/ecosystem/hooks/index.ts +341 -0
  298. package/packages/src/ecosystem/hooks/prompt-evaluator.ts +300 -0
  299. package/packages/src/ecosystem/skills/index.ts +295 -0
  300. package/packages/src/ecosystem/tools/__tests__/index.test.ts +1335 -0
  301. package/packages/src/ecosystem/tools/index.ts +1877 -0
  302. package/packages/src/index.ts +120 -0
  303. package/packages/src/interfaces/mcp/client.ts +389 -0
  304. package/packages/src/interfaces/ui/Screenshot 2026-03-02 at 9.23.10/342/200/257PM.png +0 -0
  305. package/packages/src/interfaces/ui/Screenshot 2026-03-03 at 10.55.11/342/200/257AM.png +0 -0
  306. package/packages/src/interfaces/ui/index.ts +161 -0
  307. package/packages/src/interfaces/ui/lmdb.db +0 -0
  308. package/packages/src/interfaces/ui/lmdb.db-lock +0 -0
  309. package/packages/src/interfaces/ui/spinner.ts +451 -0
  310. package/packages/src/interfaces/ui/terminal/cli/index.ts +228 -0
  311. package/packages/src/interfaces/ui/terminal/lmdb.db +0 -0
  312. package/packages/src/interfaces/ui/terminal/lmdb.db-lock +0 -0
  313. package/packages/src/interfaces/ui/terminal/shared/args.ts +222 -0
  314. package/packages/src/interfaces/ui/terminal/shared/index.ts +71 -0
  315. package/packages/src/interfaces/ui/terminal/shared/loading-state.ts +322 -0
  316. package/packages/src/interfaces/ui/terminal/shared/query.ts +146 -0
  317. package/packages/src/interfaces/ui/terminal/shared/setup.ts +295 -0
  318. package/packages/src/interfaces/ui/terminal/shared/status-line.ts +358 -0
  319. package/packages/src/interfaces/ui/terminal/shared/system-prompt.ts +146 -0
  320. package/packages/src/interfaces/ui/terminal/tui/HelpPanel.tsx +262 -0
  321. package/packages/src/interfaces/ui/terminal/tui/InputContext.tsx +232 -0
  322. package/packages/src/interfaces/ui/terminal/tui/InputField.tsx +62 -0
  323. package/packages/src/interfaces/ui/terminal/tui/InteractiveTUI.tsx +537 -0
  324. package/packages/src/interfaces/ui/terminal/tui/MessageArea.tsx +107 -0
  325. package/packages/src/interfaces/ui/terminal/tui/MessageStore.tsx +240 -0
  326. package/packages/src/interfaces/ui/terminal/tui/StatusBar.tsx +54 -0
  327. package/packages/src/interfaces/ui/terminal/tui/commands.ts +438 -0
  328. package/packages/src/interfaces/ui/terminal/tui/components/InteractiveElements.tsx +584 -0
  329. package/packages/src/interfaces/ui/terminal/tui/components/MultilineInput.tsx +614 -0
  330. package/packages/src/interfaces/ui/terminal/tui/components/PaneManager.tsx +333 -0
  331. package/packages/src/interfaces/ui/terminal/tui/components/Sidebar.tsx +604 -0
  332. package/packages/src/interfaces/ui/terminal/tui/components/index.ts +118 -0
  333. package/packages/src/interfaces/ui/terminal/tui/console.ts +49 -0
  334. package/packages/src/interfaces/ui/terminal/tui/index.ts +90 -0
  335. package/packages/src/interfaces/ui/terminal/tui/run.tsx +42 -0
  336. package/packages/src/interfaces/ui/terminal/tui/spinner.ts +69 -0
  337. package/packages/src/interfaces/ui/terminal/tui/tui-app.tsx +390 -0
  338. package/packages/src/interfaces/ui/terminal/tui/tui-footer.ts +422 -0
  339. package/packages/src/interfaces/ui/terminal/tui/types.ts +186 -0
  340. package/packages/src/interfaces/ui/terminal/tui/useInputHandler.ts +104 -0
  341. package/packages/src/interfaces/ui/terminal/tui/useNativeInput.ts +239 -0
  342. package/packages/src/lmdb.db +0 -0
  343. package/packages/src/lmdb.db-lock +0 -0
  344. package/packages/src/native/index.ts +2345 -0
  345. package/packages/src/teammates/index.ts +982 -0
  346. package/packages/src/types/index.ts +722 -0
  347. package/dist/cli.js +0 -148
  348. package/dist/index-0pkak453.js +0 -136
  349. package/dist/index-0qd0x8b4.js +0 -110
  350. package/dist/index-0x3kprq6.js +0 -240
  351. package/dist/index-1eawy937.js +0 -308
  352. package/dist/index-24m2aygy.js +0 -240
  353. package/dist/index-29xcjnne.js +0 -280
  354. package/dist/index-2avyytn5.js +0 -349
  355. package/dist/index-4ms367ey.js +0 -136
  356. package/dist/index-4w2t3b0m.js +0 -240
  357. package/dist/index-4xfgd8nz.js +0 -261
  358. package/dist/index-5acjp9gc.js +0 -157
  359. package/dist/index-5s15hr56.js +0 -136
  360. package/dist/index-6e4wf341.js +0 -349
  361. package/dist/index-6fvnkedw.js +0 -240
  362. package/dist/index-6rqpmd4g.js +0 -128
  363. package/dist/index-77ckwnbm.js +0 -280
  364. package/dist/index-9knxy49k.js +0 -128
  365. package/dist/index-9zrnw4zx.js +0 -128
  366. package/dist/index-bk21w99v.js +0 -280
  367. package/dist/index-c41n76fv.js +0 -240
  368. package/dist/index-cb4ppjdt.js +0 -255
  369. package/dist/index-cfb2edt6.js +0 -240
  370. package/dist/index-cmfa38hh.js +0 -308
  371. package/dist/index-datjz8q1.js +0 -257
  372. package/dist/index-eadf4wvn.js +0 -240
  373. package/dist/index-em5k0m3z.js +0 -345
  374. package/dist/index-gh8r333a.js +0 -110
  375. package/dist/index-gkx6k2tr.js +0 -261
  376. package/dist/index-h5cabfks.js +0 -155
  377. package/dist/index-hcrpwyy3.js +0 -261
  378. package/dist/index-hk7fwwa8.js +0 -257
  379. package/dist/index-jb8cw7f8.js +0 -136
  380. package/dist/index-kbyw4th1.js +0 -347
  381. package/dist/index-kgj5gqnm.js +0 -345
  382. package/dist/index-mdf6xp1z.js +0 -255
  383. package/dist/index-mrhv8kvc.js +0 -280
  384. package/dist/index-mt4743dd.js +0 -161
  385. package/dist/index-qnwsg97q.js +0 -240
  386. package/dist/index-qwdy6x44.js +0 -261
  387. package/dist/index-rmj77261.js +0 -157
  388. package/dist/index-sbbw1a61.js +0 -349
  389. package/dist/index-svy5bcpn.js +0 -345
  390. package/dist/index-tvmy7tm9.js +0 -261
  391. package/dist/index-tzz4vzkj.js +0 -312
  392. package/dist/index-vz80zmhe.js +0 -110
  393. package/dist/index-wed2fk67.js +0 -240
  394. package/dist/index-wksgzz8e.js +0 -280
  395. package/dist/index-wn2m4wma.js +0 -240
  396. package/dist/index-xha05vjc.js +0 -257
  397. package/dist/index-yc6eh8p8.js +0 -136
  398. package/dist/index-ycjxx9ft.js +0 -240
  399. package/dist/index-z0gzd0fc.js +0 -110
  400. package/dist/index-z8cwtf8j.js +0 -240
  401. package/dist/index-zy5mtt00.js +0 -128
@@ -0,0 +1,413 @@
1
+ /**
2
+ * LoopState Tests - State management for the agent loop
3
+ */
4
+
5
+ import { describe, it, expect, beforeEach } from "bun:test";
6
+ import { LoopState, createInitialCacheMetrics } from "../loop-state.js";
7
+ import type { Message, ToolUseBlock, CacheMetrics } from "../../../types/index.js";
8
+
9
+ describe("createInitialCacheMetrics", () => {
10
+ it("should create initial cache metrics with zeros", () => {
11
+ const metrics = createInitialCacheMetrics();
12
+
13
+ expect(metrics.cacheHits).toBe(0);
14
+ expect(metrics.cacheMisses).toBe(0);
15
+ expect(metrics.totalCacheReadTokens).toBe(0);
16
+ expect(metrics.totalCacheWriteTokens).toBe(0);
17
+ expect(metrics.cacheHitRate).toBe(0);
18
+ expect(metrics.estimatedSavingsUSD).toBe(0);
19
+ });
20
+ });
21
+
22
+ describe("LoopState", () => {
23
+ let state: LoopState;
24
+ let initialMessages: Message[];
25
+
26
+ beforeEach(() => {
27
+ initialMessages = [
28
+ { role: "user", content: [{ type: "text", text: "Hello" }] },
29
+ ];
30
+ state = new LoopState(initialMessages);
31
+ });
32
+
33
+ describe("constructor", () => {
34
+ it("should initialize with provided messages", () => {
35
+ expect(state.messages).toHaveLength(1);
36
+ expect(state.messages[0]?.role).toBe("user");
37
+ });
38
+
39
+ it("should create a copy of initial messages", () => {
40
+ // Modify original array
41
+ initialMessages.push({ role: "user", content: [{ type: "text", text: "New" }] });
42
+
43
+ // State should not be affected
44
+ expect(state.messages).toHaveLength(1);
45
+ });
46
+
47
+ it("should initialize all counters to zero", () => {
48
+ expect(state.metrics).toEqual([]);
49
+ expect(state.allToolsUsed).toEqual([]);
50
+ expect(state.totalCost).toBe(0);
51
+ expect(state.totalDuration).toBe(0);
52
+ expect(state.turnNumber).toBe(0);
53
+ expect(state.previousCost).toBe(0);
54
+ expect(state.compactionCount).toBe(0);
55
+ expect(state.totalTokensCompacted).toBe(0);
56
+ });
57
+
58
+ it("should initialize cache metrics", () => {
59
+ expect(state.cacheMetrics.cacheHits).toBe(0);
60
+ expect(state.cacheMetrics.cacheMisses).toBe(0);
61
+ });
62
+ });
63
+
64
+ describe("latestMetrics", () => {
65
+ it("should return undefined when no metrics exist", () => {
66
+ expect(state.latestMetrics).toBeUndefined();
67
+ });
68
+
69
+ it("should return the last metrics entry", () => {
70
+ state.addTurnResult({
71
+ message: { content: [], stop_reason: "end_turn", id: "msg1" },
72
+ usage: { input_tokens: 100, output_tokens: 50 },
73
+ costUSD: 0.01,
74
+ durationMs: 1000,
75
+ model: "claude-sonnet-4-6",
76
+ messageCount: 1,
77
+ });
78
+
79
+ state.addTurnResult({
80
+ message: { content: [], stop_reason: "end_turn", id: "msg2" },
81
+ usage: { input_tokens: 200, output_tokens: 100 },
82
+ costUSD: 0.02,
83
+ durationMs: 2000,
84
+ model: "claude-sonnet-4-6",
85
+ messageCount: 2,
86
+ });
87
+
88
+ expect(state.latestMetrics?.usage.input_tokens).toBe(200);
89
+ expect(state.latestMetrics?.requestId).toBe("msg2");
90
+ });
91
+ });
92
+
93
+ describe("currentUsage", () => {
94
+ it("should return zero usage when no metrics exist", () => {
95
+ expect(state.currentUsage).toEqual({
96
+ input_tokens: 0,
97
+ output_tokens: 0,
98
+ });
99
+ });
100
+
101
+ it("should return usage from latest metrics", () => {
102
+ state.addTurnResult({
103
+ message: { content: [], stop_reason: "end_turn", id: "msg1" },
104
+ usage: { input_tokens: 500, output_tokens: 250 },
105
+ costUSD: 0.05,
106
+ durationMs: 1500,
107
+ model: "claude-sonnet-4-6",
108
+ messageCount: 1,
109
+ });
110
+
111
+ expect(state.currentUsage.input_tokens).toBe(500);
112
+ expect(state.currentUsage.output_tokens).toBe(250);
113
+ });
114
+ });
115
+
116
+ describe("addTurnResult", () => {
117
+ it("should add metrics entry with correct values", () => {
118
+ const result = state.addTurnResult({
119
+ message: { content: [], stop_reason: "end_turn", id: "msg1" },
120
+ usage: { input_tokens: 100, output_tokens: 50 },
121
+ costUSD: 0.01,
122
+ durationMs: 1000,
123
+ model: "claude-sonnet-4-6",
124
+ messageCount: 1,
125
+ });
126
+
127
+ expect(result.model).toBe("claude-sonnet-4-6");
128
+ expect(result.messageCount).toBe(1);
129
+ expect(result.messageTokens).toBe(150);
130
+ expect(result.costUSD).toBe(0.01);
131
+ expect(result.stopReason).toBe("end_turn");
132
+ expect(result.requestId).toBe("msg1");
133
+ });
134
+
135
+ it("should accumulate total cost", () => {
136
+ state.addTurnResult({
137
+ message: { content: [], stop_reason: "end_turn", id: "msg1" },
138
+ usage: { input_tokens: 100, output_tokens: 50 },
139
+ costUSD: 0.01,
140
+ durationMs: 1000,
141
+ model: "claude-sonnet-4-6",
142
+ messageCount: 1,
143
+ });
144
+
145
+ state.addTurnResult({
146
+ message: { content: [], stop_reason: "end_turn", id: "msg2" },
147
+ usage: { input_tokens: 100, output_tokens: 50 },
148
+ costUSD: 0.02,
149
+ durationMs: 1000,
150
+ model: "claude-sonnet-4-6",
151
+ messageCount: 2,
152
+ });
153
+
154
+ expect(state.totalCost).toBe(0.03);
155
+ expect(state.previousCost).toBe(0.01);
156
+ });
157
+
158
+ it("should accumulate duration", () => {
159
+ state.addTurnResult({
160
+ message: { content: [], stop_reason: "end_turn", id: "msg1" },
161
+ usage: { input_tokens: 100, output_tokens: 50 },
162
+ costUSD: 0.01,
163
+ durationMs: 1000,
164
+ model: "claude-sonnet-4-6",
165
+ messageCount: 1,
166
+ });
167
+
168
+ state.addTurnResult({
169
+ message: { content: [], stop_reason: "end_turn", id: "msg2" },
170
+ usage: { input_tokens: 100, output_tokens: 50 },
171
+ costUSD: 0.01,
172
+ durationMs: 500,
173
+ model: "claude-sonnet-4-6",
174
+ messageCount: 2,
175
+ });
176
+
177
+ expect(state.totalDuration).toBe(1500);
178
+ });
179
+
180
+ it("should aggregate cache metrics", () => {
181
+ state.addTurnResult({
182
+ message: { content: [], stop_reason: "end_turn", id: "msg1" },
183
+ usage: { input_tokens: 100, output_tokens: 50 },
184
+ cacheMetrics: {
185
+ cacheHits: 5,
186
+ cacheMisses: 2,
187
+ totalCacheReadTokens: 1000,
188
+ totalCacheWriteTokens: 500,
189
+ cacheHitRate: 0.714,
190
+ estimatedSavingsUSD: 0.05,
191
+ },
192
+ costUSD: 0.01,
193
+ durationMs: 1000,
194
+ model: "claude-sonnet-4-6",
195
+ messageCount: 1,
196
+ });
197
+
198
+ state.addTurnResult({
199
+ message: { content: [], stop_reason: "end_turn", id: "msg2" },
200
+ usage: { input_tokens: 100, output_tokens: 50 },
201
+ cacheMetrics: {
202
+ cacheHits: 3,
203
+ cacheMisses: 1,
204
+ totalCacheReadTokens: 600,
205
+ totalCacheWriteTokens: 200,
206
+ cacheHitRate: 0.75,
207
+ estimatedSavingsUSD: 0.03,
208
+ },
209
+ costUSD: 0.01,
210
+ durationMs: 1000,
211
+ model: "claude-sonnet-4-6",
212
+ messageCount: 2,
213
+ });
214
+
215
+ expect(state.cacheMetrics.cacheHits).toBe(8);
216
+ expect(state.cacheMetrics.cacheMisses).toBe(3);
217
+ expect(state.cacheMetrics.totalCacheReadTokens).toBe(1600);
218
+ expect(state.cacheMetrics.totalCacheWriteTokens).toBe(700);
219
+ expect(state.cacheMetrics.estimatedSavingsUSD).toBe(0.08);
220
+ // Hit rate should be recalculated: 8/(8+3) = 0.727...
221
+ expect(state.cacheMetrics.cacheHitRate).toBeCloseTo(0.727, 2);
222
+ });
223
+
224
+ it("should handle missing cache metrics", () => {
225
+ state.addTurnResult({
226
+ message: { content: [], stop_reason: "end_turn", id: "msg1" },
227
+ usage: { input_tokens: 100, output_tokens: 50 },
228
+ costUSD: 0.01,
229
+ durationMs: 1000,
230
+ model: "claude-sonnet-4-6",
231
+ messageCount: 1,
232
+ });
233
+
234
+ // Cache metrics should remain at initial values
235
+ expect(state.cacheMetrics.cacheHits).toBe(0);
236
+ expect(state.cacheMetrics.cacheMisses).toBe(0);
237
+ });
238
+ });
239
+
240
+ describe("addAssistantMessage", () => {
241
+ it("should add assistant message to history", () => {
242
+ state.addAssistantMessage([{ type: "text", text: "Hello!" }]);
243
+
244
+ expect(state.messages).toHaveLength(2);
245
+ expect(state.messages[1]?.role).toBe("assistant");
246
+ });
247
+
248
+ it("should preserve message order", () => {
249
+ state.addAssistantMessage([{ type: "text", text: "First" }]);
250
+ state.addAssistantMessage([{ type: "text", text: "Second" }]);
251
+
252
+ expect(state.messages).toHaveLength(3);
253
+ expect(state.messages[1]?.content[0]).toEqual({ type: "text", text: "First" });
254
+ expect(state.messages[2]?.content[0]).toEqual({ type: "text", text: "Second" });
255
+ });
256
+ });
257
+
258
+ describe("addUserMessage", () => {
259
+ it("should add user message with tool results", () => {
260
+ state.addUserMessage([{
261
+ type: "tool_result",
262
+ tool_use_id: "tool1",
263
+ content: "result",
264
+ }]);
265
+
266
+ expect(state.messages).toHaveLength(2);
267
+ expect(state.messages[1]?.role).toBe("user");
268
+ });
269
+ });
270
+
271
+ describe("trackToolUse", () => {
272
+ it("should track tool use blocks", () => {
273
+ const toolBlocks: ToolUseBlock[] = [
274
+ { type: "tool_use", id: "tool1", name: "Read", input: { file_path: "/test" } },
275
+ { type: "tool_use", id: "tool2", name: "Write", input: { file_path: "/test2" } },
276
+ ];
277
+
278
+ state.trackToolUse(toolBlocks);
279
+
280
+ expect(state.allToolsUsed).toHaveLength(2);
281
+ expect(state.allToolsUsed[0]?.name).toBe("Read");
282
+ expect(state.allToolsUsed[1]?.name).toBe("Write");
283
+ });
284
+
285
+ it("should accumulate tool uses across multiple calls", () => {
286
+ state.trackToolUse([{ type: "tool_use", id: "tool1", name: "Read", input: {} }]);
287
+ state.trackToolUse([{ type: "tool_use", id: "tool2", name: "Write", input: {} }]);
288
+
289
+ expect(state.allToolsUsed).toHaveLength(2);
290
+ });
291
+ });
292
+
293
+ describe("applyCompaction", () => {
294
+ it("should not apply compaction when didCompact is false", () => {
295
+ const result = state.applyCompaction(
296
+ {
297
+ messages: state.messages,
298
+ messagesRemoved: 0,
299
+ tokensBefore: 100,
300
+ tokensAfter: 100,
301
+ didCompact: false,
302
+ },
303
+ () => ({ reductionPercent: 0, tokensSaved: 0 })
304
+ );
305
+
306
+ expect(result).toBe(false);
307
+ expect(state.compactionCount).toBe(0);
308
+ });
309
+
310
+ it("should not apply compaction when tokensAfter >= tokensBefore", () => {
311
+ const result = state.applyCompaction(
312
+ {
313
+ messages: state.messages,
314
+ messagesRemoved: 0,
315
+ tokensBefore: 100,
316
+ tokensAfter: 150, // Worse than before
317
+ didCompact: true,
318
+ },
319
+ () => ({ reductionPercent: 0, tokensSaved: 0 })
320
+ );
321
+
322
+ expect(result).toBe(false);
323
+ expect(state.compactionCount).toBe(0);
324
+ });
325
+
326
+ it("should apply compaction and update state", () => {
327
+ const newMessages: Message[] = [
328
+ { role: "user", content: [{ type: "text", text: "Compacted" }] },
329
+ ];
330
+
331
+ const result = state.applyCompaction(
332
+ {
333
+ messages: newMessages,
334
+ messagesRemoved: 1,
335
+ tokensBefore: 1000,
336
+ tokensAfter: 500,
337
+ didCompact: true,
338
+ },
339
+ () => ({ reductionPercent: 50, tokensSaved: 500 })
340
+ );
341
+
342
+ expect(result).toBe(true);
343
+ expect(state.compactionCount).toBe(1);
344
+ expect(state.totalTokensCompacted).toBe(500);
345
+ expect(state.messages).toHaveLength(1);
346
+ expect(state.messages[0]?.content[0]).toEqual({ type: "text", text: "Compacted" });
347
+ });
348
+
349
+ it("should accumulate compaction counts", () => {
350
+ state.applyCompaction(
351
+ {
352
+ messages: state.messages,
353
+ messagesRemoved: 1,
354
+ tokensBefore: 1000,
355
+ tokensAfter: 500,
356
+ didCompact: true,
357
+ },
358
+ () => ({ reductionPercent: 50, tokensSaved: 500 })
359
+ );
360
+
361
+ state.applyCompaction(
362
+ {
363
+ messages: state.messages,
364
+ messagesRemoved: 1,
365
+ tokensBefore: 500,
366
+ tokensAfter: 250,
367
+ didCompact: true,
368
+ },
369
+ () => ({ reductionPercent: 50, tokensSaved: 250 })
370
+ );
371
+
372
+ expect(state.compactionCount).toBe(2);
373
+ expect(state.totalTokensCompacted).toBe(750);
374
+ });
375
+ });
376
+
377
+ describe("incrementTurn", () => {
378
+ it("should increment turn counter and return new value", () => {
379
+ expect(state.turnNumber).toBe(0);
380
+
381
+ const turn1 = state.incrementTurn();
382
+ expect(turn1).toBe(1);
383
+ expect(state.turnNumber).toBe(1);
384
+
385
+ const turn2 = state.incrementTurn();
386
+ expect(turn2).toBe(2);
387
+ expect(state.turnNumber).toBe(2);
388
+ });
389
+ });
390
+
391
+ describe("toResult", () => {
392
+ it("should convert state to AgentLoopResult", () => {
393
+ state.addTurnResult({
394
+ message: { content: [], stop_reason: "end_turn", id: "msg1" },
395
+ usage: { input_tokens: 100, output_tokens: 50 },
396
+ costUSD: 0.01,
397
+ durationMs: 1000,
398
+ model: "claude-sonnet-4-6",
399
+ messageCount: 1,
400
+ });
401
+
402
+ const result = state.toResult();
403
+
404
+ expect(result.messages).toHaveLength(1);
405
+ expect(result.metrics).toHaveLength(1);
406
+ expect(result.totalCost).toBe(0.01);
407
+ expect(result.totalDuration).toBe(1000);
408
+ expect(result.compactionCount).toBe(0);
409
+ expect(result.totalTokensCompacted).toBe(0);
410
+ expect(result.totalCacheMetrics).toBeDefined();
411
+ });
412
+ });
413
+ });
@@ -0,0 +1,229 @@
1
+ /**
2
+ * Message Builder Tests - API message construction with reminder injection
3
+ */
4
+
5
+ import { describe, it, expect } from "bun:test";
6
+ import { buildAPIMessages, injectReminderIntoContent } from "../message-builder.js";
7
+ import type { Message, ContentBlock } from "../../../types/index.js";
8
+
9
+ describe("injectReminderIntoContent", () => {
10
+ it("should append reminder to existing text block", () => {
11
+ const content: ContentBlock[] = [
12
+ { type: "text", text: "Original text" },
13
+ ];
14
+
15
+ const result = injectReminderIntoContent(content, "System reminder");
16
+
17
+ expect(result).toHaveLength(1);
18
+ expect(result[0]?.type).toBe("text");
19
+ expect((result[0] as any).text).toBe("Original text\n\nSystem reminder");
20
+ });
21
+
22
+ it("should not mutate original content array", () => {
23
+ const content: ContentBlock[] = [
24
+ { type: "text", text: "Original text" },
25
+ ];
26
+
27
+ injectReminderIntoContent(content, "System reminder");
28
+
29
+ expect((content[0] as any).text).toBe("Original text");
30
+ });
31
+
32
+ it("should add new text block if last block is not text", () => {
33
+ const content: ContentBlock[] = [
34
+ { type: "image", source: { type: "base64", data: "abc", media_type: "image/png" } },
35
+ ];
36
+
37
+ const result = injectReminderIntoContent(content, "System reminder");
38
+
39
+ expect(result).toHaveLength(2);
40
+ expect(result[0]?.type).toBe("image");
41
+ expect(result[1]?.type).toBe("text");
42
+ expect((result[1] as any).text).toBe("\n\nSystem reminder");
43
+ });
44
+
45
+ it("should add new text block if content is empty", () => {
46
+ const content: ContentBlock[] = [];
47
+
48
+ const result = injectReminderIntoContent(content, "System reminder");
49
+
50
+ expect(result).toHaveLength(1);
51
+ expect(result[0]?.type).toBe("text");
52
+ expect((result[0] as any).text).toBe("\n\nSystem reminder");
53
+ });
54
+
55
+ it("should handle multiple content blocks", () => {
56
+ const content: ContentBlock[] = [
57
+ { type: "tool_use", id: "tool1", name: "Read", input: {} },
58
+ { type: "text", text: "After tool" },
59
+ ];
60
+
61
+ const result = injectReminderIntoContent(content, "Reminder");
62
+
63
+ expect(result).toHaveLength(2);
64
+ expect((result[1] as any).text).toBe("After tool\n\nReminder");
65
+ });
66
+
67
+ it("should handle tool_result as last block", () => {
68
+ const content: ContentBlock[] = [
69
+ { type: "tool_result", tool_use_id: "tool1", content: "result" },
70
+ ];
71
+
72
+ const result = injectReminderIntoContent(content, "Reminder");
73
+
74
+ expect(result).toHaveLength(2);
75
+ expect(result[0]?.type).toBe("tool_result");
76
+ expect(result[1]?.type).toBe("text");
77
+ });
78
+ });
79
+
80
+ describe("buildAPIMessages", () => {
81
+ it("should return messages unchanged when no reminder", () => {
82
+ const messages: Message[] = [
83
+ { role: "user", content: [{ type: "text", text: "Hello" }] },
84
+ ];
85
+
86
+ const result = buildAPIMessages(messages, "System prompt");
87
+
88
+ expect(result).toEqual(messages);
89
+ });
90
+
91
+ it("should return messages unchanged when empty", () => {
92
+ const result = buildAPIMessages([], "System prompt", "Reminder");
93
+
94
+ expect(result).toEqual([]);
95
+ });
96
+
97
+ it("should inject reminder into last user message", () => {
98
+ const messages: Message[] = [
99
+ { role: "user", content: [{ type: "text", text: "Hello" }] },
100
+ ];
101
+
102
+ const result = buildAPIMessages(messages, "System prompt", "Important reminder");
103
+
104
+ expect(result).toHaveLength(1);
105
+ expect(result[0]?.role).toBe("user");
106
+ const textBlock = result[0]?.content[0] as any;
107
+ expect(textBlock.text).toContain("Hello");
108
+ expect(textBlock.text).toContain("Important reminder");
109
+ });
110
+
111
+ it("should not mutate original messages", () => {
112
+ const messages: Message[] = [
113
+ { role: "user", content: [{ type: "text", text: "Hello" }] },
114
+ ];
115
+
116
+ buildAPIMessages(messages, "System prompt", "Reminder");
117
+
118
+ expect((messages[0]?.content[0] as any).text).toBe("Hello");
119
+ });
120
+
121
+ it("should find last user message in mixed conversation", () => {
122
+ const messages: Message[] = [
123
+ { role: "user", content: [{ type: "text", text: "First user" }] },
124
+ { role: "assistant", content: [{ type: "text", text: "Response" }] },
125
+ { role: "user", content: [{ type: "text", text: "Second user" }] },
126
+ ];
127
+
128
+ const result = buildAPIMessages(messages, "System prompt", "Reminder");
129
+
130
+ expect(result).toHaveLength(3);
131
+ // First user message should be unchanged
132
+ expect((result[0]?.content[0] as any).text).toBe("First user");
133
+ // Last user message should have reminder
134
+ const lastUserText = (result[2]?.content[0] as any).text;
135
+ expect(lastUserText).toContain("Second user");
136
+ expect(lastUserText).toContain("Reminder");
137
+ });
138
+
139
+ it("should handle user message with tool results", () => {
140
+ const messages: Message[] = [
141
+ {
142
+ role: "user",
143
+ content: [
144
+ { type: "tool_result", tool_use_id: "tool1", content: "Result 1" },
145
+ { type: "tool_result", tool_use_id: "tool2", content: "Result 2" },
146
+ ],
147
+ },
148
+ ];
149
+
150
+ const result = buildAPIMessages(messages, "System prompt", "Reminder");
151
+
152
+ expect(result).toHaveLength(1);
153
+ // Should add reminder as new text block since last block is tool_result
154
+ expect(result[0]?.content).toHaveLength(3);
155
+ expect(result[0]?.content[2]?.type).toBe("text");
156
+ });
157
+
158
+ it("should handle complex user message", () => {
159
+ const messages: Message[] = [
160
+ {
161
+ role: "user",
162
+ content: [
163
+ { type: "text", text: "Query text" },
164
+ { type: "image", source: { type: "base64", data: "img", media_type: "image/png" } },
165
+ ],
166
+ },
167
+ ];
168
+
169
+ const result = buildAPIMessages(messages, "System prompt", "Reminder");
170
+
171
+ // Reminder should be added as new block since last is image
172
+ expect(result[0]?.content).toHaveLength(3);
173
+ expect(result[0]?.content[0]?.type).toBe("text");
174
+ expect(result[0]?.content[1]?.type).toBe("image");
175
+ expect(result[0]?.content[2]?.type).toBe("text");
176
+ });
177
+
178
+ it("should handle assistant-only conversation", () => {
179
+ const messages: Message[] = [
180
+ { role: "assistant", content: [{ type: "text", text: "Hello" }] },
181
+ ];
182
+
183
+ const result = buildAPIMessages(messages, "System prompt", "Reminder");
184
+
185
+ // No user message to inject into, return unchanged
186
+ expect(result).toEqual(messages);
187
+ });
188
+
189
+ it("should handle multiline reminder", () => {
190
+ const messages: Message[] = [
191
+ { role: "user", content: [{ type: "text", text: "Hello" }] },
192
+ ];
193
+
194
+ const multilineReminder = `Line 1
195
+ Line 2
196
+ Line 3`;
197
+
198
+ const result = buildAPIMessages(messages, "System prompt", multilineReminder);
199
+
200
+ const text = (result[0]?.content[0] as any).text;
201
+ expect(text).toContain("Line 1");
202
+ expect(text).toContain("Line 2");
203
+ expect(text).toContain("Line 3");
204
+ });
205
+
206
+ it("should preserve message order", () => {
207
+ const messages: Message[] = [
208
+ { role: "user", content: [{ type: "text", text: "1" }] },
209
+ { role: "assistant", content: [{ type: "text", text: "2" }] },
210
+ { role: "user", content: [{ type: "text", text: "3" }] },
211
+ { role: "assistant", content: [{ type: "text", text: "4" }] },
212
+ { role: "user", content: [{ type: "text", text: "5" }] },
213
+ ];
214
+
215
+ const result = buildAPIMessages(messages, "System prompt", "Reminder");
216
+
217
+ expect(result[0]?.role).toBe("user");
218
+ expect(result[1]?.role).toBe("assistant");
219
+ expect(result[2]?.role).toBe("user");
220
+ expect(result[3]?.role).toBe("assistant");
221
+ expect(result[4]?.role).toBe("user");
222
+
223
+ // Only last user should have reminder
224
+ expect((result[0]?.content[0] as any).text).toBe("1");
225
+ expect((result[2]?.content[0] as any).text).toBe("3");
226
+ expect((result[4]?.content[0] as any).text).toContain("5");
227
+ expect((result[4]?.content[0] as any).text).toContain("Reminder");
228
+ });
229
+ });