nolo-cli 0.1.9 → 0.1.11

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