nolo-cli 0.1.13 → 0.1.14

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 (320) hide show
  1. package/README.md +9 -2
  2. package/agent-runtime/hostAdapter.ts +53 -0
  3. package/agent-runtime/index.ts +28 -0
  4. package/agent-runtime/localLoop.ts +62 -0
  5. package/agent-runtime/runtimeDecision.ts +70 -0
  6. package/agent-runtime/types.ts +87 -0
  7. package/agentRuntimeCommands.ts +139 -22
  8. package/agentRuntimeLocal.ts +7 -0
  9. package/ai/agent/_executeModel.ts +118 -0
  10. package/ai/agent/agentSlice.ts +544 -1
  11. package/ai/agent/appWorkingMemory.ts +126 -0
  12. package/ai/agent/avatarUtils.ts +24 -0
  13. package/ai/agent/buildEditingContext.ts +373 -0
  14. package/ai/agent/buildSystemPrompt.ts +532 -0
  15. package/ai/agent/cleanAgentMessages.ts +140 -0
  16. package/ai/agent/cliChatClient.ts +119 -0
  17. package/ai/agent/contextCompiler.ts +107 -0
  18. package/ai/agent/contextLayerContract.ts +44 -0
  19. package/ai/agent/createAgentSchema.ts +234 -0
  20. package/ai/agent/executeToolCall.ts +58 -0
  21. package/ai/agent/fetchAgentContexts.ts +42 -0
  22. package/ai/agent/generatePrompt.ts +3 -0
  23. package/ai/agent/getFullChatContextKeys.ts +168 -0
  24. package/ai/agent/hooks/fetchPublicAgents.ts +133 -0
  25. package/ai/agent/hooks/useAgentConfig.ts +61 -0
  26. package/ai/agent/hooks/useAgentDialog.ts +35 -0
  27. package/ai/agent/hooks/useAgentFormValidation.ts +202 -0
  28. package/ai/agent/hooks/usePublicAgents.ts +473 -0
  29. package/ai/agent/persistMessageWithFixedId.ts +37 -0
  30. package/ai/agent/planSlice.ts +259 -0
  31. package/ai/agent/referenceUtils.ts +229 -0
  32. package/ai/agent/runAgentBackground.ts +238 -0
  33. package/ai/agent/runAgentClientLoop.ts +138 -0
  34. package/ai/agent/runtimeGuidance.ts +97 -0
  35. package/ai/agent/runtimeServerBase.ts +37 -0
  36. package/ai/agent/server/fetchPublicAgents.ts +128 -0
  37. package/ai/agent/startParallelAgentStreams.ts +424 -0
  38. package/ai/agent/startupProtocol.ts +53 -0
  39. package/ai/agent/streamAgentChatTurn.ts +1299 -0
  40. package/ai/agent/streamAgentChatTurnUtils.ts +738 -0
  41. package/ai/agent/types.ts +71 -0
  42. package/ai/agent/utils/imageOutput.ts +39 -0
  43. package/ai/agent/utils/publicImageAgentMode.ts +26 -0
  44. package/ai/agent/utils/sortUtils.ts +250 -0
  45. package/ai/agent/web/referencePickerUtils.ts +146 -0
  46. package/ai/ai.locale.ts +1083 -0
  47. package/ai/chat/accumulateToolCallChunks.ts +95 -0
  48. package/ai/chat/fetchUtils.native.ts +276 -0
  49. package/ai/chat/fetchUtils.ts +153 -0
  50. package/ai/chat/inlineImageUrlsForCustomProvider.ts +117 -0
  51. package/ai/chat/parseApiError.ts +64 -0
  52. package/ai/chat/parseMultilineSSE.ts +95 -0
  53. package/ai/chat/sendOpenAICompletionsRequest.native.ts +682 -0
  54. package/ai/chat/sendOpenAICompletionsRequest.ts +712 -0
  55. package/ai/chat/sendOpenAIResponseRequest.ts +512 -0
  56. package/ai/chat/shouldUseServerProxy.ts +18 -0
  57. package/ai/chat/sseClient.native.ts +91 -0
  58. package/ai/chat/sseClient.ts +67 -0
  59. package/ai/chat/streamReader.native.ts +31 -0
  60. package/ai/chat/streamReader.ts +62 -0
  61. package/ai/chat/updateTotalUsage.ts +72 -0
  62. package/ai/context/buildReferenceContext.ts +437 -0
  63. package/ai/context/calculateContextUsage.ts +133 -0
  64. package/ai/context/retention.ts +165 -0
  65. package/ai/context/tokenUtils.ts +78 -0
  66. package/ai/index.ts +1 -1
  67. package/ai/llm/agentCapabilities.ts +74 -0
  68. package/ai/llm/calculateGeminiImageTokens.ts +57 -0
  69. package/ai/llm/deepinfra.ts +28 -0
  70. package/ai/llm/fireworks.ts +68 -0
  71. package/ai/llm/generateRequestBody.ts +165 -0
  72. package/ai/llm/getModelContextWindow.ts +84 -0
  73. package/ai/llm/getNoloKey.ts +37 -0
  74. package/ai/llm/getPricing.ts +232 -0
  75. package/ai/llm/hooks/useModelPricing.ts +75 -0
  76. package/ai/llm/imagePricing.ts +66 -0
  77. package/ai/llm/isResponseAPIModel.ts +13 -0
  78. package/ai/llm/kimi.ts +18 -0
  79. package/ai/llm/mimo.ts +71 -0
  80. package/ai/llm/mistral.ts +22 -0
  81. package/ai/llm/modelAvatar.ts +427 -0
  82. package/ai/llm/models.ts +45 -0
  83. package/ai/llm/openrouterModels.ts +141 -0
  84. package/ai/llm/providers.ts +307 -0
  85. package/ai/llm/reasoningModels.ts +28 -0
  86. package/ai/llm/types.ts +59 -0
  87. package/ai/llm/usageRequestOptions.ts +59 -0
  88. package/ai/memory/capture.ts +148 -0
  89. package/ai/memory/consolidate.ts +104 -0
  90. package/ai/memory/delete.ts +147 -0
  91. package/ai/memory/overlay.ts +84 -0
  92. package/ai/memory/query.ts +38 -0
  93. package/ai/memory/queryShared.ts +160 -0
  94. package/ai/memory/rank.ts +105 -0
  95. package/ai/memory/recentRelationshipRecap.ts +247 -0
  96. package/ai/memory/remember.ts +167 -0
  97. package/ai/memory/runtime.ts +76 -0
  98. package/ai/memory/store.ts +20 -0
  99. package/ai/memory/storeShared.ts +76 -0
  100. package/ai/memory/types.ts +46 -0
  101. package/ai/memory/understanding.ts +349 -0
  102. package/ai/memory/understandingGreeting.ts +264 -0
  103. package/ai/messages/type.ts +20 -0
  104. package/ai/policy/personalizationDialog.ts +333 -0
  105. package/ai/policy/runtimePolicy.ts +440 -0
  106. package/ai/policy/selfUpdateFields.ts +48 -0
  107. package/ai/policy/types.ts +64 -0
  108. package/ai/skills/referenceRuntime.ts +274 -0
  109. package/ai/skills/skillDiagnostics.ts +251 -0
  110. package/ai/skills/skillDocBuilder.ts +139 -0
  111. package/ai/skills/skillDocProtocol.ts +434 -0
  112. package/ai/skills/skillReferenceSummary.ts +63 -0
  113. package/ai/skills/skillSummaryMarker.ts +26 -0
  114. package/ai/token/calculatePrice.ts +546 -0
  115. package/ai/token/db.ts +98 -0
  116. package/ai/token/externalToolCost.ts +321 -0
  117. package/ai/token/hooks/useRecords.ts +65 -0
  118. package/ai/token/missingUsageEstimate.ts +42 -0
  119. package/ai/token/modelUsageQuery.ts +252 -0
  120. package/ai/token/normalizeUsage.ts +84 -0
  121. package/ai/token/openaiImageGenerationUsage.ts +56 -0
  122. package/ai/token/prepareTokenUsageData.ts +88 -0
  123. package/ai/token/query.ts +88 -0
  124. package/ai/token/queryUserTokens.ts +59 -0
  125. package/ai/token/resolveBillingTarget.ts +52 -0
  126. package/ai/token/saveTokenRecord.ts +53 -0
  127. package/ai/token/serverDialogProjection.ts +78 -0
  128. package/ai/token/serverTokenWriter.ts +143 -0
  129. package/ai/token/stats.ts +21 -0
  130. package/ai/token/tokenThunks.ts +24 -0
  131. package/ai/token/types.ts +93 -0
  132. package/ai/tools/agent/agentTools.ts +176 -0
  133. package/ai/tools/agent/agentUpdateShared.ts +311 -0
  134. package/ai/tools/agent/callAgentTool.ts +139 -0
  135. package/ai/tools/agent/createAgentTool.ts +512 -0
  136. package/ai/tools/agent/createDialogTool.ts +69 -0
  137. package/ai/tools/agent/createSkillAgentTool.ts +62 -0
  138. package/ai/tools/agent/parallelBudget.ts +221 -0
  139. package/ai/tools/agent/presets/appBuilderPreset.ts +147 -0
  140. package/ai/tools/agent/runLlmTool.ts +96 -0
  141. package/ai/tools/agent/runStreamingAgentTool.ts +73 -0
  142. package/ai/tools/agent/skillAgentArgs.ts +106 -0
  143. package/ai/tools/agent/skillAgentPreset.ts +89 -0
  144. package/ai/tools/agent/streamParallelAgentsTool.ts +122 -0
  145. package/ai/tools/agent/updateAgentTool.ts +96 -0
  146. package/ai/tools/agent/updateSelfTool.ts +113 -0
  147. package/ai/tools/amazonProductScraperTool.ts +86 -0
  148. package/ai/tools/apifyActorClient.ts +45 -0
  149. package/ai/tools/appEditGuard.ts +372 -0
  150. package/ai/tools/appReadSnapshot.ts +153 -0
  151. package/ai/tools/appTools.ts +1549 -0
  152. package/ai/tools/applyEditTool.ts +256 -0
  153. package/ai/tools/applyLineEditsTool.ts +312 -0
  154. package/ai/tools/browserTools/click.ts +33 -0
  155. package/ai/tools/browserTools/closeSession.ts +29 -0
  156. package/ai/tools/browserTools/common.ts +27 -0
  157. package/ai/tools/browserTools/openSession.ts +48 -0
  158. package/ai/tools/browserTools/readContent.ts +38 -0
  159. package/ai/tools/browserTools/selectOption.ts +46 -0
  160. package/ai/tools/browserTools/typeText.ts +42 -0
  161. package/ai/tools/category/createCategoryTool.ts +66 -0
  162. package/ai/tools/category/queryContentsByCategoryTool.ts +69 -0
  163. package/ai/tools/category/updateContentCategoryTool.ts +75 -0
  164. package/ai/tools/cfBrowserTools.ts +319 -0
  165. package/ai/tools/cfSpeechToTextTool.ts +49 -0
  166. package/ai/tools/checkEnvTool.ts +65 -0
  167. package/ai/tools/cloudflareCrawlTool.ts +289 -0
  168. package/ai/tools/codeSearchTool.ts +111 -0
  169. package/ai/tools/codeTools.ts +101 -0
  170. package/ai/tools/createDocTool.ts +132 -0
  171. package/ai/tools/createPlanTool.ts +999 -0
  172. package/ai/tools/createSkillDocTool.ts +155 -0
  173. package/ai/tools/createWorkflowTool.ts +154 -0
  174. package/ai/tools/deepseekOcrTool.ts +34 -0
  175. package/ai/tools/delayTool.ts +31 -0
  176. package/ai/tools/deleteSpacesTool.ts +325 -0
  177. package/ai/tools/deleteSpacesToolModel.ts +159 -0
  178. package/ai/tools/devReloadUtils.ts +29 -0
  179. package/ai/tools/dialogMessageSearch.ts +137 -0
  180. package/ai/tools/doctorSkillTool.ts +72 -0
  181. package/ai/tools/ecommerceScraperTool.ts +86 -0
  182. package/ai/tools/emailTools.ts +549 -0
  183. package/ai/tools/evalSkillTool.ts +92 -0
  184. package/ai/tools/exaSearchTool.ts +64 -0
  185. package/ai/tools/execBashTool.ts +379 -0
  186. package/ai/tools/executeSqlTool.ts +192 -0
  187. package/ai/tools/fetchWebpageSupport.ts +309 -0
  188. package/ai/tools/fetchWebpageTool.ts +84 -0
  189. package/ai/tools/geminiImagePreviewTool.ts +361 -0
  190. package/ai/tools/generateDocxTool.ts +215 -0
  191. package/ai/tools/googleSearchScraperTool.ts +106 -0
  192. package/ai/tools/importDataTool.ts +133 -0
  193. package/ai/tools/importSkillTool.ts +162 -0
  194. package/ai/tools/index.ts +1927 -0
  195. package/ai/tools/listFilesTool.ts +82 -0
  196. package/ai/tools/listUserSpacesTool.ts +113 -0
  197. package/ai/tools/modelUsageTools.ts +199 -0
  198. package/ai/tools/olmOcrTool.ts +34 -0
  199. package/ai/tools/openaiImageTool.ts +267 -0
  200. package/ai/tools/prepareTools.ts +23 -0
  201. package/ai/tools/readDocTool.ts +84 -0
  202. package/ai/tools/readFileTool.ts +211 -0
  203. package/ai/tools/readTool.ts +163 -0
  204. package/ai/tools/readXPostTool.ts +233 -0
  205. package/ai/tools/rememberMemoryTool.ts +84 -0
  206. package/ai/tools/remotionVideoTool.ts +151 -0
  207. package/ai/tools/searchDialogMessagesTool.ts +222 -0
  208. package/ai/tools/searchRepoTool.ts +115 -0
  209. package/ai/tools/searchWorkspaceTool.ts +259 -0
  210. package/ai/tools/skillFollowup.ts +86 -0
  211. package/ai/tools/surfWeatherTool.ts +169 -0
  212. package/ai/tools/table/addTableRowTool.ts +217 -0
  213. package/ai/tools/table/createTableTool.ts +315 -0
  214. package/ai/tools/table/rowTools.ts +366 -0
  215. package/ai/tools/table/schemaTools.ts +244 -0
  216. package/ai/tools/table/shareTableTool.ts +148 -0
  217. package/ai/tools/table/toolShared.ts +129 -0
  218. package/ai/tools/toolApiClient.ts +198 -0
  219. package/ai/tools/toolNameAliases.ts +57 -0
  220. package/ai/tools/toolResultError.ts +42 -0
  221. package/ai/tools/toolRunSlice.ts +303 -0
  222. package/ai/tools/toolSchemaCompatibility.ts +53 -0
  223. package/ai/tools/toolVisibility.ts +4 -0
  224. package/ai/tools/types.ts +20 -0
  225. package/ai/tools/uiAskChoiceTool.ts +104 -0
  226. package/ai/tools/updateContentTitleTool.ts +84 -0
  227. package/ai/tools/updateDocTool.ts +105 -0
  228. package/ai/tools/updateUserPreferenceProfileTool.ts +145 -0
  229. package/ai/tools/whisperTool.ts +77 -0
  230. package/ai/tools/writeFileTool.ts +210 -0
  231. package/ai/tools/youtubeScraperTool.ts +116 -0
  232. package/ai/tools/ziweiChartTool.ts +678 -0
  233. package/ai/types.ts +55 -0
  234. package/ai/workflow/workflowExecutor.ts +323 -0
  235. package/ai/workflow/workflowSlice.ts +73 -0
  236. package/ai/workflow/workflowTypes.ts +106 -0
  237. package/client/agentRun.test.ts +240 -0
  238. package/client/agentRun.ts +182 -19
  239. package/client/compactDialog.test.ts +238 -0
  240. package/client/localRuntimeAdapter.test.ts +135 -0
  241. package/client/localRuntimeAdapter.ts +244 -0
  242. package/client/profileConfig.test.ts +40 -0
  243. package/client/streamingOutput.test.ts +22 -0
  244. package/client/streamingOutput.ts +38 -0
  245. package/commandRegistry.ts +9 -2
  246. package/connector-experimental/index.ts +5 -0
  247. package/database/actions/cacheMergedUserData.ts +64 -0
  248. package/database/actions/common.ts +242 -0
  249. package/database/actions/deleteFile.ts +40 -0
  250. package/database/actions/fetchUserData.ts +16 -0
  251. package/database/actions/fileContent.ts +125 -0
  252. package/database/actions/patch.ts +155 -0
  253. package/database/actions/read.ts +337 -0
  254. package/database/actions/readAndWait.ts +224 -0
  255. package/database/actions/readRequestManager.ts +120 -0
  256. package/database/actions/remove.ts +94 -0
  257. package/database/actions/replication.ts +366 -0
  258. package/database/actions/upload.ts +174 -0
  259. package/database/actions/upsert.ts +56 -0
  260. package/database/actions/write.ts +126 -0
  261. package/database/client/db.native.ts +73 -0
  262. package/database/client/db.ts +51 -0
  263. package/database/client/fetchUserData.ts +61 -0
  264. package/database/client/handleError.ts +19 -0
  265. package/database/client/queryRequest.ts +21 -0
  266. package/database/config.ts +21 -0
  267. package/database/dbActionThunks.ts +1 -0
  268. package/database/dbSlice.ts +149 -0
  269. package/database/email.ts +42 -0
  270. package/database/fileRing.ts +51 -0
  271. package/database/fileSharding.ts +70 -0
  272. package/database/fileStorage.native.ts +92 -0
  273. package/database/fileStorage.ts +232 -0
  274. package/database/fileUrl.ts +34 -0
  275. package/database/hooks/useUserData.ts +489 -0
  276. package/database/index.ts +1 -0
  277. package/database/keys.ts +765 -0
  278. package/database/queryPrefixes.ts +14 -0
  279. package/database/requests.ts +443 -0
  280. package/database/runtimeServerContext.ts +35 -0
  281. package/database/server/MemoryDB.ts +76 -0
  282. package/database/server/actorAccess.ts +76 -0
  283. package/database/server/agentDelegation.ts +124 -0
  284. package/database/server/coreDataOwnership.ts +13 -0
  285. package/database/server/coreDataProxy.ts +76 -0
  286. package/database/server/cybotReadonly.ts +18 -0
  287. package/database/server/dataHandlers.ts +111 -0
  288. package/database/server/db.ts +118 -0
  289. package/database/server/dbPath.ts +20 -0
  290. package/database/server/delete.ts +499 -0
  291. package/database/server/emailRepository.ts +1480 -0
  292. package/database/server/ensureDbOpen.ts +12 -0
  293. package/database/server/fileRead.ts +337 -0
  294. package/database/server/fileService.ts +436 -0
  295. package/database/server/handleTransaction.ts +86 -0
  296. package/database/server/patch.ts +282 -0
  297. package/database/server/query.ts +138 -0
  298. package/database/server/read.ts +325 -0
  299. package/database/server/resourceAccess.ts +211 -0
  300. package/database/server/routes.ts +110 -0
  301. package/database/server/spaceMemberAuthority.ts +67 -0
  302. package/database/server/upload.ts +159 -0
  303. package/database/server/write.ts +494 -0
  304. package/database/server/writeAuthority.ts +133 -0
  305. package/database/sqliteDb.ts +46 -0
  306. package/database/table/deleteTable.ts +120 -0
  307. package/database/tenantPlacement.ts +57 -0
  308. package/database/tombstones.ts +52 -0
  309. package/database/userDataLoadDecision.ts +17 -0
  310. package/database/userDataMerge.ts +95 -0
  311. package/database/userPreferenceRegister.ts +108 -0
  312. package/database/utils/dbPath.ts +47 -0
  313. package/database/utils/ulid.native.ts +6 -0
  314. package/database/utils/ulid.ts +1 -0
  315. package/index.ts +25 -15
  316. package/localRuntimeDb.ts +28 -0
  317. package/package.json +16 -4
  318. package/runtimeModeArgs.ts +33 -0
  319. package/tui/readlineWorkspace.ts +1 -0
  320. package/tui/session.ts +22 -0
