@posthog/ai 6.3.2 → 6.4.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.
@@ -1 +1 @@
1
- {"version":3,"file":"index.mjs","sources":["../../src/utils.ts","../../src/typeGuards.ts","../../src/sanitization.ts","../../src/gemini/index.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 EmbeddingCreateParams = OpenAIOrignal.EmbeddingCreateParams\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 MonitoringEventPropertiesWithDefaults {\n distinctId?: string\n traceId: string\n properties?: Record<string, any>\n privacyMode: boolean\n groups?: Record<string, any>\n modelOverride?: string\n providerOverride?: string\n costOverride?: CostOverride\n captureImmediate?: boolean\n}\n\nexport type MonitoringEventProperties = Partial<MonitoringEventPropertiesWithDefaults>\n\nexport type MonitoringParams = {\n [K in keyof MonitoringEventProperties as `posthog${Capitalize<string & K>}`]: MonitoringEventProperties[K]\n}\n\nexport interface CostOverride {\n inputCost: number\n outputCost: number\n}\n\nexport const getModelParams = (\n params:\n | ((ChatCompletionCreateParamsBase | MessageCreateParams | ResponseCreateParams | EmbeddingCreateParams) &\n MonitoringParams)\n | 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\nfunction toSafeString(input: unknown): string {\n if (input === undefined || input === null) {\n return ''\n }\n if (typeof input === 'string') {\n return input\n }\n try {\n return JSON.stringify(input)\n } catch {\n console.warn('Failed to stringify input', input)\n return ''\n }\n}\n\nexport const truncate = (input: unknown): string => {\n const str = toSafeString(input)\n if (str === '') {\n return ''\n }\n\n // Check if we need to truncate and ensure STRING_FORMAT is respected\n const encoder = new TextEncoder()\n const buffer = encoder.encode(str)\n if (buffer.length <= MAX_OUTPUT_SIZE) {\n // Ensure STRING_FORMAT is respected\n return new TextDecoder(STRING_FORMAT).decode(buffer)\n }\n\n // Truncate the buffer and ensure a valid string is returned\n const truncatedBuffer = buffer.slice(0, MAX_OUTPUT_SIZE)\n // fatal: false means we get U+FFFD at the end if truncation broke the encoding\n const decoder = new TextDecoder(STRING_FORMAT, { fatal: false })\n let truncatedStr = decoder.decode(truncatedBuffer)\n if (truncatedStr.endsWith('\\uFFFD')) {\n truncatedStr = truncatedStr.slice(0, -1)\n }\n return `${truncatedStr}... [truncated]`\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 enum AIEvent {\n Generation = '$ai_generation',\n Embedding = '$ai_embedding',\n}\n\nexport type SendEventToPosthogParams = {\n client: PostHog\n eventType?: AIEvent\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 | EmbeddingCreateParams) &\n 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 eventType = AIEvent.Generation,\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 ...(usage.outputTokens !== undefined ? { $ai_output_tokens: usage.outputTokens } : {}),\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: eventType,\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 {\n GoogleGenAI,\n GenerateContentResponse as GeminiResponse,\n GenerateContentParameters,\n Part,\n GenerateContentResponseUsageMetadata,\n} from '@google/genai'\nimport { PostHog } from 'posthog-node'\nimport { v4 as uuidv4 } from 'uuid'\nimport { MonitoringParams, sendEventToPosthog, extractAvailableToolCalls, formatResponseGemini } from '../utils'\nimport { sanitizeGemini } from '../sanitization'\nimport type { TokenUsage, FormattedContent, FormattedContentItem, FormattedMessage } from '../types'\n\ninterface MonitoringGeminiConfig {\n apiKey?: string\n vertexai?: boolean\n project?: string\n location?: string\n apiVersion?: string\n posthog: PostHog\n}\n\nexport class PostHogGoogleGenAI {\n private readonly phClient: PostHog\n private readonly client: GoogleGenAI\n public models: WrappedModels\n\n constructor(config: MonitoringGeminiConfig) {\n const { posthog, ...geminiConfig } = config\n this.phClient = posthog\n this.client = new GoogleGenAI(geminiConfig)\n this.models = new WrappedModels(this.client, this.phClient)\n }\n}\n\nexport class WrappedModels {\n private readonly phClient: PostHog\n private readonly client: GoogleGenAI\n\n constructor(client: GoogleGenAI, phClient: PostHog) {\n this.client = client\n this.phClient = phClient\n }\n\n public async generateContent(params: GenerateContentParameters & MonitoringParams): Promise<GeminiResponse> {\n const { posthogDistinctId, posthogTraceId, posthogCaptureImmediate, ...geminiParams } = params\n\n const traceId = posthogTraceId ?? uuidv4()\n const startTime = Date.now()\n\n try {\n const response = await this.client.models.generateContent(geminiParams as GenerateContentParameters)\n const latency = (Date.now() - startTime) / 1000\n\n const availableTools = extractAvailableToolCalls('gemini', geminiParams)\n\n const metadata = response.usageMetadata\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n model: geminiParams.model,\n provider: 'gemini',\n input: this.formatInputForPostHog(geminiParams.contents),\n output: formatResponseGemini(response),\n latency,\n baseURL: 'https://generativelanguage.googleapis.com',\n params: params as GenerateContentParameters & MonitoringParams,\n httpStatus: 200,\n usage: {\n inputTokens: metadata?.promptTokenCount ?? 0,\n outputTokens: metadata?.candidatesTokenCount ?? 0,\n reasoningTokens:\n (metadata as GenerateContentResponseUsageMetadata & { thoughtsTokenCount?: number })?.thoughtsTokenCount ??\n 0,\n cacheReadInputTokens: metadata?.cachedContentTokenCount ?? 0,\n },\n tools: availableTools,\n captureImmediate: posthogCaptureImmediate,\n })\n\n return response\n } catch (error: unknown) {\n const latency = (Date.now() - startTime) / 1000\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n model: geminiParams.model,\n provider: 'gemini',\n input: this.formatInputForPostHog(geminiParams.contents),\n output: [],\n latency,\n baseURL: 'https://generativelanguage.googleapis.com',\n params: params as GenerateContentParameters & MonitoringParams,\n httpStatus: (error as { status?: number })?.status ?? 500,\n usage: {\n inputTokens: 0,\n outputTokens: 0,\n },\n isError: true,\n error: JSON.stringify(error),\n captureImmediate: posthogCaptureImmediate,\n })\n throw error\n }\n }\n\n public async *generateContentStream(\n params: GenerateContentParameters & MonitoringParams\n ): AsyncGenerator<GeminiResponse, void, unknown> {\n const { posthogDistinctId, posthogTraceId, posthogCaptureImmediate, ...geminiParams } = params\n\n const traceId = posthogTraceId ?? uuidv4()\n const startTime = Date.now()\n const accumulatedContent: FormattedContent = []\n let usage: TokenUsage = {\n inputTokens: 0,\n outputTokens: 0,\n }\n\n try {\n const stream = await this.client.models.generateContentStream(geminiParams as GenerateContentParameters)\n\n for await (const chunk of stream) {\n // Handle text content\n if (chunk.text) {\n // Find if we already have a text item to append to\n let lastTextItem: FormattedContentItem | undefined\n for (let i = accumulatedContent.length - 1; i >= 0; i--) {\n if (accumulatedContent[i].type === 'text') {\n lastTextItem = accumulatedContent[i]\n break\n }\n }\n\n if (lastTextItem && lastTextItem.type === 'text') {\n lastTextItem.text += chunk.text\n } else {\n accumulatedContent.push({ type: 'text', text: chunk.text })\n }\n }\n\n // Handle function calls from candidates\n if (chunk.candidates && Array.isArray(chunk.candidates)) {\n for (const candidate of chunk.candidates) {\n if (candidate.content && candidate.content.parts) {\n for (const part of candidate.content.parts) {\n // Type-safe check for functionCall\n if ('functionCall' in part) {\n const funcCall = (part as Part & { functionCall?: { name?: string; args?: unknown } }).functionCall\n if (funcCall?.name) {\n accumulatedContent.push({\n type: 'function',\n function: {\n name: funcCall.name,\n arguments: funcCall.args || {},\n },\n })\n }\n }\n }\n }\n }\n }\n\n // Update usage metadata - handle both old and new field names\n if (chunk.usageMetadata) {\n const metadata = chunk.usageMetadata as GenerateContentResponseUsageMetadata\n usage = {\n inputTokens: metadata.promptTokenCount ?? 0,\n outputTokens: metadata.candidatesTokenCount ?? 0,\n reasoningTokens:\n (metadata as GenerateContentResponseUsageMetadata & { thoughtsTokenCount?: number }).thoughtsTokenCount ??\n 0,\n cacheReadInputTokens: metadata.cachedContentTokenCount ?? 0,\n }\n }\n yield chunk\n }\n\n const latency = (Date.now() - startTime) / 1000\n\n const availableTools = extractAvailableToolCalls('gemini', geminiParams)\n\n // Format output similar to formatResponseGemini\n const output = accumulatedContent.length > 0 ? [{ role: 'assistant', content: accumulatedContent }] : []\n\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n model: geminiParams.model,\n provider: 'gemini',\n input: this.formatInputForPostHog(geminiParams.contents),\n output,\n latency,\n baseURL: 'https://generativelanguage.googleapis.com',\n params: params as GenerateContentParameters & MonitoringParams,\n httpStatus: 200,\n usage,\n tools: availableTools,\n captureImmediate: posthogCaptureImmediate,\n })\n } catch (error: unknown) {\n const latency = (Date.now() - startTime) / 1000\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n model: geminiParams.model,\n provider: 'gemini',\n input: this.formatInputForPostHog(geminiParams.contents),\n output: [],\n latency,\n baseURL: 'https://generativelanguage.googleapis.com',\n params: params as GenerateContentParameters & MonitoringParams,\n httpStatus: (error as { status?: number })?.status ?? 500,\n usage: {\n inputTokens: 0,\n outputTokens: 0,\n },\n isError: true,\n error: JSON.stringify(error),\n captureImmediate: posthogCaptureImmediate,\n })\n throw error\n }\n }\n\n private formatInput(contents: unknown): FormattedMessage[] {\n if (typeof contents === 'string') {\n return [{ role: 'user', content: contents }]\n }\n\n if (Array.isArray(contents)) {\n return contents.map((item) => {\n if (typeof item === 'string') {\n return { role: 'user', content: item }\n }\n\n if (item && typeof item === 'object') {\n const obj = item as Record<string, unknown>\n if ('text' in obj && obj.text) {\n return { role: (obj.role as string) || 'user', content: obj.text }\n }\n\n if ('content' in obj && obj.content) {\n return { role: (obj.role as string) || 'user', content: obj.content }\n }\n\n if ('parts' in obj && Array.isArray(obj.parts)) {\n return {\n role: (obj.role as string) || 'user',\n content: obj.parts.map((part: unknown) => {\n if (part && typeof part === 'object' && 'text' in part) {\n return (part as { text: unknown }).text\n }\n return part\n }),\n }\n }\n }\n\n return { role: 'user', content: String(item) }\n })\n }\n\n if (contents && typeof contents === 'object') {\n const obj = contents as Record<string, unknown>\n if ('text' in obj && obj.text) {\n return [{ role: 'user', content: obj.text }]\n }\n\n if ('content' in obj && obj.content) {\n return [{ role: 'user', content: obj.content }]\n }\n }\n\n return [{ role: 'user', content: String(contents) }]\n }\n\n private formatInputForPostHog(contents: unknown): unknown {\n const sanitized = sanitizeGemini(contents)\n return this.formatInput(sanitized)\n }\n}\n\nexport default PostHogGoogleGenAI\nexport { PostHogGoogleGenAI as Gemini }\n"],"names":["STRING_FORMAT","getModelParams","params","modelParams","paramKeys","key","undefined","formatResponseGemini","response","output","candidates","Array","isArray","candidate","content","parts","part","text","push","type","functionCall","function","name","arguments","args","length","role","withPrivacyMode","client","privacyMode","input","privacy_mode","extractAvailableToolCalls","provider","config","tools","AIEvent","sanitizeValues","obj","jsonSafe","JSON","parse","stringify","Buffer","from","toString","map","Object","fromEntries","entries","k","v","sendEventToPosthog","eventType","Generation","distinctId","traceId","model","latency","baseURL","httpStatus","usage","isError","error","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","isObject","value","REDACTED_IMAGE_PLACEHOLDER","sanitizeGeminiPart","inlineData","data","processGeminiItem","item","sanitizeGemini","PostHogGoogleGenAI","constructor","posthog","geminiConfig","phClient","GoogleGenAI","models","WrappedModels","generateContent","posthogDistinctId","posthogTraceId","posthogCaptureImmediate","geminiParams","uuidv4","startTime","Date","now","availableTools","metadata","usageMetadata","formatInputForPostHog","contents","promptTokenCount","candidatesTokenCount","thoughtsTokenCount","cachedContentTokenCount","status","generateContentStream","accumulatedContent","stream","chunk","lastTextItem","i","funcCall","formatInput","String","sanitized"],"mappings":"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"}
1
+ {"version":3,"file":"index.mjs","sources":["../../src/utils.ts","../../src/typeGuards.ts","../../src/sanitization.ts","../../src/gemini/index.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'\nimport { v4 as uuidv4 } from 'uuid'\n\ntype ChatCompletionCreateParamsBase = OpenAIOrignal.Chat.Completions.ChatCompletionCreateParams\ntype MessageCreateParams = AnthropicOriginal.Messages.MessageCreateParams\ntype ResponseCreateParams = OpenAIOrignal.Responses.ResponseCreateParams\ntype EmbeddingCreateParams = OpenAIOrignal.EmbeddingCreateParams\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 MonitoringEventPropertiesWithDefaults {\n distinctId?: string\n traceId: string\n properties?: Record<string, any>\n privacyMode: boolean\n groups?: Record<string, any>\n modelOverride?: string\n providerOverride?: string\n costOverride?: CostOverride\n captureImmediate?: boolean\n}\n\nexport type MonitoringEventProperties = Partial<MonitoringEventPropertiesWithDefaults>\n\nexport type MonitoringParams = {\n [K in keyof MonitoringEventProperties as `posthog${Capitalize<string & K>}`]: MonitoringEventProperties[K]\n}\n\nexport interface CostOverride {\n inputCost: number\n outputCost: number\n}\n\nexport const getModelParams = (\n params:\n | ((ChatCompletionCreateParamsBase | MessageCreateParams | ResponseCreateParams | EmbeddingCreateParams) &\n MonitoringParams)\n | 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\nfunction toSafeString(input: unknown): string {\n if (input === undefined || input === null) {\n return ''\n }\n if (typeof input === 'string') {\n return input\n }\n try {\n return JSON.stringify(input)\n } catch {\n console.warn('Failed to stringify input', input)\n return ''\n }\n}\n\nexport const truncate = (input: unknown): string => {\n const str = toSafeString(input)\n if (str === '') {\n return ''\n }\n\n // Check if we need to truncate and ensure STRING_FORMAT is respected\n const encoder = new TextEncoder()\n const buffer = encoder.encode(str)\n if (buffer.length <= MAX_OUTPUT_SIZE) {\n // Ensure STRING_FORMAT is respected\n return new TextDecoder(STRING_FORMAT).decode(buffer)\n }\n\n // Truncate the buffer and ensure a valid string is returned\n const truncatedBuffer = buffer.slice(0, MAX_OUTPUT_SIZE)\n // fatal: false means we get U+FFFD at the end if truncation broke the encoding\n const decoder = new TextDecoder(STRING_FORMAT, { fatal: false })\n let truncatedStr = decoder.decode(truncatedBuffer)\n if (truncatedStr.endsWith('\\uFFFD')) {\n truncatedStr = truncatedStr.slice(0, -1)\n }\n return `${truncatedStr}... [truncated]`\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 enum AIEvent {\n Generation = '$ai_generation',\n Embedding = '$ai_embedding',\n}\n\nexport type SendEventToPosthogParams = {\n client: PostHog\n eventType?: AIEvent\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 | EmbeddingCreateParams) &\n 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\nconst POSTHOG_PARAMS_MAP: Record<keyof MonitoringParams, string> = {\n posthogDistinctId: 'distinctId',\n posthogTraceId: 'traceId',\n posthogProperties: 'properties',\n posthogPrivacyMode: 'privacyMode',\n posthogGroups: 'groups',\n posthogModelOverride: 'modelOverride',\n posthogProviderOverride: 'providerOverride',\n posthogCostOverride: 'costOverride',\n posthogCaptureImmediate: 'captureImmediate',\n}\n\nexport function extractPosthogParams<T>(body: T & MonitoringParams): {\n providerParams: T\n posthogParams: MonitoringEventPropertiesWithDefaults\n} {\n const providerParams: Record<string, unknown> = {}\n const posthogParams: Record<string, unknown> = {}\n\n for (const [key, value] of Object.entries(body)) {\n if (POSTHOG_PARAMS_MAP[key as keyof MonitoringParams]) {\n posthogParams[POSTHOG_PARAMS_MAP[key as keyof MonitoringParams]] = value\n } else if (key.startsWith('posthog')) {\n console.warn(`Unknown Posthog parameter ${key}`)\n } else {\n providerParams[key] = value\n }\n }\n\n return {\n providerParams: providerParams as T,\n posthogParams: addDefaults(posthogParams),\n }\n}\n\nfunction addDefaults(params: MonitoringEventProperties): MonitoringEventPropertiesWithDefaults {\n return {\n ...params,\n privacyMode: params.privacyMode ?? false,\n traceId: params.traceId ?? uuidv4(),\n }\n}\n\nexport const sendEventToPosthog = async ({\n client,\n eventType = AIEvent.Generation,\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 ...(usage.outputTokens !== undefined ? { $ai_output_tokens: usage.outputTokens } : {}),\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: eventType,\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 {\n GoogleGenAI,\n GenerateContentResponse as GeminiResponse,\n GenerateContentParameters,\n Part,\n GenerateContentResponseUsageMetadata,\n} from '@google/genai'\nimport { PostHog } from 'posthog-node'\nimport {\n MonitoringParams,\n sendEventToPosthog,\n extractAvailableToolCalls,\n formatResponseGemini,\n extractPosthogParams,\n} from '../utils'\nimport { sanitizeGemini } from '../sanitization'\nimport type { TokenUsage, FormattedContent, FormattedContentItem, FormattedMessage } from '../types'\n\ninterface MonitoringGeminiConfig {\n apiKey?: string\n vertexai?: boolean\n project?: string\n location?: string\n apiVersion?: string\n posthog: PostHog\n}\n\nexport class PostHogGoogleGenAI {\n private readonly phClient: PostHog\n private readonly client: GoogleGenAI\n public models: WrappedModels\n\n constructor(config: MonitoringGeminiConfig) {\n const { posthog, ...geminiConfig } = config\n this.phClient = posthog\n this.client = new GoogleGenAI(geminiConfig)\n this.models = new WrappedModels(this.client, this.phClient)\n }\n}\n\nexport class WrappedModels {\n private readonly phClient: PostHog\n private readonly client: GoogleGenAI\n\n constructor(client: GoogleGenAI, phClient: PostHog) {\n this.client = client\n this.phClient = phClient\n }\n\n public async generateContent(params: GenerateContentParameters & MonitoringParams): Promise<GeminiResponse> {\n const { providerParams: geminiParams, posthogParams } = extractPosthogParams(params)\n const startTime = Date.now()\n\n try {\n const response = await this.client.models.generateContent(geminiParams as GenerateContentParameters)\n const latency = (Date.now() - startTime) / 1000\n\n const availableTools = extractAvailableToolCalls('gemini', geminiParams)\n\n const metadata = response.usageMetadata\n await sendEventToPosthog({\n client: this.phClient,\n ...posthogParams,\n model: geminiParams.model,\n provider: 'gemini',\n input: this.formatInputForPostHog(geminiParams.contents),\n output: formatResponseGemini(response),\n latency,\n baseURL: 'https://generativelanguage.googleapis.com',\n params: params as GenerateContentParameters & MonitoringParams,\n httpStatus: 200,\n usage: {\n inputTokens: metadata?.promptTokenCount ?? 0,\n outputTokens: metadata?.candidatesTokenCount ?? 0,\n reasoningTokens:\n (metadata as GenerateContentResponseUsageMetadata & { thoughtsTokenCount?: number })?.thoughtsTokenCount ??\n 0,\n cacheReadInputTokens: metadata?.cachedContentTokenCount ?? 0,\n },\n tools: availableTools,\n })\n\n return response\n } catch (error: unknown) {\n const latency = (Date.now() - startTime) / 1000\n await sendEventToPosthog({\n client: this.phClient,\n ...posthogParams,\n model: geminiParams.model,\n provider: 'gemini',\n input: this.formatInputForPostHog(geminiParams.contents),\n output: [],\n latency,\n baseURL: 'https://generativelanguage.googleapis.com',\n params: params as GenerateContentParameters & MonitoringParams,\n httpStatus: (error as { status?: number })?.status ?? 500,\n usage: {\n inputTokens: 0,\n outputTokens: 0,\n },\n isError: true,\n error: JSON.stringify(error),\n })\n throw error\n }\n }\n\n public async *generateContentStream(\n params: GenerateContentParameters & MonitoringParams\n ): AsyncGenerator<GeminiResponse, void, unknown> {\n const { providerParams: geminiParams, posthogParams } = extractPosthogParams(params)\n const startTime = Date.now()\n const accumulatedContent: FormattedContent = []\n let usage: TokenUsage = {\n inputTokens: 0,\n outputTokens: 0,\n }\n\n try {\n const stream = await this.client.models.generateContentStream(geminiParams as GenerateContentParameters)\n\n for await (const chunk of stream) {\n // Handle text content\n if (chunk.text) {\n // Find if we already have a text item to append to\n let lastTextItem: FormattedContentItem | undefined\n for (let i = accumulatedContent.length - 1; i >= 0; i--) {\n if (accumulatedContent[i].type === 'text') {\n lastTextItem = accumulatedContent[i]\n break\n }\n }\n\n if (lastTextItem && lastTextItem.type === 'text') {\n lastTextItem.text += chunk.text\n } else {\n accumulatedContent.push({ type: 'text', text: chunk.text })\n }\n }\n\n // Handle function calls from candidates\n if (chunk.candidates && Array.isArray(chunk.candidates)) {\n for (const candidate of chunk.candidates) {\n if (candidate.content && candidate.content.parts) {\n for (const part of candidate.content.parts) {\n // Type-safe check for functionCall\n if ('functionCall' in part) {\n const funcCall = (part as Part & { functionCall?: { name?: string; args?: unknown } }).functionCall\n if (funcCall?.name) {\n accumulatedContent.push({\n type: 'function',\n function: {\n name: funcCall.name,\n arguments: funcCall.args || {},\n },\n })\n }\n }\n }\n }\n }\n }\n\n // Update usage metadata - handle both old and new field names\n if (chunk.usageMetadata) {\n const metadata = chunk.usageMetadata as GenerateContentResponseUsageMetadata\n usage = {\n inputTokens: metadata.promptTokenCount ?? 0,\n outputTokens: metadata.candidatesTokenCount ?? 0,\n reasoningTokens:\n (metadata as GenerateContentResponseUsageMetadata & { thoughtsTokenCount?: number }).thoughtsTokenCount ??\n 0,\n cacheReadInputTokens: metadata.cachedContentTokenCount ?? 0,\n }\n }\n yield chunk\n }\n\n const latency = (Date.now() - startTime) / 1000\n\n const availableTools = extractAvailableToolCalls('gemini', geminiParams)\n\n // Format output similar to formatResponseGemini\n const output = accumulatedContent.length > 0 ? [{ role: 'assistant', content: accumulatedContent }] : []\n\n await sendEventToPosthog({\n client: this.phClient,\n ...posthogParams,\n model: geminiParams.model,\n provider: 'gemini',\n input: this.formatInputForPostHog(geminiParams.contents),\n output,\n latency,\n baseURL: 'https://generativelanguage.googleapis.com',\n params: params as GenerateContentParameters & MonitoringParams,\n httpStatus: 200,\n usage,\n tools: availableTools,\n })\n } catch (error: unknown) {\n const latency = (Date.now() - startTime) / 1000\n await sendEventToPosthog({\n client: this.phClient,\n ...posthogParams,\n model: geminiParams.model,\n provider: 'gemini',\n input: this.formatInputForPostHog(geminiParams.contents),\n output: [],\n latency,\n baseURL: 'https://generativelanguage.googleapis.com',\n params: params as GenerateContentParameters & MonitoringParams,\n httpStatus: (error as { status?: number })?.status ?? 500,\n usage: {\n inputTokens: 0,\n outputTokens: 0,\n },\n isError: true,\n error: JSON.stringify(error),\n })\n throw error\n }\n }\n\n private formatInput(contents: unknown): FormattedMessage[] {\n if (typeof contents === 'string') {\n return [{ role: 'user', content: contents }]\n }\n\n if (Array.isArray(contents)) {\n return contents.map((item) => {\n if (typeof item === 'string') {\n return { role: 'user', content: item }\n }\n\n if (item && typeof item === 'object') {\n const obj = item as Record<string, unknown>\n if ('text' in obj && obj.text) {\n return { role: (obj.role as string) || 'user', content: obj.text }\n }\n\n if ('content' in obj && obj.content) {\n return { role: (obj.role as string) || 'user', content: obj.content }\n }\n\n if ('parts' in obj && Array.isArray(obj.parts)) {\n return {\n role: (obj.role as string) || 'user',\n content: obj.parts.map((part: unknown) => {\n if (part && typeof part === 'object' && 'text' in part) {\n return (part as { text: unknown }).text\n }\n return part\n }),\n }\n }\n }\n\n return { role: 'user', content: String(item) }\n })\n }\n\n if (contents && typeof contents === 'object') {\n const obj = contents as Record<string, unknown>\n if ('text' in obj && obj.text) {\n return [{ role: 'user', content: obj.text }]\n }\n\n if ('content' in obj && obj.content) {\n return [{ role: 'user', content: obj.content }]\n }\n }\n\n return [{ role: 'user', content: String(contents) }]\n }\n\n private formatInputForPostHog(contents: unknown): unknown {\n const sanitized = sanitizeGemini(contents)\n return this.formatInput(sanitized)\n }\n}\n\nexport default PostHogGoogleGenAI\nexport { PostHogGoogleGenAI as Gemini }\n"],"names":["STRING_FORMAT","getModelParams","params","modelParams","paramKeys","key","undefined","formatResponseGemini","response","output","candidates","Array","isArray","candidate","content","parts","part","text","push","type","functionCall","function","name","arguments","args","length","role","withPrivacyMode","client","privacyMode","input","privacy_mode","extractAvailableToolCalls","provider","config","tools","AIEvent","sanitizeValues","obj","jsonSafe","JSON","parse","stringify","Buffer","from","toString","map","Object","fromEntries","entries","k","v","POSTHOG_PARAMS_MAP","posthogDistinctId","posthogTraceId","posthogProperties","posthogPrivacyMode","posthogGroups","posthogModelOverride","posthogProviderOverride","posthogCostOverride","posthogCaptureImmediate","extractPosthogParams","body","providerParams","posthogParams","value","startsWith","console","warn","addDefaults","traceId","uuidv4","sendEventToPosthog","eventType","Generation","distinctId","model","latency","baseURL","httpStatus","usage","isError","error","captureImmediate","capture","Promise","resolve","safeInput","safeOutput","safeError","errorData","$ai_is_error","$ai_error","costOverrideData","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","$ai_model","$ai_model_parameters","$ai_input","$ai_output_choices","$ai_http_status","$ai_input_tokens","$ai_output_tokens","$ai_latency","$ai_trace_id","$ai_base_url","$process_person_profile","$ai_tools","event","groups","isObject","REDACTED_IMAGE_PLACEHOLDER","sanitizeGeminiPart","inlineData","data","processGeminiItem","item","sanitizeGemini","PostHogGoogleGenAI","constructor","posthog","geminiConfig","phClient","GoogleGenAI","models","WrappedModels","generateContent","geminiParams","startTime","Date","now","availableTools","metadata","usageMetadata","formatInputForPostHog","contents","promptTokenCount","candidatesTokenCount","thoughtsTokenCount","cachedContentTokenCount","status","generateContentStream","accumulatedContent","stream","chunk","lastTextItem","i","funcCall","formatInput","String","sanitized"],"mappings":"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"}
package/dist/index.cjs CHANGED
@@ -26,7 +26,7 @@ function _interopNamespaceDefault(e) {
26
26
 
27
27
  var uuid__namespace = /*#__PURE__*/_interopNamespaceDefault(uuid);
28
28
 
29
- var version = "6.3.2";
29
+ var version = "6.4.0";
30
30
 
31
31
  // limit large outputs by truncating to 200kb (approx 200k bytes)
32
32
  const MAX_OUTPUT_SIZE = 200000;
@@ -311,6 +311,41 @@ function sanitizeValues(obj) {
311
311
  }
312
312
  return jsonSafe;
313
313
  }
