nolo-cli 0.1.13 → 0.1.14

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 (320) hide show
  1. package/README.md +9 -2
  2. package/agent-runtime/hostAdapter.ts +53 -0
  3. package/agent-runtime/index.ts +28 -0
  4. package/agent-runtime/localLoop.ts +62 -0
  5. package/agent-runtime/runtimeDecision.ts +70 -0
  6. package/agent-runtime/types.ts +87 -0
  7. package/agentRuntimeCommands.ts +139 -22
  8. package/agentRuntimeLocal.ts +7 -0
  9. package/ai/agent/_executeModel.ts +118 -0
  10. package/ai/agent/agentSlice.ts +544 -1
  11. package/ai/agent/appWorkingMemory.ts +126 -0
  12. package/ai/agent/avatarUtils.ts +24 -0
  13. package/ai/agent/buildEditingContext.ts +373 -0
  14. package/ai/agent/buildSystemPrompt.ts +532 -0
  15. package/ai/agent/cleanAgentMessages.ts +140 -0
  16. package/ai/agent/cliChatClient.ts +119 -0
  17. package/ai/agent/contextCompiler.ts +107 -0
  18. package/ai/agent/contextLayerContract.ts +44 -0
  19. package/ai/agent/createAgentSchema.ts +234 -0
  20. package/ai/agent/executeToolCall.ts +58 -0
  21. package/ai/agent/fetchAgentContexts.ts +42 -0
  22. package/ai/agent/generatePrompt.ts +3 -0
  23. package/ai/agent/getFullChatContextKeys.ts +168 -0
  24. package/ai/agent/hooks/fetchPublicAgents.ts +133 -0
  25. package/ai/agent/hooks/useAgentConfig.ts +61 -0
  26. package/ai/agent/hooks/useAgentDialog.ts +35 -0
  27. package/ai/agent/hooks/useAgentFormValidation.ts +202 -0
  28. package/ai/agent/hooks/usePublicAgents.ts +473 -0
  29. package/ai/agent/persistMessageWithFixedId.ts +37 -0
  30. package/ai/agent/planSlice.ts +259 -0
  31. package/ai/agent/referenceUtils.ts +229 -0
  32. package/ai/agent/runAgentBackground.ts +238 -0
  33. package/ai/agent/runAgentClientLoop.ts +138 -0
  34. package/ai/agent/runtimeGuidance.ts +97 -0
  35. package/ai/agent/runtimeServerBase.ts +37 -0
  36. package/ai/agent/server/fetchPublicAgents.ts +128 -0
  37. package/ai/agent/startParallelAgentStreams.ts +424 -0
  38. package/ai/agent/startupProtocol.ts +53 -0
  39. package/ai/agent/streamAgentChatTurn.ts +1299 -0
  40. package/ai/agent/streamAgentChatTurnUtils.ts +738 -0
  41. package/ai/agent/types.ts +71 -0
  42. package/ai/agent/utils/imageOutput.ts +39 -0
  43. package/ai/agent/utils/publicImageAgentMode.ts +26 -0
  44. package/ai/agent/utils/sortUtils.ts +250 -0
  45. package/ai/agent/web/referencePickerUtils.ts +146 -0
  46. package/ai/ai.locale.ts +1083 -0
  47. package/ai/chat/accumulateToolCallChunks.ts +95 -0
  48. package/ai/chat/fetchUtils.native.ts +276 -0
  49. package/ai/chat/fetchUtils.ts +153 -0
  50. package/ai/chat/inlineImageUrlsForCustomProvider.ts +117 -0
  51. package/ai/chat/parseApiError.ts +64 -0
  52. package/ai/chat/parseMultilineSSE.ts +95 -0
  53. package/ai/chat/sendOpenAICompletionsRequest.native.ts +682 -0
  54. package/ai/chat/sendOpenAICompletionsRequest.ts +712 -0
  55. package/ai/chat/sendOpenAIResponseRequest.ts +512 -0
  56. package/ai/chat/shouldUseServerProxy.ts +18 -0
  57. package/ai/chat/sseClient.native.ts +91 -0
  58. package/ai/chat/sseClient.ts +67 -0
  59. package/ai/chat/streamReader.native.ts +31 -0
  60. package/ai/chat/streamReader.ts +62 -0
  61. package/ai/chat/updateTotalUsage.ts +72 -0
  62. package/ai/context/buildReferenceContext.ts +437 -0
  63. package/ai/context/calculateContextUsage.ts +133 -0
  64. package/ai/context/retention.ts +165 -0
  65. package/ai/context/tokenUtils.ts +78 -0
  66. package/ai/index.ts +1 -1
  67. package/ai/llm/agentCapabilities.ts +74 -0
  68. package/ai/llm/calculateGeminiImageTokens.ts +57 -0
  69. package/ai/llm/deepinfra.ts +28 -0
  70. package/ai/llm/fireworks.ts +68 -0
  71. package/ai/llm/generateRequestBody.ts +165 -0
  72. package/ai/llm/getModelContextWindow.ts +84 -0
  73. package/ai/llm/getNoloKey.ts +37 -0
  74. package/ai/llm/getPricing.ts +232 -0
  75. package/ai/llm/hooks/useModelPricing.ts +75 -0
  76. package/ai/llm/imagePricing.ts +66 -0
  77. package/ai/llm/isResponseAPIModel.ts +13 -0
  78. package/ai/llm/kimi.ts +18 -0
  79. package/ai/llm/mimo.ts +71 -0
  80. package/ai/llm/mistral.ts +22 -0
  81. package/ai/llm/modelAvatar.ts +427 -0
  82. package/ai/llm/models.ts +45 -0
  83. package/ai/llm/openrouterModels.ts +141 -0
  84. package/ai/llm/providers.ts +307 -0
  85. package/ai/llm/reasoningModels.ts +28 -0
  86. package/ai/llm/types.ts +59 -0
  87. package/ai/llm/usageRequestOptions.ts +59 -0
  88. package/ai/memory/capture.ts +148 -0
  89. package/ai/memory/consolidate.ts +104 -0
  90. package/ai/memory/delete.ts +147 -0
  91. package/ai/memory/overlay.ts +84 -0
  92. package/ai/memory/query.ts +38 -0
  93. package/ai/memory/queryShared.ts +160 -0
  94. package/ai/memory/rank.ts +105 -0
  95. package/ai/memory/recentRelationshipRecap.ts +247 -0
  96. package/ai/memory/remember.ts +167 -0
  97. package/ai/memory/runtime.ts +76 -0
  98. package/ai/memory/store.ts +20 -0
  99. package/ai/memory/storeShared.ts +76 -0
  100. package/ai/memory/types.ts +46 -0
  101. package/ai/memory/understanding.ts +349 -0
  102. package/ai/memory/understandingGreeting.ts +264 -0
  103. package/ai/messages/type.ts +20 -0
  104. package/ai/policy/personalizationDialog.ts +333 -0
  105. package/ai/policy/runtimePolicy.ts +440 -0
  106. package/ai/policy/selfUpdateFields.ts +48 -0
  107. package/ai/policy/types.ts +64 -0
  108. package/ai/skills/referenceRuntime.ts +274 -0
  109. package/ai/skills/skillDiagnostics.ts +251 -0
  110. package/ai/skills/skillDocBuilder.ts +139 -0
  111. package/ai/skills/skillDocProtocol.ts +434 -0
  112. package/ai/skills/skillReferenceSummary.ts +63 -0
  113. package/ai/skills/skillSummaryMarker.ts +26 -0
  114. package/ai/token/calculatePrice.ts +546 -0
  115. package/ai/token/db.ts +98 -0
  116. package/ai/token/externalToolCost.ts +321 -0
  117. package/ai/token/hooks/useRecords.ts +65 -0
  118. package/ai/token/missingUsageEstimate.ts +42 -0
  119. package/ai/token/modelUsageQuery.ts +252 -0
  120. package/ai/token/normalizeUsage.ts +84 -0
  121. package/ai/token/openaiImageGenerationUsage.ts +56 -0
  122. package/ai/token/prepareTokenUsageData.ts +88 -0
  123. package/ai/token/query.ts +88 -0
  124. package/ai/token/queryUserTokens.ts +59 -0
  125. package/ai/token/resolveBillingTarget.ts +52 -0
  126. package/ai/token/saveTokenRecord.ts +53 -0
  127. package/ai/token/serverDialogProjection.ts +78 -0
  128. package/ai/token/serverTokenWriter.ts +143 -0
  129. package/ai/token/stats.ts +21 -0
  130. package/ai/token/tokenThunks.ts +24 -0
  131. package/ai/token/types.ts +93 -0
  132. package/ai/tools/agent/agentTools.ts +176 -0
  133. package/ai/tools/agent/agentUpdateShared.ts +311 -0
  134. package/ai/tools/agent/callAgentTool.ts +139 -0
  135. package/ai/tools/agent/createAgentTool.ts +512 -0
  136. package/ai/tools/agent/createDialogTool.ts +69 -0
  137. package/ai/tools/agent/createSkillAgentTool.ts +62 -0
  138. package/ai/tools/agent/parallelBudget.ts +221 -0
  139. package/ai/tools/agent/presets/appBuilderPreset.ts +147 -0
  140. package/ai/tools/agent/runLlmTool.ts +96 -0
  141. package/ai/tools/agent/runStreamingAgentTool.ts +73 -0
  142. package/ai/tools/agent/skillAgentArgs.ts +106 -0
  143. package/ai/tools/agent/skillAgentPreset.ts +89 -0
  144. package/ai/tools/agent/streamParallelAgentsTool.ts +122 -0
  145. package/ai/tools/agent/updateAgentTool.ts +96 -0
  146. package/ai/tools/agent/updateSelfTool.ts +113 -0
  147. package/ai/tools/amazonProductScraperTool.ts +86 -0
  148. package/ai/tools/apifyActorClient.ts +45 -0
  149. package/ai/tools/appEditGuard.ts +372 -0
  150. package/ai/tools/appReadSnapshot.ts +153 -0
  151. package/ai/tools/appTools.ts +1549 -0
  152. package/ai/tools/applyEditTool.ts +256 -0
  153. package/ai/tools/applyLineEditsTool.ts +312 -0
  154. package/ai/tools/browserTools/click.ts +33 -0
  155. package/ai/tools/browserTools/closeSession.ts +29 -0
  156. package/ai/tools/browserTools/common.ts +27 -0
  157. package/ai/tools/browserTools/openSession.ts +48 -0
  158. package/ai/tools/browserTools/readContent.ts +38 -0
  159. package/ai/tools/browserTools/selectOption.ts +46 -0
  160. package/ai/tools/browserTools/typeText.ts +42 -0
  161. package/ai/tools/category/createCategoryTool.ts +66 -0
  162. package/ai/tools/category/queryContentsByCategoryTool.ts +69 -0
  163. package/ai/tools/category/updateContentCategoryTool.ts +75 -0
  164. package/ai/tools/cfBrowserTools.ts +319 -0
  165. package/ai/tools/cfSpeechToTextTool.ts +49 -0
  166. package/ai/tools/checkEnvTool.ts +65 -0
  167. package/ai/tools/cloudflareCrawlTool.ts +289 -0
  168. package/ai/tools/codeSearchTool.ts +111 -0
  169. package/ai/tools/codeTools.ts +101 -0
  170. package/ai/tools/createDocTool.ts +132 -0
  171. package/ai/tools/createPlanTool.ts +999 -0
  172. package/ai/tools/createSkillDocTool.ts +155 -0
  173. package/ai/tools/createWorkflowTool.ts +154 -0
  174. package/ai/tools/deepseekOcrTool.ts +34 -0
  175. package/ai/tools/delayTool.ts +31 -0
  176. package/ai/tools/deleteSpacesTool.ts +325 -0
  177. package/ai/tools/deleteSpacesToolModel.ts +159 -0
  178. package/ai/tools/devReloadUtils.ts +29 -0
  179. package/ai/tools/dialogMessageSearch.ts +137 -0
  180. package/ai/tools/doctorSkillTool.ts +72 -0
  181. package/ai/tools/ecommerceScraperTool.ts +86 -0
  182. package/ai/tools/emailTools.ts +549 -0
  183. package/ai/tools/evalSkillTool.ts +92 -0
  184. package/ai/tools/exaSearchTool.ts +64 -0
  185. package/ai/tools/execBashTool.ts +379 -0
  186. package/ai/tools/executeSqlTool.ts +192 -0
  187. package/ai/tools/fetchWebpageSupport.ts +309 -0
  188. package/ai/tools/fetchWebpageTool.ts +84 -0
  189. package/ai/tools/geminiImagePreviewTool.ts +361 -0
  190. package/ai/tools/generateDocxTool.ts +215 -0
  191. package/ai/tools/googleSearchScraperTool.ts +106 -0
  192. package/ai/tools/importDataTool.ts +133 -0
  193. package/ai/tools/importSkillTool.ts +162 -0
  194. package/ai/tools/index.ts +1927 -0
  195. package/ai/tools/listFilesTool.ts +82 -0
  196. package/ai/tools/listUserSpacesTool.ts +113 -0
  197. package/ai/tools/modelUsageTools.ts +199 -0
  198. package/ai/tools/olmOcrTool.ts +34 -0
  199. package/ai/tools/openaiImageTool.ts +267 -0
  200. package/ai/tools/prepareTools.ts +23 -0
  201. package/ai/tools/readDocTool.ts +84 -0
  202. package/ai/tools/readFileTool.ts +211 -0
  203. package/ai/tools/readTool.ts +163 -0
  204. package/ai/tools/readXPostTool.ts +233 -0
  205. package/ai/tools/rememberMemoryTool.ts +84 -0
  206. package/ai/tools/remotionVideoTool.ts +151 -0
  207. package/ai/tools/searchDialogMessagesTool.ts +222 -0
  208. package/ai/tools/searchRepoTool.ts +115 -0
  209. package/ai/tools/searchWorkspaceTool.ts +259 -0
  210. package/ai/tools/skillFollowup.ts +86 -0
  211. package/ai/tools/surfWeatherTool.ts +169 -0
  212. package/ai/tools/table/addTableRowTool.ts +217 -0
  213. package/ai/tools/table/createTableTool.ts +315 -0
  214. package/ai/tools/table/rowTools.ts +366 -0
  215. package/ai/tools/table/schemaTools.ts +244 -0
  216. package/ai/tools/table/shareTableTool.ts +148 -0
  217. package/ai/tools/table/toolShared.ts +129 -0
  218. package/ai/tools/toolApiClient.ts +198 -0
  219. package/ai/tools/toolNameAliases.ts +57 -0
  220. package/ai/tools/toolResultError.ts +42 -0
  221. package/ai/tools/toolRunSlice.ts +303 -0
  222. package/ai/tools/toolSchemaCompatibility.ts +53 -0
  223. package/ai/tools/toolVisibility.ts +4 -0
  224. package/ai/tools/types.ts +20 -0
  225. package/ai/tools/uiAskChoiceTool.ts +104 -0
  226. package/ai/tools/updateContentTitleTool.ts +84 -0
  227. package/ai/tools/updateDocTool.ts +105 -0
  228. package/ai/tools/updateUserPreferenceProfileTool.ts +145 -0
  229. package/ai/tools/whisperTool.ts +77 -0
  230. package/ai/tools/writeFileTool.ts +210 -0
  231. package/ai/tools/youtubeScraperTool.ts +116 -0
  232. package/ai/tools/ziweiChartTool.ts +678 -0
  233. package/ai/types.ts +55 -0
  234. package/ai/workflow/workflowExecutor.ts +323 -0
  235. package/ai/workflow/workflowSlice.ts +73 -0
  236. package/ai/workflow/workflowTypes.ts +106 -0
  237. package/client/agentRun.test.ts +240 -0
  238. package/client/agentRun.ts +182 -19
  239. package/client/compactDialog.test.ts +238 -0
  240. package/client/localRuntimeAdapter.test.ts +135 -0
  241. package/client/localRuntimeAdapter.ts +244 -0
  242. package/client/profileConfig.test.ts +40 -0
  243. package/client/streamingOutput.test.ts +22 -0
  244. package/client/streamingOutput.ts +38 -0
  245. package/commandRegistry.ts +9 -2
  246. package/connector-experimental/index.ts +5 -0
  247. package/database/actions/cacheMergedUserData.ts +64 -0
  248. package/database/actions/common.ts +242 -0
  249. package/database/actions/deleteFile.ts +40 -0
  250. package/database/actions/fetchUserData.ts +16 -0
  251. package/database/actions/fileContent.ts +125 -0
  252. package/database/actions/patch.ts +155 -0
  253. package/database/actions/read.ts +337 -0
  254. package/database/actions/readAndWait.ts +224 -0
  255. package/database/actions/readRequestManager.ts +120 -0
  256. package/database/actions/remove.ts +94 -0
  257. package/database/actions/replication.ts +366 -0
  258. package/database/actions/upload.ts +174 -0
  259. package/database/actions/upsert.ts +56 -0
  260. package/database/actions/write.ts +126 -0
  261. package/database/client/db.native.ts +73 -0
  262. package/database/client/db.ts +51 -0
  263. package/database/client/fetchUserData.ts +61 -0
  264. package/database/client/handleError.ts +19 -0
  265. package/database/client/queryRequest.ts +21 -0
  266. package/database/config.ts +21 -0
  267. package/database/dbActionThunks.ts +1 -0
  268. package/database/dbSlice.ts +149 -0
  269. package/database/email.ts +42 -0
  270. package/database/fileRing.ts +51 -0
  271. package/database/fileSharding.ts +70 -0
  272. package/database/fileStorage.native.ts +92 -0
  273. package/database/fileStorage.ts +232 -0
  274. package/database/fileUrl.ts +34 -0
  275. package/database/hooks/useUserData.ts +489 -0
  276. package/database/index.ts +1 -0
  277. package/database/keys.ts +765 -0
  278. package/database/queryPrefixes.ts +14 -0
  279. package/database/requests.ts +443 -0
  280. package/database/runtimeServerContext.ts +35 -0
  281. package/database/server/MemoryDB.ts +76 -0
  282. package/database/server/actorAccess.ts +76 -0
  283. package/database/server/agentDelegation.ts +124 -0
  284. package/database/server/coreDataOwnership.ts +13 -0
  285. package/database/server/coreDataProxy.ts +76 -0
  286. package/database/server/cybotReadonly.ts +18 -0
  287. package/database/server/dataHandlers.ts +111 -0
  288. package/database/server/db.ts +118 -0
  289. package/database/server/dbPath.ts +20 -0
  290. package/database/server/delete.ts +499 -0
  291. package/database/server/emailRepository.ts +1480 -0
  292. package/database/server/ensureDbOpen.ts +12 -0
  293. package/database/server/fileRead.ts +337 -0
  294. package/database/server/fileService.ts +436 -0
  295. package/database/server/handleTransaction.ts +86 -0
  296. package/database/server/patch.ts +282 -0
  297. package/database/server/query.ts +138 -0
  298. package/database/server/read.ts +325 -0
  299. package/database/server/resourceAccess.ts +211 -0
  300. package/database/server/routes.ts +110 -0
  301. package/database/server/spaceMemberAuthority.ts +67 -0
  302. package/database/server/upload.ts +159 -0
  303. package/database/server/write.ts +494 -0
  304. package/database/server/writeAuthority.ts +133 -0
  305. package/database/sqliteDb.ts +46 -0
  306. package/database/table/deleteTable.ts +120 -0
  307. package/database/tenantPlacement.ts +57 -0
  308. package/database/tombstones.ts +52 -0
  309. package/database/userDataLoadDecision.ts +17 -0
  310. package/database/userDataMerge.ts +95 -0
  311. package/database/userPreferenceRegister.ts +108 -0
  312. package/database/utils/dbPath.ts +47 -0
  313. package/database/utils/ulid.native.ts +6 -0
  314. package/database/utils/ulid.ts +1 -0
  315. package/index.ts +25 -15
  316. package/localRuntimeDb.ts +28 -0
  317. package/package.json +16 -4
  318. package/runtimeModeArgs.ts +33 -0
  319. package/tui/readlineWorkspace.ts +1 -0
  320. package/tui/session.ts +22 -0
