@elasticdash/otel 0.0.1 → 0.0.3
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/LICENSE +24 -0
- package/README.md +2 -2
- package/dist/index.cjs +507 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.cts +234 -0
- package/dist/index.d.ts +234 -0
- package/dist/index.mjs +496 -0
- package/dist/index.mjs.map +1 -0
- package/package.json +11 -11
package/LICENSE
ADDED
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
This repository is licensed under the MIT License. Portions of this codebase are derived from posthog/posthog-js-lite by PostHog, which is also licensed under the MIT License.
|
|
4
|
+
|
|
5
|
+
Copyright (c) 2023-2026 Langfuse GmbH
|
|
6
|
+
Copyright (c) 2022 PostHog (part of Hiberly Inc)
|
|
7
|
+
|
|
8
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
9
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
10
|
+
in the Software without restriction, including without limitation the rights
|
|
11
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
12
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
13
|
+
furnished to do so, subject to the following conditions:
|
|
14
|
+
|
|
15
|
+
The above copyright notice and this permission notice shall be included in all
|
|
16
|
+
copies or substantial portions of the Software.
|
|
17
|
+
|
|
18
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
19
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
20
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
21
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
22
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
23
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
24
|
+
SOFTWARE.
|
package/README.md
CHANGED
|
@@ -6,8 +6,8 @@ This is the OTEL package of the Langfuse JS SDK containing the `LangfuseSpanProc
|
|
|
6
6
|
|
|
7
7
|
## Packages
|
|
8
8
|
|
|
9
|
-
| Package
|
|
10
|
-
|
|
|
9
|
+
| Package | NPM | Description | Environments |
|
|
10
|
+
| ---------------------------------------------- | ----------------------------------------------------------------------------------------------------------------------- | --------------------------------------------------------- | ------------ |
|
|
11
11
|
| [@elasticdash/client](./packages/client) | [](https://www.npmjs.com/package/@elasticdash/client) | Langfuse API client for universal JavaScript environments | Universal JS |
|
|
12
12
|
| [@elasticdash/tracing](./packages/tracing) | [](https://www.npmjs.com/package/@elasticdash/tracing) | Langfuse instrumentation methods based on OpenTelemetry | Node.js 20+ |
|
|
13
13
|
| [@elasticdash/otel](./packages/otel) | [](https://www.npmjs.com/package/@elasticdash/otel) | Langfuse OpenTelemetry export helpers | Node.js 20+ |
|
package/dist/index.cjs
ADDED
|
@@ -0,0 +1,507 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __defProp = Object.defineProperty;
|
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
+
var __export = (target, all) => {
|
|
7
|
+
for (var name in all)
|
|
8
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
9
|
+
};
|
|
10
|
+
var __copyProps = (to, from, except, desc) => {
|
|
11
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
12
|
+
for (let key of __getOwnPropNames(from))
|
|
13
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
14
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
15
|
+
}
|
|
16
|
+
return to;
|
|
17
|
+
};
|
|
18
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
19
|
+
|
|
20
|
+
// src/index.ts
|
|
21
|
+
var index_exports = {};
|
|
22
|
+
__export(index_exports, {
|
|
23
|
+
LangfuseSpanProcessor: () => LangfuseSpanProcessor
|
|
24
|
+
});
|
|
25
|
+
module.exports = __toCommonJS(index_exports);
|
|
26
|
+
|
|
27
|
+
// src/span-processor.ts
|
|
28
|
+
var import_core2 = require("@elasticdash/core");
|
|
29
|
+
var import_core3 = require("@opentelemetry/core");
|
|
30
|
+
var import_exporter_trace_otlp_http = require("@opentelemetry/exporter-trace-otlp-http");
|
|
31
|
+
var import_sdk_trace_base = require("@opentelemetry/sdk-trace-base");
|
|
32
|
+
|
|
33
|
+
// src/MediaService.ts
|
|
34
|
+
var import_core = require("@elasticdash/core");
|
|
35
|
+
var MediaService = class {
|
|
36
|
+
constructor(params) {
|
|
37
|
+
this.pendingMediaUploads = /* @__PURE__ */ new Set();
|
|
38
|
+
this.apiClient = params.apiClient;
|
|
39
|
+
}
|
|
40
|
+
get logger() {
|
|
41
|
+
return (0, import_core.getGlobalLogger)();
|
|
42
|
+
}
|
|
43
|
+
async flush() {
|
|
44
|
+
await Promise.all(Array.from(this.pendingMediaUploads));
|
|
45
|
+
}
|
|
46
|
+
async process(span) {
|
|
47
|
+
var _a;
|
|
48
|
+
const mediaAttributes = [
|
|
49
|
+
import_core.LangfuseOtelSpanAttributes.OBSERVATION_INPUT,
|
|
50
|
+
import_core.LangfuseOtelSpanAttributes.TRACE_INPUT,
|
|
51
|
+
import_core.LangfuseOtelSpanAttributes.OBSERVATION_OUTPUT,
|
|
52
|
+
import_core.LangfuseOtelSpanAttributes.TRACE_OUTPUT,
|
|
53
|
+
import_core.LangfuseOtelSpanAttributes.OBSERVATION_METADATA,
|
|
54
|
+
import_core.LangfuseOtelSpanAttributes.TRACE_METADATA
|
|
55
|
+
];
|
|
56
|
+
for (const mediaAttribute of mediaAttributes) {
|
|
57
|
+
const mediaRelevantAttributeKeys = Object.keys(span.attributes).filter(
|
|
58
|
+
(attributeName) => attributeName.startsWith(mediaAttribute)
|
|
59
|
+
);
|
|
60
|
+
for (const key of mediaRelevantAttributeKeys) {
|
|
61
|
+
const value = span.attributes[key];
|
|
62
|
+
if (typeof value !== "string") {
|
|
63
|
+
this.logger.warn(
|
|
64
|
+
`Span attribute ${mediaAttribute} is not a stringified object. Skipping media handling.`
|
|
65
|
+
);
|
|
66
|
+
continue;
|
|
67
|
+
}
|
|
68
|
+
let mediaReplacedValue = value;
|
|
69
|
+
const regex = /data:[^;]+;base64,[A-Za-z0-9+/]+=*/g;
|
|
70
|
+
const foundMedia = [...new Set((_a = value.match(regex)) != null ? _a : [])];
|
|
71
|
+
if (foundMedia.length === 0) continue;
|
|
72
|
+
for (const mediaDataUri of foundMedia) {
|
|
73
|
+
const media = new import_core.LangfuseMedia({
|
|
74
|
+
base64DataUri: mediaDataUri,
|
|
75
|
+
source: "base64_data_uri"
|
|
76
|
+
});
|
|
77
|
+
const langfuseMediaTag = await media.getTag();
|
|
78
|
+
if (!langfuseMediaTag) {
|
|
79
|
+
this.logger.warn(
|
|
80
|
+
"Failed to create Langfuse media tag. Skipping media item."
|
|
81
|
+
);
|
|
82
|
+
continue;
|
|
83
|
+
}
|
|
84
|
+
this.scheduleUpload({
|
|
85
|
+
span,
|
|
86
|
+
media,
|
|
87
|
+
field: mediaAttribute.includes("input") ? "input" : mediaAttribute.includes("output") ? "output" : "metadata"
|
|
88
|
+
// todo: make more robust
|
|
89
|
+
});
|
|
90
|
+
mediaReplacedValue = mediaReplacedValue.replaceAll(
|
|
91
|
+
mediaDataUri,
|
|
92
|
+
langfuseMediaTag
|
|
93
|
+
);
|
|
94
|
+
}
|
|
95
|
+
span.attributes[key] = mediaReplacedValue;
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
if (span.instrumentationScope.name === "ai") {
|
|
99
|
+
const aiSDKMediaAttributes = ["ai.prompt.messages", "ai.prompt"];
|
|
100
|
+
for (const mediaAttribute of aiSDKMediaAttributes) {
|
|
101
|
+
const value = span.attributes[mediaAttribute];
|
|
102
|
+
if (!value || typeof value !== "string") {
|
|
103
|
+
continue;
|
|
104
|
+
}
|
|
105
|
+
let mediaReplacedValue = value;
|
|
106
|
+
try {
|
|
107
|
+
const parsed = JSON.parse(value);
|
|
108
|
+
if (Array.isArray(parsed)) {
|
|
109
|
+
for (const message of parsed) {
|
|
110
|
+
if (Array.isArray(message["content"])) {
|
|
111
|
+
const contentParts = message["content"];
|
|
112
|
+
for (const part of contentParts) {
|
|
113
|
+
if (part["type"] === "file") {
|
|
114
|
+
let base64Content = null;
|
|
115
|
+
if (part["data"] != null && part["mediaType"] != null) {
|
|
116
|
+
base64Content = part["data"];
|
|
117
|
+
}
|
|
118
|
+
if (part["image"] != null && part["mediaType"] != null && !part["image"].startsWith("http")) {
|
|
119
|
+
base64Content = part["image"];
|
|
120
|
+
}
|
|
121
|
+
if (!base64Content) continue;
|
|
122
|
+
const media = new import_core.LangfuseMedia({
|
|
123
|
+
contentType: part["mediaType"],
|
|
124
|
+
contentBytes: (0, import_core.base64ToBytes)(base64Content),
|
|
125
|
+
source: "bytes"
|
|
126
|
+
});
|
|
127
|
+
const langfuseMediaTag = await media.getTag();
|
|
128
|
+
if (!langfuseMediaTag) {
|
|
129
|
+
this.logger.warn(
|
|
130
|
+
"Failed to create Langfuse media tag. Skipping media item."
|
|
131
|
+
);
|
|
132
|
+
continue;
|
|
133
|
+
}
|
|
134
|
+
this.scheduleUpload({
|
|
135
|
+
span,
|
|
136
|
+
media,
|
|
137
|
+
field: "input"
|
|
138
|
+
});
|
|
139
|
+
mediaReplacedValue = mediaReplacedValue.replaceAll(
|
|
140
|
+
base64Content,
|
|
141
|
+
langfuseMediaTag
|
|
142
|
+
);
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
span.attributes[mediaAttribute] = mediaReplacedValue;
|
|
149
|
+
} catch (err) {
|
|
150
|
+
this.logger.warn(
|
|
151
|
+
`Failed to handle media for AI SDK attribute ${mediaAttribute} for span ${span.spanContext().spanId}`,
|
|
152
|
+
err
|
|
153
|
+
);
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
scheduleUpload(params) {
|
|
159
|
+
const { span, field, media } = params;
|
|
160
|
+
const uploadPromise = this.handleUpload({
|
|
161
|
+
media,
|
|
162
|
+
traceId: span.spanContext().traceId,
|
|
163
|
+
observationId: span.spanContext().spanId,
|
|
164
|
+
field
|
|
165
|
+
}).catch((err) => {
|
|
166
|
+
this.logger.error("Media upload failed with error: ", err);
|
|
167
|
+
});
|
|
168
|
+
this.pendingMediaUploads.add(uploadPromise);
|
|
169
|
+
uploadPromise.finally(() => {
|
|
170
|
+
this.pendingMediaUploads.delete(uploadPromise);
|
|
171
|
+
});
|
|
172
|
+
}
|
|
173
|
+
async handleUpload({
|
|
174
|
+
media,
|
|
175
|
+
traceId,
|
|
176
|
+
observationId,
|
|
177
|
+
field
|
|
178
|
+
}) {
|
|
179
|
+
try {
|
|
180
|
+
const contentSha256Hash = await media.getSha256Hash();
|
|
181
|
+
if (!media.contentLength || !media._contentType || !contentSha256Hash || !media._contentBytes) {
|
|
182
|
+
return;
|
|
183
|
+
}
|
|
184
|
+
const { uploadUrl, mediaId } = await this.apiClient.media.getUploadUrl({
|
|
185
|
+
contentLength: media.contentLength,
|
|
186
|
+
traceId,
|
|
187
|
+
observationId,
|
|
188
|
+
field,
|
|
189
|
+
contentType: media._contentType,
|
|
190
|
+
sha256Hash: contentSha256Hash
|
|
191
|
+
});
|
|
192
|
+
if (!uploadUrl) {
|
|
193
|
+
this.logger.debug(
|
|
194
|
+
`Media status: Media with ID ${mediaId} already uploaded. Skipping duplicate upload.`
|
|
195
|
+
);
|
|
196
|
+
return;
|
|
197
|
+
}
|
|
198
|
+
const clientSideMediaId = await media.getId();
|
|
199
|
+
if (clientSideMediaId !== mediaId) {
|
|
200
|
+
this.logger.error(
|
|
201
|
+
`Media integrity error: Media ID mismatch between SDK (${clientSideMediaId}) and Server (${mediaId}). Upload cancelled. Please check media ID generation logic.`
|
|
202
|
+
);
|
|
203
|
+
return;
|
|
204
|
+
}
|
|
205
|
+
this.logger.debug(`Uploading media ${mediaId}...`);
|
|
206
|
+
const startTime = Date.now();
|
|
207
|
+
const uploadResponse = await this.uploadWithBackoff({
|
|
208
|
+
uploadUrl,
|
|
209
|
+
contentBytes: media._contentBytes,
|
|
210
|
+
contentType: media._contentType,
|
|
211
|
+
contentSha256Hash,
|
|
212
|
+
maxRetries: 3,
|
|
213
|
+
baseDelay: 1e3
|
|
214
|
+
});
|
|
215
|
+
if (!uploadResponse) {
|
|
216
|
+
throw Error("Media upload process failed");
|
|
217
|
+
}
|
|
218
|
+
await this.apiClient.media.patch(mediaId, {
|
|
219
|
+
uploadedAt: (/* @__PURE__ */ new Date()).toISOString(),
|
|
220
|
+
uploadHttpStatus: uploadResponse.status,
|
|
221
|
+
uploadHttpError: await uploadResponse.text(),
|
|
222
|
+
uploadTimeMs: Date.now() - startTime
|
|
223
|
+
});
|
|
224
|
+
this.logger.debug(`Media upload status reported for ${mediaId}`);
|
|
225
|
+
} catch (err) {
|
|
226
|
+
this.logger.error(`Error processing media item: ${err}`);
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
async uploadWithBackoff(params) {
|
|
230
|
+
const {
|
|
231
|
+
uploadUrl,
|
|
232
|
+
contentType,
|
|
233
|
+
contentSha256Hash,
|
|
234
|
+
contentBytes,
|
|
235
|
+
maxRetries,
|
|
236
|
+
baseDelay
|
|
237
|
+
} = params;
|
|
238
|
+
for (let attempt = 0; attempt <= maxRetries; attempt++) {
|
|
239
|
+
try {
|
|
240
|
+
let parsedHostname;
|
|
241
|
+
try {
|
|
242
|
+
parsedHostname = new URL(uploadUrl).hostname;
|
|
243
|
+
} catch {
|
|
244
|
+
parsedHostname = "";
|
|
245
|
+
}
|
|
246
|
+
const isSelfHostedGcsBucket = parsedHostname === "storage.googleapis.com" || parsedHostname.endsWith(".storage.googleapis.com");
|
|
247
|
+
const headers = isSelfHostedGcsBucket ? { "Content-Type": contentType } : {
|
|
248
|
+
"Content-Type": contentType,
|
|
249
|
+
"x-amz-checksum-sha256": contentSha256Hash,
|
|
250
|
+
"x-ms-blob-type": "BlockBlob"
|
|
251
|
+
};
|
|
252
|
+
const uploadResponse = await fetch(uploadUrl, {
|
|
253
|
+
method: "PUT",
|
|
254
|
+
body: contentBytes,
|
|
255
|
+
headers
|
|
256
|
+
});
|
|
257
|
+
if (attempt < maxRetries && uploadResponse.status !== 200 && uploadResponse.status !== 201) {
|
|
258
|
+
throw new Error(`Upload failed with status ${uploadResponse.status}`);
|
|
259
|
+
}
|
|
260
|
+
return uploadResponse;
|
|
261
|
+
} catch (e) {
|
|
262
|
+
if (attempt === maxRetries) {
|
|
263
|
+
throw e;
|
|
264
|
+
}
|
|
265
|
+
const delay = baseDelay * Math.pow(2, attempt);
|
|
266
|
+
const jitter = Math.random() * 1e3;
|
|
267
|
+
await new Promise((resolve) => setTimeout(resolve, delay + jitter));
|
|
268
|
+
}
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
};
|
|
272
|
+
|
|
273
|
+
// src/span-processor.ts
|
|
274
|
+
var LangfuseSpanProcessor = class {
|
|
275
|
+
/**
|
|
276
|
+
* Creates a new LangfuseSpanProcessor instance.
|
|
277
|
+
*
|
|
278
|
+
* @param params - Configuration parameters for the processor
|
|
279
|
+
*
|
|
280
|
+
* @example
|
|
281
|
+
* ```typescript
|
|
282
|
+
* const processor = new LangfuseSpanProcessor({
|
|
283
|
+
* publicKey: 'pk_...',
|
|
284
|
+
* secretKey: 'sk_...',
|
|
285
|
+
* environment: 'staging',
|
|
286
|
+
* flushAt: 10,
|
|
287
|
+
* flushInterval: 2,
|
|
288
|
+
* mask: ({ data }) => {
|
|
289
|
+
* // Custom masking logic
|
|
290
|
+
* return typeof data === 'string'
|
|
291
|
+
* ? data.replace(/secret_\w+/g, 'secret_***')
|
|
292
|
+
* : data;
|
|
293
|
+
* },
|
|
294
|
+
* shouldExportSpan: ({ otelSpan }) => {
|
|
295
|
+
* // Only export spans from specific services
|
|
296
|
+
* return otelSpan.name.startsWith('my-service');
|
|
297
|
+
* }
|
|
298
|
+
* });
|
|
299
|
+
* ```
|
|
300
|
+
*/
|
|
301
|
+
constructor(params) {
|
|
302
|
+
this.pendingEndedSpans = /* @__PURE__ */ new Set();
|
|
303
|
+
var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k, _l;
|
|
304
|
+
const logger = (0, import_core2.getGlobalLogger)();
|
|
305
|
+
const publicKey = (_a = params == null ? void 0 : params.publicKey) != null ? _a : (0, import_core2.getEnv)("LANGFUSE_PUBLIC_KEY");
|
|
306
|
+
const secretKey = (_b = params == null ? void 0 : params.secretKey) != null ? _b : (0, import_core2.getEnv)("LANGFUSE_SECRET_KEY");
|
|
307
|
+
const baseUrl = (_e = (_d = (_c = params == null ? void 0 : params.baseUrl) != null ? _c : (0, import_core2.getEnv)("LANGFUSE_BASE_URL")) != null ? _d : (0, import_core2.getEnv)("LANGFUSE_BASEURL")) != null ? _e : (
|
|
308
|
+
// legacy v2
|
|
309
|
+
"https://cloud.langfuse.com"
|
|
310
|
+
);
|
|
311
|
+
if (!(params == null ? void 0 : params.exporter) && !publicKey) {
|
|
312
|
+
logger.warn(
|
|
313
|
+
"No exporter configured and no public key provided in constructor or as LANGFUSE_PUBLIC_KEY env var. Span exports will fail."
|
|
314
|
+
);
|
|
315
|
+
}
|
|
316
|
+
if (!(params == null ? void 0 : params.exporter) && !secretKey) {
|
|
317
|
+
logger.warn(
|
|
318
|
+
"No exporter configured and no secret key provided in constructor or as LANGFUSE_SECRET_KEY env var. Span exports will fail."
|
|
319
|
+
);
|
|
320
|
+
}
|
|
321
|
+
const flushAt = (_f = params == null ? void 0 : params.flushAt) != null ? _f : (0, import_core2.getEnv)("LANGFUSE_FLUSH_AT");
|
|
322
|
+
const flushIntervalSeconds = (_g = params == null ? void 0 : params.flushInterval) != null ? _g : (0, import_core2.getEnv)("LANGFUSE_FLUSH_INTERVAL");
|
|
323
|
+
const authHeaderValue = (0, import_core2.base64Encode)(`${publicKey}:${secretKey}`);
|
|
324
|
+
const timeoutSeconds = (_i = params == null ? void 0 : params.timeout) != null ? _i : Number((_h = (0, import_core2.getEnv)("LANGFUSE_TIMEOUT")) != null ? _h : 5);
|
|
325
|
+
const exporter = (_j = params == null ? void 0 : params.exporter) != null ? _j : new import_exporter_trace_otlp_http.OTLPTraceExporter({
|
|
326
|
+
url: `${baseUrl}/api/public/otel/v1/traces`,
|
|
327
|
+
headers: {
|
|
328
|
+
Authorization: `Basic ${authHeaderValue}`,
|
|
329
|
+
x_langfuse_sdk_name: "javascript",
|
|
330
|
+
x_langfuse_sdk_version: import_core2.LANGFUSE_SDK_VERSION,
|
|
331
|
+
x_langfuse_public_key: publicKey != null ? publicKey : "<missing>",
|
|
332
|
+
...params == null ? void 0 : params.additionalHeaders
|
|
333
|
+
},
|
|
334
|
+
timeoutMillis: timeoutSeconds * 1e3
|
|
335
|
+
});
|
|
336
|
+
this.processor = (params == null ? void 0 : params.exportMode) === "immediate" ? new import_sdk_trace_base.SimpleSpanProcessor(exporter) : new import_sdk_trace_base.BatchSpanProcessor(exporter, {
|
|
337
|
+
maxExportBatchSize: flushAt ? Number(flushAt) : void 0,
|
|
338
|
+
scheduledDelayMillis: flushIntervalSeconds ? Number(flushIntervalSeconds) * 1e3 : void 0
|
|
339
|
+
});
|
|
340
|
+
this.publicKey = publicKey;
|
|
341
|
+
this.baseUrl = baseUrl;
|
|
342
|
+
this.environment = (_k = params == null ? void 0 : params.environment) != null ? _k : (0, import_core2.getEnv)("LANGFUSE_TRACING_ENVIRONMENT");
|
|
343
|
+
this.release = (_l = params == null ? void 0 : params.release) != null ? _l : (0, import_core2.getEnv)("LANGFUSE_RELEASE");
|
|
344
|
+
this.mask = params == null ? void 0 : params.mask;
|
|
345
|
+
this.shouldExportSpan = params == null ? void 0 : params.shouldExportSpan;
|
|
346
|
+
this.apiClient = new import_core2.LangfuseAPIClient({
|
|
347
|
+
baseUrl: this.baseUrl,
|
|
348
|
+
username: this.publicKey,
|
|
349
|
+
password: secretKey,
|
|
350
|
+
xLangfusePublicKey: this.publicKey,
|
|
351
|
+
xLangfuseSdkVersion: import_core2.LANGFUSE_SDK_VERSION,
|
|
352
|
+
xLangfuseSdkName: "javascript",
|
|
353
|
+
environment: "",
|
|
354
|
+
// noop as baseUrl is set
|
|
355
|
+
headers: params == null ? void 0 : params.additionalHeaders
|
|
356
|
+
});
|
|
357
|
+
this.mediaService = new MediaService({ apiClient: this.apiClient });
|
|
358
|
+
logger.debug("Initialized LangfuseSpanProcessor with params:", {
|
|
359
|
+
publicKey,
|
|
360
|
+
baseUrl,
|
|
361
|
+
environment: this.environment,
|
|
362
|
+
release: this.release,
|
|
363
|
+
timeoutSeconds,
|
|
364
|
+
flushAt,
|
|
365
|
+
flushIntervalSeconds
|
|
366
|
+
});
|
|
367
|
+
}
|
|
368
|
+
get logger() {
|
|
369
|
+
return (0, import_core2.getGlobalLogger)();
|
|
370
|
+
}
|
|
371
|
+
/**
|
|
372
|
+
* Called when a span is started. Adds environment, release, and propagated attributes to the span.
|
|
373
|
+
*
|
|
374
|
+
* @param span - The span that was started
|
|
375
|
+
* @param parentContext - The parent context
|
|
376
|
+
*
|
|
377
|
+
* @override
|
|
378
|
+
*/
|
|
379
|
+
onStart(span, parentContext) {
|
|
380
|
+
span.setAttributes({
|
|
381
|
+
[import_core2.LangfuseOtelSpanAttributes.ENVIRONMENT]: this.environment,
|
|
382
|
+
[import_core2.LangfuseOtelSpanAttributes.RELEASE]: this.release,
|
|
383
|
+
...(0, import_core2.getPropagatedAttributesFromContext)(parentContext)
|
|
384
|
+
});
|
|
385
|
+
return this.processor.onStart(span, parentContext);
|
|
386
|
+
}
|
|
387
|
+
/**
|
|
388
|
+
* Called when a span ends. Processes the span for export to Langfuse.
|
|
389
|
+
*
|
|
390
|
+
* This method:
|
|
391
|
+
* 1. Checks if the span should be exported using the shouldExportSpan function
|
|
392
|
+
* 2. Applies data masking to sensitive attributes
|
|
393
|
+
* 3. Handles media content extraction and upload
|
|
394
|
+
* 4. Logs span details in debug mode
|
|
395
|
+
* 5. Passes the span to the parent processor for export
|
|
396
|
+
*
|
|
397
|
+
* @param span - The span that ended
|
|
398
|
+
*
|
|
399
|
+
* @override
|
|
400
|
+
*/
|
|
401
|
+
onEnd(span) {
|
|
402
|
+
const processEndedSpanPromise = this.processEndedSpan(span).catch((err) => {
|
|
403
|
+
this.logger.error(err);
|
|
404
|
+
});
|
|
405
|
+
this.pendingEndedSpans.add(processEndedSpanPromise);
|
|
406
|
+
void processEndedSpanPromise.finally(
|
|
407
|
+
() => this.pendingEndedSpans.delete(processEndedSpanPromise)
|
|
408
|
+
);
|
|
409
|
+
}
|
|
410
|
+
async flush() {
|
|
411
|
+
await Promise.all(Array.from(this.pendingEndedSpans));
|
|
412
|
+
await this.mediaService.flush();
|
|
413
|
+
}
|
|
414
|
+
/**
|
|
415
|
+
* Forces an immediate flush of all pending spans and media uploads.
|
|
416
|
+
*
|
|
417
|
+
* @returns Promise that resolves when all pending operations are complete
|
|
418
|
+
*
|
|
419
|
+
* @override
|
|
420
|
+
*/
|
|
421
|
+
async forceFlush() {
|
|
422
|
+
await this.flush();
|
|
423
|
+
return this.processor.forceFlush();
|
|
424
|
+
}
|
|
425
|
+
/**
|
|
426
|
+
* Gracefully shuts down the processor, ensuring all pending operations are completed.
|
|
427
|
+
*
|
|
428
|
+
* @returns Promise that resolves when shutdown is complete
|
|
429
|
+
*
|
|
430
|
+
* @override
|
|
431
|
+
*/
|
|
432
|
+
async shutdown() {
|
|
433
|
+
await this.flush();
|
|
434
|
+
return this.processor.shutdown();
|
|
435
|
+
}
|
|
436
|
+
async processEndedSpan(span) {
|
|
437
|
+
var _a, _b;
|
|
438
|
+
if (this.shouldExportSpan) {
|
|
439
|
+
try {
|
|
440
|
+
if (this.shouldExportSpan({ otelSpan: span }) === false) return;
|
|
441
|
+
} catch (err) {
|
|
442
|
+
this.logger.error(
|
|
443
|
+
"ShouldExportSpan failed with error. Excluding span. Error: ",
|
|
444
|
+
err
|
|
445
|
+
);
|
|
446
|
+
return;
|
|
447
|
+
}
|
|
448
|
+
}
|
|
449
|
+
this.applyMaskInPlace(span);
|
|
450
|
+
await this.mediaService.process(span);
|
|
451
|
+
this.logger.debug(
|
|
452
|
+
`Processed span:
|
|
453
|
+
${JSON.stringify(
|
|
454
|
+
{
|
|
455
|
+
name: span.name,
|
|
456
|
+
traceId: span.spanContext().traceId,
|
|
457
|
+
spanId: span.spanContext().spanId,
|
|
458
|
+
parentSpanId: (_b = (_a = span.parentSpanContext) == null ? void 0 : _a.spanId) != null ? _b : null,
|
|
459
|
+
attributes: span.attributes,
|
|
460
|
+
startTime: new Date((0, import_core3.hrTimeToMilliseconds)(span.startTime)),
|
|
461
|
+
endTime: new Date((0, import_core3.hrTimeToMilliseconds)(span.endTime)),
|
|
462
|
+
durationMs: (0, import_core3.hrTimeToMilliseconds)(span.duration),
|
|
463
|
+
kind: span.kind,
|
|
464
|
+
status: span.status,
|
|
465
|
+
resource: span.resource.attributes,
|
|
466
|
+
instrumentationScope: span.instrumentationScope
|
|
467
|
+
},
|
|
468
|
+
null,
|
|
469
|
+
2
|
|
470
|
+
)}`
|
|
471
|
+
);
|
|
472
|
+
this.processor.onEnd(span);
|
|
473
|
+
}
|
|
474
|
+
applyMaskInPlace(span) {
|
|
475
|
+
const maskCandidates = [
|
|
476
|
+
import_core2.LangfuseOtelSpanAttributes.OBSERVATION_INPUT,
|
|
477
|
+
import_core2.LangfuseOtelSpanAttributes.TRACE_INPUT,
|
|
478
|
+
import_core2.LangfuseOtelSpanAttributes.OBSERVATION_OUTPUT,
|
|
479
|
+
import_core2.LangfuseOtelSpanAttributes.TRACE_OUTPUT,
|
|
480
|
+
import_core2.LangfuseOtelSpanAttributes.OBSERVATION_METADATA,
|
|
481
|
+
import_core2.LangfuseOtelSpanAttributes.TRACE_METADATA
|
|
482
|
+
];
|
|
483
|
+
for (const maskCandidate of maskCandidates) {
|
|
484
|
+
if (maskCandidate in span.attributes) {
|
|
485
|
+
span.attributes[maskCandidate] = this.applyMask(
|
|
486
|
+
span.attributes[maskCandidate]
|
|
487
|
+
);
|
|
488
|
+
}
|
|
489
|
+
}
|
|
490
|
+
}
|
|
491
|
+
applyMask(data) {
|
|
492
|
+
if (!this.mask) return data;
|
|
493
|
+
try {
|
|
494
|
+
return this.mask({ data });
|
|
495
|
+
} catch (err) {
|
|
496
|
+
this.logger.warn(
|
|
497
|
+
`Applying mask function failed due to error, fully masking property. Error: ${err}`
|
|
498
|
+
);
|
|
499
|
+
return "<fully masked due to failed mask function>";
|
|
500
|
+
}
|
|
501
|
+
}
|
|
502
|
+
};
|
|
503
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
504
|
+
0 && (module.exports = {
|
|
505
|
+
LangfuseSpanProcessor
|
|
506
|
+
});
|
|
507
|
+
//# sourceMappingURL=index.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/index.ts","../src/span-processor.ts","../src/MediaService.ts"],"sourcesContent":["export * from \"./span-processor.js\";\n","import {\n Logger,\n getGlobalLogger,\n LangfuseAPIClient,\n LANGFUSE_SDK_VERSION,\n LangfuseOtelSpanAttributes,\n getEnv,\n base64Encode,\n getPropagatedAttributesFromContext,\n} from \"@elasticdash/core\";\nimport { Context } from \"@opentelemetry/api\";\nimport { hrTimeToMilliseconds } from \"@opentelemetry/core\";\nimport { OTLPTraceExporter } from \"@opentelemetry/exporter-trace-otlp-http\";\nimport {\n Span,\n BatchSpanProcessor,\n SimpleSpanProcessor,\n SpanExporter,\n ReadableSpan,\n SpanProcessor,\n} from \"@opentelemetry/sdk-trace-base\";\n\nimport { MediaService } from \"./MediaService.js\";\n\n/**\n * Function type for masking sensitive data in spans before export.\n *\n * @param params - Object containing the data to be masked\n * @param params.data - The data that should be masked\n * @returns The masked data (can be of any type)\n *\n * @example\n * ```typescript\n * const maskFunction: MaskFunction = ({ data }) => {\n * if (typeof data === 'string') {\n * return data.replace(/password=\\w+/g, 'password=***');\n * }\n * return data;\n * };\n * ```\n *\n * @public\n */\nexport type MaskFunction = (params: { data: any }) => any;\n\n/**\n * Function type for determining whether a span should be exported to Langfuse.\n *\n * @param params - Object containing the span to evaluate\n * @param params.otelSpan - The OpenTelemetry span to evaluate\n * @returns `true` if the span should be exported, `false` otherwise\n *\n * @example\n * ```typescript\n * const shouldExportSpan: ShouldExportSpan = ({ otelSpan }) => {\n * // Only export spans that took longer than 100ms\n * return otelSpan.duration[0] * 1000 + otelSpan.duration[1] / 1000000 > 100;\n * };\n * ```\n *\n * @public\n */\nexport type ShouldExportSpan = (params: { otelSpan: ReadableSpan }) => boolean;\n\n/**\n * Configuration parameters for the LangfuseSpanProcessor.\n *\n * @public\n */\nexport interface LangfuseSpanProcessorParams {\n /**\n * Custom OpenTelemetry span exporter. If not provided, a default OTLP exporter will be used.\n */\n exporter?: SpanExporter;\n\n /**\n * Langfuse public API key. Can also be set via LANGFUSE_PUBLIC_KEY environment variable.\n */\n publicKey?: string;\n\n /**\n * Langfuse secret API key. Can also be set via LANGFUSE_SECRET_KEY environment variable.\n */\n secretKey?: string;\n\n /**\n * Langfuse instance base URL. Can also be set via LANGFUSE_BASE_URL environment variable.\n * @defaultValue \"https://cloud.langfuse.com\"\n */\n baseUrl?: string;\n\n /**\n * Number of spans to batch before flushing. Can also be set via LANGFUSE_FLUSH_AT environment variable.\n */\n flushAt?: number;\n\n /**\n * Flush interval in seconds. Can also be set via LANGFUSE_FLUSH_INTERVAL environment variable.\n */\n flushInterval?: number;\n\n /**\n * Function to mask sensitive data in spans before export.\n */\n mask?: MaskFunction;\n\n /**\n * Function to determine whether a span should be exported to Langfuse.\n */\n shouldExportSpan?: ShouldExportSpan;\n\n /**\n * Environment identifier for the traces. Can also be set via LANGFUSE_TRACING_ENVIRONMENT environment variable.\n */\n environment?: string;\n\n /**\n * Release identifier for the traces. Can also be set via LANGFUSE_RELEASE environment variable.\n */\n release?: string;\n\n /**\n * Request timeout in seconds. Can also be set via LANGFUSE_TIMEOUT environment variable.\n * @defaultValue 5\n */\n timeout?: number;\n\n /**\n * Additional HTTP headers to include with requests.\n */\n additionalHeaders?: Record<string, string>;\n /**\n * Span export mode to use.\n *\n * - **batched**: Recommended for production environments with long-running processes.\n * Spans are batched and exported in groups for optimal performance.\n * - **immediate**: Recommended for short-lived environments such as serverless functions.\n * Spans are exported immediately to prevent data loss when the process terminates / is frozen.\n *\n * @defaultValue \"batched\"\n */\n exportMode?: \"immediate\" | \"batched\";\n}\n\n/**\n * OpenTelemetry span processor for sending spans to Langfuse.\n *\n * This processor extends the standard BatchSpanProcessor to provide:\n * - Automatic batching and flushing of spans to Langfuse\n * - Media content extraction and upload from base64 data URIs\n * - Data masking capabilities for sensitive information\n * - Conditional span export based on custom logic\n * - Environment and release tagging\n *\n * @example\n * ```typescript\n * import { NodeSDK } from '@opentelemetry/sdk-node';\n * import { LangfuseSpanProcessor } from '@elasticdash/otel';\n *\n * const sdk = new NodeSDK({\n * spanProcessors: [\n * new LangfuseSpanProcessor({\n * publicKey: 'pk_...',\n * secretKey: 'sk_...',\n * baseUrl: 'https://cloud.langfuse.com',\n * environment: 'production',\n * mask: ({ data }) => {\n * // Mask sensitive data\n * return data.replace(/api_key=\\w+/g, 'api_key=***');\n * }\n * })\n * ]\n * });\n *\n * sdk.start();\n * ```\n *\n * @public\n */\nexport class LangfuseSpanProcessor implements SpanProcessor {\n private pendingEndedSpans: Set<Promise<void>> = new Set();\n\n private publicKey?: string;\n private baseUrl?: string;\n private environment?: string;\n private release?: string;\n private mask?: MaskFunction;\n private shouldExportSpan?: ShouldExportSpan;\n private apiClient: LangfuseAPIClient;\n private processor: SpanProcessor;\n private mediaService: MediaService;\n\n /**\n * Creates a new LangfuseSpanProcessor instance.\n *\n * @param params - Configuration parameters for the processor\n *\n * @example\n * ```typescript\n * const processor = new LangfuseSpanProcessor({\n * publicKey: 'pk_...',\n * secretKey: 'sk_...',\n * environment: 'staging',\n * flushAt: 10,\n * flushInterval: 2,\n * mask: ({ data }) => {\n * // Custom masking logic\n * return typeof data === 'string'\n * ? data.replace(/secret_\\w+/g, 'secret_***')\n * : data;\n * },\n * shouldExportSpan: ({ otelSpan }) => {\n * // Only export spans from specific services\n * return otelSpan.name.startsWith('my-service');\n * }\n * });\n * ```\n */\n constructor(params?: LangfuseSpanProcessorParams) {\n const logger = getGlobalLogger();\n\n const publicKey = params?.publicKey ?? getEnv(\"LANGFUSE_PUBLIC_KEY\");\n const secretKey = params?.secretKey ?? getEnv(\"LANGFUSE_SECRET_KEY\");\n const baseUrl =\n params?.baseUrl ??\n getEnv(\"LANGFUSE_BASE_URL\") ??\n getEnv(\"LANGFUSE_BASEURL\") ?? // legacy v2\n \"https://cloud.langfuse.com\";\n\n if (!params?.exporter && !publicKey) {\n logger.warn(\n \"No exporter configured and no public key provided in constructor or as LANGFUSE_PUBLIC_KEY env var. Span exports will fail.\",\n );\n }\n if (!params?.exporter && !secretKey) {\n logger.warn(\n \"No exporter configured and no secret key provided in constructor or as LANGFUSE_SECRET_KEY env var. Span exports will fail.\",\n );\n }\n const flushAt = params?.flushAt ?? getEnv(\"LANGFUSE_FLUSH_AT\");\n const flushIntervalSeconds =\n params?.flushInterval ?? getEnv(\"LANGFUSE_FLUSH_INTERVAL\");\n\n const authHeaderValue = base64Encode(`${publicKey}:${secretKey}`);\n const timeoutSeconds =\n params?.timeout ?? Number(getEnv(\"LANGFUSE_TIMEOUT\") ?? 5);\n\n const exporter =\n params?.exporter ??\n new OTLPTraceExporter({\n url: `${baseUrl}/api/public/otel/v1/traces`,\n headers: {\n Authorization: `Basic ${authHeaderValue}`,\n x_langfuse_sdk_name: \"javascript\",\n x_langfuse_sdk_version: LANGFUSE_SDK_VERSION,\n x_langfuse_public_key: publicKey ?? \"<missing>\",\n ...params?.additionalHeaders,\n },\n timeoutMillis: timeoutSeconds * 1_000,\n });\n\n this.processor =\n params?.exportMode === \"immediate\"\n ? new SimpleSpanProcessor(exporter)\n : new BatchSpanProcessor(exporter, {\n maxExportBatchSize: flushAt ? Number(flushAt) : undefined,\n scheduledDelayMillis: flushIntervalSeconds\n ? Number(flushIntervalSeconds) * 1_000\n : undefined,\n });\n\n this.publicKey = publicKey;\n this.baseUrl = baseUrl;\n this.environment =\n params?.environment ?? getEnv(\"LANGFUSE_TRACING_ENVIRONMENT\");\n this.release = params?.release ?? getEnv(\"LANGFUSE_RELEASE\");\n this.mask = params?.mask;\n this.shouldExportSpan = params?.shouldExportSpan;\n this.apiClient = new LangfuseAPIClient({\n baseUrl: this.baseUrl,\n username: this.publicKey,\n password: secretKey,\n xLangfusePublicKey: this.publicKey,\n xLangfuseSdkVersion: LANGFUSE_SDK_VERSION,\n xLangfuseSdkName: \"javascript\",\n environment: \"\", // noop as baseUrl is set\n headers: params?.additionalHeaders,\n });\n\n this.mediaService = new MediaService({ apiClient: this.apiClient });\n\n logger.debug(\"Initialized LangfuseSpanProcessor with params:\", {\n publicKey,\n baseUrl,\n environment: this.environment,\n release: this.release,\n timeoutSeconds,\n flushAt,\n flushIntervalSeconds,\n });\n }\n\n private get logger(): Logger {\n return getGlobalLogger();\n }\n\n /**\n * Called when a span is started. Adds environment, release, and propagated attributes to the span.\n *\n * @param span - The span that was started\n * @param parentContext - The parent context\n *\n * @override\n */\n public onStart(span: Span, parentContext: Context): void {\n // Set propagated attributes, environment and release attributes\n span.setAttributes({\n [LangfuseOtelSpanAttributes.ENVIRONMENT]: this.environment,\n [LangfuseOtelSpanAttributes.RELEASE]: this.release,\n ...getPropagatedAttributesFromContext(parentContext),\n });\n\n return this.processor.onStart(span, parentContext);\n }\n\n /**\n * Called when a span ends. Processes the span for export to Langfuse.\n *\n * This method:\n * 1. Checks if the span should be exported using the shouldExportSpan function\n * 2. Applies data masking to sensitive attributes\n * 3. Handles media content extraction and upload\n * 4. Logs span details in debug mode\n * 5. Passes the span to the parent processor for export\n *\n * @param span - The span that ended\n *\n * @override\n */\n public onEnd(span: ReadableSpan): void {\n const processEndedSpanPromise = this.processEndedSpan(span).catch((err) => {\n this.logger.error(err);\n });\n\n // Enqueue this export to the pending list so it can be flushed by the user.\n this.pendingEndedSpans.add(processEndedSpanPromise);\n\n void processEndedSpanPromise.finally(() =>\n this.pendingEndedSpans.delete(processEndedSpanPromise),\n );\n }\n\n private async flush(): Promise<void> {\n await Promise.all(Array.from(this.pendingEndedSpans));\n await this.mediaService.flush();\n }\n\n /**\n * Forces an immediate flush of all pending spans and media uploads.\n *\n * @returns Promise that resolves when all pending operations are complete\n *\n * @override\n */\n public async forceFlush(): Promise<void> {\n await this.flush();\n\n return this.processor.forceFlush();\n }\n\n /**\n * Gracefully shuts down the processor, ensuring all pending operations are completed.\n *\n * @returns Promise that resolves when shutdown is complete\n *\n * @override\n */\n public async shutdown(): Promise<void> {\n await this.flush();\n\n return this.processor.shutdown();\n }\n\n private async processEndedSpan(span: ReadableSpan) {\n if (this.shouldExportSpan) {\n try {\n if (this.shouldExportSpan({ otelSpan: span }) === false) return;\n } catch (err) {\n this.logger.error(\n \"ShouldExportSpan failed with error. Excluding span. Error: \",\n err,\n );\n\n return;\n }\n }\n\n this.applyMaskInPlace(span);\n await this.mediaService.process(span);\n\n this.logger.debug(\n `Processed span:\\n${JSON.stringify(\n {\n name: span.name,\n traceId: span.spanContext().traceId,\n spanId: span.spanContext().spanId,\n parentSpanId: span.parentSpanContext?.spanId ?? null,\n attributes: span.attributes,\n startTime: new Date(hrTimeToMilliseconds(span.startTime)),\n endTime: new Date(hrTimeToMilliseconds(span.endTime)),\n durationMs: hrTimeToMilliseconds(span.duration),\n kind: span.kind,\n status: span.status,\n resource: span.resource.attributes,\n instrumentationScope: span.instrumentationScope,\n },\n null,\n 2,\n )}`,\n );\n\n this.processor.onEnd(span);\n }\n private applyMaskInPlace(span: ReadableSpan): void {\n const maskCandidates = [\n LangfuseOtelSpanAttributes.OBSERVATION_INPUT,\n LangfuseOtelSpanAttributes.TRACE_INPUT,\n LangfuseOtelSpanAttributes.OBSERVATION_OUTPUT,\n LangfuseOtelSpanAttributes.TRACE_OUTPUT,\n LangfuseOtelSpanAttributes.OBSERVATION_METADATA,\n LangfuseOtelSpanAttributes.TRACE_METADATA,\n ];\n\n for (const maskCandidate of maskCandidates) {\n if (maskCandidate in span.attributes) {\n span.attributes[maskCandidate] = this.applyMask(\n span.attributes[maskCandidate],\n );\n }\n }\n }\n\n private applyMask<T>(data: T): T | string {\n if (!this.mask) return data;\n\n try {\n return this.mask({ data });\n } catch (err) {\n this.logger.warn(\n `Applying mask function failed due to error, fully masking property. Error: ${err}`,\n );\n\n return \"<fully masked due to failed mask function>\";\n }\n }\n}\n","import {\n LangfuseAPIClient,\n LangfuseMedia,\n LangfuseOtelSpanAttributes,\n Logger,\n base64ToBytes,\n getGlobalLogger,\n} from \"@elasticdash/core\";\nimport { ReadableSpan } from \"@opentelemetry/sdk-trace-base\";\n\nexport class MediaService {\n private pendingMediaUploads: Set<Promise<void>> = new Set();\n private apiClient: LangfuseAPIClient;\n\n constructor(params: { apiClient: LangfuseAPIClient }) {\n this.apiClient = params.apiClient;\n }\n\n get logger(): Logger {\n return getGlobalLogger();\n }\n\n public async flush(): Promise<void> {\n await Promise.all(Array.from(this.pendingMediaUploads));\n }\n\n public async process(span: ReadableSpan) {\n const mediaAttributes = [\n LangfuseOtelSpanAttributes.OBSERVATION_INPUT,\n LangfuseOtelSpanAttributes.TRACE_INPUT,\n LangfuseOtelSpanAttributes.OBSERVATION_OUTPUT,\n LangfuseOtelSpanAttributes.TRACE_OUTPUT,\n LangfuseOtelSpanAttributes.OBSERVATION_METADATA,\n LangfuseOtelSpanAttributes.TRACE_METADATA,\n ];\n\n for (const mediaAttribute of mediaAttributes) {\n const mediaRelevantAttributeKeys = Object.keys(span.attributes).filter(\n (attributeName) => attributeName.startsWith(mediaAttribute),\n );\n\n for (const key of mediaRelevantAttributeKeys) {\n const value = span.attributes[key];\n\n if (typeof value !== \"string\") {\n this.logger.warn(\n `Span attribute ${mediaAttribute} is not a stringified object. Skipping media handling.`,\n );\n\n continue;\n }\n\n // Find media base64 data URI\n let mediaReplacedValue = value;\n const regex = /data:[^;]+;base64,[A-Za-z0-9+/]+=*/g;\n const foundMedia = [...new Set(value.match(regex) ?? [])];\n\n if (foundMedia.length === 0) continue;\n\n for (const mediaDataUri of foundMedia) {\n // For each media, create media tag and initiate upload\n const media = new LangfuseMedia({\n base64DataUri: mediaDataUri,\n source: \"base64_data_uri\",\n });\n\n const langfuseMediaTag = await media.getTag();\n\n if (!langfuseMediaTag) {\n this.logger.warn(\n \"Failed to create Langfuse media tag. Skipping media item.\",\n );\n\n continue;\n }\n\n this.scheduleUpload({\n span,\n media,\n field: mediaAttribute.includes(\"input\")\n ? \"input\"\n : mediaAttribute.includes(\"output\")\n ? \"output\"\n : \"metadata\", // todo: make more robust\n });\n\n // Replace original attribute with media escaped attribute\n mediaReplacedValue = mediaReplacedValue.replaceAll(\n mediaDataUri,\n langfuseMediaTag,\n );\n }\n\n span.attributes[key] = mediaReplacedValue;\n }\n }\n\n // Handle media from Vercel AI SDK\n if (span.instrumentationScope.name === \"ai\") {\n const aiSDKMediaAttributes = [\"ai.prompt.messages\", \"ai.prompt\"];\n\n for (const mediaAttribute of aiSDKMediaAttributes) {\n const value = span.attributes[mediaAttribute];\n\n if (!value || typeof value !== \"string\") {\n continue;\n }\n\n // Find media base64 data URI\n let mediaReplacedValue = value;\n\n try {\n const parsed = JSON.parse(value);\n\n if (Array.isArray(parsed)) {\n for (const message of parsed) {\n if (Array.isArray(message[\"content\"])) {\n const contentParts = message[\"content\"];\n\n for (const part of contentParts) {\n if (part[\"type\"] === \"file\") {\n let base64Content: string | null = null;\n // FilePart\n if (part[\"data\"] != null && part[\"mediaType\"] != null) {\n base64Content = part[\"data\"];\n }\n\n //ImagePart\n if (\n part[\"image\"] != null &&\n part[\"mediaType\"] != null &&\n !part[\"image\"].startsWith(\"http\") // skip URLs\n ) {\n base64Content = part[\"image\"];\n }\n\n if (!base64Content) continue;\n\n const media = new LangfuseMedia({\n contentType: part[\"mediaType\"],\n contentBytes: base64ToBytes(base64Content),\n source: \"bytes\",\n });\n\n const langfuseMediaTag = await media.getTag();\n\n if (!langfuseMediaTag) {\n this.logger.warn(\n \"Failed to create Langfuse media tag. Skipping media item.\",\n );\n\n continue;\n }\n\n this.scheduleUpload({\n span,\n media,\n field: \"input\",\n });\n\n // Replace original attribute with media escaped attribute\n mediaReplacedValue = mediaReplacedValue.replaceAll(\n base64Content,\n langfuseMediaTag,\n );\n }\n }\n }\n }\n }\n\n span.attributes[mediaAttribute] = mediaReplacedValue;\n } catch (err) {\n this.logger.warn(\n `Failed to handle media for AI SDK attribute ${mediaAttribute} for span ${span.spanContext().spanId}`,\n err,\n );\n }\n }\n }\n }\n\n private scheduleUpload(params: {\n span: ReadableSpan;\n field: string;\n media: LangfuseMedia;\n }) {\n const { span, field, media } = params;\n\n const uploadPromise: Promise<void> = this.handleUpload({\n media,\n traceId: span.spanContext().traceId,\n observationId: span.spanContext().spanId,\n field,\n }).catch((err) => {\n this.logger.error(\"Media upload failed with error: \", err);\n });\n\n this.pendingMediaUploads.add(uploadPromise);\n\n uploadPromise.finally(() => {\n this.pendingMediaUploads.delete(uploadPromise);\n });\n }\n\n private async handleUpload({\n media,\n traceId,\n observationId,\n field,\n }: {\n media: LangfuseMedia;\n traceId: string;\n observationId?: string;\n field: string;\n }): Promise<void> {\n try {\n const contentSha256Hash = await media.getSha256Hash();\n\n if (\n !media.contentLength ||\n !media._contentType ||\n !contentSha256Hash ||\n !media._contentBytes\n ) {\n return;\n }\n\n const { uploadUrl, mediaId } = await this.apiClient.media.getUploadUrl({\n contentLength: media.contentLength,\n traceId,\n observationId,\n field,\n contentType: media._contentType,\n sha256Hash: contentSha256Hash,\n });\n\n if (!uploadUrl) {\n this.logger.debug(\n `Media status: Media with ID ${mediaId} already uploaded. Skipping duplicate upload.`,\n );\n\n return;\n }\n\n const clientSideMediaId = await media.getId();\n if (clientSideMediaId !== mediaId) {\n this.logger.error(\n `Media integrity error: Media ID mismatch between SDK (${clientSideMediaId}) and Server (${mediaId}). Upload cancelled. Please check media ID generation logic.`,\n );\n\n return;\n }\n\n this.logger.debug(`Uploading media ${mediaId}...`);\n\n const startTime = Date.now();\n\n const uploadResponse = await this.uploadWithBackoff({\n uploadUrl,\n contentBytes: media._contentBytes,\n contentType: media._contentType,\n contentSha256Hash: contentSha256Hash,\n maxRetries: 3,\n baseDelay: 1000,\n });\n\n if (!uploadResponse) {\n throw Error(\"Media upload process failed\");\n }\n\n await this.apiClient.media.patch(mediaId, {\n uploadedAt: new Date().toISOString(),\n uploadHttpStatus: uploadResponse.status,\n uploadHttpError: await uploadResponse.text(),\n uploadTimeMs: Date.now() - startTime,\n });\n\n this.logger.debug(`Media upload status reported for ${mediaId}`);\n } catch (err) {\n this.logger.error(`Error processing media item: ${err}`);\n }\n }\n\n private async uploadWithBackoff(params: {\n uploadUrl: string;\n contentType: string;\n contentSha256Hash: string;\n contentBytes: Uint8Array;\n maxRetries: number;\n baseDelay: number;\n }) {\n const {\n uploadUrl,\n contentType,\n contentSha256Hash,\n contentBytes,\n maxRetries,\n baseDelay,\n } = params;\n\n for (let attempt = 0; attempt <= maxRetries; attempt++) {\n try {\n let parsedHostname: string;\n\n try {\n parsedHostname = new URL(uploadUrl).hostname;\n } catch {\n parsedHostname = \"\";\n }\n\n const isSelfHostedGcsBucket =\n parsedHostname === \"storage.googleapis.com\" ||\n parsedHostname.endsWith(\".storage.googleapis.com\");\n\n const headers = isSelfHostedGcsBucket\n ? { \"Content-Type\": contentType }\n : {\n \"Content-Type\": contentType,\n \"x-amz-checksum-sha256\": contentSha256Hash,\n \"x-ms-blob-type\": \"BlockBlob\",\n };\n\n const uploadResponse = await fetch(uploadUrl, {\n method: \"PUT\",\n body: contentBytes,\n headers,\n });\n\n if (\n attempt < maxRetries &&\n uploadResponse.status !== 200 &&\n uploadResponse.status !== 201\n ) {\n throw new Error(`Upload failed with status ${uploadResponse.status}`);\n }\n\n return uploadResponse;\n } catch (e) {\n if (attempt === maxRetries) {\n throw e;\n }\n\n const delay = baseDelay * Math.pow(2, attempt);\n const jitter = Math.random() * 1000;\n\n await new Promise((resolve) => setTimeout(resolve, delay + jitter));\n }\n }\n }\n}\n"],"mappings":"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","names":["import_core"]}
|