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.
- package/README.md +9 -2
- package/agent-runtime/hostAdapter.ts +53 -0
- package/agent-runtime/index.ts +28 -0
- package/agent-runtime/localLoop.ts +62 -0
- package/agent-runtime/runtimeDecision.ts +70 -0
- package/agent-runtime/types.ts +87 -0
- package/agentRuntimeCommands.ts +139 -22
- package/agentRuntimeLocal.ts +7 -0
- package/ai/agent/_executeModel.ts +118 -0
- package/ai/agent/agentSlice.ts +544 -1
- package/ai/agent/appWorkingMemory.ts +126 -0
- package/ai/agent/avatarUtils.ts +24 -0
- package/ai/agent/buildEditingContext.ts +373 -0
- package/ai/agent/buildSystemPrompt.ts +532 -0
- package/ai/agent/cleanAgentMessages.ts +140 -0
- package/ai/agent/cliChatClient.ts +119 -0
- package/ai/agent/contextCompiler.ts +107 -0
- package/ai/agent/contextLayerContract.ts +44 -0
- package/ai/agent/createAgentSchema.ts +234 -0
- package/ai/agent/executeToolCall.ts +58 -0
- package/ai/agent/fetchAgentContexts.ts +42 -0
- package/ai/agent/generatePrompt.ts +3 -0
- package/ai/agent/getFullChatContextKeys.ts +168 -0
- package/ai/agent/hooks/fetchPublicAgents.ts +133 -0
- package/ai/agent/hooks/useAgentConfig.ts +61 -0
- package/ai/agent/hooks/useAgentDialog.ts +35 -0
- package/ai/agent/hooks/useAgentFormValidation.ts +202 -0
- package/ai/agent/hooks/usePublicAgents.ts +473 -0
- package/ai/agent/persistMessageWithFixedId.ts +37 -0
- package/ai/agent/planSlice.ts +259 -0
- package/ai/agent/referenceUtils.ts +229 -0
- package/ai/agent/runAgentBackground.ts +238 -0
- package/ai/agent/runAgentClientLoop.ts +138 -0
- package/ai/agent/runtimeGuidance.ts +97 -0
- package/ai/agent/runtimeServerBase.ts +37 -0
- package/ai/agent/server/fetchPublicAgents.ts +128 -0
- package/ai/agent/startParallelAgentStreams.ts +424 -0
- package/ai/agent/startupProtocol.ts +53 -0
- package/ai/agent/streamAgentChatTurn.ts +1299 -0
- package/ai/agent/streamAgentChatTurnUtils.ts +738 -0
- package/ai/agent/types.ts +71 -0
- package/ai/agent/utils/imageOutput.ts +39 -0
- package/ai/agent/utils/publicImageAgentMode.ts +26 -0
- package/ai/agent/utils/sortUtils.ts +250 -0
- package/ai/agent/web/referencePickerUtils.ts +146 -0
- package/ai/ai.locale.ts +1083 -0
- package/ai/chat/accumulateToolCallChunks.ts +95 -0
- package/ai/chat/fetchUtils.native.ts +276 -0
- package/ai/chat/fetchUtils.ts +153 -0
- package/ai/chat/inlineImageUrlsForCustomProvider.ts +117 -0
- package/ai/chat/parseApiError.ts +64 -0
- package/ai/chat/parseMultilineSSE.ts +95 -0
- package/ai/chat/sendOpenAICompletionsRequest.native.ts +682 -0
- package/ai/chat/sendOpenAICompletionsRequest.ts +712 -0
- package/ai/chat/sendOpenAIResponseRequest.ts +512 -0
- package/ai/chat/shouldUseServerProxy.ts +18 -0
- package/ai/chat/sseClient.native.ts +91 -0
- package/ai/chat/sseClient.ts +67 -0
- package/ai/chat/streamReader.native.ts +31 -0
- package/ai/chat/streamReader.ts +62 -0
- package/ai/chat/updateTotalUsage.ts +72 -0
- package/ai/context/buildReferenceContext.ts +437 -0
- package/ai/context/calculateContextUsage.ts +133 -0
- package/ai/context/retention.ts +165 -0
- package/ai/context/tokenUtils.ts +78 -0
- package/ai/index.ts +1 -1
- package/ai/llm/agentCapabilities.ts +74 -0
- package/ai/llm/calculateGeminiImageTokens.ts +57 -0
- package/ai/llm/deepinfra.ts +28 -0
- package/ai/llm/fireworks.ts +68 -0
- package/ai/llm/generateRequestBody.ts +165 -0
- package/ai/llm/getModelContextWindow.ts +84 -0
- package/ai/llm/getNoloKey.ts +37 -0
- package/ai/llm/getPricing.ts +232 -0
- package/ai/llm/hooks/useModelPricing.ts +75 -0
- package/ai/llm/imagePricing.ts +66 -0
- package/ai/llm/isResponseAPIModel.ts +13 -0
- package/ai/llm/kimi.ts +18 -0
- package/ai/llm/mimo.ts +71 -0
- package/ai/llm/mistral.ts +22 -0
- package/ai/llm/modelAvatar.ts +427 -0
- package/ai/llm/models.ts +45 -0
- package/ai/llm/openrouterModels.ts +141 -0
- package/ai/llm/providers.ts +307 -0
- package/ai/llm/reasoningModels.ts +28 -0
- package/ai/llm/types.ts +59 -0
- package/ai/llm/usageRequestOptions.ts +59 -0
- package/ai/memory/capture.ts +148 -0
- package/ai/memory/consolidate.ts +104 -0
- package/ai/memory/delete.ts +147 -0
- package/ai/memory/overlay.ts +84 -0
- package/ai/memory/query.ts +38 -0
- package/ai/memory/queryShared.ts +160 -0
- package/ai/memory/rank.ts +105 -0
- package/ai/memory/recentRelationshipRecap.ts +247 -0
- package/ai/memory/remember.ts +167 -0
- package/ai/memory/runtime.ts +76 -0
- package/ai/memory/store.ts +20 -0
- package/ai/memory/storeShared.ts +76 -0
- package/ai/memory/types.ts +46 -0
- package/ai/memory/understanding.ts +349 -0
- package/ai/memory/understandingGreeting.ts +264 -0
- package/ai/messages/type.ts +20 -0
- package/ai/policy/personalizationDialog.ts +333 -0
- package/ai/policy/runtimePolicy.ts +440 -0
- package/ai/policy/selfUpdateFields.ts +48 -0
- package/ai/policy/types.ts +64 -0
- package/ai/skills/referenceRuntime.ts +274 -0
- package/ai/skills/skillDiagnostics.ts +251 -0
- package/ai/skills/skillDocBuilder.ts +139 -0
- package/ai/skills/skillDocProtocol.ts +434 -0
- package/ai/skills/skillReferenceSummary.ts +63 -0
- package/ai/skills/skillSummaryMarker.ts +26 -0
- package/ai/token/calculatePrice.ts +546 -0
- package/ai/token/db.ts +98 -0
- package/ai/token/externalToolCost.ts +321 -0
- package/ai/token/hooks/useRecords.ts +65 -0
- package/ai/token/missingUsageEstimate.ts +42 -0
- package/ai/token/modelUsageQuery.ts +252 -0
- package/ai/token/normalizeUsage.ts +84 -0
- package/ai/token/openaiImageGenerationUsage.ts +56 -0
- package/ai/token/prepareTokenUsageData.ts +88 -0
- package/ai/token/query.ts +88 -0
- package/ai/token/queryUserTokens.ts +59 -0
- package/ai/token/resolveBillingTarget.ts +52 -0
- package/ai/token/saveTokenRecord.ts +53 -0
- package/ai/token/serverDialogProjection.ts +78 -0
- package/ai/token/serverTokenWriter.ts +143 -0
- package/ai/token/stats.ts +21 -0
- package/ai/token/tokenThunks.ts +24 -0
- package/ai/token/types.ts +93 -0
- package/ai/tools/agent/agentTools.ts +176 -0
- package/ai/tools/agent/agentUpdateShared.ts +311 -0
- package/ai/tools/agent/callAgentTool.ts +139 -0
- package/ai/tools/agent/createAgentTool.ts +512 -0
- package/ai/tools/agent/createDialogTool.ts +69 -0
- package/ai/tools/agent/createSkillAgentTool.ts +62 -0
- package/ai/tools/agent/parallelBudget.ts +221 -0
- package/ai/tools/agent/presets/appBuilderPreset.ts +147 -0
- package/ai/tools/agent/runLlmTool.ts +96 -0
- package/ai/tools/agent/runStreamingAgentTool.ts +73 -0
- package/ai/tools/agent/skillAgentArgs.ts +106 -0
- package/ai/tools/agent/skillAgentPreset.ts +89 -0
- package/ai/tools/agent/streamParallelAgentsTool.ts +122 -0
- package/ai/tools/agent/updateAgentTool.ts +96 -0
- package/ai/tools/agent/updateSelfTool.ts +113 -0
- package/ai/tools/amazonProductScraperTool.ts +86 -0
- package/ai/tools/apifyActorClient.ts +45 -0
- package/ai/tools/appEditGuard.ts +372 -0
- package/ai/tools/appReadSnapshot.ts +153 -0
- package/ai/tools/appTools.ts +1549 -0
- package/ai/tools/applyEditTool.ts +256 -0
- package/ai/tools/applyLineEditsTool.ts +312 -0
- package/ai/tools/browserTools/click.ts +33 -0
- package/ai/tools/browserTools/closeSession.ts +29 -0
- package/ai/tools/browserTools/common.ts +27 -0
- package/ai/tools/browserTools/openSession.ts +48 -0
- package/ai/tools/browserTools/readContent.ts +38 -0
- package/ai/tools/browserTools/selectOption.ts +46 -0
- package/ai/tools/browserTools/typeText.ts +42 -0
- package/ai/tools/category/createCategoryTool.ts +66 -0
- package/ai/tools/category/queryContentsByCategoryTool.ts +69 -0
- package/ai/tools/category/updateContentCategoryTool.ts +75 -0
- package/ai/tools/cfBrowserTools.ts +319 -0
- package/ai/tools/cfSpeechToTextTool.ts +49 -0
- package/ai/tools/checkEnvTool.ts +65 -0
- package/ai/tools/cloudflareCrawlTool.ts +289 -0
- package/ai/tools/codeSearchTool.ts +111 -0
- package/ai/tools/codeTools.ts +101 -0
- package/ai/tools/createDocTool.ts +132 -0
- package/ai/tools/createPlanTool.ts +999 -0
- package/ai/tools/createSkillDocTool.ts +155 -0
- package/ai/tools/createWorkflowTool.ts +154 -0
- package/ai/tools/deepseekOcrTool.ts +34 -0
- package/ai/tools/delayTool.ts +31 -0
- package/ai/tools/deleteSpacesTool.ts +325 -0
- package/ai/tools/deleteSpacesToolModel.ts +159 -0
- package/ai/tools/devReloadUtils.ts +29 -0
- package/ai/tools/dialogMessageSearch.ts +137 -0
- package/ai/tools/doctorSkillTool.ts +72 -0
- package/ai/tools/ecommerceScraperTool.ts +86 -0
- package/ai/tools/emailTools.ts +549 -0
- package/ai/tools/evalSkillTool.ts +92 -0
- package/ai/tools/exaSearchTool.ts +64 -0
- package/ai/tools/execBashTool.ts +379 -0
- package/ai/tools/executeSqlTool.ts +192 -0
- package/ai/tools/fetchWebpageSupport.ts +309 -0
- package/ai/tools/fetchWebpageTool.ts +84 -0
- package/ai/tools/geminiImagePreviewTool.ts +361 -0
- package/ai/tools/generateDocxTool.ts +215 -0
- package/ai/tools/googleSearchScraperTool.ts +106 -0
- package/ai/tools/importDataTool.ts +133 -0
- package/ai/tools/importSkillTool.ts +162 -0
- package/ai/tools/index.ts +1927 -0
- package/ai/tools/listFilesTool.ts +82 -0
- package/ai/tools/listUserSpacesTool.ts +113 -0
- package/ai/tools/modelUsageTools.ts +199 -0
- package/ai/tools/olmOcrTool.ts +34 -0
- package/ai/tools/openaiImageTool.ts +267 -0
- package/ai/tools/prepareTools.ts +23 -0
- package/ai/tools/readDocTool.ts +84 -0
- package/ai/tools/readFileTool.ts +211 -0
- package/ai/tools/readTool.ts +163 -0
- package/ai/tools/readXPostTool.ts +233 -0
- package/ai/tools/rememberMemoryTool.ts +84 -0
- package/ai/tools/remotionVideoTool.ts +151 -0
- package/ai/tools/searchDialogMessagesTool.ts +222 -0
- package/ai/tools/searchRepoTool.ts +115 -0
- package/ai/tools/searchWorkspaceTool.ts +259 -0
- package/ai/tools/skillFollowup.ts +86 -0
- package/ai/tools/surfWeatherTool.ts +169 -0
- package/ai/tools/table/addTableRowTool.ts +217 -0
- package/ai/tools/table/createTableTool.ts +315 -0
- package/ai/tools/table/rowTools.ts +366 -0
- package/ai/tools/table/schemaTools.ts +244 -0
- package/ai/tools/table/shareTableTool.ts +148 -0
- package/ai/tools/table/toolShared.ts +129 -0
- package/ai/tools/toolApiClient.ts +198 -0
- package/ai/tools/toolNameAliases.ts +57 -0
- package/ai/tools/toolResultError.ts +42 -0
- package/ai/tools/toolRunSlice.ts +303 -0
- package/ai/tools/toolSchemaCompatibility.ts +53 -0
- package/ai/tools/toolVisibility.ts +4 -0
- package/ai/tools/types.ts +20 -0
- package/ai/tools/uiAskChoiceTool.ts +104 -0
- package/ai/tools/updateContentTitleTool.ts +84 -0
- package/ai/tools/updateDocTool.ts +105 -0
- package/ai/tools/updateUserPreferenceProfileTool.ts +145 -0
- package/ai/tools/whisperTool.ts +77 -0
- package/ai/tools/writeFileTool.ts +210 -0
- package/ai/tools/youtubeScraperTool.ts +116 -0
- package/ai/tools/ziweiChartTool.ts +678 -0
- package/ai/types.ts +55 -0
- package/ai/workflow/workflowExecutor.ts +323 -0
- package/ai/workflow/workflowSlice.ts +73 -0
- package/ai/workflow/workflowTypes.ts +106 -0
- package/client/agentRun.test.ts +240 -0
- package/client/agentRun.ts +182 -19
- package/client/compactDialog.test.ts +238 -0
- package/client/localRuntimeAdapter.test.ts +135 -0
- package/client/localRuntimeAdapter.ts +244 -0
- package/client/profileConfig.test.ts +40 -0
- package/client/streamingOutput.test.ts +22 -0
- package/client/streamingOutput.ts +38 -0
- package/commandRegistry.ts +9 -2
- package/connector-experimental/index.ts +5 -0
- package/database/actions/cacheMergedUserData.ts +64 -0
- package/database/actions/common.ts +242 -0
- package/database/actions/deleteFile.ts +40 -0
- package/database/actions/fetchUserData.ts +16 -0
- package/database/actions/fileContent.ts +125 -0
- package/database/actions/patch.ts +155 -0
- package/database/actions/read.ts +337 -0
- package/database/actions/readAndWait.ts +224 -0
- package/database/actions/readRequestManager.ts +120 -0
- package/database/actions/remove.ts +94 -0
- package/database/actions/replication.ts +366 -0
- package/database/actions/upload.ts +174 -0
- package/database/actions/upsert.ts +56 -0
- package/database/actions/write.ts +126 -0
- package/database/client/db.native.ts +73 -0
- package/database/client/db.ts +51 -0
- package/database/client/fetchUserData.ts +61 -0
- package/database/client/handleError.ts +19 -0
- package/database/client/queryRequest.ts +21 -0
- package/database/config.ts +21 -0
- package/database/dbActionThunks.ts +1 -0
- package/database/dbSlice.ts +149 -0
- package/database/email.ts +42 -0
- package/database/fileRing.ts +51 -0
- package/database/fileSharding.ts +70 -0
- package/database/fileStorage.native.ts +92 -0
- package/database/fileStorage.ts +232 -0
- package/database/fileUrl.ts +34 -0
- package/database/hooks/useUserData.ts +489 -0
- package/database/index.ts +1 -0
- package/database/keys.ts +765 -0
- package/database/queryPrefixes.ts +14 -0
- package/database/requests.ts +443 -0
- package/database/runtimeServerContext.ts +35 -0
- package/database/server/MemoryDB.ts +76 -0
- package/database/server/actorAccess.ts +76 -0
- package/database/server/agentDelegation.ts +124 -0
- package/database/server/coreDataOwnership.ts +13 -0
- package/database/server/coreDataProxy.ts +76 -0
- package/database/server/cybotReadonly.ts +18 -0
- package/database/server/dataHandlers.ts +111 -0
- package/database/server/db.ts +118 -0
- package/database/server/dbPath.ts +20 -0
- package/database/server/delete.ts +499 -0
- package/database/server/emailRepository.ts +1480 -0
- package/database/server/ensureDbOpen.ts +12 -0
- package/database/server/fileRead.ts +337 -0
- package/database/server/fileService.ts +436 -0
- package/database/server/handleTransaction.ts +86 -0
- package/database/server/patch.ts +282 -0
- package/database/server/query.ts +138 -0
- package/database/server/read.ts +325 -0
- package/database/server/resourceAccess.ts +211 -0
- package/database/server/routes.ts +110 -0
- package/database/server/spaceMemberAuthority.ts +67 -0
- package/database/server/upload.ts +159 -0
- package/database/server/write.ts +494 -0
- package/database/server/writeAuthority.ts +133 -0
- package/database/sqliteDb.ts +46 -0
- package/database/table/deleteTable.ts +120 -0
- package/database/tenantPlacement.ts +57 -0
- package/database/tombstones.ts +52 -0
- package/database/userDataLoadDecision.ts +17 -0
- package/database/userDataMerge.ts +95 -0
- package/database/userPreferenceRegister.ts +108 -0
- package/database/utils/dbPath.ts +47 -0
- package/database/utils/ulid.native.ts +6 -0
- package/database/utils/ulid.ts +1 -0
- package/index.ts +25 -15
- package/localRuntimeDb.ts +28 -0
- package/package.json +16 -4
- package/runtimeModeArgs.ts +33 -0
- package/tui/readlineWorkspace.ts +1 -0
- 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
|
+
}
|