UNPKG

@sentry/core

Version:
136 lines (122 loc) 4.39 kB
import { GEN_AI_USAGE_INPUT_TOKENS_ATTRIBUTE, GEN_AI_USAGE_OUTPUT_TOKENS_ATTRIBUTE, GEN_AI_REQUEST_MESSAGES_ATTRIBUTE } from '../ai/gen-ai-attributes.js'; import { getTruncatedJsonString } from '../ai/utils.js'; import { toolCallSpanMap } from './constants.js'; import { AI_PROMPT_ATTRIBUTE, AI_PROMPT_MESSAGES_ATTRIBUTE } from './vercel-ai-attributes.js'; /** * Accumulates token data from a span to its parent in the token accumulator map. * This function extracts token usage from the current span and adds it to the * accumulated totals for its parent span. */ function accumulateTokensForParent(span, tokenAccumulator) { const parentSpanId = span.parent_span_id; if (!parentSpanId) { return; } const inputTokens = span.data[GEN_AI_USAGE_INPUT_TOKENS_ATTRIBUTE]; const outputTokens = span.data[GEN_AI_USAGE_OUTPUT_TOKENS_ATTRIBUTE]; if (typeof inputTokens === 'number' || typeof outputTokens === 'number') { const existing = tokenAccumulator.get(parentSpanId) || { inputTokens: 0, outputTokens: 0 }; if (typeof inputTokens === 'number') { existing.inputTokens += inputTokens; } if (typeof outputTokens === 'number') { existing.outputTokens += outputTokens; } tokenAccumulator.set(parentSpanId, existing); } } /** * Applies accumulated token data to the `gen_ai.invoke_agent` span. * Only immediate children of the `gen_ai.invoke_agent` span are considered, * since aggregation will automatically occur for each parent span. */ function applyAccumulatedTokens( spanOrTrace, tokenAccumulator, ) { const accumulated = tokenAccumulator.get(spanOrTrace.span_id); if (!accumulated || !spanOrTrace.data) { return; } if (accumulated.inputTokens > 0) { spanOrTrace.data[GEN_AI_USAGE_INPUT_TOKENS_ATTRIBUTE] = accumulated.inputTokens; } if (accumulated.outputTokens > 0) { spanOrTrace.data[GEN_AI_USAGE_OUTPUT_TOKENS_ATTRIBUTE] = accumulated.outputTokens; } if (accumulated.inputTokens > 0 || accumulated.outputTokens > 0) { spanOrTrace.data['gen_ai.usage.total_tokens'] = accumulated.inputTokens + accumulated.outputTokens; } } /** * Get the span associated with a tool call ID */ function _INTERNAL_getSpanForToolCallId(toolCallId) { return toolCallSpanMap.get(toolCallId); } /** * Clean up the span mapping for a tool call ID */ function _INTERNAL_cleanupToolCallSpan(toolCallId) { toolCallSpanMap.delete(toolCallId); } /** * Convert an array of tool strings to a JSON string */ function convertAvailableToolsToJsonString(tools) { const toolObjects = tools.map(tool => { if (typeof tool === 'string') { try { return JSON.parse(tool); } catch { return tool; } } return tool; }); return JSON.stringify(toolObjects); } /** * Convert the prompt string to messages array */ function convertPromptToMessages(prompt) { try { const p = JSON.parse(prompt); if (!!p && typeof p === 'object') { const { prompt, system } = p; if (typeof prompt === 'string' || typeof system === 'string') { const messages = []; if (typeof system === 'string') { messages.push({ role: 'system', content: system }); } if (typeof prompt === 'string') { messages.push({ role: 'user', content: prompt }); } return messages; } } // eslint-disable-next-line no-empty } catch {} return []; } /** * Generate a request.messages JSON array from the prompt field in the * invoke_agent op */ function requestMessagesFromPrompt(span, attributes) { if (attributes[AI_PROMPT_ATTRIBUTE]) { const truncatedPrompt = getTruncatedJsonString(attributes[AI_PROMPT_ATTRIBUTE] ); span.setAttribute('gen_ai.prompt', truncatedPrompt); } const prompt = attributes[AI_PROMPT_ATTRIBUTE]; if ( typeof prompt === 'string' && !attributes[GEN_AI_REQUEST_MESSAGES_ATTRIBUTE] && !attributes[AI_PROMPT_MESSAGES_ATTRIBUTE] ) { const messages = convertPromptToMessages(prompt); if (messages.length) span.setAttribute(GEN_AI_REQUEST_MESSAGES_ATTRIBUTE, getTruncatedJsonString(messages)); } } export { _INTERNAL_cleanupToolCallSpan, _INTERNAL_getSpanForToolCallId, accumulateTokensForParent, applyAccumulatedTokens, convertAvailableToolsToJsonString, convertPromptToMessages, requestMessagesFromPrompt }; //# sourceMappingURL=utils.js.map