nolo-cli 0.1.8 → 0.1.10

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 (239) hide show
  1. package/README.md +32 -0
  2. package/agentRuntimeCommands.ts +3 -3
  3. package/ai/agent/_executeModel.ts +118 -0
  4. package/ai/agent/agentSlice.ts +525 -0
  5. package/ai/agent/appWorkingMemory.ts +126 -0
  6. package/ai/agent/avatarUtils.ts +24 -0
  7. package/ai/agent/buildEditingContext.ts +373 -0
  8. package/ai/agent/buildSystemPrompt.ts +532 -0
  9. package/ai/agent/cleanAgentMessages.ts +140 -0
  10. package/ai/agent/cliChatClient.ts +119 -0
  11. package/ai/agent/cliExecutor.ts +733 -0
  12. package/ai/agent/cliPrompt.ts +10 -0
  13. package/ai/agent/contextCompiler.ts +107 -0
  14. package/ai/agent/contextLayerContract.ts +44 -0
  15. package/ai/agent/createAgentSchema.ts +234 -0
  16. package/ai/agent/executeToolCall.ts +58 -0
  17. package/ai/agent/fetchAgentContexts.ts +42 -0
  18. package/ai/agent/generatePrompt.ts +3 -0
  19. package/ai/agent/getFullChatContextKeys.ts +168 -0
  20. package/ai/agent/hooks/fetchPublicAgents.ts +133 -0
  21. package/ai/agent/hooks/useAgentConfig.ts +61 -0
  22. package/ai/agent/hooks/useAgentDialog.ts +35 -0
  23. package/ai/agent/hooks/useAgentFormValidation.ts +202 -0
  24. package/ai/agent/hooks/usePublicAgents.ts +473 -0
  25. package/ai/agent/machineRunPermissions.ts +95 -0
  26. package/ai/agent/persistMessageWithFixedId.ts +37 -0
  27. package/ai/agent/planSlice.ts +259 -0
  28. package/ai/agent/referenceUtils.ts +229 -0
  29. package/ai/agent/runAgentBackground.ts +238 -0
  30. package/ai/agent/runAgentClientLoop.ts +138 -0
  31. package/ai/agent/runtimeGuidance.ts +97 -0
  32. package/ai/agent/runtimeServerBase.ts +37 -0
  33. package/ai/agent/server/fetchPublicAgents.ts +128 -0
  34. package/ai/agent/startParallelAgentStreams.ts +424 -0
  35. package/ai/agent/startupProtocol.ts +53 -0
  36. package/ai/agent/streamAgentChatTurn.ts +1278 -0
  37. package/ai/agent/streamAgentChatTurnUtils.ts +738 -0
  38. package/ai/agent/types.ts +71 -0
  39. package/ai/agent/utils/imageOutput.ts +33 -0
  40. package/ai/agent/utils/sortUtils.ts +250 -0
  41. package/ai/agent/web/referencePickerUtils.ts +146 -0
  42. package/ai/ai.locale.ts +1079 -0
  43. package/ai/chat/accumulateToolCallChunks.ts +95 -0
  44. package/ai/chat/fetchUtils.native.ts +276 -0
  45. package/ai/chat/fetchUtils.ts +153 -0
  46. package/ai/chat/parseApiError.ts +64 -0
  47. package/ai/chat/parseMultilineSSE.ts +95 -0
  48. package/ai/chat/sendOpenAICompletionsRequest.native.ts +682 -0
  49. package/ai/chat/sendOpenAICompletionsRequest.ts +703 -0
  50. package/ai/chat/sendOpenAIResponseRequest.ts +491 -0
  51. package/ai/chat/shouldUseServerProxy.ts +18 -0
  52. package/ai/chat/sseClient.native.ts +91 -0
  53. package/ai/chat/sseClient.ts +67 -0
  54. package/ai/chat/streamReader.native.ts +31 -0
  55. package/ai/chat/streamReader.ts +62 -0
  56. package/ai/chat/updateTotalUsage.ts +72 -0
  57. package/ai/context/buildReferenceContext.ts +437 -0
  58. package/ai/context/calculateContextUsage.ts +133 -0
  59. package/ai/context/retention.ts +165 -0
  60. package/ai/context/tokenUtils.ts +78 -0
  61. package/ai/index.ts +1 -0
  62. package/ai/llm/calculateGeminiImageTokens.ts +57 -0
  63. package/ai/llm/deepinfra.ts +28 -0
  64. package/ai/llm/fireworks.ts +50 -0
  65. package/ai/llm/generateRequestBody.ts +165 -0
  66. package/ai/llm/getModelContextWindow.ts +84 -0
  67. package/ai/llm/getNoloKey.ts +31 -0
  68. package/ai/llm/getPricing.ts +199 -0
  69. package/ai/llm/hooks/useModelPricing.ts +75 -0
  70. package/ai/llm/imagePricing.ts +40 -0
  71. package/ai/llm/isResponseAPIModel.ts +13 -0
  72. package/ai/llm/mimo.ts +71 -0
  73. package/ai/llm/mistral.ts +22 -0
  74. package/ai/llm/modelAvatar.ts +427 -0
  75. package/ai/llm/models.ts +45 -0
  76. package/ai/llm/openrouterModels.ts +269 -0
  77. package/ai/llm/providers.ts +306 -0
  78. package/ai/llm/reasoningModels.ts +28 -0
  79. package/ai/llm/types.ts +59 -0
  80. package/ai/llm/usageRequestOptions.ts +59 -0
  81. package/ai/memory/capture.ts +148 -0
  82. package/ai/memory/consolidate.ts +104 -0
  83. package/ai/memory/delete.ts +147 -0
  84. package/ai/memory/overlay.ts +84 -0
  85. package/ai/memory/query.ts +38 -0
  86. package/ai/memory/queryShared.ts +160 -0
  87. package/ai/memory/rank.ts +105 -0
  88. package/ai/memory/recentRelationshipRecap.ts +249 -0
  89. package/ai/memory/remember.ts +167 -0
  90. package/ai/memory/runtime.ts +76 -0
  91. package/ai/memory/store.ts +20 -0
  92. package/ai/memory/storeShared.ts +76 -0
  93. package/ai/memory/types.ts +46 -0
  94. package/ai/memory/understanding.ts +349 -0
  95. package/ai/memory/understandingGreeting.ts +264 -0
  96. package/ai/messages/type.ts +20 -0
  97. package/ai/policy/personalizationDialog.ts +333 -0
  98. package/ai/policy/runtimePolicy.ts +440 -0
  99. package/ai/policy/selfUpdateFields.ts +48 -0
  100. package/ai/policy/types.ts +64 -0
  101. package/ai/skills/referenceRuntime.ts +274 -0
  102. package/ai/skills/skillDiagnostics.ts +251 -0
  103. package/ai/skills/skillDocBuilder.ts +139 -0
  104. package/ai/skills/skillDocProtocol.ts +434 -0
  105. package/ai/skills/skillReferenceSummary.ts +63 -0
  106. package/ai/skills/skillSummaryMarker.ts +26 -0
  107. package/ai/token/calculatePrice.ts +544 -0
  108. package/ai/token/db.ts +98 -0
  109. package/ai/token/externalToolCost.ts +330 -0
  110. package/ai/token/hooks/useRecords.ts +65 -0
  111. package/ai/token/missingUsageEstimate.ts +42 -0
  112. package/ai/token/modelUsageQuery.ts +252 -0
  113. package/ai/token/normalizeUsage.ts +84 -0
  114. package/ai/token/openaiImageGenerationUsage.ts +56 -0
  115. package/ai/token/prepareTokenUsageData.ts +88 -0
  116. package/ai/token/query.ts +88 -0
  117. package/ai/token/queryUserTokens.ts +59 -0
  118. package/ai/token/resolveBillingTarget.ts +52 -0
  119. package/ai/token/saveTokenRecord.ts +53 -0
  120. package/ai/token/serverDialogProjection.ts +78 -0
  121. package/ai/token/serverTokenWriter.ts +143 -0
  122. package/ai/token/stats.ts +21 -0
  123. package/ai/token/tokenThunks.ts +24 -0
  124. package/ai/token/types.ts +93 -0
  125. package/ai/tools/agent/agentTools.ts +176 -0
  126. package/ai/tools/agent/agentUpdateShared.ts +311 -0
  127. package/ai/tools/agent/callAgentTool.ts +139 -0
  128. package/ai/tools/agent/createAgentTool.ts +512 -0
  129. package/ai/tools/agent/createDialogTool.ts +69 -0
  130. package/ai/tools/agent/createSkillAgentTool.ts +62 -0
  131. package/ai/tools/agent/parallelBudget.ts +221 -0
  132. package/ai/tools/agent/presets/appBuilderPreset.ts +145 -0
  133. package/ai/tools/agent/runLlmTool.ts +96 -0
  134. package/ai/tools/agent/runStreamingAgentTool.ts +73 -0
  135. package/ai/tools/agent/skillAgentArgs.ts +106 -0
  136. package/ai/tools/agent/skillAgentPreset.ts +89 -0
  137. package/ai/tools/agent/streamParallelAgentsTool.ts +122 -0
  138. package/ai/tools/agent/updateAgentTool.ts +96 -0
  139. package/ai/tools/agent/updateSelfTool.ts +113 -0
  140. package/ai/tools/amazonProductScraperTool.ts +86 -0
  141. package/ai/tools/apifyActorClient.ts +45 -0
  142. package/ai/tools/appEditGuard.ts +372 -0
  143. package/ai/tools/appReadSnapshot.ts +153 -0
  144. package/ai/tools/appTools.ts +1549 -0
  145. package/ai/tools/applyEditTool.ts +256 -0
  146. package/ai/tools/applyLineEditsTool.ts +312 -0
  147. package/ai/tools/browserTools/click.ts +33 -0
  148. package/ai/tools/browserTools/closeSession.ts +29 -0
  149. package/ai/tools/browserTools/common.ts +27 -0
  150. package/ai/tools/browserTools/openSession.ts +48 -0
  151. package/ai/tools/browserTools/readContent.ts +38 -0
  152. package/ai/tools/browserTools/selectOption.ts +46 -0
  153. package/ai/tools/browserTools/typeText.ts +42 -0
  154. package/ai/tools/category/createCategoryTool.ts +66 -0
  155. package/ai/tools/category/queryContentsByCategoryTool.ts +69 -0
  156. package/ai/tools/category/updateContentCategoryTool.ts +75 -0
  157. package/ai/tools/cfBrowserTools.ts +319 -0
  158. package/ai/tools/cfSpeechToTextTool.ts +49 -0
  159. package/ai/tools/checkEnvTool.ts +65 -0
  160. package/ai/tools/cloudflareCrawlTool.ts +289 -0
  161. package/ai/tools/codeSearchTool.ts +111 -0
  162. package/ai/tools/codeTools.ts +101 -0
  163. package/ai/tools/createDocTool.ts +132 -0
  164. package/ai/tools/createPlanTool.ts +999 -0
  165. package/ai/tools/createSkillDocTool.ts +155 -0
  166. package/ai/tools/createWorkflowTool.ts +154 -0
  167. package/ai/tools/deepseekOcrTool.ts +34 -0
  168. package/ai/tools/delayTool.ts +31 -0
  169. package/ai/tools/deleteSpacesTool.ts +325 -0
  170. package/ai/tools/deleteSpacesToolModel.ts +159 -0
  171. package/ai/tools/devReloadUtils.ts +29 -0
  172. package/ai/tools/dialogMessageSearch.ts +137 -0
  173. package/ai/tools/doctorSkillTool.ts +72 -0
  174. package/ai/tools/ecommerceScraperTool.ts +86 -0
  175. package/ai/tools/emailTools.ts +549 -0
  176. package/ai/tools/evalSkillTool.ts +92 -0
  177. package/ai/tools/exaSearchTool.ts +64 -0
  178. package/ai/tools/execBashTool.ts +379 -0
  179. package/ai/tools/executeSqlTool.ts +192 -0
  180. package/ai/tools/fetchWebpageSupport.ts +309 -0
  181. package/ai/tools/fetchWebpageTool.ts +84 -0
  182. package/ai/tools/geminiImagePreviewTool.ts +361 -0
  183. package/ai/tools/generateDocxTool.ts +215 -0
  184. package/ai/tools/googleSearchScraperTool.ts +106 -0
  185. package/ai/tools/importDataTool.ts +133 -0
  186. package/ai/tools/importSkillTool.ts +162 -0
  187. package/ai/tools/index.ts +1858 -0
  188. package/ai/tools/listFilesTool.ts +82 -0
  189. package/ai/tools/listUserSpacesTool.ts +113 -0
  190. package/ai/tools/modelUsageTools.ts +142 -0
  191. package/ai/tools/olmOcrTool.ts +34 -0
  192. package/ai/tools/openaiImageTool.ts +218 -0
  193. package/ai/tools/paddleOcrTool.ts +34 -0
  194. package/ai/tools/prepareTools.ts +23 -0
  195. package/ai/tools/readDocTool.ts +84 -0
  196. package/ai/tools/readFileTool.ts +211 -0
  197. package/ai/tools/readTool.ts +163 -0
  198. package/ai/tools/readXPostTool.ts +233 -0
  199. package/ai/tools/rememberMemoryTool.ts +84 -0
  200. package/ai/tools/remotionVideoTool.ts +151 -0
  201. package/ai/tools/searchDialogMessagesTool.ts +222 -0
  202. package/ai/tools/searchRepoTool.ts +115 -0
  203. package/ai/tools/searchWorkspaceTool.ts +259 -0
  204. package/ai/tools/skillFollowup.ts +86 -0
  205. package/ai/tools/surfWeatherTool.ts +169 -0
  206. package/ai/tools/table/addTableRowTool.ts +217 -0
  207. package/ai/tools/table/createTableTool.ts +315 -0
  208. package/ai/tools/table/rowTools.ts +366 -0
  209. package/ai/tools/table/schemaTools.ts +244 -0
  210. package/ai/tools/table/shareTableTool.ts +148 -0
  211. package/ai/tools/table/toolShared.ts +129 -0
  212. package/ai/tools/toolApiClient.ts +198 -0
  213. package/ai/tools/toolNameAliases.ts +57 -0
  214. package/ai/tools/toolResultError.ts +42 -0
  215. package/ai/tools/toolRunSlice.ts +303 -0
  216. package/ai/tools/toolSchemaCompatibility.ts +53 -0
  217. package/ai/tools/toolVisibility.ts +4 -0
  218. package/ai/tools/types.ts +20 -0
  219. package/ai/tools/uiAskChoiceTool.ts +104 -0
  220. package/ai/tools/updateContentTitleTool.ts +84 -0
  221. package/ai/tools/updateDocTool.ts +105 -0
  222. package/ai/tools/updateUserPreferenceProfileTool.ts +145 -0
  223. package/ai/tools/whisperTool.ts +77 -0
  224. package/ai/tools/writeFileTool.ts +210 -0
  225. package/ai/tools/youtubeScraperTool.ts +116 -0
  226. package/ai/tools/ziweiChartTool.ts +678 -0
  227. package/ai/types.ts +55 -0
  228. package/ai/workflow/workflowExecutor.ts +323 -0
  229. package/ai/workflow/workflowSlice.ts +73 -0
  230. package/ai/workflow/workflowTypes.ts +106 -0
  231. package/client/compactDialog.ts +222 -0
  232. package/connector-experimental/capabilities.ts +73 -0
  233. package/connector-experimental/codexBinary.ts +41 -0
  234. package/connector-experimental/heartbeatLoop.ts +22 -0
  235. package/connector-experimental/index.ts +5 -0
  236. package/connector-experimental/machineInfo.ts +46 -0
  237. package/connector-experimental/protocol.ts +54 -0
  238. package/machineCommands.ts +4 -4
  239. package/package.json +22 -6
