nolo-cli 0.1.8 → 0.1.10

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 +1079 -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 +22 -6
@@ -0,0 +1,259 @@
1
+ // ai/agent/planSlice.ts
2
+ // 管理计划(Plan)的 Redux 状态
3
+ // 支持普通步骤和反思步骤(reflect step)
4
+
5
+ import { type PayloadAction, createSlice } from "@reduxjs/toolkit";
6
+ import type { RootState } from "app/store";
7
+
8
+ // --- Interfaces ---
9
+
10
+ export interface PlanState {
11
+ planDetails: string;
12
+ currentProgress: number;
13
+ }
14
+
15
+ // 单个工具调用的接口
16
+ export interface ToolCall {
17
+ tool_name: string;
18
+ parameters: any;
19
+ }
20
+
21
+ // 反思决策的输出结构
22
+ export interface ReflectDecision {
23
+ action: "continue" | "stop" | "insert_steps";
24
+ reason: string;
25
+ steps?: Omit<Step, "status" | "result">[]; // 仅当 action 是 insert_steps 时
26
+ }
27
+
28
+ // Step 类型:normal 或 reflect
29
+ export type StepType = "normal" | "reflect";
30
+
31
+ export interface Step {
32
+ id: string;
33
+ title: string;
34
+ type?: StepType; // 默认 "normal"
35
+ status: "pending" | "in-progress" | "completed" | "failed";
36
+
37
+ // normal step 使用
38
+ calls?: ToolCall[];
39
+
40
+ // reflect step 使用
41
+ reflectInput?: string; // 给反思 LLM 的提示
42
+
43
+ // 通用
44
+ details?: any;
45
+ result?: any[]; // normal step: 工具调用结果; reflect step: ReflectDecision
46
+ }
47
+
48
+ // Plan 执行配置
49
+ export interface PlanExecutionConfig {
50
+ maxReflectCount?: number; // 最多允许多少次 reflect step,默认不限
51
+ maxTotalSteps?: number; // 最多执行多少个 step(包括动态插入的),默认不限
52
+ maxTimeMs?: number; // 最大执行时间(毫秒),默认不限
53
+ }
54
+
55
+ interface PlanSliceState {
56
+ plan: PlanState | null;
57
+ steps: Step[];
58
+ currentStep: string | null;
59
+ executionConfig: PlanExecutionConfig | null;
60
+ // 执行统计
61
+ stats: {
62
+ reflectCount: number;
63
+ totalStepsExecuted: number;
64
+ startTime: number | null;
65
+ };
66
+ }
67
+
68
+ // --- Initial State ---
69
+
70
+ const initialState: PlanSliceState = {
71
+ plan: null,
72
+ steps: [],
73
+ currentStep: null,
74
+ executionConfig: null,
75
+ stats: {
76
+ reflectCount: 0,
77
+ totalStepsExecuted: 0,
78
+ startTime: null,
79
+ },
80
+ };
81
+
82
+ // --- Slice Definition ---
83
+
84
+ const planSlice = createSlice({
85
+ name: "plan",
86
+ initialState,
87
+ reducers: {
88
+ // 设置整个计划的顶层信息
89
+ setPlan: (state, action: PayloadAction<PlanState>) => {
90
+ state.plan = action.payload;
91
+ },
92
+
93
+ // 更新计划的整体进度
94
+ updatePlanProgress: (state, action: PayloadAction<number>) => {
95
+ if (state.plan) {
96
+ state.plan.currentProgress = action.payload;
97
+ }
98
+ },
99
+
100
+ // 清除整个计划
101
+ clearPlan: (state) => {
102
+ state.plan = null;
103
+ state.steps = [];
104
+ state.currentStep = null;
105
+ state.executionConfig = null;
106
+ state.stats = {
107
+ reflectCount: 0,
108
+ totalStepsExecuted: 0,
109
+ startTime: null,
110
+ };
111
+ },
112
+
113
+ // 设置计划的所有步骤
114
+ setSteps: (state, action: PayloadAction<Step[]>) => {
115
+ state.steps = action.payload;
116
+ },
117
+
118
+ // 更新单个步骤的状态或结果
119
+ updateStep: (
120
+ state,
121
+ action: PayloadAction<{ id: string; updates: Partial<Step> }>
122
+ ) => {
123
+ const step = state.steps.find((s) => s.id === action.payload.id);
124
+ if (step) {
125
+ Object.assign(step, action.payload.updates);
126
+ }
127
+ },
128
+
129
+ // 设置当前正在执行的步骤ID
130
+ setCurrentStep: (state, action: PayloadAction<string | null>) => {
131
+ state.currentStep = action.payload;
132
+ },
133
+
134
+ // 清除所有步骤信息
135
+ clearSteps: (state) => {
136
+ state.steps = [];
137
+ state.currentStep = null;
138
+ },
139
+
140
+ // 新增:在指定步骤之后插入新步骤
141
+ insertStepsAfter: (
142
+ state,
143
+ action: PayloadAction<{
144
+ afterStepId: string;
145
+ newSteps: Step[];
146
+ }>
147
+ ) => {
148
+ const { afterStepId, newSteps } = action.payload;
149
+ const index = state.steps.findIndex((s) => s.id === afterStepId);
150
+ if (index !== -1) {
151
+ // 在 afterStepId 之后插入新步骤
152
+ state.steps.splice(index + 1, 0, ...newSteps);
153
+ }
154
+ },
155
+
156
+ // 新增:移除指定步骤之后的所有待执行步骤
157
+ removeStepsAfter: (state, action: PayloadAction<string>) => {
158
+ const afterStepId = action.payload;
159
+ const index = state.steps.findIndex((s) => s.id === afterStepId);
160
+ if (index !== -1) {
161
+ // 只保留 afterStepId 及之前的步骤,以及已完成的步骤
162
+ state.steps = state.steps.filter(
163
+ (s, i) => i <= index || s.status === "completed"
164
+ );
165
+ }
166
+ },
167
+
168
+ // 新增:设置执行配置
169
+ setExecutionConfig: (
170
+ state,
171
+ action: PayloadAction<PlanExecutionConfig | null>
172
+ ) => {
173
+ state.executionConfig = action.payload;
174
+ },
175
+
176
+ // 新增:更新执行统计
177
+ updateStats: (
178
+ state,
179
+ action: PayloadAction<Partial<PlanSliceState["stats"]>>
180
+ ) => {
181
+ Object.assign(state.stats, action.payload);
182
+ },
183
+
184
+ // 新增:增加反思计数
185
+ incrementReflectCount: (state) => {
186
+ state.stats.reflectCount += 1;
187
+ },
188
+
189
+ // 新增:增加已执行步骤计数
190
+ incrementStepsExecuted: (state) => {
191
+ state.stats.totalStepsExecuted += 1;
192
+ },
193
+
194
+ // 新增:设置开始时间
195
+ setStartTime: (state, action: PayloadAction<number>) => {
196
+ state.stats.startTime = action.payload;
197
+ },
198
+ },
199
+ selectors: {
200
+ selectPlanState: (state: PlanSliceState) => state.plan,
201
+ selectStepsState: (state: PlanSliceState) => state.steps,
202
+ selectCurrentStepIdState: (state: PlanSliceState) => state.currentStep,
203
+ selectExecutionConfigState: (state: PlanSliceState) =>
204
+ state.executionConfig,
205
+ selectStatsState: (state: PlanSliceState) => state.stats,
206
+ },
207
+ });
208
+
209
+ // --- Exports ---
210
+
211
+ export const {
212
+ setPlan,
213
+ updatePlanProgress,
214
+ clearPlan,
215
+ setSteps,
216
+ updateStep,
217
+ setCurrentStep,
218
+ clearSteps,
219
+ insertStepsAfter,
220
+ removeStepsAfter,
221
+ setExecutionConfig,
222
+ updateStats,
223
+ incrementReflectCount,
224
+ incrementStepsExecuted,
225
+ setStartTime,
226
+ } = planSlice.actions;
227
+
228
+ export default planSlice.reducer;
229
+
230
+ // Selectors
231
+ export const selectPlan = (state: RootState): PlanState | null =>
232
+ state.plan.plan;
233
+
234
+ export const selectSteps = (state: RootState): Step[] => state.plan.steps;
235
+
236
+ export const selectCurrentStepId = (state: RootState): string | null =>
237
+ state.plan.currentStep;
238
+
239
+ export const selectCurrentStepDetails = (state: RootState): Step | null => {
240
+ if (!state.plan.currentStep) return null;
241
+ return (
242
+ state.plan.steps.find((step) => step.id === state.plan.currentStep) || null
243
+ );
244
+ };
245
+
246
+ export const selectExecutionConfig = (
247
+ state: RootState
248
+ ): PlanExecutionConfig | null => state.plan.executionConfig;
249
+
250
+ export const selectPlanStats = (state: RootState): PlanSliceState["stats"] =>
251
+ state.plan.stats;
252
+
253
+ // 新增:获取待执行的步骤
254
+ export const selectPendingSteps = (state: RootState): Step[] =>
255
+ state.plan.steps.filter((s) => s.status === "pending");
256
+
257
+ // 新增:获取已完成的步骤
258
+ export const selectCompletedSteps = (state: RootState): Step[] =>
259
+ state.plan.steps.filter((s) => s.status === "completed");
@@ -0,0 +1,229 @@
1
+ import type { AppDispatch } from "app/store";
2
+ import type { ReferenceItem } from "app/types";
3
+ import { read } from "database/dbSlice";
4
+ import {
5
+ extractRuntimePageCapabilities,
6
+ resolveSkillGraphFromRoots,
7
+ type SkillRuntimePageLike,
8
+ } from "ai/skills/referenceRuntime";
9
+
10
+ export type ResolvedReferenceAssets = {
11
+ references: ReferenceItem[];
12
+ referencedTools: string[];
13
+ recommendedSkillTools: string[];
14
+ recommendedSkillHints: string[];
15
+ skillPromptPatches: string[];
16
+ contentByKey: Map<string, any>;
17
+ };
18
+
19
+ export type ResolvedContentTools = {
20
+ tools: string[];
21
+ recommendedSkillTools: string[];
22
+ recommendedSkillHints: string[];
23
+ skillPromptPatches: string[];
24
+ contentByKey: Map<string, any>;
25
+ };
26
+
27
+ const loadContentsByKeys = async (
28
+ keys: string[],
29
+ dispatch: AppDispatch,
30
+ preloaded?: Map<string, any>
31
+ ): Promise<Map<string, any>> => {
32
+ const contentByKey = new Map<string, SkillRuntimePageLike>();
33
+ const uniqueKeys = Array.from(new Set(keys.filter(Boolean)));
34
+ await Promise.all(
35
+ uniqueKeys.map(async (dbKey) => {
36
+ if (preloaded?.has(dbKey)) {
37
+ const content = preloaded.get(dbKey);
38
+ if (content) contentByKey.set(dbKey, content);
39
+ return;
40
+ }
41
+ try {
42
+ const content = await dispatch(read({ dbKey })).unwrap();
43
+ if (content) contentByKey.set(dbKey, content);
44
+ } catch {
45
+ // ignore missing content
46
+ }
47
+ })
48
+ );
49
+ return contentByKey;
50
+ };
51
+
52
+ export const resolveReferenceAssets = async (
53
+ references: ReferenceItem[] | undefined,
54
+ dispatch: AppDispatch)
55
+ : Promise<ResolvedReferenceAssets> => {
56
+ if (!Array.isArray(references) || references.length === 0) {
57
+ return {
58
+ references: [],
59
+ referencedTools: [],
60
+ recommendedSkillTools: [],
61
+ recommendedSkillHints: [],
62
+ skillPromptPatches: [],
63
+ contentByKey: new Map(),
64
+ };
65
+ }
66
+
67
+ const entries = await Promise.all(
68
+ references.
69
+ filter((ref) => ref?.dbKey).
70
+ map(async (ref) => {
71
+ try {
72
+ const content = await dispatch(read({ dbKey: ref.dbKey })).unwrap();
73
+ return { ref, content };
74
+ } catch {
75
+ return { ref, content: null };
76
+ }
77
+ })
78
+ );
79
+
80
+ const contentByKey = new Map<string, any>();
81
+ const toolSet = new Set<string>();
82
+ const recommendedToolSet = new Set<string>();
83
+ const recommendedSkillHints = new Set<string>();
84
+ const skillPromptPatches = new Set<string>();
85
+ const normalizedReferences: ReferenceItem[] = [];
86
+ const hardSkillKeys = new Set<string>();
87
+ const softSkillKeys = new Set<string>();
88
+
89
+ for (const { ref, content } of entries) {
90
+ if (content) {
91
+ contentByKey.set(ref.dbKey, content);
92
+ }
93
+
94
+ const capabilities = extractRuntimePageCapabilities(content);
95
+ for (const tool of capabilities.directTools) {
96
+ toolSet.add(tool);
97
+ }
98
+ for (const tool of capabilities.hardSkillTools) {
99
+ toolSet.add(tool);
100
+ }
101
+ for (const hint of capabilities.softSkillHints) {
102
+ recommendedSkillHints.add(hint);
103
+ }
104
+ for (const patch of capabilities.promptPatches) {
105
+ skillPromptPatches.add(patch);
106
+ }
107
+ for (const skillKey of capabilities.hardSkillKeys) {
108
+ hardSkillKeys.add(skillKey);
109
+ }
110
+ for (const skillKey of capabilities.softSkillKeys) {
111
+ softSkillKeys.add(skillKey);
112
+ }
113
+
114
+ normalizedReferences.push({
115
+ ...ref,
116
+ type: capabilities.shouldUpgradeReference ? "instruction" : ref.type,
117
+ });
118
+ }
119
+
120
+ const resolvedSkillLinks = await resolveSkillGraphFromRoots({
121
+ roots: [
122
+ ...Array.from(hardSkillKeys).map((identifier) => ({ identifier, mode: "required" as const })),
123
+ ...Array.from(softSkillKeys).map((identifier) => ({ identifier, mode: "recommended" as const })),
124
+ ],
125
+ loadPage: async (identifier) => {
126
+ const loaded = await loadContentsByKeys([identifier], dispatch, contentByKey);
127
+ return loaded.get(identifier) ?? null;
128
+ },
129
+ contentByKey,
130
+ });
131
+ resolvedSkillLinks.contentByKey.forEach((value, key) => contentByKey.set(key, value));
132
+ for (const tool of resolvedSkillLinks.requiredTools) {
133
+ toolSet.add(tool);
134
+ }
135
+ for (const tool of resolvedSkillLinks.recommendedTools) {
136
+ recommendedToolSet.add(tool);
137
+ }
138
+ for (const hint of resolvedSkillLinks.recommendedSkillHints) {
139
+ recommendedSkillHints.add(hint);
140
+ }
141
+ for (const patch of resolvedSkillLinks.skillPromptPatches) {
142
+ skillPromptPatches.add(patch);
143
+ }
144
+
145
+ return {
146
+ references: normalizedReferences,
147
+ referencedTools: Array.from(toolSet),
148
+ recommendedSkillTools: Array.from(recommendedToolSet),
149
+ recommendedSkillHints: Array.from(recommendedSkillHints),
150
+ skillPromptPatches: Array.from(skillPromptPatches),
151
+ contentByKey
152
+ };
153
+ };
154
+
155
+ export const resolveToolsFromKeys = async (
156
+ keys: string[],
157
+ dispatch: AppDispatch,
158
+ preloaded?: Map<string, any>)
159
+ : Promise<ResolvedContentTools> => {
160
+ if (!Array.isArray(keys) || keys.length === 0) {
161
+ return {
162
+ tools: [],
163
+ recommendedSkillTools: [],
164
+ recommendedSkillHints: [],
165
+ skillPromptPatches: [],
166
+ contentByKey: new Map(),
167
+ };
168
+ }
169
+
170
+ const contentByKey = await loadContentsByKeys(keys, dispatch, preloaded);
171
+ const toolSet = new Set<string>();
172
+ const recommendedToolSet = new Set<string>();
173
+ const recommendedSkillHints = new Set<string>();
174
+ const skillPromptPatches = new Set<string>();
175
+ const hardSkillKeys = new Set<string>();
176
+ const softSkillKeys = new Set<string>();
177
+
178
+ for (const content of contentByKey.values()) {
179
+ const capabilities = extractRuntimePageCapabilities(content);
180
+ for (const tool of [...capabilities.directTools, ...capabilities.hardSkillTools]) {
181
+ toolSet.add(tool);
182
+ }
183
+ for (const hint of capabilities.softSkillHints) {
184
+ recommendedSkillHints.add(hint);
185
+ }
186
+ for (const patch of capabilities.promptPatches) {
187
+ skillPromptPatches.add(patch);
188
+ }
189
+ for (const skillKey of capabilities.hardSkillKeys) {
190
+ hardSkillKeys.add(skillKey);
191
+ }
192
+ for (const skillKey of capabilities.softSkillKeys) {
193
+ softSkillKeys.add(skillKey);
194
+ }
195
+ }
196
+
197
+ const resolvedSkillLinks = await resolveSkillGraphFromRoots({
198
+ roots: [
199
+ ...Array.from(hardSkillKeys).map((identifier) => ({ identifier, mode: "required" as const })),
200
+ ...Array.from(softSkillKeys).map((identifier) => ({ identifier, mode: "recommended" as const })),
201
+ ],
202
+ loadPage: async (identifier) => {
203
+ const loaded = await loadContentsByKeys([identifier], dispatch, contentByKey);
204
+ return loaded.get(identifier) ?? null;
205
+ },
206
+ contentByKey,
207
+ });
208
+ resolvedSkillLinks.contentByKey.forEach((value, key) => contentByKey.set(key, value));
209
+ for (const tool of resolvedSkillLinks.requiredTools) {
210
+ toolSet.add(tool);
211
+ }
212
+ for (const tool of resolvedSkillLinks.recommendedTools) {
213
+ recommendedToolSet.add(tool);
214
+ }
215
+ for (const hint of resolvedSkillLinks.recommendedSkillHints) {
216
+ recommendedSkillHints.add(hint);
217
+ }
218
+ for (const patch of resolvedSkillLinks.skillPromptPatches) {
219
+ skillPromptPatches.add(patch);
220
+ }
221
+
222
+ return {
223
+ tools: Array.from(toolSet),
224
+ recommendedSkillTools: Array.from(recommendedToolSet),
225
+ recommendedSkillHints: Array.from(recommendedSkillHints),
226
+ skillPromptPatches: Array.from(skillPromptPatches),
227
+ contentByKey
228
+ };
229
+ };