@@ -1,2 +1,545 @@
1
+ // 路径: ai/agent/agentSlice.ts
2
+
3
+ import { asyncThunkCreator, buildCreateSlice, type PayloadAction } from "@reduxjs/toolkit";
4
+ import type { Agent, AgentRuntimeBinding, ReferenceItem } from "app/types";
5
+
6
+ import { write, patch, remove } from "database/dbSlice";
7
+ import { createCybotKey, createAgentKey } from "database/keys";
8
+ import { DataType } from "create/types";
1
9
  import { ulid } from "ulid";
2
- export const createAgent = () => ({ id: ulid() });
10
+ import type { FormData as AgentFormData } from "ai/agent/createAgentSchema";
11
+
12
+ const createSliceWithThunks = buildCreateSlice({
13
+ creators: { asyncThunk: asyncThunkCreator },
14
+ });
15
+
16
+ /** Slice State 定义 */
17
+ interface AgentState {
18
+ pubCybots: {
19
+ loading: boolean;
20
+ error: string | null;
21
+ data: Agent[];
22
+ };
23
+ }
24
+
25
+ /** runLlm 参数(通用 LLM 调用) */
26
+ interface RunLlmArgs {
27
+ /** 直接传入内置 llmConfig,完全不依赖 Agent 数据 */
28
+ llmConfig?: Partial<Agent> & Pick<Agent, "provider" | "model">;
29
+ agentKey?: string;
30
+ /** 兼容旧调用:直接传入 agentConfig,跳过 DB 读取 */
31
+ agentConfig?: Partial<Agent> & Pick<Agent, "provider" | "model">;
32
+ content: unknown;
33
+ isStreaming?: boolean;
34
+ parentMessageId?: string;
35
+ /** 覆盖 Agent 配置中的 system prompt */
36
+ systemPromptOverride?: string;
37
+ /** 覆盖 Agent 配置中的工具列表(tool id 数组) */
38
+ toolsOverride?: string[];
39
+ billingDialogKey?: string;
40
+ }
41
+
42
+ /** runAgent 参数(通用 Agent 调用,多轮工具循环) */
43
+ interface RunAgentArgs {
44
+ agentKey: string;
45
+ content: unknown;
46
+ parentMessageId?: string;
47
+ billingDialogKey?: string;
48
+ }
49
+
50
+ /** createAgent 参数(新建 Agent) */
51
+ interface CreateAgentArgs {
52
+ userId: string;
53
+ formData: AgentFormData; // 已通过表单或 Tool 构造的完整数据
54
+ spaceId?: string; // 可选的空间 ID,如果提供,则将 Agent 添加到该空间
55
+ }
56
+
57
+ /** updateAgent 参数(更新 Agent,支持部分字段 patch) */
58
+ interface UpdateAgentArgs {
59
+ userId: string;
60
+ agentId: string; // 纯 id(不带 db path 前缀)
61
+ formData: Partial<AgentFormData>; // 允许只传需要修改的字段
62
+ previousAgent?: Partial<Agent>; // UI 编辑时可以传,用来保持公共副本同步
63
+ }
64
+
65
+ const initialState: AgentState = {
66
+ pubCybots: {
67
+ loading: false,
68
+ error: null,
69
+ data: [],
70
+ },
71
+ };
72
+
73
+ const normalizeAgentReferences = (references: any[]): ReferenceItem[] => {
74
+ if (!Array.isArray(references)) return [];
75
+ return references.map((ref) => ({
76
+ dbKey: ref.dbKey || "",
77
+ title: ref.title || "",
78
+ type: ref.type === "page" ? "knowledge" : ref.type || "knowledge",
79
+ }));
80
+ };
81
+
82
+ /**
83
+ * 创建场景:表单数据 -> 持久化数据(全量)
84
+ */
85
+ const processAgentCreateForm = (formData: AgentFormData, userId: string) => {
86
+ const isPublic = !!formData.isPublic;
87
+ const machineId = typeof (formData as any).machineId === "string"
88
+ ? (formData as any).machineId.trim()
89
+ : "";
90
+
91
+ const result: any = {
92
+ ...formData,
93
+ // tags: "a, b" -> ["a", "b"]
94
+ tags: formData.tags
95
+ ? formData.tags
96
+ .split(",")
97
+ .map((s) => s.trim())
98
+ .filter(Boolean)
99
+ : [],
100
+ // 归一化 references
101
+ references: normalizeAgentReferences(formData.references || []),
102
+ // 非公开时,强制清空白名单,避免脏数据
103
+ whitelist: isPublic ? formData.whitelist || [] : [],
104
+ };
105
+
106
+ delete result.machineId;
107
+ if (formData.apiSource === "cli" && machineId) {
108
+ const binding: AgentRuntimeBinding = {
109
+ ...(result.runtimeBinding && typeof result.runtimeBinding === "object"
110
+ ? result.runtimeBinding
111
+ : {}),
112
+ machineId,
113
+ ownerUserId: userId,
114
+ };
115
+ result.runtimeBinding = binding;
116
+ }
117
+
118
+ return result;
119
+ };
120
+
121
+ /**
122
+ * 更新场景:部分表单字段 -> patch changes
123
+ * - 只对传进来的字段做转换 / 归一化
124
+ * - 未出现在 formData 里的字段一律不修改
125
+ */
126
+ const processAgentUpdateChanges = (
127
+ data: Partial<AgentFormData>,
128
+ userId: string,
129
+ previousAgent?: Partial<Agent>
130
+ ) => {
131
+ const changes: any = {};
132
+
133
+ // 基本字符串字段
134
+ if ("name" in data) {
135
+ changes.name = String(data.name ?? "").trim();
136
+ }
137
+ if ("model" in data) {
138
+ changes.model = (data.model ?? "").trim();
139
+ }
140
+ if ("provider" in data) {
141
+ changes.provider = (data.provider ?? "").trim();
142
+ }
143
+ if ("prompt" in data) {
144
+ changes.prompt = (data.prompt ?? "").trim();
145
+ }
146
+ if ("introduction" in data) {
147
+ changes.introduction = (data.introduction ?? "").trim();
148
+ }
149
+ if ("customProviderUrl" in data) {
150
+ changes.customProviderUrl = (data.customProviderUrl ?? "").trim();
151
+ }
152
+ if ("apiKey" in data) {
153
+ changes.apiKey = (data.apiKey ?? "").trim();
154
+ }
155
+
156
+ // 简单标志位
157
+ if ("hasVision" in data && data.hasVision !== undefined) {
158
+ changes.hasVision = !!data.hasVision;
159
+ }
160
+ if ("apiSource" in data && data.apiSource) {
161
+ changes.apiSource = data.apiSource;
162
+ }
163
+ if ("cliProvider" in data) {
164
+ changes.cliProvider = (data as any).cliProvider || "";
165
+ }
166
+ if ("machineId" in data) {
167
+ const machineId = String((data as any).machineId ?? "").trim();
168
+
169
+ // Determine effective apiSource: prefer data.apiSource, fall back to previousAgent.apiSource
170
+ const effectiveApiSource = data.apiSource ?? previousAgent?.apiSource;
171
+
172
+ if (machineId) {
173
+ // Persist runtimeBinding when effective apiSource is 'cli' or undefined
174
+ // undefined = tool/legacy partial update without apiSource context → preserve backward compatibility
175
+ if (effectiveApiSource === "cli" || effectiveApiSource === undefined) {
176
+ const binding: AgentRuntimeBinding = {
177
+ ...((changes.runtimeBinding && typeof changes.runtimeBinding === "object")
178
+ ? changes.runtimeBinding
179
+ : {}),
180
+ machineId,
181
+ ownerUserId: userId,
182
+ };
183
+ changes.runtimeBinding = binding;
184
+ }
185
+ // If apiSource is explicitly non-cli and machineId is supplied, do not create binding
186
+ } else {
187
+ // CLEAR branch: clear runtimeBinding when effectiveApiSource is 'cli' or undefined (backward-compatible)
188
+ if (effectiveApiSource === "cli" || effectiveApiSource === undefined) {
189
+ changes.runtimeBinding = null;
190
+ }
191
+ }
192
+ }
193
+ if ("useServerProxy" in data && data.useServerProxy !== undefined) {
194
+ changes.useServerProxy = !!data.useServerProxy;
195
+ }
196
+
197
+ // greeting / tools 直接透传
198
+ if ("greeting" in data) {
199
+ changes.greeting = data.greeting as any;
200
+ }
201
+ if ("tools" in data) {
202
+ changes.tools = Array.isArray(data.tools) ? data.tools.slice() : [];
203
+ }
204
+
205
+ // 数值字段
206
+ const numericKeys: (keyof AgentFormData)[] = [
207
+ "inputPrice",
208
+ "outputPrice",
209
+ "temperature",
210
+ "top_p",
211
+ "frequency_penalty",
212
+ "presence_penalty",
213
+ "max_tokens",
214
+ ];
215
+ numericKeys.forEach((key) => {
216
+ if (key in data) {
217
+ const raw = data[key];
218
+ if (raw === undefined || raw === null) {
219
+ changes[key] = raw as any;
220
+ } else {
221
+ const num = Number(raw as any);
222
+ changes[key] = Number.isNaN(num) ? raw : num;
223
+ }
224
+ }
225
+ });
226
+
227
+ // ⚠ 唯一实质改动:reasoning_effort 允许传 null,用于“清空该字段”
228
+ // 原来是:if ("reasoning_effort" in data && data.reasoning_effort) { ... }
229
+ // 这样会把 null 吞掉,无法让后端 patch + deepMerge 删除字段。
230
+ if ("reasoning_effort" in data) {
231
+ changes.reasoning_effort = (data as any).reasoning_effort;
232
+ }
233
+
234
+ // tags: string 或 string[]
235
+ if ("tags" in data) {
236
+ const raw = (data as any).tags;
237
+ let arr: string[] = [];
238
+ if (Array.isArray(raw)) {
239
+ arr = raw.map((s) => String(s || "").trim()).filter(Boolean);
240
+ } else if (typeof raw === "string") {
241
+ arr = raw
242
+ .split(",")
243
+ .map((s) => s.trim())
244
+ .filter(Boolean);
245
+ }
246
+ changes.tags = arr;
247
+ }
248
+
249
+ // references
250
+ if ("references" in data) {
251
+ changes.references = normalizeAgentReferences(
252
+ ((data.references as any) || []) as any[]
253
+ );
254
+ }
255
+
256
+ // whitelist + isPublic
257
+ if ("whitelist" in data) {
258
+ changes.whitelist = (data.whitelist as string[]) || [];
259
+ }
260
+
261
+ if ("isPublic" in data) {
262
+ changes.isPublic = !!data.isPublic;
263
+ if (!changes.isPublic) {
264
+ // 一旦改为私有,强制清空白名单
265
+ changes.whitelist = [];
266
+ }
267
+ }
268
+
269
+ return changes;
270
+ };
271
+
272
+ export const slice = createSliceWithThunks({
273
+ // 注意:保持 name = "cybot",以兼容原有 Redux state 结构
274
+ name: "cybot",
275
+ initialState,
276
+ reducers: (create) => ({
277
+ /**
278
+ * SSR 首屏:服务端预取公开 Agent 列表后注入,走 __PRELOADED_STATE__ 链路
279
+ */
280
+ setSSRPublicAgents: create.reducer(
281
+ (state, action: PayloadAction<Agent[]>) => {
282
+ state.pubCybots.data = Array.isArray(action.payload) ? action.payload : [];
283
+ state.pubCybots.loading = false;
284
+ state.pubCybots.error = null;
285
+ }
286
+ ),
287
+
288
+ /**
289
+ * 通用 LLM 调用(不带 Agent 上下文 / 历史)
290
+ */
291
+ runLlm: create.asyncThunk(async (args: RunLlmArgs, thunkApi) => {
292
+ const overrides: Record<string, any> = {};
293
+ if (args.systemPromptOverride !== undefined) overrides.prompt = args.systemPromptOverride;
294
+ if (args.toolsOverride !== undefined) overrides.tools = args.toolsOverride;
295
+ const { _executeModel } = await import("ai/agent/_executeModel");
296
+ return _executeModel(
297
+ {
298
+ isStreaming: args.isStreaming ?? false,
299
+ withAgentContext: false,
300
+ withChatHistory: false,
301
+ agentConfigOverrides: Object.keys(overrides).length ? overrides : undefined,
302
+ },
303
+ args,
304
+ thunkApi
305
+ );
306
+ }),
307
+
308
+ /**
309
+ * 通用 Agent 调用(带 Agent 上下文,多轮工具循环)
310
+ *
311
+ * 使用客户端 runAgentClientLoop:
312
+ * - 每轮调用 LLM(非流式)
313
+ * - 遇到 tool_calls 时通过 findToolExecutor 本地执行工具
314
+ * - 循环直到无工具调用或触发其他运行时停止条件
315
+ */
316
+ runAgent: create.asyncThunk(async (args: RunAgentArgs, thunkApi) => {
317
+ const { runAgentClientLoop } = await import("ai/agent/runAgentClientLoop");
318
+ const { content: loopContent, toolCallCount } = await runAgentClientLoop(
319
+ {
320
+ agentKey: args.agentKey,
321
+ content: args.content,
322
+ parentMessageId: args.parentMessageId,
323
+ billingDialogKey: args.billingDialogKey,
324
+ },
325
+ thunkApi
326
+ );
327
+ return loopContent;
328
+ }),
329
+
330
+ /**
331
+ * 聊天轮次流式 Agent 调用
332
+ */
333
+ streamAgentChatTurn: create.asyncThunk(async (args: any, thunkApi) => {
334
+ const { streamAgentChatTurnHandler } = await import("ai/agent/streamAgentChatTurn");
335
+ return streamAgentChatTurnHandler(args, thunkApi);
336
+ }),
337
+
338
+ /**
339
+ * 创建 Agent:
340
+ * - 写入用户私有路径
341
+ * - 如 isPublic=true,则同时写入公共路径
342
+ * - 返回完整 Agent 对象(包含 id / meta 字段)
343
+ */
344
+ createAgent: create.asyncThunk(
345
+ async ({ userId, formData, spaceId }: CreateAgentArgs, thunkApi) => {
346
+ const processed = processAgentCreateForm(formData, userId);
347
+
348
+ const now = Date.now();
349
+ const id = ulid();
350
+
351
+ const privateKey = createAgentKey.private(userId, id);
352
+ const publicKey = createAgentKey.public(id);
353
+
354
+ const agent: Agent = {
355
+ ...(processed as any),
356
+ id,
357
+ type: DataType.AGENT,
358
+ userId,
359
+ createdAt: now,
360
+ updatedAt: now,
361
+ dialogCount: 0,
362
+ messageCount: 0,
363
+ tokenCount: 0,
364
+ spaceId: spaceId, // 记录 spaceId
365
+ };
366
+
367
+ // 写入私有副本
368
+ await thunkApi
369
+ .dispatch(
370
+ write({
371
+ data: agent,
372
+ customKey: privateKey,
373
+ })
374
+ )
375
+ .unwrap();
376
+
377
+ // 如需公开,再写入公共副本
378
+ if (agent.isPublic) {
379
+ await thunkApi
380
+ .dispatch(
381
+ write({
382
+ data: agent,
383
+ customKey: publicKey,
384
+ })
385
+ )
386
+ .unwrap();
387
+ }
388
+
389
+ return agent;
390
+ }
391
+ ),
392
+
393
+ /**
394
+ * 更新 Agent(支持局部字段 patch):
395
+ * - patch 私有副本
396
+ * - 如提供 previousAgent,则同步更新 / 删除公共副本
397
+ *
398
+ * 注意:
399
+ * - Tool 场景下一般不提供 previousAgent,此时只保证私有副本被更新;
400
+ * 公共副本(应用市场)不做强一致保证。
401
+ */
402
+ updateAgent: create.asyncThunk(
403
+ async (
404
+ { userId, agentId, formData, previousAgent }: UpdateAgentArgs,
405
+ thunkApi
406
+ ) => {
407
+ const normalizedAgentId = (() => {
408
+ const raw = agentId.trim();
409
+ if (raw.startsWith("agent-") || raw.startsWith("cybot-")) {
410
+ const parts = raw.split("-");
411
+ if (parts.length >= 3) return parts[parts.length - 1];
412
+ }
413
+ return raw;
414
+ })();
415
+
416
+ // 【兼容层】双前缀处理:
417
+ // 1. 如果 agentId 是以 cybot- 开头,或者 previousAgent.type 是 cybot,说明这是存量旧数据。
418
+ // 2. 存量数据必须使用 createCybotKey 才能正确定位到数据库中的位置。
419
+ // 3. 新数据则默认使用 createAgentKey (agent- 前缀)。
420
+ let privateKey = createAgentKey.private(userId, normalizedAgentId);
421
+ let publicKey = createAgentKey.public(normalizedAgentId);
422
+
423
+ if (agentId.startsWith("cybot-") || previousAgent?.type === "cybot") {
424
+ privateKey = createCybotKey.private(userId, normalizedAgentId);
425
+ publicKey = createCybotKey.public(normalizedAgentId);
426
+ }
427
+
428
+ const changes = processAgentUpdateChanges(formData || {}, userId, previousAgent);
429
+
430
+ // 1) 检查本地是否存在
431
+ let localExists = false;
432
+ try {
433
+ const { db } = thunkApi.extra as any;
434
+ const localData = await db.get(privateKey);
435
+ localExists = !!localData;
436
+ } catch (e) {
437
+ // ignore
438
+ }
439
+
440
+ if (localExists) {
441
+ // 有本地数据,直接 patch
442
+ await thunkApi
443
+ .dispatch(
444
+ patch({
445
+ dbKey: privateKey,
446
+ changes,
447
+ })
448
+ )
449
+ .unwrap();
450
+ } else if (previousAgent) {
451
+ // 无本地数据,但有 UI 传来的 previousAgent,用 write 回填
452
+ const merged = {
453
+ ...previousAgent,
454
+ ...changes,
455
+ id: normalizedAgentId,
456
+ type: previousAgent.type || DataType.AGENT,
457
+ userId,
458
+ };
459
+ await thunkApi
460
+ .dispatch(
461
+ write({
462
+ data: merged,
463
+ customKey: privateKey,
464
+ })
465
+ )
466
+ .unwrap();
467
+ } else {
468
+ // 既无本地也无 previousAgent,尝试标准 path (可能会失败 if remote also fails or path throws)
469
+ // 但既然到了 update,大概率之前 read 过。
470
+ // 兜底调用 patch,让 patch 内部去报错
471
+ await thunkApi
472
+ .dispatch(
473
+ patch({
474
+ dbKey: privateKey,
475
+ changes,
476
+ })
477
+ )
478
+ .unwrap();
479
+ }
480
+
481
+ // 2) 如提供 previousAgent,则尝试保持公共副本同步
482
+ if (previousAgent) {
483
+ const wasPublic = !!previousAgent.isPublic;
484
+ const hasIsPublicChange = Object.prototype.hasOwnProperty.call(
485
+ changes,
486
+ "isPublic"
487
+ );
488
+ const nowPublic = hasIsPublicChange
489
+ ? !!(changes as any).isPublic
490
+ : wasPublic;
491
+
492
+ if (nowPublic) {
493
+ const mergedPublic: Agent = {
494
+ ...(previousAgent as any),
495
+ ...(changes as any),
496
+ id: normalizedAgentId,
497
+ type: previousAgent.type || DataType.AGENT,
498
+ userId,
499
+ };
500
+
501
+ await thunkApi
502
+ .dispatch(
503
+ write({
504
+ data: mergedPublic,
505
+ customKey: publicKey,
506
+ })
507
+ )
508
+ .unwrap();
509
+ } else if (wasPublic && !nowPublic) {
510
+ await thunkApi
511
+ .dispatch(remove(publicKey))
512
+ .unwrap();
513
+ }
514
+ }
515
+
516
+ // 3) 返回“私有视角”的最新 Agent(主要给前端本地状态使用)
517
+ const base = previousAgent ?? ({} as Partial<Agent>);
518
+ const mergedPrivate: Agent = {
519
+ ...(base as any),
520
+ ...(changes as any),
521
+ id: normalizedAgentId,
522
+ type: base.type || DataType.AGENT,
523
+ userId,
524
+ };
525
+
526
+ return mergedPrivate;
527
+ }
528
+ ),
529
+ }),
530
+ });
531
+
532
+ export const {
533
+ runLlm,
534
+ runAgent,
535
+ streamAgentChatTurn,
536
+ createAgent,
537
+ updateAgent,
538
+ setSSRPublicAgents,
539
+ } = slice.actions;
540
+
541
+ export default slice.reducer;
542
+
543
+ /** 读取 SSR 预载的公开 Agent 列表(首页 AI 广场) */
544
+ export const selectSSRPublicAgents = (state: any): Agent[] =>
545
+ state.cybot?.pubCybots?.data ?? [];
@@ -0,0 +1,126 @@
1
+ import type { Message } from "chat/messages/types";
2
+ import type { ToolRun } from "ai/tools/toolRunSlice";
3
+
4
+ const APP_TOOL_NAMES = new Set([
5
+ "appList",
6
+ "appRead",
7
+ "appDeploy",
8
+ "appPreflight",
9
+ "appDelete",
10
+ ]);
11
+
12
+ const contentToText = (content: Message["content"] | undefined): string => {
13
+ if (!content) return "";
14
+ if (typeof content === "string") return content;
15
+ if (Array.isArray(content)) {
16
+ return content
17
+ .map((part) => ("text" in part ? part.text ?? "" : ""))
18
+ .join("\n");
19
+ }
20
+ return "";
21
+ };
22
+
23
+ const parseAppListEntries = (text: string): string[] =>
24
+ text
25
+ .split("\n")
26
+ .map((line) => line.trim())
27
+ .filter((line) => line.startsWith("- **") && line.includes("(appId:"));
28
+
29
+ export const buildRecentAppToolMemory = (
30
+ messages: Message[],
31
+ toolRuns: ToolRun[],
32
+ ): string | null => {
33
+ const messageById = new Map(messages.map((msg) => [msg.id, msg]));
34
+ const recentRuns = [...toolRuns]
35
+ .filter((run) => APP_TOOL_NAMES.has(run.toolName))
36
+ .sort((a, b) => (b.startedAt ?? 0) - (a.startedAt ?? 0))
37
+ .slice(0, 8);
38
+
39
+ if (recentRuns.length === 0) return null;
40
+
41
+ const lines: string[] = [
42
+ "这些信息来自当前对话里最近的 app 工具调用,不依赖右侧编辑态。",
43
+ ];
44
+
45
+ const latestDeployLike = recentRuns.find((run) =>
46
+ ["appDeploy", "appRead", "appPreflight"].includes(run.toolName),
47
+ );
48
+ if (latestDeployLike) {
49
+ const input = latestDeployLike.input ?? {};
50
+ const parts = [
51
+ `- 最近一次关键 app 操作: ${latestDeployLike.toolName}`,
52
+ typeof input.appId === "string" ? `appId=${input.appId}` : null,
53
+ typeof input.name === "string" ? `name=${input.name}` : null,
54
+ typeof input.framework === "string" ? `framework=${input.framework}` : null,
55
+ ].filter(Boolean);
56
+ lines.push(parts.join(","));
57
+ }
58
+
59
+ const relatedMessages = recentRuns
60
+ .map((run) => messageById.get(run.messageId))
61
+ .filter((msg): msg is Message => !!msg);
62
+
63
+ const appListMessage = relatedMessages.find((msg) => msg.toolName === "appList");
64
+ if (appListMessage) {
65
+ const entries = parseAppListEntries(contentToText(appListMessage.content)).slice(0, 5);
66
+ if (entries.length > 0) {
67
+ lines.push("- 最近一次 appList 结果:");
68
+ lines.push(...entries.map((entry) => ` ${entry}`));
69
+ }
70
+ }
71
+
72
+ const latestReadMessage = relatedMessages.find((msg) => msg.toolName === "appRead");
73
+ if (latestReadMessage) {
74
+ const text = contentToText(latestReadMessage.content);
75
+ const appId = text.match(/- appId:\s*(.+)/)?.[1]?.trim();
76
+ const url = text.match(/- 访问地址:\s*(.+)/)?.[1]?.trim();
77
+ if (appId || url) {
78
+ lines.push(
79
+ [
80
+ "- 最近一次 appRead 真值:",
81
+ appId ? `appId=${appId}` : null,
82
+ url ? `url=${url}` : null,
83
+ ]
84
+ .filter(Boolean)
85
+ .join(","),
86
+ );
87
+ }
88
+ }
89
+
90
+ const latestPreflightRun = recentRuns.find((run) => run.toolName === "appPreflight");
91
+ if (latestPreflightRun) {
92
+ lines.push(
93
+ [
94
+ "- 最近一次 appPreflight:",
95
+ latestPreflightRun.status === "failed" ? "失败" : "已执行",
96
+ latestPreflightRun.outputSummary ? `摘要=${latestPreflightRun.outputSummary}` : null,
97
+ ]
98
+ .filter(Boolean)
99
+ .join(","),
100
+ );
101
+ }
102
+
103
+ const latestDeployMessage = relatedMessages.find((msg) => msg.toolName === "appDeploy");
104
+ if (latestDeployMessage) {
105
+ const text = contentToText(latestDeployMessage.content);
106
+ const appId = text.match(/- appId:\s*(.+)/)?.[1]?.trim();
107
+ const url = text.match(/- 访问地址:\s*(.+)/)?.[1]?.trim();
108
+ if (appId || url) {
109
+ lines.push(
110
+ [
111
+ "- 最近一次 appDeploy 结果:",
112
+ appId ? `appId=${appId}` : null,
113
+ url ? `url=${url}` : null,
114
+ ]
115
+ .filter(Boolean)
116
+ .join(","),
117
+ );
118
+ }
119
+ }
120
+
121
+ lines.push(
122
+ "- 如果用户说“刚才那个 app / 那个网站”,优先把它理解为上面最近一次被读取、预检或部署的应用;若仍有歧义,再用 appList 或 appRead 确认。",
123
+ );
124
+
125
+ return lines.join("\n");
126
+ };
@@ -0,0 +1,24 @@
1
+ // packages/ai/agent/avatarUtils.ts
2
+ // Resolves an agent's avatarFileId → a displayable URL
3
+
4
+ import { buildDatabaseFileContentUrl } from "database/fileUrl";
5
+
6
+ /**
7
+ * Returns a URL for displaying the agent avatar, or null.
8
+ * - If avatarFileId is an http(s) URL → use directly
9
+ * - Otherwise build server file-content URL
10
+ */
11
+ export function resolveAvatarUrl(
12
+ avatarFileId: string | undefined | null,
13
+ server: string | undefined | null
14
+ ): string | null {
15
+ if (!avatarFileId) return null;
16
+ if (avatarFileId.startsWith("http") || avatarFileId.startsWith("blob:")) {
17
+ return avatarFileId;
18
+ }
19
+ if (avatarFileId.startsWith("/")) {
20
+ if (!server) return avatarFileId;
21
+ return `${server}${avatarFileId}`;
22
+ }
23
+ return buildDatabaseFileContentUrl(server, avatarFileId);
24
+ }