nolo-cli 0.1.12 → 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 (323) hide show
  1. package/README.md +54 -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 +141 -22
  8. package/agentRuntimeLocal.ts +7 -0
  9. package/ai/agent/_executeModel.ts +118 -0
  10. package/ai/agent/agentSlice.ts +545 -0
  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/agent.ts +2 -0
  47. package/ai/ai.locale.ts +1083 -0
  48. package/ai/chat/accumulateToolCallChunks.ts +95 -0
  49. package/ai/chat/fetchUtils.native.ts +276 -0
  50. package/ai/chat/fetchUtils.ts +153 -0
  51. package/ai/chat/inlineImageUrlsForCustomProvider.ts +117 -0
  52. package/ai/chat/parseApiError.ts +64 -0
  53. package/ai/chat/parseMultilineSSE.ts +95 -0
  54. package/ai/chat/sendOpenAICompletionsRequest.native.ts +682 -0
  55. package/ai/chat/sendOpenAICompletionsRequest.ts +712 -0
  56. package/ai/chat/sendOpenAIResponseRequest.ts +512 -0
  57. package/ai/chat/shouldUseServerProxy.ts +18 -0
  58. package/ai/chat/sseClient.native.ts +91 -0
  59. package/ai/chat/sseClient.ts +67 -0
  60. package/ai/chat/streamReader.native.ts +31 -0
  61. package/ai/chat/streamReader.ts +62 -0
  62. package/ai/chat/updateTotalUsage.ts +72 -0
  63. package/ai/context/buildReferenceContext.ts +437 -0
  64. package/ai/context/calculateContextUsage.ts +133 -0
  65. package/ai/context/retention.ts +165 -0
  66. package/ai/context/tokenUtils.ts +78 -0
  67. package/ai/index.ts +1 -0
  68. package/ai/llm/agentCapabilities.ts +74 -0
  69. package/ai/llm/calculateGeminiImageTokens.ts +57 -0
  70. package/ai/llm/deepinfra.ts +28 -0
  71. package/ai/llm/fireworks.ts +68 -0
  72. package/ai/llm/generateRequestBody.ts +165 -0
  73. package/ai/llm/getModelContextWindow.ts +84 -0
  74. package/ai/llm/getNoloKey.ts +37 -0
  75. package/ai/llm/getPricing.ts +232 -0
  76. package/ai/llm/hooks/useModelPricing.ts +75 -0
  77. package/ai/llm/imagePricing.ts +66 -0
  78. package/ai/llm/isResponseAPIModel.ts +13 -0
  79. package/ai/llm/kimi.ts +18 -0
  80. package/ai/llm/mimo.ts +71 -0
  81. package/ai/llm/mistral.ts +22 -0
  82. package/ai/llm/modelAvatar.ts +427 -0
  83. package/ai/llm/models.ts +45 -0
  84. package/ai/llm/openrouterModels.ts +141 -0
  85. package/ai/llm/providers.ts +307 -0
  86. package/ai/llm/reasoningModels.ts +28 -0
  87. package/ai/llm/types.ts +59 -0
  88. package/ai/llm/usageRequestOptions.ts +59 -0
  89. package/ai/memory/capture.ts +148 -0
  90. package/ai/memory/consolidate.ts +104 -0
  91. package/ai/memory/delete.ts +147 -0
  92. package/ai/memory/overlay.ts +84 -0
  93. package/ai/memory/query.ts +38 -0
  94. package/ai/memory/queryShared.ts +160 -0
  95. package/ai/memory/rank.ts +105 -0
  96. package/ai/memory/recentRelationshipRecap.ts +247 -0
  97. package/ai/memory/remember.ts +167 -0
  98. package/ai/memory/runtime.ts +76 -0
  99. package/ai/memory/store.ts +20 -0
  100. package/ai/memory/storeShared.ts +76 -0
  101. package/ai/memory/types.ts +46 -0
  102. package/ai/memory/understanding.ts +349 -0
  103. package/ai/memory/understandingGreeting.ts +264 -0
  104. package/ai/messages/type.ts +20 -0
  105. package/ai/policy/personalizationDialog.ts +333 -0
  106. package/ai/policy/runtimePolicy.ts +440 -0
  107. package/ai/policy/selfUpdateFields.ts +48 -0
  108. package/ai/policy/types.ts +64 -0
  109. package/ai/skills/referenceRuntime.ts +274 -0
  110. package/ai/skills/skillDiagnostics.ts +251 -0
  111. package/ai/skills/skillDocBuilder.ts +139 -0
  112. package/ai/skills/skillDocProtocol.ts +434 -0
  113. package/ai/skills/skillReferenceSummary.ts +63 -0
  114. package/ai/skills/skillSummaryMarker.ts +26 -0
  115. package/ai/token/calculatePrice.ts +546 -0
  116. package/ai/token/db.ts +98 -0
  117. package/ai/token/externalToolCost.ts +321 -0
  118. package/ai/token/hooks/useRecords.ts +65 -0
  119. package/ai/token/missingUsageEstimate.ts +42 -0
  120. package/ai/token/modelUsageQuery.ts +252 -0
  121. package/ai/token/normalizeUsage.ts +84 -0
  122. package/ai/token/openaiImageGenerationUsage.ts +56 -0
  123. package/ai/token/prepareTokenUsageData.ts +88 -0
  124. package/ai/token/query.ts +88 -0
  125. package/ai/token/queryUserTokens.ts +59 -0
  126. package/ai/token/resolveBillingTarget.ts +52 -0
  127. package/ai/token/saveTokenRecord.ts +53 -0
  128. package/ai/token/serverDialogProjection.ts +78 -0
  129. package/ai/token/serverTokenWriter.ts +143 -0
  130. package/ai/token/stats.ts +21 -0
  131. package/ai/token/tokenThunks.ts +24 -0
  132. package/ai/token/types.ts +93 -0
  133. package/ai/tools/agent/agentTools.ts +176 -0
  134. package/ai/tools/agent/agentUpdateShared.ts +311 -0
  135. package/ai/tools/agent/callAgentTool.ts +139 -0
  136. package/ai/tools/agent/createAgentTool.ts +512 -0
  137. package/ai/tools/agent/createDialogTool.ts +69 -0
  138. package/ai/tools/agent/createSkillAgentTool.ts +62 -0
  139. package/ai/tools/agent/parallelBudget.ts +221 -0
  140. package/ai/tools/agent/presets/appBuilderPreset.ts +147 -0
  141. package/ai/tools/agent/runLlmTool.ts +96 -0
  142. package/ai/tools/agent/runStreamingAgentTool.ts +73 -0
  143. package/ai/tools/agent/skillAgentArgs.ts +106 -0
  144. package/ai/tools/agent/skillAgentPreset.ts +89 -0
  145. package/ai/tools/agent/streamParallelAgentsTool.ts +122 -0
  146. package/ai/tools/agent/updateAgentTool.ts +96 -0
  147. package/ai/tools/agent/updateSelfTool.ts +113 -0
  148. package/ai/tools/amazonProductScraperTool.ts +86 -0
  149. package/ai/tools/apifyActorClient.ts +45 -0
  150. package/ai/tools/appEditGuard.ts +372 -0
  151. package/ai/tools/appReadSnapshot.ts +153 -0
  152. package/ai/tools/appTools.ts +1549 -0
  153. package/ai/tools/applyEditTool.ts +256 -0
  154. package/ai/tools/applyLineEditsTool.ts +312 -0
  155. package/ai/tools/browserTools/click.ts +33 -0
  156. package/ai/tools/browserTools/closeSession.ts +29 -0
  157. package/ai/tools/browserTools/common.ts +27 -0
  158. package/ai/tools/browserTools/openSession.ts +48 -0
  159. package/ai/tools/browserTools/readContent.ts +38 -0
  160. package/ai/tools/browserTools/selectOption.ts +46 -0
  161. package/ai/tools/browserTools/typeText.ts +42 -0
  162. package/ai/tools/category/createCategoryTool.ts +66 -0
  163. package/ai/tools/category/queryContentsByCategoryTool.ts +69 -0
  164. package/ai/tools/category/updateContentCategoryTool.ts +75 -0
  165. package/ai/tools/cfBrowserTools.ts +319 -0
  166. package/ai/tools/cfSpeechToTextTool.ts +49 -0
  167. package/ai/tools/checkEnvTool.ts +65 -0
  168. package/ai/tools/cloudflareCrawlTool.ts +289 -0
  169. package/ai/tools/codeSearchTool.ts +111 -0
  170. package/ai/tools/codeTools.ts +101 -0
  171. package/ai/tools/createDocTool.ts +132 -0
  172. package/ai/tools/createPlanTool.ts +999 -0
  173. package/ai/tools/createSkillDocTool.ts +155 -0
  174. package/ai/tools/createWorkflowTool.ts +154 -0
  175. package/ai/tools/deepseekOcrTool.ts +34 -0
  176. package/ai/tools/delayTool.ts +31 -0
  177. package/ai/tools/deleteSpacesTool.ts +325 -0
  178. package/ai/tools/deleteSpacesToolModel.ts +159 -0
  179. package/ai/tools/devReloadUtils.ts +29 -0
  180. package/ai/tools/dialogMessageSearch.ts +137 -0
  181. package/ai/tools/doctorSkillTool.ts +72 -0
  182. package/ai/tools/ecommerceScraperTool.ts +86 -0
  183. package/ai/tools/emailTools.ts +549 -0
  184. package/ai/tools/evalSkillTool.ts +92 -0
  185. package/ai/tools/exaSearchTool.ts +64 -0
  186. package/ai/tools/execBashTool.ts +379 -0
  187. package/ai/tools/executeSqlTool.ts +192 -0
  188. package/ai/tools/fetchWebpageSupport.ts +309 -0
  189. package/ai/tools/fetchWebpageTool.ts +84 -0
  190. package/ai/tools/geminiImagePreviewTool.ts +361 -0
  191. package/ai/tools/generateDocxTool.ts +215 -0
  192. package/ai/tools/googleSearchScraperTool.ts +106 -0
  193. package/ai/tools/importDataTool.ts +133 -0
  194. package/ai/tools/importSkillTool.ts +162 -0
  195. package/ai/tools/index.ts +1927 -0
  196. package/ai/tools/listFilesTool.ts +82 -0
  197. package/ai/tools/listUserSpacesTool.ts +113 -0
  198. package/ai/tools/modelUsageTools.ts +199 -0
  199. package/ai/tools/olmOcrTool.ts +34 -0
  200. package/ai/tools/openaiImageTool.ts +267 -0
  201. package/ai/tools/prepareTools.ts +23 -0
  202. package/ai/tools/readDocTool.ts +84 -0
  203. package/ai/tools/readFileTool.ts +211 -0
  204. package/ai/tools/readTool.ts +163 -0
  205. package/ai/tools/readXPostTool.ts +233 -0
  206. package/ai/tools/rememberMemoryTool.ts +84 -0
  207. package/ai/tools/remotionVideoTool.ts +151 -0
  208. package/ai/tools/searchDialogMessagesTool.ts +222 -0
  209. package/ai/tools/searchRepoTool.ts +115 -0
  210. package/ai/tools/searchWorkspaceTool.ts +259 -0
  211. package/ai/tools/skillFollowup.ts +86 -0
  212. package/ai/tools/surfWeatherTool.ts +169 -0
  213. package/ai/tools/table/addTableRowTool.ts +217 -0
  214. package/ai/tools/table/createTableTool.ts +315 -0
  215. package/ai/tools/table/rowTools.ts +366 -0
  216. package/ai/tools/table/schemaTools.ts +244 -0
  217. package/ai/tools/table/shareTableTool.ts +148 -0
  218. package/ai/tools/table/toolShared.ts +129 -0
  219. package/ai/tools/toolApiClient.ts +198 -0
  220. package/ai/tools/toolNameAliases.ts +57 -0
  221. package/ai/tools/toolResultError.ts +42 -0
  222. package/ai/tools/toolRunSlice.ts +303 -0
  223. package/ai/tools/toolSchemaCompatibility.ts +53 -0
  224. package/ai/tools/toolVisibility.ts +4 -0
  225. package/ai/tools/types.ts +20 -0
  226. package/ai/tools/uiAskChoiceTool.ts +104 -0
  227. package/ai/tools/updateContentTitleTool.ts +84 -0
  228. package/ai/tools/updateDocTool.ts +105 -0
  229. package/ai/tools/updateUserPreferenceProfileTool.ts +145 -0
  230. package/ai/tools/whisperTool.ts +77 -0
  231. package/ai/tools/writeFileTool.ts +210 -0
  232. package/ai/tools/youtubeScraperTool.ts +116 -0
  233. package/ai/tools/ziweiChartTool.ts +678 -0
  234. package/ai/types.ts +55 -0
  235. package/ai/workflow/workflowExecutor.ts +323 -0
  236. package/ai/workflow/workflowSlice.ts +73 -0
  237. package/ai/workflow/workflowTypes.ts +106 -0
  238. package/authCommands.ts +185 -21
  239. package/client/agentRun.test.ts +240 -0
  240. package/client/agentRun.ts +182 -19
  241. package/client/compactDialog.test.ts +238 -0
  242. package/client/compactDialog.ts +5 -2
  243. package/client/localRuntimeAdapter.test.ts +135 -0
  244. package/client/localRuntimeAdapter.ts +244 -0
  245. package/client/profileConfig.test.ts +40 -0
  246. package/client/streamingOutput.test.ts +22 -0
  247. package/client/streamingOutput.ts +38 -0
  248. package/commandRegistry.ts +11 -2
  249. package/connector-experimental/index.ts +5 -0
  250. package/database/actions/cacheMergedUserData.ts +64 -0
  251. package/database/actions/common.ts +242 -0
  252. package/database/actions/deleteFile.ts +40 -0
  253. package/database/actions/fetchUserData.ts +16 -0
  254. package/database/actions/fileContent.ts +125 -0
  255. package/database/actions/patch.ts +155 -0
  256. package/database/actions/read.ts +337 -0
  257. package/database/actions/readAndWait.ts +224 -0
  258. package/database/actions/readRequestManager.ts +120 -0
  259. package/database/actions/remove.ts +94 -0
  260. package/database/actions/replication.ts +366 -0
  261. package/database/actions/upload.ts +174 -0
  262. package/database/actions/upsert.ts +56 -0
  263. package/database/actions/write.ts +126 -0
  264. package/database/client/db.native.ts +73 -0
  265. package/database/client/db.ts +51 -0
  266. package/database/client/fetchUserData.ts +61 -0
  267. package/database/client/handleError.ts +19 -0
  268. package/database/client/queryRequest.ts +21 -0
  269. package/database/config.ts +21 -0
  270. package/database/dbActionThunks.ts +1 -0
  271. package/database/dbSlice.ts +149 -0
  272. package/database/email.ts +42 -0
  273. package/database/fileRing.ts +51 -0
  274. package/database/fileSharding.ts +70 -0
  275. package/database/fileStorage.native.ts +92 -0
  276. package/database/fileStorage.ts +232 -0
  277. package/database/fileUrl.ts +34 -0
  278. package/database/hooks/useUserData.ts +489 -0
  279. package/database/index.ts +1 -0
  280. package/database/keys.ts +765 -0
  281. package/database/queryPrefixes.ts +14 -0
  282. package/database/requests.ts +443 -0
  283. package/database/runtimeServerContext.ts +35 -0
  284. package/database/server/MemoryDB.ts +76 -0
  285. package/database/server/actorAccess.ts +76 -0
  286. package/database/server/agentDelegation.ts +124 -0
  287. package/database/server/coreDataOwnership.ts +13 -0
  288. package/database/server/coreDataProxy.ts +76 -0
  289. package/database/server/cybotReadonly.ts +18 -0
  290. package/database/server/dataHandlers.ts +111 -0
  291. package/database/server/db.ts +118 -0
  292. package/database/server/dbPath.ts +20 -0
  293. package/database/server/delete.ts +499 -0
  294. package/database/server/emailRepository.ts +1480 -0
  295. package/database/server/ensureDbOpen.ts +12 -0
  296. package/database/server/fileRead.ts +337 -0
  297. package/database/server/fileService.ts +436 -0
  298. package/database/server/handleTransaction.ts +86 -0
  299. package/database/server/patch.ts +282 -0
  300. package/database/server/query.ts +138 -0
  301. package/database/server/read.ts +325 -0
  302. package/database/server/resourceAccess.ts +211 -0
  303. package/database/server/routes.ts +110 -0
  304. package/database/server/spaceMemberAuthority.ts +67 -0
  305. package/database/server/upload.ts +159 -0
  306. package/database/server/write.ts +494 -0
  307. package/database/server/writeAuthority.ts +133 -0
  308. package/database/sqliteDb.ts +46 -0
  309. package/database/table/deleteTable.ts +120 -0
  310. package/database/tenantPlacement.ts +57 -0
  311. package/database/tombstones.ts +52 -0
  312. package/database/userDataLoadDecision.ts +17 -0
  313. package/database/userDataMerge.ts +95 -0
  314. package/database/userPreferenceRegister.ts +108 -0
  315. package/database/utils/dbPath.ts +47 -0
  316. package/database/utils/ulid.native.ts +6 -0
  317. package/database/utils/ulid.ts +1 -0
  318. package/index.ts +25 -15
  319. package/localRuntimeDb.ts +28 -0
  320. package/package.json +16 -4
  321. package/runtimeModeArgs.ts +33 -0
  322. package/tui/readlineWorkspace.ts +1 -0
  323. package/tui/session.ts +22 -0
