@thinkhive/sdk 2.0.1 → 3.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/MIGRATION.md +274 -0
- package/README.md +7 -22
- package/dist/api/calibration.d.ts +168 -0
- package/dist/api/calibration.js +176 -0
- package/dist/api/claims.d.ts +262 -0
- package/dist/api/claims.js +262 -0
- package/dist/api/runs.d.ts +200 -0
- package/dist/api/runs.js +262 -0
- package/dist/core/client.d.ts +29 -0
- package/dist/core/client.js +89 -0
- package/dist/core/config.d.ts +38 -0
- package/dist/core/config.js +76 -0
- package/dist/core/types.d.ts +354 -0
- package/dist/core/types.js +8 -0
- package/dist/index.d.ts +222 -512
- package/dist/index.js +169 -394
- package/dist/instrumentation/langchain.d.ts +194 -0
- package/dist/instrumentation/langchain.js +429 -0
- package/dist/instrumentation/openai.d.ts +141 -0
- package/dist/instrumentation/openai.js +279 -0
- package/dist/integrations/customer-context.d.ts +203 -0
- package/dist/integrations/customer-context.js +274 -0
- package/dist/integrations/ticket-linking.d.ts +217 -0
- package/dist/integrations/ticket-linking.js +259 -0
- package/package.json +61 -9
|
@@ -0,0 +1,141 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ThinkHive SDK v3.0 - OpenAI Instrumentation
|
|
3
|
+
*
|
|
4
|
+
* Auto-instrumentation for OpenAI API calls including:
|
|
5
|
+
* - Chat completions
|
|
6
|
+
* - Assistants API
|
|
7
|
+
* - Tool calls
|
|
8
|
+
*/
|
|
9
|
+
import type { ConversationMessage, SpanData } from '../core/types';
|
|
10
|
+
/**
|
|
11
|
+
* Wrap an OpenAI chat completion call with tracing
|
|
12
|
+
*
|
|
13
|
+
* @example
|
|
14
|
+
* ```typescript
|
|
15
|
+
* import OpenAI from 'openai';
|
|
16
|
+
* import { wrapChatCompletion } from '@thinkhive/sdk/instrumentation/openai';
|
|
17
|
+
*
|
|
18
|
+
* const openai = new OpenAI();
|
|
19
|
+
*
|
|
20
|
+
* const result = await wrapChatCompletion(
|
|
21
|
+
* () => openai.chat.completions.create({
|
|
22
|
+
* model: 'gpt-4',
|
|
23
|
+
* messages: [{ role: 'user', content: 'Hello' }],
|
|
24
|
+
* }),
|
|
25
|
+
* {
|
|
26
|
+
* model: 'gpt-4',
|
|
27
|
+
* messages: [{ role: 'user', content: 'Hello' }],
|
|
28
|
+
* }
|
|
29
|
+
* );
|
|
30
|
+
* ```
|
|
31
|
+
*/
|
|
32
|
+
export declare function wrapChatCompletion<T>(fn: () => Promise<T>, options?: {
|
|
33
|
+
model?: string;
|
|
34
|
+
messages?: Array<{
|
|
35
|
+
role: string;
|
|
36
|
+
content: string;
|
|
37
|
+
}>;
|
|
38
|
+
temperature?: number;
|
|
39
|
+
maxTokens?: number;
|
|
40
|
+
}): Promise<T>;
|
|
41
|
+
/**
|
|
42
|
+
* Wrap an OpenAI Assistants API call with tracing
|
|
43
|
+
*
|
|
44
|
+
* @example
|
|
45
|
+
* ```typescript
|
|
46
|
+
* import OpenAI from 'openai';
|
|
47
|
+
* import { wrapAssistantRun } from '@thinkhive/sdk/instrumentation/openai';
|
|
48
|
+
*
|
|
49
|
+
* const openai = new OpenAI();
|
|
50
|
+
*
|
|
51
|
+
* const run = await wrapAssistantRun(
|
|
52
|
+
* () => openai.beta.threads.runs.create(threadId, {
|
|
53
|
+
* assistant_id: assistantId,
|
|
54
|
+
* }),
|
|
55
|
+
* { assistantId, threadId }
|
|
56
|
+
* );
|
|
57
|
+
* ```
|
|
58
|
+
*/
|
|
59
|
+
export declare function wrapAssistantRun<T>(fn: () => Promise<T>, options: {
|
|
60
|
+
assistantId: string;
|
|
61
|
+
threadId: string;
|
|
62
|
+
model?: string;
|
|
63
|
+
}): Promise<T>;
|
|
64
|
+
/**
|
|
65
|
+
* Wrap an OpenAI tool call with tracing
|
|
66
|
+
*
|
|
67
|
+
* @example
|
|
68
|
+
* ```typescript
|
|
69
|
+
* const result = await wrapToolCall(
|
|
70
|
+
* () => executeMyTool(args),
|
|
71
|
+
* {
|
|
72
|
+
* toolName: 'search_orders',
|
|
73
|
+
* toolCallId: 'call_abc123',
|
|
74
|
+
* arguments: { orderId: '12345' },
|
|
75
|
+
* }
|
|
76
|
+
* );
|
|
77
|
+
* ```
|
|
78
|
+
*/
|
|
79
|
+
export declare function wrapToolCall<T>(fn: () => Promise<T>, options: {
|
|
80
|
+
toolName: string;
|
|
81
|
+
toolCallId?: string;
|
|
82
|
+
arguments?: Record<string, unknown>;
|
|
83
|
+
}): Promise<T>;
|
|
84
|
+
/**
|
|
85
|
+
* Extract conversation messages from OpenAI format
|
|
86
|
+
*/
|
|
87
|
+
export declare function extractConversationMessages(openaiMessages: Array<{
|
|
88
|
+
role: string;
|
|
89
|
+
content: string | null;
|
|
90
|
+
tool_calls?: Array<{
|
|
91
|
+
id: string;
|
|
92
|
+
function: {
|
|
93
|
+
name: string;
|
|
94
|
+
arguments: string;
|
|
95
|
+
};
|
|
96
|
+
}>;
|
|
97
|
+
}>): ConversationMessage[];
|
|
98
|
+
/**
|
|
99
|
+
* Extract span data from OpenAI response
|
|
100
|
+
*/
|
|
101
|
+
export declare function extractSpanData(response: {
|
|
102
|
+
model?: string;
|
|
103
|
+
usage?: {
|
|
104
|
+
prompt_tokens?: number;
|
|
105
|
+
completion_tokens?: number;
|
|
106
|
+
total_tokens?: number;
|
|
107
|
+
};
|
|
108
|
+
choices?: Array<{
|
|
109
|
+
message?: {
|
|
110
|
+
tool_calls?: Array<{
|
|
111
|
+
id: string;
|
|
112
|
+
function: {
|
|
113
|
+
name: string;
|
|
114
|
+
arguments: string;
|
|
115
|
+
};
|
|
116
|
+
}>;
|
|
117
|
+
};
|
|
118
|
+
finish_reason?: string;
|
|
119
|
+
}>;
|
|
120
|
+
}, durationMs: number): SpanData;
|
|
121
|
+
/**
|
|
122
|
+
* Auto-instrument OpenAI client
|
|
123
|
+
* Call this during SDK init to automatically trace all OpenAI calls
|
|
124
|
+
*
|
|
125
|
+
* @example
|
|
126
|
+
* ```typescript
|
|
127
|
+
* import OpenAI from 'openai';
|
|
128
|
+
* import { instrumentOpenAIClient } from '@thinkhive/sdk/instrumentation/openai';
|
|
129
|
+
*
|
|
130
|
+
* const openai = new OpenAI();
|
|
131
|
+
* instrumentOpenAIClient(openai);
|
|
132
|
+
*
|
|
133
|
+
* // All subsequent calls are automatically traced
|
|
134
|
+
* const response = await openai.chat.completions.create({ ... });
|
|
135
|
+
* ```
|
|
136
|
+
*/
|
|
137
|
+
export declare function instrumentOpenAIClient(client: any): void;
|
|
138
|
+
/**
|
|
139
|
+
* Remove instrumentation from OpenAI client
|
|
140
|
+
*/
|
|
141
|
+
export declare function uninstrumentOpenAIClient(client: any, originalCreate: any): void;
|
|
@@ -0,0 +1,279 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* ThinkHive SDK v3.0 - OpenAI Instrumentation
|
|
4
|
+
*
|
|
5
|
+
* Auto-instrumentation for OpenAI API calls including:
|
|
6
|
+
* - Chat completions
|
|
7
|
+
* - Assistants API
|
|
8
|
+
* - Tool calls
|
|
9
|
+
*/
|
|
10
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
11
|
+
exports.wrapChatCompletion = wrapChatCompletion;
|
|
12
|
+
exports.wrapAssistantRun = wrapAssistantRun;
|
|
13
|
+
exports.wrapToolCall = wrapToolCall;
|
|
14
|
+
exports.extractConversationMessages = extractConversationMessages;
|
|
15
|
+
exports.extractSpanData = extractSpanData;
|
|
16
|
+
exports.instrumentOpenAIClient = instrumentOpenAIClient;
|
|
17
|
+
exports.uninstrumentOpenAIClient = uninstrumentOpenAIClient;
|
|
18
|
+
const api_1 = require("@opentelemetry/api");
|
|
19
|
+
const config_1 = require("../core/config");
|
|
20
|
+
// ============================================================================
|
|
21
|
+
// OPENAI TRACING
|
|
22
|
+
// ============================================================================
|
|
23
|
+
/**
|
|
24
|
+
* Wrap an OpenAI chat completion call with tracing
|
|
25
|
+
*
|
|
26
|
+
* @example
|
|
27
|
+
* ```typescript
|
|
28
|
+
* import OpenAI from 'openai';
|
|
29
|
+
* import { wrapChatCompletion } from '@thinkhive/sdk/instrumentation/openai';
|
|
30
|
+
*
|
|
31
|
+
* const openai = new OpenAI();
|
|
32
|
+
*
|
|
33
|
+
* const result = await wrapChatCompletion(
|
|
34
|
+
* () => openai.chat.completions.create({
|
|
35
|
+
* model: 'gpt-4',
|
|
36
|
+
* messages: [{ role: 'user', content: 'Hello' }],
|
|
37
|
+
* }),
|
|
38
|
+
* {
|
|
39
|
+
* model: 'gpt-4',
|
|
40
|
+
* messages: [{ role: 'user', content: 'Hello' }],
|
|
41
|
+
* }
|
|
42
|
+
* );
|
|
43
|
+
* ```
|
|
44
|
+
*/
|
|
45
|
+
async function wrapChatCompletion(fn, options = {}) {
|
|
46
|
+
const tracer = api_1.trace.getTracer('thinkhive', '3.0.0');
|
|
47
|
+
return tracer.startActiveSpan('openai.chat.completions', {
|
|
48
|
+
attributes: {
|
|
49
|
+
'openinference.span.kind': 'LLM',
|
|
50
|
+
'llm.provider': 'openai',
|
|
51
|
+
'llm.model_name': options.model,
|
|
52
|
+
'llm.temperature': options.temperature,
|
|
53
|
+
'llm.max_tokens': options.maxTokens,
|
|
54
|
+
'llm.input_messages': options.messages
|
|
55
|
+
? JSON.stringify(options.messages).substring(0, 10000)
|
|
56
|
+
: undefined,
|
|
57
|
+
},
|
|
58
|
+
}, async (span) => {
|
|
59
|
+
const startTime = Date.now();
|
|
60
|
+
try {
|
|
61
|
+
const result = await fn();
|
|
62
|
+
// Extract response data
|
|
63
|
+
const response = result;
|
|
64
|
+
if (response?.choices?.[0]?.message) {
|
|
65
|
+
span.setAttribute('llm.output_messages', JSON.stringify(response.choices[0].message).substring(0, 10000));
|
|
66
|
+
}
|
|
67
|
+
if (response?.usage) {
|
|
68
|
+
span.setAttribute('llm.token_count.prompt', response.usage.prompt_tokens);
|
|
69
|
+
span.setAttribute('llm.token_count.completion', response.usage.completion_tokens);
|
|
70
|
+
span.setAttribute('llm.token_count.total', response.usage.total_tokens);
|
|
71
|
+
}
|
|
72
|
+
span.setStatus({ code: api_1.SpanStatusCode.OK });
|
|
73
|
+
return result;
|
|
74
|
+
}
|
|
75
|
+
catch (error) {
|
|
76
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
77
|
+
span.setStatus({ code: api_1.SpanStatusCode.ERROR, message });
|
|
78
|
+
span.recordException(error);
|
|
79
|
+
throw error;
|
|
80
|
+
}
|
|
81
|
+
finally {
|
|
82
|
+
span.setAttribute('duration_ms', Date.now() - startTime);
|
|
83
|
+
span.end();
|
|
84
|
+
}
|
|
85
|
+
});
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Wrap an OpenAI Assistants API call with tracing
|
|
89
|
+
*
|
|
90
|
+
* @example
|
|
91
|
+
* ```typescript
|
|
92
|
+
* import OpenAI from 'openai';
|
|
93
|
+
* import { wrapAssistantRun } from '@thinkhive/sdk/instrumentation/openai';
|
|
94
|
+
*
|
|
95
|
+
* const openai = new OpenAI();
|
|
96
|
+
*
|
|
97
|
+
* const run = await wrapAssistantRun(
|
|
98
|
+
* () => openai.beta.threads.runs.create(threadId, {
|
|
99
|
+
* assistant_id: assistantId,
|
|
100
|
+
* }),
|
|
101
|
+
* { assistantId, threadId }
|
|
102
|
+
* );
|
|
103
|
+
* ```
|
|
104
|
+
*/
|
|
105
|
+
async function wrapAssistantRun(fn, options) {
|
|
106
|
+
const tracer = api_1.trace.getTracer('thinkhive', '3.0.0');
|
|
107
|
+
return tracer.startActiveSpan('openai.assistants.run', {
|
|
108
|
+
attributes: {
|
|
109
|
+
'openinference.span.kind': 'AGENT',
|
|
110
|
+
'llm.provider': 'openai',
|
|
111
|
+
'llm.model_name': options.model,
|
|
112
|
+
'assistant.id': options.assistantId,
|
|
113
|
+
'assistant.thread_id': options.threadId,
|
|
114
|
+
},
|
|
115
|
+
}, async (span) => {
|
|
116
|
+
const startTime = Date.now();
|
|
117
|
+
try {
|
|
118
|
+
const result = await fn();
|
|
119
|
+
const response = result;
|
|
120
|
+
if (response?.status) {
|
|
121
|
+
span.setAttribute('assistant.run_status', response.status);
|
|
122
|
+
}
|
|
123
|
+
if (response?.usage) {
|
|
124
|
+
span.setAttribute('llm.token_count.prompt', response.usage.prompt_tokens);
|
|
125
|
+
span.setAttribute('llm.token_count.completion', response.usage.completion_tokens);
|
|
126
|
+
}
|
|
127
|
+
span.setStatus({ code: api_1.SpanStatusCode.OK });
|
|
128
|
+
return result;
|
|
129
|
+
}
|
|
130
|
+
catch (error) {
|
|
131
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
132
|
+
span.setStatus({ code: api_1.SpanStatusCode.ERROR, message });
|
|
133
|
+
span.recordException(error);
|
|
134
|
+
throw error;
|
|
135
|
+
}
|
|
136
|
+
finally {
|
|
137
|
+
span.setAttribute('duration_ms', Date.now() - startTime);
|
|
138
|
+
span.end();
|
|
139
|
+
}
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
/**
|
|
143
|
+
* Wrap an OpenAI tool call with tracing
|
|
144
|
+
*
|
|
145
|
+
* @example
|
|
146
|
+
* ```typescript
|
|
147
|
+
* const result = await wrapToolCall(
|
|
148
|
+
* () => executeMyTool(args),
|
|
149
|
+
* {
|
|
150
|
+
* toolName: 'search_orders',
|
|
151
|
+
* toolCallId: 'call_abc123',
|
|
152
|
+
* arguments: { orderId: '12345' },
|
|
153
|
+
* }
|
|
154
|
+
* );
|
|
155
|
+
* ```
|
|
156
|
+
*/
|
|
157
|
+
async function wrapToolCall(fn, options) {
|
|
158
|
+
const tracer = api_1.trace.getTracer('thinkhive', '3.0.0');
|
|
159
|
+
return tracer.startActiveSpan(`tool.${options.toolName}`, {
|
|
160
|
+
attributes: {
|
|
161
|
+
'openinference.span.kind': 'TOOL',
|
|
162
|
+
'tool.name': options.toolName,
|
|
163
|
+
'tool.call_id': options.toolCallId,
|
|
164
|
+
'tool.parameters': options.arguments
|
|
165
|
+
? JSON.stringify(options.arguments).substring(0, 10000)
|
|
166
|
+
: undefined,
|
|
167
|
+
},
|
|
168
|
+
}, async (span) => {
|
|
169
|
+
const startTime = Date.now();
|
|
170
|
+
try {
|
|
171
|
+
const result = await fn();
|
|
172
|
+
span.setAttribute('tool.output', JSON.stringify(result).substring(0, 10000));
|
|
173
|
+
span.setStatus({ code: api_1.SpanStatusCode.OK });
|
|
174
|
+
return result;
|
|
175
|
+
}
|
|
176
|
+
catch (error) {
|
|
177
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
178
|
+
span.setStatus({ code: api_1.SpanStatusCode.ERROR, message });
|
|
179
|
+
span.recordException(error);
|
|
180
|
+
throw error;
|
|
181
|
+
}
|
|
182
|
+
finally {
|
|
183
|
+
span.setAttribute('duration_ms', Date.now() - startTime);
|
|
184
|
+
span.end();
|
|
185
|
+
}
|
|
186
|
+
});
|
|
187
|
+
}
|
|
188
|
+
// ============================================================================
|
|
189
|
+
// EXTRACTION HELPERS
|
|
190
|
+
// ============================================================================
|
|
191
|
+
/**
|
|
192
|
+
* Extract conversation messages from OpenAI format
|
|
193
|
+
*/
|
|
194
|
+
function extractConversationMessages(openaiMessages) {
|
|
195
|
+
return openaiMessages.map((m) => ({
|
|
196
|
+
role: m.role,
|
|
197
|
+
content: m.content || '',
|
|
198
|
+
metadata: m.tool_calls
|
|
199
|
+
? {
|
|
200
|
+
tool_calls: m.tool_calls.map((tc) => ({
|
|
201
|
+
id: tc.id,
|
|
202
|
+
name: tc.function.name,
|
|
203
|
+
arguments: tc.function.arguments,
|
|
204
|
+
})),
|
|
205
|
+
}
|
|
206
|
+
: undefined,
|
|
207
|
+
}));
|
|
208
|
+
}
|
|
209
|
+
/**
|
|
210
|
+
* Extract span data from OpenAI response
|
|
211
|
+
*/
|
|
212
|
+
function extractSpanData(response, durationMs) {
|
|
213
|
+
const span = {
|
|
214
|
+
name: 'openai.chat.completions',
|
|
215
|
+
type: 'llm',
|
|
216
|
+
durationMs,
|
|
217
|
+
model: response.model,
|
|
218
|
+
provider: 'openai',
|
|
219
|
+
promptTokens: response.usage?.prompt_tokens,
|
|
220
|
+
completionTokens: response.usage?.completion_tokens,
|
|
221
|
+
status: 'ok',
|
|
222
|
+
};
|
|
223
|
+
// Extract tool calls as children
|
|
224
|
+
const toolCalls = response.choices?.[0]?.message?.tool_calls;
|
|
225
|
+
if (toolCalls && toolCalls.length > 0) {
|
|
226
|
+
span.children = toolCalls.map((tc) => ({
|
|
227
|
+
name: tc.function.name,
|
|
228
|
+
type: 'tool',
|
|
229
|
+
toolName: tc.function.name,
|
|
230
|
+
toolParameters: JSON.parse(tc.function.arguments),
|
|
231
|
+
}));
|
|
232
|
+
}
|
|
233
|
+
return span;
|
|
234
|
+
}
|
|
235
|
+
// ============================================================================
|
|
236
|
+
// AUTO-INSTRUMENTATION
|
|
237
|
+
// ============================================================================
|
|
238
|
+
/**
|
|
239
|
+
* Auto-instrument OpenAI client
|
|
240
|
+
* Call this during SDK init to automatically trace all OpenAI calls
|
|
241
|
+
*
|
|
242
|
+
* @example
|
|
243
|
+
* ```typescript
|
|
244
|
+
* import OpenAI from 'openai';
|
|
245
|
+
* import { instrumentOpenAIClient } from '@thinkhive/sdk/instrumentation/openai';
|
|
246
|
+
*
|
|
247
|
+
* const openai = new OpenAI();
|
|
248
|
+
* instrumentOpenAIClient(openai);
|
|
249
|
+
*
|
|
250
|
+
* // All subsequent calls are automatically traced
|
|
251
|
+
* const response = await openai.chat.completions.create({ ... });
|
|
252
|
+
* ```
|
|
253
|
+
*/
|
|
254
|
+
function instrumentOpenAIClient(client) {
|
|
255
|
+
if (!client || !client.chat?.completions?.create) {
|
|
256
|
+
(0, config_1.debugLog)('OpenAI client not found or incompatible');
|
|
257
|
+
return;
|
|
258
|
+
}
|
|
259
|
+
const originalCreate = client.chat.completions.create.bind(client.chat.completions);
|
|
260
|
+
client.chat.completions.create = async function (params, options) {
|
|
261
|
+
return wrapChatCompletion(() => originalCreate(params, options), {
|
|
262
|
+
model: params.model,
|
|
263
|
+
messages: params.messages,
|
|
264
|
+
temperature: params.temperature,
|
|
265
|
+
maxTokens: params.max_tokens,
|
|
266
|
+
});
|
|
267
|
+
};
|
|
268
|
+
(0, config_1.debugLog)('OpenAI client instrumented');
|
|
269
|
+
}
|
|
270
|
+
/**
|
|
271
|
+
* Remove instrumentation from OpenAI client
|
|
272
|
+
*/
|
|
273
|
+
function uninstrumentOpenAIClient(client, originalCreate) {
|
|
274
|
+
if (client?.chat?.completions) {
|
|
275
|
+
client.chat.completions.create = originalCreate;
|
|
276
|
+
(0, config_1.debugLog)('OpenAI client uninstrumented');
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"openai.js","sourceRoot":"","sources":["../../src/instrumentation/openai.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ThinkHive SDK v3.0 - OpenAI Instrumentation\n *\n * Auto-instrumentation for OpenAI API calls including:\n * - Chat completions\n * - Assistants API\n * - Tool calls\n */\n\nimport { trace, context, SpanStatusCode, type Span } from '@opentelemetry/api';\nimport { getConfig, debugLog } from '../core/config';\nimport type { ConversationMessage, SpanData } from '../core/types';\n\n// ============================================================================\n// OPENAI TRACING\n// ============================================================================\n\n/**\n * Wrap an OpenAI chat completion call with tracing\n *\n * @example\n * ```typescript\n * import OpenAI from 'openai';\n * import { wrapChatCompletion } from '@thinkhive/sdk/instrumentation/openai';\n *\n * const openai = new OpenAI();\n *\n * const result = await wrapChatCompletion(\n *   () => openai.chat.completions.create({\n *     model: 'gpt-4',\n *     messages: [{ role: 'user', content: 'Hello' }],\n *   }),\n *   {\n *     model: 'gpt-4',\n *     messages: [{ role: 'user', content: 'Hello' }],\n *   }\n * );\n * ```\n */\nexport async function wrapChatCompletion<T>(\n  fn: () => Promise<T>,\n  options: {\n    model?: string;\n    messages?: Array<{ role: string; content: string }>;\n    temperature?: number;\n    maxTokens?: number;\n  } = {}\n): Promise<T> {\n  const tracer = trace.getTracer('thinkhive', '3.0.0');\n\n  return tracer.startActiveSpan(\n    'openai.chat.completions',\n    {\n      attributes: {\n        'openinference.span.kind': 'LLM',\n        'llm.provider': 'openai',\n        'llm.model_name': options.model,\n        'llm.temperature': options.temperature,\n        'llm.max_tokens': options.maxTokens,\n        'llm.input_messages': options.messages\n          ? JSON.stringify(options.messages).substring(0, 10000)\n          : undefined,\n      },\n    },\n    async (span) => {\n      const startTime = Date.now();\n      try {\n        const result = await fn();\n\n        // Extract response data\n        const response = result as any;\n        if (response?.choices?.[0]?.message) {\n          span.setAttribute(\n            'llm.output_messages',\n            JSON.stringify(response.choices[0].message).substring(0, 10000)\n          );\n        }\n        if (response?.usage) {\n          span.setAttribute('llm.token_count.prompt', response.usage.prompt_tokens);\n          span.setAttribute('llm.token_count.completion', response.usage.completion_tokens);\n          span.setAttribute('llm.token_count.total', response.usage.total_tokens);\n        }\n\n        span.setStatus({ code: SpanStatusCode.OK });\n        return result;\n      } catch (error: unknown) {\n        const message = error instanceof Error ? error.message : String(error);\n        span.setStatus({ code: SpanStatusCode.ERROR, message });\n        span.recordException(error as Error);\n        throw error;\n      } finally {\n        span.setAttribute('duration_ms', Date.now() - startTime);\n        span.end();\n      }\n    }\n  );\n}\n\n/**\n * Wrap an OpenAI Assistants API call with tracing\n *\n * @example\n * ```typescript\n * import OpenAI from 'openai';\n * import { wrapAssistantRun } from '@thinkhive/sdk/instrumentation/openai';\n *\n * const openai = new OpenAI();\n *\n * const run = await wrapAssistantRun(\n *   () => openai.beta.threads.runs.create(threadId, {\n *     assistant_id: assistantId,\n *   }),\n *   { assistantId, threadId }\n * );\n * ```\n */\nexport async function wrapAssistantRun<T>(\n  fn: () => Promise<T>,\n  options: {\n    assistantId: string;\n    threadId: string;\n    model?: string;\n  }\n): Promise<T> {\n  const tracer = trace.getTracer('thinkhive', '3.0.0');\n\n  return tracer.startActiveSpan(\n    'openai.assistants.run',\n    {\n      attributes: {\n        'openinference.span.kind': 'AGENT',\n        'llm.provider': 'openai',\n        'llm.model_name': options.model,\n        'assistant.id': options.assistantId,\n        'assistant.thread_id': options.threadId,\n      },\n    },\n    async (span) => {\n      const startTime = Date.now();\n      try {\n        const result = await fn();\n\n        const response = result as any;\n        if (response?.status) {\n          span.setAttribute('assistant.run_status', response.status);\n        }\n        if (response?.usage) {\n          span.setAttribute('llm.token_count.prompt', response.usage.prompt_tokens);\n          span.setAttribute('llm.token_count.completion', response.usage.completion_tokens);\n        }\n\n        span.setStatus({ code: SpanStatusCode.OK });\n        return result;\n      } catch (error: unknown) {\n        const message = error instanceof Error ? error.message : String(error);\n        span.setStatus({ code: SpanStatusCode.ERROR, message });\n        span.recordException(error as Error);\n        throw error;\n      } finally {\n        span.setAttribute('duration_ms', Date.now() - startTime);\n        span.end();\n      }\n    }\n  );\n}\n\n/**\n * Wrap an OpenAI tool call with tracing\n *\n * @example\n * ```typescript\n * const result = await wrapToolCall(\n *   () => executeMyTool(args),\n *   {\n *     toolName: 'search_orders',\n *     toolCallId: 'call_abc123',\n *     arguments: { orderId: '12345' },\n *   }\n * );\n * ```\n */\nexport async function wrapToolCall<T>(\n  fn: () => Promise<T>,\n  options: {\n    toolName: string;\n    toolCallId?: string;\n    arguments?: Record<string, unknown>;\n  }\n): Promise<T> {\n  const tracer = trace.getTracer('thinkhive', '3.0.0');\n\n  return tracer.startActiveSpan(\n    `tool.${options.toolName}`,\n    {\n      attributes: {\n        'openinference.span.kind': 'TOOL',\n        'tool.name': options.toolName,\n        'tool.call_id': options.toolCallId,\n        'tool.parameters': options.arguments\n          ? JSON.stringify(options.arguments).substring(0, 10000)\n          : undefined,\n      },\n    },\n    async (span) => {\n      const startTime = Date.now();\n      try {\n        const result = await fn();\n        span.setAttribute('tool.output', JSON.stringify(result).substring(0, 10000));\n        span.setStatus({ code: SpanStatusCode.OK });\n        return result;\n      } catch (error: unknown) {\n        const message = error instanceof Error ? error.message : String(error);\n        span.setStatus({ code: SpanStatusCode.ERROR, message });\n        span.recordException(error as Error);\n        throw error;\n      } finally {\n        span.setAttribute('duration_ms', Date.now() - startTime);\n        span.end();\n      }\n    }\n  );\n}\n\n// ============================================================================\n// EXTRACTION HELPERS\n// ============================================================================\n\n/**\n * Extract conversation messages from OpenAI format\n */\nexport function extractConversationMessages(\n  openaiMessages: Array<{\n    role: string;\n    content: string | null;\n    tool_calls?: Array<{\n      id: string;\n      function: { name: string; arguments: string };\n    }>;\n  }>\n): ConversationMessage[] {\n  return openaiMessages.map((m) => ({\n    role: m.role as ConversationMessage['role'],\n    content: m.content || '',\n    metadata: m.tool_calls\n      ? {\n          tool_calls: m.tool_calls.map((tc) => ({\n            id: tc.id,\n            name: tc.function.name,\n            arguments: tc.function.arguments,\n          })),\n        }\n      : undefined,\n  }));\n}\n\n/**\n * Extract span data from OpenAI response\n */\nexport function extractSpanData(\n  response: {\n    model?: string;\n    usage?: {\n      prompt_tokens?: number;\n      completion_tokens?: number;\n      total_tokens?: number;\n    };\n    choices?: Array<{\n      message?: {\n        tool_calls?: Array<{\n          id: string;\n          function: { name: string; arguments: string };\n        }>;\n      };\n      finish_reason?: string;\n    }>;\n  },\n  durationMs: number\n): SpanData {\n  const span: SpanData = {\n    name: 'openai.chat.completions',\n    type: 'llm',\n    durationMs,\n    model: response.model,\n    provider: 'openai',\n    promptTokens: response.usage?.prompt_tokens,\n    completionTokens: response.usage?.completion_tokens,\n    status: 'ok',\n  };\n\n  // Extract tool calls as children\n  const toolCalls = response.choices?.[0]?.message?.tool_calls;\n  if (toolCalls && toolCalls.length > 0) {\n    span.children = toolCalls.map((tc) => ({\n      name: tc.function.name,\n      type: 'tool' as const,\n      toolName: tc.function.name,\n      toolParameters: JSON.parse(tc.function.arguments),\n    }));\n  }\n\n  return span;\n}\n\n// ============================================================================\n// AUTO-INSTRUMENTATION\n// ============================================================================\n\n/**\n * Auto-instrument OpenAI client\n * Call this during SDK init to automatically trace all OpenAI calls\n *\n * @example\n * ```typescript\n * import OpenAI from 'openai';\n * import { instrumentOpenAIClient } from '@thinkhive/sdk/instrumentation/openai';\n *\n * const openai = new OpenAI();\n * instrumentOpenAIClient(openai);\n *\n * // All subsequent calls are automatically traced\n * const response = await openai.chat.completions.create({ ... });\n * ```\n */\nexport function instrumentOpenAIClient(client: any): void {\n  if (!client || !client.chat?.completions?.create) {\n    debugLog('OpenAI client not found or incompatible');\n    return;\n  }\n\n  const originalCreate = client.chat.completions.create.bind(client.chat.completions);\n\n  client.chat.completions.create = async function (params: any, options?: any) {\n    return wrapChatCompletion(\n      () => originalCreate(params, options),\n      {\n        model: params.model,\n        messages: params.messages,\n        temperature: params.temperature,\n        maxTokens: params.max_tokens,\n      }\n    );\n  };\n\n  debugLog('OpenAI client instrumented');\n}\n\n/**\n * Remove instrumentation from OpenAI client\n */\nexport function uninstrumentOpenAIClient(client: any, originalCreate: any): void {\n  if (client?.chat?.completions) {\n    client.chat.completions.create = originalCreate;\n    debugLog('OpenAI client uninstrumented');\n  }\n}\n"]}
|
|
@@ -0,0 +1,203 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ThinkHive SDK v3.0 - Customer Context
|
|
3
|
+
*
|
|
4
|
+
* Time-series customer metrics snapshots
|
|
5
|
+
* Captures ARR, health score, segment AS OF the run time (not current values)
|
|
6
|
+
*/
|
|
7
|
+
import type { CustomerContextSnapshot } from '../core/types';
|
|
8
|
+
export interface CustomerAccount {
|
|
9
|
+
id: string;
|
|
10
|
+
companyId: string;
|
|
11
|
+
externalId?: string;
|
|
12
|
+
externalSource?: 'salesforce' | 'hubspot' | 'zendesk' | 'intercom' | 'custom';
|
|
13
|
+
name: string;
|
|
14
|
+
domain?: string;
|
|
15
|
+
segment?: string;
|
|
16
|
+
industry?: string;
|
|
17
|
+
employeeCount?: number;
|
|
18
|
+
createdAt: string;
|
|
19
|
+
updatedAt: string;
|
|
20
|
+
}
|
|
21
|
+
export interface CustomerMetricsSnapshot {
|
|
22
|
+
id: string;
|
|
23
|
+
customerAccountId: string;
|
|
24
|
+
arr?: number;
|
|
25
|
+
healthScore?: number;
|
|
26
|
+
nps?: number;
|
|
27
|
+
segment?: string;
|
|
28
|
+
churnRisk?: 'low' | 'medium' | 'high';
|
|
29
|
+
capturedAt: string;
|
|
30
|
+
source?: string;
|
|
31
|
+
createdAt: string;
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Customer context API client for time-series metrics
|
|
35
|
+
*/
|
|
36
|
+
export declare const customerContext: {
|
|
37
|
+
/**
|
|
38
|
+
* Create a customer account
|
|
39
|
+
*
|
|
40
|
+
* @example
|
|
41
|
+
* ```typescript
|
|
42
|
+
* const account = await customerContext.createAccount({
|
|
43
|
+
* name: 'Acme Corp',
|
|
44
|
+
* externalId: 'sf_001234',
|
|
45
|
+
* externalSource: 'salesforce',
|
|
46
|
+
* segment: 'enterprise',
|
|
47
|
+
* });
|
|
48
|
+
* ```
|
|
49
|
+
*/
|
|
50
|
+
createAccount(input: {
|
|
51
|
+
name: string;
|
|
52
|
+
externalId?: string;
|
|
53
|
+
externalSource?: CustomerAccount["externalSource"];
|
|
54
|
+
domain?: string;
|
|
55
|
+
segment?: string;
|
|
56
|
+
industry?: string;
|
|
57
|
+
employeeCount?: number;
|
|
58
|
+
}): Promise<CustomerAccount>;
|
|
59
|
+
/**
|
|
60
|
+
* Get a customer account by ID
|
|
61
|
+
*
|
|
62
|
+
* @example
|
|
63
|
+
* ```typescript
|
|
64
|
+
* const account = await customerContext.getAccount('cust_abc123');
|
|
65
|
+
* ```
|
|
66
|
+
*/
|
|
67
|
+
getAccount(customerId: string): Promise<CustomerAccount>;
|
|
68
|
+
/**
|
|
69
|
+
* Get a customer account by external ID
|
|
70
|
+
*
|
|
71
|
+
* @example
|
|
72
|
+
* ```typescript
|
|
73
|
+
* const account = await customerContext.getAccountByExternalId(
|
|
74
|
+
* 'sf_001234',
|
|
75
|
+
* 'salesforce'
|
|
76
|
+
* );
|
|
77
|
+
* ```
|
|
78
|
+
*/
|
|
79
|
+
getAccountByExternalId(externalId: string, source: CustomerAccount["externalSource"]): Promise<CustomerAccount | null>;
|
|
80
|
+
/**
|
|
81
|
+
* Capture a metrics snapshot for a customer
|
|
82
|
+
* This creates a point-in-time record of the customer's metrics
|
|
83
|
+
*
|
|
84
|
+
* @example
|
|
85
|
+
* ```typescript
|
|
86
|
+
* // Capture current metrics
|
|
87
|
+
* const snapshot = await customerContext.captureSnapshot('cust_abc123', {
|
|
88
|
+
* arr: 120000,
|
|
89
|
+
* healthScore: 85,
|
|
90
|
+
* nps: 45,
|
|
91
|
+
* segment: 'enterprise',
|
|
92
|
+
* });
|
|
93
|
+
*
|
|
94
|
+
* // Use this snapshot in a run
|
|
95
|
+
* const run = await runs.create({
|
|
96
|
+
* agentId: 'agent_123',
|
|
97
|
+
* customerContext: {
|
|
98
|
+
* customerId: 'cust_abc123',
|
|
99
|
+
* arr: snapshot.arr,
|
|
100
|
+
* healthScore: snapshot.healthScore,
|
|
101
|
+
* capturedAt: snapshot.capturedAt,
|
|
102
|
+
* },
|
|
103
|
+
* // ...
|
|
104
|
+
* });
|
|
105
|
+
* ```
|
|
106
|
+
*/
|
|
107
|
+
captureSnapshot(customerId: string, metrics: {
|
|
108
|
+
arr?: number;
|
|
109
|
+
healthScore?: number;
|
|
110
|
+
nps?: number;
|
|
111
|
+
segment?: string;
|
|
112
|
+
churnRisk?: "low" | "medium" | "high";
|
|
113
|
+
source?: string;
|
|
114
|
+
}): Promise<CustomerMetricsSnapshot>;
|
|
115
|
+
/**
|
|
116
|
+
* Get metrics snapshots for a customer (time-series)
|
|
117
|
+
*
|
|
118
|
+
* @example
|
|
119
|
+
* ```typescript
|
|
120
|
+
* // Get last 30 days of snapshots
|
|
121
|
+
* const snapshots = await customerContext.getSnapshots('cust_abc123', {
|
|
122
|
+
* from: new Date(Date.now() - 30 * 24 * 60 * 60 * 1000).toISOString(),
|
|
123
|
+
* });
|
|
124
|
+
* ```
|
|
125
|
+
*/
|
|
126
|
+
getSnapshots(customerId: string, options?: {
|
|
127
|
+
from?: string;
|
|
128
|
+
to?: string;
|
|
129
|
+
limit?: number;
|
|
130
|
+
}): Promise<CustomerMetricsSnapshot[]>;
|
|
131
|
+
/**
|
|
132
|
+
* Get the most recent snapshot for a customer
|
|
133
|
+
*
|
|
134
|
+
* @example
|
|
135
|
+
* ```typescript
|
|
136
|
+
* const latest = await customerContext.getLatestSnapshot('cust_abc123');
|
|
137
|
+
* ```
|
|
138
|
+
*/
|
|
139
|
+
getLatestSnapshot(customerId: string): Promise<CustomerMetricsSnapshot | null>;
|
|
140
|
+
/**
|
|
141
|
+
* Get snapshot closest to a specific timestamp
|
|
142
|
+
* Useful for retroactive analysis
|
|
143
|
+
*
|
|
144
|
+
* @example
|
|
145
|
+
* ```typescript
|
|
146
|
+
* // Get metrics as of a specific date
|
|
147
|
+
* const snapshot = await customerContext.getSnapshotAsOf(
|
|
148
|
+
* 'cust_abc123',
|
|
149
|
+
* '2024-01-15T10:00:00Z'
|
|
150
|
+
* );
|
|
151
|
+
* ```
|
|
152
|
+
*/
|
|
153
|
+
getSnapshotAsOf(customerId: string, timestamp: string | Date): Promise<CustomerMetricsSnapshot | null>;
|
|
154
|
+
};
|
|
155
|
+
/**
|
|
156
|
+
* Create a CustomerContextSnapshot from a metrics snapshot
|
|
157
|
+
*/
|
|
158
|
+
export declare function toContextSnapshot(snapshot: CustomerMetricsSnapshot): CustomerContextSnapshot;
|
|
159
|
+
/**
|
|
160
|
+
* Capture metrics and create context snapshot in one call
|
|
161
|
+
*
|
|
162
|
+
* @example
|
|
163
|
+
* ```typescript
|
|
164
|
+
* const context = await captureCustomerContext('cust_abc123', {
|
|
165
|
+
* arr: 100000,
|
|
166
|
+
* healthScore: 90,
|
|
167
|
+
* segment: 'enterprise',
|
|
168
|
+
* });
|
|
169
|
+
*
|
|
170
|
+
* // Use in run
|
|
171
|
+
* const run = await runs.create({
|
|
172
|
+
* agentId: 'agent_123',
|
|
173
|
+
* customerContext: context,
|
|
174
|
+
* // ...
|
|
175
|
+
* });
|
|
176
|
+
* ```
|
|
177
|
+
*/
|
|
178
|
+
export declare function captureCustomerContext(customerId: string, metrics: {
|
|
179
|
+
arr?: number;
|
|
180
|
+
healthScore?: number;
|
|
181
|
+
segment?: string;
|
|
182
|
+
}): Promise<CustomerContextSnapshot>;
|
|
183
|
+
/**
|
|
184
|
+
* Get customer context as of a specific time
|
|
185
|
+
*/
|
|
186
|
+
export declare function getContextAsOf(customerId: string, timestamp: string | Date): Promise<CustomerContextSnapshot | null>;
|
|
187
|
+
/**
|
|
188
|
+
* Calculate ARR change between two snapshots
|
|
189
|
+
*/
|
|
190
|
+
export declare function calculateArrChange(older: CustomerMetricsSnapshot, newer: CustomerMetricsSnapshot): {
|
|
191
|
+
absolute: number;
|
|
192
|
+
percentage: number;
|
|
193
|
+
direction: 'increase' | 'decrease' | 'stable';
|
|
194
|
+
};
|
|
195
|
+
/**
|
|
196
|
+
* Calculate health score trend
|
|
197
|
+
*/
|
|
198
|
+
export declare function calculateHealthTrend(snapshots: CustomerMetricsSnapshot[]): {
|
|
199
|
+
currentScore: number | null;
|
|
200
|
+
avgScore: number | null;
|
|
201
|
+
trend: 'improving' | 'declining' | 'stable';
|
|
202
|
+
volatility: 'low' | 'medium' | 'high';
|
|
203
|
+
};
|