@posthog/ai 5.2.2 → 6.0.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.
Files changed (60) hide show
  1. package/LICENSE +245 -0
  2. package/{lib → dist}/anthropic/index.cjs +44 -17
  3. package/dist/anthropic/index.cjs.map +1 -0
  4. package/{lib → dist}/anthropic/index.mjs +41 -10
  5. package/dist/anthropic/index.mjs.map +1 -0
  6. package/{lib → dist}/gemini/index.cjs +68 -26
  7. package/dist/gemini/index.cjs.map +1 -0
  8. package/{lib → dist}/gemini/index.d.ts +0 -1
  9. package/{lib → dist}/gemini/index.mjs +67 -25
  10. package/dist/gemini/index.mjs.map +1 -0
  11. package/{lib → dist}/index.cjs +875 -601
  12. package/dist/index.cjs.map +1 -0
  13. package/{lib → dist}/index.d.ts +3 -3
  14. package/{lib → dist}/index.mjs +859 -579
  15. package/dist/index.mjs.map +1 -0
  16. package/{lib → dist}/langchain/index.cjs +178 -118
  17. package/dist/langchain/index.cjs.map +1 -0
  18. package/{lib → dist}/langchain/index.d.ts +1 -0
  19. package/{lib → dist}/langchain/index.mjs +175 -112
  20. package/dist/langchain/index.mjs.map +1 -0
  21. package/{lib → dist}/openai/index.cjs +113 -6
  22. package/dist/openai/index.cjs.map +1 -0
  23. package/{lib → dist}/openai/index.mjs +112 -5
  24. package/dist/openai/index.mjs.map +1 -0
  25. package/{lib → dist}/vercel/index.cjs +117 -82
  26. package/dist/vercel/index.cjs.map +1 -0
  27. package/{lib → dist}/vercel/index.d.ts +2 -2
  28. package/{lib → dist}/vercel/index.mjs +118 -81
  29. package/dist/vercel/index.mjs.map +1 -0
  30. package/package.json +45 -35
  31. package/CHANGELOG.md +0 -89
  32. package/index.ts +0 -1
  33. package/lib/anthropic/index.cjs.map +0 -1
  34. package/lib/anthropic/index.mjs.map +0 -1
  35. package/lib/gemini/index.cjs.map +0 -1
  36. package/lib/gemini/index.mjs.map +0 -1
  37. package/lib/index.cjs.map +0 -1
  38. package/lib/index.mjs.map +0 -1
  39. package/lib/langchain/index.cjs.map +0 -1
  40. package/lib/langchain/index.mjs.map +0 -1
  41. package/lib/openai/index.cjs.map +0 -1
  42. package/lib/openai/index.mjs.map +0 -1
  43. package/lib/vercel/index.cjs.map +0 -1
  44. package/lib/vercel/index.mjs.map +0 -1
  45. package/src/anthropic/index.ts +0 -211
  46. package/src/gemini/index.ts +0 -254
  47. package/src/index.ts +0 -13
  48. package/src/langchain/callbacks.ts +0 -640
  49. package/src/langchain/index.ts +0 -1
  50. package/src/openai/azure.ts +0 -481
  51. package/src/openai/index.ts +0 -498
  52. package/src/utils.ts +0 -287
  53. package/src/vercel/index.ts +0 -1
  54. package/src/vercel/middleware.ts +0 -393
  55. package/tests/callbacks.test.ts +0 -48
  56. package/tests/gemini.test.ts +0 -344
  57. package/tests/openai.test.ts +0 -403
  58. package/tsconfig.json +0 -10
  59. /package/{lib → dist}/anthropic/index.d.ts +0 -0
  60. /package/{lib → dist}/openai/index.d.ts +0 -0
@@ -22,11 +22,89 @@ const getModelParams = params => {
22
22
  };