@@ -0,0 +1,14 @@
1
+ // 用户内容查询不能简单假设所有 key 都是 `${type}-${userId}-...`。
2
+ // 例如 table 的真实 meta key 是 `meta-${userId}-${tableId}`,所以 query 层必须维护 type -> key prefix 的映射。
3
+ const TYPE_PREFIX_ALIASES: Record<string, string[]> = {
4
+ table: ["meta"],
5
+ };
6
+
7
+ export function getUserDataPrefixes(type: string, userId: string): string[] {
8
+ const normalizedType = String(type || "").trim();
9
+ const normalizedUserId = String(userId || "").trim();
10
+ if (!normalizedType || !normalizedUserId) return [];
11
+
12
+ const aliases = TYPE_PREFIX_ALIASES[normalizedType] ?? [normalizedType];
13
+ return aliases.map((prefix) => `${prefix}-${normalizedUserId}`);
14
+ }
@@ -0,0 +1,443 @@
1
+ // database/requests.ts
2
+ import { API_ENDPOINTS } from "./config";
3
+
4
+
5
+ export const TIMEOUT = 5000;
6
+
7
+ type RequestFailureLogLevel = "error" | "warn" | "info" | "silent";
8
+
9
+ const logRequestFailure = (
10
+ level: RequestFailureLogLevel,
11
+ message: string
12
+ ) => {
13
+ if (level === "silent") return;
14
+ if (level === "warn") {
15
+ console.warn(message);
16
+ return;
17
+ }
18
+ if (level === "info") {
19
+ console.info(message);
20
+ return;
21
+ }
22
+ console.error(message);
23
+ };
24
+
25
+ /**
26
+ * 通用的Nolo服务器请求函数
27
+ * @param server 服务器地址
28
+ * @param config 请求配置 (url, method, body)
29
+ * @param state Redux state (用于获取token)
30
+ * @param signal AbortSignal 用于取消请求
31
+ * @returns Fetch Response Promise
32
+ */
33
+ export const noloRequest = async (
34
+ server: string,
35
+ config: {
36
+ url: string;
37
+ method?: string;
38
+ body?: string | FormData;
39
+ headers?: HeadersInit;
40
+ keepalive?: boolean;
41
+ },
42
+ state: any,
43
+ signal?: AbortSignal
44
+ ): Promise<Response> => {
45
+ const headers: HeadersInit = config.headers || {
46
+ "Content-Type": "application/json",
47
+ };
48
+ // 从 state 中安全地获取 token
49
+ const token = state?.auth?.currentToken;
50
+ if (token) {
51
+ headers["Authorization"] = `Bearer ${token}`;
52
+ }
53
+
54
+ return fetch(server + config.url, {
55
+ method: config.method || "GET",
56
+ headers,
57
+ body: config.body,
58
+ signal, // 传递 AbortSignal
59
+ ...(config.keepalive ? { keepalive: true } : {}),
60
+ });
61
+ };
62
+
63
+ /**
64
+ * 向单个服务器发送 PATCH 请求 (用于更新部分数据)
65
+ * @param server 服务器地址
66
+ * @param dbKey 数据键
67
+ * @param updates 要更新的数据对象
68
+ * @param state Redux state
69
+ * @param signal AbortSignal
70
+ * @returns Promise<boolean> 请求是否成功 (response.ok)
71
+ */
72
+ export const noloPatchRequest = async (
73
+ server: string,
74
+ dbKey: string,
75
+ updates: any,
76
+ state: any,
77
+ signal?: AbortSignal,
78
+ options?: { failureLogLevel?: RequestFailureLogLevel }
79
+ ): Promise<boolean> => {
80
+ const failureLogLevel = options?.failureLogLevel ?? "error";
81
+ try {
82
+ const response = await noloRequest(
83
+ server,
84
+ {
85
+ url: `${API_ENDPOINTS.DATABASE}/patch/${dbKey}`,
86
+ method: "PATCH",
87
+ body: JSON.stringify(updates),
88
+ },
89
+ state,
90
+ signal
91
+ );
92
+ if (!response.ok) {
93
+ logRequestFailure(
94
+ failureLogLevel,
95
+ `PATCH request failed for ${dbKey} on ${server}: HTTP ${response.status}`
96
+ );
97
+ }
98
+ return response.ok;
99
+ } catch (error: any) {
100
+ if (error.name !== "AbortError") {
101
+ logRequestFailure(
102
+ failureLogLevel,
103
+ `PATCH request failed for ${dbKey} on ${server}: ${error.message || "Unknown error"}`
104
+ );
105
+ }
106
+ // 对于 AbortError 或其他网络错误,返回 false
107
+ return false;
108
+ }
109
+ };
110
+
111
+ /**
112
+ * 向单个服务器发送 POST 请求 (用于写入完整数据)
113
+ * @param server 服务器地址
114
+ * @param writeConfig 写入配置 { data, customKey, userId }
115
+ * @param state Redux state
116
+ * @param signal AbortSignal
117
+ * @returns Promise<boolean> 请求是否成功 (response.ok)
118
+ */
119
+ export const noloWriteRequest = async (
120
+ server: string,
121
+ writeConfig: { data: any; customKey: string; userId?: string; indexKeys?: string[] },
122
+ state: any,
123
+ signal?: AbortSignal,
124
+ options?: { failureLogLevel?: RequestFailureLogLevel }
125
+ ): Promise<boolean> => {
126
+ const { data, customKey, userId, indexKeys } = writeConfig;
127
+ const failureLogLevel = options?.failureLogLevel ?? "error";
128
+ try {
129
+ const response = await noloRequest(
130
+ server,
131
+ {
132
+ url: `${API_ENDPOINTS.DATABASE}/write/`,
133
+ method: "POST",
134
+ body: JSON.stringify({ data, customKey, userId, indexKeys }),
135
+ },
136
+ state,
137
+ signal
138
+ );
139
+ if (!response.ok) {
140
+ logRequestFailure(
141
+ failureLogLevel,
142
+ `Write request failed for ${customKey} on ${server}: HTTP ${response.status}`
143
+ );
144
+ }
145
+ return response.ok;
146
+ } catch (error: any) {
147
+ if (error.name !== "AbortError") {
148
+ logRequestFailure(
149
+ failureLogLevel,
150
+ `Write request failed for ${customKey} on ${server}: ${error.message || "Unknown error"}`
151
+ );
152
+ }
153
+ // 对于 AbortError 或其他网络错误,返回 false
154
+ return false;
155
+ }
156
+ };
157
+
158
+ /**
159
+ * 向单个服务器发送 POST 请求 (用于文件上传)
160
+ * @param server 服务器地址
161
+ * @param uploadConfig 上传配置 { file, metadata, customKey, userId }
162
+ * @param state Redux state
163
+ * @param signal AbortSignal
164
+ * @returns Promise<boolean> 请求是否成功 (response.ok)
165
+ */
166
+ export const noloUploadRequest = async (
167
+ server: string,
168
+ uploadConfig: {
169
+ file: File;
170
+ metadata: any;
171
+ customKey: string;
172
+ userId?: string;
173
+ },
174
+ state: any,
175
+ signal?: AbortSignal
176
+ ): Promise<boolean> => {
177
+ const { file, metadata, customKey, userId } = uploadConfig;
178
+ try {
179
+ const isReactNative =
180
+ typeof navigator !== "undefined" && (navigator as any).product === "ReactNative";
181
+ const isRNFile = (f: any) => f && typeof f.uri === 'string' && typeof f.name === 'string' && typeof f.type === 'string';
182
+ const normalizeBlobUtilPath = (uri: string): string =>
183
+ uri.startsWith("file://") ? uri.slice("file://".length) : uri;
184
+
185
+ if (isReactNative && isRNFile(file)) {
186
+ const ReactNativeBlobUtil = (await import("react-native-blob-util")).default;
187
+ const wrappedPath = ReactNativeBlobUtil.wrap(normalizeBlobUtilPath((file as any).uri));
188
+ const token = state?.auth?.currentToken;
189
+ const headers: Record<string, string> = {
190
+ "Content-Type": "multipart/form-data",
191
+ };
192
+ if (token) {
193
+ headers.Authorization = `Bearer ${token}`;
194
+ }
195
+
196
+ const response = await ReactNativeBlobUtil.fetch(
197
+ "POST",
198
+ server + `${API_ENDPOINTS.DATABASE}/upload`,
199
+ headers,
200
+ [
201
+ {
202
+ name: "file",
203
+ filename: (file as any).name,
204
+ type: (file as any).type,
205
+ data: wrappedPath,
206
+ },
207
+ { name: "metadata", data: JSON.stringify(metadata) },
208
+ { name: "customKey", data: customKey },
209
+ ...(userId ? [{ name: "userId", data: userId }] : []),
210
+ ]
211
+ );
212
+ const status = response.info().status;
213
+ const ok = status >= 200 && status < 300;
214
+
215
+ if (!ok) {
216
+ console.error(
217
+ `Upload request failed for ${customKey} on ${server}: HTTP ${status}`
218
+ );
219
+ }
220
+
221
+ return ok;
222
+ }
223
+
224
+ // 创建 FormData 对象,用于 multipart/form-data 请求
225
+ const formData = new FormData();
226
+
227
+ if (isRNFile(file)) {
228
+ // @ts-ignore - RN 的 FormData append 允许传对象,但 TS 定义可能不匹配
229
+ formData.append("file", {
230
+ uri: (file as any).uri,
231
+ type: (file as any).type,
232
+ name: (file as any).name,
233
+ });
234
+ } else {
235
+ // Web 环境:直接 append File 对象
236
+ formData.append("file", file);
237
+ }
238
+
239
+ formData.append("metadata", JSON.stringify(metadata)); // 添加文件元数据
240
+ formData.append("customKey", customKey); // 添加自定义键
241
+ if (userId) {
242
+ formData.append("userId", userId); // 添加用户ID(如果有)
243
+ }
244
+
245
+ const response = await noloRequest(
246
+ server,
247
+ {
248
+ url: `${API_ENDPOINTS.DATABASE}/upload`,
249
+ method: "POST",
250
+ body: formData as any, // Cast to any to avoid TS mismatch with Bun/DOM FormData
251
+ headers: {}, // 不设置 Content-Type,让浏览器自动处理 multipart/form-data
252
+ },
253
+ state,
254
+ signal
255
+ );
256
+ if (!response.ok) {
257
+ console.error(
258
+ `Upload request failed for ${customKey} on ${server}: HTTP ${response.status}`
259
+ );
260
+ }
261
+ return response.ok;
262
+ } catch (error: any) {
263
+ if (error.name !== "AbortError") {
264
+ console.error(
265
+ `Upload request failed for ${customKey} on ${server}: ${error.message || "Unknown error"}`
266
+ );
267
+ }
268
+ // 对于 AbortError 或其他网络错误,返回 false
269
+ return false;
270
+ }
271
+ };
272
+
273
+ /**
274
+ * 向单个服务器发送 GET 请求 (用于读取文件内容或元数据)
275
+ * @param server 服务器地址
276
+ * @param fileId 文件ID或自定义键
277
+ * @param options 可选参数 { type: 'metadata' | 'content' }
278
+ * @param state Redux state
279
+ * @param signal AbortSignal
280
+ * @returns Promise<{ success: boolean, data?: any }> 请求是否成功以及返回的数据
281
+ */
282
+ export const noloReadFileRequest = async (
283
+ server: string,
284
+ fileId: string,
285
+ options: {
286
+ type?: "metadata" | "content"; // metadata: 只获取元数据, content: 获取文件内容
287
+ } = { type: "metadata" },
288
+ state: any,
289
+ signal?: AbortSignal
290
+ ): Promise<{ success: boolean; data?: any }> => {
291
+ const { type = "metadata" } = options;
292
+
293
+ try {
294
+ // 根据类型构建 URL
295
+ const url =
296
+ type === "content"
297
+ ? `${API_ENDPOINTS.DATABASE}/file/content/${fileId}`
298
+ : `${API_ENDPOINTS.DATABASE}/file/metadata/${fileId}`;
299
+
300
+ const response = await noloRequest(
301
+ server,
302
+ {
303
+ url,
304
+ method: "GET",
305
+ },
306
+ state,
307
+ signal
308
+ );
309
+
310
+ if (!response.ok) {
311
+ console.error(
312
+ `Read file request failed for ${fileId} on ${server}: HTTP ${response.status}`
313
+ );
314
+ return { success: false };
315
+ }
316
+
317
+ // 根据类型处理响应数据
318
+ let data;
319
+ if (type === "content") {
320
+ // 文件内容可能较大,建议以流式或 Blob 形式处理
321
+ data = await response.blob(); // 以 Blob 形式返回文件内容
322
+ } else {
323
+ data = await response.json(); // 元数据以 JSON 形式返回
324
+ }
325
+
326
+ return { success: true, data };
327
+ } catch (error: any) {
328
+ if (error.name !== "AbortError") {
329
+ console.error(
330
+ `Read file request failed for ${fileId} on ${server}: ${error.message || "Unknown error"}`
331
+ );
332
+ }
333
+ return { success: false };
334
+ }
335
+ };
336
+
337
+ /**
338
+ * 通用的服务器同步函数,带有超时和错误处理
339
+ * @param servers 服务器地址列表
340
+ * @param requestFn 实际执行请求的函数 (应返回 Promise<boolean>)
341
+ * @param errorMessage 失败时的错误消息前缀
342
+ * @param requestArgs 传递给 requestFn 的额外参数 (除了 server 和 signal)
343
+ */
344
+ export const syncWithServers = <TArgs extends any[]>(
345
+ servers: string[],
346
+ requestFn: (
347
+ server: string,
348
+ ...args: [...TArgs, AbortSignal?]
349
+ ) => Promise<boolean>,
350
+ errorMessage: string,
351
+ ...requestArgs: TArgs
352
+ ): void => {
353
+ servers.forEach((server) => {
354
+ const abortController = new AbortController();
355
+ const timeoutId = setTimeout(() => {
356
+ // console.warn(`Request to ${server} timed out after ${TIMEOUT}ms`); // 可选:超时警告
357
+ abortController.abort(); // 超时时中止请求
358
+ }, TIMEOUT);
359
+
360
+ // 执行请求函数
361
+ requestFn(server, ...requestArgs, abortController.signal)
362
+ .then((success) => {
363
+ clearTimeout(timeoutId); // 清除超时定时器
364
+ if (!success) {
365
+ // 只在请求明确失败时提示(非超时)
366
+ // 注意:noloPatchRequest/noloWriteRequest 内部已打印详细错误
367
+ // 此处 toast 可考虑移除或改为更通用的后台同步失败提示
368
+ // toast.error(`${errorMessage} ${server}`);
369
+ console.warn(`${errorMessage} ${server}`); // 使用 console.warn 代替 toast
370
+ }
371
+ })
372
+ .catch((error) => {
373
+ clearTimeout(timeoutId); // 清除超时定时器
374
+ // AbortError 通常由超时引起,已在 requestFn 中处理或此处忽略
375
+ if (error.name !== "AbortError") {
376
+ console.error(
377
+ `Unexpected error during sync with ${server}: ${error.message || "Unknown error"}`
378
+ );
379
+ // toast.error(`Sync failed with ${server}`); // 可选的通用失败提示
380
+ }
381
+ });
382
+ });
383
+ };
384
+
385
+ /**
386
+ * 向单个服务器发送 DELETE 请求
387
+ * @param server 服务器地址
388
+ * @param dbKey 数据键
389
+ * @param options 可选参数 { type: 'messages' | 'single' }
390
+ * @param state Redux state
391
+ * @param signal AbortSignal
392
+ * @returns Promise<boolean> 请求是否成功
393
+ */
394
+ export const noloDeleteRequest = async (
395
+ server: string,
396
+ dbKey: string,
397
+ options: {
398
+ // 原来: type?: "messages" | "single";
399
+ // 现在增加 "table",给删整张表用
400
+ type?: "messages" | "single" | "table";
401
+ },
402
+ state: any,
403
+ signal?: AbortSignal
404
+ ): Promise<boolean> => {
405
+ const { type = "single" } = options; // 默认为 'single'
406
+
407
+ try {
408
+ // 根据类型构建 URL
409
+ const url =
410
+ type === "messages"
411
+ ? `${API_ENDPOINTS.DATABASE}/delete/${dbKey}?type=messages`
412
+ : type === "table"
413
+ ? `${API_ENDPOINTS.DATABASE}/delete/${dbKey}?type=table`
414
+ : `${API_ENDPOINTS.DATABASE}/delete/${dbKey}`;
415
+
416
+ const response = await noloRequest(
417
+ server,
418
+ {
419
+ url,
420
+ method: "DELETE",
421
+ keepalive: true,
422
+ },
423
+ state,
424
+ signal
425
+ );
426
+
427
+ if (!response.ok) {
428
+ console.error(
429
+ `DELETE request failed for ${dbKey} on ${server}: HTTP ${response.status}`
430
+ );
431
+ return false;
432
+ }
433
+
434
+ return true; // 请求成功
435
+ } catch (error: any) {
436
+ if (error.name !== "AbortError") {
437
+ console.error(
438
+ `DELETE request failed for ${dbKey} on ${server}: ${error.message || "Unknown error"}`
439
+ );
440
+ }
441
+ return false;
442
+ }
443
+ };
@@ -0,0 +1,35 @@
1
+ import type { RootState } from "app/store";
2
+ import { selectRuntimeSnapshot } from "app/stateViews/runtime";
3
+ import { getAllServers } from "database/actions/common";
4
+
5
+ export type RuntimeServerContext = {
6
+ currentToken?: string;
7
+ currentUserId?: string;
8
+ currentServer?: string;
9
+ syncServers: string[];
10
+ remoteServers: string[];
11
+ };
12
+
13
+ export const getRuntimeServerContext = (
14
+ state: RootState,
15
+ preferredServerOrigin?: string | null
16
+ ): RuntimeServerContext => {
17
+ const {
18
+ currentToken,
19
+ currentUserId,
20
+ currentServer,
21
+ syncServers = [],
22
+ } = selectRuntimeSnapshot(state);
23
+
24
+ return {
25
+ currentToken,
26
+ currentUserId,
27
+ currentServer,
28
+ syncServers,
29
+ remoteServers: getAllServers(
30
+ currentServer,
31
+ syncServers,
32
+ preferredServerOrigin
33
+ ),
34
+ };
35
+ };
@@ -0,0 +1,76 @@
1
+ export interface BatchOp {
2
+ type: "put" | "del";
3
+ key: string;
4
+ value?: any;
5
+ }
6
+
7
+ export class MemoryDB {
8
+ private data = new Map<string, any>();
9
+
10
+ async get(key: string): Promise<any> {
11
+ if (this.data.has(key)) {
12
+ return this.data.get(key);
13
+ }
14
+ throw new Error("NotFound");
15
+ }
16
+
17
+ async put(key: string, value: any): Promise<void> {
18
+ this.data.set(key, value);
19
+ }
20
+
21
+ async del(key: string): Promise<void> {
22
+ this.data.delete(key);
23
+ }
24
+
25
+ private async applyBatch(ops: BatchOp[]): Promise<void> {
26
+ for (const op of ops) {
27
+ if (op.type === "put") {
28
+ this.data.set(op.key, op.value);
29
+ } else {
30
+ this.data.delete(op.key);
31
+ }
32
+ }
33
+ }
34
+
35
+ batch(ops?: BatchOp[]) {
36
+ if (Array.isArray(ops)) {
37
+ return this.applyBatch(ops);
38
+ }
39
+
40
+ const bufferedOps: BatchOp[] = [];
41
+ return {
42
+ put: (key: string, value: any) => {
43
+ bufferedOps.push({ type: "put", key, value });
44
+ },
45
+ del: (key: string) => {
46
+ bufferedOps.push({ type: "del", key });
47
+ },
48
+ write: async () => {
49
+ await this.applyBatch(bufferedOps);
50
+ },
51
+ };
52
+ }
53
+
54
+ async *iterator(options: { gte?: string; lte?: string; lt?: string; reverse?: boolean } = {}) {
55
+ let keys = Array.from(this.data.keys()).sort();
56
+ if (options.reverse) {
57
+ keys.reverse();
58
+ }
59
+
60
+ for (const key of keys) {
61
+ if (options.gte && key < options.gte) continue;
62
+ if (options.lte && key > options.lte) continue;
63
+ if (options.lt && key >= options.lt) continue;
64
+ yield [key, this.data.get(key)];
65
+ }
66
+ }
67
+
68
+ // Helper for tests
69
+ dump() {
70
+ return Object.fromEntries(this.data);
71
+ }
72
+
73
+ clear() {
74
+ this.data.clear();
75
+ }
76
+ }
@@ -0,0 +1,76 @@
1
+ import type { ActorContext } from "auth/actor";
2
+ import {
3
+ authorizeRecordAccess,
4
+ type AccessAction,
5
+ type AccessResult,
6
+ } from "./resourceAccess";
7
+ import {
8
+ loadAgentDelegation,
9
+ validateAgentDelegation,
10
+ } from "./agentDelegation";
11
+
12
+ type ActorAccessInput = {
13
+ action: AccessAction;
14
+ actor?: ActorContext | null;
15
+ dbKey: string;
16
+ record?: any;
17
+ };
18
+
19
+ export async function authorizeActorRecordAccess({
20
+ action,
21
+ actor,
22
+ dbKey,
23
+ record,
24
+ }: ActorAccessInput): Promise<AccessResult> {
25
+ if (!actor) {
26
+ return { allowed: false, reason: "missing_actor" };
27
+ }
28
+
29
+ if (actor.type === "user") {
30
+ return authorizeRecordAccess({
31
+ action,
32
+ actionUserId: actor.userId,
33
+ dbKey,
34
+ record,
35
+ });
36
+ }
37
+
38
+ if (!actor.principalUserId) {
39
+ return { allowed: false, reason: "missing_principal_user" };
40
+ }
41
+
42
+ const principalAccess = await authorizeRecordAccess({
43
+ action,
44
+ actionUserId: actor.principalUserId,
45
+ dbKey,
46
+ record,
47
+ });
48
+ if (!principalAccess.allowed) {
49
+ return {
50
+ ...principalAccess,
51
+ allowed: false,
52
+ reason: "principal_access_denied",
53
+ };
54
+ }
55
+
56
+ const delegation = await loadAgentDelegation(
57
+ actor.principalUserId,
58
+ actor.agentId
59
+ );
60
+ const delegationAccess = validateAgentDelegation({
61
+ delegation,
62
+ action,
63
+ dbKey,
64
+ record,
65
+ spaceId: principalAccess.spaceId,
66
+ });
67
+ if (!delegationAccess.allowed) {
68
+ return {
69
+ ...principalAccess,
70
+ allowed: false,
71
+ reason: delegationAccess.reason,
72
+ };
73
+ }
74
+
75
+ return principalAccess;
76
+ }