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,84 +0,0 @@
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;
@@ -1,211 +0,0 @@
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
- }
@@ -1,163 +0,0 @@
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
- }
@@ -1,233 +0,0 @@
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
- }