nolo-cli 0.1.13 → 0.1.15
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.
- package/README.md +9 -2
- package/agent-runtime/hostAdapter.ts +53 -0
- package/agent-runtime/index.ts +28 -0
- package/agent-runtime/localLoop.ts +62 -0
- package/agent-runtime/runtimeDecision.ts +70 -0
- package/agent-runtime/types.ts +87 -0
- package/agentRunCommand.ts +104 -0
- package/agentRuntimeCommands.ts +139 -22
- package/agentRuntimeLocal.ts +7 -0
- package/ai/agent/_executeModel.ts +118 -0
- package/ai/agent/agentSlice.ts +544 -1
- package/ai/agent/appWorkingMemory.ts +126 -0
- package/ai/agent/avatarUtils.ts +24 -0
- package/ai/agent/buildEditingContext.ts +373 -0
- package/ai/agent/buildSystemPrompt.ts +532 -0
- package/ai/agent/cleanAgentMessages.ts +140 -0
- package/ai/agent/cliChatClient.ts +119 -0
- package/ai/agent/contextCompiler.ts +107 -0
- package/ai/agent/contextLayerContract.ts +44 -0
- package/ai/agent/createAgentSchema.ts +234 -0
- package/ai/agent/executeToolCall.ts +58 -0
- package/ai/agent/fetchAgentContexts.ts +42 -0
- package/ai/agent/generatePrompt.ts +3 -0
- package/ai/agent/getFullChatContextKeys.ts +168 -0
- package/ai/agent/hooks/fetchPublicAgents.ts +133 -0
- package/ai/agent/hooks/useAgentConfig.ts +61 -0
- package/ai/agent/hooks/useAgentDialog.ts +35 -0
- package/ai/agent/hooks/useAgentFormValidation.ts +202 -0
- package/ai/agent/hooks/usePublicAgents.ts +473 -0
- package/ai/agent/persistMessageWithFixedId.ts +37 -0
- package/ai/agent/planSlice.ts +259 -0
- package/ai/agent/referenceUtils.ts +229 -0
- package/ai/agent/runAgentBackground.ts +238 -0
- package/ai/agent/runAgentClientLoop.ts +138 -0
- package/ai/agent/runtimeGuidance.ts +97 -0
- package/ai/agent/runtimeServerBase.ts +37 -0
- package/ai/agent/server/fetchPublicAgents.ts +128 -0
- package/ai/agent/startParallelAgentStreams.ts +424 -0
- package/ai/agent/startupProtocol.ts +53 -0
- package/ai/agent/streamAgentChatTurn.ts +1299 -0
- package/ai/agent/streamAgentChatTurnUtils.ts +738 -0
- package/ai/agent/types.ts +71 -0
- package/ai/agent/utils/imageOutput.ts +39 -0
- package/ai/agent/utils/publicImageAgentMode.ts +26 -0
- package/ai/agent/utils/sortUtils.ts +250 -0
- package/ai/agent/web/referencePickerUtils.ts +146 -0
- package/ai/ai.locale.ts +1083 -0
- package/ai/chat/accumulateToolCallChunks.ts +95 -0
- package/ai/chat/fetchUtils.native.ts +276 -0
- package/ai/chat/fetchUtils.ts +153 -0
- package/ai/chat/inlineImageUrlsForCustomProvider.ts +117 -0
- package/ai/chat/parseApiError.ts +64 -0
- package/ai/chat/parseMultilineSSE.ts +95 -0
- package/ai/chat/sendOpenAICompletionsRequest.native.ts +682 -0
- package/ai/chat/sendOpenAICompletionsRequest.ts +712 -0
- package/ai/chat/sendOpenAIResponseRequest.ts +512 -0
- package/ai/chat/shouldUseServerProxy.ts +18 -0
- package/ai/chat/sseClient.native.ts +91 -0
- package/ai/chat/sseClient.ts +67 -0
- package/ai/chat/streamReader.native.ts +31 -0
- package/ai/chat/streamReader.ts +62 -0
- package/ai/chat/updateTotalUsage.ts +72 -0
- package/ai/context/buildReferenceContext.ts +437 -0
- package/ai/context/calculateContextUsage.ts +133 -0
- package/ai/context/retention.ts +165 -0
- package/ai/context/tokenUtils.ts +78 -0
- package/ai/index.ts +1 -1
- package/ai/llm/agentCapabilities.ts +74 -0
- package/ai/llm/calculateGeminiImageTokens.ts +57 -0
- package/ai/llm/deepinfra.ts +28 -0
- package/ai/llm/fireworks.ts +68 -0
- package/ai/llm/generateRequestBody.ts +165 -0
- package/ai/llm/getModelContextWindow.ts +84 -0
- package/ai/llm/getNoloKey.ts +37 -0
- package/ai/llm/getPricing.ts +232 -0
- package/ai/llm/hooks/useModelPricing.ts +75 -0
- package/ai/llm/imagePricing.ts +66 -0
- package/ai/llm/isResponseAPIModel.ts +13 -0
- package/ai/llm/kimi.ts +18 -0
- package/ai/llm/mimo.ts +71 -0
- package/ai/llm/mistral.ts +22 -0
- package/ai/llm/modelAvatar.ts +427 -0
- package/ai/llm/models.ts +45 -0
- package/ai/llm/openrouterModels.ts +141 -0
- package/ai/llm/providers.ts +307 -0
- package/ai/llm/reasoningModels.ts +28 -0
- package/ai/llm/types.ts +59 -0
- package/ai/llm/usageRequestOptions.ts +59 -0
- package/ai/memory/capture.ts +148 -0
- package/ai/memory/consolidate.ts +104 -0
- package/ai/memory/delete.ts +147 -0
- package/ai/memory/overlay.ts +84 -0
- package/ai/memory/query.ts +38 -0
- package/ai/memory/queryShared.ts +160 -0
- package/ai/memory/rank.ts +105 -0
- package/ai/memory/recentRelationshipRecap.ts +247 -0
- package/ai/memory/remember.ts +167 -0
- package/ai/memory/runtime.ts +76 -0
- package/ai/memory/store.ts +20 -0
- package/ai/memory/storeShared.ts +76 -0
- package/ai/memory/types.ts +46 -0
- package/ai/memory/understanding.ts +349 -0
- package/ai/memory/understandingGreeting.ts +264 -0
- package/ai/messages/type.ts +20 -0
- package/ai/policy/personalizationDialog.ts +333 -0
- package/ai/policy/runtimePolicy.ts +440 -0
- package/ai/policy/selfUpdateFields.ts +48 -0
- package/ai/policy/types.ts +64 -0
- package/ai/skills/referenceRuntime.ts +274 -0
- package/ai/skills/skillDiagnostics.ts +251 -0
- package/ai/skills/skillDocBuilder.ts +139 -0
- package/ai/skills/skillDocProtocol.ts +434 -0
- package/ai/skills/skillReferenceSummary.ts +63 -0
- package/ai/skills/skillSummaryMarker.ts +26 -0
- package/ai/token/calculatePrice.ts +546 -0
- package/ai/token/db.ts +98 -0
- package/ai/token/externalToolCost.ts +321 -0
- package/ai/token/hooks/useRecords.ts +65 -0
- package/ai/token/missingUsageEstimate.ts +42 -0
- package/ai/token/modelUsageQuery.ts +252 -0
- package/ai/token/normalizeUsage.ts +84 -0
- package/ai/token/openaiImageGenerationUsage.ts +56 -0
- package/ai/token/prepareTokenUsageData.ts +88 -0
- package/ai/token/query.ts +88 -0
- package/ai/token/queryUserTokens.ts +59 -0
- package/ai/token/resolveBillingTarget.ts +52 -0
- package/ai/token/saveTokenRecord.ts +53 -0
- package/ai/token/serverDialogProjection.ts +78 -0
- package/ai/token/serverTokenWriter.ts +143 -0
- package/ai/token/stats.ts +21 -0
- package/ai/token/tokenThunks.ts +24 -0
- package/ai/token/types.ts +93 -0
- package/ai/tools/agent/agentTools.ts +176 -0
- package/ai/tools/agent/agentUpdateShared.ts +311 -0
- package/ai/tools/agent/callAgentTool.ts +139 -0
- package/ai/tools/agent/createAgentTool.ts +512 -0
- package/ai/tools/agent/createDialogTool.ts +69 -0
- package/ai/tools/agent/createSkillAgentTool.ts +62 -0
- package/ai/tools/agent/parallelBudget.ts +221 -0
- package/ai/tools/agent/presets/appBuilderPreset.ts +147 -0
- package/ai/tools/agent/runLlmTool.ts +96 -0
- package/ai/tools/agent/runStreamingAgentTool.ts +73 -0
- package/ai/tools/agent/skillAgentArgs.ts +106 -0
- package/ai/tools/agent/skillAgentPreset.ts +89 -0
- package/ai/tools/agent/streamParallelAgentsTool.ts +122 -0
- package/ai/tools/agent/updateAgentTool.ts +96 -0
- package/ai/tools/agent/updateSelfTool.ts +113 -0
- package/ai/tools/amazonProductScraperTool.ts +86 -0
- package/ai/tools/apifyActorClient.ts +45 -0
- package/ai/tools/appEditGuard.ts +372 -0
- package/ai/tools/appReadSnapshot.ts +153 -0
- package/ai/tools/appTools.ts +1549 -0
- package/ai/tools/applyEditTool.ts +256 -0
- package/ai/tools/applyLineEditsTool.ts +312 -0
- package/ai/tools/browserTools/click.ts +33 -0
- package/ai/tools/browserTools/closeSession.ts +29 -0
- package/ai/tools/browserTools/common.ts +27 -0
- package/ai/tools/browserTools/openSession.ts +48 -0
- package/ai/tools/browserTools/readContent.ts +38 -0
- package/ai/tools/browserTools/selectOption.ts +46 -0
- package/ai/tools/browserTools/typeText.ts +42 -0
- package/ai/tools/category/createCategoryTool.ts +66 -0
- package/ai/tools/category/queryContentsByCategoryTool.ts +69 -0
- package/ai/tools/category/updateContentCategoryTool.ts +75 -0
- package/ai/tools/cfBrowserTools.ts +319 -0
- package/ai/tools/cfSpeechToTextTool.ts +49 -0
- package/ai/tools/checkEnvTool.ts +65 -0
- package/ai/tools/cloudflareCrawlTool.ts +289 -0
- package/ai/tools/codeSearchTool.ts +111 -0
- package/ai/tools/codeTools.ts +101 -0
- package/ai/tools/createDocTool.ts +132 -0
- package/ai/tools/createPlanTool.ts +999 -0
- package/ai/tools/createSkillDocTool.ts +155 -0
- package/ai/tools/createWorkflowTool.ts +154 -0
- package/ai/tools/deepseekOcrTool.ts +34 -0
- package/ai/tools/delayTool.ts +31 -0
- package/ai/tools/deleteSpacesTool.ts +325 -0
- package/ai/tools/deleteSpacesToolModel.ts +159 -0
- package/ai/tools/devReloadUtils.ts +29 -0
- package/ai/tools/dialogMessageSearch.ts +137 -0
- package/ai/tools/doctorSkillTool.ts +72 -0
- package/ai/tools/ecommerceScraperTool.ts +86 -0
- package/ai/tools/emailTools.ts +549 -0
- package/ai/tools/evalSkillTool.ts +92 -0
- package/ai/tools/exaSearchTool.ts +64 -0
- package/ai/tools/execBashTool.ts +379 -0
- package/ai/tools/executeSqlTool.ts +192 -0
- package/ai/tools/fetchWebpageSupport.ts +309 -0
- package/ai/tools/fetchWebpageTool.ts +84 -0
- package/ai/tools/geminiImagePreviewTool.ts +361 -0
- package/ai/tools/generateDocxTool.ts +215 -0
- package/ai/tools/googleSearchScraperTool.ts +106 -0
- package/ai/tools/importDataTool.ts +133 -0
- package/ai/tools/importSkillTool.ts +162 -0
- package/ai/tools/index.ts +1927 -0
- package/ai/tools/listFilesTool.ts +82 -0
- package/ai/tools/listUserSpacesTool.ts +113 -0
- package/ai/tools/modelUsageTools.ts +199 -0
- package/ai/tools/olmOcrTool.ts +34 -0
- package/ai/tools/openaiImageTool.ts +267 -0
- package/ai/tools/prepareTools.ts +23 -0
- package/ai/tools/readDocTool.ts +84 -0
- package/ai/tools/readFileTool.ts +211 -0
- package/ai/tools/readTool.ts +163 -0
- package/ai/tools/readXPostTool.ts +233 -0
- package/ai/tools/rememberMemoryTool.ts +84 -0
- package/ai/tools/remotionVideoTool.ts +151 -0
- package/ai/tools/searchDialogMessagesTool.ts +222 -0
- package/ai/tools/searchRepoTool.ts +115 -0
- package/ai/tools/searchWorkspaceTool.ts +259 -0
- package/ai/tools/skillFollowup.ts +86 -0
- package/ai/tools/surfWeatherTool.ts +169 -0
- package/ai/tools/table/addTableRowTool.ts +217 -0
- package/ai/tools/table/createTableTool.ts +315 -0
- package/ai/tools/table/rowTools.ts +366 -0
- package/ai/tools/table/schemaTools.ts +244 -0
- package/ai/tools/table/shareTableTool.ts +148 -0
- package/ai/tools/table/toolShared.ts +129 -0
- package/ai/tools/toolApiClient.ts +198 -0
- package/ai/tools/toolNameAliases.ts +57 -0
- package/ai/tools/toolResultError.ts +42 -0
- package/ai/tools/toolRunSlice.ts +303 -0
- package/ai/tools/toolSchemaCompatibility.ts +53 -0
- package/ai/tools/toolVisibility.ts +4 -0
- package/ai/tools/types.ts +20 -0
- package/ai/tools/uiAskChoiceTool.ts +104 -0
- package/ai/tools/updateContentTitleTool.ts +84 -0
- package/ai/tools/updateDocTool.ts +105 -0
- package/ai/tools/updateUserPreferenceProfileTool.ts +145 -0
- package/ai/tools/whisperTool.ts +77 -0
- package/ai/tools/writeFileTool.ts +210 -0
- package/ai/tools/youtubeScraperTool.ts +116 -0
- package/ai/tools/ziweiChartTool.ts +678 -0
- package/ai/types.ts +55 -0
- package/ai/workflow/workflowExecutor.ts +323 -0
- package/ai/workflow/workflowSlice.ts +73 -0
- package/ai/workflow/workflowTypes.ts +106 -0
- package/client/agentRun.test.ts +240 -0
- package/client/agentRun.ts +182 -19
- package/client/compactDialog.test.ts +238 -0
- package/client/localRuntimeAdapter.test.ts +135 -0
- package/client/localRuntimeAdapter.ts +244 -0
- package/client/profileConfig.test.ts +40 -0
- package/client/streamingOutput.test.ts +22 -0
- package/client/streamingOutput.ts +38 -0
- package/commandRegistry.ts +11 -2
- package/connector-experimental/index.ts +5 -0
- package/database/actions/cacheMergedUserData.ts +64 -0
- package/database/actions/common.ts +242 -0
- package/database/actions/deleteFile.ts +40 -0
- package/database/actions/fetchUserData.ts +16 -0
- package/database/actions/fileContent.ts +125 -0
- package/database/actions/patch.ts +155 -0
- package/database/actions/read.ts +337 -0
- package/database/actions/readAndWait.ts +224 -0
- package/database/actions/readRequestManager.ts +120 -0
- package/database/actions/remove.ts +94 -0
- package/database/actions/replication.ts +366 -0
- package/database/actions/upload.ts +174 -0
- package/database/actions/upsert.ts +56 -0
- package/database/actions/write.ts +126 -0
- package/database/client/db.native.ts +73 -0
- package/database/client/db.ts +51 -0
- package/database/client/fetchUserData.ts +61 -0
- package/database/client/handleError.ts +19 -0
- package/database/client/queryRequest.ts +21 -0
- package/database/config.ts +21 -0
- package/database/dbActionThunks.ts +1 -0
- package/database/dbSlice.ts +149 -0
- package/database/email.ts +42 -0
- package/database/fileRing.ts +51 -0
- package/database/fileSharding.ts +70 -0
- package/database/fileStorage.native.ts +92 -0
- package/database/fileStorage.ts +232 -0
- package/database/fileUrl.ts +34 -0
- package/database/hooks/useUserData.ts +489 -0
- package/database/index.ts +1 -0
- package/database/keys.ts +765 -0
- package/database/queryPrefixes.ts +14 -0
- package/database/requests.ts +443 -0
- package/database/runtimeServerContext.ts +35 -0
- package/database/server/MemoryDB.ts +76 -0
- package/database/server/actorAccess.ts +76 -0
- package/database/server/agentDelegation.ts +124 -0
- package/database/server/coreDataOwnership.ts +13 -0
- package/database/server/coreDataProxy.ts +76 -0
- package/database/server/cybotReadonly.ts +18 -0
- package/database/server/dataHandlers.ts +111 -0
- package/database/server/db.ts +118 -0
- package/database/server/dbPath.ts +20 -0
- package/database/server/delete.ts +499 -0
- package/database/server/emailRepository.ts +1480 -0
- package/database/server/ensureDbOpen.ts +12 -0
- package/database/server/fileRead.ts +337 -0
- package/database/server/fileService.ts +436 -0
- package/database/server/handleTransaction.ts +86 -0
- package/database/server/patch.ts +282 -0
- package/database/server/query.ts +138 -0
- package/database/server/read.ts +325 -0
- package/database/server/resourceAccess.ts +211 -0
- package/database/server/routes.ts +110 -0
- package/database/server/spaceMemberAuthority.ts +67 -0
- package/database/server/upload.ts +159 -0
- package/database/server/write.ts +494 -0
- package/database/server/writeAuthority.ts +133 -0
- package/database/sqliteDb.ts +46 -0
- package/database/table/deleteTable.ts +120 -0
- package/database/tenantPlacement.ts +57 -0
- package/database/tombstones.ts +52 -0
- package/database/userDataLoadDecision.ts +17 -0
- package/database/userDataMerge.ts +95 -0
- package/database/userPreferenceRegister.ts +108 -0
- package/database/utils/dbPath.ts +47 -0
- package/database/utils/ulid.native.ts +6 -0
- package/database/utils/ulid.ts +1 -0
- package/index.ts +37 -19
- package/localRuntimeDb.ts +28 -0
- package/package.json +17 -4
- package/runtimeModeArgs.ts +33 -0
- package/tui/readlineWorkspace.ts +1 -0
- package/tui/session.ts +22 -0
|
@@ -0,0 +1,155 @@
|
|
|
1
|
+
import { createDoc } from "render/page/docSlice";
|
|
2
|
+
import { selectCurrentSpaceId } from "create/space/spaceSlice";
|
|
3
|
+
import type { RootState } from "app/store";
|
|
4
|
+
import {
|
|
5
|
+
buildSkillDocFromArgs,
|
|
6
|
+
type CreateSkillDocArgs,
|
|
7
|
+
} from "ai/skills/skillDocBuilder";
|
|
8
|
+
import { buildSkillFollowupResult } from "./skillFollowup";
|
|
9
|
+
|
|
10
|
+
export interface CreateSkillDocToolArgs extends CreateSkillDocArgs {
|
|
11
|
+
spaceId?: string;
|
|
12
|
+
categoryId?: string;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export const createSkillDocFunctionSchema = {
|
|
16
|
+
name: "createSkillDoc",
|
|
17
|
+
description: [
|
|
18
|
+
"创建一个本地 skill 文档,并自动写入隐藏的 skill-config / eval-config 协议块。",
|
|
19
|
+
"适合把某个能力流程沉淀成可被 agent 引用的 skill,而不是手写协议注释。",
|
|
20
|
+
"创建成功后,如果用户还没说明下一步,优先调用 ui_ask_choice 继续询问:仅保存、挂到现有 Agent,还是新建一个 Agent 来使用它。",
|
|
21
|
+
].join("\n"),
|
|
22
|
+
parameters: {
|
|
23
|
+
type: "object",
|
|
24
|
+
properties: {
|
|
25
|
+
title: {
|
|
26
|
+
type: "string",
|
|
27
|
+
description: "可选:文档标题。未提供时回退到 name。",
|
|
28
|
+
},
|
|
29
|
+
name: {
|
|
30
|
+
type: "string",
|
|
31
|
+
description: "可选:skill 名称。未提供时回退到 title。",
|
|
32
|
+
},
|
|
33
|
+
description: {
|
|
34
|
+
type: "string",
|
|
35
|
+
description: "必填:skill 的用途说明。",
|
|
36
|
+
},
|
|
37
|
+
body: {
|
|
38
|
+
type: "string",
|
|
39
|
+
description: "可选:skill 的正文说明、步骤、注意事项等 Markdown 内容。",
|
|
40
|
+
},
|
|
41
|
+
spaceId: {
|
|
42
|
+
type: "string",
|
|
43
|
+
description: '可选:目标 spaceId;不传则优先使用当前 space。可传空字符串 ""。',
|
|
44
|
+
},
|
|
45
|
+
categoryId: {
|
|
46
|
+
type: "string",
|
|
47
|
+
description: '可选:目标分类 categoryId;不传时传空字符串 ""。',
|
|
48
|
+
},
|
|
49
|
+
toolNames: {
|
|
50
|
+
type: "array",
|
|
51
|
+
items: { type: "string" },
|
|
52
|
+
description: "可选:该 skill 绑定的工具名列表。",
|
|
53
|
+
},
|
|
54
|
+
requiredSkills: {
|
|
55
|
+
type: "array",
|
|
56
|
+
items: { type: "string" },
|
|
57
|
+
description: "可选:硬依赖的 skill key/page key 列表。",
|
|
58
|
+
},
|
|
59
|
+
recommendedSkills: {
|
|
60
|
+
type: "array",
|
|
61
|
+
items: { type: "string" },
|
|
62
|
+
description: "可选:推荐但不强制的 skill key/page key 列表。",
|
|
63
|
+
},
|
|
64
|
+
promptPatch: {
|
|
65
|
+
type: "string",
|
|
66
|
+
description: "可选:附加到 system prompt 的技能提示。",
|
|
67
|
+
},
|
|
68
|
+
budgetTier: {
|
|
69
|
+
type: "string",
|
|
70
|
+
enum: ["low", "medium", "high"],
|
|
71
|
+
description: "可选:该 skill 的预算等级。",
|
|
72
|
+
},
|
|
73
|
+
dispatchPreferred: {
|
|
74
|
+
type: "boolean",
|
|
75
|
+
description: "可选:是否建议优先走 agent dispatch 而不是直接本 agent 执行。",
|
|
76
|
+
},
|
|
77
|
+
modalities: {
|
|
78
|
+
type: "array",
|
|
79
|
+
items: { type: "string", enum: ["text", "image", "video", "audio", "3d"] },
|
|
80
|
+
description: "可选:该 skill 涉及的模态。",
|
|
81
|
+
},
|
|
82
|
+
preferredAgents: {
|
|
83
|
+
type: "array",
|
|
84
|
+
items: { type: "string" },
|
|
85
|
+
description: "可选:更适合执行该 skill 的 agent 名称或 key。",
|
|
86
|
+
},
|
|
87
|
+
discoverKeywords: {
|
|
88
|
+
type: "array",
|
|
89
|
+
items: { type: "string" },
|
|
90
|
+
description: "可选:触发/发现关键字。",
|
|
91
|
+
},
|
|
92
|
+
discoverExamples: {
|
|
93
|
+
type: "array",
|
|
94
|
+
items: { type: "string" },
|
|
95
|
+
description: "可选:示例任务或问法。",
|
|
96
|
+
},
|
|
97
|
+
evalCases: {
|
|
98
|
+
type: "array",
|
|
99
|
+
items: {
|
|
100
|
+
type: "object",
|
|
101
|
+
properties: {
|
|
102
|
+
input: { type: "string" },
|
|
103
|
+
expectedTools: {
|
|
104
|
+
type: "array",
|
|
105
|
+
items: { type: "string" },
|
|
106
|
+
},
|
|
107
|
+
expectedSignals: {
|
|
108
|
+
type: "array",
|
|
109
|
+
items: { type: "string" },
|
|
110
|
+
},
|
|
111
|
+
forbiddenSignals: {
|
|
112
|
+
type: "array",
|
|
113
|
+
items: { type: "string" },
|
|
114
|
+
},
|
|
115
|
+
},
|
|
116
|
+
required: ["input"],
|
|
117
|
+
},
|
|
118
|
+
description: "可选:skill 的评估用例,写入 eval-config。",
|
|
119
|
+
},
|
|
120
|
+
},
|
|
121
|
+
required: ["description"],
|
|
122
|
+
} as const,
|
|
123
|
+
};
|
|
124
|
+
|
|
125
|
+
export async function createSkillDocFunc(
|
|
126
|
+
args: CreateSkillDocToolArgs,
|
|
127
|
+
thunkApi: any
|
|
128
|
+
): Promise<{ rawData: unknown; displayData: string }> {
|
|
129
|
+
const { dispatch, getState } = thunkApi;
|
|
130
|
+
const state = getState() as RootState;
|
|
131
|
+
const { title, content, skillConfig, evalConfig } = buildSkillDocFromArgs(args);
|
|
132
|
+
|
|
133
|
+
const explicitSpaceId = (args.spaceId ?? "").trim() || undefined;
|
|
134
|
+
const currentSpaceId = selectCurrentSpaceId(state) || undefined;
|
|
135
|
+
const spaceId = explicitSpaceId ?? currentSpaceId;
|
|
136
|
+
const categoryId = (args.categoryId ?? "").trim() || undefined;
|
|
137
|
+
|
|
138
|
+
const id = await (dispatch as any)(
|
|
139
|
+
(createDoc as any)({
|
|
140
|
+
title,
|
|
141
|
+
spaceId,
|
|
142
|
+
categoryId,
|
|
143
|
+
content,
|
|
144
|
+
})
|
|
145
|
+
).unwrap();
|
|
146
|
+
|
|
147
|
+
return buildSkillFollowupResult({
|
|
148
|
+
dbKey: id,
|
|
149
|
+
title,
|
|
150
|
+
skillId: skillConfig.id,
|
|
151
|
+
spaceId: spaceId ?? null,
|
|
152
|
+
toolNames: skillConfig.toolNames ?? [],
|
|
153
|
+
hasEvalConfig: !!evalConfig,
|
|
154
|
+
});
|
|
155
|
+
}
|
|
@@ -0,0 +1,154 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* createWorkflow tool
|
|
3
|
+
*
|
|
4
|
+
* LLM calls this ONCE to define and immediately execute a workflow.
|
|
5
|
+
* The execution engine runs all steps deterministically — no further
|
|
6
|
+
* LLM involvement needed for orchestration, so token cost is O(1)
|
|
7
|
+
* regardless of step count.
|
|
8
|
+
*
|
|
9
|
+
* Use when the task path is known upfront.
|
|
10
|
+
* Use agent loop when the path needs dynamic discovery.
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
import { runWorkflow } from "ai/workflow/workflowExecutor";
|
|
14
|
+
import type {
|
|
15
|
+
WorkflowDefinition,
|
|
16
|
+
WorkflowStep,
|
|
17
|
+
} from "ai/workflow/workflowTypes";
|
|
18
|
+
|
|
19
|
+
export const createWorkflowFunctionSchema = {
|
|
20
|
+
name: "createWorkflow",
|
|
21
|
+
description:
|
|
22
|
+
"当你已知完整执行路径时使用本工具。定义一个多步骤 workflow,引擎自动执行全部步骤," +
|
|
23
|
+
"不需要每步都调用 LLM 决策,大幅节省 token。" +
|
|
24
|
+
"步骤类型:" +
|
|
25
|
+
"'tool'——直接执行注册工具;" +
|
|
26
|
+
"'llm'——单次 LLM 调用(无 tool loop);" +
|
|
27
|
+
"'parallel'——子步骤并发执行;" +
|
|
28
|
+
"'condition'——纯逻辑判断分支,无需 LLM。" +
|
|
29
|
+
"步骤间通过 {{steps.STEP_ID.result}} 传递数据。",
|
|
30
|
+
parameters: {
|
|
31
|
+
type: "object",
|
|
32
|
+
properties: {
|
|
33
|
+
title: {
|
|
34
|
+
type: "string",
|
|
35
|
+
description: "Workflow 标题,简明描述整体目标。",
|
|
36
|
+
},
|
|
37
|
+
steps: {
|
|
38
|
+
type: "array",
|
|
39
|
+
description: "有序步骤列表。每步声明类型和所需参数,引擎顺序执行(parallel 步骤内部并发)。",
|
|
40
|
+
items: {
|
|
41
|
+
type: "object",
|
|
42
|
+
properties: {
|
|
43
|
+
id: {
|
|
44
|
+
type: "string",
|
|
45
|
+
description: "步骤唯一 ID,后续步骤通过 {{steps.ID.result}} 引用。",
|
|
46
|
+
},
|
|
47
|
+
type: {
|
|
48
|
+
type: "string",
|
|
49
|
+
enum: ["tool", "llm", "parallel", "condition"],
|
|
50
|
+
description: "步骤类型。",
|
|
51
|
+
},
|
|
52
|
+
title: {
|
|
53
|
+
type: "string",
|
|
54
|
+
description: "步骤人类可读描述(可选)。",
|
|
55
|
+
},
|
|
56
|
+
// tool step
|
|
57
|
+
tool: {
|
|
58
|
+
type: "string",
|
|
59
|
+
description: "type='tool' 时必填,工具名称。",
|
|
60
|
+
},
|
|
61
|
+
args: {
|
|
62
|
+
type: "object",
|
|
63
|
+
description:
|
|
64
|
+
"type='tool' 时必填,工具参数。支持 {{steps.ID.result}} 和 {{steps.ID.result[N]}} 占位符。" +
|
|
65
|
+
"parallel 子步骤结果也可直接通过其自身 ID 引用,例如 {{steps.subStepId.result}}。",
|
|
66
|
+
},
|
|
67
|
+
// llm step
|
|
68
|
+
prompt: {
|
|
69
|
+
type: "string",
|
|
70
|
+
description: "type='llm' 时必填,发送给模型的 prompt。支持占位符。",
|
|
71
|
+
},
|
|
72
|
+
model: {
|
|
73
|
+
type: "string",
|
|
74
|
+
description: "type='llm' 时可选,指定模型名。",
|
|
75
|
+
},
|
|
76
|
+
// parallel step
|
|
77
|
+
steps: {
|
|
78
|
+
type: "array",
|
|
79
|
+
description:
|
|
80
|
+
"type='parallel' 时必填,子步骤列表(tool 或 llm 类型),并发执行。" +
|
|
81
|
+
"子步骤执行完成后,其结果同时以子步骤自身 ID 注册到全局 steps map," +
|
|
82
|
+
"后续步骤可通过 {{steps.subStepId.result}} 直接引用,无需通过父 parallel 步骤中转。",
|
|
83
|
+
items: { type: "object" },
|
|
84
|
+
},
|
|
85
|
+
// condition step
|
|
86
|
+
check: {
|
|
87
|
+
type: "string",
|
|
88
|
+
description:
|
|
89
|
+
"type='condition' 时必填,纯 JS 布尔表达式。" +
|
|
90
|
+
"steps.<stepId>.<prop> 表示某步骤结果的属性(点路径,不支持 bracket notation,无 .result 包装)。" +
|
|
91
|
+
"支持:点路径 / 字符串和数字字面量 / 布尔 null / 比较运算符 === !== > < >= <= / 逻辑运算符 && || ! / 括号。" +
|
|
92
|
+
"例: \"steps.validate.isValid === true\" 或 \"steps.score.value >= 80 && steps.flag.ok !== false\"",
|
|
93
|
+
},
|
|
94
|
+
ifTrue: {
|
|
95
|
+
type: "array",
|
|
96
|
+
items: { type: "string" },
|
|
97
|
+
description: "条件为真时执行的步骤 ID 列表,其余步骤跳过。",
|
|
98
|
+
},
|
|
99
|
+
ifFalse: {
|
|
100
|
+
type: "array",
|
|
101
|
+
items: { type: "string" },
|
|
102
|
+
description: "条件为假时执行的步骤 ID 列表,其余步骤跳过。",
|
|
103
|
+
},
|
|
104
|
+
// error handling
|
|
105
|
+
onError: {
|
|
106
|
+
type: "string",
|
|
107
|
+
enum: ["stop", "skip", "retry"],
|
|
108
|
+
description:
|
|
109
|
+
"步骤失败时策略:stop(默认,终止 workflow)/ skip(跳过,结果记为 null 继续)" +
|
|
110
|
+
"/ retry(按 retryCount 次数重试,耗尽后若仍失败则触发 stop)。",
|
|
111
|
+
},
|
|
112
|
+
retryCount: {
|
|
113
|
+
type: "integer",
|
|
114
|
+
description: "onError='retry' 时重试次数,默认 1(即最多执行 2 次)。",
|
|
115
|
+
},
|
|
116
|
+
},
|
|
117
|
+
required: ["id", "type"],
|
|
118
|
+
},
|
|
119
|
+
},
|
|
120
|
+
},
|
|
121
|
+
required: ["title", "steps"],
|
|
122
|
+
},
|
|
123
|
+
};
|
|
124
|
+
|
|
125
|
+
interface CreateWorkflowArgs {
|
|
126
|
+
title: string;
|
|
127
|
+
steps: WorkflowStep[];
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
export async function createWorkflowFunc(
|
|
131
|
+
args: CreateWorkflowArgs,
|
|
132
|
+
thunkApi: any
|
|
133
|
+
): Promise<{ rawData: any; displayData?: string }> {
|
|
134
|
+
const { title, steps } = args;
|
|
135
|
+
|
|
136
|
+
if (!steps?.length) {
|
|
137
|
+
throw new Error("createWorkflow: steps 不能为空。");
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
const definition: WorkflowDefinition = { title, steps };
|
|
141
|
+
|
|
142
|
+
const { dispatch } = thunkApi;
|
|
143
|
+
const result = await dispatch(runWorkflow({ definition })).unwrap();
|
|
144
|
+
|
|
145
|
+
const completedCount = Object.values(result.results).filter(
|
|
146
|
+
(v) => v !== null && v !== undefined
|
|
147
|
+
).length;
|
|
148
|
+
|
|
149
|
+
const displayData = result.success
|
|
150
|
+
? `✅ **${title}** 完成,共执行 ${completedCount} 步`
|
|
151
|
+
: `❌ **${title}** 在步骤 \`${result.failedStep}\` 失败:${result.error}`;
|
|
152
|
+
|
|
153
|
+
return { rawData: result, displayData };
|
|
154
|
+
}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import { ToolDefinition } from ".";
|
|
2
|
+
import { callToolApi } from "./toolApiClient";
|
|
3
|
+
|
|
4
|
+
export const deepseekOcrSchema = {
|
|
5
|
+
name: "deepseek_ocr",
|
|
6
|
+
description: "使用 deepseek-ai/DeepSeek-OCR 模型进行光学字符识别 (OCR)。擅长中英文混合文档、手写体及复杂版式的精准识别。",
|
|
7
|
+
parameters: {
|
|
8
|
+
type: "object",
|
|
9
|
+
properties: {
|
|
10
|
+
imageUrl: {
|
|
11
|
+
type: "string",
|
|
12
|
+
description: "要进行 OCR 的图片 URL、base64 编码的图片数据,或者内部文件 ID (如 'file-...')。",
|
|
13
|
+
},
|
|
14
|
+
prompt: {
|
|
15
|
+
type: "string",
|
|
16
|
+
description: "可选的提示词,用于指导模型如何提取信息(例如:'提取所有表格数据' 或 '识别图片中的所有文字')。默认为 '识别图片中的所有文字'。",
|
|
17
|
+
default: "识别图片中的所有文字"
|
|
18
|
+
}
|
|
19
|
+
},
|
|
20
|
+
required: ["imageUrl"],
|
|
21
|
+
},
|
|
22
|
+
};
|
|
23
|
+
|
|
24
|
+
export const deepseekOcrFunc: ToolDefinition["executor"] = async (input: any, thunkApi) => {
|
|
25
|
+
const { imageUrl, prompt = "识别图片中的所有文字" } = input;
|
|
26
|
+
|
|
27
|
+
const data = await callToolApi(thunkApi, "/api/deepseek-ocr", { imageUrl, prompt }, { withAuth: true });
|
|
28
|
+
|
|
29
|
+
return {
|
|
30
|
+
summary: "OCR processing completed",
|
|
31
|
+
text: data.choices?.[0]?.message?.content || "No text detected",
|
|
32
|
+
rawData: data
|
|
33
|
+
};
|
|
34
|
+
};
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
// ai/tools/delayTool.ts
|
|
2
|
+
|
|
3
|
+
export const delayFunctionSchema = {
|
|
4
|
+
name: "delay",
|
|
5
|
+
description:
|
|
6
|
+
"让计划暂停一小段时间(毫秒),用于节流批量操作(例如连续下载多个文件)。",
|
|
7
|
+
parameters: {
|
|
8
|
+
type: "object",
|
|
9
|
+
properties: {
|
|
10
|
+
ms: {
|
|
11
|
+
type: "integer",
|
|
12
|
+
description: "需要等待的时间(毫秒),建议 200~1000 之间。",
|
|
13
|
+
},
|
|
14
|
+
},
|
|
15
|
+
required: ["ms"],
|
|
16
|
+
},
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
export async function delayFunc(
|
|
20
|
+
args: { ms: number },
|
|
21
|
+
_thunkApi: any
|
|
22
|
+
): Promise<{ rawData: object; displayData: string }> {
|
|
23
|
+
const ms = typeof args.ms === "number" && args.ms > 0 ? args.ms : 200;
|
|
24
|
+
|
|
25
|
+
await new Promise<void>((resolve) => setTimeout(resolve, ms));
|
|
26
|
+
|
|
27
|
+
const rawData = { success: true, ms };
|
|
28
|
+
const displayData = `已等待 ${ms} 毫秒,继续执行后续步骤。`;
|
|
29
|
+
|
|
30
|
+
return { rawData, displayData };
|
|
31
|
+
}
|
|
@@ -0,0 +1,325 @@
|
|
|
1
|
+
import { selectUserId } from "auth/authSlice";
|
|
2
|
+
import { createSpaceKey } from "create/space/spaceKeys";
|
|
3
|
+
import { deleteSpace, selectAllMemberSpaces } from "create/space/spaceSlice";
|
|
4
|
+
import { read } from "database/dbSlice";
|
|
5
|
+
import { getRuntimeServerContext } from "database/runtimeServerContext";
|
|
6
|
+
|
|
7
|
+
import {
|
|
8
|
+
buildDeleteSpacesPreview,
|
|
9
|
+
filterSpaceDeletionCandidates,
|
|
10
|
+
resolveConfirmedSpaceDeletionTargets,
|
|
11
|
+
type DeleteSpacesMatchMode,
|
|
12
|
+
type SpaceDeletionPreview,
|
|
13
|
+
type SpaceMembershipLike,
|
|
14
|
+
type SpaceRecordLike,
|
|
15
|
+
} from "./deleteSpacesToolModel";
|
|
16
|
+
|
|
17
|
+
export interface DeleteSpacesToolArgs {
|
|
18
|
+
query: string;
|
|
19
|
+
matchMode?: DeleteSpacesMatchMode;
|
|
20
|
+
confirmedSpaceIds?: string[];
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
interface DeleteSpacesToolDeps {
|
|
24
|
+
selectCurrentUserId: (state: any) => string | undefined;
|
|
25
|
+
selectMemberSpaces: (state: any) => SpaceMembershipLike[];
|
|
26
|
+
readSpaceRecord: (thunkApi: any, spaceId: string) => Promise<SpaceRecordLike | null>;
|
|
27
|
+
selectDeleteServers: (state: any) => string[];
|
|
28
|
+
deleteServerKey: (
|
|
29
|
+
thunkApi: any,
|
|
30
|
+
server: string,
|
|
31
|
+
dbKey: string
|
|
32
|
+
) => Promise<{ ok: boolean; status: number; detail: string }>;
|
|
33
|
+
deleteOwnedSpace: (
|
|
34
|
+
thunkApi: any,
|
|
35
|
+
input: { spaceId: string; strategy: "delete-space-only" }
|
|
36
|
+
) => Promise<void>;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
export const deleteSpacesFunctionSchema = {
|
|
40
|
+
name: "deleteSpaces",
|
|
41
|
+
description: [
|
|
42
|
+
"按名称或 ID 查找并删除当前用户拥有的 Space。",
|
|
43
|
+
"这是危险操作:第一次调用会返回待删除列表并等待用户确认;确认后才执行删除。",
|
|
44
|
+
"默认只删除 Space 壳和成员关系,不删除其中挂载的 doc/dialog/file 内容。",
|
|
45
|
+
"非 owner 的 Space 会被跳过,不会删除。",
|
|
46
|
+
].join("\n"),
|
|
47
|
+
parameters: {
|
|
48
|
+
type: "object",
|
|
49
|
+
properties: {
|
|
50
|
+
query: {
|
|
51
|
+
type: "string",
|
|
52
|
+
description:
|
|
53
|
+
"要删除的 Space 筛选词,例如 rn_owner_verify_0504,或配合 matchMode=spaceId 传入 space id。",
|
|
54
|
+
},
|
|
55
|
+
matchMode: {
|
|
56
|
+
type: "string",
|
|
57
|
+
enum: ["prefix", "exact", "contains", "spaceId"],
|
|
58
|
+
description:
|
|
59
|
+
"匹配方式。prefix=名称前缀,exact=名称完全匹配,contains=名称包含,spaceId=按 Space ID 匹配。默认 prefix。",
|
|
60
|
+
default: "prefix",
|
|
61
|
+
},
|
|
62
|
+
confirmedSpaceIds: {
|
|
63
|
+
type: "array",
|
|
64
|
+
items: { type: "string" },
|
|
65
|
+
description:
|
|
66
|
+
"可选:确认阶段要删除的 Space ID 列表。不传时,用户点击确认会删除预览列表中的全部可删除项。",
|
|
67
|
+
},
|
|
68
|
+
},
|
|
69
|
+
required: ["query"],
|
|
70
|
+
},
|
|
71
|
+
};
|
|
72
|
+
|
|
73
|
+
const defaultDeps: DeleteSpacesToolDeps = {
|
|
74
|
+
selectCurrentUserId: selectUserId,
|
|
75
|
+
selectMemberSpaces: (state) => selectAllMemberSpaces(state) as SpaceMembershipLike[],
|
|
76
|
+
readSpaceRecord: async (thunkApi, spaceId) => {
|
|
77
|
+
try {
|
|
78
|
+
const record = await thunkApi.dispatch(
|
|
79
|
+
read({ dbKey: createSpaceKey.space(spaceId), waitRemote: true } as any)
|
|
80
|
+
).unwrap();
|
|
81
|
+
return record ?? null;
|
|
82
|
+
} catch {
|
|
83
|
+
return null;
|
|
84
|
+
}
|
|
85
|
+
},
|
|
86
|
+
selectDeleteServers: (state) => {
|
|
87
|
+
const { currentServer, remoteServers } = getRuntimeServerContext(state);
|
|
88
|
+
return remoteServers.length > 0
|
|
89
|
+
? remoteServers
|
|
90
|
+
: currentServer
|
|
91
|
+
? [currentServer]
|
|
92
|
+
: [];
|
|
93
|
+
},
|
|
94
|
+
deleteServerKey: async (thunkApi, server, dbKey) => {
|
|
95
|
+
const state = thunkApi.getState();
|
|
96
|
+
const token = state?.auth?.currentToken;
|
|
97
|
+
if (!server) {
|
|
98
|
+
return { ok: false, status: 0, detail: "missing server" };
|
|
99
|
+
}
|
|
100
|
+
if (!token) {
|
|
101
|
+
return { ok: false, status: 401, detail: "missing auth token" };
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
const res = await fetch(
|
|
105
|
+
`${server.replace(/\/+$/, "")}/api/v1/db/delete/${encodeURIComponent(dbKey)}`,
|
|
106
|
+
{
|
|
107
|
+
method: "DELETE",
|
|
108
|
+
headers: { Authorization: `Bearer ${token}` },
|
|
109
|
+
}
|
|
110
|
+
);
|
|
111
|
+
return {
|
|
112
|
+
ok: res.ok,
|
|
113
|
+
status: res.status,
|
|
114
|
+
detail: await res.text().catch(() => ""),
|
|
115
|
+
};
|
|
116
|
+
},
|
|
117
|
+
deleteOwnedSpace: async (thunkApi, input) => {
|
|
118
|
+
await thunkApi.dispatch(deleteSpace(input as any)).unwrap();
|
|
119
|
+
},
|
|
120
|
+
};
|
|
121
|
+
|
|
122
|
+
const readSpaceRecordsById = async (
|
|
123
|
+
thunkApi: any,
|
|
124
|
+
memberships: SpaceMembershipLike[],
|
|
125
|
+
deps: DeleteSpacesToolDeps
|
|
126
|
+
) => {
|
|
127
|
+
const recordsById: Record<string, SpaceRecordLike | null> = {};
|
|
128
|
+
|
|
129
|
+
for (const membership of memberships) {
|
|
130
|
+
const spaceId = String(membership.spaceId ?? "").replace(/^space-/, "");
|
|
131
|
+
if (!spaceId) continue;
|
|
132
|
+
recordsById[spaceId] = await deps.readSpaceRecord(thunkApi, spaceId);
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
return recordsById;
|
|
136
|
+
};
|
|
137
|
+
|
|
138
|
+
const loadDeletionContext = async (
|
|
139
|
+
args: DeleteSpacesToolArgs,
|
|
140
|
+
thunkApi: any,
|
|
141
|
+
deps: DeleteSpacesToolDeps
|
|
142
|
+
): Promise<{
|
|
143
|
+
currentUserId: string;
|
|
144
|
+
preview: SpaceDeletionPreview;
|
|
145
|
+
spaceRecordsById: Record<string, SpaceRecordLike | null>;
|
|
146
|
+
deleteServers: string[];
|
|
147
|
+
}> => {
|
|
148
|
+
const state = thunkApi.getState();
|
|
149
|
+
const currentUserId = deps.selectCurrentUserId(state);
|
|
150
|
+
if (!currentUserId) {
|
|
151
|
+
throw new Error("deleteSpaces 需要当前登录用户。");
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
const memberships = deps.selectMemberSpaces(state);
|
|
155
|
+
const deleteServers = deps.selectDeleteServers(state);
|
|
156
|
+
const candidates = filterSpaceDeletionCandidates(memberships, {
|
|
157
|
+
query: args.query,
|
|
158
|
+
matchMode: args.matchMode,
|
|
159
|
+
});
|
|
160
|
+
const spaceRecordsById = await readSpaceRecordsById(thunkApi, candidates, deps);
|
|
161
|
+
|
|
162
|
+
const preview = buildDeleteSpacesPreview({
|
|
163
|
+
currentUserId,
|
|
164
|
+
candidates,
|
|
165
|
+
spaceRecordsById,
|
|
166
|
+
});
|
|
167
|
+
|
|
168
|
+
return { currentUserId, preview, spaceRecordsById, deleteServers };
|
|
169
|
+
};
|
|
170
|
+
|
|
171
|
+
const loadPreview = async (
|
|
172
|
+
args: DeleteSpacesToolArgs,
|
|
173
|
+
thunkApi: any,
|
|
174
|
+
deps: DeleteSpacesToolDeps
|
|
175
|
+
): Promise<SpaceDeletionPreview> => {
|
|
176
|
+
const { preview } = await loadDeletionContext(args, thunkApi, deps);
|
|
177
|
+
return preview;
|
|
178
|
+
};
|
|
179
|
+
|
|
180
|
+
const spaceMemberKeysForDelete = (
|
|
181
|
+
currentUserId: string,
|
|
182
|
+
spaceId: string,
|
|
183
|
+
record: SpaceRecordLike
|
|
184
|
+
) => {
|
|
185
|
+
const memberIds = new Set<string>([currentUserId]);
|
|
186
|
+
if (Array.isArray(record.members)) {
|
|
187
|
+
for (const memberId of record.members) {
|
|
188
|
+
if (typeof memberId === "string" && memberId.trim()) {
|
|
189
|
+
memberIds.add(memberId.trim());
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
} else if (record.members && typeof record.members === "object") {
|
|
193
|
+
for (const memberId of Object.keys(record.members)) {
|
|
194
|
+
if (memberId.trim()) memberIds.add(memberId.trim());
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
return Array.from(memberIds).map((memberId) =>
|
|
198
|
+
createSpaceKey.member(memberId, spaceId)
|
|
199
|
+
);
|
|
200
|
+
};
|
|
201
|
+
|
|
202
|
+
const formatPreview = (preview: SpaceDeletionPreview) => {
|
|
203
|
+
if (preview.deletable.length === 0 && preview.skipped.length === 0) {
|
|
204
|
+
return "没有找到匹配的 Space。";
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
const lines = [
|
|
208
|
+
`找到 ${preview.deletable.length} 个可删除 Space,${preview.skipped.length} 个跳过。`,
|
|
209
|
+
"",
|
|
210
|
+
];
|
|
211
|
+
|
|
212
|
+
if (preview.deletable.length > 0) {
|
|
213
|
+
lines.push("将删除这些 Space 壳和成员关系,不删除其中 doc/dialog/file:");
|
|
214
|
+
for (const item of preview.deletable) {
|
|
215
|
+
lines.push(
|
|
216
|
+
`- ${item.name} (${item.spaceId}),成员 ${item.memberCount},内容 ${item.contentCount}`
|
|
217
|
+
);
|
|
218
|
+
}
|
|
219
|
+
lines.push("");
|
|
220
|
+
lines.push("需要确认后才会删除。");
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
if (preview.skipped.length > 0) {
|
|
224
|
+
lines.push("跳过:");
|
|
225
|
+
for (const item of preview.skipped) {
|
|
226
|
+
const owner = item.ownerId ? `,owner=${item.ownerId}` : "";
|
|
227
|
+
lines.push(`- ${item.name || item.spaceId} (${item.spaceId}):${item.reason}${owner}`);
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
return lines.join("\n");
|
|
232
|
+
};
|
|
233
|
+
|
|
234
|
+
export function createDeleteSpacesToolHandlers(deps: DeleteSpacesToolDeps) {
|
|
235
|
+
return {
|
|
236
|
+
preview: async (args: DeleteSpacesToolArgs, thunkApi: any) => {
|
|
237
|
+
const preview = await loadPreview(args, thunkApi, deps);
|
|
238
|
+
return {
|
|
239
|
+
rawData: preview,
|
|
240
|
+
displayData: formatPreview(preview),
|
|
241
|
+
};
|
|
242
|
+
},
|
|
243
|
+
execute: async (args: DeleteSpacesToolArgs, thunkApi: any) => {
|
|
244
|
+
const { currentUserId, preview, spaceRecordsById, deleteServers } =
|
|
245
|
+
await loadDeletionContext(args, thunkApi, deps);
|
|
246
|
+
const confirmedIds =
|
|
247
|
+
Array.isArray(args.confirmedSpaceIds) && args.confirmedSpaceIds.length > 0
|
|
248
|
+
? args.confirmedSpaceIds
|
|
249
|
+
: preview.deletable.map((item) => item.spaceId);
|
|
250
|
+
const { targets, missingConfirmedSpaceIds } =
|
|
251
|
+
resolveConfirmedSpaceDeletionTargets(preview, confirmedIds);
|
|
252
|
+
|
|
253
|
+
const deletedSpaceIds: string[] = [];
|
|
254
|
+
const deletedKeys: string[] = [];
|
|
255
|
+
const deletedRecords: Array<{ server: string; dbKey: string }> = [];
|
|
256
|
+
const failures: Array<{ server?: string; dbKey: string; status: number; detail: string }> = [];
|
|
257
|
+
for (const target of targets) {
|
|
258
|
+
const record = spaceRecordsById[target.spaceId];
|
|
259
|
+
if (!record) {
|
|
260
|
+
failures.push({
|
|
261
|
+
dbKey: createSpaceKey.space(target.spaceId),
|
|
262
|
+
status: 404,
|
|
263
|
+
detail: "space record disappeared before deletion",
|
|
264
|
+
});
|
|
265
|
+
continue;
|
|
266
|
+
}
|
|
267
|
+
if (deleteServers.length === 0) {
|
|
268
|
+
failures.push({
|
|
269
|
+
dbKey: createSpaceKey.space(target.spaceId),
|
|
270
|
+
status: 0,
|
|
271
|
+
detail: "no delete server configured",
|
|
272
|
+
});
|
|
273
|
+
continue;
|
|
274
|
+
}
|
|
275
|
+
const keys = [
|
|
276
|
+
...spaceMemberKeysForDelete(currentUserId, target.spaceId, record),
|
|
277
|
+
createSpaceKey.space(target.spaceId),
|
|
278
|
+
];
|
|
279
|
+
let targetFailed = false;
|
|
280
|
+
for (const server of deleteServers) {
|
|
281
|
+
for (const dbKey of keys) {
|
|
282
|
+
const result = await deps.deleteServerKey(thunkApi, server, dbKey);
|
|
283
|
+
if (!result.ok) {
|
|
284
|
+
targetFailed = true;
|
|
285
|
+
failures.push({ server, dbKey, status: result.status, detail: result.detail });
|
|
286
|
+
} else {
|
|
287
|
+
deletedKeys.push(dbKey);
|
|
288
|
+
deletedRecords.push({ server, dbKey });
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
if (targetFailed) continue;
|
|
293
|
+
|
|
294
|
+
await deps.deleteOwnedSpace(thunkApi, {
|
|
295
|
+
spaceId: target.spaceId,
|
|
296
|
+
strategy: "delete-space-only",
|
|
297
|
+
});
|
|
298
|
+
deletedSpaceIds.push(target.spaceId);
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
const displayData =
|
|
302
|
+
deletedSpaceIds.length > 0
|
|
303
|
+
? `已删除 ${deletedSpaceIds.length} 个 Space:${deletedSpaceIds.join(", ")}。内容记录未被删除。`
|
|
304
|
+
: "没有删除任何 Space。";
|
|
305
|
+
|
|
306
|
+
return {
|
|
307
|
+
rawData: {
|
|
308
|
+
deletedSpaceIds,
|
|
309
|
+
deletedKeys,
|
|
310
|
+
deletedRecords,
|
|
311
|
+
deleteServers,
|
|
312
|
+
missingConfirmedSpaceIds,
|
|
313
|
+
skipped: preview.skipped,
|
|
314
|
+
failures,
|
|
315
|
+
},
|
|
316
|
+
displayData,
|
|
317
|
+
};
|
|
318
|
+
},
|
|
319
|
+
};
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
const defaultHandlers = createDeleteSpacesToolHandlers(defaultDeps);
|
|
323
|
+
|
|
324
|
+
export const deleteSpacesPreviewFunc = defaultHandlers.preview;
|
|
325
|
+
export const deleteSpacesFunc = defaultHandlers.execute;
|