@mastra/sentry 1.0.0-beta.2

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/index.cjs ADDED
@@ -0,0 +1,479 @@
1
+ 'use strict';
2
+
3
+ var observability$1 = require('@mastra/core/observability');
4
+ var observability = require('@mastra/observability');
5
+ var otelExporter = require('@mastra/otel-exporter');
6
+ var Sentry = require('@sentry/node');
7
+
8
+ function _interopNamespace(e) {
9
+ if (e && e.__esModule) return e;
10
+ var n = Object.create(null);
11
+ if (e) {
12
+ Object.keys(e).forEach(function (k) {
13
+ if (k !== 'default') {
14
+ var d = Object.getOwnPropertyDescriptor(e, k);
15
+ Object.defineProperty(n, k, d.get ? d : {
16
+ enumerable: true,
17
+ get: function () { return e[k]; }
18
+ });
19
+ }
20
+ });
21
+ }
22
+ n.default = e;
23
+ return Object.freeze(n);
24
+ }
25
+
26
+ var Sentry__namespace = /*#__PURE__*/_interopNamespace(Sentry);
27
+
28
+ // src/tracing.ts
29
+ var SPAN_TYPE_CONFIG = {
30
+ [observability$1.SpanType.AGENT_RUN]: { opType: "gen_ai.invoke_agent", opName: "invoke_agent" },
31
+ [observability$1.SpanType.MODEL_GENERATION]: { opType: "gen_ai.chat", opName: "chat" },
32
+ [observability$1.SpanType.TOOL_CALL]: { opType: "gen_ai.execute_tool", opName: "execute_tool" },
33
+ [observability$1.SpanType.MCP_TOOL_CALL]: { opType: "gen_ai.execute_tool", opName: "execute_tool" },
34
+ [observability$1.SpanType.WORKFLOW_RUN]: { opType: "workflow.run", opName: "workflow" },
35
+ [observability$1.SpanType.WORKFLOW_STEP]: { opType: "workflow.step", opName: "step" },
36
+ [observability$1.SpanType.WORKFLOW_CONDITIONAL]: { opType: "workflow.conditional", opName: "step" },
37
+ [observability$1.SpanType.WORKFLOW_CONDITIONAL_EVAL]: { opType: "workflow.conditional", opName: "step" },
38
+ [observability$1.SpanType.WORKFLOW_PARALLEL]: { opType: "workflow.parallel", opName: "step" },
39
+ [observability$1.SpanType.WORKFLOW_LOOP]: { opType: "workflow.loop", opName: "step" },
40
+ [observability$1.SpanType.WORKFLOW_SLEEP]: { opType: "workflow.sleep", opName: "step" },
41
+ [observability$1.SpanType.WORKFLOW_WAIT_EVENT]: { opType: "workflow.wait", opName: "step" },
42
+ [observability$1.SpanType.PROCESSOR_RUN]: { opType: "ai.processor", opName: "step" },
43
+ [observability$1.SpanType.GENERIC]: { opType: "ai.span", opName: "span" },
44
+ [observability$1.SpanType.MODEL_STEP]: { opType: "ai.span", opName: "step" },
45
+ [observability$1.SpanType.MODEL_CHUNK]: { opType: "ai.span", opName: "step" }
46
+ };
47
+ var ATTRIBUTE_KEYS = {
48
+ SPAN_TYPE: "ai.span.type",
49
+ ORIGIN: "sentry.origin",
50
+ TAGS: "tags",
51
+ INPUT: "input",
52
+ OUTPUT: "output",
53
+ GEN_AI_REQUEST_STREAM: "gen_ai.request.stream",
54
+ GEN_AI_RESPONSE_MODEL: "gen_ai.response.model",
55
+ GEN_AI_RESPONSE_STREAMING: "gen_ai.response.streaming",
56
+ GEN_AI_RESPONSE_TOOL_CALLS: "gen_ai.response.tool_calls",
57
+ GEN_AI_RESPONSE_TEXT: "gen_ai.response.text",
58
+ GEN_AI_COMPLETION_START_TIME: "gen_ai.completion_start_time",
59
+ GEN_AI_TOOL_CALL_ID: "gen_ai.tool.call.id",
60
+ TOOL_SUCCESS: "tool.success",
61
+ GEN_AI_PIPELINE_NAME: "gen_ai.pipeline.name",
62
+ GEN_AI_AGENT_PROMPT: "gen_ai.agent.prompt",
63
+ WORKFLOW_ID: "workflow.id",
64
+ WORKFLOW_STATUS: "workflow.status",
65
+ WORKFLOW_STEP_ID: "workflow.step.id",
66
+ WORKFLOW_STEP_STATUS: "workflow.step.status",
67
+ GEN_AI_USAGE_INPUT_TOKENS: "gen_ai.usage.input_tokens",
68
+ GEN_AI_USAGE_OUTPUT_TOKENS: "gen_ai.usage.output_tokens",
69
+ GEN_AI_USAGE_TOTAL_TOKENS: "gen_ai.usage.total_tokens",
70
+ GEN_AI_USAGE_CACHE_READ_TOKENS: "gen_ai.usage.cache_read_input_tokens",
71
+ GEN_AI_USAGE_CACHE_WRITE_TOKENS: "gen_ai.usage.cache_write_input_tokens",
72
+ GEN_AI_USAGE_REASONING_TOKENS: "gen_ai.usage.reasoning_tokens"
73
+ };
74
+ var SentryExporter = class extends observability.BaseExporter {
75
+ name = "sentry";
76
+ config;
77
+ spanMap = /* @__PURE__ */ new Map();
78
+ skippedSpans = /* @__PURE__ */ new Map();
79
+ initialized = false;
80
+ constructor(config = {}) {
81
+ super(config);
82
+ this.config = {
83
+ dsn: config.dsn ?? process.env.SENTRY_DSN ?? "",
84
+ environment: config.environment ?? process.env.SENTRY_ENVIRONMENT ?? "production",
85
+ tracesSampleRate: config.tracesSampleRate ?? 1,
86
+ release: config.release ?? process.env.SENTRY_RELEASE ?? ""
87
+ };
88
+ if (!this.config.dsn) {
89
+ const dsnSource = config.dsn ? "from config" : process.env.SENTRY_DSN ? "from env" : "missing";
90
+ this.setDisabled(
91
+ `Missing required DSN (dsn: ${dsnSource}). Set SENTRY_DSN environment variable or pass it in config.`
92
+ );
93
+ return;
94
+ }
95
+ try {
96
+ Sentry__namespace.init({
97
+ dsn: this.config.dsn,
98
+ environment: this.config.environment,
99
+ tracesSampleRate: this.config.tracesSampleRate,
100
+ release: this.config.release,
101
+ ...config.options
102
+ });
103
+ this.initialized = true;
104
+ } catch (error) {
105
+ this.setDisabled(`Failed to initialize Sentry: ${error}`);
106
+ }
107
+ }
108
+ // ============================================================================
109
+ // Main Event Handlers
110
+ // ============================================================================
111
+ async _exportTracingEvent(event) {
112
+ if (!this.initialized) return;
113
+ const { type, exportedSpan } = event;
114
+ if (exportedSpan.isEvent) {
115
+ this.handleEventSpan(exportedSpan);
116
+ return;
117
+ }
118
+ if (exportedSpan.type === observability$1.SpanType.MODEL_CHUNK || exportedSpan.type === observability$1.SpanType.MODEL_STEP) {
119
+ if (type === observability$1.TracingEventType.SPAN_STARTED) {
120
+ this.skippedSpans.set(exportedSpan.id, exportedSpan.parentSpanId || "");
121
+ } else if (type === observability$1.TracingEventType.SPAN_ENDED) {
122
+ this.skippedSpans.delete(exportedSpan.id);
123
+ }
124
+ return;
125
+ }
126
+ switch (type) {
127
+ case observability$1.TracingEventType.SPAN_STARTED:
128
+ await this.handleSpanStarted(exportedSpan);
129
+ break;
130
+ case observability$1.TracingEventType.SPAN_UPDATED:
131
+ await this.handleSpanUpdated(exportedSpan);
132
+ break;
133
+ case observability$1.TracingEventType.SPAN_ENDED:
134
+ await this.handleSpanEnded(exportedSpan);
135
+ break;
136
+ }
137
+ }
138
+ handleEventSpan(span) {
139
+ Sentry__namespace.addBreadcrumb({
140
+ type: "default",
141
+ category: span.type,
142
+ message: span.name,
143
+ level: span.errorInfo ? "error" : "info",
144
+ data: {
145
+ spanId: span.id,
146
+ traceId: span.traceId,
147
+ ...span.input && { input: this.serializeValue(span.input) },
148
+ ...span.output && { output: this.serializeValue(span.output) },
149
+ ...span.metadata && { metadata: span.metadata },
150
+ ...span.attributes && { attributes: span.attributes }
151
+ },
152
+ timestamp: span.startTime.getTime() / 1e3
153
+ });
154
+ }
155
+ async handleSpanStarted(span) {
156
+ const resolvedParentId = this.resolveParentSpanId(span.parentSpanId);
157
+ const sentrySpan = Sentry__namespace.startInactiveSpan({
158
+ op: this.getOperationType(span),
159
+ name: otelExporter.getSpanName(span),
160
+ startTime: span.startTime.getTime(),
161
+ forceTransaction: span.isRootSpan,
162
+ parentSpan: resolvedParentId ? this.spanMap.get(resolvedParentId)?.span : void 0
163
+ });
164
+ sentrySpan.setAttributes(this.buildSpanAttributes(span));
165
+ this.spanMap.set(span.id, {
166
+ span: sentrySpan,
167
+ spanType: span.type
168
+ });
169
+ if (span.type === observability$1.SpanType.TOOL_CALL && resolvedParentId) {
170
+ this.trackToolCallForParent(span, resolvedParentId);
171
+ }
172
+ }
173
+ async handleSpanUpdated(span) {
174
+ const spanData = this.spanMap.get(span.id);
175
+ if (!spanData) {
176
+ this.logMissingSpan(span, "span update");
177
+ return;
178
+ }
179
+ }
180
+ async handleSpanEnded(span) {
181
+ const spanData = this.spanMap.get(span.id);
182
+ if (!spanData) {
183
+ this.logMissingSpan(span, "span end");
184
+ return;
185
+ }
186
+ const { span: sentrySpan } = spanData;
187
+ sentrySpan.setAttributes(this.buildSpanAttributes(span));
188
+ if (span.type === observability$1.SpanType.MODEL_GENERATION) {
189
+ this.applyToolCallsAttribute(spanData);
190
+ const resolvedParentId = this.resolveParentSpanId(span.parentSpanId);
191
+ if (resolvedParentId) {
192
+ const parentData = this.spanMap.get(resolvedParentId);
193
+ if (parentData?.spanType === observability$1.SpanType.AGENT_RUN) {
194
+ const modelAttr = span.attributes;
195
+ parentData.generation = {
196
+ model: modelAttr.model,
197
+ output: span.output,
198
+ usage: modelAttr.usage
199
+ };
200
+ }
201
+ }
202
+ }
203
+ if (span.type === observability$1.SpanType.AGENT_RUN) {
204
+ this.applyUsageFromGeneration(spanData);
205
+ this.setGenerationResponseAttributes(spanData);
206
+ }
207
+ if (span.errorInfo) {
208
+ sentrySpan.setStatus({
209
+ code: 2,
210
+ message: span.errorInfo.message
211
+ });
212
+ Sentry__namespace.captureException(span.errorInfo.message, {
213
+ contexts: {
214
+ trace: { trace_id: span.traceId, span_id: span.id },
215
+ span_info: {
216
+ name: span.name,
217
+ type: span.type,
218
+ error_id: span.errorInfo.id,
219
+ error_category: span.errorInfo.category
220
+ }
221
+ }
222
+ });
223
+ }
224
+ const endTime = span.endTime ? span.endTime.getTime() : void 0;
225
+ sentrySpan.end(endTime);
226
+ this.spanMap.delete(span.id);
227
+ }
228
+ // ============================================================================
229
+ // Span Creation Helpers
230
+ // ============================================================================
231
+ resolveParentSpanId(parentSpanId) {
232
+ if (!parentSpanId) return void 0;
233
+ let currentParentId = parentSpanId;
234
+ while (currentParentId && this.skippedSpans.has(currentParentId)) {
235
+ currentParentId = this.skippedSpans.get(currentParentId);
236
+ if (!currentParentId) break;
237
+ }
238
+ return currentParentId;
239
+ }
240
+ getOperationType(span) {
241
+ const config = SPAN_TYPE_CONFIG[span.type];
242
+ return config ? config.opType : "ai.span";
243
+ }
244
+ buildSpanAttributes(span) {
245
+ const attributes = otelExporter.getAttributes(span);
246
+ attributes[ATTRIBUTE_KEYS.SPAN_TYPE] = span.type;
247
+ attributes[ATTRIBUTE_KEYS.ORIGIN] = "auto.ai.mastra";
248
+ if (span.metadata) {
249
+ Object.entries(span.metadata).forEach(([key, value]) => {
250
+ if (value !== void 0 && value !== null && key !== "langfuse") {
251
+ attributes[`metadata.${key}`] = this.serializeValue(value);
252
+ }
253
+ });
254
+ }
255
+ this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.TAGS, span.tags?.join(","));
256
+ this.addInputOutputAttributes(attributes, span);
257
+ if (span.type === observability$1.SpanType.MODEL_GENERATION) {
258
+ this.addModelGenerationAttributes(attributes, span);
259
+ }
260
+ if (span.type === observability$1.SpanType.TOOL_CALL) {
261
+ this.addToolCallAttributes(attributes, span);
262
+ }
263
+ if (span.type === observability$1.SpanType.AGENT_RUN) {
264
+ this.addAgentRunAttributes(attributes, span);
265
+ }
266
+ if (span.type === observability$1.SpanType.WORKFLOW_RUN) {
267
+ const workflowAttr = span.attributes;
268
+ this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.WORKFLOW_ID, this.getEntityName(span));
269
+ this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.WORKFLOW_STATUS, workflowAttr.status);
270
+ }
271
+ if (span.type === observability$1.SpanType.WORKFLOW_STEP) {
272
+ const stepAttr = span.attributes;
273
+ this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.WORKFLOW_STEP_ID, this.getEntityName(span));
274
+ this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.WORKFLOW_STEP_STATUS, stepAttr.status);
275
+ }
276
+ return attributes;
277
+ }
278
+ // ============================================================================
279
+ // Sentry-Specific Attribute Formatters
280
+ // ============================================================================
281
+ /**
282
+ * Adds Sentry-specific input/output attributes that complement GenAI semantic conventions.
283
+ * Adds 'input' and 'output' keys for Sentry UI compatibility.
284
+ */
285
+ addInputOutputAttributes(attributes, span) {
286
+ if (span.input !== void 0) {
287
+ attributes[ATTRIBUTE_KEYS.INPUT] = this.serializeValue(span.input);
288
+ }
289
+ if (span.output !== void 0) {
290
+ attributes[ATTRIBUTE_KEYS.OUTPUT] = this.serializeValue(span.output);
291
+ if (span.type === observability$1.SpanType.MODEL_GENERATION) {
292
+ const outputText = this.extractOutputText(span.output);
293
+ if (outputText) {
294
+ attributes[ATTRIBUTE_KEYS.GEN_AI_RESPONSE_TEXT] = outputText;
295
+ }
296
+ }
297
+ }
298
+ }
299
+ /**
300
+ * Adds Sentry-specific MODEL_GENERATION attributes that complement GenAI semantic conventions.
301
+ */
302
+ addModelGenerationAttributes(attributes, span) {
303
+ const modelAttr = span.attributes;
304
+ if (modelAttr.streaming !== void 0) {
305
+ attributes[ATTRIBUTE_KEYS.GEN_AI_REQUEST_STREAM] = modelAttr.streaming;
306
+ attributes[ATTRIBUTE_KEYS.GEN_AI_RESPONSE_STREAMING] = modelAttr.streaming;
307
+ }
308
+ this.setAttributeIfDefined(
309
+ attributes,
310
+ ATTRIBUTE_KEYS.GEN_AI_COMPLETION_START_TIME,
311
+ modelAttr.completionStartTime?.toISOString()
312
+ );
313
+ if (modelAttr.usage) {
314
+ const totalTokens = (modelAttr.usage.inputTokens || 0) + (modelAttr.usage.outputTokens || 0);
315
+ if (totalTokens > 0) {
316
+ attributes[ATTRIBUTE_KEYS.GEN_AI_USAGE_TOTAL_TOKENS] = totalTokens;
317
+ }
318
+ }
319
+ }
320
+ /**
321
+ * Adds Sentry-specific TOOL_CALL attributes that complement GenAI semantic conventions.
322
+ */
323
+ addToolCallAttributes(attributes, span) {
324
+ const toolAttr = span.attributes;
325
+ this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.TOOL_SUCCESS, toolAttr.success);
326
+ this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.GEN_AI_TOOL_CALL_ID, span.metadata?.toolCallId);
327
+ }
328
+ /**
329
+ * Adds Sentry-specific AGENT_RUN attributes that complement GenAI semantic conventions.
330
+ */
331
+ addAgentRunAttributes(attributes, span) {
332
+ const agentAttr = span.attributes;
333
+ const agentName = this.getEntityName(span);
334
+ if (agentName) {
335
+ attributes[ATTRIBUTE_KEYS.GEN_AI_PIPELINE_NAME] = agentName;
336
+ }
337
+ this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.GEN_AI_AGENT_PROMPT, agentAttr.prompt);
338
+ }
339
+ // ============================================================================
340
+ // Token Usage Management
341
+ // ============================================================================
342
+ /**
343
+ * Applies token usage from the MODEL_GENERATION span to the AGENT_RUN span attributes.
344
+ * Reads usage directly from the generation field.
345
+ * Called when AGENT_RUN spans end to set gen_ai.usage.* attributes.
346
+ */
347
+ applyUsageFromGeneration(spanData) {
348
+ const usage = spanData.generation?.usage;
349
+ if (!usage) return;
350
+ const inputTokens = usage.inputTokens || 0;
351
+ const outputTokens = usage.outputTokens || 0;
352
+ if (inputTokens > 0) {
353
+ spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_USAGE_INPUT_TOKENS, inputTokens);
354
+ }
355
+ if (outputTokens > 0) {
356
+ spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_USAGE_OUTPUT_TOKENS, outputTokens);
357
+ }
358
+ const totalTokens = inputTokens + outputTokens;
359
+ if (totalTokens > 0) {
360
+ spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_USAGE_TOTAL_TOKENS, totalTokens);
361
+ }
362
+ const cacheReadTokens = usage.inputDetails?.cacheRead || 0;
363
+ const cacheWriteTokens = usage.inputDetails?.cacheWrite || 0;
364
+ const reasoningTokens = usage.outputDetails?.reasoning || 0;
365
+ if (cacheReadTokens > 0) {
366
+ spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_USAGE_CACHE_READ_TOKENS, cacheReadTokens);
367
+ }
368
+ if (cacheWriteTokens > 0) {
369
+ spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_USAGE_CACHE_WRITE_TOKENS, cacheWriteTokens);
370
+ }
371
+ if (reasoningTokens > 0) {
372
+ spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_USAGE_REASONING_TOKENS, reasoningTokens);
373
+ }
374
+ }
375
+ /**
376
+ * Sets gen_ai.response.model and gen_ai.response.text from the MODEL_GENERATION.
377
+ * Only applies to AGENT_RUN spans.
378
+ */
379
+ setGenerationResponseAttributes(spanData) {
380
+ if (!spanData.generation) return;
381
+ if (spanData.generation.model) {
382
+ spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_RESPONSE_MODEL, spanData.generation.model);
383
+ }
384
+ if (spanData.generation.output) {
385
+ const outputText = this.extractOutputText(spanData.generation.output);
386
+ if (outputText) {
387
+ spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_RESPONSE_TEXT, outputText);
388
+ }
389
+ }
390
+ }
391
+ /**
392
+ * Tracks a TOOL_CALL span as a child of its parent MODEL_GENERATION span.
393
+ * This builds the tool_calls array for gen_ai.response.tool_calls attribute.
394
+ */
395
+ trackToolCallForParent(span, parentId) {
396
+ const parentSpanData = this.spanMap.get(parentId);
397
+ if (!parentSpanData || parentSpanData.spanType !== observability$1.SpanType.MODEL_GENERATION) {
398
+ return;
399
+ }
400
+ const toolAttr = span.attributes;
401
+ if (!parentSpanData.toolCalls) {
402
+ parentSpanData.toolCalls = [];
403
+ }
404
+ parentSpanData.toolCalls.push({
405
+ name: this.getEntityName(span),
406
+ id: span.metadata?.toolCallId,
407
+ type: toolAttr.toolType || "function"
408
+ });
409
+ }
410
+ /**
411
+ * Applies the gen_ai.response.tool_calls attribute to MODEL_GENERATION spans.
412
+ * Called when MODEL_GENERATION spans end if they have child tool calls.
413
+ */
414
+ applyToolCallsAttribute(spanData) {
415
+ if (!spanData.toolCalls || spanData.toolCalls.length === 0) {
416
+ return;
417
+ }
418
+ spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_RESPONSE_TOOL_CALLS, JSON.stringify(spanData.toolCalls));
419
+ }
420
+ // ============================================================================
421
+ // Utility Helpers
422
+ // ============================================================================
423
+ logMissingSpan(span, operation) {
424
+ this.logger.warn(`Sentry exporter: No Sentry span found for ${operation}`, {
425
+ traceId: span.traceId,
426
+ spanId: span.id,
427
+ spanName: span.name
428
+ });
429
+ }
430
+ getEntityName(span) {
431
+ return span.entityName || span.entityId || "unknown";
432
+ }
433
+ extractOutputText(output) {
434
+ if (!output) return void 0;
435
+ if (typeof output === "string") return output;
436
+ if (output.text && typeof output.text === "string") return output.text;
437
+ if (output.content && typeof output.content === "string") return output.content;
438
+ if (output.message?.content && typeof output.message.content === "string") return output.message.content;
439
+ return void 0;
440
+ }
441
+ serializeValue(value) {
442
+ if (value === null || value === void 0) return value;
443
+ if (typeof value === "object") {
444
+ try {
445
+ return JSON.stringify(value);
446
+ } catch {
447
+ return String(value);
448
+ }
449
+ }
450
+ return value;
451
+ }
452
+ setAttributeIfDefined(attributes, key, value) {
453
+ if (value !== void 0 && value !== null) {
454
+ attributes[key] = value;
455
+ }
456
+ }
457
+ // ============================================================================
458
+ // Shutdown
459
+ // ============================================================================
460
+ async shutdown() {
461
+ for (const [spanId, spanData] of this.spanMap.entries()) {
462
+ try {
463
+ spanData.span.end();
464
+ } catch (error) {
465
+ this.logger.error("Sentry exporter: Error ending span during shutdown", { spanId, error });
466
+ }
467
+ }
468
+ this.spanMap.clear();
469
+ this.skippedSpans.clear();
470
+ if (this.initialized) {
471
+ await Sentry__namespace.close(2e3);
472
+ }
473
+ await super.shutdown();
474
+ }
475
+ };
476
+
477
+ exports.SentryExporter = SentryExporter;
478
+ //# sourceMappingURL=index.cjs.map
479
+ //# sourceMappingURL=index.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/tracing.ts"],"names":["SpanType","BaseExporter","Sentry","TracingEventType","getGenAISpanName","getGenAIAttributes"],"mappings":"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","file":"index.cjs","sourcesContent":["/**\n * Sentry Exporter for Mastra Observability\n *\n * Sends observability data to Sentry for AI tracing and monitoring.\n * Uses Sentry's modern span model (v8+) with OpenTelemetry semantic conventions.\n *\n * Spans are hierarchically organized: AGENT_RUN -> MODEL_GENERATION -> TOOL_CALL\n * MODEL_STEP and MODEL_CHUNK spans are skipped to simplify the trace hierarchy.\n */\n\nimport type {\n TracingEvent,\n AnyExportedSpan,\n ModelGenerationAttributes,\n ToolCallAttributes,\n AgentRunAttributes,\n WorkflowRunAttributes,\n WorkflowStepAttributes,\n UsageStats,\n} from '@mastra/core/observability';\nimport { SpanType, TracingEventType } from '@mastra/core/observability';\nimport type { BaseExporterConfig } from '@mastra/observability';\nimport { BaseExporter } from '@mastra/observability';\nimport { getAttributes as getGenAIAttributes, getSpanName as getGenAISpanName } from '@mastra/otel-exporter';\nimport * as Sentry from '@sentry/node';\n\nconst SPAN_TYPE_CONFIG: Record<SpanType, { opType: string; opName: string }> = {\n [SpanType.AGENT_RUN]: { opType: 'gen_ai.invoke_agent', opName: 'invoke_agent' },\n [SpanType.MODEL_GENERATION]: { opType: 'gen_ai.chat', opName: 'chat' },\n [SpanType.TOOL_CALL]: { opType: 'gen_ai.execute_tool', opName: 'execute_tool' },\n [SpanType.MCP_TOOL_CALL]: { opType: 'gen_ai.execute_tool', opName: 'execute_tool' },\n [SpanType.WORKFLOW_RUN]: { opType: 'workflow.run', opName: 'workflow' },\n [SpanType.WORKFLOW_STEP]: { opType: 'workflow.step', opName: 'step' },\n [SpanType.WORKFLOW_CONDITIONAL]: { opType: 'workflow.conditional', opName: 'step' },\n [SpanType.WORKFLOW_CONDITIONAL_EVAL]: { opType: 'workflow.conditional', opName: 'step' },\n [SpanType.WORKFLOW_PARALLEL]: { opType: 'workflow.parallel', opName: 'step' },\n [SpanType.WORKFLOW_LOOP]: { opType: 'workflow.loop', opName: 'step' },\n [SpanType.WORKFLOW_SLEEP]: { opType: 'workflow.sleep', opName: 'step' },\n [SpanType.WORKFLOW_WAIT_EVENT]: { opType: 'workflow.wait', opName: 'step' },\n [SpanType.PROCESSOR_RUN]: { opType: 'ai.processor', opName: 'step' },\n [SpanType.GENERIC]: { opType: 'ai.span', opName: 'span' },\n [SpanType.MODEL_STEP]: { opType: 'ai.span', opName: 'step' },\n [SpanType.MODEL_CHUNK]: { opType: 'ai.span', opName: 'step' },\n};\n\nconst ATTRIBUTE_KEYS = {\n SPAN_TYPE: 'ai.span.type',\n ORIGIN: 'sentry.origin',\n TAGS: 'tags',\n INPUT: 'input',\n OUTPUT: 'output',\n GEN_AI_REQUEST_STREAM: 'gen_ai.request.stream',\n GEN_AI_RESPONSE_MODEL: 'gen_ai.response.model',\n GEN_AI_RESPONSE_STREAMING: 'gen_ai.response.streaming',\n GEN_AI_RESPONSE_TOOL_CALLS: 'gen_ai.response.tool_calls',\n GEN_AI_RESPONSE_TEXT: 'gen_ai.response.text',\n GEN_AI_COMPLETION_START_TIME: 'gen_ai.completion_start_time',\n GEN_AI_TOOL_CALL_ID: 'gen_ai.tool.call.id',\n TOOL_SUCCESS: 'tool.success',\n GEN_AI_PIPELINE_NAME: 'gen_ai.pipeline.name',\n GEN_AI_AGENT_PROMPT: 'gen_ai.agent.prompt',\n WORKFLOW_ID: 'workflow.id',\n WORKFLOW_STATUS: 'workflow.status',\n WORKFLOW_STEP_ID: 'workflow.step.id',\n WORKFLOW_STEP_STATUS: 'workflow.step.status',\n GEN_AI_USAGE_INPUT_TOKENS: 'gen_ai.usage.input_tokens',\n GEN_AI_USAGE_OUTPUT_TOKENS: 'gen_ai.usage.output_tokens',\n GEN_AI_USAGE_TOTAL_TOKENS: 'gen_ai.usage.total_tokens',\n GEN_AI_USAGE_CACHE_READ_TOKENS: 'gen_ai.usage.cache_read_input_tokens',\n GEN_AI_USAGE_CACHE_WRITE_TOKENS: 'gen_ai.usage.cache_write_input_tokens',\n GEN_AI_USAGE_REASONING_TOKENS: 'gen_ai.usage.reasoning_tokens',\n} as const;\n\nexport interface SentryExporterConfig extends BaseExporterConfig {\n // Sentry SDK options (passed to Sentry.init())\n /** Data Source Name - tells the SDK where to send events */\n dsn?: string;\n /** Deployment environment (enables filtering issues and alerts by environment) */\n environment?: string;\n /** Percentage of transactions sent to Sentry (0.0 = 0%, 1.0 = 100%) */\n tracesSampleRate?: number;\n /** Version of your code deployed (helps identify regressions and track deployments) */\n release?: string;\n /** Additional Sentry SDK options (integrations, beforeSend, etc.) */\n options?: Partial<Sentry.NodeOptions>;\n}\n\n/**\n * Internal span tracking data.\n * generation tracks the single MODEL_GENERATION for AGENT_RUN response attributes.\n * toolCalls tracks child tool calls for MODEL_GENERATION spans.\n */\ntype SpanData = {\n span: Sentry.Span;\n spanType: SpanType;\n generation?: {\n model?: string;\n output?: any;\n usage?: UsageStats;\n };\n toolCalls?: Array<{\n name: string;\n id?: string;\n type?: string;\n }>;\n};\n\nexport class SentryExporter extends BaseExporter {\n name = 'sentry';\n private config: Required<Omit<SentryExporterConfig, 'logger' | 'logLevel' | 'options'>>;\n private spanMap = new Map<string, SpanData>();\n private skippedSpans = new Map<string, string>();\n private initialized = false;\n\n constructor(config: SentryExporterConfig = {}) {\n super(config);\n\n this.config = {\n dsn: config.dsn ?? process.env.SENTRY_DSN ?? '',\n environment: config.environment ?? process.env.SENTRY_ENVIRONMENT ?? 'production',\n tracesSampleRate: config.tracesSampleRate ?? 1.0,\n release: config.release ?? process.env.SENTRY_RELEASE ?? '',\n };\n\n if (!this.config.dsn) {\n const dsnSource = config.dsn ? 'from config' : process.env.SENTRY_DSN ? 'from env' : 'missing';\n this.setDisabled(\n `Missing required DSN (dsn: ${dsnSource}). Set SENTRY_DSN environment variable or pass it in config.`,\n );\n return;\n }\n\n try {\n Sentry.init({\n dsn: this.config.dsn,\n environment: this.config.environment,\n tracesSampleRate: this.config.tracesSampleRate,\n release: this.config.release,\n ...config.options,\n });\n this.initialized = true;\n } catch (error) {\n this.setDisabled(`Failed to initialize Sentry: ${error}`);\n }\n }\n\n // ============================================================================\n // Main Event Handlers\n // ============================================================================\n\n protected async _exportTracingEvent(event: TracingEvent): Promise<void> {\n if (!this.initialized) return;\n\n const { type, exportedSpan } = event;\n\n if (exportedSpan.isEvent) {\n this.handleEventSpan(exportedSpan);\n return;\n }\n\n // Skip MODEL_CHUNK and MODEL_STEP spans to simplify trace hierarchy.\n // We store them in skippedSpans to preserve parent-child relationships:\n // when a child span references a skipped span as parent, resolveParentSpanId()\n // walks up the chain to find the first non-skipped ancestor.\n if (exportedSpan.type === SpanType.MODEL_CHUNK || exportedSpan.type === SpanType.MODEL_STEP) {\n if (type === TracingEventType.SPAN_STARTED) {\n this.skippedSpans.set(exportedSpan.id, exportedSpan.parentSpanId || '');\n } else if (type === TracingEventType.SPAN_ENDED) {\n this.skippedSpans.delete(exportedSpan.id);\n }\n return;\n }\n\n switch (type) {\n case TracingEventType.SPAN_STARTED:\n await this.handleSpanStarted(exportedSpan);\n break;\n case TracingEventType.SPAN_UPDATED:\n await this.handleSpanUpdated(exportedSpan);\n break;\n case TracingEventType.SPAN_ENDED:\n await this.handleSpanEnded(exportedSpan);\n break;\n }\n }\n\n private handleEventSpan(span: AnyExportedSpan): void {\n Sentry.addBreadcrumb({\n type: 'default',\n category: span.type,\n message: span.name,\n level: span.errorInfo ? 'error' : 'info',\n data: {\n spanId: span.id,\n traceId: span.traceId,\n ...(span.input && { input: this.serializeValue(span.input) }),\n ...(span.output && { output: this.serializeValue(span.output) }),\n ...(span.metadata && { metadata: span.metadata }),\n ...(span.attributes && { attributes: span.attributes }),\n },\n timestamp: span.startTime.getTime() / 1000,\n });\n }\n\n private async handleSpanStarted(span: AnyExportedSpan): Promise<void> {\n const resolvedParentId = this.resolveParentSpanId(span.parentSpanId);\n\n const sentrySpan = Sentry.startInactiveSpan({\n op: this.getOperationType(span),\n name: getGenAISpanName(span),\n startTime: span.startTime.getTime(),\n forceTransaction: span.isRootSpan,\n parentSpan: resolvedParentId ? this.spanMap.get(resolvedParentId)?.span : undefined,\n });\n\n sentrySpan.setAttributes(this.buildSpanAttributes(span));\n\n this.spanMap.set(span.id, {\n span: sentrySpan,\n spanType: span.type,\n });\n\n // Track tool calls as children of MODEL_GENERATION spans for gen_ai.response.tool_calls attribute\n if (span.type === SpanType.TOOL_CALL && resolvedParentId) {\n this.trackToolCallForParent(span, resolvedParentId);\n }\n }\n\n private async handleSpanUpdated(span: AnyExportedSpan): Promise<void> {\n const spanData = this.spanMap.get(span.id);\n if (!spanData) {\n this.logMissingSpan(span, 'span update');\n return;\n }\n // Attributes are set on SPAN_STARTED and finalized on SPAN_ENDED.\n // If dynamic updates become necessary, add spanData.span.setAttributes() here.\n }\n\n private async handleSpanEnded(span: AnyExportedSpan): Promise<void> {\n const spanData = this.spanMap.get(span.id);\n if (!spanData) {\n this.logMissingSpan(span, 'span end');\n return;\n }\n\n const { span: sentrySpan } = spanData;\n\n sentrySpan.setAttributes(this.buildSpanAttributes(span));\n\n if (span.type === SpanType.MODEL_GENERATION) {\n // Set gen_ai.response.tool_calls if this generation had tool calls\n this.applyToolCallsAttribute(spanData);\n\n const resolvedParentId = this.resolveParentSpanId(span.parentSpanId);\n if (resolvedParentId) {\n const parentData = this.spanMap.get(resolvedParentId);\n if (parentData?.spanType === SpanType.AGENT_RUN) {\n const modelAttr = span.attributes as ModelGenerationAttributes;\n parentData.generation = {\n model: modelAttr.model,\n output: span.output,\n usage: modelAttr.usage,\n };\n }\n }\n }\n\n if (span.type === SpanType.AGENT_RUN) {\n // Apply token usage from the single child MODEL_GENERATION span\n // (there is only ever one MODEL_GENERATION span per AGENT_RUN)\n this.applyUsageFromGeneration(spanData);\n\n this.setGenerationResponseAttributes(spanData);\n }\n\n if (span.errorInfo) {\n sentrySpan.setStatus({\n code: 2,\n message: span.errorInfo.message,\n });\n\n Sentry.captureException(span.errorInfo.message, {\n contexts: {\n trace: { trace_id: span.traceId, span_id: span.id },\n span_info: {\n name: span.name,\n type: span.type,\n error_id: span.errorInfo.id,\n error_category: span.errorInfo.category,\n },\n },\n });\n }\n\n const endTime = span.endTime ? span.endTime.getTime() : undefined;\n sentrySpan.end(endTime);\n this.spanMap.delete(span.id);\n }\n\n // ============================================================================\n // Span Creation Helpers\n // ============================================================================\n\n private resolveParentSpanId(parentSpanId: string | undefined): string | undefined {\n if (!parentSpanId) return undefined;\n\n let currentParentId: string | undefined = parentSpanId;\n while (currentParentId && this.skippedSpans.has(currentParentId)) {\n currentParentId = this.skippedSpans.get(currentParentId);\n if (!currentParentId) break;\n }\n\n return currentParentId;\n }\n\n private getOperationType(span: AnyExportedSpan): string {\n const config = SPAN_TYPE_CONFIG[span.type];\n return config ? config.opType : 'ai.span';\n }\n\n private buildSpanAttributes(span: AnyExportedSpan): Record<string, any> {\n const attributes = getGenAIAttributes(span) as Record<string, any>;\n\n attributes[ATTRIBUTE_KEYS.SPAN_TYPE] = span.type;\n attributes[ATTRIBUTE_KEYS.ORIGIN] = 'auto.ai.mastra';\n\n if (span.metadata) {\n Object.entries(span.metadata).forEach(([key, value]) => {\n if (value !== undefined && value !== null && key !== 'langfuse') {\n attributes[`metadata.${key}`] = this.serializeValue(value);\n }\n });\n }\n\n this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.TAGS, span.tags?.join(','));\n\n this.addInputOutputAttributes(attributes, span);\n\n if (span.type === SpanType.MODEL_GENERATION) {\n this.addModelGenerationAttributes(attributes, span);\n }\n\n if (span.type === SpanType.TOOL_CALL) {\n this.addToolCallAttributes(attributes, span);\n }\n\n if (span.type === SpanType.AGENT_RUN) {\n this.addAgentRunAttributes(attributes, span);\n }\n\n if (span.type === SpanType.WORKFLOW_RUN) {\n const workflowAttr = span.attributes as WorkflowRunAttributes;\n this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.WORKFLOW_ID, this.getEntityName(span));\n this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.WORKFLOW_STATUS, workflowAttr.status);\n }\n\n if (span.type === SpanType.WORKFLOW_STEP) {\n const stepAttr = span.attributes as WorkflowStepAttributes;\n this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.WORKFLOW_STEP_ID, this.getEntityName(span));\n this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.WORKFLOW_STEP_STATUS, stepAttr.status);\n }\n\n return attributes;\n }\n\n // ============================================================================\n // Sentry-Specific Attribute Formatters\n // ============================================================================\n\n /**\n * Adds Sentry-specific input/output attributes that complement GenAI semantic conventions.\n * Adds 'input' and 'output' keys for Sentry UI compatibility.\n */\n private addInputOutputAttributes(attributes: Record<string, any>, span: AnyExportedSpan): void {\n if (span.input !== undefined) {\n attributes[ATTRIBUTE_KEYS.INPUT] = this.serializeValue(span.input);\n }\n\n if (span.output !== undefined) {\n attributes[ATTRIBUTE_KEYS.OUTPUT] = this.serializeValue(span.output);\n\n // Extract text for MODEL_GENERATION spans\n if (span.type === SpanType.MODEL_GENERATION) {\n const outputText = this.extractOutputText(span.output);\n if (outputText) {\n attributes[ATTRIBUTE_KEYS.GEN_AI_RESPONSE_TEXT] = outputText;\n }\n }\n }\n }\n\n /**\n * Adds Sentry-specific MODEL_GENERATION attributes that complement GenAI semantic conventions.\n */\n private addModelGenerationAttributes(attributes: Record<string, any>, span: AnyExportedSpan): void {\n const modelAttr = span.attributes as ModelGenerationAttributes;\n\n if (modelAttr.streaming !== undefined) {\n attributes[ATTRIBUTE_KEYS.GEN_AI_REQUEST_STREAM] = modelAttr.streaming;\n attributes[ATTRIBUTE_KEYS.GEN_AI_RESPONSE_STREAMING] = modelAttr.streaming;\n }\n\n this.setAttributeIfDefined(\n attributes,\n ATTRIBUTE_KEYS.GEN_AI_COMPLETION_START_TIME,\n modelAttr.completionStartTime?.toISOString(),\n );\n\n if (modelAttr.usage) {\n const totalTokens = (modelAttr.usage.inputTokens || 0) + (modelAttr.usage.outputTokens || 0);\n if (totalTokens > 0) {\n attributes[ATTRIBUTE_KEYS.GEN_AI_USAGE_TOTAL_TOKENS] = totalTokens;\n }\n }\n }\n\n /**\n * Adds Sentry-specific TOOL_CALL attributes that complement GenAI semantic conventions.\n */\n private addToolCallAttributes(attributes: Record<string, any>, span: AnyExportedSpan): void {\n const toolAttr = span.attributes as ToolCallAttributes;\n\n this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.TOOL_SUCCESS, toolAttr.success);\n this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.GEN_AI_TOOL_CALL_ID, span.metadata?.toolCallId);\n }\n\n /**\n * Adds Sentry-specific AGENT_RUN attributes that complement GenAI semantic conventions.\n */\n private addAgentRunAttributes(attributes: Record<string, any>, span: AnyExportedSpan): void {\n const agentAttr = span.attributes as AgentRunAttributes;\n\n const agentName = this.getEntityName(span);\n if (agentName) {\n attributes[ATTRIBUTE_KEYS.GEN_AI_PIPELINE_NAME] = agentName;\n }\n\n this.setAttributeIfDefined(attributes, ATTRIBUTE_KEYS.GEN_AI_AGENT_PROMPT, agentAttr.prompt);\n }\n\n // ============================================================================\n // Token Usage Management\n // ============================================================================\n\n /**\n * Applies token usage from the MODEL_GENERATION span to the AGENT_RUN span attributes.\n * Reads usage directly from the generation field.\n * Called when AGENT_RUN spans end to set gen_ai.usage.* attributes.\n */\n private applyUsageFromGeneration(spanData: SpanData): void {\n const usage = spanData.generation?.usage;\n if (!usage) return;\n\n const inputTokens = usage.inputTokens || 0;\n const outputTokens = usage.outputTokens || 0;\n\n if (inputTokens > 0) {\n spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_USAGE_INPUT_TOKENS, inputTokens);\n }\n if (outputTokens > 0) {\n spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_USAGE_OUTPUT_TOKENS, outputTokens);\n }\n\n const totalTokens = inputTokens + outputTokens;\n if (totalTokens > 0) {\n spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_USAGE_TOTAL_TOKENS, totalTokens);\n }\n\n const cacheReadTokens = usage.inputDetails?.cacheRead || 0;\n const cacheWriteTokens = usage.inputDetails?.cacheWrite || 0;\n const reasoningTokens = usage.outputDetails?.reasoning || 0;\n\n if (cacheReadTokens > 0) {\n spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_USAGE_CACHE_READ_TOKENS, cacheReadTokens);\n }\n if (cacheWriteTokens > 0) {\n spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_USAGE_CACHE_WRITE_TOKENS, cacheWriteTokens);\n }\n if (reasoningTokens > 0) {\n spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_USAGE_REASONING_TOKENS, reasoningTokens);\n }\n }\n\n /**\n * Sets gen_ai.response.model and gen_ai.response.text from the MODEL_GENERATION.\n * Only applies to AGENT_RUN spans.\n */\n private setGenerationResponseAttributes(spanData: SpanData): void {\n if (!spanData.generation) return;\n\n if (spanData.generation.model) {\n spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_RESPONSE_MODEL, spanData.generation.model);\n }\n\n if (spanData.generation.output) {\n const outputText = this.extractOutputText(spanData.generation.output);\n if (outputText) {\n spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_RESPONSE_TEXT, outputText);\n }\n }\n }\n\n /**\n * Tracks a TOOL_CALL span as a child of its parent MODEL_GENERATION span.\n * This builds the tool_calls array for gen_ai.response.tool_calls attribute.\n */\n private trackToolCallForParent(span: AnyExportedSpan, parentId: string): void {\n const parentSpanData = this.spanMap.get(parentId);\n if (!parentSpanData || parentSpanData.spanType !== SpanType.MODEL_GENERATION) {\n return;\n }\n\n const toolAttr = span.attributes as ToolCallAttributes;\n if (!parentSpanData.toolCalls) {\n parentSpanData.toolCalls = [];\n }\n\n parentSpanData.toolCalls.push({\n name: this.getEntityName(span),\n id: span.metadata?.toolCallId,\n type: toolAttr.toolType || 'function',\n });\n }\n\n /**\n * Applies the gen_ai.response.tool_calls attribute to MODEL_GENERATION spans.\n * Called when MODEL_GENERATION spans end if they have child tool calls.\n */\n private applyToolCallsAttribute(spanData: SpanData): void {\n if (!spanData.toolCalls || spanData.toolCalls.length === 0) {\n return;\n }\n\n spanData.span.setAttribute(ATTRIBUTE_KEYS.GEN_AI_RESPONSE_TOOL_CALLS, JSON.stringify(spanData.toolCalls));\n }\n\n // ============================================================================\n // Utility Helpers\n // ============================================================================\n\n private logMissingSpan(span: AnyExportedSpan, operation: string): void {\n this.logger.warn(`Sentry exporter: No Sentry span found for ${operation}`, {\n traceId: span.traceId,\n spanId: span.id,\n spanName: span.name,\n });\n }\n\n private getEntityName(span: AnyExportedSpan): string {\n return span.entityName || span.entityId || 'unknown';\n }\n\n private extractOutputText(output: any): string | undefined {\n if (!output) return undefined;\n if (typeof output === 'string') return output;\n if (output.text && typeof output.text === 'string') return output.text;\n if (output.content && typeof output.content === 'string') return output.content;\n if (output.message?.content && typeof output.message.content === 'string') return output.message.content;\n return undefined;\n }\n\n private serializeValue(value: any): any {\n if (value === null || value === undefined) return value;\n if (typeof value === 'object') {\n try {\n return JSON.stringify(value);\n } catch {\n return String(value);\n }\n }\n return value;\n }\n\n private setAttributeIfDefined(attributes: Record<string, any>, key: string, value: any): void {\n if (value !== undefined && value !== null) {\n attributes[key] = value;\n }\n }\n\n // ============================================================================\n // Shutdown\n // ============================================================================\n\n async shutdown(): Promise<void> {\n for (const [spanId, spanData] of this.spanMap.entries()) {\n try {\n spanData.span.end();\n } catch (error) {\n this.logger.error('Sentry exporter: Error ending span during shutdown', { spanId, error });\n }\n }\n\n this.spanMap.clear();\n this.skippedSpans.clear();\n\n if (this.initialized) {\n await Sentry.close(2000);\n }\n\n await super.shutdown();\n }\n}\n"]}
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Sentry Observability Provider for Mastra
3
+ *
4
+ * This package provides Sentry-specific observability features for Mastra applications.
5
+ * Includes tracing support with OpenTelemetry semantic conventions for AI.
6
+ */
7
+ export * from './tracing.js';
8
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAGH,cAAc,WAAW,CAAC"}