nolo-cli 0.1.9 → 0.1.11

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 (240) hide show
  1. package/README.md +0 -32
  2. package/agentRuntimeCommands.ts +3 -3
  3. package/commandRegistry.ts +2 -2
  4. package/machineCommands.ts +31 -6
  5. package/package.json +6 -8
  6. package/ai/agent/_executeModel.ts +0 -118
  7. package/ai/agent/agentSlice.ts +0 -525
  8. package/ai/agent/appWorkingMemory.ts +0 -126
  9. package/ai/agent/avatarUtils.ts +0 -24
  10. package/ai/agent/buildEditingContext.ts +0 -373
  11. package/ai/agent/buildSystemPrompt.ts +0 -532
  12. package/ai/agent/cleanAgentMessages.ts +0 -140
  13. package/ai/agent/cliChatClient.ts +0 -119
  14. package/ai/agent/cliExecutor.ts +0 -733
  15. package/ai/agent/cliPrompt.ts +0 -10
  16. package/ai/agent/contextCompiler.ts +0 -107
  17. package/ai/agent/contextLayerContract.ts +0 -44
  18. package/ai/agent/createAgentSchema.ts +0 -234
  19. package/ai/agent/executeToolCall.ts +0 -58
  20. package/ai/agent/fetchAgentContexts.ts +0 -42
  21. package/ai/agent/generatePrompt.ts +0 -3
  22. package/ai/agent/getFullChatContextKeys.ts +0 -168
  23. package/ai/agent/hooks/fetchPublicAgents.ts +0 -133
  24. package/ai/agent/hooks/useAgentConfig.ts +0 -61
  25. package/ai/agent/hooks/useAgentDialog.ts +0 -35
  26. package/ai/agent/hooks/useAgentFormValidation.ts +0 -202
  27. package/ai/agent/hooks/usePublicAgents.ts +0 -473
  28. package/ai/agent/machineRunPermissions.ts +0 -95
  29. package/ai/agent/persistMessageWithFixedId.ts +0 -37
  30. package/ai/agent/planSlice.ts +0 -259
  31. package/ai/agent/referenceUtils.ts +0 -229
  32. package/ai/agent/runAgentBackground.ts +0 -238
  33. package/ai/agent/runAgentClientLoop.ts +0 -138
  34. package/ai/agent/runtimeGuidance.ts +0 -97
  35. package/ai/agent/runtimeServerBase.ts +0 -37
  36. package/ai/agent/server/fetchPublicAgents.ts +0 -128
  37. package/ai/agent/startParallelAgentStreams.ts +0 -424
  38. package/ai/agent/startupProtocol.ts +0 -53
  39. package/ai/agent/streamAgentChatTurn.ts +0 -1278
  40. package/ai/agent/streamAgentChatTurnUtils.ts +0 -738
  41. package/ai/agent/types.ts +0 -71
  42. package/ai/agent/utils/imageOutput.ts +0 -33
  43. package/ai/agent/utils/sortUtils.ts +0 -250
  44. package/ai/agent/web/referencePickerUtils.ts +0 -146
  45. package/ai/ai.locale.ts +0 -1075
  46. package/ai/chat/accumulateToolCallChunks.ts +0 -95
  47. package/ai/chat/fetchUtils.native.ts +0 -276
  48. package/ai/chat/fetchUtils.ts +0 -153
  49. package/ai/chat/parseApiError.ts +0 -64
  50. package/ai/chat/parseMultilineSSE.ts +0 -95
  51. package/ai/chat/sendOpenAICompletionsRequest.native.ts +0 -682
  52. package/ai/chat/sendOpenAICompletionsRequest.ts +0 -703
  53. package/ai/chat/sendOpenAIResponseRequest.ts +0 -491
  54. package/ai/chat/shouldUseServerProxy.ts +0 -18
  55. package/ai/chat/sseClient.native.ts +0 -91
  56. package/ai/chat/sseClient.ts +0 -67
  57. package/ai/chat/streamReader.native.ts +0 -31
  58. package/ai/chat/streamReader.ts +0 -62
  59. package/ai/chat/updateTotalUsage.ts +0 -72
  60. package/ai/context/buildReferenceContext.ts +0 -437
  61. package/ai/context/calculateContextUsage.ts +0 -133
  62. package/ai/context/retention.ts +0 -165
  63. package/ai/context/tokenUtils.ts +0 -78
  64. package/ai/index.ts +0 -1
  65. package/ai/llm/calculateGeminiImageTokens.ts +0 -57
  66. package/ai/llm/deepinfra.ts +0 -28
  67. package/ai/llm/fireworks.ts +0 -50
  68. package/ai/llm/generateRequestBody.ts +0 -165
  69. package/ai/llm/getModelContextWindow.ts +0 -84
  70. package/ai/llm/getNoloKey.ts +0 -31
  71. package/ai/llm/getPricing.ts +0 -199
  72. package/ai/llm/hooks/useModelPricing.ts +0 -75
  73. package/ai/llm/imagePricing.ts +0 -40
  74. package/ai/llm/isResponseAPIModel.ts +0 -13
  75. package/ai/llm/mimo.ts +0 -71
  76. package/ai/llm/mistral.ts +0 -22
  77. package/ai/llm/modelAvatar.ts +0 -427
  78. package/ai/llm/models.ts +0 -45
  79. package/ai/llm/openrouterModels.ts +0 -269
  80. package/ai/llm/providers.ts +0 -306
  81. package/ai/llm/reasoningModels.ts +0 -28
  82. package/ai/llm/types.ts +0 -59
  83. package/ai/llm/usageRequestOptions.ts +0 -59
  84. package/ai/memory/capture.ts +0 -148
  85. package/ai/memory/consolidate.ts +0 -104
  86. package/ai/memory/delete.ts +0 -147
  87. package/ai/memory/overlay.ts +0 -84
  88. package/ai/memory/query.ts +0 -38
  89. package/ai/memory/queryShared.ts +0 -160
  90. package/ai/memory/rank.ts +0 -105
  91. package/ai/memory/recentRelationshipRecap.ts +0 -249
  92. package/ai/memory/remember.ts +0 -167
  93. package/ai/memory/runtime.ts +0 -76
  94. package/ai/memory/store.ts +0 -20
  95. package/ai/memory/storeShared.ts +0 -76
  96. package/ai/memory/types.ts +0 -46
  97. package/ai/memory/understanding.ts +0 -349
  98. package/ai/memory/understandingGreeting.ts +0 -264
  99. package/ai/messages/type.ts +0 -20
  100. package/ai/policy/personalizationDialog.ts +0 -333
  101. package/ai/policy/runtimePolicy.ts +0 -440
  102. package/ai/policy/selfUpdateFields.ts +0 -48
  103. package/ai/policy/types.ts +0 -64
  104. package/ai/skills/referenceRuntime.ts +0 -274
  105. package/ai/skills/skillDiagnostics.ts +0 -251
  106. package/ai/skills/skillDocBuilder.ts +0 -139
  107. package/ai/skills/skillDocProtocol.ts +0 -434
  108. package/ai/skills/skillReferenceSummary.ts +0 -63
  109. package/ai/skills/skillSummaryMarker.ts +0 -26
  110. package/ai/token/calculatePrice.ts +0 -544
  111. package/ai/token/db.ts +0 -98
  112. package/ai/token/externalToolCost.ts +0 -330
  113. package/ai/token/hooks/useRecords.ts +0 -65
  114. package/ai/token/missingUsageEstimate.ts +0 -42
  115. package/ai/token/modelUsageQuery.ts +0 -252
  116. package/ai/token/normalizeUsage.ts +0 -84
  117. package/ai/token/openaiImageGenerationUsage.ts +0 -56
  118. package/ai/token/prepareTokenUsageData.ts +0 -88
  119. package/ai/token/query.ts +0 -88
  120. package/ai/token/queryUserTokens.ts +0 -59
  121. package/ai/token/resolveBillingTarget.ts +0 -52
  122. package/ai/token/saveTokenRecord.ts +0 -53
  123. package/ai/token/serverDialogProjection.ts +0 -78
  124. package/ai/token/serverTokenWriter.ts +0 -143
  125. package/ai/token/stats.ts +0 -21
  126. package/ai/token/tokenThunks.ts +0 -24
  127. package/ai/token/types.ts +0 -93
  128. package/ai/tools/agent/agentTools.ts +0 -176
  129. package/ai/tools/agent/agentUpdateShared.ts +0 -311
  130. package/ai/tools/agent/callAgentTool.ts +0 -139
  131. package/ai/tools/agent/createAgentTool.ts +0 -512
  132. package/ai/tools/agent/createDialogTool.ts +0 -69
  133. package/ai/tools/agent/createSkillAgentTool.ts +0 -62
  134. package/ai/tools/agent/parallelBudget.ts +0 -221
  135. package/ai/tools/agent/presets/appBuilderPreset.ts +0 -145
  136. package/ai/tools/agent/runLlmTool.ts +0 -96
  137. package/ai/tools/agent/runStreamingAgentTool.ts +0 -73
  138. package/ai/tools/agent/skillAgentArgs.ts +0 -106
  139. package/ai/tools/agent/skillAgentPreset.ts +0 -89
  140. package/ai/tools/agent/streamParallelAgentsTool.ts +0 -122
  141. package/ai/tools/agent/updateAgentTool.ts +0 -96
  142. package/ai/tools/agent/updateSelfTool.ts +0 -113
  143. package/ai/tools/amazonProductScraperTool.ts +0 -86
  144. package/ai/tools/apifyActorClient.ts +0 -45
  145. package/ai/tools/appEditGuard.ts +0 -372
  146. package/ai/tools/appReadSnapshot.ts +0 -153
  147. package/ai/tools/appTools.ts +0 -1549
  148. package/ai/tools/applyEditTool.ts +0 -256
  149. package/ai/tools/applyLineEditsTool.ts +0 -312
  150. package/ai/tools/browserTools/click.ts +0 -33
  151. package/ai/tools/browserTools/closeSession.ts +0 -29
  152. package/ai/tools/browserTools/common.ts +0 -27
  153. package/ai/tools/browserTools/openSession.ts +0 -48
  154. package/ai/tools/browserTools/readContent.ts +0 -38
  155. package/ai/tools/browserTools/selectOption.ts +0 -46
  156. package/ai/tools/browserTools/typeText.ts +0 -42
  157. package/ai/tools/category/createCategoryTool.ts +0 -66
  158. package/ai/tools/category/queryContentsByCategoryTool.ts +0 -69
  159. package/ai/tools/category/updateContentCategoryTool.ts +0 -75
  160. package/ai/tools/cfBrowserTools.ts +0 -319
  161. package/ai/tools/cfSpeechToTextTool.ts +0 -49
  162. package/ai/tools/checkEnvTool.ts +0 -65
  163. package/ai/tools/cloudflareCrawlTool.ts +0 -289
  164. package/ai/tools/codeSearchTool.ts +0 -111
  165. package/ai/tools/codeTools.ts +0 -101
  166. package/ai/tools/createDocTool.ts +0 -132
  167. package/ai/tools/createPlanTool.ts +0 -999
  168. package/ai/tools/createSkillDocTool.ts +0 -155
  169. package/ai/tools/createWorkflowTool.ts +0 -154
  170. package/ai/tools/deepseekOcrTool.ts +0 -34
  171. package/ai/tools/delayTool.ts +0 -31
  172. package/ai/tools/deleteSpacesTool.ts +0 -325
  173. package/ai/tools/deleteSpacesToolModel.ts +0 -159
  174. package/ai/tools/devReloadUtils.ts +0 -29
  175. package/ai/tools/dialogMessageSearch.ts +0 -137
  176. package/ai/tools/doctorSkillTool.ts +0 -72
  177. package/ai/tools/ecommerceScraperTool.ts +0 -86
  178. package/ai/tools/emailTools.ts +0 -549
  179. package/ai/tools/evalSkillTool.ts +0 -92
  180. package/ai/tools/exaSearchTool.ts +0 -64
  181. package/ai/tools/execBashTool.ts +0 -379
  182. package/ai/tools/executeSqlTool.ts +0 -192
  183. package/ai/tools/fetchWebpageSupport.ts +0 -309
  184. package/ai/tools/fetchWebpageTool.ts +0 -84
  185. package/ai/tools/geminiImagePreviewTool.ts +0 -361
  186. package/ai/tools/generateDocxTool.ts +0 -215
  187. package/ai/tools/googleSearchScraperTool.ts +0 -106
  188. package/ai/tools/importDataTool.ts +0 -133
  189. package/ai/tools/importSkillTool.ts +0 -162
  190. package/ai/tools/index.ts +0 -1858
  191. package/ai/tools/listFilesTool.ts +0 -82
  192. package/ai/tools/listUserSpacesTool.ts +0 -113
  193. package/ai/tools/modelUsageTools.ts +0 -142
  194. package/ai/tools/olmOcrTool.ts +0 -34
  195. package/ai/tools/openaiImageTool.ts +0 -218
  196. package/ai/tools/paddleOcrTool.ts +0 -34
  197. package/ai/tools/prepareTools.ts +0 -23
  198. package/ai/tools/readDocTool.ts +0 -84
  199. package/ai/tools/readFileTool.ts +0 -211
  200. package/ai/tools/readTool.ts +0 -163
  201. package/ai/tools/readXPostTool.ts +0 -233
  202. package/ai/tools/rememberMemoryTool.ts +0 -84
  203. package/ai/tools/remotionVideoTool.ts +0 -151
  204. package/ai/tools/searchDialogMessagesTool.ts +0 -222
  205. package/ai/tools/searchRepoTool.ts +0 -115
  206. package/ai/tools/searchWorkspaceTool.ts +0 -259
  207. package/ai/tools/skillFollowup.ts +0 -86
  208. package/ai/tools/surfWeatherTool.ts +0 -169
  209. package/ai/tools/table/addTableRowTool.ts +0 -217
  210. package/ai/tools/table/createTableTool.ts +0 -315
  211. package/ai/tools/table/rowTools.ts +0 -366
  212. package/ai/tools/table/schemaTools.ts +0 -244
  213. package/ai/tools/table/shareTableTool.ts +0 -148
  214. package/ai/tools/table/toolShared.ts +0 -129
  215. package/ai/tools/toolApiClient.ts +0 -198
  216. package/ai/tools/toolNameAliases.ts +0 -57
  217. package/ai/tools/toolResultError.ts +0 -42
  218. package/ai/tools/toolRunSlice.ts +0 -303
  219. package/ai/tools/toolSchemaCompatibility.ts +0 -53
  220. package/ai/tools/toolVisibility.ts +0 -4
  221. package/ai/tools/types.ts +0 -20
  222. package/ai/tools/uiAskChoiceTool.ts +0 -104
  223. package/ai/tools/updateContentTitleTool.ts +0 -84
  224. package/ai/tools/updateDocTool.ts +0 -105
  225. package/ai/tools/updateUserPreferenceProfileTool.ts +0 -145
  226. package/ai/tools/whisperTool.ts +0 -77
  227. package/ai/tools/writeFileTool.ts +0 -210
  228. package/ai/tools/youtubeScraperTool.ts +0 -116
  229. package/ai/tools/ziweiChartTool.ts +0 -678
  230. package/ai/types.ts +0 -55
  231. package/ai/workflow/workflowExecutor.ts +0 -323
  232. package/ai/workflow/workflowSlice.ts +0 -73
  233. package/ai/workflow/workflowTypes.ts +0 -106
  234. package/client/compactDialog.ts +0 -222
  235. package/connector-experimental/capabilities.ts +0 -73
  236. package/connector-experimental/codexBinary.ts +0 -41
  237. package/connector-experimental/heartbeatLoop.ts +0 -22
  238. package/connector-experimental/index.ts +0 -5
  239. package/connector-experimental/machineInfo.ts +0 -46
  240. package/connector-experimental/protocol.ts +0 -54
@@ -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
- }