@posthog/ai 6.4.3 → 6.5.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.
- package/dist/anthropic/index.cjs +1 -1
- package/dist/anthropic/index.cjs.map +1 -1
- package/dist/anthropic/index.mjs +1 -1
- package/dist/anthropic/index.mjs.map +1 -1
- package/dist/gemini/index.cjs +29 -3
- package/dist/gemini/index.cjs.map +1 -1
- package/dist/gemini/index.mjs +29 -3
- package/dist/gemini/index.mjs.map +1 -1
- package/dist/index.cjs +50 -14
- package/dist/index.cjs.map +1 -1
- package/dist/index.mjs +50 -14
- package/dist/index.mjs.map +1 -1
- package/dist/langchain/index.cjs +31 -4
- package/dist/langchain/index.cjs.map +1 -1
- package/dist/langchain/index.mjs +31 -4
- package/dist/langchain/index.mjs.map +1 -1
- package/dist/openai/index.cjs +33 -5
- package/dist/openai/index.cjs.map +1 -1
- package/dist/openai/index.mjs +33 -5
- package/dist/openai/index.mjs.map +1 -1
- package/dist/vercel/index.cjs +39 -5
- package/dist/vercel/index.cjs.map +1 -1
- package/dist/vercel/index.mjs +39 -5
- package/dist/vercel/index.mjs.map +1 -1
- package/package.json +2 -2
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.mjs","sources":["../../src/typeGuards.ts","../../src/utils.ts","../../src/sanitization.ts","../../src/gemini/index.ts"],"sourcesContent":["// 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 { 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 { ResponseCreateParamsWithTools } from 'openai/lib/ResponsesParser'\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'\nimport { isString } from './typeGuards'\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 | ((\n | ChatCompletionCreateParamsBase\n | MessageCreateParams\n | ResponseCreateParams\n | ResponseCreateParamsWithTools\n | EmbeddingCreateParams\n ) &\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: (\n | ChatCompletionCreateParamsBase\n | MessageCreateParams\n | ResponseCreateParams\n | ResponseCreateParamsWithTools\n | EmbeddingCreateParams\n ) &\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\nexport function formatOpenAIResponsesInput(input: unknown, instructions?: string | null): FormattedMessage[] {\n const messages: FormattedMessage[] = []\n\n if (instructions) {\n messages.push({\n role: 'system',\n content: instructions,\n })\n }\n\n if (Array.isArray(input)) {\n for (const item of input) {\n if (typeof item === 'string') {\n messages.push({ role: 'user', content: item })\n } else if (item && typeof item === 'object') {\n const obj = item as Record<string, unknown>\n const role = isString(obj.role) ? obj.role : 'user'\n const content = obj.content || obj.text || String(item)\n messages.push({ role, content: String(content) })\n } else {\n messages.push({ role: 'user', content: String(item) })\n }\n }\n } else if (typeof input === 'string') {\n messages.push({ role: 'user', content: input })\n } else if (input) {\n messages.push({ role: 'user', content: String(input) })\n }\n\n return messages\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'\nimport { isString } from '../typeGuards'\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),\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),\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),\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),\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: isString(obj.role) ? obj.role : 'user', content: obj.text }\n }\n\n if ('content' in obj && obj.content) {\n return { role: isString(obj.role) ? obj.role : 'user', content: obj.content }\n }\n\n if ('parts' in obj && Array.isArray(obj.parts)) {\n return {\n role: isString(obj.role) ? obj.role : '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 extractSystemInstruction(params: GenerateContentParameters): string | null {\n if (!params || typeof params !== 'object' || !params.config) {\n return null\n }\n const config = params.config as any\n if (!('systemInstruction' in config)) {\n return null\n }\n const systemInstruction = config.systemInstruction\n if (typeof systemInstruction === 'string') {\n return systemInstruction\n }\n if (systemInstruction && typeof systemInstruction === 'object' && 'text' in systemInstruction) {\n return systemInstruction.text\n }\n if (\n systemInstruction &&\n typeof systemInstruction === 'object' &&\n 'parts' in systemInstruction &&\n Array.isArray(systemInstruction.parts)\n ) {\n for (const part of systemInstruction.parts) {\n if (part && typeof part === 'object' && 'text' in part && typeof part.text === 'string') {\n return part.text\n }\n }\n }\n if (Array.isArray(systemInstruction)) {\n for (const part of systemInstruction) {\n if (typeof part === 'string') {\n return part\n }\n if (part && typeof part === 'object' && 'text' in part && typeof part.text === 'string') {\n return part.text\n }\n }\n }\n return null\n }\n\n private formatInputForPostHog(params: GenerateContentParameters): FormattedMessage[] {\n const sanitized = sanitizeGemini(params.contents)\n const messages = this.formatInput(sanitized)\n\n const systemInstruction = this.extractSystemInstruction(params)\n\n if (systemInstruction) {\n const hasSystemMessage = messages.some((msg: FormattedMessage) => msg.role === 'system')\n\n if (!hasSystemMessage) {\n return [{ role: 'system', content: systemInstruction }, ...messages]\n }\n }\n\n return messages\n }\n}\n\nexport default PostHogGoogleGenAI\nexport { PostHogGoogleGenAI as Gemini }\n"],"names":["isString","value","isObject","Array","isArray","STRING_FORMAT","getModelParams","params","modelParams","paramKeys","key","undefined","formatResponseGemini","response","output","candidates","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","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","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","promptTokenCount","candidatesTokenCount","thoughtsTokenCount","cachedContentTokenCount","status","generateContentStream","accumulatedContent","stream","chunk","lastTextItem","i","funcCall","formatInput","contents","String","extractSystemInstruction","systemInstruction","sanitized","messages","hasSystemMessage","some","msg"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.mjs","sources":["../../src/typeGuards.ts","../../src/utils.ts","../../src/sanitization.ts","../../src/gemini/index.ts"],"sourcesContent":["// 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 { 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 { ResponseCreateParamsWithTools } from 'openai/lib/ResponsesParser'\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'\nimport { isString } from './typeGuards'\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\n/**\n * Safely converts content to a string, preserving structure for objects/arrays.\n * - If content is already a string, returns it as-is\n * - If content is an object or array, stringifies it with JSON.stringify to preserve structure\n * - Otherwise, converts to string with String()\n *\n * This prevents the \"[object Object]\" bug when objects are naively converted to strings.\n *\n * @param content - The content to convert to a string\n * @returns A string representation that preserves structure for complex types\n */\nexport function toContentString(content: unknown): string {\n if (typeof content === 'string') {\n return content\n }\n if (content !== undefined && content !== null && typeof content === 'object') {\n try {\n return JSON.stringify(content)\n } catch {\n // Fallback for circular refs, BigInt, or objects with throwing toJSON\n return String(content)\n }\n }\n return String(content)\n}\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 | ((\n | ChatCompletionCreateParamsBase\n | MessageCreateParams\n | ResponseCreateParams\n | ResponseCreateParamsWithTools\n | EmbeddingCreateParams\n ) &\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: (\n | ChatCompletionCreateParamsBase\n | MessageCreateParams\n | ResponseCreateParams\n | ResponseCreateParamsWithTools\n | EmbeddingCreateParams\n ) &\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\nexport function formatOpenAIResponsesInput(input: unknown, instructions?: string | null): FormattedMessage[] {\n const messages: FormattedMessage[] = []\n\n if (instructions) {\n messages.push({\n role: 'system',\n content: instructions,\n })\n }\n\n if (Array.isArray(input)) {\n for (const item of input) {\n if (typeof item === 'string') {\n messages.push({ role: 'user', content: item })\n } else if (item && typeof item === 'object') {\n const obj = item as Record<string, unknown>\n const role = isString(obj.role) ? obj.role : 'user'\n\n // Handle content properly - preserve structure for objects/arrays\n const content = obj.content ?? obj.text ?? item\n messages.push({ role, content: toContentString(content) })\n } else {\n messages.push({ role: 'user', content: toContentString(item) })\n }\n }\n } else if (typeof input === 'string') {\n messages.push({ role: 'user', content: input })\n } else if (input) {\n messages.push({ role: 'user', content: toContentString(input) })\n }\n\n return messages\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 toContentString,\n} from '../utils'\nimport { sanitizeGemini } from '../sanitization'\nimport type { TokenUsage, FormattedContent, FormattedContentItem, FormattedMessage } from '../types'\nimport { isString } from '../typeGuards'\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),\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),\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),\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),\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: isString(obj.role) ? obj.role : 'user', content: obj.text }\n }\n\n if ('content' in obj && obj.content) {\n return { role: isString(obj.role) ? obj.role : 'user', content: obj.content }\n }\n\n if ('parts' in obj && Array.isArray(obj.parts)) {\n return {\n role: isString(obj.role) ? obj.role : '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: toContentString(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: toContentString(contents) }]\n }\n\n private extractSystemInstruction(params: GenerateContentParameters): string | null {\n if (!params || typeof params !== 'object' || !params.config) {\n return null\n }\n const config = params.config as any\n if (!('systemInstruction' in config)) {\n return null\n }\n const systemInstruction = config.systemInstruction\n if (typeof systemInstruction === 'string') {\n return systemInstruction\n }\n if (systemInstruction && typeof systemInstruction === 'object' && 'text' in systemInstruction) {\n return systemInstruction.text\n }\n if (\n systemInstruction &&\n typeof systemInstruction === 'object' &&\n 'parts' in systemInstruction &&\n Array.isArray(systemInstruction.parts)\n ) {\n for (const part of systemInstruction.parts) {\n if (part && typeof part === 'object' && 'text' in part && typeof part.text === 'string') {\n return part.text\n }\n }\n }\n if (Array.isArray(systemInstruction)) {\n for (const part of systemInstruction) {\n if (typeof part === 'string') {\n return part\n }\n if (part && typeof part === 'object' && 'text' in part && typeof part.text === 'string') {\n return part.text\n }\n }\n }\n return null\n }\n\n private formatInputForPostHog(params: GenerateContentParameters): FormattedMessage[] {\n const sanitized = sanitizeGemini(params.contents)\n const messages = this.formatInput(sanitized)\n\n const systemInstruction = this.extractSystemInstruction(params)\n\n if (systemInstruction) {\n const hasSystemMessage = messages.some((msg: FormattedMessage) => msg.role === 'system')\n\n if (!hasSystemMessage) {\n return [{ role: 'system', content: systemInstruction }, ...messages]\n }\n }\n\n return messages\n }\n}\n\nexport default PostHogGoogleGenAI\nexport { PostHogGoogleGenAI as Gemini }\n"],"names":["isString","value","isObject","Array","isArray","STRING_FORMAT","toContentString","content","undefined","JSON","stringify","String","getModelParams","params","modelParams","paramKeys","key","formatResponseGemini","response","output","candidates","candidate","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","parse","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","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","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","promptTokenCount","candidatesTokenCount","thoughtsTokenCount","cachedContentTokenCount","status","generateContentStream","accumulatedContent","stream","chunk","lastTextItem","i","funcCall","formatInput","contents","extractSystemInstruction","systemInstruction","sanitized","messages","hasSystemMessage","some","msg"],"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.
|
|
29
|
+
var version = "6.5.0";
|
|
30
30
|
|
|
31
31
|
// Type guards for safer type checking
|
|
32
32
|
const isString = value => {
|
|
@@ -39,6 +39,31 @@ const isObject = value => {
|
|
|
39
39
|
// limit large outputs by truncating to 200kb (approx 200k bytes)
|
|
40
40
|
const MAX_OUTPUT_SIZE = 200000;
|
|
41
41
|
const STRING_FORMAT = 'utf8';
|
|
42
|
+
/**
|
|
43
|
+
* Safely converts content to a string, preserving structure for objects/arrays.
|
|
44
|
+
* - If content is already a string, returns it as-is
|
|
45
|
+
* - If content is an object or array, stringifies it with JSON.stringify to preserve structure
|
|
46
|
+
* - Otherwise, converts to string with String()
|
|
47
|
+
*
|
|
48
|
+
* This prevents the "[object Object]" bug when objects are naively converted to strings.
|
|
49
|
+
*
|
|
50
|
+
* @param content - The content to convert to a string
|
|
51
|
+
* @returns A string representation that preserves structure for complex types
|
|
52
|
+
*/
|
|
53
|
+
function toContentString(content) {
|
|
54
|
+
if (typeof content === 'string') {
|
|
55
|
+
return content;
|
|
56
|
+
}
|
|
57
|
+
if (content !== undefined && content !== null && typeof content === 'object') {
|
|
58
|
+
try {
|
|
59
|
+
return JSON.stringify(content);
|
|
60
|
+
} catch {
|
|
61
|
+
// Fallback for circular refs, BigInt, or objects with throwing toJSON
|
|
62
|
+
return String(content);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
return String(content);
|
|
66
|
+
}
|
|
42
67
|
const getModelParams = params => {
|
|
43
68
|
if (!params) {
|
|
44
69
|
return {};
|
|
@@ -466,15 +491,16 @@ function formatOpenAIResponsesInput(input, instructions) {
|
|
|
466
491
|
} else if (item && typeof item === 'object') {
|
|
467
492
|
const obj = item;
|
|
468
493
|
const role = isString(obj.role) ? obj.role : 'user';
|
|
469
|
-
|
|
494
|
+
// Handle content properly - preserve structure for objects/arrays
|
|
495
|
+
const content = obj.content ?? obj.text ?? item;
|
|
470
496
|
messages.push({
|
|
471
497
|
role,
|
|
472
|
-
content:
|
|
498
|
+
content: toContentString(content)
|
|
473
499
|
});
|
|
474
500
|
} else {
|
|
475
501
|
messages.push({
|
|
476
502
|
role: 'user',
|
|
477
|
-
content:
|
|
503
|
+
content: toContentString(item)
|
|
478
504
|
});
|
|
479
505
|
}
|
|
480
506
|
}
|
|
@@ -486,7 +512,7 @@ function formatOpenAIResponsesInput(input, instructions) {
|
|
|
486
512
|
} else if (input) {
|
|
487
513
|
messages.push({
|
|
488
514
|
role: 'user',
|
|
489
|
-
content:
|
|
515
|
+
content: toContentString(input)
|
|
490
516
|
});
|
|
491
517
|
}
|
|
492
518
|
return messages;
|
|
@@ -1626,7 +1652,7 @@ const mapVercelPrompt = messages => {
|
|
|
1626
1652
|
if (message.role === 'system') {
|
|
1627
1653
|
content = [{
|
|
1628
1654
|
type: 'text',
|
|
1629
|
-
text: truncate(
|
|
1655
|
+
text: truncate(toContentString(message.content))
|
|
1630
1656
|
}];
|
|
1631
1657
|
} else {
|
|
1632
1658
|
// Handle other roles which have array content
|
|
@@ -1684,7 +1710,7 @@ const mapVercelPrompt = messages => {
|
|
|
1684
1710
|
// Fallback for non-array content
|
|
1685
1711
|
content = [{
|
|
1686
1712
|
type: 'text',
|
|
1687
|
-
text: truncate(
|
|
1713
|
+
text: truncate(toContentString(message.content))
|
|
1688
1714
|
}];
|
|
1689
1715
|
}
|
|
1690
1716
|
}
|
|
@@ -1812,7 +1838,11 @@ const createInstrumentationMiddleware = (phClient, model, options) => {
|
|
|
1812
1838
|
const startTime = Date.now();
|
|
1813
1839
|
const mergedParams = {
|
|
1814
1840
|
...options,
|
|
1815
|
-
...mapVercelParams(params)
|
|
1841
|
+
...mapVercelParams(params),
|
|
1842
|
+
posthogProperties: {
|
|
1843
|
+
...options.posthogProperties,
|
|
1844
|
+
$ai_framework: 'vercel'
|
|
1845
|
+
}
|
|
1816
1846
|
};
|
|
1817
1847
|
const availableTools = extractAvailableToolCalls('vercel', params);
|
|
1818
1848
|
try {
|
|
@@ -1888,7 +1918,11 @@ const createInstrumentationMiddleware = (phClient, model, options) => {
|
|
|
1888
1918
|
let usage = {};
|
|
1889
1919
|
const mergedParams = {
|
|
1890
1920
|
...options,
|
|
1891
|
-
...mapVercelParams(params)
|
|
1921
|
+
...mapVercelParams(params),
|
|
1922
|
+
posthogProperties: {
|
|
1923
|
+
...options.posthogProperties,
|
|
1924
|
+
$ai_framework: 'vercel'
|
|
1925
|
+
}
|
|
1892
1926
|
};
|
|
1893
1927
|
const modelId = options.posthogModelOverride ?? model.modelId;
|
|
1894
1928
|
const provider = options.posthogProviderOverride ?? extractProvider(model);
|
|
@@ -2499,7 +2533,7 @@ class WrappedModels {
|
|
|
2499
2533
|
}
|
|
2500
2534
|
return {
|
|
2501
2535
|
role: 'user',
|
|
2502
|
-
content:
|
|
2536
|
+
content: toContentString(item)
|
|
2503
2537
|
};
|
|
2504
2538
|
});
|
|
2505
2539
|
}
|
|
@@ -2520,7 +2554,7 @@ class WrappedModels {
|
|
|
2520
2554
|
}
|
|
2521
2555
|
return [{
|
|
2522
2556
|
role: 'user',
|
|
2523
|
-
content:
|
|
2557
|
+
content: toContentString(contents)
|
|
2524
2558
|
}];
|
|
2525
2559
|
}
|
|
2526
2560
|
extractSystemInstruction(params) {
|
|
@@ -3322,7 +3356,8 @@ class LangChainCallbackHandler extends BaseCallbackHandler {
|
|
|
3322
3356
|
$ai_input_state: withPrivacyMode(this.client, this.privacyMode, run.input),
|
|
3323
3357
|
$ai_latency: latency,
|
|
3324
3358
|
$ai_span_name: run.name,
|
|
3325
|
-
$ai_span_id: runId
|
|
3359
|
+
$ai_span_id: runId,
|
|
3360
|
+
$ai_framework: 'langchain'
|
|
3326
3361
|
};
|
|
3327
3362
|
if (parentRunId) {
|
|
3328
3363
|
eventProperties['$ai_parent_id'] = parentRunId;
|
|
@@ -3370,7 +3405,8 @@ class LangChainCallbackHandler extends BaseCallbackHandler {
|
|
|
3370
3405
|
$ai_input: withPrivacyMode(this.client, this.privacyMode, run.input),
|
|
3371
3406
|
$ai_http_status: 200,
|
|
3372
3407
|
$ai_latency: latency,
|
|
3373
|
-
$ai_base_url: run.baseUrl
|
|
3408
|
+
$ai_base_url: run.baseUrl,
|
|
3409
|
+
$ai_framework: 'langchain'
|
|
3374
3410
|
};
|
|
3375
3411
|
if (run.tools) {
|
|
3376
3412
|
eventProperties['$ai_tools'] = run.tools;
|
|
@@ -3512,7 +3548,7 @@ class LangChainCallbackHandler extends BaseCallbackHandler {
|
|
|
3512
3548
|
default:
|
|
3513
3549
|
messageDict = {
|
|
3514
3550
|
role: messageType,
|
|
3515
|
-
content:
|
|
3551
|
+
content: toContentString(message.content)
|
|
3516
3552
|
};
|
|
3517
3553
|
break;
|
|
3518
3554
|
}
|