@zwbigi/ink-xy 0.1.2 → 0.1.4

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 (598) hide show
  1. package/.next/BUILD_ID +1 -1
  2. package/.next/app-path-routes-manifest.json +2 -2
  3. package/.next/build-manifest.json +2 -2
  4. package/.next/server/app/_global-error.html +1 -1
  5. package/.next/server/app/_global-error.rsc +1 -1
  6. package/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  7. package/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  8. package/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  9. package/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  10. package/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  11. package/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  12. package/.next/server/app/_not-found.html +1 -1
  13. package/.next/server/app/_not-found.rsc +1 -1
  14. package/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
  15. package/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  16. package/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
  17. package/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  18. package/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  19. package/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  20. package/.next/server/app/api/inkos/route.js.nft.json +1 -1
  21. package/.next/server/app/api/skills/install/route.js.nft.json +1 -1
  22. package/.next/server/app/api/skills/search/route.js.nft.json +1 -1
  23. package/.next/server/app/index.html +1 -1
  24. package/.next/server/app/index.rsc +1 -1
  25. package/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
  26. package/.next/server/app/index.segments/_full.segment.rsc +1 -1
  27. package/.next/server/app/index.segments/_head.segment.rsc +1 -1
  28. package/.next/server/app/index.segments/_index.segment.rsc +1 -1
  29. package/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  30. package/.next/server/app-paths-manifest.json +2 -2
  31. package/.next/server/chunks/162.js +1 -1
  32. package/.next/server/middleware-build-manifest.js +1 -1
  33. package/.next/server/pages/404.html +1 -1
  34. package/.next/server/pages/500.html +1 -1
  35. package/.next/trace +4 -4
  36. package/.next/trace-build +1 -1
  37. package/bin/pi-web.js +4 -1
  38. package/inkos/.env.example +20 -0
  39. package/inkos/.node-version +1 -0
  40. package/inkos/.nvmrc +1 -0
  41. package/inkos/CHANGELOG.md +787 -0
  42. package/inkos/CONTRIBUTING.md +89 -0
  43. package/inkos/LICENSE +661 -0
  44. package/inkos/README.en.md +483 -0
  45. package/inkos/README.ja.md +461 -0
  46. package/inkos/README.md +272 -0
  47. package/inkos/assets/15qun.jpg +0 -0
  48. package/inkos/assets/41777702961_.pic.jpg +0 -0
  49. package/inkos/assets/inkos-short-demo-cover.png +0 -0
  50. package/inkos/assets/inkos-text.svg +40 -0
  51. package/inkos/assets/logo.svg +47 -0
  52. package/inkos/assets/screenshot-chapters.png +0 -0
  53. package/inkos/assets/screenshot-pipeline.png +0 -0
  54. package/inkos/assets/screenshot-state.png +0 -0
  55. package/inkos/assets/screenshot-terminal.png +0 -0
  56. package/inkos/assets/wechat-group-v8.jpg +0 -0
  57. package/inkos/package.json +42 -0
  58. package/inkos/packages/cli/package.json +74 -0
  59. package/inkos/packages/cli/src/__tests__/analytics.test.ts +154 -0
  60. package/inkos/packages/cli/src/__tests__/cli-integration.test.ts +1031 -0
  61. package/inkos/packages/cli/src/__tests__/daemon.test.ts +93 -0
  62. package/inkos/packages/cli/src/__tests__/doctor.test.ts +36 -0
  63. package/inkos/packages/cli/src/__tests__/interact-command.test.ts +142 -0
  64. package/inkos/packages/cli/src/__tests__/interaction-tools.test.ts +107 -0
  65. package/inkos/packages/cli/src/__tests__/llm-overrides.test.ts +25 -0
  66. package/inkos/packages/cli/src/__tests__/localization.test.ts +121 -0
  67. package/inkos/packages/cli/src/__tests__/progress-text.test.ts +92 -0
  68. package/inkos/packages/cli/src/__tests__/project-bootstrap.test.ts +71 -0
  69. package/inkos/packages/cli/src/__tests__/publish-package.test.ts +272 -0
  70. package/inkos/packages/cli/src/__tests__/revision-command.test.ts +82 -0
  71. package/inkos/packages/cli/src/__tests__/runtime-requirements.test.ts +89 -0
  72. package/inkos/packages/cli/src/__tests__/short-fiction-command.test.ts +48 -0
  73. package/inkos/packages/cli/src/__tests__/studio-runtime.test.ts +142 -0
  74. package/inkos/packages/cli/src/__tests__/studio.test.ts +87 -0
  75. package/inkos/packages/cli/src/__tests__/tui-activity-state.test.ts +20 -0
  76. package/inkos/packages/cli/src/__tests__/tui-agent-session.test.ts +213 -0
  77. package/inkos/packages/cli/src/__tests__/tui-chat-depth.test.ts +27 -0
  78. package/inkos/packages/cli/src/__tests__/tui-chat-draft.test.ts +44 -0
  79. package/inkos/packages/cli/src/__tests__/tui-command.test.ts +86 -0
  80. package/inkos/packages/cli/src/__tests__/tui-composer-caret.test.ts +46 -0
  81. package/inkos/packages/cli/src/__tests__/tui-composer-display.test.ts +40 -0
  82. package/inkos/packages/cli/src/__tests__/tui-dashboard.test.tsx +219 -0
  83. package/inkos/packages/cli/src/__tests__/tui-effects-i18n.test.ts +29 -0
  84. package/inkos/packages/cli/src/__tests__/tui-i18n.test.ts +22 -0
  85. package/inkos/packages/cli/src/__tests__/tui-input-chrome.test.ts +10 -0
  86. package/inkos/packages/cli/src/__tests__/tui-input-history.test.ts +40 -0
  87. package/inkos/packages/cli/src/__tests__/tui-layout.test.ts +55 -0
  88. package/inkos/packages/cli/src/__tests__/tui-local-commands.test.ts +47 -0
  89. package/inkos/packages/cli/src/__tests__/tui-session-store.test.ts +81 -0
  90. package/inkos/packages/cli/src/__tests__/tui-setup-i18n.test.ts +31 -0
  91. package/inkos/packages/cli/src/__tests__/tui-slash-autocomplete.test.ts +33 -0
  92. package/inkos/packages/cli/src/commands/agent.ts +65 -0
  93. package/inkos/packages/cli/src/commands/analytics.ts +77 -0
  94. package/inkos/packages/cli/src/commands/audit.ts +52 -0
  95. package/inkos/packages/cli/src/commands/book.ts +260 -0
  96. package/inkos/packages/cli/src/commands/compose.ts +50 -0
  97. package/inkos/packages/cli/src/commands/config.ts +328 -0
  98. package/inkos/packages/cli/src/commands/consolidate.ts +50 -0
  99. package/inkos/packages/cli/src/commands/daemon.ts +121 -0
  100. package/inkos/packages/cli/src/commands/detect.ts +125 -0
  101. package/inkos/packages/cli/src/commands/doctor.ts +391 -0
  102. package/inkos/packages/cli/src/commands/draft.ts +43 -0
  103. package/inkos/packages/cli/src/commands/eval.ts +217 -0
  104. package/inkos/packages/cli/src/commands/export.ts +45 -0
  105. package/inkos/packages/cli/src/commands/fanfic.ts +183 -0
  106. package/inkos/packages/cli/src/commands/genre.ts +160 -0
  107. package/inkos/packages/cli/src/commands/import.ts +158 -0
  108. package/inkos/packages/cli/src/commands/init.ts +47 -0
  109. package/inkos/packages/cli/src/commands/interact.ts +109 -0
  110. package/inkos/packages/cli/src/commands/plan.ts +54 -0
  111. package/inkos/packages/cli/src/commands/radar.ts +60 -0
  112. package/inkos/packages/cli/src/commands/review.ts +253 -0
  113. package/inkos/packages/cli/src/commands/revise.ts +58 -0
  114. package/inkos/packages/cli/src/commands/short-fiction.ts +294 -0
  115. package/inkos/packages/cli/src/commands/status.ts +138 -0
  116. package/inkos/packages/cli/src/commands/studio.ts +194 -0
  117. package/inkos/packages/cli/src/commands/style.ts +99 -0
  118. package/inkos/packages/cli/src/commands/tui.ts +18 -0
  119. package/inkos/packages/cli/src/commands/update.ts +45 -0
  120. package/inkos/packages/cli/src/commands/write.ts +324 -0
  121. package/inkos/packages/cli/src/index.ts +5 -0
  122. package/inkos/packages/cli/src/interaction/tools.ts +49 -0
  123. package/inkos/packages/cli/src/localization.ts +215 -0
  124. package/inkos/packages/cli/src/program.ts +106 -0
  125. package/inkos/packages/cli/src/progress-text.ts +85 -0
  126. package/inkos/packages/cli/src/project-bootstrap.ts +175 -0
  127. package/inkos/packages/cli/src/runtime-requirements.ts +135 -0
  128. package/inkos/packages/cli/src/tui/__tests__/markdown.test.ts +64 -0
  129. package/inkos/packages/cli/src/tui/activity-state.ts +41 -0
  130. package/inkos/packages/cli/src/tui/agent-input.ts +264 -0
  131. package/inkos/packages/cli/src/tui/ansi.ts +72 -0
  132. package/inkos/packages/cli/src/tui/app.ts +130 -0
  133. package/inkos/packages/cli/src/tui/chat-depth.ts +22 -0
  134. package/inkos/packages/cli/src/tui/chat-draft.ts +42 -0
  135. package/inkos/packages/cli/src/tui/composer-caret.ts +22 -0
  136. package/inkos/packages/cli/src/tui/composer-display.ts +26 -0
  137. package/inkos/packages/cli/src/tui/dashboard-model.ts +164 -0
  138. package/inkos/packages/cli/src/tui/dashboard.tsx +544 -0
  139. package/inkos/packages/cli/src/tui/effects.ts +542 -0
  140. package/inkos/packages/cli/src/tui/i18n.ts +278 -0
  141. package/inkos/packages/cli/src/tui/input-history.ts +69 -0
  142. package/inkos/packages/cli/src/tui/local-commands.ts +55 -0
  143. package/inkos/packages/cli/src/tui/markdown.ts +64 -0
  144. package/inkos/packages/cli/src/tui/session-store.ts +6 -0
  145. package/inkos/packages/cli/src/tui/setup.ts +397 -0
  146. package/inkos/packages/cli/src/tui/slash-autocomplete.ts +62 -0
  147. package/inkos/packages/cli/src/tui/theme.ts +17 -0
  148. package/inkos/packages/cli/src/utils.ts +222 -0
  149. package/inkos/packages/cli/tsconfig.json +9 -0
  150. package/inkos/packages/core/genres/cozy.md +43 -0
  151. package/inkos/packages/core/genres/cultivation.md +42 -0
  152. package/inkos/packages/core/genres/dungeon-core.md +40 -0
  153. package/inkos/packages/core/genres/horror.md +51 -0
  154. package/inkos/packages/core/genres/isekai.md +43 -0
  155. package/inkos/packages/core/genres/litrpg.md +43 -0
  156. package/inkos/packages/core/genres/other.md +24 -0
  157. package/inkos/packages/core/genres/progression.md +41 -0
  158. package/inkos/packages/core/genres/romantasy.md +45 -0
  159. package/inkos/packages/core/genres/sci-fi.md +42 -0
  160. package/inkos/packages/core/genres/system-apocalypse.md +40 -0
  161. package/inkos/packages/core/genres/tower-climber.md +41 -0
  162. package/inkos/packages/core/genres/urban.md +53 -0
  163. package/inkos/packages/core/genres/xianxia.md +46 -0
  164. package/inkos/packages/core/genres/xuanhuan.md +64 -0
  165. package/inkos/packages/core/package.json +61 -0
  166. package/inkos/packages/core/src/__tests__/agent-max-tokens-policy.test.ts +29 -0
  167. package/inkos/packages/core/src/__tests__/agent-session.test.ts +866 -0
  168. package/inkos/packages/core/src/__tests__/agent-system-prompt.test.ts +167 -0
  169. package/inkos/packages/core/src/__tests__/agent-tools-params.test.ts +197 -0
  170. package/inkos/packages/core/src/__tests__/agent-tools.test.ts +421 -0
  171. package/inkos/packages/core/src/__tests__/ai-tells.test.ts +90 -0
  172. package/inkos/packages/core/src/__tests__/architect-phase5-consolidated.test.ts +445 -0
  173. package/inkos/packages/core/src/__tests__/architect-phase5.test.ts +455 -0
  174. package/inkos/packages/core/src/__tests__/architect-phase7.test.ts +210 -0
  175. package/inkos/packages/core/src/__tests__/architect.test.ts +859 -0
  176. package/inkos/packages/core/src/__tests__/audit-parse.test.ts +78 -0
  177. package/inkos/packages/core/src/__tests__/book-id.test.ts +26 -0
  178. package/inkos/packages/core/src/__tests__/book-session-store.test.ts +447 -0
  179. package/inkos/packages/core/src/__tests__/book-session.test.ts +113 -0
  180. package/inkos/packages/core/src/__tests__/chapter-analyzer.test.ts +574 -0
  181. package/inkos/packages/core/src/__tests__/chapter-memo-parser.test.ts +247 -0
  182. package/inkos/packages/core/src/__tests__/chapter-persistence.test.ts +198 -0
  183. package/inkos/packages/core/src/__tests__/chapter-review-cycle.test.ts +294 -0
  184. package/inkos/packages/core/src/__tests__/chapter-splitter.test.ts +156 -0
  185. package/inkos/packages/core/src/__tests__/chapter-state-recovery.test.ts +235 -0
  186. package/inkos/packages/core/src/__tests__/chapter-truth-validation.test.ts +253 -0
  187. package/inkos/packages/core/src/__tests__/composer.test.ts +627 -0
  188. package/inkos/packages/core/src/__tests__/config-loader.test.ts +325 -0
  189. package/inkos/packages/core/src/__tests__/config-migration.test.ts +102 -0
  190. package/inkos/packages/core/src/__tests__/consolidator.test.ts +32 -0
  191. package/inkos/packages/core/src/__tests__/context-filter.test.ts +60 -0
  192. package/inkos/packages/core/src/__tests__/context-transform.test.ts +108 -0
  193. package/inkos/packages/core/src/__tests__/continuity.test.ts +391 -0
  194. package/inkos/packages/core/src/__tests__/detection-insights.test.ts +59 -0
  195. package/inkos/packages/core/src/__tests__/detector.test.ts +86 -0
  196. package/inkos/packages/core/src/__tests__/draft-directive-parser.test.ts +386 -0
  197. package/inkos/packages/core/src/__tests__/edit-controller.test.ts +190 -0
  198. package/inkos/packages/core/src/__tests__/effective-llm-config.test.ts +486 -0
  199. package/inkos/packages/core/src/__tests__/fanfic-dimensions.test.ts +58 -0
  200. package/inkos/packages/core/src/__tests__/fanfic-models.test.ts +69 -0
  201. package/inkos/packages/core/src/__tests__/governed-working-set.test.ts +155 -0
  202. package/inkos/packages/core/src/__tests__/hook-arbiter.test.ts +124 -0
  203. package/inkos/packages/core/src/__tests__/hook-governance.test.ts +228 -0
  204. package/inkos/packages/core/src/__tests__/hook-health.test.ts +166 -0
  205. package/inkos/packages/core/src/__tests__/hook-ledger-validator.test.ts +236 -0
  206. package/inkos/packages/core/src/__tests__/hook-promotion.test.ts +192 -0
  207. package/inkos/packages/core/src/__tests__/hook-stale-detection.test.ts +136 -0
  208. package/inkos/packages/core/src/__tests__/index-notify-lazy.test.ts +20 -0
  209. package/inkos/packages/core/src/__tests__/interaction-chat-tokens.test.ts +170 -0
  210. package/inkos/packages/core/src/__tests__/interaction-models.test.ts +155 -0
  211. package/inkos/packages/core/src/__tests__/interaction-nl-router.test.ts +223 -0
  212. package/inkos/packages/core/src/__tests__/interaction-runtime.test.ts +633 -0
  213. package/inkos/packages/core/src/__tests__/interaction-tools.test.ts +343 -0
  214. package/inkos/packages/core/src/__tests__/length-metrics.test.ts +82 -0
  215. package/inkos/packages/core/src/__tests__/length-normalizer.test.ts +331 -0
  216. package/inkos/packages/core/src/__tests__/list-models.test.ts +109 -0
  217. package/inkos/packages/core/src/__tests__/llm-env.test.ts +31 -0
  218. package/inkos/packages/core/src/__tests__/logger.test.ts +175 -0
  219. package/inkos/packages/core/src/__tests__/long-span-fatigue.test.ts +160 -0
  220. package/inkos/packages/core/src/__tests__/memory-retrieval.test.ts +1303 -0
  221. package/inkos/packages/core/src/__tests__/models.test.ts +918 -0
  222. package/inkos/packages/core/src/__tests__/outline-paths.test.ts +97 -0
  223. package/inkos/packages/core/src/__tests__/path-safety.test.ts +22 -0
  224. package/inkos/packages/core/src/__tests__/persisted-governed-plan.test.ts +134 -0
  225. package/inkos/packages/core/src/__tests__/phase5-cleanup.test.ts +393 -0
  226. package/inkos/packages/core/src/__tests__/phase5-hotfix.test.ts +288 -0
  227. package/inkos/packages/core/src/__tests__/phase7-hotfix.test.ts +614 -0
  228. package/inkos/packages/core/src/__tests__/pipeline-agent.test.ts +354 -0
  229. package/inkos/packages/core/src/__tests__/pipeline-runner-memory-sync.test.ts +317 -0
  230. package/inkos/packages/core/src/__tests__/pipeline-runner.test.ts +5200 -0
  231. package/inkos/packages/core/src/__tests__/planner-context.test.ts +137 -0
  232. package/inkos/packages/core/src/__tests__/planner-prompts-ratio.test.ts +11 -0
  233. package/inkos/packages/core/src/__tests__/planner-prompts.test.ts +171 -0
  234. package/inkos/packages/core/src/__tests__/planner.test.ts +362 -0
  235. package/inkos/packages/core/src/__tests__/planning-materials.test.ts +90 -0
  236. package/inkos/packages/core/src/__tests__/polisher.test.ts +189 -0
  237. package/inkos/packages/core/src/__tests__/post-write-validator.test.ts +291 -0
  238. package/inkos/packages/core/src/__tests__/probe.test.ts +77 -0
  239. package/inkos/packages/core/src/__tests__/project-interaction.test.ts +241 -0
  240. package/inkos/packages/core/src/__tests__/provider.test.ts +953 -0
  241. package/inkos/packages/core/src/__tests__/providers-group.test.ts +34 -0
  242. package/inkos/packages/core/src/__tests__/providers-lookup.test.ts +81 -0
  243. package/inkos/packages/core/src/__tests__/providers-schema.test.ts +158 -0
  244. package/inkos/packages/core/src/__tests__/proxy-fetch.test.ts +75 -0
  245. package/inkos/packages/core/src/__tests__/revise-foundation.test.ts +514 -0
  246. package/inkos/packages/core/src/__tests__/reviser.test.ts +859 -0
  247. package/inkos/packages/core/src/__tests__/runtime-state-store.test.ts +388 -0
  248. package/inkos/packages/core/src/__tests__/scheduler.test.ts +123 -0
  249. package/inkos/packages/core/src/__tests__/secrets-migration.test.ts +71 -0
  250. package/inkos/packages/core/src/__tests__/secrets.test.ts +95 -0
  251. package/inkos/packages/core/src/__tests__/sensitive-words.test.ts +88 -0
  252. package/inkos/packages/core/src/__tests__/service-presets-regression.test.ts +73 -0
  253. package/inkos/packages/core/src/__tests__/service-resolver-regression.test.ts +75 -0
  254. package/inkos/packages/core/src/__tests__/service-resolver.test.ts +228 -0
  255. package/inkos/packages/core/src/__tests__/session-transcript-restore.test.ts +1311 -0
  256. package/inkos/packages/core/src/__tests__/session-transcript.test.ts +195 -0
  257. package/inkos/packages/core/src/__tests__/settler-delta-parser.test.ts +133 -0
  258. package/inkos/packages/core/src/__tests__/short-fiction-public.test.ts +241 -0
  259. package/inkos/packages/core/src/__tests__/spot-fix-patches.test.ts +104 -0
  260. package/inkos/packages/core/src/__tests__/state-manager.test.ts +1298 -0
  261. package/inkos/packages/core/src/__tests__/state-projections.test.ts +130 -0
  262. package/inkos/packages/core/src/__tests__/state-reducer.test.ts +372 -0
  263. package/inkos/packages/core/src/__tests__/state-validator-agent.test.ts +165 -0
  264. package/inkos/packages/core/src/__tests__/state-validator.test.ts +122 -0
  265. package/inkos/packages/core/src/__tests__/style-analyzer.test.ts +61 -0
  266. package/inkos/packages/core/src/__tests__/temperature-constraints.test.ts +57 -0
  267. package/inkos/packages/core/src/__tests__/v13-hotfix-round4.test.ts +343 -0
  268. package/inkos/packages/core/src/__tests__/verify-service.test.ts +77 -0
  269. package/inkos/packages/core/src/__tests__/webhook.test.ts +91 -0
  270. package/inkos/packages/core/src/__tests__/writer-parser.test.ts +348 -0
  271. package/inkos/packages/core/src/__tests__/writer-prompts.test.ts +269 -0
  272. package/inkos/packages/core/src/__tests__/writer.test.ts +1360 -0
  273. package/inkos/packages/core/src/agent/agent-session.ts +737 -0
  274. package/inkos/packages/core/src/agent/agent-system-prompt.ts +199 -0
  275. package/inkos/packages/core/src/agent/agent-tools.ts +835 -0
  276. package/inkos/packages/core/src/agent/context-transform.ts +85 -0
  277. package/inkos/packages/core/src/agent/index.ts +14 -0
  278. package/inkos/packages/core/src/agents/ai-tells.ts +161 -0
  279. package/inkos/packages/core/src/agents/architect.ts +1291 -0
  280. package/inkos/packages/core/src/agents/base.ts +100 -0
  281. package/inkos/packages/core/src/agents/chapter-analyzer.ts +634 -0
  282. package/inkos/packages/core/src/agents/composer.ts +469 -0
  283. package/inkos/packages/core/src/agents/consolidator.ts +218 -0
  284. package/inkos/packages/core/src/agents/continuity.ts +824 -0
  285. package/inkos/packages/core/src/agents/detection-insights.ts +72 -0
  286. package/inkos/packages/core/src/agents/detector.ts +224 -0
  287. package/inkos/packages/core/src/agents/en-prompt-sections.ts +129 -0
  288. package/inkos/packages/core/src/agents/fanfic-canon-importer.ts +146 -0
  289. package/inkos/packages/core/src/agents/fanfic-dimensions.ts +87 -0
  290. package/inkos/packages/core/src/agents/fanfic-prompt-sections.ts +109 -0
  291. package/inkos/packages/core/src/agents/foundation-reviewer.ts +204 -0
  292. package/inkos/packages/core/src/agents/length-normalizer.ts +218 -0
  293. package/inkos/packages/core/src/agents/observer-prompts.ts +127 -0
  294. package/inkos/packages/core/src/agents/planner-context.ts +297 -0
  295. package/inkos/packages/core/src/agents/planner-prompts.ts +404 -0
  296. package/inkos/packages/core/src/agents/planner.ts +783 -0
  297. package/inkos/packages/core/src/agents/polisher.ts +153 -0
  298. package/inkos/packages/core/src/agents/post-write-validator.ts +873 -0
  299. package/inkos/packages/core/src/agents/radar-source.ts +123 -0
  300. package/inkos/packages/core/src/agents/radar.ts +120 -0
  301. package/inkos/packages/core/src/agents/reviser.ts +701 -0
  302. package/inkos/packages/core/src/agents/rules-reader.ts +155 -0
  303. package/inkos/packages/core/src/agents/sensitive-words.ts +142 -0
  304. package/inkos/packages/core/src/agents/settler-delta-parser.ts +53 -0
  305. package/inkos/packages/core/src/agents/settler-parser.ts +38 -0
  306. package/inkos/packages/core/src/agents/settler-prompts.ts +230 -0
  307. package/inkos/packages/core/src/agents/short-fiction.ts +429 -0
  308. package/inkos/packages/core/src/agents/state-validator.ts +322 -0
  309. package/inkos/packages/core/src/agents/style-analyzer.ts +93 -0
  310. package/inkos/packages/core/src/agents/writer-parser.ts +178 -0
  311. package/inkos/packages/core/src/agents/writer-prompts.ts +899 -0
  312. package/inkos/packages/core/src/agents/writer.ts +1450 -0
  313. package/inkos/packages/core/src/index.ts +392 -0
  314. package/inkos/packages/core/src/interaction/book-session-store.ts +226 -0
  315. package/inkos/packages/core/src/interaction/draft-directive-parser.ts +266 -0
  316. package/inkos/packages/core/src/interaction/edit-controller.ts +270 -0
  317. package/inkos/packages/core/src/interaction/events.ts +41 -0
  318. package/inkos/packages/core/src/interaction/export-artifact.ts +151 -0
  319. package/inkos/packages/core/src/interaction/intents.ts +63 -0
  320. package/inkos/packages/core/src/interaction/modes.ts +13 -0
  321. package/inkos/packages/core/src/interaction/nl-router.ts +258 -0
  322. package/inkos/packages/core/src/interaction/project-control.ts +150 -0
  323. package/inkos/packages/core/src/interaction/project-session-store.ts +81 -0
  324. package/inkos/packages/core/src/interaction/project-tools.ts +704 -0
  325. package/inkos/packages/core/src/interaction/request-router.ts +5 -0
  326. package/inkos/packages/core/src/interaction/runtime.ts +1167 -0
  327. package/inkos/packages/core/src/interaction/session-transcript-legacy.ts +113 -0
  328. package/inkos/packages/core/src/interaction/session-transcript-restore.ts +607 -0
  329. package/inkos/packages/core/src/interaction/session-transcript-schema.ts +76 -0
  330. package/inkos/packages/core/src/interaction/session-transcript.ts +189 -0
  331. package/inkos/packages/core/src/interaction/session.ts +226 -0
  332. package/inkos/packages/core/src/interaction/truth-authority.ts +45 -0
  333. package/inkos/packages/core/src/llm/config-migration.ts +58 -0
  334. package/inkos/packages/core/src/llm/cover-providers.ts +45 -0
  335. package/inkos/packages/core/src/llm/provider.ts +1331 -0
  336. package/inkos/packages/core/src/llm/providers/endpoints/ai360.ts +42 -0
  337. package/inkos/packages/core/src/llm/providers/endpoints/anthropic.ts +82 -0
  338. package/inkos/packages/core/src/llm/providers/endpoints/astronCodingPlan.ts +30 -0
  339. package/inkos/packages/core/src/llm/providers/endpoints/baichuan.ts +28 -0
  340. package/inkos/packages/core/src/llm/providers/endpoints/bailian.ts +65 -0
  341. package/inkos/packages/core/src/llm/providers/endpoints/bailianCodingPlan.ts +30 -0
  342. package/inkos/packages/core/src/llm/providers/endpoints/custom.ts +22 -0
  343. package/inkos/packages/core/src/llm/providers/endpoints/deepseek.ts +35 -0
  344. package/inkos/packages/core/src/llm/providers/endpoints/giteeai.ts +41 -0
  345. package/inkos/packages/core/src/llm/providers/endpoints/githubCopilot.ts +43 -0
  346. package/inkos/packages/core/src/llm/providers/endpoints/glmCodingPlan.ts +28 -0
  347. package/inkos/packages/core/src/llm/providers/endpoints/google.ts +51 -0
  348. package/inkos/packages/core/src/llm/providers/endpoints/hunyuan.ts +42 -0
  349. package/inkos/packages/core/src/llm/providers/endpoints/infiniai.ts +72 -0
  350. package/inkos/packages/core/src/llm/providers/endpoints/internlm.ts +28 -0
  351. package/inkos/packages/core/src/llm/providers/endpoints/kimiCode.ts +23 -0
  352. package/inkos/packages/core/src/llm/providers/endpoints/kimiCodingPlan.ts +24 -0
  353. package/inkos/packages/core/src/llm/providers/endpoints/kkaiapi.ts +56 -0
  354. package/inkos/packages/core/src/llm/providers/endpoints/longcat.ts +25 -0
  355. package/inkos/packages/core/src/llm/providers/endpoints/minimax.ts +39 -0
  356. package/inkos/packages/core/src/llm/providers/endpoints/minimaxCodingPlan.ts +28 -0
  357. package/inkos/packages/core/src/llm/providers/endpoints/mistral.ts +40 -0
  358. package/inkos/packages/core/src/llm/providers/endpoints/modelscope.ts +30 -0
  359. package/inkos/packages/core/src/llm/providers/endpoints/moonshot.ts +39 -0
  360. package/inkos/packages/core/src/llm/providers/endpoints/newapi.ts +21 -0
  361. package/inkos/packages/core/src/llm/providers/endpoints/ollama.ts +73 -0
  362. package/inkos/packages/core/src/llm/providers/endpoints/openai.ts +77 -0
  363. package/inkos/packages/core/src/llm/providers/endpoints/opencodeCodingPlan.ts +30 -0
  364. package/inkos/packages/core/src/llm/providers/endpoints/openrouter.ts +87 -0
  365. package/inkos/packages/core/src/llm/providers/endpoints/ppio.ts +86 -0
  366. package/inkos/packages/core/src/llm/providers/endpoints/qiniu.ts +32 -0
  367. package/inkos/packages/core/src/llm/providers/endpoints/sensenova.ts +45 -0
  368. package/inkos/packages/core/src/llm/providers/endpoints/siliconcloud.ts +126 -0
  369. package/inkos/packages/core/src/llm/providers/endpoints/spark.ts +33 -0
  370. package/inkos/packages/core/src/llm/providers/endpoints/stepfun.ts +35 -0
  371. package/inkos/packages/core/src/llm/providers/endpoints/tencentcloud.ts +25 -0
  372. package/inkos/packages/core/src/llm/providers/endpoints/volcengine.ts +52 -0
  373. package/inkos/packages/core/src/llm/providers/endpoints/volcengineCodingPlan.ts +42 -0
  374. package/inkos/packages/core/src/llm/providers/endpoints/wenxin.ts +106 -0
  375. package/inkos/packages/core/src/llm/providers/endpoints/xai.ts +34 -0
  376. package/inkos/packages/core/src/llm/providers/endpoints/xiaomimimo.ts +26 -0
  377. package/inkos/packages/core/src/llm/providers/endpoints/zeroone.ts +34 -0
  378. package/inkos/packages/core/src/llm/providers/endpoints/zhipu.ts +61 -0
  379. package/inkos/packages/core/src/llm/providers/index.ts +71 -0
  380. package/inkos/packages/core/src/llm/providers/lookup.ts +70 -0
  381. package/inkos/packages/core/src/llm/providers/probe.ts +35 -0
  382. package/inkos/packages/core/src/llm/providers/types.ts +89 -0
  383. package/inkos/packages/core/src/llm/providers/verify.ts +104 -0
  384. package/inkos/packages/core/src/llm/secrets.ts +77 -0
  385. package/inkos/packages/core/src/llm/service-presets.ts +215 -0
  386. package/inkos/packages/core/src/llm/service-resolver.ts +91 -0
  387. package/inkos/packages/core/src/models/book-rules.ts +126 -0
  388. package/inkos/packages/core/src/models/book.ts +70 -0
  389. package/inkos/packages/core/src/models/chapter.ts +42 -0
  390. package/inkos/packages/core/src/models/detection.ts +25 -0
  391. package/inkos/packages/core/src/models/genre-profile.ts +36 -0
  392. package/inkos/packages/core/src/models/input-governance.ts +99 -0
  393. package/inkos/packages/core/src/models/length-governance.ts +46 -0
  394. package/inkos/packages/core/src/models/project.ts +161 -0
  395. package/inkos/packages/core/src/models/runtime-state.ts +144 -0
  396. package/inkos/packages/core/src/models/state.ts +52 -0
  397. package/inkos/packages/core/src/models/style-profile.ts +15 -0
  398. package/inkos/packages/core/src/notify/dispatcher.ts +96 -0
  399. package/inkos/packages/core/src/notify/feishu.ts +34 -0
  400. package/inkos/packages/core/src/notify/telegram.ts +25 -0
  401. package/inkos/packages/core/src/notify/webhook.ts +58 -0
  402. package/inkos/packages/core/src/notify/wechat-work.ts +22 -0
  403. package/inkos/packages/core/src/pipeline/agent.ts +691 -0
  404. package/inkos/packages/core/src/pipeline/chapter-persistence.ts +79 -0
  405. package/inkos/packages/core/src/pipeline/chapter-review-cycle.ts +324 -0
  406. package/inkos/packages/core/src/pipeline/chapter-state-recovery.ts +236 -0
  407. package/inkos/packages/core/src/pipeline/chapter-truth-validation.ts +145 -0
  408. package/inkos/packages/core/src/pipeline/detection-runner.ts +164 -0
  409. package/inkos/packages/core/src/pipeline/persisted-governed-plan.ts +216 -0
  410. package/inkos/packages/core/src/pipeline/runner.ts +3438 -0
  411. package/inkos/packages/core/src/pipeline/scheduler.ts +411 -0
  412. package/inkos/packages/core/src/pipeline/short-fiction-runner.ts +801 -0
  413. package/inkos/packages/core/src/prompts/index.ts +1 -0
  414. package/inkos/packages/core/src/prompts/short-fiction.ts +273 -0
  415. package/inkos/packages/core/src/state/manager.ts +560 -0
  416. package/inkos/packages/core/src/state/memory-db.ts +359 -0
  417. package/inkos/packages/core/src/state/runtime-state-store.ts +164 -0
  418. package/inkos/packages/core/src/state/state-bootstrap.ts +657 -0
  419. package/inkos/packages/core/src/state/state-projections.ts +255 -0
  420. package/inkos/packages/core/src/state/state-reducer.ts +260 -0
  421. package/inkos/packages/core/src/state/state-validator.ts +117 -0
  422. package/inkos/packages/core/src/utils/analytics.ts +92 -0
  423. package/inkos/packages/core/src/utils/book-id.ts +31 -0
  424. package/inkos/packages/core/src/utils/cadence-policy.ts +46 -0
  425. package/inkos/packages/core/src/utils/chapter-cadence.ts +211 -0
  426. package/inkos/packages/core/src/utils/chapter-memo-parser.ts +157 -0
  427. package/inkos/packages/core/src/utils/chapter-splitter.ts +80 -0
  428. package/inkos/packages/core/src/utils/config-loader.ts +29 -0
  429. package/inkos/packages/core/src/utils/context-assembly.ts +98 -0
  430. package/inkos/packages/core/src/utils/context-filter.ts +190 -0
  431. package/inkos/packages/core/src/utils/effective-llm-config.ts +529 -0
  432. package/inkos/packages/core/src/utils/governed-context.ts +101 -0
  433. package/inkos/packages/core/src/utils/governed-working-set.ts +395 -0
  434. package/inkos/packages/core/src/utils/hook-arbiter.ts +332 -0
  435. package/inkos/packages/core/src/utils/hook-governance.ts +199 -0
  436. package/inkos/packages/core/src/utils/hook-health.ts +189 -0
  437. package/inkos/packages/core/src/utils/hook-ledger-validator.ts +277 -0
  438. package/inkos/packages/core/src/utils/hook-lifecycle.ts +224 -0
  439. package/inkos/packages/core/src/utils/hook-policy.ts +115 -0
  440. package/inkos/packages/core/src/utils/hook-promotion.ts +313 -0
  441. package/inkos/packages/core/src/utils/hook-stale-detection.ts +168 -0
  442. package/inkos/packages/core/src/utils/length-metrics.ts +123 -0
  443. package/inkos/packages/core/src/utils/llm-endpoint-auth.ts +40 -0
  444. package/inkos/packages/core/src/utils/llm-env.ts +74 -0
  445. package/inkos/packages/core/src/utils/logger.ts +123 -0
  446. package/inkos/packages/core/src/utils/long-span-fatigue.ts +545 -0
  447. package/inkos/packages/core/src/utils/memory-retrieval.ts +527 -0
  448. package/inkos/packages/core/src/utils/narrative-control.ts +177 -0
  449. package/inkos/packages/core/src/utils/outline-paths.ts +275 -0
  450. package/inkos/packages/core/src/utils/path-safety.ts +11 -0
  451. package/inkos/packages/core/src/utils/planning-materials.ts +185 -0
  452. package/inkos/packages/core/src/utils/pov-filter.ts +149 -0
  453. package/inkos/packages/core/src/utils/proxy-fetch.ts +44 -0
  454. package/inkos/packages/core/src/utils/runtime-writer.ts +41 -0
  455. package/inkos/packages/core/src/utils/spot-fix-patches.ts +189 -0
  456. package/inkos/packages/core/src/utils/story-markdown.ts +346 -0
  457. package/inkos/packages/core/src/utils/web-search.ts +82 -0
  458. package/inkos/packages/core/src/utils/writing-methodology.ts +164 -0
  459. package/inkos/packages/core/tsconfig.json +8 -0
  460. package/inkos/packages/core/vitest.config.ts +7 -0
  461. package/inkos/packages/studio/components.json +25 -0
  462. package/inkos/packages/studio/index.html +13 -0
  463. package/inkos/packages/studio/package.json +72 -0
  464. package/inkos/packages/studio/postcss.config.js +3 -0
  465. package/inkos/packages/studio/src/App.test.ts +25 -0
  466. package/inkos/packages/studio/src/App.tsx +280 -0
  467. package/inkos/packages/studio/src/api/__tests__/normalize-base-url.test.ts +40 -0
  468. package/inkos/packages/studio/src/api/book-create.test.ts +104 -0
  469. package/inkos/packages/studio/src/api/book-create.ts +94 -0
  470. package/inkos/packages/studio/src/api/errors.ts +17 -0
  471. package/inkos/packages/studio/src/api/index.ts +30 -0
  472. package/inkos/packages/studio/src/api/lib/run-store.ts +177 -0
  473. package/inkos/packages/studio/src/api/lib/sse.ts +50 -0
  474. package/inkos/packages/studio/src/api/phase5-hotfix.test.ts +335 -0
  475. package/inkos/packages/studio/src/api/safety.ts +6 -0
  476. package/inkos/packages/studio/src/api/server.test.ts +3162 -0
  477. package/inkos/packages/studio/src/api/server.ts +3666 -0
  478. package/inkos/packages/studio/src/api/v13-hotfix-round4.test.ts +226 -0
  479. package/inkos/packages/studio/src/app-state.test.ts +8 -0
  480. package/inkos/packages/studio/src/app-state.ts +1 -0
  481. package/inkos/packages/studio/src/components/ConfirmDialog.tsx +95 -0
  482. package/inkos/packages/studio/src/components/ServiceConfigSourceCard.tsx +139 -0
  483. package/inkos/packages/studio/src/components/ServiceQuickLinks.tsx +65 -0
  484. package/inkos/packages/studio/src/components/Sidebar.tsx +652 -0
  485. package/inkos/packages/studio/src/components/ai-elements/code-block.tsx +562 -0
  486. package/inkos/packages/studio/src/components/ai-elements/confirmation.tsx +174 -0
  487. package/inkos/packages/studio/src/components/ai-elements/message.tsx +360 -0
  488. package/inkos/packages/studio/src/components/ai-elements/prompt-input.tsx +1457 -0
  489. package/inkos/packages/studio/src/components/ai-elements/reasoning.tsx +226 -0
  490. package/inkos/packages/studio/src/components/ai-elements/shimmer.tsx +77 -0
  491. package/inkos/packages/studio/src/components/ai-elements/tool.tsx +173 -0
  492. package/inkos/packages/studio/src/components/chat/BookSidebar.tsx +291 -0
  493. package/inkos/packages/studio/src/components/chat/ChatMessage.tsx +39 -0
  494. package/inkos/packages/studio/src/components/chat/QuickActions.tsx +73 -0
  495. package/inkos/packages/studio/src/components/chat/ToolExecutionSteps.tsx +320 -0
  496. package/inkos/packages/studio/src/components/chat/__tests__/ToolExecutionSteps.test.ts +114 -0
  497. package/inkos/packages/studio/src/components/chat-utils.ts +56 -0
  498. package/inkos/packages/studio/src/components/chatbar-state.test.ts +69 -0
  499. package/inkos/packages/studio/src/components/sidebar/ChaptersSection.tsx +66 -0
  500. package/inkos/packages/studio/src/components/sidebar/CharacterSection.tsx +129 -0
  501. package/inkos/packages/studio/src/components/sidebar/FoundationSection.tsx +61 -0
  502. package/inkos/packages/studio/src/components/sidebar/ProgressSection.tsx +124 -0
  503. package/inkos/packages/studio/src/components/sidebar/SidebarCard.tsx +30 -0
  504. package/inkos/packages/studio/src/components/sidebar/SummarySection.tsx +89 -0
  505. package/inkos/packages/studio/src/components/ui/alert.tsx +76 -0
  506. package/inkos/packages/studio/src/components/ui/badge.tsx +52 -0
  507. package/inkos/packages/studio/src/components/ui/button-group.tsx +87 -0
  508. package/inkos/packages/studio/src/components/ui/button.tsx +58 -0
  509. package/inkos/packages/studio/src/components/ui/collapsible.tsx +19 -0
  510. package/inkos/packages/studio/src/components/ui/command.tsx +194 -0
  511. package/inkos/packages/studio/src/components/ui/dialog.tsx +158 -0
  512. package/inkos/packages/studio/src/components/ui/dropdown-menu.tsx +266 -0
  513. package/inkos/packages/studio/src/components/ui/hover-card.tsx +51 -0
  514. package/inkos/packages/studio/src/components/ui/input-group.tsx +158 -0
  515. package/inkos/packages/studio/src/components/ui/input.tsx +20 -0
  516. package/inkos/packages/studio/src/components/ui/select.tsx +199 -0
  517. package/inkos/packages/studio/src/components/ui/separator.tsx +23 -0
  518. package/inkos/packages/studio/src/components/ui/spinner.tsx +10 -0
  519. package/inkos/packages/studio/src/components/ui/textarea.tsx +18 -0
  520. package/inkos/packages/studio/src/components/ui/tooltip.tsx +66 -0
  521. package/inkos/packages/studio/src/constants/service-groups.ts +29 -0
  522. package/inkos/packages/studio/src/hooks/use-api.test.ts +93 -0
  523. package/inkos/packages/studio/src/hooks/use-api.ts +189 -0
  524. package/inkos/packages/studio/src/hooks/use-book-activity.test.ts +129 -0
  525. package/inkos/packages/studio/src/hooks/use-book-activity.ts +180 -0
  526. package/inkos/packages/studio/src/hooks/use-colors.ts +27 -0
  527. package/inkos/packages/studio/src/hooks/use-hash-route.test.ts +101 -0
  528. package/inkos/packages/studio/src/hooks/use-hash-route.ts +89 -0
  529. package/inkos/packages/studio/src/hooks/use-i18n.ts +289 -0
  530. package/inkos/packages/studio/src/hooks/use-session-events.ts +64 -0
  531. package/inkos/packages/studio/src/hooks/use-sse.test.ts +52 -0
  532. package/inkos/packages/studio/src/hooks/use-sse.ts +92 -0
  533. package/inkos/packages/studio/src/hooks/use-theme.test.ts +31 -0
  534. package/inkos/packages/studio/src/hooks/use-theme.ts +75 -0
  535. package/inkos/packages/studio/src/index.css +323 -0
  536. package/inkos/packages/studio/src/lib/error-copy.test.ts +34 -0
  537. package/inkos/packages/studio/src/lib/error-copy.ts +37 -0
  538. package/inkos/packages/studio/src/lib/utils.ts +6 -0
  539. package/inkos/packages/studio/src/main.tsx +10 -0
  540. package/inkos/packages/studio/src/pages/Analytics.tsx +80 -0
  541. package/inkos/packages/studio/src/pages/BookCreate.tsx +895 -0
  542. package/inkos/packages/studio/src/pages/BookDetail.tsx +652 -0
  543. package/inkos/packages/studio/src/pages/ChapterReader.tsx +266 -0
  544. package/inkos/packages/studio/src/pages/ChatPage.tsx +521 -0
  545. package/inkos/packages/studio/src/pages/DaemonControl.tsx +116 -0
  546. package/inkos/packages/studio/src/pages/Dashboard.tsx +379 -0
  547. package/inkos/packages/studio/src/pages/DoctorView.tsx +82 -0
  548. package/inkos/packages/studio/src/pages/GenreManager.tsx +464 -0
  549. package/inkos/packages/studio/src/pages/ImportManager.tsx +216 -0
  550. package/inkos/packages/studio/src/pages/LanguageSelector.tsx +74 -0
  551. package/inkos/packages/studio/src/pages/LogViewer.tsx +82 -0
  552. package/inkos/packages/studio/src/pages/RadarView.tsx +157 -0
  553. package/inkos/packages/studio/src/pages/ServiceDetailPage.tsx +393 -0
  554. package/inkos/packages/studio/src/pages/ServiceListPage.tsx +463 -0
  555. package/inkos/packages/studio/src/pages/StyleManager.tsx +225 -0
  556. package/inkos/packages/studio/src/pages/TruthFiles.tsx +194 -0
  557. package/inkos/packages/studio/src/pages/chat-page-state.test.ts +206 -0
  558. package/inkos/packages/studio/src/pages/chat-page-state.ts +112 -0
  559. package/inkos/packages/studio/src/pages/page-state.test.ts +258 -0
  560. package/inkos/packages/studio/src/pages/service-detail-state.test.ts +294 -0
  561. package/inkos/packages/studio/src/pages/service-detail-state.ts +234 -0
  562. package/inkos/packages/studio/src/pages/style-manager-state.test.ts +22 -0
  563. package/inkos/packages/studio/src/pages/truth-files-state.test.ts +61 -0
  564. package/inkos/packages/studio/src/shared/contracts.ts +143 -0
  565. package/inkos/packages/studio/src/store/chat/__tests__/message-parts.test.ts +172 -0
  566. package/inkos/packages/studio/src/store/chat/index.ts +3 -0
  567. package/inkos/packages/studio/src/store/chat/initialState.ts +8 -0
  568. package/inkos/packages/studio/src/store/chat/message-policy.test.ts +16 -0
  569. package/inkos/packages/studio/src/store/chat/message-policy.ts +5 -0
  570. package/inkos/packages/studio/src/store/chat/parts-builder.ts +187 -0
  571. package/inkos/packages/studio/src/store/chat/selectors.ts +13 -0
  572. package/inkos/packages/studio/src/store/chat/slices/create/action.ts +10 -0
  573. package/inkos/packages/studio/src/store/chat/slices/create/initialState.ts +9 -0
  574. package/inkos/packages/studio/src/store/chat/slices/message/action.ts +417 -0
  575. package/inkos/packages/studio/src/store/chat/slices/message/initialState.ts +10 -0
  576. package/inkos/packages/studio/src/store/chat/slices/message/runtime.test.ts +21 -0
  577. package/inkos/packages/studio/src/store/chat/slices/message/runtime.ts +233 -0
  578. package/inkos/packages/studio/src/store/chat/slices/message/stream-events.ts +272 -0
  579. package/inkos/packages/studio/src/store/chat/store.ts +11 -0
  580. package/inkos/packages/studio/src/store/chat/types.ts +169 -0
  581. package/inkos/packages/studio/src/store/service/index.ts +2 -0
  582. package/inkos/packages/studio/src/store/service/store.ts +123 -0
  583. package/inkos/packages/studio/src/store/service/types.ts +50 -0
  584. package/inkos/packages/studio/tsconfig.json +24 -0
  585. package/inkos/packages/studio/tsconfig.server.json +11 -0
  586. package/inkos/packages/studio/vite.config.ts +34 -0
  587. package/inkos/packages/studio/vitest.config.ts +14 -0
  588. package/inkos/pnpm-lock.yaml +9569 -0
  589. package/inkos/pnpm-workspace.yaml +2 -0
  590. package/inkos/scripts/prepare-package-for-publish.mjs +135 -0
  591. package/inkos/scripts/restore-package-json.mjs +31 -0
  592. package/inkos/scripts/set-package-versions.mjs +74 -0
  593. package/inkos/scripts/verify-no-workspace-protocol.mjs +140 -0
  594. package/inkos/skills/SKILL.md +654 -0
  595. package/inkos/tsconfig.json +19 -0
  596. package/package.json +4 -3
  597. /package/.next/static/{F2hMZMf1IyCVAWpkbtRz7 → -3vIrBZXdQ0rp7Wa3Kz40}/_buildManifest.js +0 -0
  598. /package/.next/static/{F2hMZMf1IyCVAWpkbtRz7 → -3vIrBZXdQ0rp7Wa3Kz40}/_ssgManifest.js +0 -0
