nolo-cli 0.1.7 → 0.1.9

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 (247) hide show
  1. package/README.md +107 -5
  2. package/agentRuntimeCommands.ts +464 -0
  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 +1075 -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/agentRun.ts +198 -167
  232. package/client/compactDialog.ts +222 -0
  233. package/commandRegistry.ts +14 -0
  234. package/connector-experimental/capabilities.ts +73 -0
  235. package/connector-experimental/codexBinary.ts +41 -0
  236. package/connector-experimental/heartbeatLoop.ts +22 -0
  237. package/connector-experimental/index.ts +5 -0
  238. package/connector-experimental/machineInfo.ts +46 -0
  239. package/connector-experimental/protocol.ts +54 -0
  240. package/connectorWebSocketTarget.ts +29 -0
  241. package/defaultServer.ts +1 -0
  242. package/index.ts +158 -104
  243. package/machineCommands.ts +382 -0
  244. package/package.json +12 -2
  245. package/tui/readlineWorkspace.ts +50 -0
  246. package/tui/session.ts +40 -2
  247. package/updateCommands.ts +70 -5
@@ -0,0 +1,84 @@
1
+ // 文件路径: ai/tools/readDocTool.ts
2
+
3
+ import { slateToSimplifiedMarkdown } from "create/editor/transforms/slateToSimplifiedMarkdown";
4
+ import type { PageData } from "render/page/types";
5
+
6
+ export interface ReadPageToolArgs {
7
+ id: string; // 对应 dbKey,例如 "PAGE-xxx"
8
+ }
9
+
10
+ /**
11
+ * [Schema] 定义了 'readDoc' 工具的结构。
12
+ */
13
+ export const readDocFunctionSchema = {
14
+ name: "readDoc",
15
+ description: [
16
+ "读取指定页面的内容,并将结构化的数据转换为 Markdown 格式返回。",
17
+ "如果你拿到了页面的 dbKey(如 page-xxx),请使用此工具查看页面内容。",
18
+ ].join("\n"),
19
+ parameters: {
20
+ type: "object",
21
+ properties: {
22
+ id: {
23
+ type: "string",
24
+ description: "页面/文档的数据库键(dbKey),例如 page-xxx。",
25
+ },
26
+ },
27
+ required: ["id"],
28
+ },
29
+ } as const;
30
+
31
+ export const readPageFunctionSchema = {
32
+ ...readDocFunctionSchema,
33
+ name: "readPage",
34
+ } as const;
35
+
36
+ export const buildReadDocResult = (
37
+ pageData: PageData
38
+ ): { rawData: unknown; displayData: string } => {
39
+ const markdownContent = slateToSimplifiedMarkdown(pageData.slateData || []);
40
+
41
+ const rawData = {
42
+ success: true,
43
+ id: pageData.dbKey,
44
+ title: pageData.title,
45
+ content: markdownContent,
46
+ metadata: {
47
+ spaceId: pageData.spaceId,
48
+ created: pageData.created,
49
+ },
50
+ };
51
+
52
+ const displayData = `已成功读取页面《${pageData.title}》。\n\n内容如下:\n\n${markdownContent}`;
53
+ return { rawData, displayData };
54
+ };
55
+
56
+ /**
57
+ * [Executor] 'readDoc' 工具的执行函数。
58
+ */
59
+ export async function readDocFunc(
60
+ args: ReadPageToolArgs,
61
+ thunkApi: any
62
+ ): Promise<{ rawData: unknown; displayData: string }> {
63
+ const { id } = args;
64
+
65
+ if (!id || !id.toLowerCase().startsWith("page-")) {
66
+ throw new Error(`无效的页面 ID: ${id}。页面 ID 通常以 "page-" 开头。`);
67
+ }
68
+
69
+ try {
70
+ // 1. 从数据库读取原始数据
71
+ const { readAction } = await import("database/actions/read");
72
+ const pageData = (await readAction({ dbKey: id }, thunkApi)) as PageData;
73
+
74
+ if (!pageData) {
75
+ throw new Error(`未找到 ID 为 ${id} 的页面。`);
76
+ }
77
+
78
+ return buildReadDocResult(pageData);
79
+ } catch (error: any) {
80
+ throw new Error(`读取页面时出错: ${error.message || String(error)}`);
81
+ }
82
+ }
83
+
84
+ export const readPageFunc = readDocFunc;
@@ -0,0 +1,211 @@
1
+ // 文件路径: packages/ai/tools/readFileTool.ts
2
+
3
+ import { getToolBaseUrl } from "./toolApiClient";
4
+
5
+ // ---- Types ----
6
+
7
+ export type ReadFileArgs = {
8
+ /**
9
+ * 目标文件相对项目根目录(bun-nolo)的路径,例如:
10
+ * - packages/chat/src/messages/web/ToolMessageContent.tsx
11
+ * - packages/server/entry.ts
12
+ */
13
+ filePath: string;
14
+ /**
15
+ * 可选:从该行开始读取(1-based,包含)。
16
+ */
17
+ startLine?: number;
18
+ /**
19
+ * 可选:读取到该行结束(1-based,包含)。
20
+ */
21
+ endLine?: number;
22
+ };
23
+
24
+ // ---- 工具 Schema,供 LLM 调用 ----
25
+
26
+ export const readFileFunctionSchema = {
27
+ name: "readFile",
28
+ description: [
29
+ "从后端服务器读取一个文本文件的完整内容,并辅助行级编辑工具获取精确的行号信息。",
30
+ "",
31
+ "适用场景:",
32
+ "- 在修改文件之前,先获取当前文件的完整内容用于分析。",
33
+ "- 在需要理解某个模块实现时,请求查看对应源文件。",
34
+ "- 在使用 applyLineEdits 按行号修改文件前,先读取文件并基于返回的 lines / lineCount 计算行号。",
35
+ "- 如果使用 startLine/endLine 只读取部分内容:",
36
+ " - lines[0] 对应的真实行号为 lineOffset;",
37
+ " - 全文件总行数为 totalLineCount;",
38
+ " - 计算绝对行号时,请用 lineOffset + index。",
39
+ "- 为 applyLineEdits 提供 originalSnippet:",
40
+ " - 对 replaceRange/deleteRange:从 lines 中取出将要修改的那几行,用 \\n 拼接,作为 originalSnippet 传给 applyLineEdits;",
41
+ " - 对 insertBefore/insertAfter:从 lines 中取出插入点附近的一行原始代码,作为 originalSnippet 传给 applyLineEdits;",
42
+ "",
43
+ "行为约定:",
44
+ "- 当目标文件存在:返回文件的完整文本内容。",
45
+ "- 当目标文件不存在:返回错误信息。",
46
+ "",
47
+ "返回数据约定(rawData 中):",
48
+ "- content:文件完整内容,统一使用 \\n 作为换行符。",
49
+ "- lines:按行拆分后的字符串数组;若使用 startLine/endLine,则 lines[0] 对应 lineOffset。",
50
+ "- lineCount:全文件总行数(即最大可用行号)。",
51
+ "- lineOffset:当使用 startLine/endLine 时,lines[0] 对应的真实行号。",
52
+ "- totalLineCount:全文件总行数(与 lineCount 相同,便于兼容)。",
53
+ "",
54
+ "注意:",
55
+ "- filePath 必须是相对项目根目录(bun-nolo)的相对路径,且不能越出项目根目录。",
56
+ "- 仅适用于文本文件,二进制文件的内容会被当作 UTF-8 文本读取,可能出现乱码。",
57
+ ].join("\n"),
58
+ parameters: {
59
+ type: "object",
60
+ properties: {
61
+ filePath: {
62
+ type: "string",
63
+ description:
64
+ "目标文件相对项目根目录(bun-nolo)的路径,例如: packages/chat/src/messages/web/ToolMessageContent.tsx",
65
+ },
66
+ startLine: {
67
+ type: "number",
68
+ description:
69
+ "可选:从该行开始读取(1-based,包含)。与 endLine 搭配可只读局部内容。",
70
+ },
71
+ endLine: {
72
+ type: "number",
73
+ description:
74
+ "可选:读取到该行结束(1-based,包含)。与 startLine 搭配可只读局部内容。",
75
+ },
76
+ },
77
+ required: ["filePath"],
78
+ },
79
+ };
80
+
81
+ // ---- 预览执行:不读文件,只展示信息 ----
82
+
83
+ export async function readFilePreviewFunc(
84
+ args: ReadFileArgs,
85
+ _thunkApi: any
86
+ ): Promise<{ rawData: any; displayData?: string }> {
87
+ const { filePath, startLine, endLine } = args;
88
+
89
+ if (!filePath || typeof filePath !== "string") {
90
+ throw new Error("readFile 预览失败:必须提供有效的 filePath 字符串。");
91
+ }
92
+
93
+ return {
94
+ rawData: {
95
+ previewOnly: true,
96
+ filePath,
97
+ startLine,
98
+ endLine,
99
+ },
100
+ displayData: `⏸️ 文件读取预览: ${filePath}`,
101
+ };
102
+ }
103
+
104
+ // ---- 真正执行:POST 到后端 /api/read-file ----
105
+
106
+ export async function readFileFunc(
107
+ args: ReadFileArgs,
108
+ thunkApi: any,
109
+ context?: { parentMessageId?: string; signal?: AbortSignal }
110
+ ): Promise<{ rawData: any; displayData?: string }> {
111
+ const { filePath, startLine, endLine } = args;
112
+
113
+ if (!filePath || typeof filePath !== "string") {
114
+ throw new Error("读取文件失败:必须提供有效的 filePath 字符串。");
115
+ }
116
+
117
+ try {
118
+ const baseUrl = getToolBaseUrl(thunkApi);
119
+
120
+ if (!baseUrl) {
121
+ throw new Error("读取文件失败:无法获取 readFile 服务器地址。");
122
+ }
123
+
124
+ const apiUrl = `${baseUrl.replace(/\/+$/, "")}/api/read-file`;
125
+
126
+ const response = await fetch(apiUrl, {
127
+ method: "POST",
128
+ headers: { "Content-Type": "application/json" },
129
+ signal: context?.signal,
130
+ body: JSON.stringify({ filePath, startLine, endLine }),
131
+ });
132
+
133
+ const textBody = await response.text();
134
+ let data: any = {};
135
+ try {
136
+ data = textBody ? JSON.parse(textBody) : {};
137
+ } catch {
138
+ // 非 JSON 的情况忽略解析错误
139
+ }
140
+
141
+ if (!response.ok || data?.error) {
142
+ const errMsg =
143
+ data?.error ||
144
+ `readFile API 请求失败,状态码: ${response.status}. 响应: ${textBody}`;
145
+ console.error("readFile API Error:", errMsg);
146
+ throw new Error(errMsg);
147
+ }
148
+
149
+ // 统一换行符为 \n,保证与行级编辑逻辑一致
150
+ const serverContent: string =
151
+ typeof data?.content === "string" ? data.content : "";
152
+ const normalizedContent =
153
+ serverContent.indexOf("\r\n") >= 0
154
+ ? serverContent.replace(/\r\n/g, "\n")
155
+ : serverContent;
156
+
157
+ const lines =
158
+ normalizedContent === "" ? [] : normalizedContent.split("\n");
159
+
160
+ const lineOffset: number =
161
+ typeof data?.lineOffset === "number" ? data.lineOffset : 1;
162
+
163
+ const totalLineCount: number =
164
+ typeof data?.totalLineCount === "number"
165
+ ? data.totalLineCount
166
+ : typeof data?.lineCount === "number"
167
+ ? data.lineCount
168
+ : lines.length;
169
+
170
+ const rangeStart: number =
171
+ typeof data?.rangeStart === "number"
172
+ ? data.rangeStart
173
+ : lines.length
174
+ ? lineOffset
175
+ : 1;
176
+ const rangeEnd: number =
177
+ typeof data?.rangeEnd === "number"
178
+ ? data.rangeEnd
179
+ : lines.length
180
+ ? lineOffset + lines.length - 1
181
+ : 0;
182
+
183
+ const returnedLineCount = lines.length;
184
+
185
+ const rangeInfo =
186
+ totalLineCount === 0
187
+ ? "0 行"
188
+ : rangeStart === 1 && rangeEnd === totalLineCount
189
+ ? `${totalLineCount} 行`
190
+ : `${rangeStart}-${rangeEnd} 行 / 共 ${totalLineCount} 行`;
191
+
192
+ return {
193
+ rawData: {
194
+ applied: true,
195
+ filePath,
196
+ lines,
197
+ totalLineCount,
198
+ lineOffset,
199
+ rangeStart,
200
+ rangeEnd,
201
+ returnedLineCount,
202
+ },
203
+ displayData: `📖 已读取文件: ${filePath}(${rangeInfo})`,
204
+ };
205
+ } catch (error: any) {
206
+ console.error("执行 readFile 时发生错误:", error);
207
+ throw new Error(
208
+ `读取文件 (${filePath}) 失败:${error?.message || String(error)}`
209
+ );
210
+ }
211
+ }
@@ -0,0 +1,163 @@
1
+ // 文件路径: ai/tools/readTool.ts
2
+
3
+ import { readAction } from "database/actions/read";
4
+ import { readAndWaitAction } from "database/actions/readAndWait";
5
+ import { slateToSimplifiedMarkdown } from "create/editor/transforms/slateToSimplifiedMarkdown";
6
+ import type { PageData } from "render/page/types";
7
+ import { getRuntimeServerContext } from "database/runtimeServerContext";
8
+ import { isTableMetaKey, rowKey, isAgentKey } from "database/keys";
9
+ import { DataType } from "create/types";
10
+ import { TableMeta } from "render/table/types";
11
+ import { fetchAndSerializeTable } from "render/table/utils/tableSerialization";
12
+
13
+ // ---- Types ----
14
+
15
+ export type ReadArgs = {
16
+ /**
17
+ * 对应 database 的 dbKey
18
+ */
19
+ dbKey: string;
20
+
21
+ /**
22
+ * 是否等待远程结果:
23
+ * - false(默认):使用 readAction,本地优先,远程在后台同步
24
+ * - true:使用 readAndWaitAction,等待远程与本地决策后返回“权威”结果
25
+ */
26
+ waitRemote?: boolean;
27
+ };
28
+
29
+ // ---- 工具 Schema,供 LLM 调用 ----
30
+
31
+ export const readFunctionSchema = {
32
+ name: "read",
33
+ description: [
34
+ "根据指定的 dbKey 从本地/远程数据库读取一条记录。",
35
+ "支持所有内容类型:",
36
+ "- page-xxx:页面内容(返回 Markdown)",
37
+ "- dialog-xxx:对话历史(返回消息列表)",
38
+ "- agent-xxx:Agent 配置",
39
+ "- table-xxx:表格数据",
40
+ "- space-xxx:Space 完整数据(含分类和内容目录)",
41
+ "行为说明:",
42
+ "- 默认本地优先:若本地存在则立即返回,后台同步远程;",
43
+ "- waitRemote=true:等待远程权威结果后返回(适合需要最新数据的场景)。",
44
+ ].join("\n"),
45
+ parameters: {
46
+ type: "object",
47
+ properties: {
48
+ dbKey: {
49
+ type: "string",
50
+ description: "要读取的数据键,例如 PAGE-xxx、dialog-xxx、agent-xxx、space-xxx。",
51
+ },
52
+ waitRemote: {
53
+ type: "boolean",
54
+ description:
55
+ "是否等待远程结果。false=本地优先并在后台同步远程;true=等待远程与本地决策后返回。",
56
+ default: false,
57
+ },
58
+ },
59
+ required: ["dbKey"],
60
+ },
61
+ };
62
+
63
+ // ---- 执行函数 ----
64
+
65
+ export async function readFunc(
66
+ args: ReadArgs,
67
+ thunkApi: any,
68
+ context?: { parentMessageId?: string; signal?: AbortSignal }
69
+ ): Promise<{ rawData: any; displayData?: string }> {
70
+ const { dbKey, waitRemote = false } = args || {};
71
+
72
+ if (!dbKey || typeof dbKey !== "string") {
73
+ throw new Error("read 工具需要提供字符串类型的 id(dbKey)。");
74
+ }
75
+
76
+ try {
77
+ const signal = context?.signal;
78
+
79
+ const result = waitRemote
80
+ ? await readAndWaitAction(dbKey, thunkApi)
81
+ : await readAction({ dbKey, signal }, thunkApi);
82
+
83
+ if (!result) {
84
+ throw new Error(`未找到 dbKey 为 ${dbKey} 的数据。`);
85
+ }
86
+
87
+ // 智能化处理:如果是页面,自动转 Markdown
88
+ if (dbKey.startsWith("PAGE-") || (result as any).type === DataType.DOC) {
89
+ const pageData = result as PageData;
90
+ const markdownContent = slateToSimplifiedMarkdown(pageData.slateData || []);
91
+
92
+ return {
93
+ rawData: {
94
+ ...pageData,
95
+ content: markdownContent, // 附加转换后的内容
96
+ },
97
+ displayData: `已成功读取页面《${pageData.title}》。\n\n内容预览:\n\n${markdownContent}`,
98
+ };
99
+ }
100
+
101
+ // 智能化处理:如果是表格,自动转 Markdown
102
+ if (isTableMetaKey(dbKey) || (result as any).type === DataType.TABLE) {
103
+ const tableMeta = result as TableMeta;
104
+ const title = tableMeta.displayName || `Table (${tableMeta.tableId})`;
105
+
106
+ // Use shared utility
107
+ const { rows, markdown: tableMd } = await thunkApi.dispatch(
108
+ async (_dispatch: any, getState: any, { db }: any) => {
109
+ const state = getState();
110
+ const { currentToken: token, remoteServers } =
111
+ getRuntimeServerContext(state);
112
+
113
+ return await fetchAndSerializeTable(tableMeta, db, {
114
+ token,
115
+ remoteServers,
116
+ });
117
+ }
118
+ );
119
+
120
+ return {
121
+ rawData: {
122
+ ...tableMeta,
123
+ rows,
124
+ markdown: tableMd,
125
+ },
126
+ displayData: `已成功读取表格《${title}》。共 ${rows.length} 行数据。\n\n内容预览:\n\n${tableMd}`,
127
+ };
128
+ }
129
+
130
+ // 智能化处理:如果是 Agent/Cybot,显示基本信息
131
+ if (
132
+ isAgentKey(dbKey) ||
133
+ (result as any).type === DataType.AGENT ||
134
+ (result as any).type === DataType.CYBOT
135
+ ) {
136
+ const agent = result as any;
137
+ const name = agent.name || "未命名 Agent";
138
+ const desc = agent.introduction || agent.description || "无描述";
139
+ const modelInfo = agent.model ? ` (模型: ${agent.model})` : "";
140
+
141
+ return {
142
+ rawData: agent,
143
+ displayData: `已成功读取 Agent《${name}》${modelInfo}。\n\n描述:${desc}\n提示词预览:\n${agent.prompt?.slice(0, 200) || "无"
144
+ }...`,
145
+ };
146
+ }
147
+
148
+ const sourceLabel = waitRemote
149
+ ? "已等待远程与本地完成后返回权威结果"
150
+ : "本地优先,已触发后台与远程同步";
151
+
152
+
153
+ return {
154
+ rawData: result,
155
+ displayData: `已读取数据: "${result.dbKey}"(${sourceLabel})`,
156
+ };
157
+ } catch (error: any) {
158
+ console.error("执行 read 工具时发生错误:", error);
159
+ throw new Error(
160
+ `读取数据 (${dbKey}) 失败:${error?.message || String(error)}`
161
+ );
162
+ }
163
+ }
@@ -0,0 +1,233 @@
1
+ import { createXReadFailure } from "../../integrations/x-reader/types";
2
+ import type { XPost, XReadResult } from "../../integrations/x-reader/types";
3
+ import { getRequestConfig, ToolApiError } from "./toolApiClient";
4
+
5
+ export const readXPostFunctionSchema = {
6
+ name: "read_x_post",
7
+ description:
8
+ "读取 X/Twitter status 链接的可见帖子正文、作者和结构化数据。适合用户给出 x.com/twitter.com 帖子链接并要求查看、总结、解释或抽取信息的场景。默认通过桌面本地 Chrome/CDP bridge 读取,不要求用户粘贴 cookie 或 token。",
9
+ parameters: {
10
+ type: "object",
11
+ properties: {
12
+ url: {
13
+ type: "string",
14
+ description:
15
+ "要读取的 X/Twitter status URL,例如 https://x.com/user/status/123。",
16
+ },
17
+ keepOpen: {
18
+ type: "boolean",
19
+ description: "调试时是否保留临时 Chrome bridge。默认 false。",
20
+ default: false,
21
+ },
22
+ profileDir: {
23
+ type: "string",
24
+ description:
25
+ "桌面端本地 Chrome 专用 profile 目录。用于让用户在本机登录一次 X 后复用本地账号状态;不要传入用户日常 Chrome profile。",
26
+ },
27
+ headless: {
28
+ type: "boolean",
29
+ description:
30
+ "是否用 headless 模式启动本地 Chrome。需要用户首次登录本地 X profile 时设为 false 并配合 keepOpen。",
31
+ default: true,
32
+ },
33
+ },
34
+ required: ["url"],
35
+ },
36
+ };
37
+
38
+ type ReadXPostToolContext = {
39
+ reader?: (
40
+ url: string,
41
+ args: { keepOpen?: boolean; profileDir?: string; headless?: boolean },
42
+ ) => Promise<XReadResult<XPost>>;
43
+ };
44
+
45
+ async function callLocalReadXPostApi(
46
+ thunkApi: any,
47
+ body: object,
48
+ ): Promise<XReadResult<XPost>> {
49
+ const { currentServer, token } = getRequestConfig(thunkApi);
50
+ const browserOrigin = (globalThis as any).window?.location?.origin;
51
+ const baseUrl = typeof browserOrigin === "string" && browserOrigin
52
+ ? browserOrigin
53
+ : currentServer.replace(/\/+$/, "");
54
+ const headers: Record<string, string> = { "Content-Type": "application/json" };
55
+ if (token) headers.Authorization = `Bearer ${token}`;
56
+
57
+ const response = await fetch(`${baseUrl}/api/read-x-post`, {
58
+ method: "POST",
59
+ headers,
60
+ body: JSON.stringify(body),
61
+ });
62
+ const text = await response.text();
63
+ let data: any = null;
64
+ try {
65
+ data = text ? JSON.parse(text) : null;
66
+ } catch {
67
+ data = null;
68
+ }
69
+ if (!response.ok) {
70
+ throw new ToolApiError(
71
+ data?.error?.message ?? `read_x_post API 请求失败,状态码: ${response.status}`,
72
+ {
73
+ status: response.status,
74
+ code: data?.error?.code,
75
+ details: data,
76
+ },
77
+ );
78
+ }
79
+ return data as XReadResult<XPost>;
80
+ }
81
+
82
+ async function callDesktopReadXPostApi(
83
+ thunkApi: any,
84
+ body: object,
85
+ ): Promise<XReadResult<XPost>> {
86
+ const { token } = getRequestConfig(thunkApi);
87
+ const port = Number(process.env.NOLO_DESKTOP_SERVER_PORT ?? 3233);
88
+ const headers: Record<string, string> = { "Content-Type": "application/json" };
89
+ if (token) headers.Authorization = `Bearer ${token}`;
90
+
91
+ const response = await fetch(`http://127.0.0.1:${port}/api/read-x-post`, {
92
+ method: "POST",
93
+ headers,
94
+ body: JSON.stringify(body),
95
+ });
96
+ const text = await response.text();
97
+ let data: any = null;
98
+ try {
99
+ data = text ? JSON.parse(text) : null;
100
+ } catch {
101
+ data = null;
102
+ }
103
+ if (!response.ok) {
104
+ throw new ToolApiError(
105
+ data?.error?.message ?? `desktop read_x_post API 请求失败,状态码: ${response.status}`,
106
+ {
107
+ status: response.status,
108
+ code: data?.error?.code,
109
+ details: data,
110
+ },
111
+ );
112
+ }
113
+ return data as XReadResult<XPost>;
114
+ }
115
+
116
+ async function readWithDefaultBridge(
117
+ url: string,
118
+ args: { keepOpen?: boolean; profileDir?: string; headless?: boolean },
119
+ thunkApi?: any,
120
+ ): Promise<XReadResult<XPost>> {
121
+ if (process.env.PLATFORM === "web") {
122
+ if (thunkApi?.getState) {
123
+ return callLocalReadXPostApi(thunkApi, { url, ...args });
124
+ }
125
+
126
+ if (typeof window !== "undefined" && (window as any).__NOLO_DESKTOP__) {
127
+ try {
128
+ const res = await fetch("/api/read-x-post", {
129
+ method: "POST",
130
+ headers: { "Content-Type": "application/json" },
131
+ body: JSON.stringify({ url, ...args }),
132
+ });
133
+ const data = await res.json().catch(() => null);
134
+ if (res.ok && data) {
135
+ return data as XReadResult<XPost>;
136
+ }
137
+ return createXReadFailure({
138
+ code: "network_error",
139
+ message: `desktop read_x_post endpoint failed: HTTP ${res.status}`,
140
+ nextStep: "请确认桌面端本地服务仍在运行,然后重试。",
141
+ backend: "desktop_local_browser",
142
+ });
143
+ } catch (error) {
144
+ return createXReadFailure({
145
+ code: "network_error",
146
+ message:
147
+ error instanceof Error
148
+ ? error.message
149
+ : "desktop read_x_post endpoint request failed",
150
+ nextStep: "请确认桌面端本地服务仍在运行,然后重试。",
151
+ backend: "desktop_local_browser",
152
+ });
153
+ }
154
+ }
155
+
156
+ return createXReadFailure({
157
+ code: "not_connected",
158
+ message: "read_x_post 需要通过服务器或桌面本地 bridge 执行,不能在普通浏览器 bundle 内直接启动 Chrome/CDP。",
159
+ nextStep: "请使用服务器 agent run 或桌面端本地 bridge 路径执行该工具。",
160
+ backend: "desktop_local_browser",
161
+ });
162
+ }
163
+
164
+ if (process.env.NOLO_DESKTOP === "1" && thunkApi?.getState) {
165
+ return callDesktopReadXPostApi(thunkApi, { url, ...args });
166
+ }
167
+
168
+ const importBridge = new Function("specifier", "return import(specifier)") as <
169
+ T = any,
170
+ >(
171
+ specifier: string,
172
+ ) => Promise<T>;
173
+ const bridgeModuleUrl = new URL(
174
+ "../../integrations/x-reader/bridge/readXPostWithBridge.ts",
175
+ import.meta.url,
176
+ ).href;
177
+ const { readXPostWithBridge } = await importBridge<{
178
+ readXPostWithBridge: (
179
+ url: string,
180
+ args: { keepOpen?: boolean; profileDir?: string; headless?: boolean },
181
+ ) => Promise<XReadResult<XPost>>;
182
+ }>(bridgeModuleUrl);
183
+ return readXPostWithBridge(url, args);
184
+ }
185
+
186
+ function assertXStatusUrl(url: string) {
187
+ if (!/^https?:\/\/(x|twitter)\.com\/[^/]+\/status\/\d+/i.test(url)) {
188
+ throw new Error("read_x_post 需要一个有效的 X/Twitter status URL。");
189
+ }
190
+ }
191
+
192
+ function formatDisplay(result: XReadResult<XPost>) {
193
+ if (!result.ok) {
194
+ return [
195
+ `读取 X 帖子失败:${result.message}`,
196
+ `失败代码:${result.code}`,
197
+ result.nextStep ? `下一步:${result.nextStep}` : "",
198
+ `后端:${result.backend}`,
199
+ ]
200
+ .filter(Boolean)
201
+ .join("\n");
202
+ }
203
+
204
+ const post = result.data;
205
+ return [
206
+ `已读取 X 帖子:@${post.author.handle}${post.author.displayName ? `(${post.author.displayName})` : ""}`,
207
+ `URL: ${post.url}`,
208
+ `后端:${result.backend}`,
209
+ "",
210
+ post.text,
211
+ ].join("\n");
212
+ }
213
+
214
+ export async function readXPostFunc(
215
+ args: { url: string; keepOpen?: boolean; profileDir?: string; headless?: boolean },
216
+ thunkApi: any,
217
+ context: ReadXPostToolContext = {},
218
+ ): Promise<{ rawData: XReadResult<XPost>; displayData: string }> {
219
+ const url = String(args?.url ?? "").trim();
220
+ assertXStatusUrl(url);
221
+
222
+ const keepOpen = Boolean(args?.keepOpen);
223
+ const profileDir = String(args?.profileDir ?? "").trim() || undefined;
224
+ const headless = args?.headless;
225
+ const rawData =
226
+ (await context.reader?.(url, { keepOpen, profileDir, headless })) ??
227
+ (await readWithDefaultBridge(url, { keepOpen, profileDir, headless }, thunkApi));
228
+
229
+ return {
230
+ rawData,
231
+ displayData: formatDisplay(rawData),
232
+ };
233
+ }