nolo-cli 0.1.10 → 0.1.12

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (232) hide show
  1. package/README.md +0 -32
  2. package/agentRuntimeCommands.ts +1 -1
  3. package/client/compactDialog.ts +2 -5
  4. package/commandRegistry.ts +2 -2
  5. package/machineCommands.ts +28 -3
  6. package/package.json +5 -25
  7. package/ai/agent/_executeModel.ts +0 -118
  8. package/ai/agent/agentSlice.ts +0 -525
  9. package/ai/agent/appWorkingMemory.ts +0 -126
  10. package/ai/agent/avatarUtils.ts +0 -24
  11. package/ai/agent/buildEditingContext.ts +0 -373
  12. package/ai/agent/buildSystemPrompt.ts +0 -532
  13. package/ai/agent/cleanAgentMessages.ts +0 -140
  14. package/ai/agent/cliChatClient.ts +0 -119
  15. package/ai/agent/contextCompiler.ts +0 -107
  16. package/ai/agent/contextLayerContract.ts +0 -44
  17. package/ai/agent/createAgentSchema.ts +0 -234
  18. package/ai/agent/executeToolCall.ts +0 -58
  19. package/ai/agent/fetchAgentContexts.ts +0 -42
  20. package/ai/agent/generatePrompt.ts +0 -3
  21. package/ai/agent/getFullChatContextKeys.ts +0 -168
  22. package/ai/agent/hooks/fetchPublicAgents.ts +0 -133
  23. package/ai/agent/hooks/useAgentConfig.ts +0 -61
  24. package/ai/agent/hooks/useAgentDialog.ts +0 -35
  25. package/ai/agent/hooks/useAgentFormValidation.ts +0 -202
  26. package/ai/agent/hooks/usePublicAgents.ts +0 -473
  27. package/ai/agent/persistMessageWithFixedId.ts +0 -37
  28. package/ai/agent/planSlice.ts +0 -259
  29. package/ai/agent/referenceUtils.ts +0 -229
  30. package/ai/agent/runAgentBackground.ts +0 -238
  31. package/ai/agent/runAgentClientLoop.ts +0 -138
  32. package/ai/agent/runtimeGuidance.ts +0 -97
  33. package/ai/agent/runtimeServerBase.ts +0 -37
  34. package/ai/agent/server/fetchPublicAgents.ts +0 -128
  35. package/ai/agent/startParallelAgentStreams.ts +0 -424
  36. package/ai/agent/startupProtocol.ts +0 -53
  37. package/ai/agent/streamAgentChatTurn.ts +0 -1278
  38. package/ai/agent/streamAgentChatTurnUtils.ts +0 -738
  39. package/ai/agent/types.ts +0 -71
  40. package/ai/agent/utils/imageOutput.ts +0 -33
  41. package/ai/agent/utils/sortUtils.ts +0 -250
  42. package/ai/agent/web/referencePickerUtils.ts +0 -146
  43. package/ai/ai.locale.ts +0 -1079
  44. package/ai/chat/accumulateToolCallChunks.ts +0 -95
  45. package/ai/chat/fetchUtils.native.ts +0 -276
  46. package/ai/chat/fetchUtils.ts +0 -153
  47. package/ai/chat/parseApiError.ts +0 -64
  48. package/ai/chat/parseMultilineSSE.ts +0 -95
  49. package/ai/chat/sendOpenAICompletionsRequest.native.ts +0 -682
  50. package/ai/chat/sendOpenAICompletionsRequest.ts +0 -703
  51. package/ai/chat/sendOpenAIResponseRequest.ts +0 -491
  52. package/ai/chat/shouldUseServerProxy.ts +0 -18
  53. package/ai/chat/sseClient.native.ts +0 -91
  54. package/ai/chat/sseClient.ts +0 -67
  55. package/ai/chat/streamReader.native.ts +0 -31
  56. package/ai/chat/streamReader.ts +0 -62
  57. package/ai/chat/updateTotalUsage.ts +0 -72
  58. package/ai/context/buildReferenceContext.ts +0 -437
  59. package/ai/context/calculateContextUsage.ts +0 -133
  60. package/ai/context/retention.ts +0 -165
  61. package/ai/context/tokenUtils.ts +0 -78
  62. package/ai/index.ts +0 -1
  63. package/ai/llm/calculateGeminiImageTokens.ts +0 -57
  64. package/ai/llm/deepinfra.ts +0 -28
  65. package/ai/llm/fireworks.ts +0 -50
  66. package/ai/llm/generateRequestBody.ts +0 -165
  67. package/ai/llm/getModelContextWindow.ts +0 -84
  68. package/ai/llm/getNoloKey.ts +0 -31
  69. package/ai/llm/getPricing.ts +0 -199
  70. package/ai/llm/hooks/useModelPricing.ts +0 -75
  71. package/ai/llm/imagePricing.ts +0 -40
  72. package/ai/llm/isResponseAPIModel.ts +0 -13
  73. package/ai/llm/mimo.ts +0 -71
  74. package/ai/llm/mistral.ts +0 -22
  75. package/ai/llm/modelAvatar.ts +0 -427
  76. package/ai/llm/models.ts +0 -45
  77. package/ai/llm/openrouterModels.ts +0 -269
  78. package/ai/llm/providers.ts +0 -306
  79. package/ai/llm/reasoningModels.ts +0 -28
  80. package/ai/llm/types.ts +0 -59
  81. package/ai/llm/usageRequestOptions.ts +0 -59
  82. package/ai/memory/capture.ts +0 -148
  83. package/ai/memory/consolidate.ts +0 -104
  84. package/ai/memory/delete.ts +0 -147
  85. package/ai/memory/overlay.ts +0 -84
  86. package/ai/memory/query.ts +0 -38
  87. package/ai/memory/queryShared.ts +0 -160
  88. package/ai/memory/rank.ts +0 -105
  89. package/ai/memory/recentRelationshipRecap.ts +0 -249
  90. package/ai/memory/remember.ts +0 -167
  91. package/ai/memory/runtime.ts +0 -76
  92. package/ai/memory/store.ts +0 -20
  93. package/ai/memory/storeShared.ts +0 -76
  94. package/ai/memory/types.ts +0 -46
  95. package/ai/memory/understanding.ts +0 -349
  96. package/ai/memory/understandingGreeting.ts +0 -264
  97. package/ai/messages/type.ts +0 -20
  98. package/ai/policy/personalizationDialog.ts +0 -333
  99. package/ai/policy/runtimePolicy.ts +0 -440
  100. package/ai/policy/selfUpdateFields.ts +0 -48
  101. package/ai/policy/types.ts +0 -64
  102. package/ai/skills/referenceRuntime.ts +0 -274
  103. package/ai/skills/skillDiagnostics.ts +0 -251
  104. package/ai/skills/skillDocBuilder.ts +0 -139
  105. package/ai/skills/skillDocProtocol.ts +0 -434
  106. package/ai/skills/skillReferenceSummary.ts +0 -63
  107. package/ai/skills/skillSummaryMarker.ts +0 -26
  108. package/ai/token/calculatePrice.ts +0 -544
  109. package/ai/token/db.ts +0 -98
  110. package/ai/token/externalToolCost.ts +0 -330
  111. package/ai/token/hooks/useRecords.ts +0 -65
  112. package/ai/token/missingUsageEstimate.ts +0 -42
  113. package/ai/token/modelUsageQuery.ts +0 -252
  114. package/ai/token/normalizeUsage.ts +0 -84
  115. package/ai/token/openaiImageGenerationUsage.ts +0 -56
  116. package/ai/token/prepareTokenUsageData.ts +0 -88
  117. package/ai/token/query.ts +0 -88
  118. package/ai/token/queryUserTokens.ts +0 -59
  119. package/ai/token/resolveBillingTarget.ts +0 -52
  120. package/ai/token/saveTokenRecord.ts +0 -53
  121. package/ai/token/serverDialogProjection.ts +0 -78
  122. package/ai/token/serverTokenWriter.ts +0 -143
  123. package/ai/token/stats.ts +0 -21
  124. package/ai/token/tokenThunks.ts +0 -24
  125. package/ai/token/types.ts +0 -93
  126. package/ai/tools/agent/agentTools.ts +0 -176
  127. package/ai/tools/agent/agentUpdateShared.ts +0 -311
  128. package/ai/tools/agent/callAgentTool.ts +0 -139
  129. package/ai/tools/agent/createAgentTool.ts +0 -512
  130. package/ai/tools/agent/createDialogTool.ts +0 -69
  131. package/ai/tools/agent/createSkillAgentTool.ts +0 -62
  132. package/ai/tools/agent/parallelBudget.ts +0 -221
  133. package/ai/tools/agent/presets/appBuilderPreset.ts +0 -145
  134. package/ai/tools/agent/runLlmTool.ts +0 -96
  135. package/ai/tools/agent/runStreamingAgentTool.ts +0 -73
  136. package/ai/tools/agent/skillAgentArgs.ts +0 -106
  137. package/ai/tools/agent/skillAgentPreset.ts +0 -89
  138. package/ai/tools/agent/streamParallelAgentsTool.ts +0 -122
  139. package/ai/tools/agent/updateAgentTool.ts +0 -96
  140. package/ai/tools/agent/updateSelfTool.ts +0 -113
  141. package/ai/tools/amazonProductScraperTool.ts +0 -86
  142. package/ai/tools/apifyActorClient.ts +0 -45
  143. package/ai/tools/appEditGuard.ts +0 -372
  144. package/ai/tools/appReadSnapshot.ts +0 -153
  145. package/ai/tools/appTools.ts +0 -1549
  146. package/ai/tools/applyEditTool.ts +0 -256
  147. package/ai/tools/applyLineEditsTool.ts +0 -312
  148. package/ai/tools/browserTools/click.ts +0 -33
  149. package/ai/tools/browserTools/closeSession.ts +0 -29
  150. package/ai/tools/browserTools/common.ts +0 -27
  151. package/ai/tools/browserTools/openSession.ts +0 -48
  152. package/ai/tools/browserTools/readContent.ts +0 -38
  153. package/ai/tools/browserTools/selectOption.ts +0 -46
  154. package/ai/tools/browserTools/typeText.ts +0 -42
  155. package/ai/tools/category/createCategoryTool.ts +0 -66
  156. package/ai/tools/category/queryContentsByCategoryTool.ts +0 -69
  157. package/ai/tools/category/updateContentCategoryTool.ts +0 -75
  158. package/ai/tools/cfBrowserTools.ts +0 -319
  159. package/ai/tools/cfSpeechToTextTool.ts +0 -49
  160. package/ai/tools/checkEnvTool.ts +0 -65
  161. package/ai/tools/cloudflareCrawlTool.ts +0 -289
  162. package/ai/tools/codeSearchTool.ts +0 -111
  163. package/ai/tools/codeTools.ts +0 -101
  164. package/ai/tools/createDocTool.ts +0 -132
  165. package/ai/tools/createPlanTool.ts +0 -999
  166. package/ai/tools/createSkillDocTool.ts +0 -155
  167. package/ai/tools/createWorkflowTool.ts +0 -154
  168. package/ai/tools/deepseekOcrTool.ts +0 -34
  169. package/ai/tools/delayTool.ts +0 -31
  170. package/ai/tools/deleteSpacesTool.ts +0 -325
  171. package/ai/tools/deleteSpacesToolModel.ts +0 -159
  172. package/ai/tools/devReloadUtils.ts +0 -29
  173. package/ai/tools/dialogMessageSearch.ts +0 -137
  174. package/ai/tools/doctorSkillTool.ts +0 -72
  175. package/ai/tools/ecommerceScraperTool.ts +0 -86
  176. package/ai/tools/emailTools.ts +0 -549
  177. package/ai/tools/evalSkillTool.ts +0 -92
  178. package/ai/tools/exaSearchTool.ts +0 -64
  179. package/ai/tools/execBashTool.ts +0 -379
  180. package/ai/tools/executeSqlTool.ts +0 -192
  181. package/ai/tools/fetchWebpageSupport.ts +0 -309
  182. package/ai/tools/fetchWebpageTool.ts +0 -84
  183. package/ai/tools/geminiImagePreviewTool.ts +0 -361
  184. package/ai/tools/generateDocxTool.ts +0 -215
  185. package/ai/tools/googleSearchScraperTool.ts +0 -106
  186. package/ai/tools/importDataTool.ts +0 -133
  187. package/ai/tools/importSkillTool.ts +0 -162
  188. package/ai/tools/index.ts +0 -1858
  189. package/ai/tools/listFilesTool.ts +0 -82
  190. package/ai/tools/listUserSpacesTool.ts +0 -113
  191. package/ai/tools/modelUsageTools.ts +0 -142
  192. package/ai/tools/olmOcrTool.ts +0 -34
  193. package/ai/tools/openaiImageTool.ts +0 -218
  194. package/ai/tools/paddleOcrTool.ts +0 -34
  195. package/ai/tools/prepareTools.ts +0 -23
  196. package/ai/tools/readDocTool.ts +0 -84
  197. package/ai/tools/readFileTool.ts +0 -211
  198. package/ai/tools/readTool.ts +0 -163
  199. package/ai/tools/readXPostTool.ts +0 -233
  200. package/ai/tools/rememberMemoryTool.ts +0 -84
  201. package/ai/tools/remotionVideoTool.ts +0 -151
  202. package/ai/tools/searchDialogMessagesTool.ts +0 -222
  203. package/ai/tools/searchRepoTool.ts +0 -115
  204. package/ai/tools/searchWorkspaceTool.ts +0 -259
  205. package/ai/tools/skillFollowup.ts +0 -86
  206. package/ai/tools/surfWeatherTool.ts +0 -169
  207. package/ai/tools/table/addTableRowTool.ts +0 -217
  208. package/ai/tools/table/createTableTool.ts +0 -315
  209. package/ai/tools/table/rowTools.ts +0 -366
  210. package/ai/tools/table/schemaTools.ts +0 -244
  211. package/ai/tools/table/shareTableTool.ts +0 -148
  212. package/ai/tools/table/toolShared.ts +0 -129
  213. package/ai/tools/toolApiClient.ts +0 -198
  214. package/ai/tools/toolNameAliases.ts +0 -57
  215. package/ai/tools/toolResultError.ts +0 -42
  216. package/ai/tools/toolRunSlice.ts +0 -303
  217. package/ai/tools/toolSchemaCompatibility.ts +0 -53
  218. package/ai/tools/toolVisibility.ts +0 -4
  219. package/ai/tools/types.ts +0 -20
  220. package/ai/tools/uiAskChoiceTool.ts +0 -104
  221. package/ai/tools/updateContentTitleTool.ts +0 -84
  222. package/ai/tools/updateDocTool.ts +0 -105
  223. package/ai/tools/updateUserPreferenceProfileTool.ts +0 -145
  224. package/ai/tools/whisperTool.ts +0 -77
  225. package/ai/tools/writeFileTool.ts +0 -210
  226. package/ai/tools/youtubeScraperTool.ts +0 -116
  227. package/ai/tools/ziweiChartTool.ts +0 -678
  228. package/ai/types.ts +0 -55
  229. package/ai/workflow/workflowExecutor.ts +0 -323
  230. package/ai/workflow/workflowSlice.ts +0 -73
  231. package/ai/workflow/workflowTypes.ts +0 -106
  232. package/connector-experimental/index.ts +0 -5
