@sentry/core
Version:
Base implementation for all Sentry JavaScript SDKs
186 lines (165 loc) • 5.38 kB
JavaScript
import { getClient } from '../../currentScopes.js';
import { SEMANTIC_ATTRIBUTE_SENTRY_SOURCE, SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, SEMANTIC_ATTRIBUTE_SENTRY_OP } from '../../semanticAttributes.js';
import { startSpan } from '../../tracing/trace.js';
import { buildTypeSpecificAttributes, buildTransportAttributes } from './attributeExtraction.js';
import { MCP_SERVER_OP_VALUE, MCP_METHOD_NAME_ATTRIBUTE, MCP_ROUTE_SOURCE_VALUE, MCP_NOTIFICATION_SERVER_TO_CLIENT_OP_VALUE, MCP_NOTIFICATION_ORIGIN_VALUE, MCP_NOTIFICATION_CLIENT_TO_SERVER_OP_VALUE, MCP_FUNCTION_ORIGIN_VALUE } from './attributes.js';
import { extractTargetInfo } from './methodConfig.js';
import { filterMcpPiiFromSpanData } from './piiFiltering.js';
/**
* Span creation and management functions for MCP server instrumentation
*
* Provides unified span creation following OpenTelemetry MCP semantic conventions and our opinitionated take on MCP.
* Handles both request and notification spans with attribute extraction.
*/
/**
* Creates a span name based on the method and target
* @internal
* @param method - MCP method name
* @param target - Optional target identifier
* @returns Formatted span name
*/
function createSpanName(method, target) {
return target ? `${method} ${target}` : method;
}
/**
* Build Sentry-specific attributes based on span type
* @internal
* @param type - Span type configuration
* @returns Sentry-specific attributes
*/
function buildSentryAttributes(type) {
let op;
let origin;
switch (type) {
case 'request':
op = MCP_SERVER_OP_VALUE;
origin = MCP_FUNCTION_ORIGIN_VALUE;
break;
case 'notification-incoming':
op = MCP_NOTIFICATION_CLIENT_TO_SERVER_OP_VALUE;
origin = MCP_NOTIFICATION_ORIGIN_VALUE;
break;
case 'notification-outgoing':
op = MCP_NOTIFICATION_SERVER_TO_CLIENT_OP_VALUE;
origin = MCP_NOTIFICATION_ORIGIN_VALUE;
break;
}
return {
[SEMANTIC_ATTRIBUTE_SENTRY_OP]: op,
[SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN]: origin,
[SEMANTIC_ATTRIBUTE_SENTRY_SOURCE]: MCP_ROUTE_SOURCE_VALUE,
};
}
/**
* Unified builder for creating MCP spans
* @internal
* @param config - Span configuration
* @returns Created span
*/
function createMcpSpan(config) {
const { type, message, transport, extra, callback } = config;
const { method } = message;
const params = message.params ;
// Determine span name based on type and OTEL conventions
let spanName;
if (type === 'request') {
const targetInfo = extractTargetInfo(method, params || {});
spanName = createSpanName(method, targetInfo.target);
} else {
// For notifications, use method name directly per OpenTelemetry conventions
spanName = method;
}
const rawAttributes = {
...buildTransportAttributes(transport, extra),
[MCP_METHOD_NAME_ATTRIBUTE]: method,
...buildTypeSpecificAttributes(type, message, params),
...buildSentryAttributes(type),
};
const client = getClient();
const sendDefaultPii = Boolean(client?.getOptions().sendDefaultPii);
const attributes = filterMcpPiiFromSpanData(rawAttributes, sendDefaultPii) ;
return startSpan(
{
name: spanName,
forceTransaction: true,
attributes,
},
callback,
);
}
/**
* Creates a span for incoming MCP notifications
* @param jsonRpcMessage - Notification message
* @param transport - MCP transport instance
* @param extra - Extra handler data
* @param callback - Span execution callback
* @returns Span execution result
*/
function createMcpNotificationSpan(
jsonRpcMessage,
transport,
extra,
callback,
) {
return createMcpSpan({
type: 'notification-incoming',
message: jsonRpcMessage,
transport,
extra,
callback,
});
}
/**
* Creates a span for outgoing MCP notifications
* @param jsonRpcMessage - Notification message
* @param transport - MCP transport instance
* @param callback - Span execution callback
* @returns Span execution result
*/
function createMcpOutgoingNotificationSpan(
jsonRpcMessage,
transport,
callback,
) {
return createMcpSpan({
type: 'notification-outgoing',
message: jsonRpcMessage,
transport,
callback,
});
}
/**
* Builds span configuration for MCP server requests
* @param jsonRpcMessage - Request message
* @param transport - MCP transport instance
* @param extra - Optional extra handler data
* @returns Span configuration object
*/
function buildMcpServerSpanConfig(
jsonRpcMessage,
transport,
extra,
)
{
const { method } = jsonRpcMessage;
const params = jsonRpcMessage.params ;
const targetInfo = extractTargetInfo(method, params || {});
const spanName = createSpanName(method, targetInfo.target);
const rawAttributes = {
...buildTransportAttributes(transport, extra),
[MCP_METHOD_NAME_ATTRIBUTE]: method,
...buildTypeSpecificAttributes('request', jsonRpcMessage, params),
...buildSentryAttributes('request'),
};
const client = getClient();
const sendDefaultPii = Boolean(client?.getOptions().sendDefaultPii);
const attributes = filterMcpPiiFromSpanData(rawAttributes, sendDefaultPii) ;
return {
name: spanName,
op: MCP_SERVER_OP_VALUE,
forceTransaction: true,
attributes,
};
}
export { buildMcpServerSpanConfig, createMcpNotificationSpan, createMcpOutgoingNotificationSpan };
//# sourceMappingURL=spans.js.map