@@ -0,0 +1,337 @@
1
+ // 文件路径: database/actions/read.ts
2
+
3
+ import type { AppThunkApi } from "app/store";
4
+ import { getRuntimeServerContext } from "database/runtimeServerContext";
5
+ import {
6
+ fetchFromClientDb,
7
+ fetchFromServer,
8
+ isReadTimeoutError,
9
+ logger,
10
+ } from "./common";
11
+ import { readRequestManager } from "./readRequestManager";
12
+ import { shouldReplaceWithNextRecord } from "../tombstones";
13
+ import { scheduleExistingRecordReplication } from "./replication";
14
+
15
+ // --- 辅助函数 ---
16
+
17
+ const updateClientDbIfNewer = async (
18
+ clientDb: any,
19
+ dbKey: string,
20
+ remoteData: any,
21
+ localData: any
22
+ ): Promise<void> => {
23
+ if (!clientDb) return;
24
+ try {
25
+ if (isRemoteDataNewer(remoteData, localData)) {
26
+ await clientDb.put(dbKey, remoteData);
27
+ }
28
+ } catch (err) {
29
+ throw err;
30
+ }
31
+ };
32
+
33
+ const isRemoteDataNewer = (remoteData: any, localData: any): boolean => {
34
+ if (!remoteData || typeof remoteData !== "object") return false;
35
+ if (!localData || typeof localData !== "object") return true;
36
+ return shouldReplaceWithNextRecord(remoteData, localData);
37
+ };
38
+
39
+ const shouldSyncLocalToServer = (localData: any, remoteData: any): boolean => {
40
+ return !!localData && !remoteData;
41
+ };
42
+
43
+ const syncLocalDataToServer = async (
44
+ thunkApi: AppThunkApi,
45
+ dbKey: string,
46
+ localData: any
47
+ ): Promise<void> => {
48
+ try {
49
+ const state = thunkApi.getState();
50
+ const { currentServer, syncServers } = getRuntimeServerContext(state);
51
+ scheduleExistingRecordReplication({
52
+ currentServer,
53
+ syncServers,
54
+ dbKey,
55
+ localData,
56
+ state,
57
+ });
58
+ } catch (err) {
59
+ // Error ignored
60
+ }
61
+ };
62
+
63
+ const saveRemoteDataToClientDb = async (
64
+ clientDb: any,
65
+ dbKey: string,
66
+ remoteData: any,
67
+ serverOrigin?: string | null
68
+ ): Promise<void> => {
69
+ if (!clientDb) return;
70
+ try {
71
+ await clientDb.put(
72
+ dbKey,
73
+ serverOrigin
74
+ ? {
75
+ ...remoteData,
76
+ serverOrigin,
77
+ }
78
+ : remoteData
79
+ );
80
+ } catch (err) {
81
+ // Error ignored
82
+ }
83
+ };
84
+
85
+ const getValidRemoteData = (
86
+ dbKey: string,
87
+ settledResults: PromiseSettledResult<any>[],
88
+ localData?: any
89
+ ): { data: any; index: number } | null => {
90
+ const validResults = settledResults
91
+ .map((result, index) => ({
92
+ data: result.status === "fulfilled" ? result.value : null,
93
+ index,
94
+ }))
95
+ .filter((item) => item.data !== null && typeof item.data === "object");
96
+
97
+ if (validResults.length === 0) return null;
98
+
99
+ const latest = validResults.reduce((latest, current) => {
100
+ return shouldReplaceWithNextRecord(current.data, latest.data) ? current : latest;
101
+ });
102
+
103
+ return {
104
+ ...latest,
105
+ data: latest.data,
106
+ };
107
+ };
108
+
109
+ const processRemoteDataInBackground = async (
110
+ clientDb: any,
111
+ dbKey: string,
112
+ remotePromises: Promise<any>[],
113
+ remoteServers: string[],
114
+ localData: any,
115
+ thunkApi: AppThunkApi
116
+ ): Promise<void> => {
117
+ if (!clientDb) return;
118
+ try {
119
+ const settledResults = await Promise.allSettled(remotePromises);
120
+ const remoteResult = getValidRemoteData(dbKey, settledResults, localData);
121
+ const validRemoteData = remoteResult ? remoteResult.data : null;
122
+ const serverOrigin =
123
+ remoteResult && remoteServers[remoteResult.index]
124
+ ? remoteServers[remoteResult.index]
125
+ : undefined;
126
+
127
+ if (validRemoteData && localData) {
128
+ await updateClientDbIfNewer(
129
+ clientDb,
130
+ dbKey,
131
+ serverOrigin ? { ...validRemoteData, serverOrigin } : validRemoteData,
132
+ localData
133
+ );
134
+ }
135
+ if (shouldSyncLocalToServer(localData, validRemoteData)) {
136
+ await syncLocalDataToServer(thunkApi, dbKey, localData);
137
+ }
138
+ } catch (err) {
139
+ // Background sync errors ignored
140
+ }
141
+ };
142
+
143
+ // --- 主函数 ---
144
+
145
+ export const readAction = async (
146
+ payload: {
147
+ dbKey: string;
148
+ signal?: AbortSignal;
149
+ preferredServerOrigin?: string | null;
150
+ },
151
+ thunkApi: AppThunkApi
152
+ ): Promise<any> => {
153
+ const dbKey = payload.dbKey;
154
+ const signal = payload.signal;
155
+ const preferredServerOrigin = payload.preferredServerOrigin;
156
+
157
+ if (!dbKey || typeof dbKey !== "string") {
158
+ throw new Error("readAction requires a non-empty dbKey.");
159
+ }
160
+
161
+ // 2. 尽早检查中止信号,快速退出
162
+ if (signal?.aborted) {
163
+ throw new DOMException("Aborted", "AbortError");
164
+ }
165
+
166
+ const { db: clientDb } = thunkApi.extra;
167
+ if (!clientDb) {
168
+ throw new Error("Client database is not available.");
169
+ }
170
+
171
+ const executeRead = async (): Promise<any> => {
172
+ const state = thunkApi.getState();
173
+ const { currentToken, remoteServers: allServers } =
174
+ getRuntimeServerContext(state, preferredServerOrigin);
175
+ const isLoggedIn = !!currentToken;
176
+ const now = Date.now();
177
+ const localData = await fetchFromClientDb(clientDb, dbKey);
178
+ readRequestManager.cleanupMisses(now);
179
+ readRequestManager.cleanupLocalHitRevalidations(now);
180
+
181
+ if (localData) {
182
+ readRequestManager.clearMiss(dbKey);
183
+ } else {
184
+ const retryInMs = readRequestManager.getRetryInMs(dbKey, now);
185
+ if (typeof retryInMs === "number" && retryInMs > 0) {
186
+ logger.debug(
187
+ { dbKey, retryInMs },
188
+ "[readAction] Suppressing repeated miss read"
189
+ );
190
+ throw new Error(`Read temporarily suppressed for missing key "${dbKey}".`);
191
+ }
192
+ }
193
+
194
+ // 离线 / 无可用远程服务器:只看本地
195
+ if (allServers.length === 0) {
196
+ if (localData) {
197
+ return { ...localData, dbKey };
198
+ }
199
+ readRequestManager.markMiss(dbKey, now);
200
+ throw new Error(
201
+ `Failed to fetch data for key "${dbKey}" because network is offline and no local data is available.`
202
+ );
203
+ }
204
+
205
+ const preferredServer =
206
+ typeof preferredServerOrigin === "string" && preferredServerOrigin.trim().length > 0
207
+ ? preferredServerOrigin.trim().replace(/\/+$/, "")
208
+ : null;
209
+ const remainingServers = preferredServer
210
+ ? allServers.filter((server) => server.replace(/\/+$/, "") !== preferredServer)
211
+ : allServers;
212
+
213
+ if (localData) {
214
+ // Local-first: return durable local data immediately and only revalidate
215
+ // against remote servers in the background. This avoids turning a
216
+ // preferred-server timeout into a visible read failure for data we
217
+ // already have locally (for example createDialog -> initDialog).
218
+ if (!signal?.aborted) {
219
+ const retryInMs = readRequestManager.getLocalHitRevalidateInMs(
220
+ dbKey,
221
+ now
222
+ );
223
+ if (retryInMs === null) {
224
+ readRequestManager.markLocalHitRevalidated(dbKey, now);
225
+ const revalidationServers = preferredServer
226
+ ? [preferredServer, ...remainingServers]
227
+ : remainingServers;
228
+ const remotePromises = revalidationServers.map((server) =>
229
+ fetchFromServer(server, dbKey, isLoggedIn ? currentToken : undefined, signal)
230
+ );
231
+ void (async () => {
232
+ await processRemoteDataInBackground(
233
+ clientDb,
234
+ dbKey,
235
+ remotePromises,
236
+ revalidationServers,
237
+ localData,
238
+ thunkApi
239
+ );
240
+ })();
241
+ } else {
242
+ logger.debug(
243
+ { dbKey, retryInMs },
244
+ "[readAction] Skipping frequent local-hit revalidation"
245
+ );
246
+ }
247
+ }
248
+ return { ...localData, dbKey };
249
+ }
250
+
251
+ if (preferredServer) {
252
+ try {
253
+ const preferredRemoteData = await fetchFromServer(
254
+ preferredServer,
255
+ dbKey,
256
+ isLoggedIn ? currentToken : undefined,
257
+ signal
258
+ );
259
+
260
+ if (preferredRemoteData) {
261
+ await saveRemoteDataToClientDb(
262
+ clientDb,
263
+ dbKey,
264
+ preferredRemoteData,
265
+ preferredServer
266
+ );
267
+ readRequestManager.clearMiss(dbKey);
268
+ return { ...preferredRemoteData, dbKey, serverOrigin: preferredServer };
269
+ }
270
+ } catch (error) {
271
+ if (signal?.aborted || (error as { name?: string } | null)?.name === "AbortError") {
272
+ throw error;
273
+ }
274
+ if (isReadTimeoutError(error)) {
275
+ logger.warn(
276
+ { dbKey, preferredServer, error: String((error as Error).message) },
277
+ "[readAction] Preferred server timed out; falling back to remaining servers"
278
+ );
279
+ } else {
280
+ logger.warn(
281
+ { dbKey, preferredServer, error: String(error) },
282
+ "[readAction] Preferred server read failed; falling back to remaining servers"
283
+ );
284
+ }
285
+ }
286
+ }
287
+
288
+ // 3. 将 signal 传递给所有网络请求
289
+ const remotePromises = remainingServers.map((server) =>
290
+ fetchFromServer(server, dbKey, isLoggedIn ? currentToken : undefined, signal)
291
+ );
292
+
293
+ // 如果本地没有数据,则等待网络请求结果
294
+ const settledResults = await Promise.allSettled(remotePromises);
295
+
296
+ // 4. 在处理网络结果前,再次检查中止信号
297
+ if (signal?.aborted) {
298
+ throw new DOMException("Aborted", "AbortError");
299
+ }
300
+
301
+ const remoteResult = getValidRemoteData(dbKey, settledResults, localData);
302
+ if (remoteResult) {
303
+ const { data: validRemoteData } = remoteResult;
304
+ const serverOrigin = remainingServers[remoteResult.index];
305
+ if (!signal?.aborted) {
306
+ await saveRemoteDataToClientDb(
307
+ clientDb,
308
+ dbKey,
309
+ validRemoteData,
310
+ serverOrigin
311
+ );
312
+ }
313
+ readRequestManager.clearMiss(dbKey);
314
+ return serverOrigin
315
+ ? { ...validRemoteData, dbKey, serverOrigin }
316
+ : { ...validRemoteData, dbKey };
317
+ }
318
+
319
+ readRequestManager.markMiss(dbKey, Date.now());
320
+ throw new Error(`Failed to fetch data for key "${dbKey}" from all sources.`);
321
+ };
322
+
323
+ const canDedup = !signal;
324
+ if (canDedup) {
325
+ const existing = readRequestManager.getInFlight(dbKey);
326
+ if (existing) return existing;
327
+
328
+ let inFlightPromise: Promise<any>;
329
+ inFlightPromise = executeRead().finally(() => {
330
+ readRequestManager.clearInFlight(dbKey, inFlightPromise);
331
+ });
332
+ readRequestManager.setInFlight(dbKey, inFlightPromise);
333
+ return inFlightPromise;
334
+ }
335
+
336
+ return executeRead();
337
+ };
@@ -0,0 +1,224 @@
1
+ // 文件路径: database/actions/readAndWait.ts
2
+
3
+ import type { AppThunkApi } from "app/store";
4
+ import {
5
+ fetchFromClientDb,
6
+ fetchFromServer,
7
+ } from "./common";
8
+ import { readRequestManager } from "./readRequestManager";
9
+ import { scheduleExistingRecordReplication } from "./replication";
10
+ import { getRuntimeServerContext } from "database/runtimeServerContext";
11
+
12
+ /**
13
+ * 比较远程数据和本地数据的时间戳,判断远程数据是否更新。
14
+ * @param remoteData - 从服务器获取的数据。
15
+ * @param localData - 从本地数据库获取的数据。
16
+ * @returns 如果远程数据更新,则返回 true。
17
+ */
18
+ const isRemoteDataNewer = (remoteData: any, localData: any): boolean => {
19
+ const toComparableTimestamp = (data: any): number => {
20
+ if (!data || typeof data !== "object") return 0;
21
+
22
+ const updatedAtMs = new Date(data.updatedAt).getTime();
23
+ if (Number.isFinite(updatedAtMs) && updatedAtMs > 0) return updatedAtMs;
24
+
25
+ const createdAtMs = new Date(data.createdAt).getTime();
26
+ if (Number.isFinite(createdAtMs) && createdAtMs > 0) return createdAtMs;
27
+
28
+ const metaCreatedAt = Number(data?.meta?.createdAt);
29
+ if (Number.isFinite(metaCreatedAt) && metaCreatedAt > 0) return metaCreatedAt;
30
+
31
+ return 0;
32
+ };
33
+
34
+ const remoteTs = toComparableTimestamp(remoteData);
35
+ const localTs = toComparableTimestamp(localData);
36
+ if (remoteTs <= 0) return false;
37
+ if (localTs <= 0) return true;
38
+ return remoteTs > localTs;
39
+ };
40
+
41
+ /**
42
+ * 触发一个“即发即忘”的异步任务,将本地数据上传(写入)到服务器。
43
+ * 通常在发现本地存在数据而所有远程服务器上都不存在该数据时调用。
44
+ */
45
+ const syncLocalDataToServer = async (
46
+ thunkApi: AppThunkApi,
47
+ dbKey: string,
48
+ localData: any
49
+ ): Promise<void> => {
50
+ try {
51
+ const state = thunkApi.getState();
52
+ const { currentServer, syncServers } = getRuntimeServerContext(state);
53
+ scheduleExistingRecordReplication({
54
+ currentServer,
55
+ syncServers,
56
+ dbKey,
57
+ localData,
58
+ state,
59
+ });
60
+ } catch {
61
+ // 后台同步失败不阻塞主流程,静默处理即可
62
+ }
63
+ };
64
+
65
+ /**
66
+ * 从所有远程服务器的请求结果中,筛选出有效的、时间戳最新的那一份数据。
67
+ */
68
+ const getValidRemoteData = (
69
+ dbKey: string,
70
+ settledResults: PromiseSettledResult<any>[],
71
+ localData?: any
72
+ ): { data: any; index: number } | null => {
73
+ const toComparableTimestamp = (data: any): number => {
74
+ if (!data || typeof data !== "object") return 0;
75
+
76
+ const updatedAtMs = new Date(data.updatedAt).getTime();
77
+ if (Number.isFinite(updatedAtMs) && updatedAtMs > 0) return updatedAtMs;
78
+
79
+ const createdAtMs = new Date(data.createdAt).getTime();
80
+ if (Number.isFinite(createdAtMs) && createdAtMs > 0) return createdAtMs;
81
+
82
+ const metaCreatedAt = Number(data?.meta?.createdAt);
83
+ if (Number.isFinite(metaCreatedAt) && metaCreatedAt > 0) return metaCreatedAt;
84
+
85
+ return 0;
86
+ };
87
+
88
+ const validResults = settledResults
89
+ .map((result, index) => ({
90
+ data: result.status === "fulfilled" ? result.value : null,
91
+ index,
92
+ }))
93
+ .filter((item) => item.data !== null && typeof item.data === "object");
94
+
95
+ if (validResults.length === 0) return null;
96
+
97
+ // 从所有有效的远程数据中,选出时间戳最新的那一个。
98
+ const latest = validResults.reduce((latest, current) => {
99
+ const latestTimestamp = toComparableTimestamp(latest.data);
100
+ const currentTimestamp = toComparableTimestamp(current.data);
101
+ return currentTimestamp > latestTimestamp ? current : latest;
102
+ });
103
+
104
+ return {
105
+ ...latest,
106
+ data: latest.data,
107
+ };
108
+ };
109
+
110
+ /**
111
+ * 核心处理函数:协调本地和远程数据,决定最终返回哪个版本的数据,并处理同步逻辑。
112
+ */
113
+ const processRemoteData = async (
114
+ db: any,
115
+ dbKey: string,
116
+ remotePromises: Promise<any>[],
117
+ localData: any,
118
+ thunkApi: AppThunkApi
119
+ ): Promise<any> => {
120
+ try {
121
+ // 并行执行所有远程请求,并等待它们全部完成(无论成功或失败)。
122
+ const settledResults = await Promise.allSettled(remotePromises);
123
+ // 从所有结果中,找出最“权威”的远程版本(时间戳最新)。
124
+ const remoteResult = getValidRemoteData(dbKey, settledResults, localData);
125
+ const validRemoteData = remoteResult ? remoteResult.data : null;
126
+
127
+ // --- 数据决策核心逻辑 ---
128
+
129
+ // 场景 1: 至少一个远程服务器返回了有效数据。
130
+ if (validRemoteData) {
131
+ // 如果本地没有数据,或者远程更新更新,则用远程覆盖本地缓存。
132
+ if (!localData || isRemoteDataNewer(validRemoteData, localData)) {
133
+ await db.put(dbKey, validRemoteData);
134
+ }
135
+ // 最终决策:返回权威的远程数据。
136
+ return validRemoteData;
137
+ }
138
+
139
+ // 场景 2: 所有远程服务器都没有返回有效数据,但我们本地数据库中存在数据。
140
+ if (localData) {
141
+ // 仅当“确实有远程目标”时,才尝试上传本地数据到服务器。
142
+ if (remotePromises.length > 0) {
143
+ void syncLocalDataToServer(thunkApi, dbKey, localData);
144
+ }
145
+ // 最终决策:返回本地数据。
146
+ return localData;
147
+ }
148
+
149
+ // 场景 3: 远程和本地都找不到任何数据。
150
+ throw new Error("Failed to fetch data from all sources");
151
+ } catch (err) {
152
+ // 如果在上述过程中发生错误,且有本地数据,则优先返回本地数据,避免崩溃。
153
+ if (localData) {
154
+ return localData;
155
+ }
156
+ throw err;
157
+ }
158
+ };
159
+
160
+ /**
161
+ * 主函数:读取数据,并等待远程和本地操作都完成后才返回最合适的数据。
162
+ * - 本地优先:有本地数据时,远程只用于更新缓存或回填云端。
163
+ * - 远程优先:若拿到了有效远程数据,以其为准。
164
+ */
165
+ export const readAndWaitAction = async (
166
+ dbKey: string,
167
+ thunkApi: AppThunkApi
168
+ ): Promise<any> => {
169
+ const { db } = thunkApi.extra;
170
+
171
+ if (!db) {
172
+ throw new Error(
173
+ "Database instance is not available in thunk extra argument."
174
+ );
175
+ }
176
+
177
+ const state = thunkApi.getState();
178
+ const { currentToken, remoteServers: allServers } =
179
+ getRuntimeServerContext(state);
180
+ const isLoggedIn = !!currentToken;
181
+
182
+ const executeReadAndWait = async (): Promise<any> => {
183
+ // 1. 准备所有需要访问的远程服务器(带去重 + 离线检测)
184
+ // 2. 首先,尝试从本地数据库获取数据(可能为 null)
185
+ const localData = await fetchFromClientDb(db, dbKey);
186
+
187
+ // 如果离线或没有任何可用远程服务器:
188
+ if (allServers.length === 0) {
189
+ if (localData) {
190
+ return { ...localData, dbKey };
191
+ }
192
+ throw new Error(
193
+ `Failed to fetch data for key "${dbKey}" because network is offline and no local data is available.`
194
+ );
195
+ }
196
+
197
+ // 3. 创建所有到远程服务器的并行请求
198
+ const remotePromises = allServers.map((server) =>
199
+ fetchFromServer(server, dbKey, isLoggedIn ? currentToken : undefined)
200
+ );
201
+
202
+ // 4. 将所有信息交给核心处理函数去做最终决策
203
+ const chosenData = await processRemoteData(
204
+ db,
205
+ dbKey,
206
+ remotePromises,
207
+ localData,
208
+ thunkApi
209
+ );
210
+
211
+ // ⭐ 统一:不论返回的是本地还是远程,最终都附加 dbKey 字段
212
+ return { ...chosenData, dbKey };
213
+ };
214
+
215
+ const existing = readRequestManager.getInFlight(dbKey);
216
+ if (existing) return existing;
217
+
218
+ let inFlightPromise: Promise<any>;
219
+ inFlightPromise = executeReadAndWait().finally(() => {
220
+ readRequestManager.clearInFlight(dbKey, inFlightPromise);
221
+ });
222
+ readRequestManager.setInFlight(dbKey, inFlightPromise);
223
+ return inFlightPromise;
224
+ };
@@ -0,0 +1,120 @@
1
+ const DEFAULT_MISS_COOLDOWN_MS = 2000;
2
+ const DEFAULT_LOCAL_HIT_REVALIDATE_COOLDOWN_MS = 1500;
3
+ const DEFAULT_MISS_CACHE_MAX_SIZE = 1000;
4
+
5
+ export class ReadRequestManager {
6
+ private readonly inFlightReads = new Map<string, Promise<any>>();
7
+ private readonly recentMisses = new Map<string, number>();
8
+ private readonly recentLocalHitRevalidations = new Map<string, number>();
9
+
10
+ constructor(
11
+ private readonly options: {
12
+ missCooldownMs?: number;
13
+ missCacheMaxSize?: number;
14
+ localHitRevalidateCooldownMs?: number;
15
+ } = {}
16
+ ) {}
17
+
18
+ private get missCooldownMs() {
19
+ return this.options.missCooldownMs ?? DEFAULT_MISS_COOLDOWN_MS;
20
+ }
21
+
22
+ private get missCacheMaxSize() {
23
+ return this.options.missCacheMaxSize ?? DEFAULT_MISS_CACHE_MAX_SIZE;
24
+ }
25
+
26
+ private get localHitRevalidateCooldownMs() {
27
+ return (
28
+ this.options.localHitRevalidateCooldownMs ??
29
+ DEFAULT_LOCAL_HIT_REVALIDATE_COOLDOWN_MS
30
+ );
31
+ }
32
+
33
+ getInFlight(dbKey: string) {
34
+ return this.inFlightReads.get(dbKey);
35
+ }
36
+
37
+ setInFlight(dbKey: string, promise: Promise<any>) {
38
+ this.inFlightReads.set(dbKey, promise);
39
+ }
40
+
41
+ clearInFlight(dbKey: string, promise: Promise<any>) {
42
+ if (this.inFlightReads.get(dbKey) === promise) {
43
+ this.inFlightReads.delete(dbKey);
44
+ }
45
+ }
46
+
47
+ clearMiss(dbKey: string) {
48
+ this.recentMisses.delete(dbKey);
49
+ }
50
+
51
+ getRetryInMs(dbKey: string, now: number) {
52
+ const missUntil = this.recentMisses.get(dbKey);
53
+ if (typeof missUntil !== "number") return null;
54
+ if (missUntil <= now) {
55
+ this.recentMisses.delete(dbKey);
56
+ return null;
57
+ }
58
+ return missUntil - now;
59
+ }
60
+
61
+ markMiss(dbKey: string, now: number, cooldownMs = this.missCooldownMs) {
62
+ this.recentMisses.set(dbKey, now + cooldownMs);
63
+ this.cleanupMisses(now);
64
+ }
65
+
66
+ getLocalHitRevalidateInMs(dbKey: string, now: number) {
67
+ const nextAllowedAt = this.recentLocalHitRevalidations.get(dbKey);
68
+ if (typeof nextAllowedAt !== "number") return null;
69
+ if (nextAllowedAt <= now) {
70
+ this.recentLocalHitRevalidations.delete(dbKey);
71
+ return null;
72
+ }
73
+ return nextAllowedAt - now;
74
+ }
75
+
76
+ markLocalHitRevalidated(
77
+ dbKey: string,
78
+ now: number,
79
+ cooldownMs = this.localHitRevalidateCooldownMs
80
+ ) {
81
+ this.recentLocalHitRevalidations.set(dbKey, now + cooldownMs);
82
+ this.cleanupLocalHitRevalidations(now);
83
+ }
84
+
85
+ private cleanupExpiringMap(map: Map<string, number>, now: number) {
86
+ for (const [key, expiresAt] of Array.from(map.entries())) {
87
+ if (expiresAt <= now) {
88
+ map.delete(key);
89
+ }
90
+ }
91
+
92
+ if (map.size <= this.missCacheMaxSize) return;
93
+ const overflow = map.size - this.missCacheMaxSize;
94
+ const keys = Array.from(map.keys());
95
+ for (let i = 0; i < overflow; i += 1) {
96
+ const key = keys[i];
97
+ if (key) map.delete(key);
98
+ }
99
+ }
100
+
101
+ cleanupMisses(now: number) {
102
+ this.cleanupExpiringMap(this.recentMisses, now);
103
+ }
104
+
105
+ cleanupLocalHitRevalidations(now: number) {
106
+ this.cleanupExpiringMap(this.recentLocalHitRevalidations, now);
107
+ }
108
+
109
+ // test helper
110
+ getMissCacheSize() {
111
+ return this.recentMisses.size;
112
+ }
113
+
114
+ // test helper
115
+ getLocalHitRevalidationCacheSize() {
116
+ return this.recentLocalHitRevalidations.size;
117
+ }
118
+ }
119
+
120
+ export const readRequestManager = new ReadRequestManager();