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,159 @@
1
+ import { normalizeSpaceId } from "create/space/spaceKeys";
2
+
3
+ export type DeleteSpacesMatchMode = "prefix" | "exact" | "contains" | "spaceId";
4
+
5
+ export interface DeleteSpacesQuery {
6
+ query: string;
7
+ matchMode?: DeleteSpacesMatchMode;
8
+ }
9
+
10
+ export interface SpaceMembershipLike {
11
+ spaceId?: string;
12
+ spaceName?: string;
13
+ name?: string;
14
+ title?: string;
15
+ role?: string;
16
+ ownerId?: string;
17
+ }
18
+
19
+ export interface SpaceRecordLike {
20
+ id?: string;
21
+ name?: string;
22
+ title?: string;
23
+ ownerId?: string;
24
+ members?: string[] | Record<string, unknown>;
25
+ contents?: Record<string, unknown>;
26
+ }
27
+
28
+ export interface SpaceDeletionPreviewItem {
29
+ spaceId: string;
30
+ name: string;
31
+ ownerId: string | null;
32
+ memberCount: number;
33
+ contentCount: number;
34
+ }
35
+
36
+ export interface SkippedSpaceDeletionItem {
37
+ spaceId: string;
38
+ name: string;
39
+ reason: "missing_space_id" | "missing_space_record" | "not_owner";
40
+ ownerId?: string | null;
41
+ }
42
+
43
+ export interface SpaceDeletionPreview {
44
+ deletable: SpaceDeletionPreviewItem[];
45
+ skipped: SkippedSpaceDeletionItem[];
46
+ }
47
+
48
+ const getSpaceName = (membership: SpaceMembershipLike, record?: SpaceRecordLike) =>
49
+ String(
50
+ membership.spaceName ??
51
+ membership.name ??
52
+ membership.title ??
53
+ record?.name ??
54
+ record?.title ??
55
+ membership.spaceId ??
56
+ ""
57
+ ).trim();
58
+
59
+ const countMembers = (members: SpaceRecordLike["members"]) => {
60
+ if (Array.isArray(members)) return members.length;
61
+ if (members && typeof members === "object") return Object.keys(members).length;
62
+ return 0;
63
+ };
64
+
65
+ export function filterSpaceDeletionCandidates(
66
+ memberships: SpaceMembershipLike[],
67
+ query: DeleteSpacesQuery
68
+ ) {
69
+ const rawQuery = String(query.query ?? "").trim();
70
+ if (!rawQuery) return [];
71
+ const matchMode = query.matchMode ?? "prefix";
72
+ const normalizedQuery = normalizeSpaceId(rawQuery).toLowerCase();
73
+ const textQuery = rawQuery.toLowerCase();
74
+
75
+ return memberships.filter((membership) => {
76
+ const spaceId = normalizeSpaceId(String(membership.spaceId ?? "")).toLowerCase();
77
+ const name = getSpaceName(membership).toLowerCase();
78
+
79
+ if (matchMode === "spaceId") return spaceId === normalizedQuery;
80
+ if (matchMode === "exact") return name === textQuery;
81
+ if (matchMode === "contains") return name.includes(textQuery);
82
+ return name.startsWith(textQuery);
83
+ });
84
+ }
85
+
86
+ export function buildDeleteSpacesPreview(args: {
87
+ currentUserId: string;
88
+ candidates: SpaceMembershipLike[];
89
+ spaceRecordsById: Record<string, SpaceRecordLike | null | undefined>;
90
+ }): SpaceDeletionPreview {
91
+ const deletable: SpaceDeletionPreviewItem[] = [];
92
+ const skipped: SkippedSpaceDeletionItem[] = [];
93
+
94
+ for (const membership of args.candidates) {
95
+ const spaceId = normalizeSpaceId(String(membership.spaceId ?? ""));
96
+ const fallbackName = getSpaceName(membership);
97
+ if (!spaceId) {
98
+ skipped.push({
99
+ spaceId: "",
100
+ name: fallbackName,
101
+ reason: "missing_space_id",
102
+ });
103
+ continue;
104
+ }
105
+
106
+ const record = args.spaceRecordsById[spaceId];
107
+ const name = getSpaceName(membership, record ?? undefined) || spaceId;
108
+ if (!record) {
109
+ skipped.push({
110
+ spaceId,
111
+ name,
112
+ reason: "missing_space_record",
113
+ });
114
+ continue;
115
+ }
116
+
117
+ const ownerId = typeof record.ownerId === "string"
118
+ ? record.ownerId
119
+ : typeof membership.ownerId === "string"
120
+ ? membership.ownerId
121
+ : null;
122
+ if (ownerId && ownerId !== args.currentUserId) {
123
+ skipped.push({
124
+ spaceId,
125
+ name,
126
+ reason: "not_owner",
127
+ ownerId,
128
+ });
129
+ continue;
130
+ }
131
+
132
+ deletable.push({
133
+ spaceId,
134
+ name,
135
+ ownerId,
136
+ memberCount: countMembers(record.members),
137
+ contentCount:
138
+ record.contents && typeof record.contents === "object"
139
+ ? Object.keys(record.contents).length
140
+ : 0,
141
+ });
142
+ }
143
+
144
+ return { deletable, skipped };
145
+ }
146
+
147
+ export function resolveConfirmedSpaceDeletionTargets(
148
+ preview: SpaceDeletionPreview,
149
+ confirmedSpaceIds: string[]
150
+ ) {
151
+ const wanted = new Set(
152
+ confirmedSpaceIds.map((spaceId) => normalizeSpaceId(spaceId)).filter(Boolean)
153
+ );
154
+ const targets = preview.deletable.filter((item) => wanted.has(item.spaceId));
155
+ const found = new Set(targets.map((item) => item.spaceId));
156
+ const missingConfirmedSpaceIds = Array.from(wanted).filter((spaceId) => !found.has(spaceId));
157
+
158
+ return { targets, missingConfirmedSpaceIds };
159
+ }
@@ -0,0 +1,29 @@
1
+ // 文件路径: packages/ai/tools/devReloadUtils.ts
2
+ // Dev reload 抑制工具函数 - 用于自举编辑时延迟热更新
3
+
4
+ /**
5
+ * 判断文件路径是否为项目源码(需要抑制 reload 的自举编辑)
6
+ * @param filePath 文件路径
7
+ * @returns 是否属于 packages/ 下的源码
8
+ */
9
+ export function isSelfEditingPath(filePath: string): boolean {
10
+ if (!filePath) return false;
11
+ return filePath.replace(/\\/g, "/").startsWith("packages/");
12
+ }
13
+
14
+ /**
15
+ * 如果是自举编辑(改项目源码),增加 reload 抑制计数
16
+ * 这样 dev server 的热更新会被延迟到回答结束后再执行
17
+ * @param filePath 正在编辑的文件路径
18
+ */
19
+ export function bumpDevReloadSuppressIfSelfEditing(filePath: string): void {
20
+ if (typeof window === "undefined") return;
21
+ if (!isSelfEditingPath(filePath)) return;
22
+
23
+ const w = window as any;
24
+ const cur = typeof w.__DEV_RELOAD_SUPPRESS_COUNT__ === "number" ? w.__DEV_RELOAD_SUPPRESS_COUNT__ : 0;
25
+ w.__DEV_RELOAD_SUPPRESS_COUNT__ = cur + 1;
26
+ w.__DEV_RELOAD_PENDING__ = false;
27
+
28
+ console.log("[devReloadSuppress] 自举编辑,增加 reload 抑制计数:", w.__DEV_RELOAD_SUPPRESS_COUNT__, "filePath:", filePath);
29
+ }
@@ -0,0 +1,137 @@
1
+ export type DialogMessageSearchRecord = {
2
+ id?: string;
3
+ dbKey?: string;
4
+ role?: string;
5
+ content?: unknown;
6
+ toolName?: string;
7
+ createdAt?: string | number;
8
+ };
9
+
10
+ export type DialogMessageSearchResult = {
11
+ messageId: string | null;
12
+ dbKey: string | null;
13
+ role: string | null;
14
+ toolName: string | null;
15
+ createdAt: string | number | null;
16
+ content: string;
17
+ context: Array<{
18
+ messageId: string | null;
19
+ role: string | null;
20
+ toolName: string | null;
21
+ createdAt: string | number | null;
22
+ content: string;
23
+ }>;
24
+ };
25
+
26
+ export const normalizeDialogSearchText = (value: unknown): string =>
27
+ String(value ?? "").trim();
28
+
29
+ export const clampDialogSearchNumber = (
30
+ value: unknown,
31
+ fallback: number,
32
+ min: number,
33
+ max: number,
34
+ ): number => {
35
+ const parsed = Number(value ?? fallback);
36
+ const safe = Number.isFinite(parsed) ? Math.floor(parsed) : fallback;
37
+ return Math.max(min, Math.min(max, safe));
38
+ };
39
+
40
+ export const buildDialogSearchMatcher = (query: string) => {
41
+ const normalizedQuery = query.toLowerCase();
42
+ const terms = normalizedQuery.split(/\s+/).filter(Boolean);
43
+ return (text: string) => {
44
+ const normalizedText = text.toLowerCase();
45
+ return normalizedText.includes(normalizedQuery) ||
46
+ (terms.length > 1 && terms.every((term) => normalizedText.includes(term)));
47
+ };
48
+ };
49
+
50
+ export const dialogMessageContentToText = (content: unknown): string => {
51
+ if (typeof content === "string") return content;
52
+ if (Array.isArray(content)) {
53
+ return content.map((part) => {
54
+ if (part && typeof part === "object" && "text" in part) {
55
+ return String((part as { text?: unknown }).text ?? "");
56
+ }
57
+ return JSON.stringify(part);
58
+ }).join("\n");
59
+ }
60
+ if (content == null) return "";
61
+ return JSON.stringify(content);
62
+ };
63
+
64
+ export const clipDialogSearchText = (value: string, max: number): string => {
65
+ const trimmed = value.trim();
66
+ if (trimmed.length <= max) return trimmed;
67
+ return `${trimmed.slice(0, max)}\n...[truncated ${trimmed.length - max} chars]`;
68
+ };
69
+
70
+ export const buildDialogMessageSearchResults = (args: {
71
+ messages: DialogMessageSearchRecord[];
72
+ query: string;
73
+ limit: number;
74
+ contextMessages: number;
75
+ role?: string;
76
+ includeTools?: boolean;
77
+ contentClipChars: number;
78
+ contextClipChars: number;
79
+ }): DialogMessageSearchResult[] => {
80
+ const matchesQuery = buildDialogSearchMatcher(args.query);
81
+ const includeTools = args.includeTools !== false;
82
+
83
+ return args.messages
84
+ .map((message, index) => ({
85
+ message,
86
+ index,
87
+ text: dialogMessageContentToText(message.content),
88
+ }))
89
+ .filter(({ message, text }) => {
90
+ if (!includeTools && message.role === "tool") return false;
91
+ if (args.role && message.role !== args.role) return false;
92
+ return matchesQuery(text);
93
+ })
94
+ .slice(0, args.limit)
95
+ .map(({ message, index, text }) => {
96
+ const start = Math.max(0, index - args.contextMessages);
97
+ const end = Math.min(args.messages.length, index + args.contextMessages + 1);
98
+ return {
99
+ messageId: message.id ?? null,
100
+ dbKey: message.dbKey ?? null,
101
+ role: message.role ?? null,
102
+ toolName: message.toolName ?? null,
103
+ createdAt: message.createdAt ?? null,
104
+ content: clipDialogSearchText(text, args.contentClipChars),
105
+ context: args.messages.slice(start, end).map((item) => ({
106
+ messageId: item.id ?? null,
107
+ role: item.role ?? null,
108
+ toolName: item.toolName ?? null,
109
+ createdAt: item.createdAt ?? null,
110
+ content: clipDialogSearchText(
111
+ dialogMessageContentToText(item.content),
112
+ args.contextClipChars,
113
+ ),
114
+ })),
115
+ };
116
+ });
117
+ };
118
+
119
+ export const formatDialogMessageSearchDisplay = (args: {
120
+ dialogKey: string;
121
+ query: string;
122
+ results: DialogMessageSearchResult[];
123
+ }): string => {
124
+ if (!args.results.length) {
125
+ return `No original dialog messages matched "${args.query}" in ${args.dialogKey}.`;
126
+ }
127
+
128
+ return [
129
+ `Found ${args.results.length} message match(es) in ${args.dialogKey} for "${args.query}".`,
130
+ ...args.results.map((item, index) =>
131
+ [
132
+ `Match ${index + 1}: ${item.role ?? "unknown"} id=${item.messageId ?? "unknown"}${item.toolName ? ` tool=${item.toolName}` : ""}`,
133
+ item.content,
134
+ ].join("\n")
135
+ ),
136
+ ].join("\n\n");
137
+ };
@@ -0,0 +1,72 @@
1
+ import { readAction } from "database/actions/read";
2
+ import type { PageData } from "render/page/types";
3
+ import { diagnoseSkillDocument } from "ai/skills/skillDiagnostics";
4
+
5
+ export interface DoctorSkillToolArgs {
6
+ id?: string;
7
+ content?: string;
8
+ }
9
+
10
+ export const doctorSkillFunctionSchema = {
11
+ name: "doctorSkill",
12
+ description:
13
+ "检查一个 skill 文档的协议、工具绑定和常见问题,返回错误、警告和改进建议。",
14
+ parameters: {
15
+ type: "object",
16
+ properties: {
17
+ id: {
18
+ type: "string",
19
+ description: "可选:本地 skill 文档的 dbKey(如 page-xxx)。",
20
+ },
21
+ content: {
22
+ type: "string",
23
+ description: "可选:直接传入 skill markdown 内容进行诊断。",
24
+ },
25
+ },
26
+ } as const,
27
+ };
28
+
29
+ export async function doctorSkillFunc(
30
+ args: DoctorSkillToolArgs,
31
+ thunkApi: any
32
+ ): Promise<{ rawData: unknown; displayData: string }> {
33
+ const { id, content } = args;
34
+ if (!id && !content?.trim()) {
35
+ throw new Error("doctorSkill 需要提供 id 或 content。");
36
+ }
37
+
38
+ let page: PageData | undefined;
39
+ if (id) {
40
+ page = (await readAction({ dbKey: id }, thunkApi)) as PageData;
41
+ if (!page) {
42
+ throw new Error(`未找到 skill 文档:${id}`);
43
+ }
44
+ }
45
+
46
+ const result = await diagnoseSkillDocument(
47
+ {
48
+ id: page?.dbKey ?? id,
49
+ title: page?.title,
50
+ content: content ?? page?.content ?? "",
51
+ meta: page?.meta,
52
+ tools: page?.tools,
53
+ }
54
+ );
55
+
56
+ const lines = [
57
+ `Skill 检查:${result.name ?? result.skillId ?? id ?? "inline-skill"}`,
58
+ `- 状态: ${result.ok ? "通过" : "存在错误"}`,
59
+ `- 归一化工具: ${result.canonicalToolNames.join(", ") || "(无)"}`,
60
+ `- Eval 用例数: ${result.evalCaseCount}`,
61
+ ...(result.errors.length ? ["- 错误:", ...result.errors.map((item) => ` - ${item}`)] : []),
62
+ ...(result.warnings.length
63
+ ? ["- 警告:", ...result.warnings.map((item) => ` - ${item}`)]
64
+ : []),
65
+ ...(result.notes.length ? ["- 建议:", ...result.notes.map((item) => ` - ${item}`)] : []),
66
+ ];
67
+
68
+ return {
69
+ rawData: result,
70
+ displayData: lines.join("\n"),
71
+ };
72
+ }
@@ -0,0 +1,86 @@
1
+ // /ai/tools/ecommerceScraperTool.ts
2
+
3
+ import { callApifyActor } from "./apifyActorClient";
4
+
5
+ export const ecommerceScraperFunctionSchema = {
6
+ name: "ecommerceScraper",
7
+ description:
8
+ "使用 Apify E-commerce Scraping Tool 抓取电商网站的产品、评论、卖家等信息。",
9
+ parameters: {
10
+ type: "object",
11
+ properties: {
12
+ detailsUrls: {
13
+ type: "array",
14
+ description: "产品详情页 URL 列表,对应 detailsUrls。",
15
+ items: { type: "string" },
16
+ },
17
+ listingUrls: {
18
+ type: "array",
19
+ description: "分类/列表页 URL 列表,对应 listingUrls。",
20
+ items: { type: "string" },
21
+ },
22
+ keyword: {
23
+ type: "string",
24
+ description: "用于 marketplace 搜索的关键词。",
25
+ },
26
+ marketplaces: {
27
+ type: "array",
28
+ description: "指定在哪些电商站点搜索(见 Actor 文档枚举)。",
29
+ items: { type: "string" },
30
+ },
31
+ maxProductResults: {
32
+ type: "integer",
33
+ minimum: 1,
34
+ description: "总产品数量上限。",
35
+ },
36
+ // 你可以按需继续把 review / seller 的字段加进来,
37
+ extraInput: {
38
+ type: "object",
39
+ description: "可选。直接透传给 Apify 的其他 input 字段。",
40
+ },
41
+ },
42
+ required: [],
43
+ },
44
+ };
45
+
46
+ export async function ecommerceScraperFunc(
47
+ args: {
48
+ detailsUrls?: string[];
49
+ listingUrls?: string[];
50
+ keyword?: string;
51
+ marketplaces?: string[];
52
+ maxProductResults?: number;
53
+ extraInput?: Record<string, any>;
54
+ },
55
+ thunkApi: any
56
+ ) {
57
+ const {
58
+ detailsUrls,
59
+ listingUrls,
60
+ keyword,
61
+ marketplaces,
62
+ maxProductResults,
63
+ extraInput = {},
64
+ } = args;
65
+
66
+ const input: any = {
67
+ scrapeMode: "AUTO",
68
+ ...extraInput,
69
+ };
70
+
71
+ if (detailsUrls?.length)
72
+ input.detailsUrls = detailsUrls.map((url) => ({ url }));
73
+ if (listingUrls?.length)
74
+ input.listingUrls = listingUrls.map((url) => ({ url }));
75
+ if (typeof keyword === "string" && keyword.trim()) input.keyword = keyword;
76
+ if (marketplaces?.length) input.marketplaces = marketplaces;
77
+ if (typeof maxProductResults === "number")
78
+ input.maxProductResults = maxProductResults;
79
+
80
+ return callApifyActor(thunkApi, {
81
+ actorId: "apify~e-commerce-scraping-tool",
82
+ input,
83
+ resultType: "datasetItems",
84
+ displayName: "E-commerce Scraping Tool",
85
+ });
86
+ }