@google/adk 0.3.0 → 0.4.0
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/dist/cjs/a2a/part_converter_utils.js +210 -0
- package/dist/cjs/agents/active_streaming_tool.js +1 -1
- package/dist/cjs/agents/base_agent.js +3 -3
- package/dist/cjs/agents/base_llm_processor.js +1 -1
- package/dist/cjs/agents/callback_context.js +1 -1
- package/dist/cjs/agents/content_processor_utils.js +1 -1
- package/dist/cjs/agents/functions.js +2 -1
- package/dist/cjs/agents/instructions.js +1 -1
- package/dist/cjs/agents/invocation_context.js +1 -1
- package/dist/cjs/agents/live_request_queue.js +1 -1
- package/dist/cjs/agents/llm_agent.js +58 -40
- package/dist/cjs/agents/loop_agent.js +1 -1
- package/dist/cjs/agents/parallel_agent.js +1 -1
- package/dist/cjs/agents/readonly_context.js +13 -1
- package/dist/cjs/agents/run_config.js +2 -1
- package/dist/cjs/agents/sequential_agent.js +1 -1
- package/dist/cjs/agents/transcription_entry.js +1 -1
- package/dist/cjs/artifacts/base_artifact_service.js +1 -1
- package/dist/cjs/artifacts/file_artifact_service.js +491 -0
- package/dist/cjs/artifacts/gcs_artifact_service.js +127 -48
- package/dist/cjs/artifacts/in_memory_artifact_service.js +54 -6
- package/dist/cjs/artifacts/registry.js +55 -0
- package/dist/cjs/auth/auth_credential.js +1 -1
- package/dist/cjs/auth/auth_handler.js +1 -1
- package/dist/cjs/auth/auth_schemes.js +1 -1
- package/dist/cjs/auth/auth_tool.js +1 -1
- package/dist/cjs/auth/credential_service/base_credential_service.js +1 -1
- package/dist/cjs/auth/credential_service/in_memory_credential_service.js +1 -1
- package/dist/cjs/auth/exchanger/base_credential_exchanger.js +1 -1
- package/dist/cjs/auth/exchanger/credential_exchanger_registry.js +1 -1
- package/dist/cjs/code_executors/base_code_executor.js +1 -1
- package/dist/cjs/code_executors/built_in_code_executor.js +1 -1
- package/dist/cjs/code_executors/code_execution_utils.js +1 -1
- package/dist/cjs/code_executors/code_executor_context.js +1 -1
- package/dist/cjs/common.js +14 -1
- package/dist/cjs/events/event.js +33 -4
- package/dist/cjs/events/event_actions.js +2 -2
- package/dist/cjs/events/structured_events.js +105 -0
- package/dist/cjs/examples/base_example_provider.js +1 -1
- package/dist/cjs/examples/example.js +1 -1
- package/dist/cjs/examples/example_util.js +1 -1
- package/dist/cjs/index.js +54 -83
- package/dist/cjs/index_web.js +1 -1
- package/dist/cjs/memory/base_memory_service.js +1 -1
- package/dist/cjs/memory/in_memory_memory_service.js +1 -1
- package/dist/cjs/memory/memory_entry.js +1 -1
- package/dist/cjs/models/apigee_llm.js +182 -0
- package/dist/cjs/models/base_llm.js +1 -1
- package/dist/cjs/models/base_llm_connection.js +1 -1
- package/dist/cjs/models/gemini_llm_connection.js +1 -1
- package/dist/cjs/models/google_llm.js +70 -51
- package/dist/cjs/models/llm_request.js +1 -1
- package/dist/cjs/models/llm_response.js +1 -1
- package/dist/cjs/models/registry.js +3 -1
- package/dist/cjs/plugins/base_plugin.js +1 -1
- package/dist/cjs/plugins/logging_plugin.js +1 -1
- package/dist/cjs/plugins/plugin_manager.js +1 -1
- package/dist/cjs/plugins/security_plugin.js +1 -1
- package/dist/cjs/runner/in_memory_runner.js +1 -1
- package/dist/cjs/runner/runner.js +32 -1
- package/dist/cjs/sessions/base_session_service.js +53 -3
- package/dist/cjs/sessions/database_session_service.js +364 -0
- package/dist/cjs/sessions/db/operations.js +114 -0
- package/dist/cjs/sessions/db/schema.js +204 -0
- package/dist/cjs/sessions/in_memory_session_service.js +24 -22
- package/dist/cjs/sessions/registry.js +49 -0
- package/dist/cjs/sessions/session.js +1 -1
- package/dist/cjs/sessions/state.js +1 -1
- package/dist/cjs/telemetry/google_cloud.js +1 -1
- package/dist/cjs/telemetry/setup.js +1 -1
- package/dist/cjs/telemetry/tracing.js +1 -1
- package/dist/cjs/tools/agent_tool.js +1 -1
- package/dist/cjs/tools/base_tool.js +1 -1
- package/dist/cjs/tools/base_toolset.js +1 -1
- package/dist/cjs/tools/forwarding_artifact_service.js +17 -1
- package/dist/cjs/tools/function_tool.js +1 -1
- package/dist/cjs/tools/google_search_tool.js +1 -1
- package/dist/cjs/tools/long_running_tool.js +1 -1
- package/dist/cjs/tools/mcp/mcp_session_manager.js +1 -1
- package/dist/cjs/tools/mcp/mcp_tool.js +1 -1
- package/dist/cjs/tools/mcp/mcp_toolset.js +1 -1
- package/dist/cjs/tools/tool_confirmation.js +1 -1
- package/dist/cjs/tools/tool_context.js +1 -1
- package/dist/cjs/utils/client_labels.js +1 -1
- package/dist/cjs/utils/env_aware_utils.js +10 -1
- package/dist/cjs/utils/gemini_schema_util.js +1 -1
- package/dist/cjs/utils/logger.js +1 -1
- package/dist/cjs/utils/model_name.js +1 -1
- package/dist/cjs/utils/object_notation_utils.js +78 -0
- package/dist/cjs/utils/simple_zod_to_json.js +1 -1
- package/dist/cjs/utils/variant_utils.js +3 -9
- package/dist/cjs/version.js +2 -2
- package/dist/esm/a2a/part_converter_utils.js +171 -0
- package/dist/esm/agents/base_agent.js +2 -2
- package/dist/esm/agents/functions.js +1 -0
- package/dist/esm/agents/llm_agent.js +58 -40
- package/dist/esm/agents/readonly_context.js +12 -0
- package/dist/esm/agents/run_config.js +1 -0
- package/dist/esm/artifacts/file_artifact_service.js +451 -0
- package/dist/esm/artifacts/gcs_artifact_service.js +126 -47
- package/dist/esm/artifacts/in_memory_artifact_service.js +51 -4
- package/dist/esm/artifacts/registry.js +28 -0
- package/dist/esm/common.js +9 -1
- package/dist/esm/events/event.js +29 -2
- package/dist/esm/events/event_actions.js +1 -1
- package/dist/esm/events/structured_events.js +74 -0
- package/dist/esm/index.js +18 -88
- package/dist/esm/models/apigee_llm.js +152 -0
- package/dist/esm/models/google_llm.js +67 -49
- package/dist/esm/models/registry.js +2 -0
- package/dist/esm/runner/runner.js +31 -0
- package/dist/esm/sessions/base_session_service.js +49 -1
- package/dist/esm/sessions/database_session_service.js +350 -0
- package/dist/esm/sessions/db/operations.js +87 -0
- package/dist/esm/sessions/db/schema.js +172 -0
- package/dist/esm/sessions/in_memory_session_service.js +23 -21
- package/dist/esm/sessions/registry.js +25 -0
- package/dist/esm/tools/forwarding_artifact_service.js +16 -0
- package/dist/esm/utils/env_aware_utils.js +8 -0
- package/dist/esm/utils/object_notation_utils.js +47 -0
- package/dist/esm/utils/variant_utils.js +1 -7
- package/dist/esm/version.js +1 -1
- package/dist/types/a2a/part_converter_utils.d.ts +47 -0
- package/dist/types/agents/llm_agent.d.ts +11 -11
- package/dist/types/agents/readonly_context.d.ts +8 -0
- package/dist/types/agents/run_config.d.ts +6 -0
- package/dist/types/artifacts/base_artifact_service.d.ts +31 -0
- package/dist/types/artifacts/file_artifact_service.d.ts +43 -0
- package/dist/types/artifacts/gcs_artifact_service.d.ts +3 -1
- package/dist/types/artifacts/in_memory_artifact_service.d.ts +5 -2
- package/dist/types/artifacts/registry.d.ts +7 -0
- package/dist/types/common.d.ts +11 -2
- package/dist/types/events/event.d.ts +15 -1
- package/dist/types/events/event_actions.d.ts +1 -1
- package/dist/types/events/structured_events.d.ts +106 -0
- package/dist/types/index.d.ts +5 -1
- package/dist/types/models/apigee_llm.d.ts +59 -0
- package/dist/types/models/google_llm.d.ts +5 -2
- package/dist/types/runner/runner.d.ts +15 -0
- package/dist/types/sessions/base_session_service.d.ts +20 -0
- package/dist/types/sessions/database_session_service.d.ts +31 -0
- package/dist/types/sessions/db/operations.d.ts +29 -0
- package/dist/types/sessions/db/schema.d.ts +45 -0
- package/dist/types/sessions/in_memory_session_service.d.ts +4 -1
- package/dist/types/sessions/registry.d.ts +7 -0
- package/dist/types/tools/forwarding_artifact_service.d.ts +3 -1
- package/dist/types/utils/env_aware_utils.d.ts +7 -0
- package/dist/types/utils/object_notation_utils.d.ts +21 -0
- package/dist/types/version.d.ts +1 -1
- package/dist/web/a2a/part_converter_utils.js +171 -0
- package/dist/web/agents/base_agent.js +2 -2
- package/dist/web/agents/functions.js +1 -0
- package/dist/web/agents/llm_agent.js +79 -59
- package/dist/web/agents/readonly_context.js +12 -0
- package/dist/web/agents/run_config.js +2 -1
- package/dist/web/artifacts/file_artifact_service.js +506 -0
- package/dist/web/artifacts/gcs_artifact_service.js +123 -46
- package/dist/web/artifacts/in_memory_artifact_service.js +51 -4
- package/dist/web/artifacts/registry.js +28 -0
- package/dist/web/common.js +9 -1
- package/dist/web/events/event.js +29 -2
- package/dist/web/events/event_actions.js +1 -1
- package/dist/web/events/structured_events.js +74 -0
- package/dist/web/index.js +18 -8
- package/dist/web/models/apigee_llm.js +219 -0
- package/dist/web/models/google_llm.js +67 -46
- package/dist/web/models/registry.js +2 -0
- package/dist/web/runner/runner.js +33 -0
- package/dist/web/sessions/base_session_service.js +49 -1
- package/dist/web/sessions/database_session_service.js +368 -0
- package/dist/web/sessions/db/operations.js +87 -0
- package/dist/web/sessions/db/schema.js +172 -0
- package/dist/web/sessions/in_memory_session_service.js +23 -21
- package/dist/web/sessions/registry.js +25 -0
- package/dist/web/tools/forwarding_artifact_service.js +16 -0
- package/dist/web/utils/env_aware_utils.js +8 -0
- package/dist/web/utils/object_notation_utils.js +47 -0
- package/dist/web/utils/variant_utils.js +1 -7
- package/dist/web/version.js +1 -1
- package/package.json +13 -4
- package/dist/cjs/index.js.map +0 -7
- package/dist/esm/index.js.map +0 -7
- package/dist/web/index.js.map +0 -7
|
@@ -0,0 +1,491 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright 2026 Google LLC
|
|
4
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
"use strict";
|
|
8
|
+
var __create = Object.create;
|
|
9
|
+
var __defProp = Object.defineProperty;
|
|
10
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
11
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
12
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
13
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
14
|
+
var __export = (target, all) => {
|
|
15
|
+
for (var name in all)
|
|
16
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
17
|
+
};
|
|
18
|
+
var __copyProps = (to, from, except, desc) => {
|
|
19
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
20
|
+
for (let key of __getOwnPropNames(from))
|
|
21
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
22
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
23
|
+
}
|
|
24
|
+
return to;
|
|
25
|
+
};
|
|
26
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
27
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
|
28
|
+
// file that has been converted to a CommonJS file using a Babel-
|
|
29
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
30
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
31
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
32
|
+
mod
|
|
33
|
+
));
|
|
34
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
35
|
+
var file_artifact_service_exports = {};
|
|
36
|
+
__export(file_artifact_service_exports, {
|
|
37
|
+
FileArtifactService: () => FileArtifactService
|
|
38
|
+
});
|
|
39
|
+
module.exports = __toCommonJS(file_artifact_service_exports);
|
|
40
|
+
var fs = __toESM(require("fs/promises"), 1);
|
|
41
|
+
var path = __toESM(require("path"), 1);
|
|
42
|
+
var import_url = require("url");
|
|
43
|
+
var import_logger = require("../utils/logger.js");
|
|
44
|
+
/**
|
|
45
|
+
* @license
|
|
46
|
+
* Copyright 2026 Google LLC
|
|
47
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
48
|
+
*/
|
|
49
|
+
const USER_NAMESPACE_PREFIX = "user:";
|
|
50
|
+
class FileArtifactService {
|
|
51
|
+
constructor(rootDirOrUri) {
|
|
52
|
+
try {
|
|
53
|
+
const rootDir = rootDirOrUri.startsWith("file://") ? (0, import_url.fileURLToPath)(rootDirOrUri) : rootDirOrUri;
|
|
54
|
+
this.rootDir = path.resolve(rootDir);
|
|
55
|
+
} catch (e) {
|
|
56
|
+
throw new Error(`Invalid root directory: ${rootDirOrUri}`, { cause: e });
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
async saveArtifact({
|
|
60
|
+
userId,
|
|
61
|
+
sessionId,
|
|
62
|
+
filename,
|
|
63
|
+
artifact,
|
|
64
|
+
customMetadata
|
|
65
|
+
}) {
|
|
66
|
+
if (!artifact.inlineData && !artifact.text) {
|
|
67
|
+
throw new Error("Artifact must have either inlineData or text content.");
|
|
68
|
+
}
|
|
69
|
+
const artifactDir = getArtifactDir(
|
|
70
|
+
this.rootDir,
|
|
71
|
+
userId,
|
|
72
|
+
sessionId,
|
|
73
|
+
filename
|
|
74
|
+
);
|
|
75
|
+
await fs.mkdir(artifactDir, { recursive: true });
|
|
76
|
+
const versions = await getArtifactVersionsFromDir(artifactDir);
|
|
77
|
+
const nextVersion = versions.length > 0 ? versions[versions.length - 1] + 1 : 0;
|
|
78
|
+
const versionsDir = getVersionsDir(artifactDir);
|
|
79
|
+
const versionDir = path.join(versionsDir, nextVersion.toString());
|
|
80
|
+
await fs.mkdir(versionDir, { recursive: true });
|
|
81
|
+
const storedFilename = path.basename(artifactDir);
|
|
82
|
+
const contentPath = path.join(versionDir, storedFilename);
|
|
83
|
+
let mimeType;
|
|
84
|
+
if (artifact.inlineData) {
|
|
85
|
+
const data = artifact.inlineData.data || "";
|
|
86
|
+
await fs.writeFile(contentPath, Buffer.from(data, "base64"));
|
|
87
|
+
mimeType = artifact.inlineData.mimeType || "application/octet-stream";
|
|
88
|
+
} else if (artifact.text !== void 0) {
|
|
89
|
+
await fs.writeFile(contentPath, artifact.text, "utf-8");
|
|
90
|
+
}
|
|
91
|
+
const canonicalUri = await getCanonicalUri(
|
|
92
|
+
this.rootDir,
|
|
93
|
+
userId,
|
|
94
|
+
sessionId,
|
|
95
|
+
filename,
|
|
96
|
+
nextVersion
|
|
97
|
+
);
|
|
98
|
+
const metadata = {
|
|
99
|
+
fileName: filename,
|
|
100
|
+
mimeType,
|
|
101
|
+
version: nextVersion,
|
|
102
|
+
canonicalUri,
|
|
103
|
+
customMetadata
|
|
104
|
+
};
|
|
105
|
+
await writeMetadata(path.join(versionDir, "metadata.json"), metadata);
|
|
106
|
+
return nextVersion;
|
|
107
|
+
}
|
|
108
|
+
async loadArtifact({
|
|
109
|
+
userId,
|
|
110
|
+
sessionId,
|
|
111
|
+
filename,
|
|
112
|
+
version
|
|
113
|
+
}) {
|
|
114
|
+
try {
|
|
115
|
+
const artifactDir = getArtifactDir(
|
|
116
|
+
this.rootDir,
|
|
117
|
+
userId,
|
|
118
|
+
sessionId,
|
|
119
|
+
filename
|
|
120
|
+
);
|
|
121
|
+
try {
|
|
122
|
+
await fs.access(artifactDir);
|
|
123
|
+
} catch (e) {
|
|
124
|
+
import_logger.logger.warn(
|
|
125
|
+
`[FileArtifactService] loadArtifact: Artifact ${filename} not found`,
|
|
126
|
+
e
|
|
127
|
+
);
|
|
128
|
+
return void 0;
|
|
129
|
+
}
|
|
130
|
+
const versions = await getArtifactVersionsFromDir(artifactDir);
|
|
131
|
+
if (versions.length === 0) {
|
|
132
|
+
return void 0;
|
|
133
|
+
}
|
|
134
|
+
let versionToLoad;
|
|
135
|
+
if (version === void 0) {
|
|
136
|
+
versionToLoad = versions[versions.length - 1];
|
|
137
|
+
} else {
|
|
138
|
+
if (!versions.includes(version)) {
|
|
139
|
+
import_logger.logger.warn(
|
|
140
|
+
`[FileArtifactService] loadArtifact: Artifact ${filename} version ${version} not found`
|
|
141
|
+
);
|
|
142
|
+
return void 0;
|
|
143
|
+
}
|
|
144
|
+
versionToLoad = version;
|
|
145
|
+
}
|
|
146
|
+
const versionDir = path.join(
|
|
147
|
+
getVersionsDir(artifactDir),
|
|
148
|
+
versionToLoad.toString()
|
|
149
|
+
);
|
|
150
|
+
const metadataPath = path.join(versionDir, "metadata.json");
|
|
151
|
+
const metadata = await readMetadata(metadataPath);
|
|
152
|
+
const storedFilename = path.basename(artifactDir);
|
|
153
|
+
let contentPath = path.join(versionDir, storedFilename);
|
|
154
|
+
if (metadata.canonicalUri) {
|
|
155
|
+
const uriPath = fileUriToPath(metadata.canonicalUri);
|
|
156
|
+
if (uriPath) {
|
|
157
|
+
try {
|
|
158
|
+
await fs.access(uriPath);
|
|
159
|
+
contentPath = uriPath;
|
|
160
|
+
} catch {
|
|
161
|
+
import_logger.logger.warn(
|
|
162
|
+
`[FileArtifactService] loadArtifact: Artifact ${filename} missing at ${uriPath}, falling back to content path ${contentPath}`
|
|
163
|
+
);
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
if (metadata.mimeType) {
|
|
168
|
+
try {
|
|
169
|
+
const data = await fs.readFile(contentPath);
|
|
170
|
+
return {
|
|
171
|
+
inlineData: {
|
|
172
|
+
mimeType: metadata.mimeType,
|
|
173
|
+
data: data.toString("base64")
|
|
174
|
+
}
|
|
175
|
+
};
|
|
176
|
+
} catch {
|
|
177
|
+
import_logger.logger.warn(
|
|
178
|
+
`[FileArtifactService] loadArtifact: Artifact ${filename} missing at ${contentPath}`
|
|
179
|
+
);
|
|
180
|
+
return void 0;
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
try {
|
|
184
|
+
const text = await fs.readFile(contentPath, "utf-8");
|
|
185
|
+
return { text };
|
|
186
|
+
} catch {
|
|
187
|
+
import_logger.logger.warn(
|
|
188
|
+
`[FileArtifactService] loadArtifact: Text artifact ${filename} missing at ${contentPath}`
|
|
189
|
+
);
|
|
190
|
+
return void 0;
|
|
191
|
+
}
|
|
192
|
+
} catch (e) {
|
|
193
|
+
import_logger.logger.error(
|
|
194
|
+
`[FileArtifactService] loadArtifact: Error loading artifact ${filename}`,
|
|
195
|
+
e
|
|
196
|
+
);
|
|
197
|
+
return void 0;
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
async listArtifactKeys({
|
|
201
|
+
userId,
|
|
202
|
+
sessionId
|
|
203
|
+
}) {
|
|
204
|
+
const filenames = /* @__PURE__ */ new Set();
|
|
205
|
+
const userRoot = getUserRoot(this.rootDir, userId);
|
|
206
|
+
const sessionRoot = getSessionArtifactsDir(userRoot, sessionId);
|
|
207
|
+
for await (const artifactDir of iterateArtifactDirs(sessionRoot)) {
|
|
208
|
+
const metadata = await getLatestMetadata(artifactDir);
|
|
209
|
+
if (metadata == null ? void 0 : metadata.fileName) {
|
|
210
|
+
filenames.add(metadata.fileName);
|
|
211
|
+
} else {
|
|
212
|
+
const rel = path.relative(sessionRoot, artifactDir);
|
|
213
|
+
filenames.add(asPosixPath(rel));
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
const artifactsRoot = getUserArtifactsDir(userRoot);
|
|
217
|
+
for await (const artifactDir of iterateArtifactDirs(artifactsRoot)) {
|
|
218
|
+
const metadata = await getLatestMetadata(artifactDir);
|
|
219
|
+
if (metadata == null ? void 0 : metadata.fileName) {
|
|
220
|
+
filenames.add(metadata.fileName);
|
|
221
|
+
} else {
|
|
222
|
+
const rel = path.relative(artifactsRoot, artifactDir);
|
|
223
|
+
filenames.add(`${USER_NAMESPACE_PREFIX}${asPosixPath(rel)}`);
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
return Array.from(filenames).sort();
|
|
227
|
+
}
|
|
228
|
+
async deleteArtifact({
|
|
229
|
+
userId,
|
|
230
|
+
sessionId,
|
|
231
|
+
filename
|
|
232
|
+
}) {
|
|
233
|
+
try {
|
|
234
|
+
const artifactDir = getArtifactDir(
|
|
235
|
+
this.rootDir,
|
|
236
|
+
userId,
|
|
237
|
+
sessionId,
|
|
238
|
+
filename
|
|
239
|
+
);
|
|
240
|
+
await fs.rm(artifactDir, { recursive: true, force: true });
|
|
241
|
+
} catch (e) {
|
|
242
|
+
import_logger.logger.warn(
|
|
243
|
+
`[FileArtifactService] deleteArtifact: Failed to delete artifact ${filename}`,
|
|
244
|
+
e
|
|
245
|
+
);
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
async listVersions({
|
|
249
|
+
userId,
|
|
250
|
+
sessionId,
|
|
251
|
+
filename
|
|
252
|
+
}) {
|
|
253
|
+
try {
|
|
254
|
+
const artifactDir = getArtifactDir(
|
|
255
|
+
this.rootDir,
|
|
256
|
+
userId,
|
|
257
|
+
sessionId,
|
|
258
|
+
filename
|
|
259
|
+
);
|
|
260
|
+
return await getArtifactVersionsFromDir(artifactDir);
|
|
261
|
+
} catch (e) {
|
|
262
|
+
import_logger.logger.warn(
|
|
263
|
+
`[FileArtifactService] listVersions: Failed to list versions for artifact ${filename}`,
|
|
264
|
+
e
|
|
265
|
+
);
|
|
266
|
+
return [];
|
|
267
|
+
}
|
|
268
|
+
}
|
|
269
|
+
async listArtifactVersions({
|
|
270
|
+
userId,
|
|
271
|
+
sessionId,
|
|
272
|
+
filename
|
|
273
|
+
}) {
|
|
274
|
+
try {
|
|
275
|
+
const artifactDir = getArtifactDir(
|
|
276
|
+
this.rootDir,
|
|
277
|
+
userId,
|
|
278
|
+
sessionId,
|
|
279
|
+
filename
|
|
280
|
+
);
|
|
281
|
+
const versions = await getArtifactVersionsFromDir(artifactDir);
|
|
282
|
+
const artifactVersions = [];
|
|
283
|
+
for (const version of versions) {
|
|
284
|
+
const metadataPath = path.join(
|
|
285
|
+
getVersionsDir(artifactDir),
|
|
286
|
+
version.toString(),
|
|
287
|
+
"metadata.json"
|
|
288
|
+
);
|
|
289
|
+
try {
|
|
290
|
+
const metadata = await readMetadata(metadataPath);
|
|
291
|
+
artifactVersions.push(metadata);
|
|
292
|
+
} catch (e) {
|
|
293
|
+
import_logger.logger.warn(
|
|
294
|
+
`[FileArtifactService] listArtifactVersions: Failed to read artifact version ${version} at ${artifactDir}`,
|
|
295
|
+
e
|
|
296
|
+
);
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
return artifactVersions;
|
|
300
|
+
} catch (e) {
|
|
301
|
+
import_logger.logger.warn(
|
|
302
|
+
`[FileArtifactService] listArtifactVersions: Failed to list artifact versions for userId: ${userId} sessionId: ${sessionId} filename: ${filename}`,
|
|
303
|
+
e
|
|
304
|
+
);
|
|
305
|
+
return [];
|
|
306
|
+
}
|
|
307
|
+
}
|
|
308
|
+
async getArtifactVersion({
|
|
309
|
+
userId,
|
|
310
|
+
sessionId,
|
|
311
|
+
filename,
|
|
312
|
+
version
|
|
313
|
+
}) {
|
|
314
|
+
try {
|
|
315
|
+
const artifactDir = getArtifactDir(
|
|
316
|
+
this.rootDir,
|
|
317
|
+
userId,
|
|
318
|
+
sessionId,
|
|
319
|
+
filename
|
|
320
|
+
);
|
|
321
|
+
const versions = await getArtifactVersionsFromDir(artifactDir);
|
|
322
|
+
if (versions.length === 0) {
|
|
323
|
+
return void 0;
|
|
324
|
+
}
|
|
325
|
+
let versionToRead;
|
|
326
|
+
if (version === void 0) {
|
|
327
|
+
versionToRead = versions[versions.length - 1];
|
|
328
|
+
} else {
|
|
329
|
+
if (!versions.includes(version)) {
|
|
330
|
+
return void 0;
|
|
331
|
+
}
|
|
332
|
+
versionToRead = version;
|
|
333
|
+
}
|
|
334
|
+
const metadataPath = path.join(
|
|
335
|
+
getVersionsDir(artifactDir),
|
|
336
|
+
versionToRead.toString(),
|
|
337
|
+
"metadata.json"
|
|
338
|
+
);
|
|
339
|
+
return await readMetadata(metadataPath);
|
|
340
|
+
} catch (e) {
|
|
341
|
+
import_logger.logger.warn(
|
|
342
|
+
`[FileArtifactService] getArtifactVersion: Failed to get artifact version for userId: ${userId} sessionId: ${sessionId} filename: ${filename} version: ${version}`,
|
|
343
|
+
e
|
|
344
|
+
);
|
|
345
|
+
return void 0;
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
function getUserRoot(rootDir, userId) {
|
|
350
|
+
return path.join(rootDir, "users", userId);
|
|
351
|
+
}
|
|
352
|
+
function isUserScoped(sessionId, filename) {
|
|
353
|
+
return !sessionId || filename.startsWith(USER_NAMESPACE_PREFIX);
|
|
354
|
+
}
|
|
355
|
+
function getUserArtifactsDir(userRoot) {
|
|
356
|
+
return path.join(userRoot, "artifacts");
|
|
357
|
+
}
|
|
358
|
+
function getSessionArtifactsDir(baseRoot, sessionId) {
|
|
359
|
+
return path.join(baseRoot, "sessions", sessionId, "artifacts");
|
|
360
|
+
}
|
|
361
|
+
function getVersionsDir(artifactDir) {
|
|
362
|
+
return path.join(artifactDir, "versions");
|
|
363
|
+
}
|
|
364
|
+
function getArtifactDir(rootDir, userId, sessionId, filename) {
|
|
365
|
+
const userRoot = getUserRoot(rootDir, userId);
|
|
366
|
+
let scopeRoot;
|
|
367
|
+
if (isUserScoped(sessionId, filename)) {
|
|
368
|
+
scopeRoot = getUserArtifactsDir(userRoot);
|
|
369
|
+
} else {
|
|
370
|
+
if (!sessionId) {
|
|
371
|
+
throw new Error(
|
|
372
|
+
"Session ID must be provided for session-scoped artifacts."
|
|
373
|
+
);
|
|
374
|
+
}
|
|
375
|
+
scopeRoot = getSessionArtifactsDir(userRoot, sessionId);
|
|
376
|
+
}
|
|
377
|
+
let cleanFilename = filename;
|
|
378
|
+
if (cleanFilename.startsWith(USER_NAMESPACE_PREFIX)) {
|
|
379
|
+
cleanFilename = cleanFilename.substring(USER_NAMESPACE_PREFIX.length);
|
|
380
|
+
}
|
|
381
|
+
cleanFilename = cleanFilename.trim();
|
|
382
|
+
if (path.isAbsolute(cleanFilename)) {
|
|
383
|
+
throw new Error(`Absolute artifact filename ${filename} is not permitted.`);
|
|
384
|
+
}
|
|
385
|
+
const artifactDir = path.resolve(scopeRoot, cleanFilename);
|
|
386
|
+
const relative = path.relative(scopeRoot, artifactDir);
|
|
387
|
+
if (relative.startsWith("..") || path.isAbsolute(relative)) {
|
|
388
|
+
throw new Error(`Artifact filename ${filename} escapes storage directory.`);
|
|
389
|
+
}
|
|
390
|
+
if (relative === "" || relative === ".") {
|
|
391
|
+
return path.join(scopeRoot, "artifact");
|
|
392
|
+
}
|
|
393
|
+
return artifactDir;
|
|
394
|
+
}
|
|
395
|
+
async function getArtifactVersionsFromDir(artifactDir) {
|
|
396
|
+
const versionsDir = getVersionsDir(artifactDir);
|
|
397
|
+
try {
|
|
398
|
+
const files = await fs.readdir(versionsDir, { withFileTypes: true });
|
|
399
|
+
const versions = files.filter((dirent) => dirent.isDirectory()).map((dirent) => parseInt(dirent.name, 10)).filter((v) => !isNaN(v));
|
|
400
|
+
return versions.sort((a, b) => a - b);
|
|
401
|
+
} catch (e) {
|
|
402
|
+
import_logger.logger.warn(
|
|
403
|
+
`[FileArtifactService] getArtifactVersionsFromDir: Failed to list artifact versions from ${artifactDir}`,
|
|
404
|
+
e
|
|
405
|
+
);
|
|
406
|
+
return [];
|
|
407
|
+
}
|
|
408
|
+
}
|
|
409
|
+
async function getCanonicalUri(rootDir, userId, sessionId, filename, version) {
|
|
410
|
+
const artifactDir = await getArtifactDir(
|
|
411
|
+
rootDir,
|
|
412
|
+
userId,
|
|
413
|
+
sessionId,
|
|
414
|
+
filename
|
|
415
|
+
);
|
|
416
|
+
const storedFilename = path.basename(artifactDir);
|
|
417
|
+
const versionsDir = getVersionsDir(artifactDir);
|
|
418
|
+
const payloadPath = path.join(
|
|
419
|
+
versionsDir,
|
|
420
|
+
version.toString(),
|
|
421
|
+
storedFilename
|
|
422
|
+
);
|
|
423
|
+
return (0, import_url.pathToFileURL)(payloadPath).toString();
|
|
424
|
+
}
|
|
425
|
+
async function writeMetadata(metadataPath, metadata) {
|
|
426
|
+
await fs.writeFile(metadataPath, JSON.stringify(metadata, null, 2), "utf-8");
|
|
427
|
+
}
|
|
428
|
+
async function readMetadata(metadataPath) {
|
|
429
|
+
const content = await fs.readFile(metadataPath, "utf-8");
|
|
430
|
+
return JSON.parse(content);
|
|
431
|
+
}
|
|
432
|
+
async function getLatestMetadata(artifactDir) {
|
|
433
|
+
const versions = await getArtifactVersionsFromDir(artifactDir);
|
|
434
|
+
if (versions.length === 0) {
|
|
435
|
+
return void 0;
|
|
436
|
+
}
|
|
437
|
+
const latestVersion = versions[versions.length - 1];
|
|
438
|
+
const metadataPath = path.join(
|
|
439
|
+
getVersionsDir(artifactDir),
|
|
440
|
+
latestVersion.toString(),
|
|
441
|
+
"metadata.json"
|
|
442
|
+
);
|
|
443
|
+
try {
|
|
444
|
+
return await readMetadata(metadataPath);
|
|
445
|
+
} catch (e) {
|
|
446
|
+
import_logger.logger.warn(
|
|
447
|
+
`[FileArtifactService] getLatestMetadata: Failed to read metadata from ${metadataPath}`,
|
|
448
|
+
e
|
|
449
|
+
);
|
|
450
|
+
return void 0;
|
|
451
|
+
}
|
|
452
|
+
}
|
|
453
|
+
async function* iterateArtifactDirs(dir) {
|
|
454
|
+
try {
|
|
455
|
+
const entries = await fs.readdir(dir, { withFileTypes: true });
|
|
456
|
+
const hasVersions = entries.some(
|
|
457
|
+
(e) => e.isDirectory() && e.name === "versions"
|
|
458
|
+
);
|
|
459
|
+
if (hasVersions) {
|
|
460
|
+
yield dir;
|
|
461
|
+
return;
|
|
462
|
+
}
|
|
463
|
+
for (const entry of entries) {
|
|
464
|
+
if (entry.isDirectory()) {
|
|
465
|
+
const subdir = path.join(dir, entry.name);
|
|
466
|
+
for await (const foundDir of iterateArtifactDirs(subdir)) {
|
|
467
|
+
yield foundDir;
|
|
468
|
+
}
|
|
469
|
+
}
|
|
470
|
+
}
|
|
471
|
+
} catch (_e) {
|
|
472
|
+
}
|
|
473
|
+
}
|
|
474
|
+
function fileUriToPath(uri) {
|
|
475
|
+
try {
|
|
476
|
+
return (0, import_url.fileURLToPath)(uri);
|
|
477
|
+
} catch (e) {
|
|
478
|
+
import_logger.logger.warn(
|
|
479
|
+
`[FileArtifactService] fileUriToPath: Failed to convert file URI to path: ${uri}`,
|
|
480
|
+
e
|
|
481
|
+
);
|
|
482
|
+
return void 0;
|
|
483
|
+
}
|
|
484
|
+
}
|
|
485
|
+
function asPosixPath(p) {
|
|
486
|
+
return p.split(path.sep).join("/");
|
|
487
|
+
}
|
|
488
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
489
|
+
0 && (module.exports = {
|
|
490
|
+
FileArtifactService
|
|
491
|
+
});
|