@@ -0,0 +1,953 @@
1
+ import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
2
+ import type { AssistantMessage, Model, Api } from "@mariozechner/pi-ai";
3
+ import {
4
+ __resetFixedTemperatureWarnings,
5
+ chatCompletion,
6
+ type LLMClient,
7
+ } from "../llm/provider.js";
8
+
9
+ // ── Mock @mariozechner/pi-ai ──────────────────────────────────────────────────
10
+ // We intercept streamSimple so tests don't hit the network.
11
+
12
+ const mockStreamSimple = vi.fn();
13
+ const mockCompleteSimple = vi.fn();
14
+ const mockComplete = vi.fn();
15
+
16
+ vi.mock("@mariozechner/pi-ai", async (importOriginal) => {
17
+ const original = await importOriginal<typeof import("@mariozechner/pi-ai")>();
18
+ return {
19
+ ...original,
20
+ streamSimple: (...args: unknown[]) => mockStreamSimple(...args),
21
+ completeSimple: (...args: unknown[]) => mockCompleteSimple(...args),
22
+ complete: (...args: unknown[]) => mockComplete(...args),
23
+ };
24
+ });
25
+
26
+ // ── Helpers ───────────────────────────────────────────────────────────────────
27
+
28
+ const MOCK_USAGE = {
29
+ input: 11,
30
+ output: 7,
31
+ cacheRead: 0,
32
+ cacheWrite: 0,
33
+ totalTokens: 18,
34
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 },
35
+ };
36
+
37
+ function makeAssistantMessage(text: string): AssistantMessage {
38
+ return {
39
+ role: "assistant",
40
+ content: [{ type: "text", text }],
41
+ api: "openai-completions" as Api,
42
+ provider: "openai",
43
+ model: "test-model",
44
+ usage: MOCK_USAGE,
45
+ stopReason: "stop",
46
+ timestamp: Date.now(),
47
+ };
48
+ }
49
+
50
+ /** Builds an async iterable that emits the given events. */
51
+ function makeEventStream(
52
+ events: Array<Record<string, unknown>>,
53
+ ): AsyncIterable<Record<string, unknown>> {
54
+ return {
55
+ [Symbol.asyncIterator](): AsyncIterator<Record<string, unknown>> {
56
+ let i = 0;
57
+ return {
58
+ async next() {
59
+ if (i < events.length) return { value: events[i++]!, done: false };
60
+ return { value: undefined as unknown as Record<string, unknown>, done: true };
61
+ },
62
+ };
63
+ },
64
+ };
65
+ }
66
+
67
+ /** Stream that emits one text_delta and then done. */
68
+ function makeTextStream(text: string): AsyncIterable<Record<string, unknown>> {
69
+ const msg = makeAssistantMessage(text);
70
+ return makeEventStream([
71
+ { type: "text_delta", contentIndex: 0, delta: text, partial: msg },
72
+ { type: "done", reason: "stop", message: msg },
73
+ ]);
74
+ }
75
+
76
+ /** Stream that emits only done with empty content. */
77
+ function makeEmptyStream(): AsyncIterable<Record<string, unknown>> {
78
+ const msg = makeAssistantMessage("");
79
+ return makeEventStream([
80
+ { type: "done", reason: "stop", message: msg },
81
+ ]);
82
+ }
83
+
84
+ /** Stream that throws immediately. */
85
+ function makeErrorStream(message: string): AsyncIterable<Record<string, unknown>> {
86
+ return {
87
+ [Symbol.asyncIterator](): AsyncIterator<Record<string, unknown>> {
88
+ return {
89
+ async next() {
90
+ throw new Error(message);
91
+ },
92
+ };
93
+ },
94
+ };
95
+ }
96
+
97
+ const MOCK_PI_MODEL: Model<Api> = {
98
+ id: "test-model",
99
+ name: "test-model",
100
+ api: "openai-completions",
101
+ provider: "openai",
102
+ baseUrl: "https://api.openai.com/v1",
103
+ reasoning: false,
104
+ input: ["text"],
105
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
106
+ contextWindow: 128000,
107
+ maxTokens: 8192,
108
+ };
109
+
110
+ function makeClient(temperature = 0.7, extra: Partial<LLMClient> = {}): LLMClient {
111
+ return {
112
+ provider: "openai",
113
+ service: "openai",
114
+ configSource: "studio",
115
+ apiFormat: "chat",
116
+ stream: true,
117
+ _piModel: MOCK_PI_MODEL,
118
+ _apiKey: "test-key",
119
+ defaults: {
120
+ temperature,
121
+ maxTokens: 512,
122
+ thinkingBudget: 0,
123
+
124
+ extra: {},
125
+ },
126
+ ...extra,
127
+ };
128
+ }
129
+
130
+ async function captureError(task: Promise<unknown>): Promise<Error> {
131
+ try {
132
+ await task;
133
+ } catch (error) {
134
+ return error as Error;
135
+ }
136
+ throw new Error("Expected promise to reject");
137
+ }
138
+
139
+ // ── Tests ─────────────────────────────────────────────────────────────────────
140
+
141
+ describe("chatCompletion via pi-ai", () => {
142
+ beforeEach(() => {
143
+ mockStreamSimple.mockReset();
144
+ mockCompleteSimple.mockReset();
145
+ mockComplete.mockReset();
146
+ });
147
+
148
+ it("returns text content from a successful stream", async () => {
149
+ mockStreamSimple.mockReturnValue(makeTextStream("hello world"));
150
+
151
+ const client = makeClient();
152
+ const result = await chatCompletion(client, "test-model", [
153
+ { role: "user", content: "ping" },
154
+ ]);
155
+
156
+ expect(result.content).toBe("hello world");
157
+ expect(result.usage.promptTokens).toBe(11);
158
+ expect(result.usage.completionTokens).toBe(7);
159
+ expect(result.usage.totalTokens).toBe(18);
160
+ expect(mockStreamSimple).toHaveBeenCalledOnce();
161
+ });
162
+
163
+ it("throws when stream produces no text content", async () => {
164
+ mockStreamSimple.mockReturnValue(makeEmptyStream());
165
+
166
+ const client = makeClient();
167
+ const error = await captureError(
168
+ chatCompletion(client, "test-model", [{ role: "user", content: "ping" }]),
169
+ );
170
+
171
+ expect(error.message).toContain("empty response");
172
+ });
173
+
174
+ it("wraps 400 API errors with a user-friendly message", async () => {
175
+ mockStreamSimple.mockReturnValue(makeErrorStream("400 Bad Request"));
176
+
177
+ const client = makeClient();
178
+ const error = await captureError(
179
+ chatCompletion(client, "test-model", [{ role: "user", content: "ping" }]),
180
+ );
181
+
182
+ expect(error.message).toContain("API 返回 400");
183
+ expect(error.message).toContain("temperature");
184
+ expect(error.message).not.toMatch(/kkaiapi/i);
185
+ });
186
+
187
+ it("wraps 401 errors with an unauthorized message", async () => {
188
+ mockStreamSimple.mockReturnValue(makeErrorStream("401 Unauthorized"));
189
+
190
+ const client = makeClient();
191
+ const error = await captureError(
192
+ chatCompletion(client, "test-model", [{ role: "user", content: "ping" }]),
193
+ );
194
+
195
+ expect(error.message).toContain("API 返回 401");
196
+ });
197
+
198
+ it("wraps connection errors with a friendly message", async () => {
199
+ mockStreamSimple.mockReturnValue(makeErrorStream("fetch failed: ECONNREFUSED"));
200
+
201
+ const client = makeClient();
202
+ const error = await captureError(
203
+ chatCompletion(client, "test-model", [{ role: "user", content: "ping" }]),
204
+ );
205
+
206
+ expect(error.message).toContain("无法连接到 API 服务");
207
+ expect(error.message).not.toMatch(/kkaiapi/i);
208
+ });
209
+
210
+ it("retries transient socket termination errors before failing the chapter pipeline", async () => {
211
+ mockStreamSimple
212
+ .mockReturnValueOnce(makeErrorStream("terminated: UND_ERR_SOCKET other side closed"))
213
+ .mockReturnValueOnce(makeTextStream("recovered"));
214
+
215
+ const client = makeClient();
216
+ const result = await chatCompletion(client, "test-model", [{ role: "user", content: "ping" }]);
217
+
218
+ expect(result.content).toBe("recovered");
219
+ expect(mockStreamSimple).toHaveBeenCalledTimes(2);
220
+ });
221
+
222
+ it("passes temperature and maxTokens to streamSimple", async () => {
223
+ mockStreamSimple.mockReturnValue(makeTextStream("ok"));
224
+
225
+ const client = makeClient(0.5);
226
+ await chatCompletion(client, "test-model", [{ role: "user", content: "hi" }], {
227
+ temperature: 0.3,
228
+ maxTokens: 256,
229
+ });
230
+
231
+ const opts = mockStreamSimple.mock.calls[0]?.[2] as Record<string, unknown>;
232
+ expect(opts.temperature).toBe(0.3);
233
+ expect(opts.maxTokens).toBe(256);
234
+ });
235
+
236
+ it("drops non-ByteString headers before calling pi-ai", async () => {
237
+ mockStreamSimple.mockReturnValue(makeTextStream("ok"));
238
+
239
+ const client = makeClient(0.7, {
240
+ _piModel: {
241
+ ...MOCK_PI_MODEL,
242
+ headers: {
243
+ "X-Valid": "ok",
244
+ "X-Bad": "服务测试",
245
+ },
246
+ },
247
+ });
248
+ await chatCompletion(client, "test-model", [{ role: "user", content: "hi" }]);
249
+
250
+ const opts = mockStreamSimple.mock.calls[0]?.[2] as { headers?: Record<string, string> };
251
+ expect(opts.headers).toMatchObject({ "User-Agent": "InkOS/1.3.5", "X-Valid": "ok" });
252
+ expect(opts.headers).not.toHaveProperty("X-Bad");
253
+ });
254
+
255
+ it("uses client defaults when no per-call overrides are provided", async () => {
256
+ mockStreamSimple.mockReturnValue(makeTextStream("ok"));
257
+
258
+ const client = makeClient(0.8);
259
+ await chatCompletion(client, "test-model", [{ role: "user", content: "hi" }]);
260
+
261
+ const opts = mockStreamSimple.mock.calls[0]?.[2] as Record<string, unknown>;
262
+ expect(opts.temperature).toBe(0.8);
263
+ expect(opts.maxTokens).toBe(512);
264
+ });
265
+
266
+ it("calls onTextDelta for each text chunk", async () => {
267
+ const msg = makeAssistantMessage("abc");
268
+ mockStreamSimple.mockReturnValue(makeEventStream([
269
+ { type: "text_delta", contentIndex: 0, delta: "a", partial: msg },
270
+ { type: "text_delta", contentIndex: 0, delta: "b", partial: msg },
271
+ { type: "text_delta", contentIndex: 0, delta: "c", partial: msg },
272
+ { type: "done", reason: "stop", message: msg },
273
+ ]));
274
+
275
+ const deltas: string[] = [];
276
+ const client = makeClient();
277
+ await chatCompletion(client, "test-model", [{ role: "user", content: "hi" }], {
278
+ onTextDelta: (d) => deltas.push(d),
279
+ });
280
+
281
+ expect(deltas).toEqual(["a", "b", "c"]);
282
+ });
283
+
284
+ it("uses completeSimple when client.stream is false", async () => {
285
+ mockCompleteSimple.mockResolvedValue(makeAssistantMessage("offline hello"));
286
+
287
+ const client = makeClient(0.7, { stream: false });
288
+ const result = await chatCompletion(client, "test-model", [{ role: "user", content: "hi" }]);
289
+
290
+ expect(result.content).toBe("offline hello");
291
+ expect(mockCompleteSimple).toHaveBeenCalledOnce();
292
+ expect(mockStreamSimple).not.toHaveBeenCalled();
293
+ });
294
+
295
+ it("uses native fetch transport for custom openai-compatible chat", async () => {
296
+ const fetchMock = vi.fn().mockResolvedValue({
297
+ ok: true,
298
+ json: async () => ({
299
+ choices: [{ message: { content: "你好!" } }],
300
+ usage: { prompt_tokens: 3, completion_tokens: 2, total_tokens: 5 },
301
+ }),
302
+ });
303
+ vi.stubGlobal("fetch", fetchMock);
304
+
305
+ const client = makeClient(0.7, {
306
+ service: "custom",
307
+ stream: false,
308
+ _piModel: {
309
+ ...MOCK_PI_MODEL,
310
+ provider: "openai",
311
+ baseUrl: "https://gateway.example/v1",
312
+ },
313
+ });
314
+ const result = await chatCompletion(client, "gpt-5.4", [{ role: "user", content: "nihao" }]);
315
+
316
+ expect(result.content).toBe("你好!");
317
+ expect(fetchMock).toHaveBeenCalledOnce();
318
+ expect(mockCompleteSimple).not.toHaveBeenCalled();
319
+ expect(mockStreamSimple).not.toHaveBeenCalled();
320
+
321
+ vi.unstubAllGlobals();
322
+ });
323
+
324
+ it("rejects non-ASCII API keys before native custom fetch builds headers", async () => {
325
+ const fetchMock = vi.fn();
326
+ vi.stubGlobal("fetch", fetchMock);
327
+
328
+ const client = makeClient(0.7, {
329
+ service: "custom",
330
+ stream: false,
331
+ _apiKey: "sk-test测试",
332
+ _piModel: {
333
+ ...MOCK_PI_MODEL,
334
+ provider: "openai",
335
+ baseUrl: "https://gateway.example/v1",
336
+ },
337
+ });
338
+
339
+ const error = await captureError(
340
+ chatCompletion(client, "gpt-5.4", [{ role: "user", content: "ping" }]),
341
+ );
342
+
343
+ expect(error.message).toContain("non-ASCII");
344
+ expect(fetchMock).not.toHaveBeenCalled();
345
+ vi.unstubAllGlobals();
346
+ });
347
+
348
+ it("uses native fetch transport for kkaiapi chat and sanitizes headers", async () => {
349
+ const fetchMock = vi.fn().mockResolvedValue({
350
+ ok: true,
351
+ json: async () => ({
352
+ choices: [{ message: { content: "kkai ok" } }],
353
+ usage: { prompt_tokens: 3, completion_tokens: 2, total_tokens: 5 },
354
+ }),
355
+ });
356
+ vi.stubGlobal("fetch", fetchMock);
357
+
358
+ const client = makeClient(0.7, {
359
+ service: "kkaiapi",
360
+ stream: false,
361
+ _piModel: {
362
+ ...MOCK_PI_MODEL,
363
+ provider: "openai",
364
+ baseUrl: "https://api.kkaiapi.com/v1",
365
+ headers: {
366
+ "X-Valid": "ok",
367
+ "X-Bad": "服务测试",
368
+ },
369
+ },
370
+ });
371
+ const result = await chatCompletion(client, "deepseek-v4-flash", [{ role: "user", content: "nihao" }]);
372
+
373
+ expect(result.content).toBe("kkai ok");
374
+ expect(fetchMock).toHaveBeenCalledOnce();
375
+ expect(mockCompleteSimple).not.toHaveBeenCalled();
376
+ expect(mockStreamSimple).not.toHaveBeenCalled();
377
+
378
+ const init = fetchMock.mock.calls[0]?.[1] as { headers?: Record<string, string> };
379
+ expect(init.headers).toMatchObject({
380
+ Authorization: "Bearer test-key",
381
+ "Content-Type": "application/json",
382
+ "X-Valid": "ok",
383
+ });
384
+ expect(init.headers).not.toHaveProperty("X-Bad");
385
+
386
+ vi.unstubAllGlobals();
387
+ });
388
+
389
+ it("does not leave a stream monitor timer after native non-stream chat", async () => {
390
+ vi.useFakeTimers();
391
+ const fetchMock = vi.fn().mockResolvedValue({
392
+ ok: true,
393
+ json: async () => ({
394
+ choices: [{ message: { content: "你好!" } }],
395
+ usage: { prompt_tokens: 3, completion_tokens: 2, total_tokens: 5 },
396
+ }),
397
+ });
398
+ vi.stubGlobal("fetch", fetchMock);
399
+
400
+ const client = makeClient(0.7, {
401
+ service: "custom",
402
+ stream: false,
403
+ _piModel: {
404
+ ...MOCK_PI_MODEL,
405
+ provider: "openai",
406
+ baseUrl: "https://gateway.example/v1",
407
+ },
408
+ });
409
+ const result = await chatCompletion(client, "gpt-5.4", [{ role: "user", content: "nihao" }], {
410
+ onStreamProgress: vi.fn(),
411
+ });
412
+
413
+ expect(result.content).toBe("你好!");
414
+ expect(vi.getTimerCount()).toBe(0);
415
+
416
+ vi.unstubAllGlobals();
417
+ vi.useRealTimers();
418
+ });
419
+
420
+ it("attaches a proxy dispatcher for custom openai-compatible chat when proxyUrl is configured", async () => {
421
+ const fetchMock = vi.fn().mockResolvedValue({
422
+ ok: true,
423
+ json: async () => ({
424
+ choices: [{ message: { content: "proxied" } }],
425
+ usage: { prompt_tokens: 3, completion_tokens: 2, total_tokens: 5 },
426
+ }),
427
+ });
428
+ vi.stubGlobal("fetch", fetchMock);
429
+
430
+ const client = makeClient(0.7, {
431
+ service: "custom",
432
+ stream: false,
433
+ proxyUrl: "http://127.0.0.1:9910",
434
+ _piModel: {
435
+ ...MOCK_PI_MODEL,
436
+ provider: "openai",
437
+ baseUrl: "https://gateway.example/v1",
438
+ },
439
+ });
440
+ const result = await chatCompletion(client, "gpt-5.4", [{ role: "user", content: "nihao" }]);
441
+
442
+ expect(result.content).toBe("proxied");
443
+ expect(fetchMock).toHaveBeenCalledOnce();
444
+ expect(fetchMock.mock.calls[0]?.[1]).toMatchObject({
445
+ method: "POST",
446
+ dispatcher: expect.any(Object),
447
+ });
448
+
449
+ vi.unstubAllGlobals();
450
+ });
451
+
452
+ it("uses reasoning_content for custom openai-compatible non-stream responses that omit content", async () => {
453
+ const fetchMock = vi.fn().mockResolvedValue({
454
+ ok: true,
455
+ json: async () => ({
456
+ choices: [{ message: { reasoning_content: "推理通道文本" } }],
457
+ usage: { prompt_tokens: 3, completion_tokens: 2, total_tokens: 5 },
458
+ }),
459
+ });
460
+ vi.stubGlobal("fetch", fetchMock);
461
+
462
+ const client = makeClient(0.7, {
463
+ service: "custom",
464
+ stream: false,
465
+ _piModel: {
466
+ ...MOCK_PI_MODEL,
467
+ provider: "openai",
468
+ baseUrl: "https://gateway.example/v1",
469
+ },
470
+ });
471
+ const result = await chatCompletion(client, "glm-compat", [{ role: "user", content: "nihao" }]);
472
+
473
+ expect(result.content).toBe("推理通道文本");
474
+ expect(fetchMock).toHaveBeenCalledOnce();
475
+
476
+ vi.unstubAllGlobals();
477
+ });
478
+
479
+ it("uses reasoning_content for custom openai-compatible streams that omit content deltas", async () => {
480
+ const encoder = new TextEncoder();
481
+ const sse = [
482
+ "data: {\"choices\":[{\"delta\":{\"reasoning_content\":\"你\"}}]}\n\n",
483
+ "data: {\"choices\":[{\"delta\":{\"reasoning_content\":\"好\"}}]}\n\n",
484
+ "data: {\"usage\":{\"prompt_tokens\":3,\"completion_tokens\":2,\"total_tokens\":5}}\n\n",
485
+ "data: [DONE]\n\n",
486
+ ].join("");
487
+ const fetchMock = vi.fn().mockResolvedValue({
488
+ ok: true,
489
+ body: new ReadableStream({
490
+ start(controller) {
491
+ controller.enqueue(encoder.encode(sse));
492
+ controller.close();
493
+ },
494
+ }),
495
+ });
496
+ vi.stubGlobal("fetch", fetchMock);
497
+
498
+ const client = makeClient(0.7, {
499
+ service: "custom",
500
+ stream: true,
501
+ _piModel: {
502
+ ...MOCK_PI_MODEL,
503
+ provider: "openai",
504
+ baseUrl: "https://gateway.example/v1",
505
+ },
506
+ });
507
+ const result = await chatCompletion(client, "glm-compat", [{ role: "user", content: "nihao" }]);
508
+
509
+ expect(result.content).toBe("你好");
510
+ expect(result.usage.totalTokens).toBe(5);
511
+ expect(fetchMock).toHaveBeenCalledOnce();
512
+
513
+ vi.unstubAllGlobals();
514
+ });
515
+
516
+ it("retries custom openai-compatible chat by folding system messages into user when system role is unsupported", async () => {
517
+ const fetchMock = vi.fn()
518
+ .mockResolvedValueOnce({
519
+ ok: false,
520
+ status: 400,
521
+ statusText: "Bad Request",
522
+ text: async () => JSON.stringify({ error: { message: "role system is unsupported" } }),
523
+ })
524
+ .mockResolvedValueOnce({
525
+ ok: true,
526
+ json: async () => ({
527
+ choices: [{ message: { content: "ok" } }],
528
+ usage: { prompt_tokens: 9, completion_tokens: 1, total_tokens: 10 },
529
+ }),
530
+ });
531
+ vi.stubGlobal("fetch", fetchMock);
532
+
533
+ const client = makeClient(0.7, {
534
+ service: "custom",
535
+ stream: false,
536
+ _piModel: {
537
+ ...MOCK_PI_MODEL,
538
+ provider: "openai",
539
+ baseUrl: "https://gateway.example/v1",
540
+ },
541
+ });
542
+ const result = await chatCompletion(client, "wild-compatible", [
543
+ { role: "system", content: "只输出中文。" },
544
+ { role: "user", content: "ping" },
545
+ ]);
546
+
547
+ expect(result.content).toBe("ok");
548
+ expect(fetchMock).toHaveBeenCalledTimes(2);
549
+ const firstBody = JSON.parse(fetchMock.mock.calls[0]?.[1]?.body as string);
550
+ const secondBody = JSON.parse(fetchMock.mock.calls[1]?.[1]?.body as string);
551
+ expect(firstBody.messages).toEqual([
552
+ { role: "system", content: "只输出中文。" },
553
+ { role: "user", content: "ping" },
554
+ ]);
555
+ expect(secondBody.messages).toHaveLength(1);
556
+ expect(secondBody.messages[0]).toMatchObject({ role: "user" });
557
+ expect(secondBody.messages[0].content).toContain("只输出中文。");
558
+ expect(secondBody.messages[0].content).toContain("ping");
559
+
560
+ vi.unstubAllGlobals();
561
+ });
562
+
563
+ it("keeps legacy env custom openai-compatible chat on pi-ai path", async () => {
564
+ const fetchMock = vi.fn();
565
+ vi.stubGlobal("fetch", fetchMock);
566
+ mockCompleteSimple.mockResolvedValue(makeAssistantMessage("legacy ok"));
567
+
568
+ const client = makeClient(0.7, {
569
+ service: "custom",
570
+ configSource: "env",
571
+ stream: false,
572
+ _piModel: {
573
+ ...MOCK_PI_MODEL,
574
+ provider: "openai",
575
+ baseUrl: "https://generativelanguage.googleapis.com/v1beta/openai",
576
+ },
577
+ });
578
+
579
+ const result = await chatCompletion(client, "gemma-4", [{ role: "user", content: "ping" }]);
580
+
581
+ expect(result.content).toBe("legacy ok");
582
+ expect(mockCompleteSimple).toHaveBeenCalledOnce();
583
+ expect(fetchMock).not.toHaveBeenCalled();
584
+
585
+ vi.unstubAllGlobals();
586
+ });
587
+
588
+ it("uses native fetch transport for local Ollama without an API key", async () => {
589
+ const fetchMock = vi.fn().mockResolvedValue({
590
+ ok: true,
591
+ json: async () => ({
592
+ choices: [{ message: { content: "本地 Ollama 可用" } }],
593
+ usage: { prompt_tokens: 3, completion_tokens: 4, total_tokens: 7 },
594
+ }),
595
+ });
596
+ vi.stubGlobal("fetch", fetchMock);
597
+
598
+ const client = makeClient(0.7, {
599
+ service: "ollama",
600
+ configSource: "env",
601
+ stream: false,
602
+ _apiKey: "",
603
+ _piModel: {
604
+ ...MOCK_PI_MODEL,
605
+ provider: "ollama",
606
+ baseUrl: "http://127.0.0.1:11434/v1",
607
+ },
608
+ });
609
+ const result = await chatCompletion(client, "Qwen3.6-35B-A3B-APEX-I-Mini.gguf", [
610
+ { role: "user", content: "ping" },
611
+ ]);
612
+
613
+ expect(result.content).toBe("本地 Ollama 可用");
614
+ expect(fetchMock).toHaveBeenCalledOnce();
615
+ expect(mockCompleteSimple).not.toHaveBeenCalled();
616
+
617
+ vi.unstubAllGlobals();
618
+ });
619
+
620
+ it("uses native fetch transport for local custom OpenAI-compatible endpoints without an API key", async () => {
621
+ const fetchMock = vi.fn().mockResolvedValue({
622
+ ok: true,
623
+ json: async () => ({
624
+ choices: [{ message: { content: "本地自定义端点可用" } }],
625
+ usage: { prompt_tokens: 5, completion_tokens: 6, total_tokens: 11 },
626
+ }),
627
+ });
628
+ vi.stubGlobal("fetch", fetchMock);
629
+
630
+ const client = makeClient(0.7, {
631
+ service: "custom",
632
+ configSource: "env",
633
+ stream: false,
634
+ _apiKey: "",
635
+ _piModel: {
636
+ ...MOCK_PI_MODEL,
637
+ provider: "openai",
638
+ baseUrl: "http://127.0.0.1:11434/v1",
639
+ },
640
+ });
641
+ const result = await chatCompletion(client, "local-qwen", [{ role: "user", content: "ping" }]);
642
+
643
+ expect(result.content).toBe("本地自定义端点可用");
644
+ expect(fetchMock).toHaveBeenCalledOnce();
645
+ expect(fetchMock.mock.calls[0]?.[1]?.headers).not.toHaveProperty("Authorization");
646
+ expect(mockCompleteSimple).not.toHaveBeenCalled();
647
+
648
+ vi.unstubAllGlobals();
649
+ });
650
+
651
+ it("uses native fetch transport for custom anthropic-compatible non-stream chat", async () => {
652
+ const fetchMock = vi.fn().mockResolvedValue({
653
+ ok: true,
654
+ json: async () => ({
655
+ content: [{ type: "text", text: "你好,Anthropic!" }],
656
+ usage: { input_tokens: 5, output_tokens: 3 },
657
+ }),
658
+ });
659
+ vi.stubGlobal("fetch", fetchMock);
660
+
661
+ const client = makeClient(0.7, {
662
+ provider: "anthropic",
663
+ service: "custom",
664
+ stream: false,
665
+ _piModel: {
666
+ ...MOCK_PI_MODEL,
667
+ provider: "anthropic",
668
+ api: "anthropic-messages" as Api,
669
+ baseUrl: "https://gateway.example",
670
+ },
671
+ });
672
+ const result = await chatCompletion(client, "claude-sonnet-4-6", [{ role: "user", content: "nihao" }]);
673
+
674
+ expect(result.content).toBe("你好,Anthropic!");
675
+ expect(result.usage.promptTokens).toBe(5);
676
+ expect(result.usage.completionTokens).toBe(3);
677
+ expect(fetchMock).toHaveBeenCalledOnce();
678
+ expect(mockCompleteSimple).not.toHaveBeenCalled();
679
+ expect(mockStreamSimple).not.toHaveBeenCalled();
680
+
681
+ vi.unstubAllGlobals();
682
+ });
683
+
684
+ it("uses native fetch transport for custom anthropic-compatible stream chat", async () => {
685
+ const encoder = new TextEncoder();
686
+ const sse = [
687
+ "event: message_start\n",
688
+ "data: {\"type\":\"message_start\",\"message\":{\"usage\":{\"input_tokens\":4}}}\n\n",
689
+ "event: content_block_delta\n",
690
+ "data: {\"type\":\"content_block_delta\",\"delta\":{\"type\":\"text_delta\",\"text\":\"你\"}}\n\n",
691
+ "event: content_block_delta\n",
692
+ "data: {\"type\":\"content_block_delta\",\"delta\":{\"type\":\"text_delta\",\"text\":\"好\"}}\n\n",
693
+ "event: message_delta\n",
694
+ "data: {\"type\":\"message_delta\",\"usage\":{\"output_tokens\":2}}\n\n",
695
+ "event: message_stop\n",
696
+ "data: {\"type\":\"message_stop\"}\n\n",
697
+ ].join("");
698
+ const fetchMock = vi.fn().mockResolvedValue({
699
+ ok: true,
700
+ body: new ReadableStream({
701
+ start(controller) {
702
+ controller.enqueue(encoder.encode(sse));
703
+ controller.close();
704
+ },
705
+ }),
706
+ });
707
+ vi.stubGlobal("fetch", fetchMock);
708
+
709
+ const client = makeClient(0.7, {
710
+ provider: "anthropic",
711
+ service: "custom",
712
+ stream: true,
713
+ _piModel: {
714
+ ...MOCK_PI_MODEL,
715
+ provider: "anthropic",
716
+ api: "anthropic-messages" as Api,
717
+ baseUrl: "https://gateway.example",
718
+ },
719
+ });
720
+ const result = await chatCompletion(client, "claude-sonnet-4-6", [{ role: "user", content: "nihao" }]);
721
+
722
+ expect(result.content).toBe("你好");
723
+ expect(result.usage.promptTokens).toBe(4);
724
+ expect(result.usage.completionTokens).toBe(2);
725
+ expect(result.usage.totalTokens).toBe(6);
726
+ expect(fetchMock).toHaveBeenCalledOnce();
727
+ expect(mockCompleteSimple).not.toHaveBeenCalled();
728
+ expect(mockStreamSimple).not.toHaveBeenCalled();
729
+
730
+ vi.unstubAllGlobals();
731
+ });
732
+ });
733
+
734
+ describe("chatCompletion fixed-temperature clamp (thinking models)", () => {
735
+ beforeEach(() => {
736
+ __resetFixedTemperatureWarnings();
737
+ mockStreamSimple.mockReset();
738
+ mockStreamSimple.mockReturnValue(makeTextStream("ok"));
739
+ });
740
+
741
+ afterEach(() => {
742
+ vi.restoreAllMocks();
743
+ });
744
+
745
+ it("forces temperature=1 for kimi-k2.5 even when client default is 0.7", async () => {
746
+ const client = makeClient(0.7);
747
+ const warn = vi.spyOn(console, "warn").mockImplementation(() => {});
748
+
749
+ await chatCompletion(client, "kimi-k2.5", [{ role: "user", content: "hi" }]);
750
+
751
+ const opts = mockStreamSimple.mock.calls[0]?.[2] as Record<string, unknown>;
752
+ expect(opts.temperature).toBe(1);
753
+ expect(warn).toHaveBeenCalledOnce();
754
+ expect(warn.mock.calls[0]?.[0]).toContain("kimi-k2.5");
755
+ warn.mockRestore();
756
+ });
757
+
758
+ it("clamps per-call temperature override (0.3) to 1 for kimi-k2.5", async () => {
759
+ const client = makeClient(0.7);
760
+ vi.spyOn(console, "warn").mockImplementation(() => {});
761
+
762
+ await chatCompletion(
763
+ client,
764
+ "kimi-k2.5",
765
+ [{ role: "user", content: "hi" }],
766
+ { temperature: 0.3 },
767
+ );
768
+
769
+ const opts = mockStreamSimple.mock.calls[0]?.[2] as Record<string, unknown>;
770
+ expect(opts.temperature).toBe(1);
771
+ });
772
+
773
+ it("only warns once per model name across multiple calls", async () => {
774
+ const client = makeClient(0.7);
775
+ const warn = vi.spyOn(console, "warn").mockImplementation(() => {});
776
+
777
+ await chatCompletion(client, "kimi-k2.5", [{ role: "user", content: "a" }]);
778
+ await chatCompletion(client, "kimi-k2.5", [{ role: "user", content: "b" }]);
779
+ await chatCompletion(client, "kimi-k2.5", [{ role: "user", content: "c" }]);
780
+
781
+ expect(warn).toHaveBeenCalledOnce();
782
+ warn.mockRestore();
783
+ });
784
+
785
+ it("clamps kimi-k2-thinking (bank-marked temperature:1)", async () => {
786
+ const client = makeClient(0.5);
787
+ vi.spyOn(console, "warn").mockImplementation(() => {});
788
+
789
+ await chatCompletion(client, "kimi-k2-thinking", [
790
+ { role: "user", content: "hi" },
791
+ ]);
792
+
793
+ const opts = mockStreamSimple.mock.calls[0]?.[2] as Record<string, unknown>;
794
+ expect(opts.temperature).toBe(1);
795
+ });
796
+
797
+ it("leaves regular models untouched (no clamp, no warning)", async () => {
798
+ const client = makeClient(0.7);
799
+ const warn = vi.spyOn(console, "warn").mockImplementation(() => {});
800
+
801
+ await chatCompletion(
802
+ client,
803
+ "moonshot-v1-32k",
804
+ [{ role: "user", content: "hi" }],
805
+ { temperature: 0.3 },
806
+ );
807
+
808
+ const opts = mockStreamSimple.mock.calls[0]?.[2] as Record<string, unknown>;
809
+ expect(opts.temperature).toBe(0.3);
810
+ expect(warn).not.toHaveBeenCalled();
811
+ warn.mockRestore();
812
+ });
813
+
814
+ it("does not warn when requested temperature is already 1", async () => {
815
+ const client = makeClient(1);
816
+ const warn = vi.spyOn(console, "warn").mockImplementation(() => {});
817
+
818
+ await chatCompletion(client, "kimi-k2.5", [{ role: "user", content: "hi" }]);
819
+
820
+ const opts = mockStreamSimple.mock.calls[0]?.[2] as Record<string, unknown>;
821
+ expect(opts.temperature).toBe(1);
822
+ expect(warn).not.toHaveBeenCalled();
823
+ warn.mockRestore();
824
+ });
825
+ });
826
+
827
+ // ── 回归测试:per-call maxTokens 不被裁剪(v2.0.0 精简版)─────────────
828
+ // 背景:v2.0.0 删除了 config.maxTokens / maxTokensCap 字段,provider 层不再做 cap,
829
+ // agent per-call 传的 maxTokens 原样透传到下游。
830
+ describe("createLLMClient per-call maxTokens not capped (v2.0.0)", () => {
831
+ it("per-call maxTokens 16384 reaches the API as-is", async () => {
832
+ const { createLLMClient } = await import("../llm/provider.js");
833
+ const { LLMConfigSchema } = await import("../models/project.js");
834
+
835
+ const client = createLLMClient(LLMConfigSchema.parse({
836
+ provider: "openai",
837
+ baseUrl: "http://localhost:0",
838
+ model: "test-model",
839
+ apiKey: "test-key",
840
+ }));
841
+
842
+ mockStreamSimple.mockReset();
843
+ mockStreamSimple.mockReturnValue(makeTextStream("ok"));
844
+
845
+ await chatCompletion(client, "test-model", [
846
+ { role: "user", content: "architect" },
847
+ ], { maxTokens: 16384 });
848
+
849
+ const opts = mockStreamSimple.mock.calls[0]?.[2] as Record<string, unknown>;
850
+ expect(opts.maxTokens).toBe(16384);
851
+ });
852
+ });
853
+
854
+ describe("createLLMClient with providers lookup", () => {
855
+ it("anthropic + claude-sonnet-4-6 拿到 modelCard 的 maxOutput (64000),不是未知模型兜底", async () => {
856
+ const { createLLMClient } = await import("../llm/provider.js");
857
+ const { LLMConfigSchema } = await import("../models/project.js");
858
+ const client = createLLMClient(LLMConfigSchema.parse({
859
+ provider: "anthropic",
860
+ service: "anthropic",
861
+ model: "claude-sonnet-4-6",
862
+ apiKey: "test",
863
+ baseUrl: "https://api.anthropic.com",
864
+ }));
865
+ expect(client.defaults.maxTokens).toBe(64_000);
866
+ expect(client._piModel?.maxTokens).toBe(64_000);
867
+ expect(client._piModel?.contextWindow).toBe(1_000_000);
868
+ });
869
+
870
+ it("custom service + gpt-4o 靠 Layer 2 全局扫命中 openai provider", async () => {
871
+ const { createLLMClient } = await import("../llm/provider.js");
872
+ const { LLMConfigSchema } = await import("../models/project.js");
873
+ const client = createLLMClient(LLMConfigSchema.parse({
874
+ provider: "openai",
875
+ service: "custom",
876
+ model: "gpt-4o",
877
+ apiKey: "test",
878
+ baseUrl: "https://middleman.example/v1",
879
+ }));
880
+ // lobe 数据里 gpt-4o maxOutput=4096
881
+ expect(client.defaults.maxTokens).toBe(4096);
882
+ });
883
+
884
+ it("未知 model 走 8192 * 3 的写作兜底预算", async () => {
885
+ const { createLLMClient } = await import("../llm/provider.js");
886
+ const { LLMConfigSchema } = await import("../models/project.js");
887
+ const client = createLLMClient(LLMConfigSchema.parse({
888
+ provider: "openai",
889
+ service: "custom",
890
+ model: "my-private-xyz-model-does-not-exist",
891
+ apiKey: "test",
892
+ baseUrl: "https://middleman.example/v1",
893
+ }));
894
+ expect(client.defaults.maxTokens).toBe(24_576);
895
+ expect(client._piModel?.maxTokens).toBe(24_576);
896
+ });
897
+
898
+ it("config.maxTokens 命中 modelCard 后被覆盖(用户填 4000 还是用 modelCard 的 64000)", async () => {
899
+ const { createLLMClient } = await import("../llm/provider.js");
900
+ const { LLMConfigSchema } = await import("../models/project.js");
901
+ const client = createLLMClient(LLMConfigSchema.parse({
902
+ provider: "anthropic",
903
+ service: "anthropic",
904
+ model: "claude-sonnet-4-6",
905
+ apiKey: "test",
906
+ baseUrl: "https://api.anthropic.com",
907
+ maxTokens: 4000,
908
+ }));
909
+ expect(client.defaults.maxTokens).toBe(64_000);
910
+ });
911
+
912
+ it("B7: kimiCodingPlan 的 kimi-k2.5 走 API 时 piModel.id 是 deploymentName (k2p5)", async () => {
913
+ const { createLLMClient } = await import("../llm/provider.js");
914
+ const { LLMConfigSchema } = await import("../models/project.js");
915
+ const client = createLLMClient(LLMConfigSchema.parse({
916
+ provider: "anthropic",
917
+ service: "kimiCodingPlan",
918
+ model: "kimi-k2.5",
919
+ apiKey: "test",
920
+ baseUrl: "https://api.moonshot.cn/anthropic",
921
+ }));
922
+ expect(client._piModel?.id).toBe("k2p5");
923
+ });
924
+
925
+ it("B7: 没有 deploymentName 的 model piModel.id 保持原 config.model", async () => {
926
+ const { createLLMClient } = await import("../llm/provider.js");
927
+ const { LLMConfigSchema } = await import("../models/project.js");
928
+ const client = createLLMClient(LLMConfigSchema.parse({
929
+ provider: "anthropic",
930
+ service: "kimiCodingPlan",
931
+ model: "kimi-k2-thinking",
932
+ apiKey: "test",
933
+ baseUrl: "https://api.moonshot.cn/anthropic",
934
+ }));
935
+ expect(client._piModel?.id).toBe("kimi-k2-thinking");
936
+ });
937
+
938
+ it("Google Gemini uses native google-generative-ai provider", async () => {
939
+ const { createLLMClient } = await import("../llm/provider.js");
940
+ const { LLMConfigSchema } = await import("../models/project.js");
941
+ const client = createLLMClient(LLMConfigSchema.parse({
942
+ provider: "openai",
943
+ service: "google",
944
+ model: "gemini-2.5-flash",
945
+ apiKey: "test",
946
+ baseUrl: "https://generativelanguage.googleapis.com/v1beta",
947
+ }));
948
+ expect(client._piModel?.api).toBe("google-generative-ai");
949
+ expect(client._piModel?.provider).toBe("google");
950
+ expect(client._piModel?.baseUrl).toBe("https://generativelanguage.googleapis.com/v1beta");
951
+ expect(client._piModel?.compat).toBeUndefined();
952
+ });
953
+ });