314
+ const POSTHOG_PARAMS_MAP = {
315
+ posthogDistinctId: 'distinctId',
316
+ posthogTraceId: 'traceId',
317
+ posthogProperties: 'properties',
318
+ posthogPrivacyMode: 'privacyMode',
319
+ posthogGroups: 'groups',
320
+ posthogModelOverride: 'modelOverride',
321
+ posthogProviderOverride: 'providerOverride',
322
+ posthogCostOverride: 'costOverride',
323
+ posthogCaptureImmediate: 'captureImmediate'
324
+ };
325
+ function extractPosthogParams(body) {
326
+ const providerParams = {};
327
+ const posthogParams = {};
328
+ for (const [key, value] of Object.entries(body)) {
329
+ if (POSTHOG_PARAMS_MAP[key]) {
330
+ posthogParams[POSTHOG_PARAMS_MAP[key]] = value;
331
+ } else if (key.startsWith('posthog')) {
332
+ console.warn(`Unknown Posthog parameter ${key}`);
333
+ } else {
334
+ providerParams[key] = value;
335
+ }
336
+ }
337
+ return {
338
+ providerParams: providerParams,
339
+ posthogParams: addDefaults(posthogParams)
340
+ };
341
+ }
342
+ function addDefaults(params) {
343
+ return {
344
+ ...params,
345
+ privacyMode: params.privacyMode ?? false,
346
+ traceId: params.traceId ?? uuid.v4()
347
+ };
348
+ }
314
349
  const sendEventToPosthog = async ({
315
350
  client,
316
351
  eventType = AIEvent.Generation,
@@ -604,42 +639,6 @@ const sanitizeLangChain = data => {
604
639
  return processMessages(data, sanitizeLangChainImage);
605
640
  };
606
641
 
607
- const POSTHOG_PARAMS_MAP = {
608
- posthogDistinctId: 'distinctId',
609
- posthogTraceId: 'traceId',
610
- posthogProperties: 'properties',
611
- posthogPrivacyMode: 'privacyMode',
612
- posthogGroups: 'groups',
613
- posthogModelOverride: 'modelOverride',
614
- posthogProviderOverride: 'providerOverride',
615
- posthogCostOverride: 'costOverride',
616
- posthogCaptureImmediate: 'captureImmediate'
617
- };
618
- function extractPosthogParams(body) {
619
- const openAIParams = {};
620
- const posthogParams = {};
621
- for (const [key, value] of Object.entries(body)) {
622
- if (POSTHOG_PARAMS_MAP[key]) {
623
- posthogParams[POSTHOG_PARAMS_MAP[key]] = value;
624
- } else if (key.startsWith('posthog')) {
625
- console.warn(`Unknown Posthog parameter ${key}`);
626
- } else {
627
- openAIParams[key] = value;
628
- }
629
- }
630
- return {
631
- openAIParams: openAIParams,
632
- posthogParams: addDefaults(posthogParams)
633
- };
634
- }
635
- function addDefaults(params) {
636
- return {
637
- ...params,
638
- privacyMode: params.privacyMode ?? false,
639
- traceId: params.traceId ?? uuid.v4()
640
- };
641
- }
642
-
643
642
  const Chat = openai.OpenAI.Chat;
644
643
  const Completions = Chat.Completions;
645
644
  const Responses = openai.OpenAI.Responses;
@@ -672,7 +671,7 @@ let WrappedCompletions$1 = class WrappedCompletions extends Completions {
672
671
  // --- Implementation Signature
673
672
  create(body, options) {
674
673
  const {
675
- openAIParams,
674
+ providerParams: openAIParams,
676
675
  posthogParams
677
676
  } = extractPosthogParams(body);
678
677
  const startTime = Date.now();
@@ -873,7 +872,7 @@ let WrappedResponses$1 = class WrappedResponses extends Responses {
873
872
  // --- Implementation Signature
874
873
  create(body, options) {
875
874
  const {
876
- openAIParams,
875
+ providerParams: openAIParams,
877
876
  posthogParams
878
877
  } = extractPosthogParams(body);
879
878
  const startTime = Date.now();
@@ -1003,7 +1002,7 @@ let WrappedResponses$1 = class WrappedResponses extends Responses {
1003
1002
  }
1004
1003
  parse(body, options) {
1005
1004
  const {
1006
- openAIParams,
1005
+ providerParams: openAIParams,
1007
1006
  posthogParams
1008
1007
  } = extractPosthogParams(body);
1009
1008
  const startTime = Date.now();
@@ -1074,7 +1073,7 @@ let WrappedEmbeddings$1 = class WrappedEmbeddings extends Embeddings {
1074
1073
  }
1075
1074
  create(body, options) {
1076
1075
  const {
1077
- openAIParams,
1076
+ providerParams: openAIParams,
1078
1077
  posthogParams
1079
1078
  } = extractPosthogParams(body);
1080
1079
  const startTime = Date.now();
@@ -1153,7 +1152,7 @@ class WrappedCompletions extends openai.AzureOpenAI.Chat.Completions {
1153
1152
  // --- Implementation Signature
1154
1153
  create(body, options) {
1155
1154
  const {
1156
- openAIParams,
1155
+ providerParams: openAIParams,
1157
1156
  posthogParams
1158
1157
  } = extractPosthogParams(body);
1159
1158
  const startTime = Date.now();
@@ -1350,7 +1349,7 @@ class WrappedResponses extends openai.AzureOpenAI.Responses {
1350
1349
  // --- Implementation Signature
1351
1350
  create(body, options) {
1352
1351
  const {
1353
- openAIParams,
1352
+ providerParams: openAIParams,
1354
1353
  posthogParams
1355
1354
  } = extractPosthogParams(body);
1356
1355
  const startTime = Date.now();
@@ -1474,7 +1473,7 @@ class WrappedResponses extends openai.AzureOpenAI.Responses {
1474
1473
  }
1475
1474
  parse(body, options) {
1476
1475
  const {
1477
- openAIParams,
1476
+ providerParams: openAIParams,
1478
1477
  posthogParams
1479
1478
  } = extractPosthogParams(body);
1480
1479
  const startTime = Date.now();
@@ -1534,7 +1533,7 @@ class WrappedEmbeddings extends openai.AzureOpenAI.Embeddings {
1534
1533
  }
1535
1534
  create(body, options) {
1536
1535
  const {
1537
- openAIParams,
1536
+ providerParams: openAIParams,
1538
1537
  posthogParams
1539
1538
  } = extractPosthogParams(body);
1540
1539
  const startTime = Date.now();
@@ -2050,12 +2049,9 @@ class WrappedMessages extends AnthropicOriginal.Messages {
2050
2049
  }
2051
2050
  create(body, options) {
2052
2051
  const {
2053
- posthogDistinctId,
2054
- posthogTraceId,
2055
- posthogCaptureImmediate,
2056
- ...anthropicParams
2057
- } = body;
2058
- const traceId = posthogTraceId ?? uuid.v4();
2052
+ providerParams: anthropicParams,
2053
+ posthogParams
2054
+ } = extractPosthogParams(body);
2059
2055
  const startTime = Date.now();
2060
2056
  const parentPromise = super.create(anthropicParams, options);
2061
2057
  if (anthropicParams.stream) {
@@ -2165,8 +2161,7 @@ class WrappedMessages extends AnthropicOriginal.Messages {
2165
2161
  }];
2166
2162
  await sendEventToPosthog({
2167
2163
  client: this.phClient,
2168
- distinctId: posthogDistinctId,
2169
- traceId,
2164
+ ...posthogParams,
2170
2165
  model: anthropicParams.model,
2171
2166
  provider: 'anthropic',
2172
2167
  input: sanitizeAnthropic(mergeSystemPrompt(anthropicParams, 'anthropic')),
@@ -2176,15 +2171,13 @@ class WrappedMessages extends AnthropicOriginal.Messages {
2176
2171
  params: body,
2177
2172
  httpStatus: 200,
2178
2173
  usage,
2179
- tools: availableTools,
2180
- captureImmediate: posthogCaptureImmediate
2174
+ tools: availableTools
2181
2175
  });
2182
2176
  } catch (error) {
2183
2177
  // error handling
2184
2178
  await sendEventToPosthog({
2185
2179
  client: this.phClient,
2186
- distinctId: posthogDistinctId,
2187
- traceId,
2180
+ ...posthogParams,
2188
2181
  model: anthropicParams.model,
2189
2182
  provider: 'anthropic',
2190
2183
  input: sanitizeAnthropic(mergeSystemPrompt(anthropicParams)),
@@ -2198,8 +2191,7 @@ class WrappedMessages extends AnthropicOriginal.Messages {
2198
2191
  outputTokens: 0
2199
2192
  },
2200
2193
  isError: true,
2201
- error: JSON.stringify(error),
2202
- captureImmediate: posthogCaptureImmediate
2194
+ error: JSON.stringify(error)
2203
2195
  });
2204
2196
  }
2205
2197
  })();
@@ -2215,8 +2207,7 @@ class WrappedMessages extends AnthropicOriginal.Messages {
2215
2207
  const availableTools = extractAvailableToolCalls('anthropic', anthropicParams);
2216
2208
  await sendEventToPosthog({
2217
2209
  client: this.phClient,
2218
- distinctId: posthogDistinctId,
2219
- traceId,
2210
+ ...posthogParams,
2220
2211
  model: anthropicParams.model,
2221
2212
  provider: 'anthropic',
2222
2213
  input: sanitizeAnthropic(mergeSystemPrompt(anthropicParams)),
@@ -2231,16 +2222,14 @@ class WrappedMessages extends AnthropicOriginal.Messages {
2231
2222
  cacheCreationInputTokens: result.usage.cache_creation_input_tokens ?? 0,
2232
2223
  cacheReadInputTokens: result.usage.cache_read_input_tokens ?? 0
2233
2224
  },
2234
- tools: availableTools,
2235
- captureImmediate: posthogCaptureImmediate
2225
+ tools: availableTools
2236
2226
  });
2237
2227
  }
2238
2228
  return result;
2239
2229
  }, async error => {
2240
2230
  await sendEventToPosthog({
2241
2231
  client: this.phClient,
2242
- distinctId: posthogDistinctId,
2243
- traceId,
2232
+ ...posthogParams,
2244
2233
  model: anthropicParams.model,
2245
2234
  provider: 'anthropic',
2246
2235
  input: sanitizeAnthropic(mergeSystemPrompt(anthropicParams)),
@@ -2254,8 +2243,7 @@ class WrappedMessages extends AnthropicOriginal.Messages {
2254
2243
  outputTokens: 0
2255
2244
  },
2256
2245
  isError: true,
2257
- error: JSON.stringify(error),
2258
- captureImmediate: posthogCaptureImmediate
2246
+ error: JSON.stringify(error)
2259
2247
  });
2260
2248
  throw error;
2261
2249
  });
@@ -2282,12 +2270,9 @@ class WrappedModels {
2282
2270
  }
2283
2271
  async generateContent(params) {
2284
2272
  const {
2285
- posthogDistinctId,
2286
- posthogTraceId,
2287
- posthogCaptureImmediate,
2288
- ...geminiParams
2289
- } = params;
2290
- const traceId = posthogTraceId ?? uuid.v4();
2273
+ providerParams: geminiParams,
2274
+ posthogParams
2275
+ } = extractPosthogParams(params);
2291
2276
  const startTime = Date.now();
2292
2277
  try {
2293
2278
  const response = await this.client.models.generateContent(geminiParams);
@@ -2296,8 +2281,7 @@ class WrappedModels {
2296
2281
  const metadata = response.usageMetadata;
2297
2282
  await sendEventToPosthog({
2298
2283
  client: this.phClient,
2299
- distinctId: posthogDistinctId,
2300
- traceId,
2284
+ ...posthogParams,
2301
2285
  model: geminiParams.model,
2302
2286
  provider: 'gemini',
2303
2287
  input: this.formatInputForPostHog(geminiParams.contents),
@@ -2312,16 +2296,14 @@ class WrappedModels {
2312
2296
  reasoningTokens: metadata?.thoughtsTokenCount ?? 0,
2313
2297
  cacheReadInputTokens: metadata?.cachedContentTokenCount ?? 0
2314
2298
  },
2315
- tools: availableTools,
2316
- captureImmediate: posthogCaptureImmediate
2299
+ tools: availableTools
2317
2300
  });
2318
2301
  return response;
2319
2302
  } catch (error) {
2320
2303
  const latency = (Date.now() - startTime) / 1000;
2321
2304
  await sendEventToPosthog({
2322
2305
  client: this.phClient,
2323
- distinctId: posthogDistinctId,
2324
- traceId,
2306
+ ...posthogParams,
2325
2307
  model: geminiParams.model,
2326
2308
  provider: 'gemini',
2327
2309
  input: this.formatInputForPostHog(geminiParams.contents),
@@ -2335,20 +2317,16 @@ class WrappedModels {
2335
2317
  outputTokens: 0
2336
2318
  },
2337
2319
  isError: true,
2338
- error: JSON.stringify(error),
2339
- captureImmediate: posthogCaptureImmediate
2320
+ error: JSON.stringify(error)
2340
2321
  });
2341
2322
  throw error;
2342
2323
  }
2343
2324
  }
2344
2325
  async *generateContentStream(params) {
2345
2326
  const {
2346
- posthogDistinctId,
2347
- posthogTraceId,
2348
- posthogCaptureImmediate,
2349
- ...geminiParams
2350
- } = params;
2351
- const traceId = posthogTraceId ?? uuid.v4();
2327
+ providerParams: geminiParams,
2328
+ posthogParams
2329
+ } = extractPosthogParams(params);
2352
2330
  const startTime = Date.now();
2353
2331
  const accumulatedContent = [];
2354
2332
  let usage = {
@@ -2420,8 +2398,7 @@ class WrappedModels {
2420
2398
  }] : [];
2421
2399
  await sendEventToPosthog({
2422
2400
  client: this.phClient,
2423
- distinctId: posthogDistinctId,
2424
- traceId,
2401
+ ...posthogParams,
2425
2402
  model: geminiParams.model,
2426
2403
  provider: 'gemini',
2427
2404
  input: this.formatInputForPostHog(geminiParams.contents),
@@ -2431,15 +2408,13 @@ class WrappedModels {
2431
2408
  params: params,
2432
2409
  httpStatus: 200,
2433
2410
  usage,
2434
- tools: availableTools,
2435
- captureImmediate: posthogCaptureImmediate
2411
+ tools: availableTools
2436
2412
  });
2437
2413
  } catch (error) {
2438
2414
  const latency = (Date.now() - startTime) / 1000;
2439
2415
  await sendEventToPosthog({
2440
2416
  client: this.phClient,
2441
- distinctId: posthogDistinctId,
2442
- traceId,
2417
+ ...posthogParams,
2443
2418
  model: geminiParams.model,
2444
2419
  provider: 'gemini',
2445
2420
  input: this.formatInputForPostHog(geminiParams.contents),
@@ -2453,8 +2428,7 @@ class WrappedModels {
2453
2428
  outputTokens: 0
2454
2429
  },
2455
2430
  isError: true,
2456
- error: JSON.stringify(error),
2457
- captureImmediate: posthogCaptureImmediate
2431
+ error: JSON.stringify(error)
2458
2432
  });
2459
2433
  throw error;
2460
2434
  }