nolo-cli 0.1.10 → 0.1.12

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 (232) hide show
  1. package/README.md +0 -32
  2. package/agentRuntimeCommands.ts +1 -1
  3. package/client/compactDialog.ts +2 -5
  4. package/commandRegistry.ts +2 -2
  5. package/machineCommands.ts +28 -3
  6. package/package.json +5 -25
  7. package/ai/agent/_executeModel.ts +0 -118
  8. package/ai/agent/agentSlice.ts +0 -525
  9. package/ai/agent/appWorkingMemory.ts +0 -126
  10. package/ai/agent/avatarUtils.ts +0 -24
  11. package/ai/agent/buildEditingContext.ts +0 -373
  12. package/ai/agent/buildSystemPrompt.ts +0 -532
  13. package/ai/agent/cleanAgentMessages.ts +0 -140
  14. package/ai/agent/cliChatClient.ts +0 -119
  15. package/ai/agent/contextCompiler.ts +0 -107
  16. package/ai/agent/contextLayerContract.ts +0 -44
  17. package/ai/agent/createAgentSchema.ts +0 -234
  18. package/ai/agent/executeToolCall.ts +0 -58
  19. package/ai/agent/fetchAgentContexts.ts +0 -42
  20. package/ai/agent/generatePrompt.ts +0 -3
  21. package/ai/agent/getFullChatContextKeys.ts +0 -168
  22. package/ai/agent/hooks/fetchPublicAgents.ts +0 -133
  23. package/ai/agent/hooks/useAgentConfig.ts +0 -61
  24. package/ai/agent/hooks/useAgentDialog.ts +0 -35
  25. package/ai/agent/hooks/useAgentFormValidation.ts +0 -202
  26. package/ai/agent/hooks/usePublicAgents.ts +0 -473
  27. package/ai/agent/persistMessageWithFixedId.ts +0 -37
  28. package/ai/agent/planSlice.ts +0 -259
  29. package/ai/agent/referenceUtils.ts +0 -229
  30. package/ai/agent/runAgentBackground.ts +0 -238
  31. package/ai/agent/runAgentClientLoop.ts +0 -138
  32. package/ai/agent/runtimeGuidance.ts +0 -97
  33. package/ai/agent/runtimeServerBase.ts +0 -37
  34. package/ai/agent/server/fetchPublicAgents.ts +0 -128
  35. package/ai/agent/startParallelAgentStreams.ts +0 -424
  36. package/ai/agent/startupProtocol.ts +0 -53
  37. package/ai/agent/streamAgentChatTurn.ts +0 -1278
  38. package/ai/agent/streamAgentChatTurnUtils.ts +0 -738
  39. package/ai/agent/types.ts +0 -71
  40. package/ai/agent/utils/imageOutput.ts +0 -33
  41. package/ai/agent/utils/sortUtils.ts +0 -250
  42. package/ai/agent/web/referencePickerUtils.ts +0 -146
  43. package/ai/ai.locale.ts +0 -1079
  44. package/ai/chat/accumulateToolCallChunks.ts +0 -95
  45. package/ai/chat/fetchUtils.native.ts +0 -276
  46. package/ai/chat/fetchUtils.ts +0 -153
  47. package/ai/chat/parseApiError.ts +0 -64
  48. package/ai/chat/parseMultilineSSE.ts +0 -95
  49. package/ai/chat/sendOpenAICompletionsRequest.native.ts +0 -682
  50. package/ai/chat/sendOpenAICompletionsRequest.ts +0 -703
  51. package/ai/chat/sendOpenAIResponseRequest.ts +0 -491
  52. package/ai/chat/shouldUseServerProxy.ts +0 -18
  53. package/ai/chat/sseClient.native.ts +0 -91
  54. package/ai/chat/sseClient.ts +0 -67
  55. package/ai/chat/streamReader.native.ts +0 -31
  56. package/ai/chat/streamReader.ts +0 -62
  57. package/ai/chat/updateTotalUsage.ts +0 -72
  58. package/ai/context/buildReferenceContext.ts +0 -437
  59. package/ai/context/calculateContextUsage.ts +0 -133
  60. package/ai/context/retention.ts +0 -165
  61. package/ai/context/tokenUtils.ts +0 -78
  62. package/ai/index.ts +0 -1
  63. package/ai/llm/calculateGeminiImageTokens.ts +0 -57
  64. package/ai/llm/deepinfra.ts +0 -28
  65. package/ai/llm/fireworks.ts +0 -50
  66. package/ai/llm/generateRequestBody.ts +0 -165
  67. package/ai/llm/getModelContextWindow.ts +0 -84
  68. package/ai/llm/getNoloKey.ts +0 -31
  69. package/ai/llm/getPricing.ts +0 -199
  70. package/ai/llm/hooks/useModelPricing.ts +0 -75
  71. package/ai/llm/imagePricing.ts +0 -40
  72. package/ai/llm/isResponseAPIModel.ts +0 -13
  73. package/ai/llm/mimo.ts +0 -71
  74. package/ai/llm/mistral.ts +0 -22
  75. package/ai/llm/modelAvatar.ts +0 -427
  76. package/ai/llm/models.ts +0 -45
  77. package/ai/llm/openrouterModels.ts +0 -269
  78. package/ai/llm/providers.ts +0 -306
  79. package/ai/llm/reasoningModels.ts +0 -28
  80. package/ai/llm/types.ts +0 -59
  81. package/ai/llm/usageRequestOptions.ts +0 -59
  82. package/ai/memory/capture.ts +0 -148
  83. package/ai/memory/consolidate.ts +0 -104
  84. package/ai/memory/delete.ts +0 -147
  85. package/ai/memory/overlay.ts +0 -84
  86. package/ai/memory/query.ts +0 -38
  87. package/ai/memory/queryShared.ts +0 -160
  88. package/ai/memory/rank.ts +0 -105
  89. package/ai/memory/recentRelationshipRecap.ts +0 -249
  90. package/ai/memory/remember.ts +0 -167
  91. package/ai/memory/runtime.ts +0 -76
  92. package/ai/memory/store.ts +0 -20
  93. package/ai/memory/storeShared.ts +0 -76
  94. package/ai/memory/types.ts +0 -46
  95. package/ai/memory/understanding.ts +0 -349
  96. package/ai/memory/understandingGreeting.ts +0 -264
  97. package/ai/messages/type.ts +0 -20
  98. package/ai/policy/personalizationDialog.ts +0 -333
  99. package/ai/policy/runtimePolicy.ts +0 -440
  100. package/ai/policy/selfUpdateFields.ts +0 -48
  101. package/ai/policy/types.ts +0 -64
  102. package/ai/skills/referenceRuntime.ts +0 -274
  103. package/ai/skills/skillDiagnostics.ts +0 -251
  104. package/ai/skills/skillDocBuilder.ts +0 -139
  105. package/ai/skills/skillDocProtocol.ts +0 -434
  106. package/ai/skills/skillReferenceSummary.ts +0 -63
  107. package/ai/skills/skillSummaryMarker.ts +0 -26
  108. package/ai/token/calculatePrice.ts +0 -544
  109. package/ai/token/db.ts +0 -98
  110. package/ai/token/externalToolCost.ts +0 -330
  111. package/ai/token/hooks/useRecords.ts +0 -65
  112. package/ai/token/missingUsageEstimate.ts +0 -42
  113. package/ai/token/modelUsageQuery.ts +0 -252
  114. package/ai/token/normalizeUsage.ts +0 -84
  115. package/ai/token/openaiImageGenerationUsage.ts +0 -56
  116. package/ai/token/prepareTokenUsageData.ts +0 -88
  117. package/ai/token/query.ts +0 -88
  118. package/ai/token/queryUserTokens.ts +0 -59
  119. package/ai/token/resolveBillingTarget.ts +0 -52
  120. package/ai/token/saveTokenRecord.ts +0 -53
  121. package/ai/token/serverDialogProjection.ts +0 -78
  122. package/ai/token/serverTokenWriter.ts +0 -143
  123. package/ai/token/stats.ts +0 -21
  124. package/ai/token/tokenThunks.ts +0 -24
  125. package/ai/token/types.ts +0 -93
  126. package/ai/tools/agent/agentTools.ts +0 -176
  127. package/ai/tools/agent/agentUpdateShared.ts +0 -311
  128. package/ai/tools/agent/callAgentTool.ts +0 -139
  129. package/ai/tools/agent/createAgentTool.ts +0 -512
  130. package/ai/tools/agent/createDialogTool.ts +0 -69
  131. package/ai/tools/agent/createSkillAgentTool.ts +0 -62
  132. package/ai/tools/agent/parallelBudget.ts +0 -221
  133. package/ai/tools/agent/presets/appBuilderPreset.ts +0 -145
  134. package/ai/tools/agent/runLlmTool.ts +0 -96
  135. package/ai/tools/agent/runStreamingAgentTool.ts +0 -73
  136. package/ai/tools/agent/skillAgentArgs.ts +0 -106
  137. package/ai/tools/agent/skillAgentPreset.ts +0 -89
  138. package/ai/tools/agent/streamParallelAgentsTool.ts +0 -122
  139. package/ai/tools/agent/updateAgentTool.ts +0 -96
  140. package/ai/tools/agent/updateSelfTool.ts +0 -113
  141. package/ai/tools/amazonProductScraperTool.ts +0 -86
  142. package/ai/tools/apifyActorClient.ts +0 -45
  143. package/ai/tools/appEditGuard.ts +0 -372
  144. package/ai/tools/appReadSnapshot.ts +0 -153
  145. package/ai/tools/appTools.ts +0 -1549
  146. package/ai/tools/applyEditTool.ts +0 -256
  147. package/ai/tools/applyLineEditsTool.ts +0 -312
  148. package/ai/tools/browserTools/click.ts +0 -33
  149. package/ai/tools/browserTools/closeSession.ts +0 -29
  150. package/ai/tools/browserTools/common.ts +0 -27
  151. package/ai/tools/browserTools/openSession.ts +0 -48
  152. package/ai/tools/browserTools/readContent.ts +0 -38
  153. package/ai/tools/browserTools/selectOption.ts +0 -46
  154. package/ai/tools/browserTools/typeText.ts +0 -42
  155. package/ai/tools/category/createCategoryTool.ts +0 -66
  156. package/ai/tools/category/queryContentsByCategoryTool.ts +0 -69
  157. package/ai/tools/category/updateContentCategoryTool.ts +0 -75
  158. package/ai/tools/cfBrowserTools.ts +0 -319
  159. package/ai/tools/cfSpeechToTextTool.ts +0 -49
  160. package/ai/tools/checkEnvTool.ts +0 -65
  161. package/ai/tools/cloudflareCrawlTool.ts +0 -289
  162. package/ai/tools/codeSearchTool.ts +0 -111
  163. package/ai/tools/codeTools.ts +0 -101
  164. package/ai/tools/createDocTool.ts +0 -132
  165. package/ai/tools/createPlanTool.ts +0 -999
  166. package/ai/tools/createSkillDocTool.ts +0 -155
  167. package/ai/tools/createWorkflowTool.ts +0 -154
  168. package/ai/tools/deepseekOcrTool.ts +0 -34
  169. package/ai/tools/delayTool.ts +0 -31
  170. package/ai/tools/deleteSpacesTool.ts +0 -325
  171. package/ai/tools/deleteSpacesToolModel.ts +0 -159
  172. package/ai/tools/devReloadUtils.ts +0 -29
  173. package/ai/tools/dialogMessageSearch.ts +0 -137
  174. package/ai/tools/doctorSkillTool.ts +0 -72
  175. package/ai/tools/ecommerceScraperTool.ts +0 -86
  176. package/ai/tools/emailTools.ts +0 -549
  177. package/ai/tools/evalSkillTool.ts +0 -92
  178. package/ai/tools/exaSearchTool.ts +0 -64
  179. package/ai/tools/execBashTool.ts +0 -379
  180. package/ai/tools/executeSqlTool.ts +0 -192
  181. package/ai/tools/fetchWebpageSupport.ts +0 -309
  182. package/ai/tools/fetchWebpageTool.ts +0 -84
  183. package/ai/tools/geminiImagePreviewTool.ts +0 -361
  184. package/ai/tools/generateDocxTool.ts +0 -215
  185. package/ai/tools/googleSearchScraperTool.ts +0 -106
  186. package/ai/tools/importDataTool.ts +0 -133
  187. package/ai/tools/importSkillTool.ts +0 -162
  188. package/ai/tools/index.ts +0 -1858
  189. package/ai/tools/listFilesTool.ts +0 -82
  190. package/ai/tools/listUserSpacesTool.ts +0 -113
  191. package/ai/tools/modelUsageTools.ts +0 -142
  192. package/ai/tools/olmOcrTool.ts +0 -34
  193. package/ai/tools/openaiImageTool.ts +0 -218
  194. package/ai/tools/paddleOcrTool.ts +0 -34
  195. package/ai/tools/prepareTools.ts +0 -23
  196. package/ai/tools/readDocTool.ts +0 -84
  197. package/ai/tools/readFileTool.ts +0 -211
  198. package/ai/tools/readTool.ts +0 -163
  199. package/ai/tools/readXPostTool.ts +0 -233
  200. package/ai/tools/rememberMemoryTool.ts +0 -84
  201. package/ai/tools/remotionVideoTool.ts +0 -151
  202. package/ai/tools/searchDialogMessagesTool.ts +0 -222
  203. package/ai/tools/searchRepoTool.ts +0 -115
  204. package/ai/tools/searchWorkspaceTool.ts +0 -259
  205. package/ai/tools/skillFollowup.ts +0 -86
  206. package/ai/tools/surfWeatherTool.ts +0 -169
  207. package/ai/tools/table/addTableRowTool.ts +0 -217
  208. package/ai/tools/table/createTableTool.ts +0 -315
  209. package/ai/tools/table/rowTools.ts +0 -366
  210. package/ai/tools/table/schemaTools.ts +0 -244
  211. package/ai/tools/table/shareTableTool.ts +0 -148
  212. package/ai/tools/table/toolShared.ts +0 -129
  213. package/ai/tools/toolApiClient.ts +0 -198
  214. package/ai/tools/toolNameAliases.ts +0 -57
  215. package/ai/tools/toolResultError.ts +0 -42
  216. package/ai/tools/toolRunSlice.ts +0 -303
  217. package/ai/tools/toolSchemaCompatibility.ts +0 -53
  218. package/ai/tools/toolVisibility.ts +0 -4
  219. package/ai/tools/types.ts +0 -20
  220. package/ai/tools/uiAskChoiceTool.ts +0 -104
  221. package/ai/tools/updateContentTitleTool.ts +0 -84
  222. package/ai/tools/updateDocTool.ts +0 -105
  223. package/ai/tools/updateUserPreferenceProfileTool.ts +0 -145
  224. package/ai/tools/whisperTool.ts +0 -77
  225. package/ai/tools/writeFileTool.ts +0 -210
  226. package/ai/tools/youtubeScraperTool.ts +0 -116
  227. package/ai/tools/ziweiChartTool.ts +0 -678
  228. package/ai/types.ts +0 -55
  229. package/ai/workflow/workflowExecutor.ts +0 -323
  230. package/ai/workflow/workflowSlice.ts +0 -73
  231. package/ai/workflow/workflowTypes.ts +0 -106
  232. package/connector-experimental/index.ts +0 -5
@@ -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
- }