Next.js website for Rocky Mountain Vending company featuring: - Product catalog with Stripe integration - Service areas and parts pages - Admin dashboard with Clerk authentication - SEO optimized pages with JSON-LD structured data Co-authored-by: Cursor <cursoragent@cursor.com>
186 lines
5.3 KiB
Text
186 lines
5.3 KiB
Text
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 } 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';
|
|
import { buildTransportAttributes } from './sessionExtraction.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
|