@@ -1,366 +0,0 @@
1
- import type { RootState } from "app/store";
2
- import {
3
- applyRowFilters,
4
- formatKnownColumns,
5
- normalizeRowValues,
6
- pickRowColumns,
7
- sortRows,
8
- type RowFilters,
9
- } from "render/table/toolValueUtils";
10
- import { buildNewRow, ensureRowsLoaded, loadTableMetaOrThrow, patchRecord, resolveRowOrThrow, resolveTableIdentity, writeRow } from "./toolShared";
11
-
12
- const toPreviewJson = (value: unknown, maxLength = 600): string => {
13
- try {
14
- const json = JSON.stringify(value, null, 2);
15
- if (json.length <= maxLength) return json;
16
- return `${json.slice(0, maxLength)}\n…(已截断)…`;
17
- } catch {
18
- return "[无法序列化为 JSON]";
19
- }
20
- };
21
-
22
- const requireTableIdentity = (args: { tenantId?: string; tableId?: string }, thunkApi: any) => {
23
- const state = thunkApi.getState() as RootState;
24
- const { tenantId, tableId } = resolveTableIdentity(args, state);
25
- if (!tenantId || !tableId) {
26
- throw new Error("需要显式提供 tenantId 和 tableId,或在已打开的表页面中调用。");
27
- }
28
- return { tenantId, tableId };
29
- };
30
-
31
- export const queryTableRowsFunctionSchema = {
32
- name: "queryTableRows",
33
- description: "查询指定表中的行,支持简单字段过滤、排序、分页和返回指定列。",
34
- parameters: {
35
- type: "object",
36
- properties: {
37
- tenantId: { type: "string", description: "租户 ID。聊天场景中必须显式传入。" },
38
- tableId: { type: "string", description: "表 ID。聊天场景中必须显式传入。" },
39
- filters: {
40
- type: "object",
41
- description: "按列名做精确匹配过滤,例如 {\"status\":\"todo\"}。",
42
- },
43
- columns: {
44
- type: "array",
45
- items: { type: "string" },
46
- description: "可选:只返回这些列。",
47
- },
48
- limit: { type: "number", description: "最多返回多少行,默认 20,最大 200。" },
49
- offset: { type: "number", description: "从第几行开始返回,默认 0。" },
50
- sortBy: { type: "string", description: "排序字段,默认 updatedAt。" },
51
- sortOrder: {
52
- type: "string",
53
- enum: ["asc", "desc"],
54
- description: "排序方向,默认 desc。",
55
- },
56
- },
57
- },
58
- };
59
-
60
- export async function queryTableRowsFunc(args: any, thunkApi: any) {
61
- const { tenantId, tableId } = requireTableIdentity(args, thunkApi);
62
- const tableMeta = await loadTableMetaOrThrow(thunkApi, tenantId, tableId);
63
- const rows = await ensureRowsLoaded(thunkApi, tenantId, tableId);
64
- const filters = (args?.filters ?? {}) as RowFilters;
65
- const limit = Math.min(Math.max(Number(args?.limit ?? 20), 1), 200);
66
- const offset = Math.max(Number(args?.offset ?? 0), 0);
67
- const sortBy = typeof args?.sortBy === "string" && args.sortBy.trim() ? args.sortBy.trim() : "updatedAt";
68
- const sortOrder = args?.sortOrder === "asc" ? "asc" : "desc";
69
- const columns = Array.isArray(args?.columns)
70
- ? args.columns.filter((value: unknown): value is string => typeof value === "string" && value.trim().length > 0)
71
- : undefined;
72
-
73
- const filtered = applyRowFilters(rows, filters);
74
- const sorted = sortRows(filtered, sortBy, sortOrder);
75
- const page = sorted.slice(offset, offset + limit);
76
- const items = columns ? page.map((row) => pickRowColumns(row, columns)) : page;
77
-
78
- return {
79
- rawData: {
80
- tenantId,
81
- tableId,
82
- total: filtered.length,
83
- limit,
84
- offset,
85
- items,
86
- },
87
- displayData:
88
- `在表「${tableMeta.displayName || tableId}」中查询到 ${filtered.length} 行,当前返回 ${items.length} 行。\n` +
89
- `字段:${formatKnownColumns(tableMeta)}\n\n` +
90
- `${toPreviewJson(items)}`,
91
- };
92
- }
93
-
94
- export const updateTableRowFunctionSchema = {
95
- name: "updateTableRow",
96
- description: "按 rowId 或 dbKey 更新指定表中的一行数据,只修改 changes 中提供的字段。",
97
- parameters: {
98
- type: "object",
99
- properties: {
100
- tenantId: { type: "string" },
101
- tableId: { type: "string" },
102
- rowId: { type: "string", description: "要更新的行 ID。" },
103
- dbKey: { type: "string", description: "要更新的行 dbKey。rowId 与 dbKey 二选一即可。" },
104
- changes: {
105
- type: "object",
106
- description: "要修改的字段集合,例如 {\"status\":\"done\"}。",
107
- },
108
- },
109
- required: ["changes"],
110
- },
111
- };
112
-
113
- export async function updateTableRowFunc(args: any, thunkApi: any) {
114
- const { tenantId, tableId } = requireTableIdentity(args, thunkApi);
115
- const tableMeta = await loadTableMetaOrThrow(thunkApi, tenantId, tableId);
116
- const targetRow = await resolveRowOrThrow(thunkApi, { tenantId, tableId, rowId: args?.rowId, dbKey: args?.dbKey });
117
- const changes = args?.changes;
118
-
119
- if (!changes || typeof changes !== "object" || Array.isArray(changes)) {
120
- throw new Error("updateTableRow.changes 必须是对象。");
121
- }
122
-
123
- const normalized = normalizeRowValues(tableMeta.columns, changes, { mode: "update" });
124
- if (normalized.errors.length > 0) {
125
- throw new Error(normalized.errors.join("\n"));
126
- }
127
- if (Object.keys(normalized.sanitizedValues).length === 0) {
128
- throw new Error(
129
- `updateTableRow 没有可更新的有效字段。已知字段:${formatKnownColumns(tableMeta)}`
130
- );
131
- }
132
-
133
- const updatedAt = new Date().toISOString();
134
- const updated = await patchRecord(thunkApi, targetRow.dbKey, {
135
- ...normalized.sanitizedValues,
136
- updatedAt,
137
- });
138
-
139
- const ignoredInfo =
140
- normalized.ignoredColumns.length > 0
141
- ? `\n\n注意:以下字段已忽略:${normalized.ignoredColumns.join(", ")}`
142
- : "";
143
-
144
- return {
145
- rawData: updated,
146
- displayData:
147
- `已更新表「${tableMeta.displayName || tableId}」中的行 ${targetRow.rowId}。${ignoredInfo}\n\n` +
148
- `${toPreviewJson(updated)}`,
149
- };
150
- }
151
-
152
- export const deleteTableRowFunctionSchema = {
153
- name: "deleteTableRow",
154
- description: "删除指定表中的一行数据(软删除)。",
155
- parameters: {
156
- type: "object",
157
- properties: {
158
- tenantId: { type: "string" },
159
- tableId: { type: "string" },
160
- rowId: { type: "string" },
161
- dbKey: { type: "string" },
162
- },
163
- },
164
- };
165
-
166
- export async function deleteTableRowFunc(args: any, thunkApi: any) {
167
- const { tenantId, tableId } = requireTableIdentity(args, thunkApi);
168
- const tableMeta = await loadTableMetaOrThrow(thunkApi, tenantId, tableId);
169
- const targetRow = await resolveRowOrThrow(thunkApi, { tenantId, tableId, rowId: args?.rowId, dbKey: args?.dbKey });
170
- const updatedAt = new Date().toISOString();
171
- const deleted = await patchRecord(thunkApi, targetRow.dbKey, {
172
- deletedAt: updatedAt,
173
- updatedAt,
174
- });
175
-
176
- return {
177
- rawData: deleted,
178
- displayData: `已从表「${tableMeta.displayName || tableId}」中删除行 ${targetRow.rowId}。`,
179
- };
180
- }
181
-
182
- export const addTableRowsFunctionSchema = {
183
- name: "addTableRows",
184
- description: "批量向指定表中新增多行数据。",
185
- parameters: {
186
- type: "object",
187
- properties: {
188
- tenantId: { type: "string" },
189
- tableId: { type: "string" },
190
- rows: {
191
- type: "array",
192
- items: { type: "object" },
193
- description: "每一项都是一行数据对象,key 必须是表中的列名。",
194
- },
195
- },
196
- required: ["rows"],
197
- },
198
- };
199
-
200
- export async function addTableRowsFunc(args: any, thunkApi: any) {
201
- const { tenantId, tableId } = requireTableIdentity(args, thunkApi);
202
- const tableMeta = await loadTableMetaOrThrow(thunkApi, tenantId, tableId);
203
- const rows = Array.isArray(args?.rows) ? args.rows : [];
204
- if (rows.length === 0) {
205
- throw new Error("addTableRows.rows 至少需要包含一行。");
206
- }
207
-
208
- const createdRows: any[] = [];
209
- const ignoredColumns = new Set<string>();
210
- for (const row of rows) {
211
- if (!row || typeof row !== "object" || Array.isArray(row)) {
212
- throw new Error("addTableRows.rows 中的每一项都必须是对象。");
213
- }
214
- const normalized = normalizeRowValues(tableMeta.columns, row, { mode: "create" });
215
- if (normalized.errors.length > 0) {
216
- throw new Error(normalized.errors.join("\n"));
217
- }
218
- if (Object.keys(normalized.sanitizedValues).length === 0) {
219
- throw new Error(`存在一行没有任何有效字段。已知字段:${formatKnownColumns(tableMeta)}`);
220
- }
221
- normalized.ignoredColumns.forEach((column) => ignoredColumns.add(column));
222
- const created = buildNewRow(tenantId, tableId, normalized.sanitizedValues);
223
- await writeRow(thunkApi, created);
224
- createdRows.push(created);
225
- }
226
-
227
- const ignoredInfo =
228
- ignoredColumns.size > 0 ? `\n\n注意:以下字段已忽略:${Array.from(ignoredColumns).join(", ")}` : "";
229
-
230
- return {
231
- rawData: {
232
- count: createdRows.length,
233
- items: createdRows,
234
- },
235
- displayData:
236
- `已向表「${tableMeta.displayName || tableId}」批量新增 ${createdRows.length} 行。${ignoredInfo}\n\n` +
237
- `${toPreviewJson(createdRows.slice(0, 10))}`,
238
- };
239
- }
240
-
241
- export const updateTableRowsFunctionSchema = {
242
- name: "updateTableRows",
243
- description: "批量更新多行数据。每项都需要 rowId 或 dbKey,以及 changes。",
244
- parameters: {
245
- type: "object",
246
- properties: {
247
- tenantId: { type: "string" },
248
- tableId: { type: "string" },
249
- updates: {
250
- type: "array",
251
- items: {
252
- type: "object",
253
- properties: {
254
- rowId: { type: "string" },
255
- dbKey: { type: "string" },
256
- changes: { type: "object" },
257
- },
258
- },
259
- },
260
- },
261
- required: ["updates"],
262
- },
263
- };
264
-
265
- export async function updateTableRowsFunc(args: any, thunkApi: any) {
266
- const { tenantId, tableId } = requireTableIdentity(args, thunkApi);
267
- const tableMeta = await loadTableMetaOrThrow(thunkApi, tenantId, tableId);
268
- const updates = Array.isArray(args?.updates) ? args.updates : [];
269
- if (updates.length === 0) {
270
- throw new Error("updateTableRows.updates 至少需要包含一项。");
271
- }
272
-
273
- const results: any[] = [];
274
- const ignoredColumns = new Set<string>();
275
- for (const item of updates) {
276
- if (!item || typeof item !== "object" || Array.isArray(item)) {
277
- throw new Error("updateTableRows.updates 中的每一项都必须是对象。");
278
- }
279
- const targetRow = await resolveRowOrThrow(thunkApi, {
280
- tenantId,
281
- tableId,
282
- rowId: item.rowId,
283
- dbKey: item.dbKey,
284
- });
285
- const normalized = normalizeRowValues(tableMeta.columns, item.changes ?? {}, { mode: "update" });
286
- if (normalized.errors.length > 0) {
287
- throw new Error(normalized.errors.join("\n"));
288
- }
289
- if (Object.keys(normalized.sanitizedValues).length === 0) {
290
- throw new Error(`行 ${targetRow.rowId} 没有可更新的有效字段。`);
291
- }
292
- normalized.ignoredColumns.forEach((column) => ignoredColumns.add(column));
293
- const updated = await patchRecord(thunkApi, targetRow.dbKey, {
294
- ...normalized.sanitizedValues,
295
- updatedAt: new Date().toISOString(),
296
- });
297
- results.push(updated);
298
- }
299
-
300
- const ignoredInfo =
301
- ignoredColumns.size > 0 ? `\n\n注意:以下字段已忽略:${Array.from(ignoredColumns).join(", ")}` : "";
302
-
303
- return {
304
- rawData: {
305
- count: results.length,
306
- items: results,
307
- },
308
- displayData:
309
- `已批量更新表「${tableMeta.displayName || tableId}」中的 ${results.length} 行。${ignoredInfo}`,
310
- };
311
- }
312
-
313
- export const deleteTableRowsFunctionSchema = {
314
- name: "deleteTableRows",
315
- description: "批量删除多行数据(软删除)。",
316
- parameters: {
317
- type: "object",
318
- properties: {
319
- tenantId: { type: "string" },
320
- tableId: { type: "string" },
321
- rowIds: {
322
- type: "array",
323
- items: { type: "string" },
324
- description: "要删除的 rowId 列表。",
325
- },
326
- dbKeys: {
327
- type: "array",
328
- items: { type: "string" },
329
- description: "要删除的 dbKey 列表。",
330
- },
331
- },
332
- },
333
- };
334
-
335
- export async function deleteTableRowsFunc(args: any, thunkApi: any) {
336
- const { tenantId, tableId } = requireTableIdentity(args, thunkApi);
337
- const tableMeta = await loadTableMetaOrThrow(thunkApi, tenantId, tableId);
338
- const rowIds = Array.isArray(args?.rowIds) ? args.rowIds : [];
339
- const dbKeys = Array.isArray(args?.dbKeys) ? args.dbKeys : [];
340
- const targets = [
341
- ...rowIds.map((rowId: string) => ({ rowId })),
342
- ...dbKeys.map((dbKey: string) => ({ dbKey })),
343
- ];
344
-
345
- if (targets.length === 0) {
346
- throw new Error("deleteTableRows 需要至少提供一个 rowId 或 dbKey。");
347
- }
348
-
349
- const results = [];
350
- for (const target of targets) {
351
- const row = await resolveRowOrThrow(thunkApi, { tenantId, tableId, rowId: target.rowId, dbKey: target.dbKey });
352
- const deleted = await patchRecord(thunkApi, row.dbKey, {
353
- deletedAt: new Date().toISOString(),
354
- updatedAt: new Date().toISOString(),
355
- });
356
- results.push(deleted);
357
- }
358
-
359
- return {
360
- rawData: {
361
- count: results.length,
362
- items: results,
363
- },
364
- displayData: `已从表「${tableMeta.displayName || tableId}」中批量删除 ${results.length} 行。`,
365
- };
366
- }
@@ -1,244 +0,0 @@
1
- import { patchRecord, loadTableMetaOrThrow, ensureRowsLoaded, resolveTableIdentity } from "./toolShared";
2
- import type { RootState } from "app/store";
3
- import type { TableColumn } from "render/table/types";
4
- import { ulid } from "database/utils/ulid";
5
-
6
- const requireTableIdentity = (args: { tenantId?: string; tableId?: string }, thunkApi: any) => {
7
- const state = thunkApi.getState() as RootState;
8
- const { tenantId, tableId } = resolveTableIdentity(args, state);
9
- if (!tenantId || !tableId) {
10
- throw new Error("需要显式提供 tenantId 和 tableId,或在已打开的表页面中调用。");
11
- }
12
- return { tenantId, tableId };
13
- };
14
-
15
- const ensureColumnName = (name: unknown): string => {
16
- if (typeof name !== "string" || !name.trim()) {
17
- throw new Error("列名不能为空。");
18
- }
19
- return name.trim();
20
- };
21
-
22
- const normalizeOptions = (options: unknown): string[] | undefined => {
23
- if (!Array.isArray(options)) return undefined;
24
- const normalized = options
25
- .map((value) => (typeof value === "string" ? value.trim() : ""))
26
- .filter(Boolean);
27
- return normalized.length > 0 ? normalized : undefined;
28
- };
29
-
30
- export const addTableColumnFunctionSchema = {
31
- name: "addTableColumn",
32
- description: "向指定表中新增一列。",
33
- parameters: {
34
- type: "object",
35
- properties: {
36
- tenantId: { type: "string" },
37
- tableId: { type: "string" },
38
- name: { type: "string" },
39
- label: { type: "string" },
40
- type: {
41
- type: "string",
42
- enum: ["text", "number", "boolean", "date", "datetime", "select", "multi_select"],
43
- },
44
- description: { type: "string" },
45
- required: { type: "boolean" },
46
- options: { type: "array", items: { type: "string" } },
47
- },
48
- required: ["name"],
49
- },
50
- };
51
-
52
- export async function addTableColumnFunc(args: any, thunkApi: any) {
53
- const { tenantId, tableId } = requireTableIdentity(args, thunkApi);
54
- const tableMeta = await loadTableMetaOrThrow(thunkApi, tenantId, tableId);
55
- const name = ensureColumnName(args?.name);
56
- if (tableMeta.columns.some((column) => column.name === name)) {
57
- throw new Error(`字段 ${name} 已存在。`);
58
- }
59
-
60
- const column: TableColumn = {
61
- id: ulid(),
62
- name,
63
- label: typeof args?.label === "string" && args.label.trim() ? args.label.trim() : name,
64
- type: typeof args?.type === "string" ? args.type : "text",
65
- description: typeof args?.description === "string" && args.description.trim() ? args.description.trim() : undefined,
66
- required: typeof args?.required === "boolean" ? args.required : undefined,
67
- options: normalizeOptions(args?.options),
68
- };
69
-
70
- const updatedAt = new Date().toISOString();
71
- const updated = await patchRecord(thunkApi, tableMeta.dbKey, {
72
- columns: [...tableMeta.columns, column],
73
- updatedAt,
74
- });
75
-
76
- return {
77
- rawData: updated,
78
- displayData: `已向表「${tableMeta.displayName || tableId}」新增字段 ${name}。`,
79
- };
80
- }
81
-
82
- export const deleteTableColumnFunctionSchema = {
83
- name: "deleteTableColumn",
84
- description: "删除指定表中的某一列,并同步从已有行中移除该字段。",
85
- parameters: {
86
- type: "object",
87
- properties: {
88
- tenantId: { type: "string" },
89
- tableId: { type: "string" },
90
- columnName: { type: "string" },
91
- },
92
- required: ["columnName"],
93
- },
94
- };
95
-
96
- export async function deleteTableColumnFunc(args: any, thunkApi: any) {
97
- const { tenantId, tableId } = requireTableIdentity(args, thunkApi);
98
- const tableMeta = await loadTableMetaOrThrow(thunkApi, tenantId, tableId);
99
- const columnName = ensureColumnName(args?.columnName);
100
- if (!tableMeta.columns.some((column) => column.name === columnName)) {
101
- throw new Error(`字段 ${columnName} 不存在。`);
102
- }
103
-
104
- const rows = await ensureRowsLoaded(thunkApi, tenantId, tableId);
105
- const updatedAt = new Date().toISOString();
106
- for (const row of rows) {
107
- if (Object.prototype.hasOwnProperty.call(row, columnName)) {
108
- await patchRecord(thunkApi, row.dbKey, {
109
- [columnName]: null,
110
- updatedAt,
111
- });
112
- }
113
- }
114
-
115
- const updated = await patchRecord(thunkApi, tableMeta.dbKey, {
116
- columns: tableMeta.columns.filter((column) => column.name !== columnName),
117
- updatedAt,
118
- });
119
-
120
- return {
121
- rawData: updated,
122
- displayData: `已从表「${tableMeta.displayName || tableId}」删除字段 ${columnName}。`,
123
- };
124
- }
125
-
126
- export const renameTableColumnFunctionSchema = {
127
- name: "renameTableColumn",
128
- description: "修改表字段的 machine name,并同步迁移所有已有行的数据 key。",
129
- parameters: {
130
- type: "object",
131
- properties: {
132
- tenantId: { type: "string" },
133
- tableId: { type: "string" },
134
- oldName: { type: "string" },
135
- newName: { type: "string" },
136
- },
137
- required: ["oldName", "newName"],
138
- },
139
- };
140
-
141
- export async function renameTableColumnFunc(args: any, thunkApi: any) {
142
- const { tenantId, tableId } = requireTableIdentity(args, thunkApi);
143
- const tableMeta = await loadTableMetaOrThrow(thunkApi, tenantId, tableId);
144
- const oldName = ensureColumnName(args?.oldName);
145
- const newName = ensureColumnName(args?.newName);
146
-
147
- if (!tableMeta.columns.some((column) => column.name === oldName)) {
148
- throw new Error(`字段 ${oldName} 不存在。`);
149
- }
150
- if (tableMeta.columns.some((column) => column.name === newName)) {
151
- throw new Error(`字段 ${newName} 已存在。`);
152
- }
153
-
154
- const rows = await ensureRowsLoaded(thunkApi, tenantId, tableId);
155
- const updatedAt = new Date().toISOString();
156
- for (const row of rows) {
157
- if (Object.prototype.hasOwnProperty.call(row, oldName)) {
158
- await patchRecord(thunkApi, row.dbKey, {
159
- [newName]: row[oldName],
160
- [oldName]: null,
161
- updatedAt,
162
- });
163
- }
164
- }
165
-
166
- const updated = await patchRecord(thunkApi, tableMeta.dbKey, {
167
- columns: tableMeta.columns.map((column) =>
168
- column.name === oldName ? { ...column, name: newName } : column
169
- ),
170
- updatedAt,
171
- });
172
-
173
- return {
174
- rawData: updated,
175
- displayData: `已将表「${tableMeta.displayName || tableId}」中的字段 ${oldName} 重命名为 ${newName}。`,
176
- };
177
- }
178
-
179
- export const renameTableColumnLabelFunctionSchema = {
180
- name: "renameTableColumnLabel",
181
- description: "修改表字段的显示名,不改变行数据里的 key。",
182
- parameters: {
183
- type: "object",
184
- properties: {
185
- tenantId: { type: "string" },
186
- tableId: { type: "string" },
187
- columnName: { type: "string" },
188
- newLabel: { type: "string" },
189
- },
190
- required: ["columnName", "newLabel"],
191
- },
192
- };
193
-
194
- export async function renameTableColumnLabelFunc(args: any, thunkApi: any) {
195
- const { tenantId, tableId } = requireTableIdentity(args, thunkApi);
196
- const tableMeta = await loadTableMetaOrThrow(thunkApi, tenantId, tableId);
197
- const columnName = ensureColumnName(args?.columnName);
198
- const newLabel = ensureColumnName(args?.newLabel);
199
-
200
- if (!tableMeta.columns.some((column) => column.name === columnName)) {
201
- throw new Error(`字段 ${columnName} 不存在。`);
202
- }
203
-
204
- const updated = await patchRecord(thunkApi, tableMeta.dbKey, {
205
- columns: tableMeta.columns.map((column) =>
206
- column.name === columnName ? { ...column, label: newLabel } : column
207
- ),
208
- updatedAt: new Date().toISOString(),
209
- });
210
-
211
- return {
212
- rawData: updated,
213
- displayData: `已将字段 ${columnName} 的显示名更新为 ${newLabel}。`,
214
- };
215
- }
216
-
217
- export const renameTableFunctionSchema = {
218
- name: "renameTable",
219
- description: "更新表的显示名称。",
220
- parameters: {
221
- type: "object",
222
- properties: {
223
- tenantId: { type: "string" },
224
- tableId: { type: "string" },
225
- newName: { type: "string" },
226
- },
227
- required: ["newName"],
228
- },
229
- };
230
-
231
- export async function renameTableFunc(args: any, thunkApi: any) {
232
- const { tenantId, tableId } = requireTableIdentity(args, thunkApi);
233
- const tableMeta = await loadTableMetaOrThrow(thunkApi, tenantId, tableId);
234
- const newName = ensureColumnName(args?.newName);
235
- const updated = await patchRecord(thunkApi, tableMeta.dbKey, {
236
- displayName: newName,
237
- updatedAt: new Date().toISOString(),
238
- });
239
-
240
- return {
241
- rawData: updated,
242
- displayData: `已将表名更新为「${newName}」。`,
243
- };
244
- }