23
23
  const formatResponseOpenAI = response => {
24
24
  const output = [];
25
- for (const choice of response.choices ?? []) {
26
- if (choice.message?.content) {
25
+ if (response.choices) {
26
+ for (const choice of response.choices) {
27
+ const content = [];
28
+ let role = 'assistant';
29
+ if (choice.message) {
30
+ if (choice.message.role) {
31
+ role = choice.message.role;
32
+ }
33
+ if (choice.message.content) {
34
+ content.push({
35
+ type: 'text',
36
+ text: choice.message.content
37
+ });
38
+ }
39
+ if (choice.message.tool_calls) {
40
+ for (const toolCall of choice.message.tool_calls) {
41
+ content.push({
42
+ type: 'function',
43
+ id: toolCall.id,
44
+ function: {
45
+ name: toolCall.function.name,
46
+ arguments: toolCall.function.arguments
47
+ }
48
+ });
49
+ }
50
+ }
51
+ }
52
+ if (content.length > 0) {
53
+ output.push({
54
+ role,
55
+ content
56
+ });
57
+ }
58
+ }
59
+ }
60
+
61
+ // Handle Responses API format
62
+ if (response.output) {
63
+ const content = [];
64
+ let role = 'assistant';
65
+ for (const item of response.output) {
66
+ if (item.type === 'message') {
67
+ role = item.role;
68
+ if (item.content && Array.isArray(item.content)) {
69
+ for (const contentItem of item.content) {
70
+ if (contentItem.type === 'output_text' && contentItem.text) {
71
+ content.push({
72
+ type: 'text',
73
+ text: contentItem.text
74
+ });
75
+ } else if (contentItem.text) {
76
+ content.push({
77
+ type: 'text',
78
+ text: contentItem.text
79
+ });
80
+ } else if (contentItem.type === 'input_image' && contentItem.image_url) {
81
+ content.push({
82
+ type: 'image',
83
+ image: contentItem.image_url
84
+ });
85
+ }
86
+ }
87
+ } else if (item.content) {
88
+ content.push({
89
+ type: 'text',
90
+ text: String(item.content)
91
+ });
92
+ }
93
+ } else if (item.type === 'function_call') {
94
+ content.push({
95
+ type: 'function',
96
+ id: item.call_id || item.id || '',
97
+ function: {
98
+ name: item.name,
99
+ arguments: item.arguments || {}
100
+ }
101
+ });
102
+ }
103
+ }
104
+ if (content.length > 0) {
27
105
  output.push({
28
- role: choice.message.role,
29
- content: choice.message.content
106
+ role,
107
+ content
30
108
  });
31
109
  }
32
110
  }
@@ -35,6 +113,19 @@ const formatResponseOpenAI = response => {
35
113
  const withPrivacyMode = (client, privacyMode, input) => {
36
114
  return client.privacy_mode || privacyMode ? null : input;
37
115
  };
116
+
117
+ /**
118
+ * Extract available tool calls from the request parameters.
119
+ * These are the tools provided to the LLM, not the tool calls in the response.
120
+ */
121
+ const extractAvailableToolCalls = (provider, params) => {
122
+ {
123
+ if (params.tools) {
124
+ return params.tools;
125
+ }
126
+ return null;
127
+ }
128
+ };
38
129
  function sanitizeValues(obj) {
39
130
  if (obj === undefined || obj === null) {
40
131
  return obj;
@@ -211,6 +302,7 @@ class WrappedCompletions extends Completions {
211
302
  }
212
303
  }
213
304
  const latency = (Date.now() - startTime) / 1000;
305
+ const availableTools = extractAvailableToolCalls('openai', openAIParams);
214
306
  await sendEventToPosthog({
215
307
  client: this.phClient,
216
308
  distinctId: posthogDistinctId,
@@ -227,6 +319,7 @@ class WrappedCompletions extends Completions {
227
319
  params: body,
228
320
  httpStatus: 200,
229
321
  usage,
322
+ tools: availableTools,
230
323
  captureImmediate: posthogCaptureImmediate
231
324
  });
232
325
  } catch (error) {
@@ -262,6 +355,7 @@ class WrappedCompletions extends Completions {
262
355
  const wrappedPromise = parentPromise.then(async result => {
263
356
  if ('choices' in result) {
264
357
  const latency = (Date.now() - startTime) / 1000;
358
+ const availableTools = extractAvailableToolCalls('openai', openAIParams);
265
359
  await sendEventToPosthog({
266
360
  client: this.phClient,
267
361
  distinctId: posthogDistinctId,
@@ -280,6 +374,7 @@ class WrappedCompletions extends Completions {
280
374
  reasoningTokens: result.usage?.completion_tokens_details?.reasoning_tokens ?? 0,
281
375
  cacheReadInputTokens: result.usage?.prompt_tokens_details?.cached_tokens ?? 0
282
376
  },
377
+ tools: availableTools,
283
378
  captureImmediate: posthogCaptureImmediate
284
379
  });
285
380
  }
@@ -363,10 +458,12 @@ class WrappedResponses extends Responses {
363
458
  }
364
459
  }
365
460
  const latency = (Date.now() - startTime) / 1000;
461
+ const availableTools = extractAvailableToolCalls('openai', openAIParams);
366
462
  await sendEventToPosthog({
367
463
  client: this.phClient,
368
464
  distinctId: posthogDistinctId,
369
465
  traceId,
466
+ //@ts-expect-error
370
467
  model: openAIParams.model,
371
468
  provider: 'openai',
372
469
  input: openAIParams.input,
@@ -376,6 +473,7 @@ class WrappedResponses extends Responses {
376
473
  params: body,
377
474
  httpStatus: 200,
378
475
  usage,
476
+ tools: availableTools,
379
477
  captureImmediate: posthogCaptureImmediate
380
478
  });
381
479
  } catch (error) {
@@ -383,6 +481,7 @@ class WrappedResponses extends Responses {
383
481
  client: this.phClient,
384
482
  distinctId: posthogDistinctId,
385
483
  traceId,
484
+ //@ts-expect-error
386
485
  model: openAIParams.model,
387
486
  provider: 'openai',
388
487
  input: openAIParams.input,
@@ -409,14 +508,18 @@ class WrappedResponses extends Responses {
409
508
  const wrappedPromise = parentPromise.then(async result => {
410
509
  if ('output' in result) {
411
510
  const latency = (Date.now() - startTime) / 1000;
511
+ const availableTools = extractAvailableToolCalls('openai', openAIParams);
412
512
  await sendEventToPosthog({
413
513
  client: this.phClient,
414
514
  distinctId: posthogDistinctId,
415
515
  traceId,
516
+ //@ts-expect-error
416
517
  model: openAIParams.model,
417
518
  provider: 'openai',
418
519
  input: openAIParams.input,
419
- output: result.output,
520
+ output: formatResponseOpenAI({
521
+ output: result.output
522
+ }),
420
523
  latency,
421
524
  baseURL: this.baseURL ?? '',
422
525
  params: body,
@@ -427,6 +530,7 @@ class WrappedResponses extends Responses {
427
530
  reasoningTokens: result.usage?.output_tokens_details?.reasoning_tokens ?? 0,
428
531
  cacheReadInputTokens: result.usage?.input_tokens_details?.cached_tokens ?? 0
429
532
  },
533
+ tools: availableTools,
430
534
  captureImmediate: posthogCaptureImmediate
431
535
  });
432
536
  }
@@ -436,6 +540,7 @@ class WrappedResponses extends Responses {
436
540
  client: this.phClient,
437
541
  distinctId: posthogDistinctId,
438
542
  traceId,
543
+ //@ts-expect-error
439
544
  model: openAIParams.model,
440
545
  provider: 'openai',
441
546
  input: openAIParams.input,
@@ -484,6 +589,7 @@ class WrappedResponses extends Responses {
484
589
  client: this.phClient,
485
590
  distinctId: posthogDistinctId,
486
591
  traceId,
592
+ //@ts-expect-error
487
593
  model: openAIParams.model,
488
594
  provider: 'openai',
489
595
  input: openAIParams.input,
@@ -506,6 +612,7 @@ class WrappedResponses extends Responses {
506
612
  client: this.phClient,
507
613
  distinctId: posthogDistinctId,
508
614
  traceId,
615
+ //@ts-expect-error
509
616
  model: openAIParams.model,
510
617
  provider: 'openai',
511
618
  input: openAIParams.input,
@@ -537,5 +644,5 @@ exports.PostHogOpenAI = PostHogOpenAI;
537
644
  exports.WrappedChat = WrappedChat;
538
645
  exports.WrappedCompletions = WrappedCompletions;
539
646
  exports.WrappedResponses = WrappedResponses;
540
- exports["default"] = PostHogOpenAI;
647
+ exports.default = PostHogOpenAI;
541
648
  //# sourceMappingURL=index.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.cjs","sources":["../../src/utils.ts","../../src/openai/index.ts"],"sourcesContent":["import { PostHog } from 'posthog-node'\nimport { Buffer } from 'buffer'\nimport OpenAIOrignal from 'openai'\nimport AnthropicOriginal from '@anthropic-ai/sdk'\nimport type { ChatCompletionTool } from 'openai/resources/chat/completions'\nimport type { Tool as GeminiTool } from '@google/genai'\nimport type { FormattedMessage, FormattedContent, TokenUsage } from './types'\n\ntype ChatCompletionCreateParamsBase = OpenAIOrignal.Chat.Completions.ChatCompletionCreateParams\ntype MessageCreateParams = AnthropicOriginal.Messages.MessageCreateParams\ntype ResponseCreateParams = OpenAIOrignal.Responses.ResponseCreateParams\ntype AnthropicTool = AnthropicOriginal.Tool\n\n// limit large outputs by truncating to 200kb (approx 200k bytes)\nexport const MAX_OUTPUT_SIZE = 200000\nconst STRING_FORMAT = 'utf8'\n\nexport interface MonitoringParams {\n posthogDistinctId?: string\n posthogTraceId?: string\n posthogProperties?: Record<string, any>\n posthogPrivacyMode?: boolean\n posthogGroups?: Record<string, any>\n posthogModelOverride?: string\n posthogProviderOverride?: string\n posthogCostOverride?: CostOverride\n posthogCaptureImmediate?: boolean\n}\n\nexport interface CostOverride {\n inputCost: number\n outputCost: number\n}\n\nexport const getModelParams = (\n params: ((ChatCompletionCreateParamsBase | MessageCreateParams | ResponseCreateParams) & MonitoringParams) | null\n): Record<string, any> => {\n if (!params) {\n return {}\n }\n const modelParams: Record<string, any> = {}\n const paramKeys = [\n 'temperature',\n 'max_tokens',\n 'max_completion_tokens',\n 'top_p',\n 'frequency_penalty',\n 'presence_penalty',\n 'n',\n 'stop',\n 'stream',\n 'streaming',\n ] as const\n\n for (const key of paramKeys) {\n if (key in params && (params as any)[key] !== undefined) {\n modelParams[key] = (params as any)[key]\n }\n }\n return modelParams\n}\n\n/**\n * Helper to format responses (non-streaming) for consumption, mirroring Python's openai vs. anthropic approach.\n */\nexport const formatResponse = (response: any, provider: string): FormattedMessage[] => {\n if (!response) {\n return []\n }\n if (provider === 'anthropic') {\n return formatResponseAnthropic(response)\n } else if (provider === 'openai') {\n return formatResponseOpenAI(response)\n } else if (provider === 'gemini') {\n return formatResponseGemini(response)\n }\n return []\n}\n\nexport const formatResponseAnthropic = (response: any): FormattedMessage[] => {\n const output: FormattedMessage[] = []\n const content: FormattedContent = []\n\n for (const choice of response.content ?? []) {\n if (choice?.type === 'text' && choice?.text) {\n content.push({ type: 'text', text: choice.text })\n } else if (choice?.type === 'tool_use' && choice?.name && choice?.id) {\n content.push({\n type: 'function',\n id: choice.id,\n function: {\n name: choice.name,\n arguments: choice.input || {},\n },\n })\n }\n }\n\n if (content.length > 0) {\n output.push({\n role: 'assistant',\n content,\n })\n }\n\n return output\n}\n\nexport const formatResponseOpenAI = (response: any): FormattedMessage[] => {\n const output: FormattedMessage[] = []\n\n if (response.choices) {\n for (const choice of response.choices) {\n const content: FormattedContent = []\n let role = 'assistant'\n\n if (choice.message) {\n if (choice.message.role) {\n role = choice.message.role\n }\n\n if (choice.message.content) {\n content.push({ type: 'text', text: choice.message.content })\n }\n\n if (choice.message.tool_calls) {\n for (const toolCall of choice.message.tool_calls) {\n content.push({\n type: 'function',\n id: toolCall.id,\n function: {\n name: toolCall.function.name,\n arguments: toolCall.function.arguments,\n },\n })\n }\n }\n }\n\n if (content.length > 0) {\n output.push({\n role,\n content,\n })\n }\n }\n }\n\n // Handle Responses API format\n if (response.output) {\n const content: FormattedContent = []\n let role = 'assistant'\n\n for (const item of response.output) {\n if (item.type === 'message') {\n role = item.role\n\n if (item.content && Array.isArray(item.content)) {\n for (const contentItem of item.content) {\n if (contentItem.type === 'output_text' && contentItem.text) {\n content.push({ type: 'text', text: contentItem.text })\n } else if (contentItem.text) {\n content.push({ type: 'text', text: contentItem.text })\n } else if (contentItem.type === 'input_image' && contentItem.image_url) {\n content.push({\n type: 'image',\n image: contentItem.image_url,\n })\n }\n }\n } else if (item.content) {\n content.push({ type: 'text', text: String(item.content) })\n }\n } else if (item.type === 'function_call') {\n content.push({\n type: 'function',\n id: item.call_id || item.id || '',\n function: {\n name: item.name,\n arguments: item.arguments || {},\n },\n })\n }\n }\n\n if (content.length > 0) {\n output.push({\n role,\n content,\n })\n }\n }\n\n return output\n}\n\nexport const formatResponseGemini = (response: any): FormattedMessage[] => {\n const output: FormattedMessage[] = []\n\n if (response.candidates && Array.isArray(response.candidates)) {\n for (const candidate of response.candidates) {\n if (candidate.content && candidate.content.parts) {\n const content: FormattedContent = []\n\n for (const part of candidate.content.parts) {\n if (part.text) {\n content.push({ type: 'text', text: part.text })\n } else if (part.functionCall) {\n content.push({\n type: 'function',\n function: {\n name: part.functionCall.name,\n arguments: part.functionCall.args,\n },\n })\n }\n }\n\n if (content.length > 0) {\n output.push({\n role: 'assistant',\n content,\n })\n }\n } else if (candidate.text) {\n output.push({\n role: 'assistant',\n content: [{ type: 'text', text: candidate.text }],\n })\n }\n }\n } else if (response.text) {\n output.push({\n role: 'assistant',\n content: [{ type: 'text', text: response.text }],\n })\n }\n\n return output\n}\n\nexport const mergeSystemPrompt = (params: MessageCreateParams & MonitoringParams, provider: string): any => {\n if (provider == 'anthropic') {\n const messages = params.messages || []\n if (!(params as any).system) {\n return messages\n }\n const systemMessage = (params as any).system\n return [{ role: 'system', content: systemMessage }, ...messages]\n }\n return params.messages\n}\n\nexport const withPrivacyMode = (client: PostHog, privacyMode: boolean, input: any): any => {\n return (client as any).privacy_mode || privacyMode ? null : input\n}\n\nexport const truncate = (str: string): string => {\n try {\n const buffer = Buffer.from(str, STRING_FORMAT)\n if (buffer.length <= MAX_OUTPUT_SIZE) {\n return str\n }\n const truncatedBuffer = buffer.slice(0, MAX_OUTPUT_SIZE)\n return `${truncatedBuffer.toString(STRING_FORMAT)}... [truncated]`\n } catch (error) {\n console.error('Error truncating, likely not a string')\n return str\n }\n}\n\n/**\n * Extract available tool calls from the request parameters.\n * These are the tools provided to the LLM, not the tool calls in the response.\n */\nexport const extractAvailableToolCalls = (\n provider: string,\n params: any\n): ChatCompletionTool[] | AnthropicTool[] | GeminiTool[] | null => {\n if (provider === 'anthropic') {\n if (params.tools) {\n return params.tools\n }\n\n return null\n } else if (provider === 'gemini') {\n if (params.config && params.config.tools) {\n return params.config.tools\n }\n\n return null\n } else if (provider === 'openai') {\n if (params.tools) {\n return params.tools\n }\n\n return null\n } else if (provider === 'vercel') {\n // Vercel AI SDK stores tools in params.mode.tools when mode type is 'regular'\n if (params.mode?.type === 'regular' && params.mode.tools) {\n return params.mode.tools\n }\n\n return null\n }\n\n return null\n}\n\nexport type SendEventToPosthogParams = {\n client: PostHog\n distinctId?: string\n traceId: string\n model: string\n provider: string\n input: any\n output: any\n latency: number\n baseURL: string\n httpStatus: number\n usage?: TokenUsage\n params: (ChatCompletionCreateParamsBase | MessageCreateParams | ResponseCreateParams) & MonitoringParams\n isError?: boolean\n error?: string\n tools?: ChatCompletionTool[] | AnthropicTool[] | GeminiTool[] | null\n captureImmediate?: boolean\n}\n\nfunction sanitizeValues(obj: any): any {\n if (obj === undefined || obj === null) {\n return obj\n }\n const jsonSafe = JSON.parse(JSON.stringify(obj))\n if (typeof jsonSafe === 'string') {\n return Buffer.from(jsonSafe, STRING_FORMAT).toString(STRING_FORMAT)\n } else if (Array.isArray(jsonSafe)) {\n return jsonSafe.map(sanitizeValues)\n } else if (jsonSafe && typeof jsonSafe === 'object') {\n return Object.fromEntries(Object.entries(jsonSafe).map(([k, v]) => [k, sanitizeValues(v)]))\n }\n return jsonSafe\n}\n\nexport const sendEventToPosthog = async ({\n client,\n distinctId,\n traceId,\n model,\n provider,\n input,\n output,\n latency,\n baseURL,\n params,\n httpStatus = 200,\n usage = {},\n isError = false,\n error,\n tools,\n captureImmediate = false,\n}: SendEventToPosthogParams): Promise<void> => {\n if (!client.capture) {\n return Promise.resolve()\n }\n // sanitize input and output for UTF-8 validity\n const safeInput = sanitizeValues(input)\n const safeOutput = sanitizeValues(output)\n const safeError = sanitizeValues(error)\n\n let errorData = {}\n if (isError) {\n errorData = {\n $ai_is_error: true,\n $ai_error: safeError,\n }\n }\n let costOverrideData = {}\n if (params.posthogCostOverride) {\n const inputCostUSD = (params.posthogCostOverride.inputCost ?? 0) * (usage.inputTokens ?? 0)\n const outputCostUSD = (params.posthogCostOverride.outputCost ?? 0) * (usage.outputTokens ?? 0)\n costOverrideData = {\n $ai_input_cost_usd: inputCostUSD,\n $ai_output_cost_usd: outputCostUSD,\n $ai_total_cost_usd: inputCostUSD + outputCostUSD,\n }\n }\n\n const additionalTokenValues = {\n ...(usage.reasoningTokens ? { $ai_reasoning_tokens: usage.reasoningTokens } : {}),\n ...(usage.cacheReadInputTokens ? { $ai_cache_read_input_tokens: usage.cacheReadInputTokens } : {}),\n ...(usage.cacheCreationInputTokens ? { $ai_cache_creation_input_tokens: usage.cacheCreationInputTokens } : {}),\n }\n\n const properties = {\n $ai_provider: params.posthogProviderOverride ?? provider,\n $ai_model: params.posthogModelOverride ?? model,\n $ai_model_parameters: getModelParams(params),\n $ai_input: withPrivacyMode(client, params.posthogPrivacyMode ?? false, safeInput),\n $ai_output_choices: withPrivacyMode(client, params.posthogPrivacyMode ?? false, safeOutput),\n $ai_http_status: httpStatus,\n $ai_input_tokens: usage.inputTokens ?? 0,\n $ai_output_tokens: usage.outputTokens ?? 0,\n ...additionalTokenValues,\n $ai_latency: latency,\n $ai_trace_id: traceId,\n $ai_base_url: baseURL,\n ...params.posthogProperties,\n ...(distinctId ? {} : { $process_person_profile: false }),\n ...(tools ? { $ai_tools: tools } : {}),\n ...errorData,\n ...costOverrideData,\n }\n\n const event = {\n distinctId: distinctId ?? traceId,\n event: '$ai_generation',\n properties,\n groups: params.posthogGroups,\n }\n\n if (captureImmediate) {\n // await capture promise to send single event in serverless environments\n await client.captureImmediate(event)\n } else {\n client.capture(event)\n }\n}\n","import { OpenAI as OpenAIOrignal, ClientOptions } from 'openai'\nimport { PostHog } from 'posthog-node'\nimport { v4 as uuidv4 } from 'uuid'\nimport { formatResponseOpenAI, MonitoringParams, sendEventToPosthog, extractAvailableToolCalls } from '../utils'\nimport type { APIPromise } from 'openai'\nimport type { Stream } from 'openai/streaming'\nimport type { ParsedResponse } from 'openai/resources/responses/responses'\n\nconst Chat = OpenAIOrignal.Chat\nconst Completions = Chat.Completions\nconst Responses = OpenAIOrignal.Responses\n\ntype ChatCompletion = OpenAIOrignal.ChatCompletion\ntype ChatCompletionChunk = OpenAIOrignal.ChatCompletionChunk\ntype ChatCompletionCreateParamsBase = OpenAIOrignal.Chat.Completions.ChatCompletionCreateParams\ntype ChatCompletionCreateParamsNonStreaming = OpenAIOrignal.Chat.Completions.ChatCompletionCreateParamsNonStreaming\ntype ChatCompletionCreateParamsStreaming = OpenAIOrignal.Chat.Completions.ChatCompletionCreateParamsStreaming\ntype ResponsesCreateParamsBase = OpenAIOrignal.Responses.ResponseCreateParams\ntype ResponsesCreateParamsNonStreaming = OpenAIOrignal.Responses.ResponseCreateParamsNonStreaming\ntype ResponsesCreateParamsStreaming = OpenAIOrignal.Responses.ResponseCreateParamsStreaming\n\ninterface MonitoringOpenAIConfig extends ClientOptions {\n apiKey: string\n posthog: PostHog\n baseURL?: string\n}\n\ntype RequestOptions = Record<string, any>\n\nexport class PostHogOpenAI extends OpenAIOrignal {\n private readonly phClient: PostHog\n public chat: WrappedChat\n public responses: WrappedResponses\n\n constructor(config: MonitoringOpenAIConfig) {\n const { posthog, ...openAIConfig } = config\n super(openAIConfig)\n this.phClient = posthog\n this.chat = new WrappedChat(this, this.phClient)\n this.responses = new WrappedResponses(this, this.phClient)\n }\n}\n\nexport class WrappedChat extends Chat {\n constructor(parentClient: PostHogOpenAI, phClient: PostHog) {\n super(parentClient)\n this.completions = new WrappedCompletions(parentClient, phClient)\n }\n\n public completions: WrappedCompletions\n}\n\nexport class WrappedCompletions extends Completions {\n private readonly phClient: PostHog\n\n constructor(client: OpenAIOrignal, phClient: PostHog) {\n super(client)\n this.phClient = phClient\n }\n\n // --- Overload #1: Non-streaming\n public create(\n body: ChatCompletionCreateParamsNonStreaming & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<ChatCompletion>\n\n // --- Overload #2: Streaming\n public create(\n body: ChatCompletionCreateParamsStreaming & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<Stream<ChatCompletionChunk>>\n\n // --- Overload #3: Generic base\n public create(\n body: ChatCompletionCreateParamsBase & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<ChatCompletion | Stream<ChatCompletionChunk>>\n\n // --- Implementation Signature\n public create(\n body: ChatCompletionCreateParamsBase & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<ChatCompletion | Stream<ChatCompletionChunk>> {\n const {\n posthogDistinctId,\n posthogTraceId,\n posthogProperties,\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n posthogPrivacyMode = false,\n posthogGroups,\n posthogCaptureImmediate,\n ...openAIParams\n } = body\n\n const traceId = posthogTraceId ?? uuidv4()\n const startTime = Date.now()\n\n const parentPromise = super.create(openAIParams, options)\n\n if (openAIParams.stream) {\n return parentPromise.then((value) => {\n if ('tee' in value) {\n const [stream1, stream2] = value.tee()\n ;(async () => {\n try {\n let accumulatedContent = ''\n let usage: {\n inputTokens?: number\n outputTokens?: number\n reasoningTokens?: number\n cacheReadInputTokens?: number\n } = {\n inputTokens: 0,\n outputTokens: 0,\n }\n\n for await (const chunk of stream1) {\n const delta = chunk?.choices?.[0]?.delta?.content ?? ''\n accumulatedContent += delta\n if (chunk.usage) {\n usage = {\n inputTokens: chunk.usage.prompt_tokens ?? 0,\n outputTokens: chunk.usage.completion_tokens ?? 0,\n reasoningTokens: chunk.usage.completion_tokens_details?.reasoning_tokens ?? 0,\n cacheReadInputTokens: chunk.usage.prompt_tokens_details?.cached_tokens ?? 0,\n }\n }\n }\n\n const latency = (Date.now() - startTime) / 1000\n const availableTools = extractAvailableToolCalls('openai', openAIParams)\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n model: openAIParams.model,\n provider: 'openai',\n input: openAIParams.messages,\n output: [{ content: accumulatedContent, role: 'assistant' }],\n latency,\n baseURL: (this as any).baseURL ?? '',\n params: body,\n httpStatus: 200,\n usage,\n tools: availableTools,\n captureImmediate: posthogCaptureImmediate,\n })\n } catch (error: any) {\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n model: openAIParams.model,\n provider: 'openai',\n input: openAIParams.messages,\n output: [],\n latency: 0,\n baseURL: (this as any).baseURL ?? '',\n params: body,\n httpStatus: error?.status ? error.status : 500,\n usage: { inputTokens: 0, outputTokens: 0 },\n isError: true,\n error: JSON.stringify(error),\n captureImmediate: posthogCaptureImmediate,\n })\n }\n })()\n\n // Return the other stream to the user\n return stream2\n }\n return value\n }) as APIPromise<Stream<ChatCompletionChunk>>\n } else {\n const wrappedPromise = parentPromise.then(\n async (result) => {\n if ('choices' in result) {\n const latency = (Date.now() - startTime) / 1000\n const availableTools = extractAvailableToolCalls('openai', openAIParams)\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n model: openAIParams.model,\n provider: 'openai',\n input: openAIParams.messages,\n output: formatResponseOpenAI(result),\n latency,\n baseURL: (this as any).baseURL ?? '',\n params: body,\n httpStatus: 200,\n usage: {\n inputTokens: result.usage?.prompt_tokens ?? 0,\n outputTokens: result.usage?.completion_tokens ?? 0,\n reasoningTokens: result.usage?.completion_tokens_details?.reasoning_tokens ?? 0,\n cacheReadInputTokens: result.usage?.prompt_tokens_details?.cached_tokens ?? 0,\n },\n tools: availableTools,\n captureImmediate: posthogCaptureImmediate,\n })\n }\n return result\n },\n async (error: any) => {\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n model: openAIParams.model,\n provider: 'openai',\n input: openAIParams.messages,\n output: [],\n latency: 0,\n baseURL: (this as any).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 captureImmediate: posthogCaptureImmediate,\n })\n throw error\n }\n ) as APIPromise<ChatCompletion>\n\n return wrappedPromise\n }\n }\n}\n\nexport class WrappedResponses extends Responses {\n private readonly phClient: PostHog\n\n constructor(client: OpenAIOrignal, phClient: PostHog) {\n super(client)\n this.phClient = phClient\n }\n\n // --- Overload #1: Non-streaming\n public create(\n body: ResponsesCreateParamsNonStreaming & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<OpenAIOrignal.Responses.Response>\n\n // --- Overload #2: Streaming\n public create(\n body: ResponsesCreateParamsStreaming & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<Stream<OpenAIOrignal.Responses.ResponseStreamEvent>>\n\n // --- Overload #3: Generic base\n public create(\n body: ResponsesCreateParamsBase & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<OpenAIOrignal.Responses.Response | Stream<OpenAIOrignal.Responses.ResponseStreamEvent>>\n\n // --- Implementation Signature\n public create(\n body: ResponsesCreateParamsBase & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<OpenAIOrignal.Responses.Response | Stream<OpenAIOrignal.Responses.ResponseStreamEvent>> {\n const {\n posthogDistinctId,\n posthogTraceId,\n posthogProperties,\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n posthogPrivacyMode = false,\n posthogGroups,\n posthogCaptureImmediate,\n ...openAIParams\n } = body\n\n const traceId = posthogTraceId ?? uuidv4()\n const startTime = Date.now()\n\n const parentPromise = super.create(openAIParams, options)\n\n if (openAIParams.stream) {\n return parentPromise.then((value) => {\n if ('tee' in value && typeof (value as any).tee === 'function') {\n const [stream1, stream2] = (value as any).tee()\n ;(async () => {\n try {\n let finalContent: any[] = []\n let usage: {\n inputTokens?: number\n outputTokens?: number\n reasoningTokens?: number\n cacheReadInputTokens?: number\n } = {\n inputTokens: 0,\n outputTokens: 0,\n }\n\n for await (const chunk of stream1) {\n if (\n chunk.type === 'response.completed' &&\n 'response' in chunk &&\n chunk.response?.output &&\n chunk.response.output.length > 0\n ) {\n finalContent = chunk.response.output\n }\n if ('response' in chunk && chunk.response?.usage) {\n usage = {\n inputTokens: chunk.response.usage.input_tokens ?? 0,\n outputTokens: chunk.response.usage.output_tokens ?? 0,\n reasoningTokens: chunk.response.usage.output_tokens_details?.reasoning_tokens ?? 0,\n cacheReadInputTokens: chunk.response.usage.input_tokens_details?.cached_tokens ?? 0,\n }\n }\n }\n\n const latency = (Date.now() - startTime) / 1000\n const availableTools = extractAvailableToolCalls('openai', openAIParams)\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n //@ts-expect-error\n model: openAIParams.model,\n provider: 'openai',\n input: openAIParams.input,\n output: finalContent,\n latency,\n baseURL: (this as any).baseURL ?? '',\n params: body,\n httpStatus: 200,\n usage,\n tools: availableTools,\n captureImmediate: posthogCaptureImmediate,\n })\n } catch (error: any) {\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n //@ts-expect-error\n model: openAIParams.model,\n provider: 'openai',\n input: openAIParams.input,\n output: [],\n latency: 0,\n baseURL: (this as any).baseURL ?? '',\n params: body,\n httpStatus: error?.status ? error.status : 500,\n usage: { inputTokens: 0, outputTokens: 0 },\n isError: true,\n error: JSON.stringify(error),\n captureImmediate: posthogCaptureImmediate,\n })\n }\n })()\n\n return stream2\n }\n return value\n }) as APIPromise<Stream<OpenAIOrignal.Responses.ResponseStreamEvent>>\n } else {\n const wrappedPromise = parentPromise.then(\n async (result) => {\n if ('output' in result) {\n const latency = (Date.now() - startTime) / 1000\n const availableTools = extractAvailableToolCalls('openai', openAIParams)\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n //@ts-expect-error\n model: openAIParams.model,\n provider: 'openai',\n input: openAIParams.input,\n output: formatResponseOpenAI({ output: result.output }),\n latency,\n baseURL: (this as any).baseURL ?? '',\n params: body,\n httpStatus: 200,\n usage: {\n inputTokens: result.usage?.input_tokens ?? 0,\n outputTokens: result.usage?.output_tokens ?? 0,\n reasoningTokens: result.usage?.output_tokens_details?.reasoning_tokens ?? 0,\n cacheReadInputTokens: result.usage?.input_tokens_details?.cached_tokens ?? 0,\n },\n tools: availableTools,\n captureImmediate: posthogCaptureImmediate,\n })\n }\n return result\n },\n async (error: any) => {\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n //@ts-expect-error\n model: openAIParams.model,\n provider: 'openai',\n input: openAIParams.input,\n output: [],\n latency: 0,\n baseURL: (this as any).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 captureImmediate: posthogCaptureImmediate,\n })\n throw error\n }\n ) as APIPromise<OpenAIOrignal.Responses.Response>\n\n return wrappedPromise\n }\n }\n\n public parse<Params extends ResponsesCreateParamsBase, ParsedT = any>(\n body: Params & MonitoringParams,\n options?: RequestOptions\n ): APIPromise<ParsedResponse<ParsedT>> {\n const {\n posthogDistinctId,\n posthogTraceId,\n posthogProperties,\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n posthogPrivacyMode = false,\n posthogGroups,\n posthogCaptureImmediate,\n ...openAIParams\n } = body\n\n const traceId = posthogTraceId ?? uuidv4()\n const startTime = Date.now()\n\n // Create a temporary instance that bypasses our wrapped create method\n const originalCreate = super.create.bind(this)\n const originalSelf = this as any\n const tempCreate = originalSelf.create\n originalSelf.create = originalCreate\n\n try {\n const parentPromise = super.parse(openAIParams, options)\n\n const wrappedPromise = parentPromise.then(\n async (result) => {\n const latency = (Date.now() - startTime) / 1000\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n //@ts-expect-error\n model: openAIParams.model,\n provider: 'openai',\n input: openAIParams.input,\n output: result.output,\n latency,\n baseURL: (this as any).baseURL ?? '',\n params: body,\n httpStatus: 200,\n usage: {\n inputTokens: result.usage?.input_tokens ?? 0,\n outputTokens: result.usage?.output_tokens ?? 0,\n reasoningTokens: result.usage?.output_tokens_details?.reasoning_tokens ?? 0,\n cacheReadInputTokens: result.usage?.input_tokens_details?.cached_tokens ?? 0,\n },\n captureImmediate: posthogCaptureImmediate,\n })\n return result\n },\n async (error: any) => {\n await sendEventToPosthog({\n client: this.phClient,\n distinctId: posthogDistinctId,\n traceId,\n //@ts-expect-error\n model: openAIParams.model,\n provider: 'openai',\n input: openAIParams.input,\n output: [],\n latency: 0,\n baseURL: (this as any).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 captureImmediate: posthogCaptureImmediate,\n })\n throw error\n }\n )\n\n return wrappedPromise as APIPromise<ParsedResponse<ParsedT>>\n } finally {\n // Restore our wrapped create method\n originalSelf.create = tempCreate\n }\n }\n}\n\nexport default PostHogOpenAI\n\nexport { PostHogOpenAI as OpenAI }\n"],"names":["STRING_FORMAT","getModelParams","params","modelParams","paramKeys","key","undefined","formatResponseOpenAI","response","output","choices","choice","content","role","message","push","type","text","tool_calls","toolCall","id","function","name","arguments","length","item","Array","isArray","contentItem","image_url","image","String","call_id","withPrivacyMode","client","privacyMode","input","privacy_mode","extractAvailableToolCalls","provider","tools","sanitizeValues","obj","jsonSafe","JSON","parse","stringify","Buffer","from","toString","map","Object","fromEntries","entries","k","v","sendEventToPosthog","distinctId","traceId","model","latency","baseURL","httpStatus","usage","isError","error","captureImmediate","capture","Promise","resolve","safeInput","safeOutput","safeError","errorData","$ai_is_error","$ai_error","costOverrideData","posthogCostOverride","inputCostUSD","inputCost","inputTokens","outputCostUSD","outputCost","outputTokens","$ai_input_cost_usd","$ai_output_cost_usd","$ai_total_cost_usd","additionalTokenValues","reasoningTokens","$ai_reasoning_tokens","cacheReadInputTokens","$ai_cache_read_input_tokens","cacheCreationInputTokens","$ai_cache_creation_input_tokens","properties","$ai_provider","posthogProviderOverride","$ai_model","posthogModelOverride","$ai_model_parameters","$ai_input","posthogPrivacyMode","$ai_output_choices","$ai_http_status","$ai_input_tokens","$ai_output_tokens","$ai_latency","$ai_trace_id","$ai_base_url","posthogProperties","$process_person_profile","$ai_tools","event","groups","posthogGroups","Chat","OpenAIOrignal","Completions","Responses","PostHogOpenAI","constructor","config","posthog","openAIConfig","phClient","chat","WrappedChat","responses","WrappedResponses","parentClient","completions","WrappedCompletions","create","body","options","posthogDistinctId","posthogTraceId","posthogCaptureImmediate","openAIParams","uuidv4","startTime","Date","now","parentPromise","stream","then","value","stream1","stream2","tee","accumulatedContent","chunk","delta","prompt_tokens","completion_tokens","completion_tokens_details","reasoning_tokens","prompt_tokens_details","cached_tokens","availableTools","messages","status","wrappedPromise","result","finalContent","input_tokens","output_tokens","output_tokens_details","input_tokens_details","originalCreate","bind","originalSelf","tempCreate"],"mappings":"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"}
@@ -18,11 +18,89 @@ const getModelParams = params => {
18
18
  };
19
19
  const formatResponseOpenAI = response => {
20
20
  const output = [];
21
- for (const choice of response.choices ?? []) {
22
- if (choice.message?.content) {
21
+ if (response.choices) {
22
+ for (const choice of response.choices) {
23
+ const content = [];
24
+ let role = 'assistant';
25
+ if (choice.message) {
26
+ if (choice.message.role) {
27
+ role = choice.message.role;
28
+ }
29
+ if (choice.message.content) {
30
+ content.push({
31
+ type: 'text',
32
+ text: choice.message.content
33
+ });
34
+ }
35
+ if (choice.message.tool_calls) {
36
+ for (const toolCall of choice.message.tool_calls) {
37
+ content.push({
38
+ type: 'function',
39
+ id: toolCall.id,
40
+ function: {
41
+ name: toolCall.function.name,
42
+ arguments: toolCall.function.arguments
43
+ }
44
+ });
45
+ }
46
+ }
47
+ }
48
+ if (content.length > 0) {
49
+ output.push({
50
+ role,
51
+ content
52
+ });
53
+ }
54
+ }
55
+ }
56
+
57
+ // Handle Responses API format
58
+ if (response.output) {
59
+ const content = [];
60
+ let role = 'assistant';
61
+ for (const item of response.output) {
62
+ if (item.type === 'message') {
63
+ role = item.role;
64
+ if (item.content && Array.isArray(item.content)) {
65
+ for (const contentItem of item.content) {
66
+ if (contentItem.type === 'output_text' && contentItem.text) {
67
+ content.push({
68
+ type: 'text',
69
+ text: contentItem.text
70
+ });
71
+ } else if (contentItem.text) {
72
+ content.push({
73
+ type: 'text',
74
+ text: contentItem.text
75
+ });
76
+ } else if (contentItem.type === 'input_image' && contentItem.image_url) {
77
+ content.push({
78
+ type: 'image',
79
+ image: contentItem.image_url
80
+ });
81
+ }
82
+ }
83
+ } else if (item.content) {
84
+ content.push({
85
+ type: 'text',
86
+ text: String(item.content)
87
+ });
88
+ }
89
+ } else if (item.type === 'function_call') {
90
+ content.push({
91
+ type: 'function',
92
+ id: item.call_id || item.id || '',
93
+ function: {
94
+ name: item.name,
95
+ arguments: item.arguments || {}
96
+ }
97
+ });
98
+ }
99
+ }
100
+ if (content.length > 0) {
23
101
  output.push({
24
- role: choice.message.role,
25
- content: choice.message.content
102
+ role,
103
+ content
26
104
  });
27
105
  }
28
106
  }
@@ -31,6 +109,19 @@ const formatResponseOpenAI = response => {
31
109
  const withPrivacyMode = (client, privacyMode, input) => {
32
110
  return client.privacy_mode || privacyMode ? null : input;
33
111
  };
112
+
113
+ /**
114
+ * Extract available tool calls from the request parameters.
115
+ * These are the tools provided to the LLM, not the tool calls in the response.
116
+ */
117
+ const extractAvailableToolCalls = (provider, params) => {
118
+ {
119
+ if (params.tools) {
120
+ return params.tools;
121
+ }
122
+ return null;
123
+ }
124
+ };
34
125
  function sanitizeValues(obj) {
35
126
  if (obj === undefined || obj === null) {
36
127
  return obj;
@@ -207,6 +298,7 @@ class WrappedCompletions extends Completions {
207
298
  }
208
299
  }
209
300
  const latency = (Date.now() - startTime) / 1000;
301
+ const availableTools = extractAvailableToolCalls('openai', openAIParams);
210
302
  await sendEventToPosthog({
211
303
  client: this.phClient,
212
304
  distinctId: posthogDistinctId,
@@ -223,6 +315,7 @@ class WrappedCompletions extends Completions {
223
315
  params: body,
224
316
  httpStatus: 200,
225
317
  usage,
318
+ tools: availableTools,
226
319
  captureImmediate: posthogCaptureImmediate
227
320
  });
228
321
  } catch (error) {
@@ -258,6 +351,7 @@ class WrappedCompletions extends Completions {
258
351
  const wrappedPromise = parentPromise.then(async result => {
259
352
  if ('choices' in result) {
260
353
  const latency = (Date.now() - startTime) / 1000;
354
+ const availableTools = extractAvailableToolCalls('openai', openAIParams);
261
355
  await sendEventToPosthog({
262
356
  client: this.phClient,
263
357
  distinctId: posthogDistinctId,
@@ -276,6 +370,7 @@ class WrappedCompletions extends Completions {
276
370
  reasoningTokens: result.usage?.completion_tokens_details?.reasoning_tokens ?? 0,
277
371
  cacheReadInputTokens: result.usage?.prompt_tokens_details?.cached_tokens ?? 0
278
372
  },
373
+ tools: availableTools,
279
374
  captureImmediate: posthogCaptureImmediate
280
375
  });
281
376
  }
@@ -359,10 +454,12 @@ class WrappedResponses extends Responses {
359
454
  }
360
455
  }
361
456
  const latency = (Date.now() - startTime) / 1000;
457
+ const availableTools = extractAvailableToolCalls('openai', openAIParams);
362
458
  await sendEventToPosthog({
363
459
  client: this.phClient,
364
460
  distinctId: posthogDistinctId,
365
461
  traceId,
462
+ //@ts-expect-error
366
463
  model: openAIParams.model,
367
464
  provider: 'openai',
368
465
  input: openAIParams.input,
@@ -372,6 +469,7 @@ class WrappedResponses extends Responses {
372
469
  params: body,
373
470
  httpStatus: 200,
374
471
  usage,
472
+ tools: availableTools,
375
473
  captureImmediate: posthogCaptureImmediate
376
474
  });
377
475
  } catch (error) {
@@ -379,6 +477,7 @@ class WrappedResponses extends Responses {
379
477
  client: this.phClient,
380
478
  distinctId: posthogDistinctId,
381
479
  traceId,
480
+ //@ts-expect-error
382
481
  model: openAIParams.model,
383
482
  provider: 'openai',
384
483
  input: openAIParams.input,
@@ -405,14 +504,18 @@ class WrappedResponses extends Responses {
405
504
  const wrappedPromise = parentPromise.then(async result => {
406
505
  if ('output' in result) {
407
506
  const latency = (Date.now() - startTime) / 1000;
507
+ const availableTools = extractAvailableToolCalls('openai', openAIParams);
408
508
  await sendEventToPosthog({
409
509
  client: this.phClient,
410
510
  distinctId: posthogDistinctId,
411
511
  traceId,
512
+ //@ts-expect-error
412
513
  model: openAIParams.model,
413
514
  provider: 'openai',
414
515
  input: openAIParams.input,
415
- output: result.output,
516
+ output: formatResponseOpenAI({
517
+ output: result.output
518
+ }),
416
519
  latency,
417
520
  baseURL: this.baseURL ?? '',
418
521
  params: body,
@@ -423,6 +526,7 @@ class WrappedResponses extends Responses {
423
526
  reasoningTokens: result.usage?.output_tokens_details?.reasoning_tokens ?? 0,
424
527
  cacheReadInputTokens: result.usage?.input_tokens_details?.cached_tokens ?? 0
425
528
  },
529
+ tools: availableTools,
426
530
  captureImmediate: posthogCaptureImmediate
427
531
  });
428
532
  }
@@ -432,6 +536,7 @@ class WrappedResponses extends Responses {
432
536
  client: this.phClient,
433
537
  distinctId: posthogDistinctId,
434
538
  traceId,
539
+ //@ts-expect-error
435
540
  model: openAIParams.model,
436
541
  provider: 'openai',
437
542
  input: openAIParams.input,
@@ -480,6 +585,7 @@ class WrappedResponses extends Responses {
480
585
  client: this.phClient,
481
586
  distinctId: posthogDistinctId,
482
587
  traceId,
588
+ //@ts-expect-error
483
589
  model: openAIParams.model,
484
590
  provider: 'openai',
485
591
  input: openAIParams.input,
@@ -502,6 +608,7 @@ class WrappedResponses extends Responses {
502
608
  client: this.phClient,
503
609
  distinctId: posthogDistinctId,
504
610
  traceId,
611
+ //@ts-expect-error
505
612
  model: openAIParams.model,
506
613
  provider: 'openai',
507
614
  input: openAIParams.input,