nolo-cli 0.1.8 → 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 (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 +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/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 +8 -6
@@ -0,0 +1,133 @@
1
+ // /ai/tools/importDataTool.ts
2
+
3
+ import type { RootState } from "app/store";
4
+ import { executeSqlFunc } from "./executeSqlTool";
5
+ import { selectPendingRawDataByPageKey } from "chat/dialog/dialogSlice";
6
+
7
+ // importDataFunctionSchema 保持不变
8
+ export const importDataFunctionSchema = {
9
+ name: "importData",
10
+ description:
11
+ "将用户在当前消息中附带的文件内容,导入到指定的数据库表中。必须使用 `User's Current References` 上下文中的 `DB Key` 作为 `fileReferenceId`。",
12
+ parameters: {
13
+ type: "object",
14
+ properties: {
15
+ tableName: {
16
+ type: "string",
17
+ description:
18
+ "数据要导入的目标数据库表的名称。例如:'budgets', 'sales_records'。",
19
+ },
20
+ fileReferenceId: {
21
+ type: "string",
22
+ description:
23
+ "必须是从 `User's Current References` 上下文中获取到的、文件的 DB Key (例如 'page:xyz123')。",
24
+ },
25
+ },
26
+ required: ["tableName", "fileReferenceId"],
27
+ },
28
+ };
29
+
30
+ /**
31
+ * [Executor] 'importData' 工具的执行函数。
32
+ * @param args - LLM 提供的参数: { tableName: string, fileReferenceId: string }
33
+ * @param thunkApi - Redux Thunk API
34
+ * @returns {Promise<{rawData: any, displayData: string}>} - 返回标准化的结果对象
35
+ */
36
+ export async function importDataFunc(
37
+ args: { tableName: string; fileReferenceId: string },
38
+ thunkApi: any
39
+ ): Promise<{ rawData: any; displayData: string }> {
40
+ const { tableName, fileReferenceId: pageKey } = args;
41
+
42
+ if (!tableName || !pageKey) {
43
+ throw new Error("`tableName` 和 `fileReferenceId` 都是必需的参数。");
44
+ }
45
+
46
+ try {
47
+ const state = thunkApi.getState() as RootState;
48
+
49
+ const pendingData = selectPendingRawDataByPageKey(state, pageKey);
50
+
51
+ if (!pendingData || !pendingData.jsonData) {
52
+ throw new Error(`在内存中找不到文件 (ID: ${pageKey}) 的数据。`);
53
+ }
54
+
55
+ const jsonData = pendingData.jsonData;
56
+ if (!Array.isArray(jsonData) || jsonData.length === 0) {
57
+ return { rawData: {}, displayData: "文件数据为空,未执行导入操作。" };
58
+ }
59
+
60
+ // [关键修复] 步骤 1: 查询数据库以获取真实的列名和大小写
61
+ const schemaInfoResult = await executeSqlFunc(
62
+ { sqlQuery: `PRAGMA table_info("${tableName}");` },
63
+ thunkApi
64
+ );
65
+
66
+ if (
67
+ !schemaInfoResult.rawData?.result ||
68
+ schemaInfoResult.rawData.result.length === 0
69
+ ) {
70
+ throw new Error(
71
+ `无法获取表 "${tableName}" 的结构信息,请检查表是否存在。`
72
+ );
73
+ }
74
+
75
+ // 从 PRAGMA 结果中提取数据库中的真实列名
76
+ const dbColumns = schemaInfoResult.rawData.result.map(
77
+ (col: any) => col.name
78
+ );
79
+
80
+ // [关键修复] 步骤 2: 创建从“小写键名”到“原始数据键名”的映射
81
+ // 这使得我们可以忽略大小写来匹配列
82
+ const inputKeys = Object.keys(jsonData[0]);
83
+ const inputKeyMap = new Map(
84
+ inputKeys.map((key) => [key.toLowerCase(), key])
85
+ );
86
+
87
+ // [关键修复] 步骤 3: 准备基于真实数据库列名的 INSERT 语句
88
+ const columnNamesForSql = dbColumns.map((col) => `"${col}"`).join(", ");
89
+ const valuePlaceholders = `(${dbColumns.map(() => "?").join(", ")})`;
90
+ const insertQueryTemplate = `INSERT INTO "${tableName}" (${columnNamesForSql}) VALUES ${valuePlaceholders};`;
91
+
92
+ let totalAffectedRows = 0;
93
+
94
+ await executeSqlFunc({ sqlQuery: "BEGIN TRANSACTION;" }, thunkApi);
95
+ try {
96
+ for (const row of jsonData) {
97
+ // [关键修复] 步骤 4: 按数据库列的顺序,不区分大小写地从数据行中提取值
98
+ const params = dbColumns.map((dbCol) => {
99
+ const originalInputKey = inputKeyMap.get(dbCol.toLowerCase());
100
+ // 如果在输入数据中找到了匹配的列(忽略大小写),则使用其值
101
+ if (originalInputKey) {
102
+ return row[originalInputKey] ?? null;
103
+ }
104
+ // 如果数据库有这个列,但输入数据没有,则插入 NULL
105
+ return null;
106
+ });
107
+
108
+ const result = await executeSqlFunc(
109
+ { sqlQuery: insertQueryTemplate, params },
110
+ thunkApi
111
+ );
112
+ totalAffectedRows += result.rawData?.result?.changes ?? 0;
113
+ }
114
+ await executeSqlFunc({ sqlQuery: "COMMIT;" }, thunkApi);
115
+ } catch (innerError) {
116
+ await executeSqlFunc({ sqlQuery: "ROLLBACK;" }, thunkApi);
117
+ throw innerError;
118
+ }
119
+
120
+ const displayData = `成功将文件中的 ${jsonData.length} 条记录导入到表 "${tableName}"。总共影响行数: ${totalAffectedRows}。`;
121
+
122
+ return {
123
+ rawData: {
124
+ importedCount: jsonData.length,
125
+ affectedRows: totalAffectedRows,
126
+ },
127
+ displayData,
128
+ };
129
+ } catch (error: any) {
130
+ console.error("Import Data tool error:", error);
131
+ throw new Error(`数据导入失败: ${error.message}`);
132
+ }
133
+ }
@@ -0,0 +1,162 @@
1
+ import { createDoc } from "render/page/docSlice";
2
+ import { selectCurrentSpaceId } from "create/space/spaceSlice";
3
+ import type { RootState } from "app/store";
4
+ import { canonicalizeToolNames } from "./toolNameAliases";
5
+ import {
6
+ buildSkillDocMarkdown,
7
+ parseExternalSkillMarkdown,
8
+ type SkillDocConfig,
9
+ } from "ai/skills/skillDocProtocol";
10
+ import { buildSkillFollowupResult } from "./skillFollowup";
11
+
12
+ export interface ImportSkillToolArgs {
13
+ url?: string;
14
+ content?: string;
15
+ title?: string;
16
+ spaceId?: string;
17
+ categoryId?: string;
18
+ }
19
+
20
+ const normalizeUrl = (rawUrl: string): string => {
21
+ const trimmed = rawUrl.trim();
22
+ const githubBlobMatch = trimmed.match(
23
+ /^https:\/\/github\.com\/([^/]+)\/([^/]+)\/blob\/([^/]+)\/(.+)$/
24
+ );
25
+ if (githubBlobMatch) {
26
+ const [, owner, repo, ref, path] = githubBlobMatch;
27
+ return `https://raw.githubusercontent.com/${owner}/${repo}/${ref}/${path}`;
28
+ }
29
+ return trimmed;
30
+ };
31
+
32
+ const slugify = (value: string): string =>
33
+ value
34
+ .trim()
35
+ .toLowerCase()
36
+ .replace(/[^a-z0-9]+/g, "-")
37
+ .replace(/^-+|-+$/g, "") || "imported-skill";
38
+
39
+ const mapImportedTools = (toolNames: string[]): string[] =>
40
+ canonicalizeToolNames(
41
+ toolNames.filter((toolName) => /^[a-zA-Z0-9_-]+$/.test(toolName))
42
+ );
43
+
44
+ export const importSkillFunctionSchema = {
45
+ name: "importSkill",
46
+ description: [
47
+ "从外部 SKILL.md、GitHub 地址或原始 Markdown 文本导入一个 skill,并保存为当前空间中的普通文档。",
48
+ "导入后会自动附加隐藏的 skill-config 协议块;无法映射的脚本环境会降级为说明型 skill。",
49
+ "导入成功后,如果用户还没说明下一步,优先调用 ui_ask_choice 继续询问:仅保存、挂到现有 Agent,还是新建一个 Agent 来使用它。",
50
+ ].join("\n"),
51
+ parameters: {
52
+ type: "object",
53
+ properties: {
54
+ url: {
55
+ type: "string",
56
+ description:
57
+ "可选:远程 SKILL.md 地址。支持 raw GitHub URL,也支持 github.com/.../blob/... 链接。",
58
+ },
59
+ content: {
60
+ type: "string",
61
+ description:
62
+ "可选:原始 SKILL.md 或普通 Markdown 内容。如果提供,则优先使用它而不是 url。",
63
+ },
64
+ title: {
65
+ type: "string",
66
+ description: "可选:导入后文档标题。未提供时优先使用外部 skill name。",
67
+ },
68
+ spaceId: {
69
+ type: "string",
70
+ description: '可选:目标 spaceId;不传则优先使用当前 space。可传空字符串 ""。',
71
+ },
72
+ categoryId: {
73
+ type: "string",
74
+ description: '可选:目标分类 categoryId;不传时传空字符串 ""。',
75
+ },
76
+ },
77
+ } as const,
78
+ };
79
+
80
+ export async function importSkillFunc(
81
+ args: ImportSkillToolArgs,
82
+ thunkApi: any
83
+ ): Promise<{ rawData: unknown; displayData: string }> {
84
+ const { dispatch, getState } = thunkApi;
85
+ const state = getState() as RootState;
86
+ const explicitContent = typeof args.content === "string" ? args.content.trim() : "";
87
+ const url = typeof args.url === "string" ? normalizeUrl(args.url) : "";
88
+
89
+ if (!explicitContent && !url) {
90
+ throw new Error("importSkill 需要提供 content 或 url。");
91
+ }
92
+
93
+ let rawMarkdown = explicitContent;
94
+ if (!rawMarkdown) {
95
+ const response = await fetch(url);
96
+ if (!response.ok) {
97
+ throw new Error(`导入 skill 失败:HTTP ${response.status}`);
98
+ }
99
+ rawMarkdown = (await response.text()).trim();
100
+ }
101
+
102
+ const parsed = parseExternalSkillMarkdown(rawMarkdown);
103
+ const docTitle =
104
+ (args.title ?? "").trim() ||
105
+ parsed.name ||
106
+ "Imported Skill";
107
+ const normalizedTools = mapImportedTools(parsed.allowedTools);
108
+ const description =
109
+ parsed.description ||
110
+ "Imported external skill. Review instructions and mapped tools before using it in agents.";
111
+ const skillConfig: SkillDocConfig = {
112
+ version: "0.1",
113
+ kind: "skill",
114
+ id: slugify(parsed.name || docTitle),
115
+ name: parsed.name || docTitle,
116
+ description,
117
+ triggerMode: "explicit",
118
+ toolNames: normalizedTools,
119
+ budgetTier: "medium",
120
+ dispatchPreferred: false,
121
+ modalities: ["text"],
122
+ };
123
+
124
+ const importedBody = [
125
+ parsed.body,
126
+ parsed.compatibility ? `\n## Compatibility\n${parsed.compatibility}` : "",
127
+ url ? `\nImported from: ${url}` : "",
128
+ ]
129
+ .filter(Boolean)
130
+ .join("\n\n")
131
+ .trim();
132
+
133
+ const content = buildSkillDocMarkdown({
134
+ body: importedBody,
135
+ skillConfig,
136
+ });
137
+
138
+ const explicitSpaceId = (args.spaceId ?? "").trim() || undefined;
139
+ const currentSpaceId = selectCurrentSpaceId(state) || undefined;
140
+ const spaceId = explicitSpaceId ?? currentSpaceId;
141
+ const categoryId = (args.categoryId ?? "").trim() || undefined;
142
+
143
+ const createDocResult = await (dispatch as any)(
144
+ (createDoc as any)({
145
+ title: docTitle,
146
+ spaceId,
147
+ categoryId,
148
+ content,
149
+ })
150
+ );
151
+ const id = await createDocResult.unwrap();
152
+
153
+ return buildSkillFollowupResult({
154
+ dbKey: id,
155
+ title: docTitle,
156
+ skillId: skillConfig.id,
157
+ spaceId: spaceId ?? null,
158
+ toolNames: normalizedTools,
159
+ importedFrom: url || null,
160
+ hasEvalConfig: false,
161
+ });
162
+ }