@posthog/ai 7.3.2 → 7.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.
@@ -5,12 +5,13 @@ Object.defineProperty(exports, '__esModule', { value: true });
5
5
  var AnthropicOriginal = require('@anthropic-ai/sdk');
6
6
  var buffer = require('buffer');
7
7
  var uuid = require('uuid');
8
+ var core = require('@posthog/core');
8
9
 
9
10
  function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
10
11
 
11
12
  var AnthropicOriginal__default = /*#__PURE__*/_interopDefault(AnthropicOriginal);
12
13
 
13
- var version = "7.3.2";
14
+ var version = "7.4.0";
14
15
 
15
16
  // Type guards for safer type checking
16
17
 
@@ -143,6 +144,33 @@ function addDefaults(params) {
143
144
  traceId: params.traceId ?? uuid.v4()
144
145
  };
145
146
  }
147
+ const sendEventWithErrorToPosthog = async ({
148
+ client,
149
+ traceId,
150
+ error,
151
+ ...args
152
+ }) => {
153
+ const httpStatus = error && typeof error === 'object' && 'status' in error ? error.status ?? 500 : 500;
154
+ const properties = {
155
+ client,
156
+ traceId,
157
+ httpStatus,
158
+ error: JSON.stringify(error),
159
+ ...args
160
+ };
161
+ const enrichedError = error;
162
+ if (client.options?.enableExceptionAutocapture) {
163
+ // assign a uuid that can be used to link the trace and exception events
164
+ const exceptionId = core.uuidv7();
165
+ client.captureException(error, undefined, {
166
+ $ai_trace_id: traceId
167
+ }, exceptionId);
168
+ enrichedError.__posthog_previously_captured_error = true;
169
+ properties.exceptionId = exceptionId;
170
+ }
171
+ await sendEventToPosthog(properties);
172
+ return enrichedError;
173
+ };
146
174
  const sendEventToPosthog = async ({
147
175
  client,
148
176
  eventType = AIEvent.Generation,
@@ -157,8 +185,8 @@ const sendEventToPosthog = async ({
157
185
  params,
158
186
  httpStatus = 200,
159
187
  usage = {},
160
- isError = false,
161
188
  error,
189
+ exceptionId,
162
190
  tools,
163
191
  captureImmediate = false
164
192
  }) => {
@@ -170,10 +198,11 @@ const sendEventToPosthog = async ({
170
198
  const safeOutput = sanitizeValues(output);
171
199
  const safeError = sanitizeValues(error);
172
200
  let errorData = {};
173
- if (isError) {
201
+ if (error) {
174
202
  errorData = {
175
203
  $ai_is_error: true,
176
- $ai_error: safeError
204
+ $ai_error: safeError,
205
+ $exception_event_id: exceptionId
177
206
  };
178
207
  }
179
208
  let costOverrideData = {};
@@ -239,6 +268,7 @@ const sendEventToPosthog = async ({
239
268
  } else {
240
269
  client.capture(event);
241
270
  }
271
+ return Promise.resolve();
242
272
  };
243
273
 
244
274
  const REDACTED_IMAGE_PLACEHOLDER = '[base64 image redacted]';
@@ -457,8 +487,7 @@ class WrappedMessages extends AnthropicOriginal__default.default.Messages {
457
487
  tools: availableTools
458
488
  });
459
489
  } catch (error) {
460
- // error handling
461
- await sendEventToPosthog({
490
+ const enrichedError = await sendEventWithErrorToPosthog({
462
491
  client: this.phClient,
463
492
  ...posthogParams,
464
493
  model: anthropicParams.model,
@@ -468,14 +497,13 @@ class WrappedMessages extends AnthropicOriginal__default.default.Messages {
468
497
  latency: 0,
469
498
  baseURL: this.baseURL,
470
499
  params: body,
471
- httpStatus: error?.status ? error.status : 500,
472
500
  usage: {
473
501
  inputTokens: 0,
474
502
  outputTokens: 0
475
503
  },
476
- isError: true,
477
- error: JSON.stringify(error)
504
+ error: error
478
505
  });
506
+ throw enrichedError;
479
507
  }
480
508
  })();
481
509
 
@@ -527,7 +555,6 @@ class WrappedMessages extends AnthropicOriginal__default.default.Messages {
527
555
  inputTokens: 0,
528
556
  outputTokens: 0
529
557
  },
530
- isError: true,
531
558
  error: JSON.stringify(error)
532
559
  });
533
560
  throw error;
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs","sources":["../../src/typeGuards.ts","../../src/utils.ts","../../src/sanitization.ts","../../src/anthropic/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 TranscriptionCreateParams = OpenAIOrignal.Audio.Transcriptions.TranscriptionCreateParams\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 | TranscriptionCreateParams\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 'language',\n 'response_format',\n 'timestamp_granularities',\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\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 // Handle audio output (gpt-4o-audio-preview)\n if (choice.message.audio) {\n content.push({\n type: 'audio',\n ...choice.message.audio,\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 } else if (part.inlineData) {\n // Handle audio/media inline data\n const mimeType = part.inlineData.mimeType || 'audio/pcm'\n let data = part.inlineData.data\n\n // Handle binary data (Buffer/Uint8Array -> base64)\n if (data instanceof Uint8Array || Buffer.isBuffer(data)) {\n data = Buffer.from(data).toString('base64')\n }\n\n content.push({\n type: 'audio',\n mime_type: mimeType,\n data: data,\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 * Calculate web search count from raw API response.\n *\n * Uses a two-tier detection strategy:\n * Priority 1 (Exact Count): Count actual web search calls when available\n * Priority 2 (Binary Detection): Return 1 if web search indicators are present, 0 otherwise\n *\n * @param result - Raw API response from any provider (OpenAI, Perplexity, OpenRouter, Gemini, etc.)\n * @returns Number of web searches performed (exact count or binary 1/0)\n */\nexport function calculateWebSearchCount(result: unknown): number {\n if (!result || typeof result !== 'object') {\n return 0\n }\n\n // Priority 1: Exact Count\n // Check for OpenAI Responses API web_search_call items\n if ('output' in result && Array.isArray(result.output)) {\n let count = 0\n\n for (const item of result.output) {\n if (typeof item === 'object' && item !== null && 'type' in item && item.type === 'web_search_call') {\n count++\n }\n }\n\n if (count > 0) {\n return count\n }\n }\n\n // Priority 2: Binary Detection (1 or 0)\n\n // Check for citations at root level (Perplexity)\n if ('citations' in result && Array.isArray(result.citations) && result.citations.length > 0) {\n return 1\n }\n\n // Check for search_results at root level (Perplexity via OpenRouter)\n if ('search_results' in result && Array.isArray(result.search_results) && result.search_results.length > 0) {\n return 1\n }\n\n // Check for usage.search_context_size (Perplexity via OpenRouter)\n if ('usage' in result && typeof result.usage === 'object' && result.usage !== null) {\n if ('search_context_size' in result.usage && result.usage.search_context_size) {\n return 1\n }\n }\n\n // Check for annotations with url_citation in choices[].message or choices[].delta (OpenAI/Perplexity)\n if ('choices' in result && Array.isArray(result.choices)) {\n for (const choice of result.choices) {\n if (typeof choice === 'object' && choice !== null) {\n // Check both message (non-streaming) and delta (streaming) for annotations\n const content = ('message' in choice ? choice.message : null) || ('delta' in choice ? choice.delta : null)\n\n if (typeof content === 'object' && content !== null && 'annotations' in content) {\n const annotations = content.annotations\n\n if (Array.isArray(annotations)) {\n const hasUrlCitation = annotations.some((ann: unknown) => {\n return typeof ann === 'object' && ann !== null && 'type' in ann && ann.type === 'url_citation'\n })\n\n if (hasUrlCitation) {\n return 1\n }\n }\n }\n }\n }\n }\n\n // Check for annotations in output[].content[] (OpenAI Responses API)\n if ('output' in result && Array.isArray(result.output)) {\n for (const item of result.output) {\n if (typeof item === 'object' && item !== null && 'content' in item) {\n const content = item.content\n\n if (Array.isArray(content)) {\n for (const contentItem of content) {\n if (typeof contentItem === 'object' && contentItem !== null && 'annotations' in contentItem) {\n const annotations = contentItem.annotations\n\n if (Array.isArray(annotations)) {\n const hasUrlCitation = annotations.some((ann: unknown) => {\n return typeof ann === 'object' && ann !== null && 'type' in ann && ann.type === 'url_citation'\n })\n\n if (hasUrlCitation) {\n return 1\n }\n }\n }\n }\n }\n }\n }\n }\n\n // Check for grounding_metadata (Gemini)\n if ('candidates' in result && Array.isArray(result.candidates)) {\n for (const candidate of result.candidates) {\n if (\n typeof candidate === 'object' &&\n candidate !== null &&\n 'grounding_metadata' in candidate &&\n candidate.grounding_metadata\n ) {\n return 1\n }\n }\n }\n\n return 0\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 | TranscriptionCreateParams\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 ...(usage.webSearchCount ? { $ai_web_search_count: usage.webSearchCount } : {}),\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// Multimodal Feature Toggle\n// ============================================\n\nconst isMultimodalEnabled = (): boolean => {\n const val = process.env._INTERNAL_LLMA_MULTIMODAL || ''\n return val.toLowerCase() === 'true' || val === '1' || val.toLowerCase() === 'yes'\n}\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 (isMultimodalEnabled()) return str\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 // Handle audio format\n if (item.type === 'audio' && 'data' in item) {\n if (isMultimodalEnabled()) return item\n return { ...item, data: REDACTED_IMAGE_PLACEHOLDER }\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 (isMultimodalEnabled()) return item\n if (!isObject(item)) return item\n\n // Handle Anthropic's image and document formats (same structure, different type field)\n if (\n (item.type === 'image' || item.type === 'document') &&\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 (isMultimodalEnabled()) return part\n if (!isObject(part)) return part\n\n // Handle Gemini's inline data format (images, audio, PDFs all use inlineData)\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 if (isMultimodalEnabled()) return item\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 AnthropicOriginal, { APIPromise } from '@anthropic-ai/sdk'\nimport { PostHog } from 'posthog-node'\nimport {\n formatResponseAnthropic,\n mergeSystemPrompt,\n MonitoringParams,\n sendEventToPosthog,\n extractAvailableToolCalls,\n extractPosthogParams,\n} from '../utils'\nimport type { FormattedContentItem, FormattedTextContent, FormattedFunctionCall, FormattedMessage } from '../types'\n\ntype MessageCreateParamsNonStreaming = AnthropicOriginal.Messages.MessageCreateParamsNonStreaming\ntype MessageCreateParamsStreaming = AnthropicOriginal.Messages.MessageCreateParamsStreaming\ntype MessageCreateParams = AnthropicOriginal.Messages.MessageCreateParams\ntype Message = AnthropicOriginal.Messages.Message\ntype RawMessageStreamEvent = AnthropicOriginal.Messages.RawMessageStreamEvent\ntype MessageCreateParamsBase = AnthropicOriginal.Messages.MessageCreateParams\ntype RequestOptions = AnthropicOriginal.RequestOptions\nimport type { Stream } from '@anthropic-ai/sdk/streaming'\nimport { sanitizeAnthropic } from '../sanitization'\n\ninterface ToolInProgress {\n block: FormattedFunctionCall\n inputString: string\n}\n\ninterface MonitoringAnthropicConfig {\n apiKey: string\n posthog: PostHog\n baseURL?: string\n}\n\nexport class PostHogAnthropic extends AnthropicOriginal {\n private readonly phClient: PostHog\n public messages: WrappedMessages\n\n constructor(config: MonitoringAnthropicConfig) {\n const { posthog, ...anthropicConfig } = config\n super(anthropicConfig)\n this.phClient = posthog\n this.messages = new WrappedMessages(this, this.phClient)\n }\n}\n\nexport class WrappedMessages extends AnthropicOriginal.Messages {\n private readonly phClient: PostHog\n private readonly baseURL: string\n\n constructor(parentClient: PostHogAnthropic, phClient: PostHog) {\n super(parentClient)\n this.phClient = phClient\n this.baseURL = parentClient.baseURL\n }\n\n public create(body: MessageCreateParamsNonStreaming, options?: RequestOptions): APIPromise<Message>\n public create(\n body: MessageCreateParamsStreaming & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<Stream<RawMessageStreamEvent>>\n public create(\n body: MessageCreateParamsBase & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<Stream<RawMessageStreamEvent> | Message>\n public create(\n body: MessageCreateParams & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<Message> | APIPromise<Stream<RawMessageStreamEvent>> {\n const { providerParams: anthropicParams, posthogParams } = extractPosthogParams(body)\n const startTime = Date.now()\n\n const parentPromise = super.create(anthropicParams, options)\n\n if (anthropicParams.stream) {\n return parentPromise.then((value) => {\n let accumulatedContent = ''\n const contentBlocks: FormattedContentItem[] = []\n const toolsInProgress: Map<string, ToolInProgress> = new Map()\n let currentTextBlock: FormattedTextContent | null = null\n\n const usage: {\n inputTokens: number\n outputTokens: number\n cacheCreationInputTokens?: number\n cacheReadInputTokens?: number\n webSearchCount?: number\n } = {\n inputTokens: 0,\n outputTokens: 0,\n cacheCreationInputTokens: 0,\n cacheReadInputTokens: 0,\n webSearchCount: 0,\n }\n if ('tee' in value) {\n const [stream1, stream2] = value.tee()\n ;(async () => {\n try {\n for await (const chunk of stream1) {\n // Handle content block start events\n if (chunk.type === 'content_block_start') {\n if (chunk.content_block?.type === 'text') {\n currentTextBlock = {\n type: 'text',\n text: '',\n }\n\n contentBlocks.push(currentTextBlock)\n } else if (chunk.content_block?.type === 'tool_use') {\n const toolBlock: FormattedFunctionCall = {\n type: 'function',\n id: chunk.content_block.id,\n function: {\n name: chunk.content_block.name,\n arguments: {},\n },\n }\n\n contentBlocks.push(toolBlock)\n\n toolsInProgress.set(chunk.content_block.id, {\n block: toolBlock,\n inputString: '',\n })\n\n currentTextBlock = null\n }\n }\n\n // Handle text delta events\n if ('delta' in chunk) {\n if ('text' in chunk.delta) {\n const delta = chunk.delta.text\n\n accumulatedContent += delta\n\n if (currentTextBlock) {\n currentTextBlock.text += delta\n }\n }\n }\n\n // Handle tool input delta events\n if (chunk.type === 'content_block_delta' && chunk.delta?.type === 'input_json_delta') {\n const block = chunk.index !== undefined ? contentBlocks[chunk.index] : undefined\n const toolId = block?.type === 'function' ? block.id : undefined\n\n if (toolId && toolsInProgress.has(toolId)) {\n const tool = toolsInProgress.get(toolId)\n if (tool) {\n tool.inputString += chunk.delta.partial_json || ''\n }\n }\n }\n\n // Handle content block stop events\n if (chunk.type === 'content_block_stop') {\n currentTextBlock = null\n\n // Parse accumulated tool input\n if (chunk.index !== undefined) {\n const block = contentBlocks[chunk.index]\n\n if (block?.type === 'function' && block.id && toolsInProgress.has(block.id)) {\n const tool = toolsInProgress.get(block.id)\n if (tool) {\n try {\n block.function.arguments = JSON.parse(tool.inputString)\n } catch (e) {\n // Keep empty object if parsing fails\n console.error('Error parsing tool input:', e)\n }\n }\n toolsInProgress.delete(block.id)\n }\n }\n }\n\n if (chunk.type == 'message_start') {\n usage.inputTokens = chunk.message.usage.input_tokens ?? 0\n usage.cacheCreationInputTokens = chunk.message.usage.cache_creation_input_tokens ?? 0\n usage.cacheReadInputTokens = chunk.message.usage.cache_read_input_tokens ?? 0\n usage.webSearchCount = chunk.message.usage.server_tool_use?.web_search_requests ?? 0\n }\n if ('usage' in chunk) {\n usage.outputTokens = chunk.usage.output_tokens ?? 0\n // Update web search count if present in delta\n if (chunk.usage.server_tool_use?.web_search_requests !== undefined) {\n usage.webSearchCount = chunk.usage.server_tool_use.web_search_requests\n }\n }\n }\n\n const latency = (Date.now() - startTime) / 1000\n\n const availableTools = extractAvailableToolCalls('anthropic', anthropicParams)\n\n // Format output to match non-streaming version\n const formattedOutput: FormattedMessage[] =\n contentBlocks.length > 0\n ? [\n {\n role: 'assistant',\n content: contentBlocks,\n },\n ]\n : [\n {\n role: 'assistant',\n content: [{ type: 'text', text: accumulatedContent }],\n },\n ]\n\n await sendEventToPosthog({\n client: this.phClient,\n ...posthogParams,\n model: anthropicParams.model,\n provider: 'anthropic',\n input: sanitizeAnthropic(mergeSystemPrompt(anthropicParams, 'anthropic')),\n output: formattedOutput,\n latency,\n baseURL: this.baseURL,\n params: body,\n httpStatus: 200,\n usage,\n tools: availableTools,\n })\n } catch (error: any) {\n // error handling\n await sendEventToPosthog({\n client: this.phClient,\n ...posthogParams,\n model: anthropicParams.model,\n provider: 'anthropic',\n input: sanitizeAnthropic(mergeSystemPrompt(anthropicParams, 'anthropic')),\n output: [],\n latency: 0,\n baseURL: this.baseURL,\n params: body,\n httpStatus: error?.status ? error.status : 500,\n usage: {\n inputTokens: 0,\n outputTokens: 0,\n },\n isError: true,\n error: JSON.stringify(error),\n })\n }\n })()\n\n // Return the other stream to the user\n return stream2\n }\n return value\n }) as APIPromise<Stream<RawMessageStreamEvent>>\n } else {\n const wrappedPromise = parentPromise.then(\n async (result) => {\n if ('content' in result) {\n const latency = (Date.now() - startTime) / 1000\n\n const availableTools = extractAvailableToolCalls('anthropic', anthropicParams)\n\n await sendEventToPosthog({\n client: this.phClient,\n ...posthogParams,\n model: anthropicParams.model,\n provider: 'anthropic',\n input: sanitizeAnthropic(mergeSystemPrompt(anthropicParams, 'anthropic')),\n output: formatResponseAnthropic(result),\n latency,\n baseURL: this.baseURL,\n params: body,\n httpStatus: 200,\n usage: {\n inputTokens: result.usage.input_tokens ?? 0,\n outputTokens: result.usage.output_tokens ?? 0,\n cacheCreationInputTokens: result.usage.cache_creation_input_tokens ?? 0,\n cacheReadInputTokens: result.usage.cache_read_input_tokens ?? 0,\n webSearchCount: result.usage.server_tool_use?.web_search_requests ?? 0,\n },\n tools: availableTools,\n })\n }\n return result\n },\n async (error: any) => {\n await sendEventToPosthog({\n client: this.phClient,\n ...posthogParams,\n model: anthropicParams.model,\n provider: 'anthropic',\n input: sanitizeAnthropic(mergeSystemPrompt(anthropicParams, 'anthropic')),\n output: [],\n latency: 0,\n baseURL: this.baseURL,\n params: body,\n httpStatus: error?.status ? error.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 ) as APIPromise<Message>\n\n return wrappedPromise\n }\n }\n}\n\nexport default PostHogAnthropic\n\nexport { PostHogAnthropic as Anthropic }\n"],"names":["isObject","value","Array","isArray","STRING_FORMAT","getModelParams","params","modelParams","paramKeys","key","undefined","formatResponseAnthropic","response","output","content","choice","type","text","push","name","id","function","arguments","input","length","role","mergeSystemPrompt","provider","messages","system","systemMessage","withPrivacyMode","client","privacyMode","privacy_mode","extractAvailableToolCalls","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","webSearchCount","$ai_web_search_count","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","isMultimodalEnabled","val","process","env","_INTERNAL_LLMA_MULTIMODAL","toLowerCase","processMessages","transformContent","processContent","processMessage","msg","sanitizeAnthropicImage","item","source","data","sanitizeAnthropic","PostHogAnthropic","AnthropicOriginal","constructor","config","posthog","anthropicConfig","phClient","WrappedMessages","Messages","parentClient","create","options","anthropicParams","startTime","Date","now","parentPromise","stream","then","accumulatedContent","contentBlocks","toolsInProgress","Map","currentTextBlock","stream1","stream2","tee","chunk","content_block","toolBlock","set","block","inputString","delta","index","toolId","has","tool","get","partial_json","e","delete","message","input_tokens","cache_creation_input_tokens","cache_read_input_tokens","server_tool_use","web_search_requests","output_tokens","availableTools","formattedOutput","status","wrappedPromise","result"],"mappings":";;;;;;;;;;;;;;AAAA;;AAMO,MAAMA,QAAQ,GAAIC,KAAc,IAAuC;AAC5E,EAAA,OAAOA,KAAK,KAAK,IAAI,IAAI,OAAOA,KAAK,KAAK,QAAQ,IAAI,CAACC,KAAK,CAACC,OAAO,CAACF,KAAK,CAAC;AAC7E,CAAC;;ACaD,MAAMG,aAAa,GAAG,MAAM;AAmDrB,MAAMC,cAAc,GACzBC,MAUQ,IACgB;EACxB,IAAI,CAACA,MAAM,EAAE;AACX,IAAA,OAAO,EAAE;AACX,EAAA;EACA,MAAMC,WAAgC,GAAG,EAAE;AAC3C,EAAA,MAAMC,SAAS,GAAG,CAChB,aAAa,EACb,YAAY,EACZ,uBAAuB,EACvB,OAAO,EACP,mBAAmB,EACnB,kBAAkB,EAClB,GAAG,EACH,MAAM,EACN,QAAQ,EACR,WAAW,EACX,UAAU,EACV,iBAAiB,EACjB,yBAAyB,CACjB;AAEV,EAAA,KAAK,MAAMC,GAAG,IAAID,SAAS,EAAE;IAC3B,IAAIC,GAAG,IAAIH,MAAM,IAAKA,MAAM,CAASG,GAAG,CAAC,KAAKC,SAAS,EAAE;AACvDH,MAAAA,WAAW,CAACE,GAAG,CAAC,GAAIH,MAAM,CAASG,GAAG,CAAC;AACzC,IAAA;AACF,EAAA;AACA,EAAA,OAAOF,WAAW;AACpB,CAAC;AAmBM,MAAMI,uBAAuB,GAAIC,QAAa,IAAyB;EAC5E,MAAMC,MAA0B,GAAG,EAAE;EACrC,MAAMC,OAAyB,GAAG,EAAE;EAEpC,KAAK,MAAMC,MAAM,IAAIH,QAAQ,CAACE,OAAO,IAAI,EAAE,EAAE;IAC3C,IAAIC,MAAM,EAAEC,IAAI,KAAK,MAAM,IAAID,MAAM,EAAEE,IAAI,EAAE;MAC3CH,OAAO,CAACI,IAAI,CAAC;AAAEF,QAAAA,IAAI,EAAE,MAAM;QAAEC,IAAI,EAAEF,MAAM,CAACE;AAAK,OAAC,CAAC;AACnD,IAAA,CAAC,MAAM,IAAIF,MAAM,EAAEC,IAAI,KAAK,UAAU,IAAID,MAAM,EAAEI,IAAI,IAAIJ,MAAM,EAAEK,EAAE,EAAE;MACpEN,OAAO,CAACI,IAAI,CAAC;AACXF,QAAAA,IAAI,EAAE,UAAU;QAChBI,EAAE,EAAEL,MAAM,CAACK,EAAE;AACbC,QAAAA,QAAQ,EAAE;UACRF,IAAI,EAAEJ,MAAM,CAACI,IAAI;AACjBG,UAAAA,SAAS,EAAEP,MAAM,CAACQ,KAAK,IAAI;AAC7B;AACF,OAAC,CAAC;AACJ,IAAA;AACF,EAAA;AAEA,EAAA,IAAIT,OAAO,CAACU,MAAM,GAAG,CAAC,EAAE;IACtBX,MAAM,CAACK,IAAI,CAAC;AACVO,MAAAA,IAAI,EAAE,WAAW;AACjBX,MAAAA;AACF,KAAC,CAAC;AACJ,EAAA;AAEA,EAAA,OAAOD,MAAM;AACf,CAAC;AA8JM,MAAMa,iBAAiB,GAAGA,CAACpB,MAA8C,EAAEqB,QAAgB,KAAU;EAC7E;AAC3B,IAAA,MAAMC,QAAQ,GAAGtB,MAAM,CAACsB,QAAQ,IAAI,EAAE;AACtC,IAAA,IAAI,CAAEtB,MAAM,CAASuB,MAAM,EAAE;AAC3B,MAAA,OAAOD,QAAQ;AACjB,IAAA;AACA,IAAA,MAAME,aAAa,GAAIxB,MAAM,CAASuB,MAAM;AAC5C,IAAA,OAAO,CAAC;AAAEJ,MAAAA,IAAI,EAAE,QAAQ;AAAEX,MAAAA,OAAO,EAAEgB;KAAe,EAAE,GAAGF,QAAQ,CAAC;AAClE,EAAA;AAEF,CAAC;AAEM,MAAMG,eAAe,GAAGA,CAACC,MAAe,EAAEC,WAAoB,EAAEV,KAAU,KAAU;EACzF,OAAQS,MAAM,CAASE,YAAY,IAAID,WAAW,GAAG,IAAI,GAAGV,KAAK;AACnE,CAAC;;AAgKD;AACA;AACA;AACA;AACO,MAAMY,yBAAyB,GAAGA,CACvCR,QAAgB,EAChBrB,MAAW,KACsD;EACnC;IAC5B,IAAIA,MAAM,CAAC8B,KAAK,EAAE;MAChB,OAAO9B,MAAM,CAAC8B,KAAK;AACrB,IAAA;AAEA,IAAA,OAAO,IAAI;AACb,EAAA;AAqBF,CAAC;AAED,IAAYC,OAAO,0BAAPA,OAAO,EAAA;EAAPA,OAAO,CAAA,YAAA,CAAA,GAAA,gBAAA;EAAPA,OAAO,CAAA,WAAA,CAAA,GAAA,eAAA;AAAA,EAAA,OAAPA,OAAO;AAAA,CAAA,CAAA,EAAA,CAAA;AAiCnB,SAASC,cAAcA,CAACC,GAAQ,EAAO;AACrC,EAAA,IAAIA,GAAG,KAAK7B,SAAS,IAAI6B,GAAG,KAAK,IAAI,EAAE;AACrC,IAAA,OAAOA,GAAG;AACZ,EAAA;AACA,EAAA,MAAMC,QAAQ,GAAGC,IAAI,CAACC,KAAK,CAACD,IAAI,CAACE,SAAS,CAACJ,GAAG,CAAC,CAAC;AAChD,EAAA,IAAI,OAAOC,QAAQ,KAAK,QAAQ,EAAE;AAChC,IAAA,OAAOI,aAAM,CAACC,IAAI,CAACL,QAAQ,EAAEpC,aAAa,CAAC,CAAC0C,QAAQ,CAAC1C,aAAa,CAAC;EACrE,CAAC,MAAM,IAAIF,KAAK,CAACC,OAAO,CAACqC,QAAQ,CAAC,EAAE;AAClC,IAAA,OAAOA,QAAQ,CAACO,GAAG,CAACT,cAAc,CAAC;EACrC,CAAC,MAAM,IAAIE,QAAQ,IAAI,OAAOA,QAAQ,KAAK,QAAQ,EAAE;AACnD,IAAA,OAAOQ,MAAM,CAACC,WAAW,CAACD,MAAM,CAACE,OAAO,CAACV,QAAQ,CAAC,CAACO,GAAG,CAAC,CAAC,CAACI,CAAC,EAAEC,CAAC,CAAC,KAAK,CAACD,CAAC,EAAEb,cAAc,CAACc,CAAC,CAAC,CAAC,CAAC,CAAC;AAC7F,EAAA;AACA,EAAA,OAAOZ,QAAQ;AACjB;AAEA,MAAMa,kBAA0D,GAAG;AACjEC,EAAAA,iBAAiB,EAAE,YAAY;AAC/BC,EAAAA,cAAc,EAAE,SAAS;AACzBC,EAAAA,iBAAiB,EAAE,YAAY;AAC/BC,EAAAA,kBAAkB,EAAE,aAAa;AACjCC,EAAAA,aAAa,EAAE,QAAQ;AACvBC,EAAAA,oBAAoB,EAAE,eAAe;AACrCC,EAAAA,uBAAuB,EAAE,kBAAkB;AAC3CC,EAAAA,mBAAmB,EAAE,cAAc;AACnCC,EAAAA,uBAAuB,EAAE;AAC3B,CAAC;AAEM,SAASC,oBAAoBA,CAAIC,IAA0B,EAGhE;EACA,MAAMC,cAAuC,GAAG,EAAE;EAClD,MAAMC,aAAsC,GAAG,EAAE;AAEjD,EAAA,KAAK,MAAM,CAACzD,GAAG,EAAER,KAAK,CAAC,IAAI+C,MAAM,CAACE,OAAO,CAACc,IAAI,CAAC,EAAE;AAC/C,IAAA,IAAIX,kBAAkB,CAAC5C,GAAG,CAA2B,EAAE;AACrDyD,MAAAA,aAAa,CAACb,kBAAkB,CAAC5C,GAAG,CAA2B,CAAC,GAAGR,KAAK;IAC1E,CAAC,MAAM,IAAIQ,GAAG,CAAC0D,UAAU,CAAC,SAAS,CAAC,EAAE;AACpCC,MAAAA,OAAO,CAACC,IAAI,CAAC,CAAA,0BAAA,EAA6B5D,GAAG,EAAE,CAAC;AAClD,IAAA,CAAC,MAAM;AACLwD,MAAAA,cAAc,CAACxD,GAAG,CAAC,GAAGR,KAAK;AAC7B,IAAA;AACF,EAAA;EAEA,OAAO;AACLgE,IAAAA,cAAc,EAAEA,cAAmB;IACnCC,aAAa,EAAEI,WAAW,CAACJ,aAAa;GACzC;AACH;AAEA,SAASI,WAAWA,CAAChE,MAAiC,EAAyC;EAC7F,OAAO;AACL,IAAA,GAAGA,MAAM;AACT2B,IAAAA,WAAW,EAAE3B,MAAM,CAAC2B,WAAW,IAAI,KAAK;AACxCsC,IAAAA,OAAO,EAAEjE,MAAM,CAACiE,OAAO,IAAIC,OAAM;GAClC;AACH;AAEO,MAAMC,kBAAkB,GAAG,OAAO;EACvCzC,MAAM;EACN0C,SAAS,GAAGrC,OAAO,CAACsC,UAAU;EAC9BC,UAAU;EACVL,OAAO;EACPM,KAAK;EACLlD,QAAQ;EACRJ,KAAK;EACLV,MAAM;EACNiE,OAAO;EACPC,OAAO;EACPzE,MAAM;AACN0E,EAAAA,UAAU,GAAG,GAAG;EAChBC,KAAK,GAAG,EAAE;AACVC,EAAAA,OAAO,GAAG,KAAK;EACfC,KAAK;EACL/C,KAAK;AACLgD,EAAAA,gBAAgB,GAAG;AACK,CAAC,KAAoB;AAC7C,EAAA,IAAI,CAACpD,MAAM,CAACqD,OAAO,EAAE;AACnB,IAAA,OAAOC,OAAO,CAACC,OAAO,EAAE;AAC1B,EAAA;AACA;AACA,EAAA,MAAMC,SAAS,GAAGlD,cAAc,CAACf,KAAK,CAAC;AACvC,EAAA,MAAMkE,UAAU,GAAGnD,cAAc,CAACzB,MAAM,CAAC;AACzC,EAAA,MAAM6E,SAAS,GAAGpD,cAAc,CAAC6C,KAAK,CAAC;EAEvC,IAAIQ,SAAS,GAAG,EAAE;AAClB,EAAA,IAAIT,OAAO,EAAE;AACXS,IAAAA,SAAS,GAAG;AACVC,MAAAA,YAAY,EAAE,IAAI;AAClBC,MAAAA,SAAS,EAAEH;KACZ;AACH,EAAA;EACA,IAAII,gBAAgB,GAAG,EAAE;EACzB,IAAIxF,MAAM,CAACuD,mBAAmB,EAAE;AAC9B,IAAA,MAAMkC,YAAY,GAAG,CAACzF,MAAM,CAACuD,mBAAmB,CAACmC,SAAS,IAAI,CAAC,KAAKf,KAAK,CAACgB,WAAW,IAAI,CAAC,CAAC;AAC3F,IAAA,MAAMC,aAAa,GAAG,CAAC5F,MAAM,CAACuD,mBAAmB,CAACsC,UAAU,IAAI,CAAC,KAAKlB,KAAK,CAACmB,YAAY,IAAI,CAAC,CAAC;AAC9FN,IAAAA,gBAAgB,GAAG;AACjBO,MAAAA,kBAAkB,EAAEN,YAAY;AAChCO,MAAAA,mBAAmB,EAAEJ,aAAa;MAClCK,kBAAkB,EAAER,YAAY,GAAGG;KACpC;AACH,EAAA;AAEA,EAAA,MAAMM,qBAAqB,GAAG;IAC5B,IAAIvB,KAAK,CAACwB,eAAe,GAAG;MAAEC,oBAAoB,EAAEzB,KAAK,CAACwB;KAAiB,GAAG,EAAE,CAAC;IACjF,IAAIxB,KAAK,CAAC0B,oBAAoB,GAAG;MAAEC,2BAA2B,EAAE3B,KAAK,CAAC0B;KAAsB,GAAG,EAAE,CAAC;IAClG,IAAI1B,KAAK,CAAC4B,wBAAwB,GAAG;MAAEC,+BAA+B,EAAE7B,KAAK,CAAC4B;KAA0B,GAAG,EAAE,CAAC;IAC9G,IAAI5B,KAAK,CAAC8B,cAAc,GAAG;MAAEC,oBAAoB,EAAE/B,KAAK,CAAC8B;KAAgB,GAAG,EAAE;GAC/E;AAED,EAAA,MAAME,UAAU,GAAG;AACjBC,IAAAA,OAAO,EAAE,YAAY;AACrBC,IAAAA,eAAe,EAAEC,OAAO;AACxBC,IAAAA,YAAY,EAAE/G,MAAM,CAACsD,uBAAuB,IAAIjC,QAAQ;AACxD2F,IAAAA,SAAS,EAAEhH,MAAM,CAACqD,oBAAoB,IAAIkB,KAAK;AAC/C0C,IAAAA,oBAAoB,EAAElH,cAAc,CAACC,MAAM,CAAC;AAC5CkH,IAAAA,SAAS,EAAEzF,eAAe,CAACC,MAAM,EAAE1B,MAAM,CAACmD,kBAAkB,IAAI,KAAK,EAAE+B,SAAS,CAAC;AACjFiC,IAAAA,kBAAkB,EAAE1F,eAAe,CAACC,MAAM,EAAE1B,MAAM,CAACmD,kBAAkB,IAAI,KAAK,EAAEgC,UAAU,CAAC;AAC3FiC,IAAAA,eAAe,EAAE1C,UAAU;AAC3B2C,IAAAA,gBAAgB,EAAE1C,KAAK,CAACgB,WAAW,IAAI,CAAC;AACxC,IAAA,IAAIhB,KAAK,CAACmB,YAAY,KAAK1F,SAAS,GAAG;MAAEkH,iBAAiB,EAAE3C,KAAK,CAACmB;KAAc,GAAG,EAAE,CAAC;AACtF,IAAA,GAAGI,qBAAqB;AACxBqB,IAAAA,WAAW,EAAE/C,OAAO;AACpBgD,IAAAA,YAAY,EAAEvD,OAAO;AACrBwD,IAAAA,YAAY,EAAEhD,OAAO;IACrB,GAAGzE,MAAM,CAACkD,iBAAiB;AAC3B,IAAA,IAAIoB,UAAU,GAAG,EAAE,GAAG;AAAEoD,MAAAA,uBAAuB,EAAE;AAAM,KAAC,CAAC;AACzD,IAAA,IAAI5F,KAAK,GAAG;AAAE6F,MAAAA,SAAS,EAAE7F;KAAO,GAAG,EAAE,CAAC;AACtC,IAAA,GAAGuD,SAAS;IACZ,GAAGG;GACJ;AAED,EAAA,MAAMoC,KAAK,GAAG;IACZtD,UAAU,EAAEA,UAAU,IAAIL,OAAO;AACjC2D,IAAAA,KAAK,EAAExD,SAAS;IAChBuC,UAAU;IACVkB,MAAM,EAAE7H,MAAM,CAACoD;GAChB;AAED,EAAA,IAAI0B,gBAAgB,EAAE;AACpB;AACA,IAAA,MAAMpD,MAAM,CAACoD,gBAAgB,CAAC8C,KAAK,CAAC;AACtC,EAAA,CAAC,MAAM;AACLlG,IAAAA,MAAM,CAACqD,OAAO,CAAC6C,KAAK,CAAC;AACvB,EAAA;AACF,CAAC;;AC9rBD,MAAME,0BAA0B,GAAG,yBAAyB;;AAE5D;AACA;AACA;;AAEA,MAAMC,mBAAmB,GAAGA,MAAe;EACzC,MAAMC,GAAG,GAAGC,OAAO,CAACC,GAAG,CAACC,yBAAyB,IAAI,EAAE;AACvD,EAAA,OAAOH,GAAG,CAACI,WAAW,EAAE,KAAK,MAAM,IAAIJ,GAAG,KAAK,GAAG,IAAIA,GAAG,CAACI,WAAW,EAAE,KAAK,KAAK;AACnF,CAAC;;AAkDD;AACA;AACA;;AAIA,MAAMC,eAAe,GAAGA,CAAC/G,QAAiB,EAAEgH,gBAAoC,KAAc;AAC5F,EAAA,IAAI,CAAChH,QAAQ,EAAE,OAAOA,QAAQ;EAE9B,MAAMiH,cAAc,GAAI/H,OAAgB,IAAc;AACpD,IAAA,IAAI,OAAOA,OAAO,KAAK,QAAQ,EAAE,OAAOA,OAAO;AAE/C,IAAA,IAAI,CAACA,OAAO,EAAE,OAAOA,OAAO;AAE5B,IAAA,IAAIZ,KAAK,CAACC,OAAO,CAACW,OAAO,CAAC,EAAE;AAC1B,MAAA,OAAOA,OAAO,CAACiC,GAAG,CAAC6F,gBAAgB,CAAC;AACtC,IAAA;;AAEA;IACA,OAAOA,gBAAgB,CAAC9H,OAAO,CAAC;EAClC,CAAC;EAED,MAAMgI,cAAc,GAAIC,GAAY,IAAc;AAChD,IAAA,IAAI,CAAC/I,QAAQ,CAAC+I,GAAG,CAAC,IAAI,EAAE,SAAS,IAAIA,GAAG,CAAC,EAAE,OAAOA,GAAG;IACrD,OAAO;AAAE,MAAA,GAAGA,GAAG;AAAEjI,MAAAA,OAAO,EAAE+H,cAAc,CAACE,GAAG,CAACjI,OAAO;KAAG;EACzD,CAAC;;AAED;AACA,EAAA,IAAIZ,KAAK,CAACC,OAAO,CAACyB,QAAQ,CAAC,EAAE;AAC3B,IAAA,OAAOA,QAAQ,CAACmB,GAAG,CAAC+F,cAAc,CAAC;AACrC,EAAA;EAEA,OAAOA,cAAc,CAAClH,QAAQ,CAAC;AACjC,CAAC;AA2CD,MAAMoH,sBAAsB,GAAIC,IAAa,IAAc;AACzD,EAAA,IAAIZ,mBAAmB,EAAE,EAAE,OAAOY,IAAI;AACtC,EAAA,IAAI,CAACjJ,QAAQ,CAACiJ,IAAI,CAAC,EAAE,OAAOA,IAAI;;AAEhC;AACA,EAAA,IACE,CAACA,IAAI,CAACjI,IAAI,KAAK,OAAO,IAAIiI,IAAI,CAACjI,IAAI,KAAK,UAAU,KAClD,QAAQ,IAAIiI,IAAI,IAChBjJ,QAAQ,CAACiJ,IAAI,CAACC,MAAM,CAAC,IACrBD,IAAI,CAACC,MAAM,CAAClI,IAAI,KAAK,QAAQ,IAC7B,MAAM,IAAIiI,IAAI,CAACC,MAAM,EACrB;IACA,OAAO;AACL,MAAA,GAAGD,IAAI;AACPC,MAAAA,MAAM,EAAE;QACN,GAAGD,IAAI,CAACC,MAAM;AACdC,QAAAA,IAAI,EAAEf;AACR;KACD;AACH,EAAA;AAEA,EAAA,OAAOa,IAAI;AACb,CAAC;AAiFM,MAAMG,iBAAiB,GAAID,IAAa,IAAc;AAC3D,EAAA,OAAOR,eAAe,CAACQ,IAAI,EAAEH,sBAAsB,CAAC;AACtD,CAAC;;ACjNM,MAAMK,gBAAgB,SAASC,kCAAiB,CAAC;EAItDC,WAAWA,CAACC,MAAiC,EAAE;IAC7C,MAAM;MAAEC,OAAO;MAAE,GAAGC;AAAgB,KAAC,GAAGF,MAAM;IAC9C,KAAK,CAACE,eAAe,CAAC;IACtB,IAAI,CAACC,QAAQ,GAAGF,OAAO;IACvB,IAAI,CAAC7H,QAAQ,GAAG,IAAIgI,eAAe,CAAC,IAAI,EAAE,IAAI,CAACD,QAAQ,CAAC;AAC1D,EAAA;AACF;AAEO,MAAMC,eAAe,SAASN,kCAAiB,CAACO,QAAQ,CAAC;AAI9DN,EAAAA,WAAWA,CAACO,YAA8B,EAAEH,QAAiB,EAAE;IAC7D,KAAK,CAACG,YAAY,CAAC;IACnB,IAAI,CAACH,QAAQ,GAAGA,QAAQ;AACxB,IAAA,IAAI,CAAC5E,OAAO,GAAG+E,YAAY,CAAC/E,OAAO;AACrC,EAAA;AAWOgF,EAAAA,MAAMA,CACX/F,IAA4C,EAC5CgG,OAAwB,EACyC;IACjE,MAAM;AAAE/F,MAAAA,cAAc,EAAEgG,eAAe;AAAE/F,MAAAA;AAAc,KAAC,GAAGH,oBAAoB,CAACC,IAAI,CAAC;AACrF,IAAA,MAAMkG,SAAS,GAAGC,IAAI,CAACC,GAAG,EAAE;IAE5B,MAAMC,aAAa,GAAG,KAAK,CAACN,MAAM,CAACE,eAAe,EAAED,OAAO,CAAC;IAE5D,IAAIC,eAAe,CAACK,MAAM,EAAE;AAC1B,MAAA,OAAOD,aAAa,CAACE,IAAI,CAAEtK,KAAK,IAAK;QACnC,IAAIuK,kBAAkB,GAAG,EAAE;QAC3B,MAAMC,aAAqC,GAAG,EAAE;AAChD,QAAA,MAAMC,eAA4C,GAAG,IAAIC,GAAG,EAAE;QAC9D,IAAIC,gBAA6C,GAAG,IAAI;AAExD,QAAA,MAAM3F,KAML,GAAG;AACFgB,UAAAA,WAAW,EAAE,CAAC;AACdG,UAAAA,YAAY,EAAE,CAAC;AACfS,UAAAA,wBAAwB,EAAE,CAAC;AAC3BF,UAAAA,oBAAoB,EAAE,CAAC;AACvBI,UAAAA,cAAc,EAAE;SACjB;QACD,IAAI,KAAK,IAAI9G,KAAK,EAAE;UAClB,MAAM,CAAC4K,OAAO,EAAEC,OAAO,CAAC,GAAG7K,KAAK,CAAC8K,GAAG,EAAE;AACrC,UAAA,CAAC,YAAY;YACZ,IAAI;AACF,cAAA,WAAW,MAAMC,KAAK,IAAIH,OAAO,EAAE;AACjC;AACA,gBAAA,IAAIG,KAAK,CAAChK,IAAI,KAAK,qBAAqB,EAAE;AACxC,kBAAA,IAAIgK,KAAK,CAACC,aAAa,EAAEjK,IAAI,KAAK,MAAM,EAAE;AACxC4J,oBAAAA,gBAAgB,GAAG;AACjB5J,sBAAAA,IAAI,EAAE,MAAM;AACZC,sBAAAA,IAAI,EAAE;qBACP;AAEDwJ,oBAAAA,aAAa,CAACvJ,IAAI,CAAC0J,gBAAgB,CAAC;kBACtC,CAAC,MAAM,IAAII,KAAK,CAACC,aAAa,EAAEjK,IAAI,KAAK,UAAU,EAAE;AACnD,oBAAA,MAAMkK,SAAgC,GAAG;AACvClK,sBAAAA,IAAI,EAAE,UAAU;AAChBI,sBAAAA,EAAE,EAAE4J,KAAK,CAACC,aAAa,CAAC7J,EAAE;AAC1BC,sBAAAA,QAAQ,EAAE;AACRF,wBAAAA,IAAI,EAAE6J,KAAK,CAACC,aAAa,CAAC9J,IAAI;AAC9BG,wBAAAA,SAAS,EAAE;AACb;qBACD;AAEDmJ,oBAAAA,aAAa,CAACvJ,IAAI,CAACgK,SAAS,CAAC;oBAE7BR,eAAe,CAACS,GAAG,CAACH,KAAK,CAACC,aAAa,CAAC7J,EAAE,EAAE;AAC1CgK,sBAAAA,KAAK,EAAEF,SAAS;AAChBG,sBAAAA,WAAW,EAAE;AACf,qBAAC,CAAC;AAEFT,oBAAAA,gBAAgB,GAAG,IAAI;AACzB,kBAAA;AACF,gBAAA;;AAEA;gBACA,IAAI,OAAO,IAAII,KAAK,EAAE;AACpB,kBAAA,IAAI,MAAM,IAAIA,KAAK,CAACM,KAAK,EAAE;AACzB,oBAAA,MAAMA,KAAK,GAAGN,KAAK,CAACM,KAAK,CAACrK,IAAI;AAE9BuJ,oBAAAA,kBAAkB,IAAIc,KAAK;AAE3B,oBAAA,IAAIV,gBAAgB,EAAE;sBACpBA,gBAAgB,CAAC3J,IAAI,IAAIqK,KAAK;AAChC,oBAAA;AACF,kBAAA;AACF,gBAAA;;AAEA;AACA,gBAAA,IAAIN,KAAK,CAAChK,IAAI,KAAK,qBAAqB,IAAIgK,KAAK,CAACM,KAAK,EAAEtK,IAAI,KAAK,kBAAkB,EAAE;AACpF,kBAAA,MAAMoK,KAAK,GAAGJ,KAAK,CAACO,KAAK,KAAK7K,SAAS,GAAG+J,aAAa,CAACO,KAAK,CAACO,KAAK,CAAC,GAAG7K,SAAS;AAChF,kBAAA,MAAM8K,MAAM,GAAGJ,KAAK,EAAEpK,IAAI,KAAK,UAAU,GAAGoK,KAAK,CAAChK,EAAE,GAAGV,SAAS;kBAEhE,IAAI8K,MAAM,IAAId,eAAe,CAACe,GAAG,CAACD,MAAM,CAAC,EAAE;AACzC,oBAAA,MAAME,IAAI,GAAGhB,eAAe,CAACiB,GAAG,CAACH,MAAM,CAAC;AACxC,oBAAA,IAAIE,IAAI,EAAE;sBACRA,IAAI,CAACL,WAAW,IAAIL,KAAK,CAACM,KAAK,CAACM,YAAY,IAAI,EAAE;AACpD,oBAAA;AACF,kBAAA;AACF,gBAAA;;AAEA;AACA,gBAAA,IAAIZ,KAAK,CAAChK,IAAI,KAAK,oBAAoB,EAAE;AACvC4J,kBAAAA,gBAAgB,GAAG,IAAI;;AAEvB;AACA,kBAAA,IAAII,KAAK,CAACO,KAAK,KAAK7K,SAAS,EAAE;AAC7B,oBAAA,MAAM0K,KAAK,GAAGX,aAAa,CAACO,KAAK,CAACO,KAAK,CAAC;AAExC,oBAAA,IAAIH,KAAK,EAAEpK,IAAI,KAAK,UAAU,IAAIoK,KAAK,CAAChK,EAAE,IAAIsJ,eAAe,CAACe,GAAG,CAACL,KAAK,CAAChK,EAAE,CAAC,EAAE;sBAC3E,MAAMsK,IAAI,GAAGhB,eAAe,CAACiB,GAAG,CAACP,KAAK,CAAChK,EAAE,CAAC;AAC1C,sBAAA,IAAIsK,IAAI,EAAE;wBACR,IAAI;AACFN,0BAAAA,KAAK,CAAC/J,QAAQ,CAACC,SAAS,GAAGmB,IAAI,CAACC,KAAK,CAACgJ,IAAI,CAACL,WAAW,CAAC;wBACzD,CAAC,CAAC,OAAOQ,CAAC,EAAE;AACV;AACAzH,0BAAAA,OAAO,CAACe,KAAK,CAAC,2BAA2B,EAAE0G,CAAC,CAAC;AAC/C,wBAAA;AACF,sBAAA;AACAnB,sBAAAA,eAAe,CAACoB,MAAM,CAACV,KAAK,CAAChK,EAAE,CAAC;AAClC,oBAAA;AACF,kBAAA;AACF,gBAAA;AAEA,gBAAA,IAAI4J,KAAK,CAAChK,IAAI,IAAI,eAAe,EAAE;kBACjCiE,KAAK,CAACgB,WAAW,GAAG+E,KAAK,CAACe,OAAO,CAAC9G,KAAK,CAAC+G,YAAY,IAAI,CAAC;kBACzD/G,KAAK,CAAC4B,wBAAwB,GAAGmE,KAAK,CAACe,OAAO,CAAC9G,KAAK,CAACgH,2BAA2B,IAAI,CAAC;kBACrFhH,KAAK,CAAC0B,oBAAoB,GAAGqE,KAAK,CAACe,OAAO,CAAC9G,KAAK,CAACiH,uBAAuB,IAAI,CAAC;AAC7EjH,kBAAAA,KAAK,CAAC8B,cAAc,GAAGiE,KAAK,CAACe,OAAO,CAAC9G,KAAK,CAACkH,eAAe,EAAEC,mBAAmB,IAAI,CAAC;AACtF,gBAAA;gBACA,IAAI,OAAO,IAAIpB,KAAK,EAAE;kBACpB/F,KAAK,CAACmB,YAAY,GAAG4E,KAAK,CAAC/F,KAAK,CAACoH,aAAa,IAAI,CAAC;AACnD;kBACA,IAAIrB,KAAK,CAAC/F,KAAK,CAACkH,eAAe,EAAEC,mBAAmB,KAAK1L,SAAS,EAAE;oBAClEuE,KAAK,CAAC8B,cAAc,GAAGiE,KAAK,CAAC/F,KAAK,CAACkH,eAAe,CAACC,mBAAmB;AACxE,kBAAA;AACF,gBAAA;AACF,cAAA;cAEA,MAAMtH,OAAO,GAAG,CAACqF,IAAI,CAACC,GAAG,EAAE,GAAGF,SAAS,IAAI,IAAI;AAE/C,cAAA,MAAMoC,cAAc,GAAGnK,yBAAyB,CAAC,WAAW,EAAE8H,eAAe,CAAC;;AAE9E;cACA,MAAMsC,eAAmC,GACvC9B,aAAa,CAACjJ,MAAM,GAAG,CAAC,GACpB,CACE;AACEC,gBAAAA,IAAI,EAAE,WAAW;AACjBX,gBAAAA,OAAO,EAAE2J;eACV,CACF,GACD,CACE;AACEhJ,gBAAAA,IAAI,EAAE,WAAW;AACjBX,gBAAAA,OAAO,EAAE,CAAC;AAAEE,kBAAAA,IAAI,EAAE,MAAM;AAAEC,kBAAAA,IAAI,EAAEuJ;iBAAoB;AACtD,eAAC,CACF;AAEP,cAAA,MAAM/F,kBAAkB,CAAC;gBACvBzC,MAAM,EAAE,IAAI,CAAC2H,QAAQ;AACrB,gBAAA,GAAGzF,aAAa;gBAChBW,KAAK,EAAEoF,eAAe,CAACpF,KAAK;AAC5BlD,gBAAAA,QAAQ,EAAE,WAAW;gBACrBJ,KAAK,EAAE6H,iBAAiB,CAAC1H,iBAAiB,CAACuI,eAAe,EAAE,WAAW,CAAC,CAAC;AACzEpJ,gBAAAA,MAAM,EAAE0L,eAAe;gBACvBzH,OAAO;gBACPC,OAAO,EAAE,IAAI,CAACA,OAAO;AACrBzE,gBAAAA,MAAM,EAAE0D,IAAI;AACZgB,gBAAAA,UAAU,EAAE,GAAG;gBACfC,KAAK;AACL7C,gBAAAA,KAAK,EAAEkK;AACT,eAAC,CAAC;YACJ,CAAC,CAAC,OAAOnH,KAAU,EAAE;AACnB;AACA,cAAA,MAAMV,kBAAkB,CAAC;gBACvBzC,MAAM,EAAE,IAAI,CAAC2H,QAAQ;AACrB,gBAAA,GAAGzF,aAAa;gBAChBW,KAAK,EAAEoF,eAAe,CAACpF,KAAK;AAC5BlD,gBAAAA,QAAQ,EAAE,WAAW;gBACrBJ,KAAK,EAAE6H,iBAAiB,CAAC1H,iBAAiB,CAACuI,eAA4B,CAAC,CAAC;AACzEpJ,gBAAAA,MAAM,EAAE,EAAE;AACViE,gBAAAA,OAAO,EAAE,CAAC;gBACVC,OAAO,EAAE,IAAI,CAACA,OAAO;AACrBzE,gBAAAA,MAAM,EAAE0D,IAAI;gBACZgB,UAAU,EAAEG,KAAK,EAAEqH,MAAM,GAAGrH,KAAK,CAACqH,MAAM,GAAG,GAAG;AAC9CvH,gBAAAA,KAAK,EAAE;AACLgB,kBAAAA,WAAW,EAAE,CAAC;AACdG,kBAAAA,YAAY,EAAE;iBACf;AACDlB,gBAAAA,OAAO,EAAE,IAAI;AACbC,gBAAAA,KAAK,EAAE1C,IAAI,CAACE,SAAS,CAACwC,KAAK;AAC7B,eAAC,CAAC;AACJ,YAAA;AACF,UAAA,CAAC,GAAG;;AAEJ;AACA,UAAA,OAAO2F,OAAO;AAChB,QAAA;AACA,QAAA,OAAO7K,KAAK;AACd,MAAA,CAAC,CAAC;AACJ,IAAA,CAAC,MAAM;MACL,MAAMwM,cAAc,GAAGpC,aAAa,CAACE,IAAI,CACvC,MAAOmC,MAAM,IAAK;QAChB,IAAI,SAAS,IAAIA,MAAM,EAAE;UACvB,MAAM5H,OAAO,GAAG,CAACqF,IAAI,CAACC,GAAG,EAAE,GAAGF,SAAS,IAAI,IAAI;AAE/C,UAAA,MAAMoC,cAAc,GAAGnK,yBAAyB,CAAC,WAAW,EAAE8H,eAAe,CAAC;AAE9E,UAAA,MAAMxF,kBAAkB,CAAC;YACvBzC,MAAM,EAAE,IAAI,CAAC2H,QAAQ;AACrB,YAAA,GAAGzF,aAAa;YAChBW,KAAK,EAAEoF,eAAe,CAACpF,KAAK;AAC5BlD,YAAAA,QAAQ,EAAE,WAAW;YACrBJ,KAAK,EAAE6H,iBAAiB,CAAC1H,iBAAiB,CAACuI,eAA4B,CAAC,CAAC;AACzEpJ,YAAAA,MAAM,EAAEF,uBAAuB,CAAC+L,MAAM,CAAC;YACvC5H,OAAO;YACPC,OAAO,EAAE,IAAI,CAACA,OAAO;AACrBzE,YAAAA,MAAM,EAAE0D,IAAI;AACZgB,YAAAA,UAAU,EAAE,GAAG;AACfC,YAAAA,KAAK,EAAE;AACLgB,cAAAA,WAAW,EAAEyG,MAAM,CAACzH,KAAK,CAAC+G,YAAY,IAAI,CAAC;AAC3C5F,cAAAA,YAAY,EAAEsG,MAAM,CAACzH,KAAK,CAACoH,aAAa,IAAI,CAAC;AAC7CxF,cAAAA,wBAAwB,EAAE6F,MAAM,CAACzH,KAAK,CAACgH,2BAA2B,IAAI,CAAC;AACvEtF,cAAAA,oBAAoB,EAAE+F,MAAM,CAACzH,KAAK,CAACiH,uBAAuB,IAAI,CAAC;cAC/DnF,cAAc,EAAE2F,MAAM,CAACzH,KAAK,CAACkH,eAAe,EAAEC,mBAAmB,IAAI;aACtE;AACDhK,YAAAA,KAAK,EAAEkK;AACT,WAAC,CAAC;AACJ,QAAA;AACA,QAAA,OAAOI,MAAM;MACf,CAAC,EACD,MAAOvH,KAAU,IAAK;AACpB,QAAA,MAAMV,kBAAkB,CAAC;UACvBzC,MAAM,EAAE,IAAI,CAAC2H,QAAQ;AACrB,UAAA,GAAGzF,aAAa;UAChBW,KAAK,EAAEoF,eAAe,CAACpF,KAAK;AAC5BlD,UAAAA,QAAQ,EAAE,WAAW;UACrBJ,KAAK,EAAE6H,iBAAiB,CAAC1H,iBAAiB,CAACuI,eAA4B,CAAC,CAAC;AACzEpJ,UAAAA,MAAM,EAAE,EAAE;AACViE,UAAAA,OAAO,EAAE,CAAC;UACVC,OAAO,EAAE,IAAI,CAACA,OAAO;AACrBzE,UAAAA,MAAM,EAAE0D,IAAI;UACZgB,UAAU,EAAEG,KAAK,EAAEqH,MAAM,GAAGrH,KAAK,CAACqH,MAAM,GAAG,GAAG;AAC9CvH,UAAAA,KAAK,EAAE;AACLgB,YAAAA,WAAW,EAAE,CAAC;AACdG,YAAAA,YAAY,EAAE;WACf;AACDlB,UAAAA,OAAO,EAAE,IAAI;AACbC,UAAAA,KAAK,EAAE1C,IAAI,CAACE,SAAS,CAACwC,KAAK;AAC7B,SAAC,CAAC;AACF,QAAA,MAAMA,KAAK;AACb,MAAA,CACF,CAAwB;AAExB,MAAA,OAAOsH,cAAc;AACvB,IAAA;AACF,EAAA;AACF;;;;;;;"}
1
+ {"version":3,"file":"index.cjs","sources":["../../src/typeGuards.ts","../../src/utils.ts","../../src/sanitization.ts","../../src/anthropic/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 { EventMessage, 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'\nimport { uuidv7, ErrorTracking as CoreErrorTracking } from '@posthog/core'\n\ntype ChatCompletionCreateParamsBase = OpenAIOrignal.Chat.Completions.ChatCompletionCreateParams\ntype MessageCreateParams = AnthropicOriginal.Messages.MessageCreateParams\ntype ResponseCreateParams = OpenAIOrignal.Responses.ResponseCreateParams\ntype EmbeddingCreateParams = OpenAIOrignal.EmbeddingCreateParams\ntype TranscriptionCreateParams = OpenAIOrignal.Audio.Transcriptions.TranscriptionCreateParams\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 | TranscriptionCreateParams\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 'language',\n 'response_format',\n 'timestamp_granularities',\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\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 // Handle audio output (gpt-4o-audio-preview)\n if (choice.message.audio) {\n content.push({\n type: 'audio',\n ...choice.message.audio,\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 } else if (part.inlineData) {\n // Handle audio/media inline data\n const mimeType = part.inlineData.mimeType || 'audio/pcm'\n let data = part.inlineData.data\n\n // Handle binary data (Buffer/Uint8Array -> base64)\n if (data instanceof Uint8Array || Buffer.isBuffer(data)) {\n data = Buffer.from(data).toString('base64')\n }\n\n content.push({\n type: 'audio',\n mime_type: mimeType,\n data: data,\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 * Calculate web search count from raw API response.\n *\n * Uses a two-tier detection strategy:\n * Priority 1 (Exact Count): Count actual web search calls when available\n * Priority 2 (Binary Detection): Return 1 if web search indicators are present, 0 otherwise\n *\n * @param result - Raw API response from any provider (OpenAI, Perplexity, OpenRouter, Gemini, etc.)\n * @returns Number of web searches performed (exact count or binary 1/0)\n */\nexport function calculateWebSearchCount(result: unknown): number {\n if (!result || typeof result !== 'object') {\n return 0\n }\n\n // Priority 1: Exact Count\n // Check for OpenAI Responses API web_search_call items\n if ('output' in result && Array.isArray(result.output)) {\n let count = 0\n\n for (const item of result.output) {\n if (typeof item === 'object' && item !== null && 'type' in item && item.type === 'web_search_call') {\n count++\n }\n }\n\n if (count > 0) {\n return count\n }\n }\n\n // Priority 2: Binary Detection (1 or 0)\n\n // Check for citations at root level (Perplexity)\n if ('citations' in result && Array.isArray(result.citations) && result.citations.length > 0) {\n return 1\n }\n\n // Check for search_results at root level (Perplexity via OpenRouter)\n if ('search_results' in result && Array.isArray(result.search_results) && result.search_results.length > 0) {\n return 1\n }\n\n // Check for usage.search_context_size (Perplexity via OpenRouter)\n if ('usage' in result && typeof result.usage === 'object' && result.usage !== null) {\n if ('search_context_size' in result.usage && result.usage.search_context_size) {\n return 1\n }\n }\n\n // Check for annotations with url_citation in choices[].message or choices[].delta (OpenAI/Perplexity)\n if ('choices' in result && Array.isArray(result.choices)) {\n for (const choice of result.choices) {\n if (typeof choice === 'object' && choice !== null) {\n // Check both message (non-streaming) and delta (streaming) for annotations\n const content = ('message' in choice ? choice.message : null) || ('delta' in choice ? choice.delta : null)\n\n if (typeof content === 'object' && content !== null && 'annotations' in content) {\n const annotations = content.annotations\n\n if (Array.isArray(annotations)) {\n const hasUrlCitation = annotations.some((ann: unknown) => {\n return typeof ann === 'object' && ann !== null && 'type' in ann && ann.type === 'url_citation'\n })\n\n if (hasUrlCitation) {\n return 1\n }\n }\n }\n }\n }\n }\n\n // Check for annotations in output[].content[] (OpenAI Responses API)\n if ('output' in result && Array.isArray(result.output)) {\n for (const item of result.output) {\n if (typeof item === 'object' && item !== null && 'content' in item) {\n const content = item.content\n\n if (Array.isArray(content)) {\n for (const contentItem of content) {\n if (typeof contentItem === 'object' && contentItem !== null && 'annotations' in contentItem) {\n const annotations = contentItem.annotations\n\n if (Array.isArray(annotations)) {\n const hasUrlCitation = annotations.some((ann: unknown) => {\n return typeof ann === 'object' && ann !== null && 'type' in ann && ann.type === 'url_citation'\n })\n\n if (hasUrlCitation) {\n return 1\n }\n }\n }\n }\n }\n }\n }\n }\n\n // Check for grounding_metadata (Gemini)\n if ('candidates' in result && Array.isArray(result.candidates)) {\n for (const candidate of result.candidates) {\n if (\n typeof candidate === 'object' &&\n candidate !== null &&\n 'grounding_metadata' in candidate &&\n candidate.grounding_metadata\n ) {\n return 1\n }\n }\n }\n\n return 0\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 | TranscriptionCreateParams\n ) &\n MonitoringParams\n error?: unknown\n exceptionId?: 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 sendEventWithErrorToPosthog = async ({\n client,\n traceId,\n error,\n ...args\n}: Omit<SendEventToPosthogParams, 'error' | 'httpStatus'> &\n Required<Pick<SendEventToPosthogParams, 'error'>>): Promise<unknown> => {\n const httpStatus =\n error && typeof error === 'object' && 'status' in error ? ((error as { status?: number }).status ?? 500) : 500\n\n const properties = { client, traceId, httpStatus, error: JSON.stringify(error), ...args }\n const enrichedError = error as CoreErrorTracking.PreviouslyCapturedError\n\n if (client.options?.enableExceptionAutocapture) {\n // assign a uuid that can be used to link the trace and exception events\n const exceptionId = uuidv7()\n client.captureException(error, undefined, { $ai_trace_id: traceId }, exceptionId)\n enrichedError.__posthog_previously_captured_error = true\n properties.exceptionId = exceptionId\n }\n\n await sendEventToPosthog(properties)\n\n return enrichedError\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 error,\n exceptionId,\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 (error) {\n errorData = {\n $ai_is_error: true,\n $ai_error: safeError,\n $exception_event_id: exceptionId,\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 ...(usage.webSearchCount ? { $ai_web_search_count: usage.webSearchCount } : {}),\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: EventMessage = {\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 return Promise.resolve()\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// Multimodal Feature Toggle\n// ============================================\n\nconst isMultimodalEnabled = (): boolean => {\n const val = process.env._INTERNAL_LLMA_MULTIMODAL || ''\n return val.toLowerCase() === 'true' || val === '1' || val.toLowerCase() === 'yes'\n}\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 (isMultimodalEnabled()) return str\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 // Handle audio format\n if (item.type === 'audio' && 'data' in item) {\n if (isMultimodalEnabled()) return item\n return { ...item, data: REDACTED_IMAGE_PLACEHOLDER }\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 (isMultimodalEnabled()) return item\n if (!isObject(item)) return item\n\n // Handle Anthropic's image and document formats (same structure, different type field)\n if (\n (item.type === 'image' || item.type === 'document') &&\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 (isMultimodalEnabled()) return part\n if (!isObject(part)) return part\n\n // Handle Gemini's inline data format (images, audio, PDFs all use inlineData)\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 if (isMultimodalEnabled()) return item\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 AnthropicOriginal, { APIPromise } from '@anthropic-ai/sdk'\nimport { PostHog } from 'posthog-node'\nimport {\n formatResponseAnthropic,\n mergeSystemPrompt,\n MonitoringParams,\n sendEventToPosthog,\n extractAvailableToolCalls,\n extractPosthogParams,\n sendEventWithErrorToPosthog,\n} from '../utils'\nimport type { FormattedContentItem, FormattedTextContent, FormattedFunctionCall, FormattedMessage } from '../types'\n\ntype MessageCreateParamsNonStreaming = AnthropicOriginal.Messages.MessageCreateParamsNonStreaming\ntype MessageCreateParamsStreaming = AnthropicOriginal.Messages.MessageCreateParamsStreaming\ntype MessageCreateParams = AnthropicOriginal.Messages.MessageCreateParams\ntype Message = AnthropicOriginal.Messages.Message\ntype RawMessageStreamEvent = AnthropicOriginal.Messages.RawMessageStreamEvent\ntype MessageCreateParamsBase = AnthropicOriginal.Messages.MessageCreateParams\ntype RequestOptions = AnthropicOriginal.RequestOptions\nimport type { Stream } from '@anthropic-ai/sdk/streaming'\nimport { sanitizeAnthropic } from '../sanitization'\n\ninterface ToolInProgress {\n block: FormattedFunctionCall\n inputString: string\n}\n\ninterface MonitoringAnthropicConfig {\n apiKey: string\n posthog: PostHog\n baseURL?: string\n}\n\nexport class PostHogAnthropic extends AnthropicOriginal {\n private readonly phClient: PostHog\n public messages: WrappedMessages\n\n constructor(config: MonitoringAnthropicConfig) {\n const { posthog, ...anthropicConfig } = config\n super(anthropicConfig)\n this.phClient = posthog\n this.messages = new WrappedMessages(this, this.phClient)\n }\n}\n\nexport class WrappedMessages extends AnthropicOriginal.Messages {\n private readonly phClient: PostHog\n private readonly baseURL: string\n\n constructor(parentClient: PostHogAnthropic, phClient: PostHog) {\n super(parentClient)\n this.phClient = phClient\n this.baseURL = parentClient.baseURL\n }\n\n public create(body: MessageCreateParamsNonStreaming, options?: RequestOptions): APIPromise<Message>\n public create(\n body: MessageCreateParamsStreaming & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<Stream<RawMessageStreamEvent>>\n public create(\n body: MessageCreateParamsBase & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<Stream<RawMessageStreamEvent> | Message>\n public create(\n body: MessageCreateParams & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<Message> | APIPromise<Stream<RawMessageStreamEvent>> {\n const { providerParams: anthropicParams, posthogParams } = extractPosthogParams(body)\n const startTime = Date.now()\n\n const parentPromise = super.create(anthropicParams, options)\n\n if (anthropicParams.stream) {\n return parentPromise.then((value) => {\n let accumulatedContent = ''\n const contentBlocks: FormattedContentItem[] = []\n const toolsInProgress: Map<string, ToolInProgress> = new Map()\n let currentTextBlock: FormattedTextContent | null = null\n\n const usage: {\n inputTokens: number\n outputTokens: number\n cacheCreationInputTokens?: number\n cacheReadInputTokens?: number\n webSearchCount?: number\n } = {\n inputTokens: 0,\n outputTokens: 0,\n cacheCreationInputTokens: 0,\n cacheReadInputTokens: 0,\n webSearchCount: 0,\n }\n if ('tee' in value) {\n const [stream1, stream2] = value.tee()\n ;(async () => {\n try {\n for await (const chunk of stream1) {\n // Handle content block start events\n if (chunk.type === 'content_block_start') {\n if (chunk.content_block?.type === 'text') {\n currentTextBlock = {\n type: 'text',\n text: '',\n }\n\n contentBlocks.push(currentTextBlock)\n } else if (chunk.content_block?.type === 'tool_use') {\n const toolBlock: FormattedFunctionCall = {\n type: 'function',\n id: chunk.content_block.id,\n function: {\n name: chunk.content_block.name,\n arguments: {},\n },\n }\n\n contentBlocks.push(toolBlock)\n\n toolsInProgress.set(chunk.content_block.id, {\n block: toolBlock,\n inputString: '',\n })\n\n currentTextBlock = null\n }\n }\n\n // Handle text delta events\n if ('delta' in chunk) {\n if ('text' in chunk.delta) {\n const delta = chunk.delta.text\n\n accumulatedContent += delta\n\n if (currentTextBlock) {\n currentTextBlock.text += delta\n }\n }\n }\n\n // Handle tool input delta events\n if (chunk.type === 'content_block_delta' && chunk.delta?.type === 'input_json_delta') {\n const block = chunk.index !== undefined ? contentBlocks[chunk.index] : undefined\n const toolId = block?.type === 'function' ? block.id : undefined\n\n if (toolId && toolsInProgress.has(toolId)) {\n const tool = toolsInProgress.get(toolId)\n if (tool) {\n tool.inputString += chunk.delta.partial_json || ''\n }\n }\n }\n\n // Handle content block stop events\n if (chunk.type === 'content_block_stop') {\n currentTextBlock = null\n\n // Parse accumulated tool input\n if (chunk.index !== undefined) {\n const block = contentBlocks[chunk.index]\n\n if (block?.type === 'function' && block.id && toolsInProgress.has(block.id)) {\n const tool = toolsInProgress.get(block.id)\n if (tool) {\n try {\n block.function.arguments = JSON.parse(tool.inputString)\n } catch (e) {\n // Keep empty object if parsing fails\n console.error('Error parsing tool input:', e)\n }\n }\n toolsInProgress.delete(block.id)\n }\n }\n }\n\n if (chunk.type == 'message_start') {\n usage.inputTokens = chunk.message.usage.input_tokens ?? 0\n usage.cacheCreationInputTokens = chunk.message.usage.cache_creation_input_tokens ?? 0\n usage.cacheReadInputTokens = chunk.message.usage.cache_read_input_tokens ?? 0\n usage.webSearchCount = chunk.message.usage.server_tool_use?.web_search_requests ?? 0\n }\n if ('usage' in chunk) {\n usage.outputTokens = chunk.usage.output_tokens ?? 0\n // Update web search count if present in delta\n if (chunk.usage.server_tool_use?.web_search_requests !== undefined) {\n usage.webSearchCount = chunk.usage.server_tool_use.web_search_requests\n }\n }\n }\n\n const latency = (Date.now() - startTime) / 1000\n\n const availableTools = extractAvailableToolCalls('anthropic', anthropicParams)\n\n // Format output to match non-streaming version\n const formattedOutput: FormattedMessage[] =\n contentBlocks.length > 0\n ? [\n {\n role: 'assistant',\n content: contentBlocks,\n },\n ]\n : [\n {\n role: 'assistant',\n content: [{ type: 'text', text: accumulatedContent }],\n },\n ]\n\n await sendEventToPosthog({\n client: this.phClient,\n ...posthogParams,\n model: anthropicParams.model,\n provider: 'anthropic',\n input: sanitizeAnthropic(mergeSystemPrompt(anthropicParams, 'anthropic')),\n output: formattedOutput,\n latency,\n baseURL: this.baseURL,\n params: body,\n httpStatus: 200,\n usage,\n tools: availableTools,\n })\n } catch (error: unknown) {\n const enrichedError = await sendEventWithErrorToPosthog({\n client: this.phClient,\n ...posthogParams,\n model: anthropicParams.model,\n provider: 'anthropic',\n input: sanitizeAnthropic(mergeSystemPrompt(anthropicParams, 'anthropic')),\n output: [],\n latency: 0,\n baseURL: this.baseURL,\n params: body,\n usage: {\n inputTokens: 0,\n outputTokens: 0,\n },\n error: error,\n })\n throw enrichedError\n }\n })()\n\n // Return the other stream to the user\n return stream2\n }\n return value\n }) as APIPromise<Stream<RawMessageStreamEvent>>\n } else {\n const wrappedPromise = parentPromise.then(\n async (result) => {\n if ('content' in result) {\n const latency = (Date.now() - startTime) / 1000\n\n const availableTools = extractAvailableToolCalls('anthropic', anthropicParams)\n\n await sendEventToPosthog({\n client: this.phClient,\n ...posthogParams,\n model: anthropicParams.model,\n provider: 'anthropic',\n input: sanitizeAnthropic(mergeSystemPrompt(anthropicParams, 'anthropic')),\n output: formatResponseAnthropic(result),\n latency,\n baseURL: this.baseURL,\n params: body,\n httpStatus: 200,\n usage: {\n inputTokens: result.usage.input_tokens ?? 0,\n outputTokens: result.usage.output_tokens ?? 0,\n cacheCreationInputTokens: result.usage.cache_creation_input_tokens ?? 0,\n cacheReadInputTokens: result.usage.cache_read_input_tokens ?? 0,\n webSearchCount: result.usage.server_tool_use?.web_search_requests ?? 0,\n },\n tools: availableTools,\n })\n }\n return result\n },\n async (error: any) => {\n await sendEventToPosthog({\n client: this.phClient,\n ...posthogParams,\n model: anthropicParams.model,\n provider: 'anthropic',\n input: sanitizeAnthropic(mergeSystemPrompt(anthropicParams, 'anthropic')),\n output: [],\n latency: 0,\n baseURL: this.baseURL,\n params: body,\n httpStatus: error?.status ? error.status : 500,\n usage: {\n inputTokens: 0,\n outputTokens: 0,\n },\n error: JSON.stringify(error),\n })\n throw error\n }\n ) as APIPromise<Message>\n\n return wrappedPromise\n }\n }\n}\n\nexport default PostHogAnthropic\n\nexport { PostHogAnthropic as Anthropic }\n"],"names":["isObject","value","Array","isArray","STRING_FORMAT","getModelParams","params","modelParams","paramKeys","key","undefined","formatResponseAnthropic","response","output","content","choice","type","text","push","name","id","function","arguments","input","length","role","mergeSystemPrompt","provider","messages","system","systemMessage","withPrivacyMode","client","privacyMode","privacy_mode","extractAvailableToolCalls","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","sendEventWithErrorToPosthog","error","args","httpStatus","status","properties","enrichedError","options","enableExceptionAutocapture","exceptionId","uuidv7","captureException","$ai_trace_id","__posthog_previously_captured_error","sendEventToPosthog","eventType","Generation","distinctId","model","latency","baseURL","usage","captureImmediate","capture","Promise","resolve","safeInput","safeOutput","safeError","errorData","$ai_is_error","$ai_error","$exception_event_id","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","webSearchCount","$ai_web_search_count","$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_base_url","$process_person_profile","$ai_tools","event","groups","REDACTED_IMAGE_PLACEHOLDER","isMultimodalEnabled","val","process","env","_INTERNAL_LLMA_MULTIMODAL","toLowerCase","processMessages","transformContent","processContent","processMessage","msg","sanitizeAnthropicImage","item","source","data","sanitizeAnthropic","PostHogAnthropic","AnthropicOriginal","constructor","config","posthog","anthropicConfig","phClient","WrappedMessages","Messages","parentClient","create","anthropicParams","startTime","Date","now","parentPromise","stream","then","accumulatedContent","contentBlocks","toolsInProgress","Map","currentTextBlock","stream1","stream2","tee","chunk","content_block","toolBlock","set","block","inputString","delta","index","toolId","has","tool","get","partial_json","e","delete","message","input_tokens","cache_creation_input_tokens","cache_read_input_tokens","server_tool_use","web_search_requests","output_tokens","availableTools","formattedOutput","wrappedPromise","result"],"mappings":"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"}
@@ -1,8 +1,9 @@
1
1
  import AnthropicOriginal from '@anthropic-ai/sdk';
2
2
  import { Buffer } from 'buffer';
3
3
  import { v4 } from 'uuid';
4
+ import { uuidv7 } from '@posthog/core';
4
5
 
5
- var version = "7.3.2";
6
+ var version = "7.4.0";
6
7
 
7
8
  // Type guards for safer type checking
8
9
 
@@ -135,6 +136,33 @@ function addDefaults(params) {
135
136
  traceId: params.traceId ?? v4()
136
137
  };
137
138
  }
139
+ const sendEventWithErrorToPosthog = async ({
140
+ client,
141
+ traceId,
142
+ error,
143
+ ...args
144
+ }) => {
145
+ const httpStatus = error && typeof error === 'object' && 'status' in error ? error.status ?? 500 : 500;
146
+ const properties = {
147
+ client,
148
+ traceId,
149
+ httpStatus,
150
+ error: JSON.stringify(error),
151
+ ...args
152
+ };
153
+ const enrichedError = error;
154
+ if (client.options?.enableExceptionAutocapture) {
155
+ // assign a uuid that can be used to link the trace and exception events
156
+ const exceptionId = uuidv7();
157
+ client.captureException(error, undefined, {
158
+ $ai_trace_id: traceId
159
+ }, exceptionId);
160
+ enrichedError.__posthog_previously_captured_error = true;
161
+ properties.exceptionId = exceptionId;
162
+ }
163
+ await sendEventToPosthog(properties);
164
+ return enrichedError;
165
+ };
138
166
  const sendEventToPosthog = async ({
139
167
  client,
140
168
  eventType = AIEvent.Generation,
@@ -149,8 +177,8 @@ const sendEventToPosthog = async ({
149
177
  params,
150
178
  httpStatus = 200,
151
179
  usage = {},
152
- isError = false,
153
180
  error,
181
+ exceptionId,
154
182
  tools,
155
183
  captureImmediate = false
156
184
  }) => {
@@ -162,10 +190,11 @@ const sendEventToPosthog = async ({
162
190
  const safeOutput = sanitizeValues(output);
163
191
  const safeError = sanitizeValues(error);
164
192
  let errorData = {};
165
- if (isError) {
193
+ if (error) {
166
194
  errorData = {
167
195
  $ai_is_error: true,
168
- $ai_error: safeError
196
+ $ai_error: safeError,
197
+ $exception_event_id: exceptionId
169
198
  };
170
199
  }
171
200
  let costOverrideData = {};
@@ -231,6 +260,7 @@ const sendEventToPosthog = async ({
231
260
  } else {
232
261
  client.capture(event);
233
262
  }
263
+ return Promise.resolve();
234
264
  };
235
265
 
236
266
  const REDACTED_IMAGE_PLACEHOLDER = '[base64 image redacted]';
@@ -449,8 +479,7 @@ class WrappedMessages extends AnthropicOriginal.Messages {
449
479
  tools: availableTools
450
480
  });
451
481
  } catch (error) {
452
- // error handling
453
- await sendEventToPosthog({
482
+ const enrichedError = await sendEventWithErrorToPosthog({
454
483
  client: this.phClient,
455
484
  ...posthogParams,
456
485
  model: anthropicParams.model,
@@ -460,14 +489,13 @@ class WrappedMessages extends AnthropicOriginal.Messages {
460
489
  latency: 0,
461
490
  baseURL: this.baseURL,
462
491
  params: body,
463
- httpStatus: error?.status ? error.status : 500,
464
492
  usage: {
465
493
  inputTokens: 0,
466
494
  outputTokens: 0
467
495
  },
468
- isError: true,
469
- error: JSON.stringify(error)
496
+ error: error
470
497
  });
498
+ throw enrichedError;
471
499
  }
472
500
  })();
473
501
 
@@ -519,7 +547,6 @@ class WrappedMessages extends AnthropicOriginal.Messages {
519
547
  inputTokens: 0,
520
548
  outputTokens: 0
521
549
  },
522
- isError: true,
523
550
  error: JSON.stringify(error)
524
551
  });
525
552
  throw error;