@@ -0,0 +1,256 @@
1
+ import { bumpDevReloadSuppressIfSelfEditing } from "./devReloadUtils";
2
+ import { getToolBaseUrl } from "./toolApiClient";
3
+
4
+ type EditMatchOptions = {
5
+ occurrence?: number;
6
+ expectedMatches?: number;
7
+ replaceAll?: boolean;
8
+ };
9
+
10
+ export type ApplyEdit =
11
+ | (EditMatchOptions & {
12
+ type: "replace";
13
+ oldText: string;
14
+ newText: string;
15
+ })
16
+ | (EditMatchOptions & {
17
+ type: "insertBefore";
18
+ anchor: string;
19
+ content: string;
20
+ })
21
+ | (EditMatchOptions & {
22
+ type: "insertAfter";
23
+ anchor: string;
24
+ content: string;
25
+ })
26
+ | (EditMatchOptions & {
27
+ type: "delete";
28
+ oldText: string;
29
+ });
30
+
31
+ export type ApplyEditArgs = {
32
+ filePath: string;
33
+ edits: ApplyEdit[];
34
+ };
35
+
36
+ export const applyEditFunctionSchema = {
37
+ name: "applyEdit",
38
+ description: [
39
+ "推荐默认使用的代码编辑工具:基于“精确文本片段”而不是行号进行局部修改。",
40
+ "",
41
+ "适用场景:",
42
+ "- 替换一个已知的唯一代码片段。",
43
+ "- 在某个唯一锚点片段之前或之后插入代码。",
44
+ "- 删除一个精确可定位的片段。",
45
+ "",
46
+ "推荐工作流:",
47
+ "- 先用 codeSearch 找文件或搜索代码;",
48
+ "- 再用 readFile 读取最新内容;",
49
+ "- 如果这次修改依赖用户给的 URL / 文档页,先用 fetchWebpage 或其它网页工具抓取真值,再编辑;",
50
+ "- 直接把 readFile 中复制出的精确片段作为 oldText / anchor 传入。",
51
+ "- 对从网页提取出的价格、上下文窗口、能力字段,按抓到的原值写入,不要凭记忆改写。",
52
+ "",
53
+ "准确率约定:",
54
+ "- 默认要求片段在文件中只匹配 1 次;",
55
+ "- 若匹配多次,会直接报错,要求你补更多上下文、设置 occurrence,或显式 replaceAll;",
56
+ "- replaceAll 只会在你显式传 replaceAll=true 时执行;",
57
+ "- 可用 expectedMatches 断言匹配数量,进一步防止误改。",
58
+ "",
59
+ "返回数据约定(rawData.response 中):",
60
+ "- newContent:编辑后的完整文件内容(统一使用 \\n)。",
61
+ "- diff:基于原始内容与 newContent 的行级 diff。",
62
+ "- appliedEdits:每个 edit 的匹配次数与实际应用次数。",
63
+ "",
64
+ "注意:",
65
+ "- filePath 必须是相对项目根目录(bun-nolo)的路径。",
66
+ "- 优先用 applyEdit 做最小局部修改;只有确实无法用精确片段表达时,再考虑 applyLineEdits 或整文件覆盖。",
67
+ ].join("\n"),
68
+ parameters: {
69
+ type: "object",
70
+ properties: {
71
+ filePath: {
72
+ type: "string",
73
+ description:
74
+ "要修改的文件相对项目根目录(bun-nolo)的路径,例如: packages/server/routes.ts",
75
+ },
76
+ edits: {
77
+ type: "array",
78
+ description:
79
+ "要按顺序应用的一组精确片段编辑。后一个 edit 会基于前一个 edit 应用后的最新文本继续执行。",
80
+ items: {
81
+ type: "object",
82
+ oneOf: [
83
+ {
84
+ properties: {
85
+ type: { const: "replace" },
86
+ oldText: {
87
+ type: "string",
88
+ description: "要被替换的精确原始片段。",
89
+ },
90
+ newText: {
91
+ type: "string",
92
+ description: "替换后的新片段。",
93
+ },
94
+ occurrence: {
95
+ type: "number",
96
+ description: "可选:当 oldText 匹配多次时,替换第几次匹配(1-based)。",
97
+ },
98
+ expectedMatches: {
99
+ type: "number",
100
+ description: "可选:断言 oldText 总共应匹配多少次。",
101
+ },
102
+ replaceAll: {
103
+ type: "boolean",
104
+ description: "可选:为 true 时替换所有匹配项。",
105
+ },
106
+ },
107
+ required: ["type", "oldText", "newText"],
108
+ },
109
+ {
110
+ properties: {
111
+ type: { const: "insertBefore" },
112
+ anchor: {
113
+ type: "string",
114
+ description: "作为插入锚点的精确原始片段。",
115
+ },
116
+ content: {
117
+ type: "string",
118
+ description: "要插入到锚点之前的文本。",
119
+ },
120
+ occurrence: {
121
+ type: "number",
122
+ description: "可选:当 anchor 匹配多次时,插入到第几次匹配之前(1-based)。",
123
+ },
124
+ expectedMatches: {
125
+ type: "number",
126
+ description: "可选:断言 anchor 总共应匹配多少次。",
127
+ },
128
+ replaceAll: {
129
+ type: "boolean",
130
+ description: "可选:为 true 时,对所有匹配锚点都执行插入。",
131
+ },
132
+ },
133
+ required: ["type", "anchor", "content"],
134
+ },
135
+ {
136
+ properties: {
137
+ type: { const: "insertAfter" },
138
+ anchor: {
139
+ type: "string",
140
+ description: "作为插入锚点的精确原始片段。",
141
+ },
142
+ content: {
143
+ type: "string",
144
+ description: "要插入到锚点之后的文本。",
145
+ },
146
+ occurrence: {
147
+ type: "number",
148
+ description: "可选:当 anchor 匹配多次时,插入到第几次匹配之后(1-based)。",
149
+ },
150
+ expectedMatches: {
151
+ type: "number",
152
+ description: "可选:断言 anchor 总共应匹配多少次。",
153
+ },
154
+ replaceAll: {
155
+ type: "boolean",
156
+ description: "可选:为 true 时,对所有匹配锚点都执行插入。",
157
+ },
158
+ },
159
+ required: ["type", "anchor", "content"],
160
+ },
161
+ {
162
+ properties: {
163
+ type: { const: "delete" },
164
+ oldText: {
165
+ type: "string",
166
+ description: "要删除的精确原始片段。",
167
+ },
168
+ occurrence: {
169
+ type: "number",
170
+ description: "可选:当 oldText 匹配多次时,删除第几次匹配(1-based)。",
171
+ },
172
+ expectedMatches: {
173
+ type: "number",
174
+ description: "可选:断言 oldText 总共应匹配多少次。",
175
+ },
176
+ replaceAll: {
177
+ type: "boolean",
178
+ description: "可选:为 true 时删除所有匹配项。",
179
+ },
180
+ },
181
+ required: ["type", "oldText"],
182
+ },
183
+ ],
184
+ },
185
+ },
186
+ },
187
+ required: ["filePath", "edits"],
188
+ },
189
+ };
190
+
191
+ function assertValidArgs(args: ApplyEditArgs): void {
192
+ if (!args.filePath || typeof args.filePath !== "string") {
193
+ throw new Error("应用代码编辑失败:必须提供有效的 filePath 字符串。");
194
+ }
195
+ if (!Array.isArray(args.edits) || args.edits.length === 0) {
196
+ throw new Error("应用代码编辑失败:edits 必须是非空数组。");
197
+ }
198
+ }
199
+
200
+ export async function applyEditFunc(
201
+ args: ApplyEditArgs,
202
+ thunkApi: any,
203
+ context?: { parentMessageId?: string; signal?: AbortSignal }
204
+ ): Promise<{ rawData: any; displayData?: string }> {
205
+ assertValidArgs(args);
206
+
207
+ const { filePath, edits } = args;
208
+ bumpDevReloadSuppressIfSelfEditing(filePath);
209
+
210
+ try {
211
+ const baseUrl = getToolBaseUrl(thunkApi);
212
+
213
+ if (!baseUrl) {
214
+ throw new Error("应用代码编辑失败:无法获取 applyEdit 服务器地址。");
215
+ }
216
+
217
+ const apiUrl = `${baseUrl.replace(/\/+$/, "")}/api/apply-edit`;
218
+ const response = await fetch(apiUrl, {
219
+ method: "POST",
220
+ headers: { "Content-Type": "application/json" },
221
+ signal: context?.signal,
222
+ body: JSON.stringify({ filePath, edits }),
223
+ });
224
+
225
+ const textBody = await response.text();
226
+ let data: any = {};
227
+ try {
228
+ data = textBody ? JSON.parse(textBody) : {};
229
+ } catch {
230
+ // ignore JSON parse error
231
+ }
232
+
233
+ if (!response.ok || data?.error) {
234
+ const errMsg =
235
+ data?.error ||
236
+ `applyEdit API 请求失败,状态码: ${response.status}. 响应: ${textBody}`;
237
+ console.error("applyEdit API Error:", errMsg);
238
+ throw new Error(errMsg);
239
+ }
240
+
241
+ return {
242
+ rawData: {
243
+ applied: true,
244
+ filePath,
245
+ editCount: edits.length,
246
+ response: data,
247
+ },
248
+ displayData: `✅ 已成功对文件应用 ${edits.length} 个精确片段编辑: ${filePath}`,
249
+ };
250
+ } catch (error: any) {
251
+ console.error("执行 applyEdit 时发生错误:", error);
252
+ throw new Error(
253
+ `应用代码编辑到文件 (${filePath}) 失败:${error?.message || String(error)}`
254
+ );
255
+ }
256
+ }
@@ -0,0 +1,312 @@
1
+ // 文件路径: packages/ai/tools/applyLineEditsTool.ts
2
+
3
+ import { bumpDevReloadSuppressIfSelfEditing } from "./devReloadUtils";
4
+ import { getToolBaseUrl } from "./toolApiClient";
5
+
6
+ // ---- Types ----
7
+
8
+ /**
9
+ * 提高编辑成功率用的公共字段:
10
+ * - 建议 LLM 在规划编辑前先调用 readFile 获取最新内容;
11
+ * - 然后把“将要修改的那几行原始代码”(用 \n 拼接)复制到 originalSnippet。
12
+ *
13
+ * 后端会把行号当作“粗定位”,再结合 originalSnippet 在附近搜索精确位置,
14
+ * 这样即使行号有轻微偏差,也有很大概率能对齐到正确位置再修改,而不是直接失败。
15
+ */
16
+ export type LineEditCommon = {
17
+ /**
18
+ * 可选:LLM 看到的原始代码片段。
19
+ *
20
+ * 用法建议:
21
+ * - replaceRange / deleteRange:
22
+ * 把 startLine~endLine 对应的原始行文本(用 \n 拼接)原样复制到 originalSnippet。
23
+ * - insertBefore / insertAfter:
24
+ * 把插入点附近的一行原始代码(通常是当前这一行)复制到 originalSnippet。
25
+ *
26
+ * 行为说明:
27
+ * - 后端会先按行号找到大致区域;
28
+ * - 若提供了 originalSnippet,会在该区域附近优先搜索同样的片段;
29
+ * - 若找到,就以搜索结果为准自动修正编辑位置;
30
+ * - 若找不到,则退化为按行号硬改(与旧行为一致),不会因此额外报错。
31
+ */
32
+ originalSnippet?: string;
33
+ };
34
+
35
+ export type LineEdit =
36
+ | (LineEditCommon & {
37
+ type: "replaceRange";
38
+ /**
39
+ * 期望被替换的起始、结束行(包含),1-based。
40
+ * 实际应用时,后端会以此为“粗定位”,并结合 originalSnippet 做精确对齐。
41
+ */
42
+ startLine: number;
43
+ endLine: number;
44
+ /**
45
+ * 用于替换这段行区间的新文本(可以是多行或空字符串)。
46
+ * 文本内部按 \n 分行。
47
+ */
48
+ replacement: string;
49
+ })
50
+ | (LineEditCommon & {
51
+ type: "insertBefore";
52
+ /**
53
+ * 在第 line 行“之前”插入 content。
54
+ * 1 <= line <= 文件总行数 + 1;
55
+ * 当 line === 文件总行数 + 1 时,相当于在文件尾部追加。
56
+ *
57
+ * 实际应用时,后端会把此行号作为插入点“粗定位”,
58
+ * 并优先尝试在附近用 originalSnippet 微调到更精确的位置。
59
+ */
60
+ line: number;
61
+ content: string;
62
+ })
63
+ | (LineEditCommon & {
64
+ type: "insertAfter";
65
+ /**
66
+ * 在第 line 行“之后”插入 content。
67
+ * 1 <= line <= 文件总行数;
68
+ * 若 line > 文件总行数,会退化为在文件尾部追加。
69
+ *
70
+ * 实际应用时,后端会把此行号作为插入点“粗定位”,
71
+ * 并优先尝试在附近用 originalSnippet 微调到更精确的位置。
72
+ */
73
+ line: number;
74
+ content: string;
75
+ })
76
+ | (LineEditCommon & {
77
+ type: "deleteRange";
78
+ /**
79
+ * 期望删除的起始行、结束行(包含),1-based。
80
+ * 实际应用时,后端会以此为“粗定位”,并结合 originalSnippet 做精确对齐。
81
+ */
82
+ startLine: number;
83
+ endLine: number;
84
+ });
85
+
86
+ export type ApplyLineEditsArgs = {
87
+ filePath: string;
88
+ edits: LineEdit[];
89
+ };
90
+
91
+ // ---- 工具 Schema,供 LLM 调用 ----
92
+
93
+ export const applyLineEditsFunctionSchema = {
94
+ name: "applyLineEdits",
95
+ description: [
96
+ "对指定文件按“行号”执行一组精确的文本编辑操作,而不是整文件重写。",
97
+ "",
98
+ "适用场景:",
99
+ "- 替换某个连续的行区间(一个函数体、一段 JSX 区块等)。",
100
+ "- 在某一行之前或之后插入若干行代码。",
101
+ "- 删除一段连续的行。",
102
+ "",
103
+ "约定:",
104
+ "- 所有行号均为 1-based(第一行为 1)。",
105
+ "- 建议一次调用只包含 1 个 edit;多个 edit 会顺序依次应用,后续 edit 的行号基于前一个 edit 应用后的结果。",
106
+ "",
107
+ "为了在行号有轻微偏差的情况下仍然成功修改,推荐同时提供 originalSnippet:",
108
+ "- 在规划编辑前,请先调用 readFile 工具获取最新的文件内容和行号信息。",
109
+ "- 若 readFile 使用了 startLine/endLine 只返回局部内容:",
110
+ " - lines[0] 对应的真实行号为 lineOffset;",
111
+ " - 计算 startLine/endLine 时需用 lineOffset + index。",
112
+ "- 对 replaceRange / deleteRange:",
113
+ " - 在 readFile 返回的 lines 中,取出 startLine~endLine 对应的原始行文本,用 \\n 拼接后放入 originalSnippet。",
114
+ "- 对 insertBefore / insertAfter:",
115
+ " - 取插入点附近的一行原始代码文本(通常是插入点当前这一行),放入 originalSnippet。",
116
+ "- 后端会:",
117
+ " - 将传入的行号视为“粗定位”;",
118
+ " - 然后在该行号附近优先搜索 originalSnippet;",
119
+ " - 若找到,就以搜索结果为准自动修正编辑位置,从而提高成功率;",
120
+ " - 若找不到,则退化为按行号硬改(与旧行为一致),不会因为 snippet 不匹配而额外报错。",
121
+ "",
122
+ "返回数据约定(rawData.response 中):",
123
+ "- newContent:应用行级编辑后的完整文件内容(统一使用 \\n 作为换行符),用于“最终文件预览”。",
124
+ "- diff:基于原始内容与 newContent 的行级 diff(diffLines 结果),用于 UI 中的 Diff 视图。",
125
+ ].join("\n"),
126
+ parameters: {
127
+ type: "object",
128
+ properties: {
129
+ filePath: {
130
+ type: "string",
131
+ description:
132
+ "要修改的文件相对项目根目录(bun-nolo)的路径,例如: packages/chat/src/messages/web/ToolMessageContent.tsx",
133
+ },
134
+ edits: {
135
+ type: "array",
136
+ description:
137
+ "要按顺序应用的一系列行级编辑操作。建议一次只传一个 edit,减少行号偏移带来的复杂性。",
138
+ items: {
139
+ type: "object",
140
+ oneOf: [
141
+ {
142
+ properties: {
143
+ type: { const: "replaceRange" },
144
+ startLine: {
145
+ type: "number",
146
+ description:
147
+ "期望被替换的起始行(包含),1-based。",
148
+ },
149
+ endLine: {
150
+ type: "number",
151
+ description:
152
+ "期望被替换的结束行(包含),1-based,必须 >= startLine。",
153
+ },
154
+ replacement: {
155
+ type: "string",
156
+ description:
157
+ "用于替换这段行区间的新文本(可以是多行或空字符串)。文本内部按 \\n 分行。",
158
+ },
159
+ originalSnippet: {
160
+ type: "string",
161
+ description:
162
+ "可选但强烈建议:startLine~endLine 对应的原始代码片段(用 \\n 拼接)。后端会优先在该行区附近搜索这段文本,并在行号略有偏差时自动对齐后再应用替换,从而大幅提高成功率。",
163
+ },
164
+ },
165
+ required: ["type", "startLine", "endLine", "replacement"],
166
+ },
167
+ {
168
+ properties: {
169
+ type: { const: "insertBefore" },
170
+ line: {
171
+ type: "number",
172
+ description:
173
+ "在第 line 行“之前”插入 content。1 <= line <= 文件总行数 + 1。",
174
+ },
175
+ content: {
176
+ type: "string",
177
+ description:
178
+ "要插入的文本(可以是多行),文本内部按 \\n 分行。",
179
+ },
180
+ originalSnippet: {
181
+ type: "string",
182
+ description:
183
+ "可选:插入点附近的一行原始代码文本(通常是当前第 line 行)。后端会在该行号附近优先搜索这行文本,在有轻微行号偏移时,仍然能找到合理的插入位置。",
184
+ },
185
+ },
186
+ required: ["type", "line", "content"],
187
+ },
188
+ {
189
+ properties: {
190
+ type: { const: "insertAfter" },
191
+ line: {
192
+ type: "number",
193
+ description:
194
+ "在第 line 行“之后”插入 content。1 <= line <= 文件总行数。",
195
+ },
196
+ content: {
197
+ type: "string",
198
+ description:
199
+ "要插入的文本(可以是多行),文本内部按 \\n 分行。",
200
+ },
201
+ originalSnippet: {
202
+ type: "string",
203
+ description:
204
+ "可选:插入点附近的一行原始代码文本(通常是当前第 line 行)。后端会在该行号附近优先搜索这行文本,在有轻微行号偏移时,仍然能找到合理的插入位置。",
205
+ },
206
+ },
207
+ required: ["type", "line", "content"],
208
+ },
209
+ {
210
+ properties: {
211
+ type: { const: "deleteRange" },
212
+ startLine: {
213
+ type: "number",
214
+ description:
215
+ "期望删除的起始行(包含),1-based。",
216
+ },
217
+ endLine: {
218
+ type: "number",
219
+ description:
220
+ "期望删除的结束行(包含),1-based,必须 >= startLine。",
221
+ },
222
+ originalSnippet: {
223
+ type: "string",
224
+ description:
225
+ "可选但建议:startLine~endLine 对应的原始代码片段(用 \\n 拼接)。后端会优先在该行区附近搜索这段文本,在行号略有偏差时自动对齐后再执行删除,减少误删风险且提高成功率。",
226
+ },
227
+ },
228
+ required: ["type", "startLine", "endLine"],
229
+ },
230
+ ],
231
+ },
232
+ },
233
+ },
234
+ required: ["filePath", "edits"],
235
+ },
236
+ };
237
+
238
+ // ---- 参数校验 ----
239
+
240
+ function assertValidArgs(args: ApplyLineEditsArgs): void {
241
+ const { filePath, edits } = args;
242
+
243
+ if (!filePath || typeof filePath !== "string") {
244
+ throw new Error("应用行级代码编辑失败:必须提供有效的 filePath 字符串。");
245
+ }
246
+ if (!Array.isArray(edits) || edits.length === 0) {
247
+ throw new Error("应用行级代码编辑失败:edits 必须是非空数组。");
248
+ }
249
+ }
250
+
251
+ // ---- 真正执行 ----
252
+
253
+ export async function applyLineEditsFunc(
254
+ args: ApplyLineEditsArgs,
255
+ thunkApi: any,
256
+ context?: { parentMessageId?: string; signal?: AbortSignal }
257
+ ): Promise<{ rawData: any; displayData?: string }> {
258
+ assertValidArgs(args);
259
+
260
+ const { filePath, edits } = args;
261
+
262
+ bumpDevReloadSuppressIfSelfEditing(filePath);
263
+
264
+ try {
265
+ const baseUrl = getToolBaseUrl(thunkApi);
266
+
267
+ if (!baseUrl) {
268
+ throw new Error("应用行级代码编辑失败:无法获取 applyLineEdits 服务器地址。");
269
+ }
270
+
271
+ const apiUrl = `${baseUrl.replace(/\/+$/, "")}/api/apply-line-edits`;
272
+
273
+ const response = await fetch(apiUrl, {
274
+ method: "POST",
275
+ headers: { "Content-Type": "application/json" },
276
+ signal: context?.signal,
277
+ body: JSON.stringify({ filePath, edits }),
278
+ });
279
+
280
+ const textBody = await response.text();
281
+ let data: any = {};
282
+ try {
283
+ data = textBody ? JSON.parse(textBody) : {};
284
+ } catch {
285
+ // ignore JSON parse error
286
+ }
287
+
288
+ if (!response.ok || data?.error) {
289
+ const errMsg =
290
+ data?.error ||
291
+ `applyLineEdits API 请求失败,状态码: ${response.status}. 响应: ${textBody}`;
292
+ console.error("applyLineEdits API Error:", errMsg);
293
+ throw new Error(errMsg);
294
+ }
295
+
296
+ return {
297
+ rawData: {
298
+ applied: true,
299
+ filePath,
300
+ editCount: edits.length,
301
+ response: data,
302
+ },
303
+ displayData: `✅ 已成功对文件应用 ${edits.length} 个行级编辑操作: ${filePath}`,
304
+ };
305
+ } catch (error: any) {
306
+ console.error("执行 applyLineEdits 时发生错误:", error);
307
+ throw new Error(
308
+ `应用行级代码编辑到文件 (${filePath}) 失败:${error?.message || String(error)
309
+ }`
310
+ );
311
+ }
312
+ }
@@ -0,0 +1,33 @@
1
+
2
+ import { executeBrowserTool } from "./common";
3
+
4
+ export const browser_click_Schema = {
5
+ name: "browser_click",
6
+ description:
7
+ "在当前浏览器会话中,点击匹配指定 CSS 选择器的元素。",
8
+ parameters: {
9
+ type: "object",
10
+ properties: {
11
+ sessionId: {
12
+ type: "string",
13
+ description: "由 browser_openSession 返回的活跃会话 ID。",
14
+ },
15
+ selector: {
16
+ type: "string",
17
+ description: "要点击元素的 CSS 选择器 (例如 '#submit-btn', '.nav-link').",
18
+ },
19
+ },
20
+ required: ["sessionId", "selector"],
21
+ },
22
+ };
23
+
24
+ export async function browser_click_Func(
25
+ args: { sessionId: string; selector: string },
26
+ thunkApi: any
27
+ ) {
28
+ const result = await executeBrowserTool("browser_click", args, thunkApi);
29
+ return {
30
+ rawData: result,
31
+ displayData: `✅ 已点击元素: \`${args.selector}\``,
32
+ };
33
+ }
@@ -0,0 +1,29 @@
1
+ import { executeBrowserTool } from "./common";
2
+
3
+ export const browser_closeSession_Schema = {
4
+ name: "browser_closeSession",
5
+ description:
6
+ "关闭一个已有的浏览器会话并释放服务器上的浏览器槽位。完成页面操作后应尽快调用。",
7
+ parameters: {
8
+ type: "object",
9
+ properties: {
10
+ sessionId: {
11
+ type: "string",
12
+ description: "由 browser_openSession 返回的活跃会话 ID。",
13
+ },
14
+ },
15
+ required: ["sessionId"],
16
+ },
17
+ };
18
+
19
+ export async function browser_closeSession_Func(
20
+ args: { sessionId: string },
21
+ thunkApi: any
22
+ ) {
23
+ const result = await executeBrowserTool("browser_closeSession", args, thunkApi);
24
+
25
+ return {
26
+ rawData: result.status,
27
+ displayData: `✅ 已关闭浏览器会话 ${args.sessionId}。`,
28
+ };
29
+ }
@@ -0,0 +1,27 @@
1
+ import { callToolApi } from "../toolApiClient";
2
+
3
+ /**
4
+ * 调用统一的后端浏览器工具API端点。
5
+ * @param toolName - 要调用的工具名称 (例如 'browser_openSession')。
6
+ * @param parameters - 传递给工具的参数。
7
+ * @param thunkApi - Redux Thunk API。
8
+ * @returns 后端返回的执行结果。
9
+ */
10
+ export async function executeBrowserTool(
11
+ toolName: string,
12
+ parameters: any,
13
+ thunkApi: any
14
+ ): Promise<any> {
15
+ try {
16
+ const result = await callToolApi<{ data: any }>(
17
+ thunkApi,
18
+ "/api/browser-tool",
19
+ { toolName, params: parameters },
20
+ { withAuth: true }
21
+ );
22
+ return result.data;
23
+ } catch (error: any) {
24
+ console.error(`执行浏览器工具 '${toolName}' 时发生网络或解析错误:`, error);
25
+ throw new Error(`浏览器工具 '${toolName}' 失败: ${error.message}`);
26
+ }
27
+ }