@thinkhive/sdk 2.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.
- package/LICENSE +21 -0
- package/README.md +212 -0
- package/dist/index.d.ts +629 -0
- package/dist/index.js +639 -0
- package/package.json +54 -0
package/dist/index.js
ADDED
|
@@ -0,0 +1,639 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* ThinkHive TypeScript SDK
|
|
4
|
+
* OpenTelemetry-based observability for AI agents with Explainer integration
|
|
5
|
+
*
|
|
6
|
+
* @version 2.0.0
|
|
7
|
+
*/
|
|
8
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
9
|
+
exports.analytics = exports.quality = exports.feedback = exports.explainer = void 0;
|
|
10
|
+
exports.init = init;
|
|
11
|
+
exports.getTracer = getTracer;
|
|
12
|
+
exports.isInitialized = isInitialized;
|
|
13
|
+
exports.traceLLM = traceLLM;
|
|
14
|
+
exports.traceRetrieval = traceRetrieval;
|
|
15
|
+
exports.traceTool = traceTool;
|
|
16
|
+
exports.traceChain = traceChain;
|
|
17
|
+
exports.createAndAnalyze = createAndAnalyze;
|
|
18
|
+
const api_1 = require("@opentelemetry/api");
|
|
19
|
+
const exporter_trace_otlp_proto_1 = require("@opentelemetry/exporter-trace-otlp-proto");
|
|
20
|
+
const resources_1 = require("@opentelemetry/resources");
|
|
21
|
+
const sdk_trace_node_1 = require("@opentelemetry/sdk-trace-node");
|
|
22
|
+
const sdk_trace_base_1 = require("@opentelemetry/sdk-trace-base");
|
|
23
|
+
// ============================================================================
|
|
24
|
+
// GLOBAL STATE
|
|
25
|
+
// ============================================================================
|
|
26
|
+
let tracer = null;
|
|
27
|
+
let initialized = false;
|
|
28
|
+
let config;
|
|
29
|
+
const DEFAULT_ENDPOINT = "https://thinkhivemind-h25z7pvd3q-uc.a.run.app";
|
|
30
|
+
// ============================================================================
|
|
31
|
+
// HTTP CLIENT
|
|
32
|
+
// ============================================================================
|
|
33
|
+
async function apiRequest(path, options = {}) {
|
|
34
|
+
const { method = 'GET', body, headers = {} } = options;
|
|
35
|
+
const url = `${config.endpoint}/api/v1${path}`;
|
|
36
|
+
const requestHeaders = {
|
|
37
|
+
'Content-Type': 'application/json',
|
|
38
|
+
...headers,
|
|
39
|
+
};
|
|
40
|
+
if (config.apiKey) {
|
|
41
|
+
requestHeaders['Authorization'] = `Bearer ${config.apiKey}`;
|
|
42
|
+
}
|
|
43
|
+
else if (config.agentId) {
|
|
44
|
+
requestHeaders['X-Agent-ID'] = config.agentId;
|
|
45
|
+
}
|
|
46
|
+
const response = await fetch(url, {
|
|
47
|
+
method,
|
|
48
|
+
headers: requestHeaders,
|
|
49
|
+
body: body ? JSON.stringify(body) : undefined,
|
|
50
|
+
});
|
|
51
|
+
if (!response.ok) {
|
|
52
|
+
const error = await response.text();
|
|
53
|
+
throw new Error(`ThinkHive API error: ${response.status} - ${error}`);
|
|
54
|
+
}
|
|
55
|
+
return response.json();
|
|
56
|
+
}
|
|
57
|
+
// ============================================================================
|
|
58
|
+
// INITIALIZATION
|
|
59
|
+
// ============================================================================
|
|
60
|
+
/**
|
|
61
|
+
* Initialize ThinkHive SDK
|
|
62
|
+
*
|
|
63
|
+
* @example
|
|
64
|
+
* ```typescript
|
|
65
|
+
* import { init } from '@thinkhive/sdk';
|
|
66
|
+
*
|
|
67
|
+
* init({
|
|
68
|
+
* apiKey: 'th_your_api_key',
|
|
69
|
+
* serviceName: 'my-ai-agent',
|
|
70
|
+
* autoInstrument: true,
|
|
71
|
+
* frameworks: ['langchain', 'openai'],
|
|
72
|
+
* });
|
|
73
|
+
* ```
|
|
74
|
+
*/
|
|
75
|
+
function init(options = {}) {
|
|
76
|
+
if (initialized) {
|
|
77
|
+
if (options.debug) {
|
|
78
|
+
console.log('ThinkHive SDK already initialized');
|
|
79
|
+
}
|
|
80
|
+
return;
|
|
81
|
+
}
|
|
82
|
+
const apiKey = options.apiKey || process.env.THINKHIVE_API_KEY;
|
|
83
|
+
const agentId = options.agentId || process.env.THINKHIVE_AGENT_ID;
|
|
84
|
+
if (!apiKey && !agentId) {
|
|
85
|
+
throw new Error("Either apiKey or agentId must be provided (or set THINKHIVE_API_KEY env var)");
|
|
86
|
+
}
|
|
87
|
+
config = {
|
|
88
|
+
apiKey: apiKey || '',
|
|
89
|
+
agentId: agentId || '',
|
|
90
|
+
endpoint: options.endpoint || process.env.THINKHIVE_ENDPOINT || DEFAULT_ENDPOINT,
|
|
91
|
+
serviceName: options.serviceName || process.env.THINKHIVE_SERVICE_NAME || 'my-ai-agent',
|
|
92
|
+
autoInstrument: options.autoInstrument ?? false,
|
|
93
|
+
frameworks: options.frameworks || ['langchain', 'openai'],
|
|
94
|
+
debug: options.debug ?? false,
|
|
95
|
+
};
|
|
96
|
+
// Configure OTLP exporter
|
|
97
|
+
const exporterHeaders = {};
|
|
98
|
+
if (config.apiKey) {
|
|
99
|
+
exporterHeaders["Authorization"] = `Bearer ${config.apiKey}`;
|
|
100
|
+
}
|
|
101
|
+
else if (config.agentId) {
|
|
102
|
+
exporterHeaders["X-Agent-ID"] = config.agentId;
|
|
103
|
+
}
|
|
104
|
+
const exporter = new exporter_trace_otlp_proto_1.OTLPTraceExporter({
|
|
105
|
+
url: `${config.endpoint}/v1/traces`,
|
|
106
|
+
headers: exporterHeaders,
|
|
107
|
+
});
|
|
108
|
+
// Create OpenTelemetry resource
|
|
109
|
+
const resource = resources_1.Resource.default().merge(new resources_1.Resource({
|
|
110
|
+
"service.name": config.serviceName,
|
|
111
|
+
"thinkhive.sdk.version": "2.0.0",
|
|
112
|
+
"thinkhive.sdk.language": "typescript",
|
|
113
|
+
}));
|
|
114
|
+
// Create provider with span processor
|
|
115
|
+
const provider = new sdk_trace_node_1.NodeTracerProvider({
|
|
116
|
+
resource,
|
|
117
|
+
spanProcessors: [new sdk_trace_base_1.BatchSpanProcessor(exporter)],
|
|
118
|
+
});
|
|
119
|
+
// Register provider
|
|
120
|
+
provider.register();
|
|
121
|
+
// Get tracer
|
|
122
|
+
tracer = api_1.trace.getTracer("thinkhive", "2.0.0");
|
|
123
|
+
initialized = true;
|
|
124
|
+
if (config.debug) {
|
|
125
|
+
console.log(`✅ ThinkHive SDK initialized`);
|
|
126
|
+
console.log(` Endpoint: ${config.endpoint}`);
|
|
127
|
+
console.log(` Service: ${config.serviceName}`);
|
|
128
|
+
console.log(` Auto-instrument: ${config.autoInstrument}`);
|
|
129
|
+
}
|
|
130
|
+
// Setup auto-instrumentation if enabled
|
|
131
|
+
if (config.autoInstrument) {
|
|
132
|
+
setupAutoInstrumentation(config.frameworks);
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
/**
|
|
136
|
+
* Get the global tracer
|
|
137
|
+
*/
|
|
138
|
+
function getTracer() {
|
|
139
|
+
if (!initialized || !tracer) {
|
|
140
|
+
throw new Error("ThinkHive SDK not initialized. Call init() first.");
|
|
141
|
+
}
|
|
142
|
+
return tracer;
|
|
143
|
+
}
|
|
144
|
+
/**
|
|
145
|
+
* Check if SDK is initialized
|
|
146
|
+
*/
|
|
147
|
+
function isInitialized() {
|
|
148
|
+
return initialized;
|
|
149
|
+
}
|
|
150
|
+
// ============================================================================
|
|
151
|
+
// TRACING FUNCTIONS
|
|
152
|
+
// ============================================================================
|
|
153
|
+
/**
|
|
154
|
+
* Trace an LLM call
|
|
155
|
+
*
|
|
156
|
+
* @example
|
|
157
|
+
* ```typescript
|
|
158
|
+
* const response = await traceLLM({
|
|
159
|
+
* name: 'chat_completion',
|
|
160
|
+
* modelName: 'gpt-4',
|
|
161
|
+
* provider: 'openai',
|
|
162
|
+
* }, async () => {
|
|
163
|
+
* return await openai.chat.completions.create({ ... });
|
|
164
|
+
* });
|
|
165
|
+
* ```
|
|
166
|
+
*/
|
|
167
|
+
function traceLLM(options, fn) {
|
|
168
|
+
const t = getTracer();
|
|
169
|
+
return t.startActiveSpan(options.name, {
|
|
170
|
+
attributes: {
|
|
171
|
+
"openinference.span.kind": "LLM",
|
|
172
|
+
"llm.model_name": options.modelName,
|
|
173
|
+
"llm.provider": options.provider,
|
|
174
|
+
"input.value": options.input ? JSON.stringify(options.input).substring(0, 10000) : undefined,
|
|
175
|
+
},
|
|
176
|
+
}, async (span) => {
|
|
177
|
+
const startTime = Date.now();
|
|
178
|
+
try {
|
|
179
|
+
const result = await fn();
|
|
180
|
+
span.setAttribute("output.value", JSON.stringify(result).substring(0, 10000));
|
|
181
|
+
span.setStatus({ code: api_1.SpanStatusCode.OK });
|
|
182
|
+
return result;
|
|
183
|
+
}
|
|
184
|
+
catch (error) {
|
|
185
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
186
|
+
span.setStatus({ code: api_1.SpanStatusCode.ERROR, message });
|
|
187
|
+
span.recordException(error);
|
|
188
|
+
throw error;
|
|
189
|
+
}
|
|
190
|
+
finally {
|
|
191
|
+
span.setAttribute("duration_ms", Date.now() - startTime);
|
|
192
|
+
span.end();
|
|
193
|
+
}
|
|
194
|
+
});
|
|
195
|
+
}
|
|
196
|
+
/**
|
|
197
|
+
* Trace a retrieval operation
|
|
198
|
+
*
|
|
199
|
+
* @example
|
|
200
|
+
* ```typescript
|
|
201
|
+
* const docs = await traceRetrieval({
|
|
202
|
+
* name: 'vector_search',
|
|
203
|
+
* query: 'What is the return policy?',
|
|
204
|
+
* }, async () => {
|
|
205
|
+
* return await vectorStore.similaritySearch(query, 5);
|
|
206
|
+
* });
|
|
207
|
+
* ```
|
|
208
|
+
*/
|
|
209
|
+
function traceRetrieval(options, fn) {
|
|
210
|
+
const t = getTracer();
|
|
211
|
+
return t.startActiveSpan(options.name, {
|
|
212
|
+
attributes: {
|
|
213
|
+
"openinference.span.kind": "RETRIEVER",
|
|
214
|
+
"retrieval.query": options.query,
|
|
215
|
+
"retrieval.top_k": options.topK,
|
|
216
|
+
},
|
|
217
|
+
}, async (span) => {
|
|
218
|
+
const startTime = Date.now();
|
|
219
|
+
try {
|
|
220
|
+
const result = await fn();
|
|
221
|
+
// Try to extract document count from result
|
|
222
|
+
if (Array.isArray(result)) {
|
|
223
|
+
span.setAttribute("retrieval.document_count", result.length);
|
|
224
|
+
}
|
|
225
|
+
span.setStatus({ code: api_1.SpanStatusCode.OK });
|
|
226
|
+
return result;
|
|
227
|
+
}
|
|
228
|
+
catch (error) {
|
|
229
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
230
|
+
span.setStatus({ code: api_1.SpanStatusCode.ERROR, message });
|
|
231
|
+
span.recordException(error);
|
|
232
|
+
throw error;
|
|
233
|
+
}
|
|
234
|
+
finally {
|
|
235
|
+
span.setAttribute("duration_ms", Date.now() - startTime);
|
|
236
|
+
span.end();
|
|
237
|
+
}
|
|
238
|
+
});
|
|
239
|
+
}
|
|
240
|
+
/**
|
|
241
|
+
* Trace a tool call
|
|
242
|
+
*
|
|
243
|
+
* @example
|
|
244
|
+
* ```typescript
|
|
245
|
+
* const result = await traceTool({
|
|
246
|
+
* name: 'search_orders',
|
|
247
|
+
* toolName: 'OrderLookup',
|
|
248
|
+
* parameters: { orderId: '12345' },
|
|
249
|
+
* }, async () => {
|
|
250
|
+
* return await orderService.lookup('12345');
|
|
251
|
+
* });
|
|
252
|
+
* ```
|
|
253
|
+
*/
|
|
254
|
+
function traceTool(options, fn) {
|
|
255
|
+
const t = getTracer();
|
|
256
|
+
return t.startActiveSpan(options.toolName || options.name, {
|
|
257
|
+
attributes: {
|
|
258
|
+
"openinference.span.kind": "TOOL",
|
|
259
|
+
"tool.name": options.toolName || options.name,
|
|
260
|
+
"tool.parameters": options.parameters ? JSON.stringify(options.parameters) : undefined,
|
|
261
|
+
},
|
|
262
|
+
}, async (span) => {
|
|
263
|
+
const startTime = Date.now();
|
|
264
|
+
try {
|
|
265
|
+
const result = await fn();
|
|
266
|
+
span.setAttribute("tool.output", JSON.stringify(result).substring(0, 10000));
|
|
267
|
+
span.setStatus({ code: api_1.SpanStatusCode.OK });
|
|
268
|
+
return result;
|
|
269
|
+
}
|
|
270
|
+
catch (error) {
|
|
271
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
272
|
+
span.setStatus({ code: api_1.SpanStatusCode.ERROR, message });
|
|
273
|
+
span.recordException(error);
|
|
274
|
+
throw error;
|
|
275
|
+
}
|
|
276
|
+
finally {
|
|
277
|
+
span.setAttribute("duration_ms", Date.now() - startTime);
|
|
278
|
+
span.end();
|
|
279
|
+
}
|
|
280
|
+
});
|
|
281
|
+
}
|
|
282
|
+
/**
|
|
283
|
+
* Trace a chain/workflow
|
|
284
|
+
*
|
|
285
|
+
* @example
|
|
286
|
+
* ```typescript
|
|
287
|
+
* const result = await traceChain({
|
|
288
|
+
* name: 'customer_support_chain',
|
|
289
|
+
* }, async () => {
|
|
290
|
+
* // Multiple LLM calls, tools, etc.
|
|
291
|
+
* });
|
|
292
|
+
* ```
|
|
293
|
+
*/
|
|
294
|
+
function traceChain(options, fn) {
|
|
295
|
+
const t = getTracer();
|
|
296
|
+
return t.startActiveSpan(options.name, {
|
|
297
|
+
attributes: {
|
|
298
|
+
"openinference.span.kind": "CHAIN",
|
|
299
|
+
"input.value": options.input ? JSON.stringify(options.input).substring(0, 10000) : undefined,
|
|
300
|
+
},
|
|
301
|
+
}, async (span) => {
|
|
302
|
+
const startTime = Date.now();
|
|
303
|
+
try {
|
|
304
|
+
const result = await fn();
|
|
305
|
+
span.setAttribute("output.value", JSON.stringify(result).substring(0, 10000));
|
|
306
|
+
span.setStatus({ code: api_1.SpanStatusCode.OK });
|
|
307
|
+
return result;
|
|
308
|
+
}
|
|
309
|
+
catch (error) {
|
|
310
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
311
|
+
span.setStatus({ code: api_1.SpanStatusCode.ERROR, message });
|
|
312
|
+
span.recordException(error);
|
|
313
|
+
throw error;
|
|
314
|
+
}
|
|
315
|
+
finally {
|
|
316
|
+
span.setAttribute("duration_ms", Date.now() - startTime);
|
|
317
|
+
span.end();
|
|
318
|
+
}
|
|
319
|
+
});
|
|
320
|
+
}
|
|
321
|
+
// ============================================================================
|
|
322
|
+
// EXPLAINER CLIENT
|
|
323
|
+
// ============================================================================
|
|
324
|
+
/**
|
|
325
|
+
* Explainer API client for trace analysis
|
|
326
|
+
*/
|
|
327
|
+
exports.explainer = {
|
|
328
|
+
/**
|
|
329
|
+
* Analyze a trace and get explainability insights
|
|
330
|
+
*
|
|
331
|
+
* @example
|
|
332
|
+
* ```typescript
|
|
333
|
+
* const result = await explainer.analyze({
|
|
334
|
+
* userMessage: 'Help me with my order #12345',
|
|
335
|
+
* agentResponse: 'I found your order...',
|
|
336
|
+
* outcome: 'success',
|
|
337
|
+
* spans: [
|
|
338
|
+
* { name: 'order_lookup', type: 'tool', durationMs: 150 },
|
|
339
|
+
* { name: 'gpt-4', type: 'llm', durationMs: 2500, model: 'gpt-4' },
|
|
340
|
+
* ],
|
|
341
|
+
* });
|
|
342
|
+
* ```
|
|
343
|
+
*/
|
|
344
|
+
async analyze(trace, options = {}) {
|
|
345
|
+
if (!initialized) {
|
|
346
|
+
throw new Error("ThinkHive SDK not initialized. Call init() first.");
|
|
347
|
+
}
|
|
348
|
+
const payload = {
|
|
349
|
+
trace: {
|
|
350
|
+
userMessage: trace.userMessage,
|
|
351
|
+
agentResponse: trace.agentResponse,
|
|
352
|
+
userIntent: trace.userIntent,
|
|
353
|
+
outcome: trace.outcome || 'success',
|
|
354
|
+
duration: trace.duration,
|
|
355
|
+
sessionId: trace.sessionId,
|
|
356
|
+
conversationHistory: trace.conversationHistory,
|
|
357
|
+
metadata: trace.metadata,
|
|
358
|
+
},
|
|
359
|
+
spans: trace.spans,
|
|
360
|
+
businessContext: trace.businessContext,
|
|
361
|
+
options: {
|
|
362
|
+
tier: options.tier,
|
|
363
|
+
includeRagEvaluation: options.includeRagEvaluation ?? true,
|
|
364
|
+
includeHallucinationCheck: options.includeHallucinationCheck ?? true,
|
|
365
|
+
},
|
|
366
|
+
};
|
|
367
|
+
if (options.waitForResult === false && options.webhookUrl) {
|
|
368
|
+
// Async analysis with webhook
|
|
369
|
+
return apiRequest('/explainer/analyze-async', {
|
|
370
|
+
method: 'POST',
|
|
371
|
+
body: { ...payload, webhookUrl: options.webhookUrl },
|
|
372
|
+
});
|
|
373
|
+
}
|
|
374
|
+
return apiRequest('/explainer/analyze', {
|
|
375
|
+
method: 'POST',
|
|
376
|
+
body: payload,
|
|
377
|
+
});
|
|
378
|
+
},
|
|
379
|
+
/**
|
|
380
|
+
* Batch analyze multiple traces
|
|
381
|
+
*
|
|
382
|
+
* @example
|
|
383
|
+
* ```typescript
|
|
384
|
+
* const results = await explainer.analyzeBatch([
|
|
385
|
+
* { userMessage: 'Q1', agentResponse: 'A1' },
|
|
386
|
+
* { userMessage: 'Q2', agentResponse: 'A2' },
|
|
387
|
+
* ]);
|
|
388
|
+
* ```
|
|
389
|
+
*/
|
|
390
|
+
async analyzeBatch(traces, options = {}) {
|
|
391
|
+
if (!initialized) {
|
|
392
|
+
throw new Error("ThinkHive SDK not initialized. Call init() first.");
|
|
393
|
+
}
|
|
394
|
+
return apiRequest('/explainer/analyze-batch', {
|
|
395
|
+
method: 'POST',
|
|
396
|
+
body: {
|
|
397
|
+
traces: traces.map(t => ({
|
|
398
|
+
userMessage: t.userMessage,
|
|
399
|
+
agentResponse: t.agentResponse,
|
|
400
|
+
userIntent: t.userIntent,
|
|
401
|
+
outcome: t.outcome || 'success',
|
|
402
|
+
spans: t.spans,
|
|
403
|
+
businessContext: t.businessContext,
|
|
404
|
+
metadata: t.metadata,
|
|
405
|
+
})),
|
|
406
|
+
options,
|
|
407
|
+
},
|
|
408
|
+
});
|
|
409
|
+
},
|
|
410
|
+
/**
|
|
411
|
+
* Get a previously analyzed trace
|
|
412
|
+
*/
|
|
413
|
+
async get(traceId) {
|
|
414
|
+
if (!initialized) {
|
|
415
|
+
throw new Error("ThinkHive SDK not initialized. Call init() first.");
|
|
416
|
+
}
|
|
417
|
+
return apiRequest(`/explainer/${traceId}`);
|
|
418
|
+
},
|
|
419
|
+
/**
|
|
420
|
+
* Search traces semantically
|
|
421
|
+
*
|
|
422
|
+
* @example
|
|
423
|
+
* ```typescript
|
|
424
|
+
* const results = await explainer.search({
|
|
425
|
+
* query: 'order refund issues',
|
|
426
|
+
* filters: { outcome: 'failure' },
|
|
427
|
+
* limit: 10,
|
|
428
|
+
* });
|
|
429
|
+
* ```
|
|
430
|
+
*/
|
|
431
|
+
async search(params) {
|
|
432
|
+
if (!initialized) {
|
|
433
|
+
throw new Error("ThinkHive SDK not initialized. Call init() first.");
|
|
434
|
+
}
|
|
435
|
+
return apiRequest('/explainer/search', {
|
|
436
|
+
method: 'POST',
|
|
437
|
+
body: params,
|
|
438
|
+
});
|
|
439
|
+
},
|
|
440
|
+
};
|
|
441
|
+
// ============================================================================
|
|
442
|
+
// FEEDBACK CLIENT
|
|
443
|
+
// ============================================================================
|
|
444
|
+
/**
|
|
445
|
+
* Feedback API for improving analysis quality
|
|
446
|
+
*/
|
|
447
|
+
exports.feedback = {
|
|
448
|
+
/**
|
|
449
|
+
* Submit feedback for a trace
|
|
450
|
+
*
|
|
451
|
+
* @example
|
|
452
|
+
* ```typescript
|
|
453
|
+
* await feedback.submit({
|
|
454
|
+
* traceId: 'trace_123',
|
|
455
|
+
* rating: 5,
|
|
456
|
+
* wasHelpful: true,
|
|
457
|
+
* });
|
|
458
|
+
* ```
|
|
459
|
+
*/
|
|
460
|
+
async submit(options) {
|
|
461
|
+
if (!initialized) {
|
|
462
|
+
throw new Error("ThinkHive SDK not initialized. Call init() first.");
|
|
463
|
+
}
|
|
464
|
+
return apiRequest('/feedback', {
|
|
465
|
+
method: 'POST',
|
|
466
|
+
body: options,
|
|
467
|
+
});
|
|
468
|
+
},
|
|
469
|
+
};
|
|
470
|
+
// ============================================================================
|
|
471
|
+
// QUALITY METRICS CLIENT
|
|
472
|
+
// ============================================================================
|
|
473
|
+
/**
|
|
474
|
+
* Quality metrics API for RAG evaluation and hallucination detection
|
|
475
|
+
*/
|
|
476
|
+
exports.quality = {
|
|
477
|
+
/**
|
|
478
|
+
* Get RAG evaluation scores for a trace
|
|
479
|
+
*/
|
|
480
|
+
async getRagScores(traceId) {
|
|
481
|
+
if (!initialized) {
|
|
482
|
+
throw new Error("ThinkHive SDK not initialized. Call init() first.");
|
|
483
|
+
}
|
|
484
|
+
return apiRequest(`/quality/rag-scores/${traceId}`);
|
|
485
|
+
},
|
|
486
|
+
/**
|
|
487
|
+
* Get hallucination report for a trace
|
|
488
|
+
*/
|
|
489
|
+
async getHallucinationReport(traceId) {
|
|
490
|
+
if (!initialized) {
|
|
491
|
+
throw new Error("ThinkHive SDK not initialized. Call init() first.");
|
|
492
|
+
}
|
|
493
|
+
return apiRequest(`/quality/hallucination-report/${traceId}`);
|
|
494
|
+
},
|
|
495
|
+
/**
|
|
496
|
+
* Evaluate RAG quality for custom input
|
|
497
|
+
*/
|
|
498
|
+
async evaluateRag(input) {
|
|
499
|
+
if (!initialized) {
|
|
500
|
+
throw new Error("ThinkHive SDK not initialized. Call init() first.");
|
|
501
|
+
}
|
|
502
|
+
return apiRequest('/quality/evaluate-rag', {
|
|
503
|
+
method: 'POST',
|
|
504
|
+
body: input,
|
|
505
|
+
});
|
|
506
|
+
},
|
|
507
|
+
};
|
|
508
|
+
// ============================================================================
|
|
509
|
+
// ROI ANALYTICS CLIENT
|
|
510
|
+
// ============================================================================
|
|
511
|
+
/**
|
|
512
|
+
* ROI analytics API for business impact tracking
|
|
513
|
+
*/
|
|
514
|
+
exports.analytics = {
|
|
515
|
+
/**
|
|
516
|
+
* Get ROI summary for the account
|
|
517
|
+
*/
|
|
518
|
+
async getRoiSummary() {
|
|
519
|
+
if (!initialized) {
|
|
520
|
+
throw new Error("ThinkHive SDK not initialized. Call init() first.");
|
|
521
|
+
}
|
|
522
|
+
return apiRequest('/analytics/roi/summary');
|
|
523
|
+
},
|
|
524
|
+
/**
|
|
525
|
+
* Get ROI by agent
|
|
526
|
+
*/
|
|
527
|
+
async getRoiByAgent(agentId) {
|
|
528
|
+
if (!initialized) {
|
|
529
|
+
throw new Error("ThinkHive SDK not initialized. Call init() first.");
|
|
530
|
+
}
|
|
531
|
+
return apiRequest(`/analytics/roi/by-agent/${agentId}`);
|
|
532
|
+
},
|
|
533
|
+
/**
|
|
534
|
+
* Get correlation analysis
|
|
535
|
+
*/
|
|
536
|
+
async getCorrelations() {
|
|
537
|
+
if (!initialized) {
|
|
538
|
+
throw new Error("ThinkHive SDK not initialized. Call init() first.");
|
|
539
|
+
}
|
|
540
|
+
return apiRequest('/analytics/correlations');
|
|
541
|
+
},
|
|
542
|
+
};
|
|
543
|
+
// ============================================================================
|
|
544
|
+
// AUTO-INSTRUMENTATION
|
|
545
|
+
// ============================================================================
|
|
546
|
+
/**
|
|
547
|
+
* Setup auto-instrumentation for AI frameworks
|
|
548
|
+
*/
|
|
549
|
+
function setupAutoInstrumentation(frameworks) {
|
|
550
|
+
if (config.debug) {
|
|
551
|
+
console.log(`Setting up auto-instrumentation for: ${frameworks.join(', ')}`);
|
|
552
|
+
}
|
|
553
|
+
// Note: Full auto-instrumentation requires framework-specific patches
|
|
554
|
+
// This is a placeholder for the instrumentation setup
|
|
555
|
+
// In production, use @opentelemetry/instrumentation-* packages
|
|
556
|
+
for (const framework of frameworks) {
|
|
557
|
+
try {
|
|
558
|
+
switch (framework) {
|
|
559
|
+
case 'openai':
|
|
560
|
+
instrumentOpenAI();
|
|
561
|
+
break;
|
|
562
|
+
case 'langchain':
|
|
563
|
+
instrumentLangChain();
|
|
564
|
+
break;
|
|
565
|
+
case 'anthropic':
|
|
566
|
+
instrumentAnthropic();
|
|
567
|
+
break;
|
|
568
|
+
case 'llamaindex':
|
|
569
|
+
instrumentLlamaIndex();
|
|
570
|
+
break;
|
|
571
|
+
}
|
|
572
|
+
}
|
|
573
|
+
catch (error) {
|
|
574
|
+
if (config.debug) {
|
|
575
|
+
console.warn(`Failed to instrument ${framework}:`, error);
|
|
576
|
+
}
|
|
577
|
+
}
|
|
578
|
+
}
|
|
579
|
+
}
|
|
580
|
+
function instrumentOpenAI() {
|
|
581
|
+
// Placeholder - would patch OpenAI client
|
|
582
|
+
if (config.debug) {
|
|
583
|
+
console.log('OpenAI instrumentation ready');
|
|
584
|
+
}
|
|
585
|
+
}
|
|
586
|
+
function instrumentLangChain() {
|
|
587
|
+
// Placeholder - would use LangChain callbacks
|
|
588
|
+
if (config.debug) {
|
|
589
|
+
console.log('LangChain instrumentation ready');
|
|
590
|
+
}
|
|
591
|
+
}
|
|
592
|
+
function instrumentAnthropic() {
|
|
593
|
+
// Placeholder - would patch Anthropic client
|
|
594
|
+
if (config.debug) {
|
|
595
|
+
console.log('Anthropic instrumentation ready');
|
|
596
|
+
}
|
|
597
|
+
}
|
|
598
|
+
function instrumentLlamaIndex() {
|
|
599
|
+
// Placeholder - would use LlamaIndex callbacks
|
|
600
|
+
if (config.debug) {
|
|
601
|
+
console.log('LlamaIndex instrumentation ready');
|
|
602
|
+
}
|
|
603
|
+
}
|
|
604
|
+
// ============================================================================
|
|
605
|
+
// CONVENIENCE FUNCTIONS
|
|
606
|
+
// ============================================================================
|
|
607
|
+
/**
|
|
608
|
+
* Create a trace and analyze it in one call
|
|
609
|
+
*
|
|
610
|
+
* @example
|
|
611
|
+
* ```typescript
|
|
612
|
+
* const { trace, analysis } = await createAndAnalyze({
|
|
613
|
+
* userMessage: 'Help me track my order',
|
|
614
|
+
* agentResponse: 'Your order is on the way...',
|
|
615
|
+
* });
|
|
616
|
+
* ```
|
|
617
|
+
*/
|
|
618
|
+
async function createAndAnalyze(traceData, options = {}) {
|
|
619
|
+
const analysis = await exports.explainer.analyze(traceData, options);
|
|
620
|
+
return { trace: traceData, analysis };
|
|
621
|
+
}
|
|
622
|
+
// ============================================================================
|
|
623
|
+
// EXPORTS
|
|
624
|
+
// ============================================================================
|
|
625
|
+
exports.default = {
|
|
626
|
+
init,
|
|
627
|
+
getTracer,
|
|
628
|
+
isInitialized,
|
|
629
|
+
traceLLM,
|
|
630
|
+
traceRetrieval,
|
|
631
|
+
traceTool,
|
|
632
|
+
traceChain,
|
|
633
|
+
explainer: exports.explainer,
|
|
634
|
+
feedback: exports.feedback,
|
|
635
|
+
quality: exports.quality,
|
|
636
|
+
analytics: exports.analytics,
|
|
637
|
+
createAndAnalyze,
|
|
638
|
+
};
|
|
639
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ThinkHive TypeScript SDK\n * OpenTelemetry-based observability for AI agents with Explainer integration\n *\n * @version 2.0.0\n */\n\nimport { trace, context, SpanStatusCode, Span as OTelSpan } from \"@opentelemetry/api\";\nimport { OTLPTraceExporter } from \"@opentelemetry/exporter-trace-otlp-proto\";\nimport { Resource } from \"@opentelemetry/resources\";\nimport { NodeTracerProvider } from \"@opentelemetry/sdk-trace-node\";\nimport { BatchSpanProcessor } from \"@opentelemetry/sdk-trace-base\";\n\n// ============================================================================\n// TYPES\n// ============================================================================\n\nexport interface InitOptions {\n  /** ThinkHive API key (starts with th_) */\n  apiKey?: string;\n  /** Agent ID for legacy authentication */\n  agentId?: string;\n  /** ThinkHive API endpoint */\n  endpoint?: string;\n  /** Service name for traces */\n  serviceName?: string;\n  /** Enable auto-instrumentation */\n  autoInstrument?: boolean;\n  /** Frameworks to auto-instrument */\n  frameworks?: Array<'langchain' | 'openai' | 'anthropic' | 'llamaindex'>;\n  /** Enable debug logging */\n  debug?: boolean;\n}\n\nexport interface TraceOptions {\n  /** User's message/query */\n  userMessage: string;\n  /** Agent's response */\n  agentResponse: string;\n  /** User's detected intent */\n  userIntent?: string;\n  /** Outcome: success, failure, partial_success */\n  outcome?: 'success' | 'failure' | 'partial_success';\n  /** Duration in milliseconds */\n  duration?: number;\n  /** Session ID for conversation tracking */\n  sessionId?: string;\n  /** Conversation history */\n  conversationHistory?: Array<{ role: string; content: string }>;\n  /** Span data for detailed analysis */\n  spans?: SpanData[];\n  /** Business context for ROI calculation */\n  businessContext?: BusinessContext;\n  /** Custom metadata */\n  metadata?: Record<string, unknown>;\n}\n\nexport interface SpanData {\n  id?: string;\n  name: string;\n  type: 'llm' | 'tool' | 'retrieval' | 'embedding' | 'chain' | 'custom';\n  startTime?: Date;\n  endTime?: Date;\n  durationMs?: number;\n  status?: 'ok' | 'error' | 'timeout';\n  error?: string;\n  input?: unknown;\n  output?: unknown;\n  // LLM-specific\n  model?: string;\n  provider?: string;\n  promptTokens?: number;\n  completionTokens?: number;\n  // Tool-specific\n  toolName?: string;\n  toolParameters?: Record<string, unknown>;\n  // Retrieval-specific\n  query?: string;\n  documentCount?: number;\n  topScore?: number;\n  sources?: string[];\n  // Children\n  children?: SpanData[];\n}\n\nexport interface BusinessContext {\n  /** Customer ID */\n  customerId?: string;\n  /** Transaction value in dollars */\n  transactionValue?: number;\n  /** Priority level */\n  priority?: 'low' | 'medium' | 'high' | 'critical';\n  /** Department */\n  department?: string;\n  /** Industry vertical */\n  industry?: string;\n  /** Custom fields */\n  custom?: Record<string, unknown>;\n}\n\nexport interface ExplainabilityResult {\n  traceId: string;\n  explainabilityId: string;\n  summary: string;\n  outcome: {\n    verdict: 'success' | 'partial_success' | 'failure';\n    confidence: number;\n    reasoning: string;\n  };\n  businessImpact: {\n    impactScore: number;\n    customerSatisfaction: number;\n    revenueRisk: string;\n  };\n  recommendations: Array<{\n    priority: string;\n    category: string;\n    action: string;\n    expectedImpact: string;\n  }>;\n  ragEvaluation?: {\n    groundedness: number;\n    faithfulness: number;\n    answerRelevance: number;\n  };\n  hallucinationReport?: {\n    detected: boolean;\n    types: string[];\n    severity: string;\n  };\n  processingTimeMs: number;\n}\n\nexport interface AnalyzeOptions {\n  /** Force a specific analysis tier */\n  tier?: 'rule_based' | 'fast_llm' | 'full_llm' | 'deep';\n  /** Include RAG evaluation */\n  includeRagEvaluation?: boolean;\n  /** Include hallucination detection */\n  includeHallucinationCheck?: boolean;\n  /** Wait for analysis to complete (sync mode) */\n  waitForResult?: boolean;\n  /** Webhook URL for async notification */\n  webhookUrl?: string;\n}\n\nexport interface FeedbackOptions {\n  /** Trace ID to provide feedback for */\n  traceId: string;\n  /** User rating (1-5) */\n  rating?: number;\n  /** Was the response helpful */\n  wasHelpful?: boolean;\n  /** Specific issues encountered */\n  issues?: Array<'slow_response' | 'incorrect' | 'unhelpful' | 'hallucination' | 'off_topic' | 'other'>;\n  /** Free-form comment */\n  comment?: string;\n}\n\n// ============================================================================\n// GLOBAL STATE\n// ============================================================================\n\nlet tracer: ReturnType<typeof trace.getTracer> | null = null;\nlet initialized = false;\nlet config: Required<InitOptions>;\n\nconst DEFAULT_ENDPOINT = \"https://thinkhivemind-h25z7pvd3q-uc.a.run.app\";\n\n// ============================================================================\n// HTTP CLIENT\n// ============================================================================\n\nasync function apiRequest<T>(\n  path: string,\n  options: {\n    method?: 'GET' | 'POST' | 'PUT' | 'DELETE';\n    body?: unknown;\n    headers?: Record<string, string>;\n  } = {}\n): Promise<T> {\n  const { method = 'GET', body, headers = {} } = options;\n\n  const url = `${config.endpoint}/api/v1${path}`;\n\n  const requestHeaders: Record<string, string> = {\n    'Content-Type': 'application/json',\n    ...headers,\n  };\n\n  if (config.apiKey) {\n    requestHeaders['Authorization'] = `Bearer ${config.apiKey}`;\n  } else if (config.agentId) {\n    requestHeaders['X-Agent-ID'] = config.agentId;\n  }\n\n  const response = await fetch(url, {\n    method,\n    headers: requestHeaders,\n    body: body ? JSON.stringify(body) : undefined,\n  });\n\n  if (!response.ok) {\n    const error = await response.text();\n    throw new Error(`ThinkHive API error: ${response.status} - ${error}`);\n  }\n\n  return response.json() as Promise<T>;\n}\n\n// ============================================================================\n// INITIALIZATION\n// ============================================================================\n\n/**\n * Initialize ThinkHive SDK\n *\n * @example\n * ```typescript\n * import { init } from '@thinkhive/sdk';\n *\n * init({\n *   apiKey: 'th_your_api_key',\n *   serviceName: 'my-ai-agent',\n *   autoInstrument: true,\n *   frameworks: ['langchain', 'openai'],\n * });\n * ```\n */\nexport function init(options: InitOptions = {}): void {\n  if (initialized) {\n    if (options.debug) {\n      console.log('ThinkHive SDK already initialized');\n    }\n    return;\n  }\n\n  const apiKey = options.apiKey || process.env.THINKHIVE_API_KEY;\n  const agentId = options.agentId || process.env.THINKHIVE_AGENT_ID;\n\n  if (!apiKey && !agentId) {\n    throw new Error(\"Either apiKey or agentId must be provided (or set THINKHIVE_API_KEY env var)\");\n  }\n\n  config = {\n    apiKey: apiKey || '',\n    agentId: agentId || '',\n    endpoint: options.endpoint || process.env.THINKHIVE_ENDPOINT || DEFAULT_ENDPOINT,\n    serviceName: options.serviceName || process.env.THINKHIVE_SERVICE_NAME || 'my-ai-agent',\n    autoInstrument: options.autoInstrument ?? false,\n    frameworks: options.frameworks || ['langchain', 'openai'],\n    debug: options.debug ?? false,\n  };\n\n  // Configure OTLP exporter\n  const exporterHeaders: Record<string, string> = {};\n  if (config.apiKey) {\n    exporterHeaders[\"Authorization\"] = `Bearer ${config.apiKey}`;\n  } else if (config.agentId) {\n    exporterHeaders[\"X-Agent-ID\"] = config.agentId;\n  }\n\n  const exporter = new OTLPTraceExporter({\n    url: `${config.endpoint}/v1/traces`,\n    headers: exporterHeaders,\n  });\n\n  // Create OpenTelemetry resource\n  const resource = Resource.default().merge(\n    new Resource({\n      \"service.name\": config.serviceName,\n      \"thinkhive.sdk.version\": \"2.0.0\",\n      \"thinkhive.sdk.language\": \"typescript\",\n    })\n  );\n\n  // Create provider with span processor\n  const provider = new NodeTracerProvider({\n    resource,\n    spanProcessors: [new BatchSpanProcessor(exporter)],\n  });\n\n  // Register provider\n  provider.register();\n\n  // Get tracer\n  tracer = trace.getTracer(\"thinkhive\", \"2.0.0\");\n  initialized = true;\n\n  if (config.debug) {\n    console.log(`✅ ThinkHive SDK initialized`);\n    console.log(`   Endpoint: ${config.endpoint}`);\n    console.log(`   Service: ${config.serviceName}`);\n    console.log(`   Auto-instrument: ${config.autoInstrument}`);\n  }\n\n  // Setup auto-instrumentation if enabled\n  if (config.autoInstrument) {\n    setupAutoInstrumentation(config.frameworks);\n  }\n}\n\n/**\n * Get the global tracer\n */\nexport function getTracer() {\n  if (!initialized || !tracer) {\n    throw new Error(\"ThinkHive SDK not initialized. Call init() first.\");\n  }\n  return tracer;\n}\n\n/**\n * Check if SDK is initialized\n */\nexport function isInitialized(): boolean {\n  return initialized;\n}\n\n// ============================================================================\n// TRACING FUNCTIONS\n// ============================================================================\n\n/**\n * Trace an LLM call\n *\n * @example\n * ```typescript\n * const response = await traceLLM({\n *   name: 'chat_completion',\n *   modelName: 'gpt-4',\n *   provider: 'openai',\n * }, async () => {\n *   return await openai.chat.completions.create({ ... });\n * });\n * ```\n */\nexport function traceLLM<T>(\n  options: {\n    name: string;\n    modelName?: string;\n    provider?: string;\n    input?: unknown;\n  },\n  fn: () => Promise<T>\n): Promise<T> {\n  const t = getTracer();\n\n  return t.startActiveSpan(\n    options.name,\n    {\n      attributes: {\n        \"openinference.span.kind\": \"LLM\",\n        \"llm.model_name\": options.modelName,\n        \"llm.provider\": options.provider,\n        \"input.value\": options.input ? JSON.stringify(options.input).substring(0, 10000) : undefined,\n      },\n    },\n    async (span) => {\n      const startTime = Date.now();\n      try {\n        const result = await fn();\n        span.setAttribute(\"output.value\", 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 * Trace a retrieval operation\n *\n * @example\n * ```typescript\n * const docs = await traceRetrieval({\n *   name: 'vector_search',\n *   query: 'What is the return policy?',\n * }, async () => {\n *   return await vectorStore.similaritySearch(query, 5);\n * });\n * ```\n */\nexport function traceRetrieval<T>(\n  options: {\n    name: string;\n    query?: string;\n    topK?: number;\n  },\n  fn: () => Promise<T>\n): Promise<T> {\n  const t = getTracer();\n\n  return t.startActiveSpan(\n    options.name,\n    {\n      attributes: {\n        \"openinference.span.kind\": \"RETRIEVER\",\n        \"retrieval.query\": options.query,\n        \"retrieval.top_k\": options.topK,\n      },\n    },\n    async (span) => {\n      const startTime = Date.now();\n      try {\n        const result = await fn();\n\n        // Try to extract document count from result\n        if (Array.isArray(result)) {\n          span.setAttribute(\"retrieval.document_count\", result.length);\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 * Trace a tool call\n *\n * @example\n * ```typescript\n * const result = await traceTool({\n *   name: 'search_orders',\n *   toolName: 'OrderLookup',\n *   parameters: { orderId: '12345' },\n * }, async () => {\n *   return await orderService.lookup('12345');\n * });\n * ```\n */\nexport function traceTool<T>(\n  options: {\n    name: string;\n    toolName?: string;\n    parameters?: Record<string, unknown>;\n  },\n  fn: () => Promise<T>\n): Promise<T> {\n  const t = getTracer();\n\n  return t.startActiveSpan(\n    options.toolName || options.name,\n    {\n      attributes: {\n        \"openinference.span.kind\": \"TOOL\",\n        \"tool.name\": options.toolName || options.name,\n        \"tool.parameters\": options.parameters ? JSON.stringify(options.parameters) : 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 * Trace a chain/workflow\n *\n * @example\n * ```typescript\n * const result = await traceChain({\n *   name: 'customer_support_chain',\n * }, async () => {\n *   // Multiple LLM calls, tools, etc.\n * });\n * ```\n */\nexport function traceChain<T>(\n  options: {\n    name: string;\n    input?: unknown;\n  },\n  fn: () => Promise<T>\n): Promise<T> {\n  const t = getTracer();\n\n  return t.startActiveSpan(\n    options.name,\n    {\n      attributes: {\n        \"openinference.span.kind\": \"CHAIN\",\n        \"input.value\": options.input ? JSON.stringify(options.input).substring(0, 10000) : undefined,\n      },\n    },\n    async (span) => {\n      const startTime = Date.now();\n      try {\n        const result = await fn();\n        span.setAttribute(\"output.value\", 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// EXPLAINER CLIENT\n// ============================================================================\n\n/**\n * Explainer API client for trace analysis\n */\nexport const explainer = {\n  /**\n   * Analyze a trace and get explainability insights\n   *\n   * @example\n   * ```typescript\n   * const result = await explainer.analyze({\n   *   userMessage: 'Help me with my order #12345',\n   *   agentResponse: 'I found your order...',\n   *   outcome: 'success',\n   *   spans: [\n   *     { name: 'order_lookup', type: 'tool', durationMs: 150 },\n   *     { name: 'gpt-4', type: 'llm', durationMs: 2500, model: 'gpt-4' },\n   *   ],\n   * });\n   * ```\n   */\n  async analyze(\n    trace: TraceOptions,\n    options: AnalyzeOptions = {}\n  ): Promise<ExplainabilityResult> {\n    if (!initialized) {\n      throw new Error(\"ThinkHive SDK not initialized. Call init() first.\");\n    }\n\n    const payload = {\n      trace: {\n        userMessage: trace.userMessage,\n        agentResponse: trace.agentResponse,\n        userIntent: trace.userIntent,\n        outcome: trace.outcome || 'success',\n        duration: trace.duration,\n        sessionId: trace.sessionId,\n        conversationHistory: trace.conversationHistory,\n        metadata: trace.metadata,\n      },\n      spans: trace.spans,\n      businessContext: trace.businessContext,\n      options: {\n        tier: options.tier,\n        includeRagEvaluation: options.includeRagEvaluation ?? true,\n        includeHallucinationCheck: options.includeHallucinationCheck ?? true,\n      },\n    };\n\n    if (options.waitForResult === false && options.webhookUrl) {\n      // Async analysis with webhook\n      return apiRequest<ExplainabilityResult>('/explainer/analyze-async', {\n        method: 'POST',\n        body: { ...payload, webhookUrl: options.webhookUrl },\n      });\n    }\n\n    return apiRequest<ExplainabilityResult>('/explainer/analyze', {\n      method: 'POST',\n      body: payload,\n    });\n  },\n\n  /**\n   * Batch analyze multiple traces\n   *\n   * @example\n   * ```typescript\n   * const results = await explainer.analyzeBatch([\n   *   { userMessage: 'Q1', agentResponse: 'A1' },\n   *   { userMessage: 'Q2', agentResponse: 'A2' },\n   * ]);\n   * ```\n   */\n  async analyzeBatch(\n    traces: TraceOptions[],\n    options: AnalyzeOptions = {}\n  ): Promise<{\n    results: ExplainabilityResult[];\n    summary: {\n      total: number;\n      successCount: number;\n      failureCount: number;\n      averageProcessingTime: number;\n    };\n  }> {\n    if (!initialized) {\n      throw new Error(\"ThinkHive SDK not initialized. Call init() first.\");\n    }\n\n    return apiRequest('/explainer/analyze-batch', {\n      method: 'POST',\n      body: {\n        traces: traces.map(t => ({\n          userMessage: t.userMessage,\n          agentResponse: t.agentResponse,\n          userIntent: t.userIntent,\n          outcome: t.outcome || 'success',\n          spans: t.spans,\n          businessContext: t.businessContext,\n          metadata: t.metadata,\n        })),\n        options,\n      },\n    });\n  },\n\n  /**\n   * Get a previously analyzed trace\n   */\n  async get(traceId: string): Promise<ExplainabilityResult> {\n    if (!initialized) {\n      throw new Error(\"ThinkHive SDK not initialized. Call init() first.\");\n    }\n\n    return apiRequest<ExplainabilityResult>(`/explainer/${traceId}`);\n  },\n\n  /**\n   * Search traces semantically\n   *\n   * @example\n   * ```typescript\n   * const results = await explainer.search({\n   *   query: 'order refund issues',\n   *   filters: { outcome: 'failure' },\n   *   limit: 10,\n   * });\n   * ```\n   */\n  async search(params: {\n    query: string;\n    filters?: {\n      outcome?: 'success' | 'failure' | 'partial_success';\n      minImpactScore?: number;\n      dateRange?: { from: Date; to: Date };\n    };\n    limit?: number;\n  }): Promise<{\n    results: Array<ExplainabilityResult & { similarity: number }>;\n    total: number;\n  }> {\n    if (!initialized) {\n      throw new Error(\"ThinkHive SDK not initialized. Call init() first.\");\n    }\n\n    return apiRequest('/explainer/search', {\n      method: 'POST',\n      body: params,\n    });\n  },\n};\n\n// ============================================================================\n// FEEDBACK CLIENT\n// ============================================================================\n\n/**\n * Feedback API for improving analysis quality\n */\nexport const feedback = {\n  /**\n   * Submit feedback for a trace\n   *\n   * @example\n   * ```typescript\n   * await feedback.submit({\n   *   traceId: 'trace_123',\n   *   rating: 5,\n   *   wasHelpful: true,\n   * });\n   * ```\n   */\n  async submit(options: FeedbackOptions): Promise<{ success: boolean }> {\n    if (!initialized) {\n      throw new Error(\"ThinkHive SDK not initialized. Call init() first.\");\n    }\n\n    return apiRequest('/feedback', {\n      method: 'POST',\n      body: options,\n    });\n  },\n};\n\n// ============================================================================\n// QUALITY METRICS CLIENT\n// ============================================================================\n\n/**\n * Quality metrics API for RAG evaluation and hallucination detection\n */\nexport const quality = {\n  /**\n   * Get RAG evaluation scores for a trace\n   */\n  async getRagScores(traceId: string): Promise<{\n    contextRelevance: number;\n    contextPrecision: number;\n    contextRecall: number;\n    groundedness: number;\n    faithfulness: number;\n    answerRelevance: number;\n    citationAccuracy: number;\n    citationCompleteness: number;\n    overallGrade: string;\n  }> {\n    if (!initialized) {\n      throw new Error(\"ThinkHive SDK not initialized. Call init() first.\");\n    }\n\n    return apiRequest(`/quality/rag-scores/${traceId}`);\n  },\n\n  /**\n   * Get hallucination report for a trace\n   */\n  async getHallucinationReport(traceId: string): Promise<{\n    hasHallucinations: boolean;\n    severity: 'none' | 'low' | 'medium' | 'high';\n    confidence: number;\n    detectedTypes: Array<{\n      type: string;\n      description: string;\n      evidence: string;\n      severity: string;\n    }>;\n  }> {\n    if (!initialized) {\n      throw new Error(\"ThinkHive SDK not initialized. Call init() first.\");\n    }\n\n    return apiRequest(`/quality/hallucination-report/${traceId}`);\n  },\n\n  /**\n   * Evaluate RAG quality for custom input\n   */\n  async evaluateRag(input: {\n    query: string;\n    response: string;\n    contexts: Array<{ content: string; source?: string; score?: number }>;\n  }): Promise<{\n    scores: Record<string, number>;\n    grade: string;\n    groundedSpans: string[];\n    ungroundedSpans: string[];\n  }> {\n    if (!initialized) {\n      throw new Error(\"ThinkHive SDK not initialized. Call init() first.\");\n    }\n\n    return apiRequest('/quality/evaluate-rag', {\n      method: 'POST',\n      body: input,\n    });\n  },\n};\n\n// ============================================================================\n// ROI ANALYTICS CLIENT\n// ============================================================================\n\n/**\n * ROI analytics API for business impact tracking\n */\nexport const analytics = {\n  /**\n   * Get ROI summary for the account\n   */\n  async getRoiSummary(): Promise<{\n    totalRevenueSaved: number;\n    supportCostReduction: number;\n    averageResolutionTime: number;\n    customerSatisfaction: number;\n    trends: Array<{\n      date: string;\n      revenueSaved: number;\n      tracesAnalyzed: number;\n    }>;\n  }> {\n    if (!initialized) {\n      throw new Error(\"ThinkHive SDK not initialized. Call init() first.\");\n    }\n\n    return apiRequest('/analytics/roi/summary');\n  },\n\n  /**\n   * Get ROI by agent\n   */\n  async getRoiByAgent(agentId: string): Promise<{\n    agentId: string;\n    revenueSaved: number;\n    tracesAnalyzed: number;\n    successRate: number;\n    topIssues: Array<{ issue: string; count: number; impact: number }>;\n  }> {\n    if (!initialized) {\n      throw new Error(\"ThinkHive SDK not initialized. Call init() first.\");\n    }\n\n    return apiRequest(`/analytics/roi/by-agent/${agentId}`);\n  },\n\n  /**\n   * Get correlation analysis\n   */\n  async getCorrelations(): Promise<{\n    correlations: Array<{\n      type: string;\n      coefficient: number;\n      description: string;\n      actionableInsight: string;\n    }>;\n  }> {\n    if (!initialized) {\n      throw new Error(\"ThinkHive SDK not initialized. Call init() first.\");\n    }\n\n    return apiRequest('/analytics/correlations');\n  },\n};\n\n// ============================================================================\n// AUTO-INSTRUMENTATION\n// ============================================================================\n\n/**\n * Setup auto-instrumentation for AI frameworks\n */\nfunction setupAutoInstrumentation(\n  frameworks: Array<'langchain' | 'openai' | 'anthropic' | 'llamaindex'>\n): void {\n  if (config.debug) {\n    console.log(`Setting up auto-instrumentation for: ${frameworks.join(', ')}`);\n  }\n\n  // Note: Full auto-instrumentation requires framework-specific patches\n  // This is a placeholder for the instrumentation setup\n  // In production, use @opentelemetry/instrumentation-* packages\n\n  for (const framework of frameworks) {\n    try {\n      switch (framework) {\n        case 'openai':\n          instrumentOpenAI();\n          break;\n        case 'langchain':\n          instrumentLangChain();\n          break;\n        case 'anthropic':\n          instrumentAnthropic();\n          break;\n        case 'llamaindex':\n          instrumentLlamaIndex();\n          break;\n      }\n    } catch (error) {\n      if (config.debug) {\n        console.warn(`Failed to instrument ${framework}:`, error);\n      }\n    }\n  }\n}\n\nfunction instrumentOpenAI(): void {\n  // Placeholder - would patch OpenAI client\n  if (config.debug) {\n    console.log('OpenAI instrumentation ready');\n  }\n}\n\nfunction instrumentLangChain(): void {\n  // Placeholder - would use LangChain callbacks\n  if (config.debug) {\n    console.log('LangChain instrumentation ready');\n  }\n}\n\nfunction instrumentAnthropic(): void {\n  // Placeholder - would patch Anthropic client\n  if (config.debug) {\n    console.log('Anthropic instrumentation ready');\n  }\n}\n\nfunction instrumentLlamaIndex(): void {\n  // Placeholder - would use LlamaIndex callbacks\n  if (config.debug) {\n    console.log('LlamaIndex instrumentation ready');\n  }\n}\n\n// ============================================================================\n// CONVENIENCE FUNCTIONS\n// ============================================================================\n\n/**\n * Create a trace and analyze it in one call\n *\n * @example\n * ```typescript\n * const { trace, analysis } = await createAndAnalyze({\n *   userMessage: 'Help me track my order',\n *   agentResponse: 'Your order is on the way...',\n * });\n * ```\n */\nexport async function createAndAnalyze(\n  traceData: TraceOptions,\n  options: AnalyzeOptions = {}\n): Promise<{\n  trace: TraceOptions;\n  analysis: ExplainabilityResult;\n}> {\n  const analysis = await explainer.analyze(traceData, options);\n  return { trace: traceData, analysis };\n}\n\n// ============================================================================\n// EXPORTS\n// ============================================================================\n\nexport default {\n  init,\n  getTracer,\n  isInitialized,\n  traceLLM,\n  traceRetrieval,\n  traceTool,\n  traceChain,\n  explainer,\n  feedback,\n  quality,\n  analytics,\n  createAndAnalyze,\n};\n"]}
|