0
# Media Management
1
2
The MediaManager provides functionality to resolve media references in Langfuse trace data. Media content (images, audio, video, PDFs, etc.) is stored separately and referenced using special string identifiers. The MediaManager resolves these references back to usable content like base64 data URIs.
3
4
## Capabilities
5
6
### Resolve Media References
7
8
Replace media reference strings in an object with base64 data URIs by fetching the actual media content from Langfuse.
9
10
```typescript { .api }
11
/**
12
* Resolves media reference strings in an object to base64 data URIs
13
*
14
* Recursively traverses the input object looking for media reference strings
15
* in the format "@@@langfuseMedia:...@@@". When found, fetches the actual
16
* media content and replaces the reference with a base64 data URI.
17
*
18
* If fetching fails for a reference, a warning is logged and the reference
19
* string is left unchanged in the output.
20
*
21
* @template T - The type of the object being processed
22
* @param params - Configuration object for resolving references
23
* @returns A deep copy of the input with media references resolved
24
*/
25
async resolveReferences<T>(
26
params: LangfuseMediaResolveMediaReferencesParams<T>
27
): Promise<T>;
28
29
type LangfuseMediaResolveMediaReferencesParams<T> = {
30
/** The object to process for media references */
31
obj: T;
32
/** The format to resolve media references to (currently only "base64DataUri" is supported) */
33
resolveWith: "base64DataUri";
34
/** Maximum depth to traverse when processing nested objects (default: 10) */
35
maxDepth?: number;
36
};
37
```
38
39
**Usage Examples:**
40
41
```typescript
42
import { LangfuseClient } from '@langfuse/client';
43
44
const langfuse = new LangfuseClient();
45
46
// Simple object with media reference
47
const obj = {
48
image: "@@@langfuseMedia:type=image/jpeg|id=abc123|source=bytes@@@",
49
text: "This is a regular string"
50
};
51
52
const resolved = await langfuse.media.resolveReferences({
53
obj,
54
resolveWith: "base64DataUri"
55
});
56
57
// Result:
58
// {
59
// image: "...",
60
// text: "This is a regular string"
61
// }
62
```
63
64
**Nested Objects and Arrays:**
65
66
```typescript
67
// Complex nested structure
68
const traceData = {
69
observations: [
70
{
71
input: {
72
message: "Process this image",
73
image: "@@@langfuseMedia:type=image/png|id=img001|source=bytes@@@"
74
},
75
output: {
76
result: "Processed",
77
thumbnail: "@@@langfuseMedia:type=image/jpeg|id=img002|source=bytes@@@"
78
},
79
metadata: {
80
attachments: [
81
"@@@langfuseMedia:type=application/pdf|id=doc001|source=bytes@@@",
82
"@@@langfuseMedia:type=audio/wav|id=aud001|source=bytes@@@"
83
]
84
}
85
}
86
]
87
};
88
89
// Resolves all media references throughout the nested structure
90
const resolved = await langfuse.media.resolveReferences({
91
obj: traceData,
92
resolveWith: "base64DataUri"
93
});
94
95
// All media references are now base64 data URIs
96
console.log(resolved.observations[0].input.image);
97
// "..."
98
99
console.log(resolved.observations[0].metadata.attachments[0]);
100
// "data:application/pdf;base64,JVBERi0xLjcKJeLjz9MKMSAw..."
101
```
102
103
**Controlling Traversal Depth:**
104
105
```typescript
106
// Deeply nested structure
107
const deepObject = {
108
level1: {
109
level2: {
110
level3: {
111
level4: {
112
level5: {
113
media: "@@@langfuseMedia:type=image/jpeg|id=deep001|source=bytes@@@"
114
}
115
}
116
}
117
}
118
}
119
};
120
121
// Limit traversal depth
122
const resolved = await langfuse.media.resolveReferences({
123
obj: deepObject,
124
resolveWith: "base64DataUri",
125
maxDepth: 3 // Will not process beyond level 3
126
});
127
128
// Media at level 5 will not be resolved due to maxDepth limit
129
```
130
131
**Resolving Trace Data:**
132
133
```typescript
134
// Fetch trace from API (contains media references)
135
const trace = await langfuse.api.trace.get('trace-id-123');
136
137
// Resolve all media references in the trace
138
const resolvedTrace = await langfuse.media.resolveReferences({
139
obj: trace,
140
resolveWith: "base64DataUri"
141
});
142
143
// Now you can use the resolved media content
144
if (resolvedTrace.observations) {
145
for (const obs of resolvedTrace.observations) {
146
if (obs.input?.image) {
147
// image is now a base64 data URI, can be displayed in browser
148
console.log('Image data URI:', obs.input.image);
149
}
150
}
151
}
152
```
153
154
**Error Handling:**
155
156
```typescript
157
// If media fetch fails, the reference string remains unchanged
158
const obj = {
159
validMedia: "@@@langfuseMedia:type=image/jpeg|id=valid123|source=bytes@@@",
160
invalidMedia: "@@@langfuseMedia:type=image/jpeg|id=invalid999|source=bytes@@@"
161
};
162
163
const resolved = await langfuse.media.resolveReferences({
164
obj,
165
resolveWith: "base64DataUri"
166
});
167
168
// Valid media is resolved
169
console.log(resolved.validMedia);
170
// "..."
171
172
// Invalid media remains as reference (warning logged)
173
console.log(resolved.invalidMedia);
174
// "@@@langfuseMedia:type=image/jpeg|id=invalid999|source=bytes@@@"
175
```
176
177
### Parse Media Reference String
178
179
Parse a media reference string into its component parts (static method).
180
181
```typescript { .api }
182
/**
183
* Parses a media reference string into a ParsedMediaReference object
184
*
185
* Media reference string format:
186
* "@@@langfuseMedia:type=<mime>|id=<id>|source=<source>@@@"
187
*
188
* @param referenceString - The reference string to parse
189
* @returns Parsed media reference with mediaId, source, and contentType
190
* @throws Error if the reference string is invalid or missing required fields
191
*/
192
static parseReferenceString(
193
referenceString: string
194
): ParsedMediaReference;
195
196
type ParsedMediaReference = {
197
/** The unique identifier for the media content */
198
mediaId: string;
199
/** The source type of the media (e.g., "bytes", "base64_data_uri") */
200
source: string;
201
/** The MIME type of the media content */
202
contentType: MediaContentType;
203
};
204
```
205
206
**Usage Examples:**
207
208
```typescript
209
import { MediaManager } from '@langfuse/client';
210
211
// Parse a valid reference string
212
const referenceString = "@@@langfuseMedia:type=image/jpeg|id=abc-123|source=bytes@@@";
213
214
const parsed = MediaManager.parseReferenceString(referenceString);
215
216
console.log(parsed);
217
// {
218
// mediaId: "abc-123",
219
// source: "bytes",
220
// contentType: "image/jpeg"
221
// }
222
```
223
224
**Extracting Information:**
225
226
```typescript
227
// Use parsed information to understand media type
228
const ref = "@@@langfuseMedia:type=application/pdf|id=doc001|source=base64_data_uri@@@";
229
const parsed = MediaManager.parseReferenceString(ref);
230
231
console.log(`Media ID: ${parsed.mediaId}`); // "doc001"
232
console.log(`Content Type: ${parsed.contentType}`); // "application/pdf"
233
console.log(`Source: ${parsed.source}`); // "base64_data_uri"
234
235
// Determine media category
236
if (parsed.contentType.startsWith('image/')) {
237
console.log('This is an image');
238
} else if (parsed.contentType.startsWith('audio/')) {
239
console.log('This is audio');
240
} else if (parsed.contentType === 'application/pdf') {
241
console.log('This is a PDF document');
242
}
243
```
244
245
**Error Handling:**
246
247
```typescript
248
// Invalid format - missing prefix
249
try {
250
MediaManager.parseReferenceString("type=image/jpeg|id=123|source=bytes@@@");
251
} catch (error) {
252
console.error(error.message);
253
// "Reference string does not start with '@@@langfuseMedia:type='"
254
}
255
256
// Invalid format - missing suffix
257
try {
258
MediaManager.parseReferenceString("@@@langfuseMedia:type=image/jpeg|id=123|source=bytes");
259
} catch (error) {
260
console.error(error.message);
261
// "Reference string does not end with '@@@'"
262
}
263
264
// Invalid format - missing required fields
265
try {
266
MediaManager.parseReferenceString("@@@langfuseMedia:type=image/jpeg|id=123@@@");
267
} catch (error) {
268
console.error(error.message);
269
// "Missing required fields in reference string"
270
}
271
```
272
273
**Finding References in Strings:**
274
275
```typescript
276
// Extract and parse all media references from a string
277
const text = `
278
Image 1: @@@langfuseMedia:type=image/png|id=img1|source=bytes@@@
279
Image 2: @@@langfuseMedia:type=image/jpeg|id=img2|source=bytes@@@
280
PDF: @@@langfuseMedia:type=application/pdf|id=doc1|source=bytes@@@
281
`;
282
283
const regex = /@@@langfuseMedia:.+?@@@/g;
284
const matches = text.match(regex);
285
286
if (matches) {
287
const parsed = matches.map(ref => MediaManager.parseReferenceString(ref));
288
289
console.log('Found media references:');
290
parsed.forEach(p => {
291
console.log(`- ${p.contentType}: ${p.mediaId}`);
292
});
293
// - image/png: img1
294
// - image/jpeg: img2
295
// - application/pdf: doc1
296
}
297
```
298
299
## Reference String Format
300
301
Media reference strings follow a specific format that encodes media metadata:
302
303
```
304
@@@langfuseMedia:type=<mime-type>|id=<media-id>|source=<source-type>@@@
305
```
306
307
**Components:**
308
309
- **Prefix:** `@@@langfuseMedia:`
310
- **type:** MIME type of the media (e.g., `image/jpeg`, `application/pdf`)
311
- **id:** Unique identifier for the media content
312
- **source:** Source type (e.g., `bytes`, `base64_data_uri`)
313
- **Suffix:** `@@@`
314
315
**Examples:**
316
317
```typescript
318
// Image reference
319
"@@@langfuseMedia:type=image/jpeg|id=abc-123-def|source=bytes@@@"
320
321
// PDF reference
322
"@@@langfuseMedia:type=application/pdf|id=doc-456|source=base64_data_uri@@@"
323
324
// Audio reference
325
"@@@langfuseMedia:type=audio/wav|id=sound-789|source=bytes@@@"
326
327
// Video reference
328
"@@@langfuseMedia:type=video/mp4|id=clip-012|source=bytes@@@"
329
```
330
331
## Types
332
333
### MediaContentType
334
335
Union type representing all supported MIME types for media content.
336
337
```typescript { .api }
338
type MediaContentType =
339
// Images
340
| "image/png"
341
| "image/jpeg"
342
| "image/jpg"
343
| "image/webp"
344
| "image/gif"
345
| "image/svg+xml"
346
| "image/tiff"
347
| "image/bmp"
348
// Audio
349
| "audio/mpeg"
350
| "audio/mp3"
351
| "audio/wav"
352
| "audio/ogg"
353
| "audio/oga"
354
| "audio/aac"
355
| "audio/mp4"
356
| "audio/flac"
357
// Video
358
| "video/mp4"
359
| "video/webm"
360
// Text
361
| "text/plain"
362
| "text/html"
363
| "text/css"
364
| "text/csv"
365
// Documents
366
| "application/pdf"
367
| "application/msword"
368
| "application/vnd.ms-excel"
369
// Other
370
| "application/zip"
371
| "application/json"
372
| "application/xml"
373
| "application/octet-stream";
374
375
// Constants for type-safe access
376
const MediaContentType = {
377
// Images
378
ImagePng: "image/png",
379
ImageJpeg: "image/jpeg",
380
ImageJpg: "image/jpg",
381
ImageWebp: "image/webp",
382
ImageGif: "image/gif",
383
ImageSvgXml: "image/svg+xml",
384
ImageTiff: "image/tiff",
385
ImageBmp: "image/bmp",
386
// Audio
387
AudioMpeg: "audio/mpeg",
388
AudioMp3: "audio/mp3",
389
AudioWav: "audio/wav",
390
AudioOgg: "audio/ogg",
391
AudioOga: "audio/oga",
392
AudioAac: "audio/aac",
393
AudioMp4: "audio/mp4",
394
AudioFlac: "audio/flac",
395
// Video
396
VideoMp4: "video/mp4",
397
VideoWebm: "video/webm",
398
// Text
399
TextPlain: "text/plain",
400
TextHtml: "text/html",
401
TextCss: "text/css",
402
TextCsv: "text/csv",
403
// Documents
404
ApplicationPdf: "application/pdf",
405
ApplicationMsword: "application/msword",
406
ApplicationMsExcel: "application/vnd.ms-excel",
407
// Other
408
ApplicationZip: "application/zip",
409
ApplicationJson: "application/json",
410
ApplicationXml: "application/xml",
411
ApplicationOctetStream: "application/octet-stream",
412
} as const;
413
```
414
415
### ParsedMediaReference
416
417
Result type from parsing a media reference string.
418
419
```typescript { .api }
420
type ParsedMediaReference = {
421
/** The unique identifier for the media content */
422
mediaId: string;
423
/** The source type of the media */
424
source: string;
425
/** The MIME type of the media content */
426
contentType: MediaContentType;
427
};
428
```
429
430
### LangfuseMediaResolveMediaReferencesParams
431
432
Parameters for the `resolveReferences` method.
433
434
```typescript { .api }
435
type LangfuseMediaResolveMediaReferencesParams<T> = {
436
/** The object to process for media references */
437
obj: T;
438
/** The format to resolve media references to */
439
resolveWith: "base64DataUri";
440
/** Maximum depth to traverse when processing nested objects (default: 10) */
441
maxDepth?: number;
442
};
443
```
444
445
## Common Use Cases
446
447
### Working with Images
448
449
```typescript
450
import { LangfuseClient } from '@langfuse/client';
451
452
const langfuse = new LangfuseClient();
453
454
// Fetch trace with image references
455
const trace = await langfuse.api.trace.get('trace-123');
456
457
// Resolve image references
458
const resolved = await langfuse.media.resolveReferences({
459
obj: trace,
460
resolveWith: "base64DataUri"
461
});
462
463
// Use resolved images in HTML
464
if (resolved.observations) {
465
for (const obs of resolved.observations) {
466
if (obs.input?.screenshot) {
467
// Create an img element with the data URI
468
const img = document.createElement('img');
469
img.src = obs.input.screenshot; // data:image/png;base64,...
470
document.body.appendChild(img);
471
}
472
}
473
}
474
```
475
476
### Processing PDF Documents
477
478
```typescript
479
// Object with PDF reference
480
const data = {
481
document: "@@@langfuseMedia:type=application/pdf|id=doc123|source=bytes@@@"
482
};
483
484
// Resolve to base64 data URI
485
const resolved = await langfuse.media.resolveReferences({
486
obj: data,
487
resolveWith: "base64DataUri"
488
});
489
490
// Use in iframe or download
491
console.log(resolved.document);
492
// "data:application/pdf;base64,JVBERi0xLjcK..."
493
494
// Display in iframe
495
const iframe = document.createElement('iframe');
496
iframe.src = resolved.document;
497
document.body.appendChild(iframe);
498
```
499
500
### Handling Audio Files
501
502
```typescript
503
// Trace with audio recording reference
504
const trace = {
505
input: {
506
recording: "@@@langfuseMedia:type=audio/wav|id=rec001|source=bytes@@@"
507
}
508
};
509
510
// Resolve audio reference
511
const resolved = await langfuse.media.resolveReferences({
512
obj: trace,
513
resolveWith: "base64DataUri"
514
});
515
516
// Play audio in browser
517
const audio = new Audio(resolved.input.recording);
518
audio.play();
519
520
// Or use in HTML audio element
521
const audioElement = document.createElement('audio');
522
audioElement.src = resolved.input.recording;
523
audioElement.controls = true;
524
document.body.appendChild(audioElement);
525
```
526
527
### Processing Video Content
528
529
```typescript
530
// Video reference in observation
531
const observation = {
532
output: {
533
generatedVideo: "@@@langfuseMedia:type=video/mp4|id=vid001|source=bytes@@@"
534
}
535
};
536
537
// Resolve video
538
const resolved = await langfuse.media.resolveReferences({
539
obj: observation,
540
resolveWith: "base64DataUri"
541
});
542
543
// Display video
544
const video = document.createElement('video');
545
video.src = resolved.output.generatedVideo;
546
video.controls = true;
547
video.width = 640;
548
document.body.appendChild(video);
549
```
550
551
### Batch Processing Multiple Traces
552
553
```typescript
554
// Fetch multiple traces
555
const traces = await langfuse.api.trace.list({ limit: 50 });
556
557
// Resolve all media references across all traces
558
const resolvedTraces = await langfuse.media.resolveReferences({
559
obj: traces,
560
resolveWith: "base64DataUri"
561
});
562
563
// Process each trace with resolved media
564
for (const trace of resolvedTraces.data) {
565
console.log(`Processing trace: ${trace.id}`);
566
567
// Access resolved media content
568
if (trace.observations) {
569
for (const obs of trace.observations) {
570
if (obs.input?.image?.startsWith('data:image/')) {
571
console.log('Found resolved image');
572
}
573
}
574
}
575
}
576
```
577
578
### Selective Media Resolution
579
580
```typescript
581
// Parse references first to decide what to resolve
582
const obj = {
583
largeVideo: "@@@langfuseMedia:type=video/mp4|id=large001|source=bytes@@@",
584
smallImage: "@@@langfuseMedia:type=image/jpeg|id=small001|source=bytes@@@"
585
};
586
587
// Extract reference strings
588
const videoRef = obj.largeVideo;
589
const imageRef = obj.smallImage;
590
591
// Parse to check content type
592
const videoParsed = MediaManager.parseReferenceString(videoRef);
593
const imageParsed = MediaManager.parseReferenceString(imageRef);
594
595
// Only resolve images (skip large videos)
596
const filtered = {
597
smallImage: obj.smallImage,
598
// Exclude largeVideo
599
};
600
601
const resolved = await langfuse.media.resolveReferences({
602
obj: filtered,
603
resolveWith: "base64DataUri"
604
});
605
606
console.log('Resolved only small image, skipped large video');
607
```
608
609
## Best Practices
610
611
### Memory Management
612
613
Be mindful of memory usage when resolving large media files or processing many objects:
614
615
```typescript
616
// Good: Process in batches
617
const traces = await langfuse.api.trace.list({ limit: 10 });
618
const resolved = await langfuse.media.resolveReferences({
619
obj: traces,
620
resolveWith: "base64DataUri"
621
});
622
623
// Less ideal: Processing too many large files at once
624
// const manyTraces = await langfuse.api.trace.list({ limit: 1000 });
625
// May cause memory issues if traces contain large media
626
```
627
628
### Error Resilience
629
630
The `resolveReferences` method handles individual fetch failures gracefully:
631
632
```typescript
633
// Some references may fail to fetch, but the method continues
634
const obj = {
635
validImage: "@@@langfuseMedia:type=image/jpeg|id=valid|source=bytes@@@",
636
missingImage: "@@@langfuseMedia:type=image/jpeg|id=deleted|source=bytes@@@",
637
anotherValidImage: "@@@langfuseMedia:type=image/png|id=valid2|source=bytes@@@"
638
};
639
640
const resolved = await langfuse.media.resolveReferences({
641
obj,
642
resolveWith: "base64DataUri"
643
});
644
645
// Valid images are resolved, invalid ones remain as references
646
// Warnings are logged for failed fetches
647
```
648
649
### Caching Results
650
651
Consider caching resolved media to avoid repeated fetches:
652
653
```typescript
654
const cache = new Map<string, string>();
655
656
async function getResolvedMedia(referenceString: string): Promise<string> {
657
if (cache.has(referenceString)) {
658
return cache.get(referenceString)!;
659
}
660
661
const resolved = await langfuse.media.resolveReferences({
662
obj: { media: referenceString },
663
resolveWith: "base64DataUri"
664
});
665
666
cache.set(referenceString, resolved.media);
667
return resolved.media;
668
}
669
670
// Use cached resolver
671
const image1 = await getResolvedMedia(
672
"@@@langfuseMedia:type=image/jpeg|id=abc|source=bytes@@@"
673
);
674
const image2 = await getResolvedMedia(
675
"@@@langfuseMedia:type=image/jpeg|id=abc|source=bytes@@@" // Cache hit
676
);
677
```
678
679
### Type Safety
680
681
Use TypeScript generics to maintain type information:
682
683
```typescript
684
interface TraceWithMedia {
685
id: string;
686
observations: Array<{
687
input?: {
688
image?: string;
689
};
690
}>;
691
}
692
693
// Type is preserved through resolution
694
const trace: TraceWithMedia = await langfuse.api.trace.get('trace-123');
695
const resolved: TraceWithMedia = await langfuse.media.resolveReferences({
696
obj: trace,
697
resolveWith: "base64DataUri"
698
});
699
700
// TypeScript knows the structure
701
console.log(resolved.observations[0].input?.image);
702
```
703
704
### Controlling Recursion
705
706
Use `maxDepth` to prevent excessive recursion on deeply nested structures:
707
708
```typescript
709
// Limit traversal for performance
710
const resolved = await langfuse.media.resolveReferences({
711
obj: veryNestedObject,
712
resolveWith: "base64DataUri",
713
maxDepth: 5 // Don't traverse deeper than 5 levels
714
});
715
```
716
717
## Integration Patterns
718
719
### With Web Applications
720
721
```typescript
722
// React example
723
import { LangfuseClient } from '@langfuse/client';
724
import { useState, useEffect } from 'react';
725
726
function TraceViewer({ traceId }: { traceId: string }) {
727
const [resolvedTrace, setResolvedTrace] = useState(null);
728
const langfuse = new LangfuseClient();
729
730
useEffect(() => {
731
async function loadTrace() {
732
const trace = await langfuse.api.trace.get(traceId);
733
const resolved = await langfuse.media.resolveReferences({
734
obj: trace,
735
resolveWith: "base64DataUri"
736
});
737
setResolvedTrace(resolved);
738
}
739
loadTrace();
740
}, [traceId]);
741
742
if (!resolvedTrace) return <div>Loading...</div>;
743
744
return (
745
<div>
746
{resolvedTrace.observations?.map((obs, idx) => (
747
<div key={idx}>
748
{obs.input?.image && (
749
<img src={obs.input.image} alt="Trace input" />
750
)}
751
</div>
752
))}
753
</div>
754
);
755
}
756
```
757
758
### With Server-Side Processing
759
760
```typescript
761
// Node.js server example
762
import { LangfuseClient } from '@langfuse/client';
763
import express from 'express';
764
765
const app = express();
766
const langfuse = new LangfuseClient();
767
768
app.get('/api/trace/:traceId', async (req, res) => {
769
try {
770
const trace = await langfuse.api.trace.get(req.params.traceId);
771
772
// Resolve media references before sending to client
773
const resolved = await langfuse.media.resolveReferences({
774
obj: trace,
775
resolveWith: "base64DataUri"
776
});
777
778
res.json(resolved);
779
} catch (error) {
780
res.status(500).json({ error: error.message });
781
}
782
});
783
784
app.listen(3000);
785
```
786
787
### With Data Processing Pipelines
788
789
```typescript
790
// ETL pipeline example
791
import { LangfuseClient } from '@langfuse/client';
792
793
const langfuse = new LangfuseClient();
794
795
async function exportTracesWithMedia() {
796
const traces = await langfuse.api.trace.list({ limit: 100 });
797
798
// Resolve all media in batch
799
const resolved = await langfuse.media.resolveReferences({
800
obj: traces,
801
resolveWith: "base64DataUri"
802
});
803
804
// Export to file or database with resolved media
805
for (const trace of resolved.data) {
806
await saveToDatabase(trace); // Media is now embedded as base64
807
}
808
}
809
```
810