@posthog/ai 6.1.1 → 6.2.0

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.
@@ -4,6 +4,8 @@ var ai = require('ai');
4
4
  var uuid = require('uuid');
5
5
  var buffer = require('buffer');
6
6
 
7
+ var version = "6.2.0";
8
+
7
9
  // limit large outputs by truncating to 200kb (approx 200k bytes)
8
10
  const MAX_OUTPUT_SIZE = 200000;
9
11
  const STRING_FORMAT = 'utf8';
@@ -43,9 +45,8 @@ const truncate = str => {
43
45
  */
44
46
  const extractAvailableToolCalls = (provider, params) => {
45
47
  {
46
- // Vercel AI SDK stores tools in params.mode.tools when mode type is 'regular'
47
- if (params.mode?.type === 'regular' && params.mode.tools) {
48
- return params.mode.tools;
48
+ if (params.tools) {
49
+ return params.tools;
49
50
  }
50
51
  return null;
51
52
  }
@@ -118,6 +119,8 @@ const sendEventToPosthog = async ({
118
119
  } : {})
119
120
  };
120
121
  const properties = {
122
+ $ai_lib: 'posthog-ai',
123
+ $ai_lib_version: version,
121
124
  $ai_provider: params.posthogProviderOverride ?? provider,
122
125
  $ai_model: params.posthogModelOverride ?? model,
123
126
  $ai_model_parameters: getModelParams(params),
@@ -203,6 +206,8 @@ function redactBase64DataUrl(str) {
203
206
  return str;
204
207
  }
205
208
 
209
+ // Content types for the output array
210
+
206
211
  const mapVercelParams = params => {
207
212
  return {
208
213
  temperature: params.temperature,
@@ -493,6 +498,8 @@ const createInstrumentationMiddleware = (phClient, model, options) => {
493
498
  const availableTools = extractAvailableToolCalls('vercel', params);
494
499
  const baseURL = ''; // cannot currently get baseURL from vercel
495
500
 
501
+ // Map to track in-progress tool calls
502
+ const toolCallsInProgress = new Map();
496
503
  try {
497
504
  const {
498
505
  stream,
@@ -507,6 +514,35 @@ const createInstrumentationMiddleware = (phClient, model, options) => {
507
514
  if (chunk.type === 'reasoning-delta') {
508
515
  reasoningText += chunk.delta; // New in v5
509
516
  }
517
+
518
+ // Handle tool call chunks
519
+ if (chunk.type === 'tool-input-start') {
520
+ // Initialize a new tool call
521
+ toolCallsInProgress.set(chunk.id, {
522
+ toolCallId: chunk.id,
523
+ toolName: chunk.toolName,
524
+ input: ''
525
+ });
526
+ }
527
+ if (chunk.type === 'tool-input-delta') {
528
+ // Accumulate tool call arguments
529
+ const toolCall = toolCallsInProgress.get(chunk.id);
530
+ if (toolCall) {
531
+ toolCall.input += chunk.delta;
532
+ }
533
+ }
534
+ if (chunk.type === 'tool-input-end') {
535
+ // Tool call is complete, keep it in the map for final processing
536
+ // Nothing specific to do here, the tool call is already complete
537
+ }
538
+ if (chunk.type === 'tool-call') {
539
+ // Direct tool call chunk (complete tool call)
540
+ toolCallsInProgress.set(chunk.toolCallId, {
541
+ toolCallId: chunk.toolCallId,
542
+ toolName: chunk.toolName,
543
+ input: chunk.input
544
+ });
545
+ }
510
546
  if (chunk.type === 'finish') {
511
547
  const providerMetadata = chunk.providerMetadata;
512
548
  const additionalTokenValues = {
@@ -541,6 +577,20 @@ const createInstrumentationMiddleware = (phClient, model, options) => {
541
577
  });
542
578
  }
543
579
 
580
+ // Add completed tool calls to content
581
+ for (const toolCall of toolCallsInProgress.values()) {
582
+ if (toolCall.toolName) {
583
+ content.push({
584
+ type: 'tool-call',
585
+ id: toolCall.toolCallId,
586
+ function: {
587
+ name: toolCall.toolName,
588
+ arguments: toolCall.input
589
+ }
590
+ });
591
+ }
592
+ }
593
+
544
594
  // Structure output like mapVercelOutput does
545
595
  const output = content.length > 0 ? [{
546
596
  role: 'assistant',
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs","sources":["../../src/utils.ts","../../src/typeGuards.ts","../../src/sanitization.ts","../../src/vercel/middleware.ts"],"sourcesContent":["import { PostHog } from 'posthog-node'\nimport { Buffer } from 'buffer'\nimport OpenAIOrignal from 'openai'\nimport AnthropicOriginal from '@anthropic-ai/sdk'\nimport type { ChatCompletionTool } from 'openai/resources/chat/completions'\nimport type { Tool as GeminiTool } from '@google/genai'\nimport type { FormattedMessage, FormattedContent, TokenUsage } from './types'\n\ntype ChatCompletionCreateParamsBase = OpenAIOrignal.Chat.Completions.ChatCompletionCreateParams\ntype MessageCreateParams = AnthropicOriginal.Messages.MessageCreateParams\ntype ResponseCreateParams = OpenAIOrignal.Responses.ResponseCreateParams\ntype AnthropicTool = AnthropicOriginal.Tool\n\n// limit large outputs by truncating to 200kb (approx 200k bytes)\nexport const MAX_OUTPUT_SIZE = 200000\nconst STRING_FORMAT = 'utf8'\n\nexport interface MonitoringParams {\n posthogDistinctId?: string\n posthogTraceId?: string\n posthogProperties?: Record<string, any>\n posthogPrivacyMode?: boolean\n posthogGroups?: Record<string, any>\n posthogModelOverride?: string\n posthogProviderOverride?: string\n posthogCostOverride?: CostOverride\n posthogCaptureImmediate?: boolean\n}\n\nexport interface CostOverride {\n inputCost: number\n outputCost: number\n}\n\nexport const getModelParams = (\n params: ((ChatCompletionCreateParamsBase | MessageCreateParams | ResponseCreateParams) & MonitoringParams) | null\n): Record<string, any> => {\n if (!params) {\n return {}\n }\n const modelParams: Record<string, any> = {}\n const paramKeys = [\n 'temperature',\n 'max_tokens',\n 'max_completion_tokens',\n 'top_p',\n 'frequency_penalty',\n 'presence_penalty',\n 'n',\n 'stop',\n 'stream',\n 'streaming',\n ] as const\n\n for (const key of paramKeys) {\n if (key in params && (params as any)[key] !== undefined) {\n modelParams[key] = (params as any)[key]\n }\n }\n return modelParams\n}\n\n/**\n * Helper to format responses (non-streaming) for consumption, mirroring Python's openai vs. anthropic approach.\n */\nexport const formatResponse = (response: any, provider: string): FormattedMessage[] => {\n if (!response) {\n return []\n }\n if (provider === 'anthropic') {\n return formatResponseAnthropic(response)\n } else if (provider === 'openai') {\n return formatResponseOpenAI(response)\n } else if (provider === 'gemini') {\n return formatResponseGemini(response)\n }\n return []\n}\n\nexport const formatResponseAnthropic = (response: any): FormattedMessage[] => {\n const output: FormattedMessage[] = []\n const content: FormattedContent = []\n\n for (const choice of response.content ?? []) {\n if (choice?.type === 'text' && choice?.text) {\n content.push({ type: 'text', text: choice.text })\n } else if (choice?.type === 'tool_use' && choice?.name && choice?.id) {\n content.push({\n type: 'function',\n id: choice.id,\n function: {\n name: choice.name,\n arguments: choice.input || {},\n },\n })\n }\n }\n\n if (content.length > 0) {\n output.push({\n role: 'assistant',\n content,\n })\n }\n\n return output\n}\n\nexport const formatResponseOpenAI = (response: any): FormattedMessage[] => {\n const output: FormattedMessage[] = []\n\n if (response.choices) {\n for (const choice of response.choices) {\n const content: FormattedContent = []\n let role = 'assistant'\n\n if (choice.message) {\n if (choice.message.role) {\n role = choice.message.role\n }\n\n if (choice.message.content) {\n content.push({ type: 'text', text: choice.message.content })\n }\n\n if (choice.message.tool_calls) {\n for (const toolCall of choice.message.tool_calls) {\n content.push({\n type: 'function',\n id: toolCall.id,\n function: {\n name: toolCall.function.name,\n arguments: toolCall.function.arguments,\n },\n })\n }\n }\n }\n\n if (content.length > 0) {\n output.push({\n role,\n content,\n })\n }\n }\n }\n\n // Handle Responses API format\n if (response.output) {\n const content: FormattedContent = []\n let role = 'assistant'\n\n for (const item of response.output) {\n if (item.type === 'message') {\n role = item.role\n\n if (item.content && Array.isArray(item.content)) {\n for (const contentItem of item.content) {\n if (contentItem.type === 'output_text' && contentItem.text) {\n content.push({ type: 'text', text: contentItem.text })\n } else if (contentItem.text) {\n content.push({ type: 'text', text: contentItem.text })\n } else if (contentItem.type === 'input_image' && contentItem.image_url) {\n content.push({\n type: 'image',\n image: contentItem.image_url,\n })\n }\n }\n } else if (item.content) {\n content.push({ type: 'text', text: String(item.content) })\n }\n } else if (item.type === 'function_call') {\n content.push({\n type: 'function',\n id: item.call_id || item.id || '',\n function: {\n name: item.name,\n arguments: item.arguments || {},\n },\n })\n }\n }\n\n if (content.length > 0) {\n output.push({\n role,\n content,\n })\n }\n }\n\n return output\n}\n\nexport const formatResponseGemini = (response: any): FormattedMessage[] => {\n const output: FormattedMessage[] = []\n\n if (response.candidates && Array.isArray(response.candidates)) {\n for (const candidate of response.candidates) {\n if (candidate.content && candidate.content.parts) {\n const content: FormattedContent = []\n\n for (const part of candidate.content.parts) {\n if (part.text) {\n content.push({ type: 'text', text: part.text })\n } else if (part.functionCall) {\n content.push({\n type: 'function',\n function: {\n name: part.functionCall.name,\n arguments: part.functionCall.args,\n },\n })\n }\n }\n\n if (content.length > 0) {\n output.push({\n role: 'assistant',\n content,\n })\n }\n } else if (candidate.text) {\n output.push({\n role: 'assistant',\n content: [{ type: 'text', text: candidate.text }],\n })\n }\n }\n } else if (response.text) {\n output.push({\n role: 'assistant',\n content: [{ type: 'text', text: response.text }],\n })\n }\n\n return output\n}\n\nexport const mergeSystemPrompt = (params: MessageCreateParams & MonitoringParams, provider: string): any => {\n if (provider == 'anthropic') {\n const messages = params.messages || []\n if (!(params as any).system) {\n return messages\n }\n const systemMessage = (params as any).system\n return [{ role: 'system', content: systemMessage }, ...messages]\n }\n return params.messages\n}\n\nexport const withPrivacyMode = (client: PostHog, privacyMode: boolean, input: any): any => {\n return (client as any).privacy_mode || privacyMode ? null : input\n}\n\nexport const truncate = (str: string): string => {\n try {\n const buffer = Buffer.from(str, STRING_FORMAT)\n if (buffer.length <= MAX_OUTPUT_SIZE) {\n return str\n }\n const truncatedBuffer = buffer.slice(0, MAX_OUTPUT_SIZE)\n return `${truncatedBuffer.toString(STRING_FORMAT)}... [truncated]`\n } catch (error) {\n console.error('Error truncating, likely not a string')\n return str\n }\n}\n\n/**\n * Extract available tool calls from the request parameters.\n * These are the tools provided to the LLM, not the tool calls in the response.\n */\nexport const extractAvailableToolCalls = (\n provider: string,\n params: any\n): ChatCompletionTool[] | AnthropicTool[] | GeminiTool[] | null => {\n if (provider === 'anthropic') {\n if (params.tools) {\n return params.tools\n }\n\n return null\n } else if (provider === 'gemini') {\n if (params.config && params.config.tools) {\n return params.config.tools\n }\n\n return null\n } else if (provider === 'openai') {\n if (params.tools) {\n return params.tools\n }\n\n return null\n } else if (provider === 'vercel') {\n // Vercel AI SDK stores tools in params.mode.tools when mode type is 'regular'\n if (params.mode?.type === 'regular' && params.mode.tools) {\n return params.mode.tools\n }\n\n return null\n }\n\n return null\n}\n\nexport type SendEventToPosthogParams = {\n client: PostHog\n distinctId?: string\n traceId: string\n model: string\n provider: string\n input: any\n output: any\n latency: number\n baseURL: string\n httpStatus: number\n usage?: TokenUsage\n params: (ChatCompletionCreateParamsBase | MessageCreateParams | ResponseCreateParams) & MonitoringParams\n isError?: boolean\n error?: string\n tools?: ChatCompletionTool[] | AnthropicTool[] | GeminiTool[] | null\n captureImmediate?: boolean\n}\n\nfunction sanitizeValues(obj: any): any {\n if (obj === undefined || obj === null) {\n return obj\n }\n const jsonSafe = JSON.parse(JSON.stringify(obj))\n if (typeof jsonSafe === 'string') {\n return Buffer.from(jsonSafe, STRING_FORMAT).toString(STRING_FORMAT)\n } else if (Array.isArray(jsonSafe)) {\n return jsonSafe.map(sanitizeValues)\n } else if (jsonSafe && typeof jsonSafe === 'object') {\n return Object.fromEntries(Object.entries(jsonSafe).map(([k, v]) => [k, sanitizeValues(v)]))\n }\n return jsonSafe\n}\n\nexport const sendEventToPosthog = async ({\n client,\n distinctId,\n traceId,\n model,\n provider,\n input,\n output,\n latency,\n baseURL,\n params,\n httpStatus = 200,\n usage = {},\n isError = false,\n error,\n tools,\n captureImmediate = false,\n}: SendEventToPosthogParams): Promise<void> => {\n if (!client.capture) {\n return Promise.resolve()\n }\n // sanitize input and output for UTF-8 validity\n const safeInput = sanitizeValues(input)\n const safeOutput = sanitizeValues(output)\n const safeError = sanitizeValues(error)\n\n let errorData = {}\n if (isError) {\n errorData = {\n $ai_is_error: true,\n $ai_error: safeError,\n }\n }\n let costOverrideData = {}\n if (params.posthogCostOverride) {\n const inputCostUSD = (params.posthogCostOverride.inputCost ?? 0) * (usage.inputTokens ?? 0)\n const outputCostUSD = (params.posthogCostOverride.outputCost ?? 0) * (usage.outputTokens ?? 0)\n costOverrideData = {\n $ai_input_cost_usd: inputCostUSD,\n $ai_output_cost_usd: outputCostUSD,\n $ai_total_cost_usd: inputCostUSD + outputCostUSD,\n }\n }\n\n const additionalTokenValues = {\n ...(usage.reasoningTokens ? { $ai_reasoning_tokens: usage.reasoningTokens } : {}),\n ...(usage.cacheReadInputTokens ? { $ai_cache_read_input_tokens: usage.cacheReadInputTokens } : {}),\n ...(usage.cacheCreationInputTokens ? { $ai_cache_creation_input_tokens: usage.cacheCreationInputTokens } : {}),\n }\n\n const properties = {\n $ai_provider: params.posthogProviderOverride ?? provider,\n $ai_model: params.posthogModelOverride ?? model,\n $ai_model_parameters: getModelParams(params),\n $ai_input: withPrivacyMode(client, params.posthogPrivacyMode ?? false, safeInput),\n $ai_output_choices: withPrivacyMode(client, params.posthogPrivacyMode ?? false, safeOutput),\n $ai_http_status: httpStatus,\n $ai_input_tokens: usage.inputTokens ?? 0,\n $ai_output_tokens: usage.outputTokens ?? 0,\n ...additionalTokenValues,\n $ai_latency: latency,\n $ai_trace_id: traceId,\n $ai_base_url: baseURL,\n ...params.posthogProperties,\n ...(distinctId ? {} : { $process_person_profile: false }),\n ...(tools ? { $ai_tools: tools } : {}),\n ...errorData,\n ...costOverrideData,\n }\n\n const event = {\n distinctId: distinctId ?? traceId,\n event: '$ai_generation',\n properties,\n groups: params.posthogGroups,\n }\n\n if (captureImmediate) {\n // await capture promise to send single event in serverless environments\n await client.captureImmediate(event)\n } else {\n client.capture(event)\n }\n}\n","// Type guards for safer type checking\n\nexport const isString = (value: unknown): value is string => {\n return typeof value === 'string'\n}\n\nexport const isObject = (value: unknown): value is Record<string, unknown> => {\n return value !== null && typeof value === 'object' && !Array.isArray(value)\n}\n","import { isString, isObject } from './typeGuards'\n\nconst REDACTED_IMAGE_PLACEHOLDER = '[base64 image redacted]'\n\n// ============================================\n// Base64 Detection Helpers\n// ============================================\n\nconst isBase64DataUrl = (str: string): boolean => {\n return /^data:([^;]+);base64,/.test(str)\n}\n\nconst isValidUrl = (str: string): boolean => {\n try {\n new URL(str)\n return true\n } catch {\n // Not an absolute URL, check if it's a relative URL or path\n return str.startsWith('/') || str.startsWith('./') || str.startsWith('../')\n }\n}\n\nconst isRawBase64 = (str: string): boolean => {\n // Skip if it's a valid URL or path\n if (isValidUrl(str)) {\n return false\n }\n\n // Check if it's a valid base64 string\n // Base64 images are typically at least a few hundred chars, but we'll be conservative\n return str.length > 20 && /^[A-Za-z0-9+/]+=*$/.test(str)\n}\n\nexport function redactBase64DataUrl(str: string): string\nexport function redactBase64DataUrl(str: unknown): unknown\nexport function redactBase64DataUrl(str: unknown): unknown {\n if (!isString(str)) return str\n\n // Check for data URL format\n if (isBase64DataUrl(str)) {\n return REDACTED_IMAGE_PLACEHOLDER\n }\n\n // Check for raw base64 (Vercel sends raw base64 for inline images)\n if (isRawBase64(str)) {\n return REDACTED_IMAGE_PLACEHOLDER\n }\n\n return str\n}\n\n// ============================================\n// Common Message Processing\n// ============================================\n\ntype ContentTransformer = (item: unknown) => unknown\n\nconst processMessages = (messages: unknown, transformContent: ContentTransformer): unknown => {\n if (!messages) return messages\n\n const processContent = (content: unknown): unknown => {\n if (typeof content === 'string') return content\n\n if (!content) return content\n\n if (Array.isArray(content)) {\n return content.map(transformContent)\n }\n\n // Handle single object content\n return transformContent(content)\n }\n\n const processMessage = (msg: unknown): unknown => {\n if (!isObject(msg) || !('content' in msg)) return msg\n return { ...msg, content: processContent(msg.content) }\n }\n\n // Handle both arrays and single messages\n if (Array.isArray(messages)) {\n return messages.map(processMessage)\n }\n\n return processMessage(messages)\n}\n\n// ============================================\n// Provider-Specific Image Sanitizers\n// ============================================\n\nconst sanitizeOpenAIImage = (item: unknown): unknown => {\n if (!isObject(item)) return item\n\n // Handle image_url format\n if (item.type === 'image_url' && 'image_url' in item && isObject(item.image_url) && 'url' in item.image_url) {\n return {\n ...item,\n image_url: {\n ...item.image_url,\n url: redactBase64DataUrl(item.image_url.url),\n },\n }\n }\n\n return item\n}\n\nconst sanitizeOpenAIResponseImage = (item: unknown): unknown => {\n if (!isObject(item)) return item\n\n // Handle input_image format\n if (item.type === 'input_image' && 'image_url' in item) {\n return {\n ...item,\n image_url: redactBase64DataUrl(item.image_url),\n }\n }\n\n return item\n}\n\nconst sanitizeAnthropicImage = (item: unknown): unknown => {\n if (!isObject(item)) return item\n\n // Handle Anthropic's image format\n if (\n item.type === 'image' &&\n 'source' in item &&\n isObject(item.source) &&\n item.source.type === 'base64' &&\n 'data' in item.source\n ) {\n return {\n ...item,\n source: {\n ...item.source,\n data: REDACTED_IMAGE_PLACEHOLDER,\n },\n }\n }\n\n return item\n}\n\nconst sanitizeGeminiPart = (part: unknown): unknown => {\n if (!isObject(part)) return part\n\n // Handle Gemini's inline data format\n if ('inlineData' in part && isObject(part.inlineData) && 'data' in part.inlineData) {\n return {\n ...part,\n inlineData: {\n ...part.inlineData,\n data: REDACTED_IMAGE_PLACEHOLDER,\n },\n }\n }\n\n return part\n}\n\nconst processGeminiItem = (item: unknown): unknown => {\n if (!isObject(item)) return item\n\n // If it has parts, process them\n if ('parts' in item && item.parts) {\n const parts = Array.isArray(item.parts) ? item.parts.map(sanitizeGeminiPart) : sanitizeGeminiPart(item.parts)\n\n return { ...item, parts }\n }\n\n return item\n}\n\nconst sanitizeLangChainImage = (item: unknown): unknown => {\n if (!isObject(item)) return item\n\n // OpenAI style\n if (item.type === 'image_url' && 'image_url' in item && isObject(item.image_url) && 'url' in item.image_url) {\n return {\n ...item,\n image_url: {\n ...item.image_url,\n url: redactBase64DataUrl(item.image_url.url),\n },\n }\n }\n\n // Direct image with data field\n if (item.type === 'image' && 'data' in item) {\n return { ...item, data: redactBase64DataUrl(item.data) }\n }\n\n // Anthropic style\n if (item.type === 'image' && 'source' in item && isObject(item.source) && 'data' in item.source) {\n return {\n ...item,\n source: {\n ...item.source,\n data: redactBase64DataUrl(item.source.data),\n },\n }\n }\n\n // Google style\n if (item.type === 'media' && 'data' in item) {\n return { ...item, data: redactBase64DataUrl(item.data) }\n }\n\n return item\n}\n\n// Export individual sanitizers for tree-shaking\nexport const sanitizeOpenAI = (data: unknown): unknown => {\n return processMessages(data, sanitizeOpenAIImage)\n}\n\nexport const sanitizeOpenAIResponse = (data: unknown): unknown => {\n return processMessages(data, sanitizeOpenAIResponseImage)\n}\n\nexport const sanitizeAnthropic = (data: unknown): unknown => {\n return processMessages(data, sanitizeAnthropicImage)\n}\n\nexport const sanitizeGemini = (data: unknown): unknown => {\n // Gemini has a different structure with 'parts' directly on items instead of 'content'\n // So we need custom processing instead of using processMessages\n if (!data) return data\n\n if (Array.isArray(data)) {\n return data.map(processGeminiItem)\n }\n\n return processGeminiItem(data)\n}\n\nexport const sanitizeLangChain = (data: unknown): unknown => {\n return processMessages(data, sanitizeLangChainImage)\n}\n","import { wrapLanguageModel } from 'ai'\nimport type {\n LanguageModelV2,\n LanguageModelV2Content,\n LanguageModelV2Middleware,\n LanguageModelV2Prompt,\n LanguageModelV2StreamPart,\n} from '@ai-sdk/provider'\nimport { v4 as uuidv4 } from 'uuid'\nimport { PostHog } from 'posthog-node'\nimport { CostOverride, sendEventToPosthog, truncate, MAX_OUTPUT_SIZE, extractAvailableToolCalls } from '../utils'\nimport { Buffer } from 'buffer'\nimport { redactBase64DataUrl } from '../sanitization'\nimport { isString } from '../typeGuards'\n\ninterface ClientOptions {\n posthogDistinctId?: string\n posthogTraceId?: string\n posthogProperties?: Record<string, any>\n posthogPrivacyMode?: boolean\n posthogGroups?: Record<string, any>\n posthogModelOverride?: string\n posthogProviderOverride?: string\n posthogCostOverride?: CostOverride\n posthogCaptureImmediate?: boolean\n}\n\ninterface CreateInstrumentationMiddlewareOptions {\n posthogDistinctId?: string\n posthogTraceId?: string\n posthogProperties?: Record<string, any>\n posthogPrivacyMode?: boolean\n posthogGroups?: Record<string, any>\n posthogModelOverride?: string\n posthogProviderOverride?: string\n posthogCostOverride?: CostOverride\n posthogCaptureImmediate?: boolean\n}\n\ninterface PostHogInput {\n role: string\n type?: string\n content?:\n | string\n | {\n [key: string]: any\n }\n}\n\nconst mapVercelParams = (params: any): Record<string, any> => {\n return {\n temperature: params.temperature,\n max_output_tokens: params.maxOutputTokens,\n top_p: params.topP,\n frequency_penalty: params.frequencyPenalty,\n presence_penalty: params.presencePenalty,\n stop: params.stopSequences,\n stream: params.stream,\n }\n}\n\nconst mapVercelPrompt = (messages: LanguageModelV2Prompt): PostHogInput[] => {\n // Map and truncate individual content\n const inputs: PostHogInput[] = messages.map((message) => {\n let content: any\n\n // Handle system role which has string content\n if (message.role === 'system') {\n content = [\n {\n type: 'text',\n text: truncate(String(message.content)),\n },\n ]\n } else {\n // Handle other roles which have array content\n if (Array.isArray(message.content)) {\n content = message.content.map((c: any) => {\n if (c.type === 'text') {\n return {\n type: 'text',\n text: truncate(c.text),\n }\n } else if (c.type === 'file') {\n // For file type, check if it's a data URL and redact if needed\n let fileData: string\n\n const contentData: unknown = c.data\n\n if (contentData instanceof URL) {\n fileData = contentData.toString()\n } else if (isString(contentData)) {\n // Redact base64 data URLs and raw base64 to prevent oversized events\n fileData = redactBase64DataUrl(contentData)\n } else {\n fileData = 'raw files not supported'\n }\n\n return {\n type: 'file',\n file: fileData,\n mediaType: c.mediaType,\n }\n } else if (c.type === 'reasoning') {\n return {\n type: 'reasoning',\n text: truncate(c.reasoning),\n }\n } else if (c.type === 'tool-call') {\n return {\n type: 'tool-call',\n toolCallId: c.toolCallId,\n toolName: c.toolName,\n input: c.input,\n }\n } else if (c.type === 'tool-result') {\n return {\n type: 'tool-result',\n toolCallId: c.toolCallId,\n toolName: c.toolName,\n output: c.output,\n isError: c.isError,\n }\n }\n return {\n type: 'text',\n text: '',\n }\n })\n } else {\n // Fallback for non-array content\n content = [\n {\n type: 'text',\n text: truncate(String(message.content)),\n },\n ]\n }\n }\n\n return {\n role: message.role,\n content,\n }\n })\n\n try {\n // Trim the inputs array until its JSON size fits within MAX_OUTPUT_SIZE\n let serialized = JSON.stringify(inputs)\n let removedCount = 0\n // We need to keep track of the initial size of the inputs array because we're going to be mutating it\n const initialSize = inputs.length\n for (let i = 0; i < initialSize && Buffer.byteLength(serialized, 'utf8') > MAX_OUTPUT_SIZE; i++) {\n inputs.shift()\n removedCount++\n serialized = JSON.stringify(inputs)\n }\n if (removedCount > 0) {\n // Add one placeholder to indicate how many were removed\n inputs.unshift({\n role: 'posthog',\n content: `[${removedCount} message${removedCount === 1 ? '' : 's'} removed due to size limit]`,\n })\n }\n } catch (error) {\n console.error('Error stringifying inputs', error)\n return [{ role: 'posthog', content: 'An error occurred while processing your request. Please try again.' }]\n }\n return inputs\n}\n\nconst mapVercelOutput = (result: LanguageModelV2Content[]): PostHogInput[] => {\n const content: any[] = result.map((item) => {\n if (item.type === 'text') {\n return { type: 'text', text: truncate(item.text) }\n }\n if (item.type === 'tool-call') {\n return {\n type: 'tool-call',\n id: item.toolCallId,\n function: {\n name: item.toolName,\n arguments: (item as any).args || JSON.stringify((item as any).arguments || {}),\n },\n }\n }\n if (item.type === 'reasoning') {\n return { type: 'reasoning', text: truncate(item.text) }\n }\n if (item.type === 'file') {\n // Handle files similar to input mapping - avoid large base64 data\n let fileData: string\n if (item.data instanceof URL) {\n fileData = item.data.toString()\n } else if (typeof item.data === 'string') {\n fileData = redactBase64DataUrl(item.data)\n\n // If not redacted and still large, replace with size indicator\n if (fileData === item.data && item.data.length > 1000) {\n fileData = `[${item.mediaType} file - ${item.data.length} bytes]`\n }\n } else {\n fileData = `[binary ${item.mediaType} file]`\n }\n\n return {\n type: 'file',\n name: 'generated_file',\n mediaType: item.mediaType,\n data: fileData,\n }\n }\n if (item.type === 'source') {\n return {\n type: 'source',\n sourceType: item.sourceType,\n id: item.id,\n url: (item as any).url || '',\n title: item.title || '',\n }\n }\n // Fallback for unknown types - try to extract text if possible\n return { type: 'text', text: truncate(JSON.stringify(item)) }\n })\n\n if (content.length > 0) {\n return [\n {\n role: 'assistant',\n content: content.length === 1 && content[0].type === 'text' ? content[0].text : content,\n },\n ]\n }\n // otherwise stringify and truncate\n try {\n const jsonOutput = JSON.stringify(result)\n return [{ content: truncate(jsonOutput), role: 'assistant' }]\n } catch (error) {\n console.error('Error stringifying output')\n return []\n }\n}\n\nconst extractProvider = (model: LanguageModelV2): string => {\n const provider = model.provider.toLowerCase()\n const providerName = provider.split('.')[0]\n return providerName\n}\n\nexport const createInstrumentationMiddleware = (\n phClient: PostHog,\n model: LanguageModelV2,\n options: CreateInstrumentationMiddlewareOptions\n): LanguageModelV2Middleware => {\n const middleware: LanguageModelV2Middleware = {\n wrapGenerate: async ({ doGenerate, params }) => {\n const startTime = Date.now()\n const mergedParams = {\n ...options,\n ...mapVercelParams(params),\n }\n const availableTools = extractAvailableToolCalls('vercel', params)\n\n try {\n const result = await doGenerate()\n const modelId =\n options.posthogModelOverride ?? (result.response?.modelId ? result.response.modelId : model.modelId)\n const provider = options.posthogProviderOverride ?? extractProvider(model)\n const baseURL = '' // cannot currently get baseURL from vercel\n const content = mapVercelOutput(result.content)\n const latency = (Date.now() - startTime) / 1000\n const providerMetadata = result.providerMetadata\n const additionalTokenValues = {\n ...(providerMetadata?.anthropic\n ? {\n cacheCreationInputTokens: providerMetadata.anthropic.cacheCreationInputTokens,\n }\n : {}),\n }\n const usage = {\n inputTokens: result.usage.inputTokens,\n outputTokens: result.usage.outputTokens,\n reasoningTokens: result.usage.reasoningTokens,\n cacheReadInputTokens: result.usage.cachedInputTokens,\n ...additionalTokenValues,\n }\n await sendEventToPosthog({\n client: phClient,\n distinctId: options.posthogDistinctId,\n traceId: options.posthogTraceId ?? uuidv4(),\n model: modelId,\n provider: provider,\n input: options.posthogPrivacyMode ? '' : mapVercelPrompt(params.prompt),\n output: content,\n latency,\n baseURL,\n params: mergedParams as any,\n httpStatus: 200,\n usage,\n tools: availableTools,\n captureImmediate: options.posthogCaptureImmediate,\n })\n\n return result\n } catch (error: any) {\n const modelId = model.modelId\n await sendEventToPosthog({\n client: phClient,\n distinctId: options.posthogDistinctId,\n traceId: options.posthogTraceId ?? uuidv4(),\n model: modelId,\n provider: model.provider,\n input: options.posthogPrivacyMode ? '' : mapVercelPrompt(params.prompt),\n output: [],\n latency: 0,\n baseURL: '',\n params: mergedParams as any,\n httpStatus: error?.status ? error.status : 500,\n usage: {\n inputTokens: 0,\n outputTokens: 0,\n },\n isError: true,\n error: truncate(JSON.stringify(error)),\n tools: availableTools,\n captureImmediate: options.posthogCaptureImmediate,\n })\n throw error\n }\n },\n\n wrapStream: async ({ doStream, params }) => {\n const startTime = Date.now()\n let generatedText = ''\n let reasoningText = ''\n let usage: {\n inputTokens?: number\n outputTokens?: number\n reasoningTokens?: any\n cacheReadInputTokens?: any\n cacheCreationInputTokens?: any\n } = {}\n const mergedParams = {\n ...options,\n ...mapVercelParams(params),\n }\n\n const modelId = options.posthogModelOverride ?? model.modelId\n const provider = options.posthogProviderOverride ?? extractProvider(model)\n const availableTools = extractAvailableToolCalls('vercel', params)\n const baseURL = '' // cannot currently get baseURL from vercel\n\n try {\n const { stream, ...rest } = await doStream()\n const transformStream = new TransformStream<LanguageModelV2StreamPart, LanguageModelV2StreamPart>({\n transform(chunk, controller) {\n // Handle new v5 streaming patterns\n if (chunk.type === 'text-delta') {\n generatedText += chunk.delta\n }\n if (chunk.type === 'reasoning-delta') {\n reasoningText += chunk.delta // New in v5\n }\n if (chunk.type === 'finish') {\n const providerMetadata = chunk.providerMetadata\n const additionalTokenValues = {\n ...(providerMetadata?.anthropic\n ? {\n cacheCreationInputTokens: providerMetadata.anthropic.cacheCreationInputTokens,\n }\n : {}),\n }\n usage = {\n inputTokens: chunk.usage?.inputTokens,\n outputTokens: chunk.usage?.outputTokens,\n reasoningTokens: chunk.usage?.reasoningTokens,\n cacheReadInputTokens: chunk.usage?.cachedInputTokens,\n ...additionalTokenValues,\n }\n }\n controller.enqueue(chunk)\n },\n\n flush: async () => {\n const latency = (Date.now() - startTime) / 1000\n // Build content array similar to mapVercelOutput structure\n const content = []\n if (reasoningText) {\n content.push({ type: 'reasoning', text: truncate(reasoningText) })\n }\n if (generatedText) {\n content.push({ type: 'text', text: truncate(generatedText) })\n }\n\n // Structure output like mapVercelOutput does\n const output =\n content.length > 0\n ? [\n {\n role: 'assistant',\n content: content.length === 1 && content[0].type === 'text' ? content[0].text : content,\n },\n ]\n : []\n\n await sendEventToPosthog({\n client: phClient,\n distinctId: options.posthogDistinctId,\n traceId: options.posthogTraceId ?? uuidv4(),\n model: modelId,\n provider: provider,\n input: options.posthogPrivacyMode ? '' : mapVercelPrompt(params.prompt),\n output: output,\n latency,\n baseURL,\n params: mergedParams as any,\n httpStatus: 200,\n usage,\n tools: availableTools,\n captureImmediate: options.posthogCaptureImmediate,\n })\n },\n })\n\n return {\n stream: stream.pipeThrough(transformStream),\n ...rest,\n }\n } catch (error: any) {\n await sendEventToPosthog({\n client: phClient,\n distinctId: options.posthogDistinctId,\n traceId: options.posthogTraceId ?? uuidv4(),\n model: modelId,\n provider: provider,\n input: options.posthogPrivacyMode ? '' : mapVercelPrompt(params.prompt),\n output: [],\n latency: 0,\n baseURL: '',\n params: mergedParams as any,\n httpStatus: error?.status ? error.status : 500,\n usage: {\n inputTokens: 0,\n outputTokens: 0,\n },\n isError: true,\n error: truncate(JSON.stringify(error)),\n tools: availableTools,\n captureImmediate: options.posthogCaptureImmediate,\n })\n throw error\n }\n },\n }\n\n return middleware\n}\n\nexport const wrapVercelLanguageModel = (\n model: LanguageModelV2,\n phClient: PostHog,\n options: ClientOptions\n): LanguageModelV2 => {\n const traceId = options.posthogTraceId ?? uuidv4()\n const middleware = createInstrumentationMiddleware(phClient, model, {\n ...options,\n posthogTraceId: traceId,\n posthogDistinctId: options.posthogDistinctId,\n })\n\n const wrappedModel = wrapLanguageModel({\n model,\n middleware,\n })\n\n return wrappedModel\n}\n"],"names":["MAX_OUTPUT_SIZE","STRING_FORMAT","getModelParams","params","modelParams","paramKeys","key","undefined","withPrivacyMode","client","privacyMode","input","privacy_mode","truncate","str","buffer","Buffer","from","length","truncatedBuffer","slice","toString","error","console","extractAvailableToolCalls","provider","mode","type","tools","sanitizeValues","obj","jsonSafe","JSON","parse","stringify","Array","isArray","map","Object","fromEntries","entries","k","v","sendEventToPosthog","distinctId","traceId","model","output","latency","baseURL","httpStatus","usage","isError","captureImmediate","capture","Promise","resolve","safeInput","safeOutput","safeError","errorData","$ai_is_error","$ai_error","costOverrideData","posthogCostOverride","inputCostUSD","inputCost","inputTokens","outputCostUSD","outputCost","outputTokens","$ai_input_cost_usd","$ai_output_cost_usd","$ai_total_cost_usd","additionalTokenValues","reasoningTokens","$ai_reasoning_tokens","cacheReadInputTokens","$ai_cache_read_input_tokens","cacheCreationInputTokens","$ai_cache_creation_input_tokens","properties","$ai_provider","posthogProviderOverride","$ai_model","posthogModelOverride","$ai_model_parameters","$ai_input","posthogPrivacyMode","$ai_output_choices","$ai_http_status","$ai_input_tokens","$ai_output_tokens","$ai_latency","$ai_trace_id","$ai_base_url","posthogProperties","$process_person_profile","$ai_tools","event","groups","posthogGroups","isString","value","REDACTED_IMAGE_PLACEHOLDER","isBase64DataUrl","test","isValidUrl","URL","startsWith","isRawBase64","redactBase64DataUrl","mapVercelParams","temperature","max_output_tokens","maxOutputTokens","top_p","topP","frequency_penalty","frequencyPenalty","presence_penalty","presencePenalty","stop","stopSequences","stream","mapVercelPrompt","messages","inputs","message","content","role","text","String","c","fileData","contentData","data","file","mediaType","reasoning","toolCallId","toolName","serialized","removedCount","initialSize","i","byteLength","shift","unshift","mapVercelOutput","result","item","id","function","name","arguments","args","sourceType","url","title","jsonOutput","extractProvider","toLowerCase","providerName","split","createInstrumentationMiddleware","phClient","options","middleware","wrapGenerate","doGenerate","startTime","Date","now","mergedParams","availableTools","modelId","response","providerMetadata","anthropic","cachedInputTokens","posthogDistinctId","posthogTraceId","uuidv4","prompt","posthogCaptureImmediate","status","wrapStream","doStream","generatedText","reasoningText","rest","transformStream","TransformStream","transform","chunk","controller","delta","enqueue","flush","push","pipeThrough","wrapVercelLanguageModel","wrappedModel","wrapLanguageModel"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs","sources":["../../src/utils.ts","../../src/typeGuards.ts","../../src/sanitization.ts","../../src/vercel/middleware.ts"],"sourcesContent":["import { PostHog } from 'posthog-node'\nimport { Buffer } from 'buffer'\nimport OpenAIOrignal from 'openai'\nimport AnthropicOriginal from '@anthropic-ai/sdk'\nimport type { ChatCompletionTool } from 'openai/resources/chat/completions'\nimport type { Tool as GeminiTool } from '@google/genai'\nimport type { FormattedMessage, FormattedContent, TokenUsage } from './types'\nimport { version } from '../package.json'\n\ntype ChatCompletionCreateParamsBase = OpenAIOrignal.Chat.Completions.ChatCompletionCreateParams\ntype MessageCreateParams = AnthropicOriginal.Messages.MessageCreateParams\ntype ResponseCreateParams = OpenAIOrignal.Responses.ResponseCreateParams\ntype AnthropicTool = AnthropicOriginal.Tool\n\n// limit large outputs by truncating to 200kb (approx 200k bytes)\nexport const MAX_OUTPUT_SIZE = 200000\nconst STRING_FORMAT = 'utf8'\n\nexport interface MonitoringParams {\n posthogDistinctId?: string\n posthogTraceId?: string\n posthogProperties?: Record<string, any>\n posthogPrivacyMode?: boolean\n posthogGroups?: Record<string, any>\n posthogModelOverride?: string\n posthogProviderOverride?: string\n posthogCostOverride?: CostOverride\n posthogCaptureImmediate?: boolean\n}\n\nexport interface CostOverride {\n inputCost: number\n outputCost: number\n}\n\nexport const getModelParams = (\n params: ((ChatCompletionCreateParamsBase | MessageCreateParams | ResponseCreateParams) & MonitoringParams) | null\n): Record<string, any> => {\n if (!params) {\n return {}\n }\n const modelParams: Record<string, any> = {}\n const paramKeys = [\n 'temperature',\n 'max_tokens',\n 'max_completion_tokens',\n 'top_p',\n 'frequency_penalty',\n 'presence_penalty',\n 'n',\n 'stop',\n 'stream',\n 'streaming',\n ] as const\n\n for (const key of paramKeys) {\n if (key in params && (params as any)[key] !== undefined) {\n modelParams[key] = (params as any)[key]\n }\n }\n return modelParams\n}\n\n/**\n * Helper to format responses (non-streaming) for consumption, mirroring Python's openai vs. anthropic approach.\n */\nexport const formatResponse = (response: any, provider: string): FormattedMessage[] => {\n if (!response) {\n return []\n }\n if (provider === 'anthropic') {\n return formatResponseAnthropic(response)\n } else if (provider === 'openai') {\n return formatResponseOpenAI(response)\n } else if (provider === 'gemini') {\n return formatResponseGemini(response)\n }\n return []\n}\n\nexport const formatResponseAnthropic = (response: any): FormattedMessage[] => {\n const output: FormattedMessage[] = []\n const content: FormattedContent = []\n\n for (const choice of response.content ?? []) {\n if (choice?.type === 'text' && choice?.text) {\n content.push({ type: 'text', text: choice.text })\n } else if (choice?.type === 'tool_use' && choice?.name && choice?.id) {\n content.push({\n type: 'function',\n id: choice.id,\n function: {\n name: choice.name,\n arguments: choice.input || {},\n },\n })\n }\n }\n\n if (content.length > 0) {\n output.push({\n role: 'assistant',\n content,\n })\n }\n\n return output\n}\n\nexport const formatResponseOpenAI = (response: any): FormattedMessage[] => {\n const output: FormattedMessage[] = []\n\n if (response.choices) {\n for (const choice of response.choices) {\n const content: FormattedContent = []\n let role = 'assistant'\n\n if (choice.message) {\n if (choice.message.role) {\n role = choice.message.role\n }\n\n if (choice.message.content) {\n content.push({ type: 'text', text: choice.message.content })\n }\n\n if (choice.message.tool_calls) {\n for (const toolCall of choice.message.tool_calls) {\n content.push({\n type: 'function',\n id: toolCall.id,\n function: {\n name: toolCall.function.name,\n arguments: toolCall.function.arguments,\n },\n })\n }\n }\n }\n\n if (content.length > 0) {\n output.push({\n role,\n content,\n })\n }\n }\n }\n\n // Handle Responses API format\n if (response.output) {\n const content: FormattedContent = []\n let role = 'assistant'\n\n for (const item of response.output) {\n if (item.type === 'message') {\n role = item.role\n\n if (item.content && Array.isArray(item.content)) {\n for (const contentItem of item.content) {\n if (contentItem.type === 'output_text' && contentItem.text) {\n content.push({ type: 'text', text: contentItem.text })\n } else if (contentItem.text) {\n content.push({ type: 'text', text: contentItem.text })\n } else if (contentItem.type === 'input_image' && contentItem.image_url) {\n content.push({\n type: 'image',\n image: contentItem.image_url,\n })\n }\n }\n } else if (item.content) {\n content.push({ type: 'text', text: String(item.content) })\n }\n } else if (item.type === 'function_call') {\n content.push({\n type: 'function',\n id: item.call_id || item.id || '',\n function: {\n name: item.name,\n arguments: item.arguments || {},\n },\n })\n }\n }\n\n if (content.length > 0) {\n output.push({\n role,\n content,\n })\n }\n }\n\n return output\n}\n\nexport const formatResponseGemini = (response: any): FormattedMessage[] => {\n const output: FormattedMessage[] = []\n\n if (response.candidates && Array.isArray(response.candidates)) {\n for (const candidate of response.candidates) {\n if (candidate.content && candidate.content.parts) {\n const content: FormattedContent = []\n\n for (const part of candidate.content.parts) {\n if (part.text) {\n content.push({ type: 'text', text: part.text })\n } else if (part.functionCall) {\n content.push({\n type: 'function',\n function: {\n name: part.functionCall.name,\n arguments: part.functionCall.args,\n },\n })\n }\n }\n\n if (content.length > 0) {\n output.push({\n role: 'assistant',\n content,\n })\n }\n } else if (candidate.text) {\n output.push({\n role: 'assistant',\n content: [{ type: 'text', text: candidate.text }],\n })\n }\n }\n } else if (response.text) {\n output.push({\n role: 'assistant',\n content: [{ type: 'text', text: response.text }],\n })\n }\n\n return output\n}\n\nexport const mergeSystemPrompt = (params: MessageCreateParams & MonitoringParams, provider: string): any => {\n if (provider == 'anthropic') {\n const messages = params.messages || []\n if (!(params as any).system) {\n return messages\n }\n const systemMessage = (params as any).system\n return [{ role: 'system', content: systemMessage }, ...messages]\n }\n return params.messages\n}\n\nexport const withPrivacyMode = (client: PostHog, privacyMode: boolean, input: any): any => {\n return (client as any).privacy_mode || privacyMode ? null : input\n}\n\nexport const truncate = (str: string): string => {\n try {\n const buffer = Buffer.from(str, STRING_FORMAT)\n if (buffer.length <= MAX_OUTPUT_SIZE) {\n return str\n }\n const truncatedBuffer = buffer.slice(0, MAX_OUTPUT_SIZE)\n return `${truncatedBuffer.toString(STRING_FORMAT)}... [truncated]`\n } catch (error) {\n console.error('Error truncating, likely not a string')\n return str\n }\n}\n\n/**\n * Extract available tool calls from the request parameters.\n * These are the tools provided to the LLM, not the tool calls in the response.\n */\nexport const extractAvailableToolCalls = (\n provider: string,\n params: any\n): ChatCompletionTool[] | AnthropicTool[] | GeminiTool[] | null => {\n if (provider === 'anthropic') {\n if (params.tools) {\n return params.tools\n }\n\n return null\n } else if (provider === 'gemini') {\n if (params.config && params.config.tools) {\n return params.config.tools\n }\n\n return null\n } else if (provider === 'openai') {\n if (params.tools) {\n return params.tools\n }\n\n return null\n } else if (provider === 'vercel') {\n if (params.tools) {\n return params.tools\n }\n\n return null\n }\n\n return null\n}\n\nexport type SendEventToPosthogParams = {\n client: PostHog\n distinctId?: string\n traceId: string\n model: string\n provider: string\n input: any\n output: any\n latency: number\n baseURL: string\n httpStatus: number\n usage?: TokenUsage\n params: (ChatCompletionCreateParamsBase | MessageCreateParams | ResponseCreateParams) & MonitoringParams\n isError?: boolean\n error?: string\n tools?: ChatCompletionTool[] | AnthropicTool[] | GeminiTool[] | null\n captureImmediate?: boolean\n}\n\nfunction sanitizeValues(obj: any): any {\n if (obj === undefined || obj === null) {\n return obj\n }\n const jsonSafe = JSON.parse(JSON.stringify(obj))\n if (typeof jsonSafe === 'string') {\n return Buffer.from(jsonSafe, STRING_FORMAT).toString(STRING_FORMAT)\n } else if (Array.isArray(jsonSafe)) {\n return jsonSafe.map(sanitizeValues)\n } else if (jsonSafe && typeof jsonSafe === 'object') {\n return Object.fromEntries(Object.entries(jsonSafe).map(([k, v]) => [k, sanitizeValues(v)]))\n }\n return jsonSafe\n}\n\nexport const sendEventToPosthog = async ({\n client,\n distinctId,\n traceId,\n model,\n provider,\n input,\n output,\n latency,\n baseURL,\n params,\n httpStatus = 200,\n usage = {},\n isError = false,\n error,\n tools,\n captureImmediate = false,\n}: SendEventToPosthogParams): Promise<void> => {\n if (!client.capture) {\n return Promise.resolve()\n }\n // sanitize input and output for UTF-8 validity\n const safeInput = sanitizeValues(input)\n const safeOutput = sanitizeValues(output)\n const safeError = sanitizeValues(error)\n\n let errorData = {}\n if (isError) {\n errorData = {\n $ai_is_error: true,\n $ai_error: safeError,\n }\n }\n let costOverrideData = {}\n if (params.posthogCostOverride) {\n const inputCostUSD = (params.posthogCostOverride.inputCost ?? 0) * (usage.inputTokens ?? 0)\n const outputCostUSD = (params.posthogCostOverride.outputCost ?? 0) * (usage.outputTokens ?? 0)\n costOverrideData = {\n $ai_input_cost_usd: inputCostUSD,\n $ai_output_cost_usd: outputCostUSD,\n $ai_total_cost_usd: inputCostUSD + outputCostUSD,\n }\n }\n\n const additionalTokenValues = {\n ...(usage.reasoningTokens ? { $ai_reasoning_tokens: usage.reasoningTokens } : {}),\n ...(usage.cacheReadInputTokens ? { $ai_cache_read_input_tokens: usage.cacheReadInputTokens } : {}),\n ...(usage.cacheCreationInputTokens ? { $ai_cache_creation_input_tokens: usage.cacheCreationInputTokens } : {}),\n }\n\n const properties = {\n $ai_lib: 'posthog-ai',\n $ai_lib_version: version,\n $ai_provider: params.posthogProviderOverride ?? provider,\n $ai_model: params.posthogModelOverride ?? model,\n $ai_model_parameters: getModelParams(params),\n $ai_input: withPrivacyMode(client, params.posthogPrivacyMode ?? false, safeInput),\n $ai_output_choices: withPrivacyMode(client, params.posthogPrivacyMode ?? false, safeOutput),\n $ai_http_status: httpStatus,\n $ai_input_tokens: usage.inputTokens ?? 0,\n $ai_output_tokens: usage.outputTokens ?? 0,\n ...additionalTokenValues,\n $ai_latency: latency,\n $ai_trace_id: traceId,\n $ai_base_url: baseURL,\n ...params.posthogProperties,\n ...(distinctId ? {} : { $process_person_profile: false }),\n ...(tools ? { $ai_tools: tools } : {}),\n ...errorData,\n ...costOverrideData,\n }\n\n const event = {\n distinctId: distinctId ?? traceId,\n event: '$ai_generation',\n properties,\n groups: params.posthogGroups,\n }\n\n if (captureImmediate) {\n // await capture promise to send single event in serverless environments\n await client.captureImmediate(event)\n } else {\n client.capture(event)\n }\n}\n","// Type guards for safer type checking\n\nexport const isString = (value: unknown): value is string => {\n return typeof value === 'string'\n}\n\nexport const isObject = (value: unknown): value is Record<string, unknown> => {\n return value !== null && typeof value === 'object' && !Array.isArray(value)\n}\n","import { isString, isObject } from './typeGuards'\n\nconst REDACTED_IMAGE_PLACEHOLDER = '[base64 image redacted]'\n\n// ============================================\n// Base64 Detection Helpers\n// ============================================\n\nconst isBase64DataUrl = (str: string): boolean => {\n return /^data:([^;]+);base64,/.test(str)\n}\n\nconst isValidUrl = (str: string): boolean => {\n try {\n new URL(str)\n return true\n } catch {\n // Not an absolute URL, check if it's a relative URL or path\n return str.startsWith('/') || str.startsWith('./') || str.startsWith('../')\n }\n}\n\nconst isRawBase64 = (str: string): boolean => {\n // Skip if it's a valid URL or path\n if (isValidUrl(str)) {\n return false\n }\n\n // Check if it's a valid base64 string\n // Base64 images are typically at least a few hundred chars, but we'll be conservative\n return str.length > 20 && /^[A-Za-z0-9+/]+=*$/.test(str)\n}\n\nexport function redactBase64DataUrl(str: string): string\nexport function redactBase64DataUrl(str: unknown): unknown\nexport function redactBase64DataUrl(str: unknown): unknown {\n if (!isString(str)) return str\n\n // Check for data URL format\n if (isBase64DataUrl(str)) {\n return REDACTED_IMAGE_PLACEHOLDER\n }\n\n // Check for raw base64 (Vercel sends raw base64 for inline images)\n if (isRawBase64(str)) {\n return REDACTED_IMAGE_PLACEHOLDER\n }\n\n return str\n}\n\n// ============================================\n// Common Message Processing\n// ============================================\n\ntype ContentTransformer = (item: unknown) => unknown\n\nconst processMessages = (messages: unknown, transformContent: ContentTransformer): unknown => {\n if (!messages) return messages\n\n const processContent = (content: unknown): unknown => {\n if (typeof content === 'string') return content\n\n if (!content) return content\n\n if (Array.isArray(content)) {\n return content.map(transformContent)\n }\n\n // Handle single object content\n return transformContent(content)\n }\n\n const processMessage = (msg: unknown): unknown => {\n if (!isObject(msg) || !('content' in msg)) return msg\n return { ...msg, content: processContent(msg.content) }\n }\n\n // Handle both arrays and single messages\n if (Array.isArray(messages)) {\n return messages.map(processMessage)\n }\n\n return processMessage(messages)\n}\n\n// ============================================\n// Provider-Specific Image Sanitizers\n// ============================================\n\nconst sanitizeOpenAIImage = (item: unknown): unknown => {\n if (!isObject(item)) return item\n\n // Handle image_url format\n if (item.type === 'image_url' && 'image_url' in item && isObject(item.image_url) && 'url' in item.image_url) {\n return {\n ...item,\n image_url: {\n ...item.image_url,\n url: redactBase64DataUrl(item.image_url.url),\n },\n }\n }\n\n return item\n}\n\nconst sanitizeOpenAIResponseImage = (item: unknown): unknown => {\n if (!isObject(item)) return item\n\n // Handle input_image format\n if (item.type === 'input_image' && 'image_url' in item) {\n return {\n ...item,\n image_url: redactBase64DataUrl(item.image_url),\n }\n }\n\n return item\n}\n\nconst sanitizeAnthropicImage = (item: unknown): unknown => {\n if (!isObject(item)) return item\n\n // Handle Anthropic's image format\n if (\n item.type === 'image' &&\n 'source' in item &&\n isObject(item.source) &&\n item.source.type === 'base64' &&\n 'data' in item.source\n ) {\n return {\n ...item,\n source: {\n ...item.source,\n data: REDACTED_IMAGE_PLACEHOLDER,\n },\n }\n }\n\n return item\n}\n\nconst sanitizeGeminiPart = (part: unknown): unknown => {\n if (!isObject(part)) return part\n\n // Handle Gemini's inline data format\n if ('inlineData' in part && isObject(part.inlineData) && 'data' in part.inlineData) {\n return {\n ...part,\n inlineData: {\n ...part.inlineData,\n data: REDACTED_IMAGE_PLACEHOLDER,\n },\n }\n }\n\n return part\n}\n\nconst processGeminiItem = (item: unknown): unknown => {\n if (!isObject(item)) return item\n\n // If it has parts, process them\n if ('parts' in item && item.parts) {\n const parts = Array.isArray(item.parts) ? item.parts.map(sanitizeGeminiPart) : sanitizeGeminiPart(item.parts)\n\n return { ...item, parts }\n }\n\n return item\n}\n\nconst sanitizeLangChainImage = (item: unknown): unknown => {\n if (!isObject(item)) return item\n\n // OpenAI style\n if (item.type === 'image_url' && 'image_url' in item && isObject(item.image_url) && 'url' in item.image_url) {\n return {\n ...item,\n image_url: {\n ...item.image_url,\n url: redactBase64DataUrl(item.image_url.url),\n },\n }\n }\n\n // Direct image with data field\n if (item.type === 'image' && 'data' in item) {\n return { ...item, data: redactBase64DataUrl(item.data) }\n }\n\n // Anthropic style\n if (item.type === 'image' && 'source' in item && isObject(item.source) && 'data' in item.source) {\n return {\n ...item,\n source: {\n ...item.source,\n data: redactBase64DataUrl(item.source.data),\n },\n }\n }\n\n // Google style\n if (item.type === 'media' && 'data' in item) {\n return { ...item, data: redactBase64DataUrl(item.data) }\n }\n\n return item\n}\n\n// Export individual sanitizers for tree-shaking\nexport const sanitizeOpenAI = (data: unknown): unknown => {\n return processMessages(data, sanitizeOpenAIImage)\n}\n\nexport const sanitizeOpenAIResponse = (data: unknown): unknown => {\n return processMessages(data, sanitizeOpenAIResponseImage)\n}\n\nexport const sanitizeAnthropic = (data: unknown): unknown => {\n return processMessages(data, sanitizeAnthropicImage)\n}\n\nexport const sanitizeGemini = (data: unknown): unknown => {\n // Gemini has a different structure with 'parts' directly on items instead of 'content'\n // So we need custom processing instead of using processMessages\n if (!data) return data\n\n if (Array.isArray(data)) {\n return data.map(processGeminiItem)\n }\n\n return processGeminiItem(data)\n}\n\nexport const sanitizeLangChain = (data: unknown): unknown => {\n return processMessages(data, sanitizeLangChainImage)\n}\n","import { wrapLanguageModel } from 'ai'\nimport type {\n LanguageModelV2,\n LanguageModelV2Content,\n LanguageModelV2Middleware,\n LanguageModelV2Prompt,\n LanguageModelV2StreamPart,\n} from '@ai-sdk/provider'\nimport { v4 as uuidv4 } from 'uuid'\nimport { PostHog } from 'posthog-node'\nimport { CostOverride, sendEventToPosthog, truncate, MAX_OUTPUT_SIZE, extractAvailableToolCalls } from '../utils'\nimport { Buffer } from 'buffer'\nimport { redactBase64DataUrl } from '../sanitization'\nimport { isString } from '../typeGuards'\n\ninterface ClientOptions {\n posthogDistinctId?: string\n posthogTraceId?: string\n posthogProperties?: Record<string, any>\n posthogPrivacyMode?: boolean\n posthogGroups?: Record<string, any>\n posthogModelOverride?: string\n posthogProviderOverride?: string\n posthogCostOverride?: CostOverride\n posthogCaptureImmediate?: boolean\n}\n\ninterface CreateInstrumentationMiddlewareOptions {\n posthogDistinctId?: string\n posthogTraceId?: string\n posthogProperties?: Record<string, any>\n posthogPrivacyMode?: boolean\n posthogGroups?: Record<string, any>\n posthogModelOverride?: string\n posthogProviderOverride?: string\n posthogCostOverride?: CostOverride\n posthogCaptureImmediate?: boolean\n}\n\ninterface PostHogInput {\n role: string\n type?: string\n content?:\n | string\n | {\n [key: string]: any\n }\n}\n\n// Content types for the output array\ntype OutputContentItem =\n | { type: 'text'; text: string }\n | { type: 'reasoning'; text: string }\n | { type: 'tool-call'; id: string; function: { name: string; arguments: string } }\n | { type: 'file'; name: string; mediaType: string; data: string }\n | { type: 'source'; sourceType: string; id: string; url: string; title: string }\n\nconst mapVercelParams = (params: any): Record<string, any> => {\n return {\n temperature: params.temperature,\n max_output_tokens: params.maxOutputTokens,\n top_p: params.topP,\n frequency_penalty: params.frequencyPenalty,\n presence_penalty: params.presencePenalty,\n stop: params.stopSequences,\n stream: params.stream,\n }\n}\n\nconst mapVercelPrompt = (messages: LanguageModelV2Prompt): PostHogInput[] => {\n // Map and truncate individual content\n const inputs: PostHogInput[] = messages.map((message) => {\n let content: any\n\n // Handle system role which has string content\n if (message.role === 'system') {\n content = [\n {\n type: 'text',\n text: truncate(String(message.content)),\n },\n ]\n } else {\n // Handle other roles which have array content\n if (Array.isArray(message.content)) {\n content = message.content.map((c: any) => {\n if (c.type === 'text') {\n return {\n type: 'text',\n text: truncate(c.text),\n }\n } else if (c.type === 'file') {\n // For file type, check if it's a data URL and redact if needed\n let fileData: string\n\n const contentData: unknown = c.data\n\n if (contentData instanceof URL) {\n fileData = contentData.toString()\n } else if (isString(contentData)) {\n // Redact base64 data URLs and raw base64 to prevent oversized events\n fileData = redactBase64DataUrl(contentData)\n } else {\n fileData = 'raw files not supported'\n }\n\n return {\n type: 'file',\n file: fileData,\n mediaType: c.mediaType,\n }\n } else if (c.type === 'reasoning') {\n return {\n type: 'reasoning',\n text: truncate(c.reasoning),\n }\n } else if (c.type === 'tool-call') {\n return {\n type: 'tool-call',\n toolCallId: c.toolCallId,\n toolName: c.toolName,\n input: c.input,\n }\n } else if (c.type === 'tool-result') {\n return {\n type: 'tool-result',\n toolCallId: c.toolCallId,\n toolName: c.toolName,\n output: c.output,\n isError: c.isError,\n }\n }\n return {\n type: 'text',\n text: '',\n }\n })\n } else {\n // Fallback for non-array content\n content = [\n {\n type: 'text',\n text: truncate(String(message.content)),\n },\n ]\n }\n }\n\n return {\n role: message.role,\n content,\n }\n })\n\n try {\n // Trim the inputs array until its JSON size fits within MAX_OUTPUT_SIZE\n let serialized = JSON.stringify(inputs)\n let removedCount = 0\n // We need to keep track of the initial size of the inputs array because we're going to be mutating it\n const initialSize = inputs.length\n for (let i = 0; i < initialSize && Buffer.byteLength(serialized, 'utf8') > MAX_OUTPUT_SIZE; i++) {\n inputs.shift()\n removedCount++\n serialized = JSON.stringify(inputs)\n }\n if (removedCount > 0) {\n // Add one placeholder to indicate how many were removed\n inputs.unshift({\n role: 'posthog',\n content: `[${removedCount} message${removedCount === 1 ? '' : 's'} removed due to size limit]`,\n })\n }\n } catch (error) {\n console.error('Error stringifying inputs', error)\n return [{ role: 'posthog', content: 'An error occurred while processing your request. Please try again.' }]\n }\n return inputs\n}\n\nconst mapVercelOutput = (result: LanguageModelV2Content[]): PostHogInput[] => {\n const content: OutputContentItem[] = result.map((item) => {\n if (item.type === 'text') {\n return { type: 'text', text: truncate(item.text) }\n }\n if (item.type === 'tool-call') {\n return {\n type: 'tool-call',\n id: item.toolCallId,\n function: {\n name: item.toolName,\n arguments: (item as any).args || JSON.stringify((item as any).arguments || {}),\n },\n }\n }\n if (item.type === 'reasoning') {\n return { type: 'reasoning', text: truncate(item.text) }\n }\n if (item.type === 'file') {\n // Handle files similar to input mapping - avoid large base64 data\n let fileData: string\n if (item.data instanceof URL) {\n fileData = item.data.toString()\n } else if (typeof item.data === 'string') {\n fileData = redactBase64DataUrl(item.data)\n\n // If not redacted and still large, replace with size indicator\n if (fileData === item.data && item.data.length > 1000) {\n fileData = `[${item.mediaType} file - ${item.data.length} bytes]`\n }\n } else {\n fileData = `[binary ${item.mediaType} file]`\n }\n\n return {\n type: 'file',\n name: 'generated_file',\n mediaType: item.mediaType,\n data: fileData,\n }\n }\n if (item.type === 'source') {\n return {\n type: 'source',\n sourceType: item.sourceType,\n id: item.id,\n url: (item as any).url || '',\n title: item.title || '',\n }\n }\n // Fallback for unknown types - try to extract text if possible\n return { type: 'text', text: truncate(JSON.stringify(item)) }\n })\n\n if (content.length > 0) {\n return [\n {\n role: 'assistant',\n content: content.length === 1 && content[0].type === 'text' ? content[0].text : content,\n },\n ]\n }\n // otherwise stringify and truncate\n try {\n const jsonOutput = JSON.stringify(result)\n return [{ content: truncate(jsonOutput), role: 'assistant' }]\n } catch (error) {\n console.error('Error stringifying output')\n return []\n }\n}\n\nconst extractProvider = (model: LanguageModelV2): string => {\n const provider = model.provider.toLowerCase()\n const providerName = provider.split('.')[0]\n return providerName\n}\n\nexport const createInstrumentationMiddleware = (\n phClient: PostHog,\n model: LanguageModelV2,\n options: CreateInstrumentationMiddlewareOptions\n): LanguageModelV2Middleware => {\n const middleware: LanguageModelV2Middleware = {\n wrapGenerate: async ({ doGenerate, params }) => {\n const startTime = Date.now()\n const mergedParams = {\n ...options,\n ...mapVercelParams(params),\n }\n const availableTools = extractAvailableToolCalls('vercel', params)\n\n try {\n const result = await doGenerate()\n const modelId =\n options.posthogModelOverride ?? (result.response?.modelId ? result.response.modelId : model.modelId)\n const provider = options.posthogProviderOverride ?? extractProvider(model)\n const baseURL = '' // cannot currently get baseURL from vercel\n const content = mapVercelOutput(result.content)\n const latency = (Date.now() - startTime) / 1000\n const providerMetadata = result.providerMetadata\n const additionalTokenValues = {\n ...(providerMetadata?.anthropic\n ? {\n cacheCreationInputTokens: providerMetadata.anthropic.cacheCreationInputTokens,\n }\n : {}),\n }\n const usage = {\n inputTokens: result.usage.inputTokens,\n outputTokens: result.usage.outputTokens,\n reasoningTokens: result.usage.reasoningTokens,\n cacheReadInputTokens: result.usage.cachedInputTokens,\n ...additionalTokenValues,\n }\n await sendEventToPosthog({\n client: phClient,\n distinctId: options.posthogDistinctId,\n traceId: options.posthogTraceId ?? uuidv4(),\n model: modelId,\n provider: provider,\n input: options.posthogPrivacyMode ? '' : mapVercelPrompt(params.prompt),\n output: content,\n latency,\n baseURL,\n params: mergedParams as any,\n httpStatus: 200,\n usage,\n tools: availableTools,\n captureImmediate: options.posthogCaptureImmediate,\n })\n\n return result\n } catch (error: any) {\n const modelId = model.modelId\n await sendEventToPosthog({\n client: phClient,\n distinctId: options.posthogDistinctId,\n traceId: options.posthogTraceId ?? uuidv4(),\n model: modelId,\n provider: model.provider,\n input: options.posthogPrivacyMode ? '' : mapVercelPrompt(params.prompt),\n output: [],\n latency: 0,\n baseURL: '',\n params: mergedParams as any,\n httpStatus: error?.status ? error.status : 500,\n usage: {\n inputTokens: 0,\n outputTokens: 0,\n },\n isError: true,\n error: truncate(JSON.stringify(error)),\n tools: availableTools,\n captureImmediate: options.posthogCaptureImmediate,\n })\n throw error\n }\n },\n\n wrapStream: async ({ doStream, params }) => {\n const startTime = Date.now()\n let generatedText = ''\n let reasoningText = ''\n let usage: {\n inputTokens?: number\n outputTokens?: number\n reasoningTokens?: any\n cacheReadInputTokens?: any\n cacheCreationInputTokens?: any\n } = {}\n const mergedParams = {\n ...options,\n ...mapVercelParams(params),\n }\n\n const modelId = options.posthogModelOverride ?? model.modelId\n const provider = options.posthogProviderOverride ?? extractProvider(model)\n const availableTools = extractAvailableToolCalls('vercel', params)\n const baseURL = '' // cannot currently get baseURL from vercel\n\n // Map to track in-progress tool calls\n const toolCallsInProgress = new Map<\n string,\n {\n toolCallId: string\n toolName: string\n input: string\n }\n >()\n\n try {\n const { stream, ...rest } = await doStream()\n const transformStream = new TransformStream<LanguageModelV2StreamPart, LanguageModelV2StreamPart>({\n transform(chunk, controller) {\n // Handle new v5 streaming patterns\n if (chunk.type === 'text-delta') {\n generatedText += chunk.delta\n }\n if (chunk.type === 'reasoning-delta') {\n reasoningText += chunk.delta // New in v5\n }\n\n // Handle tool call chunks\n if (chunk.type === 'tool-input-start') {\n // Initialize a new tool call\n toolCallsInProgress.set(chunk.id, {\n toolCallId: chunk.id,\n toolName: chunk.toolName,\n input: '',\n })\n }\n if (chunk.type === 'tool-input-delta') {\n // Accumulate tool call arguments\n const toolCall = toolCallsInProgress.get(chunk.id)\n if (toolCall) {\n toolCall.input += chunk.delta\n }\n }\n if (chunk.type === 'tool-input-end') {\n // Tool call is complete, keep it in the map for final processing\n // Nothing specific to do here, the tool call is already complete\n }\n if (chunk.type === 'tool-call') {\n // Direct tool call chunk (complete tool call)\n toolCallsInProgress.set(chunk.toolCallId, {\n toolCallId: chunk.toolCallId,\n toolName: chunk.toolName,\n input: chunk.input,\n })\n }\n\n if (chunk.type === 'finish') {\n const providerMetadata = chunk.providerMetadata\n const additionalTokenValues = {\n ...(providerMetadata?.anthropic\n ? {\n cacheCreationInputTokens: providerMetadata.anthropic.cacheCreationInputTokens,\n }\n : {}),\n }\n usage = {\n inputTokens: chunk.usage?.inputTokens,\n outputTokens: chunk.usage?.outputTokens,\n reasoningTokens: chunk.usage?.reasoningTokens,\n cacheReadInputTokens: chunk.usage?.cachedInputTokens,\n ...additionalTokenValues,\n }\n }\n controller.enqueue(chunk)\n },\n\n flush: async () => {\n const latency = (Date.now() - startTime) / 1000\n // Build content array similar to mapVercelOutput structure\n const content: OutputContentItem[] = []\n if (reasoningText) {\n content.push({ type: 'reasoning', text: truncate(reasoningText) })\n }\n if (generatedText) {\n content.push({ type: 'text', text: truncate(generatedText) })\n }\n\n // Add completed tool calls to content\n for (const toolCall of toolCallsInProgress.values()) {\n if (toolCall.toolName) {\n content.push({\n type: 'tool-call',\n id: toolCall.toolCallId,\n function: {\n name: toolCall.toolName,\n arguments: toolCall.input,\n },\n })\n }\n }\n\n // Structure output like mapVercelOutput does\n const output =\n content.length > 0\n ? [\n {\n role: 'assistant',\n content: content.length === 1 && content[0].type === 'text' ? content[0].text : content,\n },\n ]\n : []\n\n await sendEventToPosthog({\n client: phClient,\n distinctId: options.posthogDistinctId,\n traceId: options.posthogTraceId ?? uuidv4(),\n model: modelId,\n provider: provider,\n input: options.posthogPrivacyMode ? '' : mapVercelPrompt(params.prompt),\n output: output,\n latency,\n baseURL,\n params: mergedParams as any,\n httpStatus: 200,\n usage,\n tools: availableTools,\n captureImmediate: options.posthogCaptureImmediate,\n })\n },\n })\n\n return {\n stream: stream.pipeThrough(transformStream),\n ...rest,\n }\n } catch (error: any) {\n await sendEventToPosthog({\n client: phClient,\n distinctId: options.posthogDistinctId,\n traceId: options.posthogTraceId ?? uuidv4(),\n model: modelId,\n provider: provider,\n input: options.posthogPrivacyMode ? '' : mapVercelPrompt(params.prompt),\n output: [],\n latency: 0,\n baseURL: '',\n params: mergedParams as any,\n httpStatus: error?.status ? error.status : 500,\n usage: {\n inputTokens: 0,\n outputTokens: 0,\n },\n isError: true,\n error: truncate(JSON.stringify(error)),\n tools: availableTools,\n captureImmediate: options.posthogCaptureImmediate,\n })\n throw error\n }\n },\n }\n\n return middleware\n}\n\nexport const wrapVercelLanguageModel = (\n model: LanguageModelV2,\n phClient: PostHog,\n options: ClientOptions\n): LanguageModelV2 => {\n const traceId = options.posthogTraceId ?? uuidv4()\n const middleware = createInstrumentationMiddleware(phClient, model, {\n ...options,\n posthogTraceId: traceId,\n posthogDistinctId: options.posthogDistinctId,\n })\n\n const wrappedModel = wrapLanguageModel({\n model,\n middleware,\n })\n\n return wrappedModel\n}\n"],"names":["MAX_OUTPUT_SIZE","STRING_FORMAT","getModelParams","params","modelParams","paramKeys","key","undefined","withPrivacyMode","client","privacyMode","input","privacy_mode","truncate","str","buffer","Buffer","from","length","truncatedBuffer","slice","toString","error","console","extractAvailableToolCalls","provider","tools","sanitizeValues","obj","jsonSafe","JSON","parse","stringify","Array","isArray","map","Object","fromEntries","entries","k","v","sendEventToPosthog","distinctId","traceId","model","output","latency","baseURL","httpStatus","usage","isError","captureImmediate","capture","Promise","resolve","safeInput","safeOutput","safeError","errorData","$ai_is_error","$ai_error","costOverrideData","posthogCostOverride","inputCostUSD","inputCost","inputTokens","outputCostUSD","outputCost","outputTokens","$ai_input_cost_usd","$ai_output_cost_usd","$ai_total_cost_usd","additionalTokenValues","reasoningTokens","$ai_reasoning_tokens","cacheReadInputTokens","$ai_cache_read_input_tokens","cacheCreationInputTokens","$ai_cache_creation_input_tokens","properties","$ai_lib","$ai_lib_version","version","$ai_provider","posthogProviderOverride","$ai_model","posthogModelOverride","$ai_model_parameters","$ai_input","posthogPrivacyMode","$ai_output_choices","$ai_http_status","$ai_input_tokens","$ai_output_tokens","$ai_latency","$ai_trace_id","$ai_base_url","posthogProperties","$process_person_profile","$ai_tools","event","groups","posthogGroups","isString","value","REDACTED_IMAGE_PLACEHOLDER","isBase64DataUrl","test","isValidUrl","URL","startsWith","isRawBase64","redactBase64DataUrl","mapVercelParams","temperature","max_output_tokens","maxOutputTokens","top_p","topP","frequency_penalty","frequencyPenalty","presence_penalty","presencePenalty","stop","stopSequences","stream","mapVercelPrompt","messages","inputs","message","content","role","type","text","String","c","fileData","contentData","data","file","mediaType","reasoning","toolCallId","toolName","serialized","removedCount","initialSize","i","byteLength","shift","unshift","mapVercelOutput","result","item","id","function","name","arguments","args","sourceType","url","title","jsonOutput","extractProvider","toLowerCase","providerName","split","createInstrumentationMiddleware","phClient","options","middleware","wrapGenerate","doGenerate","startTime","Date","now","mergedParams","availableTools","modelId","response","providerMetadata","anthropic","cachedInputTokens","posthogDistinctId","posthogTraceId","uuidv4","prompt","posthogCaptureImmediate","status","wrapStream","doStream","generatedText","reasoningText","toolCallsInProgress","Map","rest","transformStream","TransformStream","transform","chunk","controller","delta","set","toolCall","get","enqueue","flush","push","values","pipeThrough","wrapVercelLanguageModel","wrappedModel","wrapLanguageModel"],"mappings":"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"}
@@ -2,6 +2,8 @@ import { wrapLanguageModel } from 'ai';
2
2
  import { v4 } from 'uuid';
3
3
  import { Buffer } from 'buffer';
4
4
 
5
+ var version = "6.2.0";
6
+
5
7
  // limit large outputs by truncating to 200kb (approx 200k bytes)
6
8
  const MAX_OUTPUT_SIZE = 200000;
7
9
  const STRING_FORMAT = 'utf8';
@@ -41,9 +43,8 @@ const truncate = str => {
41
43
  */
42
44
  const extractAvailableToolCalls = (provider, params) => {
43
45
  {
44
- // Vercel AI SDK stores tools in params.mode.tools when mode type is 'regular'
45
- if (params.mode?.type === 'regular' && params.mode.tools) {
46
- return params.mode.tools;
46
+ if (params.tools) {
47
+ return params.tools;
47
48
  }
48
49
  return null;
49
50
  }
@@ -116,6 +117,8 @@ const sendEventToPosthog = async ({
116
117
  } : {})
117
118
  };
118
119
  const properties = {
120
+ $ai_lib: 'posthog-ai',
121
+ $ai_lib_version: version,
119
122
  $ai_provider: params.posthogProviderOverride ?? provider,
120
123
  $ai_model: params.posthogModelOverride ?? model,
121
124
  $ai_model_parameters: getModelParams(params),
@@ -201,6 +204,8 @@ function redactBase64DataUrl(str) {
201
204
  return str;
202
205
  }
203
206
 
207
+ // Content types for the output array
208
+
204
209
  const mapVercelParams = params => {
205
210
  return {
206
211
  temperature: params.temperature,
@@ -491,6 +496,8 @@ const createInstrumentationMiddleware = (phClient, model, options) => {
491
496
  const availableTools = extractAvailableToolCalls('vercel', params);
492
497
  const baseURL = ''; // cannot currently get baseURL from vercel
493
498
 
499
+ // Map to track in-progress tool calls
500
+ const toolCallsInProgress = new Map();
494
501
  try {
495
502
  const {
496
503
  stream,
@@ -505,6 +512,35 @@ const createInstrumentationMiddleware = (phClient, model, options) => {
505
512
  if (chunk.type === 'reasoning-delta') {
506
513
  reasoningText += chunk.delta; // New in v5
507
514
  }
515
+
516
+ // Handle tool call chunks
517
+ if (chunk.type === 'tool-input-start') {
518
+ // Initialize a new tool call
519
+ toolCallsInProgress.set(chunk.id, {
520
+ toolCallId: chunk.id,
521
+ toolName: chunk.toolName,
522
+ input: ''
523
+ });
524
+ }
525
+ if (chunk.type === 'tool-input-delta') {
526
+ // Accumulate tool call arguments
527
+ const toolCall = toolCallsInProgress.get(chunk.id);
528
+ if (toolCall) {
529
+ toolCall.input += chunk.delta;
530
+ }
531
+ }
532
+ if (chunk.type === 'tool-input-end') {
533
+ // Tool call is complete, keep it in the map for final processing
534
+ // Nothing specific to do here, the tool call is already complete
535
+ }
536
+ if (chunk.type === 'tool-call') {
537
+ // Direct tool call chunk (complete tool call)
538
+ toolCallsInProgress.set(chunk.toolCallId, {
539
+ toolCallId: chunk.toolCallId,
540
+ toolName: chunk.toolName,
541
+ input: chunk.input
542
+ });
543
+ }
508
544
  if (chunk.type === 'finish') {
509
545
  const providerMetadata = chunk.providerMetadata;
510
546
  const additionalTokenValues = {
@@ -539,6 +575,20 @@ const createInstrumentationMiddleware = (phClient, model, options) => {
539
575
  });
540
576
  }
541
577
 
578
+ // Add completed tool calls to content
579
+ for (const toolCall of toolCallsInProgress.values()) {
580
+ if (toolCall.toolName) {
581
+ content.push({
582
+ type: 'tool-call',
583
+ id: toolCall.toolCallId,
584
+ function: {
585
+ name: toolCall.toolName,
586
+ arguments: toolCall.input
587
+ }
588
+ });
589
+ }
590
+ }
591
+
542
592
  // Structure output like mapVercelOutput does
543
593
  const output = content.length > 0 ? [{
544
594
  role: 'assistant',